mirror of
https://github.com/MarijnDoeve/TijdVoorDeTest.git
synced 2026-03-07 13:14:20 +01:00
Compare commits
34 Commits
f6715de330
...
v0.0.2
| Author | SHA1 | Date | |
|---|---|---|---|
|
0319224979
|
|||
|
81e471a760
|
|||
|
cfb69c8dab
|
|||
|
35ec71302b
|
|||
|
b7a570928a
|
|||
|
d80436534f
|
|||
|
|
14e2dd490e | ||
|
|
68e54b1110 | ||
|
|
e615b2cdea | ||
|
|
4b45a2c557 | ||
| c6fe553341 | |||
|
69a2b9c811
|
|||
|
f31a7d527d
|
|||
| ed3cf7644f | |||
| 77d21b004f | |||
|
379fafcd16
|
|||
|
7586d2d8ac
|
|||
|
9e41376244
|
|||
| 2bfef94bbe | |||
|
a8c4cba968
|
|||
|
d5566d4737
|
|||
|
366bc36520
|
|||
|
79b24b0d44
|
|||
|
7f93680987
|
|||
|
bdbff32256
|
|||
| ebadc24b59 | |||
| e0075fdcdc | |||
| 06aafefffc | |||
| ff6534fa81 | |||
| 6a77df402d | |||
| 79236d84e9 | |||
| beb8d13dde | |||
| 3e724ff1fb | |||
|
e131d3b8d9
|
@@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"image": "mcr.microsoft.com/devcontainers/universal:2",
|
|
||||||
"features": {"ghcr.io/devcontainers/features/php:1": {
|
|
||||||
"version": "8.3"
|
|
||||||
}}
|
|
||||||
}
|
|
||||||
@@ -35,6 +35,9 @@ indent_size = 4
|
|||||||
trim_trailing_whitespace = false
|
trim_trailing_whitespace = false
|
||||||
indent_size = 2
|
indent_size = 2
|
||||||
|
|
||||||
|
[config/**/*.{yaml,yml}]
|
||||||
|
indent_size = 4
|
||||||
|
|
||||||
[.github/workflows/*.yml]
|
[.github/workflows/*.yml]
|
||||||
indent_size = 2
|
indent_size = 2
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# define your env variables for the test env here
|
# define your env variables for the test env here
|
||||||
KERNEL_CLASS='App\Kernel'
|
KERNEL_CLASS='Tvdt\Kernel'
|
||||||
APP_SECRET='$ecretf0rt3st'
|
APP_SECRET='$ecretf0rt3st'
|
||||||
SYMFONY_DEPRECATIONS_HELPER=999999
|
SYMFONY_DEPRECATIONS_HELPER=999999
|
||||||
|
|||||||
11
.github/dependabot.yml
vendored
Normal file
11
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
# To get started with Dependabot version updates, you'll need to specify which
|
||||||
|
# package ecosystems to update and where the package manifests are located.
|
||||||
|
# Please see the documentation for all configuration options:
|
||||||
|
# https://docs.github.com/code-security/dependabot/dependabot-version-updates/configuration-options-for-the-dependabot.yml-file
|
||||||
|
|
||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: "composer" # See documentation for possible values
|
||||||
|
directory: "/" # Location of package manifests
|
||||||
|
schedule:
|
||||||
|
interval: "weekly"
|
||||||
26
.github/workflows/ci.yml
vendored
26
.github/workflows/ci.yml
vendored
@@ -4,6 +4,8 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
|
tags:
|
||||||
|
- '*'
|
||||||
pull_request: ~
|
pull_request: ~
|
||||||
workflow_dispatch: ~
|
workflow_dispatch: ~
|
||||||
|
|
||||||
@@ -18,10 +20,12 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
- name: Lint Dockerfile
|
||||||
|
uses: hadolint/hadolint-action@v3.1.0
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3
|
uses: docker/setup-buildx-action@v3
|
||||||
- name: Build Docker images
|
- name: Build Docker images
|
||||||
uses: docker/bake-action@v4
|
uses: docker/bake-action@v5
|
||||||
with:
|
with:
|
||||||
pull: true
|
pull: true
|
||||||
load: true
|
load: true
|
||||||
@@ -50,16 +54,20 @@ jobs:
|
|||||||
- name: Run migrations
|
- name: Run migrations
|
||||||
run: docker compose exec -T php bin/console -e test doctrine:migrations:migrate --no-interaction
|
run: docker compose exec -T php bin/console -e test doctrine:migrations:migrate --no-interaction
|
||||||
- name: Run PHPUnit
|
- name: Run PHPUnit
|
||||||
if: false # Remove this line when the tests are ready
|
|
||||||
run: docker compose exec -T php vendor/bin/phpunit
|
run: docker compose exec -T php vendor/bin/phpunit
|
||||||
- name: Doctrine Schema Validator
|
- name: Doctrine Schema Validator
|
||||||
run: docker compose exec -T php bin/console -e test doctrine:schema:validate
|
run: docker compose exec -T php bin/console -e test doctrine:schema:validate
|
||||||
lint:
|
deploy:
|
||||||
name: Docker Lint
|
name: Deploy
|
||||||
|
environment:
|
||||||
|
name: ${{ startsWith(github.ref, 'refs/tags/') && 'production' || (github.ref == 'refs/heads/main' && 'acceptance' || '') }}
|
||||||
|
url: ${{ vars.URL }}
|
||||||
|
needs: tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- shell: bash
|
||||||
uses: actions/checkout@v4
|
env:
|
||||||
- name: Lint Dockerfile
|
PORTAINER_WEBHOOK: ${{secrets.PORTAINER_WEBHOOK}}
|
||||||
uses: hadolint/hadolint-action@v3.1.0
|
run: |
|
||||||
|
curl -v -X POST "$PORTAINER_WEBHOOK"
|
||||||
|
|||||||
9
.idea/TijdVoorDeTest.iml
generated
9
.idea/TijdVoorDeTest.iml
generated
@@ -2,8 +2,8 @@
|
|||||||
<module type="WEB_MODULE" version="4">
|
<module type="WEB_MODULE" version="4">
|
||||||
<component name="NewModuleRootManager">
|
<component name="NewModuleRootManager">
|
||||||
<content url="file://$MODULE_DIR$">
|
<content url="file://$MODULE_DIR$">
|
||||||
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" packagePrefix="App\" />
|
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" packagePrefix="Tvdt\" />
|
||||||
<sourceFolder url="file://$MODULE_DIR$/tests" isTestSource="true" packagePrefix="App\Tests\" />
|
<sourceFolder url="file://$MODULE_DIR$/tests" isTestSource="true" packagePrefix="Tvdt\Tests\" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/clue/ndjson-react" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/clue/ndjson-react" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/composer" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/composer" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/doctrine/collections" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/doctrine/collections" />
|
||||||
@@ -134,7 +134,6 @@
|
|||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/browser-kit" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/browser-kit" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/dom-crawler" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/dom-crawler" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/mailer" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/mailer" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/polyfill-php84" />
|
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfonycasts/verify-email-bundle" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfonycasts/verify-email-bundle" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/guzzlehttp/psr7" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/guzzlehttp/psr7" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/jean85/pretty-package-versions" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/jean85/pretty-package-versions" />
|
||||||
@@ -164,6 +163,10 @@
|
|||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/http-client-contracts" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/http-client-contracts" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/symfonycasts/sass-bundle" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfonycasts/sass-bundle" />
|
||||||
<excludeFolder url="file://$MODULE_DIR$/vendor/twig/intl-extra" />
|
<excludeFolder url="file://$MODULE_DIR$/vendor/twig/intl-extra" />
|
||||||
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/stimulus-bundle" />
|
||||||
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/ux-turbo" />
|
||||||
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/polyfill-uuid" />
|
||||||
|
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/brevo-mailer" />
|
||||||
</content>
|
</content>
|
||||||
<orderEntry type="inheritedJdk" />
|
<orderEntry type="inheritedJdk" />
|
||||||
<orderEntry type="sourceFolder" forTests="false" />
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
|
|||||||
26
.idea/php.xml
generated
26
.idea/php.xml
generated
@@ -7,11 +7,6 @@
|
|||||||
</laravel_pint_by_interpreter>
|
</laravel_pint_by_interpreter>
|
||||||
</laravel_pint_settings>
|
</laravel_pint_settings>
|
||||||
</component>
|
</component>
|
||||||
<component name="MessDetector">
|
|
||||||
<phpmd_settings>
|
|
||||||
<phpmd_by_interpreter asDefaultInterpreter="true" interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" timeout="30000" />
|
|
||||||
</phpmd_settings>
|
|
||||||
</component>
|
|
||||||
<component name="MessDetectorOptionsConfiguration">
|
<component name="MessDetectorOptionsConfiguration">
|
||||||
<option name="transferred" value="true" />
|
<option name="transferred" value="true" />
|
||||||
</component>
|
</component>
|
||||||
@@ -108,7 +103,6 @@
|
|||||||
<path value="$PROJECT_DIR$/vendor/symfony/http-client" />
|
<path value="$PROJECT_DIR$/vendor/symfony/http-client" />
|
||||||
<path value="$PROJECT_DIR$/vendor/symfony/stopwatch" />
|
<path value="$PROJECT_DIR$/vendor/symfony/stopwatch" />
|
||||||
<path value="$PROJECT_DIR$/vendor/symfony/finder" />
|
<path value="$PROJECT_DIR$/vendor/symfony/finder" />
|
||||||
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-php84" />
|
|
||||||
<path value="$PROJECT_DIR$/vendor/symfony/yaml" />
|
<path value="$PROJECT_DIR$/vendor/symfony/yaml" />
|
||||||
<path value="$PROJECT_DIR$/vendor/symfony/mime" />
|
<path value="$PROJECT_DIR$/vendor/symfony/mime" />
|
||||||
<path value="$PROJECT_DIR$/vendor/symfony/psr-http-message-bridge" />
|
<path value="$PROJECT_DIR$/vendor/symfony/psr-http-message-bridge" />
|
||||||
@@ -199,11 +193,16 @@
|
|||||||
<path value="$PROJECT_DIR$/vendor/nikic/php-parser" />
|
<path value="$PROJECT_DIR$/vendor/nikic/php-parser" />
|
||||||
<path value="$PROJECT_DIR$/vendor/twig/html-extra" />
|
<path value="$PROJECT_DIR$/vendor/twig/html-extra" />
|
||||||
<path value="$PROJECT_DIR$/vendor/twig/twig" />
|
<path value="$PROJECT_DIR$/vendor/twig/twig" />
|
||||||
|
<path value="$PROJECT_DIR$/vendor/symfony/ux-turbo" />
|
||||||
|
<path value="$PROJECT_DIR$/vendor/symfony/stimulus-bundle" />
|
||||||
|
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-uuid" />
|
||||||
|
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-php84" />
|
||||||
|
<path value="$PROJECT_DIR$/vendor/symfony/brevo-mailer" />
|
||||||
</include_path>
|
</include_path>
|
||||||
</component>
|
</component>
|
||||||
<component name="PhpInterpreters">
|
<component name="PhpInterpreters">
|
||||||
<interpreters>
|
<interpreters>
|
||||||
<interpreter id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" name="Compose PHP 8.3" home="docker-compose://DATA" auto="false" debugger_id="php.debugger.XDebug">
|
<interpreter id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" name="Compose PHP 8.4" home="docker-compose://DATA" auto="false" debugger_id="php.debugger.XDebug">
|
||||||
<remote_data INTERPRETER_PATH="php" HELPERS_PATH="/opt/.phpstorm_helpers" VALID="true" RUN_AS_ROOT_VIA_SUDO="false" DOCKER_ACCOUNT_NAME="Colima" DOCKER_COMPOSE_SERVICE_NAME="php" DOCKER_REMOTE_PROJECT_PATH="/opt/project">
|
<remote_data INTERPRETER_PATH="php" HELPERS_PATH="/opt/.phpstorm_helpers" VALID="true" RUN_AS_ROOT_VIA_SUDO="false" DOCKER_ACCOUNT_NAME="Colima" DOCKER_COMPOSE_SERVICE_NAME="php" DOCKER_REMOTE_PROJECT_PATH="/opt/project">
|
||||||
<type_data command="EXEC" />
|
<type_data command="EXEC" />
|
||||||
<dockerComposeConfigurationPaths>
|
<dockerComposeConfigurationPaths>
|
||||||
@@ -217,15 +216,15 @@
|
|||||||
</component>
|
</component>
|
||||||
<component name="PhpInterpretersPhpInfoCache">
|
<component name="PhpInterpretersPhpInfoCache">
|
||||||
<phpInfoCache>
|
<phpInfoCache>
|
||||||
<interpreter name="Compose PHP 8.3">
|
<interpreter name="Compose PHP 8.4">
|
||||||
<phpinfo binary_type="PHP" php_cgi="/usr/local/bin/php-cgi" php_cli="/usr/local/bin/php" path_separator=":" version="8.3.19">
|
<phpinfo binary_type="PHP" php_cgi="/usr/local/bin/php-cgi" php_cli="/usr/local/bin/php" path_separator=":" version="8.4.8">
|
||||||
<additional_php_ini>/usr/local/etc/php/conf.d/docker-php-ext-apcu.ini, /usr/local/etc/php/conf.d/docker-php-ext-intl.ini, /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini, /usr/local/etc/php/conf.d/docker-php-ext-pdo_pgsql.ini, /usr/local/etc/php/conf.d/docker-php-ext-sodium.ini, /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini, /usr/local/etc/php/conf.d/docker-php-ext-zip.ini, /usr/local/etc/php/app.conf.d/10-app.ini, /usr/local/etc/php/app.conf.d/20-app.dev.ini</additional_php_ini>
|
<additional_php_ini>/usr/local/etc/php/conf.d/docker-php-ext-apcu.ini, /usr/local/etc/php/conf.d/docker-php-ext-excimer.ini, /usr/local/etc/php/conf.d/docker-php-ext-gd.ini, /usr/local/etc/php/conf.d/docker-php-ext-intl.ini, /usr/local/etc/php/conf.d/docker-php-ext-opcache.ini, /usr/local/etc/php/conf.d/docker-php-ext-pdo_pgsql.ini, /usr/local/etc/php/conf.d/docker-php-ext-sodium.ini, /usr/local/etc/php/conf.d/docker-php-ext-uuid.ini, /usr/local/etc/php/conf.d/docker-php-ext-xdebug.ini, /usr/local/etc/php/conf.d/docker-php-ext-zip.ini, /usr/local/etc/php/app.conf.d/10-app.ini, /usr/local/etc/php/app.conf.d/20-app.dev.ini</additional_php_ini>
|
||||||
<configuration_file>/usr/local/etc/php/php.ini</configuration_file>
|
<configuration_file>/usr/local/etc/php/php.ini</configuration_file>
|
||||||
<configuration_options>
|
<configuration_options>
|
||||||
<configuration_option name="include_path" value=".:/usr/local/lib/php" />
|
<configuration_option name="include_path" value=".:/usr/local/lib/php" />
|
||||||
</configuration_options>
|
</configuration_options>
|
||||||
<debuggers>
|
<debuggers>
|
||||||
<debugger_info debugger="xdebug" debugger_version="3.4.2">
|
<debugger_info debugger="xdebug" debugger_version="3.4.3">
|
||||||
<debug_extensions />
|
<debug_extensions />
|
||||||
</debugger_info>
|
</debugger_info>
|
||||||
</debuggers>
|
</debuggers>
|
||||||
@@ -242,8 +241,10 @@
|
|||||||
<extension name="curl" />
|
<extension name="curl" />
|
||||||
<extension name="date" />
|
<extension name="date" />
|
||||||
<extension name="dom" />
|
<extension name="dom" />
|
||||||
|
<extension name="excimer" />
|
||||||
<extension name="fileinfo" />
|
<extension name="fileinfo" />
|
||||||
<extension name="filter" />
|
<extension name="filter" />
|
||||||
|
<extension name="gd" />
|
||||||
<extension name="hash" />
|
<extension name="hash" />
|
||||||
<extension name="iconv" />
|
<extension name="iconv" />
|
||||||
<extension name="intl" />
|
<extension name="intl" />
|
||||||
@@ -263,6 +264,7 @@
|
|||||||
<extension name="sqlite3" />
|
<extension name="sqlite3" />
|
||||||
<extension name="standard" />
|
<extension name="standard" />
|
||||||
<extension name="tokenizer" />
|
<extension name="tokenizer" />
|
||||||
|
<extension name="uuid" />
|
||||||
<extension name="xdebug" />
|
<extension name="xdebug" />
|
||||||
<extension name="xml" />
|
<extension name="xml" />
|
||||||
<extension name="xmlreader" />
|
<extension name="xmlreader" />
|
||||||
@@ -274,7 +276,7 @@
|
|||||||
</interpreter>
|
</interpreter>
|
||||||
</phpInfoCache>
|
</phpInfoCache>
|
||||||
</component>
|
</component>
|
||||||
<component name="PhpProjectSharedConfiguration" php_language_level="8.3" />
|
<component name="PhpProjectSharedConfiguration" php_language_level="8.4" />
|
||||||
<component name="PhpStan">
|
<component name="PhpStan">
|
||||||
<PhpStan_settings>
|
<PhpStan_settings>
|
||||||
<phpstan_by_interpreter interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" tool_path="vendor/bin/phpstan" timeout="60000" />
|
<phpstan_by_interpreter interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" tool_path="vendor/bin/phpstan" timeout="60000" />
|
||||||
|
|||||||
1
.idea/symfony2.xml
generated
1
.idea/symfony2.xml
generated
@@ -2,5 +2,6 @@
|
|||||||
<project version="4">
|
<project version="4">
|
||||||
<component name="Symfony2PluginSettings">
|
<component name="Symfony2PluginSettings">
|
||||||
<option name="pluginEnabled" value="true" />
|
<option name="pluginEnabled" value="true" />
|
||||||
|
<option name="profilerCsvPath" value="" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
#syntax=docker/dockerfile:1
|
#syntax=docker/dockerfile:1
|
||||||
|
|
||||||
# Versions
|
# Versions
|
||||||
FROM dunglas/frankenphp:1-php8.3 AS frankenphp_upstream
|
FROM dunglas/frankenphp:1-php8.4 AS frankenphp_upstream
|
||||||
|
|
||||||
# The different stages of this Dockerfile are meant to be built into separate images
|
# The different stages of this Dockerfile are meant to be built into separate images
|
||||||
# https://docs.docker.com/develop/develop-images/multistage-build/#stop-at-a-specific-build-stage
|
# https://docs.docker.com/develop/develop-images/multistage-build/#stop-at-a-specific-build-stage
|
||||||
@@ -31,7 +31,6 @@ RUN set -eux; \
|
|||||||
intl \
|
intl \
|
||||||
opcache \
|
opcache \
|
||||||
zip \
|
zip \
|
||||||
uuid \
|
|
||||||
gd \
|
gd \
|
||||||
excimer-1.2.3 \
|
excimer-1.2.3 \
|
||||||
;
|
;
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import 'bootstrap/dist/css/bootstrap.min.css'
|
|
||||||
import * as bootstrap from 'bootstrap'
|
import * as bootstrap from 'bootstrap'
|
||||||
|
import './bootstrap.js';
|
||||||
|
import 'bootstrap/dist/css/bootstrap.min.css'
|
||||||
|
|
||||||
import './styles/backoffice.scss';
|
import './styles/backoffice.scss';
|
||||||
|
|||||||
5
assets/bootstrap.js
vendored
Normal file
5
assets/bootstrap.js
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import { startStimulusApp } from '@symfony/stimulus-bundle';
|
||||||
|
|
||||||
|
const app = startStimulusApp();
|
||||||
|
// register any custom, 3rd party controllers here
|
||||||
|
// app.register('some_controller_name', SomeImportedController);
|
||||||
15
assets/controllers.json
Normal file
15
assets/controllers.json
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"controllers": {
|
||||||
|
"@symfony/ux-turbo": {
|
||||||
|
"turbo-core": {
|
||||||
|
"enabled": false,
|
||||||
|
"fetch": "eager"
|
||||||
|
},
|
||||||
|
"mercure-turbo-stream": {
|
||||||
|
"enabled": false,
|
||||||
|
"fetch": "eager"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"entrypoints": []
|
||||||
|
}
|
||||||
17
assets/controllers/bo/quiz_controller.js
Normal file
17
assets/controllers/bo/quiz_controller.js
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import {Controller} from '@hotwired/stimulus';
|
||||||
|
import * as bootstrap from 'bootstrap'
|
||||||
|
|
||||||
|
export default class extends Controller {
|
||||||
|
connect() {
|
||||||
|
const tooltipTriggerList = document.querySelectorAll('[data-bs-toggle="tooltip"]')
|
||||||
|
const tooltipList = [...tooltipTriggerList].map(tooltipTriggerEl => new bootstrap.Tooltip(tooltipTriggerEl))
|
||||||
|
}
|
||||||
|
|
||||||
|
clearQuiz() {
|
||||||
|
new bootstrap.Modal('#clearQuizModal').show();
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteQuiz() {
|
||||||
|
new bootstrap.Modal('#deleteQuizModal').show();
|
||||||
|
}
|
||||||
|
}
|
||||||
79
assets/controllers/csrf_protection_controller.js
Normal file
79
assets/controllers/csrf_protection_controller.js
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
const nameCheck = /^[-_a-zA-Z0-9]{4,22}$/;
|
||||||
|
const tokenCheck = /^[-_\/+a-zA-Z0-9]{24,}$/;
|
||||||
|
|
||||||
|
// Generate and double-submit a CSRF token in a form field and a cookie, as defined by Symfony's SameOriginCsrfTokenManager
|
||||||
|
document.addEventListener('submit', function (event) {
|
||||||
|
generateCsrfToken(event.target);
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
// When @hotwired/turbo handles form submissions, send the CSRF token in a header in addition to a cookie
|
||||||
|
// The `framework.csrf_protection.check_header` config option needs to be enabled for the header to be checked
|
||||||
|
document.addEventListener('turbo:submit-start', function (event) {
|
||||||
|
const h = generateCsrfHeaders(event.detail.formSubmission.formElement);
|
||||||
|
Object.keys(h).map(function (k) {
|
||||||
|
event.detail.formSubmission.fetchRequest.headers[k] = h[k];
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// When @hotwired/turbo handles form submissions, remove the CSRF cookie once a form has been submitted
|
||||||
|
document.addEventListener('turbo:submit-end', function (event) {
|
||||||
|
removeCsrfToken(event.detail.formSubmission.formElement);
|
||||||
|
});
|
||||||
|
|
||||||
|
export function generateCsrfToken (formElement) {
|
||||||
|
const csrfField = formElement.querySelector('input[data-controller="csrf-protection"], input[name="_csrf_token"]');
|
||||||
|
|
||||||
|
if (!csrfField) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let csrfCookie = csrfField.getAttribute('data-csrf-protection-cookie-value');
|
||||||
|
let csrfToken = csrfField.value;
|
||||||
|
|
||||||
|
if (!csrfCookie && nameCheck.test(csrfToken)) {
|
||||||
|
csrfField.setAttribute('data-csrf-protection-cookie-value', csrfCookie = csrfToken);
|
||||||
|
csrfField.defaultValue = csrfToken = btoa(String.fromCharCode.apply(null, (window.crypto || window.msCrypto).getRandomValues(new Uint8Array(18))));
|
||||||
|
csrfField.dispatchEvent(new Event('change', { bubbles: true }));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (csrfCookie && tokenCheck.test(csrfToken)) {
|
||||||
|
const cookie = csrfCookie + '_' + csrfToken + '=' + csrfCookie + '; path=/; samesite=strict';
|
||||||
|
document.cookie = window.location.protocol === 'https:' ? '__Host-' + cookie + '; secure' : cookie;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function generateCsrfHeaders (formElement) {
|
||||||
|
const headers = {};
|
||||||
|
const csrfField = formElement.querySelector('input[data-controller="csrf-protection"], input[name="_csrf_token"]');
|
||||||
|
|
||||||
|
if (!csrfField) {
|
||||||
|
return headers;
|
||||||
|
}
|
||||||
|
|
||||||
|
const csrfCookie = csrfField.getAttribute('data-csrf-protection-cookie-value');
|
||||||
|
|
||||||
|
if (tokenCheck.test(csrfField.value) && nameCheck.test(csrfCookie)) {
|
||||||
|
headers[csrfCookie] = csrfField.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return headers;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function removeCsrfToken (formElement) {
|
||||||
|
const csrfField = formElement.querySelector('input[data-controller="csrf-protection"], input[name="_csrf_token"]');
|
||||||
|
|
||||||
|
if (!csrfField) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const csrfCookie = csrfField.getAttribute('data-csrf-protection-cookie-value');
|
||||||
|
|
||||||
|
if (tokenCheck.test(csrfField.value) && nameCheck.test(csrfCookie)) {
|
||||||
|
const cookie = csrfCookie + '_' + csrfField.value + '=0; path=/; samesite=strict; max-age=0';
|
||||||
|
|
||||||
|
document.cookie = window.location.protocol === 'https:' ? '__Host-' + cookie + '; secure' : cookie;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* stimulusFetch: 'lazy' */
|
||||||
|
export default 'csrf-protection-controller';
|
||||||
10
assets/controllers/elimination_controller.js
Normal file
10
assets/controllers/elimination_controller.js
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import {Controller} from '@hotwired/stimulus';
|
||||||
|
|
||||||
|
export default class extends Controller {
|
||||||
|
next() {
|
||||||
|
const currentUrl = window.location.href;
|
||||||
|
const urlParts = currentUrl.split('/');
|
||||||
|
urlParts.pop();
|
||||||
|
window.location.href = urlParts.join('/');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,24 +1,6 @@
|
|||||||
|
import './bootstrap.js';
|
||||||
import 'bootstrap/dist/css/bootstrap.min.css'
|
import 'bootstrap/dist/css/bootstrap.min.css'
|
||||||
import * as bootstrap from 'bootstrap'
|
import * as bootstrap from 'bootstrap'
|
||||||
|
|
||||||
import './styles/app.scss'
|
import './styles/app.scss'
|
||||||
|
|
||||||
document.addEventListener('DOMContentLoaded', function () {
|
|
||||||
// Check if we're on the elimination candidate screen
|
|
||||||
const eliminationScreen = document.querySelector('.elimination-screen');
|
|
||||||
if (eliminationScreen) {
|
|
||||||
// Add event listener for any keypress
|
|
||||||
document.addEventListener('keydown', function (event) {
|
|
||||||
// Get the current URL
|
|
||||||
const currentUrl = window.location.href;
|
|
||||||
// Extract the elimination ID from the URL
|
|
||||||
const urlParts = currentUrl.split('/');
|
|
||||||
// Remove the candidate hash (last part of the URL)
|
|
||||||
urlParts.pop();
|
|
||||||
// Construct the URL to the main elimination page
|
|
||||||
const redirectUrl = urlParts.join('/');
|
|
||||||
// Redirect to the main elimination page
|
|
||||||
window.location.href = redirectUrl;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
use App\Kernel;
|
use Tvdt\Kernel;
|
||||||
use Symfony\Bundle\FrameworkBundle\Console\Application;
|
use Symfony\Bundle\FrameworkBundle\Console\Application;
|
||||||
|
|
||||||
if (!is_dir(dirname(__DIR__).'/vendor')) {
|
if (!is_dir(dirname(__DIR__).'/vendor')) {
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ services:
|
|||||||
# See https://xdebug.org/docs/all_settings#mode
|
# See https://xdebug.org/docs/all_settings#mode
|
||||||
XDEBUG_MODE: "${XDEBUG_MODE:-off}"
|
XDEBUG_MODE: "${XDEBUG_MODE:-off}"
|
||||||
MAILER_DSN: "smtp://mailer:1025"
|
MAILER_DSN: "smtp://mailer:1025"
|
||||||
|
PHP_CS_FIXER_IGNORE_ENV: 1
|
||||||
extra_hosts:
|
extra_hosts:
|
||||||
# Ensure that host.docker.internal is correctly defined on Linux
|
# Ensure that host.docker.internal is correctly defined on Linux
|
||||||
- host.docker.internal:host-gateway
|
- host.docker.internal:host-gateway
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ services:
|
|||||||
APP_SECRET: ${APP_SECRET}
|
APP_SECRET: ${APP_SECRET}
|
||||||
MERCURE_PUBLISHER_JWT_KEY: ${CADDY_MERCURE_JWT_SECRET}
|
MERCURE_PUBLISHER_JWT_KEY: ${CADDY_MERCURE_JWT_SECRET}
|
||||||
MERCURE_SUBSCRIBER_JWT_KEY: ${CADDY_MERCURE_JWT_SECRET}
|
MERCURE_SUBSCRIBER_JWT_KEY: ${CADDY_MERCURE_JWT_SECRET}
|
||||||
|
MAILER_DSN: ${MAILER_DSN}
|
||||||
MAILER_SENDER: ${MAILER_SENDER}
|
MAILER_SENDER: ${MAILER_SENDER}
|
||||||
SENTRY_DSN: ${SENTRY_DSN}
|
SENTRY_DSN: ${SENTRY_DSN}
|
||||||
labels:
|
labels:
|
||||||
@@ -23,6 +24,8 @@ services:
|
|||||||
database:
|
database:
|
||||||
networks:
|
networks:
|
||||||
- internal
|
- internal
|
||||||
|
ports:
|
||||||
|
- "5430:5432"
|
||||||
networks:
|
networks:
|
||||||
web:
|
web:
|
||||||
external: true
|
external: true
|
||||||
|
|||||||
@@ -6,38 +6,40 @@
|
|||||||
"minimum-stability": "stable",
|
"minimum-stability": "stable",
|
||||||
"prefer-stable": true,
|
"prefer-stable": true,
|
||||||
"require": {
|
"require": {
|
||||||
"php": ">=8.3.15",
|
"php": ">=8.4",
|
||||||
"ext-ctype": "*",
|
"ext-ctype": "*",
|
||||||
"ext-iconv": "*",
|
"ext-iconv": "*",
|
||||||
"doctrine/dbal": "^4.2.3",
|
"doctrine/dbal": "^4.3.3",
|
||||||
"doctrine/doctrine-bundle": "^2.14.0",
|
"doctrine/doctrine-bundle": "^2.16.2",
|
||||||
"doctrine/doctrine-migrations-bundle": "^3.4.2",
|
"doctrine/doctrine-migrations-bundle": "^3.4.2",
|
||||||
"doctrine/orm": "^3.3.3",
|
"doctrine/orm": "^3.5.2",
|
||||||
"easycorp/easyadmin-bundle": "^4.24.7",
|
"easycorp/easyadmin-bundle": "^4.25.1",
|
||||||
"phpdocumentor/reflection-docblock": "^5.6.2",
|
"phpdocumentor/reflection-docblock": "^5.6.3",
|
||||||
"phpoffice/phpspreadsheet": "^4.2.0",
|
"phpoffice/phpspreadsheet": "^5.1",
|
||||||
"phpstan/phpdoc-parser": "^2.1",
|
"phpstan/phpdoc-parser": "^2.3",
|
||||||
"runtime/frankenphp-symfony": "^0.2.0",
|
"runtime/frankenphp-symfony": "^0.2.0",
|
||||||
"sentry/sentry-symfony": "^5.2",
|
"sentry/sentry-symfony": "^5.6",
|
||||||
"symfony/asset": "7.2.*",
|
"symfony/asset": "7.3.*",
|
||||||
"symfony/asset-mapper": "7.2.*",
|
"symfony/asset-mapper": "7.3.*",
|
||||||
"symfony/console": "7.2.*",
|
"symfony/brevo-mailer": "7.3.*",
|
||||||
"symfony/dotenv": "7.2.*",
|
"symfony/console": "7.3.*",
|
||||||
"symfony/flex": "^2.7.0",
|
"symfony/dotenv": "7.3.*",
|
||||||
"symfony/form": "7.2.*",
|
"symfony/flex": "^2.8.2",
|
||||||
"symfony/framework-bundle": "7.2.*",
|
"symfony/form": "7.3.*",
|
||||||
"symfony/mailer": "7.2.*",
|
"symfony/framework-bundle": "7.3.*",
|
||||||
"symfony/property-access": "7.2.*",
|
"symfony/mailer": "7.3.*",
|
||||||
"symfony/property-info": "7.2.*",
|
"symfony/property-access": "7.3.*",
|
||||||
"symfony/runtime": "7.2.*",
|
"symfony/property-info": "7.3.*",
|
||||||
"symfony/security-bundle": "7.2.*",
|
"symfony/runtime": "7.3.*",
|
||||||
"symfony/security-csrf": "7.2.*",
|
"symfony/security-bundle": "7.3.*",
|
||||||
"symfony/serializer": "7.2.*",
|
"symfony/security-csrf": "7.3.*",
|
||||||
"symfony/twig-bundle": "7.2.*",
|
"symfony/serializer": "7.3.*",
|
||||||
"symfony/uid": "7.2.*",
|
"symfony/twig-bundle": "7.3.*",
|
||||||
"symfony/yaml": "7.2.*",
|
"symfony/uid": "7.3.*",
|
||||||
"symfonycasts/sass-bundle": "^0.8.2",
|
"symfony/ux-turbo": "^2.30.0",
|
||||||
"symfonycasts/verify-email-bundle": "^1.17.3",
|
"symfony/yaml": "7.3.*",
|
||||||
|
"symfonycasts/sass-bundle": "^0.8.3",
|
||||||
|
"symfonycasts/verify-email-bundle": "^1.17.4",
|
||||||
"thecodingmachine/safe": "^3.3.0",
|
"thecodingmachine/safe": "^3.3.0",
|
||||||
"twig/extra-bundle": "^3.21",
|
"twig/extra-bundle": "^3.21",
|
||||||
"twig/intl-extra": "^3.21",
|
"twig/intl-extra": "^3.21",
|
||||||
@@ -45,23 +47,23 @@
|
|||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"doctrine/doctrine-fixtures-bundle": "^4.1",
|
"doctrine/doctrine-fixtures-bundle": "^4.1",
|
||||||
"friendsofphp/php-cs-fixer": "^3.75.0",
|
"friendsofphp/php-cs-fixer": "^3.88.2",
|
||||||
"phpstan/extension-installer": "^1.4.3",
|
"phpstan/extension-installer": "^1.4.3",
|
||||||
"phpstan/phpstan": "^2.1.17",
|
"phpstan/phpstan": "^2.1.29",
|
||||||
"phpstan/phpstan-doctrine": "^2.0.3",
|
"phpstan/phpstan-doctrine": "^2.0.6",
|
||||||
"phpstan/phpstan-phpunit": "^2.0.6",
|
"phpstan/phpstan-phpunit": "^2.0.7",
|
||||||
"phpstan/phpstan-symfony": "^2.0.6",
|
"phpstan/phpstan-symfony": "^2.0.8",
|
||||||
"phpunit/phpunit": "^12.1.6",
|
"phpunit/phpunit": "^12.3.15",
|
||||||
"rector/rector": "^2.0.16",
|
"rector/rector": "^2.1.7",
|
||||||
"roave/security-advisories": "dev-latest",
|
"roave/security-advisories": "dev-latest",
|
||||||
"symfony/browser-kit": "7.2.*",
|
"symfony/browser-kit": "7.3.*",
|
||||||
"symfony/css-selector": "7.2.*",
|
"symfony/css-selector": "7.3.*",
|
||||||
"symfony/maker-bundle": "^1.63.0",
|
"symfony/maker-bundle": "^1.64.0",
|
||||||
"symfony/phpunit-bridge": "7.2.*",
|
"symfony/phpunit-bridge": "7.3.*",
|
||||||
"symfony/stopwatch": "7.2.*",
|
"symfony/stopwatch": "7.3.*",
|
||||||
"symfony/web-profiler-bundle": "7.2.*",
|
"symfony/web-profiler-bundle": "7.3.*",
|
||||||
"thecodingmachine/phpstan-safe-rule": "^1.4.1",
|
"thecodingmachine/phpstan-safe-rule": "^1.4.1",
|
||||||
"vincentlanglet/twig-cs-fixer": "^3.7.1"
|
"vincentlanglet/twig-cs-fixer": "^3.10.0"
|
||||||
},
|
},
|
||||||
"config": {
|
"config": {
|
||||||
"allow-plugins": {
|
"allow-plugins": {
|
||||||
@@ -75,12 +77,12 @@
|
|||||||
},
|
},
|
||||||
"autoload": {
|
"autoload": {
|
||||||
"psr-4": {
|
"psr-4": {
|
||||||
"App\\": "src/"
|
"Tvdt\\": "src/"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"autoload-dev": {
|
"autoload-dev": {
|
||||||
"psr-4": {
|
"psr-4": {
|
||||||
"App\\Tests\\": "tests/"
|
"Tvdt\\Tests\\": "tests/"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"replace": {
|
"replace": {
|
||||||
@@ -94,7 +96,7 @@
|
|||||||
"symfony/polyfill-php81": "*",
|
"symfony/polyfill-php81": "*",
|
||||||
"symfony/polyfill-php82": "*",
|
"symfony/polyfill-php82": "*",
|
||||||
"symfony/polyfill-php83": "*",
|
"symfony/polyfill-php83": "*",
|
||||||
"symfony/polyfill-uuid": "*"
|
"symfony/polyfill-php84": "*"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"auto-scripts": {
|
"auto-scripts": {
|
||||||
@@ -115,7 +117,7 @@
|
|||||||
"extra": {
|
"extra": {
|
||||||
"symfony": {
|
"symfony": {
|
||||||
"allow-contrib": false,
|
"allow-contrib": false,
|
||||||
"require": "7.2.*",
|
"require": "7.3.*",
|
||||||
"docker": true
|
"docker": true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
2226
composer.lock
generated
2226
composer.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -12,6 +12,8 @@ use Symfony\Bundle\MakerBundle\MakerBundle;
|
|||||||
use Symfony\Bundle\SecurityBundle\SecurityBundle;
|
use Symfony\Bundle\SecurityBundle\SecurityBundle;
|
||||||
use Symfony\Bundle\TwigBundle\TwigBundle;
|
use Symfony\Bundle\TwigBundle\TwigBundle;
|
||||||
use Symfony\Bundle\WebProfilerBundle\WebProfilerBundle;
|
use Symfony\Bundle\WebProfilerBundle\WebProfilerBundle;
|
||||||
|
use Symfony\UX\StimulusBundle\StimulusBundle;
|
||||||
|
use Symfony\UX\Turbo\TurboBundle;
|
||||||
use Symfony\UX\TwigComponent\TwigComponentBundle;
|
use Symfony\UX\TwigComponent\TwigComponentBundle;
|
||||||
use SymfonyCasts\Bundle\VerifyEmail\SymfonyCastsVerifyEmailBundle;
|
use SymfonyCasts\Bundle\VerifyEmail\SymfonyCastsVerifyEmailBundle;
|
||||||
use Symfonycasts\SassBundle\SymfonycastsSassBundle;
|
use Symfonycasts\SassBundle\SymfonycastsSassBundle;
|
||||||
@@ -32,4 +34,6 @@ return [
|
|||||||
SymfonyCastsVerifyEmailBundle::class => ['all' => true],
|
SymfonyCastsVerifyEmailBundle::class => ['all' => true],
|
||||||
SentryBundle::class => ['prod' => true],
|
SentryBundle::class => ['prod' => true],
|
||||||
SymfonycastsSassBundle::class => ['all' => true],
|
SymfonycastsSassBundle::class => ['all' => true],
|
||||||
|
StimulusBundle::class => ['all' => true],
|
||||||
|
TurboBundle::class => ['all' => true],
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -18,12 +18,14 @@ doctrine:
|
|||||||
Doctrine\DBAL\Platforms\PostgreSQLPlatform: identity
|
Doctrine\DBAL\Platforms\PostgreSQLPlatform: identity
|
||||||
auto_mapping: true
|
auto_mapping: true
|
||||||
mappings:
|
mappings:
|
||||||
App:
|
Tvdt:
|
||||||
type: attribute
|
type: attribute
|
||||||
is_bundle: false
|
is_bundle: false
|
||||||
dir: '%kernel.project_dir%/src/Entity'
|
dir: '%kernel.project_dir%/src/Entity'
|
||||||
prefix: 'App\Entity'
|
prefix: 'Tvdt\Entity'
|
||||||
alias: App
|
alias: Tvdt
|
||||||
|
controller_resolver:
|
||||||
|
auto_mapping: false
|
||||||
|
|
||||||
when@test:
|
when@test:
|
||||||
doctrine:
|
doctrine:
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
doctrine_migrations:
|
doctrine_migrations:
|
||||||
migrations_paths:
|
migrations_paths:
|
||||||
# namespace is arbitrary but should be different from App\Migrations
|
# namespace is arbitrary but should be different from Tvdt\Migrations
|
||||||
# as migrations classes should NOT be autoloaded
|
# as migrations classes should NOT be autoloaded
|
||||||
'DoctrineMigrations': '%kernel.project_dir%/migrations'
|
'DoctrineMigrations': '%kernel.project_dir%/migrations'
|
||||||
enable_profiler: false
|
enable_profiler: false
|
||||||
|
|||||||
3
config/packages/property_info.yaml
Normal file
3
config/packages/property_info.yaml
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
framework:
|
||||||
|
property_info:
|
||||||
|
with_constructor_extractor: true
|
||||||
@@ -5,9 +5,9 @@ security:
|
|||||||
# https://symfony.com/doc/current/security.html#loading-the-user-the-user-provider
|
# https://symfony.com/doc/current/security.html#loading-the-user-the-user-provider
|
||||||
providers:
|
providers:
|
||||||
# used to reload user from session & other features (e.g. switch_user)
|
# used to reload user from session & other features (e.g. switch_user)
|
||||||
app_user_provider:
|
tvdt_user_provider:
|
||||||
entity:
|
entity:
|
||||||
class: App\Entity\User
|
class: Tvdt\Entity\User
|
||||||
property: email
|
property: email
|
||||||
# used to reload user from session & other features (e.g. switch_user)
|
# used to reload user from session & other features (e.g. switch_user)
|
||||||
firewalls:
|
firewalls:
|
||||||
@@ -16,16 +16,16 @@ security:
|
|||||||
security: false
|
security: false
|
||||||
main:
|
main:
|
||||||
lazy: true
|
lazy: true
|
||||||
provider: app_user_provider
|
provider: tvdt_user_provider
|
||||||
form_login:
|
form_login:
|
||||||
login_path: app_login_login
|
login_path: tvdt_login_login
|
||||||
check_path: app_login_login
|
check_path: tvdt_login_login
|
||||||
enable_csrf: true
|
enable_csrf: true
|
||||||
default_target_path: app_backoffice_index
|
default_target_path: tvdt_backoffice_index
|
||||||
logout:
|
logout:
|
||||||
path: app_login_logout
|
path: tvdt_login_logout
|
||||||
# where to redirect after logout
|
# where to redirect after logout
|
||||||
# target: app_any_route
|
# target: tvdt_any_route
|
||||||
|
|
||||||
# activate different ways to authenticate
|
# activate different ways to authenticate
|
||||||
# https://symfony.com/doc/current/security.html#the-firewall
|
# https://symfony.com/doc/current/security.html#the-firewall
|
||||||
@@ -37,7 +37,7 @@ security:
|
|||||||
# Note: Only the *first* access control that matches will be used
|
# Note: Only the *first* access control that matches will be used
|
||||||
access_control:
|
access_control:
|
||||||
- { path: ^/admin, roles: ROLE_ADMIN }
|
- { path: ^/admin, roles: ROLE_ADMIN }
|
||||||
# - { path: ^/profile, roles: ROLE_USER }
|
- { path: ^/backoffice, roles: ROLE_USER }
|
||||||
|
|
||||||
when@test:
|
when@test:
|
||||||
security:
|
security:
|
||||||
|
|||||||
@@ -2,4 +2,4 @@ twig_component:
|
|||||||
anonymous_template_directory: 'components/'
|
anonymous_template_directory: 'components/'
|
||||||
defaults:
|
defaults:
|
||||||
# Namespace & directory for components
|
# Namespace & directory for components
|
||||||
App\Twig\Components\: 'components/'
|
Tvdt\Twig\Components\: 'components/'
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ framework:
|
|||||||
# Enables validator auto-mapping support.
|
# Enables validator auto-mapping support.
|
||||||
# For instance, basic validation constraints will be inferred from Doctrine's metadata.
|
# For instance, basic validation constraints will be inferred from Doctrine's metadata.
|
||||||
#auto_mapping:
|
#auto_mapping:
|
||||||
# App\Entity\: []
|
# Tvdt\Entity\: []
|
||||||
|
|
||||||
when@test:
|
when@test:
|
||||||
framework:
|
framework:
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
controllers:
|
controllers:
|
||||||
resource:
|
resource:
|
||||||
path: ../src/Controller/
|
path: ../src/Controller/
|
||||||
namespace: App\Controller
|
namespace: Tvdt\Controller
|
||||||
type: attribute
|
type: attribute
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
controllers:
|
controllers:
|
||||||
resource:
|
resource:
|
||||||
path: ../../src/Controller/
|
path: ../../src/Controller/
|
||||||
namespace: App\Controller
|
namespace: Tvdt\Controller
|
||||||
type: attribute
|
type: attribute
|
||||||
|
|
||||||
kernel:
|
kernel:
|
||||||
resource: App\Kernel
|
resource: Tvdt\Kernel
|
||||||
type: attribute
|
type: attribute
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
when@dev:
|
when@dev:
|
||||||
_errors:
|
_errors:
|
||||||
resource: '@FrameworkBundle/Resources/config/routing/errors.xml'
|
resource: '@FrameworkBundle/Resources/config/routing/errors.php'
|
||||||
prefix: /_error
|
prefix: /_error
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
when@dev:
|
when@dev:
|
||||||
web_profiler_wdt:
|
web_profiler_wdt:
|
||||||
resource: '@WebProfilerBundle/Resources/config/routing/wdt.xml'
|
resource: '@WebProfilerBundle/Resources/config/routing/wdt.php'
|
||||||
prefix: /_wdt
|
prefix: /_wdt
|
||||||
|
|
||||||
web_profiler_profiler:
|
web_profiler_profiler:
|
||||||
resource: '@WebProfilerBundle/Resources/config/routing/profiler.xml'
|
resource: '@WebProfilerBundle/Resources/config/routing/profiler.php'
|
||||||
prefix: /_profiler
|
prefix: /_profiler
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ services:
|
|||||||
|
|
||||||
# makes classes in src/ available to be used as services
|
# makes classes in src/ available to be used as services
|
||||||
# this creates a service per class whose id is the fully-qualified class name
|
# this creates a service per class whose id is the fully-qualified class name
|
||||||
App\:
|
Tvdt\:
|
||||||
resource: '../src/'
|
resource: '../src/'
|
||||||
exclude:
|
exclude:
|
||||||
- '../src/DependencyInjection/'
|
- '../src/DependencyInjection/'
|
||||||
|
|||||||
@@ -32,4 +32,13 @@ return [
|
|||||||
'version' => '5.3.6',
|
'version' => '5.3.6',
|
||||||
'type' => 'css',
|
'type' => 'css',
|
||||||
],
|
],
|
||||||
|
'@hotwired/stimulus' => [
|
||||||
|
'version' => '3.2.2',
|
||||||
|
],
|
||||||
|
'@symfony/stimulus-bundle' => [
|
||||||
|
'path' => './vendor/symfony/stimulus-bundle/assets/dist/loader.js',
|
||||||
|
],
|
||||||
|
'@hotwired/turbo' => [
|
||||||
|
'version' => '7.3.0',
|
||||||
|
],
|
||||||
];
|
];
|
||||||
|
|||||||
90
migrations/Version20250606192337.php
Normal file
90
migrations/Version20250606192337.php
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20250606192337 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return 'Ze Big migration';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE TABLE quiz_candidate (id UUID NOT NULL, corrections DOUBLE PRECISION NOT NULL, created TIMESTAMP(0) WITH TIME ZONE NOT NULL, quiz_id UUID NOT NULL, candidate_id UUID NOT NULL, PRIMARY KEY(id))
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE INDEX IDX_CED2FFA2853CD175 ON quiz_candidate (quiz_id)
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE INDEX IDX_CED2FFA291BD8781 ON quiz_candidate (candidate_id)
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE UNIQUE INDEX UNIQ_CED2FFA291BD8781853CD175 ON quiz_candidate (candidate_id, quiz_id)
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE quiz_candidate ADD CONSTRAINT FK_CED2FFA2853CD175 FOREIGN KEY (quiz_id) REFERENCES quiz (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE quiz_candidate ADD CONSTRAINT FK_CED2FFA291BD8781 FOREIGN KEY (candidate_id) REFERENCES candidate (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE correction DROP CONSTRAINT fk_a29da1b891bd8781
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE correction DROP CONSTRAINT fk_a29da1b8853cd175
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
DROP TABLE correction
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE elimination ALTER created TYPE TIMESTAMP(0) WITH TIME ZONE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ALTER created TYPE TIMESTAMP(0) WITH TIME ZONE
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE TABLE correction (id UUID NOT NULL, candidate_id UUID NOT NULL, quiz_id UUID NOT NULL, amount DOUBLE PRECISION NOT NULL, PRIMARY KEY(id))
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE UNIQUE INDEX uniq_a29da1b891bd8781853cd175 ON correction (candidate_id, quiz_id)
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE INDEX idx_a29da1b8853cd175 ON correction (quiz_id)
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE INDEX idx_a29da1b891bd8781 ON correction (candidate_id)
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE correction ADD CONSTRAINT fk_a29da1b891bd8781 FOREIGN KEY (candidate_id) REFERENCES candidate (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE correction ADD CONSTRAINT fk_a29da1b8853cd175 FOREIGN KEY (quiz_id) REFERENCES quiz (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE quiz_candidate DROP CONSTRAINT FK_CED2FFA2853CD175
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE quiz_candidate DROP CONSTRAINT FK_CED2FFA291BD8781
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
DROP TABLE quiz_candidate
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ALTER created TYPE TIMESTAMP(0) WITHOUT TIME ZONE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE elimination ALTER created TYPE TIMESTAMP(0) WITHOUT TIME ZONE
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
}
|
||||||
42
migrations/Version20250606195952.php
Normal file
42
migrations/Version20250606195952.php
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Auto-generated Migration: Please modify to your needs!
|
||||||
|
*/
|
||||||
|
final class Version20250606195952 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
delete from given_answer where answer_id is null
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ALTER answer_id TYPE UUID
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ALTER answer_id SET NOT NULL
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ALTER answer_id TYPE UUID
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ALTER answer_id DROP NOT NULL
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
}
|
||||||
41
migrations/Version20250607154730.php
Normal file
41
migrations/Version20250607154730.php
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Auto-generated Migration: Please modify to your needs!
|
||||||
|
*/
|
||||||
|
final class Version20250607154730 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
// this up() migration is auto-generated, please modify it to your needs
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season DROP CONSTRAINT FK_F0E45BA96706D6B
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season ADD CONSTRAINT FK_F0E45BA96706D6B FOREIGN KEY (active_quiz_id) REFERENCES quiz (id) ON DELETE SET NULL NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
// this down() migration is auto-generated, please modify it to your needs
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season DROP CONSTRAINT fk_f0e45ba96706d6b
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season ADD CONSTRAINT fk_f0e45ba96706d6b FOREIGN KEY (active_quiz_id) REFERENCES quiz (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
}
|
||||||
53
migrations/Version20250607184525.php
Normal file
53
migrations/Version20250607184525.php
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Auto-generated Migration: Please modify to your needs!
|
||||||
|
*/
|
||||||
|
final class Version20250607184525 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
// this up() migration is auto-generated, please modify it to your needs
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE elimination DROP CONSTRAINT FK_5947284F853CD175
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE elimination ADD CONSTRAINT FK_5947284F853CD175 FOREIGN KEY (quiz_id) REFERENCES quiz (id) ON DELETE CASCADE NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer DROP CONSTRAINT FK_9AC61A30853CD175
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ADD CONSTRAINT FK_9AC61A30853CD175 FOREIGN KEY (quiz_id) REFERENCES quiz (id) ON DELETE CASCADE NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
// this down() migration is auto-generated, please modify it to your needs
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer DROP CONSTRAINT fk_9ac61a30853cd175
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE given_answer ADD CONSTRAINT fk_9ac61a30853cd175 FOREIGN KEY (quiz_id) REFERENCES quiz (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE elimination DROP CONSTRAINT fk_5947284f853cd175
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE elimination ADD CONSTRAINT fk_5947284f853cd175 FOREIGN KEY (quiz_id) REFERENCES quiz (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
}
|
||||||
48
migrations/Version20250610210417.php
Normal file
48
migrations/Version20250610210417.php
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace DoctrineMigrations;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Schema\Schema;
|
||||||
|
use Doctrine\Migrations\AbstractMigration;
|
||||||
|
|
||||||
|
final class Version20250610210417 extends AbstractMigration
|
||||||
|
{
|
||||||
|
public function getDescription(): string
|
||||||
|
{
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function up(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE TABLE season_settings (id UUID NOT NULL, show_numbers BOOLEAN DEFAULT false NOT NULL, confirm_answers BOOLEAN DEFAULT false NOT NULL, PRIMARY KEY(id))
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season ADD settings_id UUID DEFAULT NULL
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season ADD CONSTRAINT FK_F0E45BA959949888 FOREIGN KEY (settings_id) REFERENCES season_settings (id) NOT DEFERRABLE INITIALLY IMMEDIATE
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
CREATE UNIQUE INDEX UNIQ_F0E45BA959949888 ON season (settings_id)
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(Schema $schema): void
|
||||||
|
{
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
DROP TABLE season_settings
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season DROP CONSTRAINT FK_F0E45BA959949888
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
DROP INDEX UNIQ_F0E45BA959949888
|
||||||
|
SQL);
|
||||||
|
$this->addSql(<<<'SQL'
|
||||||
|
ALTER TABLE season DROP settings_id
|
||||||
|
SQL);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,3 +7,4 @@ parameters:
|
|||||||
- public/
|
- public/
|
||||||
- src/
|
- src/
|
||||||
- tests/
|
- tests/
|
||||||
|
treatPhpDocTypesAsCertain: false
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
use App\Kernel;
|
use Tvdt\Kernel;
|
||||||
|
|
||||||
require_once dirname(__DIR__).'/vendor/autoload_runtime.php';
|
require_once dirname(__DIR__).'/vendor/autoload_runtime.php';
|
||||||
|
|
||||||
|
|||||||
41
src/Command/AddSettingsCommand.php
Normal file
41
src/Command/AddSettingsCommand.php
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Command;
|
||||||
|
|
||||||
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Symfony\Component\Console\Attribute\AsCommand;
|
||||||
|
use Symfony\Component\Console\Command\Command;
|
||||||
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
use Symfony\Component\Console\Style\SymfonyStyle;
|
||||||
|
use Tvdt\Entity\SeasonSettings;
|
||||||
|
use Tvdt\Repository\SeasonRepository;
|
||||||
|
|
||||||
|
#[AsCommand(
|
||||||
|
name: 'tvdt:add-settings',
|
||||||
|
description: 'Add a short description for your command',
|
||||||
|
)]
|
||||||
|
readonly class AddSettingsCommand
|
||||||
|
{
|
||||||
|
public function __construct(private SeasonRepository $seasonRepository, private EntityManagerInterface $entityManager) {}
|
||||||
|
|
||||||
|
public function __invoke(InputInterface $input, OutputInterface $output): int
|
||||||
|
{
|
||||||
|
$io = new SymfonyStyle($input, $output);
|
||||||
|
|
||||||
|
foreach ($this->seasonRepository->findAll() as $season) {
|
||||||
|
if (null !== $season->getSettings()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$io->text('Adding settings to season : '.$season->getSeasonCode());
|
||||||
|
$season->setSettings(new SeasonSettings());
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->entityManager->flush();
|
||||||
|
|
||||||
|
return Command::SUCCESS;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,45 +2,35 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Command;
|
namespace Tvdt\Command;
|
||||||
|
|
||||||
use App\Repository\SeasonRepository;
|
|
||||||
use App\Repository\UserRepository;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Symfony\Component\Console\Attribute\Argument;
|
||||||
use Symfony\Component\Console\Attribute\AsCommand;
|
use Symfony\Component\Console\Attribute\AsCommand;
|
||||||
use Symfony\Component\Console\Command\Command;
|
use Symfony\Component\Console\Command\Command;
|
||||||
use Symfony\Component\Console\Input\InputArgument;
|
|
||||||
use Symfony\Component\Console\Input\InputInterface;
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
use Symfony\Component\Console\Output\OutputInterface;
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
use Symfony\Component\Console\Style\SymfonyStyle;
|
use Symfony\Component\Console\Style\SymfonyStyle;
|
||||||
|
use Tvdt\Repository\SeasonRepository;
|
||||||
|
use Tvdt\Repository\UserRepository;
|
||||||
|
|
||||||
#[AsCommand(
|
#[AsCommand(
|
||||||
name: 'app:claim-season',
|
name: 'tvdt:claim-season',
|
||||||
description: 'Give a user owner rights on a season',
|
description: 'Give a user owner rights on a season',
|
||||||
)]
|
)]
|
||||||
class ClaimSeasonCommand extends Command
|
readonly class ClaimSeasonCommand
|
||||||
{
|
{
|
||||||
public function __construct(
|
public function __construct(private UserRepository $userRepository, private SeasonRepository $seasonRepository, private EntityManagerInterface $entityManager) {}
|
||||||
private readonly UserRepository $userRepository,
|
|
||||||
private readonly SeasonRepository $seasonRepository,
|
|
||||||
private readonly EntityManagerInterface $entityManager)
|
|
||||||
{
|
|
||||||
parent::__construct();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function configure(): void
|
public function __invoke(
|
||||||
{
|
#[Argument]
|
||||||
$this
|
string $seasonCode,
|
||||||
->addArgument('email', InputArgument::REQUIRED, 'The email of the user thats claims the season')
|
#[Argument]
|
||||||
->addArgument('season', InputArgument::REQUIRED, 'The season to claim')
|
string $email,
|
||||||
;
|
InputInterface $input,
|
||||||
}
|
OutputInterface $output,
|
||||||
|
): int {
|
||||||
protected function execute(InputInterface $input, OutputInterface $output): int
|
|
||||||
{
|
|
||||||
$io = new SymfonyStyle($input, $output);
|
$io = new SymfonyStyle($input, $output);
|
||||||
$email = $input->getArgument('email');
|
|
||||||
$seasonCode = $input->getArgument('season');
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$season = $this->seasonRepository->findOneBy(['seasonCode' => $seasonCode]);
|
$season = $this->seasonRepository->findOneBy(['seasonCode' => $seasonCode]);
|
||||||
|
|||||||
@@ -2,39 +2,31 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Command;
|
namespace Tvdt\Command;
|
||||||
|
|
||||||
use App\Repository\UserRepository;
|
use Symfony\Component\Console\Attribute\Argument;
|
||||||
use Symfony\Component\Console\Attribute\AsCommand;
|
use Symfony\Component\Console\Attribute\AsCommand;
|
||||||
use Symfony\Component\Console\Command\Command;
|
use Symfony\Component\Console\Command\Command;
|
||||||
use Symfony\Component\Console\Input\InputArgument;
|
|
||||||
use Symfony\Component\Console\Input\InputInterface;
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
use Symfony\Component\Console\Output\OutputInterface;
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
use Symfony\Component\Console\Style\SymfonyStyle;
|
use Symfony\Component\Console\Style\SymfonyStyle;
|
||||||
|
use Tvdt\Repository\UserRepository;
|
||||||
|
|
||||||
#[AsCommand(
|
#[AsCommand(
|
||||||
name: 'app:make-admin',
|
name: 'tvdt:make-admin',
|
||||||
description: 'Give a user the role admin',
|
description: 'Give a user the role admin',
|
||||||
)]
|
)]
|
||||||
class MakeAdminCommand extends Command
|
readonly class MakeAdminCommand
|
||||||
{
|
{
|
||||||
public function __construct(private readonly UserRepository $userRepository)
|
public function __construct(private UserRepository $userRepository) {}
|
||||||
{
|
|
||||||
parent::__construct();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function configure(): void
|
public function __invoke(
|
||||||
{
|
#[Argument]
|
||||||
$this
|
string $email,
|
||||||
->addArgument('email', InputArgument::REQUIRED, 'The email of the user to make admin')
|
InputInterface $input,
|
||||||
;
|
OutputInterface $output,
|
||||||
}
|
): int {
|
||||||
|
|
||||||
protected function execute(InputInterface $input, OutputInterface $output): int
|
|
||||||
{
|
|
||||||
$io = new SymfonyStyle($input, $output);
|
$io = new SymfonyStyle($input, $output);
|
||||||
$email = $input->getArgument('email');
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$this->userRepository->makeAdmin($email);
|
$this->userRepository->makeAdmin($email);
|
||||||
} catch (\InvalidArgumentException) {
|
} catch (\InvalidArgumentException) {
|
||||||
|
|||||||
@@ -2,13 +2,17 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller;
|
namespace Tvdt\Controller;
|
||||||
|
|
||||||
use App\Enum\FlashType;
|
|
||||||
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController as AbstractBaseController;
|
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController as AbstractBaseController;
|
||||||
|
use Tvdt\Enum\FlashType;
|
||||||
|
|
||||||
abstract class AbstractController extends AbstractBaseController
|
abstract class AbstractController extends AbstractBaseController
|
||||||
{
|
{
|
||||||
|
protected const string SEASON_CODE_REGEX = '[A-Za-z\d]{5}';
|
||||||
|
|
||||||
|
protected const string CANDIDATE_HASH_REGEX = '[\w\-=]+';
|
||||||
|
|
||||||
#[\Override]
|
#[\Override]
|
||||||
protected function addFlash(FlashType|string $type, mixed $message): void
|
protected function addFlash(FlashType|string $type, mixed $message): void
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\Answer;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\Answer;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<Answer> */
|
||||||
class AnswerCrudController extends AbstractCrudController
|
class AnswerCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<Candidate> */
|
||||||
class CandidateCrudController extends AbstractCrudController
|
class CandidateCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
@@ -1,16 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
declare(strict_types=1);
|
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
|
||||||
|
|
||||||
use App\Entity\Correction;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
|
||||||
|
|
||||||
class CorrectionCrudController extends AbstractCrudController
|
|
||||||
{
|
|
||||||
public static function getEntityFqcn(): string
|
|
||||||
{
|
|
||||||
return Correction::class;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -2,22 +2,22 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\Answer;
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\Correction;
|
|
||||||
use App\Entity\GivenAnswer;
|
|
||||||
use App\Entity\Question;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Entity\User;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Attribute\AdminDashboard;
|
use EasyCorp\Bundle\EasyAdminBundle\Attribute\AdminDashboard;
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Config\Dashboard;
|
use EasyCorp\Bundle\EasyAdminBundle\Config\Dashboard;
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Config\MenuItem;
|
use EasyCorp\Bundle\EasyAdminBundle\Config\MenuItem;
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractDashboardController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractDashboardController;
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Router\AdminUrlGenerator;
|
use EasyCorp\Bundle\EasyAdminBundle\Router\AdminUrlGenerator;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Tvdt\Entity\Answer;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\GivenAnswer;
|
||||||
|
use Tvdt\Entity\Question;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\QuizCandidate;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Entity\User;
|
||||||
|
|
||||||
#[AdminDashboard(routePath: '/admin', routeName: 'admin')]
|
#[AdminDashboard(routePath: '/admin', routeName: 'admin')]
|
||||||
class DashboardController extends AbstractDashboardController
|
class DashboardController extends AbstractDashboardController
|
||||||
@@ -58,7 +58,7 @@ class DashboardController extends AbstractDashboardController
|
|||||||
yield MenuItem::linkToCrud('Quiz', 'fas fa-list', Quiz::class);
|
yield MenuItem::linkToCrud('Quiz', 'fas fa-list', Quiz::class);
|
||||||
yield MenuItem::linkToCrud('Question', 'fas fa-list', Question::class);
|
yield MenuItem::linkToCrud('Question', 'fas fa-list', Question::class);
|
||||||
yield MenuItem::linkToCrud('Candidate', 'fas fa-list', Candidate::class);
|
yield MenuItem::linkToCrud('Candidate', 'fas fa-list', Candidate::class);
|
||||||
yield MenuItem::linkToCrud('Correction', 'fas fa-list', Correction::class);
|
yield MenuItem::linkToCrud('Correction', 'fas fa-list', QuizCandidate::class);
|
||||||
yield MenuItem::linkToCrud('User', 'fas fa-list', User::class);
|
yield MenuItem::linkToCrud('User', 'fas fa-list', User::class);
|
||||||
yield MenuItem::linkToCrud('Given Answer', 'fas fa-list', GivenAnswer::class);
|
yield MenuItem::linkToCrud('Given Answer', 'fas fa-list', GivenAnswer::class);
|
||||||
yield MenuItem::linkToCrud('Answer', 'fas fa-list', Answer::class);
|
yield MenuItem::linkToCrud('Answer', 'fas fa-list', Answer::class);
|
||||||
|
|||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\GivenAnswer;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\GivenAnswer;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<GivenAnswer> */
|
||||||
class GivenAnswerCrudController extends AbstractCrudController
|
class GivenAnswerCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\Question;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\Question;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<Question> */
|
||||||
class QuestionCrudController extends AbstractCrudController
|
class QuestionCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
17
src/Controller/Admin/QuizCorrectionCrudController.php
Normal file
17
src/Controller/Admin/QuizCorrectionCrudController.php
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\QuizCandidate;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<QuizCandidate> */
|
||||||
|
class QuizCorrectionCrudController extends AbstractCrudController
|
||||||
|
{
|
||||||
|
public static function getEntityFqcn(): string
|
||||||
|
{
|
||||||
|
return QuizCandidate::class;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<Quiz> */
|
||||||
class QuizCrudController extends AbstractCrudController
|
class QuizCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\Season;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<Season> */
|
||||||
class SeasonCrudController extends AbstractCrudController
|
class SeasonCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
@@ -2,11 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Admin;
|
namespace Tvdt\Controller\Admin;
|
||||||
|
|
||||||
use App\Entity\User;
|
|
||||||
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
use EasyCorp\Bundle\EasyAdminBundle\Controller\AbstractCrudController;
|
||||||
|
use Tvdt\Entity\User;
|
||||||
|
|
||||||
|
/** @extends AbstractCrudController<User> */
|
||||||
class UserCrudController extends AbstractCrudController
|
class UserCrudController extends AbstractCrudController
|
||||||
{
|
{
|
||||||
public static function getEntityFqcn(): string
|
public static function getEntityFqcn(): string
|
||||||
|
|||||||
@@ -2,14 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Backoffice;
|
namespace Tvdt\Controller\Backoffice;
|
||||||
|
|
||||||
use App\Controller\AbstractController;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Entity\User;
|
|
||||||
use App\Form\CreateSeasonFormType;
|
|
||||||
use App\Repository\SeasonRepository;
|
|
||||||
use App\Service\QuizSpreadsheetService;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Symfony\Bundle\SecurityBundle\Security;
|
use Symfony\Bundle\SecurityBundle\Security;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
@@ -18,6 +12,12 @@ use Symfony\Component\HttpFoundation\StreamedResponse;
|
|||||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
||||||
|
use Tvdt\Controller\AbstractController;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Entity\User;
|
||||||
|
use Tvdt\Form\CreateSeasonFormType;
|
||||||
|
use Tvdt\Repository\SeasonRepository;
|
||||||
|
use Tvdt\Service\QuizSpreadsheetService;
|
||||||
|
|
||||||
#[AsController]
|
#[AsController]
|
||||||
#[IsGranted('ROLE_USER')]
|
#[IsGranted('ROLE_USER')]
|
||||||
@@ -28,7 +28,7 @@ final class BackofficeController extends AbstractController
|
|||||||
private readonly Security $security,
|
private readonly Security $security,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
#[Route('/backoffice/', name: 'app_backoffice_index')]
|
#[Route('/backoffice/', name: 'tvdt_backoffice_index')]
|
||||||
public function index(): Response
|
public function index(): Response
|
||||||
{
|
{
|
||||||
$user = $this->getUser();
|
$user = $this->getUser();
|
||||||
@@ -43,7 +43,7 @@ final class BackofficeController extends AbstractController
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/backoffice/add', name: 'app_backoffice_season_add', priority: 10)]
|
#[Route('/backoffice/season/add', name: 'tvdt_backoffice_season_add', priority: 10)]
|
||||||
public function addSeason(Request $request, EntityManagerInterface $em): Response
|
public function addSeason(Request $request, EntityManagerInterface $em): Response
|
||||||
{
|
{
|
||||||
$season = new Season();
|
$season = new Season();
|
||||||
@@ -61,13 +61,13 @@ final class BackofficeController extends AbstractController
|
|||||||
$em->persist($season);
|
$em->persist($season);
|
||||||
$em->flush();
|
$em->flush();
|
||||||
|
|
||||||
return $this->redirectToRoute('app_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('backoffice/season_add.html.twig', ['form' => $form]);
|
return $this->render('backoffice/season_add.html.twig', ['form' => $form]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/backoffice/template', name: 'app_backoffice_template', priority: 10)]
|
#[Route('/backoffice/template', name: 'tvdt_backoffice_template', priority: 10)]
|
||||||
public function getTemplate(QuizSpreadsheetService $excel): Response
|
public function getTemplate(QuizSpreadsheetService $excel): Response
|
||||||
{
|
{
|
||||||
$response = new StreamedResponse($excel->generateTemplate());
|
$response = new StreamedResponse($excel->generateTemplate());
|
||||||
|
|||||||
@@ -2,38 +2,48 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Backoffice;
|
namespace Tvdt\Controller\Backoffice;
|
||||||
|
|
||||||
use App\Entity\Elimination;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Factory\EliminationFactory;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
|
use Symfony\Component\Routing\Requirement\Requirement;
|
||||||
|
use Tvdt\Controller\AbstractController;
|
||||||
|
use Tvdt\Entity\Elimination;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Factory\EliminationFactory;
|
||||||
|
|
||||||
final class PrepareEliminationController extends AbstractController
|
final class PrepareEliminationController extends AbstractController
|
||||||
{
|
{
|
||||||
#[Route('/backoffice/elimination/{seasonCode}/{quiz}/prepare', name: 'app_prepare_elimination')]
|
#[Route(
|
||||||
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/elimination/prepare',
|
||||||
|
name: 'tvdt_prepare_elimination',
|
||||||
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
||||||
|
)]
|
||||||
public function index(Season $season, Quiz $quiz, EliminationFactory $eliminationFactory): Response
|
public function index(Season $season, Quiz $quiz, EliminationFactory $eliminationFactory): Response
|
||||||
{
|
{
|
||||||
$elimination = $eliminationFactory->createEliminationFromQuiz($quiz);
|
$elimination = $eliminationFactory->createEliminationFromQuiz($quiz);
|
||||||
|
|
||||||
return $this->redirectToRoute('app_prepare_elimination_view', ['elimination' => $elimination->getId()]);
|
return $this->redirectToRoute('tvdt_prepare_elimination_view', ['elimination' => $elimination->getId()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/backoffice/elimination/{elimination}', name: 'app_prepare_elimination_view')]
|
#[Route(
|
||||||
|
'/backoffice/elimination/{elimination}',
|
||||||
|
name: 'tvdt_prepare_elimination_view',
|
||||||
|
requirements: ['elimination' => Requirement::UUID],
|
||||||
|
)]
|
||||||
public function viewElimination(Elimination $elimination, Request $request, EntityManagerInterface $em): Response
|
public function viewElimination(Elimination $elimination, Request $request, EntityManagerInterface $em): Response
|
||||||
{
|
{
|
||||||
if ('POST' === $request->getMethod()) {
|
if ('POST' === $request->getMethod()) {
|
||||||
$elimination->updateFromInputBag($request->request);
|
$elimination->updateFromInputBag($request->request);
|
||||||
$em->flush();
|
$em->flush();
|
||||||
|
|
||||||
if (true === $request->request->getBoolean('start')) {
|
if ($request->request->getBoolean('start')) {
|
||||||
return $this->redirectToRoute('app_elimination', ['elimination' => $elimination->getId()]);
|
return $this->redirectToRoute('tvdt_elimination', ['elimination' => $elimination->getId()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->addFlash('success', 'Elimination updated');
|
$this->addFlash('success', 'Elimination updated');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,18 +2,27 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Backoffice;
|
namespace Tvdt\Controller\Backoffice;
|
||||||
|
|
||||||
use App\Controller\AbstractController;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Repository\CandidateRepository;
|
|
||||||
use App\Security\Voter\SeasonVoter;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Symfony\Component\HttpFoundation\RedirectResponse;
|
||||||
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\MethodNotAllowedHttpException;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
|
use Symfony\Component\Routing\Requirement\Requirement;
|
||||||
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Controller\AbstractController;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Exception\ErrorClearingQuizException;
|
||||||
|
use Tvdt\Repository\CandidateRepository;
|
||||||
|
use Tvdt\Repository\QuizCandidateRepository;
|
||||||
|
use Tvdt\Repository\QuizRepository;
|
||||||
|
use Tvdt\Security\Voter\SeasonVoter;
|
||||||
|
|
||||||
#[AsController]
|
#[AsController]
|
||||||
#[IsGranted('ROLE_USER')]
|
#[IsGranted('ROLE_USER')]
|
||||||
@@ -21,9 +30,14 @@ class QuizController extends AbstractController
|
|||||||
{
|
{
|
||||||
public function __construct(
|
public function __construct(
|
||||||
private readonly CandidateRepository $candidateRepository,
|
private readonly CandidateRepository $candidateRepository,
|
||||||
|
private readonly TranslatorInterface $translator,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
#[Route('/backoffice/season/{seasonCode}/quiz/{quiz}', name: 'app_backoffice_quiz')]
|
#[Route(
|
||||||
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}',
|
||||||
|
name: 'tvdt_backoffice_quiz',
|
||||||
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
|
||||||
|
)]
|
||||||
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
||||||
public function index(Season $season, Quiz $quiz): Response
|
public function index(Season $season, Quiz $quiz): Response
|
||||||
{
|
{
|
||||||
@@ -34,17 +48,73 @@ class QuizController extends AbstractController
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/backoffice/season/{seasonCode}/quiz/{quiz}/enable', name: 'app_backoffice_enable')]
|
#[Route(
|
||||||
|
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/enable',
|
||||||
|
name: 'tvdt_backoffice_enable',
|
||||||
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID.'|null'],
|
||||||
|
)]
|
||||||
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
||||||
public function enableQuiz(Season $season, ?Quiz $quiz, EntityManagerInterface $em): Response
|
public function enableQuiz(Season $season, ?Quiz $quiz, EntityManagerInterface $em): RedirectResponse
|
||||||
{
|
{
|
||||||
$season->setActiveQuiz($quiz);
|
$season->setActiveQuiz($quiz);
|
||||||
$em->flush();
|
$em->flush();
|
||||||
|
|
||||||
if ($quiz instanceof Quiz) {
|
if ($quiz instanceof Quiz) {
|
||||||
return $this->redirectToRoute('app_backoffice_quiz', ['seasonCode' => $season->getSeasonCode(), 'quiz' => $quiz->getId()]);
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $season->getSeasonCode(), 'quiz' => $quiz->getId()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->redirectToRoute('app_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route(
|
||||||
|
'/backoffice/quiz/{quiz}/clear',
|
||||||
|
name: 'tvdt_backoffice_quiz_clear',
|
||||||
|
requirements: ['quiz' => Requirement::UUID],
|
||||||
|
)]
|
||||||
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
||||||
|
public function clearQuiz(Quiz $quiz, QuizRepository $quizRepository): RedirectResponse
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$quizRepository->clearQuiz($quiz);
|
||||||
|
$this->addFlash('success', $this->translator->trans('Quiz cleared'));
|
||||||
|
} catch (ErrorClearingQuizException) {
|
||||||
|
$this->addFlash('error', $this->translator->trans('Error clearing quiz'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->getSeason()->getSeasonCode(), 'quiz' => $quiz->getId()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route(
|
||||||
|
'/backoffice/quiz/{quiz}/delete',
|
||||||
|
name: 'tvdt_backoffice_quiz_delete',
|
||||||
|
requirements: ['quiz' => Requirement::UUID],
|
||||||
|
)]
|
||||||
|
#[IsGranted(SeasonVoter::DELETE, subject: 'quiz')]
|
||||||
|
public function deleteQuiz(Quiz $quiz, QuizRepository $quizRepository): RedirectResponse
|
||||||
|
{
|
||||||
|
$quizRepository->deleteQuiz($quiz);
|
||||||
|
|
||||||
|
$this->addFlash('success', $this->translator->trans('Quiz deleted'));
|
||||||
|
|
||||||
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $quiz->getSeason()->getSeasonCode()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[Route(
|
||||||
|
'/backoffice/quiz/{quiz}/candidate/{candidate}/modify_correction',
|
||||||
|
name: 'tvdt_backoffice_modify_correction',
|
||||||
|
requirements: ['quiz' => Requirement::UUID, 'candidate' => Requirement::UUID],
|
||||||
|
)]
|
||||||
|
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
|
||||||
|
public function modifyCorrection(Quiz $quiz, Candidate $candidate, QuizCandidateRepository $quizCandidateRepository, Request $request): RedirectResponse
|
||||||
|
{
|
||||||
|
if (!$request->isMethod('POST')) {
|
||||||
|
throw new MethodNotAllowedHttpException(['POST']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$corrections = (float) $request->request->get('corrections');
|
||||||
|
|
||||||
|
$quizCandidateRepository->setCorrectionsForCandidate($quiz, $candidate, $corrections);
|
||||||
|
|
||||||
|
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->getSeason()->getSeasonCode(), 'quiz' => $quiz->getId()]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,17 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller\Backoffice;
|
namespace Tvdt\Controller\Backoffice;
|
||||||
|
|
||||||
use App\Controller\AbstractController;
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Enum\FlashType;
|
|
||||||
use App\Form\AddCandidatesFormType;
|
|
||||||
use App\Form\UploadQuizFormType;
|
|
||||||
use App\Security\Voter\SeasonVoter;
|
|
||||||
use App\Service\QuizSpreadsheetService;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Symfony\Component\HttpFoundation\File\UploadedFile;
|
use Symfony\Component\HttpFoundation\File\UploadedFile;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
@@ -21,24 +12,54 @@ use Symfony\Component\HttpKernel\Attribute\AsController;
|
|||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Controller\AbstractController;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Enum\FlashType;
|
||||||
|
use Tvdt\Form\AddCandidatesFormType;
|
||||||
|
use Tvdt\Form\SettingsForm;
|
||||||
|
use Tvdt\Form\UploadQuizFormType;
|
||||||
|
use Tvdt\Security\Voter\SeasonVoter;
|
||||||
|
use Tvdt\Service\QuizSpreadsheetService;
|
||||||
|
|
||||||
#[AsController]
|
#[AsController]
|
||||||
#[IsGranted('ROLE_USER')]
|
#[IsGranted('ROLE_USER')]
|
||||||
class SeasonController extends AbstractController
|
class SeasonController extends AbstractController
|
||||||
{
|
{
|
||||||
public function __construct(private readonly TranslatorInterface $translator, private EntityManagerInterface $em,
|
public function __construct(
|
||||||
|
private readonly TranslatorInterface $translator,
|
||||||
|
private readonly EntityManagerInterface $em,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
#[Route('/backoffice/season/{seasonCode}', name: 'app_backoffice_season')]
|
#[Route(
|
||||||
|
'/backoffice/season/{seasonCode:season}',
|
||||||
|
name: 'tvdt_backoffice_season',
|
||||||
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX],
|
||||||
|
)]
|
||||||
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
||||||
public function index(Season $season): Response
|
public function index(Season $season, Request $request): Response
|
||||||
{
|
{
|
||||||
|
$form = $this->createForm(SettingsForm::class, $season->getSettings());
|
||||||
|
|
||||||
|
$form->handleRequest($request);
|
||||||
|
|
||||||
|
if ($form->isSubmitted() && $form->isValid()) {
|
||||||
|
$this->em->flush();
|
||||||
|
}
|
||||||
|
|
||||||
return $this->render('backoffice/season.html.twig', [
|
return $this->render('backoffice/season.html.twig', [
|
||||||
'season' => $season,
|
'season' => $season,
|
||||||
|
'form' => $form,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/backoffice/season/{seasonCode}/add_candidate', name: 'app_backoffice_add_candidates', priority: 10)]
|
#[Route(
|
||||||
|
'/backoffice/season/{seasonCode:season}/add-candidate',
|
||||||
|
name: 'tvdt_backoffice_add_candidates',
|
||||||
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX],
|
||||||
|
priority: 10,
|
||||||
|
)]
|
||||||
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
||||||
public function addCandidates(Season $season, Request $request): Response
|
public function addCandidates(Season $season, Request $request): Response
|
||||||
{
|
{
|
||||||
@@ -47,19 +68,24 @@ class SeasonController extends AbstractController
|
|||||||
|
|
||||||
if ($form->isSubmitted() && $form->isValid()) {
|
if ($form->isSubmitted() && $form->isValid()) {
|
||||||
$candidates = $form->get('candidates')->getData();
|
$candidates = $form->get('candidates')->getData();
|
||||||
foreach (explode("\r\n", (string) $candidates) as $candidate) {
|
foreach (explode("\n", (string) $candidates) as $candidate) {
|
||||||
$season->addCandidate(new Candidate($candidate));
|
$season->addCandidate(new Candidate(mb_rtrim($candidate)));
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->em->flush();
|
$this->em->flush();
|
||||||
|
|
||||||
return $this->redirectToRoute('app_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('backoffice/season_add_candidates.html.twig', ['form' => $form]);
|
return $this->render('backoffice/season_add_candidates.html.twig', ['form' => $form]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/backoffice/season/{seasonCode}/add', name: 'app_backoffice_quiz_add', priority: 10)]
|
#[Route(
|
||||||
|
'/backoffice/season/{seasonCode:season}/add-quiz',
|
||||||
|
name: 'tvdt_backoffice_quiz_add',
|
||||||
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX],
|
||||||
|
priority: 10,
|
||||||
|
)]
|
||||||
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
|
||||||
public function addQuiz(Request $request, Season $season, QuizSpreadsheetService $quizSpreadsheet): Response
|
public function addQuiz(Request $request, Season $season, QuizSpreadsheetService $quizSpreadsheet): Response
|
||||||
{
|
{
|
||||||
@@ -80,7 +106,7 @@ class SeasonController extends AbstractController
|
|||||||
|
|
||||||
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz Added!'));
|
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz Added!'));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('/backoffice/quiz_add.html.twig', ['form' => $form, 'season' => $season]);
|
return $this->render('/backoffice/quiz_add.html.twig', ['form' => $form, 'season' => $season]);
|
||||||
|
|||||||
@@ -2,22 +2,23 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller;
|
namespace Tvdt\Controller;
|
||||||
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\Elimination;
|
|
||||||
use App\Enum\FlashType;
|
|
||||||
use App\Form\EliminationEnterNameType;
|
|
||||||
use App\Helpers\Base64;
|
|
||||||
use App\Repository\CandidateRepository;
|
|
||||||
use App\Security\Voter\SeasonVoter;
|
|
||||||
use Symfony\Bridge\Doctrine\Attribute\MapEntity;
|
use Symfony\Bridge\Doctrine\Attribute\MapEntity;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
|
use Symfony\Component\Routing\Requirement\Requirement;
|
||||||
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
use Symfony\Component\Security\Http\Attribute\IsGranted;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Elimination;
|
||||||
|
use Tvdt\Enum\FlashType;
|
||||||
|
use Tvdt\Form\EliminationEnterNameType;
|
||||||
|
use Tvdt\Helpers\Base64;
|
||||||
|
use Tvdt\Repository\CandidateRepository;
|
||||||
|
use Tvdt\Security\Voter\SeasonVoter;
|
||||||
|
|
||||||
use function Symfony\Component\Translation\t;
|
use function Symfony\Component\Translation\t;
|
||||||
|
|
||||||
@@ -27,7 +28,7 @@ final class EliminationController extends AbstractController
|
|||||||
{
|
{
|
||||||
public function __construct(private readonly TranslatorInterface $translator) {}
|
public function __construct(private readonly TranslatorInterface $translator) {}
|
||||||
|
|
||||||
#[Route('/elimination/{elimination}', name: 'app_elimination')]
|
#[Route('/elimination/{elimination}', name: 'tvdt_elimination', requirements: ['elimination' => Requirement::UUID])]
|
||||||
#[IsGranted(SeasonVoter::ELIMINATION, 'elimination')]
|
#[IsGranted(SeasonVoter::ELIMINATION, 'elimination')]
|
||||||
public function index(#[MapEntity] Elimination $elimination, Request $request): Response
|
public function index(#[MapEntity] Elimination $elimination, Request $request): Response
|
||||||
{
|
{
|
||||||
@@ -38,7 +39,7 @@ final class EliminationController extends AbstractController
|
|||||||
if ($form->isSubmitted() && $form->isValid()) {
|
if ($form->isSubmitted() && $form->isValid()) {
|
||||||
$name = $form->get('name')->getData();
|
$name = $form->get('name')->getData();
|
||||||
|
|
||||||
return $this->redirectToRoute('app_elimination_candidate', ['elimination' => $elimination->getId(), 'candidateHash' => Base64::base64UrlEncode($name)]);
|
return $this->redirectToRoute('tvdt_elimination_candidate', ['elimination' => $elimination->getId(), 'candidateHash' => Base64::base64UrlEncode($name)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('quiz/elimination/index.html.twig', [
|
return $this->render('quiz/elimination/index.html.twig', [
|
||||||
@@ -47,17 +48,17 @@ final class EliminationController extends AbstractController
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/elimination/{elimination}/{candidateHash}', name: 'app_elimination_candidate')]
|
#[Route('/elimination/{elimination}/{candidateHash}', name: 'tvdt_elimination_candidate', requirements: ['elimination' => Requirement::UUID, 'candidateHash' => self::CANDIDATE_HASH_REGEX])]
|
||||||
#[IsGranted(SeasonVoter::ELIMINATION, 'elimination')]
|
#[IsGranted(SeasonVoter::ELIMINATION, 'elimination')]
|
||||||
public function candidateScreen(Elimination $elimination, string $candidateHash, CandidateRepository $candidateRepository): Response
|
public function candidateScreen(Elimination $elimination, string $candidateHash, CandidateRepository $candidateRepository): Response
|
||||||
{
|
{
|
||||||
$candidate = $candidateRepository->getCandidateByHash($elimination->getQuiz()->getSeason(), $candidateHash);
|
$candidate = $candidateRepository->getCandidateByHash($elimination->getQuiz()->getSeason(), $candidateHash);
|
||||||
if (!$candidate instanceof Candidate) {
|
if (!$candidate instanceof Candidate) {
|
||||||
$this->addFlash(FlashType::Warning,
|
$this->addFlash(FlashType::Warning,
|
||||||
t('Cound not find candidate with name %name%', ['%name%' => Base64::base64UrlDecode($candidateHash)])->trans($this->translator)
|
t('Cound not find candidate with name %name%', ['%name%' => Base64::base64UrlDecode($candidateHash)])->trans($this->translator),
|
||||||
);
|
);
|
||||||
|
|
||||||
return $this->redirectToRoute('app_elimination', ['elimination' => $elimination->getId()]);
|
return $this->redirectToRoute('tvdt_elimination', ['elimination' => $elimination->getId()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$screenColour = $elimination->getScreenColour($candidate->getName());
|
$screenColour = $elimination->getScreenColour($candidate->getName());
|
||||||
@@ -65,7 +66,7 @@ final class EliminationController extends AbstractController
|
|||||||
if (null === $screenColour) {
|
if (null === $screenColour) {
|
||||||
$this->addFlash(FlashType::Warning, $this->translator->trans('Cound not find candidate with name %name% in elimination.', ['%name%' => $candidate->getName()]));
|
$this->addFlash(FlashType::Warning, $this->translator->trans('Cound not find candidate with name %name% in elimination.', ['%name%' => $candidate->getName()]));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_elimination', ['elimination' => $elimination->getId()]);
|
return $this->redirectToRoute('tvdt_elimination', ['elimination' => $elimination->getId()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('quiz/elimination/candidate.html.twig', [
|
return $this->render('quiz/elimination/candidate.html.twig', [
|
||||||
|
|||||||
@@ -2,20 +2,20 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller;
|
namespace Tvdt\Controller;
|
||||||
|
|
||||||
use App\Enum\FlashType;
|
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
use Symfony\Component\Security\Core\Exception\AuthenticationException;
|
use Symfony\Component\Security\Core\Exception\AuthenticationException;
|
||||||
use Symfony\Component\Security\Http\Authentication\AuthenticationUtils;
|
use Symfony\Component\Security\Http\Authentication\AuthenticationUtils;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Enum\FlashType;
|
||||||
|
|
||||||
#[AsController]
|
#[AsController]
|
||||||
final class LoginController extends AbstractController
|
final class LoginController extends AbstractController
|
||||||
{
|
{
|
||||||
#[Route(path: '/login', name: 'app_login_login')]
|
#[Route(path: '/login', name: 'tvdt_login_login')]
|
||||||
public function login(AuthenticationUtils $authenticationUtils, TranslatorInterface $translator): Response
|
public function login(AuthenticationUtils $authenticationUtils, TranslatorInterface $translator): Response
|
||||||
{
|
{
|
||||||
// get the login error if there is one
|
// get the login error if there is one
|
||||||
@@ -34,7 +34,7 @@ final class LoginController extends AbstractController
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route(path: '/logout', name: 'app_login_logout')]
|
#[Route(path: '/logout', name: 'tvdt_login_logout')]
|
||||||
public function logout(): never
|
public function logout(): never
|
||||||
{
|
{
|
||||||
throw new \LogicException('This method can be blank - it will be intercepted by the logout key on your firewall.');
|
throw new \LogicException('This method can be blank - it will be intercepted by the logout key on your firewall.');
|
||||||
|
|||||||
@@ -2,40 +2,37 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller;
|
namespace Tvdt\Controller;
|
||||||
|
|
||||||
use App\Entity\Answer;
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\GivenAnswer;
|
|
||||||
use App\Entity\Question;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Enum\FlashType;
|
|
||||||
use App\Form\EnterNameType;
|
|
||||||
use App\Form\SelectSeasonType;
|
|
||||||
use App\Helpers\Base64;
|
|
||||||
use App\Repository\AnswerRepository;
|
|
||||||
use App\Repository\CandidateRepository;
|
|
||||||
use App\Repository\GivenAnswerRepository;
|
|
||||||
use App\Repository\QuestionRepository;
|
|
||||||
use App\Repository\SeasonRepository;
|
|
||||||
use Symfony\Bridge\Doctrine\Attribute\MapEntity;
|
|
||||||
use Symfony\Component\HttpFoundation\Exception\BadRequestException;
|
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
use Symfony\Component\HttpKernel\Attribute\AsController;
|
use Symfony\Component\HttpKernel\Attribute\AsController;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Entity\Answer;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\GivenAnswer;
|
||||||
|
use Tvdt\Entity\Question;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Enum\FlashType;
|
||||||
|
use Tvdt\Form\EnterNameType;
|
||||||
|
use Tvdt\Form\SelectSeasonType;
|
||||||
|
use Tvdt\Helpers\Base64;
|
||||||
|
use Tvdt\Repository\AnswerRepository;
|
||||||
|
use Tvdt\Repository\CandidateRepository;
|
||||||
|
use Tvdt\Repository\GivenAnswerRepository;
|
||||||
|
use Tvdt\Repository\QuestionRepository;
|
||||||
|
use Tvdt\Repository\QuizCandidateRepository;
|
||||||
|
use Tvdt\Repository\SeasonRepository;
|
||||||
|
|
||||||
#[AsController]
|
#[AsController]
|
||||||
final class QuizController extends AbstractController
|
final class QuizController extends AbstractController
|
||||||
{
|
{
|
||||||
public const string SEASON_CODE_REGEX = '[A-Za-z\d]{5}';
|
|
||||||
|
|
||||||
private const string CANDIDATE_HASH_REGEX = '[\w\-=]+';
|
|
||||||
|
|
||||||
public function __construct(private readonly TranslatorInterface $translator) {}
|
public function __construct(private readonly TranslatorInterface $translator) {}
|
||||||
|
|
||||||
#[Route(path: '/', name: 'app_quiz_selectseason', methods: ['GET', 'POST'])]
|
#[Route(path: '/', name: 'tvdt_quiz_select_season', methods: ['GET', 'POST'])]
|
||||||
public function selectSeason(Request $request, SeasonRepository $seasonRepository): Response
|
public function selectSeason(Request $request, SeasonRepository $seasonRepository): Response
|
||||||
{
|
{
|
||||||
$form = $this->createForm(SelectSeasonType::class);
|
$form = $this->createForm(SelectSeasonType::class);
|
||||||
@@ -47,19 +44,18 @@ final class QuizController extends AbstractController
|
|||||||
if ([] === $seasonRepository->findBy(['seasonCode' => $seasonCode])) {
|
if ([] === $seasonRepository->findBy(['seasonCode' => $seasonCode])) {
|
||||||
$this->addFlash(FlashType::Warning, $this->translator->trans('Invalid season code'));
|
$this->addFlash(FlashType::Warning, $this->translator->trans('Invalid season code'));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_quiz_selectseason');
|
return $this->redirectToRoute('tvdt_quiz_select_season');
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->redirectToRoute('app_quiz_entername', ['seasonCode' => $seasonCode]);
|
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $seasonCode]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('quiz/select_season.html.twig', ['form' => $form]);
|
return $this->render('quiz/select_season.html.twig', ['form' => $form]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route(path: '/{seasonCode}', name: 'app_quiz_entername', requirements: ['seasonCode' => self::SEASON_CODE_REGEX])]
|
#[Route(path: '/{seasonCode:season}', name: 'tvdt_quiz_enter_name', requirements: ['seasonCode' => self::SEASON_CODE_REGEX])]
|
||||||
public function enterName(
|
public function enterName(
|
||||||
Request $request,
|
Request $request,
|
||||||
#[MapEntity(mapping: ['seasonCode' => 'seasonCode'])]
|
|
||||||
Season $season,
|
Season $season,
|
||||||
): Response {
|
): Response {
|
||||||
$form = $this->createForm(EnterNameType::class);
|
$form = $this->createForm(EnterNameType::class);
|
||||||
@@ -69,47 +65,54 @@ final class QuizController extends AbstractController
|
|||||||
if ($form->isSubmitted() && $form->isValid()) {
|
if ($form->isSubmitted() && $form->isValid()) {
|
||||||
$name = $form->get('name')->getData();
|
$name = $form->get('name')->getData();
|
||||||
|
|
||||||
return $this->redirectToRoute('app_quiz_quizpage', ['seasonCode' => $season->getSeasonCode(), 'nameHash' => Base64::base64UrlEncode($name)]);
|
return $this->redirectToRoute('tvdt_quiz_quiz_page', ['seasonCode' => $season->getSeasonCode(), 'nameHash' => Base64::base64UrlEncode($name)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->render('quiz/enter_name.twig', ['season' => $season, 'form' => $form]);
|
return $this->render('quiz/enter_name.twig', ['season' => $season, 'form' => $form]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route(
|
#[Route(
|
||||||
path: '/{seasonCode}/{nameHash}',
|
path: '/{seasonCode:season}/{nameHash}',
|
||||||
name: 'app_quiz_quizpage',
|
name: 'tvdt_quiz_quiz_page',
|
||||||
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'nameHash' => self::CANDIDATE_HASH_REGEX],
|
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'nameHash' => self::CANDIDATE_HASH_REGEX],
|
||||||
)]
|
)]
|
||||||
public function quizPage(
|
public function quizPage(
|
||||||
#[MapEntity(mapping: ['seasonCode' => 'seasonCode'])]
|
|
||||||
Season $season,
|
Season $season,
|
||||||
string $nameHash,
|
string $nameHash,
|
||||||
|
Request $request,
|
||||||
CandidateRepository $candidateRepository,
|
CandidateRepository $candidateRepository,
|
||||||
QuestionRepository $questionRepository,
|
QuestionRepository $questionRepository,
|
||||||
AnswerRepository $answerRepository,
|
AnswerRepository $answerRepository,
|
||||||
GivenAnswerRepository $givenAnswerRepository,
|
GivenAnswerRepository $givenAnswerRepository,
|
||||||
Request $request,
|
QuizCandidateRepository $quizCandidateRepository,
|
||||||
): Response {
|
): Response {
|
||||||
$candidate = $candidateRepository->getCandidateByHash($season, $nameHash);
|
$candidate = $candidateRepository->getCandidateByHash($season, $nameHash);
|
||||||
|
|
||||||
if (!$candidate instanceof Candidate) {
|
if (!$candidate instanceof Candidate) {
|
||||||
$this->addFlash(FlashType::Danger, $this->translator->trans('Candidate not found'));
|
$this->addFlash(FlashType::Danger, $this->translator->trans('Candidate not found'));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_quiz_entername', ['seasonCode' => $season->getSeasonCode()]);
|
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$quiz = $season->getActiveQuiz();
|
||||||
|
|
||||||
|
if (!$quiz instanceof Quiz) {
|
||||||
|
$this->addFlash(FlashType::Warning, $this->translator->trans('There is no active quiz'));
|
||||||
|
|
||||||
|
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ('POST' === $request->getMethod()) {
|
if ('POST' === $request->getMethod()) {
|
||||||
$answer = $answerRepository->findOneBy(['id' => $request->request->get('answer')]);
|
$answer = $answerRepository->findOneBy(['id' => $request->request->get('answer')]);
|
||||||
|
|
||||||
if (!$answer instanceof Answer) {
|
if (!$answer instanceof Answer) {
|
||||||
throw new BadRequestException('Invalid Answer ID');
|
throw new BadRequestHttpException('Invalid Answer ID');
|
||||||
}
|
}
|
||||||
|
|
||||||
$givenAnswer = (new GivenAnswer())
|
$givenAnswer = new GivenAnswer($candidate, $answer->getQuestion()->getQuiz(), $answer);
|
||||||
->setCandidate($candidate)
|
|
||||||
->setAnswer($answer)
|
|
||||||
->setQuiz($answer->getQuestion()->getQuiz());
|
|
||||||
$givenAnswerRepository->save($givenAnswer);
|
$givenAnswerRepository->save($givenAnswer);
|
||||||
|
|
||||||
|
return $this->redirectToRoute('tvdt_quiz_quiz_page', ['seasonCode' => $season->getSeasonCode(), 'nameHash' => $nameHash]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$question = $questionRepository->findNextQuestionForCandidate($candidate);
|
$question = $questionRepository->findNextQuestionForCandidate($candidate);
|
||||||
@@ -117,10 +120,11 @@ final class QuizController extends AbstractController
|
|||||||
if (!$question instanceof Question) {
|
if (!$question instanceof Question) {
|
||||||
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz completed'));
|
$this->addFlash(FlashType::Success, $this->translator->trans('Quiz completed'));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_quiz_entername', ['seasonCode' => $season->getSeasonCode()]);
|
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->getSeasonCode()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO One first question record time
|
$quizCandidateRepository->createIfNotExist($quiz, $candidate);
|
||||||
return $this->render('quiz/question.twig', ['candidate' => $candidate, 'question' => $question]);
|
|
||||||
|
return $this->render('quiz/question.twig', ['candidate' => $candidate, 'question' => $question, 'season' => $season]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,33 +2,36 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Controller;
|
namespace Tvdt\Controller;
|
||||||
|
|
||||||
use App\Entity\User;
|
|
||||||
use App\Form\RegistrationFormType;
|
|
||||||
use App\Repository\UserRepository;
|
|
||||||
use App\Security\EmailVerifier;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
use Symfony\Bridge\Twig\Mime\TemplatedEmail;
|
use Symfony\Bridge\Twig\Mime\TemplatedEmail;
|
||||||
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
|
||||||
use Symfony\Bundle\SecurityBundle\Security;
|
use Symfony\Bundle\SecurityBundle\Security;
|
||||||
use Symfony\Component\HttpFoundation\Request;
|
use Symfony\Component\HttpFoundation\Request;
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Symfony\Component\Mailer\Exception\TransportExceptionInterface;
|
||||||
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
|
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
|
||||||
use Symfony\Component\Routing\Attribute\Route;
|
use Symfony\Component\Routing\Attribute\Route;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
use SymfonyCasts\Bundle\VerifyEmail\Exception\VerifyEmailExceptionInterface;
|
use SymfonyCasts\Bundle\VerifyEmail\Exception\VerifyEmailExceptionInterface;
|
||||||
|
use Tvdt\Entity\User;
|
||||||
|
use Tvdt\Form\RegistrationFormType;
|
||||||
|
use Tvdt\Repository\UserRepository;
|
||||||
|
use Tvdt\Security\EmailVerifier;
|
||||||
|
|
||||||
final class RegistrationController extends AbstractController
|
final class RegistrationController extends AbstractController
|
||||||
{
|
{
|
||||||
public function __construct(private readonly EmailVerifier $emailVerifier, private readonly TranslatorInterface $translator) {}
|
public function __construct(private readonly EmailVerifier $emailVerifier, private readonly TranslatorInterface $translator) {}
|
||||||
|
|
||||||
#[Route('/register', name: 'app_register')]
|
#[Route('/register', name: 'tvdt_register')]
|
||||||
public function register(
|
public function register(
|
||||||
Request $request,
|
Request $request,
|
||||||
UserPasswordHasherInterface $userPasswordHasher,
|
UserPasswordHasherInterface $userPasswordHasher,
|
||||||
Security $security,
|
Security $security,
|
||||||
EntityManagerInterface $entityManager,
|
EntityManagerInterface $entityManager,
|
||||||
|
LoggerInterface $logger,
|
||||||
): Response {
|
): Response {
|
||||||
$user = new User();
|
$user = new User();
|
||||||
$form = $this->createForm(RegistrationFormType::class, $user);
|
$form = $this->createForm(RegistrationFormType::class, $user);
|
||||||
@@ -43,13 +46,17 @@ final class RegistrationController extends AbstractController
|
|||||||
$entityManager->persist($user);
|
$entityManager->persist($user);
|
||||||
$entityManager->flush();
|
$entityManager->flush();
|
||||||
|
|
||||||
|
try {
|
||||||
// generate a signed url and email it to the user
|
// generate a signed url and email it to the user
|
||||||
$this->emailVerifier->sendEmailConfirmation('app_verify_email', $user,
|
$this->emailVerifier->sendEmailConfirmation('tvdt_verify_email', $user,
|
||||||
(new TemplatedEmail())
|
new TemplatedEmail()
|
||||||
->to((string) $user->getEmail())
|
->to((string) $user->getEmail())
|
||||||
->subject($this->translator->trans('Please Confirm your Email'))
|
->subject($this->translator->trans('Please Confirm your Email'))
|
||||||
->htmlTemplate('backoffice/registration/confirmation_email.html.twig')
|
->htmlTemplate('backoffice/registration/confirmation_email.html.twig'),
|
||||||
);
|
);
|
||||||
|
} catch (TransportExceptionInterface $e) {
|
||||||
|
$logger->error($e->getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
$response = $security->login($user, 'form_login', 'main');
|
$response = $security->login($user, 'form_login', 'main');
|
||||||
\assert($response instanceof Response);
|
\assert($response instanceof Response);
|
||||||
@@ -62,19 +69,19 @@ final class RegistrationController extends AbstractController
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[Route('/verify/email', name: 'app_verify_email')]
|
#[Route('/verify/email', name: 'tvdt_verify_email')]
|
||||||
public function verifyUserEmail(Request $request, TranslatorInterface $translator, UserRepository $userRepository): Response
|
public function verifyUserEmail(Request $request, TranslatorInterface $translator, UserRepository $userRepository): Response
|
||||||
{
|
{
|
||||||
$id = $request->query->get('id');
|
$id = $request->query->get('id');
|
||||||
|
|
||||||
if (null === $id) {
|
if (null === $id) {
|
||||||
return $this->redirectToRoute('app_register');
|
return $this->redirectToRoute('tvdt_register');
|
||||||
}
|
}
|
||||||
|
|
||||||
$user = $userRepository->find($id);
|
$user = $userRepository->find($id);
|
||||||
|
|
||||||
if (null === $user) {
|
if (null === $user) {
|
||||||
return $this->redirectToRoute('app_register');
|
return $this->redirectToRoute('tvdt_register');
|
||||||
}
|
}
|
||||||
|
|
||||||
// validate email confirmation link, sets User::isVerified=true and persists
|
// validate email confirmation link, sets User::isVerified=true and persists
|
||||||
@@ -83,11 +90,11 @@ final class RegistrationController extends AbstractController
|
|||||||
} catch (VerifyEmailExceptionInterface $verifyEmailException) {
|
} catch (VerifyEmailExceptionInterface $verifyEmailException) {
|
||||||
$this->addFlash('verify_email_error', $translator->trans($verifyEmailException->getReason(), [], 'VerifyEmailBundle'));
|
$this->addFlash('verify_email_error', $translator->trans($verifyEmailException->getReason(), [], 'VerifyEmailBundle'));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_register');
|
return $this->redirectToRoute('tvdt_register');
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->addFlash('success', $this->translator->trans('Your email address has been verified.'));
|
$this->addFlash('success', $this->translator->trans('Your email address has been verified.'));
|
||||||
|
|
||||||
return $this->redirectToRoute('app_backoffice_index');
|
return $this->redirectToRoute('tvdt_backoffice_index');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,15 +2,15 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\DataFixtures;
|
namespace Tvdt\DataFixtures;
|
||||||
|
|
||||||
use App\Entity\Answer;
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\Question;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use Doctrine\Bundle\FixturesBundle\Fixture;
|
use Doctrine\Bundle\FixturesBundle\Fixture;
|
||||||
use Doctrine\Persistence\ObjectManager;
|
use Doctrine\Persistence\ObjectManager;
|
||||||
|
use Tvdt\Entity\Answer;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Question;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
|
||||||
class KrtekFixtures extends Fixture
|
class KrtekFixtures extends Fixture
|
||||||
{
|
{
|
||||||
@@ -44,42 +44,42 @@ class KrtekFixtures extends Fixture
|
|||||||
|
|
||||||
private function createQuiz1(Season $season): Quiz
|
private function createQuiz1(Season $season): Quiz
|
||||||
{
|
{
|
||||||
return (new Quiz())
|
return new Quiz()
|
||||||
->setName('Quiz 1')
|
->setName('Quiz 1')
|
||||||
->setSeason($season)
|
->setSeason($season)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Is de Krtek een man of een vrouw?')
|
->setQuestion('Is de Krtek een man of een vrouw?')
|
||||||
->addAnswer(new Answer('Vrouw', true))
|
->addAnswer(new Answer('Vrouw', true))
|
||||||
->addAnswer(new Answer('Man'))
|
->addAnswer(new Answer('Man'))
|
||||||
->setOrdering(1)
|
->setOrdering(1),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Hoeveel broers heeft de Krtek?')
|
->setQuestion('Hoeveel broers heeft de Krtek?')
|
||||||
->addAnswer(new Answer('Geen', true))
|
->addAnswer(new Answer('Geen', true))
|
||||||
->addAnswer(new Answer('1'))
|
->addAnswer(new Answer('1'))
|
||||||
->addAnswer(new Answer('2'))
|
->addAnswer(new Answer('2'))
|
||||||
->setOrdering(2)
|
->setOrdering(2),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wat is de lievelingsfeestdag van de Krtek?')
|
->setQuestion('Wat is de lievelingsfeestdag van de Krtek?')
|
||||||
->addAnswer(new Answer('Geen'))
|
->addAnswer(new Answer('Geen'))
|
||||||
->addAnswer(new Answer('Diens eigen verjaardag'))
|
->addAnswer(new Answer('Diens eigen verjaardag'))
|
||||||
->addAnswer(new Answer('Koningsdag'))
|
->addAnswer(new Answer('Koningsdag'))
|
||||||
->addAnswer(new Answer('Kerst', true))
|
->addAnswer(new Answer('Kerst', true))
|
||||||
->addAnswer(new Answer('Oud en Nieuw'))
|
->addAnswer(new Answer('Oud en Nieuw'))
|
||||||
->setOrdering(3)
|
->setOrdering(3),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Hoe kwam de Krtek naar Kersteren vandaag?')
|
->setQuestion('Hoe kwam de Krtek naar Kersteren vandaag?')
|
||||||
->addAnswer(new Answer('Met het OV', true))
|
->addAnswer(new Answer('Met het OV', true))
|
||||||
->addAnswer(new Answer('Met de auto'))
|
->addAnswer(new Answer('Met de auto'))
|
||||||
->setOrdering(4)
|
->setOrdering(4),
|
||||||
)
|
)
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Met wie keek de Krtek video bij binnenkomst?')
|
->setQuestion('Met wie keek de Krtek video bij binnenkomst?')
|
||||||
->addAnswer(new Answer('Claudia'))
|
->addAnswer(new Answer('Claudia'))
|
||||||
->addAnswer(new Answer('Eelco'))
|
->addAnswer(new Answer('Eelco'))
|
||||||
@@ -94,10 +94,10 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Remy'))
|
->addAnswer(new Answer('Remy'))
|
||||||
->addAnswer(new Answer('Robbert'))
|
->addAnswer(new Answer('Robbert'))
|
||||||
->addAnswer(new Answer('Tom', true))
|
->addAnswer(new Answer('Tom', true))
|
||||||
->setOrdering(5)
|
->setOrdering(5),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Welk advies zou de Krtek zichzelf als kind geven?')
|
->setQuestion('Welk advies zou de Krtek zichzelf als kind geven?')
|
||||||
->addAnswer(new Answer('Geef je vader een knuffel.'))
|
->addAnswer(new Answer('Geef je vader een knuffel.'))
|
||||||
->addAnswer(new Answer('Trek je wat minder aan van anderen.'))
|
->addAnswer(new Answer('Trek je wat minder aan van anderen.'))
|
||||||
@@ -107,10 +107,10 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Probeer ook eens buiten de lijntjes te kleuren', true))
|
->addAnswer(new Answer('Probeer ook eens buiten de lijntjes te kleuren', true))
|
||||||
->addAnswer(new Answer('Ga als je groot bent op groepsreis! '))
|
->addAnswer(new Answer('Ga als je groot bent op groepsreis! '))
|
||||||
->addAnswer(new Answer('Trek minder aan van de mening van anderen, het is oké om anders te zijn.'))
|
->addAnswer(new Answer('Trek minder aan van de mening van anderen, het is oké om anders te zijn.'))
|
||||||
->setOrdering(6)
|
->setOrdering(6),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wat voor soort schoenen droeg de Krtek bij het diner?')
|
->setQuestion('Wat voor soort schoenen droeg de Krtek bij het diner?')
|
||||||
->addAnswer(new Answer('Sneakers'))
|
->addAnswer(new Answer('Sneakers'))
|
||||||
->addAnswer(new Answer('Wandel-/bergschoenen', true))
|
->addAnswer(new Answer('Wandel-/bergschoenen', true))
|
||||||
@@ -118,25 +118,25 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Pantoffels'))
|
->addAnswer(new Answer('Pantoffels'))
|
||||||
->addAnswer(new Answer('Hakken'))
|
->addAnswer(new Answer('Hakken'))
|
||||||
->addAnswer(new Answer('Geen schoenen, alleen sokken'))
|
->addAnswer(new Answer('Geen schoenen, alleen sokken'))
|
||||||
->setOrdering(7)
|
->setOrdering(7),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Met welk vervoersmiddel reist de Krtek het liefste?')
|
->setQuestion('Met welk vervoersmiddel reist de Krtek het liefste?')
|
||||||
->addAnswer(new Answer('Fiets', true))
|
->addAnswer(new Answer('Fiets', true))
|
||||||
->addAnswer(new Answer('Auto'))
|
->addAnswer(new Answer('Auto'))
|
||||||
->addAnswer(new Answer('Trein'))
|
->addAnswer(new Answer('Trein'))
|
||||||
->setOrdering(8)
|
->setOrdering(8),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Heeft de Krtek een eigen auto?')
|
->setQuestion('Heeft de Krtek een eigen auto?')
|
||||||
->addAnswer(new Answer('Ja'))
|
->addAnswer(new Answer('Ja'))
|
||||||
->addAnswer(new Answer('Nee', true))
|
->addAnswer(new Answer('Nee', true))
|
||||||
->setOrdering(9)
|
->setOrdering(9),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Van wie is de quote die de Krtek gepakt heeft')
|
->setQuestion('Van wie is de quote die de Krtek gepakt heeft')
|
||||||
->addAnswer(new Answer('Karen'))
|
->addAnswer(new Answer('Karen'))
|
||||||
->addAnswer(new Answer('Gilles de Coster'))
|
->addAnswer(new Answer('Gilles de Coster'))
|
||||||
@@ -153,41 +153,41 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Pieter'))
|
->addAnswer(new Answer('Pieter'))
|
||||||
->addAnswer(new Answer('Renée Fokker'))
|
->addAnswer(new Answer('Renée Fokker'))
|
||||||
->addAnswer(new Answer('Sam, Davy', true))
|
->addAnswer(new Answer('Sam, Davy', true))
|
||||||
->setOrdering(10)
|
->setOrdering(10),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Zou de Krtek molboekjes, jokers, vrijstellingen of topito’s uit iemands rugzak stelen om te kunnen winnen?')
|
->setQuestion('Zou de Krtek molboekjes, jokers, vrijstellingen of topito’s uit iemands rugzak stelen om te kunnen winnen?')
|
||||||
->addAnswer(new Answer('Ja'))
|
->addAnswer(new Answer('Ja'))
|
||||||
->addAnswer(new Answer('Nee', true))
|
->addAnswer(new Answer('Nee', true))
|
||||||
->setOrdering(11)
|
->setOrdering(11),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('In wat voor bed slaapt de Krtek dit weekend?')
|
->setQuestion('In wat voor bed slaapt de Krtek dit weekend?')
|
||||||
->addAnswer(new Answer('Éénpersoons, losstaand bed'))
|
->addAnswer(new Answer('Éénpersoons, losstaand bed'))
|
||||||
->addAnswer(new Answer('Éénpersoonsbed, tegen een ander bed aan', true))
|
->addAnswer(new Answer('Éénpersoonsbed, tegen een ander bed aan', true))
|
||||||
->addAnswer(new Answer('Tweepersoons bed'))
|
->addAnswer(new Answer('Tweepersoons bed'))
|
||||||
->setOrdering(12)
|
->setOrdering(12),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Hoeveel jaar heeft de Krtek gedaan over de middelbare school?')
|
->setQuestion('Hoeveel jaar heeft de Krtek gedaan over de middelbare school?')
|
||||||
->addAnswer(new Answer('5'))
|
->addAnswer(new Answer('5'))
|
||||||
->addAnswer(new Answer('6', true))
|
->addAnswer(new Answer('6', true))
|
||||||
->addAnswer(new Answer('7'))
|
->addAnswer(new Answer('7'))
|
||||||
->addAnswer(new Answer('8'))
|
->addAnswer(new Answer('8'))
|
||||||
->setOrdering(13)
|
->setOrdering(13),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Waar zat de Krtek aan tafel bij het diner?')
|
->setQuestion('Waar zat de Krtek aan tafel bij het diner?')
|
||||||
->addAnswer(new Answer('Met de rug naar de accommodatie'))
|
->addAnswer(new Answer('Met de rug naar de accommodatie'))
|
||||||
->addAnswer(new Answer('Met de rug naar de buitenmuur', true))
|
->addAnswer(new Answer('Met de rug naar de buitenmuur', true))
|
||||||
->setOrdering(14)
|
->setOrdering(14),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wie is de Krtek?')
|
->setQuestion('Wie is de Krtek?')
|
||||||
->addAnswer(new Answer('Claudia', true))
|
->addAnswer(new Answer('Claudia', true))
|
||||||
->addAnswer(new Answer('Eelco'))
|
->addAnswer(new Answer('Eelco'))
|
||||||
@@ -202,25 +202,25 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Remy'))
|
->addAnswer(new Answer('Remy'))
|
||||||
->addAnswer(new Answer('Robbert'))
|
->addAnswer(new Answer('Robbert'))
|
||||||
->addAnswer(new Answer('Tom'))
|
->addAnswer(new Answer('Tom'))
|
||||||
->setOrdering(15)
|
->setOrdering(15),
|
||||||
)
|
)
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function createQuiz2(Season $season): Quiz
|
private function createQuiz2(Season $season): Quiz
|
||||||
{
|
{
|
||||||
return (new Quiz())
|
return new Quiz()
|
||||||
->setName('Quiz 2')
|
->setName('Quiz 2')
|
||||||
->setSeason($season)
|
->setSeason($season)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Is de Krtek een man of een vrouw?')
|
->setQuestion('Is de Krtek een man of een vrouw?')
|
||||||
->addAnswer(new Answer('Man'))
|
->addAnswer(new Answer('Man'))
|
||||||
->addAnswer(new Answer('Vrouw', true))
|
->addAnswer(new Answer('Vrouw', true))
|
||||||
->setOrdering(1)
|
->setOrdering(1),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Heeft de Krtek dieetwensen of allergieën?')
|
->setQuestion('Heeft de Krtek dieetwensen of allergieën?')
|
||||||
->addAnswer(new Answer('nee'))
|
->addAnswer(new Answer('nee'))
|
||||||
->addAnswer(new Answer('De Krtek is vegetariër', true))
|
->addAnswer(new Answer('De Krtek is vegetariër', true))
|
||||||
@@ -229,10 +229,10 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('De Krtek heeft een intolerantie'))
|
->addAnswer(new Answer('De Krtek heeft een intolerantie'))
|
||||||
->addAnswer(new Answer('De Krtek eet geen rundvlees'))
|
->addAnswer(new Answer('De Krtek eet geen rundvlees'))
|
||||||
->addAnswer(new Answer('De Krtek eet geen waterdieren'))
|
->addAnswer(new Answer('De Krtek eet geen waterdieren'))
|
||||||
->setOrdering(2)
|
->setOrdering(2),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Hoe heet het huisdier/de huisdieren van de Krtek?')
|
->setQuestion('Hoe heet het huisdier/de huisdieren van de Krtek?')
|
||||||
->addAnswer(new Answer('Amy, Karel en Floyd'))
|
->addAnswer(new Answer('Amy, Karel en Floyd'))
|
||||||
->addAnswer(new Answer('Flip en Majoor'))
|
->addAnswer(new Answer('Flip en Majoor'))
|
||||||
@@ -241,10 +241,10 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Tom'))
|
->addAnswer(new Answer('Tom'))
|
||||||
->addAnswer(new Answer('De huisdieren van de Krtek hebben geen naam'))
|
->addAnswer(new Answer('De huisdieren van de Krtek hebben geen naam'))
|
||||||
->addAnswer(new Answer('De Krtek heeft geen huisdieren', true))
|
->addAnswer(new Answer('De Krtek heeft geen huisdieren', true))
|
||||||
->setOrdering(3)
|
->setOrdering(3),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wat dronk de Krtek deze ochtend bij het ontbijt?')
|
->setQuestion('Wat dronk de Krtek deze ochtend bij het ontbijt?')
|
||||||
->addAnswer(new Answer('Koffie'))
|
->addAnswer(new Answer('Koffie'))
|
||||||
->addAnswer(new Answer('Thee'))
|
->addAnswer(new Answer('Thee'))
|
||||||
@@ -252,10 +252,10 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Melk'))
|
->addAnswer(new Answer('Melk'))
|
||||||
->addAnswer(new Answer('Sap'))
|
->addAnswer(new Answer('Sap'))
|
||||||
->addAnswer(new Answer('Niks'))
|
->addAnswer(new Answer('Niks'))
|
||||||
->setOrdering(4)
|
->setOrdering(4),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Waar ging de eerste vakantie die de Krtek zich nog herinnert heen?')
|
->setQuestion('Waar ging de eerste vakantie die de Krtek zich nog herinnert heen?')
|
||||||
->addAnswer(new Answer('Denemarken'))
|
->addAnswer(new Answer('Denemarken'))
|
||||||
->addAnswer(new Answer('Drenthe'))
|
->addAnswer(new Answer('Drenthe'))
|
||||||
@@ -264,54 +264,54 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Oostenrijk'))
|
->addAnswer(new Answer('Oostenrijk'))
|
||||||
->addAnswer(new Answer('Turkije'))
|
->addAnswer(new Answer('Turkije'))
|
||||||
->addAnswer(new Answer('Zweden', true))
|
->addAnswer(new Answer('Zweden', true))
|
||||||
->setOrdering(5)
|
->setOrdering(5),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Met welk groepje ging de Krtek als eerste het Douanespel in?')
|
->setQuestion('Met welk groepje ging de Krtek als eerste het Douanespel in?')
|
||||||
->addAnswer(new Answer('Het eerste groepje', true))
|
->addAnswer(new Answer('Het eerste groepje', true))
|
||||||
->addAnswer(new Answer('Het tweede groepje'))
|
->addAnswer(new Answer('Het tweede groepje'))
|
||||||
->addAnswer(new Answer('Het derde groepje'))
|
->addAnswer(new Answer('Het derde groepje'))
|
||||||
->addAnswer(new Answer('Het vierde groepje'))
|
->addAnswer(new Answer('Het vierde groepje'))
|
||||||
->addAnswer(new Answer('Het vijfde groepje'))
|
->addAnswer(new Answer('Het vijfde groepje'))
|
||||||
->setOrdering(6)
|
->setOrdering(6),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Gelooft de Krtek ergens in?')
|
->setQuestion('Gelooft de Krtek ergens in?')
|
||||||
->addAnswer(new Answer('Nee'))
|
->addAnswer(new Answer('Nee'))
|
||||||
->addAnswer(new Answer('Het universum', true))
|
->addAnswer(new Answer('Het universum', true))
|
||||||
->addAnswer(new Answer('Toeval'))
|
->addAnswer(new Answer('Toeval'))
|
||||||
->addAnswer(new Answer('De Krtek is hindoeïstisch'))
|
->addAnswer(new Answer('De Krtek is hindoeïstisch'))
|
||||||
->setOrdering(7)
|
->setOrdering(7),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('At de Krtek op vrijdagavond heksenkaas tijdens het diner?')
|
->setQuestion('At de Krtek op vrijdagavond heksenkaas tijdens het diner?')
|
||||||
->addAnswer(new Answer('Ja', true))
|
->addAnswer(new Answer('Ja', true))
|
||||||
->addAnswer(new Answer('Nee'))
|
->addAnswer(new Answer('Nee'))
|
||||||
->setOrdering(8)
|
->setOrdering(8),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Hoe laat ging de Krtek gisteravond naar bed?')
|
->setQuestion('Hoe laat ging de Krtek gisteravond naar bed?')
|
||||||
->addAnswer(new Answer('Tussen 0:00 en 0:59 uur'))
|
->addAnswer(new Answer('Tussen 0:00 en 0:59 uur'))
|
||||||
->addAnswer(new Answer('Tussen 1:00 en 1:59 uur', true))
|
->addAnswer(new Answer('Tussen 1:00 en 1:59 uur', true))
|
||||||
->addAnswer(new Answer('Tussen 2:00 en 2:59 uur'))
|
->addAnswer(new Answer('Tussen 2:00 en 2:59 uur'))
|
||||||
->addAnswer(new Answer('Na 3:00'))
|
->addAnswer(new Answer('Na 3:00'))
|
||||||
->setOrdering(9)
|
->setOrdering(9),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Hoeveel batterijen heeft de Krtek naar het bord gebracht bij het douanespel?')
|
->setQuestion('Hoeveel batterijen heeft de Krtek naar het bord gebracht bij het douanespel?')
|
||||||
->addAnswer(new Answer('1'))
|
->addAnswer(new Answer('1'))
|
||||||
->addAnswer(new Answer('2'))
|
->addAnswer(new Answer('2'))
|
||||||
->addAnswer(new Answer('3'))
|
->addAnswer(new Answer('3'))
|
||||||
->addAnswer(new Answer('geen', true))
|
->addAnswer(new Answer('geen', true))
|
||||||
->setOrdering(10)
|
->setOrdering(10),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wat keek de Krtek als kind graag op TV?')
|
->setQuestion('Wat keek de Krtek als kind graag op TV?')
|
||||||
->addAnswer(new Answer('Digimon', true))
|
->addAnswer(new Answer('Digimon', true))
|
||||||
->addAnswer(new Answer('Floris'))
|
->addAnswer(new Answer('Floris'))
|
||||||
@@ -319,18 +319,18 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Sesamstraat'))
|
->addAnswer(new Answer('Sesamstraat'))
|
||||||
->addAnswer(new Answer('Spongebob Squarepants'))
|
->addAnswer(new Answer('Spongebob Squarepants'))
|
||||||
->addAnswer(new Answer('Teletubbies'))
|
->addAnswer(new Answer('Teletubbies'))
|
||||||
->setOrdering(11)
|
->setOrdering(11),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Waarin zat op de heenreis de bagage van de Krtek (voornamelijk)?')
|
->setQuestion('Waarin zat op de heenreis de bagage van de Krtek (voornamelijk)?')
|
||||||
->addAnswer(new Answer('In koffer(s)', true))
|
->addAnswer(new Answer('In koffer(s)', true))
|
||||||
->addAnswer(new Answer('In losse tas(sen)'))
|
->addAnswer(new Answer('In losse tas(sen)'))
|
||||||
->addAnswer(new Answer('In een rugzak'))
|
->addAnswer(new Answer('In een rugzak'))
|
||||||
->setOrdering(12)
|
->setOrdering(12),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Van welk geluid gaan de haren van de Krtek overeind staan?')
|
->setQuestion('Van welk geluid gaan de haren van de Krtek overeind staan?')
|
||||||
->addAnswer(new Answer('Een vork die door een metalen pan krast '))
|
->addAnswer(new Answer('Een vork die door een metalen pan krast '))
|
||||||
->addAnswer(new Answer('Smakkende mensen'))
|
->addAnswer(new Answer('Smakkende mensen'))
|
||||||
@@ -340,17 +340,17 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Servies dat tegen elkaar klettert'))
|
->addAnswer(new Answer('Servies dat tegen elkaar klettert'))
|
||||||
->addAnswer(new Answer('Het geroekoe van een duif', true))
|
->addAnswer(new Answer('Het geroekoe van een duif', true))
|
||||||
->addAnswer(new Answer('Piepschuim'))
|
->addAnswer(new Answer('Piepschuim'))
|
||||||
->setOrdering(13)
|
->setOrdering(13),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wilde de Krtek penningmeester worden?')
|
->setQuestion('Wilde de Krtek penningmeester worden?')
|
||||||
->addAnswer(new Answer('Ja'))
|
->addAnswer(new Answer('Ja'))
|
||||||
->addAnswer(new Answer('Nee', true))
|
->addAnswer(new Answer('Nee', true))
|
||||||
->setOrdering(14)
|
->setOrdering(14),
|
||||||
)
|
)
|
||||||
|
|
||||||
->addQuestion((new Question())
|
->addQuestion(new Question()
|
||||||
->setQuestion('Wie is de Krtek?')
|
->setQuestion('Wie is de Krtek?')
|
||||||
->addAnswer(new Answer('Claudia', true))
|
->addAnswer(new Answer('Claudia', true))
|
||||||
->addAnswer(new Answer('Eelco'))
|
->addAnswer(new Answer('Eelco'))
|
||||||
@@ -365,7 +365,7 @@ class KrtekFixtures extends Fixture
|
|||||||
->addAnswer(new Answer('Remy'))
|
->addAnswer(new Answer('Remy'))
|
||||||
->addAnswer(new Answer('Robbert'))
|
->addAnswer(new Answer('Robbert'))
|
||||||
->addAnswer(new Answer('Tom'))
|
->addAnswer(new Answer('Tom'))
|
||||||
->setOrdering(15)
|
->setOrdering(15),
|
||||||
)
|
)
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\AnswerRepository;
|
|
||||||
use Doctrine\Common\Collections\ArrayCollection;
|
use Doctrine\Common\Collections\ArrayCollection;
|
||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\DBAL\Types\Types;
|
use Doctrine\DBAL\Types\Types;
|
||||||
@@ -12,6 +11,7 @@ use Doctrine\ORM\Mapping as ORM;
|
|||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\AnswerRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: AnswerRepository::class)]
|
#[ORM\Entity(repositoryClass: AnswerRepository::class)]
|
||||||
class Answer
|
class Answer
|
||||||
@@ -116,16 +116,6 @@ class Answer
|
|||||||
return $this->givenAnswers;
|
return $this->givenAnswers;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function addGivenAnswer(GivenAnswer $givenAnswer): static
|
|
||||||
{
|
|
||||||
if (!$this->givenAnswers->contains($givenAnswer)) {
|
|
||||||
$this->givenAnswers->add($givenAnswer);
|
|
||||||
$givenAnswer->setAnswer($this);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getOrdering(): int
|
public function getOrdering(): int
|
||||||
{
|
{
|
||||||
return $this->ordering;
|
return $this->ordering;
|
||||||
|
|||||||
@@ -2,16 +2,16 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Helpers\Base64;
|
|
||||||
use App\Repository\CandidateRepository;
|
|
||||||
use Doctrine\Common\Collections\ArrayCollection;
|
use Doctrine\Common\Collections\ArrayCollection;
|
||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Helpers\Base64;
|
||||||
|
use Tvdt\Repository\CandidateRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: CandidateRepository::class)]
|
#[ORM\Entity(repositoryClass: CandidateRepository::class)]
|
||||||
#[ORM\UniqueConstraint(fields: ['name', 'season'])]
|
#[ORM\UniqueConstraint(fields: ['name', 'season'])]
|
||||||
@@ -21,7 +21,7 @@ class Candidate
|
|||||||
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
||||||
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
private ?Uuid $id = null;
|
private Uuid $id;
|
||||||
|
|
||||||
#[ORM\ManyToOne(inversedBy: 'candidates')]
|
#[ORM\ManyToOne(inversedBy: 'candidates')]
|
||||||
#[ORM\JoinColumn(nullable: false)]
|
#[ORM\JoinColumn(nullable: false)]
|
||||||
@@ -35,9 +35,9 @@ class Candidate
|
|||||||
#[ORM\OneToMany(targetEntity: GivenAnswer::class, mappedBy: 'candidate', orphanRemoval: true)]
|
#[ORM\OneToMany(targetEntity: GivenAnswer::class, mappedBy: 'candidate', orphanRemoval: true)]
|
||||||
private Collection $givenAnswers;
|
private Collection $givenAnswers;
|
||||||
|
|
||||||
/** @var Collection<int, Correction> */
|
/** @var Collection<int, QuizCandidate> */
|
||||||
#[ORM\OneToMany(targetEntity: Correction::class, mappedBy: 'candidate', orphanRemoval: true)]
|
#[ORM\OneToMany(targetEntity: QuizCandidate::class, mappedBy: 'candidate', orphanRemoval: true)]
|
||||||
private Collection $corrections;
|
private Collection $quizData;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
#[ORM\Column(length: 16)]
|
#[ORM\Column(length: 16)]
|
||||||
@@ -45,10 +45,10 @@ class Candidate
|
|||||||
) {
|
) {
|
||||||
$this->answersOnCandidate = new ArrayCollection();
|
$this->answersOnCandidate = new ArrayCollection();
|
||||||
$this->givenAnswers = new ArrayCollection();
|
$this->givenAnswers = new ArrayCollection();
|
||||||
$this->corrections = new ArrayCollection();
|
$this->quizData = new ArrayCollection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
public function getId(): Uuid
|
||||||
{
|
{
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
@@ -108,30 +108,10 @@ class Candidate
|
|||||||
return $this->givenAnswers;
|
return $this->givenAnswers;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function addGivenAnswer(GivenAnswer $givenAnswer): static
|
/** @return Collection<int, QuizCandidate> */
|
||||||
|
public function getQuizData(): Collection
|
||||||
{
|
{
|
||||||
if (!$this->givenAnswers->contains($givenAnswer)) {
|
return $this->quizData;
|
||||||
$this->givenAnswers->add($givenAnswer);
|
|
||||||
$givenAnswer->setCandidate($this);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @return Collection<int, Correction> */
|
|
||||||
public function getCorrections(): Collection
|
|
||||||
{
|
|
||||||
return $this->corrections;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function addCorrection(Correction $correction): static
|
|
||||||
{
|
|
||||||
if (!$this->corrections->contains($correction)) {
|
|
||||||
$this->corrections->add($correction);
|
|
||||||
$correction->setCandidate($this);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getNameHash(): string
|
public function getNameHash(): string
|
||||||
|
|||||||
@@ -1,74 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
declare(strict_types=1);
|
|
||||||
|
|
||||||
namespace App\Entity;
|
|
||||||
|
|
||||||
use App\Repository\CorrectionRepository;
|
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
|
||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
|
||||||
use Symfony\Component\Uid\Uuid;
|
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: CorrectionRepository::class)]
|
|
||||||
#[ORM\UniqueConstraint(columns: ['candidate_id', 'quiz_id'])]
|
|
||||||
class Correction
|
|
||||||
{
|
|
||||||
#[ORM\Id]
|
|
||||||
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
|
||||||
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
|
||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
|
||||||
private ?Uuid $id = null;
|
|
||||||
|
|
||||||
#[ORM\ManyToOne(inversedBy: 'corrections')]
|
|
||||||
#[ORM\JoinColumn(nullable: false)]
|
|
||||||
private Candidate $candidate;
|
|
||||||
|
|
||||||
#[ORM\ManyToOne(inversedBy: 'corrections')]
|
|
||||||
#[ORM\JoinColumn(nullable: false)]
|
|
||||||
private Quiz $quiz;
|
|
||||||
|
|
||||||
#[ORM\Column]
|
|
||||||
private float $amount = 0;
|
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
|
||||||
{
|
|
||||||
return $this->id;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getCandidate(): Candidate
|
|
||||||
{
|
|
||||||
return $this->candidate;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setCandidate(Candidate $candidate): static
|
|
||||||
{
|
|
||||||
$this->candidate = $candidate;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getQuiz(): Quiz
|
|
||||||
{
|
|
||||||
return $this->quiz;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setQuiz(Quiz $quiz): static
|
|
||||||
{
|
|
||||||
$this->quiz = $quiz;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getAmount(): ?float
|
|
||||||
{
|
|
||||||
return $this->amount;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setAmount(float $amount): static
|
|
||||||
{
|
|
||||||
$this->amount = $amount;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\EliminationRepository;
|
|
||||||
use Doctrine\DBAL\Types\Types;
|
use Doctrine\DBAL\Types\Types;
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
use Safe\DateTimeImmutable;
|
use Safe\DateTimeImmutable;
|
||||||
@@ -12,12 +11,14 @@ use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
|||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\HttpFoundation\InputBag;
|
use Symfony\Component\HttpFoundation\InputBag;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\EliminationRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: EliminationRepository::class)]
|
#[ORM\Entity(repositoryClass: EliminationRepository::class)]
|
||||||
#[ORM\HasLifecycleCallbacks]
|
#[ORM\HasLifecycleCallbacks]
|
||||||
class Elimination
|
class Elimination
|
||||||
{
|
{
|
||||||
public const string SCREEN_GREEN = 'green';
|
public const string SCREEN_GREEN = 'green';
|
||||||
|
|
||||||
public const string SCREEN_RED = 'red';
|
public const string SCREEN_RED = 'red';
|
||||||
|
|
||||||
#[ORM\Id]
|
#[ORM\Id]
|
||||||
@@ -30,12 +31,12 @@ class Elimination
|
|||||||
#[ORM\Column(type: Types::JSON)]
|
#[ORM\Column(type: Types::JSON)]
|
||||||
private array $data = [];
|
private array $data = [];
|
||||||
|
|
||||||
#[ORM\Column(type: Types::DATETIME_IMMUTABLE, nullable: false)]
|
#[ORM\Column(type: Types::DATETIMETZ_IMMUTABLE, nullable: false)]
|
||||||
private \DateTimeImmutable $created;
|
private \DateTimeImmutable $created;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
#[ORM\ManyToOne(inversedBy: 'eliminations')]
|
#[ORM\ManyToOne(inversedBy: 'eliminations')]
|
||||||
#[ORM\JoinColumn(nullable: false)]
|
#[ORM\JoinColumn(nullable: false, onDelete: 'CASCADE')]
|
||||||
private Quiz $quiz,
|
private Quiz $quiz,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@@ -66,7 +67,7 @@ class Elimination
|
|||||||
/** @param InputBag<bool|float|int|string> $inputBag */
|
/** @param InputBag<bool|float|int|string> $inputBag */
|
||||||
public function updateFromInputBag(InputBag $inputBag): self
|
public function updateFromInputBag(InputBag $inputBag): self
|
||||||
{
|
{
|
||||||
foreach ($this->data as $name => $screenColour) {
|
foreach (array_keys($this->data) as $name) {
|
||||||
$newColour = $inputBag->get('colour-'.mb_strtolower($name));
|
$newColour = $inputBag->get('colour-'.mb_strtolower($name));
|
||||||
if (\is_string($newColour)) {
|
if (\is_string($newColour)) {
|
||||||
$this->data[$name] = $inputBag->get('colour-'.mb_strtolower($name));
|
$this->data[$name] = $inputBag->get('colour-'.mb_strtolower($name));
|
||||||
|
|||||||
@@ -2,15 +2,15 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\GivenAnswerRepository;
|
|
||||||
use Doctrine\DBAL\Types\Types;
|
use Doctrine\DBAL\Types\Types;
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
use Safe\DateTimeImmutable;
|
use Safe\DateTimeImmutable;
|
||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\GivenAnswerRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: GivenAnswerRepository::class)]
|
#[ORM\Entity(repositoryClass: GivenAnswerRepository::class)]
|
||||||
#[ORM\HasLifecycleCallbacks]
|
#[ORM\HasLifecycleCallbacks]
|
||||||
@@ -22,22 +22,24 @@ class GivenAnswer
|
|||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
private Uuid $id;
|
private Uuid $id;
|
||||||
|
|
||||||
#[ORM\ManyToOne(inversedBy: 'givenAnswers')]
|
#[ORM\Column(type: Types::DATETIMETZ_IMMUTABLE, nullable: false)]
|
||||||
#[ORM\JoinColumn(nullable: false)]
|
|
||||||
private Candidate $candidate;
|
|
||||||
|
|
||||||
#[ORM\ManyToOne]
|
|
||||||
#[ORM\JoinColumn(nullable: false)]
|
|
||||||
private Quiz $quiz;
|
|
||||||
|
|
||||||
#[ORM\ManyToOne(inversedBy: 'givenAnswers')]
|
|
||||||
#[ORM\JoinColumn(nullable: true)]
|
|
||||||
private ?Answer $answer = null;
|
|
||||||
|
|
||||||
#[ORM\Column(type: Types::DATETIME_IMMUTABLE, nullable: false)]
|
|
||||||
private \DateTimeImmutable $created;
|
private \DateTimeImmutable $created;
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
public function __construct(
|
||||||
|
#[ORM\ManyToOne(inversedBy: 'givenAnswers')]
|
||||||
|
#[ORM\JoinColumn(nullable: false)]
|
||||||
|
private Candidate $candidate,
|
||||||
|
|
||||||
|
#[ORM\ManyToOne]
|
||||||
|
#[ORM\JoinColumn(nullable: false, onDelete: 'CASCADE')]
|
||||||
|
private Quiz $quiz,
|
||||||
|
|
||||||
|
#[ORM\ManyToOne(inversedBy: 'givenAnswers')]
|
||||||
|
#[ORM\JoinColumn(nullable: false)]
|
||||||
|
private Answer $answer,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function getId(): Uuid
|
||||||
{
|
{
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
@@ -47,37 +49,16 @@ class GivenAnswer
|
|||||||
return $this->candidate;
|
return $this->candidate;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setCandidate(Candidate $candidate): static
|
public function getQuiz(): Quiz
|
||||||
{
|
|
||||||
$this->candidate = $candidate;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getQuiz(): ?Quiz
|
|
||||||
{
|
{
|
||||||
return $this->quiz;
|
return $this->quiz;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setQuiz(Quiz $quiz): static
|
public function getAnswer(): Answer
|
||||||
{
|
|
||||||
$this->quiz = $quiz;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getAnswer(): ?Answer
|
|
||||||
{
|
{
|
||||||
return $this->answer;
|
return $this->answer;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setAnswer(?Answer $answer): static
|
|
||||||
{
|
|
||||||
$this->answer = $answer;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getCreated(): \DateTimeImmutable
|
public function getCreated(): \DateTimeImmutable
|
||||||
{
|
{
|
||||||
return $this->created;
|
return $this->created;
|
||||||
|
|||||||
@@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\QuestionRepository;
|
|
||||||
use Doctrine\Common\Collections\ArrayCollection;
|
use Doctrine\Common\Collections\ArrayCollection;
|
||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\DBAL\Types\Types;
|
use Doctrine\DBAL\Types\Types;
|
||||||
@@ -12,6 +11,7 @@ use Doctrine\ORM\Mapping as ORM;
|
|||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\QuestionRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: QuestionRepository::class)]
|
#[ORM\Entity(repositoryClass: QuestionRepository::class)]
|
||||||
class Question
|
class Question
|
||||||
@@ -20,7 +20,7 @@ class Question
|
|||||||
#[ORM\Column(type: UuidType::NAME)]
|
#[ORM\Column(type: UuidType::NAME)]
|
||||||
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
private ?Uuid $id = null;
|
private Uuid $id;
|
||||||
|
|
||||||
#[ORM\Column(type: Types::SMALLINT, options: ['default' => 0])]
|
#[ORM\Column(type: Types::SMALLINT, options: ['default' => 0])]
|
||||||
private int $ordering;
|
private int $ordering;
|
||||||
@@ -45,7 +45,7 @@ class Question
|
|||||||
$this->answers = new ArrayCollection();
|
$this->answers = new ArrayCollection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
public function getId(): Uuid
|
||||||
{
|
{
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,15 +2,15 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\QuizRepository;
|
|
||||||
use Doctrine\Common\Collections\ArrayCollection;
|
use Doctrine\Common\Collections\ArrayCollection;
|
||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\QuizRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: QuizRepository::class)]
|
#[ORM\Entity(repositoryClass: QuizRepository::class)]
|
||||||
#[ORM\UniqueConstraint(fields: ['name', 'season'])]
|
#[ORM\UniqueConstraint(fields: ['name', 'season'])]
|
||||||
@@ -20,7 +20,7 @@ class Quiz
|
|||||||
#[ORM\Column(type: UuidType::NAME)]
|
#[ORM\Column(type: UuidType::NAME)]
|
||||||
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
private ?Uuid $id = null;
|
private Uuid $id;
|
||||||
|
|
||||||
#[ORM\Column(length: 64)]
|
#[ORM\Column(length: 64)]
|
||||||
private string $name;
|
private string $name;
|
||||||
@@ -34,9 +34,9 @@ class Quiz
|
|||||||
#[ORM\OrderBy(['ordering' => 'ASC'])]
|
#[ORM\OrderBy(['ordering' => 'ASC'])]
|
||||||
private Collection $questions;
|
private Collection $questions;
|
||||||
|
|
||||||
/** @var Collection<int, Correction> */
|
/** @var Collection<int, QuizCandidate> */
|
||||||
#[ORM\OneToMany(targetEntity: Correction::class, mappedBy: 'quiz', orphanRemoval: true)]
|
#[ORM\OneToMany(targetEntity: QuizCandidate::class, mappedBy: 'quiz', orphanRemoval: true)]
|
||||||
private Collection $corrections;
|
private Collection $candidateData;
|
||||||
|
|
||||||
#[ORM\Column(nullable: false, options: ['default' => 1])]
|
#[ORM\Column(nullable: false, options: ['default' => 1])]
|
||||||
private int $dropouts = 1;
|
private int $dropouts = 1;
|
||||||
@@ -49,11 +49,11 @@ class Quiz
|
|||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
||||||
$this->questions = new ArrayCollection();
|
$this->questions = new ArrayCollection();
|
||||||
$this->corrections = new ArrayCollection();
|
$this->candidateData = new ArrayCollection();
|
||||||
$this->eliminations = new ArrayCollection();
|
$this->eliminations = new ArrayCollection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
public function getId(): Uuid
|
||||||
{
|
{
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
@@ -98,20 +98,10 @@ class Quiz
|
|||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @return Collection<int, Correction> */
|
/** @return Collection<int, QuizCandidate> */
|
||||||
public function getCorrections(): Collection
|
public function getCandidateData(): Collection
|
||||||
{
|
{
|
||||||
return $this->corrections;
|
return $this->candidateData;
|
||||||
}
|
|
||||||
|
|
||||||
public function addCorrection(Correction $correction): static
|
|
||||||
{
|
|
||||||
if (!$this->corrections->contains($correction)) {
|
|
||||||
$this->corrections->add($correction);
|
|
||||||
$correction->setQuiz($this);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getDropouts(): int
|
public function getDropouts(): int
|
||||||
|
|||||||
79
src/Entity/QuizCandidate.php
Normal file
79
src/Entity/QuizCandidate.php
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Types\Types;
|
||||||
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
|
use Safe\DateTimeImmutable;
|
||||||
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\QuizCandidateRepository;
|
||||||
|
|
||||||
|
#[ORM\Entity(repositoryClass: QuizCandidateRepository::class)]
|
||||||
|
#[ORM\UniqueConstraint(columns: ['candidate_id', 'quiz_id'])]
|
||||||
|
#[ORM\HasLifecycleCallbacks]
|
||||||
|
class QuizCandidate
|
||||||
|
{
|
||||||
|
#[ORM\Id]
|
||||||
|
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
||||||
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
|
private Uuid $id;
|
||||||
|
|
||||||
|
#[ORM\Column]
|
||||||
|
private float $corrections = 0;
|
||||||
|
|
||||||
|
#[ORM\Column(type: Types::DATETIMETZ_IMMUTABLE)]
|
||||||
|
private \DateTimeImmutable $created;
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
#[ORM\ManyToOne(inversedBy: 'candidateData')]
|
||||||
|
#[ORM\JoinColumn(nullable: false)]
|
||||||
|
private Quiz $quiz,
|
||||||
|
|
||||||
|
#[ORM\ManyToOne(inversedBy: 'quizData')]
|
||||||
|
#[ORM\JoinColumn(nullable: false)]
|
||||||
|
private Candidate $candidate,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function getId(): Uuid
|
||||||
|
{
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCandidate(): Candidate
|
||||||
|
{
|
||||||
|
return $this->candidate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getQuiz(): Quiz
|
||||||
|
{
|
||||||
|
return $this->quiz;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCorrections(): ?float
|
||||||
|
{
|
||||||
|
return $this->corrections;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setCorrections(float $corrections): static
|
||||||
|
{
|
||||||
|
$this->corrections = $corrections;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCreated(): \DateTimeImmutable
|
||||||
|
{
|
||||||
|
return $this->created;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[ORM\PrePersist]
|
||||||
|
public function setCreatedAtValue(): void
|
||||||
|
{
|
||||||
|
$this->created = new DateTimeImmutable();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,15 +2,15 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\SeasonRepository;
|
|
||||||
use Doctrine\Common\Collections\ArrayCollection;
|
use Doctrine\Common\Collections\ArrayCollection;
|
||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\ORM\Mapping as ORM;
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
use Symfony\Bridge\Doctrine\Types\UuidType;
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\SeasonRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: SeasonRepository::class)]
|
#[ORM\Entity(repositoryClass: SeasonRepository::class)]
|
||||||
class Season
|
class Season
|
||||||
@@ -21,7 +21,7 @@ class Season
|
|||||||
#[ORM\Column(type: UuidType::NAME)]
|
#[ORM\Column(type: UuidType::NAME)]
|
||||||
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
private ?Uuid $id = null;
|
private Uuid $id;
|
||||||
|
|
||||||
#[ORM\Column(length: 64)]
|
#[ORM\Column(length: 64)]
|
||||||
private string $name;
|
private string $name;
|
||||||
@@ -43,16 +43,22 @@ class Season
|
|||||||
private Collection $owners;
|
private Collection $owners;
|
||||||
|
|
||||||
#[ORM\ManyToOne]
|
#[ORM\ManyToOne]
|
||||||
|
#[ORM\JoinColumn(nullable: true, onDelete: 'SET NULL')]
|
||||||
private ?Quiz $ActiveQuiz = null;
|
private ?Quiz $ActiveQuiz = null;
|
||||||
|
|
||||||
|
#[ORM\OneToOne(cascade: ['persist', 'remove'])]
|
||||||
|
#[ORM\JoinColumn(nullable: true)]
|
||||||
|
private ?SeasonSettings $settings = null;
|
||||||
|
|
||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
||||||
|
$this->settings = new SeasonSettings();
|
||||||
$this->quizzes = new ArrayCollection();
|
$this->quizzes = new ArrayCollection();
|
||||||
$this->candidates = new ArrayCollection();
|
$this->candidates = new ArrayCollection();
|
||||||
$this->owners = new ArrayCollection();
|
$this->owners = new ArrayCollection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
public function getId(): Uuid
|
||||||
{
|
{
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
@@ -165,4 +171,16 @@ class Season
|
|||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getSettings(): ?SeasonSettings
|
||||||
|
{
|
||||||
|
return $this->settings;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setSettings(SeasonSettings $settings): static
|
||||||
|
{
|
||||||
|
$this->settings = $settings;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
57
src/Entity/SeasonSettings.php
Normal file
57
src/Entity/SeasonSettings.php
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
|
use Doctrine\DBAL\Types\Types;
|
||||||
|
use Doctrine\ORM\Mapping as ORM;
|
||||||
|
use Symfony\Bridge\Doctrine\IdGenerator\UuidGenerator;
|
||||||
|
use Symfony\Bridge\Doctrine\Types\UuidType;
|
||||||
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\SeasonSettingsRepository;
|
||||||
|
|
||||||
|
#[ORM\Entity(repositoryClass: SeasonSettingsRepository::class)]
|
||||||
|
class SeasonSettings
|
||||||
|
{
|
||||||
|
#[ORM\Id]
|
||||||
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
|
#[ORM\Column(type: UuidType::NAME)]
|
||||||
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
|
private Uuid $id;
|
||||||
|
|
||||||
|
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
|
||||||
|
private bool $showNumbers = false;
|
||||||
|
|
||||||
|
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => false])]
|
||||||
|
private bool $confirmAnswers = false;
|
||||||
|
|
||||||
|
public function getId(): Uuid
|
||||||
|
{
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isShowNumbers(): bool
|
||||||
|
{
|
||||||
|
return $this->showNumbers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setShowNumbers(bool $showNumbers): self
|
||||||
|
{
|
||||||
|
$this->showNumbers = $showNumbers;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isConfirmAnswers(): bool
|
||||||
|
{
|
||||||
|
return $this->confirmAnswers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setConfirmAnswers(bool $confirmAnswers): self
|
||||||
|
{
|
||||||
|
$this->confirmAnswers = $confirmAnswers;
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Entity;
|
namespace Tvdt\Entity;
|
||||||
|
|
||||||
use App\Repository\UserRepository;
|
|
||||||
use Doctrine\Common\Collections\ArrayCollection;
|
use Doctrine\Common\Collections\ArrayCollection;
|
||||||
use Doctrine\Common\Collections\Collection;
|
use Doctrine\Common\Collections\Collection;
|
||||||
use Doctrine\DBAL\Types\Types;
|
use Doctrine\DBAL\Types\Types;
|
||||||
@@ -15,6 +14,7 @@ use Symfony\Bridge\Doctrine\Validator\Constraints\UniqueEntity;
|
|||||||
use Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface;
|
use Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface;
|
||||||
use Symfony\Component\Security\Core\User\UserInterface;
|
use Symfony\Component\Security\Core\User\UserInterface;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Repository\UserRepository;
|
||||||
|
|
||||||
#[ORM\Entity(repositoryClass: UserRepository::class)]
|
#[ORM\Entity(repositoryClass: UserRepository::class)]
|
||||||
#[ORM\Table(name: '`user`')]
|
#[ORM\Table(name: '`user`')]
|
||||||
@@ -26,7 +26,7 @@ class User implements UserInterface, PasswordAuthenticatedUserInterface
|
|||||||
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
#[ORM\Column(type: UuidType::NAME, unique: true)]
|
||||||
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
#[ORM\GeneratedValue(strategy: 'CUSTOM')]
|
||||||
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
#[ORM\CustomIdGenerator(class: UuidGenerator::class)]
|
||||||
private ?Uuid $id = null;
|
private Uuid $id;
|
||||||
|
|
||||||
#[ORM\Column(length: 180)]
|
#[ORM\Column(length: 180)]
|
||||||
private string $email;
|
private string $email;
|
||||||
@@ -51,7 +51,7 @@ class User implements UserInterface, PasswordAuthenticatedUserInterface
|
|||||||
$this->seasons = new ArrayCollection();
|
$this->seasons = new ArrayCollection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getId(): ?Uuid
|
public function getId(): Uuid
|
||||||
{
|
{
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Enum;
|
namespace Tvdt\Enum;
|
||||||
|
|
||||||
enum FlashType: string
|
enum FlashType: string
|
||||||
{
|
{
|
||||||
|
|||||||
7
src/Exception/ErrorClearingQuizException.php
Normal file
7
src/Exception/ErrorClearingQuizException.php
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Exception;
|
||||||
|
|
||||||
|
class ErrorClearingQuizException extends \Exception {}
|
||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Exception;
|
namespace Tvdt\Exception;
|
||||||
|
|
||||||
class SpreadsheetDataException extends SpreadsheetException
|
class SpreadsheetDataException extends SpreadsheetException
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -2,6 +2,6 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Exception;
|
namespace Tvdt\Exception;
|
||||||
|
|
||||||
class SpreadsheetException extends \Exception {}
|
class SpreadsheetException extends \Exception {}
|
||||||
|
|||||||
@@ -2,12 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Factory;
|
namespace Tvdt\Factory;
|
||||||
|
|
||||||
use App\Entity\Elimination;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Repository\CandidateRepository;
|
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
|
use Tvdt\Entity\Elimination;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Repository\CandidateRepository;
|
||||||
|
|
||||||
final readonly class EliminationFactory
|
final readonly class EliminationFactory
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\TextareaType;
|
use Symfony\Component\Form\Extension\Core\Type\TextareaType;
|
||||||
|
|||||||
@@ -2,14 +2,14 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use App\Entity\Season;
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
||||||
use Symfony\Component\Form\FormBuilderInterface;
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
use Symfony\Component\OptionsResolver\OptionsResolver;
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
|
||||||
/** @extends AbstractType<Season> */
|
/** @extends AbstractType<Season> */
|
||||||
class CreateSeasonFormType extends AbstractType
|
class CreateSeasonFormType extends AbstractType
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
||||||
|
|||||||
@@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use App\Entity\User;
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\EmailType;
|
use Symfony\Component\Form\Extension\Core\Type\EmailType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\PasswordType;
|
use Symfony\Component\Form\Extension\Core\Type\PasswordType;
|
||||||
@@ -14,6 +13,7 @@ use Symfony\Component\OptionsResolver\OptionsResolver;
|
|||||||
use Symfony\Component\Validator\Constraints\Length;
|
use Symfony\Component\Validator\Constraints\Length;
|
||||||
use Symfony\Component\Validator\Constraints\NotBlank;
|
use Symfony\Component\Validator\Constraints\NotBlank;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Entity\User;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends AbstractType<User>
|
* @extends AbstractType<User>
|
||||||
@@ -39,15 +39,8 @@ class RegistrationFormType extends AbstractType
|
|||||||
'second_options' => ['label' => $this->translator->trans('Repeat Password')],
|
'second_options' => ['label' => $this->translator->trans('Repeat Password')],
|
||||||
'mapped' => false,
|
'mapped' => false,
|
||||||
'constraints' => [
|
'constraints' => [
|
||||||
new NotBlank([
|
new NotBlank(message: 'Please enter a password'),
|
||||||
'message' => 'Please enter a password',
|
new Length(min: 8, max: 4096, minMessage: 'Your password should be at least {{ limit }} characters'),
|
||||||
]),
|
|
||||||
new Length([
|
|
||||||
'min' => 8,
|
|
||||||
'minMessage' => 'Your password should be at least {{ limit }} characters',
|
|
||||||
// max length allowed by Symfony for security reasons
|
|
||||||
'max' => 4096,
|
|
||||||
]),
|
|
||||||
],
|
],
|
||||||
'translation_domain' => false,
|
'translation_domain' => false,
|
||||||
])
|
])
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
||||||
|
|||||||
35
src/Form/SettingsForm.php
Normal file
35
src/Form/SettingsForm.php
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
|
use Symfony\Component\Form\AbstractType;
|
||||||
|
use Symfony\Component\Form\Extension\Core\Type\SubmitType;
|
||||||
|
use Symfony\Component\Form\FormBuilderInterface;
|
||||||
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
|
use Tvdt\Entity\SeasonSettings;
|
||||||
|
|
||||||
|
/** @extends AbstractType<SeasonSettings> */
|
||||||
|
class SettingsForm extends AbstractType
|
||||||
|
{
|
||||||
|
public function buildForm(FormBuilderInterface $builder, array $options): void
|
||||||
|
{
|
||||||
|
$builder
|
||||||
|
->add('showNumbers', options: [
|
||||||
|
'label_attr' => ['class' => 'checkbox-switch'],
|
||||||
|
'attr' => ['role' => 'switch', 'switch' => null]])
|
||||||
|
->add('confirmAnswers', options: [
|
||||||
|
'label_attr' => ['class' => 'checkbox-switch'],
|
||||||
|
'attr' => ['role' => 'switch', 'switch' => null]])
|
||||||
|
->add('save', SubmitType::class)
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function configureOptions(OptionsResolver $resolver): void
|
||||||
|
{
|
||||||
|
$resolver->setDefaults([
|
||||||
|
'data_class' => SeasonSettings::class,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Form;
|
namespace Tvdt\Form;
|
||||||
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use Symfony\Component\Form\AbstractType;
|
use Symfony\Component\Form\AbstractType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\FileType;
|
use Symfony\Component\Form\Extension\Core\Type\FileType;
|
||||||
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
use Symfony\Component\Form\Extension\Core\Type\TextType;
|
||||||
@@ -12,6 +11,7 @@ use Symfony\Component\Form\FormBuilderInterface;
|
|||||||
use Symfony\Component\OptionsResolver\OptionsResolver;
|
use Symfony\Component\OptionsResolver\OptionsResolver;
|
||||||
use Symfony\Component\Validator\Constraints\File;
|
use Symfony\Component\Validator\Constraints\File;
|
||||||
use Symfony\Contracts\Translation\TranslatorInterface;
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
|
||||||
/** @extends AbstractType<Quiz> */
|
/** @extends AbstractType<Quiz> */
|
||||||
class UploadQuizFormType extends AbstractType
|
class UploadQuizFormType extends AbstractType
|
||||||
@@ -31,13 +31,9 @@ class UploadQuizFormType extends AbstractType
|
|||||||
'required' => true,
|
'required' => true,
|
||||||
'translation_domain' => false,
|
'translation_domain' => false,
|
||||||
'constraints' => [
|
'constraints' => [
|
||||||
new File([
|
new File(maxSize: '1024k', mimeTypes: [
|
||||||
'maxSize' => '1024k',
|
|
||||||
'mimeTypes' => [
|
|
||||||
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
|
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
|
||||||
],
|
], mimeTypesMessage: $this->translator->trans('Please upload a valid XLSX file')),
|
||||||
'mimeTypesMessage' => $this->translator->trans('Please upload a valid XLSX file'),
|
|
||||||
]),
|
|
||||||
],
|
],
|
||||||
])
|
])
|
||||||
;
|
;
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Helpers;
|
namespace Tvdt\Helpers;
|
||||||
|
|
||||||
use Safe\Exceptions\UrlException;
|
use Safe\Exceptions\UrlException;
|
||||||
|
|
||||||
@@ -10,7 +10,7 @@ class Base64
|
|||||||
{
|
{
|
||||||
public static function base64UrlEncode(string $input): string
|
public static function base64UrlEncode(string $input): string
|
||||||
{
|
{
|
||||||
return rtrim(strtr(base64_encode($input), '+/', '-_'), '=');
|
return mb_rtrim(strtr(base64_encode($input), '+/', '-_'), '=');
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @throws UrlException */
|
/** @throws UrlException */
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App;
|
namespace Tvdt;
|
||||||
|
|
||||||
use Symfony\Bundle\FrameworkBundle\Kernel\MicroKernelTrait;
|
use Symfony\Bundle\FrameworkBundle\Kernel\MicroKernelTrait;
|
||||||
use Symfony\Component\HttpKernel\Kernel as BaseKernel;
|
use Symfony\Component\HttpKernel\Kernel as BaseKernel;
|
||||||
|
|||||||
@@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\Answer;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\Answer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<Answer>
|
* @extends ServiceEntityRepository<Answer>
|
||||||
|
|||||||
@@ -2,23 +2,21 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\Correction;
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Helpers\Base64;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\ORM\Query\Expr\Join;
|
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
use Safe\Exceptions\UrlException;
|
use Safe\Exceptions\UrlException;
|
||||||
use Symfony\Component\Uid\Uuid;
|
use Symfony\Component\Uid\Uuid;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Helpers\Base64;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<Candidate>
|
* @extends ServiceEntityRepository<Candidate>
|
||||||
*
|
*
|
||||||
* @phpstan-type Result array{id: Uuid, name: string, correct: int, time: \DateInterval, corrections?: float, score: float}
|
* @phpstan-type Result array{id: Uuid, name: string, correct: int, time: \DateInterval, corrections: float, score: float}
|
||||||
* @phpstan-type ResultList list<Result>
|
* @phpstan-type ResultList list<Result>
|
||||||
*/
|
*/
|
||||||
class CandidateRepository extends ServiceEntityRepository
|
class CandidateRepository extends ServiceEntityRepository
|
||||||
@@ -36,12 +34,14 @@ class CandidateRepository extends ServiceEntityRepository
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->createQueryBuilder('c')
|
return $this->getEntityManager()->createQuery(<<<DQL
|
||||||
->where('c.season = :season')
|
select c from Tvdt\Entity\Candidate c
|
||||||
->andWhere('lower(c.name) = lower(:name)')
|
where c.season = :season
|
||||||
->setParameter('season', $season)
|
and lower(c.name) = lower(:name)
|
||||||
|
DQL
|
||||||
|
)->setParameter('season', $season)
|
||||||
->setParameter('name', $name)
|
->setParameter('name', $name)
|
||||||
->getQuery()->getOneOrNullResult();
|
->getOneOrNullResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function save(Candidate $candidate, bool $flush = true): void
|
public function save(Candidate $candidate, bool $flush = true): void
|
||||||
@@ -56,46 +56,22 @@ class CandidateRepository extends ServiceEntityRepository
|
|||||||
/** @return ResultList */
|
/** @return ResultList */
|
||||||
public function getScores(Quiz $quiz): array
|
public function getScores(Quiz $quiz): array
|
||||||
{
|
{
|
||||||
$scoreTimeQb = $this->createQueryBuilder('c', 'c.id')
|
return $this->getEntityManager()->createQuery(<<<DQL
|
||||||
->select('c.id', 'c.name', 'sum(case when a.isRightAnswer = true then 1 else 0 end) as correct', 'max(ga.created) - min(ga.created) as time')
|
select
|
||||||
->join('c.givenAnswers', 'ga')
|
c.id,
|
||||||
->join('ga.answer', 'a')
|
c.name,
|
||||||
->where('ga.quiz = :quiz')
|
sum(case when a.isRightAnswer = true then 1 else 0 end) as correct,
|
||||||
->groupBy('c.id')
|
qc.corrections,
|
||||||
->setParameter('quiz', $quiz);
|
max(ga.created) - qc.created as time,
|
||||||
|
(sum(case when a.isRightAnswer = true then 1 else 0 end) + qc.corrections) as score
|
||||||
$correctionsQb = $this->createQueryBuilder('c', 'c.id')
|
from Tvdt\Entity\Candidate c
|
||||||
->select('c.id', 'cor.amount as corrections')
|
join c.givenAnswers ga
|
||||||
->innerJoin(Correction::class, 'cor', Join::WITH, 'cor.candidate = c and cor.quiz = :quiz')
|
join ga.answer a
|
||||||
->setParameter('quiz', $quiz);
|
join c.quizData qc
|
||||||
|
where qc.quiz = :quiz and ga.quiz = :quiz
|
||||||
$merged = array_merge_recursive($scoreTimeQb->getQuery()->getArrayResult(), $correctionsQb->getQuery()->getArrayResult());
|
group by ga.quiz, c.id, qc.id
|
||||||
|
order by score desc, time asc
|
||||||
return $this->sortResults($this->calculateScore($merged));
|
DQL
|
||||||
}
|
)->setParameter('quiz', $quiz)->getResult();
|
||||||
|
|
||||||
/**
|
|
||||||
* @param array<string, array{id: Uuid, name: string, correct: int, time: \DateInterval, corrections?: float}> $in
|
|
||||||
*
|
|
||||||
* @return array<string, Result>
|
|
||||||
* */
|
|
||||||
private function calculateScore(array $in): array
|
|
||||||
{
|
|
||||||
return array_map(static fn ($candidate): array => [
|
|
||||||
...$candidate,
|
|
||||||
'score' => $candidate['correct'] + ($candidate['corrections'] ?? 0.0),
|
|
||||||
], $in);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param array<string, Result> $results
|
|
||||||
*
|
|
||||||
* @return ResultList
|
|
||||||
* */
|
|
||||||
private function sortResults(array $results): array
|
|
||||||
{
|
|
||||||
usort($results, static fn ($a, $b): int => $b['score'] <=> $a['score']);
|
|
||||||
|
|
||||||
return $results;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
declare(strict_types=1);
|
|
||||||
|
|
||||||
namespace App\Repository;
|
|
||||||
|
|
||||||
use App\Entity\Correction;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @extends ServiceEntityRepository<Correction>
|
|
||||||
*/
|
|
||||||
class CorrectionRepository extends ServiceEntityRepository
|
|
||||||
{
|
|
||||||
public function __construct(ManagerRegistry $registry)
|
|
||||||
{
|
|
||||||
parent::__construct($registry, Correction::class);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\Elimination;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\Elimination;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<Elimination>
|
* @extends ServiceEntityRepository<Elimination>
|
||||||
@@ -17,29 +17,4 @@ class EliminationRepository extends ServiceEntityRepository
|
|||||||
{
|
{
|
||||||
parent::__construct($registry, Elimination::class);
|
parent::__construct($registry, Elimination::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
// /**
|
|
||||||
// * @return Elimination[] Returns an array of Elimination objects
|
|
||||||
// */
|
|
||||||
// public function findByExampleField($value): array
|
|
||||||
// {
|
|
||||||
// return $this->createQueryBuilder('e')
|
|
||||||
// ->andWhere('e.exampleField = :val')
|
|
||||||
// ->setParameter('val', $value)
|
|
||||||
// ->orderBy('e.id', 'ASC')
|
|
||||||
// ->setMaxResults(10)
|
|
||||||
// ->getQuery()
|
|
||||||
// ->getResult()
|
|
||||||
// ;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public function findOneBySomeField($value): ?Elimination
|
|
||||||
// {
|
|
||||||
// return $this->createQueryBuilder('e')
|
|
||||||
// ->andWhere('e.exampleField = :val')
|
|
||||||
// ->setParameter('val', $value)
|
|
||||||
// ->getQuery()
|
|
||||||
// ->getOneOrNullResult()
|
|
||||||
// ;
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\GivenAnswer;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\GivenAnswer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<GivenAnswer>
|
* @extends ServiceEntityRepository<GivenAnswer>
|
||||||
|
|||||||
@@ -2,13 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\Candidate;
|
|
||||||
use App\Entity\GivenAnswer;
|
|
||||||
use App\Entity\Question;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Question;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<Question>
|
* @extends ServiceEntityRepository<Question>
|
||||||
@@ -22,22 +21,21 @@ class QuestionRepository extends ServiceEntityRepository
|
|||||||
|
|
||||||
public function findNextQuestionForCandidate(Candidate $candidate): ?Question
|
public function findNextQuestionForCandidate(Candidate $candidate): ?Question
|
||||||
{
|
{
|
||||||
$qb = $this->createQueryBuilder('q');
|
return $this->getEntityManager()->createQuery(<<<DQL
|
||||||
|
select q from Tvdt\Entity\Question q
|
||||||
return $qb->join('q.quiz', 'qz')
|
join q.quiz qz
|
||||||
->andWhere($qb->expr()->notIn('q.id', $this->getEntityManager()->createQueryBuilder()
|
where q.id not in (
|
||||||
->select('q1')
|
select q1.id from Tvdt\Entity\GivenAnswer ga
|
||||||
->from(GivenAnswer::class, 'ga')
|
join ga.answer a
|
||||||
->join('ga.answer', 'a')
|
join a.question q1
|
||||||
->join('a.question', 'q1')
|
where ga.candidate = :candidate
|
||||||
->andWhere($qb->expr()->isNotNull('ga.answer'))
|
and q1.quiz = :quiz
|
||||||
->andWhere('ga.candidate = :candidate')
|
)
|
||||||
->andWhere('q1.quiz = :quiz')
|
and qz = :quiz
|
||||||
->getDQL()))
|
DQL)
|
||||||
->andWhere('qz = :quiz')
|
|
||||||
->setMaxResults(1)
|
->setMaxResults(1)
|
||||||
->setParameter('candidate', $candidate)
|
->setParameter('candidate', $candidate)
|
||||||
->setParameter('quiz', $candidate->getSeason()->getActiveQuiz())
|
->setParameter('quiz', $candidate->getSeason()->getActiveQuiz())
|
||||||
->getQuery()->getOneOrNullResult();
|
->getOneOrNullResult();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
47
src/Repository/QuizCandidateRepository.php
Normal file
47
src/Repository/QuizCandidateRepository.php
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\Candidate;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\QuizCandidate;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @extends ServiceEntityRepository<QuizCandidate>
|
||||||
|
*/
|
||||||
|
class QuizCandidateRepository extends ServiceEntityRepository
|
||||||
|
{
|
||||||
|
public function __construct(ManagerRegistry $registry)
|
||||||
|
{
|
||||||
|
parent::__construct($registry, QuizCandidate::class);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return bool true if a new entry was created */
|
||||||
|
public function createIfNotExist(Quiz $quiz, Candidate $candidate): bool
|
||||||
|
{
|
||||||
|
if (0 !== $this->count(['candidate' => $candidate, 'quiz' => $quiz])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$quizCandidate = new QuizCandidate($quiz, $candidate);
|
||||||
|
$this->getEntityManager()->persist($quizCandidate);
|
||||||
|
$this->getEntityManager()->flush();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setCorrectionsForCandidate(Quiz $quiz, Candidate $candidate, float $corrections): void
|
||||||
|
{
|
||||||
|
$quizCandidate = $this->findOneBy(['candidate' => $candidate, 'quiz' => $quiz]);
|
||||||
|
if (!$quizCandidate instanceof QuizCandidate) {
|
||||||
|
throw new \InvalidArgumentException('Quiz candidate not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
$quizCandidate->setCorrections($corrections);
|
||||||
|
$this->getEntityManager()->flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,19 +2,61 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\Quiz;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Psr\Log\LoggerInterface;
|
||||||
|
use Tvdt\Entity\Elimination;
|
||||||
|
use Tvdt\Entity\GivenAnswer;
|
||||||
|
use Tvdt\Entity\Quiz;
|
||||||
|
use Tvdt\Entity\QuizCandidate;
|
||||||
|
use Tvdt\Exception\ErrorClearingQuizException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<Quiz>
|
* @extends ServiceEntityRepository<Quiz>
|
||||||
*/
|
*/
|
||||||
class QuizRepository extends ServiceEntityRepository
|
class QuizRepository extends ServiceEntityRepository
|
||||||
{
|
{
|
||||||
public function __construct(ManagerRegistry $registry)
|
public function __construct(ManagerRegistry $registry, private readonly LoggerInterface $logger)
|
||||||
{
|
{
|
||||||
parent::__construct($registry, Quiz::class);
|
parent::__construct($registry, Quiz::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @throws ErrorClearingQuizException */
|
||||||
|
public function clearQuiz(Quiz $quiz): void
|
||||||
|
{
|
||||||
|
$em = $this->getEntityManager();
|
||||||
|
$em->beginTransaction();
|
||||||
|
try {
|
||||||
|
$em->createQueryBuilder()
|
||||||
|
->delete()->from(QuizCandidate::class, 'qc')
|
||||||
|
->where('qc.quiz = :quiz')
|
||||||
|
->setParameter('quiz', $quiz)
|
||||||
|
->getQuery()->execute();
|
||||||
|
|
||||||
|
$em->createQueryBuilder()
|
||||||
|
->delete()->from(GivenAnswer::class, 'ga')
|
||||||
|
->where('ga.quiz = :quiz')
|
||||||
|
->setParameter('quiz', $quiz)
|
||||||
|
->getQuery()->execute();
|
||||||
|
$em->createQueryBuilder()
|
||||||
|
->delete()->from(Elimination::class, 'e')
|
||||||
|
->where('e.quiz = :quiz')
|
||||||
|
->setParameter('quiz', $quiz)
|
||||||
|
->getQuery()->execute();
|
||||||
|
} catch (\Throwable $throwable) {
|
||||||
|
$this->logger->error($throwable->getMessage());
|
||||||
|
$em->rollback();
|
||||||
|
throw new ErrorClearingQuizException(previous: $throwable);
|
||||||
|
}
|
||||||
|
|
||||||
|
$em->commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function deleteQuiz(Quiz $quiz): void
|
||||||
|
{
|
||||||
|
$this->getEntityManager()->remove($quiz);
|
||||||
|
$this->getEntityManager()->flush();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,12 +2,12 @@
|
|||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
namespace App\Repository;
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
use App\Entity\Season;
|
|
||||||
use App\Entity\User;
|
|
||||||
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
use Doctrine\Persistence\ManagerRegistry;
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\Season;
|
||||||
|
use Tvdt\Entity\User;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @extends ServiceEntityRepository<Season>
|
* @extends ServiceEntityRepository<Season>
|
||||||
@@ -22,11 +22,9 @@ class SeasonRepository extends ServiceEntityRepository
|
|||||||
/** @return list<Season> Returns an array of Season objects */
|
/** @return list<Season> Returns an array of Season objects */
|
||||||
public function getSeasonsForUser(User $user): array
|
public function getSeasonsForUser(User $user): array
|
||||||
{
|
{
|
||||||
$qb = $this->createQueryBuilder('s')
|
return $this->getEntityManager()->createQuery(<<<DQL
|
||||||
->where(':user MEMBER OF s.owners')
|
select s from Tvdt\Entity\Season s where :user member of s.owners order by s.name
|
||||||
->orderBy('s.name')
|
DQL
|
||||||
->setParameter('user', $user);
|
)->setParameter('user', $user)->getResult();
|
||||||
|
|
||||||
return $qb->getQuery()->getResult();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
20
src/Repository/SeasonSettingsRepository.php
Normal file
20
src/Repository/SeasonSettingsRepository.php
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Tvdt\Repository;
|
||||||
|
|
||||||
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
||||||
|
use Doctrine\Persistence\ManagerRegistry;
|
||||||
|
use Tvdt\Entity\SeasonSettings;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @extends ServiceEntityRepository<SeasonSettings>
|
||||||
|
*/
|
||||||
|
class SeasonSettingsRepository extends ServiceEntityRepository
|
||||||
|
{
|
||||||
|
public function __construct(ManagerRegistry $registry)
|
||||||
|
{
|
||||||
|
parent::__construct($registry, SeasonSettings::class);
|
||||||
|
}
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user