mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-25 22:23:09 +02:00 
			
		
		
		
	Compare commits
	
		
			30 Commits
		
	
	
		
			1347c7f174
			...
			6dc25e2645
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 6dc25e2645 | ||
|  | a94c937c6a | ||
|  | 0a261e6ad5 | ||
|  | ab9329f62b | ||
|  | b97b79e2ea | ||
|  | 483ea26f02 | ||
|  | 695ce63e08 | ||
|  | 79f50c27f1 | ||
|  | 5989721bc9 | ||
|  | bcc3e7cc53 | ||
|  | 608804db30 | ||
|  | 82a06c29dd | ||
|  | cf9d208586 | ||
|  | 432f50e49a | ||
|  | 883589e08c | ||
|  | c36f8c25a2 | ||
|  | 8783a63d7f | ||
|  | 4cc43fe4b6 | ||
|  | b7c0986a5f | ||
|  | 85ea43a7cf | ||
|  | f54dd30482 | ||
|  | 7eafe33945 | ||
|  | 6edef619aa | ||
|  | 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-django42: | py310-django52: | ||||||
|   stage: test |   stage: test | ||||||
|   image: ubuntu:22.04 |   image: ubuntu:22.04 | ||||||
|   before_script: |   before_script: | ||||||
| @@ -22,10 +22,10 @@ py310-django42: | |||||||
|         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-django42 |   script: tox -e py310-django52 | ||||||
|  |  | ||||||
| # Debian Bookworm | # Debian Bookworm | ||||||
| py311-django42: | py311-django52: | ||||||
|   stage: test |   stage: test | ||||||
|   image: debian:bookworm |   image: debian:bookworm | ||||||
|   before_script: |   before_script: | ||||||
| @@ -37,7 +37,7 @@ py311-django42: | |||||||
|         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-django42 |   script: tox -e py311-django52 | ||||||
|  |  | ||||||
| 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; | ||||||
|   | |||||||
| @@ -145,7 +145,7 @@ class AddIngredientForms(forms.ModelForm): | |||||||
|             polymorphic_ctype__model="transformedfood", |             polymorphic_ctype__model="transformedfood", | ||||||
|             is_ready=False, |             is_ready=False, | ||||||
|             end_of_life='', |             end_of_life='', | ||||||
|         ).filter(PermissionBackend.filter_queryset(get_current_request(), TransformedFood, "change")).exclude(pk=pk) |         ).filter(PermissionBackend.filter_queryset(get_current_request(), Food, "change")).exclude(pk=pk) | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         model = TransformedFood |         model = TransformedFood | ||||||
|   | |||||||
| @@ -12,6 +12,9 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|   </h3> |   </h3> | ||||||
|   <div class="card-body"> |   <div class="card-body"> | ||||||
|     <ul> |     <ul> | ||||||
|  |       {% if QR_code %} | ||||||
|  |       <li> {{QR_code}} </li> | ||||||
|  |       {% endif %} | ||||||
|       {% for field, value in fields %} |       {% for field, value in fields %} | ||||||
|       <li> {{ field }} : {{ value }}</li> |       <li> {{ field }} : {{ value }}</li> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|   | |||||||
| @@ -7,7 +7,52 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
| {% load i18n %} | {% load i18n %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {{ block.super }} | <div class="card bg-light"> | ||||||
|  |   <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"> | ||||||
| @@ -68,4 +113,20 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|   {% endfor %} |   {% endfor %} | ||||||
|   {% endif %} |   {% endif %} | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <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 %} | {% endblock %} | ||||||
| @@ -18,4 +18,5 @@ 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,6 +10,7 @@ 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 _ | ||||||
| @@ -454,6 +455,8 @@ class FoodDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): | |||||||
|         context["fields"] = [( |         context["fields"] = [( | ||||||
|             Food._meta.get_field(field).verbose_name.capitalize(), |             Food._meta.get_field(field).verbose_name.capitalize(), | ||||||
|             value) for field, value in fields.items()] |             value) for field, value in fields.items()] | ||||||
|  |         if self.object.QR_code.exists(): | ||||||
|  |             context["QR_code"] = self.object.QR_code.first() | ||||||
|         context["meals"] = self.object.transformed_ingredient_inv.all() |         context["meals"] = self.object.transformed_ingredient_inv.all() | ||||||
|         context["update"] = PermissionBackend.check_perm(self.request, "food.change_food") |         context["update"] = PermissionBackend.check_perm(self.request, "food.change_food") | ||||||
|         context["add_ingredient"] = (self.object.end_of_life == '' and PermissionBackend.check_perm(self.request, "food.change_transformedfood")) |         context["add_ingredient"] = (self.object.end_of_life == '' and PermissionBackend.check_perm(self.request, "food.change_transformedfood")) | ||||||
| @@ -507,3 +510,14 @@ 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.get(reverse("logout")) |         response = self.client.post(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) |     router.register(path + '/consumer', ConsumerViewSet, basename='alias2') | ||||||
|  |  | ||||||
|     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,7 +18,18 @@ 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): |     def get_all_scopes(self, **kwargs): | ||||||
|  |         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,6 +13,7 @@ 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,14 +164,24 @@ class ScopesView(LoginRequiredMixin, TemplateView): | |||||||
|         from oauth2_provider.models import Application |         from oauth2_provider.models import Application | ||||||
|         from .scopes import PermissionScopes |         from .scopes import PermissionScopes | ||||||
|  |  | ||||||
|         scopes = PermissionScopes() |         oidc = False | ||||||
|         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 = scopes.get_available_scopes(app) |             available_scopes = PermissionScopes().get_available_scopes(app) | ||||||
|             context["scopes"][app] = OrderedDict() |             context["scopes"][app] = OrderedDict() | ||||||
|             items = [(k, v) for (k, v) in all_scopes.items() if k in available_scopes] |             all_scopes = PermissionScopes().get_all_scopes(scopes=available_scopes) | ||||||
|             # items.sort(key=lambda x: (int(x[0].split("_")[1]), int(x[0].split("_")[0]))) |             scopes = {} | ||||||
|  |             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 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -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-06-27 19:15+0200\n" | "POT-Creation-Date: 2025-07-11 16:10+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:78 | #: note_kfet/templates/base.html:79 | ||||||
| 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:96 | #: apps/activity/views.py:71 note_kfet/templates/base.html:97 | ||||||
| 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:34 | #: apps/food/models.py:107 apps/food/views.py:35 | ||||||
| #: note_kfet/templates/base.html:72 | #: note_kfet/templates/base.html:72 | ||||||
| msgid "Food" | msgid "Food" | ||||||
| msgstr "Bouffe" | msgstr "Bouffe" | ||||||
| @@ -657,61 +657,75 @@ 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:19 | #: apps/food/templates/food/food_detail.html:22 | ||||||
| msgid "Contained in" | msgid "Contained in" | ||||||
| msgstr "Contenu dans" | msgstr "Contenu dans" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:26 | #: apps/food/templates/food/food_detail.html:29 | ||||||
| msgid "Contain" | msgid "Contain" | ||||||
| msgstr "Contient" | msgstr "Contient" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:35 | #: apps/food/templates/food/food_detail.html:38 | ||||||
| msgid "Update" | msgid "Update" | ||||||
| msgstr "Modifier" | msgstr "Modifier" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_detail.html:40 | #: apps/food/templates/food/food_detail.html:43 | ||||||
| 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:45 | #: apps/food/templates/food/food_detail.html:48 | ||||||
| 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:49 | #: apps/food/templates/food/food_detail.html:52 | ||||||
| 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:14 | #: apps/food/templates/food/food_list.html:32 | ||||||
|  | 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:19 | #: apps/food/templates/food/food_list.html:63 | ||||||
| msgid "New meal" | msgid "New meal" | ||||||
| msgstr "Nouveau plat" | msgstr "Nouveau plat" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:28 | #: apps/food/templates/food/food_list.html:72 | ||||||
| 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:35 | #: apps/food/templates/food/food_list.html:79 | ||||||
| msgid "Free food" | msgid "Free food" | ||||||
| msgstr "Open" | msgstr "Open" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:42 | #: apps/food/templates/food/food_list.html:86 | ||||||
| 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:50 | #: apps/food/templates/food/food_list.html:94 | ||||||
| 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:56 | #: apps/food/templates/food/food_list.html:100 | ||||||
| msgid "Food of club" | msgid "Food of club" | ||||||
| msgstr "Bouffe du club" | msgstr "Bouffe du club" | ||||||
|  |  | ||||||
| #: apps/food/templates/food/food_list.html:63 | #: apps/food/templates/food/food_list.html:107 | ||||||
| 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" | ||||||
|  |  | ||||||
| @@ -785,49 +799,49 @@ msgstr "semaines" | |||||||
| msgid "and" | msgid "and" | ||||||
| msgstr "et" | msgstr "et" | ||||||
|  |  | ||||||
| #: apps/food/views.py:118 | #: apps/food/views.py:120 | ||||||
| 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:167 | #: apps/food/views.py:169 | ||||||
| msgid "Add an aliment" | msgid "Add an aliment" | ||||||
| msgstr "Ajouter un nouvel aliment" | msgstr "Ajouter un nouvel aliment" | ||||||
|  |  | ||||||
| #: apps/food/views.py:235 | #: apps/food/views.py:228 | ||||||
| msgid "Add a meal" | msgid "Add a meal" | ||||||
| msgstr "Ajouter un plat" | msgstr "Ajouter un plat" | ||||||
|  |  | ||||||
| #: apps/food/views.py:275 | #: apps/food/views.py:259 | ||||||
| msgid "Manage ingredients of:" | msgid "Manage ingredients of:" | ||||||
| msgstr "Gestion des ingrédienrs de :" | msgstr "Gestion des ingrédienrs de :" | ||||||
|  |  | ||||||
| #: apps/food/views.py:289 apps/food/views.py:297 | #: apps/food/views.py:273 apps/food/views.py:281 | ||||||
| #, 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:344 | #: apps/food/views.py:320 | ||||||
| msgid "Add the ingredient:" | msgid "Add the ingredient:" | ||||||
| msgstr "Ajouter l'ingrédient" | msgstr "Ajouter l'ingrédient" | ||||||
|  |  | ||||||
| #: apps/food/views.py:370 | #: apps/food/views.py:346 | ||||||
| #, 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:389 | #: apps/food/views.py:365 | ||||||
| msgid "Update an aliment" | msgid "Update an aliment" | ||||||
| msgstr "Modifier un aliment" | msgstr "Modifier un aliment" | ||||||
|  |  | ||||||
| #: apps/food/views.py:437 | #: apps/food/views.py:413 | ||||||
| msgid "Details of:" | msgid "Details of:" | ||||||
| msgstr "Détails de :" | msgstr "Détails de :" | ||||||
|  |  | ||||||
| #: apps/food/views.py:447 apps/treasury/tables.py:149 | #: apps/food/views.py:423 apps/treasury/tables.py:149 | ||||||
| msgid "Yes" | msgid "Yes" | ||||||
| msgstr "Oui" | msgstr "Oui" | ||||||
|  |  | ||||||
| #: apps/food/views.py:449 apps/member/models.py:99 apps/treasury/tables.py:149 | #: apps/food/views.py:425 apps/member/models.py:99 apps/treasury/tables.py:149 | ||||||
| msgid "No" | msgid "No" | ||||||
| msgstr "Non" | msgstr "Non" | ||||||
|  |  | ||||||
| @@ -1962,8 +1976,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:1097 | #: apps/note/models/transactions.py:363 apps/wei/views.py:1105 | ||||||
| #: apps/wei/views.py:1101 | #: apps/wei/views.py:1109 | ||||||
| msgid "This field is required." | msgid "This field is required." | ||||||
| msgstr "Ce champ est requis." | msgstr "Ce champ est requis." | ||||||
|  |  | ||||||
| @@ -2065,6 +2079,8 @@ 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" | ||||||
|  |  | ||||||
| @@ -2176,7 +2192,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:66 | #: apps/note/views.py:156 note_kfet/templates/base.html:67 | ||||||
| msgid "Consumptions" | msgid "Consumptions" | ||||||
| msgstr "Consommations" | msgstr "Consommations" | ||||||
|  |  | ||||||
| @@ -2269,7 +2285,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:73 | #: apps/permission/signals.py:75 | ||||||
| #, 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 " | ||||||
| @@ -2278,7 +2294,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:83 apps/permission/views.py:104 | #: apps/permission/signals.py:85 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}." | ||||||
| @@ -2287,7 +2303,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:112 | #: apps/permission/signals.py:114 | ||||||
| #, 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}." | ||||||
| @@ -2375,7 +2391,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:120 | #: apps/permission/views.py:111 note_kfet/templates/base.html:121 | ||||||
| msgid "Rights" | msgid "Rights" | ||||||
| msgstr "Droits" | msgstr "Droits" | ||||||
|  |  | ||||||
| @@ -2580,7 +2596,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:102 | #: apps/treasury/apps.py:12 note_kfet/templates/base.html:103 | ||||||
| msgid "Treasury" | msgid "Treasury" | ||||||
| msgstr "Trésorerie" | msgstr "Trésorerie" | ||||||
|  |  | ||||||
| @@ -2996,7 +3012,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:108 | #: note_kfet/templates/base.html:109 | ||||||
| msgid "WEI" | msgid "WEI" | ||||||
| msgstr "WEI" | msgstr "WEI" | ||||||
|  |  | ||||||
| @@ -3041,7 +3057,7 @@ 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:147 | #: apps/wei/forms/registration.py:160 | ||||||
| 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." | ||||||
|  |  | ||||||
| @@ -3120,7 +3136,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:984 | #: apps/wei/models.py:202 apps/wei/views.py:992 | ||||||
| msgid "Caution check given" | msgid "Caution check given" | ||||||
| msgstr "Chèque de caution donné" | msgstr "Chèque de caution donné" | ||||||
|  |  | ||||||
| @@ -3352,7 +3368,6 @@ msgid "View club" | |||||||
| msgstr "Voir le club" | msgstr "Voir le club" | ||||||
|  |  | ||||||
| #: apps/wei/templates/wei/bus_detail.html:26 | #: apps/wei/templates/wei/bus_detail.html:26 | ||||||
| #| msgid "survey information" |  | ||||||
| msgid "Edit information" | msgid "Edit information" | ||||||
| msgstr "Modifier les informations" | msgstr "Modifier les informations" | ||||||
|  |  | ||||||
| @@ -3373,8 +3388,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:1159 | #: apps/wei/templates/wei/survey_end.html:11 apps/wei/views.py:1167 | ||||||
| #: apps/wei/views.py:1214 apps/wei/views.py:1261 | #: apps/wei/views.py:1222 apps/wei/views.py:1269 | ||||||
| msgid "Survey WEI" | msgid "Survey WEI" | ||||||
| msgstr "Questionnaire WEI" | msgstr "Questionnaire WEI" | ||||||
|  |  | ||||||
| @@ -3654,51 +3669,51 @@ msgstr "" | |||||||
| msgid "Update WEI Registration" | msgid "Update WEI Registration" | ||||||
| msgstr "Modifier l'inscription WEI" | msgstr "Modifier l'inscription WEI" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:810 | #: apps/wei/views.py:811 | ||||||
| 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:819 | #: apps/wei/views.py:820 | ||||||
| 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:825 | #: apps/wei/views.py:826 | ||||||
| #, 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:870 | #: apps/wei/views.py:871 | ||||||
| msgid "Delete WEI registration" | msgid "Delete WEI registration" | ||||||
| msgstr "Supprimer l'inscription WEI" | msgstr "Supprimer l'inscription WEI" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:881 | #: apps/wei/views.py:882 | ||||||
| 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:899 | #: apps/wei/views.py:900 | ||||||
| msgid "Validate WEI registration" | msgid "Validate WEI registration" | ||||||
| msgstr "Valider l'inscription WEI" | msgstr "Valider l'inscription WEI" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:985 | #: apps/wei/views.py:993 | ||||||
| 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:991 | #: apps/wei/views.py:999 | ||||||
| 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:992 | #: apps/wei/views.py:1000 | ||||||
| #, 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:1087 | #: apps/wei/views.py:1095 | ||||||
| #, 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. " | ||||||
| @@ -3708,21 +3723,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:1140 | #: apps/wei/views.py:1148 | ||||||
| #, 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:1354 | #: apps/wei/views.py:1362 | ||||||
| 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:1380 | #: apps/wei/views.py:1388 | ||||||
| msgid "Attribute bus" | msgid "Attribute bus" | ||||||
| msgstr "Attribuer un bus" | msgstr "Attribuer un bus" | ||||||
|  |  | ||||||
| #: apps/wei/views.py:1420 | #: apps/wei/views.py:1428 | ||||||
| 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." | ||||||
| @@ -3746,13 +3761,13 @@ msgstr "bde" | |||||||
|  |  | ||||||
| #: apps/wrapped/models.py:65 | #: apps/wrapped/models.py:65 | ||||||
| msgid "data json" | msgid "data json" | ||||||
| msgstr "donnée json" | msgstr "données 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:114 | #: apps/wrapped/models.py:70 note_kfet/templates/base.html:115 | ||||||
| msgid "Wrapped" | msgid "Wrapped" | ||||||
| msgstr "Wrapped" | msgstr "Wrapped" | ||||||
|  |  | ||||||
| @@ -3785,7 +3800,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:14 | #: note_kfet/templates/base.html:15 | ||||||
| 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." | ||||||
|  |  | ||||||
| @@ -3888,7 +3903,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'oublies pas de demander la permission des personnes apparaissant dans un " | "N'oublie 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 | ||||||
| @@ -3907,19 +3922,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:177 | #: note_kfet/settings/base.py:180 | ||||||
| msgid "German" | msgid "German" | ||||||
| msgstr "Allemand" | msgstr "Allemand" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:178 | #: note_kfet/settings/base.py:181 | ||||||
| msgid "English" | msgid "English" | ||||||
| msgstr "Anglais" | msgstr "Anglais" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:179 | #: note_kfet/settings/base.py:182 | ||||||
| msgid "Spanish" | msgid "Spanish" | ||||||
| msgstr "Espagnol" | msgstr "Espagnol" | ||||||
|  |  | ||||||
| #: note_kfet/settings/base.py:180 | #: note_kfet/settings/base.py:183 | ||||||
| msgid "French" | msgid "French" | ||||||
| msgstr "Français" | msgstr "Français" | ||||||
|  |  | ||||||
| @@ -3980,34 +3995,34 @@ msgstr "" | |||||||
| msgid "Reset" | msgid "Reset" | ||||||
| msgstr "Réinitialiser" | msgstr "Réinitialiser" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:84 | #: note_kfet/templates/base.html:85 | ||||||
| msgid "Users" | msgid "Users" | ||||||
| msgstr "Utilisateur·rices" | msgstr "Utilisateur·rices" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:90 | #: note_kfet/templates/base.html:91 | ||||||
| msgid "Clubs" | msgid "Clubs" | ||||||
| msgstr "Clubs" | msgstr "Clubs" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:125 | #: note_kfet/templates/base.html:126 | ||||||
| msgid "Admin" | msgid "Admin" | ||||||
| msgstr "Admin" | msgstr "Admin" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:139 | #: note_kfet/templates/base.html:140 | ||||||
| msgid "My account" | msgid "My account" | ||||||
| msgstr "Mon compte" | msgstr "Mon compte" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:142 | #: note_kfet/templates/base.html:145 | ||||||
| msgid "Log out" | msgid "Log out" | ||||||
| msgstr "Se déconnecter" | msgstr "Se déconnecter" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:150 | #: note_kfet/templates/base.html:154 | ||||||
| #: 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:157 | #: note_kfet/templates/base.html:161 | ||||||
| #: 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 | ||||||
| @@ -4015,7 +4030,7 @@ msgstr "Inscription" | |||||||
| msgid "Log in" | msgid "Log in" | ||||||
| msgstr "Se connecter" | msgstr "Se connecter" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:171 | #: note_kfet/templates/base.html:175 | ||||||
| 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." | ||||||
| @@ -4023,7 +4038,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:177 | #: note_kfet/templates/base.html:181 | ||||||
| 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." | ||||||
| @@ -4031,7 +4046,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:183 | #: note_kfet/templates/base.html:187 | ||||||
| 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 " | ||||||
| @@ -4045,22 +4060,38 @@ 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:206 | #: note_kfet/templates/base.html:214 | ||||||
| msgid "Contact us" | msgid "Contact us" | ||||||
| msgstr "Nous contacter" | msgstr "Nous contacter" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:208 | #: note_kfet/templates/base.html:216 | ||||||
| msgid "Technical Support" | msgid "Technical Support" | ||||||
| msgstr "Support technique" | msgstr "Support technique" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:210 | #: note_kfet/templates/base.html:218 | ||||||
| msgid "Charte Info (FR)" | msgid "Charte Info (FR)" | ||||||
| msgstr "Charte Info (FR)" | msgstr "Charte Info (FR)" | ||||||
|  |  | ||||||
| #: note_kfet/templates/base.html:212 | #: note_kfet/templates/base.html:220 | ||||||
| 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..." | ||||||
| @@ -4069,6 +4100,41 @@ 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" | ||||||
| @@ -4289,10 +4355,86 @@ 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, python-format | #, fuzzy | ||||||
| #~| msgid "Creation date" | #~| msgid "QR-code" | ||||||
| #~ msgid "Deposit %(name)s" | #~ msgid "Go to QR-code" | ||||||
| #~ msgstr "Caution %(name)s" | #~ msgstr "QR-code" | ||||||
|  |  | ||||||
|  | #, 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." | ||||||
| @@ -4706,7 +4848,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é" | ||||||
|   | |||||||
| @@ -28,4 +28,5 @@ MAILTO=notekfet2020@lists.crans.org | |||||||
|  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,3 +56,8 @@ 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,7 +39,9 @@ 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', | ||||||
| @@ -111,6 +113,7 @@ 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', | ||||||
| @@ -307,6 +310,30 @@ 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,6 +5,7 @@ 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"> | ||||||
| @@ -138,9 +139,12 @@ 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> | ||||||
|                                 <a class="dropdown-item" href="{% url 'logout' %}"> | 				<form method="post" action="{% url 'logout' %}"> | ||||||
|  | 				    {% csrf_token %} | ||||||
|  | 				    <button class="dropdown-item" type=submit"> | ||||||
| 					<i class="fa fa-sign-out"></i> {% trans "Log out" %} | 					<i class="fa fa-sign-out"></i> {% trans "Log out" %} | ||||||
|                                 </a> | 				    </button> | ||||||
|  |                                 </form> | ||||||
|                             </div> |                             </div> | ||||||
|                         </li> |                         </li> | ||||||
|                     {% else %} |                     {% else %} | ||||||
| @@ -188,7 +192,11 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                     {% endblocktrans %} |                     {% endblocktrans %} | ||||||
|                 </div> |                 </div> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|             {# TODO Add banners #} | 	    {% if config.BANNER_MESSAGE and user.is_authenticated %} | ||||||
|  | 	    <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> | ||||||
| @@ -210,6 +218,10 @@ 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" | ||||||
| @@ -246,4 +258,15 @@ 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> | ||||||
|   | |||||||
							
								
								
									
										28
									
								
								note_kfet/templates/cas/logged.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								note_kfet/templates/cas/logged.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | {% 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 %} | ||||||
							
								
								
									
										42
									
								
								note_kfet/templates/cas/login.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								note_kfet/templates/cas/login.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | {% 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 %} | ||||||
							
								
								
									
										10
									
								
								note_kfet/templates/cas/logout.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								note_kfet/templates/cas/logout.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | {% 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 %} | ||||||
|  |    | ||||||
							
								
								
									
										19
									
								
								note_kfet/templates/cas/warn.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								note_kfet/templates/cas/warn.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | {% 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,20 +1,21 @@ | |||||||
| beautifulsoup4~=4.12.3 | beautifulsoup4~=4.13.4 | ||||||
| crispy-bootstrap4~=2023.1 | crispy-bootstrap4~=2025.6 | ||||||
| Django~=4.2.9 | Django~=5.2.4 | ||||||
| django-bootstrap-datepicker-plus~=5.0.5 | django-bootstrap-datepicker-plus~=5.0.5 | ||||||
| #django-cas-server~=2.0.0 | django-cas-server~=3.1.0 | ||||||
| django-colorfield~=0.11.0 | django-colorfield~=0.14.0 | ||||||
| django-crispy-forms~=2.1.0 | django-constance~=4.3.2 | ||||||
| django-extensions>=3.2.3 | django-crispy-forms~=2.4.0 | ||||||
| django-filter~=23.5 | django-extensions>=4.1.0 | ||||||
|  | django-filter~=25.1 | ||||||
| #django-htcpcp-tea~=0.8.1 | #django-htcpcp-tea~=0.8.1 | ||||||
| django-mailer~=2.3.1 | django-mailer~=2.3.2 | ||||||
| django-oauth-toolkit~=2.3.0 | django-oauth-toolkit~=3.0.1 | ||||||
| django-phonenumber-field~=7.3.0 | django-phonenumber-field~=8.1.0 | ||||||
| django-polymorphic~=3.1.0 | django-polymorphic~=3.1.0 | ||||||
| djangorestframework~=3.14.0 | djangorestframework~=3.16.0 | ||||||
| django-rest-polymorphic~=0.1.10 | django-rest-polymorphic~=0.1.10 | ||||||
| django-tables2~=2.7.0 | django-tables2~=2.7.5 | ||||||
| python-memcached~=1.62 | python-memcached~=1.62 | ||||||
| phonenumbers~=8.13.28 | phonenumbers~=9.0.8 | ||||||
| Pillow>=10.2.0 | Pillow>=11.3.0 | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | |||||||
| [tox] | [tox] | ||||||
| envlist = | envlist = | ||||||
|     # Ubuntu 22.04 Python |     # Ubuntu 22.04 Python | ||||||
|     py310-django42 |     py310-django52 | ||||||
|  |  | ||||||
|     # Debian Bookworm Python |     # Debian Bookworm Python | ||||||
|     py311-django42 |     py311-django52 | ||||||
|  |  | ||||||
|     # Ubuntu 24.04 Python |     # Ubuntu 24.04 Python | ||||||
|     py312-django42 |     py312-django52 | ||||||
|  |  | ||||||
|     linters |     linters | ||||||
| skipsdist = True | skipsdist = True | ||||||
| @@ -32,8 +32,7 @@ deps = | |||||||
|     pep8-naming |     pep8-naming | ||||||
|     pyflakes |     pyflakes | ||||||
| commands = | commands = | ||||||
|     flake8 apps --extend-exclude apps/scripts,apps/wrapped/management/commands |     flake8 apps --extend-exclude apps/scripts | ||||||
|     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