diff --git a/LICENSE b/LICENSE index 206754f..4275a6f 100644 --- a/LICENSE +++ b/LICENSE @@ -18,4 +18,4 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. \ No newline at end of file +SOFTWARE. diff --git a/README.md b/README.md index 4fd0ff8..a867a85 100644 --- a/README.md +++ b/README.md @@ -37,7 +37,7 @@ React Query có cơ chế caching hơi khác một chút so với RTK Query, nê - `inactive`: là khi data đó không còn component nào subcribe cả ```tsx -const result = useQuery({ queryKey: ['todos'], queryFn: fetchTodoList }) +const result = useQuery({ queryKey: ["todos"], queryFn: fetchTodoList }); ``` `result` là một object chứa một vài state rất quan trọng: `status`, `fetchStatus`,... @@ -88,13 +88,13 @@ Giả sử chúng ta dùng `cacheTime` mặc định là **5 phút** và `staleT ```jsx function A() { - const result = useQuery({ queryKey: ['todos'], queryFn: fetchTodos }) + const result = useQuery({ queryKey: ["todos"], queryFn: fetchTodos }); } function B() { - const result = useQuery({ queryKey: ['todos'], queryFn: fetchTodos }) + const result = useQuery({ queryKey: ["todos"], queryFn: fetchTodos }); } function C() { - const result = useQuery({ queryKey: ['todos'], queryFn: fetchTodos }) + const result = useQuery({ queryKey: ["todos"], queryFn: fetchTodos }); } ``` diff --git a/server/db.json b/server/db.json index 690b7df..27b5450 100644 --- a/server/db.json +++ b/server/db.json @@ -1,45 +1,5 @@ { "students": [ - { - "id": 1, - "first_name": "Martie", - "last_name": "Kite", - "email": "mkite0@statcounter.com", - "gender": "Female", - "country": "Portugal", - "avatar": "", - "btc_address": "1JADzZcCizitPooetjtrTpFUuThFmi5qvi" - }, - { - "id": 2, - "first_name": "Sherilyn", - "last_name": "Paddon", - "email": "spaddon1@washington.edu", - "gender": "Female", - "country": "Cameroon", - "avatar": "", - "btc_address": "14dvHxKVi6Cqic1dmitcfXUsikr6X7URgt" - }, - { - "id": 3, - "first_name": "Winona", - "last_name": "Kennerley", - "email": "wkennerley2@slate.com", - "gender": "Female", - "country": "Portugal", - "avatar": "", - "btc_address": "1QFU1svgpSfvKrXsLHmECf92D9vn6tNwoy" - }, - { - "id": 4, - "first_name": "Urbanus", - "last_name": "Kille", - "email": "ukille3@google.ca", - "gender": "Male", - "country": "Solomon Islands", - "avatar": "", - "btc_address": "1PWkrbbLAqyjbAa9H2wFMKuyWG5bd4TgpN" - }, { "id": 5, "first_name": "Shea", @@ -70,26 +30,6 @@ "avatar": "", "btc_address": "1CZXCy88sbPxThH3fvpWSA34hhLoVzWBwT" }, - { - "id": 8, - "first_name": "Christean", - "last_name": "McAleese", - "email": "cmcaleese7@earthlink.net", - "gender": "Agender", - "country": "Vietnam", - "avatar": "", - "btc_address": "19sQuXkrNyVXB2D8i3SbBci4AHnLobpsqs" - }, - { - "id": 9, - "first_name": "Max", - "last_name": "de Tocqueville", - "email": "mdetocqueville8@oaic.gov.au", - "gender": "Agender", - "country": "Indonesia", - "avatar": "", - "btc_address": "18U4YqWvv5EGgh9mxUNEswFUkWTDspgKFi" - }, { "id": 10, "first_name": "Fernande", @@ -120,16 +60,6 @@ "avatar": "", "btc_address": "15v7JuRYocw4djokjJPQrt7HPohoEjyDMV" }, - { - "id": 13, - "first_name": "Glenine", - "last_name": "Duffit", - "email": "gduffitc@addthis.com", - "gender": "Female", - "country": "Syria", - "avatar": "", - "btc_address": "17APzB8nXDDo3MVfSRuKR9YneMLrFUAiJM" - }, { "id": 14, "first_name": "Rae", @@ -999,6 +929,36 @@ "country": "Colombia", "avatar": "", "btc_address": "12tTumUNmtwbrLz3Xf6mDXn3qqBuNN2UPB" + }, + { + "avatar": "123123", + "email": "lspals@gmail.com", + "btc_address": "12312312", + "country": "21312", + "first_name": "123312", + "last_name": "123123", + "gender": "male", + "id": 101 + }, + { + "avatar": "123123", + "email": "lspals@gmail.com", + "btc_address": "12312312", + "country": "213", + "first_name": "123312", + "last_name": "123123", + "gender": "male", + "id": 102 + }, + { + "avatar": "3123", + "email": "lspals@gmail.com", + "btc_address": "3123123", + "country": "12312", + "first_name": "312312", + "last_name": "31231", + "gender": "other", + "id": 103 } ] -} +} \ No newline at end of file diff --git a/server/server.js b/server/server.js index 724d025..6bb9385 100644 --- a/server/server.js +++ b/server/server.js @@ -1,64 +1,64 @@ -const jsonServer = require('json-server') -const server = jsonServer.create() -const router = jsonServer.router('db.json') -const middlewares = jsonServer.defaults() +const jsonServer = require("json-server"); +const server = jsonServer.create(); +const router = jsonServer.router("db.json"); +const middlewares = jsonServer.defaults(); -const PORT = 4000 -const DELAY = 8000 +const PORT = 4000; +const DELAY = 8000; const validateEmail = (email) => { return String(email) .toLowerCase() .match( /^(([^<>()[\]\\.,;:\s@"]+(\.[^<>()[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/ - ) -} + ); +}; // Set default middlewares (logger, static, cors and no-cache) -server.use(middlewares) +server.use(middlewares); // To handle POST, PUT and PATCH you need to use a body-parser // You can use the one used by JSON Server -server.use(jsonServer.bodyParser) +server.use(jsonServer.bodyParser); server.use((req, res, next) => { - if (['POST', 'PUT', 'PATCH'].includes(req.method)) { + if (["POST", "PUT", "PATCH"].includes(req.method)) { if (!validateEmail(req.body.email)) { return res.status(422).send({ error: { - email: 'Email không đúng định dạng' - } - }) + email: "Email không đúng định dạng", + }, + }); } - if (req.body.last_name === 'admin') { + if (req.body.last_name === "admin") { return res.status(500).send({ - error: 'Server bị lỗi' - }) + error: "Server bị lỗi", + }); } } setTimeout(() => { - next() - }, DELAY) -}) + next(); + }, DELAY); +}); router.render = (req, res) => { - let data = res.locals.data - const { originalUrl } = req + let data = res.locals.data; + const { originalUrl } = req; if ( - req.method === 'GET' && - (originalUrl === '/students' || /^\/students\?.*$/.test(originalUrl)) + req.method === "GET" && + (originalUrl === "/students" || /^\/students\?.*$/.test(originalUrl)) ) { data = data.map((student) => ({ id: student.id, avatar: student.avatar, last_name: student.last_name, - email: student.email - })) + email: student.email, + })); } - res.jsonp(data) -} + res.jsonp(data); +}; // Use default router -server.use(router) +server.use(router); server.listen(PORT, () => { - console.log(`JSON Server is running at http://localhost:${PORT}`) -}) + console.log(`JSON Server is running at http://localhost:${PORT}`); +}); diff --git a/starter-template/package.json b/starter-template/package.json index fcb6627..1a89da0 100644 --- a/starter-template/package.json +++ b/starter-template/package.json @@ -3,6 +3,8 @@ "version": "0.1.0", "private": true, "dependencies": { + "@tanstack/react-query": "^5.56.2", + "@tanstack/react-query-devtools": "^5.56.2", "@testing-library/jest-dom": "^5.14.1", "@testing-library/react": "^13.0.0", "@testing-library/user-event": "^13.2.1", @@ -11,10 +13,12 @@ "@types/react": "^18.0.0", "@types/react-dom": "^18.0.0", "axios": "^1.1.3", + "classnames": "^2.5.1", "react": "^18.2.0", "react-dom": "^18.2.0", "react-router-dom": "^6.4.2", "react-scripts": "5.0.1", + "react-toastify": "^10.0.5", "typescript": "^4.4.2", "web-vitals": "^2.1.0" }, diff --git a/starter-template/src/App.tsx b/starter-template/src/App.tsx index ef30c96..b230f98 100644 --- a/starter-template/src/App.tsx +++ b/starter-template/src/App.tsx @@ -1,3 +1,5 @@ +import { useIsFetching, useIsMutating } from '@tanstack/react-query' +import Spinner from 'components/Spinner' import MainLayout from 'layouts/MainLayout' import About from 'pages/About' import AddStudent from 'pages/AddStudent' @@ -5,6 +7,8 @@ import Dashboard from 'pages/Dashboard' import NotFound from 'pages/NotFound' import Students from 'pages/Students' import { useRoutes } from 'react-router-dom' +import { ToastContainer } from 'react-toastify' +import 'react-toastify/dist/ReactToastify.css' function App() { const elements = useRoutes([ @@ -34,8 +38,13 @@ function App() { } ]) + const isFetching = useIsFetching() + const isMutating = useIsMutating() + return (