diff --git a/backend/prisma/migrations/20231211235124_/migration.sql b/backend/prisma/migrations/20231211235124_/migration.sql new file mode 100644 index 0000000..ce77f7d --- /dev/null +++ b/backend/prisma/migrations/20231211235124_/migration.sql @@ -0,0 +1,2 @@ +-- AlterTable +ALTER TABLE "Drill" ALTER COLUMN "image" SET DEFAULT 'abc'; diff --git a/backend/prisma/migrations/20231211235631_/migration.sql b/backend/prisma/migrations/20231211235631_/migration.sql new file mode 100644 index 0000000..12c0cac --- /dev/null +++ b/backend/prisma/migrations/20231211235631_/migration.sql @@ -0,0 +1,2 @@ +-- AlterTable +ALTER TABLE "Drill" ALTER COLUMN "image" SET DEFAULT ''; diff --git a/backend/prisma/schema.prisma b/backend/prisma/schema.prisma index 6417eda..11c0658 100644 --- a/backend/prisma/schema.prisma +++ b/backend/prisma/schema.prisma @@ -31,7 +31,7 @@ model Drill { title String description String observations String - image String @default("") + image String @default("") classPlanId String classPlan ClassPlan @relation(fields: [classPlanId], references: [id]) drillElements DrillElement[] diff --git a/backend/src/controllers/classPlanController.ts b/backend/src/controllers/classPlanController.ts index 86505d9..83a072c 100644 --- a/backend/src/controllers/classPlanController.ts +++ b/backend/src/controllers/classPlanController.ts @@ -40,14 +40,9 @@ export default class classPlanController { where: { userId: userId, }, - include: { - user: { - select: { - id: true, - email: true, - name: true, - }, - }, + select: { + id: true, + title: true, }, }) res.status(200).json(classPlan) @@ -62,14 +57,12 @@ export default class classPlanController { where: { id: id, }, - include: { - user: { - select: { - id: true, - email: true, - name: true, - }, - }, + select: { + id: true, + title: true, + goals: true, + observations: true, + userId: true, }, }) res.status(200).json(classPlan) @@ -78,20 +71,113 @@ export default class classPlanController { } } + searchByCreatedAtOrTitle = async (req: Request, res: Response) => { + try { + const { userId, title, startDate, finalDate } = req.params + if (startDate == "_") { + const classPlan = await prisma.classPlan.findMany({ + where: { + userId: userId, + title: { contains: title }, + }, + select: { + id: true, + title: true, + }, + }) + res.status(200).json(classPlan) + } else if (title == "_") { + const start = new Date(startDate) + const final = new Date(finalDate) + const classPlan = await prisma.classPlan.findMany({ + where: { + userId: userId, + createdAt: { + lte: final, + gte: start, + }, + }, + select: { + id: true, + title: true, + }, + }) + res.status(200).json(classPlan) + } else { + const start = new Date(startDate) + const final = new Date(finalDate) + const classPlan = await prisma.classPlan.findMany({ + where: { + userId: userId, + createdAt: { + lte: final, + gte: start, + }, + title: { contains: title }, + }, + select: { + id: true, + title: true, + }, + }) + res.status(200).json(classPlan) + } + } catch (err) { + res.status(500).json({ error: "Internal Server Error" }) + } + } + delete = async (req: Request, res: Response) => { try { const { id } = req.params + const data = await prisma.classPlan.findMany({ + where: { + id, + }, + select: { + id: true, + drills: { + select: { + id: true, + drillElements: { + select: { + id: true, + }, + }, + }, + }, + }, + }) - const classPlan = await prisma.classPlan.deleteMany({ - where: { id }, + data.map(async (item) => { + item.drills.map(async (drill) => { + drill.drillElements.map(async (drillElement) => { + await prisma.drillElement.delete({ + where: { + id: drillElement.id, + }, + }) + }) + await prisma.drill.delete({ + where: { + id: drill.id, + }, + }) + }) }) - res.status(200).json(classPlan) + const plan = await prisma.classPlan.delete({ + where: { + id, + }, + }) + res.status(200).json(plan) } catch (err) { err as Prisma.PrismaClientKnownRequestError res.status(500).json({ errors: { server: "Server error" } }) } } + updateById = async (req: Request, res: Response) => { try { const id = req.params.id diff --git a/backend/src/controllers/drillController.ts b/backend/src/controllers/drillController.ts index 821c203..51292c7 100644 --- a/backend/src/controllers/drillController.ts +++ b/backend/src/controllers/drillController.ts @@ -84,6 +84,7 @@ export default class drillController { image, }, }) + console.log("Foi") res.status(204).json(updatedDrill) } catch (err) { res.status(500).json({ error: "Internal Server Error" }) @@ -93,12 +94,36 @@ export default class drillController { deleteById = async (req: Request, res: Response) => { try { const id = req.params.id - const deletedDrill = await prisma.drill.deleteMany({ + + const drills = await prisma.drill.findMany({ where: { id, }, + select: { + drillElements: { + select: { + id: true, + }, + }, + }, + }) + + drills.map(async (drill) => { + drill.drillElements.map(async (drillElement) => { + await prisma.drillElement.delete({ + where: { + id: drillElement.id, + }, + }) + }) + await prisma.drill.delete({ + where: { + id, + }, + }) }) - res.status(204).json(deletedDrill) + + res.status(204).json({ message: "Drill deleted" }) } catch (err) { console.log(err) res.status(500).json({ error: "Internal Server Error" }) diff --git a/backend/src/middleware/authUser.ts b/backend/src/middleware/authUser.ts index dd8d6aa..cc3adc2 100644 --- a/backend/src/middleware/authUser.ts +++ b/backend/src/middleware/authUser.ts @@ -1,4 +1,5 @@ -import { sign } from "jsonwebtoken" +import { sign, verify } from "jsonwebtoken" +import { Request, Response, NextFunction } from "express" interface Idata { id: string @@ -11,4 +12,22 @@ export default class AuthUser { expiresIn: "30d", }) } + + async auth( + request: Request, + response: Response, + next: NextFunction, + ): Promise { + let token + try { + token = request.headers.authorization?.split(" ")[1] + verify(token as string, process.env.AUTH_CONFIG_SECRET as string) + } catch (err) { + return response.status(401).json({ + message: "Token inválido!", + }) + } + next() + return null + } } diff --git a/backend/src/routes/classPlanRouters.ts b/backend/src/routes/classPlanRouters.ts index 042fce4..0f9bf74 100644 --- a/backend/src/routes/classPlanRouters.ts +++ b/backend/src/routes/classPlanRouters.ts @@ -1,13 +1,20 @@ import { Router } from "express" import classPlanController from "../controllers/classPlanController" +import AuthUser from "../middleware/authUser" const classPlanRouters = Router() +const authenticateUser = new AuthUser() const classPlan = new classPlanController() -classPlanRouters.post("/", classPlan.create) -classPlanRouters.get("/:id", classPlan.show) +classPlanRouters.post("/", authenticateUser.auth, classPlan.create) +classPlanRouters.get("/:id", authenticateUser.auth, classPlan.show) +classPlanRouters.get( + "/pesquisa-data-titulo/:userId/:title/:startDate/:finalDate", + authenticateUser.auth, + classPlan.searchByCreatedAtOrTitle, +) classPlanRouters.get("/planos-usuario/:userId", classPlan.list) -classPlanRouters.put("/:id", classPlan.updateById) -classPlanRouters.delete("/:id", classPlan.delete) +classPlanRouters.put("/:id", authenticateUser.auth, classPlan.updateById) +classPlanRouters.delete("/:id", authenticateUser.auth, classPlan.delete) export default classPlanRouters diff --git a/backend/src/routes/drillElementRouters.ts b/backend/src/routes/drillElementRouters.ts index c2e2835..b7bac00 100644 --- a/backend/src/routes/drillElementRouters.ts +++ b/backend/src/routes/drillElementRouters.ts @@ -1,12 +1,22 @@ import { Router } from "express" import drillElementController from "@/controllers/drillElementController" +import AuthUser from "../middleware/authUser" const drillElementRouters = Router() +const authenticateUser = new AuthUser() const drillElement = new drillElementController() -drillElementRouters.post("/", drillElement.create) -drillElementRouters.get("/:drillId", drillElement.getManyByDrillId) -drillElementRouters.put("/:id", drillElement.updateById) -drillElementRouters.delete("/:id", drillElement.deleteById) +drillElementRouters.post("/", authenticateUser.auth, drillElement.create) +drillElementRouters.get( + "/:drillId", + authenticateUser.auth, + drillElement.getManyByDrillId, +) +drillElementRouters.put("/:id", authenticateUser.auth, drillElement.updateById) +drillElementRouters.delete( + "/:id", + authenticateUser.auth, + drillElement.deleteById, +) export default drillElementRouters diff --git a/backend/src/routes/drillRouters.ts b/backend/src/routes/drillRouters.ts index 169cf22..360a3d5 100644 --- a/backend/src/routes/drillRouters.ts +++ b/backend/src/routes/drillRouters.ts @@ -1,14 +1,20 @@ import { Router } from "express" import drillController from "../controllers/drillController" +import AuthUser from "../middleware/authUser" +const authenticateUser = new AuthUser() const drillRouters = Router() const drill = new drillController() -drillRouters.post("/", drill.create) -drillRouters.get("/:classPlanId", drill.getManyByClassPlanId) -drillRouters.get("/drill/:id", drill.show) -drillRouters.put("/:id", drill.updateById) -drillRouters.put("/image/:id", drill.updateImage) -drillRouters.delete("/:id", drill.deleteById) +drillRouters.post("/", authenticateUser.auth, drill.create) +drillRouters.get( + "/:classPlanId", + authenticateUser.auth, + drill.getManyByClassPlanId, +) +drillRouters.get("/drill/:id", authenticateUser.auth, drill.show) +drillRouters.put("/:id", authenticateUser.auth, drill.updateById) +drillRouters.put("/image/:id", authenticateUser.auth, drill.updateImage) +drillRouters.delete("/:id", authenticateUser.auth, drill.deleteById) export default drillRouters diff --git a/backend/src/zodSchemas/drill.zod.ts b/backend/src/zodSchemas/drill.zod.ts index bad0550..d947046 100644 --- a/backend/src/zodSchemas/drill.zod.ts +++ b/backend/src/zodSchemas/drill.zod.ts @@ -9,7 +9,6 @@ const drillSchema = z.object({ .string() .max(500, "Observações deve ter no máximo 500 caracteres"), classPlanId: z.string().uuid("Id do plano de aula inválido"), - image: z.string(), }) export { drillSchema } diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 3d5d101..c326979 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -12,6 +12,7 @@ "apexcharts": "^3.41.0", "axios": "^1.6.2", "headlessui": "^0.0.0", + "html2canvas": "^1.4.1", "jsvectormap": "^1.5.3", "match-sorter": "^6.3.1", "react": "^18.2.0", @@ -24,6 +25,7 @@ "sort-by": "^0.0.2" }, "devDependencies": { + "@iconify/react": "^4.1.1", "@types/react": "^18.2.17", "@types/react-dom": "^18.2.7", "@vitejs/plugin-react": "^4.0.3", @@ -758,6 +760,7 @@ "version": "4.1.1", "resolved": "https://registry.npmjs.org/@iconify/react/-/react-4.1.1.tgz", "integrity": "sha512-jed14EjvKjee8mc0eoscGxlg7mSQRkwQG3iX3cPBCO7UlOjz0DtlvTqxqEcHUJGh+z1VJ31Yhu5B9PxfO0zbdg==", + "dev": true, "dependencies": { "@iconify/types": "^2.0.0" }, @@ -771,7 +774,8 @@ "node_modules/@iconify/types": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/@iconify/types/-/types-2.0.0.tgz", - "integrity": "sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg==" + "integrity": "sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg==", + "dev": true }, "node_modules/@jridgewell/gen-mapping": { "version": "0.3.3", @@ -1276,6 +1280,14 @@ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", "dev": true }, + "node_modules/base64-arraybuffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz", + "integrity": "sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ==", + "engines": { + "node": ">= 0.6.0" + } + }, "node_modules/big.js": { "version": "5.2.2", "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", @@ -1500,6 +1512,14 @@ "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==", "dev": true }, + "node_modules/css-line-break": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/css-line-break/-/css-line-break-2.1.0.tgz", + "integrity": "sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w==", + "dependencies": { + "utrie": "^1.0.2" + } + }, "node_modules/cssesc": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", @@ -1943,6 +1963,18 @@ "resolved": "https://registry.npmjs.org/headlessui/-/headlessui-0.0.0.tgz", "integrity": "sha512-CHvacVPbl8AqIg2sBNKySUmumu7o15jSrCaTrIh9GW2Eq4y/krCN/vZFOsKCwlrhWQbO4267a8xvvP8bs+qREQ==" }, + "node_modules/html2canvas": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/html2canvas/-/html2canvas-1.4.1.tgz", + "integrity": "sha512-fPU6BHNpsyIhr8yyMpTLLxAbkaK8ArIBcmZIRiBLiDhjeqvXolaEmDGmELFuX9I4xDcaKKcJl+TKZLqruBbmWA==", + "dependencies": { + "css-line-break": "^2.1.0", + "text-segmentation": "^1.0.3" + }, + "engines": { + "node": ">=8.0.0" + } + }, "node_modules/inflight": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", @@ -3188,6 +3220,14 @@ "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", "dev": true }, + "node_modules/text-segmentation": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/text-segmentation/-/text-segmentation-1.0.3.tgz", + "integrity": "sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw==", + "dependencies": { + "utrie": "^1.0.2" + } + }, "node_modules/thenify": { "version": "3.3.1", "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz", @@ -3281,6 +3321,14 @@ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", "dev": true }, + "node_modules/utrie": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/utrie/-/utrie-1.0.2.tgz", + "integrity": "sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw==", + "dependencies": { + "base64-arraybuffer": "^1.0.2" + } + }, "node_modules/vite": { "version": "4.4.7", "resolved": "https://registry.npmjs.org/vite/-/vite-4.4.7.tgz", diff --git a/frontend/src/App.tsx b/frontend/src/App.tsx index 6ec304b..bfde9dd 100644 --- a/frontend/src/App.tsx +++ b/frontend/src/App.tsx @@ -8,6 +8,7 @@ import Loader from './common/Loader'; import routes from './routes'; import ViewPlan from './pages/ViewPlan'; import Drill from './pages/Drill'; +import FilteredClassPlans from './pages/FilteredClassPlans'; const DefaultLayout = lazy(() => import('./layout/DefaultLayout')); @@ -39,6 +40,7 @@ function App() { } /> + } /> } /> }> {/* } /> */} diff --git a/frontend/src/components/DrillElement.tsx b/frontend/src/components/DrillElement.tsx index 6dd3dff..330c7aa 100644 --- a/frontend/src/components/DrillElement.tsx +++ b/frontend/src/components/DrillElement.tsx @@ -1,12 +1,30 @@ -import React from "react"; -import useDragger from "./useDragger"; +import React from 'react'; +import useDragger from './useDragger'; +const DrillElement: React.FC<{ + id: string; + deleteItem: (idDeleted: string) => void; + image: any; + elementWidth: number; + top: number; + left: number; +}> = ({ id, deleteItem, image, elementWidth, top, left }) => { + useDragger(id, deleteItem); + return ( + Drill Element + ); +}; -const DrillElement: React.FC<{id : string, deleteItem : (idDeleted: string) => void, image: any, elementWidth: number, top: number, left: number}> = ({id, deleteItem, image, elementWidth, top, left}) => { - useDragger(id, deleteItem); - return( - Drill Element - ); -} - -export default DrillElement; \ No newline at end of file +export default DrillElement; diff --git a/frontend/src/components/Header.tsx b/frontend/src/components/Header.tsx index 406b651..2f52090 100644 --- a/frontend/src/components/Header.tsx +++ b/frontend/src/components/Header.tsx @@ -1,13 +1,24 @@ -import { Link } from 'react-router-dom'; import { Icon } from '@iconify/react/dist/iconify.js'; +import { useEffect, useState } from 'react'; -const Header = (props: { path: string; hasReturnArrow: boolean }) => { +const Header = (props: { + hasReturnArrow: boolean; + changeScreenFunction: () => void; +}) => { const logout = () => { localStorage.removeItem('token'); localStorage.removeItem('user'); window.location.href = '/login'; }; + useEffect(() => { + console.log("Aqui"); + const user = localStorage.getItem('user'); + if(!user){ + logout(); + } + }, []); + return (
{ >
{props.hasReturnArrow ? ( - + ) : (
)} diff --git a/frontend/src/components/useDragger.tsx b/frontend/src/components/useDragger.tsx index e300b98..9ee688b 100644 --- a/frontend/src/components/useDragger.tsx +++ b/frontend/src/components/useDragger.tsx @@ -1,64 +1,72 @@ -import { useEffect, useRef } from "react"; +import { useEffect, useRef } from 'react'; -function useDragger(id: string, deleteItem : (idDeleted: string) => void) : void{ +function useDragger(id: string, deleteItem: (idDeleted: string) => void): void { const isClicked = useRef(0); const coords = useRef<{ - startX: number, - startY: number + startX: number; + startY: number; }>({ startX: 0, - startY: 0 - }) + startY: 0, + }); useEffect(() => { const target = document.getElementById(id); - if(!target) throw new Error("Target with given id doesn't exist"); + if (!target) throw new Error("Target with given id doesn't exist"); const container = target.parentElement; - if(!container) throw new Error("Element must have a parent"); - const trash = document.getElementById("apagar"); - if(!trash) throw new Error("Element with given id doesn't exist"); + if (!container) throw new Error('Element must have a parent'); + const trash = document.getElementById('apagar'); + if (!trash) throw new Error("Element with given id doesn't exist"); const mouseDown = (e: MouseEvent) => { - if(!isClicked.current) { + if (!isClicked.current) { target.style.border = '2px solid rgb(59, 152, 245)'; - isClicked.current = 1; - trash.style.visibility = "visible"; + isClicked.current = 1; + trash.style.visibility = 'visible'; return; - } - trash.style.visibility = "hidden"; + } + trash.style.visibility = 'hidden'; if (isClicked.current === 2) { const bodyRect = container.getBoundingClientRect(); coords.current.startX = bodyRect.left; coords.current.startY = bodyRect.top; - target.style.top = `${e.clientY - coords.current.startY - (target.offsetHeight-4)/2}px`; - target.style.left = `${e.clientX - coords.current.startX - (target.offsetWidth-4)/2}px`; + target.style.top = `${ + e.clientY - coords.current.startY - (target.offsetHeight - 4) / 2 + }px`; + target.style.left = `${ + e.clientX - coords.current.startX - (target.offsetWidth - 4) / 2 + }px`; target.style.border = 'none'; } isClicked.current = 0; - } + }; const mouseDownContainer = (e: MouseEvent) => { - if(!isClicked.current) return; - if(isClicked.current === 1){ + if (!isClicked.current) return; + if (isClicked.current === 1) { isClicked.current = 2; } else { const bodyRect = container.getBoundingClientRect(); coords.current.startX = bodyRect.left; coords.current.startY = bodyRect.top; - target.style.top = `${e.clientY - coords.current.startY - (target.offsetHeight-4)/2}px`; - target.style.left = `${e.clientX - coords.current.startX - (target.offsetWidth-4)/2}px`; + target.style.top = `${ + e.clientY - coords.current.startY - (target.offsetHeight - 4) / 2 + }px`; + target.style.left = `${ + e.clientX - coords.current.startX - (target.offsetWidth - 4) / 2 + }px`; isClicked.current = 0; target.style.border = 'none'; - trash.style.visibility = "hidden"; + trash.style.visibility = 'hidden'; } - } + }; const clickTrash = () => { - if(!isClicked.current) return; + if (!isClicked.current) return; deleteItem(id); trash.style.backgroundColor = 'inherit'; - trash.style.visibility = "hidden"; - } + trash.style.visibility = 'hidden'; + }; target.addEventListener('mousedown', mouseDown); container.addEventListener('mousedown', mouseDownContainer); @@ -68,10 +76,10 @@ function useDragger(id: string, deleteItem : (idDeleted: string) => void) : void target.removeEventListener('mousedown', mouseDown); container.removeEventListener('mousedown', mouseDownContainer); trash.removeEventListener('mousedown', clickTrash); - } + }; return cleanup; - }, [id, deleteItem]) + }, [id, deleteItem]); } -export default useDragger; \ No newline at end of file +export default useDragger; diff --git a/frontend/src/index.css b/frontend/src/index.css index 043ac03..5e289f4 100644 --- a/frontend/src/index.css +++ b/frontend/src/index.css @@ -288,12 +288,33 @@ } .buttonsPanel { - width: 100vw; + width: 100%; display: grid; grid-template-columns: repeat(auto-fit, minmax(330px, 330px)); grid-gap: 20px; } +.searchClassPlanPanel { + width: 100%; + display: grid; + grid-template-columns: repeat(auto-fit, minmax(250px, 250px)); + grid-gap: 20px; + margin: 15px 0px; +} + +input[type='date']::-webkit-calendar-picker-indicator { + opacity: 1; + color-scheme: dark; + width: 25px; + height: 20px; + border-width: thin; + box-sizing: content-box; + margin: 0px; +} + +.datepickerbg { + color-scheme: dark; +} .itemButton { height: 115px; background-color: #5899eb; @@ -342,13 +363,39 @@ padding-right: 10px; } -.contentDrill { +.titleLayoutClassPLan { + display: flex; + flex-direction: row; + margin: auto; + width: 90%; +} + +.titleLayoutClassPLan { + display: flex; + flex-direction: row; + margin: auto; + width: 90%; +} + +.contentDrill{ display: flex; justify-content: space-between; padding: 0px 15px; } -#drillGraphicContainer { +.contentClassPlan{ + display: flex; + flex-direction: column; + align-items: center; +} + +.contentClassPlan { + display: flex; + flex-direction: column; + align-items: center; +} + +#drillGraphicContainer{ margin-top: 5px; height: 406px; width: 800px; @@ -377,15 +424,19 @@ background-color: #494949; padding: 15px; border-radius: 20px; - height: 300px; + min-height: 300px; width: 350px; + box-sizing: content-box; + height: fit-content; } .pClassPlan { - color: white; + margin: 10px auto; background-color: #494949; padding: 15px; border-radius: 20px; + min-height: 300px; + box-sizing: content-box; height: fit-content; width: 90%; } @@ -410,14 +461,22 @@ border: '1px solid black'; } -.textAreaDrill { - background-color: #494949; - border-radius: 20px; - height: 300px; - width: 250px; - height: 300px; - width: 350px; - padding: 15px; +.drillButtonsPanel { + width: 100%; + margin-top: 5px; + display: grid; + grid-template-columns: repeat(auto-fit, minmax(40px, 40px)); + grid-gap: 2px; + justify-content: center; +} + +.buttonAddElementDrill { + background-color: gray; + border-radius: 5px; + display: flex; + justify-content: center; + align-items: center; + min-height: 45px; } @media (max-width: 910px) { @@ -431,11 +490,23 @@ .drillPage { color: white; - width: 100vw; - height: 100vh; + width: 100%; + height: 100%; display: flex; flex-direction: column; background-color: #272727; padding-bottom: 10px; } -} \ No newline at end of file + + .contentDrill { + flex-direction: column; + } + + #img_drill { + display: none; + } + + .pageDrill { + padding: 0px; + } +} diff --git a/frontend/src/lib.d.ts b/frontend/src/lib.d.ts index 77fa6c8..091d25e 100644 --- a/frontend/src/lib.d.ts +++ b/frontend/src/lib.d.ts @@ -1,4 +1,4 @@ declare module '*.svg' { const content: any; export default content; -} \ No newline at end of file +} diff --git a/frontend/src/pages/Authentication/SignIn.tsx b/frontend/src/pages/Authentication/SignIn.tsx index 6836977..4d38efd 100644 --- a/frontend/src/pages/Authentication/SignIn.tsx +++ b/frontend/src/pages/Authentication/SignIn.tsx @@ -7,10 +7,12 @@ import { Icon } from '@iconify/react/dist/iconify.js'; import '../../index.css'; import Logo from '../../images/logoBeachTennisCoordimate.png'; import { Link } from 'react-router-dom'; +import ClassPlanService from '../../service/classPlanService'; const SignIn = () => { const [email, setEmail] = useState(''); const [password, setPassword] = useState(''); + const classPlan = new ClassPlanService(); const handleSignIn = async () => { try { @@ -25,11 +27,20 @@ const SignIn = () => { if (data.status === 200) { localStorage.setItem('token', data.data.token); console.log(data.data); - localStorage.setItem('user', JSON.stringify(data.data)); - toast.success('Login realizado com sucesso!'); - setTimeout(() => { - window.location.href = '/'; - }, 3000); + const data_user = JSON.stringify(data.data); + localStorage.setItem('user', data_user); + try { + const user_id = String(JSON.parse(data_user).id); + let userClassPlans = await classPlan.getManyById(user_id as string); + localStorage.setItem( + 'userClassPlans', + JSON.stringify(userClassPlans.data), + ); + toast.success('Login realizado com sucesso!'); + setTimeout(() => { + window.location.href = '/'; + }, 3000); + } catch (error) {} } return; } diff --git a/frontend/src/pages/ClassPlans.tsx b/frontend/src/pages/ClassPlans.tsx index 74ed2ef..f8c9e1d 100644 --- a/frontend/src/pages/ClassPlans.tsx +++ b/frontend/src/pages/ClassPlans.tsx @@ -1,17 +1,23 @@ -import { useEffect, useState } from 'react'; +import { useState } from 'react'; import '../index.css'; import { Icon } from '@iconify/react'; import { toast, ToastContainer } from 'react-toastify'; import ClassPlanService from '../service/classPlanService'; -import { Link } from 'react-router-dom'; import Header from '../components/Header'; +import DrillService from '../service/drillService'; const ClassPlans = () => { - const [classPlans, setClassPlans] = useState([]); + const userClassPlans = localStorage.getItem('userClassPlans'); + const [classPlans, setClassPlans] = useState( + JSON.parse(userClassPlans ? userClassPlans : ''), + ); const userString = localStorage.getItem('user'); const [deletedItem, setDeletedItem] = useState(''); const [deletedItemTitle, setDeletedItemTitle] = useState(''); const [visibleDel, setVisibleDel] = useState(false); + const [searchedTitle, setSearchedTitle] = useState(''); + const [initialDate, setInitialDate] = useState(''); + const [finalDate, setFinalDate] = useState(''); let id: String; if (userString !== null) { const user = JSON.parse(userString); @@ -20,13 +26,11 @@ const ClassPlans = () => { id = ''; } - useEffect(() => { - loadPlans(); - }, []); - const [title, setTitle] = useState(''); const [visible, setVisible] = useState(false); const classPlan = new ClassPlanService(); + const drill = new DrillService(); + let data = { id: '', title: title, @@ -34,12 +38,6 @@ const ClassPlans = () => { observations: '', userId: id, }; - const [plans, setPlans] = useState([data]); - - async function loadPlans() { - const response = await classPlan.get('/'); - setPlans(response.data); - } const addClassPlan = () => { setVisible(true); @@ -58,7 +56,8 @@ const ClassPlans = () => { else if (title.length < 5) toast.warning('O título deve ter no mínimo 5 caracteres'); else { - await classPlan.save(data); + let newClassPlanData = await classPlan.save(data); + setClassPlans((prevArray) => [...prevArray, newClassPlanData.data]); toast.success('Plano de aula cadastrado com sucesso'); closeNewItemPanel(); setTitle(''); @@ -82,6 +81,9 @@ const ClassPlans = () => { const handleDeleteClassPlan = async () => { try { await classPlan.remove(deletedItem); + setClassPlans((prevItems) => + prevItems.filter((item) => item.id != deletedItem), + ); toast.success('Plano de aula excluído com sucesso'); closeDeleteItemPanel(); setDeletedItem(''); @@ -91,31 +93,155 @@ const ClassPlans = () => { return; }; - useEffect(() => { - if (id != '') { - classPlan - .getManyById(id as string) - .then((response) => { - setClassPlans(response.data); - }) - .catch(() => { - setClassPlans([]); - }); + const redirectToViewPlan = async (selectedId: string) => { + try { + const response = await classPlan.getById(selectedId); + localStorage.setItem('selectedClassPlan', JSON.stringify(response.data)); + const drillsClassPlan = await drill.getManyByClassPlanId(selectedId); + const drillData = drillsClassPlan.data; + localStorage.setItem( + 'drillsSelectedClassPlan', + JSON.stringify(drillData), + ); + setTimeout(() => { + window.location.href = `/plano-aula/${selectedId}`; + }, 3000); + } catch (error) { + toast.error('Erro ao buscar informações do plano'); + } + }; + + const searchClassPlans = async () => { + try { + let tit_proc, data_ini_proc, data_final_proc; + if (searchedTitle) { + tit_proc = searchedTitle; + if ((initialDate && !finalDate) || (!initialDate && finalDate)) { + toast.warning( + 'Preencha os dois campos de data ou limpe os dois caso deseje pesquisar apenas por título', + ); + return; + } + if (initialDate) { + data_ini_proc = initialDate; + data_final_proc = finalDate; + } else { + data_ini_proc = '_'; + data_final_proc = '_'; + } + } else if (initialDate || finalDate) { + if (!finalDate || !initialDate) { + toast.warning( + 'Preencha os dois campos de data caso deseje pesquisar pela data de criação', + ); + return; + } + tit_proc = '_'; + data_ini_proc = initialDate; + data_final_proc = finalDate; + } else { + toast.warning( + 'Preencha os campos para pesquisar por título, data de criação ou pelos dois filtros', + ); + return; + } + let userClassPlans = await classPlan.getManyByTitleOrDate( + id as string, + tit_proc, + data_ini_proc, + data_final_proc, + ); + console.log(userClassPlans.data); + localStorage.setItem( + 'userClassPlans', + JSON.stringify(userClassPlans.data), + ); + toast.success('Pesquisa realizada com sucesso!'); + setTimeout(() => { + window.location.href = '/pesquisa'; + }, 3000); + } catch (error) { + toast.error('Erro ao procurar planos'); } - }, [classPlan, id]); - console.log(classPlans); + }; + return ( <> } /> -
+
{}} hasReturnArrow={false}>

Meus Planos de Aula

+
+
+

Título

+ setSearchedTitle(e.target.value)} + > +
+
+

Data Início

+ setInitialDate(e.target.value)} + > +
+
+

Data Fim

+ setFinalDate(e.target.value)} + > +
+
+ +
+
{ ); }; -export default ClassPlans; \ No newline at end of file +export default ClassPlans; diff --git a/frontend/src/pages/Drill.tsx b/frontend/src/pages/Drill.tsx index fd105b7..bbeee85 100644 --- a/frontend/src/pages/Drill.tsx +++ b/frontend/src/pages/Drill.tsx @@ -1,26 +1,26 @@ -import { useEffect, useState } from 'react'; +import { useState } from 'react'; import '../index.css'; import { Icon } from '@iconify/react'; import { toast, ToastContainer } from 'react-toastify'; import DrillService from '../service/drillService'; -import { useParams } from 'react-router-dom'; import html2canvas from 'html2canvas'; import Header from '../components/Header'; import DrillElement from '../components/DrillElement'; -import arco from "../images/drillElements/arco.png"; -import avatar_x_neg from "../images/drillElements/avatar_x-.png"; -import avatar_x_pos from "../images/drillElements/avatar_x+.png"; -import avatar_y_neg from "../images/drillElements/avatar_y-.png"; -import avatar_y_pos from "../images/drillElements/avatar_y+.png"; -import bola_tenis from "../images/drillElements/bola_tenis.png"; -import cano_alt_x from "../images/drillElements/cano_alt_x.png"; -import cano_alt_y from "../images/drillElements/cano_alt_y.png"; -import cone from "../images/drillElements/cone.png"; -import seta_x_neg from "../images/drillElements/seta_x-.png"; -import seta_x_pos from "../images/drillElements/seta_x+.png"; -import seta_y_neg from "../images/drillElements/seta_y-.png"; -import seta_y_pos from "../images/drillElements/seta_y+.png"; +import arco from '../images/drillElements/arco.png'; +import avatar_x_neg from '../images/drillElements/avatar_x-.png'; +import avatar_x_pos from '../images/drillElements/avatar_x+.png'; +import avatar_y_neg from '../images/drillElements/avatar_y-.png'; +import avatar_y_pos from '../images/drillElements/avatar_y+.png'; +import bola_tenis from '../images/drillElements/bola_tenis.png'; +import cano_alt_x from '../images/drillElements/cano_alt_x.png'; +import cano_alt_y from '../images/drillElements/cano_alt_y.png'; +import cone from '../images/drillElements/cone.png'; +import seta_x_neg from '../images/drillElements/seta_x-.png'; +import seta_x_pos from '../images/drillElements/seta_x+.png'; +import seta_y_neg from '../images/drillElements/seta_y-.png'; +import seta_y_pos from '../images/drillElements/seta_y+.png'; import DrillElementService from '../service/drillElementService'; +import ClassPlanService from '../service/classPlanService'; const Drill = () => { const listImages = [ @@ -36,7 +36,7 @@ const Drill = () => { seta_x_neg, seta_x_pos, seta_y_neg, - seta_y_pos + seta_y_pos, ]; const listButtons = [ [0, 50, arco, 35], @@ -51,167 +51,128 @@ const Drill = () => { [9, 30, seta_x_neg, 25], [10, 30, seta_x_pos, 25], [11, 30, seta_y_neg, 25], - [12, 30, seta_y_pos, 25] - ] - const { id } = useParams(); - const [title, setTitle] = useState(''); + [12, 30, seta_y_pos, 25], + ]; + const stringSelectedDrill = localStorage.getItem('selectedDrill'); + const selectedDrill = JSON.parse( + stringSelectedDrill ? stringSelectedDrill : '', + ); + const stringDrillSelectedElements = localStorage.getItem( + 'drillSelectedElements', + ); + const drillSelectedElements = JSON.parse( + stringDrillSelectedElements ? stringDrillSelectedElements : '', + ); + const [title, setTitle] = useState(selectedDrill.title); const [imagemBase64, setimagemBase64] = useState(''); const [titleAux, setTitleAux] = useState(''); - const [description, setDescription] = useState(''); + const [description, setDescription] = useState(selectedDrill.description); const [descriptionAux, setDescriptionAux] = useState(''); - const [observations, setObservations] = useState(''); + const [observations, setObservations] = useState(selectedDrill.observations); const [observationsAux, setObservationsAux] = useState(''); const [titleNotEdited, setTitleNotEdited] = useState(true); const [descriptionNotEdited, setDescriptionNotEdited] = useState(true); const [observationsNotEdited, setObservationsNotEdited] = useState(true); const drill = new DrillService(); const drillElement = new DrillElementService(); - const [drillUpdated, setDrillUpdated] = useState({ - id: '', - title: '', - description: '', - observations: '', - image: '', - classPlanId: '', - }); const [newItems, setNewItems] = useState(Array<[string, number, number]>); - const [savedItems, setSavedItems] = useState([{ - id: 'id0', - index: 0, - left: 0, - top: 0 - }] - ); + const [savedItems, setSavedItems] = useState(drillSelectedElements); const [deletedSavedIds, setDeletedSavedIds] = useState([]); const [count, setCount] = useState(0); - const updateImagen = () => { - const captureElement = document.querySelector('#drillGraphicContainer') - if(captureElement != null){ - html2canvas(captureElement as HTMLElement).then(canvas => { - const imgData = canvas.toDataURL('image/png'); - setimagemBase64(imgData) - }); - } - } - + const captureElement = document.querySelector('#drillGraphicContainer'); + if (captureElement != null) { + html2canvas(captureElement as HTMLElement).then((canvas) => { + const imgData = canvas.toDataURL('image/png'); + setimagemBase64(imgData); + }); + } + }; - const addDrillElement = (typeImage : number, width: number) => { + const addDrillElement = (typeImage: number, width: number) => { setNewItems([...newItems, [String(count), typeImage, width]]); setCount(count + 1); - } + }; - const deleteNewItem = (idDeleted : string) => { - setNewItems((prevItems) => prevItems.filter((id) => id[0] !== idDeleted)); - } + const deleteNewItem = (idDeleted: string) => { + setNewItems((prevItems) => prevItems.filter((id) => id[0] !== idDeleted)); + }; const deleteSavedItem = (idDeleted: string) => { - const node = document.getElementById(idDeleted); - if(node){ - node.style.display = 'none'; - setDeletedSavedIds((prevArray) => [...prevArray, idDeleted]); - } - } - - async function saveDrillState(){ - updateImagen() - let data = { - id: drillUpdated.id, - title: titleAux, - image: imagemBase64, - description: drillUpdated.description, - observations: drillUpdated.observations, - classPlanId: drillUpdated.classPlanId, - }; - console.log(data) - await drill.updateById(id as string, data); - for(const [idNewItem, indexNewItem] of newItems){ - try{ + setSavedItems((prevItems) => + prevItems.filter((item) => item.id !== idDeleted), + ); + setDeletedSavedIds((prevArray) => [...prevArray, idDeleted]); + }; + + async function saveDrillState() { + updateImagen(); + let data = { + image: imagemBase64 + }; + console.log(data.image) + await drill.updateImage(selectedDrill.id, data); + for (const [idNewItem, indexNewItem] of newItems) { + try { const element = document.getElementById(idNewItem); - if(element != null){ + if (element != null) { let data = { index: indexNewItem, top: parseInt(element.style.top.slice(0, -2)), left: parseInt(element.style.left.slice(0, -2)), - drillId: id + drillId: selectedDrill.id, }; await drillElement.save(data); } } catch (error) { toast.error('Erro ao salvar elementos'); return; - } + } } - for(const savedItem of savedItems){ - try{ + for (const savedItem of savedItems) { + try { const element = document.getElementById(savedItem.id); - if(element != null){ + if (element != null) { let data = { id: savedItem.id, index: savedItem.index, top: parseInt(element.style.top.slice(0, -2)), left: parseInt(element.style.left.slice(0, -2)), - drillId: id + drillId: selectedDrill.id, }; await drillElement.updateById(savedItem.id, data); } } catch (error) { toast.error('Erro ao salvar elementos'); return; - } + } } - for(const deletedSavedId of deletedSavedIds){ - try{ + for (const deletedSavedId of deletedSavedIds) { + try { await drillElement.deleteById(deletedSavedId); } catch (error) { toast.error('Erro ao salvar elementos'); return; - } + } } toast.success('Atualizações no desenho do drill salvas com sucesso'); - setNewItems([]); const trash = document.getElementById('apagar'); if (trash) trash.style.visibility = 'hidden'; } - async function loadData() { - if (id != null) { - const response = await drill.getById(id); - setDrillUpdated(response.data); - } - } - console.log("id: " + id) - useEffect(() => { - loadData(); - setTitle(drillUpdated.title); - setDescription(drillUpdated.description); - setObservations(drillUpdated.observations); - - if (id != null) { - drillElement - .getManyByDrillId(id) - .then((response) => { - setSavedItems(response.data); - }) - .catch(() => { - setSavedItems([]); - }); - } - }, [drillUpdated]); - const startEditingTitle = () => { - setTitleAux(drillUpdated.title); + setTitleAux(title); setTitleNotEdited(false); }; const startEditingDescription = () => { - setDescriptionAux(drillUpdated.description); + setDescriptionAux(description); setDescriptionNotEdited(false); }; const startEditingObservations = () => { - setObservationsAux(drillUpdated.observations); + setObservationsAux(observations); setObservationsNotEdited(false); }; @@ -221,36 +182,40 @@ const Drill = () => { else if (titleAux.length < 5) toast.warning('O título deve ter no mínimo 5 caracteres'); else { - updateImagen() + updateImagen(); let data = { - id: drillUpdated.id, + id: selectedDrill.id, title: titleAux, image: imagemBase64, - description: drillUpdated.description, - observations: drillUpdated.observations, - classPlanId: drillUpdated.classPlanId, + description: description, + observations: observations, + classPlanId: selectedDrill.classPlanId, }; - await drill.updateById(id as string, data); + await drill.updateById(selectedDrill.id, data); + setTitle(titleAux); toast.success('Título atualizado com sucesso'); } } catch (error) { toast.error('Erro ao atualizar drill'); + } finally { + setTitleAux(''); + setTitleNotEdited(true); } - return; }; const finishEditingDescription = async () => { try { - updateImagen() + updateImagen(); let data = { - id: drillUpdated.id, - title: drillUpdated.title, + id: selectedDrill.id, + title: title, description: descriptionAux, - observations: drillUpdated.observations, + observations: observations, image: imagemBase64, - classPlanId: drillUpdated.classPlanId, + classPlanId: selectedDrill.classPlanId, }; - await drill.updateById(id as string, data); + await drill.updateById(selectedDrill.id, data); + setDescription(descriptionAux); toast.success('Descrição atualizada com sucesso'); } catch (error) { toast.error('Erro ao atualizar drill'); @@ -258,22 +223,22 @@ const Drill = () => { setDescriptionAux(''); setDescriptionNotEdited(true); } - return; }; const finishEditingObservations = async () => { try { setObservationsAux(''); - updateImagen() + updateImagen(); let data = { - id: drillUpdated.id, - title: drillUpdated.title, - description: drillUpdated.description, + id: selectedDrill.id, + title: title, + description: description, observations: observationsAux, image: imagemBase64, - classPlanId: drillUpdated.classPlanId, + classPlanId: selectedDrill.classPlanId, }; - await drill.updateById(id as string, data); + await drill.updateById(selectedDrill.id, data); + setObservations(observationsAux); toast.success('Observações atualizadas com sucesso'); } catch (error) { toast.error('Erro ao atualizar drill'); @@ -281,7 +246,28 @@ const Drill = () => { setObservationsAux(''); setObservationsNotEdited(true); } - return; + }; + + const redirectToViewPlan = async () => { + try { + console.log('Aqui 1'); + const classPlan = new ClassPlanService(); + const response = await classPlan.getById(selectedDrill.classPlanId); + localStorage.setItem('selectedClassPlan', JSON.stringify(response.data)); + const drillsClassPlan = await drill.getManyByClassPlanId( + selectedDrill.classPlanId, + ); + const drillData = drillsClassPlan.data; + localStorage.setItem( + 'drillsSelectedClassPlan', + JSON.stringify(drillData), + ); + setTimeout(() => { + window.location.href = `/plano-aula/${selectedDrill.classPlanId}`; + }, 3000); + } catch (error) { + toast.error('Erro ao buscar informações do plano'); + } }; return ( @@ -297,10 +283,7 @@ const Drill = () => { /> } /> -
+
{titleNotEdited ? (
@@ -317,7 +300,6 @@ const Drill = () => {
) : (
- { />
)} -
+
-

Descrição

+

Descrição

{descriptionNotEdited ? ( {

) : (