import Button from "@/components/Low/Button"; import ProgressBar from "@/components/Low/ProgressBar"; import Modal from "@/components/Modal"; import useUsers from "@/hooks/useUsers"; import { Module } from "@/interfaces"; import { Assignment } from "@/interfaces/results"; import { Stat, User } from "@/interfaces/user"; import useExamStore from "@/stores/exam"; import { getExamById } from "@/utils/exams"; import { sortByModule } from "@/utils/moduleUtils"; import { calculateBandScore } from "@/utils/score"; import { getUserName } from "@/utils/users"; import axios from "axios"; import clsx from "clsx"; import { capitalize, uniqBy } from "lodash"; import moment from "moment"; import { useRouter } from "next/router"; import { BsBook, BsClipboard, BsHeadphones, BsMegaphone, BsPen } from "react-icons/bs"; import { toast } from "react-toastify"; import { futureAssignmentFilter } from "@/utils/assignments"; interface Props { isOpen: boolean; users: User[]; assignment?: Assignment; onClose: () => void; } export default function AssignmentView({ isOpen, users, assignment, onClose }: Props) { const router = useRouter(); const dispatch = useExamStore((s) => s.dispatch); const deleteAssignment = async () => { if (!confirm("Are you sure you want to delete this assignment?")) return; axios .delete(`/api/assignments/${assignment?.id}`) .then(() => toast.success(`Successfully deleted the assignment "${assignment?.name}".`)) .catch(() => toast.error("Something went wrong, please try again later.")) .finally(onClose); }; const startAssignment = () => { if (assignment) { axios .post(`/api/assignments/${assignment.id}/start`) .then(() => { toast.success(`The assignment "${assignment.name}" has been started successfully!`); }) .catch((e) => { console.log(e); toast.error("Something went wrong, please try again later!"); }); } }; const formatTimestamp = (timestamp: string) => { const date = moment(parseInt(timestamp)); const formatter = "YYYY/MM/DD - HH:mm"; return date.format(formatter); }; const calculateAverageModuleScore = (module: Module) => { if (!assignment) return -1; const resultModuleBandScores = assignment.results.map((r) => { const moduleStats = r.stats.filter((s) => s.module === module); const correct = moduleStats.reduce((acc, curr) => acc + curr.score.correct, 0); const total = moduleStats.reduce((acc, curr) => acc + curr.score.total, 0); return calculateBandScore(correct, total, module, r.type); }); return resultModuleBandScores.length === 0 ? -1 : resultModuleBandScores.reduce((acc, curr) => acc + curr, 0) / assignment.results.length; }; const aggregateScoresByModule = (stats: Stat[]): { module: Module; total: number; missing: number; correct: number }[] => { const scores: { [key in Module]: { total: number; missing: number; correct: number }; } = { reading: { total: 0, correct: 0, missing: 0, }, listening: { total: 0, correct: 0, missing: 0, }, writing: { total: 0, correct: 0, missing: 0, }, speaking: { total: 0, correct: 0, missing: 0, }, level: { total: 0, correct: 0, missing: 0, }, }; stats.filter(x => !x.isPractice).forEach((x) => { scores[x.module!] = { total: scores[x.module!].total + x.score.total, correct: scores[x.module!].correct + x.score.correct, missing: scores[x.module!].missing + x.score.missing, }; }); return Object.keys(scores) .filter((x) => scores[x as Module].total > 0) .map((x) => ({ module: x as Module, ...scores[x as Module] })); }; const customContent = (stats: Stat[], user: string, focus: "academic" | "general") => { const correct = stats.reduce((accumulator, current) => accumulator + current.score.correct, 0); const total = stats.reduce((accumulator, current) => accumulator + current.score.total, 0); const aggregatedScores = aggregateScoresByModule(stats).filter((x) => x.total > 0); const aggregatedLevels = aggregatedScores.map((x) => ({ module: x.module, level: calculateBandScore(x.correct, x.total, x.module, focus), })); const timeSpent = stats[0].timeSpent; const selectExam = () => { const examPromises = uniqBy(stats, "exam").map((stat) => getExamById(stat.module, stat.exam)); Promise.all(examPromises).then((exams) => { if (exams.every((x) => !!x)) { dispatch({ type: 'INIT_SOLUTIONS', payload: { exams: exams.map((x) => x!).sort(sortByModule), modules: exams .map((x) => x!) .sort(sortByModule) .map((x) => x!.module), stats } }); router.push("/exam"); } }); }; const content = ( <>