/* eslint-disable @next/next/no-img-element */ import { MultipleChoiceExercise, MultipleChoiceQuestion, ShuffleMap } from "@/interfaces/exam"; import useExamStore from "@/stores/examStore"; import clsx from "clsx"; import { useEffect, useState } from "react"; import reactStringReplace from "react-string-replace"; import { CommonProps } from "."; import Button from "../Low/Button"; import { v4 } from "uuid"; import PracticeBadge from "../Low/PracticeBadge"; function Question({ id, variant, prompt, options, userSolution, onSelectOption, isPractice = false }: MultipleChoiceQuestion & { userSolution: string | undefined; onSelectOption?: (option: string) => void; showSolution?: boolean; isPractice?: boolean }) { const renderPrompt = (prompt: string) => { return reactStringReplace(prompt, /(.*?<\/u>)/g, (match) => { const word = match.replaceAll("", "").replaceAll("", ""); return word.length > 0 ? {word} : null; }); }; return (
{isPractice && } {isNaN(Number(id)) ? ( {renderPrompt(prompt).filter((x) => x?.toString() !== "")} ) : ( <> {id} - {renderPrompt(prompt).filter((x) => x?.toString() !== "")} )}
{variant === "image" && options.map((option) => (
(onSelectOption ? onSelectOption(option.id.toString()) : null)} className={clsx( "flex flex-col items-center border border-mti-gray-platinum p-4 px-8 rounded-xl gap-4 cursor-pointer bg-white relative select-none", userSolution === option.id.toString() && "border-mti-purple-light", )}> {option.id.toString()} {`Option
))} {variant === "text" && options.map((option) => (
(onSelectOption ? onSelectOption(option.id.toString()) : null)} className={clsx( "flex border p-4 rounded-xl gap-2 cursor-pointer bg-white text-base select-none", userSolution === option.id.toString() && "!bg-mti-purple-light !text-white", )}> {option.id.toString()}. {option.text}
))}
); } export default function MultipleChoice({ id, prompt, type, questions, userSolutions, isPractice = false, onNext, onBack, disableProgressButtons = false }: MultipleChoiceExercise & CommonProps) { const [answers, setAnswers] = useState<{ question: string; option: string }[]>(userSolutions || []); const { questionIndex, exerciseIndex, exam, shuffles, hasExamEnded, partIndex, setQuestionIndex, setCurrentSolution } = useExamStore( (state) => state, ); const shuffleMaps = shuffles.find((x) => x.exerciseID == id)?.shuffles; const scrollToTop = () => Array.from(document.getElementsByTagName("body")).forEach((body) => body.scrollTo(0, 0)); useEffect(() => { if (hasExamEnded) onNext({ exercise: id, solutions: answers, score: calculateScore(), type, isPractice }); // eslint-disable-next-line react-hooks/exhaustive-deps }, [hasExamEnded]); const onSelectOption = (option: string, question: MultipleChoiceQuestion) => { setAnswers((prev) => [...prev.filter((x) => x.question !== question.id), { option, question: question.id }]); }; useEffect(() => { setCurrentSolution({ exercise: id, solutions: answers, score: calculateScore(), type, shuffleMaps: shuffleMaps, isPractice }); // eslint-disable-next-line react-hooks/exhaustive-deps }, [answers, setAnswers]); const getShuffledSolution = (originalSolution: string, questionShuffleMap: ShuffleMap) => { for (const [newPosition, originalPosition] of Object.entries(questionShuffleMap.map)) { if (originalPosition === originalSolution) { return newPosition; } } return originalSolution; }; const calculateScore = () => { const total = questions.length; const correct = answers.filter((x) => { const matchingQuestion = questions.find((y) => { return y.id.toString() === x.question.toString(); }); let isSolutionCorrect; if (!shuffleMaps) { isSolutionCorrect = matchingQuestion?.solution === x.option; } else { const shuffleMap = shuffleMaps.find((map) => map.questionID == x.question); if (shuffleMap) { isSolutionCorrect = getShuffledSolution(x.option, shuffleMap) == matchingQuestion?.solution; } else { isSolutionCorrect = matchingQuestion?.solution === x.option; } } return isSolutionCorrect || false; }).length; const missing = total - answers!.filter((x) => questions.find((y) => x.question.toString() === y.id.toString())).length; return { total, correct, missing }; }; useEffect(() => { if (disableProgressButtons) onNext({ exercise: id, solutions: answers, score: calculateScore(), type, shuffleMaps: shuffleMaps, isPractice }); // eslint-disable-next-line react-hooks/exhaustive-deps }, [answers, disableProgressButtons]) const next = () => { if (questionIndex + 1 >= questions.length - 1) { onNext({ exercise: id, solutions: answers, score: calculateScore(), type, shuffleMaps: shuffleMaps, isPractice }); } else { setQuestionIndex(questionIndex + 2); } scrollToTop(); }; const back = () => { if (questionIndex === 0) { onBack({ exercise: id, solutions: answers, score: calculateScore(), type, shuffleMaps: shuffleMaps, isPractice }); } else { if (exam?.module === "level" && typeof exam.parts[0].intro !== "undefined" && questionIndex === 0) return; setQuestionIndex(questionIndex - 2); } scrollToTop(); }; const progressButtons = () => (
) const renderAllQuestions = () => questions.map(question => (
question.id === x.question)?.option} onSelectOption={(option) => onSelectOption(option, question)} />
)) const renderTwoQuestions = () => ( <>
{questionIndex < questions.length && ( questions[questionIndex].id === x.question)?.option} onSelectOption={(option) => onSelectOption(option, questions[questionIndex])} /> )}
{questionIndex + 1 < questions.length && (
questions[questionIndex + 1].id === x.question)?.option} onSelectOption={(option) => onSelectOption(option, questions[questionIndex + 1])} />
)} ) return (
{!disableProgressButtons && progressButtons()}
{disableProgressButtons ? renderAllQuestions() : renderTwoQuestions()}
{!disableProgressButtons && progressButtons()}
); }