mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-25 14:23:07 +02:00 
			
		
		
		
	Compare commits
	
		
			8 Commits
		
	
	
		
			beta
			...
			5c3e63b204
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 5c3e63b204 | ||
|  | e6f3084588 | ||
|  | 145e55da75 | ||
|  | d3ba95cdca | ||
|  | 8ffb0ebb56 | ||
|  | 5038af9e34 | ||
|  | 819b4214c9 | ||
|  | b8a93b0b75 | 
| @@ -8,7 +8,7 @@ variables: | |||||||
|   GIT_SUBMODULE_STRATEGY: recursive |   GIT_SUBMODULE_STRATEGY: recursive | ||||||
|  |  | ||||||
| # Ubuntu 22.04 | # Ubuntu 22.04 | ||||||
| py310-django52: | py310-django42: | ||||||
|   stage: test |   stage: test | ||||||
|   image: ubuntu:22.04 |   image: ubuntu:22.04 | ||||||
|   before_script: |   before_script: | ||||||
| @@ -22,10 +22,10 @@ py310-django52: | |||||||
|         python3-djangorestframework python3-django-oauth-toolkit python3-psycopg2 python3-pil |         python3-djangorestframework python3-django-oauth-toolkit python3-psycopg2 python3-pil | ||||||
|         python3-babel python3-lockfile python3-pip python3-phonenumbers python3-memcache |         python3-babel python3-lockfile python3-pip python3-phonenumbers python3-memcache | ||||||
|         python3-bs4 python3-setuptools tox texlive-xetex |         python3-bs4 python3-setuptools tox texlive-xetex | ||||||
|   script: tox -e py310-django52 |   script: tox -e py310-django42 | ||||||
|  |  | ||||||
| # Debian Bookworm | # Debian Bookworm | ||||||
| py311-django52: | py311-django42: | ||||||
|   stage: test |   stage: test | ||||||
|   image: debian:bookworm |   image: debian:bookworm | ||||||
|   before_script: |   before_script: | ||||||
| @@ -37,7 +37,7 @@ py311-django52: | |||||||
|         python3-djangorestframework python3-django-oauth-toolkit python3-psycopg2 python3-pil |         python3-djangorestframework python3-django-oauth-toolkit python3-psycopg2 python3-pil | ||||||
|         python3-babel python3-lockfile python3-pip python3-phonenumbers python3-memcache |         python3-babel python3-lockfile python3-pip python3-phonenumbers python3-memcache | ||||||
|         python3-bs4 python3-setuptools tox texlive-xetex |         python3-bs4 python3-setuptools tox texlive-xetex | ||||||
|   script: tox -e py311-django52 |   script: tox -e py311-django42 | ||||||
|  |  | ||||||
| linters: | linters: | ||||||
|   stage: quality-assurance |   stage: quality-assurance | ||||||
|   | |||||||
| @@ -38,6 +38,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
| </a> | </a> | ||||||
|  |  | ||||||
| <input id="alias" type="text" class="form-control" placeholder="Nom/note ..."> | <input id="alias" type="text" class="form-control" placeholder="Nom/note ..."> | ||||||
|  | <button id="trigger" class="btn btn-secondary">Click me !</button> | ||||||
|  |  | ||||||
| <hr> | <hr> | ||||||
|  |  | ||||||
| @@ -63,15 +64,46 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|         refreshBalance(); |         refreshBalance(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     function process_qrcode() { | ||||||
|  |         let name = alias_obj.val(); | ||||||
|  |         $.get("/api/note/note?search=" + name + "&format=json").done( | ||||||
|  |             function (res) { | ||||||
|  |                 let note = res.results[0]; | ||||||
|  |                 $.post("/api/activity/entry/?format=json", { | ||||||
|  |                     csrfmiddlewaretoken: CSRF_TOKEN, | ||||||
|  |                     activity: {{ activity.id }}, | ||||||
|  |                     note: note.id, | ||||||
|  |                     guest: null | ||||||
|  |                 }).done(function () { | ||||||
|  |                     addMsg(interpolate(gettext( | ||||||
|  |                         "Entry made for %s whose balance is %s €"), | ||||||
|  |                         [note.name, note.balance / 100]), "success", 4000); | ||||||
|  |                     reloadTable(true); | ||||||
|  |                 }).fail(function (xhr) { | ||||||
|  |                     errMsg(xhr.responseJSON, 4000); | ||||||
|  |                 }); | ||||||
|  |             }).fail(function (xhr) { | ||||||
|  |                 errMsg(xhr.responseJSON, 4000); | ||||||
|  |             }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     alias_obj.keyup(function(event) { |     alias_obj.keyup(function(event) { | ||||||
|         let code = event.originalEvent.keyCode |         let code = event.originalEvent.keyCode | ||||||
|         if (65 <= code <= 122 || code === 13) { |         if (65 <= code <= 122 || code === 13) { | ||||||
|             debounce(reloadTable)() |             debounce(reloadTable)() | ||||||
|         } |         } | ||||||
|  |         if (code === 0) | ||||||
|  |             process_qrcode(); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     $(document).ready(init); |     $(document).ready(init); | ||||||
|  |  | ||||||
|  |     alias_obj2 = document.getElementById("alias"); | ||||||
|  |     $("#trigger").click(function (e) { | ||||||
|  |         addMsg("Clicked", "success", 1000); | ||||||
|  |         alias_obj.val(alias_obj.val() + "\0"); | ||||||
|  |         alias_obj2.dispatchEvent(new KeyboardEvent('keyup')); | ||||||
|  |     }) | ||||||
|     function init() { |     function init() { | ||||||
|         $(".table-row").click(function (e) { |         $(".table-row").click(function (e) { | ||||||
|             let target = e.target.parentElement; |             let target = e.target.parentElement; | ||||||
| @@ -168,4 +200,4 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| </script> | </script> | ||||||
| {% endblock %} | {% endblock %} | ||||||
|   | |||||||
| @@ -7,52 +7,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
| {% load i18n %} | {% load i18n %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| <div class="card bg-light"> | {{ block.super }} | ||||||
|   <h3 class="card-header text-center"> |  | ||||||
|       {{ title }} |  | ||||||
|   </h3> |  | ||||||
|   <div class="card-body"> |  | ||||||
|     <style> |  | ||||||
|       input[type=number]::-webkit-inner-spin-button, |  | ||||||
|       input[type=number]::-webkit-outer-spin-button { |  | ||||||
|           -webkit-appearance: none; |  | ||||||
|           margin: 0; |  | ||||||
|       } |  | ||||||
|       input[type=number] { |  | ||||||
|           appearance: textfield; |  | ||||||
|           padding: 6px; |  | ||||||
|           border: 1px solid #ccc; |  | ||||||
|           border-radius: 4px; |  | ||||||
|           width: 100px; |  | ||||||
|       } |  | ||||||
|     </style> |  | ||||||
|     <div class="d-flex align-items-center" style="max-width: 300px;"> |  | ||||||
|       <form method="get" action="{% url 'food:redirect_view' %}" class="d-flex w-100"> |  | ||||||
|         <input type="number" name="slug" placeholder="QR-code" required class="form-control form-control-sm" style="max-width: 120px;"> |  | ||||||
|         <button type="submit" class="btn btn-sm btn-primary">{% trans "View food" %}</button> |  | ||||||
|       </form> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
|   <div class="card-body"> |  | ||||||
|       <input id="searchbar" type="text" class="form-control" |  | ||||||
|           placeholder="{% trans "Search by attribute such as name..." %}"> |  | ||||||
|   </div> |  | ||||||
|  |  | ||||||
|   {% block extra_inside_card %} |  | ||||||
|   {% endblock %} |  | ||||||
|  |  | ||||||
|   <div id="dynamic-table"> |  | ||||||
|       {% if table.data %} |  | ||||||
|       {% render_table table %} |  | ||||||
|       {% else %} |  | ||||||
|       <div class="card-body"> |  | ||||||
|           <div class="alert alert-warning"> |  | ||||||
|               {% trans "There is no results." %} |  | ||||||
|           </div> |  | ||||||
|       </div> |  | ||||||
|       {% endif %} |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| <br> | <br> | ||||||
| <div class="card bg-light mb-3"> | <div class="card bg-light mb-3"> | ||||||
|   <h3 class="card-header text-center"> |   <h3 class="card-header text-center"> | ||||||
| @@ -113,20 +68,4 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|   {% endfor %} |   {% endfor %} | ||||||
|   {% endif %} |   {% endif %} | ||||||
| </div> | </div> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
| <script> |  | ||||||
|   document.addEventListener('DOMContentLoaded', function() { |  | ||||||
|       document.getElementById('goButton').addEventListener('click', function(event) { |  | ||||||
|           event.preventDefault(); |  | ||||||
|           const slug = document.getElementById('slugInput').value; |  | ||||||
|           if (slug && !isNaN(slug)) { |  | ||||||
|               window.location.href = `/food/${slug}/`; |  | ||||||
|           } else { |  | ||||||
|               alert("Veuillez entrer un nombre valide."); |  | ||||||
|           } |  | ||||||
|       }); |  | ||||||
|   }); |  | ||||||
|   </script> |  | ||||||
|  |  | ||||||
| {% endblock %} |  | ||||||
|   | |||||||
| @@ -18,5 +18,4 @@ urlpatterns = [ | |||||||
|     path('detail/basic/<int:pk>', views.BasicFoodDetailView.as_view(), name='basicfood_view'), |     path('detail/basic/<int:pk>', views.BasicFoodDetailView.as_view(), name='basicfood_view'), | ||||||
|     path('detail/transformed/<int:pk>', views.TransformedFoodDetailView.as_view(), name='transformedfood_view'), |     path('detail/transformed/<int:pk>', views.TransformedFoodDetailView.as_view(), name='transformedfood_view'), | ||||||
|     path('add/ingredient/<int:pk>', views.AddIngredientView.as_view(), name='add_ingredient'), |     path('add/ingredient/<int:pk>', views.AddIngredientView.as_view(), name='add_ingredient'), | ||||||
|     path('redirect/', views.QRCodeRedirectView.as_view(), name='redirect_view'), |  | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ from django.db.models import Q | |||||||
| from django.http import HttpResponseRedirect, Http404 | from django.http import HttpResponseRedirect, Http404 | ||||||
| from django.views.generic import DetailView, UpdateView, CreateView | from django.views.generic import DetailView, UpdateView, CreateView | ||||||
| from django.views.generic.list import ListView | from django.views.generic.list import ListView | ||||||
| from django.views.generic.base import RedirectView |  | ||||||
| from django.urls import reverse_lazy | from django.urls import reverse_lazy | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| @@ -508,14 +507,3 @@ class TransformedFoodDetailView(FoodDetailView): | |||||||
|         if Food.objects.filter(pk=kwargs['pk']).count() == 1: |         if Food.objects.filter(pk=kwargs['pk']).count() == 1: | ||||||
|             kwargs['stop_redirect'] = (Food.objects.get(pk=kwargs['pk']).polymorphic_ctype.model == 'transformedfood') |             kwargs['stop_redirect'] = (Food.objects.get(pk=kwargs['pk']).polymorphic_ctype.model == 'transformedfood') | ||||||
|         return super().get(*args, **kwargs) |         return super().get(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeRedirectView(RedirectView): |  | ||||||
|     """ |  | ||||||
|     Redirects to the QR code creation page from Food List |  | ||||||
|     """ |  | ||||||
|     def get_redirect_url(self, *args, **kwargs): |  | ||||||
|         slug = self.request.GET.get('slug') |  | ||||||
|         if slug: |  | ||||||
|             return reverse_lazy('food:qrcode_create', kwargs={'slug': slug}) |  | ||||||
|         return reverse_lazy('food:list') |  | ||||||
|   | |||||||
| @@ -60,7 +60,10 @@ | |||||||
| {% if user_object.pk == user.pk %} | {% if user_object.pk == user.pk %} | ||||||
|     <div class="text-center"> |     <div class="text-center"> | ||||||
|         <a class="small badge badge-secondary" href="{% url 'member:auth_token' %}"> |         <a class="small badge badge-secondary" href="{% url 'member:auth_token' %}"> | ||||||
|             <i class="fa fa-cogs"></i>{% trans 'API token' %} |             <i class="fa fa-cogs"></i> {% trans 'API token' %} | ||||||
|  |         </a> | ||||||
|  |         <a class="small badge badge-secondary" href="{% url 'member:qr_code' user_object.pk %}"> | ||||||
|  |             <i class="fa fa-qrcode"></i> {% trans 'QR Code' %} | ||||||
|         </a> |         </a> | ||||||
|     </div> |     </div> | ||||||
| {% endif %} | {% endif %} | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								apps/member/templates/member/qr_code.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								apps/member/templates/member/qr_code.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  | {% comment %} | ||||||
|  | SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | {% endcomment %} | ||||||
|  | {% load i18n %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | <div class="card bg-light"> | ||||||
|  |   	<h3 class="card-header text-center"> | ||||||
|  | 		{% trans "QR Code for" %} {{ user_object.username }} ({{ user_object.first_name }} {{user_object.last_name }}) | ||||||
|  |   	</h3> | ||||||
|  |   	<div class="text-center" id="qrcode"> | ||||||
|  |   	</div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block extrajavascript %} | ||||||
|  | <script src="https://cdnjs.cloudflare.com/ajax/libs/qrcodejs/1.0.0/qrcode.min.js" integrity="sha512-CNgIRecGo7nphbeZ04Sc13ka07paqdeTu0WR1IM4kNcpmBAUSHSQX0FslNhTDadL4O5SAGapGt4FodqL8My0mA==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||||
|  | <script> | ||||||
|  | 	var qrc = new QRCode(document.getElementById("qrcode"), { | ||||||
|  | 		text: "{{ user_object.pk }}\0", | ||||||
|  | 		width: 1024, | ||||||
|  | 		height: 1024 | ||||||
|  | 	}); | ||||||
|  | </script> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block extracss %} | ||||||
|  | <style> | ||||||
|  | img { | ||||||
|  |     width: 100% | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | {% endblock %} | ||||||
| @@ -44,7 +44,7 @@ class TemplateLoggedInTests(TestCase): | |||||||
|         self.assertRedirects(response, settings.LOGIN_REDIRECT_URL, 302, 302) |         self.assertRedirects(response, settings.LOGIN_REDIRECT_URL, 302, 302) | ||||||
|  |  | ||||||
|     def test_logout(self): |     def test_logout(self): | ||||||
|         response = self.client.post(reverse("logout")) |         response = self.client.get(reverse("logout")) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|  |  | ||||||
|     def test_admin_index(self): |     def test_admin_index(self): | ||||||
|   | |||||||
| @@ -25,4 +25,5 @@ urlpatterns = [ | |||||||
|     path('user/<int:pk>/aliases/', views.ProfileAliasView.as_view(), name="user_alias"), |     path('user/<int:pk>/aliases/', views.ProfileAliasView.as_view(), name="user_alias"), | ||||||
|     path('user/<int:pk>/trust', views.ProfileTrustView.as_view(), name="user_trust"), |     path('user/<int:pk>/trust', views.ProfileTrustView.as_view(), name="user_trust"), | ||||||
|     path('manage-auth-token/', views.ManageAuthTokens.as_view(), name='auth_token'), |     path('manage-auth-token/', views.ManageAuthTokens.as_view(), name='auth_token'), | ||||||
|  |     path('user/<int:pk>/qr_code/', views.QRCodeView.as_view(), name='qr_code'), | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -402,6 +402,14 @@ class ManageAuthTokens(LoginRequiredMixin, TemplateView): | |||||||
|         context['token'] = Token.objects.get_or_create(user=self.request.user)[0] |         context['token'] = Token.objects.get_or_create(user=self.request.user)[0] | ||||||
|         return context |         return context | ||||||
|  |  | ||||||
|  | class QRCodeView(LoginRequiredMixin, DetailView): | ||||||
|  |     """ | ||||||
|  |     Affiche le QR Code | ||||||
|  |     """ | ||||||
|  |     model = User | ||||||
|  |     context_object_name = "user_object" | ||||||
|  |     template_name = "member/qr_code.html" | ||||||
|  |     extra_context = {"title": _("QR Code")} | ||||||
|  |  | ||||||
| # ******************************* # | # ******************************* # | ||||||
| #              CLUB               # | #              CLUB               # | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ def register_note_urls(router, path): | |||||||
|     router.register(path + '/note', NotePolymorphicViewSet) |     router.register(path + '/note', NotePolymorphicViewSet) | ||||||
|     router.register(path + '/alias', AliasViewSet) |     router.register(path + '/alias', AliasViewSet) | ||||||
|     router.register(path + '/trust', TrustViewSet) |     router.register(path + '/trust', TrustViewSet) | ||||||
|     router.register(path + '/consumer', ConsumerViewSet, basename='alias2') |     router.register(path + '/consumer', ConsumerViewSet) | ||||||
|  |  | ||||||
|     router.register(path + '/transaction/category', TemplateCategoryViewSet) |     router.register(path + '/transaction/category', TemplateCategoryViewSet) | ||||||
|     router.register(path + '/transaction/transaction', TransactionViewSet) |     router.register(path + '/transaction/transaction', TransactionViewSet) | ||||||
|   | |||||||
| @@ -18,18 +18,7 @@ class PermissionScopes(BaseScopes): | |||||||
|     and can be useful to make queries through the API with limited privileges. |     and can be useful to make queries through the API with limited privileges. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def get_all_scopes(self, **kwargs): |     def get_all_scopes(self): | ||||||
|         scopes = {} |  | ||||||
|         if 'scopes' in kwargs: |  | ||||||
|             for scope in kwargs['scopes']: |  | ||||||
|                 if scope == 'openid': |  | ||||||
|                     scopes['openid'] = "OpenID Connect" |  | ||||||
|                 else: |  | ||||||
|                     p = Permission.objects.get(id=scope.split('_')[0]) |  | ||||||
|                     club = Club.objects.get(id=scope.split('_')[1]) |  | ||||||
|                     scopes[scope] = f"{p.description} (club {club.name})" |  | ||||||
|             return scopes |  | ||||||
|  |  | ||||||
|         scopes = {f"{p.id}_{club.id}": f"{p.description} (club {club.name})" |         scopes = {f"{p.id}_{club.id}": f"{p.description} (club {club.name})" | ||||||
|                   for p in Permission.objects.all() for club in Club.objects.all()} |                   for p in Permission.objects.all() for club in Club.objects.all()} | ||||||
|         scopes['openid'] = "OpenID Connect" |         scopes['openid'] = "OpenID Connect" | ||||||
|   | |||||||
| @@ -13,7 +13,6 @@ EXCLUDED = [ | |||||||
|     'cas_server.serviceticket', |     'cas_server.serviceticket', | ||||||
|     'cas_server.user', |     'cas_server.user', | ||||||
|     'cas_server.userattributes', |     'cas_server.userattributes', | ||||||
|     'constance.constance', |  | ||||||
|     'contenttypes.contenttype', |     'contenttypes.contenttype', | ||||||
|     'logs.changelog', |     'logs.changelog', | ||||||
|     'migrations.migration', |     'migrations.migration', | ||||||
|   | |||||||
| @@ -164,24 +164,14 @@ class ScopesView(LoginRequiredMixin, TemplateView): | |||||||
|         from oauth2_provider.models import Application |         from oauth2_provider.models import Application | ||||||
|         from .scopes import PermissionScopes |         from .scopes import PermissionScopes | ||||||
|  |  | ||||||
|         oidc = False |         scopes = PermissionScopes() | ||||||
|         context["scopes"] = {} |         context["scopes"] = {} | ||||||
|  |         all_scopes = scopes.get_all_scopes() | ||||||
|         for app in Application.objects.filter(user=self.request.user).all(): |         for app in Application.objects.filter(user=self.request.user).all(): | ||||||
|             available_scopes = PermissionScopes().get_available_scopes(app) |             available_scopes = scopes.get_available_scopes(app) | ||||||
|             context["scopes"][app] = OrderedDict() |             context["scopes"][app] = OrderedDict() | ||||||
|             all_scopes = PermissionScopes().get_all_scopes(scopes=available_scopes) |             items = [(k, v) for (k, v) in all_scopes.items() if k in available_scopes] | ||||||
|             scopes = {} |             # items.sort(key=lambda x: (int(x[0].split("_")[1]), int(x[0].split("_")[0]))) | ||||||
|             for scope in available_scopes: |  | ||||||
|                 scopes[scope] = all_scopes[scope] |  | ||||||
|             # remove OIDC scope for sort |  | ||||||
|             if 'openid' in scopes: |  | ||||||
|                 del scopes['openid'] |  | ||||||
|                 oidc = True |  | ||||||
|             items = [(k, v) for (k, v) in scopes.items()] |  | ||||||
|             items.sort(key=lambda x: (int(x[0].split("_")[1]), int(x[0].split("_")[0]))) |  | ||||||
|             # add oidc if necessary |  | ||||||
|             if oidc: |  | ||||||
|                 items.append(('openid', PermissionScopes().get_all_scopes(scopes=['openid'])['openid'])) |  | ||||||
|             for k, v in items: |             for k, v in items: | ||||||
|                 context["scopes"][app][k] = v |                 context["scopes"][app][k] = v | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ from bootstrap_datepicker_plus.widgets import DatePickerInput | |||||||
| from django import forms | from django import forms | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.db.models import Q | from django.db.models import Q | ||||||
| from django.forms import CheckboxSelectMultiple, RadioSelect | from django.forms import CheckboxSelectMultiple | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from note.models import NoteSpecial, NoteUser | from note.models import NoteSpecial, NoteUser | ||||||
| from note_kfet.inputs import AmountInput, Autocomplete, ColorWidget | from note_kfet.inputs import AmountInput, Autocomplete, ColorWidget | ||||||
| @@ -140,19 +140,6 @@ class WEIMembershipForm(forms.ModelForm): | |||||||
|         required=False, |         required=False, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def __init__(self, *args, wei=None, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         if 'bus' in self.fields: |  | ||||||
|             if wei is not None: |  | ||||||
|                 self.fields['bus'].queryset = Bus.objects.filter(wei=wei) |  | ||||||
|             else: |  | ||||||
|                 self.fields['bus'].queryset = Bus.objects.none() |  | ||||||
|         if 'team' in self.fields: |  | ||||||
|             if wei is not None: |  | ||||||
|                 self.fields['team'].queryset = BusTeam.objects.filter(bus__wei=wei) |  | ||||||
|             else: |  | ||||||
|                 self.fields['team'].queryset = BusTeam.objects.none() |  | ||||||
|  |  | ||||||
|     def clean(self): |     def clean(self): | ||||||
|         cleaned_data = super().clean() |         cleaned_data = super().clean() | ||||||
|         if 'team' in cleaned_data and cleaned_data["team"] is not None \ |         if 'team' in cleaned_data and cleaned_data["team"] is not None \ | ||||||
| @@ -164,8 +151,21 @@ class WEIMembershipForm(forms.ModelForm): | |||||||
|         model = WEIMembership |         model = WEIMembership | ||||||
|         fields = ('roles', 'bus', 'team',) |         fields = ('roles', 'bus', 'team',) | ||||||
|         widgets = { |         widgets = { | ||||||
|             "bus": RadioSelect(), |             "bus": Autocomplete( | ||||||
|             "team": RadioSelect(), |                 Bus, | ||||||
|  |                 attrs={ | ||||||
|  |                     'api_url': '/api/wei/bus/', | ||||||
|  |                     'placeholder': 'Bus ...', | ||||||
|  |                 } | ||||||
|  |             ), | ||||||
|  |             "team": Autocomplete( | ||||||
|  |                 BusTeam, | ||||||
|  |                 attrs={ | ||||||
|  |                     'api_url': '/api/wei/team/', | ||||||
|  |                     'placeholder': 'Équipe ...', | ||||||
|  |                 }, | ||||||
|  |                 resetable=True, | ||||||
|  |             ), | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -213,3 +213,4 @@ class BusTeamForm(forms.ModelForm): | |||||||
|             ), |             ), | ||||||
|             "color": ColorWidget(), |             "color": ColorWidget(), | ||||||
|         } |         } | ||||||
|  |         # "color": ColorWidget(), | ||||||
|   | |||||||
| @@ -2,11 +2,11 @@ | |||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm | from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm | ||||||
| from .wei2025 import WEISurvey2025 | from .wei2024 import WEISurvey2024 | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = [ | __all__ = [ | ||||||
|     'WEISurvey', 'WEISurveyInformation', 'WEISurveyAlgorithm', 'CurrentSurvey', |     'WEISurvey', 'WEISurveyInformation', 'WEISurveyAlgorithm', 'CurrentSurvey', | ||||||
| ] | ] | ||||||
|  |  | ||||||
| CurrentSurvey = WEISurvey2025 | CurrentSurvey = WEISurvey2024 | ||||||
|   | |||||||
| @@ -121,13 +121,6 @@ class WEISurveyAlgorithm: | |||||||
|         """ |         """ | ||||||
|         raise NotImplementedError |         raise NotImplementedError | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_bus_information_form(cls): |  | ||||||
|         """ |  | ||||||
|         The class of the form to update the bus information. |  | ||||||
|         """ |  | ||||||
|         raise NotImplementedError |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurvey: | class WEISurvey: | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -1,347 +0,0 @@ | |||||||
| # Copyright (C) 2018-2025 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| import time |  | ||||||
| import json |  | ||||||
| from functools import lru_cache |  | ||||||
| from random import Random |  | ||||||
|  |  | ||||||
| from django import forms |  | ||||||
| from django.db import transaction |  | ||||||
| from django.db.models import Q |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
|  |  | ||||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm, WEIBusInformation |  | ||||||
| from ...models import WEIMembership, Bus |  | ||||||
|  |  | ||||||
| WORDS = [ |  | ||||||
|     '13 organisé', '3ième mi temps', 'Années 2000', 'Apéro', 'BBQ', 'BP', 'Beauf', 'Binge drinking', 'Bon enfant', |  | ||||||
|     'Cartouche', 'Catacombes', 'Chansons paillardes', 'Chansons populaires', 'Chanteur', 'Chartreuse', 'Chill', |  | ||||||
|     'Core', 'DJ', 'Dancefloor', 'Danse', 'David Guetta', 'Disco', 'Eau de vie', 'Électro', 'Escalade', 'Familial', |  | ||||||
|     'Fanfare', 'Fracassage', 'Féria', 'Hard rock', 'Hoeggarden', 'House', 'Huit-six', 'IPA', 'Inclusif', 'Inferno', |  | ||||||
|     'Introverti', 'Jager bomb', 'Jazz', 'Jeux d\'alcool', 'Jeux de rôles', 'Jeux vidéo', 'Jul', 'Jus de fruit', |  | ||||||
|     'Karaoké', 'LGBTQI+', 'Lady Gaga', 'Loup garou', 'Morning beer', 'Métal', 'Nuit blanche', 'Ovalie', 'Psychedelic', |  | ||||||
|     'Pétanque', 'Rave', 'Reggae', 'Rhum', 'Ricard', 'Rock', 'Rosé', 'Rétro', 'Séducteur', 'Techno', 'Thérapie taxi', |  | ||||||
|     'Théâtre', 'Trap', 'Turn up', 'Underground', 'Volley', 'Wati B', 'Zinédine Zidane', |  | ||||||
| ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyForm2025(forms.Form): |  | ||||||
|     """ |  | ||||||
|     Survey form for the year 2025. |  | ||||||
|     Members choose 20 words, from which we calculate the best associated bus. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     word = forms.ChoiceField( |  | ||||||
|         label=_("Choose a word:"), |  | ||||||
|         widget=forms.RadioSelect(), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def set_registration(self, registration): |  | ||||||
|         """ |  | ||||||
|         Filter the bus selector with the buses of the current WEI. |  | ||||||
|         """ |  | ||||||
|         information = WEISurveyInformation2025(registration) |  | ||||||
|         if not information.seed: |  | ||||||
|             information.seed = int(1000 * time.time()) |  | ||||||
|             information.save(registration) |  | ||||||
|             registration._force_save = True |  | ||||||
|             registration.save() |  | ||||||
|  |  | ||||||
|         if self.data: |  | ||||||
|             self.fields["word"].choices = [(w, w) for w in WORDS] |  | ||||||
|             if self.is_valid(): |  | ||||||
|                 return |  | ||||||
|  |  | ||||||
|         rng = Random((information.step + 1) * information.seed) |  | ||||||
|  |  | ||||||
|         buses = WEISurveyAlgorithm2025.get_buses() |  | ||||||
|         informations = {bus: WEIBusInformation2025(bus) for bus in buses} |  | ||||||
|         scores = sum((list(informations[bus].scores.values()) for bus in buses), []) |  | ||||||
|         if scores: |  | ||||||
|             average_score = sum(scores) / len(scores) |  | ||||||
|         else: |  | ||||||
|             average_score = 0 |  | ||||||
|  |  | ||||||
|         preferred_words = {bus: [word for word in WORDS |  | ||||||
|                                  if informations[bus].scores[word] >= average_score] |  | ||||||
|                            for bus in buses} |  | ||||||
|  |  | ||||||
|         # Correction : proposer plusieurs mots différents à chaque étape |  | ||||||
|         n_choices = 4  # Nombre de mots à proposer à chaque étape |  | ||||||
|         all_preferred_words = set() |  | ||||||
|         for bus_words in preferred_words.values(): |  | ||||||
|             all_preferred_words.update(bus_words) |  | ||||||
|         all_preferred_words = list(all_preferred_words) |  | ||||||
|         rng.shuffle(all_preferred_words) |  | ||||||
|         words = all_preferred_words[:n_choices] |  | ||||||
|         self.fields["word"].choices = [(w, w) for w in words] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEIBusInformation2025(WEIBusInformation): |  | ||||||
|     """ |  | ||||||
|     For each word, the bus has a score |  | ||||||
|     """ |  | ||||||
|     scores: dict |  | ||||||
|  |  | ||||||
|     def __init__(self, bus): |  | ||||||
|         self.scores = {} |  | ||||||
|         for word in WORDS: |  | ||||||
|             self.scores[word] = 0 |  | ||||||
|         super().__init__(bus) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BusInformationForm2025(forms.ModelForm): |  | ||||||
|     class Meta: |  | ||||||
|         model = Bus |  | ||||||
|         fields = ['information_json'] |  | ||||||
|         widgets = {} |  | ||||||
|  |  | ||||||
|     def __init__(self, *args, words=None, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|  |  | ||||||
|         initial_scores = {} |  | ||||||
|         if self.instance and self.instance.information_json: |  | ||||||
|             try: |  | ||||||
|                 info = json.loads(self.instance.information_json) |  | ||||||
|                 initial_scores = info.get("scores", {}) |  | ||||||
|             except (json.JSONDecodeError, TypeError, AttributeError): |  | ||||||
|                 initial_scores = {} |  | ||||||
|         if words is None: |  | ||||||
|             words = WORDS |  | ||||||
|         self.words = words |  | ||||||
|  |  | ||||||
|         choices = [(i, str(i)) for i in range(6)]  # [(0, '0'), (1, '1'), ..., (5, '5')] |  | ||||||
|         for word in words: |  | ||||||
|             self.fields[word] = forms.TypedChoiceField( |  | ||||||
|                 label=word, |  | ||||||
|                 choices=choices, |  | ||||||
|                 coerce=int, |  | ||||||
|                 initial=initial_scores.get(word, 0), |  | ||||||
|                 required=True, |  | ||||||
|                 widget=forms.RadioSelect, |  | ||||||
|                 help_text=_("Rate between 0 and 5."), |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|     def clean(self): |  | ||||||
|         cleaned_data = super().clean() |  | ||||||
|         scores = {} |  | ||||||
|         for word in self.words: |  | ||||||
|             value = cleaned_data.get(word) |  | ||||||
|             if value is not None: |  | ||||||
|                 scores[word] = value |  | ||||||
|         # On encode en JSON |  | ||||||
|         cleaned_data['information_json'] = json.dumps({"scores": scores}) |  | ||||||
|         return cleaned_data |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyInformation2025(WEISurveyInformation): |  | ||||||
|     """ |  | ||||||
|     We store the id of the selected bus. We store only the name, but is not used in the selection: |  | ||||||
|     that's only for humans that try to read data. |  | ||||||
|     """ |  | ||||||
|     # Random seed that is stored at the first time to ensure that words are generated only once |  | ||||||
|     seed = 0 |  | ||||||
|     step = 0 |  | ||||||
|  |  | ||||||
|     def __init__(self, registration): |  | ||||||
|         for i in range(1, 21): |  | ||||||
|             setattr(self, "word" + str(i), None) |  | ||||||
|         super().__init__(registration) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurvey2025(WEISurvey): |  | ||||||
|     """ |  | ||||||
|     Survey for the year 2025. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_year(cls): |  | ||||||
|         return 2025 |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_survey_information_class(cls): |  | ||||||
|         return WEISurveyInformation2025 |  | ||||||
|  |  | ||||||
|     def get_form_class(self): |  | ||||||
|         return WEISurveyForm2025 |  | ||||||
|  |  | ||||||
|     def update_form(self, form): |  | ||||||
|         """ |  | ||||||
|         Filter the bus selector with the buses of the WEI. |  | ||||||
|         """ |  | ||||||
|         form.set_registration(self.registration) |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         word = form.cleaned_data["word"] |  | ||||||
|         self.information.step += 1 |  | ||||||
|         setattr(self.information, "word" + str(self.information.step), word) |  | ||||||
|         self.save() |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_algorithm_class(cls): |  | ||||||
|         return WEISurveyAlgorithm2025 |  | ||||||
|  |  | ||||||
|     def is_complete(self) -> bool: |  | ||||||
|         """ |  | ||||||
|         The survey is complete once the bus is chosen. |  | ||||||
|         """ |  | ||||||
|         return self.information.step == 20 |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     @lru_cache() |  | ||||||
|     def word_mean(cls, word): |  | ||||||
|         """ |  | ||||||
|         Calculate the mid-score given by all buses. |  | ||||||
|         """ |  | ||||||
|         buses = cls.get_algorithm_class().get_buses() |  | ||||||
|         return sum([cls.get_algorithm_class().get_bus_information(bus).scores[word] for bus in buses]) / buses.count() |  | ||||||
|  |  | ||||||
|     @lru_cache() |  | ||||||
|     def score(self, bus): |  | ||||||
|         if not self.is_complete(): |  | ||||||
|             raise ValueError("Survey is not ended, can't calculate score") |  | ||||||
|  |  | ||||||
|         bus_info = self.get_algorithm_class().get_bus_information(bus) |  | ||||||
|         # Score is the given score by the bus subtracted to the mid-score of the buses. |  | ||||||
|         s = sum(bus_info.scores[getattr(self.information, 'word' + str(i))] |  | ||||||
|                 - self.word_mean(getattr(self.information, 'word' + str(i))) for i in range(1, 21)) / 20 |  | ||||||
|         return s |  | ||||||
|  |  | ||||||
|     @lru_cache() |  | ||||||
|     def scores_per_bus(self): |  | ||||||
|         return {bus: self.score(bus) for bus in self.get_algorithm_class().get_buses()} |  | ||||||
|  |  | ||||||
|     @lru_cache() |  | ||||||
|     def ordered_buses(self): |  | ||||||
|         values = list(self.scores_per_bus().items()) |  | ||||||
|         values.sort(key=lambda item: -item[1]) |  | ||||||
|         return values |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def clear_cache(cls): |  | ||||||
|         cls.word_mean.cache_clear() |  | ||||||
|         return super().clear_cache() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyAlgorithm2025(WEISurveyAlgorithm): |  | ||||||
|     """ |  | ||||||
|     The algorithm class for the year 2025. |  | ||||||
|     We use Gale-Shapley algorithm to attribute 1y students into buses. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_survey_class(cls): |  | ||||||
|         return WEISurvey2025 |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_bus_information_class(cls): |  | ||||||
|         return WEIBusInformation2025 |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_bus_information_form(cls): |  | ||||||
|         return BusInformationForm2025 |  | ||||||
|  |  | ||||||
|     def run_algorithm(self, display_tqdm=False): |  | ||||||
|         """ |  | ||||||
|         Gale-Shapley algorithm implementation. |  | ||||||
|         We modify it to allow buses to have multiple "weddings". |  | ||||||
|         """ |  | ||||||
|         surveys = list(self.get_survey_class()(r) for r in self.get_registrations())  # All surveys |  | ||||||
|         surveys = [s for s in surveys if s.is_complete()]  # Don't consider invalid surveys |  | ||||||
|         # Don't manage hardcoded people |  | ||||||
|         surveys = [s for s in surveys if not hasattr(s.information, 'hardcoded') or not s.information.hardcoded] |  | ||||||
|  |  | ||||||
|         # Reset previous algorithm run |  | ||||||
|         for survey in surveys: |  | ||||||
|             survey.free() |  | ||||||
|             survey.save() |  | ||||||
|  |  | ||||||
|         non_men = [s for s in surveys if s.registration.gender != 'male'] |  | ||||||
|         men = [s for s in surveys if s.registration.gender == 'male'] |  | ||||||
|  |  | ||||||
|         quotas = {} |  | ||||||
|         registrations = self.get_registrations() |  | ||||||
|         non_men_total = registrations.filter(~Q(gender='male')).count() |  | ||||||
|         for bus in self.get_buses(): |  | ||||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() |  | ||||||
|             # Remove hardcoded people |  | ||||||
|             free_seats -= WEIMembership.objects.filter(bus=bus, registration__first_year=True, |  | ||||||
|                                                        registration__information_json__icontains="hardcoded").count() |  | ||||||
|             quotas[bus] = 4 + int(non_men_total / registrations.count() * free_seats) |  | ||||||
|  |  | ||||||
|         tqdm_obj = None |  | ||||||
|         if display_tqdm: |  | ||||||
|             from tqdm import tqdm |  | ||||||
|             tqdm_obj = tqdm(total=len(non_men), desc="Non-hommes") |  | ||||||
|  |  | ||||||
|         # Repartition for non men people first |  | ||||||
|         self.make_repartition(non_men, quotas, tqdm_obj=tqdm_obj) |  | ||||||
|  |  | ||||||
|         quotas = {} |  | ||||||
|         for bus in self.get_buses(): |  | ||||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() |  | ||||||
|             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) |  | ||||||
|             # Remove hardcoded people |  | ||||||
|             free_seats -= WEIMembership.objects.filter(bus=bus, registration__first_year=True, |  | ||||||
|                                                        registration__information_json__icontains="hardcoded").count() |  | ||||||
|             quotas[bus] = free_seats |  | ||||||
|  |  | ||||||
|         if display_tqdm: |  | ||||||
|             tqdm_obj.close() |  | ||||||
|  |  | ||||||
|             from tqdm import tqdm |  | ||||||
|             tqdm_obj = tqdm(total=len(men), desc="Hommes") |  | ||||||
|  |  | ||||||
|         self.make_repartition(men, quotas, tqdm_obj=tqdm_obj) |  | ||||||
|  |  | ||||||
|         if display_tqdm: |  | ||||||
|             tqdm_obj.close() |  | ||||||
|  |  | ||||||
|         # Clear cache information after running algorithm |  | ||||||
|         WEISurvey2025.clear_cache() |  | ||||||
|  |  | ||||||
|     def make_repartition(self, surveys, quotas=None, tqdm_obj=None): |  | ||||||
|         free_surveys = surveys.copy()  # Remaining surveys |  | ||||||
|         while free_surveys:  # Some students are not affected |  | ||||||
|             survey = free_surveys[0] |  | ||||||
|             buses = survey.ordered_buses()  # Preferences of the student |  | ||||||
|             for bus, current_score in buses: |  | ||||||
|                 if self.get_bus_information(bus).has_free_seats(surveys, quotas): |  | ||||||
|                     # Selected bus has free places. Put student in the bus |  | ||||||
|                     survey.select_bus(bus) |  | ||||||
|                     survey.save() |  | ||||||
|                     free_surveys.remove(survey) |  | ||||||
|                     break |  | ||||||
|                 else: |  | ||||||
|                     # Current bus has not enough places. Remove the least preferred student from the bus if existing |  | ||||||
|                     least_preferred_survey = None |  | ||||||
|                     least_score = -1 |  | ||||||
|                     # Find the least student in the bus that has a lower score than the current student |  | ||||||
|                     for survey2 in surveys: |  | ||||||
|                         if not survey2.information.valid or survey2.information.get_selected_bus() != bus: |  | ||||||
|                             continue |  | ||||||
|                         score2 = survey2.score(bus) |  | ||||||
|                         if current_score <= score2:  # Ignore better students |  | ||||||
|                             continue |  | ||||||
|                         if least_preferred_survey is None or score2 < least_score: |  | ||||||
|                             least_preferred_survey = survey2 |  | ||||||
|                             least_score = score2 |  | ||||||
|  |  | ||||||
|                     if least_preferred_survey is not None: |  | ||||||
|                         # Remove the least student from the bus and put the current student in. |  | ||||||
|                         # If it does not exist, choose the next bus. |  | ||||||
|                         least_preferred_survey.free() |  | ||||||
|                         least_preferred_survey.save() |  | ||||||
|                         free_surveys.append(least_preferred_survey) |  | ||||||
|                         survey.select_bus(bus) |  | ||||||
|                         survey.save() |  | ||||||
|                         free_surveys.remove(survey) |  | ||||||
|                         break |  | ||||||
|             else: |  | ||||||
|                 raise ValueError(f"User {survey.registration.user} has no free seat") |  | ||||||
|  |  | ||||||
|             if tqdm_obj is not None: |  | ||||||
|                 tqdm_obj.n = len(surveys) - len(free_surveys) |  | ||||||
|                 tqdm_obj.refresh() |  | ||||||
| @@ -22,8 +22,6 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|         {% endif %} |         {% endif %} | ||||||
|         <a class="btn btn-primary btn-sm my-1" href="{% url 'wei:update_bus' pk=object.pk %}" |         <a class="btn btn-primary btn-sm my-1" href="{% url 'wei:update_bus' pk=object.pk %}" | ||||||
|             data-turbolinks="false">{% trans "Edit" %}</a> |             data-turbolinks="false">{% trans "Edit" %}</a> | ||||||
|             <a class="btn btn-primary btn-sm my-1" href="{% url 'wei:update_bus_info' pk=object.pk %}" |  | ||||||
|             data-turbolinks="false">{% trans "Edit information" %}</a> |  | ||||||
|         <a class="btn btn-primary btn-sm my-1" href="{% url 'wei:add_team' pk=object.pk %}" |         <a class="btn btn-primary btn-sm my-1" href="{% url 'wei:add_team' pk=object.pk %}" | ||||||
|             data-turbolinks="false">{% trans "Add team" %}</a> |             data-turbolinks="false">{% trans "Add team" %}</a> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -210,27 +210,4 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     </script> |     </script> | ||||||
|     <script> |  | ||||||
|         $(document).ready(function () { |  | ||||||
|             function refreshTeams() { |  | ||||||
|                 let buses = []; |  | ||||||
|                 $("input[name='bus']:checked").each(function (ignored) { |  | ||||||
|                     buses.push($(this).parent().text().trim()); |  | ||||||
|                 }); |  | ||||||
|                 console.log(buses); |  | ||||||
|                 $("input[name='team']").each(function () { |  | ||||||
|                     let label = $(this).parent(); |  | ||||||
|                     $(this).parent().addClass('d-none'); |  | ||||||
|                     buses.forEach(function (bus) { |  | ||||||
|                         if (label.text().includes(bus)) |  | ||||||
|                             label.removeClass('d-none'); |  | ||||||
|                     }); |  | ||||||
|                 }); |  | ||||||
|             } |  | ||||||
|      |  | ||||||
|             $("input[name='bus']").change(refreshTeams); |  | ||||||
|      |  | ||||||
|             refreshTeams(); |  | ||||||
|         }); |  | ||||||
|     </script> |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
|   | |||||||
| @@ -6,6 +6,8 @@ from datetime import date, timedelta | |||||||
|  |  | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
|  | from django.urls import reverse | ||||||
|  | from note.models import NoteUser | ||||||
|  |  | ||||||
| from ..forms.surveys.wei2024 import WEIBusInformation2024, WEISurvey2024, WORDS, WEISurveyInformation2024 | from ..forms.surveys.wei2024 import WEIBusInformation2024, WEISurvey2024, WORDS, WEISurveyInformation2024 | ||||||
| from ..models import Bus, WEIClub, WEIRegistration | from ..models import Bus, WEIClub, WEIRegistration | ||||||
| @@ -127,3 +129,44 @@ class TestWEIAlgorithm(TestCase): | |||||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance |             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance | ||||||
|  |  | ||||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % |         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % | ||||||
|  |  | ||||||
|  |     def test_register_1a(self): | ||||||
|  |         """ | ||||||
|  |         Test register a first year member to the WEI and complete the survey | ||||||
|  |         """ | ||||||
|  |         response = self.client.get(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk))) | ||||||
|  |         self.assertEqual(response.status_code, 200) | ||||||
|  |  | ||||||
|  |         user = User.objects.create(username="toto", email="toto@example.com") | ||||||
|  |         NoteUser.objects.create(user=user) | ||||||
|  |         response = self.client.post(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk)), dict( | ||||||
|  |             user=user.id, | ||||||
|  |             soge_credit=True, | ||||||
|  |             birth_date=date(2000, 1, 1), | ||||||
|  |             gender='nonbinary', | ||||||
|  |             clothing_cut='female', | ||||||
|  |             clothing_size='XS', | ||||||
|  |             health_issues='I am a bot', | ||||||
|  |             emergency_contact_name='NoteKfet2020', | ||||||
|  |             emergency_contact_phone='+33123456789', | ||||||
|  |         )) | ||||||
|  |         qs = WEIRegistration.objects.filter(user_id=user.id) | ||||||
|  |         self.assertTrue(qs.exists()) | ||||||
|  |         registration = qs.get() | ||||||
|  |         self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, 200) | ||||||
|  |         for question in WORDS: | ||||||
|  |             # Fill 1A Survey, 10 pages | ||||||
|  |             # be careful if questionnary form change (number of page, type of answer...) | ||||||
|  |             response = self.client.post(reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), { | ||||||
|  |                 question: "1" | ||||||
|  |             }) | ||||||
|  |             registration.refresh_from_db() | ||||||
|  |             survey = WEISurvey2024(registration) | ||||||
|  |             self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, | ||||||
|  |                                  302 if survey.is_complete() else 200) | ||||||
|  |             self.assertIsNotNone(getattr(survey.information, question), "Survey page " + question + " failed") | ||||||
|  |         survey = WEISurvey2024(registration) | ||||||
|  |         self.assertTrue(survey.is_complete()) | ||||||
|  |         survey.select_bus(self.buses[0]) | ||||||
|  |         survey.save() | ||||||
|  |         self.assertIsNotNone(survey.information.get_selected_bus()) | ||||||
|   | |||||||
| @@ -1,111 +0,0 @@ | |||||||
| # Copyright (C) 2018-2025 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| import random |  | ||||||
|  |  | ||||||
| from django.contrib.auth.models import User |  | ||||||
| from django.test import TestCase |  | ||||||
|  |  | ||||||
| from ..forms.surveys.wei2025 import WEIBusInformation2025, WEISurvey2025, WORDS, WEISurveyInformation2025 |  | ||||||
| from ..models import Bus, WEIClub, WEIRegistration |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestWEIAlgorithm(TestCase): |  | ||||||
|     """ |  | ||||||
|     Run some tests to ensure that the WEI algorithm is working well. |  | ||||||
|     """ |  | ||||||
|     fixtures = ('initial',) |  | ||||||
|  |  | ||||||
|     def setUp(self): |  | ||||||
|         """ |  | ||||||
|         Create some test data, with one WEI and 10 buses with random score attributions. |  | ||||||
|         """ |  | ||||||
|         self.wei = WEIClub.objects.create( |  | ||||||
|             name="WEI 2025", |  | ||||||
|             email="wei2025@example.com", |  | ||||||
|             date_start='2025-09-12', |  | ||||||
|             date_end='2025-09-14', |  | ||||||
|             year=2025, |  | ||||||
|             membership_start='2025-06-01' |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         self.buses = [] |  | ||||||
|         for i in range(10): |  | ||||||
|             bus = Bus.objects.create(wei=self.wei, name=f"Bus {i}", size=10) |  | ||||||
|             self.buses.append(bus) |  | ||||||
|             information = WEIBusInformation2025(bus) |  | ||||||
|             for word in WORDS: |  | ||||||
|                 information.scores[word] = random.randint(0, 101) |  | ||||||
|             information.save() |  | ||||||
|             bus.save() |  | ||||||
|  |  | ||||||
|     def test_survey_algorithm_small(self): |  | ||||||
|         """ |  | ||||||
|         There are only a few people in each bus, ensure that each person has its best bus |  | ||||||
|         """ |  | ||||||
|         # Add a few users |  | ||||||
|         for i in range(10): |  | ||||||
|             user = User.objects.create(username=f"user{i}") |  | ||||||
|             registration = WEIRegistration.objects.create( |  | ||||||
|                 user=user, |  | ||||||
|                 wei=self.wei, |  | ||||||
|                 first_year=True, |  | ||||||
|                 birth_date='2000-01-01', |  | ||||||
|             ) |  | ||||||
|             information = WEISurveyInformation2025(registration) |  | ||||||
|             for j in range(1, 21): |  | ||||||
|                 setattr(information, f'word{j}', random.choice(WORDS)) |  | ||||||
|             information.step = 20 |  | ||||||
|             information.save(registration) |  | ||||||
|             registration.save() |  | ||||||
|  |  | ||||||
|         # Run algorithm |  | ||||||
|         WEISurvey2025.get_algorithm_class()().run_algorithm() |  | ||||||
|  |  | ||||||
|         # Ensure that everyone has its first choice |  | ||||||
|         for r in WEIRegistration.objects.filter(wei=self.wei).all(): |  | ||||||
|             survey = WEISurvey2025(r) |  | ||||||
|             preferred_bus = survey.ordered_buses()[0][0] |  | ||||||
|             chosen_bus = survey.information.get_selected_bus() |  | ||||||
|             self.assertEqual(preferred_bus, chosen_bus) |  | ||||||
|  |  | ||||||
|     def test_survey_algorithm_full(self): |  | ||||||
|         """ |  | ||||||
|         Buses are full of first year people, ensure that they are happy |  | ||||||
|         """ |  | ||||||
|         # Add a lot of users |  | ||||||
|         for i in range(95): |  | ||||||
|             user = User.objects.create(username=f"user{i}") |  | ||||||
|             registration = WEIRegistration.objects.create( |  | ||||||
|                 user=user, |  | ||||||
|                 wei=self.wei, |  | ||||||
|                 first_year=True, |  | ||||||
|                 birth_date='2000-01-01', |  | ||||||
|             ) |  | ||||||
|             information = WEISurveyInformation2025(registration) |  | ||||||
|             for j in range(1, 21): |  | ||||||
|                 setattr(information, f'word{j}', random.choice(WORDS)) |  | ||||||
|             information.step = 20 |  | ||||||
|             information.save(registration) |  | ||||||
|             registration.save() |  | ||||||
|  |  | ||||||
|         # Run algorithm |  | ||||||
|         WEISurvey2025.get_algorithm_class()().run_algorithm() |  | ||||||
|  |  | ||||||
|         penalty = 0 |  | ||||||
|         # Ensure that everyone seems to be happy |  | ||||||
|         # We attribute a penalty for each user that didn't have its first choice |  | ||||||
|         # The penalty is the square of the distance between the score of the preferred bus |  | ||||||
|         # and the score of the attributed bus |  | ||||||
|         # We consider it acceptable if the mean of this distance is lower than 5 % |  | ||||||
|         for r in WEIRegistration.objects.filter(wei=self.wei).all(): |  | ||||||
|             survey = WEISurvey2025(r) |  | ||||||
|             chosen_bus = survey.information.get_selected_bus() |  | ||||||
|             buses = survey.ordered_buses() |  | ||||||
|             score = min(v for bus, v in buses if bus == chosen_bus) |  | ||||||
|             max_score = buses[0][1] |  | ||||||
|             penalty += (max_score - score) ** 2 |  | ||||||
|  |  | ||||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance |  | ||||||
|  |  | ||||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % |  | ||||||
| @@ -778,7 +778,7 @@ class TestDefaultWEISurvey(TestCase): | |||||||
|         WEISurvey.update_form(None, None) |         WEISurvey.update_form(None, None) | ||||||
|  |  | ||||||
|         self.assertEqual(CurrentSurvey.get_algorithm_class().get_survey_class(), CurrentSurvey) |         self.assertEqual(CurrentSurvey.get_algorithm_class().get_survey_class(), CurrentSurvey) | ||||||
|         self.assertEqual(CurrentSurvey.get_year(), 2025) |         self.assertEqual(CurrentSurvey.get_year(), 2024) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestWeiAPI(TestAPI): | class TestWeiAPI(TestAPI): | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ | |||||||
| from django.urls import path | from django.urls import path | ||||||
|  |  | ||||||
| from .views import CurrentWEIDetailView, WEI1AListView, WEIListView, WEICreateView, WEIDetailView, WEIUpdateView, \ | from .views import CurrentWEIDetailView, WEI1AListView, WEIListView, WEICreateView, WEIDetailView, WEIUpdateView, \ | ||||||
|     WEIRegistrationsView, WEIMembershipsView, MemberListRenderView, BusInformationUpdateView, \ |     WEIRegistrationsView, WEIMembershipsView, MemberListRenderView, \ | ||||||
|     BusCreateView, BusManageView, BusUpdateView, BusTeamCreateView, BusTeamManageView, BusTeamUpdateView, \ |     BusCreateView, BusManageView, BusUpdateView, BusTeamCreateView, BusTeamManageView, BusTeamUpdateView, \ | ||||||
|     WEIAttributeBus1AView, WEIAttributeBus1ANextView, WEIRegister1AView, WEIRegister2AView, WEIUpdateRegistrationView, \ |     WEIAttributeBus1AView, WEIAttributeBus1ANextView, WEIRegister1AView, WEIRegister2AView, WEIUpdateRegistrationView, \ | ||||||
|     WEIDeleteRegistrationView, WEIValidateRegistrationView, WEISurveyView, WEISurveyEndView, WEIClosedView |     WEIDeleteRegistrationView, WEIValidateRegistrationView, WEISurveyView, WEISurveyEndView, WEIClosedView | ||||||
| @@ -42,5 +42,4 @@ urlpatterns = [ | |||||||
|     path('detail/<int:pk>/closed/', WEIClosedView.as_view(), name="wei_closed"), |     path('detail/<int:pk>/closed/', WEIClosedView.as_view(), name="wei_closed"), | ||||||
|     path('bus-1A/<int:pk>/', WEIAttributeBus1AView.as_view(), name="wei_bus_1A"), |     path('bus-1A/<int:pk>/', WEIAttributeBus1AView.as_view(), name="wei_bus_1A"), | ||||||
|     path('bus-1A/next/<int:pk>/', WEIAttributeBus1ANextView.as_view(), name="wei_bus_1A_next"), |     path('bus-1A/next/<int:pk>/', WEIAttributeBus1ANextView.as_view(), name="wei_bus_1A_next"), | ||||||
|     path('update-bus-info/<int:pk>/', BusInformationUpdateView.as_view(), name="update_bus_info"), |  | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -788,8 +788,7 @@ class WEIUpdateRegistrationView(ProtectQuerysetMixin, LoginRequiredMixin, Update | |||||||
|         return form |         return form | ||||||
|  |  | ||||||
|     def get_membership_form(self, data=None, instance=None): |     def get_membership_form(self, data=None, instance=None): | ||||||
|         registration = self.get_object() |         membership_form = WEIMembershipForm(data if data else None, instance=instance) | ||||||
|         membership_form = WEIMembershipForm(data if data else None, instance=instance, wei=registration.wei) |  | ||||||
|         del membership_form.fields["credit_type"] |         del membership_form.fields["credit_type"] | ||||||
|         del membership_form.fields["credit_amount"] |         del membership_form.fields["credit_amount"] | ||||||
|         del membership_form.fields["first_name"] |         del membership_form.fields["first_name"] | ||||||
| @@ -970,13 +969,6 @@ class WEIValidateRegistrationView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|             return WEIMembership1AForm |             return WEIMembership1AForm | ||||||
|         return WEIMembershipForm |         return WEIMembershipForm | ||||||
|  |  | ||||||
|     def get_form_kwargs(self): |  | ||||||
|         kwargs = super().get_form_kwargs() |  | ||||||
|         registration = WEIRegistration.objects.get(pk=self.kwargs["pk"]) |  | ||||||
|         wei = registration.wei |  | ||||||
|         kwargs['wei'] = wei |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|     def get_form(self, form_class=None): |     def get_form(self, form_class=None): | ||||||
|         form = super().get_form(form_class) |         form = super().get_form(form_class) | ||||||
|         registration = WEIRegistration.objects.get(pk=self.kwargs["pk"]) |         registration = WEIRegistration.objects.get(pk=self.kwargs["pk"]) | ||||||
| @@ -1430,29 +1422,3 @@ class WEIAttributeBus1ANextView(LoginRequiredMixin, RedirectView): | |||||||
|  |  | ||||||
|         # On redirige vers la page d'attribution pour le premier étudiant trouvé |         # On redirige vers la page d'attribution pour le premier étudiant trouvé | ||||||
|         return reverse_lazy('wei:wei_bus_1A', args=(qs.first().pk,)) |         return reverse_lazy('wei:wei_bus_1A', args=(qs.first().pk,)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class BusInformationUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView): |  | ||||||
|     model = Bus |  | ||||||
|  |  | ||||||
|     def get_form_class(self): |  | ||||||
|         return CurrentSurvey.get_algorithm_class().get_bus_information_form() |  | ||||||
|  |  | ||||||
|     def dispatch(self, request, *args, **kwargs): |  | ||||||
|         wei = self.get_object().wei |  | ||||||
|         today = date.today() |  | ||||||
|         # We can't update a bus once the WEI is started |  | ||||||
|         if today >= wei.date_start: |  | ||||||
|             return redirect(reverse_lazy('wei:wei_closed', args=(wei.pk,))) |  | ||||||
|         return super().dispatch(request, *args, **kwargs) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|         context["club"] = self.object.wei |  | ||||||
|         context["information"] = CurrentSurvey.get_algorithm_class().get_bus_information(self.object) |  | ||||||
|         self.object.save() |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|     def get_success_url(self): |  | ||||||
|         self.object.refresh_from_db() |  | ||||||
|         return reverse_lazy("wei:manage_bus", kwargs={"pk": self.object.pk}) |  | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ msgid "" | |||||||
| msgstr "" | msgstr "" | ||||||
| "Project-Id-Version: \n" | "Project-Id-Version: \n" | ||||||
| "Report-Msgid-Bugs-To: \n" | "Report-Msgid-Bugs-To: \n" | ||||||
| "POT-Creation-Date: 2025-07-11 16:10+0200\n" | "POT-Creation-Date: 2025-06-20 13:50+0200\n" | ||||||
| "PO-Revision-Date: 2022-04-11 22:05+0200\n" | "PO-Revision-Date: 2022-04-11 22:05+0200\n" | ||||||
| "Last-Translator: bleizi <bleizi@crans.org>\n" | "Last-Translator: bleizi <bleizi@crans.org>\n" | ||||||
| "Language-Team: French <http://translate.ynerant.fr/projects/nk20/nk20/fr/>\n" | "Language-Team: French <http://translate.ynerant.fr/projects/nk20/nk20/fr/>\n" | ||||||
| @@ -357,7 +357,7 @@ msgstr "Détails de l'activité" | |||||||
| #: apps/note/models/transactions.py:261 | #: apps/note/models/transactions.py:261 | ||||||
| #: apps/note/templates/note/transaction_form.html:17 | #: apps/note/templates/note/transaction_form.html:17 | ||||||
| #: apps/note/templates/note/transaction_form.html:152 | #: apps/note/templates/note/transaction_form.html:152 | ||||||
| #: note_kfet/templates/base.html:79 | #: note_kfet/templates/base.html:78 | ||||||
| msgid "Transfer" | msgid "Transfer" | ||||||
| msgstr "Virement" | msgstr "Virement" | ||||||
|  |  | ||||||
| @@ -474,7 +474,7 @@ msgstr "Inviter" | |||||||
| msgid "Create new activity" | msgid "Create new activity" | ||||||
| msgstr "Créer une nouvelle activité" | msgstr "Créer une nouvelle activité" | ||||||
|  |  | ||||||
| #: apps/activity/views.py:71 note_kfet/templates/base.html:97 | #: apps/activity/views.py:71 note_kfet/templates/base.html:96 | ||||||
| msgid "Activities" | msgid "Activities" | ||||||
| msgstr "Activités" | msgstr "Activités" | ||||||
|  |  | ||||||
| @@ -563,7 +563,7 @@ msgstr "Nom" | |||||||
| #, fuzzy | #, fuzzy | ||||||
| #| msgid "QR-code number" | #| msgid "QR-code number" | ||||||
| msgid "QR code number" | msgid "QR code number" | ||||||
| msgstr "Numéro de QR-code" | msgstr "numéro de QR-code" | ||||||
|  |  | ||||||
| #: apps/food/models.py:23 | #: apps/food/models.py:23 | ||||||
| msgid "Allergen" | msgid "Allergen" | ||||||
| @@ -597,7 +597,7 @@ msgstr "est prêt" | |||||||
| msgid "order" | msgid "order" | ||||||
| msgstr "consigne" | msgstr "consigne" | ||||||
|  |  | ||||||
| #: apps/food/models.py:107 apps/food/views.py:35 | #: apps/food/models.py:107 apps/food/views.py:34 | ||||||
| #: note_kfet/templates/base.html:72 | #: note_kfet/templates/base.html:72 | ||||||
| msgid "Food" | msgid "Food" | ||||||
| msgstr "Bouffe" | msgstr "Bouffe" | ||||||
| @@ -657,75 +657,61 @@ msgstr "QR-codes" | |||||||
| #: apps/food/models.py:286 | #: apps/food/models.py:286 | ||||||
| #: apps/food/templates/food/transformedfood_update.html:24 | #: apps/food/templates/food/transformedfood_update.html:24 | ||||||
| msgid "QR-code number" | msgid "QR-code number" | ||||||
| msgstr "Numéro de QR-code" | msgstr "numéro de QR-code" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:22 | #: apps/food/templates/food/food_detail.html:19 | ||||||
| msgid "Contained in" | msgid "Contained in" | ||||||
| msgstr "Contenu dans" | msgstr "Contenu dans" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:29 | #: apps/food/templates/food/food_detail.html:26 | ||||||
| msgid "Contain" | msgid "Contain" | ||||||
| msgstr "Contient" | msgstr "Contient" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:38 | #: apps/food/templates/food/food_detail.html:35 | ||||||
| msgid "Update" | msgid "Update" | ||||||
| msgstr "Modifier" | msgstr "Modifier" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:43 | #: apps/food/templates/food/food_detail.html:40 | ||||||
| msgid "Add to a meal" | msgid "Add to a meal" | ||||||
| msgstr "Ajouter à un plat" | msgstr "Ajouter à un plat" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:48 | #: apps/food/templates/food/food_detail.html:45 | ||||||
| msgid "Manage ingredients" | msgid "Manage ingredients" | ||||||
| msgstr "Gérer les ingrédients" | msgstr "Gérer les ingrédients" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:52 | #: apps/food/templates/food/food_detail.html:49 | ||||||
| msgid "Return to the food list" | msgid "Return to the food list" | ||||||
| msgstr "Retour à la liste de nourriture" | msgstr "Retour à la liste de nourriture" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:32 | #: apps/food/templates/food/food_list.html:14 | ||||||
| msgid "View food" |  | ||||||
| msgstr "Voir l'aliment" |  | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:37 |  | ||||||
| #: note_kfet/templates/base_search.html:15 |  | ||||||
| msgid "Search by attribute such as name..." |  | ||||||
| msgstr "Chercher par un attribut tel que le nom..." |  | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:49 |  | ||||||
| #: note_kfet/templates/base_search.html:23 |  | ||||||
| msgid "There is no results." |  | ||||||
| msgstr "Il n'y a pas de résultat." |  | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:58 |  | ||||||
| msgid "Meal served" | msgid "Meal served" | ||||||
| msgstr "Plat servis" | msgstr "Plat servis" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:63 | #: apps/food/templates/food/food_list.html:19 | ||||||
| msgid "New meal" | msgid "New meal" | ||||||
| msgstr "Nouveau plat" | msgstr "Nouveau plat" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:72 | #: apps/food/templates/food/food_list.html:28 | ||||||
| msgid "There is no meal served." | msgid "There is no meal served." | ||||||
| msgstr "Il n'y a pas de plat servi." | msgstr "Il n'y a pas de plat servi." | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:79 | #: apps/food/templates/food/food_list.html:35 | ||||||
| msgid "Free food" | msgid "Free food" | ||||||
| msgstr "Open" | msgstr "Open" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:86 | #: apps/food/templates/food/food_list.html:42 | ||||||
| msgid "There is no free food." | msgid "There is no free food." | ||||||
| msgstr "Il n'y a pas de bouffe en open" | msgstr "Il n'y a pas de bouffe en open" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:94 | #: apps/food/templates/food/food_list.html:50 | ||||||
| msgid "Food of your clubs" | msgid "Food of your clubs" | ||||||
| msgstr "Bouffe de tes clubs" | msgstr "Bouffe de tes clubs" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:100 | #: apps/food/templates/food/food_list.html:56 | ||||||
| msgid "Food of club" | msgid "Food of club" | ||||||
| msgstr "Bouffe du club" | msgstr "Bouffe du club" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:107 | #: apps/food/templates/food/food_list.html:63 | ||||||
| msgid "Yours club has not food yet." | msgid "Yours club has not food yet." | ||||||
| msgstr "Ton club n'a pas de bouffe pour l'instant" | msgstr "Ton club n'a pas de bouffe pour l'instant" | ||||||
|  |  | ||||||
| @@ -799,49 +785,49 @@ msgstr "semaines" | |||||||
| msgid "and" | msgid "and" | ||||||
| msgstr "et" | msgstr "et" | ||||||
|  |  | ||||||
| #: apps/food/views.py:120 | #: apps/food/views.py:118 | ||||||
| msgid "Add a new QRCode" | msgid "Add a new QRCode" | ||||||
| msgstr "Ajouter un nouveau QR-code" | msgstr "Ajouter un nouveau QR-code" | ||||||
|  |  | ||||||
| #: apps/food/views.py:169 | #: apps/food/views.py:167 | ||||||
| msgid "Add an aliment" | msgid "Add an aliment" | ||||||
| msgstr "Ajouter un nouvel aliment" | msgstr "Ajouter un nouvel aliment" | ||||||
|  |  | ||||||
| #: apps/food/views.py:228 | #: apps/food/views.py:235 | ||||||
| msgid "Add a meal" | msgid "Add a meal" | ||||||
| msgstr "Ajouter un plat" | msgstr "Ajouter un plat" | ||||||
|  |  | ||||||
| #: apps/food/views.py:259 | #: apps/food/views.py:275 | ||||||
| msgid "Manage ingredients of:" | msgid "Manage ingredients of:" | ||||||
| msgstr "Gestion des ingrédienrs de :" | msgstr "Gestion des ingrédienrs de :" | ||||||
|  |  | ||||||
| #: apps/food/views.py:273 apps/food/views.py:281 | #: apps/food/views.py:289 apps/food/views.py:297 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "Fully used in {meal}" | msgid "Fully used in {meal}" | ||||||
| msgstr "Aliment entièrement utilisé dans : {meal}" | msgstr "Aliment entièrement utilisé dans : {meal}" | ||||||
|  |  | ||||||
| #: apps/food/views.py:320 | #: apps/food/views.py:344 | ||||||
| msgid "Add the ingredient:" | msgid "Add the ingredient:" | ||||||
| msgstr "Ajouter l'ingrédient" | msgstr "Ajouter l'ingrédient" | ||||||
|  |  | ||||||
| #: apps/food/views.py:346 | #: apps/food/views.py:370 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "Food fully used in : {meal.name}" | msgid "Food fully used in : {meal.name}" | ||||||
| msgstr "Aliment entièrement utilisé dans : {meal.name}" | msgstr "Aliment entièrement utilisé dans : {meal.name}" | ||||||
|  |  | ||||||
| #: apps/food/views.py:365 | #: apps/food/views.py:389 | ||||||
| msgid "Update an aliment" | msgid "Update an aliment" | ||||||
| msgstr "Modifier un aliment" | msgstr "Modifier un aliment" | ||||||
|  |  | ||||||
| #: apps/food/views.py:413 | #: apps/food/views.py:437 | ||||||
| msgid "Details of:" | msgid "Details of:" | ||||||
| msgstr "Détails de :" | msgstr "Détails de :" | ||||||
|  |  | ||||||
| #: apps/food/views.py:423 apps/treasury/tables.py:149 | #: apps/food/views.py:447 apps/treasury/tables.py:149 | ||||||
| msgid "Yes" | msgid "Yes" | ||||||
| msgstr "Oui" | msgstr "Oui" | ||||||
|  |  | ||||||
| #: apps/food/views.py:425 apps/member/models.py:99 apps/treasury/tables.py:149 | #: apps/food/views.py:449 apps/member/models.py:99 apps/treasury/tables.py:149 | ||||||
| msgid "No" | msgid "No" | ||||||
| msgstr "Non" | msgstr "Non" | ||||||
|  |  | ||||||
| @@ -1976,8 +1962,8 @@ msgstr "" | |||||||
| "mode de paiement et un⋅e utilisateur⋅rice ou un club" | "mode de paiement et un⋅e utilisateur⋅rice ou un club" | ||||||
|  |  | ||||||
| #: apps/note/models/transactions.py:357 apps/note/models/transactions.py:360 | #: apps/note/models/transactions.py:357 apps/note/models/transactions.py:360 | ||||||
| #: apps/note/models/transactions.py:363 apps/wei/views.py:1105 | #: apps/note/models/transactions.py:363 apps/wei/views.py:1097 | ||||||
| #: apps/wei/views.py:1109 | #: apps/wei/views.py:1101 | ||||||
| msgid "This field is required." | msgid "This field is required." | ||||||
| msgstr "Ce champ est requis." | msgstr "Ce champ est requis." | ||||||
|  |  | ||||||
| @@ -2079,8 +2065,6 @@ msgstr "Historique des transactions récentes" | |||||||
| #: apps/note/templates/note/mails/weekly_report.txt:32 | #: apps/note/templates/note/mails/weekly_report.txt:32 | ||||||
| #: apps/registration/templates/registration/mails/email_validation_email.html:40 | #: apps/registration/templates/registration/mails/email_validation_email.html:40 | ||||||
| #: apps/registration/templates/registration/mails/email_validation_email.txt:16 | #: apps/registration/templates/registration/mails/email_validation_email.txt:16 | ||||||
| #: apps/scripts/templates/scripts/food_report.html:48 |  | ||||||
| #: apps/scripts/templates/scripts/food_report.txt:14 |  | ||||||
| msgid "Mail generated by the Note Kfet on the" | msgid "Mail generated by the Note Kfet on the" | ||||||
| msgstr "Mail généré par la Note Kfet le" | msgstr "Mail généré par la Note Kfet le" | ||||||
|  |  | ||||||
| @@ -2192,7 +2176,7 @@ msgstr "Chercher un bouton" | |||||||
| msgid "Update button" | msgid "Update button" | ||||||
| msgstr "Modifier le bouton" | msgstr "Modifier le bouton" | ||||||
|  |  | ||||||
| #: apps/note/views.py:156 note_kfet/templates/base.html:67 | #: apps/note/views.py:156 note_kfet/templates/base.html:66 | ||||||
| msgid "Consumptions" | msgid "Consumptions" | ||||||
| msgstr "Consommations" | msgstr "Consommations" | ||||||
|  |  | ||||||
| @@ -2285,7 +2269,7 @@ msgstr "s'applique au club" | |||||||
| msgid "role permissions" | msgid "role permissions" | ||||||
| msgstr "permissions par rôles" | msgstr "permissions par rôles" | ||||||
|  |  | ||||||
| #: apps/permission/signals.py:75 | #: apps/permission/signals.py:73 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "" | msgid "" | ||||||
| "You don't have the permission to change the field {field} on this instance " | "You don't have the permission to change the field {field} on this instance " | ||||||
| @@ -2294,7 +2278,7 @@ msgstr "" | |||||||
| "Vous n'avez pas la permission de modifier le champ {field} sur l'instance du " | "Vous n'avez pas la permission de modifier le champ {field} sur l'instance du " | ||||||
| "modèle {app_label}.{model_name}." | "modèle {app_label}.{model_name}." | ||||||
|  |  | ||||||
| #: apps/permission/signals.py:85 apps/permission/views.py:104 | #: apps/permission/signals.py:83 apps/permission/views.py:104 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "" | msgid "" | ||||||
| "You don't have the permission to add an instance of model {app_label}." | "You don't have the permission to add an instance of model {app_label}." | ||||||
| @@ -2303,7 +2287,7 @@ msgstr "" | |||||||
| "Vous n'avez pas la permission d'ajouter une instance du modèle {app_label}." | "Vous n'avez pas la permission d'ajouter une instance du modèle {app_label}." | ||||||
| "{model_name}." | "{model_name}." | ||||||
|  |  | ||||||
| #: apps/permission/signals.py:114 | #: apps/permission/signals.py:112 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "" | msgid "" | ||||||
| "You don't have the permission to delete this instance of model {app_label}." | "You don't have the permission to delete this instance of model {app_label}." | ||||||
| @@ -2391,7 +2375,7 @@ msgstr "" | |||||||
| "Vous n'avez pas la permission d'ajouter une instance du modèle « {model} » " | "Vous n'avez pas la permission d'ajouter une instance du modèle « {model} » " | ||||||
| "avec ces paramètres. Merci de les corriger et de réessayer." | "avec ces paramètres. Merci de les corriger et de réessayer." | ||||||
|  |  | ||||||
| #: apps/permission/views.py:111 note_kfet/templates/base.html:121 | #: apps/permission/views.py:111 note_kfet/templates/base.html:120 | ||||||
| msgid "Rights" | msgid "Rights" | ||||||
| msgstr "Droits" | msgstr "Droits" | ||||||
|  |  | ||||||
| @@ -2596,7 +2580,7 @@ msgstr "" | |||||||
| msgid "Invalidate pre-registration" | msgid "Invalidate pre-registration" | ||||||
| msgstr "Invalider l'inscription" | msgstr "Invalider l'inscription" | ||||||
|  |  | ||||||
| #: apps/treasury/apps.py:12 note_kfet/templates/base.html:103 | #: apps/treasury/apps.py:12 note_kfet/templates/base.html:102 | ||||||
| msgid "Treasury" | msgid "Treasury" | ||||||
| msgstr "Trésorerie" | msgstr "Trésorerie" | ||||||
|  |  | ||||||
| @@ -3012,7 +2996,7 @@ msgstr "Gérer les crédits de la Société générale" | |||||||
|  |  | ||||||
| #: apps/wei/apps.py:10 apps/wei/models.py:42 apps/wei/models.py:43 | #: apps/wei/apps.py:10 apps/wei/models.py:42 apps/wei/models.py:43 | ||||||
| #: apps/wei/models.py:67 apps/wei/models.py:192 | #: apps/wei/models.py:67 apps/wei/models.py:192 | ||||||
| #: note_kfet/templates/base.html:109 | #: note_kfet/templates/base.html:108 | ||||||
| msgid "WEI" | msgid "WEI" | ||||||
| msgstr "WEI" | msgstr "WEI" | ||||||
|  |  | ||||||
| @@ -3057,19 +3041,14 @@ msgstr "Rôles au WEI" | |||||||
| msgid "Select the roles that you are interested in." | msgid "Select the roles that you are interested in." | ||||||
| msgstr "Sélectionnez les rôles qui vous intéressent." | msgstr "Sélectionnez les rôles qui vous intéressent." | ||||||
|  |  | ||||||
| #: apps/wei/forms/registration.py:160 | #: apps/wei/forms/registration.py:147 | ||||||
| msgid "This team doesn't belong to the given bus." | msgid "This team doesn't belong to the given bus." | ||||||
| msgstr "Cette équipe n'appartient pas à ce bus." | msgstr "Cette équipe n'appartient pas à ce bus." | ||||||
|  |  | ||||||
| #: apps/wei/forms/surveys/wei2021.py:35 apps/wei/forms/surveys/wei2022.py:38 | #: apps/wei/forms/surveys/wei2021.py:35 apps/wei/forms/surveys/wei2022.py:38 | ||||||
| #: apps/wei/forms/surveys/wei2025.py:36 |  | ||||||
| msgid "Choose a word:" | msgid "Choose a word:" | ||||||
| msgstr "Choisissez un mot :" | msgstr "Choisissez un mot :" | ||||||
|  |  | ||||||
| #: apps/wei/forms/surveys/wei2025.py:123 |  | ||||||
| msgid "Rate between 0 and 5." |  | ||||||
| msgstr "Note entre 0 et 5." |  | ||||||
|  |  | ||||||
| #: apps/wei/models.py:25 apps/wei/templates/wei/base.html:36 | #: apps/wei/models.py:25 apps/wei/templates/wei/base.html:36 | ||||||
| msgid "year" | msgid "year" | ||||||
| msgstr "année" | msgstr "année" | ||||||
| @@ -3136,7 +3115,7 @@ msgstr "Rôle au WEI" | |||||||
| msgid "Credit from Société générale" | msgid "Credit from Société générale" | ||||||
| msgstr "Crédit de la Société générale" | msgstr "Crédit de la Société générale" | ||||||
|  |  | ||||||
| #: apps/wei/models.py:202 apps/wei/views.py:992 | #: apps/wei/models.py:202 apps/wei/views.py:984 | ||||||
| msgid "Caution check given" | msgid "Caution check given" | ||||||
| msgstr "Chèque de caution donné" | msgstr "Chèque de caution donné" | ||||||
|  |  | ||||||
| @@ -3271,7 +3250,7 @@ msgstr "Année" | |||||||
| msgid "preferred bus" | msgid "preferred bus" | ||||||
| msgstr "bus préféré" | msgstr "bus préféré" | ||||||
|  |  | ||||||
| #: apps/wei/tables.py:210 apps/wei/templates/wei/bus_detail.html:38 | #: apps/wei/tables.py:210 apps/wei/templates/wei/bus_detail.html:36 | ||||||
| #: apps/wei/templates/wei/busteam_detail.html:52 | #: apps/wei/templates/wei/busteam_detail.html:52 | ||||||
| msgid "Teams" | msgid "Teams" | ||||||
| msgstr "Équipes" | msgstr "Équipes" | ||||||
| @@ -3365,22 +3344,18 @@ msgstr "Voir le WEI" | |||||||
|  |  | ||||||
| #: apps/wei/templates/wei/bus_detail.html:21 | #: apps/wei/templates/wei/bus_detail.html:21 | ||||||
| msgid "View club" | msgid "View club" | ||||||
| msgstr "Voir le club" | msgstr "Voir le lub" | ||||||
|  |  | ||||||
| #: apps/wei/templates/wei/bus_detail.html:26 | #: apps/wei/templates/wei/bus_detail.html:26 | ||||||
| msgid "Edit information" |  | ||||||
| msgstr "Modifier les informations" |  | ||||||
|  |  | ||||||
| #: apps/wei/templates/wei/bus_detail.html:28 |  | ||||||
| #: apps/wei/templates/wei/busteam_detail.html:24 | #: apps/wei/templates/wei/busteam_detail.html:24 | ||||||
| msgid "Add team" | msgid "Add team" | ||||||
| msgstr "Ajouter une équipe" | msgstr "Ajouter une équipe" | ||||||
|  |  | ||||||
| #: apps/wei/templates/wei/bus_detail.html:51 | #: apps/wei/templates/wei/bus_detail.html:49 | ||||||
| msgid "Members" | msgid "Members" | ||||||
| msgstr "Membres" | msgstr "Membres" | ||||||
|  |  | ||||||
| #: apps/wei/templates/wei/bus_detail.html:60 | #: apps/wei/templates/wei/bus_detail.html:58 | ||||||
| #: apps/wei/templates/wei/busteam_detail.html:62 | #: apps/wei/templates/wei/busteam_detail.html:62 | ||||||
| #: apps/wei/templates/wei/weimembership_list.html:31 | #: apps/wei/templates/wei/weimembership_list.html:31 | ||||||
| msgid "View as PDF" | msgid "View as PDF" | ||||||
| @@ -3388,8 +3363,8 @@ msgstr "Télécharger au format PDF" | |||||||
|  |  | ||||||
| #: apps/wei/templates/wei/survey.html:11 | #: apps/wei/templates/wei/survey.html:11 | ||||||
| #: apps/wei/templates/wei/survey_closed.html:11 | #: apps/wei/templates/wei/survey_closed.html:11 | ||||||
| #: apps/wei/templates/wei/survey_end.html:11 apps/wei/views.py:1167 | #: apps/wei/templates/wei/survey_end.html:11 apps/wei/views.py:1159 | ||||||
| #: apps/wei/views.py:1222 apps/wei/views.py:1269 | #: apps/wei/views.py:1214 apps/wei/views.py:1261 | ||||||
| msgid "Survey WEI" | msgid "Survey WEI" | ||||||
| msgstr "Questionnaire WEI" | msgstr "Questionnaire WEI" | ||||||
|  |  | ||||||
| @@ -3669,51 +3644,51 @@ msgstr "" | |||||||
| msgid "Update WEI Registration" | msgid "Update WEI Registration" | ||||||
| msgstr "Modifier l'inscription WEI" | msgstr "Modifier l'inscription WEI" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:811 | #: apps/wei/views.py:810 | ||||||
| msgid "No membership found for this registration" | msgid "No membership found for this registration" | ||||||
| msgstr "Pas d'adhésion trouvée pour cette inscription" | msgstr "Pas d'adhésion trouvée pour cette inscription" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:820 | #: apps/wei/views.py:819 | ||||||
| msgid "You don't have the permission to update memberships" | msgid "You don't have the permission to update memberships" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Vous n'avez pas la permission d'ajouter une instance du modèle {app_label}." | "Vous n'avez pas la permission d'ajouter une instance du modèle {app_label}." | ||||||
| "{model_name}." | "{model_name}." | ||||||
|  |  | ||||||
| #: apps/wei/views.py:826 | #: apps/wei/views.py:825 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "You don't have the permission to update the field %(field)s" | msgid "You don't have the permission to update the field %(field)s" | ||||||
| msgstr "Vous n'avez pas la permission de modifier le champ %(field)s" | msgstr "Vous n'avez pas la permission de modifier le champ %(field)s" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:871 | #: apps/wei/views.py:870 | ||||||
| msgid "Delete WEI registration" | msgid "Delete WEI registration" | ||||||
| msgstr "Supprimer l'inscription WEI" | msgstr "Supprimer l'inscription WEI" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:882 | #: apps/wei/views.py:881 | ||||||
| msgid "You don't have the right to delete this WEI registration." | msgid "You don't have the right to delete this WEI registration." | ||||||
| msgstr "Vous n'avez pas la permission de supprimer cette inscription au WEI." | msgstr "Vous n'avez pas la permission de supprimer cette inscription au WEI." | ||||||
|  |  | ||||||
| #: apps/wei/views.py:900 | #: apps/wei/views.py:899 | ||||||
| msgid "Validate WEI registration" | msgid "Validate WEI registration" | ||||||
| msgstr "Valider l'inscription WEI" | msgstr "Valider l'inscription WEI" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:993 | #: apps/wei/views.py:985 | ||||||
| msgid "Please make sure the check is given before validating the registration" | msgid "Please make sure the check is given before validating the registration" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Merci de vous assurer que le chèque a bien été donné avant de valider " | "Merci de vous assurer que le chèque a bien été donné avant de valider " | ||||||
| "l'adhésion" | "l'adhésion" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:999 | #: apps/wei/views.py:991 | ||||||
| msgid "Create deposit transaction" | msgid "Create deposit transaction" | ||||||
| msgstr "Créer une transaction de caution" | msgstr "Créer une transaction de caution" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1000 | #: apps/wei/views.py:992 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "" | msgid "" | ||||||
| "A transaction of %(amount).2f€ will be created from the user's Note account" | "A transaction of %(amount).2f€ will be created from the user's Note account" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Un transaction de %(amount).2f€ va être créée depuis la note de l'utilisateur" | "Un transaction de %(amount).2f€ va être créée depuis la note de l'utilisateur" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1095 | #: apps/wei/views.py:1087 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "" | msgid "" | ||||||
| "This user doesn't have enough money to join this club and pay the deposit. " | "This user doesn't have enough money to join this club and pay the deposit. " | ||||||
| @@ -3723,21 +3698,21 @@ msgstr "" | |||||||
| "payer la cautionSolde actuel : %(balance)d€, crédit : %(credit)d€, requis : " | "payer la cautionSolde actuel : %(balance)d€, crédit : %(credit)d€, requis : " | ||||||
| "%(needed)d€" | "%(needed)d€" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1148 | #: apps/wei/views.py:1140 | ||||||
| #, fuzzy, python-format | #, fuzzy, python-format | ||||||
| #| msgid "total amount" | #| msgid "total amount" | ||||||
| msgid "Caution %(name)s" | msgid "Caution %(name)s" | ||||||
| msgstr "montant total" | msgstr "montant total" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1362 | #: apps/wei/views.py:1354 | ||||||
| msgid "Attribute buses to first year members" | msgid "Attribute buses to first year members" | ||||||
| msgstr "Répartir les 1A dans les bus" | msgstr "Répartir les 1A dans les bus" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1388 | #: apps/wei/views.py:1379 | ||||||
| msgid "Attribute bus" | msgid "Attribute bus" | ||||||
| msgstr "Attribuer un bus" | msgstr "Attribuer un bus" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1428 | #: apps/wei/views.py:1419 | ||||||
| msgid "" | msgid "" | ||||||
| "No first year student without a bus found. Either all of them have a bus, or " | "No first year student without a bus found. Either all of them have a bus, or " | ||||||
| "none has filled the survey yet." | "none has filled the survey yet." | ||||||
| @@ -3761,13 +3736,13 @@ msgstr "bde" | |||||||
|  |  | ||||||
| #: apps/wrapped/models.py:65 | #: apps/wrapped/models.py:65 | ||||||
| msgid "data json" | msgid "data json" | ||||||
| msgstr "données json" | msgstr "donnée json" | ||||||
|  |  | ||||||
| #: apps/wrapped/models.py:66 | #: apps/wrapped/models.py:66 | ||||||
| msgid "data in the wrapped and generated by the script generate_wrapped" | msgid "data in the wrapped and generated by the script generate_wrapped" | ||||||
| msgstr "donnée dans le wrapped et générée par le script generate_wrapped" | msgstr "donnée dans le wrapped et générée par le script generate_wrapped" | ||||||
|  |  | ||||||
| #: apps/wrapped/models.py:70 note_kfet/templates/base.html:115 | #: apps/wrapped/models.py:70 note_kfet/templates/base.html:114 | ||||||
| msgid "Wrapped" | msgid "Wrapped" | ||||||
| msgstr "Wrapped" | msgstr "Wrapped" | ||||||
|  |  | ||||||
| @@ -3800,7 +3775,7 @@ msgid "Copy link" | |||||||
| msgstr "Copier le lien" | msgstr "Copier le lien" | ||||||
|  |  | ||||||
| #: apps/wrapped/templates/wrapped/1/wrapped_base.html:16 | #: apps/wrapped/templates/wrapped/1/wrapped_base.html:16 | ||||||
| #: note_kfet/templates/base.html:15 | #: note_kfet/templates/base.html:14 | ||||||
| msgid "The ENS Paris-Saclay BDE note." | msgid "The ENS Paris-Saclay BDE note." | ||||||
| msgstr "La note du BDE de l'ENS Paris-Saclay." | msgstr "La note du BDE de l'ENS Paris-Saclay." | ||||||
|  |  | ||||||
| @@ -3903,7 +3878,7 @@ msgid "" | |||||||
| "Do not forget to ask permission to people who are in your wrapped before to " | "Do not forget to ask permission to people who are in your wrapped before to " | ||||||
| "make them public" | "make them public" | ||||||
| msgstr "" | msgstr "" | ||||||
| "N'oublie pas de demander la permission des personnes apparaissant dans un " | "N'oublies pas de demander la permission des personnes apparaissant dans un " | ||||||
| "wrapped avant de le rendre public" | "wrapped avant de le rendre public" | ||||||
|  |  | ||||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:40 | #: apps/wrapped/templates/wrapped/wrapped_list.html:40 | ||||||
| @@ -3922,19 +3897,19 @@ msgstr "Le wrapped est public" | |||||||
| msgid "List of wrapped" | msgid "List of wrapped" | ||||||
| msgstr "Liste des wrapped" | msgstr "Liste des wrapped" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:180 | #: note_kfet/settings/base.py:177 | ||||||
| msgid "German" | msgid "German" | ||||||
| msgstr "Allemand" | msgstr "Allemand" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:181 | #: note_kfet/settings/base.py:178 | ||||||
| msgid "English" | msgid "English" | ||||||
| msgstr "Anglais" | msgstr "Anglais" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:182 | #: note_kfet/settings/base.py:179 | ||||||
| msgid "Spanish" | msgid "Spanish" | ||||||
| msgstr "Espagnol" | msgstr "Espagnol" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:183 | #: note_kfet/settings/base.py:180 | ||||||
| msgid "French" | msgid "French" | ||||||
| msgstr "Français" | msgstr "Français" | ||||||
|  |  | ||||||
| @@ -3995,34 +3970,34 @@ msgstr "" | |||||||
| msgid "Reset" | msgid "Reset" | ||||||
| msgstr "Réinitialiser" | msgstr "Réinitialiser" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:85 | #: note_kfet/templates/base.html:84 | ||||||
| msgid "Users" | msgid "Users" | ||||||
| msgstr "Utilisateur·rices" | msgstr "Utilisateur·rices" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:91 | #: note_kfet/templates/base.html:90 | ||||||
| msgid "Clubs" | msgid "Clubs" | ||||||
| msgstr "Clubs" | msgstr "Clubs" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:126 | #: note_kfet/templates/base.html:125 | ||||||
| msgid "Admin" | msgid "Admin" | ||||||
| msgstr "Admin" | msgstr "Admin" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:140 | #: note_kfet/templates/base.html:139 | ||||||
| msgid "My account" | msgid "My account" | ||||||
| msgstr "Mon compte" | msgstr "Mon compte" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:145 | #: note_kfet/templates/base.html:142 | ||||||
| msgid "Log out" | msgid "Log out" | ||||||
| msgstr "Se déconnecter" | msgstr "Se déconnecter" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:154 | #: note_kfet/templates/base.html:150 | ||||||
| #: note_kfet/templates/registration/signup.html:6 | #: note_kfet/templates/registration/signup.html:6 | ||||||
| #: note_kfet/templates/registration/signup.html:11 | #: note_kfet/templates/registration/signup.html:11 | ||||||
| #: note_kfet/templates/registration/signup.html:28 | #: note_kfet/templates/registration/signup.html:28 | ||||||
| msgid "Sign up" | msgid "Sign up" | ||||||
| msgstr "Inscription" | msgstr "Inscription" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:161 | #: note_kfet/templates/base.html:157 | ||||||
| #: note_kfet/templates/registration/login.html:6 | #: note_kfet/templates/registration/login.html:6 | ||||||
| #: note_kfet/templates/registration/login.html:15 | #: note_kfet/templates/registration/login.html:15 | ||||||
| #: note_kfet/templates/registration/login.html:38 | #: note_kfet/templates/registration/login.html:38 | ||||||
| @@ -4030,7 +4005,7 @@ msgstr "Inscription" | |||||||
| msgid "Log in" | msgid "Log in" | ||||||
| msgstr "Se connecter" | msgstr "Se connecter" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:175 | #: note_kfet/templates/base.html:171 | ||||||
| msgid "" | msgid "" | ||||||
| "You are not a BDE member anymore. Please renew your membership if you want " | "You are not a BDE member anymore. Please renew your membership if you want " | ||||||
| "to use the note." | "to use the note." | ||||||
| @@ -4038,7 +4013,7 @@ msgstr "" | |||||||
| "Vous n'êtes plus adhérent·e BDE. Merci de réadhérer si vous voulez profiter " | "Vous n'êtes plus adhérent·e BDE. Merci de réadhérer si vous voulez profiter " | ||||||
| "de la note." | "de la note." | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:181 | #: note_kfet/templates/base.html:177 | ||||||
| msgid "" | msgid "" | ||||||
| "Your e-mail address is not validated. Please check your mail inbox and click " | "Your e-mail address is not validated. Please check your mail inbox and click " | ||||||
| "on the validation link." | "on the validation link." | ||||||
| @@ -4046,7 +4021,7 @@ msgstr "" | |||||||
| "Votre adresse e-mail n'est pas validée. Merci de vérifier votre boîte mail " | "Votre adresse e-mail n'est pas validée. Merci de vérifier votre boîte mail " | ||||||
| "et de cliquer sur le lien de validation." | "et de cliquer sur le lien de validation." | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:187 | #: note_kfet/templates/base.html:183 | ||||||
| msgid "" | msgid "" | ||||||
| "You declared that you opened a bank account in the Société générale. The " | "You declared that you opened a bank account in the Société générale. The " | ||||||
| "bank did not validate the creation of the account to the BDE, so the " | "bank did not validate the creation of the account to the BDE, so the " | ||||||
| @@ -4060,38 +4035,22 @@ msgstr "" | |||||||
| "vérification peut durer quelques jours. Merci de vous assurer de bien aller " | "vérification peut durer quelques jours. Merci de vous assurer de bien aller " | ||||||
| "au bout de vos démarches." | "au bout de vos démarches." | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:214 | #: note_kfet/templates/base.html:206 | ||||||
| msgid "Contact us" | msgid "Contact us" | ||||||
| msgstr "Nous contacter" | msgstr "Nous contacter" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:216 | #: note_kfet/templates/base.html:208 | ||||||
| msgid "Technical Support" | msgid "Technical Support" | ||||||
| msgstr "Support technique" | msgstr "Support technique" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:218 | #: note_kfet/templates/base.html:210 | ||||||
| msgid "Charte Info (FR)" | msgid "Charte Info (FR)" | ||||||
| msgstr "Charte Info (FR)" | msgstr "Charte Info (FR)" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:220 | #: note_kfet/templates/base.html:212 | ||||||
| msgid "FAQ (FR)" | msgid "FAQ (FR)" | ||||||
| msgstr "FAQ (FR)" | msgstr "FAQ (FR)" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:222 |  | ||||||
| msgid "Managed by BDE" |  | ||||||
| msgstr "Géré par le BDE" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:224 |  | ||||||
| msgid "Hosted by Cr@ns" |  | ||||||
| msgstr "Hébergé par le Cr@ans" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:266 |  | ||||||
| msgid "The note is not available for now" |  | ||||||
| msgstr "La note est indisponible pour le moment" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:268 |  | ||||||
| msgid "Thank you for your understanding -- The Respos Info of BDE" |  | ||||||
| msgstr "Merci de votre compréhension -- Les Respos Info du BDE" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/base_search.html:15 | #: note_kfet/templates/base_search.html:15 | ||||||
| msgid "Search by attribute such as name..." | msgid "Search by attribute such as name..." | ||||||
| msgstr "Chercher par un attribut tel que le nom..." | msgstr "Chercher par un attribut tel que le nom..." | ||||||
| @@ -4100,41 +4059,6 @@ msgstr "Chercher par un attribut tel que le nom..." | |||||||
| msgid "There is no results." | msgid "There is no results." | ||||||
| msgstr "Il n'y a pas de résultat." | msgstr "Il n'y a pas de résultat." | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/logged.html:8 |  | ||||||
| msgid "" |  | ||||||
| "<h3>Log In Successful</h3>You have successfully logged into the Central " |  | ||||||
| "Authentication Service.<br/>For security reasons, please Log Out and Exit " |  | ||||||
| "your web browser when you are done accessing services that require " |  | ||||||
| "authentication!" |  | ||||||
| msgstr "" |  | ||||||
| "<h3>Connection réussie</h3>Vous vous êtes bien connecté au Service Central d'Authentification." |  | ||||||
| "<br/>Pour des raisons de sécurité, veuillez vous déconnecter et fermer votre navigateur internet " |  | ||||||
| "une fois que vous aurez fini d'accéder aux services qui requiert une authentification !" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/logged.html:14 |  | ||||||
| msgid "Log me out from all my sessions" |  | ||||||
| msgstr "Me déconnecter de toutes mes sessions" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/logged.html:20 |  | ||||||
| msgid "Forget the identity provider" |  | ||||||
| msgstr "Oublier le fournisseur d'identité" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/logged.html:24 |  | ||||||
| msgid "Logout" |  | ||||||
| msgstr "Déconnexion" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/login.html:11 |  | ||||||
| msgid "Please log in" |  | ||||||
| msgstr "Veuillez vous connecter" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/login.html:23 |  | ||||||
| msgid "Login" |  | ||||||
| msgstr "Connexion" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/cas/warn.html:14 |  | ||||||
| msgid "Connect to the service" |  | ||||||
| msgstr "Connexion au service" |  | ||||||
|  |  | ||||||
| #: note_kfet/templates/oauth2_provider/application_confirm_delete.html:8 | #: note_kfet/templates/oauth2_provider/application_confirm_delete.html:8 | ||||||
| msgid "Are you sure to delete the application" | msgid "Are you sure to delete the application" | ||||||
| msgstr "Êtes-vous sûr⋅e de vouloir supprimer l'application" | msgstr "Êtes-vous sûr⋅e de vouloir supprimer l'application" | ||||||
| @@ -4355,86 +4279,10 @@ msgstr "" | |||||||
| "d'adhésion. Vous devez également valider votre adresse email en suivant le " | "d'adhésion. Vous devez également valider votre adresse email en suivant le " | ||||||
| "lien que vous avez reçu." | "lien que vous avez reçu." | ||||||
|  |  | ||||||
| #, fuzzy | #, fuzzy, python-format | ||||||
| #~| msgid "QR-code" | #~| msgid "Creation date" | ||||||
| #~ msgid "Go to QR-code" | #~ msgid "Deposit %(name)s" | ||||||
| #~ msgstr "QR-code" | #~ msgstr "Caution %(name)s" | ||||||
|  |  | ||||||
| #, python-brace-format |  | ||||||
| #~ msgid "QR-code number {qr_code_number}" |  | ||||||
| #~ msgstr "Numéro du QR-code {qr_code_number}" |  | ||||||
|  |  | ||||||
| #~ msgid "was eaten" |  | ||||||
| #~ msgstr "a été mangé" |  | ||||||
|  |  | ||||||
| #~ msgid "is active" |  | ||||||
| #~ msgstr "est en cours" |  | ||||||
|  |  | ||||||
| #~ msgid "foods" |  | ||||||
| #~ msgstr "bouffes" |  | ||||||
|  |  | ||||||
| #~ msgid "Arrival date" |  | ||||||
| #~ msgstr "Date d'arrivée" |  | ||||||
|  |  | ||||||
| #~ msgid "Active" |  | ||||||
| #~ msgstr "Actif" |  | ||||||
|  |  | ||||||
| #~ msgid "Eaten" |  | ||||||
| #~ msgstr "Mangé" |  | ||||||
|  |  | ||||||
| #~ msgid "number" |  | ||||||
| #~ msgstr "numéro" |  | ||||||
|  |  | ||||||
| #~ msgid "View details" |  | ||||||
| #~ msgstr "Voir plus" |  | ||||||
|  |  | ||||||
| #~ msgid "Ready" |  | ||||||
| #~ msgstr "Prêt" |  | ||||||
|  |  | ||||||
| #~ msgid "Creation date" |  | ||||||
| #~ msgstr "Date de création" |  | ||||||
|  |  | ||||||
| #~ msgid "Ingredients" |  | ||||||
| #~ msgstr "Ingrédients" |  | ||||||
|  |  | ||||||
| #~ msgid "Open" |  | ||||||
| #~ msgstr "Open" |  | ||||||
|  |  | ||||||
| #~ msgid "All meals" |  | ||||||
| #~ msgstr "Tout les plats" |  | ||||||
|  |  | ||||||
| #~ msgid "There is no meal." |  | ||||||
| #~ msgstr "Il n'y a pas de plat" |  | ||||||
|  |  | ||||||
| #~ msgid "The product is already prepared" |  | ||||||
| #~ msgstr "Le produit est déjà prêt" |  | ||||||
|  |  | ||||||
| #~ msgid "Add a new basic food with QRCode" |  | ||||||
| #~ msgstr "Ajouter un nouvel ingrédient avec un QR-code" |  | ||||||
|  |  | ||||||
| #~ msgid "QRCode" |  | ||||||
| #~ msgstr "QR-code" |  | ||||||
|  |  | ||||||
| #~ msgid "Add a new meal" |  | ||||||
| #~ msgstr "Ajouter un nouveau plat" |  | ||||||
|  |  | ||||||
| #~ msgid "Update a meal" |  | ||||||
| #~ msgstr "Modifier le plat" |  | ||||||
|  |  | ||||||
| #, fuzzy |  | ||||||
| #~| msgid "invalidate" |  | ||||||
| #~ msgid "Enter a valid color." |  | ||||||
| #~ msgstr "dévalider" |  | ||||||
|  |  | ||||||
| #, fuzzy |  | ||||||
| #~| msgid "invalidate" |  | ||||||
| #~ msgid "Enter a valid value." |  | ||||||
| #~ msgstr "dévalider" |  | ||||||
|  |  | ||||||
| #, fuzzy |  | ||||||
| #~| msgid "Invitation" |  | ||||||
| #~ msgid "Syndication" |  | ||||||
| #~ msgstr "Invitation" |  | ||||||
|  |  | ||||||
| #, fuzzy | #, fuzzy | ||||||
| #~| msgid "There is no results." | #~| msgid "There is no results." | ||||||
| @@ -4848,7 +4696,7 @@ msgstr "" | |||||||
|  |  | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| #~ msgid "QR-code number {qr_code_number}" | #~ msgid "QR-code number {qr_code_number}" | ||||||
| #~ msgstr "Numéro du QR-code {qr_code_number}" | #~ msgstr "numéro du QR-code {qr_code_number}" | ||||||
|  |  | ||||||
| #~ msgid "was eaten" | #~ msgid "was eaten" | ||||||
| #~ msgstr "a été mangé" | #~ msgstr "a été mangé" | ||||||
|   | |||||||
| @@ -27,6 +27,5 @@ MAILTO=notekfet2020@lists.crans.org | |||||||
| # Vider les tokens Oauth2 | # Vider les tokens Oauth2 | ||||||
|  00  6     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py cleartokens -v 0 |  00  6     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py cleartokens -v 0 | ||||||
| # Envoyer la liste des abonnés à la NL BDA | # Envoyer la liste des abonnés à la NL BDA | ||||||
|  00  10    *   *   0     root   cd /var/www/note_kfet && env/bin/python manage.py extract_ml_registrations -t art -e "bda.ensparissaclay@gmail.com" |  00  10     *   *   0     root   cd /var/www/note_kfet && env/bin/python manage.py extract_ml_registrations -t art -e "bda.ensparissaclay@gmail.com" | ||||||
| # Envoyer la liste de la bouffe au club et aux GCKs |   | ||||||
|  00  8     *   *   1     root   cd /var/www/note_kfet && env/bin/python manage.py send_mail_for_food --report --club |  | ||||||
| @@ -56,8 +56,3 @@ if "cas_server" in settings.INSTALLED_APPS: | |||||||
|     from cas_server.models import * |     from cas_server.models import * | ||||||
|     admin_site.register(ServicePattern, ServicePatternAdmin) |     admin_site.register(ServicePattern, ServicePatternAdmin) | ||||||
|     admin_site.register(FederatedIendityProvider, FederatedIendityProviderAdmin) |     admin_site.register(FederatedIendityProvider, FederatedIendityProviderAdmin) | ||||||
|  |  | ||||||
| if "constance" in settings.INSTALLED_APPS: |  | ||||||
|     from constance.admin import * |  | ||||||
|     from constance.models import * |  | ||||||
|     admin_site.register([Config], ConstanceAdmin) |  | ||||||
|   | |||||||
| @@ -39,9 +39,7 @@ SECURE_HSTS_PRELOAD = True | |||||||
| INSTALLED_APPS = [ | INSTALLED_APPS = [ | ||||||
|     # External apps |     # External apps | ||||||
|     'bootstrap_datepicker_plus', |     'bootstrap_datepicker_plus', | ||||||
|     'cas_server', |  | ||||||
|     'colorfield', |     'colorfield', | ||||||
|     'constance', |  | ||||||
|     'crispy_bootstrap4', |     'crispy_bootstrap4', | ||||||
|     'crispy_forms', |     'crispy_forms', | ||||||
| #    'django_htcpcp_tea', | #    'django_htcpcp_tea', | ||||||
| @@ -113,7 +111,6 @@ TEMPLATES = [ | |||||||
|         'APP_DIRS': True, |         'APP_DIRS': True, | ||||||
|         'OPTIONS': { |         'OPTIONS': { | ||||||
|             'context_processors': [ |             'context_processors': [ | ||||||
|                 'constance.context_processors.config', |  | ||||||
|                 'django.template.context_processors.debug', |                 'django.template.context_processors.debug', | ||||||
|                 'django.template.context_processors.request', |                 'django.template.context_processors.request', | ||||||
|                 'django.contrib.auth.context_processors.auth', |                 'django.contrib.auth.context_processors.auth', | ||||||
| @@ -310,30 +307,6 @@ PHONENUMBER_DEFAULT_REGION = 'FR' | |||||||
|  |  | ||||||
| # We add custom information to CAS, in order to give a normalized name to other services | # We add custom information to CAS, in order to give a normalized name to other services | ||||||
| CAS_AUTH_CLASS = 'member.auth.CustomAuthUser' | CAS_AUTH_CLASS = 'member.auth.CustomAuthUser' | ||||||
| CAS_LOGIN_TEMPLATE = 'cas/login.html' |  | ||||||
| CAS_LOGOUT_TEMPLATE = 'cas/logout.html' |  | ||||||
| CAS_WARN_TEMPLATE = 'cas/warn.html' |  | ||||||
| CAS_LOGGED_TEMPLATE = 'cas/logged.html' |  | ||||||
|  |  | ||||||
| # Default field for primary key | # Default field for primary key | ||||||
| DEFAULT_AUTO_FIELD = "django.db.models.AutoField" | DEFAULT_AUTO_FIELD = "django.db.models.AutoField" | ||||||
|  |  | ||||||
| # Constance settings |  | ||||||
| CONSTANCE_ADDITIONAL_FIELDS = { |  | ||||||
|     'banner_type': ['django.forms.fields.ChoiceField', { |  | ||||||
|         'widget': 'django.forms.Select', |  | ||||||
|         'choices': (('info', 'Info'), ('success', 'Success'), ('warning', 'Warning'), ('danger', 'Danger')) |  | ||||||
|     }], |  | ||||||
| } |  | ||||||
| CONSTANCE_CONFIG = { |  | ||||||
|     'BANNER_MESSAGE': ('', 'Some message', str), |  | ||||||
|     'BANNER_TYPE': ('info', 'Banner type', 'banner_type'), |  | ||||||
|     'MAINTENANCE': (False, 'check for mainteance mode', bool), |  | ||||||
|     'MAINTENANCE_MESSAGE': ('', 'Some maintenance message', str), |  | ||||||
| } |  | ||||||
| CONSTANCE_CONFIG_FIELDSETS = { |  | ||||||
|     'Maintenance': ('MAINTENANCE_MESSAGE', 'MAINTENANCE'), |  | ||||||
|     'Banner': ('BANNER_MESSAGE', 'BANNER_TYPE'), |  | ||||||
| } |  | ||||||
| CONSTANCE_BACKEND = 'constance.backends.database.DatabaseBackend' |  | ||||||
| CONSTANCE_SUPERUSER_ONLY = True |  | ||||||
|   | |||||||
| @@ -5,7 +5,6 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
| <!DOCTYPE html> | <!DOCTYPE html> | ||||||
| {% get_current_language as LANGUAGE_CODE %}{% get_current_language_bidi as LANGUAGE_BIDI %} | {% get_current_language as LANGUAGE_CODE %}{% get_current_language_bidi as LANGUAGE_BIDI %} | ||||||
| <html lang="{{ LANGUAGE_CODE|default:"en" }}" {% if LANGUAGE_BIDI %}dir="rtl"{% endif %} class="position-relative h-100"> | <html lang="{{ LANGUAGE_CODE|default:"en" }}" {% if LANGUAGE_BIDI %}dir="rtl"{% endif %} class="position-relative h-100"> | ||||||
| {% if not config.MAINTENANCE %} |  | ||||||
| <head> | <head> | ||||||
|     <meta charset="utf-8"> |     <meta charset="utf-8"> | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"> |     <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"> | ||||||
| @@ -139,12 +138,9 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                                 <a class="dropdown-item" href="{% url 'member:user_detail' pk=request.user.pk %}"> |                                 <a class="dropdown-item" href="{% url 'member:user_detail' pk=request.user.pk %}"> | ||||||
|                                     <i class="fa fa-user"></i> {% trans "My account" %} |                                     <i class="fa fa-user"></i> {% trans "My account" %} | ||||||
|                                 </a> |                                 </a> | ||||||
| 				<form method="post" action="{% url 'logout' %}"> |                                 <a class="dropdown-item" href="{% url 'logout' %}"> | ||||||
| 				    {% csrf_token %} |                                     <i class="fa fa-sign-out"></i> {% trans "Log out" %} | ||||||
| 				    <button class="dropdown-item" type=submit"> |                                 </a> | ||||||
| 					<i class="fa fa-sign-out"></i> {% trans "Log out" %} |  | ||||||
| 				    </button> |  | ||||||
|                                 </form> |  | ||||||
|                             </div> |                             </div> | ||||||
|                         </li> |                         </li> | ||||||
|                     {% else %} |                     {% else %} | ||||||
| @@ -192,11 +188,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                     {% endblocktrans %} |                     {% endblocktrans %} | ||||||
|                 </div> |                 </div> | ||||||
|             {% endif %} |             {% endif %} | ||||||
| 	    {% if config.BANNER_MESSAGE and user.is_authenticated %} |             {# TODO Add banners #} | ||||||
| 	    <div class="alert alert-{{ config.BANNER_TYPE }}"> |  | ||||||
| 	      {{ config.BANNER_MESSAGE }} |  | ||||||
| 	    </div> |  | ||||||
| 	    {% endif %} |  | ||||||
|         </div> |         </div> | ||||||
|         {% block content %} |         {% block content %} | ||||||
|             <p>Default content...</p> |             <p>Default content...</p> | ||||||
| @@ -218,10 +210,6 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                            class="text-muted">{% trans "Charte Info (FR)" %}</a> — |                            class="text-muted">{% trans "Charte Info (FR)" %}</a> — | ||||||
|                         <a href="https://note.crans.org/doc/faq/" |                         <a href="https://note.crans.org/doc/faq/" | ||||||
|                            class="text-muted">{% trans "FAQ (FR)" %}</a> — |                            class="text-muted">{% trans "FAQ (FR)" %}</a> — | ||||||
| 		   	<a href="https://bde.ens-cachan.fr" |  | ||||||
| 			   class="text-muted">{% trans "Managed by BDE" %}</a> — |  | ||||||
| 		   	<a href="https://crans.org" |  | ||||||
| 			   class="text-muted">{% trans "Hosted by Cr@ns" %}</a> — |  | ||||||
|                     </span> |                     </span> | ||||||
|                     {% csrf_token %} |                     {% csrf_token %} | ||||||
|                     <select title="language" name="language" |                     <select title="language" name="language" | ||||||
| @@ -258,15 +246,4 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|  |  | ||||||
| {% block extrajavascript %}{% endblock %} | {% block extrajavascript %}{% endblock %} | ||||||
| </body> | </body> | ||||||
| {% endif %} |  | ||||||
| {% if config.MAINTENANCE %} |  | ||||||
| <body> |  | ||||||
|   <div style="text-align:center"> |  | ||||||
|     <br /> |  | ||||||
|     {% trans "The note is not available for now" %}<br /><br /> |  | ||||||
|     {{ config.MAINTENANCE_MESSAGE }}<br /><br /> |  | ||||||
|     {% trans "Thank you for your understanding -- The Respos Info of BDE" %} |  | ||||||
|   </div> |  | ||||||
| </body> |  | ||||||
| {% endif %} |  | ||||||
| </html> | </html> | ||||||
|   | |||||||
| @@ -1,28 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| Copyright (C) by BDE ENS-Paris-Saclay |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n %} |  | ||||||
| {% block content %} |  | ||||||
| <div class="alert alert-success" role="alert">{% blocktrans %}<h3>Log In Successful</h3>You have successfully logged into the Central Authentication Service.<br/>For security reasons, please Log Out and Exit your web browser when you are done accessing services that require authentication!{% endblocktrans %}</div> |  | ||||||
| <div class="card bg-light mx-auto" style="max-width:30rem;"> |  | ||||||
|   <div class="card-body"> |  | ||||||
|     <form class="form-signin" method="get" action="logout"> |  | ||||||
|       <div class="checkbox"> |  | ||||||
| 	<label> |  | ||||||
| 	  <input type="checkbox" name="all" value="1">{% trans "Log me out from all my sessions" %} |  | ||||||
| 	</label> |  | ||||||
|       </div> |  | ||||||
|       {% if settings.CAS_FEDERATE and request.COOKIES.remember_provider %} |  | ||||||
|       <div class="checkbox"> |  | ||||||
| 	<label> |  | ||||||
| 	  <input type="checkbox" name="forget_provider" value="1">{% trans "Forget the identity provider" %} |  | ||||||
| 	</label> |  | ||||||
|       </div> |  | ||||||
|       {% endif %} |  | ||||||
|       <button class="btn btn-danger btn-block btn-lg" type="submit">{% trans "Logout" %}</button> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,42 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| Copyright (C) by BDE ENS-Paris-Saclay |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n %} |  | ||||||
|  |  | ||||||
| {% block ante_messages %} |  | ||||||
|   {% if auto_submit %}<noscript>{% endif %} |  | ||||||
|   <div class="card-header text-center"> |  | ||||||
|     <h2 class="form-signin-heading">{% trans "Please log in" %}</h2> |  | ||||||
|   </div> |  | ||||||
|   {% if auto_submit %}</noscript>{% endif %} |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
|   <div class="card bg-light mx-auto" style="max-width: 30rem;"> |  | ||||||
|     <div class="card-body"> |  | ||||||
| 	<form class="form-signin" method="post" id="login_form"{% if post_url %} action="{{post_url}}"{% endif %}> |  | ||||||
| 	  {% csrf_token %} |  | ||||||
| 	  {% include "cas_server/bs4/form.html" %} |  | ||||||
| 	  {% if auto_submit %}<noscript>{% endif %} |  | ||||||
| 	  <button class="btn btn-primary btn-block btn-lg" type="submit">{% trans "Login" %}</button> |  | ||||||
| 	  {% if auto_submit %}</noscript>{% endif %} |  | ||||||
| 	</div> |  | ||||||
|       </form> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
| {% block javascript_inline %} |  | ||||||
| jQuery(function( $ ){ |  | ||||||
|   $("#id_warn").click(function(e){ |  | ||||||
|     if($("#id_warn").is(':checked')){ |  | ||||||
|       createCookie("warn", "on", 10 * 365); |  | ||||||
|     } else { |  | ||||||
|       eraseCookie("warn"); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| {% if auto_submit %}document.getElementById('login_form').submit(); // SUBMIT FORM{% endif %} |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| Copyright (C) by BDE ENS-Paris-Saclay |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n static %} |  | ||||||
| {% block content %} |  | ||||||
|     <div class="alert alert-success" role="alert">{{ logout_msg }}</div> |  | ||||||
| {% endblock %} |  | ||||||
|    |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| Copyright (C) by BDE ENS-Paris-Saclay |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n static %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
|   <div class="card bg-light mx-auto" style="max-width: 30rem;"> |  | ||||||
|     <div class="card-body"> |  | ||||||
|       <form class="form-signin" method="post"> |  | ||||||
| 	{% csrf_token %} |  | ||||||
| 	{% include "cas_server/bs4/form.html" %} |  | ||||||
| 	<button class="btn btn-primary btn-block btn-lg" type="submit">{% trans "Connect to the service" %}</button> |  | ||||||
|       </form> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| {% endblock %} |  | ||||||
|    |  | ||||||
| @@ -1,21 +1,20 @@ | |||||||
| beautifulsoup4~=4.13.4 | beautifulsoup4~=4.12.3 | ||||||
| crispy-bootstrap4~=2025.6 | crispy-bootstrap4~=2023.1 | ||||||
| Django~=5.2.4 | Django~=4.2.9 | ||||||
| django-bootstrap-datepicker-plus~=5.0.5 | django-bootstrap-datepicker-plus~=5.0.5 | ||||||
| django-cas-server~=3.1.0 | #django-cas-server~=2.0.0 | ||||||
| django-colorfield~=0.14.0 | django-colorfield~=0.11.0 | ||||||
| django-constance~=4.3.2 | django-crispy-forms~=2.1.0 | ||||||
| django-crispy-forms~=2.4.0 | django-extensions>=3.2.3 | ||||||
| django-extensions>=4.1.0 | django-filter~=23.5 | ||||||
| django-filter~=25.1 |  | ||||||
| #django-htcpcp-tea~=0.8.1 | #django-htcpcp-tea~=0.8.1 | ||||||
| django-mailer~=2.3.2 | django-mailer~=2.3.1 | ||||||
| django-oauth-toolkit~=3.0.1 | django-oauth-toolkit~=2.3.0 | ||||||
| django-phonenumber-field~=8.1.0 | django-phonenumber-field~=7.3.0 | ||||||
| django-polymorphic~=3.1.0 | django-polymorphic~=3.1.0 | ||||||
| djangorestframework~=3.16.0 | djangorestframework~=3.14.0 | ||||||
| django-rest-polymorphic~=0.1.10 | django-rest-polymorphic~=0.1.10 | ||||||
| django-tables2~=2.7.5 | django-tables2~=2.7.0 | ||||||
| python-memcached~=1.62 | python-memcached~=1.62 | ||||||
| phonenumbers~=9.0.8 | phonenumbers~=8.13.28 | ||||||
| Pillow>=11.3.0 | Pillow>=10.2.0 | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | |||||||
| [tox] | [tox] | ||||||
| envlist = | envlist = | ||||||
|     # Ubuntu 22.04 Python |     # Ubuntu 22.04 Python | ||||||
|     py310-django52 |     py310-django42 | ||||||
|  |  | ||||||
|     # Debian Bookworm Python |     # Debian Bookworm Python | ||||||
|     py311-django52 |     py311-django42 | ||||||
|  |  | ||||||
|     # Ubuntu 24.04 Python |     # Ubuntu 24.04 Python | ||||||
|     py312-django52 |     py312-django42 | ||||||
|  |  | ||||||
|     linters |     linters | ||||||
| skipsdist = True | skipsdist = True | ||||||
| @@ -32,7 +32,8 @@ deps = | |||||||
|     pep8-naming |     pep8-naming | ||||||
|     pyflakes |     pyflakes | ||||||
| commands = | commands = | ||||||
|     flake8 apps --extend-exclude apps/scripts |     flake8 apps --extend-exclude apps/scripts,apps/wrapped/management/commands | ||||||
|  |     flake8 apps/wrapped/management/commands --extend-ignore=C901 | ||||||
|  |  | ||||||
| [flake8] | [flake8] | ||||||
| ignore = W503, I100, I101, B019 | ignore = W503, I100, I101, B019 | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user