mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-24 22:03:06 +02:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			14474d309b
			...
			migration-
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | bb9f58e497 | 
| @@ -7,25 +7,10 @@ stages: | |||||||
| variables: | variables: | ||||||
|   GIT_SUBMODULE_STRATEGY: recursive |   GIT_SUBMODULE_STRATEGY: recursive | ||||||
|  |  | ||||||
| # Debian Buster | # Ubuntu 22.04 | ||||||
| #  py37-django22: | py310-django50: | ||||||
| #   stage: test |  | ||||||
| #   image: debian:buster-backports |  | ||||||
| #   before_script: |  | ||||||
| #     - > |  | ||||||
| #         apt-get update && |  | ||||||
| #         apt-get install --no-install-recommends -t buster-backports -y |  | ||||||
| #         python3-django python3-django-crispy-forms |  | ||||||
| #         python3-django-extensions python3-django-filters python3-django-polymorphic |  | ||||||
| #         python3-djangorestframework python3-django-oauth-toolkit python3-psycopg2 python3-pil |  | ||||||
| #         python3-babel python3-lockfile python3-pip python3-phonenumbers python3-memcache |  | ||||||
| #         python3-bs4 python3-setuptools tox texlive-xetex |  | ||||||
| #   script: tox -e py37-django22 |  | ||||||
|  |  | ||||||
| # Ubuntu 20.04 |  | ||||||
| py38-django22: |  | ||||||
|   stage: test |   stage: test | ||||||
|   image: ubuntu:20.04 |   image: ubuntu:22.04 | ||||||
|   before_script: |   before_script: | ||||||
|     # Fix tzdata prompt |     # Fix tzdata prompt | ||||||
|     - ln -sf /usr/share/zoneinfo/Europe/Paris /etc/localtime && echo Europe/Paris > /etc/timezone |     - ln -sf /usr/share/zoneinfo/Europe/Paris /etc/localtime && echo Europe/Paris > /etc/timezone | ||||||
| @@ -37,12 +22,12 @@ py38-django22: | |||||||
|         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 py38-django22 |   script: tox -e py310-django50 | ||||||
|  |  | ||||||
| # Debian Bullseye | # Debian Bookworm | ||||||
| py39-django22: | py311-django50: | ||||||
|   stage: test |   stage: test | ||||||
|   image: debian:bullseye |   image: debian:bookworm | ||||||
|   before_script: |   before_script: | ||||||
|     - > |     - > | ||||||
|         apt-get update && |         apt-get update && | ||||||
| @@ -52,7 +37,7 @@ py39-django22: | |||||||
|         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 py39-django22 |   script: tox -e py311-django50 | ||||||
|  |  | ||||||
| linters: | linters: | ||||||
|   stage: quality-assurance |   stage: quality-assurance | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
|  | from bootstrap_datepicker_plus.widgets import DateTimePickerInput | ||||||
|  |  | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| from random import shuffle | from random import shuffle | ||||||
|  |  | ||||||
| @@ -10,7 +12,7 @@ from django.utils import timezone | |||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from member.models import Club | from member.models import Club | ||||||
| from note.models import Note, NoteUser | from note.models import Note, NoteUser | ||||||
| from note_kfet.inputs import Autocomplete, DateTimePickerInput | from note_kfet.inputs import Autocomplete | ||||||
| from note_kfet.middlewares import get_current_request | from note_kfet.middlewares import get_current_request | ||||||
| from permission.backends import PermissionBackend | from permission.backends import PermissionBackend | ||||||
|  |  | ||||||
|   | |||||||
| @@ -17,8 +17,7 @@ from django.utils.translation import gettext_lazy as _ | |||||||
| from django.views import View | from django.views import View | ||||||
| from django.views.decorators.cache import cache_page | from django.views.decorators.cache import cache_page | ||||||
| from django.views.generic import DetailView, TemplateView, UpdateView | from django.views.generic import DetailView, TemplateView, UpdateView | ||||||
| from django.views.generic.list import ListView | from django_tables2.views import SingleTableView | ||||||
| from django_tables2.views import MultiTableMixin |  | ||||||
| from note.models import Alias, NoteSpecial, NoteUser | from note.models import Alias, NoteSpecial, NoteUser | ||||||
| from permission.backends import PermissionBackend | from permission.backends import PermissionBackend | ||||||
| from permission.views import ProtectQuerysetMixin, ProtectedCreateView | from permission.views import ProtectQuerysetMixin, ProtectedCreateView | ||||||
| @@ -58,40 +57,27 @@ class ActivityCreateView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|         return reverse_lazy('activity:activity_detail', kwargs={"pk": self.object.pk}) |         return reverse_lazy('activity:activity_detail', kwargs={"pk": self.object.pk}) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, ListView): | class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView): | ||||||
|     """ |     """ | ||||||
|     Displays all Activities, and classify if they are on-going or upcoming ones. |     Displays all Activities, and classify if they are on-going or upcoming ones. | ||||||
|     """ |     """ | ||||||
|     model = Activity |     model = Activity | ||||||
|     tables = [ActivityTable, ActivityTable] |     table_class = ActivityTable | ||||||
|  |     ordering = ('-date_start',) | ||||||
|     extra_context = {"title": _("Activities")} |     extra_context = {"title": _("Activities")} | ||||||
|  |  | ||||||
|     def get_queryset(self, **kwargs): |     def get_queryset(self, **kwargs): | ||||||
|         return super().get_queryset(**kwargs).distinct() |         return super().get_queryset(**kwargs).distinct() | ||||||
|  |  | ||||||
|     def get_tables(self): |  | ||||||
|         tables = super().get_tables() |  | ||||||
|  |  | ||||||
|         tables[0].prefix = "all-" |  | ||||||
|         tables[1].prefix = "upcoming-" |  | ||||||
|         return tables |  | ||||||
|  |  | ||||||
|     def get_tables_data(self): |  | ||||||
|         # first table = all activities, second table = upcoming |  | ||||||
|         return [ |  | ||||||
|             self.get_queryset().order_by("-date_start"), |  | ||||||
|             Activity.objects.filter(date_end__gt=timezone.now()) |  | ||||||
|                             .filter(PermissionBackend.filter_queryset(self.request, Activity, "view")) |  | ||||||
|                             .distinct() |  | ||||||
|                             .order_by("date_start") |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         context = super().get_context_data(**kwargs) |         context = super().get_context_data(**kwargs) | ||||||
|  |  | ||||||
|         tables = context["tables"] |         upcoming_activities = Activity.objects.filter(date_end__gt=timezone.now()) | ||||||
|         for name, table in zip(["table", "upcoming"], tables): |         context['upcoming'] = ActivityTable( | ||||||
|             context[name] = table |             data=upcoming_activities.filter(PermissionBackend.filter_queryset(self.request, Activity, "view")), | ||||||
|  |             prefix='upcoming-', | ||||||
|  |             order_by='date_start', | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         started_activities = self.get_queryset().filter(open=True, valid=True).distinct().all() |         started_activities = self.get_queryset().filter(open=True, valid=True).distinct().all() | ||||||
|         context["started_activities"] = started_activities |         context["started_activities"] = started_activities | ||||||
| @@ -330,8 +316,8 @@ X-WR-CALNAME:Kfet Calendar | |||||||
| NAME:Kfet Calendar | NAME:Kfet Calendar | ||||||
| CALSCALE:GREGORIAN | CALSCALE:GREGORIAN | ||||||
| BEGIN:VTIMEZONE | BEGIN:VTIMEZONE | ||||||
| TZID:Europe/Paris | TZID:Europe/Berlin | ||||||
| X-LIC-LOCATION:Europe/Paris | X-LIC-LOCATION:Europe/Berlin | ||||||
| BEGIN:DAYLIGHT | BEGIN:DAYLIGHT | ||||||
| TZOFFSETFROM:+0100 | TZOFFSETFROM:+0100 | ||||||
| TZOFFSETTO:+0200 | TZOFFSETTO:+0200 | ||||||
| @@ -353,10 +339,10 @@ END:VTIMEZONE | |||||||
| DTSTAMP:{"{:%Y%m%dT%H%M%S}".format(activity.date_start)}Z | DTSTAMP:{"{:%Y%m%dT%H%M%S}".format(activity.date_start)}Z | ||||||
| UID:{md5((activity.name + "$" + str(activity.id) + str(activity.date_start)).encode("UTF-8")).hexdigest()} | UID:{md5((activity.name + "$" + str(activity.id) + str(activity.date_start)).encode("UTF-8")).hexdigest()} | ||||||
| SUMMARY;CHARSET=UTF-8:{self.multilines(activity.name, 75, 22)} | SUMMARY;CHARSET=UTF-8:{self.multilines(activity.name, 75, 22)} | ||||||
| DTSTART:{"{:%Y%m%dT%H%M%S}Z".format(activity.date_start)} | DTSTART;TZID=Europe/Berlin:{"{:%Y%m%dT%H%M%S}".format(activity.date_start)} | ||||||
| DTEND:{"{:%Y%m%dT%H%M%S}Z".format(activity.date_end)} | DTEND;TZID=Europe/Berlin:{"{:%Y%m%dT%H%M%S}".format(activity.date_end)} | ||||||
| LOCATION:{self.multilines(activity.location, 75, 9) if activity.location else "Kfet"} | LOCATION:{self.multilines(activity.location, 75, 9) if activity.location else "Kfet"} | ||||||
| DESCRIPTION;CHARSET=UTF-8:""" + self.multilines(activity.description.replace("\n", "\\n"), 75, 26) + f""" | DESCRIPTION;CHARSET=UTF-8:""" + self.multilines(activity.description.replace("\n", "\\n"), 75, 26) + """ | ||||||
|  -- {activity.organizer.name} |  -- {activity.organizer.name} | ||||||
| END:VEVENT | END:VEVENT | ||||||
| """ | """ | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.conf.urls import url, include | from django.urls import include, re_path | ||||||
| from rest_framework import routers | from rest_framework import routers | ||||||
|  |  | ||||||
| from .views import UserInformationView | from .views import UserInformationView | ||||||
| @@ -47,7 +47,7 @@ app_name = 'api' | |||||||
| # Wire up our API using automatic URL routing. | # Wire up our API using automatic URL routing. | ||||||
| # Additionally, we include login URLs for the browsable API. | # Additionally, we include login URLs for the browsable API. | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
|     url('^', include(router.urls)), |     re_path('^', include(router.urls)), | ||||||
|     url('^me/', UserInformationView.as_view()), |     re_path('^me/', UserInformationView.as_view()), | ||||||
|     url('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), |     re_path('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
|  |  | ||||||
| import io | import io | ||||||
|  |  | ||||||
|  | from bootstrap_datepicker_plus.widgets import DatePickerInput | ||||||
| from PIL import Image, ImageSequence | from PIL import Image, ImageSequence | ||||||
| from django import forms | from django import forms | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @@ -13,7 +14,7 @@ from django.forms import CheckboxSelectMultiple | |||||||
| 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 _ | ||||||
| from note.models import NoteSpecial, Alias | from note.models import NoteSpecial, Alias | ||||||
| from note_kfet.inputs import Autocomplete, AmountInput, DatePickerInput | from note_kfet.inputs import Autocomplete, AmountInput | ||||||
| from permission.models import PermissionMask, Role | from permission.models import PermissionMask, Role | ||||||
|  |  | ||||||
| from .models import Profile, Club, Membership | from .models import Profile, Club, Membership | ||||||
| @@ -32,7 +33,7 @@ class UserForm(forms.ModelForm): | |||||||
|         # Django usernames can only contain letters, numbers, @, ., +, - and _. |         # Django usernames can only contain letters, numbers, @, ., +, - and _. | ||||||
|         # We want to allow users to have uncommon and unpractical usernames: |         # We want to allow users to have uncommon and unpractical usernames: | ||||||
|         # That is their problem, and we have normalized aliases for us. |         # That is their problem, and we have normalized aliases for us. | ||||||
|         return super()._get_validation_exclusions() + ["username"] |         return super()._get_validation_exclusions() | {"username"} | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         model = User |         model = User | ||||||
| @@ -138,9 +139,6 @@ class ImageForm(forms.Form): | |||||||
|  |  | ||||||
|         return cleaned_data |         return cleaned_data | ||||||
|  |  | ||||||
|     def is_valid(self): |  | ||||||
|         return super().is_valid() or super().clean().get('image') is None |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ClubForm(forms.ModelForm): | class ClubForm(forms.ModelForm): | ||||||
|     def clean(self): |     def clean(self): | ||||||
| @@ -154,7 +152,7 @@ class ClubForm(forms.ModelForm): | |||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         model = Club |         model = Club | ||||||
|         exclude = ("add_registration_form",) |         fields = '__all__' | ||||||
|         widgets = { |         widgets = { | ||||||
|             "membership_fee_paid": AmountInput(), |             "membership_fee_paid": AmountInput(), | ||||||
|             "membership_fee_unpaid": AmountInput(), |             "membership_fee_unpaid": AmountInput(), | ||||||
|   | |||||||
| @@ -1,18 +0,0 @@ | |||||||
| # Generated by Django 2.2.28 on 2024-07-15 09:24 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('member', '0011_profile_vss_charter_read'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='club', |  | ||||||
|             name='add_registration_form', |  | ||||||
|             field=models.BooleanField(default=False, verbose_name='add to registration form'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -259,11 +259,6 @@ class Club(models.Model): | |||||||
|         help_text=_('Maximal date of a membership, after which members must renew it.'), |         help_text=_('Maximal date of a membership, after which members must renew it.'), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     add_registration_form = models.BooleanField( |  | ||||||
|         verbose_name=_("add to registration form"), |  | ||||||
|         default=False, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         verbose_name = _("club") |         verbose_name = _("club") | ||||||
|         verbose_name_plural = _("clubs") |         verbose_name_plural = _("clubs") | ||||||
| @@ -295,14 +290,7 @@ class Club(models.Model): | |||||||
|  |  | ||||||
|         today = datetime.date.today() |         today = datetime.date.today() | ||||||
|  |  | ||||||
|         # Avoid any problems on February 29 |         while (today - self.membership_start).days >= 365: | ||||||
|         if self.membership_start.month == 2 and self.membership_start.day == 29: |  | ||||||
|             self.membership_start -= datetime.timedelta(days=1) |  | ||||||
|         if self.membership_end.month == 2 and self.membership_end.day == 29: |  | ||||||
|             self.membership_end += datetime.timedelta(days=1) |  | ||||||
|  |  | ||||||
|         while today >= datetime.date(self.membership_start.year + 1, |  | ||||||
|                                      self.membership_start.month, self.membership_start.day): |  | ||||||
|             if self.membership_start: |             if self.membership_start: | ||||||
|                 self.membership_start = datetime.date(self.membership_start.year + 1, |                 self.membership_start = datetime.date(self.membership_start.year + 1, | ||||||
|                                                       self.membership_start.month, self.membership_start.day) |                                                       self.membership_start.month, self.membership_start.day) | ||||||
|   | |||||||
| @@ -14,9 +14,6 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|       <form method="post" enctype="multipart/form-data" id="formUpload"> |       <form method="post" enctype="multipart/form-data" id="formUpload"> | ||||||
|         {% csrf_token %} |         {% csrf_token %} | ||||||
|         {{ form |crispy }} |         {{ form |crispy }} | ||||||
|         {% if user.note.display_image != "pic/default.png" %} |  | ||||||
|           <input type="submit" class="btn btn-primary" value="{% trans "Remove" %}"> |  | ||||||
|         {% endif %} |  | ||||||
|       </form> |       </form> | ||||||
|     </div> |     </div> | ||||||
|     <!-- MODAL TO CROP THE IMAGE --> |     <!-- MODAL TO CROP THE IMAGE --> | ||||||
|   | |||||||
| @@ -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): | ||||||
|   | |||||||
| @@ -326,9 +326,6 @@ class PictureUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, Det | |||||||
|         """Save image to note""" |         """Save image to note""" | ||||||
|         image = form.cleaned_data['image'] |         image = form.cleaned_data['image'] | ||||||
|  |  | ||||||
|         if image is None: |  | ||||||
|             image = "pic/default.png" |  | ||||||
|         else: |  | ||||||
|         # Rename as a PNG or GIF |         # Rename as a PNG or GIF | ||||||
|         extension = image.name.split(".")[-1] |         extension = image.name.split(".")[-1] | ||||||
|         if extension == "gif": |         if extension == "gif": | ||||||
|   | |||||||
| @@ -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="consumer") | ||||||
|  |  | ||||||
|     router.register(path + '/transaction/category', TemplateCategoryViewSet) |     router.register(path + '/transaction/category', TemplateCategoryViewSet) | ||||||
|     router.register(path + '/transaction/transaction', TransactionViewSet) |     router.register(path + '/transaction/transaction', TransactionViewSet) | ||||||
|   | |||||||
| @@ -179,19 +179,10 @@ class ConsumerViewSet(ReadOnlyProtectedModelViewSet): | |||||||
|             # We match first an alias if it is matched without normalization, |             # We match first an alias if it is matched without normalization, | ||||||
|             # then if the normalized pattern matches a normalized alias. |             # then if the normalized pattern matches a normalized alias. | ||||||
|             queryset = queryset.filter( |             queryset = queryset.filter( | ||||||
|                 **{f'name{suffix}': alias_prefix + alias} |                 Q(**{f'name{suffix}': alias_prefix + alias}) | ||||||
|             ).union( |                 | Q(**{f'normalized_name{suffix}': alias_prefix + Alias.normalize(alias)}) | ||||||
|                 queryset.filter( |                 | Q(**{f'normalized_name{suffix}': alias_prefix + alias.lower()}) | ||||||
|                     Q(**{f'normalized_name{suffix}': alias_prefix + Alias.normalize(alias)}) |             ) | ||||||
|                     & ~Q(**{f'name{suffix}': alias_prefix + alias}) |  | ||||||
|                 ), |  | ||||||
|                 all=True).union( |  | ||||||
|                 queryset.filter( |  | ||||||
|                     Q(**{f'normalized_name{suffix}': alias_prefix + alias.lower()}) |  | ||||||
|                     & ~Q(**{f'normalized_name{suffix}': alias_prefix + Alias.normalize(alias)}) |  | ||||||
|                     & ~Q(**{f'name{suffix}': alias_prefix + alias}) |  | ||||||
|                 ), |  | ||||||
|                 all=True) |  | ||||||
|  |  | ||||||
|         queryset = queryset if settings.DATABASES[queryset.db]["ENGINE"] == 'django.db.backends.postgresql' \ |         queryset = queryset if settings.DATABASES[queryset.db]["ENGINE"] == 'django.db.backends.postgresql' \ | ||||||
|             else queryset.order_by("name") |             else queryset.order_by("name") | ||||||
|   | |||||||
| @@ -1,13 +1,15 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
|  |  | ||||||
|  | from bootstrap_datepicker_plus.widgets import DateTimePickerInput | ||||||
| from django import forms | from django import forms | ||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
| from django.forms import CheckboxSelectMultiple | from django.forms import CheckboxSelectMultiple | ||||||
| from django.utils.timezone import make_aware | from django.utils.timezone import make_aware | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from note_kfet.inputs import Autocomplete, AmountInput, DateTimePickerInput | from note_kfet.inputs import Autocomplete, AmountInput | ||||||
|  |  | ||||||
| from .models import TransactionTemplate, NoteClub, Alias | from .models import TransactionTemplate, NoteClub, Alias | ||||||
|  |  | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ def create_special_notes(apps, schema_editor): | |||||||
| class Migration(migrations.Migration): | class Migration(migrations.Migration): | ||||||
|     dependencies = [ |     dependencies = [ | ||||||
|         ('note', '0001_initial'), |         ('note', '0001_initial'), | ||||||
|  |         ('logs', '0001_initial'), | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|     operations = [ |     operations = [ | ||||||
|   | |||||||
| @@ -0,0 +1,25 @@ | |||||||
|  | # Generated by Django 5.0.7 on 2024-07-11 09:24 | ||||||
|  |  | ||||||
|  | import django.db.models.deletion | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('contenttypes', '0002_remove_content_type_name'), | ||||||
|  |         ('note', '0006_trust'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='note', | ||||||
|  |             name='polymorphic_ctype', | ||||||
|  |             field=models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_%(app_label)s.%(class)s_set+', to='contenttypes.contenttype'), | ||||||
|  |         ), | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='transaction', | ||||||
|  |             name='polymorphic_ctype', | ||||||
|  |             field=models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_%(app_label)s.%(class)s_set+', to='contenttypes.contenttype'), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
| @@ -9,7 +9,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|            name="{{ widget.name }}" |            name="{{ widget.name }}" | ||||||
|            {# Other attributes are loaded  #} |            {# Other attributes are loaded  #} | ||||||
|            {% for name, value in widget.attrs.items %} |            {% for name, value in widget.attrs.items %} | ||||||
|                 {% ifnotequal value False %}{{ name }}{% ifnotequal value True %}="{{ value|stringformat:'s' }}"{% endifnotequal %}{% endifnotequal %} |                 {% if value != False %}{{ name }}{% if value != True %}="{{ value|stringformat:'s' }}"{% endif %}{% endif %} | ||||||
|             {% endfor %}> |             {% endfor %}> | ||||||
|     <div class="input-group-append"> |     <div class="input-group-append"> | ||||||
|         <span class="input-group-text">€</span> |         <span class="input-group-text">€</span> | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ | |||||||
|  |  | ||||||
| <p> | <p> | ||||||
|     Par ailleurs, le BDE ne sert pas d'alcool aux adhérents dont le solde |     Par ailleurs, le BDE ne sert pas d'alcool aux adhérents dont le solde | ||||||
|     est inférieur à 0 €. |     est inférieur à 0 € depuis plus de 24h. | ||||||
| </p> | </p> | ||||||
|  |  | ||||||
| <p> | <p> | ||||||
|   | |||||||
| @@ -2591,12 +2591,12 @@ | |||||||
| 				"note", | 				"note", | ||||||
| 				"transaction" | 				"transaction" | ||||||
| 			], | 			], | ||||||
| 			"query": "[\"OR\", {\"source__balance__gte\": 0}, [\"AND\", [\"NOT\", {\"recurrenttransaction__template__category__name\": \"Alcool\"}], {\"source__balance__gte\": {\"F\": [\"SUB\", [\"MUL\", [\"F\", \"amount\"], [\"F\", \"quantity\"]], 2000]}}], {\"valid\": false}]", | 			"query": "[\"OR\", {\"source__balance__gte\": {\"F\": [\"SUB\", [\"MUL\", [\"F\", \"amount\"], [\"F\", \"quantity\"]], 2000]}}, {\"valid\": false}]", | ||||||
| 			"type": "add", | 			"type": "add", | ||||||
| 			"mask": 2, | 			"mask": 2, | ||||||
| 			"field": "", | 			"field": "", | ||||||
| 			"permanent": false, | 			"permanent": false, | ||||||
| 			"description": "Créer une transaction quelconque tant que la source reste positive s'il s'agit d'alcool, sinon au-dessus de -20€" | 			"description": "Créer une transaction quelconque tant que la source reste au-dessus de -20 €" | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ | 	{ | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ from django import forms | |||||||
| from django.contrib.auth.forms import UserCreationForm | from django.contrib.auth.forms import UserCreationForm | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  | # from member.models import Club | ||||||
| from note.models import NoteSpecial, Alias | from note.models import NoteSpecial, Alias | ||||||
| from note_kfet.inputs import AmountInput | from note_kfet.inputs import AmountInput | ||||||
|  |  | ||||||
| @@ -114,3 +115,12 @@ class ValidationForm(forms.Form): | |||||||
|         required=False, |         required=False, | ||||||
|         initial=True, |         initial=True, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  | #     If the bda exists | ||||||
|  | #     if Club.objects.filter(name__iexact="bda").exists(): | ||||||
|  | #         The user can join the bda club at the inscription | ||||||
|  | #         join_bda = forms.BooleanField( | ||||||
|  | #             label=_("Join BDA Club"), | ||||||
|  | #             required=False, | ||||||
|  | #             initial=True, | ||||||
|  | #         ) | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django import forms |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.auth.mixins import LoginRequiredMixin | from django.contrib.auth.mixins import LoginRequiredMixin | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| @@ -239,8 +238,9 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|         fee += bde.membership_fee_paid if user.profile.paid else bde.membership_fee_unpaid |         fee += bde.membership_fee_paid if user.profile.paid else bde.membership_fee_unpaid | ||||||
|         kfet = Club.objects.get(name="Kfet") |         kfet = Club.objects.get(name="Kfet") | ||||||
|         fee += kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid |         fee += kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid | ||||||
|         for club in Club.objects.filter(add_registration_form=True): |         if Club.objects.filter(name__iexact="BDA").exists(): | ||||||
|             fee += club.membership_fee_paid if user.profile.paid else club.membership_fee_unpaid |             bda = Club.objects.get(name__iexact="BDA") | ||||||
|  |             fee += bda.membership_fee_paid if user.profile.paid else bda.membership_fee_unpaid | ||||||
|         ctx["total_fee"] = "{:.02f}".format(fee / 100, ) |         ctx["total_fee"] = "{:.02f}".format(fee / 100, ) | ||||||
|  |  | ||||||
| #        ctx["declare_soge_account"] = SogeCredit.objects.filter(user=user).exists() | #        ctx["declare_soge_account"] = SogeCredit.objects.filter(user=user).exists() | ||||||
| @@ -249,16 +249,6 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|  |  | ||||||
|     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) | ||||||
|  |  | ||||||
|         # add clubs that are in registration form |  | ||||||
|         for club in Club.objects.filter(add_registration_form=True).order_by("name"): |  | ||||||
|             form_join_club = forms.BooleanField( |  | ||||||
|                 label=_("Join %(club)s Club") % {'club': club.name}, |  | ||||||
|                 required=False, |  | ||||||
|                 initial=False, |  | ||||||
|             ) |  | ||||||
|             form.fields.update({f"join_{club.id}": form_join_club}) |  | ||||||
|  |  | ||||||
|         user = self.get_object() |         user = self.get_object() | ||||||
|         form.fields["last_name"].initial = user.last_name |         form.fields["last_name"].initial = user.last_name | ||||||
|         form.fields["first_name"].initial = user.first_name |         form.fields["first_name"].initial = user.first_name | ||||||
| @@ -276,6 +266,11 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|             form.add_error(None, _("An alias with a similar name already exists.")) |             form.add_error(None, _("An alias with a similar name already exists.")) | ||||||
|             return self.form_invalid(form) |             return self.form_invalid(form) | ||||||
|  |  | ||||||
|  |         # Check if BDA exist to propose membership at regisration | ||||||
|  |         bda_exists = False | ||||||
|  |         if Club.objects.filter(name__iexact="BDA").exists(): | ||||||
|  |             bda_exists = True | ||||||
|  |  | ||||||
|         # Get form data |         # Get form data | ||||||
| #        soge = form.cleaned_data["soge"] | #        soge = form.cleaned_data["soge"] | ||||||
|         credit_type = form.cleaned_data["credit_type"] |         credit_type = form.cleaned_data["credit_type"] | ||||||
| @@ -285,9 +280,8 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|         bank = form.cleaned_data["bank"] |         bank = form.cleaned_data["bank"] | ||||||
|         join_bde = form.cleaned_data["join_bde"] |         join_bde = form.cleaned_data["join_bde"] | ||||||
|         join_kfet = form.cleaned_data["join_kfet"] |         join_kfet = form.cleaned_data["join_kfet"] | ||||||
|  |         if bda_exists: | ||||||
|         clubs_registration = Club.objects.filter(add_registration_form=True).order_by("name") |             join_bda = form.cleaned_data["join_bda"] | ||||||
|         join_clubs = [(club, form.cleaned_data[f"join_{club.id}"]) for club in clubs_registration] |  | ||||||
|  |  | ||||||
| #        if soge: | #        if soge: | ||||||
| #            # If Société Générale pays the inscription, the user automatically joins the two clubs. | #            # If Société Générale pays the inscription, the user automatically joins the two clubs. | ||||||
| @@ -309,12 +303,11 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|         kfet_fee = kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid |         kfet_fee = kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid | ||||||
|         # Add extra fee for the full membership |         # Add extra fee for the full membership | ||||||
|         fee += kfet_fee if join_kfet else 0 |         fee += kfet_fee if join_kfet else 0 | ||||||
|         clubs_fee = dict() |         if bda_exists: | ||||||
|         for club, join_club in join_clubs: |             bda = Club.objects.get(name__iexact="BDA") | ||||||
|             club_fee = club.membership_fee_paid if user.profile.paid else club.membership_fee_unpaid |             bda_fee = bda.membership_fee_paid if user.profile.paid else bda.membership_fee_unpaid | ||||||
|             # Add extra fee for the club membership |             # Add extra fee for the bda membership | ||||||
|             clubs_fee[club] = club_fee |             fee += bda_fee if join_bda else 0 | ||||||
|             fee += club_fee if join_club else 0 |  | ||||||
|  |  | ||||||
| #        # If the bank pays, then we don't credit now. Treasurers will validate the transaction | #        # If the bank pays, then we don't credit now. Treasurers will validate the transaction | ||||||
| #        # and credit the note later. | #        # and credit the note later. | ||||||
| @@ -394,13 +387,12 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|             membership.roles.add(Role.objects.get(name="Adhérent Kfet")) |             membership.roles.add(Role.objects.get(name="Adhérent Kfet")) | ||||||
|             membership.save() |             membership.save() | ||||||
|  |  | ||||||
|         for club, join_club in join_clubs: |         if bda_exists and join_bda: | ||||||
|             if join_club: |  | ||||||
|             # Create membership for the user to the BDA starting today |             # Create membership for the user to the BDA starting today | ||||||
|             membership = Membership( |             membership = Membership( | ||||||
|                     club=club, |                 club=bda, | ||||||
|                 user=user, |                 user=user, | ||||||
|                     fee=clubs_fee[club], |                 fee=bda_fee, | ||||||
|             ) |             ) | ||||||
|             membership.save() |             membership.save() | ||||||
|             membership.refresh_from_db() |             membership.refresh_from_db() | ||||||
|   | |||||||
| @@ -0,0 +1,19 @@ | |||||||
|  | # Generated by Django 5.0.7 on 2024-07-11 09:24 | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('note', '0007_alter_note_polymorphic_ctype_and_more'), | ||||||
|  |         ('treasury', '0008_auto_20240322_0045'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='sogecredit', | ||||||
|  |             name='transactions', | ||||||
|  |             field=models.ManyToManyField(blank=True, related_name='+', to='note.membershiptransaction', verbose_name='membership transactions'), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
| @@ -1,13 +1,14 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
|  | 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 | 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, DatePickerInput, Autocomplete, ColorWidget | from note_kfet.inputs import AmountInput, Autocomplete, ColorWidget | ||||||
|  |  | ||||||
| from ..models import WEIClub, WEIRegistration, Bus, BusTeam, WEIMembership, WEIRole | from ..models import WEIClub, WEIRegistration, Bus, BusTeam, WEIMembership, WEIRole | ||||||
|  |  | ||||||
|   | |||||||
| @@ -439,7 +439,7 @@ class TestWEIRegistration(TestCase): | |||||||
|             emergency_contact_phone='+33123456789', |             emergency_contact_phone='+33123456789', | ||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertTrue("This user can't be in her/his first year since he/she has already participated to a WEI." |         self.assertTrue("This user can't be in her/his first year since he/she has already participated to a WEI." | ||||||
|                         in str(response.context["form"].errors)) |                         in str(response.context["form"].errors)) | ||||||
|  |  | ||||||
|         # Check that if the WEI is started, we can't register anyone |         # Check that if the WEI is started, we can't register anyone | ||||||
| @@ -635,7 +635,7 @@ class TestWEIRegistration(TestCase): | |||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertFalse(response.context["form"].is_valid()) |         self.assertFalse(response.context["form"].is_valid()) | ||||||
|         self.assertTrue("This team doesn't belong to the given bus." in str(response.context["form"].errors)) |         self.assertTrue("This team doesn't belong to the given bus." in str(response.context["form"].errors)) | ||||||
|  |  | ||||||
|         response = self.client.post(reverse("wei:validate_registration", kwargs=dict(pk=self.registration.pk)), dict( |         response = self.client.post(reverse("wei:validate_registration", kwargs=dict(pk=self.registration.pk)), dict( | ||||||
|             roles=[WEIRole.objects.get(name="GC WEI").id], |             roles=[WEIRole.objects.get(name="GC WEI").id], | ||||||
|   | |||||||
| @@ -114,7 +114,7 @@ msgstr "Lieu où l'activité est organisée, par exemple la Kfet." | |||||||
| msgid "type" | msgid "type" | ||||||
| msgstr "type" | msgstr "type" | ||||||
|  |  | ||||||
| #: apps/activity/models.py:89 apps/logs/models.py:22 apps/member/models.py:318 | #: apps/activity/models.py:89 apps/logs/models.py:22 apps/member/models.py:313 | ||||||
| #: apps/note/models/notes.py:148 apps/treasury/models.py:293 | #: apps/note/models/notes.py:148 apps/treasury/models.py:293 | ||||||
| #: apps/wei/models.py:171 apps/wei/templates/wei/attribute_bus_1A.html:13 | #: apps/wei/models.py:171 apps/wei/templates/wei/attribute_bus_1A.html:13 | ||||||
| #: apps/wei/templates/wei/survey.html:15 | #: apps/wei/templates/wei/survey.html:15 | ||||||
| @@ -247,7 +247,6 @@ msgid "The validation of the activity is pending." | |||||||
| msgstr "La validation de cette activité est en attente." | msgstr "La validation de cette activité est en attente." | ||||||
|  |  | ||||||
| #: apps/activity/tables.py:43 apps/treasury/tables.py:107 | #: apps/activity/tables.py:43 apps/treasury/tables.py:107 | ||||||
| #: apps/member/templates/member/picture_update.html:18 |  | ||||||
| msgid "Remove" | msgid "Remove" | ||||||
| msgstr "Supprimer" | msgstr "Supprimer" | ||||||
|  |  | ||||||
| @@ -263,13 +262,13 @@ msgstr "supprimer" | |||||||
| msgid "Type" | msgid "Type" | ||||||
| msgstr "Type" | msgstr "Type" | ||||||
|  |  | ||||||
| #: apps/activity/tables.py:84 apps/member/forms.py:196 | #: apps/activity/tables.py:84 apps/member/forms.py:193 | ||||||
| #: apps/registration/forms.py:92 apps/treasury/forms.py:131 | #: apps/registration/forms.py:92 apps/treasury/forms.py:131 | ||||||
| #: apps/wei/forms/registration.py:104 | #: apps/wei/forms/registration.py:104 | ||||||
| msgid "Last name" | msgid "Last name" | ||||||
| msgstr "Nom de famille" | msgstr "Nom de famille" | ||||||
|  |  | ||||||
| #: apps/activity/tables.py:86 apps/member/forms.py:201 | #: apps/activity/tables.py:86 apps/member/forms.py:198 | ||||||
| #: apps/note/templates/note/transaction_form.html:138 | #: apps/note/templates/note/transaction_form.html:138 | ||||||
| #: apps/registration/forms.py:97 apps/treasury/forms.py:133 | #: apps/registration/forms.py:97 apps/treasury/forms.py:133 | ||||||
| #: apps/wei/forms/registration.py:109 | #: apps/wei/forms/registration.py:109 | ||||||
| @@ -401,37 +400,37 @@ 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:68 note_kfet/templates/base.html:90 | #: apps/activity/views.py:67 note_kfet/templates/base.html:90 | ||||||
| msgid "Activities" | msgid "Activities" | ||||||
| msgstr "Activités" | msgstr "Activités" | ||||||
|  |  | ||||||
| #: apps/activity/views.py:108 | #: apps/activity/views.py:93 | ||||||
| msgid "Activity detail" | msgid "Activity detail" | ||||||
| msgstr "Détails de l'activité" | msgstr "Détails de l'activité" | ||||||
|  |  | ||||||
| #: apps/activity/views.py:128 | #: apps/activity/views.py:113 | ||||||
| msgid "Update activity" | msgid "Update activity" | ||||||
| msgstr "Modifier l'activité" | msgstr "Modifier l'activité" | ||||||
|  |  | ||||||
| #: apps/activity/views.py:155 | #: apps/activity/views.py:140 | ||||||
| msgid "Invite guest to the activity \"{}\"" | msgid "Invite guest to the activity \"{}\"" | ||||||
| msgstr "Invitation pour l'activité « {} »" | msgstr "Invitation pour l'activité « {} »" | ||||||
|  |  | ||||||
| #: apps/activity/views.py:193 | #: apps/activity/views.py:178 | ||||||
| msgid "You are not allowed to display the entry interface for this activity." | msgid "You are not allowed to display the entry interface for this activity." | ||||||
| msgstr "" | msgstr "" | ||||||
| "Vous n'êtes pas autorisé·e à afficher l'interface des entrées pour cette " | "Vous n'êtes pas autorisé·e à afficher l'interface des entrées pour cette " | ||||||
| "activité." | "activité." | ||||||
|  |  | ||||||
| #: apps/activity/views.py:196 | #: apps/activity/views.py:181 | ||||||
| msgid "This activity does not support activity entries." | msgid "This activity does not support activity entries." | ||||||
| msgstr "Cette activité ne requiert pas d'entrées." | msgstr "Cette activité ne requiert pas d'entrées." | ||||||
|  |  | ||||||
| #: apps/activity/views.py:199 | #: apps/activity/views.py:184 | ||||||
| msgid "This activity is closed." | msgid "This activity is closed." | ||||||
| msgstr "Cette activité est fermée." | msgstr "Cette activité est fermée." | ||||||
|  |  | ||||||
| #: apps/activity/views.py:295 | #: apps/activity/views.py:280 | ||||||
| msgid "Entry for activity \"{}\"" | msgid "Entry for activity \"{}\"" | ||||||
| msgstr "Entrées pour l'activité « {} »" | msgstr "Entrées pour l'activité « {} »" | ||||||
|  |  | ||||||
| @@ -508,11 +507,11 @@ msgstr "cotisation pour adhérer (normalien·ne élève)" | |||||||
| msgid "membership fee (unpaid students)" | msgid "membership fee (unpaid students)" | ||||||
| msgstr "cotisation pour adhérer (normalien·ne étudiant·e)" | msgstr "cotisation pour adhérer (normalien·ne étudiant·e)" | ||||||
|  |  | ||||||
| #: apps/member/admin.py:65 apps/member/models.py:330 | #: apps/member/admin.py:65 apps/member/models.py:325 | ||||||
| msgid "roles" | msgid "roles" | ||||||
| msgstr "rôles" | msgstr "rôles" | ||||||
|  |  | ||||||
| #: apps/member/admin.py:66 apps/member/models.py:344 | #: apps/member/admin.py:66 apps/member/models.py:339 | ||||||
| msgid "fee" | msgid "fee" | ||||||
| msgstr "cotisation" | msgstr "cotisation" | ||||||
|  |  | ||||||
| @@ -564,29 +563,29 @@ msgid "This image cannot be loaded." | |||||||
| msgstr "Cette image ne peut pas être chargée." | msgstr "Cette image ne peut pas être chargée." | ||||||
|  |  | ||||||
| #: apps/member/forms.py:148 apps/member/views.py:102 | #: apps/member/forms.py:148 apps/member/views.py:102 | ||||||
| #: apps/registration/forms.py:34 apps/registration/views.py:276 | #: apps/registration/forms.py:34 apps/registration/views.py:266 | ||||||
| msgid "An alias with a similar name already exists." | msgid "An alias with a similar name already exists." | ||||||
| msgstr "Un alias avec un nom similaire existe déjà." | msgstr "Un alias avec un nom similaire existe déjà." | ||||||
|  |  | ||||||
| #: apps/member/forms.py:175 | #: apps/member/forms.py:172 | ||||||
| msgid "Inscription paid by Société Générale" | msgid "Inscription paid by Société Générale" | ||||||
| msgstr "Inscription payée par la Société générale" | msgstr "Inscription payée par la Société générale" | ||||||
|  |  | ||||||
| #: apps/member/forms.py:177 | #: apps/member/forms.py:174 | ||||||
| msgid "Check this case if the Société Générale paid the inscription." | msgid "Check this case if the Société Générale paid the inscription." | ||||||
| msgstr "Cochez cette case si la Société Générale a payé l'inscription." | msgstr "Cochez cette case si la Société Générale a payé l'inscription." | ||||||
|  |  | ||||||
| #: apps/member/forms.py:182 apps/registration/forms.py:79 | #: apps/member/forms.py:179 apps/registration/forms.py:79 | ||||||
| #: apps/wei/forms/registration.py:91 | #: apps/wei/forms/registration.py:91 | ||||||
| msgid "Credit type" | msgid "Credit type" | ||||||
| msgstr "Type de rechargement" | msgstr "Type de rechargement" | ||||||
|  |  | ||||||
| #: apps/member/forms.py:183 apps/registration/forms.py:80 | #: apps/member/forms.py:180 apps/registration/forms.py:80 | ||||||
| #: apps/wei/forms/registration.py:92 | #: apps/wei/forms/registration.py:92 | ||||||
| msgid "No credit" | msgid "No credit" | ||||||
| msgstr "Pas de rechargement" | msgstr "Pas de rechargement" | ||||||
|  |  | ||||||
| #: apps/member/forms.py:185 | #: apps/member/forms.py:182 | ||||||
| msgid "You can credit the note of the user." | msgid "You can credit the note of the user." | ||||||
| msgstr "Vous pouvez créditer la note de l'utilisateur·ice avant l'adhésion." | msgstr "Vous pouvez créditer la note de l'utilisateur·ice avant l'adhésion." | ||||||
|  |  | ||||||
| @@ -595,17 +594,17 @@ msgstr "Vous pouvez créditer la note de l'utilisateur·ice avant l'adhésion." | |||||||
| msgid "Credit amount" | msgid "Credit amount" | ||||||
| msgstr "Montant à créditer" | msgstr "Montant à créditer" | ||||||
|  |  | ||||||
| #: apps/member/forms.py:206 apps/note/templates/note/transaction_form.html:144 | #: apps/member/forms.py:203 apps/note/templates/note/transaction_form.html:144 | ||||||
| #: apps/registration/forms.py:102 apps/treasury/forms.py:135 | #: apps/registration/forms.py:102 apps/treasury/forms.py:135 | ||||||
| #: apps/wei/forms/registration.py:114 | #: apps/wei/forms/registration.py:114 | ||||||
| msgid "Bank" | msgid "Bank" | ||||||
| msgstr "Banque" | msgstr "Banque" | ||||||
|  |  | ||||||
| #: apps/member/forms.py:233 | #: apps/member/forms.py:230 | ||||||
| msgid "User" | msgid "User" | ||||||
| msgstr "Utilisateur·ice" | msgstr "Utilisateur·ice" | ||||||
|  |  | ||||||
| #: apps/member/forms.py:247 | #: apps/member/forms.py:244 | ||||||
| msgid "Roles" | msgid "Roles" | ||||||
| msgstr "Rôles" | msgstr "Rôles" | ||||||
|  |  | ||||||
| @@ -853,50 +852,46 @@ msgstr "" | |||||||
| "Date maximale d'une fin d'adhésion, après laquelle les adhérent·e·s doivent la " | "Date maximale d'une fin d'adhésion, après laquelle les adhérent·e·s doivent la " | ||||||
| "renouveler." | "renouveler." | ||||||
|  |  | ||||||
| #: apps/member/models.py:263 | #: apps/member/models.py:263 apps/member/models.py:319 | ||||||
| msgid "add to registration form" |  | ||||||
| msgstr "ajouter au formulaire d'inscription" |  | ||||||
|  |  | ||||||
| #: apps/member/models.py:268 apps/member/models.py:324 |  | ||||||
| #: apps/note/models/notes.py:176 | #: apps/note/models/notes.py:176 | ||||||
| msgid "club" | msgid "club" | ||||||
| msgstr "club" | msgstr "club" | ||||||
|  |  | ||||||
| #: apps/member/models.py:269 | #: apps/member/models.py:264 | ||||||
| msgid "clubs" | msgid "clubs" | ||||||
| msgstr "clubs" | msgstr "clubs" | ||||||
|  |  | ||||||
| #: apps/member/models.py:335 | #: apps/member/models.py:330 | ||||||
| msgid "membership starts on" | msgid "membership starts on" | ||||||
| msgstr "l'adhésion commence le" | msgstr "l'adhésion commence le" | ||||||
|  |  | ||||||
| #: apps/member/models.py:339 | #: apps/member/models.py:334 | ||||||
| msgid "membership ends on" | msgid "membership ends on" | ||||||
| msgstr "l'adhésion finit le" | msgstr "l'adhésion finit le" | ||||||
|  |  | ||||||
| #: apps/member/models.py:348 apps/note/models/transactions.py:385 | #: apps/member/models.py:343 apps/note/models/transactions.py:385 | ||||||
| msgid "membership" | msgid "membership" | ||||||
| msgstr "adhésion" | msgstr "adhésion" | ||||||
|  |  | ||||||
| #: apps/member/models.py:349 | #: apps/member/models.py:344 | ||||||
| msgid "memberships" | msgid "memberships" | ||||||
| msgstr "adhésions" | msgstr "adhésions" | ||||||
|  |  | ||||||
| #: apps/member/models.py:353 | #: apps/member/models.py:348 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "Membership of {user} for the club {club}" | msgid "Membership of {user} for the club {club}" | ||||||
| msgstr "Adhésion de {user} pour le club {club}" | msgstr "Adhésion de {user} pour le club {club}" | ||||||
|  |  | ||||||
| #: apps/member/models.py:372 | #: apps/member/models.py:367 | ||||||
| #, python-brace-format | #, python-brace-format | ||||||
| msgid "The role {role} does not apply to the club {club}." | msgid "The role {role} does not apply to the club {club}." | ||||||
| msgstr "Le rôle {role} ne s'applique pas au club {club}." | msgstr "Le rôle {role} ne s'applique pas au club {club}." | ||||||
|  |  | ||||||
| #: apps/member/models.py:381 apps/member/views.py:712 | #: apps/member/models.py:376 apps/member/views.py:712 | ||||||
| msgid "User is already a member of the club" | msgid "User is already a member of the club" | ||||||
| msgstr "L'utilisateur·ice est déjà membre du club" | msgstr "L'utilisateur·ice est déjà membre du club" | ||||||
|  |  | ||||||
| #: apps/member/models.py:393 apps/member/views.py:721 | #: apps/member/models.py:388 apps/member/views.py:721 | ||||||
| msgid "User is not a member of the parent club" | msgid "User is not a member of the parent club" | ||||||
| msgstr "L'utilisateur·ice n'est pas membre du club parent" | msgstr "L'utilisateur·ice n'est pas membre du club parent" | ||||||
|  |  | ||||||
| @@ -1158,11 +1153,11 @@ msgstr "Introspection :" | |||||||
| msgid "Show my applications" | msgid "Show my applications" | ||||||
| msgstr "Voir mes applications" | msgstr "Voir mes applications" | ||||||
|  |  | ||||||
| #: apps/member/templates/member/picture_update.html:38 | #: apps/member/templates/member/picture_update.html:35 | ||||||
| msgid "Nevermind" | msgid "Nevermind" | ||||||
| msgstr "Annuler" | msgstr "Annuler" | ||||||
|  |  | ||||||
| #: apps/member/templates/member/picture_update.html:39 | #: apps/member/templates/member/picture_update.html:36 | ||||||
| msgid "Crop and upload" | msgid "Crop and upload" | ||||||
| msgstr "Recadrer et envoyer" | msgstr "Recadrer et envoyer" | ||||||
|  |  | ||||||
| @@ -2188,23 +2183,18 @@ msgstr "Utilisateur·ice·s en attente d'inscription" | |||||||
| msgid "Registration detail" | msgid "Registration detail" | ||||||
| msgstr "Détails de l'inscription" | msgstr "Détails de l'inscription" | ||||||
|  |  | ||||||
| #: apps/registration/views.py:256 | #: apps/registration/views.py:293 | ||||||
| #, python-format |  | ||||||
| msgid "Join %(club)s Club" |  | ||||||
| msgstr "Adhérer au club %(club)s" |  | ||||||
|  |  | ||||||
| #: apps/registration/views.py:299 |  | ||||||
| msgid "You must join the BDE." | msgid "You must join the BDE." | ||||||
| msgstr "Vous devez adhérer au BDE." | msgstr "Vous devez adhérer au BDE." | ||||||
|  |  | ||||||
| #: apps/registration/views.py:330 | #: apps/registration/views.py:323 | ||||||
| msgid "" | msgid "" | ||||||
| "The entered amount is not enough for the memberships, should be at least {}" | "The entered amount is not enough for the memberships, should be at least {}" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Le montant crédité est trop faible pour adhérer, il doit être au minimum de " | "Le montant crédité est trop faible pour adhérer, il doit être au minimum de " | ||||||
| "{}" | "{}" | ||||||
|  |  | ||||||
| #: apps/registration/views.py:425 | #: apps/registration/views.py:417 | ||||||
| msgid "Invalidate pre-registration" | msgid "Invalidate pre-registration" | ||||||
| msgstr "Invalider l'inscription" | msgstr "Invalider l'inscription" | ||||||
|  |  | ||||||
| @@ -3630,6 +3620,9 @@ 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." | ||||||
|  |  | ||||||
|  | #~ msgid "Join BDA Club" | ||||||
|  | #~ msgstr "Adhérer au club BDA" | ||||||
|  |  | ||||||
| #, fuzzy | #, fuzzy | ||||||
| #~| msgid "People having you as a friend" | #~| msgid "People having you as a friend" | ||||||
| #~ msgid "You already have that person as a friend" | #~ msgid "You already have that person as a friend" | ||||||
|   | |||||||
| @@ -25,19 +25,13 @@ admin_site.register(Site, SiteAdmin) | |||||||
|  |  | ||||||
| # Add external apps model | # Add external apps model | ||||||
| if "oauth2_provider" in settings.INSTALLED_APPS: | if "oauth2_provider" in settings.INSTALLED_APPS: | ||||||
|     from oauth2_provider.admin import Application, ApplicationAdmin, Grant, \ |     from oauth2_provider.admin import ApplicationAdmin, GrantAdmin, AccessTokenAdmin, RefreshTokenAdmin | ||||||
|         GrantAdmin, AccessToken, AccessTokenAdmin, RefreshToken, RefreshTokenAdmin |     from oauth2_provider.models import Application, Grant, AccessToken, RefreshToken | ||||||
|     admin_site.register(Application, ApplicationAdmin) |     admin_site.register(Application, ApplicationAdmin) | ||||||
|     admin_site.register(Grant, GrantAdmin) |     admin_site.register(Grant, GrantAdmin) | ||||||
|     admin_site.register(AccessToken, AccessTokenAdmin) |     admin_site.register(AccessToken, AccessTokenAdmin) | ||||||
|     admin_site.register(RefreshToken, RefreshTokenAdmin) |     admin_site.register(RefreshToken, RefreshTokenAdmin) | ||||||
|  |  | ||||||
| if "django_htcpcp_tea" in settings.INSTALLED_APPS: |  | ||||||
|     from django_htcpcp_tea.admin import * |  | ||||||
|     from django_htcpcp_tea.models import * |  | ||||||
|     admin_site.register(Pot, PotAdmin) |  | ||||||
|     admin_site.register(TeaType, TeaTypeAdmin) |  | ||||||
|     admin_site.register(Addition, AdditionAdmin) |  | ||||||
|  |  | ||||||
| if "mailer" in settings.INSTALLED_APPS: | if "mailer" in settings.INSTALLED_APPS: | ||||||
|     from mailer.admin import * |     from mailer.admin import * | ||||||
| @@ -50,9 +44,3 @@ if "rest_framework" in settings.INSTALLED_APPS: | |||||||
|     from rest_framework.authtoken.admin import * |     from rest_framework.authtoken.admin import * | ||||||
|     from rest_framework.authtoken.models import * |     from rest_framework.authtoken.models import * | ||||||
|     admin_site.register(Token, TokenAdmin) |     admin_site.register(Token, TokenAdmin) | ||||||
|  |  | ||||||
| if "cas_server" in settings.INSTALLED_APPS: |  | ||||||
|     from cas_server.admin import * |  | ||||||
|     from cas_server.models import * |  | ||||||
|     admin_site.register(ServicePattern, ServicePatternAdmin) |  | ||||||
|     admin_site.register(FederatedIendityProvider, FederatedIendityProviderAdmin) |  | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ INSTALLED_APPS = [ | |||||||
|     'bootstrap_datepicker_plus', |     'bootstrap_datepicker_plus', | ||||||
|     'colorfield', |     'colorfield', | ||||||
|     'crispy_forms', |     'crispy_forms', | ||||||
|     'django_htcpcp_tea', |     'crispy_bootstrap4', | ||||||
|     'django_tables2', |     'django_tables2', | ||||||
|     'mailer', |     'mailer', | ||||||
|     'phonenumber_field', |     'phonenumber_field', | ||||||
| @@ -90,7 +90,6 @@ MIDDLEWARE = [ | |||||||
|     'django.middleware.clickjacking.XFrameOptionsMiddleware', |     'django.middleware.clickjacking.XFrameOptionsMiddleware', | ||||||
|     'django.middleware.locale.LocaleMiddleware', |     'django.middleware.locale.LocaleMiddleware', | ||||||
|     'django.contrib.sites.middleware.CurrentSiteMiddleware', |     'django.contrib.sites.middleware.CurrentSiteMiddleware', | ||||||
|     'django_htcpcp_tea.middleware.HTCPCPTeaMiddleware', |  | ||||||
|     'note_kfet.middlewares.SessionMiddleware', |     'note_kfet.middlewares.SessionMiddleware', | ||||||
|     'note_kfet.middlewares.LoginByIPMiddleware', |     'note_kfet.middlewares.LoginByIPMiddleware', | ||||||
|     'note_kfet.middlewares.TurbolinksMiddleware', |     'note_kfet.middlewares.TurbolinksMiddleware', | ||||||
| @@ -295,3 +294,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' | ||||||
|  |  | ||||||
|  | # Default field for primary key | ||||||
|  | DEFAULT_AUTO_FIELD = "django.db.models.AutoField" | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|    {% if widget.value != None and widget.value != "" %}value="{{ widget.value }}"{% endif %} |    {% if widget.value != None and widget.value != "" %}value="{{ widget.value }}"{% endif %} | ||||||
|    name="{{ widget.name }}_name" autocomplete="off" |    name="{{ widget.name }}_name" autocomplete="off" | ||||||
|     {% for name, value in widget.attrs.items %} |     {% for name, value in widget.attrs.items %} | ||||||
|         {% ifnotequal value False %}{{ name }}{% ifnotequal value True %}="{{ value|stringformat:'s' }}"{% endifnotequal %}{% endifnotequal %} |         {% if value != False %}{{ name }}{% if value != True %}="{{ value|stringformat:'s' }}"{% endif %}{% endif %} | ||||||
|     {% endfor %} |     {% endfor %} | ||||||
|     aria-describedby="{{widget.attrs.id}}_tooltip"> |     aria-describedby="{{widget.attrs.id}}_tooltip"> | ||||||
|     {% if widget.resetable %} |     {% if widget.resetable %} | ||||||
|   | |||||||
| @@ -126,9 +126,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 %} | ||||||
|   | |||||||
| @@ -30,9 +30,6 @@ urlpatterns = [ | |||||||
|     path('accounts/', include('django.contrib.auth.urls')), |     path('accounts/', include('django.contrib.auth.urls')), | ||||||
|     path('api/', include('api.urls')), |     path('api/', include('api.urls')), | ||||||
|     path('permission/', include('permission.urls')), |     path('permission/', include('permission.urls')), | ||||||
|  |  | ||||||
|     # Make coffee |  | ||||||
|     path('coffee/', include('django_htcpcp_tea.urls')), |  | ||||||
| ] | ] | ||||||
|  |  | ||||||
| # During development, serve static and media files | # During development, serve static and media files | ||||||
| @@ -46,11 +43,6 @@ if "oauth2_provider" in settings.INSTALLED_APPS: | |||||||
|         path('o/', include('oauth2_provider.urls', namespace='oauth2_provider')) |         path('o/', include('oauth2_provider.urls', namespace='oauth2_provider')) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| if "cas_server" in settings.INSTALLED_APPS: |  | ||||||
|     urlpatterns.append( |  | ||||||
|         path('cas/', include('cas_server.urls', namespace='cas_server')) |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
| if "debug_toolbar" in settings.INSTALLED_APPS: | if "debug_toolbar" in settings.INSTALLED_APPS: | ||||||
|     import debug_toolbar |     import debug_toolbar | ||||||
|     urlpatterns = [ |     urlpatterns = [ | ||||||
|   | |||||||
| @@ -1,19 +1,17 @@ | |||||||
| beautifulsoup4~=4.7.1 | beautifulsoup4~=4.12.3 | ||||||
| Django~=2.2.15 | crispy-bootstrap4~=2024.1 | ||||||
| django-bootstrap-datepicker-plus~=3.0.5 | Django~=5.0.7 | ||||||
| django-cas-server~=1.2.0 | django-bootstrap-datepicker-plus~=5.0.5 | ||||||
| django-colorfield~=0.3.2 | django-colorfield~=0.11.0 | ||||||
| django-crispy-forms~=1.7.2 | django-crispy-forms~=2.2 | ||||||
| django-extensions>=2.1.4 | django-extensions~=3.2.3 | ||||||
| django-filter~=2.1 | django-filter~=24.2 | ||||||
| django-htcpcp-tea~=0.3.1 | django-mailer~=2.3.2 | ||||||
| django-mailer~=2.0.1 | django-oauth-toolkit~=2.4.0 | ||||||
| django-oauth-toolkit~=1.3.3 | django-phonenumber-field~=8.0.0 | ||||||
| django-phonenumber-field~=5.0.0 | django-polymorphic~=3.1.0 | ||||||
| django-polymorphic>=2.0.3,<3.0.0 | django-rest-polymorphic~=0.1.10 | ||||||
| djangorestframework>=3.9.0,<3.13.0 | django-tables2~=2.7.0 | ||||||
| django-rest-polymorphic~=0.1.9 | djangorestframework~=3.15.2 | ||||||
| django-tables2~=2.3.1 | phonenumbers~=8.13.40 | ||||||
| python-memcached~=1.59 |  | ||||||
| phonenumbers~=8.9.10 |  | ||||||
| Pillow>=5.4.1 | Pillow>=5.4.1 | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,13 +1,10 @@ | |||||||
| [tox] | [tox] | ||||||
| envlist = | envlist = | ||||||
|     # Debian Buster Python |  | ||||||
|     py37-django22 |  | ||||||
|  |  | ||||||
|     # Ubuntu 20.04 Python |     # Ubuntu 20.04 Python | ||||||
|     py38-django22 |     py310-django50 | ||||||
|  |  | ||||||
|     # Debian Bullseye Python |     # Debian Bullseye Python | ||||||
|     py39-django22 |     py311-django50 | ||||||
|  |  | ||||||
|     linters |     linters | ||||||
| skipsdist = True | skipsdist = True | ||||||
| @@ -51,4 +48,4 @@ max-complexity = 15 | |||||||
| max-line-length = 160 | max-line-length = 160 | ||||||
| import-order-style = google | import-order-style = google | ||||||
| application-import-names = flake8 | application-import-names = flake8 | ||||||
| format = %(cyan)s%(path)s%(reset)s:%(yellow)s%(bold)s%(row)d%(reset)s:%(green)s%(bold)s%(col)d%(reset)s: %(red)s%(bold)s%(code)s%(reset)s %(text)s | format = ${cyan}%(path)s${reset}:${yellow_bold}%(row)d${reset}:${green_bold}%(col)d${reset}: ${red_bold}%(code)s${reset} %(text)s | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user