first commit
This commit is contained in:
		
						commit
						174910e6aa
					
				|  | @ -0,0 +1,21 @@ | |||
| module.exports = { | ||||
|   root: true, | ||||
|   env: { browser: true, es2020: true }, | ||||
|   extends: [ | ||||
|     'eslint:recommended', | ||||
|     'plugin:react/recommended', | ||||
|     'plugin:react/jsx-runtime', | ||||
|     'plugin:react-hooks/recommended', | ||||
|   ], | ||||
|   ignorePatterns: ['dist', '.eslintrc.cjs'], | ||||
|   parserOptions: { ecmaVersion: 'latest', sourceType: 'module' }, | ||||
|   settings: { react: { version: '18.2' } }, | ||||
|   plugins: ['react-refresh'], | ||||
|   rules: { | ||||
|     'react/jsx-no-target-blank': 'off', | ||||
|     'react-refresh/only-export-components': [ | ||||
|       'warn', | ||||
|       { allowConstantExport: true }, | ||||
|     ], | ||||
|   }, | ||||
| } | ||||
|  | @ -0,0 +1,24 @@ | |||
| # Logs | ||||
| logs | ||||
| *.log | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| pnpm-debug.log* | ||||
| lerna-debug.log* | ||||
| 
 | ||||
| node_modules | ||||
| dist | ||||
| dist-ssr | ||||
| *.local | ||||
| 
 | ||||
| # Editor directories and files | ||||
| .vscode/* | ||||
| !.vscode/extensions.json | ||||
| .idea | ||||
| .DS_Store | ||||
| *.suo | ||||
| *.ntvs* | ||||
| *.njsproj | ||||
| *.sln | ||||
| *.sw? | ||||
|  | @ -0,0 +1,8 @@ | |||
| # React + Vite | ||||
| 
 | ||||
| This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. | ||||
| 
 | ||||
| Currently, two official plugins are available: | ||||
| 
 | ||||
| - [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh | ||||
| - [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh | ||||
|  | @ -0,0 +1,18 @@ | |||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="UTF-8" /> | ||||
|     <!-- <link rel="icon" type="image/svg+xml" href="/vite.svg" /> --> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|     <link | ||||
|       rel="stylesheet" | ||||
|       href="https://maxcdn.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css" | ||||
|     /> | ||||
| 
 | ||||
|     <title>exampaper.vidh.ai</title> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="root" class="w-100 vh-100"></div> | ||||
|     <script type="module" src="/src/main.jsx"></script> | ||||
|   </body> | ||||
| </html> | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,36 @@ | |||
| { | ||||
|   "name": "vite-project", | ||||
|   "private": true, | ||||
|   "version": "0.0.0", | ||||
|   "type": "module", | ||||
|   "scripts": { | ||||
|     "dev": "vite", | ||||
|     "build": "vite build", | ||||
|     "lint": "eslint . --ext js,jsx --report-unused-disable-directives --max-warnings 0", | ||||
|     "preview": "vite preview" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@emotion/react": "^11.11.4", | ||||
|     "@emotion/styled": "^11.11.5", | ||||
|     "@mui/icons-material": "^5.15.18", | ||||
|     "@mui/material": "^5.15.18", | ||||
|     "antd": "^5.17.3", | ||||
|     "bootstrap": "^5.3.3", | ||||
|     "dotenv": "^16.4.5", | ||||
|     "react": "^18.2.0", | ||||
|     "react-bootstrap": "^2.10.2", | ||||
|     "react-dom": "^18.2.0", | ||||
|     "react-router-dom": "^6.23.1", | ||||
|     "react-toastify": "^10.0.5" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/react": "^18.2.66", | ||||
|     "@types/react-dom": "^18.2.22", | ||||
|     "@vitejs/plugin-react": "^4.2.1", | ||||
|     "eslint": "^8.57.0", | ||||
|     "eslint-plugin-react": "^7.34.1", | ||||
|     "eslint-plugin-react-hooks": "^4.6.0", | ||||
|     "eslint-plugin-react-refresh": "^0.4.6", | ||||
|     "vite": "^5.2.0" | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg> | ||||
| After Width: | Height: | Size: 1.5 KiB | 
|  | @ -0,0 +1,42 @@ | |||
| #root { | ||||
|   /* max-width: 1280px; */ | ||||
|   margin: 0 auto; | ||||
|   /* padding: 1rem; */ | ||||
|   text-align: center; | ||||
| } | ||||
| 
 | ||||
| .logo { | ||||
|   height: 6em; | ||||
|   padding: 1.5em; | ||||
|   will-change: filter; | ||||
|   transition: filter 300ms; | ||||
| } | ||||
| .logo:hover { | ||||
|   filter: drop-shadow(0 0 2em #646cffaa); | ||||
| } | ||||
| .logo.react:hover { | ||||
|   filter: drop-shadow(0 0 2em #61dafbaa); | ||||
| } | ||||
| 
 | ||||
| @keyframes logo-spin { | ||||
|   from { | ||||
|     transform: rotate(0deg); | ||||
|   } | ||||
|   to { | ||||
|     transform: rotate(360deg); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @media (prefers-reduced-motion: no-preference) { | ||||
|   a:nth-of-type(2) .logo { | ||||
|     animation: logo-spin infinite 20s linear; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .card { | ||||
|   padding: 2em; | ||||
| } | ||||
| 
 | ||||
| .read-the-docs { | ||||
|   color: #888; | ||||
| } | ||||
|  | @ -0,0 +1,54 @@ | |||
| import { useState } from "react"; | ||||
| import reactLogo from "./assets/react.svg"; | ||||
| import viteLogo from "/vite.svg"; | ||||
| import "./App.css"; | ||||
| import AttendenceCorrection from "./Components/AttendenceCorrection"; | ||||
| import { BrowserRouter, Routes, Route } from "react-router-dom"; | ||||
| import AnomolyReassigned from "./Components/AnomolyReassigned"; | ||||
| import Home from "./Components/Home"; | ||||
| import ReassignedStats from "./Components/ReassignedStats"; | ||||
| import IndividualExamCentreStats from "./Components/IndividualExamCentreStats"; | ||||
| import PartAReassigned from "./Components/PartAReassigned"; | ||||
| import PartACorrection from "./Components/PartACorrection"; | ||||
| 
 | ||||
| function App() { | ||||
|   return ( | ||||
|     <> | ||||
|       <BrowserRouter> | ||||
|         <Routes> | ||||
|           <Route path="/" element={<Home />}></Route> | ||||
|           <Route | ||||
|             path="/anomoly/reassigned" | ||||
|             element={<AnomolyReassigned />} | ||||
|           ></Route> | ||||
|           <Route | ||||
|             path="/anomoly/reassigned/booklet" | ||||
|             element={<AttendenceCorrection />} | ||||
|           > | ||||
|             </Route> | ||||
|             <Route | ||||
|               path="/anomoly/reassigned/stats" | ||||
|               element={<ReassignedStats />} | ||||
|             > | ||||
|           </Route> | ||||
|           <Route | ||||
|               path="/anomoly/reassigned/stats/:exam_centre_code" | ||||
|               element={<IndividualExamCentreStats/>} | ||||
|             > | ||||
|           </Route> | ||||
|           {/* <Route | ||||
|                path="/anomoly/PartA" | ||||
|                element={<PartAReassigned/>} | ||||
|           > | ||||
|           </Route> | ||||
|           <Route path="/anomoly/partA/booklet" | ||||
|           element={<PartACorrection/>} | ||||
|           > | ||||
|           </Route> */} | ||||
|         </Routes> | ||||
|       </BrowserRouter> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export default App; | ||||
|  | @ -0,0 +1,236 @@ | |||
| import React, { useState } from "react"; | ||||
| import { | ||||
|   DesktopOutlined, | ||||
|   FileOutlined, | ||||
|   PieChartOutlined, | ||||
|   TeamOutlined, | ||||
|   UserOutlined, | ||||
| } from "@ant-design/icons"; | ||||
| import { Breadcrumb, Layout, Menu, Typography, theme } from "antd"; | ||||
| import BookletInput from "./BookletInput"; | ||||
| import { Box, Button } from "@mui/material"; | ||||
| import { useEffect } from "react"; | ||||
| import TextField from "@mui/material/TextField"; | ||||
| import EditButton from "./EditButton"; | ||||
| import { width } from "@mui/system"; | ||||
| import { ToastContainer, toast } from "react-toastify"; | ||||
| import "react-toastify/dist/ReactToastify.css"; | ||||
| import { TablePagination } from "@mui/base/TablePagination"; | ||||
| import TableComponent from "./TableComponent"; | ||||
| import LoadingContainer from "./LoadingContainer"; | ||||
| import HomeIcon from "@mui/icons-material/Home"; | ||||
| import ArrowBackIcon from "@mui/icons-material/ArrowBack"; | ||||
| import { useNavigate } from "react-router-dom"; | ||||
| import QueryStatsIcon from "@mui/icons-material/QueryStats"; | ||||
| 
 | ||||
| const { Header, Content, Footer, Sider } = Layout; | ||||
| function getItem(label, key, icon, children) { | ||||
|   return { | ||||
|     key, | ||||
|     icon, | ||||
|     children, | ||||
|     label, | ||||
|   }; | ||||
| } | ||||
| // const items = [ | ||||
| //   getItem('Option 1', '1', <PieChartOutlined />), | ||||
| //   getItem('Option 2', '2', <DesktopOutlined />), | ||||
| //   getItem('User', 'sub1', <UserOutlined />, [ | ||||
| //     getItem('Tom', '3'), | ||||
| //     getItem('Bill', '4'), | ||||
| //     getItem('Alex', '5'), | ||||
| //   ]), | ||||
| //   getItem('Team', 'sub2', <TeamOutlined />, [getItem('Team 1', '6'), getItem('Team 2', '8')]), | ||||
| //   getItem('Files', '9', <FileOutlined />), | ||||
| // ]; | ||||
| 
 | ||||
| const items = [getItem("Reassigned Booklet No", "1", <PieChartOutlined />)]; | ||||
| 
 | ||||
| const AnomolyReassigned = () => { | ||||
|   const [collapsed, setCollapsed] = useState(false); | ||||
|   const [anomolyData, setAnomolyData] = useState([]); | ||||
|   const [tableRowData, setTableRowData] = useState([]); | ||||
|   const [isLoading, setIsLoading] = useState(false); | ||||
|   const [windowWidth, setWindowWidth] = useState(window.innerWidth); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleResize = () => { | ||||
|       setWindowWidth(window.innerWidth); | ||||
|     }; | ||||
| 
 | ||||
|     window.addEventListener("resize", handleResize); | ||||
| 
 | ||||
|     return () => { | ||||
|       window.removeEventListener("resize", handleResize); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (windowWidth < 800) { | ||||
|       setCollapsed(true); | ||||
|     } | ||||
|     if (windowWidth > 800) { | ||||
|       setCollapsed(false); | ||||
|     } | ||||
|   }, [windowWidth]); | ||||
| 
 | ||||
|   const navigate = useNavigate(); | ||||
|   function createData( | ||||
|     attendence_serial_no, | ||||
|     student_slno, | ||||
|     exam_centre_code, | ||||
|     exam_centre, | ||||
|     student_name, | ||||
|     register_number, | ||||
|     reassigned_serial_no | ||||
|   ) { | ||||
|     return { | ||||
|       attendence_serial_no, | ||||
|       student_slno, | ||||
|       exam_centre_code, | ||||
|       exam_centre, | ||||
|       student_name, | ||||
|       register_number, | ||||
|       reassigned_serial_no, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
| 
 | ||||
|   const fetchAnomalyData = () => { | ||||
|     console.log("Fetching......."); | ||||
|     setIsLoading(true); | ||||
|     fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/fetchAnamolyAttendenceData`, { | ||||
|       method: "GET", | ||||
|       headers: { | ||||
|         "Content-Type": "application/json", | ||||
|       }, | ||||
|     }) | ||||
|       .then((response) => { | ||||
|         console.log("Response fetched.."); | ||||
|         return response.json(); | ||||
|       }) | ||||
|       .then((responseData) => { | ||||
|         console.log("Response Data is : ", responseData); | ||||
|         setIsLoading(false); | ||||
|         if (responseData.status === "success") { | ||||
|           setAnomolyData(responseData.data); | ||||
|           const tmpData = []; | ||||
|           for (const data of responseData.data) { | ||||
|             tmpData.push( | ||||
|               createData( | ||||
|                 data.attendence_serial_no, | ||||
|                 data.student_slno, | ||||
|                 data.exam_centre_code, | ||||
|                 data.exam_centre, | ||||
|                 data.student_name, | ||||
|                 data.register_number, | ||||
|                 data.reassigned_serial_no | ||||
|               ) | ||||
|             ); | ||||
|           } | ||||
|           console.log("Tmp data is : ", tmpData); | ||||
|           if (tmpData.length > 0) { | ||||
|             setTableRowData(tmpData); | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|       .catch((error) => { | ||||
|         console.error("Error fetching data: ", error); | ||||
|         setIsLoading(false); | ||||
|       }); | ||||
|   }; | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchAnomalyData(); | ||||
|   }, []); | ||||
| 
 | ||||
| 
 | ||||
|   const { | ||||
|     token: { colorBgContainer, borderRadiusLG }, | ||||
|   } = theme.useToken(); | ||||
| 
 | ||||
|   return ( | ||||
|     <Layout | ||||
|       style={{ | ||||
|         minHeight: "100vh", | ||||
|       }} | ||||
|     > | ||||
|       <ToastContainer /> | ||||
|       <Sider | ||||
|         collapsible | ||||
|         collapsed={collapsed} | ||||
|         onCollapse={(value) => setCollapsed(value)} | ||||
|       > | ||||
|         <div className="demo-logo-vertical" /> | ||||
|         <Menu | ||||
|           theme="dark" | ||||
|           defaultSelectedKeys={["1"]} | ||||
|           mode="inline" | ||||
|           items={items} | ||||
|         /> | ||||
|       </Sider> | ||||
|       <Layout> | ||||
|         <Header | ||||
|           style={{ | ||||
|             padding: 0, | ||||
|             background: colorBgContainer, | ||||
|           }} | ||||
|         > | ||||
|           <Box className="d-flex justify-content-between h-100 py-1 px-2"> | ||||
|             <Button | ||||
|               className="bg-primary p-1 text-light" | ||||
|               onClick={() => { | ||||
|                 navigate(-1); | ||||
|               }} | ||||
|             > | ||||
|               <ArrowBackIcon /> | ||||
|             </Button> | ||||
|             <Box className="d-flex justify-content-between gap-2"> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/anomoly/reassigned/stats"); | ||||
|                 }} | ||||
|               > | ||||
|                 <QueryStatsIcon /> | ||||
|               </Button> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/"); | ||||
|                 }} | ||||
|               > | ||||
|                 <HomeIcon /> | ||||
|               </Button> | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Header> | ||||
|         <Content | ||||
|           style={{ | ||||
|             margin: "16px 16px", | ||||
|           }} | ||||
|         > | ||||
|           <Box className="w-100 d-flex justify-content-between"> | ||||
|             <Box className="w-100 d-flex justify-content-center"> | ||||
|               {tableRowData.length > 0 && ( | ||||
|                 <TableComponent | ||||
|                   rows={tableRowData} | ||||
|                   type={"AnomolyReassigned"} | ||||
|                 /> | ||||
|               )} | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Content> | ||||
|         <Footer | ||||
|           style={{ | ||||
|             textAlign: "center", | ||||
|           }} | ||||
|         > | ||||
|           exampaper.vidh.ai ©{new Date().getFullYear()} | ||||
|         </Footer> | ||||
|       </Layout> | ||||
|       {isLoading && <LoadingContainer loadingText={"Loading"} />} | ||||
|     </Layout> | ||||
|   ); | ||||
| }; | ||||
| export default AnomolyReassigned; | ||||
|  | @ -0,0 +1,478 @@ | |||
| import React, { useState } from "react"; | ||||
| import { | ||||
|   DesktopOutlined, | ||||
|   FileOutlined, | ||||
|   PieChartOutlined, | ||||
|   TeamOutlined, | ||||
|   UserOutlined, | ||||
| } from "@ant-design/icons"; | ||||
| import { Breadcrumb, Layout, Menu, Typography, theme } from "antd"; | ||||
| import BookletInput from "./BookletInput"; | ||||
| import { Box, Button } from "@mui/material"; | ||||
| import { useEffect } from "react"; | ||||
| import TextField from "@mui/material/TextField"; | ||||
| import EditButton from "./EditButton"; | ||||
| import { width } from "@mui/system"; | ||||
| import { ToastContainer, toast } from "react-toastify"; | ||||
| import "react-toastify/dist/ReactToastify.css"; | ||||
| import { useSearchParams } from "react-router-dom"; | ||||
| import LoadingContainer from "./LoadingContainer"; | ||||
| import HomeIcon from '@mui/icons-material/Home'; | ||||
| import ArrowBackIcon from '@mui/icons-material/ArrowBack'; | ||||
| import { useNavigate } from "react-router-dom"; | ||||
| import QueryStatsIcon from '@mui/icons-material/QueryStats'; | ||||
| 
 | ||||
| 
 | ||||
| const { Header, Content, Footer, Sider } = Layout; | ||||
| function getItem(label, key, icon, children) { | ||||
|   return { | ||||
|     key, | ||||
|     icon, | ||||
|     children, | ||||
|     label, | ||||
|   }; | ||||
| } | ||||
| // const items = [ | ||||
| //   getItem('Option 1', '1', <PieChartOutlined />), | ||||
| //   getItem('Option 2', '2', <DesktopOutlined />), | ||||
| //   getItem('User', 'sub1', <UserOutlined />, [ | ||||
| //     getItem('Tom', '3'), | ||||
| //     getItem('Bill', '4'), | ||||
| //     getItem('Alex', '5'), | ||||
| //   ]), | ||||
| //   getItem('Team', 'sub2', <TeamOutlined />, [getItem('Team 1', '6'), getItem('Team 2', '8')]), | ||||
| //   getItem('Files', '9', <FileOutlined />), | ||||
| // ]; | ||||
| 
 | ||||
| const items = [getItem("Reassigned Booklet No", "1", <PieChartOutlined />)]; | ||||
| 
 | ||||
| const AttendenceCorrection = () => { | ||||
|   const [collapsed, setCollapsed] = useState(false); | ||||
|   const [bookletInput, setBookletInput] = useState(null); | ||||
|   const [dataFetched, setDataFetched] = useState(false); | ||||
|   const [studentData, setStudentData] = useState(null); | ||||
|   const [updateReassigned, setUpdateReassigned] = useState(false); | ||||
|   const [reasssingedSno, setReassignedSno] = useState(null); | ||||
|   const [isLoading, setIsLoading] = useState(false); | ||||
|   const navigate = useNavigate() | ||||
|   let [searchParams, setSearchParams] = useSearchParams(); | ||||
|   const searchParamsBookletSerialNo = searchParams.get("sno"); | ||||
|   console.log("Serach parmas sno : ", searchParamsBookletSerialNo); | ||||
|   const [windowWidth, setWindowWidth] = useState(window.innerWidth); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleResize = () => { | ||||
|       setWindowWidth(window.innerWidth); | ||||
|     }; | ||||
| 
 | ||||
|     window.addEventListener('resize', handleResize); | ||||
| 
 | ||||
|     return () => { | ||||
|       window.removeEventListener('resize', handleResize); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (windowWidth < 800) { | ||||
|       setCollapsed(true) | ||||
|     } | ||||
|     if(windowWidth > 800){ | ||||
|       setCollapsed(false) | ||||
|     } | ||||
|   }, [windowWidth]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (searchParamsBookletSerialNo) { | ||||
|       setBookletInput(searchParamsBookletSerialNo); | ||||
|     } | ||||
|   }, [searchParamsBookletSerialNo]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     submitBookletInput(); | ||||
|   }, [bookletInput]); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   const { | ||||
|     token: { colorBgContainer, borderRadiusLG }, | ||||
|   } = theme.useToken(); | ||||
| 
 | ||||
|   const submitBookletInput = async () => { | ||||
|     setIsLoading(true); | ||||
|     setStudentData(null); | ||||
|     if (!bookletInput) { | ||||
|       console.log("Returning"); | ||||
|       return; | ||||
|     } | ||||
|     const payload = { | ||||
|       bookletInput, | ||||
|     }; | ||||
|     const response = await fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/fetchBookletInfo`, { | ||||
|       method: "POST", | ||||
|       headers: { | ||||
|         "Content-Type": "application/json", | ||||
|       }, | ||||
|       body: JSON.stringify(payload), | ||||
|     }); | ||||
|     const responseData = await response.json(); | ||||
|     console.log("Response Data is : ", responseData); | ||||
|     setDataFetched(true); | ||||
|     setIsLoading(false); | ||||
|     if (responseData.status === "success") { | ||||
|       console.log("Into top if"); | ||||
|       if (!responseData.data) { | ||||
|         setStudentData(null); | ||||
|       } | ||||
|       if (responseData.data) { | ||||
|         setStudentData(responseData.data[0]); | ||||
|       } | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const updateReassignedSno = async () => { | ||||
|     const payload = { | ||||
|       bookletInput, | ||||
|       reasssingedSno, | ||||
|     }; | ||||
|     const response = await fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/updateReassingedSno`, { | ||||
|       method: "POST", | ||||
|       headers: { | ||||
|         "Content-Type": "application/json", | ||||
|       }, | ||||
|       body: JSON.stringify(payload), | ||||
|     }); | ||||
|     const responseData = await response.json(); | ||||
|     if (responseData.status === "success" && responseData?.status_code == 200) { | ||||
|       console.log("Updation success"); | ||||
|       toast.success("Record Updated Successfully !!"); | ||||
|     } else if ( | ||||
|       responseData.status === "success" && | ||||
|       responseData?.status_code == 405 | ||||
|     ) { | ||||
|       console.log("Updation not allowed"); | ||||
|       toast.error("Reassigned Serial No is Invalid !!"); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <Layout | ||||
|       style={{ | ||||
|         minHeight: "100vh", | ||||
|       }} | ||||
|     > | ||||
|       <ToastContainer /> | ||||
|       <Sider | ||||
|         collapsible | ||||
|         collapsed={collapsed} | ||||
|         onCollapse={(value) => setCollapsed(value)} | ||||
|       > | ||||
|         <div className="demo-logo-vertical" /> | ||||
|         <Menu | ||||
|           theme="dark" | ||||
|           defaultSelectedKeys={["1"]} | ||||
|           mode="inline" | ||||
|           items={items} | ||||
|         /> | ||||
|       </Sider> | ||||
|       <Layout> | ||||
|         <Header | ||||
|           style={{ | ||||
|             padding: 0, | ||||
|             background: colorBgContainer, | ||||
|           }} | ||||
|         > | ||||
|            <Box className="d-flex justify-content-between h-100 py-1 px-2"> | ||||
|             <Button | ||||
|               className="bg-primary p-1 text-light" | ||||
|               onClick={() => { | ||||
|                 navigate(-1); | ||||
|               }} | ||||
|             > | ||||
|               <ArrowBackIcon /> | ||||
|             </Button> | ||||
|             <Box className="d-flex justify-content-between gap-2"> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/anomoly/reassigned/stats"); | ||||
|                 }} | ||||
|               > | ||||
|                 <QueryStatsIcon /> | ||||
|               </Button> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/"); | ||||
|                 }} | ||||
|               > | ||||
|                 <HomeIcon /> | ||||
|               </Button> | ||||
|             </Box> | ||||
|           </Box> | ||||
|           {/* <Box>Reassigned Booklet Serial Manual Updation :</Box> */} | ||||
|         </Header> | ||||
|         <Content | ||||
|           style={{ | ||||
|             margin: "16px 16px", | ||||
|           }} | ||||
|         > | ||||
|           {/* <Breadcrumb | ||||
|             style={{ | ||||
|               margin: "16px 0", | ||||
|             }} | ||||
|           > | ||||
|             <Breadcrumb.Item>User</Breadcrumb.Item> | ||||
|             <Breadcrumb.Item>Bill</Breadcrumb.Item> | ||||
|           </Breadcrumb> */} | ||||
|           {/* <div | ||||
|             style={{ | ||||
|               padding: 24, | ||||
|               minHeight: 360, | ||||
|               background: colorBgContainer, | ||||
|               borderRadius: borderRadiusLG, | ||||
|             }} | ||||
|           > | ||||
|             Bill is a cat. | ||||
|           </div> */} | ||||
|           <Box className="w-100 d-flex flex-column flex-md-row justify-content-between"> | ||||
|             <Box className="w-25 d-none d-md-flex flex-column"> | ||||
|               <Box className="d-flex justify-content-between"> | ||||
|                 <BookletInput | ||||
|                   className="d-flex flex-1" | ||||
|                   setBookletInput={setBookletInput} | ||||
|                   bookletInput={bookletInput} | ||||
|                   setDataFetched={setDataFetched} | ||||
|                 /> | ||||
|                 <Button | ||||
|                   className="px-md-5 px-2 mx-1 mx-md-2 text-light bg-primary" | ||||
|                   onClick={submitBookletInput} | ||||
|                 > | ||||
|                   Submit | ||||
|                 </Button> | ||||
|               </Box> | ||||
|               {!isLoading && !studentData && ( | ||||
|                 <Box className="w-100 py-5"> | ||||
|                   <h6>Invalid Booklet Serial No !!</h6> | ||||
|                 </Box> | ||||
|               )} | ||||
|               {dataFetched && studentData && ( | ||||
|                 <> | ||||
|                   <Box> | ||||
|                     <Box className="d-flex justify-content-start px-2 py-3"> | ||||
|                       <h6>Booklet Info:</h6> | ||||
|                     </Box> | ||||
| 
 | ||||
|                     <Box className="px-2"> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="student-name-input">Student Name:</label> | ||||
|                         <TextField | ||||
|                           id="student-name-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.student_name} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-code-input"> | ||||
|                           Exam Center Code: | ||||
|                         </label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-code-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.exam_centre_code} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-input">Exam Centre:</label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.exam_centre} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-date-input">Exam Date:</label> | ||||
|                         <TextField | ||||
|                           id="exam-date-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.exam_date} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="reassigned-serial-no-input"> | ||||
|                           Reassigned Serial No: | ||||
|                         </label> | ||||
|                         <Box className="d-flex justify-content-between w-100"> | ||||
|                           <TextField | ||||
|                             id="reassigned-serial-no-input" | ||||
|                             disabled={!updateReassigned} | ||||
|                             className="w-100" | ||||
|                             value={studentData.reassigned_serial_no} | ||||
|                             autoComplete="off" | ||||
|                             onChange={(e) => { | ||||
|                               setReassignedSno(e.target.value); | ||||
|                             }} | ||||
|                           /> | ||||
|                           <Button | ||||
|                             onClick={() => { | ||||
|                               setUpdateReassigned(!updateReassigned); | ||||
|                             }} | ||||
|                           > | ||||
|                             <EditButton /> | ||||
|                           </Button> | ||||
|                         </Box> | ||||
|                       </Box> | ||||
|                       <Box className="py-2 d-flex justify-content-start"> | ||||
|                         <Button | ||||
|                           className="text-light bg-primary p-3" | ||||
|                           disabled={!updateReassigned} | ||||
|                           onClick={updateReassignedSno} | ||||
|                         > | ||||
|                           Update | ||||
|                         </Button> | ||||
|                       </Box> | ||||
|                     </Box> | ||||
|                   </Box> | ||||
|                 </> | ||||
|               )} | ||||
|             </Box> | ||||
|             <Box className="w-md-25 d-flex d-md-none flex-column"> | ||||
|               <Box className="d-flex justify-content-between"> | ||||
|                 <BookletInput | ||||
|                   className="d-flex flex-1" | ||||
|                   setBookletInput={setBookletInput} | ||||
|                   bookletInput={bookletInput} | ||||
|                   setDataFetched={setDataFetched} | ||||
|                 /> | ||||
|                 <Button | ||||
|                   className="px-md-5 px-2 mx-1 mx-md-2 text-light bg-primary" | ||||
|                   onClick={submitBookletInput} | ||||
|                 > | ||||
|                   Submit | ||||
|                 </Button> | ||||
|               </Box> | ||||
|               {!isLoading && !studentData && ( | ||||
|                 <Box className="w-100 py-5"> | ||||
|                   <h6>Invalid Booklet Serial No !!</h6> | ||||
|                 </Box> | ||||
|               )} | ||||
|               {dataFetched && studentData && ( | ||||
|                 <> | ||||
|                   <Box> | ||||
|                     <Box className="d-flex justify-content-start px-2 py-3"> | ||||
|                       <h6>Booklet Info:</h6> | ||||
|                     </Box> | ||||
| 
 | ||||
|                     <Box className="px-2"> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="student-name-input">Student Name:</label> | ||||
|                         <TextField | ||||
|                           id="student-name-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.student_name} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-code-input"> | ||||
|                           Exam Center Code: | ||||
|                         </label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-code-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.exam_centre_code} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-input">Exam Centre:</label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.exam_centre} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-date-input">Exam Date:</label> | ||||
|                         <TextField | ||||
|                           id="exam-date-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.exam_date} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="reassigned-serial-no-input"> | ||||
|                           Reassigned Serial No: | ||||
|                         </label> | ||||
|                         <Box className="d-flex justify-content-between w-100"> | ||||
|                           <TextField | ||||
|                             id="reassigned-serial-no-input" | ||||
|                             disabled={!updateReassigned} | ||||
|                             className="w-100" | ||||
|                             value={studentData.reassigned_serial_no} | ||||
|                             autoComplete="off" | ||||
|                             onChange={(e) => { | ||||
|                               setReassignedSno(e.target.value); | ||||
|                             }} | ||||
|                           /> | ||||
|                           <Button | ||||
|                             onClick={() => { | ||||
|                               setUpdateReassigned(!updateReassigned); | ||||
|                             }} | ||||
|                           > | ||||
|                             <EditButton /> | ||||
|                           </Button> | ||||
|                         </Box> | ||||
|                       </Box> | ||||
|                       <Box className="py-2 d-flex justify-content-start"> | ||||
|                         <Button | ||||
|                           className="text-light bg-primary p-3" | ||||
|                           disabled={!updateReassigned} | ||||
|                           onClick={updateReassignedSno} | ||||
|                         > | ||||
|                           Update | ||||
|                         </Button> | ||||
|                       </Box> | ||||
|                     </Box> | ||||
|                   </Box> | ||||
|                 </> | ||||
|               )} | ||||
|             </Box> | ||||
|             <Box className="w-75 d-none d-md-block"> | ||||
|               {dataFetched && studentData && ( | ||||
|                 <img | ||||
|                   src={`https://docs.exampaper.vidh.ai/${studentData.s3_path}`} | ||||
|                   width="60%" | ||||
|                   height="auto" | ||||
|                 /> | ||||
|               )} | ||||
|             </Box> | ||||
|             <Box className="w-md-75 d-block d-md-none"> | ||||
|               {dataFetched && studentData && ( | ||||
|                 <img | ||||
|                   src={`https://docs.exampaper.vidh.ai/${studentData.s3_path}`} | ||||
|                   width="60%" | ||||
|                   height="auto" | ||||
|                 /> | ||||
|               )} | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Content> | ||||
|         <Footer | ||||
|           style={{ | ||||
|             textAlign: "center", | ||||
|           }} | ||||
|         > | ||||
|           exampaper.vidh.ai ©{new Date().getFullYear()} | ||||
|         </Footer> | ||||
|       </Layout> | ||||
|       {isLoading && <LoadingContainer loadingText={"Fetching"} />} | ||||
|     </Layout> | ||||
|   ); | ||||
| }; | ||||
| export default AttendenceCorrection; | ||||
|  | @ -0,0 +1,23 @@ | |||
| import * as React from 'react'; | ||||
| import Box from '@mui/material/Box'; | ||||
| import TextField from '@mui/material/TextField'; | ||||
| 
 | ||||
| const BarcodeInput = ({setBookletInput,setDataFetched,bookletInput}) =>{ | ||||
|     return ( | ||||
|         <Box | ||||
|         //   component="form" | ||||
|         //   sx={{ | ||||
|         //     '& > :not(style)': { m: 1, width: '25ch' }, | ||||
|         //   }} | ||||
|         //   noValidate | ||||
|           autoComplete="off" | ||||
|         > | ||||
|           <TextField  id="outlined-basic" label="Barcode" variant="outlined" className='w-100' autoComplete="off" value={bookletInput} onChange={(e)=>{ | ||||
|             setBookletInput(e.target.value) | ||||
|             setDataFetched(false) | ||||
|             }}/> | ||||
|         </Box> | ||||
|       );     | ||||
| } | ||||
| 
 | ||||
| export default BarcodeInput; | ||||
|  | @ -0,0 +1,23 @@ | |||
| import * as React from 'react'; | ||||
| import Box from '@mui/material/Box'; | ||||
| import TextField from '@mui/material/TextField'; | ||||
| 
 | ||||
| const BookletInput = ({setBookletInput,setDataFetched,bookletInput}) =>{ | ||||
|     return ( | ||||
|         <Box | ||||
|         //   component="form" | ||||
|         //   sx={{ | ||||
|         //     '& > :not(style)': { m: 1, width: '25ch' }, | ||||
|         //   }} | ||||
|         //   noValidate | ||||
|           autoComplete="off" | ||||
|         > | ||||
|           <TextField  id="outlined-basic" label="Booklet Serial No" variant="outlined" className='w-100' autoComplete="off" value={bookletInput} onChange={(e)=>{ | ||||
|             setBookletInput(e.target.value) | ||||
|             setDataFetched(false) | ||||
|             }}/> | ||||
|         </Box> | ||||
|       );     | ||||
| } | ||||
| 
 | ||||
| export default BookletInput; | ||||
|  | @ -0,0 +1,9 @@ | |||
| import EditIcon from '@mui/icons-material/Edit'; | ||||
| 
 | ||||
| const EditButton = () =>{ | ||||
|  return( | ||||
|     <EditIcon /> | ||||
|  ) | ||||
| } | ||||
| 
 | ||||
| export default EditButton | ||||
|  | @ -0,0 +1,37 @@ | |||
| import { Box } from "@mui/material"; | ||||
| import HomepageCard from "./HomepageCard"; | ||||
| 
 | ||||
| 
 | ||||
| const Home = () => { | ||||
|   // const cards = [ | ||||
|   //   { | ||||
|   //     title: "Reassingned Serial No Anomoly Manual Updation", | ||||
|   //     url: "/anomoly/reassigned", | ||||
|   //   }, | ||||
|   //   { | ||||
|   //     title: "Part A OCR Anomoly Manual Updation", | ||||
|   //     url: "/anomoly/partA", | ||||
|   //   } | ||||
|   // ]; | ||||
|   const cards = [ | ||||
|     { | ||||
|       title: "Reassingned Serial No Anomoly Manual Updation", | ||||
|       url: "/anomoly/reassigned", | ||||
|     }] | ||||
|   return ( | ||||
|     <> | ||||
|       <Box> | ||||
|         <Box className="d-flex justify-content-center text-light bg-primary rounded py-3"> | ||||
|           <h1>Welcome to exampaper.vidh.ai</h1> | ||||
|         </Box> | ||||
|         <Box className="p-3"> | ||||
|           {cards.map((card) => ( | ||||
|             <HomepageCard title={card?.title} url={card?.url} /> | ||||
|           ))} | ||||
|         </Box> | ||||
|       </Box> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export default Home; | ||||
|  | @ -0,0 +1,23 @@ | |||
| import React from "react"; | ||||
| import { Card, Col, Row } from "antd"; | ||||
| import { useNavigate } from "react-router"; | ||||
| 
 | ||||
| const HomepageCard = ({ title, url }) => { | ||||
|   const navigate = useNavigate(); | ||||
|   return ( | ||||
|     <Row | ||||
|       gutter={16} | ||||
|       className="p-2" | ||||
|       onClick={() => { | ||||
|         navigate(url); | ||||
|       }} | ||||
|     > | ||||
|       <Col span={12}> | ||||
|         <Card title="" className="shadow" bordered={false}> | ||||
|           {title} | ||||
|         </Card> | ||||
|       </Col> | ||||
|     </Row> | ||||
|   ); | ||||
| }; | ||||
| export default HomepageCard; | ||||
|  | @ -0,0 +1,219 @@ | |||
| import React, { useState } from "react"; | ||||
| import { | ||||
|   DesktopOutlined, | ||||
|   FileOutlined, | ||||
|   PieChartOutlined, | ||||
|   TeamOutlined, | ||||
|   UserOutlined, | ||||
| } from "@ant-design/icons"; | ||||
| import { Breadcrumb, Layout, Menu, Typography, theme } from "antd"; | ||||
| import BookletInput from "./BookletInput"; | ||||
| import { Box, Button } from "@mui/material"; | ||||
| import { useEffect } from "react"; | ||||
| import TextField from "@mui/material/TextField"; | ||||
| import EditButton from "./EditButton"; | ||||
| import { width } from "@mui/system"; | ||||
| import { ToastContainer, toast } from "react-toastify"; | ||||
| import "react-toastify/dist/ReactToastify.css"; | ||||
| import { TablePagination } from "@mui/base/TablePagination"; | ||||
| import TableComponent from "./TableComponent"; | ||||
| import LoadingContainer from "./LoadingContainer"; | ||||
| import HomeIcon from "@mui/icons-material/Home"; | ||||
| import ArrowBackIcon from "@mui/icons-material/ArrowBack"; | ||||
| import { useNavigate } from "react-router-dom"; | ||||
| import QueryStatsIcon from "@mui/icons-material/QueryStats"; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| 
 | ||||
| 
 | ||||
| const { Header, Content, Footer, Sider } = Layout; | ||||
| function getItem(label, key, icon, children) { | ||||
|   return { | ||||
|     key, | ||||
|     icon, | ||||
|     children, | ||||
|     label, | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| const items = [getItem("Reassigned Booklet No", "1", <PieChartOutlined />)]; | ||||
| 
 | ||||
| const IndividualExamCentreStats = () => { | ||||
|     console.log("BACKEND URL: ", import.meta.env.VITE_REACT_APP_BACKEND_URL); | ||||
|   const [collapsed, setCollapsed] = useState(false); | ||||
|   const [anomolyData, setAnomolyData] = useState([]); | ||||
|   const [tableRowData, setTableRowData] = useState([]); | ||||
|   const [isLoading, setIsLoading] = useState(false); | ||||
|   const { exam_centre_code } = useParams(); | ||||
|   const navigate = useNavigate(); | ||||
|   const [windowWidth, setWindowWidth] = useState(window.innerWidth); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleResize = () => { | ||||
|       setWindowWidth(window.innerWidth); | ||||
|     }; | ||||
| 
 | ||||
|     window.addEventListener('resize', handleResize); | ||||
| 
 | ||||
|     return () => { | ||||
|       window.removeEventListener('resize', handleResize); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (windowWidth < 800) { | ||||
|       setCollapsed(true) | ||||
|     } | ||||
|     if(windowWidth > 800){ | ||||
|       setCollapsed(false) | ||||
|     } | ||||
|   }, [windowWidth]); | ||||
| 
 | ||||
| 
 | ||||
|   function createData( | ||||
|     exam_centre_code, | ||||
|     exam_date, | ||||
|     manual_verification_needed_count, | ||||
|     subject_code, | ||||
|     total_count | ||||
|   ) { | ||||
|     return { | ||||
|       exam_centre_code, | ||||
|       exam_date, | ||||
|       manual_verification_needed_count, | ||||
|       subject_code, | ||||
|       total_count | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
| 
 | ||||
|   const fetchIndividualReassignedStats = () => { | ||||
|    useEffect(() => { | ||||
|      setIsLoading(true); | ||||
|      const payload = { | ||||
|         exam_centre_code, | ||||
|      } | ||||
| 
 | ||||
|      fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/fetchIndividualReassignedStats`, { | ||||
|        method: "POST", | ||||
|        headers: { | ||||
|          "Content-Type": "application/json", | ||||
|        }, | ||||
|        body:JSON.stringify(payload) | ||||
|      }) | ||||
|        .then((response) => response.json()) | ||||
|        .then((responseData) => { | ||||
|          console.log("Response Data is : ", responseData); | ||||
|          setIsLoading(false); | ||||
|          if (responseData.status === "success") { | ||||
|             setAnomolyData(responseData?.data); | ||||
|             const tmpData = []; | ||||
|             for (var data of responseData?.data) { | ||||
|               // console.log("Student name : ",data?.student_name) | ||||
|               tmpData.push( | ||||
|                 createData( | ||||
|                   data?.exam_centre_code, | ||||
|                   data?.exam_date, | ||||
|                   data?.manual_verification_needed_count, | ||||
|                   data?.subject_code, | ||||
|                   data?.total_count | ||||
|                 ) | ||||
|               ); | ||||
|             } | ||||
|             console.log("Tmp data is : ", tmpData); | ||||
|             if (tmpData.length > 0) { | ||||
|               setTableRowData(tmpData); | ||||
|             } | ||||
|           } | ||||
|        }); | ||||
|    }, []); | ||||
|  }; | ||||
| fetchIndividualReassignedStats(); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   const { | ||||
|     token: { colorBgContainer, borderRadiusLG }, | ||||
|   } = theme.useToken(); | ||||
| 
 | ||||
|   return ( | ||||
|     <Layout | ||||
|       style={{ | ||||
|         minHeight: "100vh", | ||||
|       }} | ||||
|     > | ||||
|       <ToastContainer /> | ||||
|       <Sider | ||||
|         collapsible | ||||
|         collapsed={collapsed} | ||||
|         onCollapse={(value) => setCollapsed(value)} | ||||
|       > | ||||
|         <div className="demo-logo-vertical" /> | ||||
|         <Menu | ||||
|           theme="dark" | ||||
|           defaultSelectedKeys={["1"]} | ||||
|           mode="inline" | ||||
|           items={items} | ||||
|         /> | ||||
|       </Sider> | ||||
|       <Layout> | ||||
|         <Header | ||||
|           style={{ | ||||
|             padding: 0, | ||||
|             background: colorBgContainer, | ||||
|           }} | ||||
|         > | ||||
|           <Box className="d-flex justify-content-between h-100 py-1 px-2"> | ||||
|             <Button | ||||
|               className="bg-primary p-1 text-light" | ||||
|               onClick={() => { | ||||
|                 navigate(-1); | ||||
|               }} | ||||
|             > | ||||
|               <ArrowBackIcon /> | ||||
|             </Button> | ||||
|             <Box className="d-flex justify-content-between gap-2"> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/anomoly/reassigned/stats"); | ||||
|                 }} | ||||
|               > | ||||
|                 <QueryStatsIcon /> | ||||
|               </Button> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/"); | ||||
|                 }} | ||||
|               > | ||||
|                 <HomeIcon /> | ||||
|               </Button> | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Header> | ||||
|         <Content | ||||
|           style={{ | ||||
|             margin: "16px 16px", | ||||
|           }} | ||||
|         > | ||||
|           <Box className="w-100 d-flex justify-content-between"> | ||||
|             <Box className="w-100 d-flex justify-content-center"> | ||||
|               {tableRowData.length > 0 && ( | ||||
|                 <TableComponent rows={tableRowData} type={"IndividualReassignedStats"}/> | ||||
|               )} | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Content> | ||||
|         <Footer | ||||
|           style={{ | ||||
|             textAlign: "center", | ||||
|           }} | ||||
|         > | ||||
|           exampaper.vidh.ai ©{new Date().getFullYear()} | ||||
|         </Footer> | ||||
|       </Layout> | ||||
|       {isLoading && <LoadingContainer loadingText={"Loading"} />} | ||||
|     </Layout> | ||||
|   ); | ||||
| }; | ||||
| export default IndividualExamCentreStats; | ||||
|  | @ -0,0 +1,25 @@ | |||
| import React from "react"; | ||||
| import { Box } from "@mui/material"; | ||||
| import Spinner from "react-bootstrap/Spinner"; | ||||
| 
 | ||||
| const LoadingContainer = ({ loadingText }) => { | ||||
|   const LoadingContainerStyle = { | ||||
|     backgroundColor: "rgba(0,0,0,0.5)", | ||||
|     overflow: "auto", | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <> | ||||
|       <Box | ||||
|         className="d-flex justify-content-center position-absolute w-100 h-100 align-items-center" | ||||
|         style={LoadingContainerStyle} | ||||
|       > | ||||
|         <Spinner animation="border" variant="light" role="status"> | ||||
|           <span className="visually-hidden">Loading...</span> | ||||
|         </Spinner> | ||||
|       </Box> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export default LoadingContainer; | ||||
|  | @ -0,0 +1,480 @@ | |||
| import React, { useState } from "react"; | ||||
| import { | ||||
|   DesktopOutlined, | ||||
|   FileOutlined, | ||||
|   PieChartOutlined, | ||||
|   TeamOutlined, | ||||
|   UserOutlined, | ||||
| } from "@ant-design/icons"; | ||||
| import { Breadcrumb, Layout, Menu, Typography, theme } from "antd"; | ||||
| import BookletInput from "./BookletInput"; | ||||
| import BarcodeInput from "./BarcodeInput"; | ||||
| import { Box, Button } from "@mui/material"; | ||||
| import { useEffect } from "react"; | ||||
| import TextField from "@mui/material/TextField"; | ||||
| import EditButton from "./EditButton"; | ||||
| import { width } from "@mui/system"; | ||||
| import { ToastContainer, toast } from "react-toastify"; | ||||
| import "react-toastify/dist/ReactToastify.css"; | ||||
| import { useSearchParams } from "react-router-dom"; | ||||
| import LoadingContainer from "./LoadingContainer"; | ||||
| import HomeIcon from '@mui/icons-material/Home'; | ||||
| import ArrowBackIcon from '@mui/icons-material/ArrowBack'; | ||||
| import { useNavigate } from "react-router-dom"; | ||||
| import QueryStatsIcon from '@mui/icons-material/QueryStats'; | ||||
| 
 | ||||
| 
 | ||||
| const { Header, Content, Footer, Sider } = Layout; | ||||
| function getItem(label, key, icon, children) { | ||||
|   return { | ||||
|     key, | ||||
|     icon, | ||||
|     children, | ||||
|     label, | ||||
|   }; | ||||
| } | ||||
| // const items = [ | ||||
| //   getItem('Option 1', '1', <PieChartOutlined />), | ||||
| //   getItem('Option 2', '2', <DesktopOutlined />), | ||||
| //   getItem('User', 'sub1', <UserOutlined />, [ | ||||
| //     getItem('Tom', '3'), | ||||
| //     getItem('Bill', '4'), | ||||
| //     getItem('Alex', '5'), | ||||
| //   ]), | ||||
| //   getItem('Team', 'sub2', <TeamOutlined />, [getItem('Team 1', '6'), getItem('Team 2', '8')]), | ||||
| //   getItem('Files', '9', <FileOutlined />), | ||||
| // ]; | ||||
| 
 | ||||
| const items = [getItem("Reassigned Booklet No", "1", <PieChartOutlined />)]; | ||||
| 
 | ||||
| const PartACorrection = () => { | ||||
|   const [collapsed, setCollapsed] = useState(false); | ||||
|   const [bookletInput, setBookletInput] = useState(null); | ||||
|   const [dataFetched, setDataFetched] = useState(false); | ||||
|   const [studentData, setStudentData] = useState(null); | ||||
|   const [updateReassigned, setUpdateReassigned] = useState(false); | ||||
|   const [reasssingedSno, setReassignedSno] = useState(null); | ||||
|   const [isLoading, setIsLoading] = useState(false); | ||||
|   const navigate = useNavigate() | ||||
|   let [searchParams, setSearchParams] = useSearchParams(); | ||||
|   const searchParamsBarcode = searchParams.get("barcode"); | ||||
|   console.log("Serach parmas sno : ", searchParamsBarcode); | ||||
|   const [windowWidth, setWindowWidth] = useState(window.innerWidth); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleResize = () => { | ||||
|       setWindowWidth(window.innerWidth); | ||||
|     }; | ||||
| 
 | ||||
|     window.addEventListener('resize', handleResize); | ||||
| 
 | ||||
|     return () => { | ||||
|       window.removeEventListener('resize', handleResize); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (windowWidth < 800) { | ||||
|       setCollapsed(true) | ||||
|     } | ||||
|     if(windowWidth > 800){ | ||||
|       setCollapsed(false) | ||||
|     } | ||||
|   }, [windowWidth]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (searchParamsBarcode) { | ||||
|       setBookletInput(searchParamsBarcode); | ||||
|     } | ||||
|   }, [searchParamsBarcode]); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     submitBookletInput(); | ||||
|   }, [bookletInput]); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   const { | ||||
|     token: { colorBgContainer, borderRadiusLG }, | ||||
|   } = theme.useToken(); | ||||
| 
 | ||||
|   const submitBookletInput = async () => { | ||||
|     setIsLoading(true); | ||||
|     setStudentData(null); | ||||
|     if (!bookletInput) { | ||||
|       console.log("Returning"); | ||||
|       return; | ||||
|     } | ||||
|     const payload = { | ||||
|       bookletInput, | ||||
|     }; | ||||
|     const response = await fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/fetchBarcodeInfo`, { | ||||
|       method: "POST", | ||||
|       headers: { | ||||
|         "Content-Type": "application/json", | ||||
|       }, | ||||
|       body: JSON.stringify(payload), | ||||
|     }); | ||||
|     const responseData = await response.json(); | ||||
|     console.log("Response Data is : ", responseData); | ||||
|     setDataFetched(true); | ||||
|     setIsLoading(false); | ||||
|     if (responseData.status === "success") { | ||||
|       console.log("Into top if"); | ||||
|       if (!responseData.data) { | ||||
|         setStudentData(null); | ||||
|       } | ||||
|       if (responseData.data) { | ||||
|         console.log("Students Data :",responseData.data[0]) | ||||
|         setStudentData(responseData.data[0]); | ||||
|       } | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   const updateReassignedSno = async () => { | ||||
|     const payload = { | ||||
|       bookletInput, | ||||
|       reasssingedSno, | ||||
|     }; | ||||
|     const response = await fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/updateReassingedSno`, { | ||||
|       method: "POST", | ||||
|       headers: { | ||||
|         "Content-Type": "application/json", | ||||
|       }, | ||||
|       body: JSON.stringify(payload), | ||||
|     }); | ||||
|     const responseData = await response.json(); | ||||
|     if (responseData.status === "success" && responseData?.status_code == 200) { | ||||
|       console.log("Updation success"); | ||||
|       toast.success("Record Updated Successfully !!"); | ||||
|     } else if ( | ||||
|       responseData.status === "success" && | ||||
|       responseData?.status_code == 405 | ||||
|     ) { | ||||
|       console.log("Updation not allowed"); | ||||
|       toast.error("Reassigned Serial No is Invalid !!"); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <Layout | ||||
|       style={{ | ||||
|         minHeight: "100vh", | ||||
|       }} | ||||
|     > | ||||
|       <ToastContainer /> | ||||
|       <Sider | ||||
|         collapsible | ||||
|         collapsed={collapsed} | ||||
|         onCollapse={(value) => setCollapsed(value)} | ||||
|       > | ||||
|         <div className="demo-logo-vertical" /> | ||||
|         <Menu | ||||
|           theme="dark" | ||||
|           defaultSelectedKeys={["1"]} | ||||
|           mode="inline" | ||||
|           items={items} | ||||
|         /> | ||||
|       </Sider> | ||||
|       <Layout> | ||||
|         <Header | ||||
|           style={{ | ||||
|             padding: 0, | ||||
|             background: colorBgContainer, | ||||
|           }} | ||||
|         > | ||||
|            <Box className="d-flex justify-content-between h-100 py-1 px-2"> | ||||
|             <Button | ||||
|               className="bg-primary p-1 text-light" | ||||
|               onClick={() => { | ||||
|                 navigate(-1); | ||||
|               }} | ||||
|             > | ||||
|               <ArrowBackIcon /> | ||||
|             </Button> | ||||
|             <Box className="d-flex justify-content-between gap-2"> | ||||
|               {/* <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/anomoly/reassigned/stats"); | ||||
|                 }} | ||||
|               > | ||||
|                 <QueryStatsIcon /> | ||||
|               </Button> */} | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/"); | ||||
|                 }} | ||||
|               > | ||||
|                 <HomeIcon /> | ||||
|               </Button> | ||||
|             </Box> | ||||
|           </Box> | ||||
|           {/* <Box>Reassigned Booklet Serial Manual Updation :</Box> */} | ||||
|         </Header> | ||||
|         <Content | ||||
|           style={{ | ||||
|             margin: "16px 16px", | ||||
|           }} | ||||
|         > | ||||
|           {/* <Breadcrumb | ||||
|             style={{ | ||||
|               margin: "16px 0", | ||||
|             }} | ||||
|           > | ||||
|             <Breadcrumb.Item>User</Breadcrumb.Item> | ||||
|             <Breadcrumb.Item>Bill</Breadcrumb.Item> | ||||
|           </Breadcrumb> */} | ||||
|           {/* <div | ||||
|             style={{ | ||||
|               padding: 24, | ||||
|               minHeight: 360, | ||||
|               background: colorBgContainer, | ||||
|               borderRadius: borderRadiusLG, | ||||
|             }} | ||||
|           > | ||||
|             Bill is a cat. | ||||
|           </div> */} | ||||
|           <Box className="w-100 d-flex flex-column flex-md-row justify-content-between"> | ||||
|             <Box className="w-25 d-none d-md-flex flex-column"> | ||||
|               <Box className="d-flex justify-content-between"> | ||||
|                 <BarcodeInput | ||||
|                   className="d-flex flex-1" | ||||
|                   setBookletInput={setBookletInput} | ||||
|                   bookletInput={bookletInput} | ||||
|                   setDataFetched={setDataFetched} | ||||
|                 /> | ||||
|                 <Button | ||||
|                   className="px-md-5 px-2 mx-1 mx-md-2 text-light bg-primary" | ||||
|                   onClick={submitBookletInput} | ||||
|                 > | ||||
|                   Submit | ||||
|                 </Button> | ||||
|               </Box> | ||||
|               {!isLoading && !studentData && ( | ||||
|                 <Box className="w-100 py-5"> | ||||
|                   <h6>Invalid Booklet Serial No !!</h6> | ||||
|                 </Box> | ||||
|               )} | ||||
|               {dataFetched && studentData && ( | ||||
|                 <> | ||||
|                   <Box> | ||||
|                     <Box className="d-flex justify-content-start px-2 py-3"> | ||||
|                       <h6>Barcode Info:</h6> | ||||
|                     </Box> | ||||
| 
 | ||||
|                     <Box className="px-2"> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="student-name-input">Register Number:</label> | ||||
|                         <TextField | ||||
|                           id="student-name-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.register_number} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-code-input"> | ||||
|                           Subject Code: | ||||
|                         </label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-code-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.subject_code} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-input">Type:</label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.type} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       {/* <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-date-input">Exam Date:</label> | ||||
|                         <TextField | ||||
|                           id="exam-date-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.ref_exam_date} | ||||
|                         /> | ||||
|                       </Box> */} | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="reassigned-serial-no-input"> | ||||
|                           Reassigned Serial No: | ||||
|                         </label> | ||||
|                         <Box className="d-flex justify-content-between w-100"> | ||||
|                           <TextField | ||||
|                             id="reassigned-serial-no-input" | ||||
|                             disabled={!updateReassigned} | ||||
|                             className="w-100" | ||||
|                             value={studentData.reassigned_serial_no} | ||||
|                             autoComplete="off" | ||||
|                             onChange={(e) => { | ||||
|                               setReassignedSno(e.target.value); | ||||
|                             }} | ||||
|                           /> | ||||
|                           <Button | ||||
|                             onClick={() => { | ||||
|                               setUpdateReassigned(!updateReassigned); | ||||
|                             }} | ||||
|                           > | ||||
|                             <EditButton /> | ||||
|                           </Button> | ||||
|                         </Box> | ||||
|                       </Box> | ||||
|                       <Box className="py-2 d-flex justify-content-start"> | ||||
|                         <Button | ||||
|                           className="text-light bg-primary p-3" | ||||
|                           disabled={!updateReassigned} | ||||
|                           onClick={updateReassignedSno} | ||||
|                         > | ||||
|                           Update | ||||
|                         </Button> | ||||
|                       </Box> | ||||
|                     </Box> | ||||
|                   </Box> | ||||
|                 </> | ||||
|               )} | ||||
|             </Box> | ||||
|             <Box className="w-md-25 d-flex d-md-none flex-column"> | ||||
|               <Box className="d-flex justify-content-between"> | ||||
|                 <BarcodeInput | ||||
|                   className="d-flex flex-1" | ||||
|                   setBookletInput={setBookletInput} | ||||
|                   bookletInput={bookletInput} | ||||
|                   setDataFetched={setDataFetched} | ||||
|                 /> | ||||
|                 <Button | ||||
|                   className="px-md-5 px-2 mx-1 mx-md-2 text-light bg-primary" | ||||
|                   onClick={submitBookletInput} | ||||
|                 > | ||||
|                   Submit | ||||
|                 </Button> | ||||
|               </Box> | ||||
|               {!isLoading && !studentData && ( | ||||
|                 <Box className="w-100 py-5"> | ||||
|                   <h6>Invalid Booklet Serial No !!</h6> | ||||
|                 </Box> | ||||
|               )} | ||||
|               {dataFetched && studentData && ( | ||||
|                 <> | ||||
|                   <Box> | ||||
|                     <Box className="d-flex justify-content-start px-2 py-3"> | ||||
|                       <h6>Barcode Info:</h6> | ||||
|                     </Box> | ||||
| 
 | ||||
|                     <Box className="px-2"> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="student-name-input">Register Number:</label> | ||||
|                         <TextField | ||||
|                           id="student-name-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.register_number} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-code-input"> | ||||
|                           Subject Code: | ||||
|                         </label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-code-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.subject_code} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-centre-input">Type:</label> | ||||
|                         <TextField | ||||
|                           id="exam-centre-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.type} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       {/* <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="exam-date-input">Exam Date:</label> | ||||
|                         <TextField | ||||
|                           id="exam-date-input" | ||||
|                           disabled="true" | ||||
|                           className="w-100" | ||||
|                           value={studentData.ref_exam_date} | ||||
|                         /> | ||||
|                       </Box> */} | ||||
|                       <Box className="d-flex flex-column align-items-start gap-2 py-2"> | ||||
|                         <label for="reassigned-serial-no-input"> | ||||
|                           Reassigned Serial No: | ||||
|                         </label> | ||||
|                         <Box className="d-flex justify-content-between w-100"> | ||||
|                           <TextField | ||||
|                             id="reassigned-serial-no-input" | ||||
|                             disabled={!updateReassigned} | ||||
|                             className="w-100" | ||||
|                             value={studentData.reassigned_serial_no} | ||||
|                             autoComplete="off" | ||||
|                             onChange={(e) => { | ||||
|                               setReassignedSno(e.target.value); | ||||
|                             }} | ||||
|                           /> | ||||
|                           <Button | ||||
|                             onClick={() => { | ||||
|                               setUpdateReassigned(!updateReassigned); | ||||
|                             }} | ||||
|                           > | ||||
|                             <EditButton /> | ||||
|                           </Button> | ||||
|                         </Box> | ||||
|                       </Box> | ||||
|                       <Box className="py-2 d-flex justify-content-start"> | ||||
|                         <Button | ||||
|                           className="text-light bg-primary p-3" | ||||
|                           disabled={!updateReassigned} | ||||
|                           onClick={updateReassignedSno} | ||||
|                         > | ||||
|                           Update | ||||
|                         </Button> | ||||
|                       </Box> | ||||
|                     </Box> | ||||
|                   </Box> | ||||
|                 </> | ||||
|               )} | ||||
|             </Box> | ||||
|             <Box className="w-75 d-none d-md-block"> | ||||
|               {dataFetched && studentData && ( | ||||
|                 <img | ||||
|                   src={`https://docs.exampaper.vidh.ai/${studentData.s3_path}`} | ||||
|                   width="60%" | ||||
|                   height="auto" | ||||
|                 /> | ||||
|               )} | ||||
|             </Box> | ||||
|             <Box className="w-md-75 d-block d-md-none"> | ||||
|               {dataFetched && studentData && ( | ||||
|                 <img | ||||
|                   src={`https://docs.exampaper.vidh.ai/${studentData.s3_path}`} | ||||
|                   width="60%" | ||||
|                   height="auto" | ||||
|                 /> | ||||
|               )} | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Content> | ||||
|         <Footer | ||||
|           style={{ | ||||
|             textAlign: "center", | ||||
|           }} | ||||
|         > | ||||
|           exampaper.vidh.ai ©{new Date().getFullYear()} | ||||
|         </Footer> | ||||
|       </Layout> | ||||
|       {isLoading && <LoadingContainer loadingText={"Fetching"} />} | ||||
|     </Layout> | ||||
|   ); | ||||
| }; | ||||
| export default PartACorrection; | ||||
|  | @ -0,0 +1,213 @@ | |||
| import React, { useState } from "react"; | ||||
| import { | ||||
|   DesktopOutlined, | ||||
|   FileOutlined, | ||||
|   PieChartOutlined, | ||||
|   TeamOutlined, | ||||
|   UserOutlined, | ||||
| } from "@ant-design/icons"; | ||||
| import { Breadcrumb, Layout, Menu, Typography, theme } from "antd"; | ||||
| import BookletInput from "./BookletInput"; | ||||
| import { Box, Button } from "@mui/material"; | ||||
| import { useEffect } from "react"; | ||||
| import TextField from "@mui/material/TextField"; | ||||
| import EditButton from "./EditButton"; | ||||
| import { width } from "@mui/system"; | ||||
| import { ToastContainer, toast } from "react-toastify"; | ||||
| import "react-toastify/dist/ReactToastify.css"; | ||||
| import { TablePagination } from "@mui/base/TablePagination"; | ||||
| import TableComponent from "./TableComponent"; | ||||
| import LoadingContainer from "./LoadingContainer"; | ||||
| import HomeIcon from "@mui/icons-material/Home"; | ||||
| import ArrowBackIcon from "@mui/icons-material/ArrowBack"; | ||||
| import { useNavigate } from "react-router-dom"; | ||||
| import QueryStatsIcon from "@mui/icons-material/QueryStats"; | ||||
| 
 | ||||
| const { Header, Content, Footer, Sider } = Layout; | ||||
| function getItem(label, key, icon, children) { | ||||
|   return { | ||||
|     key, | ||||
|     icon, | ||||
|     children, | ||||
|     label, | ||||
|   }; | ||||
| } | ||||
| // const items = [ | ||||
| //   getItem('Option 1', '1', <PieChartOutlined />), | ||||
| //   getItem('Option 2', '2', <DesktopOutlined />), | ||||
| //   getItem('User', 'sub1', <UserOutlined />, [ | ||||
| //     getItem('Tom', '3'), | ||||
| //     getItem('Bill', '4'), | ||||
| //     getItem('Alex', '5'), | ||||
| //   ]), | ||||
| //   getItem('Team', 'sub2', <TeamOutlined />, [getItem('Team 1', '6'), getItem('Team 2', '8')]), | ||||
| //   getItem('Files', '9', <FileOutlined />), | ||||
| // ]; | ||||
| 
 | ||||
| const items = [getItem("Reassigned Booklet No", "1", <PieChartOutlined />)]; | ||||
| 
 | ||||
| const PartAReassigned = () => { | ||||
|   const [collapsed, setCollapsed] = useState(false); | ||||
|   const [anomolyData, setAnomolyData] = useState([]); | ||||
|   const [tableRowData, setTableRowData] = useState([]); | ||||
|   const [isLoading, setIsLoading] = useState(false); | ||||
|   const [windowWidth, setWindowWidth] = useState(window.innerWidth); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleResize = () => { | ||||
|       setWindowWidth(window.innerWidth); | ||||
|     }; | ||||
| 
 | ||||
|     window.addEventListener("resize", handleResize); | ||||
| 
 | ||||
|     return () => { | ||||
|       window.removeEventListener("resize", handleResize); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (windowWidth < 800) { | ||||
|       setCollapsed(true); | ||||
|     } | ||||
|     if (windowWidth > 800) { | ||||
|       setCollapsed(false); | ||||
|     } | ||||
|   }, [windowWidth]); | ||||
| 
 | ||||
|   const navigate = useNavigate(); | ||||
|   function createData(barcode,register_number,subject_code,type) { | ||||
|     return { | ||||
|         barcode,register_number,subject_code,type | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   const fetchAnomalyData = () => { | ||||
|     console.log("Fetching......."); | ||||
|     setIsLoading(true); | ||||
|     fetch( | ||||
|       `${import.meta.env.VITE_REACT_APP_BACKEND_URL}/fetchAnamolyPartAData`, | ||||
|       { | ||||
|         method: "GET", | ||||
|         headers: { | ||||
|           "Content-Type": "application/json", | ||||
|         }, | ||||
|       } | ||||
|     ) | ||||
|       .then((response) => { | ||||
|         console.log("Response fetched.."); | ||||
|         return response.json(); | ||||
|       }) | ||||
|       .then((responseData) => { | ||||
|         console.log("Response Data is : ", responseData); | ||||
|         setIsLoading(false); | ||||
|         if (responseData.status === "success") { | ||||
|           setAnomolyData(responseData.data); | ||||
|           const tmpData = []; | ||||
|           for (const data of responseData.data) { | ||||
|             tmpData.push(createData(data.barcode,data.register_number,data.subject_code,data.type)); | ||||
|           } | ||||
|           console.log("Tmp data is : ", tmpData); | ||||
|           if (tmpData.length > 0) { | ||||
|             setTableRowData(tmpData); | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|       .catch((error) => { | ||||
|         console.error("Error fetching data: ", error); | ||||
|         setIsLoading(false); | ||||
|       }); | ||||
|   }; | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     fetchAnomalyData(); | ||||
|   }, []); | ||||
| 
 | ||||
|   const { | ||||
|     token: { colorBgContainer, borderRadiusLG }, | ||||
|   } = theme.useToken(); | ||||
| 
 | ||||
|   return ( | ||||
|     <Layout | ||||
|       style={{ | ||||
|         minHeight: "100vh", | ||||
|       }} | ||||
|     > | ||||
|       <ToastContainer /> | ||||
|       <Sider | ||||
|         collapsible | ||||
|         collapsed={collapsed} | ||||
|         onCollapse={(value) => setCollapsed(value)} | ||||
|       > | ||||
|         <div className="demo-logo-vertical" /> | ||||
|         <Menu | ||||
|           theme="dark" | ||||
|           defaultSelectedKeys={["1"]} | ||||
|           mode="inline" | ||||
|           items={items} | ||||
|         /> | ||||
|       </Sider> | ||||
|       <Layout> | ||||
|         <Header | ||||
|           style={{ | ||||
|             padding: 0, | ||||
|             background: colorBgContainer, | ||||
|           }} | ||||
|         > | ||||
|           <Box className="d-flex justify-content-between h-100 py-1 px-2"> | ||||
|             <Button | ||||
|               className="bg-primary p-1 text-light" | ||||
|               onClick={() => { | ||||
|                 navigate(-1); | ||||
|               }} | ||||
|             > | ||||
|               <ArrowBackIcon /> | ||||
|             </Button> | ||||
|             <Box className="d-flex justify-content-between gap-2"> | ||||
|               {/* <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/anomoly/reassigned/stats"); | ||||
|                 }} | ||||
|               > | ||||
|                 <QueryStatsIcon /> | ||||
|               </Button> */} | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/"); | ||||
|                 }} | ||||
|               > | ||||
|                 <HomeIcon /> | ||||
|               </Button> | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Header> | ||||
|         <Content | ||||
|           style={{ | ||||
|             margin: "16px 16px", | ||||
|           }} | ||||
|         > | ||||
|           <Box className="w-100 d-flex justify-content-between"> | ||||
|             <Box className="w-100 d-flex justify-content-center"> | ||||
|               {tableRowData.length > 0 && ( | ||||
|                 <TableComponent | ||||
|                   rows={tableRowData} | ||||
|                   type={"PartAReassigned"} | ||||
|                 /> | ||||
|               )} | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Content> | ||||
|         <Footer | ||||
|           style={{ | ||||
|             textAlign: "center", | ||||
|           }} | ||||
|         > | ||||
|           exampaper.vidh.ai ©{new Date().getFullYear()} | ||||
|         </Footer> | ||||
|       </Layout> | ||||
|       {isLoading && <LoadingContainer loadingText={"Loading"} />} | ||||
|     </Layout> | ||||
|   ); | ||||
| }; | ||||
| export default PartAReassigned; | ||||
|  | @ -0,0 +1,204 @@ | |||
| import React, { useState } from "react"; | ||||
| import { | ||||
|   DesktopOutlined, | ||||
|   FileOutlined, | ||||
|   PieChartOutlined, | ||||
|   TeamOutlined, | ||||
|   UserOutlined, | ||||
| } from "@ant-design/icons"; | ||||
| import { Breadcrumb, Layout, Menu, Typography, theme } from "antd"; | ||||
| import BookletInput from "./BookletInput"; | ||||
| import { Box, Button } from "@mui/material"; | ||||
| import { useEffect } from "react"; | ||||
| import TextField from "@mui/material/TextField"; | ||||
| import EditButton from "./EditButton"; | ||||
| import { width } from "@mui/system"; | ||||
| import { ToastContainer, toast } from "react-toastify"; | ||||
| import "react-toastify/dist/ReactToastify.css"; | ||||
| import { TablePagination } from "@mui/base/TablePagination"; | ||||
| import TableComponent from "./TableComponent"; | ||||
| import LoadingContainer from "./LoadingContainer"; | ||||
| import HomeIcon from "@mui/icons-material/Home"; | ||||
| import ArrowBackIcon from "@mui/icons-material/ArrowBack"; | ||||
| import { useNavigate } from "react-router-dom"; | ||||
| import QueryStatsIcon from "@mui/icons-material/QueryStats"; | ||||
| 
 | ||||
| 
 | ||||
| const { Header, Content, Footer, Sider } = Layout; | ||||
| function getItem(label, key, icon, children) { | ||||
|   return { | ||||
|     key, | ||||
|     icon, | ||||
|     children, | ||||
|     label, | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| const items = [getItem("Reassigned Booklet No", "1", <PieChartOutlined />)]; | ||||
| 
 | ||||
| const ReassignedStats = () => { | ||||
|   const [collapsed, setCollapsed] = useState(false); | ||||
|   const [anomolyData, setAnomolyData] = useState([]); | ||||
|   const [tableRowData, setTableRowData] = useState([]); | ||||
|   const [isLoading, setIsLoading] = useState(false); | ||||
|   const navigate = useNavigate(); | ||||
|   const [windowWidth, setWindowWidth] = useState(window.innerWidth); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     const handleResize = () => { | ||||
|       setWindowWidth(window.innerWidth); | ||||
|     }; | ||||
| 
 | ||||
|     window.addEventListener('resize', handleResize); | ||||
| 
 | ||||
|     return () => { | ||||
|       window.removeEventListener('resize', handleResize); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (windowWidth < 800) { | ||||
|       setCollapsed(true) | ||||
|     } | ||||
|     if(windowWidth > 800){ | ||||
|       setCollapsed(false) | ||||
|     } | ||||
|   }, [windowWidth]); | ||||
| 
 | ||||
|   function createData( | ||||
|     exam_centre_code, | ||||
|     total_count, | ||||
|     manual_verification_count | ||||
|   ) { | ||||
|     return { | ||||
|       exam_centre_code, | ||||
|       total_count, | ||||
|       manual_verification_count | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
| 
 | ||||
|   const fetchReassignedStats = () => { | ||||
|    useEffect(() => { | ||||
|      setIsLoading(true); | ||||
|      fetch(`${import.meta.env.VITE_REACT_APP_BACKEND_URL}/fetchReassignedStats`, { | ||||
|        method: "GET", | ||||
|        headers: { | ||||
|          "Content-Type": "application/json", | ||||
|        }, | ||||
|      }) | ||||
|        .then((response) => response.json()) | ||||
|        .then((responseData) => { | ||||
|          console.log("Response Data is : ", responseData); | ||||
|          setIsLoading(false); | ||||
|          if (responseData.status === "success") { | ||||
|             setAnomolyData(responseData?.data); | ||||
|             const tmpData = []; | ||||
|             for (var data of responseData?.data) { | ||||
|               // console.log("Student name : ",data?.student_name) | ||||
|               tmpData.push( | ||||
|                 createData( | ||||
|                   data?.exam_centre_code, | ||||
|                   data?.total_count, | ||||
|                   data?.manual_verification_count | ||||
|                 ) | ||||
|               ); | ||||
|             } | ||||
|             console.log("Tmp data is : ", tmpData); | ||||
|             if (tmpData.length > 0) { | ||||
|               setTableRowData(tmpData); | ||||
|             } | ||||
|           } | ||||
|        }); | ||||
|    }, []); | ||||
|  }; | ||||
|  fetchReassignedStats(); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   const { | ||||
|     token: { colorBgContainer, borderRadiusLG }, | ||||
|   } = theme.useToken(); | ||||
| 
 | ||||
|   return ( | ||||
|     <Layout | ||||
|       style={{ | ||||
|         minHeight: "100vh", | ||||
|       }} | ||||
|     > | ||||
|       <ToastContainer /> | ||||
|       <Sider | ||||
|         collapsible | ||||
|         collapsed={collapsed} | ||||
|         onCollapse={(value) => setCollapsed(value)} | ||||
|       > | ||||
|         <div className="demo-logo-vertical" /> | ||||
|         <Menu | ||||
|           theme="dark" | ||||
|           defaultSelectedKeys={["1"]} | ||||
|           mode="inline" | ||||
|           items={items} | ||||
|         /> | ||||
|       </Sider> | ||||
|       <Layout> | ||||
|         <Header | ||||
|           style={{ | ||||
|             padding: 0, | ||||
|             background: colorBgContainer, | ||||
|           }} | ||||
|         > | ||||
|           <Box className="d-flex justify-content-between h-100 py-1 px-2"> | ||||
|             <Button | ||||
|               className="bg-primary p-1 text-light" | ||||
|               onClick={() => { | ||||
|                 navigate(-1); | ||||
|               }} | ||||
|             > | ||||
|               <ArrowBackIcon /> | ||||
|             </Button> | ||||
|             <Box className="d-flex justify-content-between gap-2"> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/anomoly/reassigned/stats"); | ||||
|                 }} | ||||
|               > | ||||
|                 <QueryStatsIcon /> | ||||
|               </Button> | ||||
|               <Button | ||||
|                 className="bg-primary p-1 text-light" | ||||
|                 onClick={() => { | ||||
|                   navigate("/"); | ||||
|                 }} | ||||
|               > | ||||
|                 <HomeIcon /> | ||||
|               </Button> | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Header> | ||||
|         <Content | ||||
|           style={{ | ||||
|             margin: "16px 16px", | ||||
|           }} | ||||
|         > | ||||
|           <Box className="w-100 d-flex justify-content-between"> | ||||
|             <Box className="w-100 d-flex justify-content-center"> | ||||
|               {tableRowData.length > 0 && ( | ||||
|                 <TableComponent rows={tableRowData} type={"ReassignedStats"}/> | ||||
|               )} | ||||
|             </Box> | ||||
|           </Box> | ||||
|         </Content> | ||||
|         <Footer | ||||
|           style={{ | ||||
|             textAlign: "center", | ||||
|           }} | ||||
|         > | ||||
|           exampaper.vidh.ai ©{new Date().getFullYear()} | ||||
|         </Footer> | ||||
|       </Layout> | ||||
|       {isLoading && <LoadingContainer loadingText={"Loading"} />} | ||||
|     </Layout> | ||||
|   ); | ||||
| }; | ||||
| export default ReassignedStats; | ||||
|  | @ -0,0 +1,383 @@ | |||
| import * as React from "react"; | ||||
| import { styled } from "@mui/system"; | ||||
| import { | ||||
|   TablePagination, | ||||
|   tablePaginationClasses as classes, | ||||
| } from "@mui/base/TablePagination"; | ||||
| 
 | ||||
| export default function TableComponent({ rows, type }) { | ||||
|   console.log("Rows in table component : ", rows); | ||||
|   console.log("Type is : ", type); | ||||
|   const [page, setPage] = React.useState(0); | ||||
|   const [rowsPerPage, setRowsPerPage] = React.useState(5); | ||||
| 
 | ||||
|   // Avoid a layout jump when reaching the last page with empty rows. | ||||
|   const emptyRows = | ||||
|     page > 0 ? Math.max(0, (1 + page) * rowsPerPage - rows.length) : 0; | ||||
| 
 | ||||
|   const handleChangePage = (event, newPage) => { | ||||
|     setPage(newPage); | ||||
|   }; | ||||
| 
 | ||||
|   const handleChangeRowsPerPage = (event) => { | ||||
|     setRowsPerPage(parseInt(event.target.value, 10)); | ||||
|     setPage(0); | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <Root className="overflow-auto" sx={{ maxWidth: "100%", width: 1200 }}> | ||||
|       {type === "AnomolyReassigned" && ( | ||||
|         <> | ||||
|           <h5 className="py-2">Manual Verification Needed Students :</h5> | ||||
|           <table aria-label="custom pagination table"> | ||||
|             <thead> | ||||
|               <tr> | ||||
|                 <th>Attendence Serial No</th> | ||||
|                 <th>Answer Booklet Serial No</th> | ||||
|                 <th>Exam Centre Code</th> | ||||
|                 <th>Exam Centre</th> | ||||
|                 <th>Student Name</th> | ||||
|                 <th>Register Number</th> | ||||
|                 <th>Reassigned Booklet Serial No</th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               {(rowsPerPage > 0 | ||||
|                 ? rows.slice( | ||||
|                     page * rowsPerPage, | ||||
|                     page * rowsPerPage + rowsPerPage | ||||
|                   ) | ||||
|                 : rows | ||||
|               ).map((row) => ( | ||||
|                 <tr key={row.attendence_serial_no}> | ||||
|                   <td>{row.attendence_serial_no}</td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     <a | ||||
|                       href={`/anomoly/reassigned/booklet?sno=${row.student_slno}`} | ||||
|                     > | ||||
|                       {row.student_slno} | ||||
|                     </a> | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.exam_centre_code} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.exam_centre} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.student_name} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.register_number} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.reassigned_serial_no} | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|               {emptyRows > 0 && ( | ||||
|                 <tr style={{ height: 41 * emptyRows }}> | ||||
|                   <td colSpan={3} aria-hidden /> | ||||
|                 </tr> | ||||
|               )} | ||||
|             </tbody> | ||||
|             <tfoot> | ||||
|               <tr> | ||||
|                 <CustomTablePagination | ||||
|                   rowsPerPageOptions={[5, 10, 25, { label: "All", value: -1 }]} | ||||
|                   colSpan={3} | ||||
|                   count={rows.length} | ||||
|                   rowsPerPage={rowsPerPage} | ||||
|                   page={page} | ||||
|                   slotProps={{ | ||||
|                     select: { | ||||
|                       "aria-label": "rows per page", | ||||
|                     }, | ||||
|                     actions: { | ||||
|                       showFirstButton: true, | ||||
|                       showLastButton: true, | ||||
|                     }, | ||||
|                   }} | ||||
|                   onPageChange={handleChangePage} | ||||
|                   onRowsPerPageChange={handleChangeRowsPerPage} | ||||
|                 /> | ||||
|               </tr> | ||||
|             </tfoot> | ||||
|           </table> | ||||
|         </> | ||||
|       )} | ||||
|       {type === "ReassignedStats" && ( | ||||
|         <> | ||||
|           <h5 className="py-2">Exam Centre Code Complete MetaData :</h5> | ||||
|           <table aria-label="custom pagination table"> | ||||
|             <thead> | ||||
|               <tr> | ||||
|                 <th>Exam Centre Code</th> | ||||
|                 <th>Total Scanned Count</th> | ||||
|                 <th>Manual Verification Count</th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               {(rowsPerPage > 0 | ||||
|                 ? rows.slice( | ||||
|                     page * rowsPerPage, | ||||
|                     page * rowsPerPage + rowsPerPage | ||||
|                   ) | ||||
|                 : rows | ||||
|               ).map((row) => ( | ||||
|                 <tr key={row.exam_centre_code}> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     <a | ||||
|                       href={`/anomoly/reassigned/stats/${row.exam_centre_code}`} | ||||
|                     > | ||||
|                       {row.exam_centre_code} | ||||
|                     </a> | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.total_count} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.manual_verification_count} | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|               {emptyRows > 0 && ( | ||||
|                 <tr style={{ height: 41 * emptyRows }}> | ||||
|                   <td colSpan={3} aria-hidden /> | ||||
|                 </tr> | ||||
|               )} | ||||
|             </tbody> | ||||
|             <tfoot> | ||||
|               <tr> | ||||
|                 <CustomTablePagination | ||||
|                   rowsPerPageOptions={[5, 10, 25, { label: "All", value: -1 }]} | ||||
|                   colSpan={3} | ||||
|                   count={rows.length} | ||||
|                   rowsPerPage={rowsPerPage} | ||||
|                   page={page} | ||||
|                   slotProps={{ | ||||
|                     select: { | ||||
|                       "aria-label": "rows per page", | ||||
|                     }, | ||||
|                     actions: { | ||||
|                       showFirstButton: true, | ||||
|                       showLastButton: true, | ||||
|                     }, | ||||
|                   }} | ||||
|                   onPageChange={handleChangePage} | ||||
|                   onRowsPerPageChange={handleChangeRowsPerPage} | ||||
|                 /> | ||||
|               </tr> | ||||
|             </tfoot> | ||||
|           </table> | ||||
|         </> | ||||
|       )} | ||||
|       {type === "IndividualReassignedStats" && ( | ||||
|         <> | ||||
|           <h5 className="py-2">Exam Centre Code-Date Wise MetaData :</h5> | ||||
|           <table aria-label="custom pagination table"> | ||||
|             <thead> | ||||
|               <tr> | ||||
|                 <th>Exam Centre Code</th> | ||||
|                 <th>Exam Date</th> | ||||
|                 <th>Subject Code</th> | ||||
|                 <th>Total Count</th> | ||||
|                 <th>Manual Verification Count</th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               {(rowsPerPage > 0 | ||||
|                 ? rows.slice( | ||||
|                     page * rowsPerPage, | ||||
|                     page * rowsPerPage + rowsPerPage | ||||
|                   ) | ||||
|                 : rows | ||||
|               ).map((row) => ( | ||||
|                 <tr key={row.exam_centre_code}> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.exam_centre_code} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.exam_date} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.subject_code} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.total_count} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.manual_verification_needed_count} | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|               {emptyRows > 0 && ( | ||||
|                 <tr style={{ height: 41 * emptyRows }}> | ||||
|                   <td colSpan={3} aria-hidden /> | ||||
|                 </tr> | ||||
|               )} | ||||
|             </tbody> | ||||
|             <tfoot> | ||||
|               <tr> | ||||
|                 <CustomTablePagination | ||||
|                   rowsPerPageOptions={[5, 10, 25, { label: "All", value: -1 }]} | ||||
|                   colSpan={3} | ||||
|                   count={rows.length} | ||||
|                   rowsPerPage={rowsPerPage} | ||||
|                   page={page} | ||||
|                   slotProps={{ | ||||
|                     select: { | ||||
|                       "aria-label": "rows per page", | ||||
|                     }, | ||||
|                     actions: { | ||||
|                       showFirstButton: true, | ||||
|                       showLastButton: true, | ||||
|                     }, | ||||
|                   }} | ||||
|                   onPageChange={handleChangePage} | ||||
|                   onRowsPerPageChange={handleChangeRowsPerPage} | ||||
|                 /> | ||||
|               </tr> | ||||
|             </tfoot> | ||||
|           </table> | ||||
|         </> | ||||
|       )} | ||||
|       {type === "PartAReassigned" && ( | ||||
|         <> | ||||
|           <h5 className="py-2">Part-A Anomoly Data :</h5> | ||||
|           <table aria-label="custom pagination table"> | ||||
|             <thead> | ||||
|               <tr> | ||||
|                 <th>Barcode</th> | ||||
|                 <th>Subject Code</th> | ||||
|                 <th>Register Number</th> | ||||
|                 <th>Type</th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               {(rowsPerPage > 0 | ||||
|                 ? rows.slice( | ||||
|                     page * rowsPerPage, | ||||
|                     page * rowsPerPage + rowsPerPage | ||||
|                   ) | ||||
|                 : rows | ||||
|               ).map((row) => ( | ||||
|                 <tr key={row.exam_centre_code}> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     <a href={`/anomoly/partA/booklet?barcode=${row.barcode}`}>{row.barcode}</a> | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.subject_code} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.register_number} | ||||
|                   </td> | ||||
|                   <td style={{ width: 160 }} align="right"> | ||||
|                     {row.type} | ||||
|                   </td> | ||||
|                 </tr> | ||||
|               ))} | ||||
|               {emptyRows > 0 && ( | ||||
|                 <tr style={{ height: 41 * emptyRows }}> | ||||
|                   <td colSpan={3} aria-hidden /> | ||||
|                 </tr> | ||||
|               )} | ||||
|             </tbody> | ||||
|             <tfoot> | ||||
|               <tr> | ||||
|                 <CustomTablePagination | ||||
|                   rowsPerPageOptions={[5, 10, 25, { label: "All", value: -1 }]} | ||||
|                   colSpan={3} | ||||
|                   count={rows.length} | ||||
|                   rowsPerPage={rowsPerPage} | ||||
|                   page={page} | ||||
|                   slotProps={{ | ||||
|                     select: { | ||||
|                       "aria-label": "rows per page", | ||||
|                     }, | ||||
|                     actions: { | ||||
|                       showFirstButton: true, | ||||
|                       showLastButton: true, | ||||
|                     }, | ||||
|                   }} | ||||
|                   onPageChange={handleChangePage} | ||||
|                   onRowsPerPageChange={handleChangeRowsPerPage} | ||||
|                 /> | ||||
|               </tr> | ||||
|             </tfoot> | ||||
|           </table> | ||||
|         </> | ||||
|       )} | ||||
|     </Root> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| const grey = { | ||||
|   50: "#F3F6F9", | ||||
|   100: "#E5EAF2", | ||||
|   200: "#DAE2ED", | ||||
|   300: "#C7D0DD", | ||||
|   400: "#B0B8C4", | ||||
|   500: "#9DA8B7", | ||||
|   600: "#6B7A90", | ||||
|   700: "#434D5B", | ||||
|   800: "#303740", | ||||
|   900: "#1C2025", | ||||
| }; | ||||
| 
 | ||||
| const Root = styled("div")( | ||||
|   ({ theme }) => ` | ||||
|   table { | ||||
|     font-family: 'IBM Plex Sans', sans-serif; | ||||
|     font-size: 0.875rem; | ||||
|     border-collapse: collapse; | ||||
|     width: 100%; | ||||
|   } | ||||
| 
 | ||||
|   td, | ||||
|   th { | ||||
|     border: 1px solid ${theme.palette.mode === "dark" ? grey[800] : grey[200]}; | ||||
|     text-align: left; | ||||
|     padding: 8px; | ||||
|   } | ||||
| 
 | ||||
|   th { | ||||
|     background-color: ${theme.palette.mode === "dark" ? grey[900] : "#fff"}; | ||||
|   } | ||||
|   ` | ||||
| ); | ||||
| 
 | ||||
| const CustomTablePagination = styled(TablePagination)` | ||||
|   & .${classes.toolbar} { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     align-items: flex-start; | ||||
|     gap: 10px; | ||||
| 
 | ||||
|     @media (min-width: 768px) { | ||||
|       flex-direction: row; | ||||
|       align-items: center; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   & .${classes.selectLabel} { | ||||
|     margin: 0; | ||||
|   } | ||||
| 
 | ||||
|   & .${classes.displayedRows} { | ||||
|     margin: 0; | ||||
| 
 | ||||
|     @media (min-width: 768px) { | ||||
|       margin-left: auto; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   & .${classes.spacer} { | ||||
|     display: none; | ||||
|   } | ||||
| 
 | ||||
|   & .${classes.actions} { | ||||
|     display: flex; | ||||
|     gap: 0.25rem; | ||||
|   } | ||||
| `; | ||||
|  | @ -0,0 +1 @@ | |||
| <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="35.93" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 228"><path fill="#00D8FF" d="M210.483 73.824a171.49 171.49 0 0 0-8.24-2.597c.465-1.9.893-3.777 1.273-5.621c6.238-30.281 2.16-54.676-11.769-62.708c-13.355-7.7-35.196.329-57.254 19.526a171.23 171.23 0 0 0-6.375 5.848a155.866 155.866 0 0 0-4.241-3.917C100.759 3.829 77.587-4.822 63.673 3.233C50.33 10.957 46.379 33.89 51.995 62.588a170.974 170.974 0 0 0 1.892 8.48c-3.28.932-6.445 1.924-9.474 2.98C17.309 83.498 0 98.307 0 113.668c0 15.865 18.582 31.778 46.812 41.427a145.52 145.52 0 0 0 6.921 2.165a167.467 167.467 0 0 0-2.01 9.138c-5.354 28.2-1.173 50.591 12.134 58.266c13.744 7.926 36.812-.22 59.273-19.855a145.567 145.567 0 0 0 5.342-4.923a168.064 168.064 0 0 0 6.92 6.314c21.758 18.722 43.246 26.282 56.54 18.586c13.731-7.949 18.194-32.003 12.4-61.268a145.016 145.016 0 0 0-1.535-6.842c1.62-.48 3.21-.974 4.76-1.488c29.348-9.723 48.443-25.443 48.443-41.52c0-15.417-17.868-30.326-45.517-39.844Zm-6.365 70.984c-1.4.463-2.836.91-4.3 1.345c-3.24-10.257-7.612-21.163-12.963-32.432c5.106-11 9.31-21.767 12.459-31.957c2.619.758 5.16 1.557 7.61 2.4c23.69 8.156 38.14 20.213 38.14 29.504c0 9.896-15.606 22.743-40.946 31.14Zm-10.514 20.834c2.562 12.94 2.927 24.64 1.23 33.787c-1.524 8.219-4.59 13.698-8.382 15.893c-8.067 4.67-25.32-1.4-43.927-17.412a156.726 156.726 0 0 1-6.437-5.87c7.214-7.889 14.423-17.06 21.459-27.246c12.376-1.098 24.068-2.894 34.671-5.345a134.17 134.17 0 0 1 1.386 6.193ZM87.276 214.515c-7.882 2.783-14.16 2.863-17.955.675c-8.075-4.657-11.432-22.636-6.853-46.752a156.923 156.923 0 0 1 1.869-8.499c10.486 2.32 22.093 3.988 34.498 4.994c7.084 9.967 14.501 19.128 21.976 27.15a134.668 134.668 0 0 1-4.877 4.492c-9.933 8.682-19.886 14.842-28.658 17.94ZM50.35 144.747c-12.483-4.267-22.792-9.812-29.858-15.863c-6.35-5.437-9.555-10.836-9.555-15.216c0-9.322 13.897-21.212 37.076-29.293c2.813-.98 5.757-1.905 8.812-2.773c3.204 10.42 7.406 21.315 12.477 32.332c-5.137 11.18-9.399 22.249-12.634 32.792a134.718 134.718 0 0 1-6.318-1.979Zm12.378-84.26c-4.811-24.587-1.616-43.134 6.425-47.789c8.564-4.958 27.502 2.111 47.463 19.835a144.318 144.318 0 0 1 3.841 3.545c-7.438 7.987-14.787 17.08-21.808 26.988c-12.04 1.116-23.565 2.908-34.161 5.309a160.342 160.342 0 0 1-1.76-7.887Zm110.427 27.268a347.8 347.8 0 0 0-7.785-12.803c8.168 1.033 15.994 2.404 23.343 4.08c-2.206 7.072-4.956 14.465-8.193 22.045a381.151 381.151 0 0 0-7.365-13.322Zm-45.032-43.861c5.044 5.465 10.096 11.566 15.065 18.186a322.04 322.04 0 0 0-30.257-.006c4.974-6.559 10.069-12.652 15.192-18.18ZM82.802 87.83a323.167 323.167 0 0 0-7.227 13.238c-3.184-7.553-5.909-14.98-8.134-22.152c7.304-1.634 15.093-2.97 23.209-3.984a321.524 321.524 0 0 0-7.848 12.897Zm8.081 65.352c-8.385-.936-16.291-2.203-23.593-3.793c2.26-7.3 5.045-14.885 8.298-22.6a321.187 321.187 0 0 0 7.257 13.246c2.594 4.48 5.28 8.868 8.038 13.147Zm37.542 31.03c-5.184-5.592-10.354-11.779-15.403-18.433c4.902.192 9.899.29 14.978.29c5.218 0 10.376-.117 15.453-.343c-4.985 6.774-10.018 12.97-15.028 18.486Zm52.198-57.817c3.422 7.8 6.306 15.345 8.596 22.52c-7.422 1.694-15.436 3.058-23.88 4.071a382.417 382.417 0 0 0 7.859-13.026a347.403 347.403 0 0 0 7.425-13.565Zm-16.898 8.101a358.557 358.557 0 0 1-12.281 19.815a329.4 329.4 0 0 1-23.444.823c-7.967 0-15.716-.248-23.178-.732a310.202 310.202 0 0 1-12.513-19.846h.001a307.41 307.41 0 0 1-10.923-20.627a310.278 310.278 0 0 1 10.89-20.637l-.001.001a307.318 307.318 0 0 1 12.413-19.761c7.613-.576 15.42-.876 23.31-.876H128c7.926 0 15.743.303 23.354.883a329.357 329.357 0 0 1 12.335 19.695a358.489 358.489 0 0 1 11.036 20.54a329.472 329.472 0 0 1-11 20.722Zm22.56-122.124c8.572 4.944 11.906 24.881 6.52 51.026c-.344 1.668-.73 3.367-1.15 5.09c-10.622-2.452-22.155-4.275-34.23-5.408c-7.034-10.017-14.323-19.124-21.64-27.008a160.789 160.789 0 0 1 5.888-5.4c18.9-16.447 36.564-22.941 44.612-18.3ZM128 90.808c12.625 0 22.86 10.235 22.86 22.86s-10.235 22.86-22.86 22.86s-22.86-10.235-22.86-22.86s10.235-22.86 22.86-22.86Z"></path></svg> | ||||
| After Width: | Height: | Size: 4.0 KiB | 
|  | @ -0,0 +1,68 @@ | |||
| :root { | ||||
|   font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif; | ||||
|   line-height: 1.5; | ||||
|   font-weight: 400; | ||||
| 
 | ||||
|   color-scheme: light dark; | ||||
|   color: rgba(255, 255, 255, 0.87); | ||||
|   background-color: #242424; | ||||
| 
 | ||||
|   font-synthesis: none; | ||||
|   text-rendering: optimizeLegibility; | ||||
|   -webkit-font-smoothing: antialiased; | ||||
|   -moz-osx-font-smoothing: grayscale; | ||||
| } | ||||
| 
 | ||||
| a { | ||||
|   font-weight: 500; | ||||
|   color: #646cff; | ||||
|   text-decoration: inherit; | ||||
| } | ||||
| a:hover { | ||||
|   color: #535bf2; | ||||
| } | ||||
| 
 | ||||
| body { | ||||
|   margin: 0; | ||||
|   display: flex; | ||||
|   place-items: center; | ||||
|   min-width: 320px; | ||||
|   min-height: 100vh; | ||||
| } | ||||
| 
 | ||||
| h1 { | ||||
|   font-size: 3.2em; | ||||
|   line-height: 1.1; | ||||
| } | ||||
| 
 | ||||
| button { | ||||
|   border-radius: 8px; | ||||
|   border: 1px solid transparent; | ||||
|   padding: 0.6em 1.2em; | ||||
|   font-size: 1em; | ||||
|   font-weight: 500; | ||||
|   font-family: inherit; | ||||
|   background-color: #1a1a1a; | ||||
|   cursor: pointer; | ||||
|   transition: border-color 0.25s; | ||||
| } | ||||
| button:hover { | ||||
|   border-color: #646cff; | ||||
| } | ||||
| button:focus, | ||||
| button:focus-visible { | ||||
|   outline: 4px auto -webkit-focus-ring-color; | ||||
| } | ||||
| 
 | ||||
| @media (prefers-color-scheme: light) { | ||||
|   :root { | ||||
|     color: #213547; | ||||
|     background-color: #ffffff; | ||||
|   } | ||||
|   a:hover { | ||||
|     color: #747bff; | ||||
|   } | ||||
|   button { | ||||
|     background-color: #f9f9f9; | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,12 @@ | |||
| import React from 'react' | ||||
| import ReactDOM from 'react-dom/client' | ||||
| import App from './App.jsx' | ||||
| import './index.css' | ||||
| import 'bootstrap/dist/css/bootstrap.min.css'; | ||||
| 
 | ||||
| 
 | ||||
| ReactDOM.createRoot(document.getElementById('root')).render( | ||||
|   <React.StrictMode> | ||||
|     <App /> | ||||
|   </React.StrictMode>, | ||||
| ) | ||||
|  | @ -0,0 +1,24 @@ | |||
| // import { defineConfig } from 'vite'
 | ||||
| // import react from '@vitejs/plugin-react'
 | ||||
| 
 | ||||
| // // https://vitejs.dev/config/
 | ||||
| // export default defineConfig({
 | ||||
| //   plugins: [react()],
 | ||||
| // })
 | ||||
| 
 | ||||
| 
 | ||||
| import { defineConfig, loadEnv } from 'vite' | ||||
| import react from '@vitejs/plugin-react' | ||||
| 
 | ||||
| // https://vitejs.dev/config/
 | ||||
| export default defineConfig(({ mode }) => { | ||||
|   // Load environment variables based on the current mode (e.g., 'development' or 'production')
 | ||||
|   const env = loadEnv(mode, process.cwd()); | ||||
| 
 | ||||
|   return { | ||||
|     define: { | ||||
|       'process.env': env // expose all environment variables prefixed with VITE_ to process.env
 | ||||
|     }, | ||||
|     plugins: [react()], | ||||
|   } | ||||
| }) | ||||
		Loading…
	
		Reference in New Issue