Compare commits

..

1 Commits

Author SHA1 Message Date
Marijn 7a29bb49ea Try with services.yaml 2025-12-01 10:02:52 +01:00
90 changed files with 4453 additions and 5288 deletions
-6
View File
@@ -1,6 +0,0 @@
# yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json
language: "en-GB"
reviews:
sequence_diagrams: false
path_filters:
- "!config/reference.php"
+1 -1
View File
@@ -54,7 +54,7 @@ indent_size = 4
[{compose,docker-compose}.*.{yaml,yml}] [{compose,docker-compose}.*.{yaml,yml}]
indent_size = 2 indent_size = 2
[{*.*Dockerfile,Dockerfile}] [*.*Dockerfile]
indent_style = tab indent_style = tab
[{*.*Caddyfile,Caddyfile}] [{*.*Caddyfile,Caddyfile}]
-7
View File
@@ -17,7 +17,6 @@
###> symfony/framework-bundle ### ###> symfony/framework-bundle ###
APP_ENV=dev APP_ENV=dev
APP_SECRET= APP_SECRET=
APP_SHARE_DIR=var/share
###< symfony/framework-bundle ### ###< symfony/framework-bundle ###
###> doctrine/doctrine-bundle ### ###> doctrine/doctrine-bundle ###
@@ -38,9 +37,3 @@ MAILER_DSN=null://null
SENTRY_DSN= SENTRY_DSN=
###< sentry/sentry-symfony ### ###< sentry/sentry-symfony ###
XDEBUG_MODE=coverage XDEBUG_MODE=coverage
###> symfony/routing ###
# Configure how to generate URLs in non-HTTP contexts, such as CLI commands.
# See https://symfony.com/doc/current/routing.html#generating-urls-in-commands
DEFAULT_URI=http://localhost
###< symfony/routing ###
-1
View File
@@ -15,4 +15,3 @@ composer.lock text eol=lf merge=ours
*.ico binary *.ico binary
*.png binary *.png binary
config/reference.php linguist-generated
-13
View File
@@ -68,19 +68,6 @@ Icon
# Thumbnails # Thumbnails
._* ._*
# IDEs
/.idea/
/.vscode/
# Junie
!/.junie/
/.junie/memory/
/.junie/plans/
# Windows
Thumbs.db
Desktop.ini
# Files that might appear in the root of a volume # Files that might appear in the root of a volume
.DocumentRevisions-V100 .DocumentRevisions-V100
.fseventsd .fseventsd
+3
View File
@@ -45,6 +45,7 @@
<excludeFolder url="file://$MODULE_DIR$/vendor/react/socket" /> <excludeFolder url="file://$MODULE_DIR$/vendor/react/socket" />
<excludeFolder url="file://$MODULE_DIR$/vendor/react/stream" /> <excludeFolder url="file://$MODULE_DIR$/vendor/react/stream" />
<excludeFolder url="file://$MODULE_DIR$/vendor/rector/rector" /> <excludeFolder url="file://$MODULE_DIR$/vendor/rector/rector" />
<excludeFolder url="file://$MODULE_DIR$/vendor/runtime/frankenphp-symfony" />
<excludeFolder url="file://$MODULE_DIR$/vendor/sebastian/cli-parser" /> <excludeFolder url="file://$MODULE_DIR$/vendor/sebastian/cli-parser" />
<excludeFolder url="file://$MODULE_DIR$/vendor/sebastian/comparator" /> <excludeFolder url="file://$MODULE_DIR$/vendor/sebastian/comparator" />
<excludeFolder url="file://$MODULE_DIR$/vendor/sebastian/complexity" /> <excludeFolder url="file://$MODULE_DIR$/vendor/sebastian/complexity" />
@@ -126,6 +127,7 @@
<excludeFolder url="file://$MODULE_DIR$/vendor/vincentlanglet/twig-cs-fixer" /> <excludeFolder url="file://$MODULE_DIR$/vendor/vincentlanglet/twig-cs-fixer" />
<excludeFolder url="file://$MODULE_DIR$/vendor/webmozart/assert" /> <excludeFolder url="file://$MODULE_DIR$/vendor/webmozart/assert" />
<excludeFolder url="file://$MODULE_DIR$/vendor/egulias/email-validator" /> <excludeFolder url="file://$MODULE_DIR$/vendor/egulias/email-validator" />
<excludeFolder url="file://$MODULE_DIR$/vendor/masterminds/html5" />
<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" />
@@ -151,6 +153,7 @@
<excludeFolder url="file://$MODULE_DIR$/vendor/markbaker/complex" /> <excludeFolder url="file://$MODULE_DIR$/vendor/markbaker/complex" />
<excludeFolder url="file://$MODULE_DIR$/vendor/markbaker/matrix" /> <excludeFolder url="file://$MODULE_DIR$/vendor/markbaker/matrix" />
<excludeFolder url="file://$MODULE_DIR$/vendor/phpoffice/phpspreadsheet" /> <excludeFolder url="file://$MODULE_DIR$/vendor/phpoffice/phpspreadsheet" />
<excludeFolder url="file://$MODULE_DIR$/vendor/psr/http-client" />
<excludeFolder url="file://$MODULE_DIR$/vendor/psr/simple-cache" /> <excludeFolder url="file://$MODULE_DIR$/vendor/psr/simple-cache" />
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/asset-mapper" /> <excludeFolder url="file://$MODULE_DIR$/vendor/symfony/asset-mapper" />
<excludeFolder url="file://$MODULE_DIR$/vendor/symfony/http-client" /> <excludeFolder url="file://$MODULE_DIR$/vendor/symfony/http-client" />
-5
View File
@@ -6,11 +6,6 @@
<synchronize>true</synchronize> <synchronize>true</synchronize>
<jdbc-driver>org.postgresql.Driver</jdbc-driver> <jdbc-driver>org.postgresql.Driver</jdbc-driver>
<jdbc-url>jdbc:postgresql://localhost:5432/app</jdbc-url> <jdbc-url>jdbc:postgresql://localhost:5432/app</jdbc-url>
<jdbc-additional-properties>
<property name="com.intellij.clouds.kubernetes.db.host.port" />
<property name="com.intellij.clouds.kubernetes.db.enabled" value="false" />
<property name="com.intellij.clouds.kubernetes.db.container.port" />
</jdbc-additional-properties>
<working-dir>$ProjectFileDir$</working-dir> <working-dir>$ProjectFileDir$</working-dir>
</data-source> </data-source>
</component> </component>
-8
View File
@@ -1,8 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="InertiaPackage">
<option name="directoryPaths">
<list />
</option>
</component>
</project>
Generated
+161 -265
View File
@@ -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>
@@ -26,9 +21,9 @@
</component> </component>
<component name="PhpCSFixer"> <component name="PhpCSFixer">
<phpcsfixer_settings> <phpcsfixer_settings>
<phpcs_fixer_by_interpreter interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" standards="DoctrineAnnotation;PER;PER-CS;PER-CS1.0;PER-CS1x0;PER-CS2.0;PER-CS2x0;PER-CS3.0;PER-CS3x0;PHP54Migration;PHP56Migration;PHP5x4Migration;PHP5x6Migration;PHP70Migration;PHP71Migration;PHP73Migration;PHP74Migration;PHP7x0Migration;PHP7x1Migration;PHP7x3Migration;PHP7x4Migration;PHP80Migration;PHP81Migration;PHP82Migration;PHP83Migration;PHP84Migration;PHP85Migration;PHP8x0Migration;PHP8x1Migration;PHP8x2Migration;PHP8x3Migration;PHP8x4Migration;PHP8x5Migration;PHPUnit100Migration;PHPUnit10x0Migration;PHPUnit11x0Migration;PHPUnit30Migration;PHPUnit32Migration;PHPUnit35Migration;PHPUnit3x0Migration;PHPUnit3x2Migration;PHPUnit3x5Migration;PHPUnit43Migration;PHPUnit48Migration;PHPUnit4x3Migration;PHPUnit4x8Migration;PHPUnit50Migration;PHPUnit52Migration;PHPUnit54Migration;PHPUnit55Migration;PHPUnit56Migration;PHPUnit57Migration;PHPUnit5x0Migration;PHPUnit5x2Migration;PHPUnit5x4Migration;PHPUnit5x5Migration;PHPUnit5x6Migration;PHPUnit5x7Migration;PHPUnit60Migration;PHPUnit6x0Migration;PHPUnit75Migration;PHPUnit7x5Migration;PHPUnit84Migration;PHPUnit8x4Migration;PHPUnit91Migration;PHPUnit9x1Migration;PSR1;PSR12;PSR2;PhpCsFixer;Symfony;auto;autoPHPMigration;autoPHPUnitMigration" tool_path="vendor/bin/php-cs-fixer" timeout="30000" /> <phpcs_fixer_by_interpreter interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" standards="DoctrineAnnotation;PER;PER-CS;PER-CS1.0;PER-CS2.0;PHP54Migration;PHP56Migration;PHP70Migration;PHP71Migration;PHP73Migration;PHP74Migration;PHP80Migration;PHP81Migration;PHP82Migration;PHP83Migration;PHP84Migration;PHPUnit100Migration;PHPUnit30Migration;PHPUnit32Migration;PHPUnit35Migration;PHPUnit43Migration;PHPUnit48Migration;PHPUnit50Migration;PHPUnit52Migration;PHPUnit54Migration;PHPUnit55Migration;PHPUnit56Migration;PHPUnit57Migration;PHPUnit60Migration;PHPUnit75Migration;PHPUnit84Migration;PHPUnit91Migration;PSR1;PSR12;PSR2;PhpCsFixer;Symfony" tool_path="vendor/bin/php-cs-fixer" timeout="30000" />
<PhpCSFixerConfiguration deletedFromTheList="true" standards="PSR1;PSR2;Symfony;DoctrineAnnotation;PHP70Migration;PHP71Migration" tool_path="$PROJECT_DIR$/vendor/bin/php-cs-fixer" /> <PhpCSFixerConfiguration deletedFromTheList="true" standards="PSR1;PSR2;Symfony;DoctrineAnnotation;PHP70Migration;PHP71Migration" tool_path="$PROJECT_DIR$/vendor/bin/php-cs-fixer" />
<phpcs_fixer_by_interpreter asDefaultInterpreter="true" deletedFromTheList="true" interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" standards="DoctrineAnnotation;PER;PER-CS;PER-CS1.0;PER-CS1x0;PER-CS2.0;PER-CS2x0;PER-CS3.0;PER-CS3x0;PHP54Migration;PHP56Migration;PHP5x4Migration;PHP5x6Migration;PHP70Migration;PHP71Migration;PHP73Migration;PHP74Migration;PHP7x0Migration;PHP7x1Migration;PHP7x3Migration;PHP7x4Migration;PHP80Migration;PHP81Migration;PHP82Migration;PHP83Migration;PHP84Migration;PHP85Migration;PHP8x0Migration;PHP8x1Migration;PHP8x2Migration;PHP8x3Migration;PHP8x4Migration;PHP8x5Migration;PHPUnit100Migration;PHPUnit10x0Migration;PHPUnit11x0Migration;PHPUnit30Migration;PHPUnit32Migration;PHPUnit35Migration;PHPUnit3x0Migration;PHPUnit3x2Migration;PHPUnit3x5Migration;PHPUnit43Migration;PHPUnit48Migration;PHPUnit4x3Migration;PHPUnit4x8Migration;PHPUnit50Migration;PHPUnit52Migration;PHPUnit54Migration;PHPUnit55Migration;PHPUnit56Migration;PHPUnit57Migration;PHPUnit5x0Migration;PHPUnit5x2Migration;PHPUnit5x4Migration;PHPUnit5x5Migration;PHPUnit5x6Migration;PHPUnit5x7Migration;PHPUnit60Migration;PHPUnit6x0Migration;PHPUnit75Migration;PHPUnit7x5Migration;PHPUnit84Migration;PHPUnit8x4Migration;PHPUnit91Migration;PHPUnit9x1Migration;PSR1;PSR12;PSR2;PhpCsFixer;Symfony;auto;autoPHPMigration;autoPHPUnitMigration" tool_path="vendor/bin/php-cs-fixer" timeout="30000" /> <phpcs_fixer_by_interpreter asDefaultInterpreter="true" deletedFromTheList="true" interpreter_id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" standards="DoctrineAnnotation;PER;PER-CS;PER-CS1.0;PER-CS2.0;PHP54Migration;PHP56Migration;PHP70Migration;PHP71Migration;PHP73Migration;PHP74Migration;PHP80Migration;PHP81Migration;PHP82Migration;PHP83Migration;PHP84Migration;PHPUnit100Migration;PHPUnit30Migration;PHPUnit32Migration;PHPUnit35Migration;PHPUnit43Migration;PHPUnit48Migration;PHPUnit50Migration;PHPUnit52Migration;PHPUnit54Migration;PHPUnit55Migration;PHPUnit56Migration;PHPUnit57Migration;PHPUnit60Migration;PHPUnit75Migration;PHPUnit84Migration;PHPUnit91Migration;PSR1;PSR12;PSR2;PhpCsFixer;Symfony" tool_path="vendor/bin/php-cs-fixer" timeout="30000" />
</phpcsfixer_settings> </phpcsfixer_settings>
</component> </component>
<component name="PhpCodeSniffer"> <component name="PhpCodeSniffer">
@@ -41,170 +36,174 @@
</component> </component>
<component name="PhpIncludePathManager"> <component name="PhpIncludePathManager">
<include_path> <include_path>
<path value="$PROJECT_DIR$/vendor/symfony/cache" />
<path value="$PROJECT_DIR$/vendor/symfony/validator" />
<path value="$PROJECT_DIR$/vendor/symfony/dotenv" />
<path value="$PROJECT_DIR$/vendor/symfony/serializer" />
<path value="$PROJECT_DIR$/vendor/symfony/intl" />
<path value="$PROJECT_DIR$/vendor/symfony/security-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/browser-kit" />
<path value="$PROJECT_DIR$/vendor/symfony/asset" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-file-iterator" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-invoker" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-code-coverage" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-text-template" />
<path value="$PROJECT_DIR$/vendor/symfony/ux-turbo" />
<path value="$PROJECT_DIR$/vendor/symfony/form" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-icu" />
<path value="$PROJECT_DIR$/vendor/symfony/framework-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/mailer" />
<path value="$PROJECT_DIR$/vendor/symfony/brevo-mailer" />
<path value="$PROJECT_DIR$/vendor/symfony/var-dumper" />
<path value="$PROJECT_DIR$/vendor/symfony/phpunit-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-idn" />
<path value="$PROJECT_DIR$/vendor/symfony/security-csrf" />
<path value="$PROJECT_DIR$/vendor/symfony/options-resolver" />
<path value="$PROJECT_DIR$/vendor/symfony/filesystem" />
<path value="$PROJECT_DIR$/vendor/symfony/security-http" />
<path value="$PROJECT_DIR$/vendor/symfony/config" />
<path value="$PROJECT_DIR$/vendor/symfony/translation-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/translation" />
<path value="$PROJECT_DIR$/vendor/symfony/runtime" />
<path value="$PROJECT_DIR$/vendor/symfony/event-dispatcher" />
<path value="$PROJECT_DIR$/vendor/symfony/asset-mapper" />
<path value="$PROJECT_DIR$/vendor/symfony/uid" />
<path value="$PROJECT_DIR$/vendor/symfony/http-client" />
<path value="$PROJECT_DIR$/vendor/symfony/finder" />
<path value="$PROJECT_DIR$/vendor/symfony/twig-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/property-access" />
<path value="$PROJECT_DIR$/vendor/symfony/yaml" />
<path value="$PROJECT_DIR$/vendor/symfony/http-kernel" />
<path value="$PROJECT_DIR$/vendor/symfony/psr-http-message-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/stopwatch" />
<path value="$PROJECT_DIR$/vendor/symfony/web-profiler-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-uuid" />
<path value="$PROJECT_DIR$/vendor/symfony/twig-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/maker-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/dom-crawler" />
<path value="$PROJECT_DIR$/vendor/symfony/event-dispatcher-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/css-selector" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-normalizer" />
<path value="$PROJECT_DIR$/vendor/symfony/var-exporter" />
<path value="$PROJECT_DIR$/vendor/symfony/doctrine-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/http-foundation" />
<path value="$PROJECT_DIR$/vendor/symfony/type-info" />
<path value="$PROJECT_DIR$/vendor/symfony/dependency-injection" />
<path value="$PROJECT_DIR$/vendor/symfony/cache-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/error-handler" />
<path value="$PROJECT_DIR$/vendor/symfony/flex" />
<path value="$PROJECT_DIR$/vendor/symfony/deprecation-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/security-core" />
<path value="$PROJECT_DIR$/vendor/symfony/string" />
<path value="$PROJECT_DIR$/vendor/symfony/mime" />
<path value="$PROJECT_DIR$/vendor/symfony/property-info" />
<path value="$PROJECT_DIR$/vendor/symfony/http-client-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/service-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/routing" />
<path value="$PROJECT_DIR$/vendor/symfony/process" />
<path value="$PROJECT_DIR$/vendor/symfony/password-hasher" />
<path value="$PROJECT_DIR$/vendor/symfony/clock" />
<path value="$PROJECT_DIR$/vendor/symfony/stimulus-bundle" />
<path value="$PROJECT_DIR$/vendor/theseer/tokenizer" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-grapheme" />
<path value="$PROJECT_DIR$/vendor/symfony/console" />
<path value="$PROJECT_DIR$/vendor/doctrine/instantiator" />
<path value="$PROJECT_DIR$/vendor/doctrine/doctrine-fixtures-bundle" />
<path value="$PROJECT_DIR$/vendor/doctrine/persistence" />
<path value="$PROJECT_DIR$/vendor/doctrine/inflector" />
<path value="$PROJECT_DIR$/vendor/doctrine/sql-formatter" />
<path value="$PROJECT_DIR$/vendor/doctrine/collections" />
<path value="$PROJECT_DIR$/vendor/doctrine/event-manager" />
<path value="$PROJECT_DIR$/vendor/doctrine/orm" />
<path value="$PROJECT_DIR$/vendor/doctrine/doctrine-migrations-bundle" />
<path value="$PROJECT_DIR$/vendor/doctrine/dbal" />
<path value="$PROJECT_DIR$/vendor/doctrine/lexer" />
<path value="$PROJECT_DIR$/vendor/doctrine/data-fixtures" />
<path value="$PROJECT_DIR$/vendor/doctrine/migrations" />
<path value="$PROJECT_DIR$/vendor/sebastian/object-reflector" />
<path value="$PROJECT_DIR$/vendor/sebastian/object-enumerator" />
<path value="$PROJECT_DIR$/vendor/sebastian/global-state" />
<path value="$PROJECT_DIR$/vendor/sebastian/comparator" />
<path value="$PROJECT_DIR$/vendor/sebastian/lines-of-code" />
<path value="$PROJECT_DIR$/vendor/sebastian/recursion-context" />
<path value="$PROJECT_DIR$/vendor/sebastian/diff" />
<path value="$PROJECT_DIR$/vendor/sebastian/complexity" />
<path value="$PROJECT_DIR$/vendor/maennchen/zipstream-php" />
<path value="$PROJECT_DIR$/vendor/evenement/evenement" />
<path value="$PROJECT_DIR$/vendor/doctrine/doctrine-bundle" />
<path value="$PROJECT_DIR$/vendor/doctrine/deprecations" />
<path value="$PROJECT_DIR$/vendor/ralouphie/getallheaders" />
<path value="$PROJECT_DIR$/vendor/phpoffice/phpspreadsheet" />
<path value="$PROJECT_DIR$/vendor/markbaker/matrix" />
<path value="$PROJECT_DIR$/vendor/markbaker/complex" />
<path value="$PROJECT_DIR$/vendor/phpdocumentor/type-resolver" />
<path value="$PROJECT_DIR$/vendor/phpdocumentor/reflection-common" />
<path value="$PROJECT_DIR$/vendor/symfonycasts/sass-bundle" />
<path value="$PROJECT_DIR$/vendor/symfonycasts/verify-email-bundle" />
<path value="$PROJECT_DIR$/vendor/thecodingmachine/safe" />
<path value="$PROJECT_DIR$/vendor/martin-georgiev/postgresql-for-doctrine" />
<path value="$PROJECT_DIR$/vendor/vincentlanglet/twig-cs-fixer" />
<path value="$PROJECT_DIR$/vendor/phpdocumentor/reflection-docblock" />
<path value="$PROJECT_DIR$/vendor/sebastian/environment" />
<path value="$PROJECT_DIR$/vendor/sebastian/cli-parser" />
<path value="$PROJECT_DIR$/vendor/sebastian/version" />
<path value="$PROJECT_DIR$/vendor/sebastian/type" />
<path value="$PROJECT_DIR$/vendor/friendsofphp/php-cs-fixer" />
<path value="$PROJECT_DIR$/vendor/guzzlehttp/psr7" />
<path value="$PROJECT_DIR$/vendor/webmozart/assert" />
<path value="$PROJECT_DIR$/vendor/sebastian/exporter" />
<path value="$PROJECT_DIR$/vendor/thecodingmachine/phpstan-safe-rule" />
<path value="$PROJECT_DIR$/vendor/composer" /> <path value="$PROJECT_DIR$/vendor/composer" />
<path value="$PROJECT_DIR$/vendor/psr/log" /> <path value="$PROJECT_DIR$/vendor/webmozart/assert" />
<path value="$PROJECT_DIR$/vendor/martin-georgiev/postgresql-for-doctrine" />
<path value="$PROJECT_DIR$/vendor/dama/doctrine-test-bundle" />
<path value="$PROJECT_DIR$/vendor/staabm/side-effects-detector" />
<path value="$PROJECT_DIR$/vendor/psr/http-client" />
<path value="$PROJECT_DIR$/vendor/psr/container" /> <path value="$PROJECT_DIR$/vendor/psr/container" />
<path value="$PROJECT_DIR$/vendor/psr/simple-cache" />
<path value="$PROJECT_DIR$/vendor/psr/log" />
<path value="$PROJECT_DIR$/vendor/psr/cache" />
<path value="$PROJECT_DIR$/vendor/psr/http-message" /> <path value="$PROJECT_DIR$/vendor/psr/http-message" />
<path value="$PROJECT_DIR$/vendor/psr/event-dispatcher" /> <path value="$PROJECT_DIR$/vendor/psr/event-dispatcher" />
<path value="$PROJECT_DIR$/vendor/clue/ndjson-react" /> <path value="$PROJECT_DIR$/vendor/symfony/cache" />
<path value="$PROJECT_DIR$/vendor/psr/http-factory" /> <path value="$PROJECT_DIR$/vendor/symfony/validator" />
<path value="$PROJECT_DIR$/vendor/psr/clock" /> <path value="$PROJECT_DIR$/vendor/symfony/asset" />
<path value="$PROJECT_DIR$/vendor/psr/simple-cache" />
<path value="$PROJECT_DIR$/vendor/psr/cache" />
<path value="$PROJECT_DIR$/vendor/react/promise" />
<path value="$PROJECT_DIR$/vendor/react/stream" />
<path value="$PROJECT_DIR$/vendor/react/cache" />
<path value="$PROJECT_DIR$/vendor/react/event-loop" />
<path value="$PROJECT_DIR$/vendor/jean85/pretty-package-versions" />
<path value="$PROJECT_DIR$/vendor/react/child-process" />
<path value="$PROJECT_DIR$/vendor/react/socket" />
<path value="$PROJECT_DIR$/vendor/react/dns" />
<path value="$PROJECT_DIR$/vendor/twig/intl-extra" />
<path value="$PROJECT_DIR$/vendor/twig/extra-bundle" />
<path value="$PROJECT_DIR$/vendor/stof/doctrine-extensions-bundle" />
<path value="$PROJECT_DIR$/vendor/dama/doctrine-test-bundle" />
<path value="$PROJECT_DIR$/vendor/nikic/php-parser" />
<path value="$PROJECT_DIR$/vendor/gedmo/doctrine-extensions" />
<path value="$PROJECT_DIR$/vendor/fidry/cpu-core-counter" /> <path value="$PROJECT_DIR$/vendor/fidry/cpu-core-counter" />
<path value="$PROJECT_DIR$/vendor/twig/twig" /> <path value="$PROJECT_DIR$/vendor/symfony/ux-turbo" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan-phpunit" /> <path value="$PROJECT_DIR$/vendor/symfony/form" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan-symfony" /> <path value="$PROJECT_DIR$/vendor/vincentlanglet/twig-cs-fixer" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan-doctrine" /> <path value="$PROJECT_DIR$/vendor/symfony/dotenv" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan" /> <path value="$PROJECT_DIR$/vendor/symfony/serializer" />
<path value="$PROJECT_DIR$/vendor/phpunit/phpunit" /> <path value="$PROJECT_DIR$/vendor/rector/rector" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-timer" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpdoc-parser" /> <path value="$PROJECT_DIR$/vendor/phpstan/phpdoc-parser" />
<path value="$PROJECT_DIR$/vendor/phpstan/extension-installer" /> <path value="$PROJECT_DIR$/vendor/phpstan/extension-installer" />
<path value="$PROJECT_DIR$/vendor/staabm/side-effects-detector" /> <path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-icu" />
<path value="$PROJECT_DIR$/vendor/symfonycasts/sass-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/framework-bundle" />
<path value="$PROJECT_DIR$/vendor/symfonycasts/verify-email-bundle" />
<path value="$PROJECT_DIR$/vendor/psr/http-factory" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan-doctrine" />
<path value="$PROJECT_DIR$/vendor/psr/clock" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan" />
<path value="$PROJECT_DIR$/vendor/theseer/tokenizer" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan-phpunit" />
<path value="$PROJECT_DIR$/vendor/phpstan/phpstan-symfony" />
<path value="$PROJECT_DIR$/vendor/symfony/mailer" />
<path value="$PROJECT_DIR$/vendor/symfony/brevo-mailer" />
<path value="$PROJECT_DIR$/vendor/symfony/options-resolver" />
<path value="$PROJECT_DIR$/vendor/symfony/filesystem" />
<path value="$PROJECT_DIR$/vendor/symfony/runtime" />
<path value="$PROJECT_DIR$/vendor/symfony/event-dispatcher" />
<path value="$PROJECT_DIR$/vendor/symfony/var-dumper" />
<path value="$PROJECT_DIR$/vendor/symfony/phpunit-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/security-http" />
<path value="$PROJECT_DIR$/vendor/symfony/config" />
<path value="$PROJECT_DIR$/vendor/symfony/asset-mapper" />
<path value="$PROJECT_DIR$/vendor/symfony/uid" />
<path value="$PROJECT_DIR$/vendor/symfony/intl" />
<path value="$PROJECT_DIR$/vendor/symfony/security-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/translation-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/translation" />
<path value="$PROJECT_DIR$/vendor/symfony/http-client" />
<path value="$PROJECT_DIR$/vendor/friendsofphp/php-cs-fixer" />
<path value="$PROJECT_DIR$/vendor/gedmo/doctrine-extensions" />
<path value="$PROJECT_DIR$/vendor/symfony/finder" />
<path value="$PROJECT_DIR$/vendor/symfony/psr-http-message-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/stopwatch" />
<path value="$PROJECT_DIR$/vendor/masterminds/html5" />
<path value="$PROJECT_DIR$/vendor/symfony/event-dispatcher-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/twig-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/property-access" />
<path value="$PROJECT_DIR$/vendor/symfony/web-profiler-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-uuid" />
<path value="$PROJECT_DIR$/vendor/symfony/css-selector" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-normalizer" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-idn" />
<path value="$PROJECT_DIR$/vendor/symfony/security-csrf" />
<path value="$PROJECT_DIR$/vendor/symfony/twig-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/maker-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/var-exporter" />
<path value="$PROJECT_DIR$/vendor/symfony/doctrine-bridge" />
<path value="$PROJECT_DIR$/vendor/symfony/error-handler" />
<path value="$PROJECT_DIR$/vendor/symfony/flex" />
<path value="$PROJECT_DIR$/vendor/symfony/property-info" />
<path value="$PROJECT_DIR$/vendor/symfony/http-client-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/http-foundation" />
<path value="$PROJECT_DIR$/vendor/symfony/type-info" />
<path value="$PROJECT_DIR$/vendor/guzzlehttp/psr7" />
<path value="$PROJECT_DIR$/vendor/symfony/deprecation-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/security-core" />
<path value="$PROJECT_DIR$/vendor/symfony/service-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/routing" />
<path value="$PROJECT_DIR$/vendor/symfony/yaml" />
<path value="$PROJECT_DIR$/vendor/symfony/http-kernel" />
<path value="$PROJECT_DIR$/vendor/symfony/string" />
<path value="$PROJECT_DIR$/vendor/symfony/mime" />
<path value="$PROJECT_DIR$/vendor/markbaker/matrix" />
<path value="$PROJECT_DIR$/vendor/markbaker/complex" />
<path value="$PROJECT_DIR$/vendor/ralouphie/getallheaders" />
<path value="$PROJECT_DIR$/vendor/nikic/php-parser" />
<path value="$PROJECT_DIR$/vendor/symfony/process" />
<path value="$PROJECT_DIR$/vendor/symfony/password-hasher" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-intl-grapheme" />
<path value="$PROJECT_DIR$/vendor/symfony/console" />
<path value="$PROJECT_DIR$/vendor/symfony/dependency-injection" />
<path value="$PROJECT_DIR$/vendor/symfony/cache-contracts" />
<path value="$PROJECT_DIR$/vendor/symfony/clock" />
<path value="$PROJECT_DIR$/vendor/symfony/stimulus-bundle" />
<path value="$PROJECT_DIR$/vendor/phpdocumentor/type-resolver" />
<path value="$PROJECT_DIR$/vendor/phpdocumentor/reflection-common" />
<path value="$PROJECT_DIR$/vendor/phpdocumentor/reflection-docblock" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-code-coverage" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-text-template" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-file-iterator" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-invoker" />
<path value="$PROJECT_DIR$/vendor/phar-io/manifest" />
<path value="$PROJECT_DIR$/vendor/phpoffice/phpspreadsheet" />
<path value="$PROJECT_DIR$/vendor/phpunit/phpunit" />
<path value="$PROJECT_DIR$/vendor/phar-io/version" />
<path value="$PROJECT_DIR$/vendor/phpunit/php-timer" />
<path value="$PROJECT_DIR$/vendor/clue/ndjson-react" />
<path value="$PROJECT_DIR$/vendor/react/socket" />
<path value="$PROJECT_DIR$/vendor/react/dns" />
<path value="$PROJECT_DIR$/vendor/react/child-process" />
<path value="$PROJECT_DIR$/vendor/react/cache" />
<path value="$PROJECT_DIR$/vendor/react/event-loop" />
<path value="$PROJECT_DIR$/vendor/react/promise" />
<path value="$PROJECT_DIR$/vendor/react/stream" />
<path value="$PROJECT_DIR$/vendor/myclabs/deep-copy" />
<path value="$PROJECT_DIR$/vendor/runtime/frankenphp-symfony" />
<path value="$PROJECT_DIR$/vendor/twig/twig" />
<path value="$PROJECT_DIR$/vendor/twig/intl-extra" />
<path value="$PROJECT_DIR$/vendor/thecodingmachine/phpstan-safe-rule" />
<path value="$PROJECT_DIR$/vendor/thecodingmachine/safe" />
<path value="$PROJECT_DIR$/vendor/twig/extra-bundle" />
<path value="$PROJECT_DIR$/vendor/sebastian/global-state" />
<path value="$PROJECT_DIR$/vendor/sebastian/comparator" />
<path value="$PROJECT_DIR$/vendor/sebastian/object-reflector" />
<path value="$PROJECT_DIR$/vendor/sebastian/object-enumerator" />
<path value="$PROJECT_DIR$/vendor/maennchen/zipstream-php" />
<path value="$PROJECT_DIR$/vendor/sebastian/exporter" />
<path value="$PROJECT_DIR$/vendor/sebastian/version" />
<path value="$PROJECT_DIR$/vendor/sebastian/type" />
<path value="$PROJECT_DIR$/vendor/sebastian/environment" />
<path value="$PROJECT_DIR$/vendor/sebastian/cli-parser" />
<path value="$PROJECT_DIR$/vendor/sebastian/diff" />
<path value="$PROJECT_DIR$/vendor/sebastian/complexity" />
<path value="$PROJECT_DIR$/vendor/sebastian/lines-of-code" />
<path value="$PROJECT_DIR$/vendor/sebastian/recursion-context" />
<path value="$PROJECT_DIR$/vendor/sentry/sentry" /> <path value="$PROJECT_DIR$/vendor/sentry/sentry" />
<path value="$PROJECT_DIR$/vendor/sentry/sentry-symfony" /> <path value="$PROJECT_DIR$/vendor/sentry/sentry-symfony" />
<path value="$PROJECT_DIR$/vendor/rector/rector" />
<path value="$PROJECT_DIR$/vendor/phar-io/version" />
<path value="$PROJECT_DIR$/vendor/phar-io/manifest" />
<path value="$PROJECT_DIR$/vendor/myclabs/deep-copy" />
<path value="$PROJECT_DIR$/vendor/egulias/email-validator" /> <path value="$PROJECT_DIR$/vendor/egulias/email-validator" />
<path value="$PROJECT_DIR$/vendor/jean85/pretty-package-versions" />
<path value="$PROJECT_DIR$/vendor/doctrine/sql-formatter" />
<path value="$PROJECT_DIR$/vendor/doctrine/collections" />
<path value="$PROJECT_DIR$/vendor/doctrine/doctrine-bundle" />
<path value="$PROJECT_DIR$/vendor/doctrine/deprecations" />
<path value="$PROJECT_DIR$/vendor/doctrine/instantiator" />
<path value="$PROJECT_DIR$/vendor/doctrine/doctrine-fixtures-bundle" />
<path value="$PROJECT_DIR$/vendor/doctrine/event-manager" />
<path value="$PROJECT_DIR$/vendor/doctrine/orm" />
<path value="$PROJECT_DIR$/vendor/doctrine/dbal" />
<path value="$PROJECT_DIR$/vendor/doctrine/lexer" />
<path value="$PROJECT_DIR$/vendor/doctrine/persistence" />
<path value="$PROJECT_DIR$/vendor/doctrine/inflector" />
<path value="$PROJECT_DIR$/vendor/doctrine/doctrine-migrations-bundle" />
<path value="$PROJECT_DIR$/vendor/doctrine/data-fixtures" />
<path value="$PROJECT_DIR$/vendor/doctrine/migrations" />
<path value="$PROJECT_DIR$/vendor/evenement/evenement" />
<path value="$PROJECT_DIR$/vendor/stof/doctrine-extensions-bundle" />
<path value="$PROJECT_DIR$/vendor/symfony/polyfill-php84" />
<path value="$PROJECT_DIR$/vendor/symfony/dom-crawler" />
<path value="$PROJECT_DIR$/vendor/symfony/browser-kit" />
</include_path> </include_path>
</component> </component>
<component name="PhpInterpreters"> <component name="PhpInterpreters">
<interpreters> <interpreters>
<interpreter id="96512cb2-7b9e-4e1d-bfa2-bf7f3be424c8" name="Compose PHP" 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>
@@ -218,15 +217,15 @@
</component> </component>
<component name="PhpInterpretersPhpInfoCache"> <component name="PhpInterpretersPhpInfoCache">
<phpInfoCache> <phpInfoCache>
<interpreter name="Compose PHP"> <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.5.2"> <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-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-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.5.0"> <debugger_info debugger="xdebug" debugger_version="3.4.3">
<debug_extensions /> <debug_extensions />
</debugger_info> </debugger_info>
</debuggers> </debuggers>
@@ -251,7 +250,6 @@
<extension name="iconv" /> <extension name="iconv" />
<extension name="intl" /> <extension name="intl" />
<extension name="json" /> <extension name="json" />
<extension name="lexbor" />
<extension name="libxml" /> <extension name="libxml" />
<extension name="mbstring" /> <extension name="mbstring" />
<extension name="mysqlnd" /> <extension name="mysqlnd" />
@@ -267,7 +265,7 @@
<extension name="sqlite3" /> <extension name="sqlite3" />
<extension name="standard" /> <extension name="standard" />
<extension name="tokenizer" /> <extension name="tokenizer" />
<extension name="uri" /> <extension name="uuid" />
<extension name="xdebug" /> <extension name="xdebug" />
<extension name="xml" /> <extension name="xml" />
<extension name="xmlreader" /> <extension name="xmlreader" />
@@ -280,108 +278,6 @@
</phpInfoCache> </phpInfoCache>
</component> </component>
<component name="PhpProjectSharedConfiguration" php_language_level="8.5" /> <component name="PhpProjectSharedConfiguration" php_language_level="8.5" />
<component name="PhpRuntimeConfiguration">
<extensions>
<extension name="Ev" enabled="false" />
<extension name="SQLite" enabled="false" />
<extension name="SplType" enabled="false" />
<extension name="ZendDebugger" enabled="false" />
<extension name="ZendUtils" enabled="false" />
<extension name="amqp" enabled="false" />
<extension name="apache" enabled="false" />
<extension name="bcmath" enabled="false" />
<extension name="brotli" enabled="false" />
<extension name="bz2" enabled="false" />
<extension name="calendar" enabled="false" />
<extension name="cassandra" enabled="false" />
<extension name="couchbase" enabled="false" />
<extension name="cubrid" enabled="false" />
<extension name="dba" enabled="false" />
<extension name="decimal" enabled="false" />
<extension name="dio" enabled="false" />
<extension name="elastic_apm" enabled="false" />
<extension name="enchant" enabled="false" />
<extension name="exif" enabled="false" />
<extension name="fann" enabled="false" />
<extension name="ffmpeg" enabled="false" />
<extension name="frankenphp" enabled="false" />
<extension name="ftp" enabled="false" />
<extension name="gearman" enabled="false" />
<extension name="geoip" enabled="false" />
<extension name="geos" enabled="false" />
<extension name="gettext" enabled="false" />
<extension name="gmagick" enabled="false" />
<extension name="gmp" enabled="false" />
<extension name="gnupg" enabled="false" />
<extension name="grpc" enabled="false" />
<extension name="http" enabled="false" />
<extension name="ibm_db2" enabled="false" />
<extension name="igbinary" enabled="false" />
<extension name="imagick" enabled="false" />
<extension name="imap" enabled="false" />
<extension name="inotify" enabled="false" />
<extension name="interbase" enabled="false" />
<extension name="jsonpath" enabled="false" />
<extension name="judy" enabled="false" />
<extension name="ldap" enabled="false" />
<extension name="libevent" enabled="false" />
<extension name="libsodium" enabled="false" />
<extension name="mailparse" enabled="false" />
<extension name="mcrypt" enabled="false" />
<extension name="memcache" enabled="false" />
<extension name="memcached" enabled="false" />
<extension name="ming" enabled="false" />
<extension name="mongo" enabled="false" />
<extension name="mongodb" enabled="false" />
<extension name="mosquitto-php" enabled="false" />
<extension name="mqseries" enabled="false" />
<extension name="mssql" enabled="false" />
<extension name="mysql" enabled="false" />
<extension name="mysql_xdevapi" enabled="false" />
<extension name="mysqli" enabled="false" />
<extension name="ncurses" enabled="false" />
<extension name="newrelic" enabled="false" />
<extension name="oauth" enabled="false" />
<extension name="oci8" enabled="false" />
<extension name="odbc" enabled="false" />
<extension name="opentelemetry" enabled="false" />
<extension name="pcntl" enabled="false" />
<extension name="pdflib" enabled="false" />
<extension name="pgsql" enabled="false" />
<extension name="pspell" enabled="false" />
<extension name="pthreads" enabled="false" />
<extension name="rar" enabled="false" />
<extension name="recode" enabled="false" />
<extension name="redis" enabled="false" />
<extension name="relay" enabled="false" />
<extension name="rrd" enabled="false" />
<extension name="shmop" enabled="false" />
<extension name="snappy" enabled="false" />
<extension name="snmp" enabled="false" />
<extension name="soap" enabled="false" />
<extension name="sockets" enabled="false" />
<extension name="sqlsrv" enabled="false" />
<extension name="ssh2" enabled="false" />
<extension name="suhosin" enabled="false" />
<extension name="svn" enabled="false" />
<extension name="sybase" enabled="false" />
<extension name="sysvmsg" enabled="false" />
<extension name="sysvsem" enabled="false" />
<extension name="sysvshm" enabled="false" />
<extension name="tidy" enabled="false" />
<extension name="v8js" enabled="false" />
<extension name="wddx" enabled="false" />
<extension name="win32service" enabled="false" />
<extension name="wincache" enabled="false" />
<extension name="xhprof" enabled="false" />
<extension name="xlswriter" enabled="false" />
<extension name="xmlrpc" enabled="false" />
<extension name="xsl" enabled="false" />
<extension name="yaml" enabled="false" />
<extension name="zend" enabled="false" />
<extension name="zmq" enabled="false" />
</extensions>
</component>
<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" />
+6
View File
@@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="SqlDialectMappings">
<file url="PROJECT" dialect="PostgreSQL" />
</component>
</project>
-1
View File
@@ -1 +0,0 @@
AGENTS.md
+1 -9
View File
@@ -25,15 +25,7 @@ return new Config()
'no_unreachable_default_argument_value' => true, 'no_unreachable_default_argument_value' => true,
'no_useless_else' => true, 'no_useless_else' => true,
'no_useless_return' => true, 'no_useless_return' => true,
'phpdoc_line_span' => [ 'phpdoc_line_span' => ['const' => 'single', 'method' => 'single', 'property' => 'single'],
'case' => 'single',
'class' => 'single',
'const' => 'single',
'method' => 'single',
'other' => 'single',
'property' => 'single',
'trait_import' => 'single',
],
'phpdoc_order' => true, 'phpdoc_order' => true,
'single_line_empty_body' => true, 'single_line_empty_body' => true,
'strict_comparison' => true, 'strict_comparison' => true,
-82
View File
@@ -1,82 +0,0 @@
# Agent Guide: Tijd Voor De Test (Tvdt)
This document provides essential context and instructions for AI agents working on the **Tijd Voor De Test** project.
## Project Overview
A web application for managing "Wie is de Mol?" style tests, including seasons, quizzes, candidates, and eliminations.
- **Namespace**: `Tvdt`
- **PHP Version**: 8.5+
- **Framework**: Symfony 8.0
## Tech Stack
- **Server**: FrankenPHP (Caddy-based PHP server)
- **Database**: PostgreSQL
- **Frontend**: Symfony Asset Mapper (no Node.js/Webpack), Stimulus, Turbo
- **Styling**: Sass (via `symfonycasts/sass-bundle`)
- **Persistence**: Doctrine ORM 3.x
## Core Domain Entities
- **Season**: Groups quizzes and candidates for a specific period.
- **SeasonSettings**: Configuration for a season.
- **Quiz**: A test within a season containing multiple questions.
- **Question**: Questions belonging to a quiz.
- **Answer**: Possible answers for a question.
- **Candidate**: A participant in the season.
- **QuizCandidate**: Represents a candidate's attempt at a specific quiz (tracking start/end time).
- **GivenAnswer**: The specific answer a candidate selected during a quiz.
- **Elimination**: Records of red/green screens and forced results.
- **User**: Administrative accounts for managing the system.
## Development Workflow
The project uses `just` as the primary task runner. Always prefer `just` commands over manual docker calls.
### Common Commands
- `just up`: Start the environment.
- `just down`: Stop the environment.
- `just shell`: Enter the PHP container.
- `just migrate`: Run database migrations.
- `just fixtures`: Load development fixtures.
- `just fix-cs`: Run `php-cs-fixer` and `twig-cs-fixer`.
- `just phpstan`: Run static analysis.
- `just rector`: Run Rector for automated refactorings.
- `just reload-tests`: Reset the test database and load test fixtures.
## Coding Standards
- **PSR-12**: Follow standard PHP coding styles.
- **Strict Typing**: Use strict types in all PHP files.
- **Doctrine ORM 3**: Be aware of ORM 3 changes (e.g., lazy loading behavior, attribute-based mapping).
- **Symfony 8**: Use modern Symfony features (Attributes, Type-hinting).
- **Safe Functions**: Use `thecodingmachine/safe` for standard PHP functions that throw exceptions instead of returning false.
## Testing
- **Framework**: PHPUnit
- **Bundle**: `dama/doctrine-test-bundle` is used to wrap tests in transactions.
- **Location**: `tests/` directory mirroring `src/`.
- **Execution**: Run via `bin/phpunit` inside the container or `just reload-tests` to prepare the environment.
## Frontend Development
- JavaScript is managed via **Import Maps**.
- Stimulus controllers are located in `assets/controllers/`.
- CSS/Sass is in `assets/styles/`.
- Assets are compiled on-the-fly or mapped; do not look for a `node_modules` folder.
## Key Components
### Controllers
- **Backoffice**: Located in `src/Controller/Backoffice`, handles season and quiz management.
- **Quiz**: `src/Controller/QuizController` handles the candidate-facing side of quizzes.
- **Elimination**: `src/Controller/EliminationController` handles elimination screens.
### Services
- **QuizSpreadsheetService**: Handles importing quizzes from XLSX files.
### Base Classes & Enums
- **AbstractController**: Base class for all controllers, containing common regexes and flash helpers.
- **FlashType Enum**: Used for consistent flash messaging (`FlashType::Success`, `FlashType::Danger`, etc.).
## Key Files
- `composer.json`: Dependency management.
- `importmap.php`: JavaScript module mapping.
- `Justfile`: Automation shortcuts.
- `config/`: Application configuration.
- `templates/`: Twig templates.
+9 -7
View File
@@ -22,7 +22,7 @@ RUN apt-get update && apt-get install -y --no-install-recommends \
file \ file \
gettext \ gettext \
git \ git \
&& rm -rf /var/lib/apt/lists/* && rm -rf /var/lib/apt/lists/*
RUN set -eux; \ RUN set -eux; \
install-php-extensions \ install-php-extensions \
@@ -62,8 +62,8 @@ ENV APP_ENV=dev XDEBUG_MODE=off
# hadolint ignore=DL3008 # hadolint ignore=DL3008
RUN apt-get update && apt-get install -y --no-install-recommends \ RUN apt-get update && apt-get install -y --no-install-recommends \
bash-completion \ bash-completion \
&& rm -rf /var/lib/apt/lists/* && rm -rf /var/lib/apt/lists/*
COPY --link frankenphp/console-complete.bash /usr/share/bash-completion/completions/console COPY --link frankenphp/console-complete.bash /usr/share/bash-completion/completions/console
COPY --link frankenphp/composer-complete.bash /usr/share/bash-completion/completions/composer COPY --link frankenphp/composer-complete.bash /usr/share/bash-completion/completions/composer
@@ -72,7 +72,7 @@ RUN mv "$PHP_INI_DIR/php.ini-development" "$PHP_INI_DIR/php.ini"
RUN set -eux; \ RUN set -eux; \
install-php-extensions \ install-php-extensions \
xdebug/xdebug@3.5.0 \ xdebug/xdebug@3.5.0alpha3 \
; ;
COPY --link frankenphp/conf.d/20-app.dev.ini $PHP_INI_DIR/app.conf.d/ COPY --link frankenphp/conf.d/20-app.dev.ini $PHP_INI_DIR/app.conf.d/
@@ -87,10 +87,12 @@ FROM frankenphp_base AS frankenphp_prod
RUN rm -rf /var/lib/apt/lists/* RUN rm -rf /var/lib/apt/lists/*
ENV APP_ENV=prod ENV APP_ENV=prod
ENV FRANKENPHP_CONFIG="import worker.Caddyfile"
RUN mv "$PHP_INI_DIR/php.ini-production" "$PHP_INI_DIR/php.ini" RUN mv "$PHP_INI_DIR/php.ini-production" "$PHP_INI_DIR/php.ini"
COPY --link frankenphp/conf.d/20-app.prod.ini $PHP_INI_DIR/app.conf.d/ COPY --link frankenphp/conf.d/20-app.prod.ini $PHP_INI_DIR/app.conf.d/
COPY --link frankenphp/worker.Caddyfile /etc/caddy/worker.Caddyfile
# prevent the reinstallation of vendors at every changes in the source code # prevent the reinstallation of vendors at every changes in the source code
COPY --link composer.* symfony.* ./ COPY --link composer.* symfony.* ./
@@ -107,6 +109,6 @@ RUN set -eux; \
composer dump-env prod; \ composer dump-env prod; \
composer run-script --no-dev post-install-cmd; \ composer run-script --no-dev post-install-cmd; \
chmod +x bin/console; \ chmod +x bin/console; \
bin/console sass:build; \ bin/console sass:build; \
bin/console asset-map:compile --no-debug --quiet --no-ansi; \ bin/console asset-map:compile --no-debug --quiet --no-ansi; \
sync; sync;
+1 -10
View File
@@ -23,7 +23,7 @@ fixtures:
docker compose exec php bin/console doctrine:fixtures:load --purge-with-truncate --no-interaction --group=dev docker compose exec php bin/console doctrine:fixtures:load --purge-with-truncate --no-interaction --group=dev
translations: translations:
docker compose exec php bin/console translation:extract --force --format=xliff --sort=asc nl docker compose exec php bin/console translation:extract --force --format=xliff --sort=asc --clean nl
fix-cs: fix-cs:
docker compose exec php vendor/bin/php-cs-fixer fix docker compose exec php vendor/bin/php-cs-fixer fix
@@ -35,9 +35,6 @@ rector *args:
phpstan *args: phpstan *args:
docker compose exec php vendor/bin/phpstan analyse {{ args }} docker compose exec php vendor/bin/phpstan analyse {{ args }}
test *args:
docker compose exec php vendor/bin/phpunit {{ args }}
[confirm] [confirm]
clean: clean:
docker compose down -v --remove-orphans docker compose down -v --remove-orphans
@@ -48,9 +45,3 @@ reload-tests:
@docker compose exec php bin/console --env=test doctrine:database:create @docker compose exec php bin/console --env=test doctrine:database:create
@docker compose exec php bin/console --env=test doctrine:migrations:migrate -n @docker compose exec php bin/console --env=test doctrine:migrations:migrate -n
@docker compose exec php bin/console --env=test doctrine:fixtures:load -n --group=test @docker compose exec php bin/console --env=test doctrine:fixtures:load -n --group=test
trust-cert:
sudo security add-trusted-cer -d \
-r trustRoot \
-k "$HOME/Library/Keychains/login.keychain" \
./frankenphp/data/caddy/pki/authorities/local/root.crt
+4 -3
View File
@@ -1,4 +1,5 @@
import 'bootstrap/dist/css/bootstrap.min.css'; import * as bootstrap from 'bootstrap'
import './styles/backoffice.scss';
import './stimulus.js';
import './bootstrap.js'; import './bootstrap.js';
import 'bootstrap/dist/css/bootstrap.min.css'
import './styles/backoffice.scss';
+5 -1
View File
@@ -1 +1,5 @@
import * as bootstrap from 'bootstrap' import { startStimulusApp } from '@symfony/stimulus-bundle';
const app = startStimulusApp();
// register any custom, 3rd party controllers here
// app.register('some_controller_name', SomeImportedController);
+5 -16
View File
@@ -1,28 +1,17 @@
import {Controller} from '@hotwired/stimulus'; import {Controller} from '@hotwired/stimulus';
import {Tooltip, Modal} from 'bootstrap'; import * as bootstrap from 'bootstrap'
export default class extends Controller { export default class extends Controller {
static targets = ['clearModal', 'deleteModal'];
connect() { connect() {
this.tooltips = []; const tooltipTriggerList = document.querySelectorAll('[data-bs-toggle="tooltip"]')
const tooltipTriggerList = this.element.querySelectorAll('[data-bs-toggle="tooltip"]'); const tooltipList = [...tooltipTriggerList].map(tooltipTriggerEl => new bootstrap.Tooltip(tooltipTriggerEl))
[...tooltipTriggerList].forEach(tooltipTriggerEl => {
this.tooltips.push(Tooltip.getOrCreateInstance(tooltipTriggerEl));
});
}
disconnect() {
this.tooltips.forEach(tooltip => tooltip.dispose());
} }
clearQuiz() { clearQuiz() {
const modal = Modal.getOrCreateInstance(this.clearModalTarget); new bootstrap.Modal('#clearQuizModal').show();
modal.show();
} }
deleteQuiz() { deleteQuiz() {
const modal = Modal.getOrCreateInstance(this.deleteModalTarget); new bootstrap.Modal('#deleteQuizModal').show();
modal.show();
} }
} }
@@ -1,9 +1,7 @@
const nameCheck = /^[-_a-zA-Z0-9]{4,22}$/; const nameCheck = /^[-_a-zA-Z0-9]{4,22}$/;
const tokenCheck = /^[-_/+a-zA-Z0-9]{24,}$/; 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 // Generate and double-submit a CSRF token in a form field and a cookie, as defined by Symfony's SameOriginCsrfTokenManager
// Use `form.requestSubmit()` to ensure that the submit event is triggered. Using `form.submit()` will not trigger the event
// and thus this event-listener will not be executed.
document.addEventListener('submit', function (event) { document.addEventListener('submit', function (event) {
generateCsrfToken(event.target); generateCsrfToken(event.target);
}, true); }, true);
@@ -35,8 +33,8 @@ export function generateCsrfToken (formElement) {
if (!csrfCookie && nameCheck.test(csrfToken)) { if (!csrfCookie && nameCheck.test(csrfToken)) {
csrfField.setAttribute('data-csrf-protection-cookie-value', csrfCookie = 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.defaultValue = csrfToken = btoa(String.fromCharCode.apply(null, (window.crypto || window.msCrypto).getRandomValues(new Uint8Array(18))));
csrfField.dispatchEvent(new Event('change', { bubbles: true }));
} }
csrfField.dispatchEvent(new Event('change', { bubbles: true }));
if (csrfCookie && tokenCheck.test(csrfToken)) { if (csrfCookie && tokenCheck.test(csrfToken)) {
const cookie = csrfCookie + '_' + csrfToken + '=' + csrfCookie + '; path=/; samesite=strict'; const cookie = csrfCookie + '_' + csrfToken + '=' + csrfCookie + '; path=/; samesite=strict';
+4 -8
View File
@@ -2,13 +2,9 @@ import {Controller} from '@hotwired/stimulus';
export default class extends Controller { export default class extends Controller {
next() { next() {
const currentUrl = new URL(window.location.href); const currentUrl = window.location.href;
const pathParts = currentUrl.pathname.split('/'); const urlParts = currentUrl.split('/');
// Remove the last segment urlParts.pop();
pathParts.pop(); window.location.href = urlParts.join('/');
// Update the pathname
currentUrl.pathname = pathParts.join('/');
// Navigate
window.location.href = currentUrl.href;
} }
} }
+5 -3
View File
@@ -1,4 +1,6 @@
import 'bootstrap/dist/css/bootstrap.min.css';
import './styles/quiz.scss';
import './stimulus.js';
import './bootstrap.js'; import './bootstrap.js';
import 'bootstrap/dist/css/bootstrap.min.css'
import * as bootstrap from 'bootstrap'
import './styles/app.scss'
-3
View File
@@ -1,3 +0,0 @@
import { startStimulusApp } from '@symfony/stimulus-bundle';
const app = startStimulusApp();
-1
View File
@@ -1,4 +1,3 @@
# yaml-language-server: $schema=https://raw.githubusercontent.com/compose-spec/compose-go/master/schema/compose-spec.json
services: services:
php: php:
build: build:
-1
View File
@@ -1,4 +1,3 @@
# yaml-language-server: $schema=https://raw.githubusercontent.com/compose-spec/compose-go/master/schema/compose-spec.json
# Development environment override # Development environment override
services: services:
php: php:
-3
View File
@@ -1,4 +1,3 @@
# yaml-language-server: $schema=https://raw.githubusercontent.com/compose-spec/compose-go/master/schema/compose-spec.json
# Production environment override # Production environment override
services: services:
php: php:
@@ -10,8 +9,6 @@ services:
MAILER_DSN: ${MAILER_DSN} MAILER_DSN: ${MAILER_DSN}
MAILER_SENDER: ${MAILER_SENDER} MAILER_SENDER: ${MAILER_SENDER}
SENTRY_DSN: ${SENTRY_DSN} SENTRY_DSN: ${SENTRY_DSN}
SENTRY_RELEASE: ${IMAGE_TAG}
SENTRY_ENVIRONMENT: ${SENTRY_ENVIRONMENT}
labels: labels:
- "traefik.enable=true" - "traefik.enable=true"
- "traefik.http.routers.tvdt.rule=Host(`tijdvoordetest.nl`)" - "traefik.http.routers.tvdt.rule=Host(`tijdvoordetest.nl`)"
-1
View File
@@ -1,4 +1,3 @@
# yaml-language-server: $schema=https://raw.githubusercontent.com/compose-spec/compose-go/master/schema/compose-spec.json
services: services:
php: php:
image: ${IMAGES_PREFIX:-}app-php image: ${IMAGES_PREFIX:-}app-php
+55 -53
View File
@@ -9,63 +9,65 @@
"php": ">=8.5", "php": ">=8.5",
"ext-ctype": "*", "ext-ctype": "*",
"ext-iconv": "*", "ext-iconv": "*",
"doctrine/dbal": "^4.4.3", "doctrine/dbal": "^4.3.4",
"doctrine/doctrine-bundle": "^3.2.2", "doctrine/doctrine-bundle": "^3.0",
"doctrine/doctrine-migrations-bundle": "^4.0", "doctrine/doctrine-migrations-bundle": "^3.7.0",
"doctrine/orm": "^3.6.2", "doctrine/orm": "^3.5.7",
"martin-georgiev/postgresql-for-doctrine": "^4.4", "martin-georgiev/postgresql-for-doctrine": "^3.6.2",
"phpdocumentor/reflection-docblock": "^6.0.3", "phpdocumentor/reflection-docblock": "^5.6.4",
"phpoffice/phpspreadsheet": "^5.5", "phpoffice/phpspreadsheet": "^5.3",
"phpstan/phpdoc-parser": "^2.3.2", "phpstan/phpdoc-parser": "^2.3",
"sentry/sentry-symfony": "^5.9.0", "runtime/frankenphp-symfony": "^0.2.0",
"stof/doctrine-extensions-bundle": "^1.15.3", "sentry/sentry-symfony": "^5.6",
"symfony/asset": "8.0.*", "stof/doctrine-extensions-bundle": "^1.14",
"symfony/asset-mapper": "8.0.*", "symfony/asset": "7.4.*",
"symfony/brevo-mailer": "8.0.*", "symfony/asset-mapper": "7.4.*",
"symfony/console": "8.0.*", "symfony/brevo-mailer": "7.4.*",
"symfony/dotenv": "8.0.*", "symfony/console": "7.4.*",
"symfony/dotenv": "7.4.*",
"symfony/flex": "^2.10.0", "symfony/flex": "^2.10.0",
"symfony/form": "8.0.*", "symfony/form": "7.4.*",
"symfony/framework-bundle": "8.0.*", "symfony/framework-bundle": "7.4.*",
"symfony/mailer": "8.0.*", "symfony/mailer": "7.4.*",
"symfony/property-access": "8.0.*", "symfony/property-access": "7.4.*",
"symfony/property-info": "8.0.*", "symfony/property-info": "7.4.*",
"symfony/runtime": "8.0.*", "symfony/runtime": "7.4.*",
"symfony/security-bundle": "8.0.*", "symfony/security-bundle": "7.4.*",
"symfony/security-csrf": "8.0.*", "symfony/security-csrf": "7.4.*",
"symfony/serializer": "8.0.*", "symfony/serializer": "7.4.*",
"symfony/translation": "8.0.*", "symfony/translation": "7.4.*",
"symfony/twig-bundle": "8.0.*", "symfony/twig-bundle": "7.4.*",
"symfony/uid": "8.0.*", "symfony/uid": "7.4.*",
"symfony/ux-turbo": "^2.33.0", "symfony/ux-turbo": "^2.31.0",
"symfony/validator": "8.0.*", "symfony/validator": "7.4.*",
"symfony/yaml": "8.0.*", "symfony/yaml": "7.4.*",
"symfonycasts/sass-bundle": "^0.9", "symfonycasts/sass-bundle": "^0.8.3",
"symfonycasts/verify-email-bundle": "^1.18.0", "symfonycasts/verify-email-bundle": "^1.17.4",
"thecodingmachine/safe": "^3.4.0", "thecodingmachine/safe": "^3.3.0",
"twig/extra-bundle": "^3.24.0", "twig/extra-bundle": "^3.22.1",
"twig/intl-extra": "^3.24.0", "twig/intl-extra": "^3.22.1",
"twig/twig": "^3.24.0" "twig/twig": "^3.22.0"
}, },
"require-dev": { "require-dev": {
"dama/doctrine-test-bundle": "^8.6", "dama/doctrine-test-bundle": "^8.4",
"doctrine/doctrine-fixtures-bundle": "^4.3.1", "doctrine/doctrine-fixtures-bundle": "^4.3",
"friendsofphp/php-cs-fixer": "^3.94.2", "friendsofphp/php-cs-fixer": "^3.90.0",
"phpstan/extension-installer": "^1.4.3", "phpstan/extension-installer": "^1.4.3",
"phpstan/phpstan": "^2.1.42", "phpstan/phpstan": "^2.1.32",
"phpstan/phpstan-doctrine": "^2.0.20", "phpstan/phpstan-doctrine": "^2.0.11",
"phpstan/phpstan-phpunit": "^2.0.16", "phpstan/phpstan-phpunit": "^2.0.8",
"phpstan/phpstan-symfony": "^2.0.15", "phpstan/phpstan-symfony": "^2.0.8",
"phpunit/phpunit": "^13.0.5", "phpunit/phpunit": "^12.4.4",
"rector/rector": "^2.3.9", "rector/rector": "^2.2.9",
"symfony/browser-kit": "8.0.*", "roave/security-advisories": "dev-latest",
"symfony/css-selector": "8.0.*", "symfony/browser-kit": "7.4.*",
"symfony/maker-bundle": "^1.67.0", "symfony/css-selector": "7.4.*",
"symfony/phpunit-bridge": "8.0.*", "symfony/maker-bundle": "^1.65.0",
"symfony/stopwatch": "8.0.*", "symfony/phpunit-bridge": "7.4.*",
"symfony/web-profiler-bundle": "8.0.*", "symfony/stopwatch": "7.4.*",
"symfony/web-profiler-bundle": "7.4.*",
"thecodingmachine/phpstan-safe-rule": "^1.4.3", "thecodingmachine/phpstan-safe-rule": "^1.4.3",
"vincentlanglet/twig-cs-fixer": "^3.14.0" "vincentlanglet/twig-cs-fixer": "^3.11.0"
}, },
"config": { "config": {
"allow-plugins": { "allow-plugins": {
@@ -120,7 +122,7 @@
"extra": { "extra": {
"symfony": { "symfony": {
"allow-contrib": false, "allow-contrib": false,
"require": "8.0.*", "require": "7.4.*",
"docker": true "docker": true
} }
} }
Generated
+2826 -1717
View File
File diff suppressed because it is too large Load Diff
+3
View File
@@ -17,6 +17,7 @@ doctrine:
orm: orm:
enable_native_lazy_objects: true
validate_xml_mapping: true validate_xml_mapping: true
naming_strategy: doctrine.orm.naming_strategy.underscore_number_aware naming_strategy: doctrine.orm.naming_strategy.underscore_number_aware
identity_generation_preferences: identity_generation_preferences:
@@ -29,6 +30,8 @@ doctrine:
dir: '%kernel.project_dir%/src/Entity' dir: '%kernel.project_dir%/src/Entity'
prefix: 'Tvdt\Entity' prefix: 'Tvdt\Entity'
alias: Tvdt alias: Tvdt
controller_resolver:
auto_mapping: false
when@test: when@test:
doctrine: doctrine:
-6
View File
@@ -1,6 +0,0 @@
# yaml-language-server: $schema=../../vendor/symfony/dependency-injection/Loader/schema/services.schema.json
when@dev:
maker:
root_namespace: 'Tvdt'
generate_final_classes: true
generate_final_entities: false
+3 -3
View File
@@ -1,8 +1,8 @@
framework: framework:
router: router:
# Configure how to generate URLs in non-HTTP contexts, such as CLI commands. # Configure how to generate URLs in non-HTTP contexts, such as CLI commands.
# See https://symfony.com/doc/current/routing.html#generating-urls-in-commands # See https://symfony.com/doc/current/routing.html#generating-urls-in-commands
default_uri: '%env(DEFAULT_URI)%' #default_uri: http://localhost
when@prod: when@prod:
framework: framework:
+17 -5
View File
@@ -2,17 +2,17 @@ security:
# https://symfony.com/doc/current/security.html#registering-the-user-hashing-passwords # https://symfony.com/doc/current/security.html#registering-the-user-hashing-passwords
password_hashers: password_hashers:
Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface: 'auto' Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface: 'auto'
# 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)
tvdt_user_provider: tvdt_user_provider:
entity: entity:
class: Tvdt\Entity\User class: Tvdt\Entity\User
property: email property: email
# used to reload user from session & other features (e.g. switch_user)
firewalls: firewalls:
dev: dev:
# Ensure dev tools and static assets are always allowed pattern: ^/(_(profiler|wdt)|css|images|js)/
pattern: ^/(_profiler|_wdt|assets|build)/
security: false security: false
main: main:
lazy: true lazy: true
@@ -24,7 +24,17 @@ security:
default_target_path: tvdt_backoffice_index default_target_path: tvdt_backoffice_index
logout: logout:
path: tvdt_login_logout path: tvdt_login_logout
# where to redirect after logout
# target: tvdt_any_route
# activate different ways to authenticate
# https://symfony.com/doc/current/security.html#the-firewall
# https://symfony.com/doc/current/security/impersonating_user.html
# switch_user: true
# Easy way to control access for large sections of your site
# 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: ^/backoffice, roles: ROLE_USER } - { path: ^/backoffice, roles: ROLE_USER }
@@ -32,8 +42,10 @@ security:
when@test: when@test:
security: security:
password_hashers: password_hashers:
# Password hashers are resource-intensive by design to ensure security. # By default, password hashers are resource intensive and take time. This is
# In tests, it's safe to reduce their cost to improve performance. # important to generate secure password hashes. In tests however, secure hashes
# are not important, waste resources and increase test times. The following
# reduces the work factor to the lowest possible values.
Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface: Symfony\Component\Security\Core\User\PasswordAuthenticatedUserInterface:
algorithm: auto algorithm: auto
cost: 4 # Lowest possible value for bcrypt cost: 4 # Lowest possible value for bcrypt
+6 -10
View File
@@ -8,10 +8,9 @@ when@prod:
ignore_exceptions: ignore_exceptions:
- 'Symfony\Component\ErrorHandler\Error\FatalError' - 'Symfony\Component\ErrorHandler\Error\FatalError'
- 'Symfony\Component\Debug\Exception\FatalErrorException' - 'Symfony\Component\Debug\Exception\FatalErrorException'
- 'Symfony\Component\HttpKernel\Exception\NotFoundHttpException'
# #
# # If you are using Monolog, you also need this additional configuration to log the errors correctly: # # If you are using Monolog, you also need this additional configuration to log the errors correctly:
# # https://docs.sentry.io/platforms/php/guides/symfony/integrations/monolog/ # # https://docs.sentry.io/platforms/php/guides/symfony/#monolog-integration
# register_error_listener: false # register_error_listener: false
# register_error_handler: false # register_error_handler: false
# #
@@ -20,21 +19,18 @@ when@prod:
# # Use this only if you don't want to use structured logging and instead receive # # Use this only if you don't want to use structured logging and instead receive
# # certain log levels as errors. # # certain log levels as errors.
# sentry: # sentry:
# type: service # type: sentry
# id: Sentry\Monolog\Handler # level: !php/const Monolog\Logger::ERROR
# hub_id: Sentry\State\HubInterface
# fill_extra_context: true # Enables sending monolog context to Sentry
# process_psr_3_messages: false # Disables the resolution of PSR-3 placeholders
# #
# # Use this for structured log integration # # Use this for structured log integration
# sentry_logs: # sentry_logs:
# type: service # type: service
# id: Sentry\SentryBundle\Monolog\LogsHandler # id: Sentry\SentryBundle\Monolog\LogsHandler
# #
# # Enable one of the two services below, depending on your choice above
# services: # services:
# Sentry\Monolog\Handler:
# arguments:
# $hub: '@Sentry\State\HubInterface'
# $level: !php/const Monolog\Logger::ERROR
# $fillExtraContext: true # Enables sending monolog context to Sentry
# Sentry\SentryBundle\Monolog\LogsHandler: # Sentry\SentryBundle\Monolog\LogsHandler:
# arguments: # arguments:
# - !php/const Monolog\Logger::INFO # - !php/const Monolog\Logger::INFO
-4
View File
@@ -1,4 +0,0 @@
symfonycasts_sass:
root_sass:
- './assets/styles/quiz.scss'
- './assets/styles/backoffice.scss'
+969 -968
View File
File diff suppressed because it is too large Load Diff
+4 -10
View File
@@ -1,11 +1,5 @@
# yaml-language-server: $schema=../vendor/symfony/routing/Loader/schema/routing.schema.json
# This file is the entry point to configure the routes of your app.
# Methods with the #[Route] attribute are automatically imported.
# See also https://symfony.com/doc/current/routing.html
# To list all registered routes, run the following command:
# bin/console debug:router
controllers: controllers:
resource: routing.controllers resource:
path: ../src/Controller/
namespace: Tvdt\Controller
type: attribute
+10 -3
View File
@@ -1,8 +1,5 @@
# yaml-language-server: $schema=../vendor/symfony/dependency-injection/Loader/schema/services.schema.json
# This file is the entry point to configure your own services. # This file is the entry point to configure your own services.
# Files in the packages/ subdirectory configure your dependencies. # Files in the packages/ subdirectory configure your dependencies.
# See also https://symfony.com/doc/current/service_container/import.html
# Put parameters here that don't need to change on each machine where the app is deployed # Put parameters here that don't need to change on each machine where the app is deployed
# https://symfony.com/doc/current/best_practices.html#use-parameters-for-application-configuration # https://symfony.com/doc/current/best_practices.html#use-parameters-for-application-configuration
@@ -25,3 +22,13 @@ services:
# add more service definitions when explicit configuration is needed # add more service definitions when explicit configuration is needed
# please note that last definitions always *replace* previous ones # please note that last definitions always *replace* previous ones
when@prod:
services:
Tvdt\:
resource: '../src/'
exclude:
- '../src/DependencyInjection/'
- '../src/Entity/'
- '../src/Kernel.php'
- '../src/DataFixtures'
+14 -14
View File
@@ -22,20 +22,20 @@
root /app/public root /app/public
encode zstd br gzip encode zstd br gzip
# mercure { mercure {
# # Transport to use (default to Bolt) # Transport to use (default to Bolt)
## transport_url {$MERCURE_TRANSPORT_URL:bolt:///data/mercure.db} transport_url {$MERCURE_TRANSPORT_URL:bolt:///data/mercure.db}
# Publisher JWT key # Publisher JWT key
# publisher_jwt {env.MERCURE_PUBLISHER_JWT_KEY} {env.MERCURE_PUBLISHER_JWT_ALG} publisher_jwt {env.MERCURE_PUBLISHER_JWT_KEY} {env.MERCURE_PUBLISHER_JWT_ALG}
# # Subscriber JWT key # Subscriber JWT key
# subscriber_jwt {env.MERCURE_SUBSCRIBER_JWT_KEY} {env.MERCURE_SUBSCRIBER_JWT_ALG} subscriber_jwt {env.MERCURE_SUBSCRIBER_JWT_KEY} {env.MERCURE_SUBSCRIBER_JWT_ALG}
# # Allow anonymous subscribers (double-check that it's what you want) # Allow anonymous subscribers (double-check that it's what you want)
# anonymous anonymous
# # Enable the subscription API (double-check that it's what you want) # Enable the subscription API (double-check that it's what you want)
# subscriptions subscriptions
# # Extra directives # Extra directives
# {$MERCURE_EXTRA_DIRECTIVES} {$MERCURE_EXTRA_DIRECTIVES}
# } }
vulcain vulcain
+4
View File
@@ -0,0 +1,4 @@
worker {
file ./public/index.php
env APP_RUNTIME Runtime\FrankenPhpSymfony\Runtime
}
+6 -6
View File
@@ -22,23 +22,23 @@ return [
'path' => './assets/backoffice.js', 'path' => './assets/backoffice.js',
'entrypoint' => true, 'entrypoint' => true,
], ],
'@symfony/stimulus-bundle' => [
'path' => './vendor/symfony/stimulus-bundle/assets/dist/loader.js',
],
'bootstrap' => [ 'bootstrap' => [
'version' => '5.3.8', 'version' => '5.3.6',
], ],
'@popperjs/core' => [ '@popperjs/core' => [
'version' => '2.11.8', 'version' => '2.11.8',
], ],
'bootstrap/dist/css/bootstrap.min.css' => [ 'bootstrap/dist/css/bootstrap.min.css' => [
'version' => '5.3.8', 'version' => '5.3.6',
'type' => 'css', 'type' => 'css',
], ],
'@hotwired/stimulus' => [ '@hotwired/stimulus' => [
'version' => '3.2.2', 'version' => '3.2.2',
], ],
'@symfony/stimulus-bundle' => [
'path' => './vendor/symfony/stimulus-bundle/assets/dist/loader.js',
],
'@hotwired/turbo' => [ '@hotwired/turbo' => [
'version' => '8.0.23', 'version' => '7.3.0',
], ],
]; ];
-29
View File
@@ -1,29 +0,0 @@
<?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 Version20260125191247 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('ALTER TABLE quiz_candidate ADD penalty_seconds SMALLINT DEFAULT 0 NOT NULL');
}
public function down(Schema $schema): void
{
// this down() migration is auto-generated, please modify it to your needs
$this->addSql('ALTER TABLE quiz_candidate DROP penalty_seconds');
}
}
-29
View File
@@ -1,29 +0,0 @@
<?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 Version20260309215703 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('ALTER TABLE quiz_candidate ADD active BOOLEAN DEFAULT true NOT NULL');
}
public function down(Schema $schema): void
{
// this down() migration is auto-generated, please modify it to your needs
$this->addSql('ALTER TABLE quiz_candidate DROP active');
}
}
-32
View File
@@ -1,32 +0,0 @@
<?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 Version20260309220448 extends AbstractMigration
{
public function getDescription(): string
{
return 'Add started field to quiz_candidate and copy existing created values';
}
public function up(Schema $schema): void
{
// this up() migration is auto-generated, please modify it to your needs
$this->addSql('ALTER TABLE quiz_candidate ADD started TIMESTAMP(0) WITH TIME ZONE DEFAULT NULL');
// Copy created to started for existing rows (these were created when quiz started)
$this->addSql('UPDATE quiz_candidate SET started = created WHERE started IS NULL');
}
public function down(Schema $schema): void
{
// this down() migration is auto-generated, please modify it to your needs
$this->addSql('ALTER TABLE quiz_candidate DROP started');
}
}
-2
View File
@@ -7,8 +7,6 @@ parameters:
- public/ - public/
- src/ - src/
- tests/ - tests/
excludePaths:
- config/reference.php
treatPhpDocTypesAsCertain: false treatPhpDocTypesAsCertain: false
symfony: symfony:
containerXmlPath: var/cache/dev/Tvdt_KernelDevDebugContainer.xml containerXmlPath: var/cache/dev/Tvdt_KernelDevDebugContainer.xml
@@ -27,7 +27,6 @@ final class BackofficeController extends AbstractController
private readonly SeasonRepository $seasonRepository, private readonly SeasonRepository $seasonRepository,
private readonly Security $security, private readonly Security $security,
private readonly QuizSpreadsheetService $excel, private readonly QuizSpreadsheetService $excel,
private readonly EntityManagerInterface $em,
) {} ) {}
#[Route('/backoffice/', name: 'tvdt_backoffice_index')] #[Route('/backoffice/', name: 'tvdt_backoffice_index')]
@@ -46,7 +45,7 @@ final class BackofficeController extends AbstractController
} }
#[Route('/backoffice/season/add', name: 'tvdt_backoffice_season_add', priority: 10)] #[Route('/backoffice/season/add', name: 'tvdt_backoffice_season_add', priority: 10)]
public function addSeason(Request $request): Response public function addSeason(Request $request, EntityManagerInterface $em): Response
{ {
$season = new Season(); $season = new Season();
$form = $this->createForm(CreateSeasonFormType::class, $season); $form = $this->createForm(CreateSeasonFormType::class, $season);
@@ -60,8 +59,8 @@ final class BackofficeController extends AbstractController
$season->addOwner($user); $season->addOwner($user);
$season->generateSeasonCode(); $season->generateSeasonCode();
$this->em->persist($season); $em->persist($season);
$this->em->flush(); $em->flush();
return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->seasonCode]); return $this->redirectToRoute('tvdt_backoffice_season', ['seasonCode' => $season->seasonCode]);
} }
@@ -18,7 +18,7 @@ use Tvdt\Factory\EliminationFactory;
final class PrepareEliminationController extends AbstractController final class PrepareEliminationController extends AbstractController
{ {
public function __construct(private readonly EliminationFactory $eliminationFactory, private readonly EntityManagerInterface $em) {} public function __construct(private readonly EliminationFactory $eliminationFactory) {}
#[Route( #[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/elimination/prepare', '/backoffice/season/{seasonCode:season}/quiz/{quiz}/elimination/prepare',
@@ -37,11 +37,11 @@ final class PrepareEliminationController extends AbstractController
name: 'tvdt_prepare_elimination_view', name: 'tvdt_prepare_elimination_view',
requirements: ['elimination' => Requirement::UUID], requirements: ['elimination' => Requirement::UUID],
)] )]
public function viewElimination(Elimination $elimination, Request $request): 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);
$this->em->flush(); $em->flush();
if ($request->request->getBoolean('start')) { if ($request->request->getBoolean('start')) {
return $this->redirectToRoute('tvdt_elimination', ['elimination' => $elimination->id]); return $this->redirectToRoute('tvdt_elimination', ['elimination' => $elimination->id]);
+3 -245
View File
@@ -9,18 +9,14 @@ use Symfony\Component\HttpFoundation\RedirectResponse;
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\HttpKernel\Exception\MethodNotAllowedHttpException; 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\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\Controller\AbstractController; use Tvdt\Controller\AbstractController;
use Tvdt\Entity\Answer;
use Tvdt\Entity\Candidate; use Tvdt\Entity\Candidate;
use Tvdt\Entity\Question;
use Tvdt\Entity\Quiz; use Tvdt\Entity\Quiz;
use Tvdt\Entity\QuizCandidate;
use Tvdt\Entity\Season; use Tvdt\Entity\Season;
use Tvdt\Exception\ErrorClearingQuizException; use Tvdt\Exception\ErrorClearingQuizException;
use Tvdt\Repository\QuizCandidateRepository; use Tvdt\Repository\QuizCandidateRepository;
@@ -35,7 +31,6 @@ class QuizController extends AbstractController
private readonly QuizRepository $quizRepository, private readonly QuizRepository $quizRepository,
private readonly TranslatorInterface $translator, private readonly TranslatorInterface $translator,
private readonly QuizCandidateRepository $quizCandidateRepository, private readonly QuizCandidateRepository $quizCandidateRepository,
private readonly EntityManagerInterface $em,
) {} ) {}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')] #[IsGranted(SeasonVoter::EDIT, subject: 'season')]
@@ -46,198 +41,10 @@ class QuizController extends AbstractController
)] )]
public function index(Season $season, Quiz $quiz): Response public function index(Season $season, Quiz $quiz): Response
{ {
return $this->redirectToRoute('tvdt_backoffice_quiz_overview', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
#[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/overview',
name: 'tvdt_backoffice_quiz_overview',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
)]
public function overview(Season $season, Quiz $quiz): Response
{
$fetchedQuiz = $this->quizRepository->fetchWithQuestionsAndCandidates($quiz->id);
// Create indexed lookup for quiz candidates by candidate ID
$quizCandidatesByCandidateId = [];
foreach ($fetchedQuiz->candidateData as $qc) {
$quizCandidatesByCandidateId[$qc->candidate->id->toString()] = $qc;
}
// Get given answers counts efficiently via database query
$givenAnswersCountByCandidateId = $this->quizRepository->getGivenAnswersCountPerCandidate($quiz);
// Pre-compute candidate data to avoid nested loops in template
$candidateData = [];
foreach ($season->candidates as $candidate) {
$candidateIdString = $candidate->id->toString();
$candidateData[] = [
'candidate' => $candidate,
'quizCandidate' => $quizCandidatesByCandidateId[$candidateIdString] ?? null,
'givenAnswersCount' => $givenAnswersCountByCandidateId[$candidateIdString] ?? 0,
];
}
return $this->render('backoffice/quiz.html.twig', [ return $this->render('backoffice/quiz.html.twig', [
'season' => $season, 'season' => $season,
'quiz' => $fetchedQuiz, 'quiz' => $quiz,
'questionErrors' => $fetchedQuiz->getQuestionErrors(),
'candidateData' => $candidateData,
'activeTab' => 'overview',
'template' => 'backoffice/quiz/tab_overview.html.twig',
]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
#[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/result',
name: 'tvdt_backoffice_quiz_result',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
)]
public function result(Season $season, Quiz $quiz): Response
{
$fetchedQuiz = $this->quizRepository->fetchWithQuestions($quiz->id);
return $this->render('backoffice/quiz.html.twig', [
'season' => $season,
'quiz' => $fetchedQuiz,
'result' => $this->quizRepository->getScores($quiz), 'result' => $this->quizRepository->getScores($quiz),
'activeTab' => 'result',
'template' => 'backoffice/quiz/tab_result.html.twig',
]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
#[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/candidates-list',
name: 'tvdt_backoffice_quiz_candidates_tab',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
)]
public function candidatesTab(Season $season, Quiz $quiz): Response
{
// Create indexed lookup for quiz candidates by candidate ID
$quizCandidatesByCandidateId = [];
foreach ($quiz->candidateData as $qc) {
$quizCandidatesByCandidateId[$qc->candidate->id->toString()] = $qc;
}
// Get given answers counts efficiently via database query
$givenAnswersCountByCandidateId = $this->quizRepository->getGivenAnswersCountPerCandidate($quiz);
// Pre-compute candidate data to avoid nested loops in template
$candidateData = [];
foreach ($season->candidates as $candidate) {
$candidateIdString = $candidate->id->toString();
$candidateData[] = [
'candidate' => $candidate,
'quizCandidate' => $quizCandidatesByCandidateId[$candidateIdString] ?? null,
'givenAnswersCount' => $givenAnswersCountByCandidateId[$candidateIdString] ?? 0,
];
}
return $this->render('backoffice/quiz.html.twig', [
'season' => $season,
'quiz' => $quiz,
'candidateData' => $candidateData,
'activeTab' => 'candidates',
'template' => 'backoffice/quiz/tab_candidates_list.html.twig',
]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
#[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/answer-mapping',
name: 'tvdt_backoffice_quiz_candidates',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
)]
public function answerMapping(Season $season, Quiz $quiz): Response
{
$fetchedQuiz = $this->quizRepository->fetchWithQuestions($quiz->id);
\assert($fetchedQuiz->questions->count() > 0);
$firstQuestion = $fetchedQuiz->questions->first();
\assert($firstQuestion instanceof Question);
return $this->redirectToRoute('tvdt_backoffice_quiz_candidates_question', [
'seasonCode' => $season->seasonCode,
'quiz' => $quiz->id,
'question' => $firstQuestion->id,
]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
#[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/candidates/{question}',
name: 'tvdt_backoffice_quiz_candidates_question',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
methods: ['GET'],
)]
public function candidates_question(Season $season, Quiz $quiz, Question $question): Response
{
return $this->render('backoffice/quiz.html.twig', [
'season' => $season,
'quiz' => $quiz,
'question' => $question,
'candidates' => $season->candidates,
'activeTab' => 'answers',
'template' => 'backoffice/quiz/tab_candidates.html.twig',
]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'season')]
#[Route(
'/backoffice/season/{seasonCode:season}/quiz/{quiz}/candidates/{question}',
name: 'tvdt_backoffice_quiz_candidates_question_save',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID],
methods: ['POST'],
)]
public function saveCandidateAnswers(Season $season, Quiz $quiz, Question $question, Request $request): RedirectResponse
{
if (false === $season->quizzes->contains($quiz)
|| false === $quiz->questions->contains($question)) {
throw new BadRequestHttpException('Invalid quiz or question');
}
$candidateAnswers = $request->request->all('candidate_answer');
// Clear existing candidate-answer associations for this question
foreach ($question->answers as $answer) {
if (false === $quiz->questions->contains($answer->question)) {
throw new BadRequestHttpException('Invalid question');
}
$answer->candidates->clear();
}
// Add new associations
foreach ($candidateAnswers as $candidateId => $answerIds) {
$candidate = $this->em->getRepository(Candidate::class)->find($candidateId);
if (false === $season->candidates->contains($candidate)) {
throw new BadRequestHttpException('Invalid candidate');
}
foreach ((array) $answerIds as $answerId) {
$answer = $this->em->getRepository(Answer::class)->find($answerId);
if (false === $question->answers->contains($answer)) {
throw new BadRequestHttpException('Invalid answer');
}
if ($answer && $candidate) {
$answer->addCandidate($candidate);
}
}
}
$this->em->flush();
$this->addFlash('success', $this->translator->trans('Candidate answers saved'));
return $this->redirectToRoute('tvdt_backoffice_quiz_candidates_question', [
'seasonCode' => $season->seasonCode,
'quiz' => $quiz->id,
'question' => $question->id,
]); ]);
} }
@@ -247,10 +54,10 @@ class QuizController extends AbstractController
name: 'tvdt_backoffice_enable', name: 'tvdt_backoffice_enable',
requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID.'|null'], requirements: ['seasonCode' => self::SEASON_CODE_REGEX, 'quiz' => Requirement::UUID.'|null'],
)] )]
public function enableQuiz(Season $season, ?Quiz $quiz): RedirectResponse public function enableQuiz(Season $season, ?Quiz $quiz, EntityManagerInterface $em): RedirectResponse
{ {
$season->activeQuiz = $quiz; $season->activeQuiz = $quiz;
$this->em->flush(); $em->flush();
if ($quiz instanceof Quiz) { if ($quiz instanceof Quiz) {
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]); return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $season->seasonCode, 'quiz' => $quiz->id]);
@@ -310,53 +117,4 @@ class QuizController extends AbstractController
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]); return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
} }
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
#[Route(
'/backoffice/quiz/{quiz}/candidate/{candidate}/modify_penalty',
name: 'tvdt_backoffice_modify_penalty',
requirements: ['quiz' => Requirement::UUID, 'candidate' => Requirement::UUID],
)]
public function modifyPenalty(Quiz $quiz, Candidate $candidate, Request $request): RedirectResponse
{
if (!$request->isMethod('POST')) {
throw new MethodNotAllowedHttpException(['POST']);
}
$penalty = (int) $request->request->get('penalty');
$this->quizCandidateRepository->setPenaltyForCandidate($quiz, $candidate, $penalty);
return $this->redirectToRoute('tvdt_backoffice_quiz', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
}
#[IsGranted(SeasonVoter::EDIT, subject: 'quiz')]
#[Route(
'/backoffice/quiz/{quiz}/candidate/{candidate}/toggle',
name: 'tvdt_backoffice_toggle_candidate',
requirements: ['quiz' => Requirement::UUID, 'candidate' => Requirement::UUID],
methods: ['GET'],
)]
public function toggleCandidate(Quiz $quiz, Candidate $candidate): RedirectResponse
{
$quizCandidate = $this->quizCandidateRepository->findOneBy([
'quiz' => $quiz,
'candidate' => $candidate,
]);
if (!$quizCandidate instanceof QuizCandidate) {
// Create new QuizCandidate if it doesn't exist (inactive by default when first toggling)
$quizCandidate = new QuizCandidate($quiz, $candidate);
$quizCandidate->active = false;
$this->em->persist($quizCandidate);
} else {
$quizCandidate->active = !$quizCandidate->active;
}
$this->em->flush();
$this->addFlash('success', $this->translator->trans('Candidate status updated'));
return $this->redirectToRoute('tvdt_backoffice_quiz_candidates_tab', ['seasonCode' => $quiz->season->seasonCode, 'quiz' => $quiz->id]);
}
} }
-5
View File
@@ -8,7 +8,6 @@ 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\Core\User\UserInterface;
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; use Tvdt\Enum\FlashType;
@@ -21,10 +20,6 @@ final class LoginController extends AbstractController
#[Route(path: '/login', name: 'tvdt_login_login')] #[Route(path: '/login', name: 'tvdt_login_login')]
public function login(): Response public function login(): Response
{ {
if ($this->getUser() instanceof UserInterface) {
return $this->redirectToRoute('tvdt_backoffice_index');
}
// get the login error if there is one // get the login error if there is one
$error = $this->authenticationUtils->getLastAuthenticationError(); $error = $this->authenticationUtils->getLastAuthenticationError();
// last username entered by the user // last username entered by the user
+1 -16
View File
@@ -99,14 +99,6 @@ final class QuizController extends AbstractController
if ('POST' === $request->getMethod()) { if ('POST' === $request->getMethod()) {
// TODO: Extract saving answer logic to a service // TODO: Extract saving answer logic to a service
// Check if candidate is inactive for this quiz
$quizCandidate = $this->quizCandidateRepository->findOneBy(['quiz' => $quiz, 'candidate' => $candidate]);
if (null !== $quizCandidate && !$quizCandidate->active) {
$this->addFlash(FlashType::Danger, $this->translator->trans('You are not allowed to answer this quiz'));
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->seasonCode]);
}
$answer = $this->answerRepository->findOneBy(['id' => $request->request->get('answer')]); $answer = $this->answerRepository->findOneBy(['id' => $request->request->get('answer')]);
if (!$answer instanceof Answer) { if (!$answer instanceof Answer) {
@@ -131,14 +123,7 @@ final class QuizController extends AbstractController
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->seasonCode]); return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->seasonCode]);
} }
$result = $this->quizCandidateRepository->createIfNotExist($quiz, $candidate); $this->quizCandidateRepository->createIfNotExist($quiz, $candidate);
// Check if candidate is inactive
if (null === $result) {
$this->addFlash(FlashType::Danger, $this->translator->trans('You are not allowed to answer this quiz'));
return $this->redirectToRoute('tvdt_quiz_enter_name', ['seasonCode' => $season->seasonCode]);
}
// end of extracting getting next question logic // end of extracting getting next question logic
return $this->render('quiz/question.twig', ['candidate' => $candidate, 'question' => $question, 'season' => $season]); return $this->render('quiz/question.twig', ['candidate' => $candidate, 'question' => $question, 'season' => $season]);
+4 -8
View File
@@ -15,7 +15,6 @@ use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\Mailer\Exception\TransportExceptionInterface; 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\Component\Security\Core\User\UserInterface;
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\Entity\User;
@@ -25,16 +24,13 @@ 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, private readonly UserPasswordHasherInterface $userPasswordHasher, private readonly Security $security, private readonly LoggerInterface $logger, private readonly UserRepository $userRepository, private readonly EntityManagerInterface $entityManager) {} public function __construct(private readonly EmailVerifier $emailVerifier, private readonly TranslatorInterface $translator, private readonly UserPasswordHasherInterface $userPasswordHasher, private readonly Security $security, private readonly LoggerInterface $logger, private readonly UserRepository $userRepository) {}
#[Route('/register', name: 'tvdt_register')] #[Route('/register', name: 'tvdt_register')]
public function register( public function register(
Request $request, Request $request,
EntityManagerInterface $entityManager,
): Response { ): Response {
if ($this->getUser() instanceof UserInterface) {
return $this->redirectToRoute('tvdt_backoffice_index');
}
$user = new User(); $user = new User();
$form = $this->createForm(RegistrationFormType::class, $user); $form = $this->createForm(RegistrationFormType::class, $user);
$form->handleRequest($request); $form->handleRequest($request);
@@ -45,8 +41,8 @@ final class RegistrationController extends AbstractController
$user->password = $this->userPasswordHasher->hashPassword($user, $plainPassword); $user->password = $this->userPasswordHasher->hashPassword($user, $plainPassword);
$this->entityManager->persist($user); $entityManager->persist($user);
$this->entityManager->flush(); $entityManager->flush();
try { try {
// generate a signed url and email it to the user // generate a signed url and email it to the user
+1
View File
@@ -9,6 +9,7 @@ use Doctrine\Bundle\FixturesBundle\FixtureGroupInterface;
use Doctrine\Common\DataFixtures\DependentFixtureInterface; use Doctrine\Common\DataFixtures\DependentFixtureInterface;
use Doctrine\Persistence\ObjectManager; use Doctrine\Persistence\ObjectManager;
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface; use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
use Tvdt\DataFixtures\KrtekFixtures;
use Tvdt\Entity\Season; use Tvdt\Entity\Season;
use Tvdt\Entity\User; use Tvdt\Entity\User;
-1
View File
@@ -13,7 +13,6 @@ final readonly class Result
public string $name, public string $name,
public int $correct, public int $correct,
public float $corrections, public float $corrections,
public int $penaltySeconds,
public \DateInterval $time, public \DateInterval $time,
public float $score, public float $score,
) {} ) {}
+1 -6
View File
@@ -13,7 +13,7 @@ use Symfony\Component\Uid\Uuid;
use Tvdt\Repository\AnswerRepository; use Tvdt\Repository\AnswerRepository;
#[ORM\Entity(repositoryClass: AnswerRepository::class)] #[ORM\Entity(repositoryClass: AnswerRepository::class)]
class Answer implements \Stringable class Answer
{ {
#[ORM\Column(type: UuidType::NAME)] #[ORM\Column(type: UuidType::NAME)]
#[ORM\CustomIdGenerator(class: 'doctrine.uuid_generator')] #[ORM\CustomIdGenerator(class: 'doctrine.uuid_generator')]
@@ -57,9 +57,4 @@ class Answer implements \Stringable
{ {
$this->candidates->removeElement($candidate); $this->candidates->removeElement($candidate);
} }
public function __toString(): string
{
return $this->text;
}
} }
+17 -3
View File
@@ -13,7 +13,7 @@ use Symfony\Component\Uid\Uuid;
use Tvdt\Repository\QuestionRepository; use Tvdt\Repository\QuestionRepository;
#[ORM\Entity(repositoryClass: QuestionRepository::class)] #[ORM\Entity(repositoryClass: QuestionRepository::class)]
class Question implements \Stringable class Question
{ {
#[ORM\Column(type: UuidType::NAME)] #[ORM\Column(type: UuidType::NAME)]
#[ORM\CustomIdGenerator(class: 'doctrine.uuid_generator')] #[ORM\CustomIdGenerator(class: 'doctrine.uuid_generator')]
@@ -54,8 +54,22 @@ class Question implements \Stringable
return $this; return $this;
} }
public function __toString(): string public function getErrors(): ?string
{ {
return $this->question ?? ''; if (0 === \count($this->answers)) {
return 'This question has no answers';
}
$correctAnswers = $this->answers->filter(static fn (Answer $answer): bool => $answer->isRightAnswer)->count();
if (0 === $correctAnswers) {
return 'This question has no correct answers';
}
if ($correctAnswers > 1) {
return 'This question has multiple correct answers';
}
return null;
} }
} }
-111
View File
@@ -68,115 +68,4 @@ class Quiz
return $this; return $this;
} }
/**
* Get errors for all questions in the quiz.
* Returns an array where keys are question IDs and values are error messages.
*
* @return array<string, string>
*/
public function getQuestionErrors(): array
{
$errors = [];
// Check if any answer in the entire quiz has candidate relations
$hasCandidateRelations = false;
foreach ($this->questions as $question) {
foreach ($question->answers as $answer) {
if ($answer->candidates->count() > 0) {
$hasCandidateRelations = true;
break 2;
}
}
}
// Pre-compute active candidates once for all questions
$activeCandidates = [];
if ($hasCandidateRelations) {
foreach ($this->candidateData as $quizCandidate) {
if ($quizCandidate->active) {
$activeCandidates[] = $quizCandidate->candidate;
}
}
}
foreach ($this->questions as $question) {
$error = $this->getQuestionError($question, $hasCandidateRelations, $activeCandidates);
if (null !== $error) {
$errors[$question->id->toString()] = $error;
}
}
return $errors;
}
/** @param list<Candidate> $activeCandidates */
private function getQuestionError(Question $question, bool $hasCandidateRelations, array $activeCandidates): ?string
{
if (0 === \count($question->answers)) {
return 'This question has no answers';
}
$correctAnswers = $question->answers->filter(static fn (Answer $answer): bool => $answer->isRightAnswer)->count();
if (0 === $correctAnswers) {
return 'This question has no correct answers';
}
if ($correctAnswers > 1) {
return 'This question has multiple correct answers';
}
// Only validate candidate-answer relations if at least one exists in the quiz
if ($hasCandidateRelations) {
$candidateCounts = [];
// Count how many times each candidate appears in answers
foreach ($question->answers as $answer) {
foreach ($answer->candidates as $candidate) {
$candidateId = $candidate->id->toString();
if (!isset($candidateCounts[$candidateId])) {
$candidateCounts[$candidateId] = ['name' => $candidate->name, 'count' => 0];
}
++$candidateCounts[$candidateId]['count'];
}
}
// Check for missing and duplicate candidates (only active ones)
$missing = [];
$duplicates = [];
foreach ($activeCandidates as $candidate) {
$candidateId = $candidate->id->toString();
$count = $candidateCounts[$candidateId]['count'] ?? 0;
if (0 === $count) {
$missing[] = $candidate->name;
} elseif ($count > 1) {
$duplicates[] = $candidate->name;
}
}
if ([] !== $missing || [] !== $duplicates) {
$errors = [];
if ([] !== $missing) {
// If all active candidates are missing, show a special message
if (\count($missing) === \count($activeCandidates)) {
$errors[] = 'No candidates assigned to this question';
} else {
$errors[] = 'Missing candidates: '.implode(', ', $missing);
}
}
if ([] !== $duplicates) {
$errors[] = 'Duplicate candidates: '.implode(', ', $duplicates);
}
return implode('. ', $errors);
}
}
return null;
}
} }
-9
View File
@@ -24,15 +24,6 @@ class QuizCandidate
#[ORM\Column] #[ORM\Column]
public float $corrections = 0; public float $corrections = 0;
#[ORM\Column(type: Types::SMALLINT, options: ['default' => 0])]
public int $penaltySeconds = 0;
#[ORM\Column(type: Types::BOOLEAN, options: ['default' => true])]
public bool $active = true;
#[ORM\Column(type: Types::DATETIMETZ_IMMUTABLE, nullable: true)]
public ?\DateTimeImmutable $started = null;
#[Gedmo\Timestampable(on: 'create')] #[Gedmo\Timestampable(on: 'create')]
#[ORM\Column(type: Types::DATETIMETZ_IMMUTABLE)] #[ORM\Column(type: Types::DATETIMETZ_IMMUTABLE)]
public private(set) \DateTimeImmutable $created; public private(set) \DateTimeImmutable $created;
-2
View File
@@ -30,7 +30,6 @@ class Season
/** @var Collection<int, Quiz> */ /** @var Collection<int, Quiz> */
#[ORM\OneToMany(targetEntity: Quiz::class, mappedBy: 'season', cascade: ['persist'], orphanRemoval: true)] #[ORM\OneToMany(targetEntity: Quiz::class, mappedBy: 'season', cascade: ['persist'], orphanRemoval: true)]
#[ORM\OrderBy(['id' => 'ASC'])]
public private(set) Collection $quizzes; public private(set) Collection $quizzes;
/** @var Collection<int, Candidate> */ /** @var Collection<int, Candidate> */
@@ -40,7 +39,6 @@ class Season
/** @var Collection<int, User> */ /** @var Collection<int, User> */
#[ORM\ManyToMany(targetEntity: User::class, inversedBy: 'seasons')] #[ORM\ManyToMany(targetEntity: User::class, inversedBy: 'seasons')]
#[ORM\OrderBy(['email' => 'ASC'])]
public private(set) Collection $owners; public private(set) Collection $owners;
#[ORM\JoinColumn(nullable: true, onDelete: 'SET NULL')] #[ORM\JoinColumn(nullable: true, onDelete: 'SET NULL')]
+3 -1
View File
@@ -15,7 +15,9 @@ use Symfony\Component\Validator\Constraints\NotBlank;
use Symfony\Contracts\Translation\TranslatorInterface; use Symfony\Contracts\Translation\TranslatorInterface;
use Tvdt\Entity\User; use Tvdt\Entity\User;
/** @extends AbstractType<User> */ /**
* @extends AbstractType<User>
*/
class RegistrationFormType extends AbstractType class RegistrationFormType extends AbstractType
{ {
public function __construct(private readonly TranslatorInterface $translator) {} public function __construct(private readonly TranslatorInterface $translator) {}
+1 -5
View File
@@ -17,16 +17,12 @@ class SettingsForm extends AbstractType
{ {
$builder $builder
->add('showNumbers', options: [ ->add('showNumbers', options: [
'label' => 'Show Numbers',
'label_attr' => ['class' => 'checkbox-switch'], 'label_attr' => ['class' => 'checkbox-switch'],
'attr' => ['role' => 'switch', 'switch' => null]]) 'attr' => ['role' => 'switch', 'switch' => null]])
->add('confirmAnswers', options: [ ->add('confirmAnswers', options: [
'label' => 'Confirm Answers',
'label_attr' => ['class' => 'checkbox-switch'], 'label_attr' => ['class' => 'checkbox-switch'],
'attr' => ['role' => 'switch', 'switch' => null]]) 'attr' => ['role' => 'switch', 'switch' => null]])
->add('save', SubmitType::class, [ ->add('save', SubmitType::class)
'label' => 'Save',
])
; ;
} }
+3 -1
View File
@@ -8,7 +8,9 @@ use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Persistence\ManagerRegistry; use Doctrine\Persistence\ManagerRegistry;
use Tvdt\Entity\Answer; use Tvdt\Entity\Answer;
/** @extends ServiceEntityRepository<Answer> */ /**
* @extends ServiceEntityRepository<Answer>
*/
class AnswerRepository extends ServiceEntityRepository class AnswerRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+3 -1
View File
@@ -11,7 +11,9 @@ use Tvdt\Entity\Candidate;
use Tvdt\Entity\Season; use Tvdt\Entity\Season;
use Tvdt\Helpers\Base64; use Tvdt\Helpers\Base64;
/** @extends ServiceEntityRepository<Candidate> */ /**
* @extends ServiceEntityRepository<Candidate>
*/
class CandidateRepository extends ServiceEntityRepository class CandidateRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+3 -1
View File
@@ -8,7 +8,9 @@ use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Persistence\ManagerRegistry; use Doctrine\Persistence\ManagerRegistry;
use Tvdt\Entity\Elimination; use Tvdt\Entity\Elimination;
/** @extends ServiceEntityRepository<Elimination> */ /**
* @extends ServiceEntityRepository<Elimination>
*/
class EliminationRepository extends ServiceEntityRepository class EliminationRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+3 -1
View File
@@ -8,7 +8,9 @@ use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Persistence\ManagerRegistry; use Doctrine\Persistence\ManagerRegistry;
use Tvdt\Entity\GivenAnswer; use Tvdt\Entity\GivenAnswer;
/** @extends ServiceEntityRepository<GivenAnswer> */ /**
* @extends ServiceEntityRepository<GivenAnswer>
*/
class GivenAnswerRepository extends ServiceEntityRepository class GivenAnswerRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+3 -1
View File
@@ -9,7 +9,9 @@ use Doctrine\Persistence\ManagerRegistry;
use Tvdt\Entity\Candidate; use Tvdt\Entity\Candidate;
use Tvdt\Entity\Question; use Tvdt\Entity\Question;
/** @extends ServiceEntityRepository<Question> */ /**
* @extends ServiceEntityRepository<Question>
*/
class QuestionRepository extends ServiceEntityRepository class QuestionRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+6 -30
View File
@@ -6,12 +6,13 @@ namespace Tvdt\Repository;
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository; use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Persistence\ManagerRegistry; use Doctrine\Persistence\ManagerRegistry;
use Safe\DateTimeImmutable;
use Tvdt\Entity\Candidate; use Tvdt\Entity\Candidate;
use Tvdt\Entity\Quiz; use Tvdt\Entity\Quiz;
use Tvdt\Entity\QuizCandidate; use Tvdt\Entity\QuizCandidate;
/** @extends ServiceEntityRepository<QuizCandidate> */ /**
* @extends ServiceEntityRepository<QuizCandidate>
*/
class QuizCandidateRepository extends ServiceEntityRepository class QuizCandidateRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
@@ -19,28 +20,14 @@ class QuizCandidateRepository extends ServiceEntityRepository
parent::__construct($registry, QuizCandidate::class); parent::__construct($registry, QuizCandidate::class);
} }
/** @return bool|null true if a new entry was created, false if it already exists, null if candidate is inactive */ /** @return bool true if a new entry was created */
public function createIfNotExist(Quiz $quiz, Candidate $candidate): ?bool public function createIfNotExist(Quiz $quiz, Candidate $candidate): bool
{ {
$quizCandidate = $this->findOneBy(['candidate' => $candidate, 'quiz' => $quiz]); if (0 !== $this->count(['candidate' => $candidate, 'quiz' => $quiz])) {
if (null !== $quizCandidate) {
// Check if candidate is inactive
if (!$quizCandidate->active) {
return null;
}
// If QuizCandidate exists but hasn't started yet, set the started timestamp
if (null === $quizCandidate->started) {
$quizCandidate->started = new DateTimeImmutable();
$this->getEntityManager()->flush();
}
return false; return false;
} }
$quizCandidate = new QuizCandidate($quiz, $candidate); $quizCandidate = new QuizCandidate($quiz, $candidate);
$quizCandidate->started = new DateTimeImmutable();
$this->getEntityManager()->persist($quizCandidate); $this->getEntityManager()->persist($quizCandidate);
$this->getEntityManager()->flush(); $this->getEntityManager()->flush();
@@ -57,15 +44,4 @@ class QuizCandidateRepository extends ServiceEntityRepository
$quizCandidate->corrections = $corrections; $quizCandidate->corrections = $corrections;
$this->getEntityManager()->flush(); $this->getEntityManager()->flush();
} }
public function setPenaltyForCandidate(Quiz $quiz, Candidate $candidate, int $penalty): void
{
$quizCandidate = $this->findOneBy(['candidate' => $candidate, 'quiz' => $quiz]);
if (!$quizCandidate instanceof QuizCandidate) {
throw new \InvalidArgumentException('Quiz candidate not found');
}
$quizCandidate->penaltySeconds = $penalty;
$this->getEntityManager()->flush();
}
} }
+15 -74
View File
@@ -9,12 +9,13 @@ use Doctrine\Persistence\ManagerRegistry;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use Safe\DateTimeImmutable; use Safe\DateTimeImmutable;
use Safe\Exceptions\DatetimeException; use Safe\Exceptions\DatetimeException;
use Symfony\Component\Uid\Uuid;
use Tvdt\Dto\Result; use Tvdt\Dto\Result;
use Tvdt\Entity\Quiz; use Tvdt\Entity\Quiz;
use Tvdt\Exception\ErrorClearingQuizException; use Tvdt\Exception\ErrorClearingQuizException;
/** @extends ServiceEntityRepository<Quiz> */ /**
* @extends ServiceEntityRepository<Quiz>
*/
class QuizRepository extends ServiceEntityRepository class QuizRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry, private readonly LoggerInterface $logger) public function __construct(ManagerRegistry $registry, private readonly LoggerInterface $logger)
@@ -53,7 +54,7 @@ class QuizRepository extends ServiceEntityRepository
catch (\Throwable $throwable) { catch (\Throwable $throwable) {
$this->logger->error($throwable->getMessage()); $this->logger->error($throwable->getMessage());
$em->rollback(); $em->rollback();
throw new ErrorClearingQuizException(message: $throwable->getMessage(), code: $throwable->getCode(), previous: $throwable); throw new ErrorClearingQuizException(previous: $throwable);
} }
// @codeCoverageIgnoreEnd // @codeCoverageIgnoreEnd
@@ -80,86 +81,26 @@ class QuizRepository extends ServiceEntityRepository
c.name, c.name,
sum(case when a.isRightAnswer = true then 1 else 0 end) as correct, sum(case when a.isRightAnswer = true then 1 else 0 end) as correct,
qd.corrections, qd.corrections,
qd.penaltySeconds,
max(ga.created) as end_time, max(ga.created) as end_time,
qd.started as start_time, qd.created as start_time,
(sum(case when a.isRightAnswer = true then 1 else 0 end) + qd.corrections) as score (sum(case when a.isRightAnswer = true then 1 else 0 end) + qd.corrections) as score
from Tvdt\Entity\Candidate c from Tvdt\Entity\Candidate c
join c.givenAnswers ga join c.givenAnswers ga
join ga.answer a join ga.answer a
join c.quizData qd join c.quizData qd
where qd.quiz = :quiz and ga.quiz = :quiz and qd.started is not null where qd.quiz = :quiz and ga.quiz = :quiz
group by ga.quiz, c.id, qd.id group by ga.quiz, c.id, qd.id
order by score desc, max(ga.created) - qd.started asc order by score desc, max(ga.created) - qd.created asc
DQL DQL
)->setParameter('quiz', $quiz)->getResult(); )->setParameter('quiz', $quiz)->getResult();
return array_map(static function (array $row): Result { return array_map(static fn (array $row): Result => new Result(
\assert($row['start_time'] instanceof \DateTimeImmutable); id: $row['id'],
name: $row['name'],
return new Result( correct: (int) $row['correct'],
id: $row['id'], corrections: $row['corrections'],
name: $row['name'], time: $row['start_time']->diff(new DateTimeImmutable($row['end_time'])),
correct: (int) $row['correct'], score: $row['score'],
corrections: $row['corrections'], ), $result);
penaltySeconds: $row['penaltySeconds'],
time: $row['start_time']->diff(new DateTimeImmutable($row['end_time'])),
score: $row['score'],
);
}, $result);
}
public function fetchWithQuestions(Uuid $id): Quiz
{
return $this->getEntityManager()->createQuery(<<<dql
select q, qz, a from Tvdt\Entity\Quiz q
join q.questions qz
join qz.answers a
where q.id = :id
dql)->setParameter('id', $id)->getSingleResult();
}
/**
* Fetch quiz with all relations needed for error checking.
* This includes: questions, answers, answer candidates, and season candidates.
*/
public function fetchWithQuestionsAndCandidates(Uuid $id): Quiz
{
return $this->getEntityManager()->createQuery(<<<dql
select q, qz, a, ac, s, sc, qc from Tvdt\Entity\Quiz q
join q.questions qz
join qz.answers a
left join a.candidates ac
join q.season s
left join s.candidates sc
left join q.candidateData qc
where q.id = :id
dql)->setParameter('id', $id)->getSingleResult();
}
/**
* Get given answers count per candidate for a quiz.
*
* @return array<string, int> Array with candidate ID as key and count as value
*/
public function getGivenAnswersCountPerCandidate(Quiz $quiz): array
{
$results = $this->getEntityManager()->createQuery(<<<DQL
select c.id as candidateId, count(ga.id) as answerCount
from Tvdt\Entity\Candidate c
left join c.givenAnswers ga with ga.quiz = :quiz
where c.season = :season
group by c.id
DQL
)->setParameter('quiz', $quiz)
->setParameter('season', $quiz->season)
->getResult();
$counts = [];
foreach ($results as $row) {
$counts[$row['candidateId']->toString()] = (int) $row['answerCount'];
}
return $counts;
} }
} }
+3 -1
View File
@@ -9,7 +9,9 @@ use Doctrine\Persistence\ManagerRegistry;
use Tvdt\Entity\Season; use Tvdt\Entity\Season;
use Tvdt\Entity\User; use Tvdt\Entity\User;
/** @extends ServiceEntityRepository<Season> */ /**
* @extends ServiceEntityRepository<Season>
*/
class SeasonRepository extends ServiceEntityRepository class SeasonRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+3 -1
View File
@@ -8,7 +8,9 @@ use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Persistence\ManagerRegistry; use Doctrine\Persistence\ManagerRegistry;
use Tvdt\Entity\SeasonSettings; use Tvdt\Entity\SeasonSettings;
/** @extends ServiceEntityRepository<SeasonSettings> */ /**
* @extends ServiceEntityRepository<SeasonSettings>
*/
class SeasonSettingsRepository extends ServiceEntityRepository class SeasonSettingsRepository extends ServiceEntityRepository
{ {
public function __construct(ManagerRegistry $registry) public function __construct(ManagerRegistry $registry)
+5 -9
View File
@@ -95,15 +95,11 @@ class QuizSpreadsheetService
$arrCounter = 1; $arrCounter = 1;
while (true) { while (true) {
try { if (null === $questionArr[$arrCounter]) {
if (null === $questionArr[$arrCounter]) { if (1 === $answerCounter) {
if (1 === $answerCounter) { $errors[] = \sprintf('Question %d has no answers', $answerCounter);
$errors[] = \sprintf('Question %d has no answers', $answerCounter);
}
break;
} }
} catch (\ErrorException) {
break; break;
} }
@@ -134,6 +130,6 @@ class QuizSpreadsheetService
private function isSpreadsheetFile(File $file): bool private function isSpreadsheetFile(File $file): bool
{ {
return 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' === $file->getMimeType(); return 'xlsx' === $file->getExtension();
} }
} }
+25 -21
View File
@@ -99,12 +99,12 @@
] ]
}, },
"sentry/sentry-symfony": { "sentry/sentry-symfony": {
"version": "5.8", "version": "5.6",
"recipe": { "recipe": {
"repo": "github.com/symfony/recipes-contrib", "repo": "github.com/symfony/recipes-contrib",
"branch": "main", "branch": "main",
"version": "5.0", "version": "5.0",
"ref": "12f504985eb24e3b20a9e41e0ec7e398798d18f0" "ref": "b6cb4b34429dadecd7187852123be19d628fa37a"
}, },
"files": [ "files": [
"config/packages/sentry.yaml" "config/packages/sentry.yaml"
@@ -184,15 +184,14 @@
] ]
}, },
"symfony/framework-bundle": { "symfony/framework-bundle": {
"version": "8.0", "version": "7.2",
"recipe": { "recipe": {
"repo": "github.com/symfony/recipes", "repo": "github.com/symfony/recipes",
"branch": "main", "branch": "main",
"version": "7.4", "version": "7.2",
"ref": "09f6e081c763a206802674ce0cb34a022f0ffc6d" "ref": "87bcf6f7c55201f345d8895deda46d2adbdbaa89"
}, },
"files": [ "files": [
".editorconfig",
"config/packages/cache.yaml", "config/packages/cache.yaml",
"config/packages/framework.yaml", "config/packages/framework.yaml",
"config/preload.php", "config/preload.php",
@@ -225,14 +224,19 @@
} }
}, },
"symfony/phpunit-bridge": { "symfony/phpunit-bridge": {
"version": "8.0", "version": "7.2",
"recipe": { "recipe": {
"repo": "github.com/symfony/recipes", "repo": "github.com/symfony/recipes",
"branch": "main", "branch": "main",
"version": "7.3", "version": "6.3",
"ref": "dc13fec96bd527bd399c3c01f0aab915c67fd544" "ref": "a411a0480041243d97382cac7984f7dce7813c08"
}, },
"files": [] "files": [
".env.test",
"bin/phpunit",
"phpunit.xml.dist",
"tests/bootstrap.php"
]
}, },
"symfony/property-info": { "symfony/property-info": {
"version": "7.3", "version": "7.3",
@@ -247,12 +251,12 @@
] ]
}, },
"symfony/routing": { "symfony/routing": {
"version": "8.0", "version": "7.2",
"recipe": { "recipe": {
"repo": "github.com/symfony/recipes", "repo": "github.com/symfony/recipes",
"branch": "main", "branch": "main",
"version": "7.4", "version": "7.0",
"ref": "bc94c4fd86f393f3ab3947c18b830ea343e51ded" "ref": "21b72649d5622d8f7da329ffb5afb232a023619d"
}, },
"files": [ "files": [
"config/packages/routing.yaml", "config/packages/routing.yaml",
@@ -260,12 +264,12 @@
] ]
}, },
"symfony/security-bundle": { "symfony/security-bundle": {
"version": "8.0", "version": "7.2",
"recipe": { "recipe": {
"repo": "github.com/symfony/recipes", "repo": "github.com/symfony/recipes",
"branch": "main", "branch": "main",
"version": "7.4", "version": "6.4",
"ref": "c42fee7802181cdd50f61b8622715829f5d2335c" "ref": "2ae08430db28c8eb4476605894296c82a642028f"
}, },
"files": [ "files": [
"config/packages/security.yaml", "config/packages/security.yaml",
@@ -273,18 +277,18 @@
] ]
}, },
"symfony/stimulus-bundle": { "symfony/stimulus-bundle": {
"version": "2.32", "version": "2.26",
"recipe": { "recipe": {
"repo": "github.com/symfony/recipes", "repo": "github.com/symfony/recipes",
"branch": "main", "branch": "main",
"version": "2.24", "version": "2.20",
"ref": "3357f2fa6627b93658d8e13baa416b2a94a50c5f" "ref": "3acc494b566816514a6873a89023a35440b6386d"
}, },
"files": [ "files": [
"assets/bootstrap.js",
"assets/controllers.json", "assets/controllers.json",
"assets/controllers/csrf_protection_controller.js", "assets/controllers/csrf_protection_controller.js",
"assets/controllers/hello_controller.js", "assets/controllers/hello_controller.js"
"assets/stimulus_bootstrap.js"
] ]
}, },
"symfony/translation": { "symfony/translation": {
-13
View File
@@ -2,16 +2,3 @@
{% block importmap %}{{ importmap('backoffice') }}{% endblock %} {% block importmap %}{{ importmap('backoffice') }}{% endblock %}
{% block title %}Tijd voor de test | {% endblock %} {% block title %}Tijd voor de test | {% endblock %}
{% block nav %}{{ include('backoffice/nav.html.twig') }}{% endblock %} {% block nav %}{{ include('backoffice/nav.html.twig') }}{% endblock %}
{% block main %}
<div class="container">
<div class="mt-3">
{% block breadcrumbs %}{% endblock %}
</div>
{{ include('flashes.html.twig') }}
<div class="mb-5">
{% block body %}
{% endblock body %}
</div>
</div>
{% endblock %}
+3 -11
View File
@@ -2,17 +2,9 @@
{% block title %}{{ parent() }}Backoffice{% endblock %} {% block title %}{{ parent() }}Backoffice{% endblock %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item active" aria-current="page">{{ 'Home'|trans }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
<div class="d-flex flex-row align-items-center mb-3"> <div class="d-flex flex-row align-items-center">
<h2 class="mb-0 pe-2"> <h2 class="py-2 pe-2">
{{ is_granted('ROLE_ADMIN') ? 'All Seasons'|trans : 'Your Seasons'|trans }} {{ is_granted('ROLE_ADMIN') ? 'All Seasons'|trans : 'Your Seasons'|trans }}
</h2> </h2>
<a class="link" href="{{ path('tvdt_backoffice_season_add') }}"> <a class="link" href="{{ path('tvdt_backoffice_season_add') }}">
@@ -20,7 +12,7 @@
</a> </a>
</div> </div>
{% if seasons %} {% if seasons %}
<table class="table table-hover mb-3"> <table class="table table-hover">
<thead> <thead>
<tr> <tr>
{% if is_granted('ROLE_ADMIN') %} {% if is_granted('ROLE_ADMIN') %}
@@ -1,17 +1,20 @@
{% extends 'backoffice/base.html.twig' %} {% extends 'backoffice/base.html.twig' %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">{{ 'Home'|trans }}</a></li>
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_season', {seasonCode: elimination.quiz.season.seasonCode}) }}">{{ elimination.quiz.season.name }}</a></li>
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_quiz', {seasonCode: elimination.quiz.season.seasonCode, quiz: elimination.quiz.id}) }}">{{ elimination.quiz.name }}</a></li>
<li class="breadcrumb-item active" aria-current="page">{{ 'Prepare Elimination'|trans }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
<div class="row">
<nav aria-label="breadcrumb">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">Home</a></li>
<li class="breadcrumb-item"><a
href="{{ path('tvdt_backoffice_season', {seasonCode: elimination.quiz.season.seasonCode}) }}">{{ elimination.quiz.season.name }}</a>
</li>
<li class="breadcrumb-item"><a
href="{{ path('tvdt_backoffice_quiz', {seasonCode: elimination.quiz.season.seasonCode, quiz: elimination.quiz.id}) }}">{{ elimination.quiz.name }}</a>
</li>
<li class="breadcrumb-item active" aria-current="page">Prepare Elimination</li>
</ol>
</nav>
</div>
<div class="row"> <div class="row">
<div class="col-12 col-md-6"> <div class="col-12 col-md-6">
<form method="post"> <form method="post">
@@ -29,7 +32,7 @@
</div> </div>
</div> </div>
{% endfor %} {% endfor %}
<div class="btn-group mb-3"> <div class="btn-group py-2">
<button type="submit" class="btn btn-primary" name="start" value="0">{{ 'Save'|trans }}</button> <button type="submit" class="btn btn-primary" name="start" value="0">{{ 'Save'|trans }}</button>
<button type="submit" class="btn btn-success" name="start" <button type="submit" class="btn btn-success" name="start"
value="1">{{ 'Save and start elimination'|trans }}</button> value="1">{{ 'Save and start elimination'|trans }}</button>
@@ -39,7 +42,7 @@
</form> </form>
</div> </div>
<div class="col-12 col-md-6"> <div class="col-12 col-md-6">
<p class="mb-3">{{ 'Help text for preparing elimination'|trans }}</p> <p>Hier kan dus weer wat uitleg komen</p>
</div> </div>
</div> </div>
{% endblock %} {% endblock %}
+159 -29
View File
@@ -1,36 +1,166 @@
{% extends 'backoffice/base.html.twig' %} {% extends 'backoffice/base.html.twig' %}
{% block title %}{{ parent() }}{{ season.name }}{% endblock %} {% block title %}{{ parent() }}{{ quiz.season.name }}{% endblock %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">{{ 'Home'|trans }}</a></li>
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_season', {seasonCode: season.seasonCode}) }}">{{ season.name }}</a></li>
<li class="breadcrumb-item active" aria-current="page">{{ quiz.name }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
{% set tabs = [ <h2 class="py-2">{{ 'Quiz'|trans }}: {{ quiz.season.name }} - {{ quiz.name }}</h2>
{id: 'overview', label: 'Overview'|trans, route: 'tvdt_backoffice_quiz_overview'}, <div class="py-2 btn-group" data-controller="bo--quiz">
{id: 'candidates', label: 'Candidates'|trans, route: 'tvdt_backoffice_quiz_candidates_tab'}, <a class="btn btn-primary {% if quiz is same as(season.activeQuiz) %}disabled{% endif %}"
{id: 'result', label: 'Results & Elimination'|trans, route: 'tvdt_backoffice_quiz_result'}, href="{{ path('tvdt_backoffice_enable', {seasonCode: season.seasonCode, quiz: quiz.id}) }}">{{ 'Make active'|trans }}</a>
{id: 'answers', label: 'Answer Mapping'|trans, route: 'tvdt_backoffice_quiz_candidates'}, {% if quiz is same as (season.activeQuiz) %}
] %} <a class="btn btn-secondary"
href="{{ path('tvdt_backoffice_enable', {seasonCode: season.seasonCode, quiz: 'null'}) }}">{{ 'Deactivate Quiz'|trans }}</a>
{% endif %}
<button class="btn btn-danger" data-action="click->bo--quiz#clearQuiz">
{{ 'Clear quiz...'|trans }}
</button>
<button class="btn btn-danger" data-action="click->bo--quiz#deleteQuiz">
{{ 'Delete Quiz...'|trans }}
</button>
</div>
<h2 class="mb-3">{{ 'Quiz'|trans }}: {{ season.name }} - {{ quiz.name }}</h2> <div id="questions">
<ul class="nav nav-tabs mb-3"> <h4 class="py-2">{{ 'Questions'|trans }}</h4>
{% for tab in tabs %} <div class="accordion">
<li class="nav-item"> {%~ for question in quiz.questions ~%}
<a class="nav-link{{ activeTab == tab.id ? ' active' }}" href="{{ path(tab.route, {seasonCode: season.seasonCode, quiz: quiz.id}) }}"> <div class="accordion-item">
{{ tab.label }} <h2 class="accordion-header">
</a> <button class="accordion-button collapsed"
</li> type="button"
{% endfor %} data-bs-toggle="collapse"
</ul> data-bs-target="#question-{{ loop.index0 }}"
<div class="pt-3"> aria-controls="question-{{ loop.index0 }}">
{{ include(template) }} {% set questionErrors = question.getErrors %}
{%~ if questionErrors -%}
<span data-bs-toggle="tooltip"
title="{{ questionErrors }}"
class="badge text-bg-danger rounded-pill me-2">!</span>
{% endif %}
{{~ loop.index -}}. {{ question.question -}}
</button>
</h2>
<div id="question-{{ loop.index0 }}"
class="accordion-collapse collapse">
<div class="accordion-body">
<ul>
{%~ for answer in question.answers %}
<li{% if answer.isRightAnswer %} class="text-decoration-underline"{% endif %}>{{ answer.text -}}</li>
{%~ else %}
{{ 'There are no answers for this question'|trans -}}
{%~ endfor %}
</ul>
</div>
</div>
</div>
{% else %}
{{ 'EMPTY'|trans }}
{% endfor %}
</div>
</div>
<div class="scores">
<h4 class="py-2">{{ 'Score'|trans }}</h4>
<div class="btn-toolbar" role="toolbar">
<div class="btn-group btn-group-lg me-2">
{# <a class="btn btn-primary">{{ 'Start Elimination'|trans }}</a> #}
<a href="{{ path('tvdt_prepare_elimination', {seasonCode: season.seasonCode, quiz: quiz.id}) }}"
class="btn btn-secondary">{{ 'Prepare Custom Elimination'|trans }}</a>
{%~ if not quiz.eliminations.empty %}
<button class="btn btn-secondary dropdown-toggle"
data-bs-toggle="dropdown">{{ 'Load Prepared Elimination'|trans }}</button>
<ul class="dropdown-menu">
{%~ for elimination in quiz.eliminations %}
<li><a class="dropdown-item"
href="{{ path('tvdt_prepare_elimination_view', {elimination: elimination.id}) }}">{{ elimination.created|format_datetime() }}</a>
</li>
{%~ endfor %}
</ul>
{% endif %}
</div>
</div>
<p>{{ 'Number of dropouts:'|trans }} {{ quiz.dropouts }} </p>
<table class="table table-hover">
<thead>
<tr>
<th scope="col">{{ 'Candidate'|trans }}</th>
<th style="width: 15%" scope="col">{{ 'Correct Answers'|trans }}</th>
<th style="width: 20%" scope="col">{{ 'Corrections'|trans }}</th>
<th style="width: 10%" scope="col">{{ 'Score'|trans }}</th>
<th style="width: 20%" scope="col">{{ 'Time'|trans }}</th>
</tr>
</thead>
<tbody>
{%~ for candidate in result ~%}
<tr class="table-{% if loop.revindex > quiz.dropouts %}success{% else %}danger{% endif %}">
<td>{{ candidate.name }}</td>
<td>{{ candidate.correct|default('0') }}</td>
<td>
<form method="post"
action="{{ path('tvdt_backoffice_modify_correction', {quiz: quiz.id, candidate: candidate.id}) }}">
<div class="row">
<div class="col-8">
<input class="form-control form-control-sm" type="number"
value="{{ candidate.corrections }}" step="0.5"
name="corrections">
</div>
<div class="col-2">
<button class="btn btn-sm btn-primary" type="submit">{{ 'Save'|trans }}</button>
</div>
</div>
</form>
</td>
<td>{{ candidate.score|default('x') }}</td>
<td>{{ candidate.time.format('%i:%S') }}</td>
</tr>
{% else %}
<tr>
<td colspan="5">{{ 'No results'|trans }}</td>
</tr>
{% endfor %}
</tbody>
</table>
</div>
{# Modal Clear #}
<div class="modal fade" id="clearQuizModal" data-bs-backdrop="static"
tabindex="-1"
aria-labelledby="staticBackdropLabel" aria-hidden="true">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<h1 class="modal-title fs-5" id="staticBackdropLabel">{{ 'Please Confirm'|trans }}</h1>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
{{ 'Are you sure you want to clear all the results? This will also delete al the eliminations.'|trans }}
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">{{ 'No'|trans }}</button>
<a href="{{ path('tvdt_backoffice_quiz_clear', {quiz: quiz.id}) }}"
class="btn btn-danger">{{ 'Yes'|trans }}</a>
</div>
</div>
</div>
</div>
{# Modal Delete #}
<div class="modal fade" id="deleteQuizModal" data-bs-backdrop="static"
tabindex="-1"
aria-labelledby="staticBackdropLabel" aria-hidden="true">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<h1 class="modal-title fs-5" id="staticBackdropLabel">{{ 'Please Confirm'|trans }}</h1>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
{{ 'Are you sure you want to delete this quiz?'|trans }}
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">{{ 'No'|trans }}</button>
<a href="{{ path('tvdt_backoffice_quiz_delete', {quiz: quiz.id}) }}"
class="btn btn-danger">{{ 'Yes'|trans }}</a>
</div>
</div>
</div>
</div> </div>
{% endblock %} {% endblock %}
@@ -1,66 +0,0 @@
<div class="d-flex justify-content-between align-items-center mb-3">
<div>
{% set questions = quiz.questions %}
{% set currentIndex = null %}
{% for index, q in questions %}
{% if q.id.toString == question.id.toString %}
{% set currentIndex = index %}
{% endif %}
{% endfor %}
{% if currentIndex > 0 %}
{% set prevQuestion = questions[currentIndex - 1] %}
<a href="{{ path('tvdt_backoffice_quiz_candidates_question', {seasonCode: season.seasonCode, quiz: quiz.id, question: prevQuestion.id}) }}"
class="btn btn-secondary">
{{ 'Previous'|trans }}
</a>
{% endif %}
</div>
<h4 class="mb-0">{{ currentIndex + 1 }}. {{ question }}</h4>
<div>
{% if currentIndex is not null and currentIndex < (questions|length - 1) %}
{% set nextQuestion = questions[currentIndex + 1] %}
<a href="{{ path('tvdt_backoffice_quiz_candidates_question', {seasonCode: season.seasonCode, quiz: quiz.id, question: nextQuestion.id}) }}"
class="btn btn-secondary">
{{ 'Next'|trans }}
</a>
{% endif %}
</div>
</div>
<form method="post">
<input type="hidden" name="_token" value="{{ csrf_token('candidate_answer') }}">
<table class="table table-hover table-striped mb-3">
<thead>
<tr>
<th scope="col">{{ 'Candidate'|trans }}</th>
{% for answer in question.answers %}
<th scope="col">{{ answer }}</th>
{% endfor %}
</tr>
</thead>
<tbody>
{% for candidate in candidates %}
<tr>
<th scope="row">{{ candidate.name }}</th>
{% for answer in question.answers %}
<td>
<input type="checkbox"
id="candidate_{{ candidate.id }}_answer_{{ answer.id }}"
name="candidate_answer[{{ candidate.id }}][]"
value="{{ answer.id }}"
class="form-check-input"
{{ answer.candidates.contains(candidate) ? 'checked' : '' }}>
<label for="candidate_{{ candidate.id }}_answer_{{ answer.id }}" class="visually-hidden">
{{ candidate.name }} - {{ answer }}
</label>
</td>
{% endfor %}
</tr>
{% endfor %}
</tbody>
</table>
<button type="submit" class="btn btn-primary">{{ 'Save'|trans }}</button>
</form>
@@ -1,50 +0,0 @@
<h4 class="mb-3">{{ 'Candidates'|trans }}</h4>
<table class="table table-hover mb-3">
<thead>
<tr>
<th scope="col">{{ 'Name'|trans }}</th>
<th scope="col">{{ 'Quiz Status'|trans }}</th>
<th scope="col">{{ 'Candidate Status'|trans }}</th>
<th scope="col">{{ 'Actions'|trans }}</th>
</tr>
</thead>
<tbody>
{% for data in candidateData %}
{% set candidate = data.candidate %}
{% set quizCandidate = data.quizCandidate %}
{% set givenAnswersCount = data.givenAnswersCount %}
<tr>
<td>{{ candidate.name }}</td>
<td>
{% if quizCandidate and quizCandidate.started %}
{% if givenAnswersCount >= quiz.questions|length %}
<span class="badge text-bg-success">{{ 'Completed'|trans }}</span>
{% else %}
<span class="badge text-bg-warning">{{ 'In Progress'|trans }} ({{ givenAnswersCount }}/{{ quiz.questions|length }})</span>
{% endif %}
{% else %}
<span class="badge text-bg-secondary">{{ 'Not Started'|trans }}</span>
{% endif %}
</td>
<td>
{% if quizCandidate == null or quizCandidate.active %}
<span class="badge text-bg-success">{{ 'Active'|trans }}</span>
{% else %}
<span class="badge text-bg-secondary">{{ 'Inactive'|trans }}</span>
{% endif %}
</td>
<td>
<a href="{{ path('tvdt_backoffice_toggle_candidate', {quiz: quiz.id, candidate: candidate.id}) }}"
class="btn btn-sm btn-outline-secondary">
{% if quizCandidate == null or quizCandidate.active %}
{{ 'Deactivate'|trans }}
{% else %}
{{ 'Activate'|trans }}
{% endif %}
</a>
</td>
</tr>
{% endfor %}
</tbody>
</table>
@@ -1,105 +0,0 @@
<div data-controller="bo--quiz">
<h4 class="mb-3">Quick actions</h4>
<div class="mb-3 btn-group">
{% if quiz is same as (season.activeQuiz) %}
<a class="btn btn-secondary"
href="{{ path('tvdt_backoffice_enable', {seasonCode: season.seasonCode, quiz: 'null'}) }}">{{ 'Deactivate Quiz'|trans }}</a>
{% else %}
<a class="btn btn-primary"
href="{{ path('tvdt_backoffice_enable', {seasonCode: season.seasonCode, quiz: quiz.id}) }}">{{ 'Make active'|trans }}</a>
{% endif %}
<button class="btn btn-danger" data-action="click->bo--quiz#clearQuiz">
{{ 'Clear Quiz...'|trans }}
</button>
<button class="btn btn-danger" data-action="click->bo--quiz#deleteQuiz">
{{ 'Delete Quiz...'|trans }}
</button>
</div>
<h4 class="mb-3">{{ 'Questions'|trans }}</h4>
<div class="accordion">
{%~ for question in quiz.questions ~%}
<div class="accordion-item">
<h2 class="accordion-header">
<button class="accordion-button collapsed"
type="button"
data-bs-toggle="collapse"
data-bs-target="#question-{{ loop.index0 }}"
aria-controls="question-{{ loop.index0 }}">
{% set questionError = questionErrors[question.id.toString] ?? null %}
<span class="badge rounded-pill me-2{% if questionError %} text-bg-danger{% else %} invisible{% endif %}"{% if questionError %} data-bs-toggle="tooltip" title="{{ questionError }}"{% endif %}>!</span>
{{~ loop.index -}}. {{ question.question -}}
</button>
</h2>
<div id="question-{{ loop.index0 }}"
class="accordion-collapse collapse">
<div class="accordion-body">
<ul>
{%~ for answer in question.answers %}
<li{% if answer.isRightAnswer %} class="text-decoration-underline"{% endif %}>
{{ answer.text }}
{% if answer.candidates|length > 0 %}
<small class="text-muted">
({{ answer.candidates|map(c => c.name)|join(', ') }})
</small>
{% endif %}
</li>
{%~ else %}
{{ 'There are no answers for this question'|trans -}}
{%~ endfor %}
</ul>
</div>
</div>
</div>
{% else %}
{{ 'EMPTY'|trans }}
{% endfor %}
</div>
{# Modal Clear #}
<div class="modal fade" id="clearQuizModal" data-bs-backdrop="static"
tabindex="-1"
data-bo--quiz-target="clearModal"
aria-labelledby="staticBackdropLabel" aria-hidden="true">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<h1 class="modal-title fs-5" id="staticBackdropLabel">{{ 'Please Confirm'|trans }}</h1>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
{{ 'Are you sure you want to clear all the results? This will also delete all the eliminations.'|trans }}
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">{{ 'No'|trans }}</button>
<a href="{{ path('tvdt_backoffice_quiz_clear', {quiz: quiz.id}) }}"
class="btn btn-danger">{{ 'Yes'|trans }}</a>
</div>
</div>
</div>
</div>
{# Modal Delete #}
<div class="modal fade" id="deleteQuizModal" data-bs-backdrop="static"
tabindex="-1"
data-bo--quiz-target="deleteModal"
aria-labelledby="staticBackdropLabel" aria-hidden="true">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<h1 class="modal-title fs-5" id="staticBackdropLabel">{{ 'Please Confirm'|trans }}</h1>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<div class="modal-body">
{{ 'Are you sure you want to delete this quiz?'|trans }}
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">{{ 'No'|trans }}</button>
<a href="{{ path('tvdt_backoffice_quiz_delete', {quiz: quiz.id}) }}"
class="btn btn-danger">{{ 'Yes'|trans }}</a>
</div>
</div>
</div>
</div>
</div>
@@ -1,77 +0,0 @@
<h4 class="mb-3">{{ 'Score'|trans }}</h4>
<div class="btn-toolbar mb-3" role="toolbar">
<div class="btn-group me-2">
{# <a class="btn btn-primary">{{ 'Start Elimination'|trans }}</a> #}
<a href="{{ path('tvdt_prepare_elimination', {seasonCode: season.seasonCode, quiz: quiz.id}) }}"
class="btn btn-secondary">{{ 'Prepare Custom Elimination'|trans }}</a>
{%~ if not quiz.eliminations.empty %}
<button class="btn btn-secondary dropdown-toggle"
data-bs-toggle="dropdown">{{ 'Load Prepared Elimination'|trans }}</button>
<ul class="dropdown-menu">
{%~ for elimination in quiz.eliminations %}
<li><a class="dropdown-item"
href="{{ path('tvdt_prepare_elimination_view', {elimination: elimination.id}) }}">{{ elimination.created|format_datetime() }}</a>
</li>
{%~ endfor %}
</ul>
{% endif %}
</div>
</div>
<p class="mb-3">{{ 'Number of dropouts:'|trans }} {{ quiz.dropouts }} </p>
<table class="table table-hover mb-3">
<thead>
<tr>
<th scope="col">{{ 'Candidate'|trans }}</th>
<th style="width: 15%" scope="col">{{ 'Correct Answers'|trans }}</th>
<th style="width: 20%" scope="col">{{ 'Corrections'|trans }}</th>
<th style="width: 20%" scope="col">{{ 'Penalty'|trans }}</th>
<th style="width: 10%" scope="col">{{ 'Score'|trans }}</th>
<th style="width: 20%" scope="col">{{ 'Time'|trans }}</th>
</tr>
</thead>
<tbody>
{%~ for candidate in result ~%}
<tr class="table-{% if loop.revindex > quiz.dropouts %}success{% else %}danger{% endif %}">
<td>{{ candidate.name }}</td>
<td>{{ candidate.correct|default('0') }}</td>
<td>
<form method="post"
action="{{ path('tvdt_backoffice_modify_correction', {quiz: quiz.id, candidate: candidate.id}) }}">
<div class="row">
<div class="col-8">
<input class="form-control form-control-sm" type="number"
value="{{ candidate.corrections }}" step="0.5"
name="corrections">
</div>
<div class="col-2">
<button class="btn btn-sm btn-primary" type="submit">{{ 'Save'|trans }}</button>
</div>
</div>
</form>
</td>
<td>
<form method="post"
action="{{ path('tvdt_backoffice_modify_penalty', {quiz: quiz.id, candidate: candidate.id}) }}">
<input type="hidden" name="_token" value="{{ csrf_token('candidate_answer') }}">
<div class="row">
<div class="col-8">
<input class="form-control form-control-sm" type="number"
value="{{ candidate.penaltySeconds }}" step="1"
name="penalty">
</div>
<div class="col-2">
<button class="btn btn-sm btn-primary" type="submit">{{ 'Save'|trans }}</button>
</div>
</div>
</form>
</td>
<td>{{ candidate.score|default('x') }}</td>
<td>{{ candidate.time.format('%i:%S') }}</td>
</tr>
{% else %}
<tr>
<td colspan="5">{{ 'No results'|trans }}</td>
</tr>
{% endfor %}
</tbody>
</table>
+5 -13
View File
@@ -1,19 +1,9 @@
{% extends 'backoffice/base.html.twig' %} {% extends 'backoffice/base.html.twig' %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">{{ 'Home'|trans }}</a></li>
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_season', {seasonCode: season.seasonCode}) }}">{{ season.name }}</a></li>
<li class="breadcrumb-item active" aria-current="page">{{ 'Add Quiz'|trans }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
<div class="row"> <div class="row">
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<h2 class="mb-3">{{ t('Add a quiz to %name%', {'%name%': season.name})|trans }} </h2> <h2 class="py-2">{{ t('Add a quiz to %name%', {'%name%': season.name})|trans }} </h2>
{{ form_start(form) }} {{ form_start(form) }}
{{ form_row(form.name) }} {{ form_row(form.name) }}
{{ form_row(form.sheet) }} {{ form_row(form.sheet) }}
@@ -21,11 +11,13 @@
{{ form_end(form) }} {{ form_end(form) }}
</div> </div>
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<p class="mb-3"> <p class="pt-5">
{{ 'Help text for adding a quiz'|trans }} Hier kan nog tekst komen met wat uitleg
</p> </p>
</div> </div>
</div> </div>
{% endblock %} {% endblock %}
{% block title %} {% block title %}
+10 -20
View File
@@ -1,47 +1,37 @@
{% extends 'backoffice/base.html.twig' %} {% extends 'backoffice/base.html.twig' %}
{% block title %}{{ parent() }}{{ season.name }}{% endblock %} {% block title %}{{ parent() }}{{ season.name }}{% endblock %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">{{ 'Home'|trans }}</a></li>
<li class="breadcrumb-item active" aria-current="page">{{ season.name }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
<h2 class="mb-3">{{ 'Season'|trans }}: {{ season.name }}</h2> <h2 class="py-2">{{ 'Season'|trans }}: {{ season.name }}</h2>
<div class="row"> <div class="row">
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<div class="d-flex flex-row align-items-center mb-3"> <div class="d-flex flex-row align-items-center">
<h4 class="mb-0 pe-2">{{ 'Quizzes'|trans }}</h4> <h4 class="py-2 pe-2">{{ 'Quizzes'|trans }}</h4>
<a class="link" <a class="link"
href="{{ path('tvdt_backoffice_quiz_add', {seasonCode: season.seasonCode}) }}">{{ 'Add'|trans }}</a> href="{{ path('tvdt_backoffice_quiz_add', {seasonCode: season.seasonCode}) }}">{{ 'Add'|trans }}</a>
</div> </div>
<div class="list-group mb-3"> <div class="list-group">
{% for quiz in season.quizzes %} {% for quiz in season.quizzes %}
<a class="list-group-item list-group-item-action{% if season.activeQuiz == quiz %} active{% endif %}" <a class="list-group-item list-group-item-action{% if season.activeQuiz == quiz %} active{% endif %}"
href="{{ path('tvdt_backoffice_quiz', {seasonCode: season.seasonCode, quiz: quiz.id}) }}">{{ quiz.name }}</a> href="{{ path('tvdt_backoffice_quiz', {seasonCode: season.seasonCode, quiz: quiz.id}) }}">{{ quiz.name }}</a>
{% else %} {% else %}
{{ 'No quizzes'|trans }} No quizzes
{% endfor %} {% endfor %}
</div> </div>
</div> </div>
<div class="col-md-3 col-12"> <div class="col-md-3 col-12">
<div class="d-flex flex-row align-items-center mb-3"> <div class="d-flex flex-row align-items-center">
<h4 class="mb-0 pe-2">{{ 'Candidates'|trans }}</h4> <h4 class="py-2 pe-2">{{ 'Candidates'|trans }}</h4>
<a class="link" <a class="link"
href="{{ path('tvdt_backoffice_add_candidates', {seasonCode: season.seasonCode}) }}">{{ 'Add Candidate'|trans }} href="{{ path('tvdt_backoffice_add_candidates', {seasonCode: season.seasonCode}) }}">{{ 'Add Candidate'|trans }}
</a> </a>
</div> </div>
<ul class="mb-3"> <ul>
{% for candidate in season.candidates %} {% for candidate in season.candidates %}
<li>{{ candidate.name }}</li>{% endfor %} <li>{{ candidate.name }}</li>{% endfor %}
</ul> </ul>
<div class="d-flex flex-row align-items-center mb-3"> <div class="d-flex flex-row align-items-center">
<h4 class="mb-0 pe-2">{{ 'Settings'|trans }}</h4> <h4 class="py-2 pe-2">{{ 'Settings'|trans }}</h4>
</div> </div>
{{ form(form) }} {{ form(form) }}
</div> </div>
+3 -12
View File
@@ -1,26 +1,17 @@
{% extends 'backoffice/base.html.twig' %} {% extends 'backoffice/base.html.twig' %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">{{ 'Home'|trans }}</a></li>
<li class="breadcrumb-item active" aria-current="page">{{ 'Create a season'|trans }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
<div class="row"> <div class="row">
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<h2 class="mb-3">{{ 'Create a season'|trans }}</h2> <h2 class="py-2">{{ 'Create a season'|trans }}</h2>
{{ form_start(form) }} {{ form_start(form) }}
{{ form_row(form.name) }} {{ form_row(form.name) }}
<button type="submit" class="btn btn-primary">{{ 'Submit'|trans }}</button> <button type="submit" class="btn btn-primary">{{ 'Submit'|trans }}</button>
{{ form_end(form) }} {{ form_end(form) }}
</div> </div>
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<p class="mb-3"> <p class="pt-5">
{{ 'Help text for creating a season'|trans }} Hier kan nog tekst komen met wat uitleg
</p> </p>
</div> </div>
</div> </div>
@@ -1,27 +1,17 @@
{% extends 'backoffice/base.html.twig' %} {% extends 'backoffice/base.html.twig' %}
{% block breadcrumbs %}
<nav aria-label="breadcrumb" class="mb-3">
<ol class="breadcrumb">
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_index') }}">{{ 'Home'|trans }}</a></li>
<li class="breadcrumb-item"><a href="{{ path('tvdt_backoffice_season', {seasonCode: season.seasonCode}) }}">{{ season.name }}</a></li>
<li class="breadcrumb-item active" aria-current="page">{{ 'Add Candidates'|trans }}</li>
</ol>
</nav>
{% endblock %}
{% block body %} {% block body %}
<div class="row"> <div class="row">
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<h2 class="mb-3">{{ 'Add Candidates'|trans }}</h2> <h2 class="py-2">{{ 'Add Candidates'|trans }}</h2>
{{ form_start(form) }} {{ form_start(form) }}
{{ form_row(form.candidates) }} {{ form_row(form.candidates) }}
<button type="submit" class="btn btn-primary">{{ 'Submit'|trans }}</button> <button type="submit" class="btn btn-primary">{{ 'Submit'|trans }}</button>
{{ form_end(form) }} {{ form_end(form) }}
</div> </div>
<div class="col-md-6 col-12"> <div class="col-md-6 col-12">
<p class="mb-3"> <p class="pt-5">
{{ 'Help text for adding candidates'|trans }} Hier kan nog tekst komen met wat uitleg
</p> </p>
</div> </div>
</div> </div>
+2 -1
View File
@@ -10,7 +10,8 @@
<title> <title>
{% block title %}Tijd voor de test{% endblock title %} {% block title %}Tijd voor de test{% endblock title %}
</title> </title>
{% block importmap %}{% endblock %} {% block stylesheets %}{% endblock %}
{% block javascripts %}{% block importmap %}{% endblock %}{% endblock %}
</head> </head>
<body> <body>
{% block nav %} {% block nav %}
-12
View File
@@ -61,8 +61,6 @@ final class QuizRepositoryTest extends DatabaseTestCase
// Start Quiz // Start Quiz
$qc = new QuizCandidate($quiz, $candidate); $qc = new QuizCandidate($quiz, $candidate);
$qc->started = $clock->now();
$this->entityManager->persist($qc); $this->entityManager->persist($qc);
$this->entityManager->flush(); $this->entityManager->flush();
@@ -101,11 +99,7 @@ final class QuizRepositoryTest extends DatabaseTestCase
$this->assertInstanceOf(Quiz::class, $quiz); $this->assertInstanceOf(Quiz::class, $quiz);
$qc1 = new QuizCandidate($quiz, $candidate1); $qc1 = new QuizCandidate($quiz, $candidate1);
$qc1->started = $clock->now();
$qc2 = new QuizCandidate($quiz, $candidate2); $qc2 = new QuizCandidate($quiz, $candidate2);
$qc2->started = $clock->now();
$this->entityManager->persist($qc1); $this->entityManager->persist($qc1);
$this->entityManager->persist($qc2); $this->entityManager->persist($qc2);
$this->entityManager->flush(); $this->entityManager->flush();
@@ -148,8 +142,6 @@ final class QuizRepositoryTest extends DatabaseTestCase
$this->assertInstanceOf(Quiz::class, $quiz); $this->assertInstanceOf(Quiz::class, $quiz);
$qc = new QuizCandidate($quiz, $candidate); $qc = new QuizCandidate($quiz, $candidate);
$qc->started = $clock->now();
$this->entityManager->persist($qc); $this->entityManager->persist($qc);
$this->entityManager->flush(); $this->entityManager->flush();
@@ -186,13 +178,9 @@ final class QuizRepositoryTest extends DatabaseTestCase
$this->assertInstanceOf(Quiz::class, $quiz); $this->assertInstanceOf(Quiz::class, $quiz);
$qc1 = new QuizCandidate($quiz, $candidate1); $qc1 = new QuizCandidate($quiz, $candidate1);
$qc1->started = $clock->now();
$this->entityManager->persist($qc1); $this->entityManager->persist($qc1);
$clock->sleep(10); $clock->sleep(10);
$qc2 = new QuizCandidate($quiz, $candidate2); $qc2 = new QuizCandidate($quiz, $candidate2);
$qc2->started = $clock->now();
$this->entityManager->persist($qc2); $this->entityManager->persist($qc2);
$this->entityManager->flush(); $this->entityManager->flush();
+1 -1
View File
@@ -6,8 +6,8 @@ namespace Tvdt\Tests\Repository;
use PHPUnit\Framework\Attributes\CoversClass; use PHPUnit\Framework\Attributes\CoversClass;
use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface; use Symfony\Component\PasswordHasher\Hasher\UserPasswordHasherInterface;
use Tvdt\DataFixtures\TestFixtures;
use Tvdt\Repository\UserRepository; use Tvdt\Repository\UserRepository;
use Tvdt\DataFixtures\TestFixtures;
use function PHPUnit\Framework\assertEmpty; use function PHPUnit\Framework\assertEmpty;
-42
View File
@@ -1,42 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<xliff xmlns="urn:oasis:names:tc:xliff:document:1.2" version="1.2">
<file source-language="nl" target-language="nl" datatype="plaintext" original="file.ext">
<header>
<tool tool-id="symfony" tool-name="Symfony"/>
</header>
<body>
<trans-unit id="E69bPZ6" resname="%count% day|%count% days">
<source>%count% day|%count% days</source>
<target>%count% dag|%count% dagen</target>
</trans-unit>
<trans-unit id="H3jnVKg" resname="%count% hour|%count% hours">
<source>%count% hour|%count% hours</source>
<target>%count% uur|%count% uren</target>
</trans-unit>
<trans-unit id="30tji6_" resname="%count% minute|%count% minutes">
<source>%count% minute|%count% minutes</source>
<target>%count% minuut|%count% minuten</target>
</trans-unit>
<trans-unit id="NCe6niW" resname="%count% month|%count% months">
<source>%count% month|%count% months</source>
<target>%count% maand|%count% maanden</target>
</trans-unit>
<trans-unit id="U_NoHhq" resname="%count% year|%count% years">
<source>%count% year|%count% years</source>
<target>%count% jaar|%count% jaar</target>
</trans-unit>
<trans-unit id="GTlrnoK" resname="The link to verify your email appears to be for a different account or email. Please request a new link.">
<source>The link to verify your email appears to be for a different account or email. Please request a new link.</source>
<target>De link om je e-mailadres te verifiëren is voor een andere gebruiker of e-mailadres. Vraag een nieuwe link aan.</target>
</trans-unit>
<trans-unit id="F0PcQoi" resname="The link to verify your email has expired. Please request a new link.">
<source>The link to verify your email has expired. Please request a new link.</source>
<target>De link om je e-mailadres te verifiëren is verlopen. Vraag een nieuwe link aan.</target>
</trans-unit>
<trans-unit id="dSF4hGL" resname="The link to verify your email is invalid. Please request a new link.">
<source>The link to verify your email is invalid. Please request a new link.</source>
<target>De link om je e-mailadres te verifiëren is ongeldig. Vraag een nieuwe link aan.</target>
</trans-unit>
</body>
</file>
</xliff>
+13 -129
View File
@@ -5,18 +5,6 @@
<tool tool-id="symfony" tool-name="Symfony"/> <tool tool-id="symfony" tool-name="Symfony"/>
</header> </header>
<body> <body>
<trans-unit id="1_MZEgP" resname="Actions">
<source>Actions</source>
<target>Acties</target>
</trans-unit>
<trans-unit id="dRNHWgg" resname="Activate">
<source>Activate</source>
<target>Activeren</target>
</trans-unit>
<trans-unit id="JjO6Nuw" resname="Active">
<source>Active</source>
<target>Actief</target>
</trans-unit>
<trans-unit id="uyMngrK" resname="Active Quiz"> <trans-unit id="uyMngrK" resname="Active Quiz">
<source>Active Quiz</source> <source>Active Quiz</source>
<target>Actieve test</target> <target>Actieve test</target>
@@ -33,10 +21,6 @@
<source>Add Candidates</source> <source>Add Candidates</source>
<target>Voeg kandidaten toe</target> <target>Voeg kandidaten toe</target>
</trans-unit> </trans-unit>
<trans-unit id="g4GCvSW" resname="Add Quiz">
<source>Add Quiz</source>
<target>Test toevoegen</target>
</trans-unit>
<trans-unit id="ehB6pAw" resname="Add a quiz to %name%"> <trans-unit id="ehB6pAw" resname="Add a quiz to %name%">
<source>Add a quiz to %name%</source> <source>Add a quiz to %name%</source>
<target>Voeg een test toe aan %name%</target> <target>Voeg een test toe aan %name%</target>
@@ -49,17 +33,13 @@
<source>Already have an account? Log in</source> <source>Already have an account? Log in</source>
<target>Heb je al een account? Log in</target> <target>Heb je al een account? Log in</target>
</trans-unit> </trans-unit>
<trans-unit id="3A2JPqn" resname="Answer Mapping">
<source>Answer Mapping</source>
<target>Antwoord-kandidaat koppeling</target>
</trans-unit>
<trans-unit id="Qu1euq_" resname="Are you sure you want to clear all the results? This will also delete al the eliminations."> <trans-unit id="Qu1euq_" resname="Are you sure you want to clear all the results? This will also delete al the eliminations.">
<source>Are you sure you want to clear all the results? This will also delete all the eliminations.</source> <source>Are you sure you want to clear all the results? This will also delete al the eliminations.</source>
<target>Weet je zeker dat je de resultaten wilt leegmaken? Dit gooit ook alle eliminaties weg.</target> <target>Weet je zeker datatype je de resultaten will leegmaken? Dit gooit ook alle eliminaties weg.</target>
</trans-unit> </trans-unit>
<trans-unit id="Ec4twG8" resname="Are you sure you want to delete this quiz?"> <trans-unit id="Ec4twG8" resname="Are you sure you want to delete this quiz?">
<source>Are you sure you want to delete this quiz?</source> <source>Are you sure you want to delete this quiz?</source>
<target>Weet je zeker dat je deze test wilt verwijderen?</target> <target>Weet je zeker datatype je deze test will verwijderen?</target>
</trans-unit> </trans-unit>
<trans-unit id=".QFPbFe" resname="Back"> <trans-unit id=".QFPbFe" resname="Back">
<source>Back</source> <source>Back</source>
@@ -69,38 +49,18 @@
<source>Candidate</source> <source>Candidate</source>
<target>Kandidaat</target> <target>Kandidaat</target>
</trans-unit> </trans-unit>
<trans-unit id="LMsl4pc" resname="Candidate Status">
<source>Candidate Status</source>
<target>Kandidaatstatus</target>
</trans-unit>
<trans-unit id="Wk50LUM" resname="Candidate answers saved">
<source>Candidate answers saved</source>
<target>Kandidaatantwoorden opgeslagen</target>
</trans-unit>
<trans-unit id="TiTLBGW" resname="Candidate not found"> <trans-unit id="TiTLBGW" resname="Candidate not found">
<source>Candidate not found</source> <source>Candidate not found</source>
<target>Kandidaat niet gevonden</target> <target>Kandidaat niet gevonden</target>
</trans-unit> </trans-unit>
<trans-unit id="6QiGbuz" resname="Candidate status updated">
<source>Candidate status updated</source>
<target>Kandidaatstatus bijgewerkt</target>
</trans-unit>
<trans-unit id="WJJE4q_" resname="Candidates"> <trans-unit id="WJJE4q_" resname="Candidates">
<source>Candidates</source> <source>Candidates</source>
<target>Kandidaten</target> <target>Kandidaten</target>
</trans-unit> </trans-unit>
<trans-unit id="J1c2y63" resname="Clear Quiz..."> <trans-unit id="FNY513f" resname="Clear quiz...">
<source>Clear Quiz...</source> <source>Clear quiz...</source>
<target>Test leegmaken...</target> <target>Test leegmaken...</target>
</trans-unit> </trans-unit>
<trans-unit id="5LgCFqF" resname="Completed">
<source>Completed</source>
<target>Voltooid</target>
</trans-unit>
<trans-unit id="7sfvWUb" resname="Confirm Answers">
<source>Confirm Answers</source>
<target>Bevestig antwoorden</target>
</trans-unit>
<trans-unit id="sFpB4C2" resname="Correct Answers"> <trans-unit id="sFpB4C2" resname="Correct Answers">
<source>Correct Answers</source> <source>Correct Answers</source>
<target>Goede antwoorden</target> <target>Goede antwoorden</target>
@@ -125,10 +85,6 @@
<source>Create an account</source> <source>Create an account</source>
<target>Maak een account aan</target> <target>Maak een account aan</target>
</trans-unit> </trans-unit>
<trans-unit id="S5P7nQd" resname="Deactivate">
<source>Deactivate</source>
<target>Deactiveren</target>
</trans-unit>
<trans-unit id="w9AyAnn" resname="Deactivate Quiz"> <trans-unit id="w9AyAnn" resname="Deactivate Quiz">
<source>Deactivate Quiz</source> <source>Deactivate Quiz</source>
<target>Deactiveer test</target> <target>Deactiveer test</target>
@@ -159,40 +115,12 @@
</trans-unit> </trans-unit>
<trans-unit id="HNMwvRn" resname="Error clearing quiz"> <trans-unit id="HNMwvRn" resname="Error clearing quiz">
<source>Error clearing quiz</source> <source>Error clearing quiz</source>
<target>Fout bij het leegmaken van de test</target> <target>Fout bij leegmaken test</target>
</trans-unit> </trans-unit>
<trans-unit id="OGiIhMH" resname="Green"> <trans-unit id="OGiIhMH" resname="Green">
<source>Green</source> <source>Green</source>
<target>Groen</target> <target>Groen</target>
</trans-unit> </trans-unit>
<trans-unit id="b7rTx0P" resname="Help text for adding a quiz">
<source>Help text for adding a quiz</source>
<target>Upload een XLSX-bestand met vragen en antwoorden voor je test.</target>
</trans-unit>
<trans-unit id="0byYjDw" resname="Help text for adding candidates">
<source>Help text for adding candidates</source>
<target>Voeg kandidaten toe aan dit seizoen. Eén naam per regel.</target>
</trans-unit>
<trans-unit id="SipST._" resname="Help text for creating a season">
<source>Help text for creating a season</source>
<target>Maak een nieuw seizoen aan om tests en kandidaten te beheren.</target>
</trans-unit>
<trans-unit id="Q0rWWbg" resname="Help text for preparing elimination">
<source>Help text for preparing elimination</source>
<target>Kies welke kandidaten groen of rood krijgen voor de eliminatie.</target>
</trans-unit>
<trans-unit id="ExFLJqx" resname="Home">
<source>Home</source>
<target>Home</target>
</trans-unit>
<trans-unit id="8n2mlHc" resname="In Progress">
<source>In Progress</source>
<target>Bezig</target>
</trans-unit>
<trans-unit id="ybqiXt5" resname="Inactive">
<source>Inactive</source>
<target>Inactief</target>
</trans-unit>
<trans-unit id="k1X7w12" resname="Invalid season code"> <trans-unit id="k1X7w12" resname="Invalid season code">
<source>Invalid season code</source> <source>Invalid season code</source>
<target>Ongeldige seizoencode</target> <target>Ongeldige seizoencode</target>
@@ -221,53 +149,37 @@
<source>Name</source> <source>Name</source>
<target>Naam</target> <target>Naam</target>
</trans-unit> </trans-unit>
<trans-unit id="gefhnBC" resname="Next">
<source>Next</source>
<target>Volgende</target>
</trans-unit>
<trans-unit id="wd1MvZW" resname="No"> <trans-unit id="wd1MvZW" resname="No">
<source>No</source> <source>No</source>
<target>Nee</target> <target>Nee</target>
</trans-unit> </trans-unit>
<trans-unit id="gefhnBC" resname="Next">
<source>Next</source>
<target>Volgende</target>
</trans-unit>
<trans-unit id="nOHriCl" resname="No active quiz"> <trans-unit id="nOHriCl" resname="No active quiz">
<source>No active quiz</source> <source>No active quiz</source>
<target>Geen actieve test</target> <target>Geen actieve test</target>
</trans-unit> </trans-unit>
<trans-unit id="oNXT2zu" resname="No quizzes">
<source>No quizzes</source>
<target>Geen tests</target>
</trans-unit>
<trans-unit id="swW4qFE" resname="No results"> <trans-unit id="swW4qFE" resname="No results">
<source>No results</source> <source>No results</source>
<target>Geen resultaten</target> <target>Geen resultaten</target>
</trans-unit> </trans-unit>
<trans-unit id="tbd1luF" resname="Not Started">
<source>Not Started</source>
<target>Niet gestart</target>
</trans-unit>
<trans-unit id="k7Eqnjt" resname="Number of dropouts:"> <trans-unit id="k7Eqnjt" resname="Number of dropouts:">
<source>Number of dropouts:</source> <source>Number of dropouts:</source>
<target>Aantal afvallers:</target> <target>Aantal afvallers:</target>
</trans-unit> </trans-unit>
<trans-unit id="HmgPmMV" resname="Overview">
<source>Overview</source>
<target>Overzicht</target>
</trans-unit>
<trans-unit id="PywqOf4" resname="Owner(s)"> <trans-unit id="PywqOf4" resname="Owner(s)">
<source>Owner(s)</source> <source>Owner(s)</source>
<target>Eigenaar(s)</target> <target>Eigena(a)r(en)</target>
</trans-unit> </trans-unit>
<trans-unit id="GqmFSHc" resname="Password"> <trans-unit id="GqmFSHc" resname="Password">
<source>Password</source> <source>Password</source>
<target>Wachtwoord</target> <target>Wachtwoord</target>
</trans-unit> </trans-unit>
<trans-unit id="1ne1Zlc" resname="Penalty">
<source>Penalty</source>
<target>Straftijd</target>
</trans-unit>
<trans-unit id="VbgD9L8" resname="Please Confirm"> <trans-unit id="VbgD9L8" resname="Please Confirm">
<source>Please Confirm</source> <source>Please Confirm</source>
<target>Bevestig alsjeblieft</target> <target>Bevestig Alsjeblieft</target>
</trans-unit> </trans-unit>
<trans-unit id="6EclFME" resname="Please Confirm your Email"> <trans-unit id="6EclFME" resname="Please Confirm your Email">
<source>Please Confirm your Email</source> <source>Please Confirm your Email</source>
@@ -285,14 +197,6 @@
<source>Prepare Custom Elimination</source> <source>Prepare Custom Elimination</source>
<target>Bereid aangepaste eliminatie voor</target> <target>Bereid aangepaste eliminatie voor</target>
</trans-unit> </trans-unit>
<trans-unit id="xe_UxWT" resname="Prepare Elimination">
<source>Prepare Elimination</source>
<target>Bereid eliminatie voor</target>
</trans-unit>
<trans-unit id="ouNrIYq" resname="Previous">
<source>Previous</source>
<target>Vorige</target>
</trans-unit>
<trans-unit id="Rx5irUP" resname="Questions"> <trans-unit id="Rx5irUP" resname="Questions">
<source>Questions</source> <source>Questions</source>
<target>Vragen</target> <target>Vragen</target>
@@ -309,10 +213,6 @@
<source>Quiz Added!</source> <source>Quiz Added!</source>
<target>Test toegevoegd!</target> <target>Test toegevoegd!</target>
</trans-unit> </trans-unit>
<trans-unit id="IwEF7a0" resname="Quiz Status">
<source>Quiz Status</source>
<target>Teststatus</target>
</trans-unit>
<trans-unit id="vXN8b2w" resname="Quiz cleared"> <trans-unit id="vXN8b2w" resname="Quiz cleared">
<source>Quiz cleared</source> <source>Quiz cleared</source>
<target>Test leeggemaakt</target> <target>Test leeggemaakt</target>
@@ -349,17 +249,13 @@
<source>Repeat Password</source> <source>Repeat Password</source>
<target>Herhaal wachtwoord</target> <target>Herhaal wachtwoord</target>
</trans-unit> </trans-unit>
<trans-unit id="7UvBPrb" resname="Results &amp; Elimination">
<source>Results &amp; Elimination</source>
<target><![CDATA[Resultaat & Eliminatie]]></target>
</trans-unit>
<trans-unit id="z9OKodR" resname="Save"> <trans-unit id="z9OKodR" resname="Save">
<source>Save</source> <source>Save</source>
<target>Opslaan</target> <target>Opslaan</target>
</trans-unit> </trans-unit>
<trans-unit id="8HUcmWU" resname="Save and start elimination"> <trans-unit id="8HUcmWU" resname="Save and start elimination">
<source>Save and start elimination</source> <source>Save and start elimination</source>
<target>Opslaan en eliminatie starten</target> <target>Opslaan en start eliminatie</target>
</trans-unit> </trans-unit>
<trans-unit id="uRWqG15" resname="Score"> <trans-unit id="uRWqG15" resname="Score">
<source>Score</source> <source>Score</source>
@@ -385,18 +281,10 @@
<source>Settings</source> <source>Settings</source>
<target>Instellingen</target> <target>Instellingen</target>
</trans-unit> </trans-unit>
<trans-unit id="tKaIbhp" resname="Show Numbers">
<source>Show Numbers</source>
<target>Toon nummers</target>
</trans-unit>
<trans-unit id="pNIxNSX" resname="Sign in"> <trans-unit id="pNIxNSX" resname="Sign in">
<source>Sign in</source> <source>Sign in</source>
<target>Log in</target> <target>Log in</target>
</trans-unit> </trans-unit>
<trans-unit id="6xCSWiZ" resname="Status">
<source>Status</source>
<target>Status</target>
</trans-unit>
<trans-unit id="9m8DOBg" resname="Submit"> <trans-unit id="9m8DOBg" resname="Submit">
<source>Submit</source> <source>Submit</source>
<target>Verstuur</target> <target>Verstuur</target>
@@ -421,10 +309,6 @@
<source>Yes</source> <source>Yes</source>
<target>Ja</target> <target>Ja</target>
</trans-unit> </trans-unit>
<trans-unit id="9A8F6VB" resname="You are not allowed to answer this quiz">
<source>You are not allowed to answer this quiz</source>
<target>Je mag deze test niet beantwoorden</target>
</trans-unit>
<trans-unit id="0afY1NF" resname="You have no seasons yet."> <trans-unit id="0afY1NF" resname="You have no seasons yet.">
<source>You have no seasons yet.</source> <source>You have no seasons yet.</source>
<target>Je hebt nog geen seizoenen.</target> <target>Je hebt nog geen seizoenen.</target>
-86
View File
@@ -1,86 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<xliff xmlns="urn:oasis:names:tc:xliff:document:1.2" version="1.2">
<file source-language="nl" target-language="nl" datatype="plaintext" original="file.ext">
<header>
<tool tool-id="symfony" tool-name="Symfony"/>
</header>
<body>
<trans-unit id="Sjc.F3G" resname="Account has expired.">
<source>Account has expired.</source>
<target>Account is verlopen.</target>
</trans-unit>
<trans-unit id=".WYXRk3" resname="Account is disabled.">
<source>Account is disabled.</source>
<target>Account is gedeactiveerd.</target>
</trans-unit>
<trans-unit id="Gp85C7l" resname="Account is locked.">
<source>Account is locked.</source>
<target>Account is geblokkeerd.</target>
</trans-unit>
<trans-unit id="lDjpeYu" resname="An authentication exception occurred.">
<source>An authentication exception occurred.</source>
<target>Er heeft zich een authenticatieprobleem voorgedaan.</target>
</trans-unit>
<trans-unit id="AJOf7xb" resname="Authentication credentials could not be found.">
<source>Authentication credentials could not be found.</source>
<target>Authenticatiegegevens konden niet worden gevonden.</target>
</trans-unit>
<trans-unit id="DPttBWR" resname="Authentication request could not be processed due to a system problem.">
<source>Authentication request could not be processed due to a system problem.</source>
<target>Authenticatieaanvraag kon niet worden verwerkt door een technisch probleem.</target>
</trans-unit>
<trans-unit id="D8oD9D6" resname="Cookie has already been used by someone else.">
<source>Cookie has already been used by someone else.</source>
<target>Cookie is al door een ander persoon gebruikt.</target>
</trans-unit>
<trans-unit id="99yINK3" resname="Credentials have expired.">
<source>Credentials have expired.</source>
<target>Authenticatiegegevens zijn verlopen.</target>
</trans-unit>
<trans-unit id="bbMNxJD" resname="Invalid CSRF token.">
<source>Invalid CSRF token.</source>
<target>CSRF-code is ongeldig.</target>
</trans-unit>
<trans-unit id="6ZX9PdF" resname="Invalid credentials.">
<source>Invalid credentials.</source>
<target>Ongeldige inloggegevens.</target>
</trans-unit>
<trans-unit id="Tz4Owtv" resname="Invalid or expired login link.">
<source>Invalid or expired login link.</source>
<target>Ongeldige of verlopen inloglink.</target>
</trans-unit>
<trans-unit id="V6vcZYp" resname="No authentication provider found to support the authentication token.">
<source>No authentication provider found to support the authentication token.</source>
<target>Geen authenticatieprovider gevonden die de authenticatietoken ondersteunt.</target>
</trans-unit>
<trans-unit id="mvKBTGQ" resname="No session available, it either timed out or cookies are not enabled.">
<source>No session available, it either timed out or cookies are not enabled.</source>
<target>Geen sessie beschikbaar, mogelijk is deze verlopen of cookies zijn uitgeschakeld.</target>
</trans-unit>
<trans-unit id="BMgfoWp" resname="No token could be found.">
<source>No token could be found.</source>
<target>Er kon geen authenticatietoken worden gevonden.</target>
</trans-unit>
<trans-unit id="6rEex5P" resname="Not privileged to request the resource.">
<source>Not privileged to request the resource.</source>
<target>Onvoldoende rechten om de aanvraag te verwerken.</target>
</trans-unit>
<trans-unit id="CPhXNYL" resname="Too many failed login attempts, please try again in %minutes% minute.">
<source>Too many failed login attempts, please try again in %minutes% minute.</source>
<target>Te veel onjuiste inlogpogingen, probeer het opnieuw over %minutes% minuut.</target>
</trans-unit>
<trans-unit id="KpyCIMn" resname="Too many failed login attempts, please try again in %minutes% minutes.">
<source>Too many failed login attempts, please try again in %minutes% minutes.</source>
<target>Te veel onjuiste inlogpogingen, probeer het opnieuw over %minutes% minuten.</target>
</trans-unit>
<trans-unit id="PvhvanK" resname="Too many failed login attempts, please try again later.">
<source>Too many failed login attempts, please try again later.</source>
<target>Te veel onjuiste inlogpogingen, probeer het later nogmaals.</target>
</trans-unit>
<trans-unit id="VZija00" resname="Username could not be found.">
<source>Username could not be found.</source>
<target>Gebruikersnaam kon niet worden gevonden.</target>
</trans-unit>
</body>
</file>
</xliff>
-680
View File
@@ -5,698 +5,18 @@
<tool tool-id="symfony" tool-name="Symfony"/> <tool tool-id="symfony" tool-name="Symfony"/>
</header> </header>
<body> <body>
<trans-unit id="_x64C6y" resname="A PHP extension caused the upload to fail.">
<source>A PHP extension caused the upload to fail.</source>
<target>De upload is mislukt vanwege een PHP-extensie.</target>
</trans-unit>
<trans-unit id="L9tKe2y" resname="An empty file is not allowed.">
<source>An empty file is not allowed.</source>
<target>Lege bestanden zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="ivltd6_" resname="Cannot write temporary file to disk.">
<source>Cannot write temporary file to disk.</source>
<target>Kan het tijdelijke bestand niet wegschrijven op de schijf.</target>
</trans-unit>
<trans-unit id="lxL.CBF" resname="Each element of this collection should satisfy its own set of constraints.">
<source>Each element of this collection should satisfy its own set of constraints.</source>
<target>Elk element van deze collectie moet voldoen aan zijn eigen set voorwaarden.</target>
</trans-unit>
<trans-unit id="B9r0Hn5" resname="Error"> <trans-unit id="B9r0Hn5" resname="Error">
<source>Error</source> <source>Error</source>
<target>Fout</target> <target>Fout</target>
</trans-unit> </trans-unit>
<trans-unit id="Ia5ts7h" resname="Invalid card number.">
<source>Invalid card number.</source>
<target>Ongeldig creditcardnummer.</target>
</trans-unit>
<trans-unit id="DpnrQ0a" resname="Mixing numbers from different scripts is not allowed.">
<source>Mixing numbers from different scripts is not allowed.</source>
<target>Het mengen van cijfers uit verschillende schriften is niet toegestaan.</target>
</trans-unit>
<trans-unit id="qUKi9W_" resname="No file was uploaded.">
<source>No file was uploaded.</source>
<target>Er is geen bestand geüpload.</target>
</trans-unit>
<trans-unit id="MRZ.kUM" resname="No temporary folder was configured in php.ini.">
<source>No temporary folder was configured in php.ini.</source>
<target>Er is geen tijdelijke map geconfigureerd in php.ini, of de gespecificeerde map bestaat niet.</target>
</trans-unit>
<trans-unit id="bV9ETHe" resname="One or more of the given values is invalid.">
<source>One or more of the given values is invalid.</source>
<target>Eén of meer van de ingegeven waarden zijn ongeldig.</target>
</trans-unit>
<trans-unit id="gwI.OzY" resname="Please choose a valid date interval.">
<source>Please choose a valid date interval.</source>
<target>Kies een geldig tijdinterval.</target>
</trans-unit>
<trans-unit id="c5ToMGG" resname="Please enter a number.">
<source>Please enter a number.</source>
<target>Vul een geldig getal in.</target>
</trans-unit>
<trans-unit id="B7yoMOR" resname="Please enter a password"> <trans-unit id="B7yoMOR" resname="Please enter a password">
<source>Please enter a password</source> <source>Please enter a password</source>
<target>Voer je wachtwoord in</target> <target>Voer je wachtwoord in</target>
</trans-unit> </trans-unit>
<trans-unit id="HviZ2Cc" resname="Please enter a percentage value.">
<source>Please enter a percentage value.</source>
<target>Vul een geldig percentage in.</target>
</trans-unit>
<trans-unit id="ywSABzT" resname="Please enter a valid URL.">
<source>Please enter a valid URL.</source>
<target>Vul een geldige URL in.</target>
</trans-unit>
<trans-unit id="e_2ZUCK" resname="Please enter a valid birthdate.">
<source>Please enter a valid birthdate.</source>
<target>Vul een geldige geboortedatum in.</target>
</trans-unit>
<trans-unit id="yeUbmWc" resname="Please enter a valid date and time.">
<source>Please enter a valid date and time.</source>
<target>Vul een geldige datum en tijd in.</target>
</trans-unit>
<trans-unit id="eC1bGok" resname="Please enter a valid date.">
<source>Please enter a valid date.</source>
<target>Vul een geldige datum in.</target>
</trans-unit>
<trans-unit id="sv_zo_u" resname="Please enter a valid email address.">
<source>Please enter a valid email address.</source>
<target>Vul een geldig e-mailadres in.</target>
</trans-unit>
<trans-unit id="GoIKSyM" resname="Please enter a valid money amount.">
<source>Please enter a valid money amount.</source>
<target>Vul een geldig bedrag in.</target>
</trans-unit>
<trans-unit id="58j92KU" resname="Please enter a valid search term.">
<source>Please enter a valid search term.</source>
<target>Vul een geldige zoekterm in.</target>
</trans-unit>
<trans-unit id="beHTTAG" resname="Please enter a valid time.">
<source>Please enter a valid time.</source>
<target>Vul een geldige tijd in.</target>
</trans-unit>
<trans-unit id="2MEQpjY" resname="Please enter a valid week.">
<source>Please enter a valid week.</source>
<target>Vul een geldige week in.</target>
</trans-unit>
<trans-unit id="o.y86J1" resname="Please enter an integer.">
<source>Please enter an integer.</source>
<target>Vul een geldig getal in.</target>
</trans-unit>
<trans-unit id="9RtYCIm" resname="Please provide a valid phone number.">
<source>Please provide a valid phone number.</source>
<target>Vul een geldig telefoonnummer in.</target>
</trans-unit>
<trans-unit id="qiPYgGj" resname="Please select a valid color.">
<source>Please select a valid color.</source>
<target>Kies een geldige kleur.</target>
</trans-unit>
<trans-unit id="qf.GaaJ" resname="Please select a valid country.">
<source>Please select a valid country.</source>
<target>Kies een geldige landnaam.</target>
</trans-unit>
<trans-unit id="WtFFCQ8" resname="Please select a valid currency.">
<source>Please select a valid currency.</source>
<target>Kies een geldige valuta.</target>
</trans-unit>
<trans-unit id=".3DoUU." resname="Please select a valid file.">
<source>Please select a valid file.</source>
<target>Kies een geldig bestand.</target>
</trans-unit>
<trans-unit id="MhFFYoK" resname="Please select a valid language.">
<source>Please select a valid language.</source>
<target>Kies een geldige taal.</target>
</trans-unit>
<trans-unit id="LKAkY.h" resname="Please select a valid locale.">
<source>Please select a valid locale.</source>
<target>Kies een geldige locale.</target>
</trans-unit>
<trans-unit id="WGMBjaJ" resname="Please select a valid option.">
<source>Please select a valid option.</source>
<target>Kies een geldige optie.</target>
</trans-unit>
<trans-unit id="X1pJkwv" resname="Please select a valid range.">
<source>Please select a valid range.</source>
<target>Kies een geldig bereik.</target>
</trans-unit>
<trans-unit id="MDu2tNQ" resname="Please select a valid timezone.">
<source>Please select a valid timezone.</source>
<target>Vul een geldige tijdzone in.</target>
</trans-unit>
<trans-unit id="b_5.52u" resname="The CSRF token is invalid. Please try to resubmit the form.">
<source>The CSRF token is invalid. Please try to resubmit the form.</source>
<target>De CSRF-token is ongeldig. Probeer het formulier opnieuw te versturen.</target>
</trans-unit>
<trans-unit id="eXUouzq" resname="The checkbox has an invalid value.">
<source>The checkbox has an invalid value.</source>
<target>De checkbox heeft een incorrecte waarde.</target>
</trans-unit>
<trans-unit id="Fkuy1Kw" resname="The collection is invalid.">
<source>The collection is invalid.</source>
<target>Deze collectie is ongeldig.</target>
</trans-unit>
<trans-unit id="uyl6nQ7" resname="The detected character encoding is invalid ({{ detected }}). Allowed encodings are {{ encodings }}.">
<source>The detected character encoding is invalid ({{ detected }}). Allowed encodings are {{ encodings }}.</source>
<target>De gedetecteerde karaktercodering is ongeldig ({{ detected }}). De toegestane coderingen zijn {{ encodings }}.</target>
</trans-unit>
<trans-unit id="1seErdG" resname="The extension of the file is invalid ({{ extension }}). Allowed extensions are {{ extensions }}.">
<source>The extension of the file is invalid ({{ extension }}). Allowed extensions are {{ extensions }}.</source>
<target>De bestandsextensie is ongeldig ({{ extension }}). De toegestane extensies zijn {{ extensions }}.</target>
</trans-unit>
<trans-unit id="OW6NCiE" resname="The file could not be found.">
<source>The file could not be found.</source>
<target>Het bestand kon niet gevonden worden.</target>
</trans-unit>
<trans-unit id="DAd59Rr" resname="The file could not be uploaded.">
<source>The file could not be uploaded.</source>
<target>Het bestand kon niet worden geüpload.</target>
</trans-unit>
<trans-unit id="uuuT8w4" resname="The file is not readable.">
<source>The file is not readable.</source>
<target>Het bestand is niet leesbaar.</target>
</trans-unit>
<trans-unit id="xmVWVqA" resname="The file is too large ({{ size }} {{ suffix }}). Allowed maximum size is {{ limit }} {{ suffix }}.">
<source>The file is too large ({{ size }} {{ suffix }}). Allowed maximum size is {{ limit }} {{ suffix }}.</source>
<target>Het bestand is te groot ({{ size }} {{ suffix }}). De maximale grootte is {{ limit }} {{ suffix }}.</target>
</trans-unit>
<trans-unit id="OEeVt7G" resname="The file is too large.">
<source>The file is too large.</source>
<target>Het bestand is te groot.</target>
</trans-unit>
<trans-unit id="UPTnJIu" resname="The file is too large. Allowed maximum size is {{ limit }} {{ suffix }}.">
<source>The file is too large. Allowed maximum size is {{ limit }} {{ suffix }}.</source>
<target>Het bestand is te groot. De maximale grootte is {{ limit }} {{ suffix }}.</target>
</trans-unit>
<trans-unit id="H.ZNP57" resname="The file was only partially uploaded.">
<source>The file was only partially uploaded.</source>
<target>Het bestand is slechts gedeeltelijk geüpload.</target>
</trans-unit>
<trans-unit id="7db7kQD" resname="The filename is too long. It should have {{ filename_max_length }} character or less.|The filename is too long. It should have {{ filename_max_length }} characters or less.">
<source>The filename is too long. It should have {{ filename_max_length }} character or less.|The filename is too long. It should have {{ filename_max_length }} characters or less.</source>
<target>De bestandsnaam is te lang. Het moet {{ filename_max_length }} of minder karakters zijn.|De bestandsnaam is te lang. Het moet {{ filename_max_length }} of minder karakters zijn.</target>
</trans-unit>
<trans-unit id="wwjjKAn" resname="The hidden field is invalid.">
<source>The hidden field is invalid.</source>
<target>Het verborgen veld is incorrect.</target>
</trans-unit>
<trans-unit id="4421dQm" resname="The host could not be resolved.">
<source>The host could not be resolved.</source>
<target>De hostnaam kon niet worden gevonden.</target>
</trans-unit>
<trans-unit id="jAJTV_I" resname="The image file is corrupted.">
<source>The image file is corrupted.</source>
<target>Het afbeeldingsbestand is beschadigd.</target>
</trans-unit>
<trans-unit id="aw1rRIE" resname="The image has too few pixels ({{ pixels }} pixels). Minimum amount expected is {{ min_pixels }} pixels.">
<source>The image has too few pixels ({{ pixels }} pixels). Minimum amount expected is {{ min_pixels }} pixels.</source>
<target>De afbeelding heeft te weinig pixels ({{ pixels }}). Verwachte minimumhoeveelheid is {{ min_pixels }}.</target>
</trans-unit>
<trans-unit id="KbWb1FD" resname="The image has too many pixels ({{ pixels }} pixels). Maximum amount expected is {{ max_pixels }} pixels.">
<source>The image has too many pixels ({{ pixels }} pixels). Maximum amount expected is {{ max_pixels }} pixels.</source>
<target>De afbeelding heeft te veel pixels ({{ pixels }}). Het verwachte maximum is {{ max_pixels }}.</target>
</trans-unit>
<trans-unit id="zV9L7pv" resname="The image height is too big ({{ height }}px). Allowed maximum height is {{ max_height }}px.">
<source>The image height is too big ({{ height }}px). Allowed maximum height is {{ max_height }}px.</source>
<target>De afbeelding is te hoog ({{ height }}px). De maximale hoogte is {{ max_height }}px.</target>
</trans-unit>
<trans-unit id="A7vLGPA" resname="The image height is too small ({{ height }}px). Minimum height expected is {{ min_height }}px.">
<source>The image height is too small ({{ height }}px). Minimum height expected is {{ min_height }}px.</source>
<target>De afbeelding is niet hoog genoeg ({{ height }}px). De minimaal verwachte hoogte is {{ min_height }}px.</target>
</trans-unit>
<trans-unit id="wpV7oxR" resname="The image is landscape oriented ({{ width }}x{{ height }}px). Landscape oriented images are not allowed.">
<source>The image is landscape oriented ({{ width }}x{{ height }}px). Landscape oriented images are not allowed.</source>
<target>De afbeelding is liggend ({{ width }}x{{ height }}px). Liggende afbeeldingen zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="t7f5lY9" resname="The image is portrait oriented ({{ width }}x{{ height }}px). Portrait oriented images are not allowed.">
<source>The image is portrait oriented ({{ width }}x{{ height }}px). Portrait oriented images are not allowed.</source>
<target>De afbeelding is staand ({{ width }}x{{ height }}px). Staande afbeeldingen zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="SVgw5SP" resname="The image is square ({{ width }}x{{ height }}px). Square images are not allowed.">
<source>The image is square ({{ width }}x{{ height }}px). Square images are not allowed.</source>
<target>De afbeelding is vierkant ({{ width }}x{{ height }}px). Vierkante afbeeldingen zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="x5xpcfK" resname="The image ratio is too big ({{ ratio }}). Allowed maximum ratio is {{ max_ratio }}.">
<source>The image ratio is too big ({{ ratio }}). Allowed maximum ratio is {{ max_ratio }}.</source>
<target>De afbeeldingsverhouding is te groot ({{ ratio }}). De maximale verhouding is {{ max_ratio }}.</target>
</trans-unit>
<trans-unit id="5cB4RCN" resname="The image ratio is too small ({{ ratio }}). Minimum ratio expected is {{ min_ratio }}.">
<source>The image ratio is too small ({{ ratio }}). Minimum ratio expected is {{ min_ratio }}.</source>
<target>De afbeeldingsverhouding is te klein ({{ ratio }}). De minimale verhouding is {{ min_ratio }}.</target>
</trans-unit>
<trans-unit id="YD_yE46" resname="The image width is too big ({{ width }}px). Allowed maximum width is {{ max_width }}px.">
<source>The image width is too big ({{ width }}px). Allowed maximum width is {{ max_width }}px.</source>
<target>De afbeelding is te breed ({{ width }}px). De maximale breedte is {{ max_width }}px.</target>
</trans-unit>
<trans-unit id="pfL2.fo" resname="The image width is too small ({{ width }}px). Minimum width expected is {{ min_width }}px.">
<source>The image width is too small ({{ width }}px). Minimum width expected is {{ min_width }}px.</source>
<target>De afbeelding is niet breed genoeg ({{ width }}px). De minimaal verwachte breedte is {{ min_width }}px.</target>
</trans-unit>
<trans-unit id="COERTaP" resname="The mime type of the file is invalid ({{ type }}). Allowed mime types are {{ types }}.">
<source>The mime type of the file is invalid ({{ type }}). Allowed mime types are {{ types }}.</source>
<target>Het mediatype van het bestand is ongeldig ({{ type }}). De toegestane mediatypes zijn {{ types }}.</target>
</trans-unit>
<trans-unit id="Co_98pi" resname="The number of elements in this collection should be a multiple of {{ compared_value }}.">
<source>The number of elements in this collection should be a multiple of {{ compared_value }}.</source>
<target>Het aantal elementen van deze collectie moet een veelvoud zijn van {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="AKUMRqn" resname="The password is invalid.">
<source>The password is invalid.</source>
<target>Het wachtwoord is incorrect.</target>
</trans-unit>
<trans-unit id="7JLHuYX" resname="The password strength is too low. Please use a stronger password.">
<source>The password strength is too low. Please use a stronger password.</source>
<target>Het wachtwoord is niet sterk genoeg. Probeer een sterker wachtwoord.</target>
</trans-unit>
<trans-unit id="TPma293" resname="The selected choice is invalid.">
<source>The selected choice is invalid.</source>
<target>Deze keuze is ongeldig.</target>
</trans-unit>
<trans-unit id="LAGu_iq" resname="The size of the image could not be detected.">
<source>The size of the image could not be detected.</source>
<target>De grootte van de afbeelding kon niet bepaald worden.</target>
</trans-unit>
<trans-unit id="7ZjICRC" resname="The size of the video could not be detected.">
<source>The size of the video could not be detected.</source>
<target>De grootte van de video kon niet worden gedetecteerd.</target>
</trans-unit>
<trans-unit id="yShl.HD" resname="The two values should be equal.">
<source>The two values should be equal.</source>
<target>De twee waarden moeten gelijk zijn.</target>
</trans-unit>
<trans-unit id="SlPljy." resname="The uploaded file was too large. Please try to upload a smaller file.">
<source>The uploaded file was too large. Please try to upload a smaller file.</source>
<target>Het geüploade bestand is te groot. Probeer een kleiner bestand te uploaden.</target>
</trans-unit>
<trans-unit id="ERGRJhK" resname="The value of the netmask should be between {{ min }} and {{ max }}.">
<source>The value of the netmask should be between {{ min }} and {{ max }}.</source>
<target>De waarde van het netmasker moet tussen {{ min }} en {{ max }} liggen.</target>
</trans-unit>
<trans-unit id="KaWXEsW" resname="The value you selected is not a valid choice.">
<source>The value you selected is not a valid choice.</source>
<target>De geselecteerde waarde is geen geldige optie.</target>
</trans-unit>
<trans-unit id="u4rxsHj" resname="The values do not match.">
<source>The values do not match.</source>
<target>De waardes komen niet overeen.</target>
</trans-unit>
<trans-unit id="nzXY1iu" resname="The video contains multiple streams. Only one stream is allowed.">
<source>The video contains multiple streams. Only one stream is allowed.</source>
<target>De video bevat meerdere streams. Slechts één stream is toegestaan.</target>
</trans-unit>
<trans-unit id="x8fsjeI" resname="The video file is corrupted.">
<source>The video file is corrupted.</source>
<target>Het videobestand is beschadigd.</target>
</trans-unit>
<trans-unit id="Dp2k5ge" resname="The video has too few pixels ({{ pixels }} pixels). Minimum amount expected is {{ min_pixels }} pixels.">
<source>The video has too few pixels ({{ pixels }} pixels). Minimum amount expected is {{ min_pixels }} pixels.</source>
<target>De video heeft te weinig pixels ({{ pixels }}). Verwachte minimumaantal is {{ min_pixels }}.</target>
</trans-unit>
<trans-unit id="IDGrrj_" resname="The video has too many pixels ({{ pixels }} pixels). Maximum amount expected is {{ max_pixels }} pixels.">
<source>The video has too many pixels ({{ pixels }} pixels). Maximum amount expected is {{ max_pixels }} pixels.</source>
<target>De video heeft te veel pixels ({{ pixels }}). Het verwachte maximum is {{ max_pixels }}.</target>
</trans-unit>
<trans-unit id="Liu7sjp" resname="The video height is too big ({{ height }}px). Allowed maximum height is {{ max_height }}px.">
<source>The video height is too big ({{ height }}px). Allowed maximum height is {{ max_height }}px.</source>
<target>De videohoogte is te groot ({{ height }}px). Toegestane maximale hoogte is {{ max_height }}px.</target>
</trans-unit>
<trans-unit id="2FkdfHN" resname="The video height is too small ({{ height }}px). Minimum height expected is {{ min_height }}px.">
<source>The video height is too small ({{ height }}px). Minimum height expected is {{ min_height }}px.</source>
<target>De videohoogte is te klein ({{ height }}px). Verwachte minimale hoogte is {{ min_height }}px.</target>
</trans-unit>
<trans-unit id="mXgOwd4" resname="The video is landscape oriented ({{ width }}x{{ height }}px). Landscape oriented videos are not allowed.">
<source>The video is landscape oriented ({{ width }}x{{ height }}px). Landscape oriented videos are not allowed.</source>
<target>De video is in liggende oriëntatie ({{ width }}x{{ height }} px). Liggende video's zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="VrsdBNF" resname="The video is portrait oriented ({{ width }}x{{ height }}px). Portrait oriented videos are not allowed.">
<source>The video is portrait oriented ({{ width }}x{{ height }}px). Portrait oriented videos are not allowed.</source>
<target>De video is in portretstand ({{ width }}x{{ height }}px). Video's in portretstand zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="OT6Evve" resname="The video is square ({{ width }}x{{ height }}px). Square videos are not allowed.">
<source>The video is square ({{ width }}x{{ height }}px). Square videos are not allowed.</source>
<target>De video is vierkant ({{ width }}x{{ height }}px). Vierkante video's zijn niet toegestaan.</target>
</trans-unit>
<trans-unit id="J1cZmuA" resname="The video ratio is too big ({{ ratio }}). Allowed maximum ratio is {{ max_ratio }}.">
<source>The video ratio is too big ({{ ratio }}). Allowed maximum ratio is {{ max_ratio }}.</source>
<target>De videoratio is te groot ({{ ratio }}). Toegestane maximale ratio is {{ max_ratio }}.</target>
</trans-unit>
<trans-unit id="WQTwvFh" resname="The video ratio is too small ({{ ratio }}). Minimum ratio expected is {{ min_ratio }}.">
<source>The video ratio is too small ({{ ratio }}). Minimum ratio expected is {{ min_ratio }}.</source>
<target>De videoratio is te klein ({{ ratio }}). Verwachte minimumverhouding is {{ min_ratio }}.</target>
</trans-unit>
<trans-unit id="WS2rIIU" resname="The video width is too big ({{ width }}px). Allowed maximum width is {{ max_width }}px.">
<source>The video width is too big ({{ width }}px). Allowed maximum width is {{ max_width }}px.</source>
<target>De videobreedte is te groot ({{ width }}px). Toegestane maximale breedte is {{ max_width }}px.</target>
</trans-unit>
<trans-unit id="Ey8lNRo" resname="The video width is too small ({{ width }}px). Minimum width expected is {{ min_width }}px.">
<source>The video width is too small ({{ width }}px). Minimum width expected is {{ min_width }}px.</source>
<target>De videobreedte is te klein ({{ width }}px). Verwachte minimum breedte is {{ min_width }}px.</target>
</trans-unit>
<trans-unit id="vsM4tSv" resname="There is already an account with this email"> <trans-unit id="vsM4tSv" resname="There is already an account with this email">
<source>There is already an account with this email</source> <source>There is already an account with this email</source>
<target>Er is al een account met dit e-mailadres</target> <target>Er is al een account met dit e-mailadres</target>
</trans-unit> </trans-unit>
<trans-unit id="7urGdp4" resname="This Business Identifier Code (BIC) is not associated with IBAN {{ iban }}.">
<source>This Business Identifier Code (BIC) is not associated with IBAN {{ iban }}.</source>
<target>Deze bankidentificatiecode (BIC) is niet gekoppeld aan IBAN {{ iban }}.</target>
</trans-unit>
<trans-unit id="Pm0aj_f" resname="This URL is missing a top-level domain.">
<source>This URL is missing a top-level domain.</source>
<target>Deze URL mist een top-level domein.</target>
</trans-unit>
<trans-unit id="d4HuxpA" resname="This collection should contain exactly {{ limit }} element.|This collection should contain exactly {{ limit }} elements.">
<source>This collection should contain exactly {{ limit }} element.|This collection should contain exactly {{ limit }} elements.</source>
<target>Deze collectie moet exact één element bevatten.|Deze collectie moet exact {{ limit }} elementen bevatten.</target>
</trans-unit>
<trans-unit id="NAJHeXU" resname="This collection should contain only unique elements.">
<source>This collection should contain only unique elements.</source>
<target>Deze collectie mag alleen unieke elementen bevatten.</target>
</trans-unit>
<trans-unit id="aSvx2eb" resname="This collection should contain {{ limit }} element or less.|This collection should contain {{ limit }} elements or less.">
<source>This collection should contain {{ limit }} element or less.|This collection should contain {{ limit }} elements or less.</source>
<target>Deze collectie moet één of minder elementen bevatten.|Deze collectie moet {{ limit }} of minder elementen bevatten.</target>
</trans-unit>
<trans-unit id=".K7ySMo" resname="This collection should contain {{ limit }} element or more.|This collection should contain {{ limit }} elements or more.">
<source>This collection should contain {{ limit }} element or more.|This collection should contain {{ limit }} elements or more.</source>
<target>Deze collectie moet één of meer elementen bevatten.|Deze collectie moet {{ limit }} of meer elementen bevatten.</target>
</trans-unit>
<trans-unit id="m5HGv6j" resname="This field is missing.">
<source>This field is missing.</source>
<target>Dit veld ontbreekt.</target>
</trans-unit>
<trans-unit id="hzMv9Ky" resname="This field was not expected.">
<source>This field was not expected.</source>
<target>Dit veld werd niet verwacht.</target>
</trans-unit>
<trans-unit id="aeFJJFS" resname="This file is not a valid image.">
<source>This file is not a valid image.</source>
<target>Dit bestand is geen geldige afbeelding.</target>
</trans-unit>
<trans-unit id="ZuGpyhA" resname="This file is not a valid video.">
<source>This file is not a valid video.</source>
<target>Dit bestand is geen geldige video.</target>
</trans-unit>
<trans-unit id="tPeCNvN" resname="This filename does not match the expected charset.">
<source>This filename does not match the expected charset.</source>
<target>Deze bestandsnaam komt niet overeen met de verwachte tekencodering.</target>
</trans-unit>
<trans-unit id="tDkL4Jc" resname="This form should not contain extra fields.">
<source>This form should not contain extra fields.</source>
<target>Dit formulier mag geen extra velden bevatten.</target>
</trans-unit>
<trans-unit id="w2dRFcU" resname="This is not a valid Business Identifier Code (BIC).">
<source>This is not a valid Business Identifier Code (BIC).</source>
<target>Deze waarde is geen geldige bankidentificatiecode (BIC).</target>
</trans-unit>
<trans-unit id="r9nOjqT" resname="This is not a valid IP address.">
<source>This is not a valid IP address.</source>
<target>Deze waarde is geen geldig IP-adres.</target>
</trans-unit>
<trans-unit id="Bp_BiDs" resname="This is not a valid International Bank Account Number (IBAN).">
<source>This is not a valid International Bank Account Number (IBAN).</source>
<target>Deze waarde is geen geldig internationaal bankrekeningnummer (IBAN).</target>
</trans-unit>
<trans-unit id="GO8.wou" resname="This is not a valid UUID.">
<source>This is not a valid UUID.</source>
<target>Deze waarde is geen geldige UUID.</target>
</trans-unit>
<trans-unit id="o0vqbDQ" resname="This password has been leaked in a data breach, it must not be used. Please use another password.">
<source>This password has been leaked in a data breach, it must not be used. Please use another password.</source>
<target>Dit wachtwoord is gelekt bij een datalek en mag niet worden gebruikt. Kies een ander wachtwoord.</target>
</trans-unit>
<trans-unit id="Pgo2CSS" resname="This value contains characters that are not allowed by the current restriction-level.">
<source>This value contains characters that are not allowed by the current restriction-level.</source>
<target>Deze waarde bevat tekens die niet zijn toegestaan volgens het huidige beperkingsniveau.</target>
</trans-unit>
<trans-unit id="tD.RkRe" resname="This value does not match the expected {{ charset }} charset.">
<source>This value does not match the expected {{ charset }} charset.</source>
<target>Deze waarde is niet in de verwachte tekencodering {{ charset }}.</target>
</trans-unit>
<trans-unit id="egUw40j" resname="This value does not represent a valid week in the ISO 8601 format.">
<source>This value does not represent a valid week in the ISO 8601 format.</source>
<target>Deze waarde vertegenwoordigt geen geldige week in het ISO 8601-formaat.</target>
</trans-unit>
<trans-unit id="nmXj9fT" resname="This value is already used.">
<source>This value is already used.</source>
<target>Deze waarde wordt al gebruikt.</target>
</trans-unit>
<trans-unit id="w8oSYvN" resname="This value is neither a valid ISBN-10 nor a valid ISBN-13.">
<source>This value is neither a valid ISBN-10 nor a valid ISBN-13.</source>
<target>Deze waarde is geen geldige ISBN-10 of ISBN-13.</target>
</trans-unit>
<trans-unit id="fye9wG1" resname="This value is not a valid CIDR notation.">
<source>This value is not a valid CIDR notation.</source>
<target>Deze waarde is geen geldige CIDR-notatie.</target>
</trans-unit>
<trans-unit id="2_9V4lo" resname="This value is not a valid CSS color.">
<source>This value is not a valid CSS color.</source>
<target>Deze waarde is geen geldige CSS kleur.</target>
</trans-unit>
<trans-unit id="wlGn1pP" resname="This value is not a valid HTML5 color.">
<source>This value is not a valid HTML5 color.</source>
<target>Dit is geen geldige HTML5 kleur.</target>
</trans-unit>
<trans-unit id="HqngxWf" resname="This value is not a valid ISBN-10.">
<source>This value is not a valid ISBN-10.</source>
<target>Deze waarde is geen geldige ISBN-10.</target>
</trans-unit>
<trans-unit id="Pm2IygX" resname="This value is not a valid ISBN-13.">
<source>This value is not a valid ISBN-13.</source>
<target>Deze waarde is geen geldige ISBN-13.</target>
</trans-unit>
<trans-unit id="aEf2fyJ" resname="This value is not a valid ISSN.">
<source>This value is not a valid ISSN.</source>
<target>Deze waarde is geen geldige ISSN.</target>
</trans-unit>
<trans-unit id="1nUIbF0" resname="This value is not a valid International Securities Identification Number (ISIN).">
<source>This value is not a valid International Securities Identification Number (ISIN).</source>
<target>Deze waarde is geen geldig International Securities Identification Number (ISIN).</target>
</trans-unit>
<trans-unit id="65jFl8b" resname="This value is not a valid MAC address.">
<source>This value is not a valid MAC address.</source>
<target>Deze waarde is geen geldig MAC-adres.</target>
</trans-unit>
<trans-unit id="3QNCFgi" resname="This value is not a valid Twig template.">
<source>This value is not a valid Twig template.</source>
<target>Deze waarde is geen geldige Twig-template.</target>
</trans-unit>
<trans-unit id="X5msM9I" resname="This value is not a valid URL.">
<source>This value is not a valid URL.</source>
<target>Deze waarde is geen geldige URL.</target>
</trans-unit>
<trans-unit id="QUviSFS" resname="This value is not a valid country.">
<source>This value is not a valid country.</source>
<target>Deze waarde is geen geldig land.</target>
</trans-unit>
<trans-unit id="OeFZFII" resname="This value is not a valid currency.">
<source>This value is not a valid currency.</source>
<target>Deze waarde is geen geldige valuta.</target>
</trans-unit>
<trans-unit id="ccjuUaj" resname="This value is not a valid date.">
<source>This value is not a valid date.</source>
<target>Deze waarde is geen geldige datum.</target>
</trans-unit>
<trans-unit id="4jAy6GT" resname="This value is not a valid datetime.">
<source>This value is not a valid datetime.</source>
<target>Deze waarde is geen geldige datum en tijd.</target>
</trans-unit>
<trans-unit id="J8MwqDW" resname="This value is not a valid email address.">
<source>This value is not a valid email address.</source>
<target>Deze waarde is geen geldig e-mailadres.</target>
</trans-unit>
<trans-unit id="q0JXpA2" resname="This value is not a valid hostname.">
<source>This value is not a valid hostname.</source>
<target>Deze waarde is geen geldige hostnaam.</target>
</trans-unit>
<trans-unit id="izWIhKR" resname="This value is not a valid language.">
<source>This value is not a valid language.</source>
<target>Deze waarde is geen geldige taal.</target>
</trans-unit>
<trans-unit id="gBF5fE7" resname="This value is not a valid locale.">
<source>This value is not a valid locale.</source>
<target>Deze waarde is geen geldige landinstelling.</target>
</trans-unit>
<trans-unit id="Uu2fZ7m" resname="This value is not a valid time.">
<source>This value is not a valid time.</source>
<target>Deze waarde is geen geldige tijd.</target>
</trans-unit>
<trans-unit id="0bSR5cp" resname="This value is not a valid timezone.">
<source>This value is not a valid timezone.</source>
<target>Deze waarde is geen geldige tijdzone.</target>
</trans-unit>
<trans-unit id="qj68TOe" resname="This value is not a valid week.">
<source>This value is not a valid week.</source>
<target>Deze waarde is geen geldige week.</target>
</trans-unit>
<trans-unit id="TZg2uqx" resname="This value is not valid.">
<source>This value is not valid.</source>
<target>Deze waarde is niet geldig.</target>
</trans-unit>
<trans-unit id="1WpHUSt" resname="This value is too long. It should contain one word.|This value is too long. It should contain {{ max }} words or less.">
<source>This value is too long. It should contain one word.|This value is too long. It should contain {{ max }} words or less.</source>
<target>Deze waarde is te lang. Het moet één woord zijn.|Deze waarde is te lang. Het mag maximaal {{ max }} woorden bevatten.</target>
</trans-unit>
<trans-unit id="Zuairxy" resname="This value is too long. It should have {{ limit }} character or less.|This value is too long. It should have {{ limit }} characters or less.">
<source>This value is too long. It should have {{ limit }} character or less.|This value is too long. It should have {{ limit }} characters or less.</source>
<target>Deze waarde is te lang. Deze mag maximaal één teken bevatten.|Deze waarde is te lang. Deze mag maximaal {{ limit }} tekens bevatten.</target>
</trans-unit>
<trans-unit id="gbpWGsC" resname="This value is too short. It should contain at least one word.|This value is too short. It should contain at least {{ min }} words.">
<source>This value is too short. It should contain at least one word.|This value is too short. It should contain at least {{ min }} words.</source>
<target>Deze waarde is te kort. Het moet ten minste één woord bevatten.|Deze waarde is te kort. Het moet ten minste {{ min }} woorden bevatten.</target>
</trans-unit>
<trans-unit id="Muq3W.6" resname="This value is too short. It should have {{ limit }} character or more.|This value is too short. It should have {{ limit }} characters or more.">
<source>This value is too short. It should have {{ limit }} character or more.|This value is too short. It should have {{ limit }} characters or more.</source>
<target>Deze waarde is te kort. Deze moet ten minste één teken bevatten.|Deze waarde is te kort. Deze moet ten minste {{ limit }} tekens bevatten.</target>
</trans-unit>
<trans-unit id="BmDYW0B" resname="This value should be a multiple of {{ compared_value }}.">
<source>This value should be a multiple of {{ compared_value }}.</source>
<target>Deze waarde moet een meervoud van {{ compared_value }} zijn.</target>
</trans-unit>
<trans-unit id="kG_VGA1" resname="This value should be a valid expression.">
<source>This value should be a valid expression.</source>
<target>Deze waarde moet een geldige expressie zijn.</target>
</trans-unit>
<trans-unit id="jpVY2B5" resname="This value should be a valid number.">
<source>This value should be a valid number.</source>
<target>Deze waarde moet een geldig getal zijn.</target>
</trans-unit>
<trans-unit id="nNegxai" resname="This value should be between {{ min }} and {{ max }}.">
<source>This value should be between {{ min }} and {{ max }}.</source>
<target>Deze waarde moet zich tussen {{ min }} en {{ max }} bevinden.</target>
</trans-unit>
<trans-unit id="K5oLZkw" resname="This value should be blank.">
<source>This value should be blank.</source>
<target>Deze waarde moet leeg zijn.</target>
</trans-unit>
<trans-unit id="INgX870" resname="This value should be either negative or zero.">
<source>This value should be either negative or zero.</source>
<target>Deze waarde moet negatief of gelijk aan nul zijn.</target>
</trans-unit>
<trans-unit id="Q07tS4h" resname="This value should be either positive or zero.">
<source>This value should be either positive or zero.</source>
<target>Deze waarde moet positief of gelijk aan nul zijn.</target>
</trans-unit>
<trans-unit id="G8HNxQM" resname="This value should be equal to {{ compared_value }}.">
<source>This value should be equal to {{ compared_value }}.</source>
<target>Deze waarde moet gelijk zijn aan {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="jysioaA" resname="This value should be false.">
<source>This value should be false.</source>
<target>Deze waarde moet onwaar zijn.</target>
</trans-unit>
<trans-unit id="OrsJXWO" resname="This value should be greater than or equal to {{ compared_value }}.">
<source>This value should be greater than or equal to {{ compared_value }}.</source>
<target>Deze waarde moet groter of gelijk aan {{ compared_value }} zijn.</target>
</trans-unit>
<trans-unit id="Xgf_Y8C" resname="This value should be greater than {{ compared_value }}.">
<source>This value should be greater than {{ compared_value }}.</source>
<target>Deze waarde moet groter zijn dan {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="wni8iKB" resname="This value should be identical to {{ compared_value_type }} {{ compared_value }}.">
<source>This value should be identical to {{ compared_value_type }} {{ compared_value }}.</source>
<target>Deze waarde moet identiek zijn aan {{ compared_value_type }} {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="vNn72MQ" resname="This value should be less than or equal to {{ compared_value }}.">
<source>This value should be less than or equal to {{ compared_value }}.</source>
<target>Deze waarde moet minder dan of gelijk aan {{ compared_value }} zijn.</target>
</trans-unit>
<trans-unit id="EgRc_Qq" resname="This value should be less than {{ compared_value }}.">
<source>This value should be less than {{ compared_value }}.</source>
<target>Deze waarde moet minder zijn dan {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="cwaFWZt" resname="This value should be negative.">
<source>This value should be negative.</source>
<target>Deze waarde moet negatief zijn.</target>
</trans-unit>
<trans-unit id="yIMqPF6" resname="This value should be null.">
<source>This value should be null.</source>
<target>Deze waarde moet null zijn.</target>
</trans-unit>
<trans-unit id="HL55su4" resname="This value should be of type {{ type }}.">
<source>This value should be of type {{ type }}.</source>
<target>Deze waarde moet van het type {{ type }} zijn.</target>
</trans-unit>
<trans-unit id="mBNZ2p4" resname="This value should be positive.">
<source>This value should be positive.</source>
<target>Deze waarde moet positief zijn.</target>
</trans-unit>
<trans-unit id="hbSSYMh" resname="This value should be the user's current password.">
<source>This value should be the user's current password.</source>
<target>Deze waarde moet het huidige wachtwoord van de gebruiker zijn.</target>
</trans-unit>
<trans-unit id="1zVPJo4" resname="This value should be true.">
<source>This value should be true.</source>
<target>Deze waarde moet waar zijn.</target>
</trans-unit>
<trans-unit id="NpIt6EV" resname="This value should be valid JSON.">
<source>This value should be valid JSON.</source>
<target>Deze waarde moet geldige JSON zijn.</target>
</trans-unit>
<trans-unit id="xJV2eT8" resname="This value should be {{ limit }} or less.">
<source>This value should be {{ limit }} or less.</source>
<target>Deze waarde moet {{ limit }} of minder zijn.</target>
</trans-unit>
<trans-unit id="3bh29vT" resname="This value should be {{ limit }} or more.">
<source>This value should be {{ limit }} or more.</source>
<target>Deze waarde moet {{ limit }} of meer zijn.</target>
</trans-unit>
<trans-unit id="A1Hv7nv" resname="This value should have exactly {{ limit }} character.|This value should have exactly {{ limit }} characters.">
<source>This value should have exactly {{ limit }} character.|This value should have exactly {{ limit }} characters.</source>
<target>Deze waarde moet exact één teken lang zijn.|Deze waarde moet exact {{ limit }} tekens lang zijn.</target>
</trans-unit>
<trans-unit id="pOcsUec" resname="This value should not be after week &quot;{{ max }}&quot;.">
<source>This value should not be after week "{{ max }}".</source>
<target>Deze waarde mag niet na week "{{ max }}" liggen.</target>
</trans-unit>
<trans-unit id="RX1vHSU" resname="This value should not be before week &quot;{{ min }}&quot;.">
<source>This value should not be before week "{{ min }}".</source>
<target>Deze waarde mag niet vóór week "{{ min }}" liggen.</target>
</trans-unit>
<trans-unit id="vLBiKCD" resname="This value should not be blank.">
<source>This value should not be blank.</source>
<target>Deze waarde mag niet leeg zijn.</target>
</trans-unit>
<trans-unit id="N4F464Y" resname="This value should not be equal to {{ compared_value }}.">
<source>This value should not be equal to {{ compared_value }}.</source>
<target>Deze waarde mag niet gelijk zijn aan {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="T8lYq4L" resname="This value should not be identical to {{ compared_value_type }} {{ compared_value }}.">
<source>This value should not be identical to {{ compared_value_type }} {{ compared_value }}.</source>
<target>Deze waarde mag niet identiek zijn aan {{ compared_value_type }} {{ compared_value }}.</target>
</trans-unit>
<trans-unit id="FUEH3cZ" resname="This value should not be null.">
<source>This value should not be null.</source>
<target>Deze waarde mag niet null zijn.</target>
</trans-unit>
<trans-unit id="rfebLrY" resname="This value should satisfy at least one of the following constraints:">
<source>This value should satisfy at least one of the following constraints:</source>
<target>Deze waarde moet voldoen aan tenminste een van de volgende voorwaarden:</target>
</trans-unit>
<trans-unit id="Y8YNvWG" resname="Unsupported card type or invalid card number.">
<source>Unsupported card type or invalid card number.</source>
<target>Niet-ondersteund type creditcard of ongeldig nummer.</target>
</trans-unit>
<trans-unit id="hCzL7Mv" resname="Unsupported video codec &quot;{{ codec }}&quot;.">
<source>Unsupported video codec "{{ codec }}".</source>
<target>Niet-ondersteunde videocodec {{ codec }}.</target>
</trans-unit>
<trans-unit id="nTnX6nj" resname="Unsupported video container &quot;{{ container }}&quot;.">
<source>Unsupported video container "{{ container }}".</source>
<target>Niet-ondersteunde videocontainer "{{ container }}".</target>
</trans-unit>
<trans-unit id="Z4taMVn" resname="Using hidden overlay characters is not allowed.">
<source>Using hidden overlay characters is not allowed.</source>
<target>Het gebruik van verborgen overlay-tekens is niet toegestaan.</target>
</trans-unit>
<trans-unit id="7hhZedg" resname="Using invisible characters is not allowed.">
<source>Using invisible characters is not allowed.</source>
<target>Het gebruik van onzichtbare tekens is niet toegestaan.</target>
</trans-unit>
<trans-unit id="4xOpbwN" resname="You must select at least {{ limit }} choice.|You must select at least {{ limit }} choices.">
<source>You must select at least {{ limit }} choice.|You must select at least {{ limit }} choices.</source>
<target>Selecteer ten minste {{ limit }} optie.|Selecteer ten minste {{ limit }} opties.</target>
</trans-unit>
<trans-unit id="65n9dGQ" resname="You must select at most {{ limit }} choice.|You must select at most {{ limit }} choices.">
<source>You must select at most {{ limit }} choice.|You must select at most {{ limit }} choices.</source>
<target>Selecteer maximaal {{ limit }} optie.|Selecteer maximaal {{ limit }} opties.</target>
</trans-unit>
<trans-unit id="_OmnKuR" resname="Your password should be at least {{ limit }} characters"> <trans-unit id="_OmnKuR" resname="Your password should be at least {{ limit }} characters">
<source>Your password should be at least {{ limit }} characters</source> <source>Your password should be at least {{ limit }} characters</source>
<target>Je wachtwoord moet minimaal {{ limit }} karakters lang zijn</target> <target>Je wachtwoord moet minimaal {{ limit }} karakters lang zijn</target>