mirror of
https://github.com/MarijnDoeve/TijdVoorDeTest.git
synced 2026-07-05 23:20:18 +02:00
829028c807
- syncUsagesAfterEdit: use isLocked() instead of !isFinalized() to avoid syncing quizzes with started candidates (would have destroyed GivenAnswer records) - syncToQuiz action: use isLocked() instead of hasStartedCandidates() to block syncing into finalized quizzes that have no started candidates - QuestionBankService::syncToQuiz: remove internal flush(); callers now flush once (syncUsagesAfterEdit after the loop, syncToQuiz action after the call) - QuizRepository::clearQuiz: delete BankQuestionUsage rows and reset finalized_at inside the transaction (previously orphaned usages blocked bank question reassignment; finalizedAt reset was a separate non-atomic flush) - QuizController::finalizeQuiz: add flash when quiz is already finalized - QuestionBankController::delete: block deletion when any usage references a locked quiz - BankQuestion::__toString: remove dead null-coalescing on non-nullable string - SeasonController::addBlankQuiz: align form field with UploadQuizFormType (add translation_domain: false, use translator for label)
436 lines
18 KiB
PHP
436 lines
18 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Tvdt\Controller\Backoffice;
|
|
|
|
use Doctrine\ORM\EntityManagerInterface;
|
|
use Safe\DateTimeImmutable;
|
|
use Symfony\Component\HttpFoundation\RedirectResponse;
|
|
use Symfony\Component\HttpFoundation\Request;
|
|
use Symfony\Component\HttpFoundation\Response;
|
|
use Symfony\Component\HttpKernel\Attribute\AsController;
|
|
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
|
|
use Symfony\Component\Routing\Attribute\Route;
|
|
use Symfony\Component\Routing\Requirement\Requirement;
|
|
use Symfony\Component\Security\Http\Attribute\IsCsrfTokenValid;
|
|
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
|
use Tvdt\Controller\AbstractController;
|
|
use Tvdt\Entity\Answer;
|
|
use Tvdt\Entity\Candidate;
|
|
use Tvdt\Entity\Question;
|
|
use Tvdt\Entity\Quiz;
|
|
use Tvdt\Entity\QuizCandidate;
|
|
use Tvdt\Entity\Season;
|
|
use Tvdt\Enum\FlashType;
|
|
use Tvdt\Exception\ErrorClearingQuizException;
|
|
use Tvdt\Repository\QuizCandidateRepository;
|
|
use Tvdt\Repository\QuizRepository;
|
|
use Tvdt\Security\Voter\SeasonVoter;
|
|
|
|
#[AsController]
|
|
#[IsGranted('ROLE_USER')]
|
|
class QuizController extends AbstractController
|
|
{
|
|
public function __construct(
|
|
private readonly QuizRepository $quizRepository,
|
|
private readonly TranslatorInterface $translator,
|
|
private readonly QuizCandidateRepository $quizCandidateRepository,
|
|
private readonly EntityManagerInterface $em,
|
|
) {}
|
|
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}',
|
|
name: 'tvdt_backoffice_quiz',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
)]
|
|
public function index(Season $season, Quiz $quiz): Response
|
|
{
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_overview', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/overview',
|
|
name: 'tvdt_backoffice_quiz_overview',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
)]
|
|
public function overview(Season $season, Quiz $quiz): Response
|
|
{
|
|
$fetchedQuiz = $this->quizRepository->fetchWithQuestionsAndCandidates($quiz->id);
|
|
|
|
// Create indexed lookup for quiz candidates by candidate ID
|
|
$quizCandidatesByCandidateId = [];
|
|
foreach ($fetchedQuiz->candidateData as $qc) {
|
|
$quizCandidatesByCandidateId[$qc->candidate->id->toString()] = $qc;
|
|
}
|
|
|
|
// Get given answers counts efficiently via database query
|
|
$givenAnswersCountByCandidateId = $this->quizRepository->getGivenAnswersCountPerCandidate($quiz);
|
|
|
|
// Pre-compute candidate data to avoid nested loops in template
|
|
$candidateData = [];
|
|
foreach ($season->candidates as $candidate) {
|
|
$candidateIdString = $candidate->id->toString();
|
|
$candidateData[] = [
|
|
'candidate' => $candidate,
|
|
'quizCandidate' => $quizCandidatesByCandidateId[$candidateIdString] ?? null,
|
|
'givenAnswersCount' => $givenAnswersCountByCandidateId[$candidateIdString] ?? 0,
|
|
];
|
|
}
|
|
|
|
return $this->render('backoffice/quiz.html.twig', [
|
|
'season' => $season,
|
|
'quiz' => $fetchedQuiz,
|
|
'questionErrors' => $fetchedQuiz->getQuestionErrors(),
|
|
'candidateData' => $candidateData,
|
|
'activeTab' => 'overview',
|
|
'template' => 'backoffice/quiz/tab_overview.html.twig',
|
|
]);
|
|
}
|
|
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/result',
|
|
name: 'tvdt_backoffice_quiz_result',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
)]
|
|
public function result(Season $season, Quiz $quiz): Response
|
|
{
|
|
$fetchedQuiz = $this->quizRepository->fetchWithQuestions($quiz->id);
|
|
|
|
return $this->render('backoffice/quiz.html.twig', [
|
|
'season' => $season,
|
|
'quiz' => $fetchedQuiz,
|
|
'result' => $this->quizRepository->getScores($quiz),
|
|
'activeTab' => 'result',
|
|
'template' => 'backoffice/quiz/tab_result.html.twig',
|
|
]);
|
|
}
|
|
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/candidates-list',
|
|
name: 'tvdt_backoffice_quiz_candidates_tab',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
)]
|
|
public function candidatesTab(Season $season, Quiz $quiz): Response
|
|
{
|
|
// Create indexed lookup for quiz candidates by candidate ID
|
|
$quizCandidatesByCandidateId = [];
|
|
foreach ($quiz->candidateData as $qc) {
|
|
$quizCandidatesByCandidateId[$qc->candidate->id->toString()] = $qc;
|
|
}
|
|
|
|
// Get given answers counts efficiently via database query
|
|
$givenAnswersCountByCandidateId = $this->quizRepository->getGivenAnswersCountPerCandidate($quiz);
|
|
|
|
// Pre-compute candidate data to avoid nested loops in template
|
|
$candidateData = [];
|
|
foreach ($season->candidates as $candidate) {
|
|
$candidateIdString = $candidate->id->toString();
|
|
$candidateData[] = [
|
|
'candidate' => $candidate,
|
|
'quizCandidate' => $quizCandidatesByCandidateId[$candidateIdString] ?? null,
|
|
'givenAnswersCount' => $givenAnswersCountByCandidateId[$candidateIdString] ?? 0,
|
|
];
|
|
}
|
|
|
|
return $this->render('backoffice/quiz.html.twig', [
|
|
'season' => $season,
|
|
'quiz' => $quiz,
|
|
'candidateData' => $candidateData,
|
|
'activeTab' => 'candidates',
|
|
'template' => 'backoffice/quiz/tab_candidates_list.html.twig',
|
|
]);
|
|
}
|
|
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/answer-mapping',
|
|
name: 'tvdt_backoffice_quiz_candidates',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
)]
|
|
public function answerMapping(Season $season, Quiz $quiz): Response
|
|
{
|
|
$fetchedQuiz = $this->quizRepository->fetchWithQuestions($quiz->id);
|
|
|
|
if ($fetchedQuiz->questions->isEmpty()) {
|
|
$this->addFlash(FlashType::Warning, $this->translator->trans('This quiz has no questions yet'));
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_overview', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
$firstQuestion = $fetchedQuiz->questions->first();
|
|
\assert($firstQuestion instanceof Question);
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_candidates_question', [
|
|
'seasonCode' => $season->seasonCode,
|
|
'quiz' => $quiz->id,
|
|
'question' => $firstQuestion->id,
|
|
]);
|
|
}
|
|
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/candidates/{question}',
|
|
name: 'tvdt_backoffice_quiz_candidates_question',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
methods: ['GET'],
|
|
)]
|
|
public function candidates_question(Season $season, Quiz $quiz, Question $question): Response
|
|
{
|
|
return $this->render('backoffice/quiz.html.twig', [
|
|
'season' => $season,
|
|
'quiz' => $quiz,
|
|
'question' => $question,
|
|
'candidates' => $season->candidates,
|
|
'activeTab' => 'answers',
|
|
'template' => 'backoffice/quiz/tab_candidates.html.twig',
|
|
]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('candidate_answer')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/candidates/{question}',
|
|
name: 'tvdt_backoffice_quiz_candidates_question_save',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function saveCandidateAnswers(Season $season, Quiz $quiz, Question $question, Request $request): RedirectResponse
|
|
{
|
|
if (false === $season->quizzes->contains($quiz)
|
|
|| false === $quiz->questions->contains($question)) {
|
|
throw new BadRequestHttpException('Invalid quiz or question');
|
|
}
|
|
|
|
$candidateAnswers = $request->request->all('candidate_answer');
|
|
|
|
// Clear existing candidate-answer associations for this question
|
|
foreach ($question->answers as $answer) {
|
|
if (false === $quiz->questions->contains($answer->question)) {
|
|
throw new BadRequestHttpException('Invalid question');
|
|
}
|
|
|
|
$answer->candidates->clear();
|
|
}
|
|
|
|
// Add new associations
|
|
foreach ($candidateAnswers as $candidateId => $answerIds) {
|
|
$candidate = $this->em->getRepository(Candidate::class)->find($candidateId);
|
|
|
|
if (false === $season->candidates->contains($candidate)) {
|
|
throw new BadRequestHttpException('Invalid candidate');
|
|
}
|
|
|
|
foreach ((array) $answerIds as $answerId) {
|
|
$answer = $this->em->getRepository(Answer::class)->find($answerId);
|
|
|
|
if (false === $question->answers->contains($answer)) {
|
|
throw new BadRequestHttpException('Invalid answer');
|
|
}
|
|
|
|
if ($answer && $candidate) {
|
|
$answer->addCandidate($candidate);
|
|
}
|
|
}
|
|
}
|
|
|
|
$this->em->flush();
|
|
|
|
$this->addFlash(FlashType::Success, $this->translator->trans('Candidate answers saved'));
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_candidates_question', [
|
|
'seasonCode' => $season->seasonCode,
|
|
'quiz' => $quiz->id,
|
|
'question' => $question->id,
|
|
]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('enable_quiz')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
|
#[Route(
|
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/enable',
|
|
name: 'tvdt_backoffice_enable',
|
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID.'|null'],
|
|
methods: ['POST'],
|
|
)]
|
|
public function enableQuiz(Season $season, ?Quiz $quiz, Request $request): RedirectResponse
|
|
{
|
|
if ($quiz instanceof Quiz && !$quiz->isFinalized()) {
|
|
$this->addFlash(FlashType::Danger, $this->translator->trans('The quiz must be finalized before it can be activated'));
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_overview', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
$season->activeQuiz = $quiz;
|
|
$this->em->flush();
|
|
|
|
if ($quiz instanceof Quiz) {
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_overview', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
// When deactivating, stay on the quiz page if one was passed
|
|
$previousQuizId = $request->request->getString('redirect_quiz');
|
|
if ('' !== $previousQuizId) {
|
|
$previousQuiz = $this->em->getRepository(Quiz::class)->find($previousQuizId);
|
|
if ($previousQuiz instanceof Quiz && $previousQuiz->season === $season) {
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_overview', ['seasonCode' => $season->seasonCode, 'quiz' => $previousQuiz->id]);
|
|
}
|
|
}
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->seasonCode]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('clear_quiz')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/clear',
|
|
name: 'tvdt_backoffice_quiz_clear',
|
|
requirements: ['quiz' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function clearQuiz(Quiz $quiz): RedirectResponse
|
|
{
|
|
try {
|
|
$this->quizRepository->clearQuiz($quiz);
|
|
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz cleared and no longer finalized'));
|
|
} catch (ErrorClearingQuizException) {
|
|
$this->addFlash(FlashType::Danger, $this->translator->trans('Error clearing quiz'));
|
|
}
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('finalize_quiz')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/finalize',
|
|
name: 'tvdt_backoffice_quiz_finalize',
|
|
requirements: ['quiz' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function finalizeQuiz(Quiz $quiz): RedirectResponse
|
|
{
|
|
if ($quiz->questions->isEmpty() || [] !== $quiz->getQuestionErrors()) {
|
|
$this->addFlash(FlashType::Warning, $this->translator->trans('The quiz cannot be finalized while it has errors'));
|
|
} elseif (!$quiz->isFinalized()) {
|
|
$quiz->finalizedAt = new DateTimeImmutable();
|
|
$this->em->flush();
|
|
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz finalized'));
|
|
} else {
|
|
$this->addFlash(FlashType::Warning, $this->translator->trans('The quiz is already finalized'));
|
|
}
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('unfinalize_quiz')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/unfinalize',
|
|
name: 'tvdt_backoffice_quiz_unfinalize',
|
|
requirements: ['quiz' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function unfinalizeQuiz(Quiz $quiz): RedirectResponse
|
|
{
|
|
if ($quiz->hasStartedCandidates()) {
|
|
$this->addFlash(FlashType::Danger, $this->translator->trans('The quiz has already been filled in and can no longer be altered'));
|
|
} elseif ($quiz->season->activeQuiz === $quiz) {
|
|
$this->addFlash(FlashType::Danger, $this->translator->trans('Deactivate the quiz before undoing the finalization'));
|
|
} else {
|
|
$quiz->finalizedAt = null;
|
|
$this->em->flush();
|
|
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz is no longer finalized'));
|
|
}
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('delete_quiz')]
|
|
#[IsGranted(SeasonVoter::DELETE, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/delete',
|
|
name: 'tvdt_backoffice_quiz_delete',
|
|
requirements: ['quiz' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function deleteQuiz(Quiz $quiz): RedirectResponse
|
|
{
|
|
$this->quizRepository->deleteQuiz($quiz);
|
|
|
|
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz deleted'));
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $quiz->season->seasonCode]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('candidate_correction')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/candidate/{candidate}/modify_correction',
|
|
name: 'tvdt_backoffice_modify_correction',
|
|
requirements: ['quiz' => Requirement::UUID, 'candidate' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function modifyCorrection(Quiz $quiz, Candidate $candidate, Request $request): RedirectResponse
|
|
{
|
|
$corrections = (float) $request->request->get('corrections');
|
|
|
|
$this->quizCandidateRepository->setCorrectionsForCandidate($quiz, $candidate, $corrections);
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('candidate_penalty')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/candidate/{candidate}/modify_penalty',
|
|
name: 'tvdt_backoffice_modify_penalty',
|
|
requirements: ['quiz' => Requirement::UUID, 'candidate' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function modifyPenalty(Quiz $quiz, Candidate $candidate, Request $request): RedirectResponse
|
|
{
|
|
$penalty = (int) $request->request->get('penalty');
|
|
|
|
$this->quizCandidateRepository->setPenaltyForCandidate($quiz, $candidate, $penalty);
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
|
|
#[IsCsrfTokenValid('toggle_candidate')]
|
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
|
#[Route(
|
|
'/backoffice/quiz/{quiz}/candidate/{candidate}/toggle',
|
|
name: 'tvdt_backoffice_toggle_candidate',
|
|
requirements: ['quiz' => Requirement::UUID, 'candidate' => Requirement::UUID],
|
|
methods: ['POST'],
|
|
)]
|
|
public function toggleCandidate(Quiz $quiz, Candidate $candidate): RedirectResponse
|
|
{
|
|
$quizCandidate = $this->quizCandidateRepository->findOneBy([
|
|
'quiz' => $quiz,
|
|
'candidate' => $candidate,
|
|
]);
|
|
|
|
if (!$quizCandidate instanceof QuizCandidate) {
|
|
// Create new QuizCandidate if it doesn't exist (inactive by default when first toggling)
|
|
$quizCandidate = new QuizCandidate($quiz, $candidate);
|
|
$quizCandidate->active = false;
|
|
$this->em->persist($quizCandidate);
|
|
} else {
|
|
$quizCandidate->active = !$quizCandidate->active;
|
|
}
|
|
|
|
$this->em->flush();
|
|
|
|
$this->addFlash(FlashType::Success, $this->translator->trans('Candidate status updated'));
|
|
|
|
return $this->redirectToRoute('tvdt_backoffice_quiz_candidates_tab', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
|
|
}
|
|
}
|