/* eslint-disable @next/next/no-img-element */ import Head from "next/head"; import {withIronSessionSsr} from "iron-session/next"; import {sessionOptions} from "@/lib/session"; import {Stat, User} from "@/interfaces/user"; import {useEffect, useMemo, useRef, useState} from "react"; import useFilterRecordsByUser from "@/hooks/useFilterRecordsByUser"; import {groupByDate} from "@/utils/stats"; import moment from "moment"; import useUsers from "@/hooks/useUsers"; import useExamStore from "@/stores/examStore"; import {ToastContainer} from "react-toastify"; import Layout from "@/components/High/Layout"; import clsx from "clsx"; import {shouldRedirectHome} from "@/utils/navigation.disabled"; import useAssignments from "@/hooks/useAssignments"; import {uuidv4} from "@firebase/util"; import {usePDFDownload} from "@/hooks/usePDFDownload"; import useRecordStore from "@/stores/recordStore"; import StatsGridItem from "@/components/Medium/StatGridItem"; import RecordFilter from "@/components/Medium/RecordFilter"; import {useRouter} from "next/router"; import useTrainingContentStore from "@/stores/trainingContentStore"; import {Assignment} from "@/interfaces/results"; import {getEntitiesUsers, getUsers} from "@/utils/users.be"; import {getAssignments, getAssignmentsByAssigner, getEntitiesAssignments} from "@/utils/assignments.be"; import useGradingSystem from "@/hooks/useGrading"; import { mapBy, redirect, serialize } from "@/utils"; import { getEntitiesWithRoles } from "@/utils/entities.be"; import { checkAccess } from "@/utils/permissions"; import { getGroups, getGroupsByEntities } from "@/utils/groups.be"; import { getGradingSystemByEntity } from "@/utils/grading.be"; import { Grading } from "@/interfaces"; import { EntityWithRoles } from "@/interfaces/entity"; import { useListSearch } from "@/hooks/useListSearch"; import CardList from "@/components/High/CardList"; import { requestUser } from "@/utils/api"; export const getServerSideProps = withIronSessionSsr(async ({req, res}) => { const user = await requestUser(req, res) if (!user) return redirect("/login") if (shouldRedirectHome(user)) return redirect("/") const entityIDs = mapBy(user.entities, 'id') const entities = await getEntitiesWithRoles(checkAccess(user, ["admin", "developer"]) ? undefined : entityIDs) const users = await (checkAccess(user, ["admin", "developer"]) ? getUsers() : getEntitiesUsers(mapBy(entities, 'id'))) const groups = await (checkAccess(user, ["admin", "developer"]) ? getGroups() : getGroupsByEntities(mapBy(entities, 'id'))) const assignments = await (checkAccess(user, ["admin", "developer"]) ? getAssignments() : getEntitiesAssignments(mapBy(entities, 'id'))) const gradingSystems = await Promise.all(entityIDs.map(getGradingSystemByEntity)) return { props: serialize({user, users, assignments, entities, gradingSystems}), }; }, sessionOptions); type Filter = "months" | "weeks" | "days" | "assignments" | undefined; interface Props { user: User; users: User[]; assignments: Assignment[]; gradingSystems: Grading[] entities: EntityWithRoles[] } const MAX_TRAINING_EXAMS = 10; export default function History({user, users, assignments, entities, gradingSystems}: Props) { const router = useRouter(); const [statsUserId, setStatsUserId, training, setTraining] = useRecordStore((state) => [ state.selectedUser, state.setSelectedUser, state.training, state.setTraining, ]); const [filter, setFilter] = useState(); const {data: stats, isLoading: isStatsLoading} = useFilterRecordsByUser(statsUserId || user?.id); const {gradingSystem} = useGradingSystem(); const setExams = useExamStore((state) => state.setExams); const setShowSolutions = useExamStore((state) => state.setShowSolutions); const setUserSolutions = useExamStore((state) => state.setUserSolutions); const setSelectedModules = useExamStore((state) => state.setSelectedModules); const setInactivity = useExamStore((state) => state.setInactivity); const setTimeSpent = useExamStore((state) => state.setTimeSpent); const renderPdfIcon = usePDFDownload("stats"); const [selectedTrainingExams, setSelectedTrainingExams] = useState([]); const setTrainingStats = useTrainingContentStore((state) => state.setStats); const groupedStats = useMemo(() => groupByDate( stats.filter((x) => { if ( (x.module === "writing" || x.module === "speaking") && !x.isDisabled && !x.solutions.every((y) => Object.keys(y).includes("evaluation")) ) return false; return true; }), ), [stats]) useEffect(() => setStatsUserId(user.id), [setStatsUserId, user]); useEffect(() => { const handleRouteChange = (url: string) => { setTraining(false); }; router.events.on("routeChangeStart", handleRouteChange); return () => { router.events.off("routeChangeStart", handleRouteChange); }; }, [router.events, setTraining]); const filterStatsByDate = (stats: {[key: string]: Stat[]}) => { if (filter && filter !== "assignments") { const filterDate = moment() .subtract({[filter as string]: 1}) .format("x"); const filteredStats: {[key: string]: Stat[]} = {}; Object.keys(stats).forEach((timestamp) => { if (timestamp >= filterDate) filteredStats[timestamp] = stats[timestamp]; }); return filteredStats; } if (filter && filter === "assignments") { const filteredStats: {[key: string]: Stat[]} = {}; Object.keys(stats).forEach((timestamp) => { if (stats[timestamp].map((s) => s.assignment === undefined).includes(false)) filteredStats[timestamp] = [...stats[timestamp].filter((s) => !!s.assignment)]; }); return filteredStats; } return stats; }; const handleTrainingContentSubmission = () => { if (groupedStats) { const groupedStatsByDate = filterStatsByDate(groupedStats); const allStats = Object.keys(groupedStatsByDate); const selectedStats = selectedTrainingExams.reduce>((accumulator, moduleAndTimestamp) => { const timestamp = moduleAndTimestamp.split("-")[1]; if (allStats.includes(timestamp) && !accumulator.hasOwnProperty(timestamp)) { accumulator[timestamp] = groupedStatsByDate[timestamp]; } return accumulator; }, {}); setTrainingStats(Object.values(selectedStats).flat()); router.push("/training"); } }; const filteredStats = useMemo(() => Object.keys(filterStatsByDate(groupedStats)) .sort((a, b) => parseInt(b) - parseInt(a)), // eslint-disable-next-line react-hooks/exhaustive-deps [groupedStats, filter]) const customContent = (timestamp: string) => { const dateStats = groupedStats[timestamp]; return ( ); }; return ( <> Record | EnCoach {user && ( {training && (
Select up to 10 exercises {`(${selectedTrainingExams.length}/${MAX_TRAINING_EXAMS})`}
)}
{filteredStats.length > 0 && !isStatsLoading && ( )} {filteredStats.length === 0 && !isStatsLoading && ( No record to display... )} {isStatsLoading && (
)}
)} ); }