mirror of
				https://gitlab.com/animath/si/plateforme.git
				synced 2025-10-25 18:33:13 +02:00 
			
		
		
		
	Compare commits
	
		
			12 Commits
		
	
	
		
			84eb08ec46
			...
			dev
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 8aec72d712 | ||
|  | 6a521b6121 | ||
|  | 62abfa94d6 | ||
|  | 952315ea4d | ||
|  | 2e613799c9 | ||
|  | 08805a6360 | ||
|  | 6841659e41 | ||
|  | a84ffcf0a3 | ||
|  | 203fc3cd54 | ||
|  | 60f5236dee | ||
|  | ab459ecc17 | ||
|  | 7ad7659d78 | 
| @@ -4,12 +4,10 @@ ENV PYTHONUNBUFFERED 1 | ||||
| ENV DJANGO_ALLOW_ASYNC_UNSAFE 1 | ||||
|  | ||||
| RUN apk add --no-cache gettext nginx gcc git libc-dev libffi-dev libpq-dev libxml2-dev libxslt-dev \ | ||||
|     npm libmagic texlive texmf-dist-fontsrecommended texmf-dist-lang texmf-dist-latexextra | ||||
|     libmagic texlive texmf-dist-fontsrecommended texmf-dist-lang texmf-dist-latexextra uglify-js | ||||
|  | ||||
| RUN apk add --no-cache bash | ||||
|  | ||||
| RUN npm install -g yuglify | ||||
|  | ||||
| RUN mkdir /code /code/docs | ||||
| WORKDIR /code | ||||
| COPY requirements.txt /code/requirements.txt | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,7 +1,9 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.contrib import admin | ||||
| from django.http import HttpRequest | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from .models import Note, Participation, Passage, Pool, Solution, Team, Tournament, Tweak, WrittenReview | ||||
| @@ -51,9 +53,14 @@ class PassageInline(admin.TabularInline): | ||||
|     model = Passage | ||||
|     extra = 0 | ||||
|     ordering = ('position',) | ||||
|     autocomplete_fields = ('reporter', 'opponent', 'reviewer', 'observer',) | ||||
|     show_change_link = True | ||||
|  | ||||
|     def get_autocomplete_fields(self, request: HttpRequest) -> tuple[str]: | ||||
|         fields = ('reporter', 'opponent', 'reviewer',) | ||||
|         if settings.HAS_OBSERVER: | ||||
|             fields += ('observer',) | ||||
|         return fields | ||||
|  | ||||
|  | ||||
| class NoteInline(admin.TabularInline): | ||||
|     model = Note | ||||
| @@ -113,12 +120,9 @@ class PoolAdmin(admin.ModelAdmin): | ||||
|  | ||||
| @admin.register(Passage) | ||||
| class PassageAdmin(admin.ModelAdmin): | ||||
|     list_display = ('__str__', 'reporter_trigram', 'solution_number', 'opponent_trigram', 'reviewer_trigram', | ||||
|                     'observer_trigram', 'pool_abbr', 'position', 'tournament') | ||||
|     list_filter = ('pool__tournament', 'pool__round', 'pool__letter', 'solution_number',) | ||||
|     search_fields = ('pool__participations__team__name', 'pool__participations__team__trigram',) | ||||
|     ordering = ('pool__tournament', 'pool__round', 'pool__letter', 'position',) | ||||
|     autocomplete_fields = ('pool', 'reporter', 'opponent', 'reviewer', 'observer',) | ||||
|     inlines = (NoteInline,) | ||||
|  | ||||
|     @admin.display(description=_("reporter"), ordering='reporter__team__trigram') | ||||
| @@ -135,7 +139,7 @@ class PassageAdmin(admin.ModelAdmin): | ||||
|  | ||||
|     @admin.display(description=_("observer"), ordering='observer__team__trigram') | ||||
|     def observer_trigram(self, record: Passage): | ||||
|         return record.observer.team.trigram | ||||
|         return record.observer.team.trigram if record.observer else None | ||||
|  | ||||
|     @admin.display(description=_("pool"), ordering='pool__letter') | ||||
|     def pool_abbr(self, record): | ||||
| @@ -145,15 +149,23 @@ class PassageAdmin(admin.ModelAdmin): | ||||
|     def tournament(self, record: Passage): | ||||
|         return record.pool.tournament | ||||
|  | ||||
|     def get_list_display(self, request: HttpRequest) -> tuple[str]: | ||||
|         if settings.HAS_OBSERVER: | ||||
|             return ('__str__', 'reporter_trigram', 'solution_number', 'opponent_trigram', | ||||
|                     'reviewer_trigram', 'observer_trigram', 'pool_abbr', 'position', 'tournament') | ||||
|         else: | ||||
|             return ('__str__', 'reporter_trigram', 'solution_number', 'opponent_trigram', | ||||
|                     'reviewer_trigram', 'pool_abbr', 'position', 'tournament') | ||||
|  | ||||
|     def get_autocomplete_fields(self, request: HttpRequest) -> tuple[str]: | ||||
|         fields = ('pool', 'reporter', 'opponent', 'reviewer',) | ||||
|         if settings.HAS_OBSERVER: | ||||
|             fields += ('observer',) | ||||
|         return fields | ||||
|  | ||||
|  | ||||
| @admin.register(Note) | ||||
| class NoteAdmin(admin.ModelAdmin): | ||||
|     list_display = ('passage', 'pool', 'jury', 'reporter_writing', 'reporter_oral', | ||||
|                     'opponent_writing', 'opponent_oral', 'reviewer_writing', 'reviewer_oral', | ||||
|                     'observer_writing', 'observer_oral',) | ||||
|     list_filter = ('passage__pool__letter', 'passage__solution_number', 'jury', | ||||
|                    'reporter_writing', 'reporter_oral', 'opponent_writing', 'opponent_oral', | ||||
|                    'reviewer_writing', 'reviewer_oral', 'observer_writing', 'observer_oral') | ||||
|     search_fields = ('jury__user__last_name', 'jury__user__first_name', 'passage__reporter__team__trigram',) | ||||
|     autocomplete_fields = ('jury', 'passage',) | ||||
|  | ||||
| @@ -161,6 +173,21 @@ class NoteAdmin(admin.ModelAdmin): | ||||
|     def pool(self, record): | ||||
|         return record.passage.pool.short_name | ||||
|  | ||||
|     def get_list_display(self, request: HttpRequest) -> tuple[str]: | ||||
|         fields = ('passage', 'pool', 'jury', 'reporter_writing', 'reporter_oral', | ||||
|                   'opponent_writing', 'opponent_oral', 'reviewer_writing', 'reviewer_oral',) | ||||
|         if settings.HAS_OBSERVER: | ||||
|             fields += ('observer_writing', 'observer_oral',) | ||||
|         return fields | ||||
|  | ||||
|     def get_list_filter(self, request: HttpRequest) -> tuple[str]: | ||||
|         fields = ('passage__pool__letter', 'passage__solution_number', 'jury', | ||||
|                   'reporter_writing', 'reporter_oral', 'opponent_writing', 'opponent_oral', | ||||
|                   'reviewer_writing', 'reviewer_oral',) | ||||
|         if settings.HAS_OBSERVER: | ||||
|             fields += ('observer_writing', 'observer_oral',) | ||||
|         return fields | ||||
|  | ||||
|  | ||||
| @admin.register(Solution) | ||||
| class SolutionAdmin(admin.ModelAdmin): | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import re | ||||
| from crispy_forms.helper import FormHelper | ||||
| from crispy_forms.layout import Div, Field, HTML, Layout, Submit | ||||
| from django import forms | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.models import User | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.core.validators import FileExtensionValidator | ||||
| @@ -14,7 +15,6 @@ from django.utils.translation import gettext_lazy as _ | ||||
| import pandas | ||||
| from pypdf import PdfReader | ||||
| from registration.models import VolunteerRegistration | ||||
| from tfjm import settings | ||||
|  | ||||
| from .models import Note, Participation, Passage, Pool, Solution, Team, Tournament, WrittenReview | ||||
|  | ||||
| @@ -405,6 +405,12 @@ class WrittenReviewForm(forms.ModelForm): | ||||
|  | ||||
|  | ||||
| class NoteForm(forms.ModelForm): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         if not settings.HAS_OBSERVER: | ||||
|             del self.fields['observer_writing'] | ||||
|             del self.fields['observer_oral'] | ||||
|  | ||||
|     class Meta: | ||||
|         model = Note | ||||
|         fields = ('reporter_writing', 'reporter_oral', 'opponent_writing', | ||||
|   | ||||
| @@ -5,11 +5,13 @@ from pathlib import Path | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.core.management import BaseCommand | ||||
| from django.utils.translation import activate | ||||
| from participation.models import Solution, Tournament | ||||
|  | ||||
|  | ||||
| class Command(BaseCommand): | ||||
|     def handle(self, *args, **kwargs): | ||||
|         activate(settings.PREFERRED_LANGUAGE_CODE) | ||||
|         base_dir = Path(__file__).parent.parent.parent.parent | ||||
|         base_dir /= "output" | ||||
|         if not base_dir.is_dir(): | ||||
|   | ||||
| @@ -440,6 +440,10 @@ class Tournament(models.Model): | ||||
|             return Participation.objects.filter(final=True) | ||||
|         return self.participation_set | ||||
|  | ||||
|     @property | ||||
|     def organizers_and_presidents(self): | ||||
|         return VolunteerRegistration.objects.filter(Q(admin=True) | Q(organized_tournaments=self) | Q(pools_presided__tournament=self)) | ||||
|  | ||||
|     @property | ||||
|     def solutions(self): | ||||
|         if self.final: | ||||
| @@ -932,10 +936,10 @@ class Participation(models.Model): | ||||
|                 'content': content, | ||||
|             }) | ||||
|         elif timezone.now() <= tournament.reviews_first_phase_limit + timedelta(hours=2): | ||||
|             reporter_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=1, reporter=self) | ||||
|             opponent_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=1, opponent=self) | ||||
|             reviewer_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=1, reviewer=self) | ||||
|             observer_passage = Passage.objects.filter(pool__tournament=self.tournament, pool__round=1, observer=self) | ||||
|             reporter_passage = Passage.objects.get(pool__tournament=tournament, pool__round=1, reporter=self) | ||||
|             opponent_passage = Passage.objects.get(pool__tournament=tournament, pool__round=1, opponent=self) | ||||
|             reviewer_passage = Passage.objects.get(pool__tournament=tournament, pool__round=1, reviewer=self) | ||||
|             observer_passage = Passage.objects.filter(pool__tournament=tournament, pool__round=1, observer=self) | ||||
|             observer_passage = observer_passage.get() if observer_passage.exists() else None | ||||
|  | ||||
|             reporter_text = _("<p>The solutions draw is ended. You can check the result on " | ||||
| @@ -997,10 +1001,10 @@ class Participation(models.Model): | ||||
|                 'content': content, | ||||
|             }) | ||||
|         elif timezone.now() <= tournament.reviews_second_phase_limit + timedelta(hours=2): | ||||
|             reporter_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=2, reporter=self) | ||||
|             opponent_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=2, opponent=self) | ||||
|             reviewer_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=2, reviewer=self) | ||||
|             observer_passage = Passage.objects.filter(pool__tournament=self.tournament, pool__round=2, observer=self) | ||||
|             reporter_passage = Passage.objects.get(pool__tournament=tournament, pool__round=2, reporter=self) | ||||
|             opponent_passage = Passage.objects.get(pool__tournament=tournament, pool__round=2, opponent=self) | ||||
|             reviewer_passage = Passage.objects.get(pool__tournament=tournament, pool__round=2, reviewer=self) | ||||
|             observer_passage = Passage.objects.filter(pool__tournament=tournament, pool__round=2, observer=self) | ||||
|             observer_passage = observer_passage.get() if observer_passage.exists() else None | ||||
|  | ||||
|             reporter_text = _("<p>For the second round, you will present " | ||||
| @@ -1061,10 +1065,10 @@ class Participation(models.Model): | ||||
|             }) | ||||
|         elif settings.NB_ROUNDS >= 3 \ | ||||
|                 and timezone.now() <= tournament.reviews_third_phase_limit + timedelta(hours=2): | ||||
|             reporter_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=3, reporter=self) | ||||
|             opponent_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=3, opponent=self) | ||||
|             reviewer_passage = Passage.objects.get(pool__tournament=self.tournament, pool__round=3, reviewer=self) | ||||
|             observer_passage = Passage.objects.filter(pool__tournament=self.tournament, pool__round=3, observer=self) | ||||
|             reporter_passage = Passage.objects.get(pool__tournament=tournament, pool__round=3, reporter=self) | ||||
|             opponent_passage = Passage.objects.get(pool__tournament=tournament, pool__round=3, opponent=self) | ||||
|             reviewer_passage = Passage.objects.get(pool__tournament=tournament, pool__round=3, reviewer=self) | ||||
|             observer_passage = Passage.objects.filter(pool__tournament=tournament, pool__round=3, observer=self) | ||||
|             observer_passage = observer_passage.get() if observer_passage.exists() else None | ||||
|  | ||||
|             reporter_text = _("<p>For the third round, you will present " | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| # Copyright (C) 2020 by Animath | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.utils import formats | ||||
| from django.utils.safestring import mark_safe | ||||
| from django.utils.text import format_lazy | ||||
| @@ -106,8 +107,6 @@ class PoolTable(tables.Table): | ||||
|  | ||||
|  | ||||
| class PassageTable(tables.Table): | ||||
|     # FIXME Ne pas afficher l'équipe observatrice si non nécessaire | ||||
|  | ||||
|     reporter = tables.LinkColumn( | ||||
|         "participation:passage_detail", | ||||
|         args=[tables.A("id")], | ||||
| @@ -131,7 +130,9 @@ class PassageTable(tables.Table): | ||||
|             'class': 'table table-condensed table-striped text-center', | ||||
|         } | ||||
|         model = Passage | ||||
|         fields = ('reporter', 'opponent', 'reviewer', 'observer', 'solution_number', ) | ||||
|         fields = ('reporter', 'opponent', 'reviewer',) \ | ||||
|             + (('observer',) if settings.HAS_OBSERVER else ()) \ | ||||
|             + ('solution_number', ) | ||||
|  | ||||
|  | ||||
| class NoteTable(tables.Table): | ||||
| @@ -160,4 +161,6 @@ class NoteTable(tables.Table): | ||||
|         } | ||||
|         model = Note | ||||
|         fields = ('jury', 'reporter_writing', 'reporter_oral', 'opponent_writing', 'opponent_oral', | ||||
|                   'reviewer_writing', 'reviewer_oral', 'observer_writing', 'observer_oral', 'update',) | ||||
|                   'reviewer_writing', 'reviewer_oral',) + \ | ||||
|                  (('observer_writing', 'observer_oral') if settings.HAS_OBSERVER else ()) + \ | ||||
|                  ('update',) | ||||
|   | ||||
| @@ -58,7 +58,7 @@ | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%DEFENSEUR | ||||
| \begin{tabular}{|c|p{25mm}|p{11cm}|c|{% for passage in passages.all %}p{2cm}|{% endfor %}}\hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Reporter" %}} \normalsize presents their ideas and major results for the solution of the problem.} {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.reporter.team.trigram }} {% endfor %}\\ \hline \hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Reporter" %}} \normalsize presents their ideas and major results for the solution of the problem.} {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.reporter.team.trigram }} {% endfor %}\\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{7}{3mm}{\bf \begin{turn}{90}WRITING\end{turn}} & \multirow{3}{20mm}{ {% trans "Scientific part" %}} & {% trans "Depth and difficulty of the elements presented" %} & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| @@ -86,7 +86,7 @@ | ||||
| %%%%%%%%%%%%%%%%%OPPOSANT⋅E | ||||
| \begin{tabular}{|c|p{25mm}|p{11cm}|c{% for passage in passages.all %}|p{2cm}{% endfor %}|}\hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Opponent" %}} \normalsize provides a critical analysis of the solution and presentation.} | ||||
| {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.opponent.team.trigram }} {% endfor %} \\ \hline \hline | ||||
| {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.opponent.team.trigram }} {% endfor %} \\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{6}{3mm}{\bf \begin{turn}{90}WRITING\end{turn}} &\multirow{4}{25mm}{ {% trans "Scientific part" %}} & {% trans "Critical thinking and perspective on the proposed solution" %} & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| @@ -108,7 +108,7 @@ | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%RAPPORTEUR⋅RICE | ||||
| \begin{tabular}{|c|p{25mm}|p{11cm}|c{% for passage in passages.all %}|p{2cm}{% endfor %}|}\hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Reviewer" %}} \normalsize evaluates the debate between the Reporter and the Opponent.} {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.reviewer.team.trigram }} {% endfor %}\\ \hline \hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Reviewer" %}} \normalsize evaluates the debate between the Reporter and the Opponent.} {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.reviewer.team.trigram }} {% endfor %}\\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{6}{3mm}{\bf \begin{turn}{90}WRITING\end{turn}} &\multirow{4}{25mm}{ {% trans "Scientific part" %}} & {% trans "Critical thinking and perspective on the proposed solution" %} & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| @@ -131,7 +131,7 @@ | ||||
| {% if TFJM.APP == "ETEAM" and pool.participations.count >= 4 %} | ||||
| %%%%%%%%%%%%%%%%%%%%%%OBSERVATEUR⋅RICE | ||||
| \begin{tabular}{|c|p{25mm}|p{11cm}|c{% for passage in passages.all %}|p{2cm}{% endfor %}|}\hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Observer" %}} \normalsize makes useful remarks on crucial points missed by the other participants.} {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.observer.team.trigram }} {% endfor %}\\ \hline \hline | ||||
| \multicolumn{4}{|l|}{The {\bf {% trans "Observer" %}} \normalsize makes useful remarks on crucial points missed by the other participants.} {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.observer.team.trigram }} {% endfor %}\\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{6}{3mm}{\bf \begin{turn}{90}WRITING\end{turn}} &\multirow{4}{25mm}{ {% trans "Scientific part" %}} & {% trans "Critical thinking and perspective on the proposed solution" %} & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
|   | ||||
| @@ -52,7 +52,7 @@ | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%DEFENSEUR | ||||
| \begin{tabular}{|c|p{24mm}|p{11cm}|c|{% for passage in passages.all %}p{2cm}|{% endfor %}}\hline | ||||
| \multicolumn{4}{|l|}{Læ {\bf D\'efenseur⋅se} \normalsize pr\'esente les id\'ees et r\'esultats principaux pour la solution du probl\`eme.} {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.reporter.team.trigram }} {% endfor %}\\ \hline \hline | ||||
| \multicolumn{4}{|l|}{Læ {\bf D\'efenseur⋅se} \normalsize pr\'esente les id\'ees et r\'esultats principaux pour la solution du probl\`eme.} {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.reporter.team.trigram }} {% endfor %}\\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{7}{3mm}{\bf \begin{turn}{90}ÉCRIT\end{turn}} & \multirow{3}{24mm}{Partie scientifique} & Profondeur et difficulté des éléments présentés & [0,6] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| @@ -80,7 +80,7 @@ | ||||
| %%%%%%%%%%%%%%%%%OPPOSANT | ||||
| \begin{tabular}{|c|p{24mm}|p{11cm}|c{% for passage in passages.all %}|p{2cm}{% endfor %}|}\hline | ||||
| \multicolumn{4}{|l|}{L' {\bf Opposant⋅e} \normalsize fournit une analyse critique de la solution et de la pr\'esentation.} | ||||
| {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.opponent.team.trigram }} {% endfor %} \\ \hline \hline | ||||
| {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.opponent.team.trigram }} {% endfor %} \\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{6}{3mm}{\bf \begin{turn}{90}ÉCRIT\end{turn}} &\multirow{4}{24mm}{Partie scientifique} & Recul et esprit critique par rapport à la solution proposée & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| @@ -102,7 +102,7 @@ | ||||
|  | ||||
| %%%%%%%%%%%%%%%%%%%%%%RAPPORTEUR.RICE | ||||
| \begin{tabular}{|c|p{24mm}|p{11cm}|c{% for passage in passages.all %}|p{2cm}{% endfor %}|}\hline | ||||
| \multicolumn{4}{|l|}{Læ {\bf Rapporteur⋅rice} \normalsize \'evalue le d\'ebat entre læ D\'efenseur⋅se et l'Opposant⋅e.} {% for passage in passages.all %}& P.{{ forloop.counter }} - {{ passage.reviewer.team.trigram }} {% endfor %}\\ \hline \hline | ||||
| \multicolumn{4}{|l|}{Læ {\bf Rapporteur⋅rice} \normalsize \'evalue le d\'ebat entre læ D\'efenseur⋅se et l'Opposant⋅e.} {% for passage in passages.all %}& Pb. {{ passage.solution_number }} - {{ passage.reviewer.team.trigram }} {% endfor %}\\ \hline \hline | ||||
|  | ||||
| %ECRIT | ||||
| \multirow{6}{3mm}{\bf \begin{turn}{90}ÉCRIT\end{turn}} &\multirow{4}{24mm}{Partie scientifique} & Recul et esprit critique par rapport à la solution proposée & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| @@ -115,7 +115,7 @@ | ||||
| \multirow{9}{3mm}{\bf \begin{turn}{90}ORAL\end{turn}} & \multirow{5}{24mm}{Questions et discours de læ rapporteur⋅rice} & \footnotesize Faire prendre de la hauteur au débat (par les sujets abordés, la pertinence des questions posées, les points soulevés, gestion du temps) & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| && \footnotesize Créer un échange constructif entre les participants (formulation des questions, réaction aux réponses, articulation entre les questions, circulation de la parole) & [0,3] {{ esp|safe }}\\ \cline{3-{{ passages.count|add:4 }}} | ||||
| && Capacité à évaluer la qualité des échanges (Défenseur⋅se-Opposant⋅e et à trois) & [0,2] {{ esp|safe }}\\ \cline{2-{{ passages.count|add:4 }}} | ||||
| && Réponses aux questions de læ Rapporteur⋅rice et du jury (fond et capacité à faire avancer le débat) & [0,2] {{ esp|safe }}\\ \cline{2-{{ passages.count|add:4 }}} | ||||
| && Réponses aux questions du jury (fond et capacité à faire avancer le débat) & [0,2] {{ esp|safe }}\\ \cline{2-{{ passages.count|add:4 }}} | ||||
| & Malus & Attitude irrespectueuse ? & [-3,0] {{ esp|safe }}\\ \cline{2-{{ passages.count|add:4 }}} | ||||
| &\multicolumn{3}{|l|}{\bf TOTAL ORAL (/10)} {{ esp|safe }}\\ \hline | ||||
| \end{tabular} | ||||
|   | ||||
| @@ -23,44 +23,80 @@ | ||||
|                     <dd class="col-sm-6">{% if tournament.price %}{{ tournament.price }} €{% else %}{% trans "Free" %}{% endif %}</dd> | ||||
|                 {% endif %} | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'remote'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "remote"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.remote|yesno }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'dates'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "dates"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{% trans "From" %} {{ tournament.date_start }} {% trans "to" %} {{ tournament.date_end }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'date of registration closing'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "date of registration closing"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.inscription_limit }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'date of maximal solution submission'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "date of maximal solution submission"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.solution_limit }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'date of the random draw'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "date of the random draw"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.solutions_draw }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'date of maximal written reviews submission for the first round'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "date of maximal written reviews submission for the first round"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.reviews_first_phase_limit }}</dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'date of maximal written reviews submission for the second round'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "Solutions available for the second round" %}</dt> | ||||
|                 <dd class="col-sm-6"> | ||||
|                     {{ tournament.solutions_available_second_phase|yesno }} | ||||
|                     {% if user.is_authenticated and user.registration in tournament.organizers_and_presidents.all %} | ||||
|                         {% now 'Y-m-d' as today %} | ||||
|                         {% if not tournament.solutions_available_second_phase %} | ||||
|                             {% if today >= tournament.date_first_phase|date:"Y-m-d" %} | ||||
|                                 <a href="{% url 'participation:tournament_publish_solutions' pk=tournament.pk round=2 %}" class="btn btn-sm btn-info"><i class="fas fa-eye"></i> {% trans "Publish" %}</a> | ||||
|                             {% endif %} | ||||
|                         {% else %} | ||||
|                             {% if today <= tournament.date_second_phase|date:"Y-m-d" %} | ||||
|                                 <a href="{% url 'participation:tournament_publish_solutions' pk=tournament.pk round=2 %}?hide" class="btn btn-sm bg-danger"><i class="fas fa-eye-slash"></i> {% trans "Unpublish" %}</a> | ||||
|                             {% endif %} | ||||
|                         {% endif %} | ||||
|                     {% endif %} | ||||
|                 </dd> | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "date of maximal written reviews submission for the second round"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.reviews_second_phase_limit }}</dd> | ||||
|  | ||||
|                 {% if TFJM.APP == "ETEAM" %} | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans 'date of maximal written reviews submission for the third round'|capfirst %}</dt> | ||||
|                 {% if TFJM.NB_ROUNDS == 3 %} | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans "Solutions available for the third round" %}</dt> | ||||
|                     <dd class="col-sm-6"> | ||||
|                         {{ tournament.solutions_available_third_phase|yesno }} | ||||
|                         {% if tournament.solutions_available_second_phase and user.is_authenticated and user.registration in tournament.organizers_and_presidents.all %} | ||||
|                             {% now 'Y-m-d' as today %} | ||||
|                             {% if not tournament.solutions_available_third_phase %} | ||||
|                                 {% if today >= tournament.date_second_phase|date:"Y-m-d" %} | ||||
|                                     <a href="{% url 'participation:tournament_publish_solutions' pk=tournament.pk round=3 %}" class="btn btn-sm btn-info"><i class="fas fa-eye"></i> {% trans "Publish" %}</a> | ||||
|                                 {% endif %} | ||||
|                             {% else %} | ||||
|                                 {% if today <= tournament.date_third_phase|date:"Y-m-d" %} | ||||
|                                     <a href="{% url 'participation:tournament_publish_solutions' pk=tournament.pk round=3 %}?hide" class="btn btn-sm bg-danger"><i class="fas fa-eye-slash"></i> {% trans "Unpublish" %}</a> | ||||
|                                 {% endif %} | ||||
|                             {% endif %} | ||||
|                         {% endif %} | ||||
|                     </dd> | ||||
|  | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans "date of maximal written reviews submission for the third round"|capfirst %}</dt> | ||||
|                     <dd class="col-sm-6">{{ tournament.reviews_third_phase_limit }}</dd> | ||||
|                 {% endif %} | ||||
|  | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans 'description'|capfirst %}</dt> | ||||
|                 <dt class="col-sm-6 text-sm-end">{% trans "description"|capfirst %}</dt> | ||||
|                 <dd class="col-sm-6">{{ tournament.description }}</dd> | ||||
|  | ||||
|                 {% if TFJM.ML_MANAGEMENT %} | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans 'To contact organizers' %}</dt> | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans "To contact organizers" %}</dt> | ||||
|                     <dd class="col-sm-6"><a href="mailto:{{ tournament.organizers_email }}">{{ tournament.organizers_email }}</a></dd> | ||||
|  | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans 'To contact juries' %}</dt> | ||||
|                     <dd class="col-sm-6"><a href="mailto:{{ tournament.jurys_email }}">{{ tournament.jurys_email }}</a></dd> | ||||
|                     {% if user.is_authenticated and user.registration.is_volunteer %} | ||||
|                         <dt class="col-sm-6 text-sm-end">{% trans "To contact juries" %}</dt> | ||||
|                         <dd class="col-sm-6"><a href="mailto:{{ tournament.jurys_email }}">{{ tournament.jurys_email }}</a></dd> | ||||
|  | ||||
|                     <dt class="col-sm-6 text-sm-end">{% trans 'To contact valid teams' %}</dt> | ||||
|                     <dd class="col-sm-6"><a href="mailto:{{ tournament.teams_email }}">{{ tournament.teams_email }}</a></dd> | ||||
|                         <dt class="col-sm-6 text-sm-end">{% trans "To contact valid teams" %}</dt> | ||||
|                         <dd class="col-sm-6"><a href="mailto:{{ tournament.teams_email }}">{{ tournament.teams_email }}</a></dd> | ||||
|                     {% endif %} | ||||
|                 {% endif %} | ||||
|             </dl> | ||||
|         </div> | ||||
| @@ -199,7 +235,7 @@ | ||||
|                     </div> | ||||
|                 </div> | ||||
|             {% endif %} | ||||
|         </div> | ||||
|             </div> | ||||
|     {% endif %} | ||||
|  | ||||
|     {% if user.registration.is_admin or user.registration in tournament.organizers.all %} | ||||
|   | ||||
| @@ -12,7 +12,7 @@ from .views import CreateTeamView, FinalNotationSheetTemplateView, GSheetNotific | ||||
|     TeamAuthorizationsView, TeamDetailView, TeamLeaveView, TeamListView, TeamUpdateView, \ | ||||
|     TeamUploadMotivationLetterView, TournamentCreateView, TournamentDetailView, TournamentExportCSVView, \ | ||||
|     TournamentHarmonizeNoteView, TournamentHarmonizeView, TournamentListView, TournamentPaymentsView, \ | ||||
|     TournamentPublishNotesView, TournamentUpdateView, WrittenReviewUploadView | ||||
|     TournamentPublishNotesView, TournamentPublishSolutionsView, TournamentUpdateView, WrittenReviewUploadView | ||||
|  | ||||
|  | ||||
| app_name = "participation" | ||||
| @@ -48,6 +48,8 @@ urlpatterns = [ | ||||
|          name="tournament_notation_sheets"), | ||||
|     path("tournament/<int:pk>/notation/notifications/", GSheetNotificationsView.as_view(), | ||||
|          name="tournament_gsheet_notifications"), | ||||
|     path("tournament/<int:pk>/publish-solutions/<int:round>/", TournamentPublishSolutionsView.as_view(), | ||||
|          name="tournament_publish_solutions"), | ||||
|     path("tournament/<int:pk>/publish-notes/<int:round>/", TournamentPublishNotesView.as_view(), | ||||
|          name="tournament_publish_notes"), | ||||
|     path("tournament/<int:pk>/harmonize/<int:round>/", TournamentHarmonizeView.as_view(), | ||||
|   | ||||
| @@ -672,7 +672,7 @@ class TournamentPaymentsView(VolunteerMixin, SingleTableMixin, DetailView): | ||||
|         if self.object.final: | ||||
|             payments = Payment.objects.filter(final=True) | ||||
|         else: | ||||
|             payments = Payment.objects.filter(registrations__team__participation__tournament=self.get_object()) | ||||
|             payments = Payment.objects.filter(registrations__team__participation__tournament=self.get_object(), final=False) | ||||
|         return payments.annotate(team_id=F('registrations__team')).order_by('-valid', 'registrations__team__trigram') \ | ||||
|             .distinct().all() | ||||
|  | ||||
| @@ -747,12 +747,12 @@ class TournamentPublishNotesView(VolunteerMixin, SingleObjectMixin, RedirectView | ||||
|             return self.handle_no_permission() | ||||
|         tournament = self.get_object() | ||||
|         reg = request.user.registration | ||||
|         if not reg.is_admin and (not reg.is_volunteer or tournament not in reg.organized_tournaments.all()): | ||||
|         if not reg.is_volunteer or reg not in tournament.organizers_and_presidents.all(): | ||||
|             return self.handle_no_permission() | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         if int(kwargs["round"]) not in (1, 2): | ||||
|         if int(kwargs["round"]) not in range(1, settings.NB_ROUNDS + 1): | ||||
|             raise Http404 | ||||
|  | ||||
|         tournament = Tournament.objects.get(pk=kwargs["pk"]) | ||||
| @@ -767,6 +767,45 @@ class TournamentPublishNotesView(VolunteerMixin, SingleObjectMixin, RedirectView | ||||
|         return reverse_lazy("participation:tournament_detail", args=(kwargs['pk'],)) | ||||
|  | ||||
|  | ||||
| class TournamentPublishSolutionsView(VolunteerMixin, SingleObjectMixin, RedirectView): | ||||
|     """ | ||||
|     On rend les solutions du tour suivant accessibles aux équipes. | ||||
|     """ | ||||
|     model = Tournament | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         """ | ||||
|         Les admins, orgas et PJ peuvent rendre les solutions accessibles. | ||||
|         """ | ||||
|         if not request.user.is_authenticated: | ||||
|             return self.handle_no_permission() | ||||
|         tournament = self.get_object() | ||||
|         reg = request.user.registration | ||||
|         if not reg.is_volunteer or reg not in tournament.organizers_and_presidents.all(): | ||||
|             return self.handle_no_permission() | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         if int(kwargs["round"]) not in range(2, settings.NB_ROUNDS + 1): | ||||
|             raise Http404 | ||||
|  | ||||
|         tournament = Tournament.objects.get(pk=kwargs["pk"]) | ||||
|         publish_solutions = 'hide' not in request.GET | ||||
|         if int(kwargs['round']) == 2: | ||||
|             tournament.solutions_available_second_phase = publish_solutions | ||||
|         elif int(kwargs['round']) == 3: | ||||
|             tournament.solutions_available_third_phase = publish_solutions | ||||
|         tournament.save() | ||||
|         if 'hide' not in request.GET: | ||||
|             messages.success(request, _("Solutions are now available to teams!")) | ||||
|         else: | ||||
|             messages.warning(request, _("Solutions are not available to teams anymore.")) | ||||
|         return super().get(request, *args, **kwargs) | ||||
|  | ||||
|     def get_redirect_url(self, *args, **kwargs): | ||||
|         return reverse_lazy("participation:tournament_detail", args=(kwargs['pk'],)) | ||||
|  | ||||
|  | ||||
| class TournamentHarmonizeView(VolunteerMixin, DetailView): | ||||
|     """ | ||||
|     Harmonize the notes of a tournament. | ||||
| @@ -779,7 +818,7 @@ class TournamentHarmonizeView(VolunteerMixin, DetailView): | ||||
|             return self.handle_no_permission() | ||||
|         tournament = self.get_object() | ||||
|         reg = request.user.registration | ||||
|         if not reg.is_admin and (not reg.is_volunteer or tournament not in reg.organized_tournaments.all()): | ||||
|         if not reg.is_volunteer or reg not in tournament.organizers_and_presidents.all(): | ||||
|             return self.handle_no_permission() | ||||
|         if self.kwargs['round'] not in range(1, settings.NB_ROUNDS + 1): | ||||
|             raise Http404 | ||||
| @@ -812,7 +851,7 @@ class TournamentHarmonizeNoteView(VolunteerMixin, DetailView): | ||||
|             return self.handle_no_permission() | ||||
|         tournament = self.get_object() | ||||
|         reg = request.user.registration | ||||
|         if not reg.is_admin and (not reg.is_volunteer or tournament not in reg.organized_tournaments.all()): | ||||
|         if not reg.is_volunteer or reg not in tournament.organizers_and_presidents.all(): | ||||
|             return self.handle_no_permission() | ||||
|         if self.kwargs['round'] not in range(1, settings.NB_ROUNDS + 1) \ | ||||
|                 or self.kwargs['action'] not in ('add', 'remove') \ | ||||
| @@ -852,7 +891,7 @@ class SelectTeamFinalView(VolunteerMixin, DetailView): | ||||
|             return self.handle_no_permission() | ||||
|         tournament = self.get_object() | ||||
|         reg = request.user.registration | ||||
|         if not reg.is_admin and (not reg.is_volunteer or tournament not in reg.organized_tournaments.all()): | ||||
|         if not reg.is_volunteer or reg not in tournament.organizers_and_presidents.all(): | ||||
|             return self.handle_no_permission() | ||||
|         participation_qs = tournament.participations.filter(pk=self.kwargs["participation_id"]) | ||||
|         if not participation_qs.exists(): | ||||
| @@ -1003,17 +1042,14 @@ class SolutionsDownloadView(VolunteerMixin, View): | ||||
|                 return super().dispatch(request, *args, **kwargs) | ||||
|         elif 'tournament_id' in kwargs: | ||||
|             tournament = Tournament.objects.get(pk=kwargs["tournament_id"]) | ||||
|             if reg.is_volunteer \ | ||||
|                     and (tournament in reg.organized_tournaments.all() | ||||
|                          or reg.pools_presided.filter(tournament=tournament).exists()): | ||||
|             if reg.is_volunteer and reg in tournament.organizers_and_presidents.all(): | ||||
|                 return super().dispatch(request, *args, **kwargs) | ||||
|         else: | ||||
|             pool = Pool.objects.get(pk=kwargs["pool_id"]) | ||||
|             tournament = pool.tournament | ||||
|             if reg.is_volunteer \ | ||||
|                     and (reg in tournament.organizers.all() | ||||
|                          or reg in pool.juries.all() | ||||
|                          or reg.pools_presided.filter(tournament=tournament).exists()): | ||||
|                     and (reg in tournament.organizers_and_presidents.all() | ||||
|                          or reg in pool.juries.all()): | ||||
|                 return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|         return self.handle_no_permission() | ||||
| @@ -2001,7 +2037,7 @@ class PassageDetailView(LoginRequiredMixin, DetailView): | ||||
|         reg = request.user.registration | ||||
|         passage = self.get_object() | ||||
|         if reg.is_admin or reg.is_volunteer \ | ||||
|                 and (self.get_object().pool.tournament in reg.organized_tournaments.all() | ||||
|                 and (reg in self.get_object().pool.tournament.organizers_and_presidents.all() | ||||
|                      or reg in passage.pool.juries.all() | ||||
|                      or reg.pools_presided.filter(tournament=passage.pool.tournament).exists()) \ | ||||
|                 or reg.participates and reg.team \ | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| {{ object.user.last_name }} | ||||
| {{ object.user.first_name }} | ||||
| {{ object.user.email }} | ||||
| @@ -70,7 +70,7 @@ class Survey(models.Model): | ||||
|             teams = Team.objects.filter(participation__valid=True) | ||||
|             if self.tournament: | ||||
|                 teams = teams.filter(participation__tournament=self.tournament) | ||||
|             return teams.all() | ||||
|             return teams.order_by('participation__tournament__name', 'trigram').all() | ||||
|         else: | ||||
|             if self.invite_coaches: | ||||
|                 registrations = ParticipantRegistration.objects.filter(team__participation__valid=True) | ||||
| @@ -78,7 +78,7 @@ class Survey(models.Model): | ||||
|                 registrations = StudentRegistration.objects.filter(team__participation__valid=True) | ||||
|             if self.tournament: | ||||
|                 registrations = registrations.filter(team__participation__tournament=self.tournament) | ||||
|             return registrations.all() | ||||
|             return registrations.order_by('team__participation__tournament__name', 'team__trigram').all() | ||||
|  | ||||
|     @property | ||||
|     def completed(self): | ||||
|   | ||||
| @@ -48,6 +48,7 @@ | ||||
|         <thead> | ||||
|             <tr> | ||||
|                 <th>{% trans "participant"|capfirst %}</th> | ||||
|                 <th>{% trans "tournament"|capfirst %}</th> | ||||
|                 <th>{% trans "completed"|capfirst %}</th> | ||||
|             </tr> | ||||
|         </thead> | ||||
| @@ -56,8 +57,10 @@ | ||||
|                 <tr class="{% if participant in survey.completed.all %}table-success{% else %}table-danger{% endif %}"> | ||||
|                     {% if survey.invite_team %} | ||||
|                         <td>{% trans "Team" %} {{ participant.name }} ({{ participant.trigram }})</td> | ||||
|                         <td>{{ participant.participation.tournament.name }}</td> | ||||
|                     {% else %} | ||||
|                         <td>{{ participant.user.first_name }} {{ participant.user.last_name }} ({% trans "team" %} {{ participant.team.trigram }})</td> | ||||
|                         <td>{{ participant.team.participation.tournament.name }}</td> | ||||
|                     {% endif %} | ||||
|                     {% if participant in survey.completed.all %} | ||||
|                         <td>{% trans "Yes" %}</td> | ||||
|   | ||||
| @@ -213,6 +213,7 @@ STATICFILES_FINDERS = ( | ||||
|  | ||||
| PIPELINE = { | ||||
|     'DISABLE_WRAPPER': True, | ||||
|     'JS_COMPRESSOR': 'pipeline.compressors.uglifyjs.UglifyJSCompressor', | ||||
|     'JAVASCRIPT': { | ||||
|         'main': { | ||||
|             'source_filenames': ( | ||||
|   | ||||
| @@ -23,7 +23,7 @@ | ||||
|         </div> | ||||
|  | ||||
|         <div id="sidebar-card" class="collapse d-lg-block"> | ||||
|             <div class="card-body"> | ||||
|             <div class="card-body px-2 py-1"> | ||||
|                 {% for information in user.registration.important_informations %} | ||||
|                     <div class="card my-2"> | ||||
|                         <div class="card-header bg-dark-subtle"> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user