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 = ( <>
{formatTimestamp(stats[0].date.toString())} {timeSpent && ( <> • {Math.floor(timeSpent / 60)} minutes )}
= 0.7 && "text-mti-purple", correct / total >= 0.3 && correct / total < 0.7 && "text-mti-red", correct / total < 0.3 && "text-mti-rose", )}> Level{" "} {(aggregatedLevels.reduce((accumulator, current) => accumulator + current.level, 0) / aggregatedLevels.length).toFixed(1)}
{aggregatedLevels.map(({ module, level }) => (
{module === "reading" && } {module === "listening" && } {module === "writing" && } {module === "speaking" && } {module === "level" && } {level.toFixed(1)}
))}
); return (
{(() => { const student = users.find((u) => u.id === user); return `${student?.name} (${student?.email})`; })()}
= 0.7 && "hover:border-mti-purple", correct / total >= 0.3 && correct / total < 0.7 && "hover:border-mti-red", correct / total < 0.3 && "hover:border-mti-rose", )} onClick={selectExam} role="button"> {content}
= 0.7 && "hover:border-mti-purple", correct / total >= 0.3 && correct / total < 0.7 && "hover:border-mti-red", correct / total < 0.3 && "hover:border-mti-rose", )} data-tip="Your screen size is too small to view previous exams." role="button"> {content}
); }; const shouldRenderStart = () => { if (assignment) { if (futureAssignmentFilter(assignment)) { return true; } } return false; }; return (
Start Date: {moment(assignment?.startDate).format("DD/MM/YY, HH:mm")} End Date: {moment(assignment?.endDate).format("DD/MM/YY, HH:mm")}
Assignees:{" "} {users .filter((u) => assignment?.assignees.includes(u.id)) .map((u) => `${u.name} (${u.email})`) .join(", ")} Assigner: {getUserName(users.find((x) => x.id === assignment?.assigner))}
Average Scores
{assignment && uniqBy(assignment.exams, (x) => x.module).map(({ module }) => (
{module === "reading" && } {module === "listening" && } {module === "writing" && } {module === "speaking" && } {module === "level" && } {calculateAverageModuleScore(module) > -1 && ( {calculateAverageModuleScore(module).toFixed(1)} )}
))}
Results ({assignment?.results.length}/{assignment?.assignees.length})
{assignment && assignment?.results.length > 0 && (
{assignment.results.map((r) => customContent(r.stats, r.user, r.type))}
)} {assignment && assignment?.results.length === 0 && No results yet...}
{assignment && (assignment.results.length === assignment.assignees.length || moment().isAfter(moment(assignment.endDate))) && ( )} {/** if the assignment is not deemed as active yet, display start */} {shouldRenderStart() && ( )}
); }