mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-22 21:08:02 +02:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			survey_wei
			...
			83de99c28e
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 83de99c28e | 
| @@ -7,25 +7,25 @@ stages: | |||||||
| variables: | variables: | ||||||
|   GIT_SUBMODULE_STRATEGY: recursive |   GIT_SUBMODULE_STRATEGY: recursive | ||||||
|  |  | ||||||
| # Debian Bullseye | # Debian Buster | ||||||
| py39-django42: | #  py37-django22: | ||||||
|   stage: test | #   stage: test | ||||||
|   image: debian:bullseye | #   image: debian:buster-backports | ||||||
|   before_script: | #   before_script: | ||||||
|     - > | #     - > | ||||||
|         apt-get update && | #         apt-get update && | ||||||
|         apt-get install --no-install-recommends -y | #         apt-get install --no-install-recommends -t buster-backports -y | ||||||
|         python3-django python3-django-crispy-forms | #         python3-django python3-django-crispy-forms | ||||||
|         python3-django-extensions python3-django-filters python3-django-polymorphic | #         python3-django-extensions python3-django-filters python3-django-polymorphic | ||||||
|         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-django42 | #   script: tox -e py37-django22 | ||||||
|  |  | ||||||
| # Ubuntu 22.04 | # Ubuntu 20.04 | ||||||
| py310-django42: | py38-django22: | ||||||
|   stage: test |   stage: test | ||||||
|   image: ubuntu:22.04 |   image: ubuntu:20.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 +37,12 @@ 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 py38-django22 | ||||||
|  |  | ||||||
| # Debian Bookworm | # Debian Bullseye | ||||||
| py311-django42: | py39-django22: | ||||||
|   stage: test |   stage: test | ||||||
|   image: debian:bookworm |   image: debian:bullseye | ||||||
|   before_script: |   before_script: | ||||||
|     - > |     - > | ||||||
|         apt-get update && |         apt-get update && | ||||||
| @@ -52,13 +52,11 @@ 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 py39-django22 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| linters: | linters: | ||||||
|   stage: quality-assurance |   stage: quality-assurance | ||||||
|   image: debian:bookworm |   image: debian:bullseye | ||||||
|   before_script: |   before_script: | ||||||
|     - apt-get update && apt-get install -y tox |     - apt-get update && apt-get install -y tox | ||||||
|   script: tox -e linters |   script: tox -e linters | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ from django.contrib import admin | |||||||
| from note_kfet.admin import admin_site | from note_kfet.admin import admin_site | ||||||
|  |  | ||||||
| from .forms import GuestForm | from .forms import GuestForm | ||||||
| from .models import Activity, ActivityType, Entry, Guest, Opener | from .models import Activity, ActivityType, Entry, Guest | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(Activity, site=admin_site) | @admin.register(Activity, site=admin_site) | ||||||
| @@ -45,11 +45,3 @@ class EntryAdmin(admin.ModelAdmin): | |||||||
|     Admin customisation for Entry |     Admin customisation for Entry | ||||||
|     """ |     """ | ||||||
|     list_display = ('note', 'activity', 'time', 'guest') |     list_display = ('note', 'activity', 'time', 'guest') | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(Opener, site=admin_site) |  | ||||||
| class OpenerAdmin(admin.ModelAdmin): |  | ||||||
|     """ |  | ||||||
|     Admin customisation for Opener |  | ||||||
|     """ |  | ||||||
|     list_display = ('activity', 'opener') |  | ||||||
|   | |||||||
| @@ -4,14 +4,13 @@ | |||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| from random import shuffle | from random import shuffle | ||||||
|  |  | ||||||
| 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.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 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 | from note_kfet.inputs import Autocomplete, DateTimePickerInput | ||||||
| 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 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| # Generated by Django 4.2.15 on 2024-08-28 08:00 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
| import django.db.models.deletion |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('note', '0006_trust'), |  | ||||||
|         ('activity', '0004_opener'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AlterModelOptions( |  | ||||||
|             name='opener', |  | ||||||
|             options={'verbose_name': 'Opener', 'verbose_name_plural': 'Openers'}, |  | ||||||
|         ), |  | ||||||
|         migrations.AlterField( |  | ||||||
|             model_name='opener', |  | ||||||
|             name='opener', |  | ||||||
|             field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='activity_responsible', to='note.note', verbose_name='Opener'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| /** | /** | ||||||
|  * On form submit, add a new opener |  * On form submit, create a new friendship | ||||||
|  */ |  */ | ||||||
| function form_create_opener (e) { | function form_create_opener (e) { | ||||||
|   // Do not submit HTML form |   // Do not submit HTML form | ||||||
| @@ -16,9 +16,9 @@ function form_create_opener (e) { | |||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Add an opener between an activity and a user |  * Create a trust between users | ||||||
|  * @param activity:Integer activity id |  * @param trusting:Integer trusting note id | ||||||
|  * @param opener:Integer user note id |  * @param trusted:Integer trusted note id | ||||||
|  */ |  */ | ||||||
| function create_opener(activity, opener) { | function create_opener(activity, opener) { | ||||||
|   $.post('/api/activity/opener/', { |   $.post('/api/activity/opener/', { | ||||||
| @@ -28,15 +28,36 @@ function create_opener(activity, opener) { | |||||||
|   }).done(function () { |   }).done(function () { | ||||||
|   // Reload tables |   // Reload tables | ||||||
|   $('#opener_table').load(location.pathname + ' #opener_table') |   $('#opener_table').load(location.pathname + ' #opener_table') | ||||||
|     addMsg(gettext('Opener successfully added'), 'success') |     addMsg(gettext('Friendship successfully added'), 'success') | ||||||
|   }).fail(function (xhr, _textStatus, _error) { |   }).fail(function (xhr, _textStatus, _error) { | ||||||
|     errMsg(xhr.responseJSON) |     errMsg(xhr.responseJSON) | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * On click of "delete", delete the opener |  * On form submit, create a new friendship | ||||||
|  * @param button_id:Integer Opener id to remove | function create_opener (e) { | ||||||
|  |   // Do not submit HTML form | ||||||
|  |   e.preventDefault() | ||||||
|  |  | ||||||
|  |   // Get data and send to API | ||||||
|  |   const formData = new FormData(e.target) | ||||||
|  |   $.post('/api/activity/opener/', { | ||||||
|  |     csrfmiddlewaretoken: formData.get('csrfmiddlewaretoken'), | ||||||
|  |     activity: formData.get('activity'), | ||||||
|  |     opener: formData.get('opener') | ||||||
|  |   }).done(function () { | ||||||
|  |     // Reload table | ||||||
|  |     $('#opener_table').load(location.pathname + ' #opener_table') | ||||||
|  |     addMsg(gettext('Alias successfully added'), 'success') | ||||||
|  |   }).fail(function (xhr, _textStatus, _error) { | ||||||
|  |     errMsg(xhr.responseJSON) | ||||||
|  |   }) | ||||||
|  | }*/ | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * On click of "delete", delete the trust | ||||||
|  |  * @param button_id:Integer Trust id to remove | ||||||
|  */ |  */ | ||||||
| function delete_button (button_id) { | function delete_button (button_id) { | ||||||
|   $.ajax({ |   $.ajax({ | ||||||
| @@ -44,7 +65,7 @@ function delete_button (button_id) { | |||||||
|     method: 'DELETE', |     method: 'DELETE', | ||||||
|     headers: { 'X-CSRFTOKEN': CSRF_TOKEN } |     headers: { 'X-CSRFTOKEN': CSRF_TOKEN } | ||||||
|   }).done(function () { |   }).done(function () { | ||||||
|     addMsg(gettext('Opener successfully deleted'), 'success') |     addMsg(gettext('Friendship successfully deleted'), 'success') | ||||||
|     $('#opener_table').load(location.pathname + ' #opener_table') |     $('#opener_table').load(location.pathname + ' #opener_table') | ||||||
|   }).fail(function (xhr, _textStatus, _error) { |   }).fail(function (xhr, _textStatus, _error) { | ||||||
|     errMsg(xhr.responseJSON) |     errMsg(xhr.responseJSON) | ||||||
|   | |||||||
| @@ -2,8 +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 include | from django.conf.urls import url, include | ||||||
| from django.urls import re_path |  | ||||||
| from rest_framework import routers | from rest_framework import routers | ||||||
|  |  | ||||||
| from .views import UserInformationView | from .views import UserInformationView | ||||||
| @@ -15,33 +14,29 @@ router = routers.DefaultRouter() | |||||||
| router.register('models', ContentTypeViewSet) | router.register('models', ContentTypeViewSet) | ||||||
| router.register('user', UserViewSet) | router.register('user', UserViewSet) | ||||||
|  |  | ||||||
| if "activity" in settings.INSTALLED_APPS: |  | ||||||
|     from activity.api.urls import register_activity_urls |  | ||||||
|     register_activity_urls(router, 'activity') |  | ||||||
|  |  | ||||||
| if "food" in settings.INSTALLED_APPS: |  | ||||||
|     from food.api.urls import register_food_urls |  | ||||||
|     register_food_urls(router, 'food') |  | ||||||
|  |  | ||||||
| if "logs" in settings.INSTALLED_APPS: |  | ||||||
|     from logs.api.urls import register_logs_urls |  | ||||||
|     register_logs_urls(router, 'logs') |  | ||||||
|  |  | ||||||
| if "member" in settings.INSTALLED_APPS: | if "member" in settings.INSTALLED_APPS: | ||||||
|     from member.api.urls import register_members_urls |     from member.api.urls import register_members_urls | ||||||
|     register_members_urls(router, 'members') |     register_members_urls(router, 'members') | ||||||
|  |  | ||||||
|  | if "member" in settings.INSTALLED_APPS: | ||||||
|  |     from activity.api.urls import register_activity_urls | ||||||
|  |     register_activity_urls(router, 'activity') | ||||||
|  |  | ||||||
| if "note" in settings.INSTALLED_APPS: | if "note" in settings.INSTALLED_APPS: | ||||||
|     from note.api.urls import register_note_urls |     from note.api.urls import register_note_urls | ||||||
|     register_note_urls(router, 'note') |     register_note_urls(router, 'note') | ||||||
|  |  | ||||||
|  | if "treasury" in settings.INSTALLED_APPS: | ||||||
|  |     from treasury.api.urls import register_treasury_urls | ||||||
|  |     register_treasury_urls(router, 'treasury') | ||||||
|  |  | ||||||
| if "permission" in settings.INSTALLED_APPS: | if "permission" in settings.INSTALLED_APPS: | ||||||
|     from permission.api.urls import register_permission_urls |     from permission.api.urls import register_permission_urls | ||||||
|     register_permission_urls(router, 'permission') |     register_permission_urls(router, 'permission') | ||||||
|  |  | ||||||
| if "treasury" in settings.INSTALLED_APPS: | if "logs" in settings.INSTALLED_APPS: | ||||||
|     from treasury.api.urls import register_treasury_urls |     from logs.api.urls import register_logs_urls | ||||||
|     register_treasury_urls(router, 'treasury') |     register_logs_urls(router, 'logs') | ||||||
|  |  | ||||||
| if "wei" in settings.INSTALLED_APPS: | if "wei" in settings.INSTALLED_APPS: | ||||||
|     from wei.api.urls import register_wei_urls |     from wei.api.urls import register_wei_urls | ||||||
| @@ -52,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 = [ | ||||||
|     re_path('^', include(router.urls)), |     url('^', include(router.urls)), | ||||||
|     re_path('^me/', UserInformationView.as_view()), |     url('^me/', UserInformationView.as_view()), | ||||||
|     re_path('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), |     url('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -1,37 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from django.contrib import admin |  | ||||||
| from django.db import transaction |  | ||||||
| from note_kfet.admin import admin_site |  | ||||||
|  |  | ||||||
| from .models import Allergen, BasicFood, QRCode, TransformedFood |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(QRCode, site=admin_site) |  | ||||||
| class QRCodeAdmin(admin.ModelAdmin): |  | ||||||
|     pass |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(BasicFood, site=admin_site) |  | ||||||
| class BasicFoodAdmin(admin.ModelAdmin): |  | ||||||
|     @transaction.atomic |  | ||||||
|     def save_related(self, *args, **kwargs): |  | ||||||
|         ans = super().save_related(*args, **kwargs) |  | ||||||
|         args[1].instance.update() |  | ||||||
|         return ans |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(TransformedFood, site=admin_site) |  | ||||||
| class TransformedFoodAdmin(admin.ModelAdmin): |  | ||||||
|     exclude = ["allergens", "expiry_date"] |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def save_related(self, request, form, *args, **kwargs): |  | ||||||
|         super().save_related(request, form, *args, **kwargs) |  | ||||||
|         form.instance.update() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(Allergen, site=admin_site) |  | ||||||
| class AllergenAdmin(admin.ModelAdmin): |  | ||||||
|     pass |  | ||||||
| @@ -1,50 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from rest_framework import serializers |  | ||||||
|  |  | ||||||
| from ..models import Allergen, BasicFood, QRCode, TransformedFood |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AllergenSerializer(serializers.ModelSerializer): |  | ||||||
|     """ |  | ||||||
|     REST API Serializer for Allergen. |  | ||||||
|     The djangorestframework plugin will analyse the model `Allergen` and parse all fields in the API. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = Allergen |  | ||||||
|         fields = '__all__' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BasicFoodSerializer(serializers.ModelSerializer): |  | ||||||
|     """ |  | ||||||
|     REST API Serializer for BasicFood. |  | ||||||
|     The djangorestframework plugin will analyse the model `BasicFood` and parse all fields in the API. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = BasicFood |  | ||||||
|         fields = '__all__' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeSerializer(serializers.ModelSerializer): |  | ||||||
|     """ |  | ||||||
|     REST API Serializer for QRCode. |  | ||||||
|     The djangorestframework plugin will analyse the model `QRCode` and parse all fields in the API. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = QRCode |  | ||||||
|         fields = '__all__' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFoodSerializer(serializers.ModelSerializer): |  | ||||||
|     """ |  | ||||||
|     REST API Serializer for TransformedFood. |  | ||||||
|     The djangorestframework plugin will analyse the model `TransformedFood` and parse all fields in the API. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = TransformedFood |  | ||||||
|         fields = '__all__' |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from .views import AllergenViewSet, BasicFoodViewSet, QRCodeViewSet, TransformedFoodViewSet |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def register_food_urls(router, path): |  | ||||||
|     """ |  | ||||||
|     Configure router for Food REST API. |  | ||||||
|     """ |  | ||||||
|     router.register(path + '/allergen', AllergenViewSet) |  | ||||||
|     router.register(path + '/basic_food', BasicFoodViewSet) |  | ||||||
|     router.register(path + '/qrcode', QRCodeViewSet) |  | ||||||
|     router.register(path + '/transformed_food', TransformedFoodViewSet) |  | ||||||
| @@ -1,61 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from api.viewsets import ReadProtectedModelViewSet |  | ||||||
| from django_filters.rest_framework import DjangoFilterBackend |  | ||||||
| from rest_framework.filters import SearchFilter |  | ||||||
|  |  | ||||||
| from .serializers import AllergenSerializer, BasicFoodSerializer, QRCodeSerializer, TransformedFoodSerializer |  | ||||||
| from ..models import Allergen, BasicFood, QRCode, TransformedFood |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AllergenViewSet(ReadProtectedModelViewSet): |  | ||||||
|     """ |  | ||||||
|     REST API View set. |  | ||||||
|     The djangorestframework plugin will get all `Allergen` objects, serialize it to JSON with the given serializer, |  | ||||||
|     then render it on /api/food/allergen/ |  | ||||||
|     """ |  | ||||||
|     queryset = Allergen.objects.order_by('id') |  | ||||||
|     serializer_class = AllergenSerializer |  | ||||||
|     filter_backends = [DjangoFilterBackend, SearchFilter] |  | ||||||
|     filterset_fields = ['name', ] |  | ||||||
|     search_fields = ['$name', ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BasicFoodViewSet(ReadProtectedModelViewSet): |  | ||||||
|     """ |  | ||||||
|     REST API View set. |  | ||||||
|     The djangorestframework plugin will get all `BasicFood` objects, serialize it to JSON with the given serializer, |  | ||||||
|     then render it on /api/food/basic_food/ |  | ||||||
|     """ |  | ||||||
|     queryset = BasicFood.objects.order_by('id') |  | ||||||
|     serializer_class = BasicFoodSerializer |  | ||||||
|     filter_backends = [DjangoFilterBackend, SearchFilter] |  | ||||||
|     filterset_fields = ['name', ] |  | ||||||
|     search_fields = ['$name', ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeViewSet(ReadProtectedModelViewSet): |  | ||||||
|     """ |  | ||||||
|     REST API View set. |  | ||||||
|     The djangorestframework plugin will get all `QRCode` objects, serialize it to JSON with the given serializer, |  | ||||||
|     then render it on /api/food/qrcode/ |  | ||||||
|     """ |  | ||||||
|     queryset = QRCode.objects.order_by('id') |  | ||||||
|     serializer_class = QRCodeSerializer |  | ||||||
|     filter_backends = [DjangoFilterBackend, SearchFilter] |  | ||||||
|     filterset_fields = ['qr_code_number', ] |  | ||||||
|     search_fields = ['$qr_code_number', ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFoodViewSet(ReadProtectedModelViewSet): |  | ||||||
|     """ |  | ||||||
|     REST API View set. |  | ||||||
|     The djangorestframework plugin will get all `TransformedFood` objects, serialize it to JSON with the given serializer, |  | ||||||
|     then render it on /api/food/transformed_food/ |  | ||||||
|     """ |  | ||||||
|     queryset = TransformedFood.objects.order_by('id') |  | ||||||
|     serializer_class = TransformedFoodSerializer |  | ||||||
|     filter_backends = [DjangoFilterBackend, SearchFilter] |  | ||||||
|     filterset_fields = ['name', ] |  | ||||||
|     search_fields = ['$name', ] |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
|  |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
| from django.apps import AppConfig |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class FoodkfetConfig(AppConfig): |  | ||||||
|     name = 'food' |  | ||||||
|     verbose_name = _('food') |  | ||||||
| @@ -1,114 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from random import shuffle |  | ||||||
|  |  | ||||||
| from django import forms |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
| from django.utils import timezone |  | ||||||
| from member.models import Club |  | ||||||
| from bootstrap_datepicker_plus.widgets import DateTimePickerInput |  | ||||||
| from note_kfet.inputs import Autocomplete |  | ||||||
| from note_kfet.middlewares import get_current_request |  | ||||||
| from permission.backends import PermissionBackend |  | ||||||
|  |  | ||||||
| from .models import BasicFood, QRCode, TransformedFood |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AddIngredientForms(forms.ModelForm): |  | ||||||
|     """ |  | ||||||
|     Form for add an ingredient |  | ||||||
|     """ |  | ||||||
|     def __init__(self, *args, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         self.fields['ingredient'].queryset = self.fields['ingredient'].queryset.filter( |  | ||||||
|             polymorphic_ctype__model='transformedfood', |  | ||||||
|             is_ready=False, |  | ||||||
|             is_active=True, |  | ||||||
|             was_eaten=False, |  | ||||||
|         ) |  | ||||||
|         # Caution, the logic is inverted here, we flip the logic on saving in AddIngredientView |  | ||||||
|         self.fields['is_active'].initial = True |  | ||||||
|         self.fields['is_active'].label = _("Fully used") |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = TransformedFood |  | ||||||
|         fields = ('ingredient', 'is_active') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BasicFoodForms(forms.ModelForm): |  | ||||||
|     """ |  | ||||||
|     Form for add non-transformed food |  | ||||||
|     """ |  | ||||||
|     def __init__(self, *args, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         self.fields['name'].widget.attrs.update({"autofocus": "autofocus"}) |  | ||||||
|         self.fields['name'].required = True |  | ||||||
|         self.fields['owner'].required = True |  | ||||||
|  |  | ||||||
|         # Some example |  | ||||||
|         self.fields['name'].widget.attrs.update({"placeholder": _("Pasta METRO 5kg")}) |  | ||||||
|         clubs = list(Club.objects.filter(PermissionBackend.filter_queryset(get_current_request(), Club, "change")).all()) |  | ||||||
|         shuffle(clubs) |  | ||||||
|         self.fields['owner'].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..." |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = BasicFood |  | ||||||
|         fields = ('name', 'owner', 'date_type', 'expiry_date', 'is_active', 'was_eaten', 'allergens',) |  | ||||||
|         widgets = { |  | ||||||
|             "owner": Autocomplete( |  | ||||||
|                 model=Club, |  | ||||||
|                 attrs={"api_url": "/api/members/club/"}, |  | ||||||
|             ), |  | ||||||
|             'expiry_date': DateTimePickerInput(), |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeForms(forms.ModelForm): |  | ||||||
|     """ |  | ||||||
|     Form for create QRCode |  | ||||||
|     """ |  | ||||||
|     def __init__(self, *args, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         self.fields['food_container'].queryset = self.fields['food_container'].queryset.filter( |  | ||||||
|             is_active=True, |  | ||||||
|             was_eaten=False, |  | ||||||
|             polymorphic_ctype__model='transformedfood', |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = QRCode |  | ||||||
|         fields = ('food_container',) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFoodForms(forms.ModelForm): |  | ||||||
|     """ |  | ||||||
|     Form for add transformed food |  | ||||||
|     """ |  | ||||||
|     def __init__(self, *args, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         self.fields['name'].widget.attrs.update({"autofocus": "autofocus"}) |  | ||||||
|         self.fields['name'].required = True |  | ||||||
|         self.fields['owner'].required = True |  | ||||||
|         self.fields['creation_date'].required = True |  | ||||||
|         self.fields['creation_date'].initial = timezone.now |  | ||||||
|         self.fields['is_active'].initial = True |  | ||||||
|         self.fields['is_ready'].initial = False |  | ||||||
|         self.fields['was_eaten'].initial = False |  | ||||||
|  |  | ||||||
|         # Some example |  | ||||||
|         self.fields['name'].widget.attrs.update({"placeholder": _("Lasagna")}) |  | ||||||
|         clubs = list(Club.objects.filter(PermissionBackend.filter_queryset(get_current_request(), Club, "change")).all()) |  | ||||||
|         shuffle(clubs) |  | ||||||
|         self.fields['owner'].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..." |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = TransformedFood |  | ||||||
|         fields = ('name', 'creation_date', 'owner', 'is_active', 'is_ready', 'was_eaten', 'shelf_life') |  | ||||||
|         widgets = { |  | ||||||
|             "owner": Autocomplete( |  | ||||||
|                 model=Club, |  | ||||||
|                 attrs={"api_url": "/api/members/club/"}, |  | ||||||
|             ), |  | ||||||
|             'creation_date': DateTimePickerInput(), |  | ||||||
|         } |  | ||||||
| @@ -1,84 +0,0 @@ | |||||||
| # Generated by Django 2.2.28 on 2024-07-05 08:57 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
| import django.db.models.deletion |  | ||||||
| import django.utils.timezone |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     initial = True |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('contenttypes', '0002_remove_content_type_name'), |  | ||||||
|         ('member', '0011_profile_vss_charter_read'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='Allergen', |  | ||||||
|             fields=[ |  | ||||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |  | ||||||
|                 ('name', models.CharField(max_length=255, verbose_name='name')), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Allergen', |  | ||||||
|                 'verbose_name_plural': 'Allergens', |  | ||||||
|             }, |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='Food', |  | ||||||
|             fields=[ |  | ||||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |  | ||||||
|                 ('name', models.CharField(max_length=255, verbose_name='name')), |  | ||||||
|                 ('expiry_date', models.DateTimeField(verbose_name='expiry date')), |  | ||||||
|                 ('was_eaten', models.BooleanField(default=False, verbose_name='was eaten')), |  | ||||||
|                 ('is_ready', models.BooleanField(default=False, verbose_name='is ready')), |  | ||||||
|                 ('allergens', models.ManyToManyField(blank=True, to='food.Allergen', verbose_name='allergen')), |  | ||||||
|                 ('owner', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='+', to='member.Club', verbose_name='owner')), |  | ||||||
|                 ('polymorphic_ctype', models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_food.food_set+', to='contenttypes.ContentType')), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'foods', |  | ||||||
|             }, |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='BasicFood', |  | ||||||
|             fields=[ |  | ||||||
|                 ('food_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='food.Food')), |  | ||||||
|                 ('date_type', models.CharField(choices=[('DLC', 'DLC'), ('DDM', 'DDM')], max_length=255)), |  | ||||||
|                 ('arrival_date', models.DateTimeField(default=django.utils.timezone.now, verbose_name='arrival date')), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Basic food', |  | ||||||
|                 'verbose_name_plural': 'Basic foods', |  | ||||||
|             }, |  | ||||||
|             bases=('food.food',), |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='QRCode', |  | ||||||
|             fields=[ |  | ||||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), |  | ||||||
|                 ('qr_code_number', models.PositiveIntegerField(unique=True, verbose_name='QR-code number')), |  | ||||||
|                 ('food_container', models.OneToOneField(on_delete=django.db.models.deletion.PROTECT, related_name='QR_code', to='food.Food', verbose_name='food container')), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'QR-code', |  | ||||||
|                 'verbose_name_plural': 'QR-codes', |  | ||||||
|             }, |  | ||||||
|         ), |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name='TransformedFood', |  | ||||||
|             fields=[ |  | ||||||
|                 ('food_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='food.Food')), |  | ||||||
|                 ('creation_date', models.DateTimeField(verbose_name='creation date')), |  | ||||||
|                 ('is_active', models.BooleanField(default=True, verbose_name='is active')), |  | ||||||
|                 ('ingredient', models.ManyToManyField(blank=True, related_name='transformed_ingredient_inv', to='food.Food', verbose_name='transformed ingredient')), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 'verbose_name': 'Transformed food', |  | ||||||
|                 'verbose_name_plural': 'Transformed foods', |  | ||||||
|             }, |  | ||||||
|             bases=('food.food',), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| # Generated by Django 2.2.28 on 2024-07-06 20:37 |  | ||||||
|  |  | ||||||
| import datetime |  | ||||||
| from django.db import migrations, models |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('food', '0001_initial'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='transformedfood', |  | ||||||
|             name='shelf_life', |  | ||||||
|             field=models.DurationField(default=datetime.timedelta(days=3), verbose_name='shelf life'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,62 +0,0 @@ | |||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
| def create_14_mandatory_allergens(apps, schema_editor): |  | ||||||
|     """ |  | ||||||
|     There are 14 mandatory allergens, they are pre-injected |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     Allergen = apps.get_model("food", "allergen") |  | ||||||
|      |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Gluten", |  | ||||||
|     )  |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Fruits à coques", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Crustacés", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Céléri", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Oeufs", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Moutarde", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Poissons", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Soja", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Lait", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Sulfites", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Sésame", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Lupin", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Arachides", |  | ||||||
|     ) |  | ||||||
|     Allergen.objects.get_or_create( |  | ||||||
|         name="Mollusques", |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|     dependencies = [ |  | ||||||
|         ('food', '0002_transformedfood_shelf_life'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.RunPython(create_14_mandatory_allergens), |  | ||||||
|     ] |  | ||||||
|      |  | ||||||
|      |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| # Generated by Django 2.2.28 on 2024-08-13 21:58 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
| import django.db.models.deletion |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('food', '0003_create_14_allergens_mandatory'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.RemoveField( |  | ||||||
|             model_name='transformedfood', |  | ||||||
|             name='is_active', |  | ||||||
|         ), |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='food', |  | ||||||
|             name='is_active', |  | ||||||
|             field=models.BooleanField(default=True, verbose_name='is active'), |  | ||||||
|         ), |  | ||||||
|         migrations.AlterField( |  | ||||||
|             model_name='qrcode', |  | ||||||
|             name='food_container', |  | ||||||
|             field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='QR_code', to='food.Food', verbose_name='food container'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| # Generated by Django 4.2.15 on 2024-08-28 08:00 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
| import django.db.models.deletion |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('contenttypes', '0002_remove_content_type_name'), |  | ||||||
|         ('food', '0004_auto_20240813_2358'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AlterField( |  | ||||||
|             model_name='food', |  | ||||||
|             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'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,226 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from datetime import timedelta |  | ||||||
|  |  | ||||||
| from django.db import models, transaction |  | ||||||
| from django.utils import timezone |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
| from member.models import Club |  | ||||||
| from polymorphic.models import PolymorphicModel |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCode(models.Model): |  | ||||||
|     """ |  | ||||||
|     An QRCode model |  | ||||||
|     """ |  | ||||||
|     qr_code_number = models.PositiveIntegerField( |  | ||||||
|         verbose_name=_("QR-code number"), |  | ||||||
|         unique=True, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     food_container = models.ForeignKey( |  | ||||||
|         'Food', |  | ||||||
|         on_delete=models.CASCADE, |  | ||||||
|         related_name='QR_code', |  | ||||||
|         verbose_name=_('food container'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _("QR-code") |  | ||||||
|         verbose_name_plural = _("QR-codes") |  | ||||||
|  |  | ||||||
|     def __str__(self): |  | ||||||
|         return _("QR-code number {qr_code_number}").format(qr_code_number=self.qr_code_number) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Allergen(models.Model): |  | ||||||
|     """ |  | ||||||
|     A list of allergen and alimentary restrictions |  | ||||||
|     """ |  | ||||||
|     name = models.CharField( |  | ||||||
|         verbose_name=_('name'), |  | ||||||
|         max_length=255, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _('Allergen') |  | ||||||
|         verbose_name_plural = _('Allergens') |  | ||||||
|  |  | ||||||
|     def __str__(self): |  | ||||||
|         return self.name |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Food(PolymorphicModel): |  | ||||||
|     name = models.CharField( |  | ||||||
|         verbose_name=_('name'), |  | ||||||
|         max_length=255, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     owner = models.ForeignKey( |  | ||||||
|         Club, |  | ||||||
|         on_delete=models.PROTECT, |  | ||||||
|         related_name='+', |  | ||||||
|         verbose_name=_('owner'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     allergens = models.ManyToManyField( |  | ||||||
|         Allergen, |  | ||||||
|         blank=True, |  | ||||||
|         verbose_name=_('allergen'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     expiry_date = models.DateTimeField( |  | ||||||
|         verbose_name=_('expiry date'), |  | ||||||
|         null=False, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     was_eaten = models.BooleanField( |  | ||||||
|         default=False, |  | ||||||
|         verbose_name=_('was eaten'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     # is_ready != is_active : is_ready signifie que la nourriture est prête à être manger, |  | ||||||
|     #                         is_active signifie que la nourriture n'est pas encore archivé |  | ||||||
|     # il sert dans les cas où il est plus intéressant que de l'open soit conservé (confiture par ex) |  | ||||||
|  |  | ||||||
|     is_ready = models.BooleanField( |  | ||||||
|         default=False, |  | ||||||
|         verbose_name=_('is ready'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     is_active = models.BooleanField( |  | ||||||
|         default=True, |  | ||||||
|         verbose_name=_('is active'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def __str__(self): |  | ||||||
|         return self.name |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def save(self, force_insert=False, force_update=False, using=None, update_fields=None): |  | ||||||
|         return super().save(force_insert, force_update, using, update_fields) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _('food') |  | ||||||
|         verbose_name = _('foods') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BasicFood(Food): |  | ||||||
|     """ |  | ||||||
|     Food which has been directly buy on supermarket |  | ||||||
|     """ |  | ||||||
|     date_type = models.CharField( |  | ||||||
|         max_length=255, |  | ||||||
|         choices=( |  | ||||||
|             ("DLC", "DLC"), |  | ||||||
|             ("DDM", "DDM"), |  | ||||||
|         ) |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     arrival_date = models.DateTimeField( |  | ||||||
|         verbose_name=_('arrival date'), |  | ||||||
|         default=timezone.now, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     # label = models.ImageField( |  | ||||||
|     #     verbose_name=_('food label'), |  | ||||||
|     #     max_length=255, |  | ||||||
|     #     blank=False, |  | ||||||
|     #     null=False, |  | ||||||
|     #     upload_to='label/', |  | ||||||
|     # ) |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def update_allergens(self): |  | ||||||
|         # update parents |  | ||||||
|         for parent in self.transformed_ingredient_inv.iterator(): |  | ||||||
|             parent.update_allergens() |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def update_expiry_date(self): |  | ||||||
|         # update parents |  | ||||||
|         for parent in self.transformed_ingredient_inv.iterator(): |  | ||||||
|             parent.update_expiry_date() |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def update(self): |  | ||||||
|         self.update_allergens() |  | ||||||
|         self.update_expiry_date() |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _('Basic food') |  | ||||||
|         verbose_name_plural = _('Basic foods') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFood(Food): |  | ||||||
|     """ |  | ||||||
|     Transformed food  are a mix between basic food and meal |  | ||||||
|     """ |  | ||||||
|     creation_date = models.DateTimeField( |  | ||||||
|         verbose_name=_('creation date'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     ingredient = models.ManyToManyField( |  | ||||||
|         Food, |  | ||||||
|         blank=True, |  | ||||||
|         symmetrical=False, |  | ||||||
|         related_name='transformed_ingredient_inv', |  | ||||||
|         verbose_name=_('transformed ingredient'), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     # Without microbiological analyzes, the storage time is 3 days |  | ||||||
|     shelf_life = models.DurationField( |  | ||||||
|         verbose_name=_("shelf life"), |  | ||||||
|         default=timedelta(days=3), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def archive(self): |  | ||||||
|         # When a meal are archived, if it was eaten, update ingredient fully used for this meal |  | ||||||
|         raise NotImplementedError |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def update_allergens(self): |  | ||||||
|         # When allergens are changed, simply update the parents' allergens |  | ||||||
|         old_allergens = list(self.allergens.all()) |  | ||||||
|         self.allergens.clear() |  | ||||||
|         for ingredient in self.ingredient.iterator(): |  | ||||||
|             self.allergens.set(self.allergens.union(ingredient.allergens.all())) |  | ||||||
|  |  | ||||||
|         if old_allergens == list(self.allergens.all()): |  | ||||||
|             return |  | ||||||
|         super().save() |  | ||||||
|  |  | ||||||
|         # update parents |  | ||||||
|         for parent in self.transformed_ingredient_inv.iterator(): |  | ||||||
|             parent.update_allergens() |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def update_expiry_date(self): |  | ||||||
|         # When expiry_date is changed, simply update the parents' expiry_date |  | ||||||
|         old_expiry_date = self.expiry_date |  | ||||||
|         self.expiry_date = self.creation_date + self.shelf_life |  | ||||||
|         for ingredient in self.ingredient.iterator(): |  | ||||||
|             self.expiry_date = min(self.expiry_date, ingredient.expiry_date) |  | ||||||
|  |  | ||||||
|         if old_expiry_date == self.expiry_date: |  | ||||||
|             return |  | ||||||
|         super().save() |  | ||||||
|  |  | ||||||
|         # update parents |  | ||||||
|         for parent in self.transformed_ingredient_inv.iterator(): |  | ||||||
|             parent.update_expiry_date() |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def update(self): |  | ||||||
|         self.update_allergens() |  | ||||||
|         self.update_expiry_date() |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def save(self, *args, **kwargs): |  | ||||||
|         super().save(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _('Transformed food') |  | ||||||
|         verbose_name_plural = _('Transformed foods') |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| import django_tables2 as tables |  | ||||||
| from django_tables2 import A |  | ||||||
|  |  | ||||||
| from .models import TransformedFood |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFoodTable(tables.Table): |  | ||||||
|     name = tables.LinkColumn( |  | ||||||
|         'food:food_view', |  | ||||||
|         args=[A('pk'), ], |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = TransformedFood |  | ||||||
|         template_name = 'django_tables2/bootstrap4.html' |  | ||||||
|         fields = ('name', "owner", "allergens", "expiry_date") |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|   <h3 class="card-header text-center"> |  | ||||||
|     {{ title }} |  | ||||||
|   </h3> |  | ||||||
|   <div class="card-body" id="form"> |  | ||||||
|     <form method="post"> |  | ||||||
|       {%  csrf_token %} |  | ||||||
|       {{ form|crispy }} |  | ||||||
|       <button class="btn btn-primary" type="submit">{% trans "Submit"%}</button> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,37 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|   <h3 class="card-header text-center"> |  | ||||||
|     {{ title }} {{ food.name }} |  | ||||||
|   </h3> |  | ||||||
|   <div class="card-body"> |  | ||||||
|     <ul> |  | ||||||
|       <li><p>{% trans 'Owner' %} : {{ food.owner }}</p></li> |  | ||||||
|       <li><p>{% trans 'Arrival date' %} : {{ food.arrival_date }}</p></li> |  | ||||||
|       <li><p>{% trans 'Expiry date' %} : {{ food.expiry_date }} ({{ food.date_type }})</p></li> |  | ||||||
|       <li>{% trans 'Allergens' %} :</li> |  | ||||||
|       <ul> |  | ||||||
|       {% for allergen in food.allergens.iterator %} |  | ||||||
|         <li>{{ allergen.name }}</li> |  | ||||||
|       {% endfor %} |  | ||||||
|       </ul> |  | ||||||
| 	<p> |  | ||||||
| 	<li><p>{% trans 'Active' %} : {{ food.is_active }}<p></li> |  | ||||||
| 	<li><p>{% trans 'Eaten' %} : {{ food.was_eaten }}<p></li> |  | ||||||
|     </ul> |  | ||||||
|     {% if can_update %} |  | ||||||
| 	<a class="btn btn-sm btn-warning" href="{% url "food:basic_update" pk=food.pk %}">{% trans 'Update' %}</a> |  | ||||||
|     {% endif %} |  | ||||||
|     {% if can_add_ingredient %} |  | ||||||
|     	<a class="btn btn-sm btn-success" href="{% url "food:add_ingredient" pk=food.pk %}"> |  | ||||||
| 		{% trans 'Add to a meal' %} |  | ||||||
| 	</a> |  | ||||||
|     {% endif %} |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|   <h3 class="card-header text-center"> |  | ||||||
|     {{ title }} |  | ||||||
|   </h3> |  | ||||||
|   <div class="card-body" id="form"> |  | ||||||
|     <form method="post"> |  | ||||||
|       {% csrf_token %} |  | ||||||
|       {{ form | crispy }} |  | ||||||
|       <button class="btn btn-primary" type="submit">{% trans "Submit"%}</button> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,55 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load render_table from django_tables2 %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|   <h3 class="card-header text-center"> |  | ||||||
|     {{ title }} |  | ||||||
|   </h3> |  | ||||||
|   <div class="card-body" id="form"> |  | ||||||
|     <a class="btn btn-sm btn-success" href="{% url "food:qrcode_basic_create" slug=slug %}"> |  | ||||||
|       {% trans 'New basic food' %} |  | ||||||
|     </a> |  | ||||||
|     <form method="post"> |  | ||||||
|       {%  csrf_token %} |  | ||||||
|       {{ form|crispy }} |  | ||||||
|       <button class="btn btn-primary" type="submit">{% trans "Submit" %}</button> |  | ||||||
|     </form> |  | ||||||
|     <div class="card-body" id="profile_infos"> |  | ||||||
|       <h4>{% trans "Copy constructor" %}</h4> |  | ||||||
|       <table class="table"> |  | ||||||
|         <thead> |  | ||||||
|           <tr> |  | ||||||
|             <th class="orderable"> |  | ||||||
|               {% trans "Name" %} |  | ||||||
|             </th> |  | ||||||
|             <th class="orderable"> |  | ||||||
|               {% trans "Owner" %} |  | ||||||
|             </th> |  | ||||||
|             <th class="orderable"> |  | ||||||
|               {% trans "Arrival date" %} |  | ||||||
|             </th> |  | ||||||
|             <th class="orderable"> |  | ||||||
|               {% trans "Expiry date" %} |  | ||||||
|             </th> |  | ||||||
|           </tr> |  | ||||||
|         </thead> |  | ||||||
|         <tbody> |  | ||||||
|           {% for basic in last_basic %} |  | ||||||
|             <tr> |  | ||||||
|               <td><a href="{% url "food:qrcode_basic_create" slug=slug %}?copy={{ basic.pk }}">{{ basic.name }}</a></td> |  | ||||||
|               <td>{{ basic.owner }}</td> |  | ||||||
|               <td>{{ basic.arrival_date }}</td> |  | ||||||
|               <td>{{ basic.expiry_date }}</td> |  | ||||||
|             </tr> |  | ||||||
|           {% endfor %} |  | ||||||
|         </tbody> |  | ||||||
|       </table> |  | ||||||
|     </div> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|     <h3 class="card-header text-center"> |  | ||||||
| 	{{ title }} {% trans 'number' %} {{ qrcode.qr_code_number }} |  | ||||||
|     </h3> |  | ||||||
| 	<div class="card-body"> |  | ||||||
| 	    <ul> |  | ||||||
| 		<li><p>{% trans 'Name' %} : {{ qrcode.food_container.name }}</p></li> |  | ||||||
| 		<li><p>{% trans 'Owner' %} : {{ qrcode.food_container.owner }}</p></li> |  | ||||||
| 		<li><p>{% trans 'Expiry date' %} : {{ qrcode.food_container.expiry_date  }}</p></li> |  | ||||||
| 	    </ul> |  | ||||||
| 	{% if qrcode.food_container.polymorphic_ctype.model == 'basicfood' and can_update_basic %} |  | ||||||
| 	    <a class="btn btn-sm btn-warning" href="{% url "food:basic_update" pk=qrcode.food_container.pk %}" data-turbolinks="false"> |  | ||||||
| 		{% trans 'Update' %} |  | ||||||
| 	    </a> |  | ||||||
| 	{% elif can_update_transformed %} |  | ||||||
| 	    <a class="btn btn-sm btn-warning" href="{% url "food:transformed_update" pk=qrcode.food_container.pk %}"> |  | ||||||
| 		{% trans 'Update' %} |  | ||||||
| 	    </a> |  | ||||||
| 	{% endif %} |  | ||||||
| 	{% if can_view_detail %} |  | ||||||
| 	    <a class="btn btn-sm btn-primary" href="{% url "food:food_view" pk=qrcode.food_container.pk %}"> |  | ||||||
| 		{% trans 'View details' %} |  | ||||||
| 	    </a> |  | ||||||
| 	{% endif %} |  | ||||||
| 	{% if can_add_ingredient %} |  | ||||||
| 	    <a class="btn btn-sm btn-success" href="{% url "food:add_ingredient" pk=qrcode.food_container.pk %}"> |  | ||||||
| 		{% trans 'Add to a meal' %} |  | ||||||
| 	    </a> |  | ||||||
| 	{% endif %} |  | ||||||
| 	</div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,51 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|     <h3 class="card-header text-center"> |  | ||||||
| 	{{ title }} {{ food.name }} |  | ||||||
|     </h3> |  | ||||||
| 	<div class="card-body"> |  | ||||||
| 	    <ul> |  | ||||||
| 		<li><p>{% trans 'Owner' %} : {{ food.owner }}</p></li> |  | ||||||
| 		{% if can_see_ready %} |  | ||||||
| 		<li><p>{% trans 'Ready' %} : {{ food.is_ready }}</p></li> |  | ||||||
| 		{% endif %} |  | ||||||
| 		<li><p>{% trans 'Creation date' %} : {{ food.creation_date }}</p></li> |  | ||||||
| 		<li><p>{% trans 'Expiry date' %} : {{ food.expiry_date }}</p></li> |  | ||||||
| 		<li>{% trans 'Allergens' %} :</li> |  | ||||||
| 		<ul> |  | ||||||
| 		    {% for allergen in food.allergens.iterator %} |  | ||||||
| 		    <li>{{ allergen.name }}</li> |  | ||||||
| 		    {% endfor %} |  | ||||||
| 	        </ul> |  | ||||||
| 		<p> |  | ||||||
| 		<li>{% trans 'Ingredients' %} :</li> |  | ||||||
| 		<ul> |  | ||||||
| 		    {% for ingredient in food.ingredient.iterator %} |  | ||||||
| 		    <li><a href="{% url "food:food_view" pk=ingredient.pk %}">{{ ingredient.name }}</a></li> |  | ||||||
| 		    {% endfor %} |  | ||||||
| 		</ul> |  | ||||||
| 		<p> |  | ||||||
| 		<li><p>{% trans 'Shelf life' %} : {{ food.shelf_life }}</p></li> |  | ||||||
| 		<li><p>{% trans 'Ready' %} : {{ food.is_ready }}</p></li> |  | ||||||
| 		<li><p>{% trans 'Active' %} : {{ food.is_active }}</p></li> |  | ||||||
| 		<li><p>{% trans 'Eaten' %} : {{ food.was_eaten }}</p></li> |  | ||||||
| 	    </ul> |  | ||||||
| 	    {% if can_update %} |  | ||||||
| 	        <a class="btn btn-sm btn-warning" href="{% url "food:transformed_update" pk=food.pk %}"> |  | ||||||
| 		    {% trans 'Update' %} |  | ||||||
| 		</a> |  | ||||||
| 	    {% endif %} |  | ||||||
| 	    {% if can_add_ingredient %} |  | ||||||
| 	        <a class="btn btn-sm btn-success" href="{% url "food:add_ingredient" pk=food.pk %}"> |  | ||||||
| 		    {% trans 'Add to a meal' %} |  | ||||||
| 		</a> |  | ||||||
| 	    {% endif %} |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load i18n crispy_forms_tags %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-white mb-3"> |  | ||||||
|   <h3 class="card-header text-center"> |  | ||||||
|     {{ title }} |  | ||||||
|   </h3> |  | ||||||
|   <div class="card-body" id="form"> |  | ||||||
|     <form method="post"> |  | ||||||
|       {%  csrf_token %} |  | ||||||
|       {{ form|crispy }} |  | ||||||
|       <button class="btn btn-primary" type="submit">{% trans "Submit"%}</button> |  | ||||||
|     </form> |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,60 +0,0 @@ | |||||||
| {% extends "base.html" %} |  | ||||||
| {% comment %} |  | ||||||
| SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| {% endcomment %} |  | ||||||
| {% load render_table from django_tables2 %} |  | ||||||
| {% load i18n %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
| <div class="card bg-light mb-3"> |  | ||||||
|     <h3 class="card-header text-center"> |  | ||||||
| 	{% trans "Meal served" %} |  | ||||||
|     </h3> |  | ||||||
|     {% if can_create_meal %} |  | ||||||
|     <div class="card-footer"> |  | ||||||
| 	<a class="btn btn-sm btn-success" href="{% url 'food:transformed_create' %}" data-turbolinks="false"> |  | ||||||
| 	    {% trans 'New meal' %} |  | ||||||
| 	</a> |  | ||||||
|     </div> |  | ||||||
|     {% endif %} |  | ||||||
|     {% if served.data %} |  | ||||||
|     {% render_table served %} |  | ||||||
|     {% else %} |  | ||||||
|     <div class="card-body"> |  | ||||||
| 	<div class="alert alert-warning"> |  | ||||||
| 	    {% trans "There is no meal served." %} |  | ||||||
| 	</div> |  | ||||||
|     </div> |  | ||||||
|     {% endif %} |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="card bg-light mb-3"> |  | ||||||
|     <h3 class="card-header text-center"> |  | ||||||
| 	{% trans "Open" %} |  | ||||||
|     </h3> |  | ||||||
|     {% if open.data %} |  | ||||||
|     {% render_table open %} |  | ||||||
|     {% else %} |  | ||||||
|     <div class="card-body"> |  | ||||||
| 	<div class="alert alert-warning"> |  | ||||||
| 	    {% trans "There is no free meal." %} |  | ||||||
| 	</div> |  | ||||||
|     </div> |  | ||||||
|     {% endif %} |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="card bg-light mb-3"> |  | ||||||
|     <h3 class="card-header text-center"> |  | ||||||
|         {% trans "All meals" %} |  | ||||||
|     </h3> |  | ||||||
|     {% if table.data %} |  | ||||||
|     {% render_table table %} |  | ||||||
|     {% else %} |  | ||||||
|     <div class="card-body"> |  | ||||||
|         <div class="alert alert-warning"> |  | ||||||
|             {% trans "There is no meal." %} |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
|     {% endif %} |  | ||||||
| </div> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # from django.test import TestCase |  | ||||||
|  |  | ||||||
| # Create your tests here. |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from django.urls import path |  | ||||||
|  |  | ||||||
| from . import views |  | ||||||
|  |  | ||||||
| app_name = 'food' |  | ||||||
|  |  | ||||||
| urlpatterns = [ |  | ||||||
|     path('', views.TransformedListView.as_view(), name='food_list'), |  | ||||||
|     path('<int:slug>', views.QRCodeView.as_view(), name='qrcode_view'), |  | ||||||
|     path('detail/<int:pk>', views.FoodView.as_view(), name='food_view'), |  | ||||||
|  |  | ||||||
|     path('<int:slug>/create_qrcode', views.QRCodeCreateView.as_view(), name='qrcode_create'), |  | ||||||
|     path('<int:slug>/create_qrcode/basic', views.QRCodeBasicFoodCreateView.as_view(), name='qrcode_basic_create'), |  | ||||||
|     path('create/transformed', views.TransformedFoodCreateView.as_view(), name='transformed_create'), |  | ||||||
|     path('update/basic/<int:pk>', views.BasicFoodUpdateView.as_view(), name='basic_update'), |  | ||||||
|     path('update/transformed/<int:pk>', views.TransformedFoodUpdateView.as_view(), name='transformed_update'), |  | ||||||
|     path('add/<int:pk>', views.AddIngredientView.as_view(), name='add_ingredient'), |  | ||||||
| ] |  | ||||||
| @@ -1,421 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from django.db import transaction |  | ||||||
| from django.contrib.auth.mixins import LoginRequiredMixin |  | ||||||
| from django.http import HttpResponseRedirect |  | ||||||
| from django_tables2.views import MultiTableMixin |  | ||||||
| from django.urls import reverse |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
| from django.utils import timezone |  | ||||||
| from django.views.generic import DetailView, UpdateView |  | ||||||
| from django.views.generic.list import ListView |  | ||||||
| from django.forms import HiddenInput |  | ||||||
| from permission.backends import PermissionBackend |  | ||||||
| from permission.views import ProtectQuerysetMixin, ProtectedCreateView |  | ||||||
|  |  | ||||||
| from .forms import AddIngredientForms, BasicFoodForms, QRCodeForms, TransformedFoodForms |  | ||||||
| from .models import BasicFood, Food, QRCode, TransformedFood |  | ||||||
| from .tables import TransformedFoodTable |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AddIngredientView(ProtectQuerysetMixin, UpdateView): |  | ||||||
|     """ |  | ||||||
|     A view to add an ingredient |  | ||||||
|     """ |  | ||||||
|     model = Food |  | ||||||
|     template_name = 'food/add_ingredient_form.html' |  | ||||||
|     extra_context = {"title": _("Add the ingredient")} |  | ||||||
|     form_class = AddIngredientForms |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|         context["pk"] = self.kwargs["pk"] |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.instance.creater = self.request.user |  | ||||||
|         food = Food.objects.get(pk=self.kwargs['pk']) |  | ||||||
|         add_ingredient_form = AddIngredientForms(data=self.request.POST) |  | ||||||
|         if food.is_ready: |  | ||||||
|             form.add_error(None, _("The product is already prepared")) |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|         if not add_ingredient_form.is_valid(): |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         # We flip logic ""fully used = not is_active"" |  | ||||||
|         food.is_active = not food.is_active |  | ||||||
|         # Save the aliment and the allergens associed |  | ||||||
|         for transformed_pk in self.request.POST.getlist('ingredient'): |  | ||||||
|             transformed = TransformedFood.objects.get(pk=transformed_pk) |  | ||||||
|             if not transformed.is_ready: |  | ||||||
|                 transformed.ingredient.add(food) |  | ||||||
|                 transformed.update() |  | ||||||
|         food.save() |  | ||||||
|  |  | ||||||
|         return HttpResponseRedirect(self.get_success_url()) |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |  | ||||||
|         return reverse('food:food_list') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BasicFoodUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView): |  | ||||||
|     """ |  | ||||||
|     A view to update a basic food |  | ||||||
|     """ |  | ||||||
|     model = BasicFood |  | ||||||
|     form_class = BasicFoodForms |  | ||||||
|     template_name = 'food/basicfood_form.html' |  | ||||||
|     extra_context = {"title": _("Update an aliment")} |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.instance.creater = self.request.user |  | ||||||
|         basic_food_form = BasicFoodForms(data=self.request.POST) |  | ||||||
|         if not basic_food_form.is_valid(): |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         ans = super().form_valid(form) |  | ||||||
|         form.instance.update() |  | ||||||
|         return ans |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |  | ||||||
|         self.object.refresh_from_db() |  | ||||||
|         return reverse('food:food_view', kwargs={"pk": self.object.pk}) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class FoodView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): |  | ||||||
|     """ |  | ||||||
|     A view to see a food |  | ||||||
|     """ |  | ||||||
|     model = Food |  | ||||||
|     extra_context = {"title": _("Details of:")} |  | ||||||
|     context_object_name = "food" |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|  |  | ||||||
|         context["can_update"] = PermissionBackend.check_perm(self.request, "food.change_food") |  | ||||||
|         context["can_add_ingredient"] = PermissionBackend.check_perm(self.request, "food.change_transformedfood") |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeBasicFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): |  | ||||||
|     ##################################################################### |  | ||||||
|     # TO DO |  | ||||||
|     # - this feature is very pratical for meat or fish, nevertheless we can implement this later |  | ||||||
|     # - fix picture save |  | ||||||
|     # - implement solution crop and convert image (reuse or recode ImageForm from members apps) |  | ||||||
|     ##################################################################### |  | ||||||
|     """ |  | ||||||
|     A view to add a basic food with a qrcode |  | ||||||
|     """ |  | ||||||
|     model = BasicFood |  | ||||||
|     form_class = BasicFoodForms |  | ||||||
|     template_name = 'food/basicfood_form.html' |  | ||||||
|     extra_context = {"title": _("Add a new basic food with QRCode")} |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.instance.creater = self.request.user |  | ||||||
|         basic_food_form = BasicFoodForms(data=self.request.POST) |  | ||||||
|         if not basic_food_form.is_valid(): |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         # Save the aliment and the allergens associed |  | ||||||
|         basic_food = form.save(commit=False) |  | ||||||
|         # We assume the date of labeling and the same as the date of arrival |  | ||||||
|         basic_food.arrival_date = timezone.now() |  | ||||||
|         basic_food.is_ready = False |  | ||||||
|         basic_food.is_active = True |  | ||||||
|         basic_food.was_eaten = False |  | ||||||
|         basic_food._force_save = True |  | ||||||
|         basic_food.save() |  | ||||||
|         basic_food.refresh_from_db() |  | ||||||
|  |  | ||||||
|         qrcode = QRCode() |  | ||||||
|         qrcode.qr_code_number = self.kwargs['slug'] |  | ||||||
|         qrcode.food_container = basic_food |  | ||||||
|         qrcode.save() |  | ||||||
|  |  | ||||||
|         return super().form_valid(form) |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |  | ||||||
|         self.object.refresh_from_db() |  | ||||||
|         return reverse('food:qrcode_view', kwargs={"slug": self.kwargs['slug']}) |  | ||||||
|  |  | ||||||
|     def get_sample_object(self): |  | ||||||
|  |  | ||||||
|         # We choose a club which may work or BDE else |  | ||||||
|         owner_id = 1 |  | ||||||
|         for membership in self.request.user.memberships.all(): |  | ||||||
|             club_id = membership.club.id |  | ||||||
|             food = BasicFood(name="", expiry_date=timezone.now(), owner_id=club_id) |  | ||||||
|             if PermissionBackend.check_perm(self.request, "food.add_basicfood", food): |  | ||||||
|                 owner_id = club_id |  | ||||||
|  |  | ||||||
|         return BasicFood( |  | ||||||
|             name="", |  | ||||||
|             expiry_date=timezone.now(), |  | ||||||
|             owner_id=owner_id, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         # Some field are hidden on create |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|  |  | ||||||
|         form = context['form'] |  | ||||||
|         form.fields['is_active'].widget = HiddenInput() |  | ||||||
|         form.fields['was_eaten'].widget = HiddenInput() |  | ||||||
|  |  | ||||||
|         copy = self.request.GET.get('copy', None) |  | ||||||
|         if copy is not None: |  | ||||||
|             basic = BasicFood.objects.get(pk=copy) |  | ||||||
|             for field in ['date_type', 'expiry_date', 'name', 'owner']: |  | ||||||
|                 form.fields[field].initial = getattr(basic, field) |  | ||||||
|             for field in ['allergens']: |  | ||||||
|                 form.fields[field].initial = getattr(basic, field).all() |  | ||||||
|  |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeCreateView(ProtectQuerysetMixin, ProtectedCreateView): |  | ||||||
|     """ |  | ||||||
|     A view to add a new qrcode |  | ||||||
|     """ |  | ||||||
|     model = QRCode |  | ||||||
|     template_name = 'food/create_qrcode_form.html' |  | ||||||
|     form_class = QRCodeForms |  | ||||||
|     extra_context = {"title": _("Add a new QRCode")} |  | ||||||
|  |  | ||||||
|     def get(self, *args, **kwargs): |  | ||||||
|         qrcode = kwargs["slug"] |  | ||||||
|         if self.model.objects.filter(qr_code_number=qrcode).count() > 0: |  | ||||||
|             return HttpResponseRedirect(reverse("food:qrcode_view", kwargs=kwargs)) |  | ||||||
|         else: |  | ||||||
|             return super().get(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|         context["slug"] = self.kwargs["slug"] |  | ||||||
|  |  | ||||||
|         context["last_basic"] = BasicFood.objects.order_by('-pk').all()[:10] |  | ||||||
|  |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.instance.creater = self.request.user |  | ||||||
|         qrcode_food_form = QRCodeForms(data=self.request.POST) |  | ||||||
|         if not qrcode_food_form.is_valid(): |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         # Save the qrcode |  | ||||||
|         qrcode = form.save(commit=False) |  | ||||||
|         qrcode.qr_code_number = self.kwargs["slug"] |  | ||||||
|         qrcode._force_save = True |  | ||||||
|         qrcode.save() |  | ||||||
|         qrcode.refresh_from_db() |  | ||||||
|  |  | ||||||
|         qrcode.food_container.save() |  | ||||||
|  |  | ||||||
|         return super().form_valid(form) |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |  | ||||||
|         self.object.refresh_from_db() |  | ||||||
|         return reverse('food:qrcode_view', kwargs={"slug": self.kwargs['slug']}) |  | ||||||
|  |  | ||||||
|     def get_sample_object(self): |  | ||||||
|         return QRCode( |  | ||||||
|             qr_code_number=self.kwargs["slug"], |  | ||||||
|             food_container_id=1 |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QRCodeView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): |  | ||||||
|     """ |  | ||||||
|     A view to see a qrcode |  | ||||||
|     """ |  | ||||||
|     model = QRCode |  | ||||||
|     extra_context = {"title": _("QRCode")} |  | ||||||
|     context_object_name = "qrcode" |  | ||||||
|     slug_field = "qr_code_number" |  | ||||||
|  |  | ||||||
|     def get(self, *args, **kwargs): |  | ||||||
|         qrcode = kwargs["slug"] |  | ||||||
|         if self.model.objects.filter(qr_code_number=qrcode).count() > 0: |  | ||||||
|             return super().get(*args, **kwargs) |  | ||||||
|         else: |  | ||||||
|             return HttpResponseRedirect(reverse("food:qrcode_create", kwargs=kwargs)) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|  |  | ||||||
|         qr_code_number = self.kwargs['slug'] |  | ||||||
|         qrcode = self.model.objects.get(qr_code_number=qr_code_number) |  | ||||||
|  |  | ||||||
|         model = qrcode.food_container.polymorphic_ctype.model |  | ||||||
|  |  | ||||||
|         if model == "basicfood": |  | ||||||
|             context["can_update_basic"] = PermissionBackend.check_perm(self.request, "food.change_basicfood") |  | ||||||
|             context["can_view_detail"] = PermissionBackend.check_perm(self.request, "food.view_basicfood") |  | ||||||
|         if model == "transformedfood": |  | ||||||
|             context["can_update_transformed"] = PermissionBackend.check_perm(self.request, "food.change_transformedfood") |  | ||||||
|             context["can_view_detail"] = PermissionBackend.check_perm(self.request, "food.view_transformedfood") |  | ||||||
|         context["can_add_ingredient"] = PermissionBackend.check_perm(self.request, "food.change_transformedfood") |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): |  | ||||||
|     """ |  | ||||||
|     A view to add a tranformed food |  | ||||||
|     """ |  | ||||||
|     model = TransformedFood |  | ||||||
|     template_name = 'food/transformedfood_form.html' |  | ||||||
|     form_class = TransformedFoodForms |  | ||||||
|     extra_context = {"title": _("Add a new meal")} |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.instance.creater = self.request.user |  | ||||||
|         transformed_food_form = TransformedFoodForms(data=self.request.POST) |  | ||||||
|         if not transformed_food_form.is_valid(): |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         # Save the aliment and allergens associated |  | ||||||
|         transformed_food = form.save(commit=False) |  | ||||||
|         transformed_food.expiry_date = transformed_food.creation_date |  | ||||||
|         transformed_food.is_active = True |  | ||||||
|         transformed_food.is_ready = False |  | ||||||
|         transformed_food.was_eaten = False |  | ||||||
|         transformed_food._force_save = True |  | ||||||
|         transformed_food.save() |  | ||||||
|         transformed_food.refresh_from_db() |  | ||||||
|         ans = super().form_valid(form) |  | ||||||
|         transformed_food.update() |  | ||||||
|         return ans |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |  | ||||||
|         self.object.refresh_from_db() |  | ||||||
|         return reverse('food:food_view', kwargs={"pk": self.object.pk}) |  | ||||||
|  |  | ||||||
|     def get_sample_object(self): |  | ||||||
|         # We choose a club which may work or BDE else |  | ||||||
|         owner_id = 1 |  | ||||||
|         for membership in self.request.user.memberships.all(): |  | ||||||
|             club_id = membership.club.id |  | ||||||
|             food = TransformedFood(name="", |  | ||||||
|                                    creation_date=timezone.now(), |  | ||||||
|                                    expiry_date=timezone.now(), |  | ||||||
|                                    owner_id=club_id) |  | ||||||
|             if PermissionBackend.check_perm(self.request, "food.add_transformedfood", food): |  | ||||||
|                 owner_id = club_id |  | ||||||
|                 break |  | ||||||
|  |  | ||||||
|         return TransformedFood( |  | ||||||
|             name="", |  | ||||||
|             owner_id=owner_id, |  | ||||||
|             creation_date=timezone.now(), |  | ||||||
|             expiry_date=timezone.now(), |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|  |  | ||||||
|         # Some field are hidden on create |  | ||||||
|         form = context['form'] |  | ||||||
|         form.fields['is_active'].widget = HiddenInput() |  | ||||||
|         form.fields['is_ready'].widget = HiddenInput() |  | ||||||
|         form.fields['was_eaten'].widget = HiddenInput() |  | ||||||
|         form.fields['shelf_life'].widget = HiddenInput() |  | ||||||
|  |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedFoodUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView): |  | ||||||
|     """ |  | ||||||
|     A view to update transformed product |  | ||||||
|     """ |  | ||||||
|     model = TransformedFood |  | ||||||
|     template_name = 'food/transformedfood_form.html' |  | ||||||
|     form_class = TransformedFoodForms |  | ||||||
|     extra_context = {'title': _('Update a meal')} |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.instance.creater = self.request.user |  | ||||||
|         transformedfood_form = TransformedFoodForms(data=self.request.POST) |  | ||||||
|         if not transformedfood_form.is_valid(): |  | ||||||
|             return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         ans = super().form_valid(form) |  | ||||||
|         form.instance.update() |  | ||||||
|         return ans |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |  | ||||||
|         self.object.refresh_from_db() |  | ||||||
|         return reverse('food:food_view', kwargs={"pk": self.object.pk}) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|         return context |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TransformedListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, ListView): |  | ||||||
|     """ |  | ||||||
|     Displays ready TransformedFood |  | ||||||
|     """ |  | ||||||
|     model = TransformedFood |  | ||||||
|     tables = [TransformedFoodTable, TransformedFoodTable, TransformedFoodTable] |  | ||||||
|     extra_context = {"title": _("Transformed food")} |  | ||||||
|  |  | ||||||
|     def get_queryset(self, **kwargs): |  | ||||||
|         return super().get_queryset(**kwargs).distinct() |  | ||||||
|  |  | ||||||
|     def get_tables(self): |  | ||||||
|         tables = super().get_tables() |  | ||||||
|  |  | ||||||
|         tables[0].prefix = "all-" |  | ||||||
|         tables[1].prefix = "open-" |  | ||||||
|         tables[2].prefix = "served-" |  | ||||||
|         return tables |  | ||||||
|  |  | ||||||
|     def get_tables_data(self): |  | ||||||
|         # first table = all transformed food, second table = free, third = served |  | ||||||
|         return [ |  | ||||||
|             self.get_queryset().order_by("-creation_date"), |  | ||||||
|             TransformedFood.objects.filter(is_ready=True, is_active=True, was_eaten=False, expiry_date__lt=timezone.now()) |  | ||||||
|                                    .filter(PermissionBackend.filter_queryset(self.request, TransformedFood, "view")) |  | ||||||
|                                    .distinct() |  | ||||||
|                                    .order_by("-creation_date"), |  | ||||||
|             TransformedFood.objects.filter(is_ready=True, is_active=True, was_eaten=False, expiry_date__gte=timezone.now()) |  | ||||||
|                                    .filter(PermissionBackend.filter_queryset(self.request, TransformedFood, "view")) |  | ||||||
|                                    .distinct() |  | ||||||
|                                    .order_by("-creation_date") |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         context = super().get_context_data(**kwargs) |  | ||||||
|  |  | ||||||
|         # We choose a club which should work |  | ||||||
|         for membership in self.request.user.memberships.all(): |  | ||||||
|             club_id = membership.club.id |  | ||||||
|             food = TransformedFood( |  | ||||||
|                 name="", |  | ||||||
|                 owner_id=club_id, |  | ||||||
|                 creation_date=timezone.now(), |  | ||||||
|                 expiry_date=timezone.now(), |  | ||||||
|             ) |  | ||||||
|             if PermissionBackend.check_perm(self.request, "food.add_transformedfood", food): |  | ||||||
|                 context['can_create_meal'] = True |  | ||||||
|                 break |  | ||||||
|  |  | ||||||
|         tables = context["tables"] |  | ||||||
|         for name, table in zip(["table", "open", "served"], tables): |  | ||||||
|             context[name] = table |  | ||||||
|         return context |  | ||||||
| @@ -3,7 +3,7 @@ | |||||||
|  |  | ||||||
| import io | import io | ||||||
|  |  | ||||||
| from bootstrap_datepicker_plus.widgets import DatePickerInput | from PIL import Image, ImageSequence | ||||||
| from django import forms | from django import forms | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.auth.forms import AuthenticationForm | from django.contrib.auth.forms import AuthenticationForm | ||||||
| @@ -13,9 +13,8 @@ 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 | from note_kfet.inputs import Autocomplete, AmountInput, DatePickerInput | ||||||
| from permission.models import PermissionMask, Role | from permission.models import PermissionMask, Role | ||||||
| from PIL import Image, ImageSequence |  | ||||||
|  |  | ||||||
| from .models import Profile, Club, Membership | from .models import Profile, Club, Membership | ||||||
|  |  | ||||||
| @@ -33,7 +32,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 | ||||||
|   | |||||||
| @@ -166,7 +166,8 @@ class UserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): | |||||||
|         # Display only the most recent membership |         # Display only the most recent membership | ||||||
|         club_list = club_list.distinct("club__name")\ |         club_list = club_list.distinct("club__name")\ | ||||||
|             if settings.DATABASES["default"]["ENGINE"] == 'django.db.backends.postgresql' else club_list |             if settings.DATABASES["default"]["ENGINE"] == 'django.db.backends.postgresql' else club_list | ||||||
|         membership_table = MembershipTable(data=club_list, prefix='membership-') |         club_list_order_by = self.request.GET.getlist("membership-sort", ("club__name", "-date_start")) | ||||||
|  |         membership_table = MembershipTable(data=club_list, prefix='membership-', order_by=club_list_order_by) | ||||||
|         membership_table.paginate(per_page=10, page=self.request.GET.get("membership-page", 1)) |         membership_table.paginate(per_page=10, page=self.request.GET.get("membership-page", 1)) | ||||||
|         context['club_list'] = membership_table |         context['club_list'] = membership_table | ||||||
|  |  | ||||||
| @@ -476,7 +477,8 @@ class ClubDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): | |||||||
|         managers = Membership.objects.filter(club=self.object, roles__name="Bureau de club", |         managers = Membership.objects.filter(club=self.object, roles__name="Bureau de club", | ||||||
|                                              date_start__lte=date.today(), date_end__gte=date.today())\ |                                              date_start__lte=date.today(), date_end__gte=date.today())\ | ||||||
|             .order_by('user__last_name').all() |             .order_by('user__last_name').all() | ||||||
|         context["managers"] = ClubManagerTable(data=managers, prefix="managers-") |         managers_order_by = self.request.GET.getlist("managers-sort", ('user__last_name')) | ||||||
|  |         context["managers"] = ClubManagerTable(data=managers, prefix="managers-", order_by=managers_order_by) | ||||||
|         # transaction history |         # transaction history | ||||||
|         club_transactions = Transaction.objects.all().filter(Q(source=club.note) | Q(destination=club.note))\ |         club_transactions = Transaction.objects.all().filter(Q(source=club.note) | Q(destination=club.note))\ | ||||||
|             .filter(PermissionBackend.filter_queryset(self.request, Transaction, "view"))\ |             .filter(PermissionBackend.filter_queryset(self.request, Transaction, "view"))\ | ||||||
| @@ -494,7 +496,8 @@ class ClubDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): | |||||||
|         club_member = club_member.distinct("user__username")\ |         club_member = club_member.distinct("user__username")\ | ||||||
|             if settings.DATABASES["default"]["ENGINE"] == 'django.db.backends.postgresql' else club_member |             if settings.DATABASES["default"]["ENGINE"] == 'django.db.backends.postgresql' else club_member | ||||||
|  |  | ||||||
|         membership_table = MembershipTable(data=club_member, prefix="membership-") |         membership_order_by = self.request.GET.getlist("membership-sort", ("user__username", "-date_start")) | ||||||
|  |         membership_table = MembershipTable(data=club_member, prefix="membership-", order_by=membership_order_by) | ||||||
|         membership_table.paginate(per_page=5, page=self.request.GET.get('membership-page', 1)) |         membership_table.paginate(per_page=5, page=self.request.GET.get('membership-page', 1)) | ||||||
|         context['member_list'] = membership_table |         context['member_list'] = membership_table | ||||||
|  |  | ||||||
|   | |||||||
| @@ -183,10 +183,19 @@ 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( | ||||||
|                 Q(**{f'name{suffix}': alias_prefix + alias}) |                 **{f'name{suffix}': alias_prefix + alias} | ||||||
|                 | Q(**{f'normalized_name{suffix}': alias_prefix + Alias.normalize(alias)}) |             ).union( | ||||||
|                 | Q(**{f'normalized_name{suffix}': alias_prefix + alias.lower()}) |                 queryset.filter( | ||||||
|             ) |                     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") | ||||||
|   | |||||||
| @@ -2,13 +2,12 @@ | |||||||
| # 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 | from note_kfet.inputs import Autocomplete, AmountInput, DateTimePickerInput | ||||||
|  |  | ||||||
| from .models import TransactionTemplate, NoteClub, Alias | from .models import TransactionTemplate, NoteClub, Alias | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,25 +0,0 @@ | |||||||
| # Generated by Django 4.2.15 on 2024-08-28 08:00 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
| import django.db.models.deletion |  | ||||||
|  |  | ||||||
|  |  | ||||||
| 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 %} | ||||||
|                 {% if value is not False %}{{ name }}{% if value is not True %}="{{ value|stringformat:'s' }}"{% endif %}{% endif %} |                 {% ifnotequal value False %}{{ name }}{% ifnotequal value True %}="{{ value|stringformat:'s' }}"{% endifnotequal %}{% endifnotequal %} | ||||||
|             {% endfor %}> |             {% endfor %}> | ||||||
|     <div class="input-group-append"> |     <div class="input-group-append"> | ||||||
|         <span class="input-group-text">€</span> |         <span class="input-group-text">€</span> | ||||||
|   | |||||||
| @@ -3167,12 +3167,12 @@ | |||||||
| 				"activity", | 				"activity", | ||||||
| 				"entry" | 				"entry" | ||||||
| 			], | 			], | ||||||
| 			"query": "{\"activity__opener__in\": [\"user\", \"note\", \"activity_responsible\", [\"all\"]]}", | 			"query": "{\"activity__opener__in\": [\"user\", \"note\", \"activity_responsible\", [\"all\"]], \"activity__open\": true, \"activity__activity_type__manage_entries\":true}", | ||||||
| 			"type": "view", | 			"type": "view", | ||||||
| 			"mask": 2, | 			"mask": 2, | ||||||
| 			"field": "", | 			"field": "", | ||||||
| 			"permanent": false, | 			"permanent": false, | ||||||
| 			"description": "Voir les entrées des activités dont l'utilisateur⋅rice est ouvreur⋅se" | 			"description": "Voir les entrées des activités ouvertes dont l'utilisateur⋅rice est ouvreur⋅se" | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ | 	{ | ||||||
| @@ -3183,12 +3183,12 @@ | |||||||
| 				"activity", | 				"activity", | ||||||
| 				"guest" | 				"guest" | ||||||
| 			], | 			], | ||||||
| 			"query": "{\"activity__opener__in\": [\"user\", \"note\", \"activity_responsible\", [\"all\"]]}", | 			"query": "{\"activity__pk__in\": [\"user\", \"note\", \"activity_responsible\", [\"all\"]], \"activity__open\": true, \"activity__activity_type__manage_entries\":true}", | ||||||
| 			"type": "view", | 			"type": "view", | ||||||
| 			"mask": 2, | 			"mask": 2, | ||||||
| 			"field": "", | 			"field": "", | ||||||
| 			"permanent": false, | 			"permanent": false, | ||||||
| 			"description": "Voir les invité⋅es des activités dont l'utilisateur⋅rice est ouvreur⋅se" | 			"description": "Voir les invité⋅es des activités ouvertes dont l'utilisateur⋅rice est ouvreur⋅se" | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ | 	{ | ||||||
| @@ -3304,454 +3304,6 @@ | |||||||
| 			"description": "Voir le tableau des ouvreur⋅ses" | 			"description": "Voir le tableau des ouvreur⋅ses" | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 211, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir tout les plats" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 212, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"owner\": [\"club\"]}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir tout les plats de son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 213, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_ready\": true, \"is_active\": true, \"was_eaten\": false}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 1, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir les plats préparés actifs servis" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 214, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"qrcode" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "add", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Initialiser un QR code de traçabilité" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 215, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"owner\": [\"club\"]}", |  | ||||||
| 			"type": "add", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Créer un nouvel ingrédient pour son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 216, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "add", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Créer un nouvel ingrédient" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 217, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir toute la bouffe" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 218, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir toute la bouffe active" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 219, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true, \"owner\": [\"club\"]}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir la bouffe active de son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 220, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Modifier de la bouffe" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 221, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true, \"was_eaten\": false}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "allergens", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Modifier les allergènes de la bouffe existante" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 222, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true, \"was_eaten\": false, \"owner\": [\"club\"]}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "allergens", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Modifier les allergènes de la bouffe appartenant à son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 223, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "add", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Créer un plat" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 224, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"owner\": [\"club\"]}", |  | ||||||
| 			"type": "add", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Créer un plat pour son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 225, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Modifier tout les plats" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 226, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "was_eaten", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Indiquer si un plat a été mangé" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 227, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true, \"owner\": [\"club\"]}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "is_ready", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Indiquer si un plat de son club est prêt" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 228, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "is_active", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Archiver un plat" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 229, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"basicfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "is_active", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Archiver de la bouffe" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 230, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir tout les plats actifs" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 231, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"qrcode" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir tous les QR codes" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 232, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"qrcode" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"food_container__is_active\": true}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir tous les QR codes actifs" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 233, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"qrcode" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"food_container__owner\": [\"club\"], \"food_container__is_active\": true}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir tous les QR codes actifs de son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk" : 234, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"transformedfood" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"owner\": [\"club\"], \"is_active\": true}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "ingredients", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Changer les ingrédients d'un plat actif de son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 235, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"food" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir bouffe" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 236, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"food" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir bouffe active" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 237, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"food" |  | ||||||
| 			], |  | ||||||
| 			"query": "{\"is_active\": true, \"owner\": [\"club\"]}", |  | ||||||
| 			"type": "view", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Voir bouffe active de son club" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"model": "permission.permission", |  | ||||||
| 		"pk": 238, |  | ||||||
| 		"fields": { |  | ||||||
| 			"model": [ |  | ||||||
| 				"food", |  | ||||||
| 				"food" |  | ||||||
| 			], |  | ||||||
| 			"query": "{}", |  | ||||||
| 			"type": "change", |  | ||||||
| 			"mask": 3, |  | ||||||
| 			"field": "", |  | ||||||
| 			"permanent": false, |  | ||||||
| 			"description": "Modifier bouffe" |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ | 	{ | ||||||
| 		"model": "permission.role", | 		"model": "permission.role", | ||||||
| 		"pk": 1, | 		"pk": 1, | ||||||
| @@ -3839,8 +3391,7 @@ | |||||||
| 				157, | 				157, | ||||||
| 				158, | 				158, | ||||||
| 				159, | 				159, | ||||||
| 				160, | 				160 | ||||||
| 				213 |  | ||||||
| 			] | 			] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| @@ -3866,17 +3417,7 @@ | |||||||
| 				49, | 				49, | ||||||
| 				50, | 				50, | ||||||
| 				141, | 				141, | ||||||
| 				169, | 				169 | ||||||
| 				212, |  | ||||||
| 				214, |  | ||||||
| 				215, |  | ||||||
| 				219, |  | ||||||
| 				222, |  | ||||||
| 				224, |  | ||||||
| 				227, |  | ||||||
| 				233, |  | ||||||
| 				234, |  | ||||||
| 				237 |  | ||||||
| 			] | 			] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| @@ -4050,21 +3591,7 @@ | |||||||
| 				166, | 				166, | ||||||
| 				167, | 				167, | ||||||
| 				168, | 				168, | ||||||
| 				182, | 				182 | ||||||
| 				212, |  | ||||||
| 				214, |  | ||||||
| 				215, |  | ||||||
| 				218, |  | ||||||
| 				221, |  | ||||||
| 				224, |  | ||||||
| 				226, |  | ||||||
| 				227, |  | ||||||
| 				228, |  | ||||||
| 				229, |  | ||||||
| 				230, |  | ||||||
| 				232, |  | ||||||
| 				234, |  | ||||||
| 				236 |  | ||||||
| 			] | 			] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| @@ -4272,8 +3799,7 @@ | |||||||
| 				168, | 				168, | ||||||
| 				176, | 				176, | ||||||
| 				177, | 				177, | ||||||
| 				197, | 				197 | ||||||
| 				211 |  | ||||||
| 			] | 			] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| @@ -4291,27 +3817,6 @@ | |||||||
| 			] | 			] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
| 	{ |  | ||||||
| 		"model": "permission.role", |  | ||||||
| 		"pk": 22, |  | ||||||
| 	        "fields": { |  | ||||||
| 			"for_club": 2, |  | ||||||
| 			"name": "Respo Bouffe", |  | ||||||
| 			"permissions": [ |  | ||||||
| 				137, |  | ||||||
| 				211, |  | ||||||
| 				214, |  | ||||||
| 				216, |  | ||||||
| 				217, |  | ||||||
| 				220, |  | ||||||
| 				223, |  | ||||||
| 				225, |  | ||||||
| 				231, |  | ||||||
| 				235, |  | ||||||
| 				238 |  | ||||||
| 			] |  | ||||||
| 		} |  | ||||||
| 	}, |  | ||||||
| 	{ | 	{ | ||||||
| 		"model": "wei.weirole", | 		"model": "wei.weirole", | ||||||
| 		"pk": 12, | 		"pk": 12, | ||||||
|   | |||||||
| @@ -35,8 +35,6 @@ class PermissionScopes(BaseScopes): | |||||||
|  |  | ||||||
|  |  | ||||||
| class PermissionOAuth2Validator(OAuth2Validator): | class PermissionOAuth2Validator(OAuth2Validator): | ||||||
|     oidc_claim_scope = None  # fix breaking change of django-oauth-toolkit 2.0.0 |  | ||||||
|  |  | ||||||
|     def validate_scopes(self, client_id, scopes, client, request, *args, **kwargs): |     def validate_scopes(self, client_id, scopes, client, request, *args, **kwargs): | ||||||
|         """ |         """ | ||||||
|         User can request as many scope as he wants, including invalid scopes, |         User can request as many scope as he wants, including invalid scopes, | ||||||
|   | |||||||
 Submodule apps/scripts updated: f580f9b9e9...472c9c33ce
									
								
							| @@ -1,19 +0,0 @@ | |||||||
| # Generated by Django 4.2.15 on 2024-08-28 08:00 |  | ||||||
|  |  | ||||||
| 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'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -109,7 +109,7 @@ | |||||||
| \renewcommand{\headrulewidth}{0pt} | \renewcommand{\headrulewidth}{0pt} | ||||||
| \cfoot{ | \cfoot{ | ||||||
|     \small{\MonNom  ~--~ \MonAdresseRue ~ \MonAdresseVille ~--~ Téléphone : +33(0)7 78 17 22 34\newline |     \small{\MonNom  ~--~ \MonAdresseRue ~ \MonAdresseVille ~--~ Téléphone : +33(0)7 78 17 22 34\newline | ||||||
|      E-mail : tresorerie.bde@lists.crans.org ~--~ Numéro SIRET : 399 485 838 00029 |      Site web : bde.ens-cachan.fr ~--~ E-mail : tresorerie.bde@lists.crans.org \newline Numéro SIRET : 399 485 838 00029 | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,14 +1,13 @@ | |||||||
| # 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, Autocomplete, ColorWidget | from note_kfet.inputs import AmountInput, DatePickerInput, Autocomplete, ColorWidget | ||||||
|  |  | ||||||
| from ..models import WEIClub, WEIRegistration, Bus, BusTeam, WEIMembership, WEIRole | from ..models import WEIClub, WEIRegistration, Bus, BusTeam, WEIMembership, WEIRole | ||||||
|  |  | ||||||
| @@ -81,11 +80,6 @@ class WEIChooseBusForm(forms.Form): | |||||||
|  |  | ||||||
|  |  | ||||||
| class WEIMembershipForm(forms.ModelForm): | class WEIMembershipForm(forms.ModelForm): | ||||||
|     caution_check = forms.BooleanField( |  | ||||||
|         required=False, |  | ||||||
|         label=_("Caution check given"), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     roles = forms.ModelMultipleChoiceField( |     roles = forms.ModelMultipleChoiceField( | ||||||
|         queryset=WEIRole.objects, |         queryset=WEIRole.objects, | ||||||
|         label=_("WEI Roles"), |         label=_("WEI Roles"), | ||||||
| @@ -154,7 +148,6 @@ class WEIMembership1AForm(WEIMembershipForm): | |||||||
|     """ |     """ | ||||||
|     Used to confirm registrations of first year members without choosing a bus now. |     Used to confirm registrations of first year members without choosing a bus now. | ||||||
|     """ |     """ | ||||||
|     caution_check = None |  | ||||||
|     roles = None |     roles = None | ||||||
|  |  | ||||||
|     def clean(self): |     def clean(self): | ||||||
|   | |||||||
| @@ -2,11 +2,11 @@ | |||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm | from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm | ||||||
| from .wei2024 import WEISurvey2024 | from .wei2023 import WEISurvey2023 | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = [ | __all__ = [ | ||||||
|     'WEISurvey', 'WEISurveyInformation', 'WEISurveyAlgorithm', 'CurrentSurvey', |     'WEISurvey', 'WEISurveyInformation', 'WEISurveyAlgorithm', 'CurrentSurvey', | ||||||
| ] | ] | ||||||
|  |  | ||||||
| CurrentSurvey = WEISurvey2024 | CurrentSurvey = WEISurvey2023 | ||||||
|   | |||||||
| @@ -1,381 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| from functools import lru_cache |  | ||||||
|  |  | ||||||
| from django import forms |  | ||||||
| from django.utils.safestring import mark_safe |  | ||||||
| from django.db import transaction |  | ||||||
| from django.db.models import Q |  | ||||||
|  |  | ||||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm, WEIBusInformation |  | ||||||
| from ...models import WEIMembership |  | ||||||
|  |  | ||||||
|  |  | ||||||
| buses_descr = [ |  | ||||||
|     [ |  | ||||||
|         "Magi[Kar]p 🐙🎮🎲", "#ef5568", 1, |  | ||||||
|         """Vous l'aurez compris au nom du bus, l'ambiance est aux jeux et à la culture geek ! Ici, vous trouverez une ambiance |  | ||||||
|         calme avec une bonne dose d'autodérision et de second degré. Que vous ayez besoin de beaucoup dormir pour tenir la soirée |  | ||||||
|         du lendemain, ou que vous souhaitiez faire nuit blanche pour jouer toute la nuit, vous pouvez nous rejoindre. Votre voix |  | ||||||
|         n'y survivra peut-être pas à force de chanter. PS : les meilleurs cocktails du WEI sont chez nous, à déguster, pas à |  | ||||||
|         siphonner !""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "Va[car]me 🎷🍎🔊", "#fd7a28", 3, |  | ||||||
|         """Ici, c'est le bus du bruit. Si vous voulez réveiller les autres bus en musique, apprendre de merveilleuses |  | ||||||
|         mélodies au kazoo tout le week-end, ou simplement profiter d'une bonne ambiance musicale, le BDA et la |  | ||||||
|         F[ENS]foire sont là pour vous. Vous pourrez également goûter au célèbre cocktail de la fanfare, concocté |  | ||||||
|         pour l'occasion par les tout nouveaux "meilleurs artisans v*********** de France" ! Alors que vous soyez artiste |  | ||||||
|         dans l'âme ou que vous souhaitiez juste faire le plus grand Vacarme, rejoignez-nous !""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "[Kar]aïbes 🏝️🏴☠️🥥", "#a5cfdd", 3, |  | ||||||
|         """Ahoy, explorateurs du WEI ! Le bus Karaibes t’invite à une traversée sous les tropiques, où l’ambiance est |  | ||||||
|         toujours au beau fixe ! ☀️🍹 Ici, c’est soleil, rhum, et bonne humeur assurée : une atmosphère de vacances où |  | ||||||
|         l’on se laisse porter par la chaleur humaine et la fête. Que tu sois un pirate en quête de sensations fortes ou |  | ||||||
|         un amateur de chill avec un cocktail à la main, tu seras à ta place dans notre bus. Les soirées seront marquées |  | ||||||
|         par des rythmes tropicaux qui te feront vibrer jusqu’à l’aube. Prêt à embarquer pour une aventure inoubliable |  | ||||||
|         avec les meilleurs matelots du WEI ? On t’attend sur le pont du Karaibes pour lever l’ancre ensemble !""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "[Kar]di [Bus] 🎙️💅", "#e46398", 2.5, |  | ||||||
|         """Bienvenue à bord du Kardi Bus, la seul, l’unique, l’inimitable pépite de ce weekend d’intégration ! Inspiré par les |  | ||||||
|         icônes suprêmes de la pop culture telles les Bratz, les Winx et autres Mean Girls, notre bus est un sanctuaire de style, |  | ||||||
|         d’audace et de pur plaisir. A nos cotés attends toi à siroter tes meilleurs Cosmo, sex on the Beach et autres cocktails |  | ||||||
|         de maxi pétasse tout en papotant entre copains copines ! Si tu rejoins le Kardi Bus, tu entres dans un monde où tu |  | ||||||
|         pourras te déhancher sur du Beyoncé, Britney, Aya et autres reines de la pop ! À très vite, les futures stars du Kardi |  | ||||||
|         Bus !""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "Sparta[bus] 🐺🐒🏉", "#ebdac2", 5, |  | ||||||
|         """Dans notre bus, on vous donne un avant goût des plus grandes assos de l'ENS : les Kyottes et l'Aspique (clubs de rugby |  | ||||||
|         féminin et masculin, mais pas que). Bien entendu, qui dit rugby dit les copaings, le pastaga et la Pena Bayona, mais vous |  | ||||||
|         verrez par vous même qu'on est ouvert⋅e à toutes propositions quand il s'agit de faire la fête. Pour les casse-cous comme |  | ||||||
|         pour les plus calmes, vous trouverez au bus Aspique-Kyottes les 2A+ qui vous feront kiffer votre WEI.""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "Zanzo[Bus] 🤯🚸🐒", "#FFFF", 3, |  | ||||||
|         """Dans un entre-trois bien senti entre zinzinerie, enfance et vieillerie, le Zanzo[BUS] est un concentré de fun mêlé à |  | ||||||
|         de la dinguerie à gogo. N'hésitez plus et rejoignez-nous pour un WEI toujours plus déjanté !""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "Bran[Kar] 🍹🥳", "#6da1ac", 4, |  | ||||||
|         """Si vous ne connaissez pas le Bran[Kar], c’est comme une grande famille qui fait un apéro, qui se bourre un peu la |  | ||||||
|         gueule en discutant des heures autour d’une table remplie de bouffe et de super bons cocktails (la plupart des |  | ||||||
|         barmen/barwomen du bus sont les barmans de Shakens), sauf qu’on est un bus du Wei (vous comprendrez bien le nom de notre |  | ||||||
|         bus en voyant l’état de certain·e·s). Il nous arrive de faire quelques conneries, mais surtout de jouer au Bière-pong en |  | ||||||
|         musique !""", |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "Techno [kar]ade 🔊🚩", "#8065a3", 3, |  | ||||||
|         """Avis à tous·tes les gauchos, amoureux·ses de la fête et des manifs : le Techno [kar]ade vous ouvre grand ses bras pour |  | ||||||
|         finir en beauté votre première inté. Préparez-vous à vous abreuver de cocktails (savamment élaborés) à la vibration d’un |  | ||||||
|         système son fabriqué pour l’occasion. Des sets technos à « Mon père était tellement de gauche » en passant par « Female |  | ||||||
|         Body », le car accueillant les meilleures DJs du plateau saura animer le trajet aussi bien que les soirées. Si alcool et |  | ||||||
|         musique seront au rendez-vous, les maîtres mots sont sécurité et inclusivité. Qui que vous soyez et quelle que soit votre |  | ||||||
|         manière de vous amuser, notre objectif est que vous vous sentiez à l’aise pour rencontrer au mieux les 1A, les 2A et les |  | ||||||
|         (nombreux⋅ses) 3A+ qui auront répondu à l’appel. Bref, rejoignez-nous, on est super cools :)""" |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|         "[Bus]ka-P 🥇🍻🎤", "#7c4768", 4.5, |  | ||||||
|         """Booska-p, c’est le « site N°1 du Rap français ». Le [Bus]ka-p ? Le bus N°1 sur l’ambiance au WEI. Les nuits vont être |  | ||||||
|         courtes, les cocktails vont couler à flots : tout sera réuni pour vivre un week-end dont tu te souviendras toute ta vie. |  | ||||||
|         Au programme pas un seul temps mort et un maximum de rencontres pour bien commencer ta première année à l’ENS. Et bien |  | ||||||
|         entendu, le tout accompagné des meilleurs sons, de Jul à Aya, en passant par ABBA et Sexion d’Assaut. Bref, si tu veux |  | ||||||
|         vivre un WEI d’anthologie et faire la fête, de jour comme de nuit, nous t’accueillons avec plaisir !""", |  | ||||||
|     ], |  | ||||||
| ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def print_bus(i): |  | ||||||
|     return f"""<h1 style="color:{buses_descr[i][1]};-webkit-text-stroke: 2px black;font-size: 50px;">{buses_descr[i][0]}</h1><br> |  | ||||||
|     <b>Alcoolomètre : {buses_descr[i][2]} / 5 🍻</b><br><br>{buses_descr[i][3]}<br>""" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def print_all_buses(): |  | ||||||
|     liste = [print_bus(i) for i in range(len(buses_descr))] |  | ||||||
|     return "<br><br><br><br>".join(liste) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_number_comment(i): |  | ||||||
|     if i == 1: |  | ||||||
|         return "Même pas en rêve" |  | ||||||
|     elif i == 2: |  | ||||||
|         return "Pas envie" |  | ||||||
|     elif i == 3: |  | ||||||
|         return "Mouais..." |  | ||||||
|     elif i == 4: |  | ||||||
|         return "Pourquoi pas !" |  | ||||||
|     elif i == 5: |  | ||||||
|         return "Ce bus ou rien !!!" |  | ||||||
|     else: |  | ||||||
|         return "" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| WORDS = { |  | ||||||
|     "recap": |  | ||||||
|         [ |  | ||||||
|             """<b>Chèr⋅e 1A, te voilà arrivé⋅e au moment fatidique du choix de ton bus !<br><br><br> |  | ||||||
|             Ton bus est constitué des gens avec qui tu passeras la majorité de ton temps : que ce soit le voyage d'aller et de |  | ||||||
|             retour et les différentes activité qu'ils pourront te proposer tout au long du WEI donc choisis le bien ! |  | ||||||
|             <br><br>Tu trouveras ci-dessous la liste de tous les bus ainsi qu'une description détaillée de ces derniers. |  | ||||||
|             Prends ton temps pour étudier chacun d'eux et quand tu te sens prêt⋅e, appuie sur le bouton « J'ai pris connaissance |  | ||||||
|             des bus » pour continuer |  | ||||||
|             <br>(pas besoin d'apprendre par cœur chaque bus, la description de chaque bus te sera rappeler avant de lui attribuer |  | ||||||
|             une note !)</b><br><br><br>""" + print_all_buses(), |  | ||||||
|             { |  | ||||||
|                 "1": "J'ai pris connaissance des différents bus et me sent fin prêt à choisir celui qui me convient le mieux !", |  | ||||||
|             } |  | ||||||
|         ] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| WORDS.update({ |  | ||||||
|     f"bus{id}": [print_bus(id), {i: f"{get_number_comment(i)}   ({i}/5)" for i in range(1, 5 + 1)}] for id in range(len(buses_descr)) |  | ||||||
| }) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyForm2024(forms.Form): |  | ||||||
|     """ |  | ||||||
|     Survey form for the year 2024. |  | ||||||
|     Members score the different buses, from which we calculate the best associated bus. |  | ||||||
|     """ |  | ||||||
|     def set_registration(self, registration): |  | ||||||
|         """ |  | ||||||
|         Filter the bus selector with the buses of the current WEI. |  | ||||||
|         """ |  | ||||||
|         information = WEISurveyInformation2024(registration) |  | ||||||
|  |  | ||||||
|         question = information.questions[information.step] |  | ||||||
|         self.fields[question] = forms.ChoiceField( |  | ||||||
|             label=mark_safe(WORDS[question][0]), |  | ||||||
|             widget=forms.RadioSelect(), |  | ||||||
|         ) |  | ||||||
|         answers = [(answer, WORDS[question][1][answer]) for answer in WORDS[question][1]] |  | ||||||
|         self.fields[question].choices = answers |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEIBusInformation2024(WEIBusInformation): |  | ||||||
|     """ |  | ||||||
|     For each question, the bus has ordered answers |  | ||||||
|     """ |  | ||||||
|     scores: dict |  | ||||||
|  |  | ||||||
|     def __init__(self, bus): |  | ||||||
|         self.scores = {} |  | ||||||
|         for question in WORDS: |  | ||||||
|             self.scores[question] = [] |  | ||||||
|         super().__init__(bus) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyInformation2024(WEISurveyInformation): |  | ||||||
|     """ |  | ||||||
|     We store the id of the selected bus. We store only the name, but is not used in the selection: |  | ||||||
|     that's only for humans that try to read data. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     step = 0 |  | ||||||
|     questions = list(WORDS.keys()) |  | ||||||
|  |  | ||||||
|     def __init__(self, registration): |  | ||||||
|         for question in WORDS: |  | ||||||
|             setattr(self, str(question), None) |  | ||||||
|         super().__init__(registration) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurvey2024(WEISurvey): |  | ||||||
|     """ |  | ||||||
|     Survey for the year 2024. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_year(cls): |  | ||||||
|         return 2024 |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_survey_information_class(cls): |  | ||||||
|         return WEISurveyInformation2024 |  | ||||||
|  |  | ||||||
|     def get_form_class(self): |  | ||||||
|         return WEISurveyForm2024 |  | ||||||
|  |  | ||||||
|     def update_form(self, form): |  | ||||||
|         """ |  | ||||||
|         Filter the bus selector with the buses of the WEI. |  | ||||||
|         """ |  | ||||||
|         form.set_registration(self.registration) |  | ||||||
|  |  | ||||||
|     @transaction.atomic |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         self.information.step += 1 |  | ||||||
|         for question in WORDS: |  | ||||||
|             if question in form.cleaned_data: |  | ||||||
|                 answer = form.cleaned_data[question] |  | ||||||
|                 setattr(self.information, question, answer) |  | ||||||
|         self.save() |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_algorithm_class(cls): |  | ||||||
|         return WEISurveyAlgorithm2024 |  | ||||||
|  |  | ||||||
|     def is_complete(self) -> bool: |  | ||||||
|         """ |  | ||||||
|         The survey is complete once the bus is chosen. |  | ||||||
|         """ |  | ||||||
|         for question in WORDS: |  | ||||||
|             if not getattr(self.information, question): |  | ||||||
|                 return False |  | ||||||
|         return True |  | ||||||
|  |  | ||||||
|     @lru_cache() |  | ||||||
|     def score(self, bus): |  | ||||||
|         if not self.is_complete(): |  | ||||||
|             raise ValueError("Survey is not ended, can't calculate score") |  | ||||||
|  |  | ||||||
|         bus_info = self.get_algorithm_class().get_bus_information(bus) |  | ||||||
|         # Score is the given score by the bus subtracted to the mid-score of the buses. |  | ||||||
|         s = 0 |  | ||||||
|         for question in WORDS: |  | ||||||
|             s += bus_info.scores[question][str(getattr(self.information, question))] |  | ||||||
|         return s |  | ||||||
|  |  | ||||||
|     @lru_cache() |  | ||||||
|     def scores_per_bus(self): |  | ||||||
|         return {bus: self.score(bus) for bus in self.get_algorithm_class().get_buses()} |  | ||||||
|  |  | ||||||
|     @lru_cache() |  | ||||||
|     def ordered_buses(self): |  | ||||||
|         values = list(self.scores_per_bus().items()) |  | ||||||
|         values.sort(key=lambda item: -item[1]) |  | ||||||
|         return values |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def clear_cache(cls): |  | ||||||
|         return super().clear_cache() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class WEISurveyAlgorithm2024(WEISurveyAlgorithm): |  | ||||||
|     """ |  | ||||||
|     The algorithm class for the year 2024. |  | ||||||
|     We use Gale-Shapley algorithm to attribute 1y students into buses. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_survey_class(cls): |  | ||||||
|         return WEISurvey2024 |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def get_bus_information_class(cls): |  | ||||||
|         return WEIBusInformation2024 |  | ||||||
|  |  | ||||||
|     def run_algorithm(self, display_tqdm=False): |  | ||||||
|         """ |  | ||||||
|         Gale-Shapley algorithm implementation. |  | ||||||
|         We modify it to allow buses to have multiple "weddings". |  | ||||||
|         """ |  | ||||||
|         surveys = list(self.get_survey_class()(r) for r in self.get_registrations())  # All surveys |  | ||||||
|         surveys = [s for s in surveys if s.is_complete()]  # Don't consider invalid surveys |  | ||||||
|         # Don't manage hardcoded people |  | ||||||
|         # surveys = [s for s in surveys if s.bus_id != None] |  | ||||||
|         # surveys = [s for s in surveys if not hasattr(s.information, 'hardcoded') or not s.information.hardcoded] |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         # surveys = [s for s in surveys if s.registration.user_id in free_users] |  | ||||||
|  |  | ||||||
|         # hardcoded_first_year_mb = WEIMembership.objects.filter(bus != None,registration__first_year=True) |  | ||||||
|         # hardcoded_first_year = hardcoded_first_year_mb.values_list('user__id', 'bus__id') |  | ||||||
|  |  | ||||||
|         hardcoded_first_year_mb = WEIMembership.objects.filter(registration__first_year=True) |  | ||||||
|         hardcoded_first_year = {mb.user.id if mb.bus else None: mb.bus.id if mb.bus else None for mb in hardcoded_first_year_mb} |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         # Reset previous algorithm run |  | ||||||
|         for survey in surveys: |  | ||||||
|             survey.free() |  | ||||||
|             if survey.registration.user_id in hardcoded_first_year.keys(): |  | ||||||
|                 survey.select_bus(hardcoded_first_year[s.registration.user_id]) |  | ||||||
|             survey.save() |  | ||||||
|  |  | ||||||
|  |  | ||||||
|         non_men = [s for s in surveys if s.registration.gender != 'male'] |  | ||||||
|         men = [s for s in surveys if s.registration.gender == 'male'] |  | ||||||
|  |  | ||||||
|         quotas = {} |  | ||||||
|         registrations = self.get_registrations() |  | ||||||
|         non_men_total = registrations.filter(~Q(gender='male')).count() |  | ||||||
|         for bus in self.get_buses(): |  | ||||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() |  | ||||||
|             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) |  | ||||||
|             quotas[bus] = 4 + int(non_men_total / registrations.count() * free_seats) |  | ||||||
|  |  | ||||||
|         tqdm_obj = None |  | ||||||
|         if display_tqdm: |  | ||||||
|             from tqdm import tqdm |  | ||||||
|             tqdm_obj = tqdm(total=len(non_men), desc="Non-hommes") |  | ||||||
|  |  | ||||||
|         # Repartition for non men people first |  | ||||||
|         self.make_repartition(non_men, quotas, tqdm_obj=tqdm_obj) |  | ||||||
|  |  | ||||||
|         quotas = {} |  | ||||||
|         for bus in self.get_buses(): |  | ||||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() |  | ||||||
|             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) |  | ||||||
|             quotas[bus] = free_seats |  | ||||||
|  |  | ||||||
|         if display_tqdm: |  | ||||||
|             tqdm_obj.close() |  | ||||||
|  |  | ||||||
|             from tqdm import tqdm |  | ||||||
|             tqdm_obj = tqdm(total=len(men), desc="Hommes") |  | ||||||
|  |  | ||||||
|         self.make_repartition(men, quotas, tqdm_obj=tqdm_obj) |  | ||||||
|  |  | ||||||
|         if display_tqdm: |  | ||||||
|             tqdm_obj.close() |  | ||||||
|  |  | ||||||
|         # Clear cache information after running algorithm |  | ||||||
|         WEISurvey2024.clear_cache() |  | ||||||
|  |  | ||||||
|     def make_repartition(self, surveys, quotas=None, tqdm_obj=None): |  | ||||||
|         free_surveys = surveys.copy()  # Remaining surveys |  | ||||||
|         while free_surveys:  # Some students are not affected |  | ||||||
|             survey = free_surveys[0] |  | ||||||
|             buses = survey.ordered_buses()  # Preferences of the student |  | ||||||
|             for bus, current_score in buses: |  | ||||||
|                 if self.get_bus_information(bus).has_free_seats(surveys, quotas): |  | ||||||
|                     # Selected bus has free places. Put student in the bus |  | ||||||
|                     survey.select_bus(bus) |  | ||||||
|                     survey.save() |  | ||||||
|                     free_surveys.remove(survey) |  | ||||||
|                     break |  | ||||||
|                 else: |  | ||||||
|                     # Current bus has not enough places. Remove the least preferred student from the bus if existing |  | ||||||
|                     least_preferred_survey = None |  | ||||||
|                     least_score = -1 |  | ||||||
|                     # Find the least student in the bus that has a lower score than the current student |  | ||||||
|                     for survey2 in surveys: |  | ||||||
|                         if not survey2.information.valid or survey2.information.get_selected_bus() != bus: |  | ||||||
|                             continue |  | ||||||
|                         score2 = survey2.score(bus) |  | ||||||
|                         if current_score <= score2:  # Ignore better students |  | ||||||
|                             continue |  | ||||||
|                         if least_preferred_survey is None or score2 < least_score: |  | ||||||
|                             least_preferred_survey = survey2 |  | ||||||
|                             least_score = score2 |  | ||||||
|  |  | ||||||
|                     if least_preferred_survey is not None: |  | ||||||
|                         # Remove the least student from the bus and put the current student in. |  | ||||||
|                         # If it does not exist, choose the next bus. |  | ||||||
|                         least_preferred_survey.free() |  | ||||||
|                         least_preferred_survey.save() |  | ||||||
|                         free_surveys.append(least_preferred_survey) |  | ||||||
|                         survey.select_bus(bus) |  | ||||||
|                         survey.save() |  | ||||||
|                         free_surveys.remove(survey) |  | ||||||
|                         break |  | ||||||
|             else: |  | ||||||
|                 raise ValueError(f"User {survey.registration.user} has no free seat") |  | ||||||
|  |  | ||||||
|             if tqdm_obj is not None: |  | ||||||
|                 tqdm_obj.n = len(surveys) - len(free_surveys) |  | ||||||
|                 tqdm_obj.refresh() |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| # Generated by Django 4.2.15 on 2024-08-28 20:47 |  | ||||||
|  |  | ||||||
| from django.db import migrations, models |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('wei', '0008_auto_20240111_1545'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.AddField( |  | ||||||
|             model_name='weiregistration', |  | ||||||
|             name='specific_diet', |  | ||||||
|             field=models.TextField(blank=True, default='', verbose_name='specific diet'), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| # Generated by Django 4.2.15 on 2024-08-29 20:15 |  | ||||||
|  |  | ||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|  |  | ||||||
|     dependencies = [ |  | ||||||
|         ('wei', '0009_weiregistration_specific_diet'), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.RemoveField( |  | ||||||
|             model_name='weiregistration', |  | ||||||
|             name='specific_diet', |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -12,7 +12,7 @@ | |||||||
|         <div class="card-body"> |         <div class="card-body"> | ||||||
|             {% render_table bus_repartition_table %} |             {% render_table bus_repartition_table %} | ||||||
|             <hr> |             <hr> | ||||||
|             <a href="{% url 'wei:wei_bus_1A_next' pk=club.pk %}" class="btn btn-block btn-success">{% trans "Start attribution !" %}</a> |             <a href="{% url 'wei:wei_bus_1A_next' pk=club.pk %}" class="btn btn-block btn-success">{% trans "Start attribution!" %}</a> | ||||||
|             <hr> |             <hr> | ||||||
|             {% render_table table %} |             {% render_table table %} | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ | |||||||
|                 <dt class="col-xl-6">{% trans 'department'|capfirst %}</dt> |                 <dt class="col-xl-6">{% trans 'department'|capfirst %}</dt> | ||||||
|                 <dd class="col-xl-6">{{ object.user.profile.get_department_display }}</dd> |                 <dd class="col-xl-6">{{ object.user.profile.get_department_display }}</dd> | ||||||
|  |  | ||||||
|                 <dt class="col-xl-6">{% trans 'health issues or specific diet'|capfirst %}</dt> |                 <dt class="col-xl-6">{% trans 'health issues'|capfirst %}</dt> | ||||||
|                 <dd class="col-xl-6">{{ object.health_issues|default:"—" }}</dd> |                 <dd class="col-xl-6">{{ object.health_issues|default:"—" }}</dd> | ||||||
|  |  | ||||||
|                 <dt class="col-xl-6">{% trans 'suggested bus'|capfirst %}</dt> |                 <dt class="col-xl-6">{% trans 'suggested bus'|capfirst %}</dt> | ||||||
|   | |||||||
| @@ -64,7 +64,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                 <dt class="col-xl-6">{% trans 'birth date'|capfirst %}</dt> |                 <dt class="col-xl-6">{% trans 'birth date'|capfirst %}</dt> | ||||||
|                 <dd class="col-xl-6">{{ registration.birth_date }}</dd> |                 <dd class="col-xl-6">{{ registration.birth_date }}</dd> | ||||||
|  |  | ||||||
|                 <dt class="col-xl-6">{% trans 'health issues or specific diet'|capfirst %}</dt> |                 <dt class="col-xl-6">{% trans 'health issues'|capfirst %}</dt> | ||||||
|                 <dd class="col-xl-6">{{ registration.health_issues }}</dd> |                 <dd class="col-xl-6">{{ registration.health_issues }}</dd> | ||||||
|  |  | ||||||
|                 <dt class="col-xl-6">{% trans 'emergency contact name'|capfirst %}</dt> |                 <dt class="col-xl-6">{% trans 'emergency contact name'|capfirst %}</dt> | ||||||
|   | |||||||
| @@ -6,6 +6,8 @@ from datetime import date, timedelta | |||||||
|  |  | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
|  | from django.urls import reverse | ||||||
|  | from note.models import NoteUser | ||||||
|  |  | ||||||
| from ..forms.surveys.wei2023 import WEIBusInformation2023, WEISurvey2023, WORDS, WEISurveyInformation2023 | from ..forms.surveys.wei2023 import WEIBusInformation2023, WEISurvey2023, WORDS, WEISurveyInformation2023 | ||||||
| from ..models import Bus, WEIClub, WEIRegistration | from ..models import Bus, WEIClub, WEIRegistration | ||||||
| @@ -125,3 +127,44 @@ class TestWEIAlgorithm(TestCase): | |||||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance |             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance | ||||||
|  |  | ||||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % |         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % | ||||||
|  |  | ||||||
|  |     def test_register_1a(self): | ||||||
|  |         """ | ||||||
|  |         Test register a first year member to the WEI and complete the survey | ||||||
|  |         """ | ||||||
|  |         response = self.client.get(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk))) | ||||||
|  |         self.assertEqual(response.status_code, 200) | ||||||
|  |  | ||||||
|  |         user = User.objects.create(username="toto", email="toto@example.com") | ||||||
|  |         NoteUser.objects.create(user=user) | ||||||
|  |         response = self.client.post(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk)), dict( | ||||||
|  |             user=user.id, | ||||||
|  |             soge_credit=True, | ||||||
|  |             birth_date=date(2000, 1, 1), | ||||||
|  |             gender='nonbinary', | ||||||
|  |             clothing_cut='female', | ||||||
|  |             clothing_size='XS', | ||||||
|  |             health_issues='I am a bot', | ||||||
|  |             emergency_contact_name='NoteKfet2020', | ||||||
|  |             emergency_contact_phone='+33123456789', | ||||||
|  |         )) | ||||||
|  |         qs = WEIRegistration.objects.filter(user_id=user.id) | ||||||
|  |         self.assertTrue(qs.exists()) | ||||||
|  |         registration = qs.get() | ||||||
|  |         self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, 200) | ||||||
|  |         for question in WORDS: | ||||||
|  |             # Fill 1A Survey, 20 pages | ||||||
|  |             # be careful if questionnary form change (number of page, type of answer...) | ||||||
|  |             response = self.client.post(reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), { | ||||||
|  |                 question: "1" | ||||||
|  |             }) | ||||||
|  |             registration.refresh_from_db() | ||||||
|  |             survey = WEISurvey2023(registration) | ||||||
|  |             self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, | ||||||
|  |                                  302 if survey.is_complete() else 200) | ||||||
|  |             self.assertIsNotNone(getattr(survey.information, question), "Survey page " + question + " failed") | ||||||
|  |         survey = WEISurvey2023(registration) | ||||||
|  |         self.assertTrue(survey.is_complete()) | ||||||
|  |         survey.select_bus(self.buses[0]) | ||||||
|  |         survey.save() | ||||||
|  |         self.assertIsNotNone(survey.information.get_selected_bus()) | ||||||
|   | |||||||
| @@ -1,172 +0,0 @@ | |||||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
|  |  | ||||||
| import random |  | ||||||
| from datetime import date, timedelta |  | ||||||
|  |  | ||||||
| from django.contrib.auth.models import User |  | ||||||
| from django.test import TestCase |  | ||||||
| from django.urls import reverse |  | ||||||
| from note.models import NoteUser |  | ||||||
|  |  | ||||||
| from ..forms.surveys.wei2024 import WEIBusInformation2024, WEISurvey2024, WORDS, WEISurveyInformation2024 |  | ||||||
| from ..models import Bus, WEIClub, WEIRegistration |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestWEIAlgorithm(TestCase): |  | ||||||
|     """ |  | ||||||
|     Run some tests to ensure that the WEI algorithm is working well. |  | ||||||
|     """ |  | ||||||
|     fixtures = ('initial',) |  | ||||||
|  |  | ||||||
|     def setUp(self): |  | ||||||
|         """ |  | ||||||
|         Create some test data, with one WEI and 10 buses with random score attributions. |  | ||||||
|         """ |  | ||||||
|         self.user = User.objects.create_superuser( |  | ||||||
|             username="weiadmin", |  | ||||||
|             password="admin", |  | ||||||
|             email="admin@example.com", |  | ||||||
|         ) |  | ||||||
|         self.user.save() |  | ||||||
|         self.client.force_login(self.user) |  | ||||||
|         sess = self.client.session |  | ||||||
|         sess["permission_mask"] = 42 |  | ||||||
|         sess.save() |  | ||||||
|  |  | ||||||
|         self.wei = WEIClub.objects.create( |  | ||||||
|             name="WEI 2024", |  | ||||||
|             email="wei2024@example.com", |  | ||||||
|             parent_club_id=2, |  | ||||||
|             membership_fee_paid=12500, |  | ||||||
|             membership_fee_unpaid=5500, |  | ||||||
|             membership_start='2024-01-01', |  | ||||||
|             membership_end='2024-12-31', |  | ||||||
|             date_start=date.today() + timedelta(days=2), |  | ||||||
|             date_end='2024-12-31', |  | ||||||
|             year=2024, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         self.buses = [] |  | ||||||
|         for i in range(10): |  | ||||||
|             bus = Bus.objects.create(wei=self.wei, name=f"Bus {i}", size=10) |  | ||||||
|             self.buses.append(bus) |  | ||||||
|             information = WEIBusInformation2024(bus) |  | ||||||
|             for question in WORDS: |  | ||||||
|                 information.scores[question] = {answer: random.randint(1, 5) for answer in WORDS[question][1]} |  | ||||||
|             information.save() |  | ||||||
|             bus.save() |  | ||||||
|  |  | ||||||
|     def test_survey_algorithm_small(self): |  | ||||||
|         """ |  | ||||||
|         There are only a few people in each bus, ensure that each person has its best bus |  | ||||||
|         """ |  | ||||||
|         # Add a few users |  | ||||||
|         for i in range(10): |  | ||||||
|             user = User.objects.create(username=f"user{i}") |  | ||||||
|             registration = WEIRegistration.objects.create( |  | ||||||
|                 user=user, |  | ||||||
|                 wei=self.wei, |  | ||||||
|                 first_year=True, |  | ||||||
|                 birth_date='2000-01-01', |  | ||||||
|             ) |  | ||||||
|             information = WEISurveyInformation2024(registration) |  | ||||||
|             for question in WORDS: |  | ||||||
|                 options = list(WORDS[question][1].keys()) |  | ||||||
|                 setattr(information, question, random.choice(options)) |  | ||||||
|             information.step = 20 |  | ||||||
|             information.save(registration) |  | ||||||
|             registration.save() |  | ||||||
|  |  | ||||||
|         # Run algorithm |  | ||||||
|         WEISurvey2024.get_algorithm_class()().run_algorithm() |  | ||||||
|  |  | ||||||
|         # Ensure that everyone has its first choice |  | ||||||
|         for r in WEIRegistration.objects.filter(wei=self.wei).all(): |  | ||||||
|             survey = WEISurvey2024(r) |  | ||||||
|             preferred_bus = survey.ordered_buses()[0][0] |  | ||||||
|             chosen_bus = survey.information.get_selected_bus() |  | ||||||
|             self.assertEqual(preferred_bus, chosen_bus) |  | ||||||
|  |  | ||||||
|     def test_survey_algorithm_full(self): |  | ||||||
|         """ |  | ||||||
|         Buses are full of first year people, ensure that they are happy |  | ||||||
|         """ |  | ||||||
|         # Add a lot of users |  | ||||||
|         for i in range(95): |  | ||||||
|             user = User.objects.create(username=f"user{i}") |  | ||||||
|             registration = WEIRegistration.objects.create( |  | ||||||
|                 user=user, |  | ||||||
|                 wei=self.wei, |  | ||||||
|                 first_year=True, |  | ||||||
|                 birth_date='2000-01-01', |  | ||||||
|             ) |  | ||||||
|             information = WEISurveyInformation2024(registration) |  | ||||||
|             for question in WORDS: |  | ||||||
|                 options = list(WORDS[question][1].keys()) |  | ||||||
|                 setattr(information, question, random.choice(options)) |  | ||||||
|             information.step = 20 |  | ||||||
|             information.save(registration) |  | ||||||
|             registration.save() |  | ||||||
|  |  | ||||||
|         # Run algorithm |  | ||||||
|         WEISurvey2024.get_algorithm_class()().run_algorithm() |  | ||||||
|  |  | ||||||
|         penalty = 0 |  | ||||||
|         # Ensure that everyone seems to be happy |  | ||||||
|         # We attribute a penalty for each user that didn't have its first choice |  | ||||||
|         # The penalty is the square of the distance between the score of the preferred bus |  | ||||||
|         # and the score of the attributed bus |  | ||||||
|         # We consider it acceptable if the mean of this distance is lower than 5 % |  | ||||||
|         for r in WEIRegistration.objects.filter(wei=self.wei).all(): |  | ||||||
|             survey = WEISurvey2024(r) |  | ||||||
|             chosen_bus = survey.information.get_selected_bus() |  | ||||||
|             buses = survey.ordered_buses() |  | ||||||
|             score = min(v for bus, v in buses if bus == chosen_bus) |  | ||||||
|             max_score = buses[0][1] |  | ||||||
|             penalty += (max_score - score) ** 2 |  | ||||||
|  |  | ||||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance |  | ||||||
|  |  | ||||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % |  | ||||||
|  |  | ||||||
|     def test_register_1a(self): |  | ||||||
|         """ |  | ||||||
|         Test register a first year member to the WEI and complete the survey |  | ||||||
|         """ |  | ||||||
|         response = self.client.get(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk))) |  | ||||||
|         self.assertEqual(response.status_code, 200) |  | ||||||
|  |  | ||||||
|         user = User.objects.create(username="toto", email="toto@example.com") |  | ||||||
|         NoteUser.objects.create(user=user) |  | ||||||
|         response = self.client.post(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk)), dict( |  | ||||||
|             user=user.id, |  | ||||||
|             soge_credit=True, |  | ||||||
|             birth_date=date(2000, 1, 1), |  | ||||||
|             gender='nonbinary', |  | ||||||
|             clothing_cut='female', |  | ||||||
|             clothing_size='XS', |  | ||||||
|             health_issues='I am a bot', |  | ||||||
|             emergency_contact_name='NoteKfet2020', |  | ||||||
|             emergency_contact_phone='+33123456789', |  | ||||||
|         )) |  | ||||||
|         qs = WEIRegistration.objects.filter(user_id=user.id) |  | ||||||
|         self.assertTrue(qs.exists()) |  | ||||||
|         registration = qs.get() |  | ||||||
|         self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, 200) |  | ||||||
|         for question in WORDS: |  | ||||||
|             # Fill 1A Survey, 10 pages |  | ||||||
|             # be careful if questionnary form change (number of page, type of answer...) |  | ||||||
|             response = self.client.post(reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), { |  | ||||||
|                 question: "1" |  | ||||||
|             }) |  | ||||||
|             registration.refresh_from_db() |  | ||||||
|             survey = WEISurvey2024(registration) |  | ||||||
|             self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, |  | ||||||
|                                  302 if survey.is_complete() else 200) |  | ||||||
|             self.assertIsNotNone(getattr(survey.information, question), "Survey page " + question + " failed") |  | ||||||
|         survey = WEISurvey2024(registration) |  | ||||||
|         self.assertTrue(survey.is_complete()) |  | ||||||
|         survey.select_bus(self.buses[0]) |  | ||||||
|         survey.save() |  | ||||||
|         self.assertIsNotNone(survey.information.get_selected_bus()) |  | ||||||
| @@ -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], | ||||||
| @@ -767,7 +767,7 @@ class TestDefaultWEISurvey(TestCase): | |||||||
|         WEISurvey.update_form(None, None) |         WEISurvey.update_form(None, None) | ||||||
|  |  | ||||||
|         self.assertEqual(CurrentSurvey.get_algorithm_class().get_survey_class(), CurrentSurvey) |         self.assertEqual(CurrentSurvey.get_algorithm_class().get_survey_class(), CurrentSurvey) | ||||||
|         self.assertEqual(CurrentSurvey.get_year(), 2024) |         self.assertEqual(CurrentSurvey.get_year(), 2023) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestWeiAPI(TestAPI): | class TestWeiAPI(TestAPI): | ||||||
|   | |||||||
| @@ -900,9 +900,6 @@ class WEIValidateRegistrationView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|         form.fields["last_name"].initial = registration.user.last_name |         form.fields["last_name"].initial = registration.user.last_name | ||||||
|         form.fields["first_name"].initial = registration.user.first_name |         form.fields["first_name"].initial = registration.user.first_name | ||||||
|  |  | ||||||
|         if "caution_check" in form.fields: |  | ||||||
|             form.fields["caution_check"].initial = registration.caution_check |  | ||||||
|  |  | ||||||
|         if registration.soge_credit: |         if registration.soge_credit: | ||||||
|             form.fields["credit_type"].disabled = True |             form.fields["credit_type"].disabled = True | ||||||
|             form.fields["credit_type"].initial = NoteSpecial.objects.get(special_type="Virement bancaire") |             form.fields["credit_type"].initial = NoteSpecial.objects.get(special_type="Virement bancaire") | ||||||
| @@ -944,9 +941,6 @@ class WEIValidateRegistrationView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|         club = registration.wei |         club = registration.wei | ||||||
|         user = registration.user |         user = registration.user | ||||||
|  |  | ||||||
|         if "caution_check" in form.data: |  | ||||||
|             registration.caution_check = form.data["caution_check"] == "on" |  | ||||||
|             registration.save() |  | ||||||
|         membership = form.instance |         membership = form.instance | ||||||
|         membership.user = user |         membership.user = user | ||||||
|         membership.club = club |         membership.club = club | ||||||
|   | |||||||
| @@ -1,83 +0,0 @@ | |||||||
| Application Food |  | ||||||
| ================ |  | ||||||
|  |  | ||||||
| L'application ``food`` s'occupe de la traçabilité et permet notamment l'obtention de la liste des allergènes. |  | ||||||
|  |  | ||||||
| Modèles |  | ||||||
| ------- |  | ||||||
|  |  | ||||||
| L'application comporte 5 modèles : Allergen, QRCode, Food, BasicFood, TransformedFood. |  | ||||||
|  |  | ||||||
| Food |  | ||||||
| ~~~~ |  | ||||||
|  |  | ||||||
| Ce modèle est un PolymorphicModel et ne sert uniquement à créer BasicFood et TransformedFood. |  | ||||||
|  |  | ||||||
| Le modèle regroupe : |  | ||||||
|  |  | ||||||
| * Nom du produit |  | ||||||
| * Propriétaire (doit-être un Club) |  | ||||||
| * Allergènes (ManyToManyField) |  | ||||||
| * date d'expiration |  | ||||||
| * a été mangé (booléen) |  | ||||||
| * est prêt (booléen) |  | ||||||
|  |  | ||||||
| BasicFood |  | ||||||
| ~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Les BasicFood correspondent aux produits non modifiés à la Kfet. Ils peuvent correspondre à la fois à des produits achetés en magasin ou à des produits Terre à Terre. Ces produits seront les ingrédients de tous les plats préparés et en conséquent sont les seuls produits à nécessité une saisie manuelle des allergènes. |  | ||||||
|  |  | ||||||
| Le modèle regroupe : |  | ||||||
|  |  | ||||||
| * Type de date (DLC = date limite de consommation, DDM = date de durabilité minimale) |  | ||||||
| * Date d'arrivée |  | ||||||
| * Champs de Food |  | ||||||
|  |  | ||||||
| TransformedFood |  | ||||||
| ~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Les TransformedFood correspondent aux produits préparés à la Kfet. Ils peuvent être composés de BasicFood et/ou de TransformedFood. La date d'expiration et les allergènes sont automatiquement mis à jour par update (qui doit être exécuté après modification des ingrédients dans les forms par exemple). |  | ||||||
|  |  | ||||||
| Le modèle regroupe : |  | ||||||
|  |  | ||||||
| * Durée de consommation (par défaut 3 jours) |  | ||||||
| * Ingrédients (ManyToManyField vers Food) |  | ||||||
| * Date de création |  | ||||||
| * Champs de Food |  | ||||||
|  |  | ||||||
| Allergen |  | ||||||
| ~~~~~~~~ |  | ||||||
|  |  | ||||||
| Le modèle regroupe : |  | ||||||
|  |  | ||||||
| * Nom |  | ||||||
|  |  | ||||||
| QRCode |  | ||||||
| ~~~~~~ |  | ||||||
|  |  | ||||||
| Le modèle regroupe : |  | ||||||
|  |  | ||||||
| * nombre (unique, entier positif) |  | ||||||
| * food (OneToOneField vers Food) |  | ||||||
|  |  | ||||||
| Création de BasicFood |  | ||||||
| ~~~~~~~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Un BasicFood a toujours besoin d'un QRCode (depuis l'interface web). Il convient donc de coller le QRCode puis de le scanner et de compléter le formulaire. |  | ||||||
|  |  | ||||||
| Création de TransformedFood |  | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Pour créer un TransformedFood, il suffit d'aller dans l'onglet ``traçabilité`` et de cliquer sur l'onglet. |  | ||||||
|  |  | ||||||
| Ajouter un ingrédient |  | ||||||
| ~~~~~~~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Un ingrédient a forcément un QRCode. Il convient donc de scanner le QRCode de l'ingrédient et de sélectionner le produit auquel il doit être ajouté. |  | ||||||
|  |  | ||||||
| Remarque : Un produit fini doit avoir un QRCode et inversement. |  | ||||||
|  |  | ||||||
| Terminer un plat |  | ||||||
| ~~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Il suffit de coller le QRCode sur le plat, de le scanner et de sélectionner le produit. |  | ||||||
							
								
								
									
										12
									
								
								docs/faq.rst
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								docs/faq.rst
									
									
									
									
									
								
							| @@ -177,13 +177,11 @@ Contributeur⋅rices | |||||||
|  |  | ||||||
|    Liste des contributeur⋅rices majeur⋅es, par ordre alphabétique : |    Liste des contributeur⋅rices majeur⋅es, par ordre alphabétique : | ||||||
|  |  | ||||||
|    * bleizi |    * Pierre-André « PAC » COMBY | ||||||
|    * erdnaxe |    * Emmy « ÿnérant » D'ANELLO | ||||||
|    * esum |    * Benjamin « esum » GRAILLOT | ||||||
|    * korenst1 |    * Alexandre « erdnaxe » IOOSS | ||||||
|    * nicomarg |    * Nicolas « nicomarg » MARGULIES | ||||||
|    * PAC |  | ||||||
|    * ÿnérant |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Hébergement | Hébergement | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -17,14 +17,6 @@ msgstr "" | |||||||
| "Content-Transfer-Encoding: 8bit\n" | "Content-Transfer-Encoding: 8bit\n" | ||||||
| "Plural-Forms: nplurals=2; plural=(n > 1);\n" | "Plural-Forms: nplurals=2; plural=(n > 1);\n" | ||||||
|  |  | ||||||
| #: apps/member/static/member/js/alias.js:17 |  | ||||||
| msgid "Opener successfully added" |  | ||||||
| msgstr "Ouvreureuse ajouté avec succès" |  | ||||||
|  |  | ||||||
| #: apps/member/static/member/js/alias.js:17 |  | ||||||
| msgid "Opener successfully deleted" |  | ||||||
| msgstr "Ouvreureuse supprimé avec succès" |  | ||||||
|  |  | ||||||
| #: apps/member/static/member/js/alias.js:17 | #: apps/member/static/member/js/alias.js:17 | ||||||
| msgid "Alias successfully added" | msgid "Alias successfully added" | ||||||
| msgstr "Alias ajouté avec succès" | msgstr "Alias ajouté avec succès" | ||||||
|   | |||||||
| @@ -25,8 +25,8 @@ 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 ApplicationAdmin, GrantAdmin, AccessTokenAdmin, RefreshTokenAdmin |     from oauth2_provider.admin import Application, ApplicationAdmin, Grant, \ | ||||||
|     from oauth2_provider.models import Application, Grant, AccessToken, RefreshToken |         GrantAdmin, AccessToken, AccessTokenAdmin, RefreshToken, RefreshTokenAdmin | ||||||
|     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) | ||||||
|   | |||||||
| @@ -68,3 +68,264 @@ class ColorWidget(Widget): | |||||||
|     def value_from_datadict(self, data, files, name): |     def value_from_datadict(self, data, files, name): | ||||||
|         val = super().value_from_datadict(data, files, name) |         val = super().value_from_datadict(data, files, name) | ||||||
|         return int(val[1:], 16) |         return int(val[1:], 16) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """ | ||||||
|  | The remaining of this file comes from the project `django-bootstrap-datepicker-plus` available on Github: | ||||||
|  | https://github.com/monim67/django-bootstrap-datepicker-plus | ||||||
|  | This is distributed under Apache License 2.0. | ||||||
|  |  | ||||||
|  | This adds datetime pickers with bootstrap. | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | """Contains Base Date-Picker input class for widgets of this package.""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatePickerDictionary: | ||||||
|  |     """Keeps track of all date-picker input classes.""" | ||||||
|  |  | ||||||
|  |     _i = 0 | ||||||
|  |     items = dict() | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def generate_id(cls): | ||||||
|  |         """Return a unique ID for each date-picker input class.""" | ||||||
|  |         cls._i += 1 | ||||||
|  |         return 'dp_%s' % cls._i | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BasePickerInput(DateTimeBaseInput): | ||||||
|  |     """Base Date-Picker input class for widgets of this package.""" | ||||||
|  |  | ||||||
|  |     template_name = 'bootstrap_datepicker_plus/date-picker.html' | ||||||
|  |     picker_type = 'DATE' | ||||||
|  |     format = '%Y-%m-%d' | ||||||
|  |     config = {} | ||||||
|  |     _default_config = { | ||||||
|  |         'id': None, | ||||||
|  |         'picker_type': None, | ||||||
|  |         'linked_to': None, | ||||||
|  |         'options': {}  # final merged options | ||||||
|  |     } | ||||||
|  |     options = {}  # options extended by user | ||||||
|  |     options_param = {}  # options passed as parameter | ||||||
|  |     _default_options = { | ||||||
|  |         'showClose': True, | ||||||
|  |         'showClear': True, | ||||||
|  |         'showTodayButton': True, | ||||||
|  |         "locale": "fr", | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # source: https://github.com/tutorcruncher/django-bootstrap3-datetimepicker | ||||||
|  |     # file: /blob/31fbb09/bootstrap3_datetime/widgets.py#L33 | ||||||
|  |     format_map = ( | ||||||
|  |         ('DDD', r'%j'), | ||||||
|  |         ('DD', r'%d'), | ||||||
|  |         ('MMMM', r'%B'), | ||||||
|  |         ('MMM', r'%b'), | ||||||
|  |         ('MM', r'%m'), | ||||||
|  |         ('YYYY', r'%Y'), | ||||||
|  |         ('YY', r'%y'), | ||||||
|  |         ('HH', r'%H'), | ||||||
|  |         ('hh', r'%I'), | ||||||
|  |         ('mm', r'%M'), | ||||||
|  |         ('ss', r'%S'), | ||||||
|  |         ('a', r'%p'), | ||||||
|  |         ('ZZ', r'%z'), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     class Media: | ||||||
|  |         """JS/CSS resources needed to render the date-picker calendar.""" | ||||||
|  |  | ||||||
|  |         js = ( | ||||||
|  |             'https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.9.0/' | ||||||
|  |             'moment-with-locales.min.js', | ||||||
|  |             'https://cdnjs.cloudflare.com/ajax/libs/bootstrap-datetimepicker/' | ||||||
|  |             '4.17.47/js/bootstrap-datetimepicker.min.js', | ||||||
|  |             'bootstrap_datepicker_plus/js/datepicker-widget.js' | ||||||
|  |         ) | ||||||
|  |         css = {'all': ( | ||||||
|  |             'https://cdnjs.cloudflare.com/ajax/libs/bootstrap-datetimepicker/' | ||||||
|  |             '4.17.47/css/bootstrap-datetimepicker.css', | ||||||
|  |             'bootstrap_datepicker_plus/css/datepicker-widget.css' | ||||||
|  |         ), } | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def format_py2js(cls, datetime_format): | ||||||
|  |         """Convert python datetime format to moment datetime format.""" | ||||||
|  |         for js_format, py_format in cls.format_map: | ||||||
|  |             datetime_format = datetime_format.replace(py_format, js_format) | ||||||
|  |         return datetime_format | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def format_js2py(cls, datetime_format): | ||||||
|  |         """Convert moment datetime format to python datetime format.""" | ||||||
|  |         for js_format, py_format in cls.format_map: | ||||||
|  |             datetime_format = datetime_format.replace(js_format, py_format) | ||||||
|  |         return datetime_format | ||||||
|  |  | ||||||
|  |     def __init__(self, attrs=None, format=None, options=None): | ||||||
|  |         """Initialize the Date-picker widget.""" | ||||||
|  |         self.format_param = format | ||||||
|  |         self.options_param = options if options else {} | ||||||
|  |         self.config = self._default_config.copy() | ||||||
|  |         self.config['id'] = DatePickerDictionary.generate_id() | ||||||
|  |         self.config['picker_type'] = self.picker_type | ||||||
|  |         self.config['options'] = self._calculate_options() | ||||||
|  |         attrs = attrs if attrs else {} | ||||||
|  |         if 'class' not in attrs: | ||||||
|  |             attrs['class'] = 'form-control' | ||||||
|  |         super().__init__(attrs, self._calculate_format()) | ||||||
|  |  | ||||||
|  |     def _calculate_options(self): | ||||||
|  |         """Calculate and Return the options.""" | ||||||
|  |         _options = self._default_options.copy() | ||||||
|  |         _options.update(self.options) | ||||||
|  |         if self.options_param: | ||||||
|  |             _options.update(self.options_param) | ||||||
|  |         return _options | ||||||
|  |  | ||||||
|  |     def _calculate_format(self): | ||||||
|  |         """Calculate and Return the datetime format.""" | ||||||
|  |         _format = self.format_param if self.format_param else self.format | ||||||
|  |         if self.config['options'].get('format'): | ||||||
|  |             _format = self.format_js2py(self.config['options'].get('format')) | ||||||
|  |         else: | ||||||
|  |             self.config['options']['format'] = self.format_py2js(_format) | ||||||
|  |         return _format | ||||||
|  |  | ||||||
|  |     def get_context(self, name, value, attrs): | ||||||
|  |         """Return widget context dictionary.""" | ||||||
|  |         context = super().get_context( | ||||||
|  |             name, value, attrs) | ||||||
|  |         context['widget']['attrs']['dp_config'] = json_dumps(self.config) | ||||||
|  |         return context | ||||||
|  |  | ||||||
|  |     def start_of(self, event_id): | ||||||
|  |         """ | ||||||
|  |         Set Date-Picker as the start-date of a date-range. | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             - event_id (string): User-defined unique id for linking two fields | ||||||
|  |         """ | ||||||
|  |         DatePickerDictionary.items[str(event_id)] = self | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def end_of(self, event_id, import_options=True): | ||||||
|  |         """ | ||||||
|  |         Set Date-Picker as the end-date of a date-range. | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             - event_id (string): User-defined unique id for linking two fields | ||||||
|  |             - import_options (bool): inherit options from start-date input, | ||||||
|  |               default: TRUE | ||||||
|  |         """ | ||||||
|  |         event_id = str(event_id) | ||||||
|  |         if event_id in DatePickerDictionary.items: | ||||||
|  |             linked_picker = DatePickerDictionary.items[event_id] | ||||||
|  |             self.config['linked_to'] = linked_picker.config['id'] | ||||||
|  |             if import_options: | ||||||
|  |                 backup_moment_format = self.config['options']['format'] | ||||||
|  |                 self.config['options'].update(linked_picker.config['options']) | ||||||
|  |                 self.config['options'].update(self.options_param) | ||||||
|  |                 if self.format_param or 'format' in self.options_param: | ||||||
|  |                     self.config['options']['format'] = backup_moment_format | ||||||
|  |                 else: | ||||||
|  |                     self.format = linked_picker.format | ||||||
|  |             # Setting useCurrent is necessary, see following issue | ||||||
|  |             # https://github.com/Eonasdan/bootstrap-datetimepicker/issues/1075 | ||||||
|  |             self.config['options']['useCurrent'] = False | ||||||
|  |             self._link_to(linked_picker) | ||||||
|  |         else: | ||||||
|  |             raise KeyError( | ||||||
|  |                 'start-date not specified for event_id "%s"' % event_id) | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def _link_to(self, linked_picker): | ||||||
|  |         """ | ||||||
|  |         Executed when two date-inputs are linked together. | ||||||
|  |  | ||||||
|  |         This method for sub-classes to override to customize the linking. | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatePickerInput(BasePickerInput): | ||||||
|  |     """ | ||||||
|  |     Widget to display a Date-Picker Calendar on a DateField property. | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         - attrs (dict): HTML attributes of rendered HTML input | ||||||
|  |         - format (string): Python DateTime format eg. "%Y-%m-%d" | ||||||
|  |         - options (dict): Options to customize the widget, see README | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     picker_type = 'DATE' | ||||||
|  |     format = '%Y-%m-%d' | ||||||
|  |     format_key = 'DATE_INPUT_FORMATS' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TimePickerInput(BasePickerInput): | ||||||
|  |     """ | ||||||
|  |     Widget to display a Time-Picker Calendar on a TimeField property. | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         - attrs (dict): HTML attributes of rendered HTML input | ||||||
|  |         - format (string): Python DateTime format eg. "%Y-%m-%d" | ||||||
|  |         - options (dict): Options to customize the widget, see README | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     picker_type = 'TIME' | ||||||
|  |     format = '%H:%M' | ||||||
|  |     format_key = 'TIME_INPUT_FORMATS' | ||||||
|  |     template_name = 'bootstrap_datepicker_plus/time_picker.html' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DateTimePickerInput(BasePickerInput): | ||||||
|  |     """ | ||||||
|  |     Widget to display a DateTime-Picker Calendar on a DateTimeField property. | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         - attrs (dict): HTML attributes of rendered HTML input | ||||||
|  |         - format (string): Python DateTime format eg. "%Y-%m-%d" | ||||||
|  |         - options (dict): Options to customize the widget, see README | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     picker_type = 'DATETIME' | ||||||
|  |     format = '%Y-%m-%d %H:%M' | ||||||
|  |     format_key = 'DATETIME_INPUT_FORMATS' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MonthPickerInput(BasePickerInput): | ||||||
|  |     """ | ||||||
|  |     Widget to display a Month-Picker Calendar on a DateField property. | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         - attrs (dict): HTML attributes of rendered HTML input | ||||||
|  |         - format (string): Python DateTime format eg. "%Y-%m-%d" | ||||||
|  |         - options (dict): Options to customize the widget, see README | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     picker_type = 'MONTH' | ||||||
|  |     format = '01/%m/%Y' | ||||||
|  |     format_key = 'DATE_INPUT_FORMATS' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class YearPickerInput(BasePickerInput): | ||||||
|  |     """ | ||||||
|  |     Widget to display a Year-Picker Calendar on a DateField property. | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         - attrs (dict): HTML attributes of rendered HTML input | ||||||
|  |         - format (string): Python DateTime format eg. "%Y-%m-%d" | ||||||
|  |         - options (dict): Options to customize the widget, see README | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     picker_type = 'YEAR' | ||||||
|  |     format = '01/01/%Y' | ||||||
|  |     format_key = 'DATE_INPUT_FORMATS' | ||||||
|  |  | ||||||
|  |     def _link_to(self, linked_picker): | ||||||
|  |         """Customize the options when linked with other date-time input""" | ||||||
|  |         yformat = self.config['options']['format'].replace('-01-01', '-12-31') | ||||||
|  |         self.config['options']['format'] = yformat | ||||||
|   | |||||||
| @@ -40,9 +40,8 @@ INSTALLED_APPS = [ | |||||||
|     # External apps |     # External apps | ||||||
|     'bootstrap_datepicker_plus', |     'bootstrap_datepicker_plus', | ||||||
|     'colorfield', |     'colorfield', | ||||||
|     'crispy_bootstrap4', |  | ||||||
|     'crispy_forms', |     'crispy_forms', | ||||||
| #    'django_htcpcp_tea', |     'django_htcpcp_tea', | ||||||
|     'django_tables2', |     'django_tables2', | ||||||
|     'mailer', |     'mailer', | ||||||
|     'phonenumber_field', |     'phonenumber_field', | ||||||
| @@ -70,7 +69,6 @@ INSTALLED_APPS = [ | |||||||
|     # Note apps |     # Note apps | ||||||
|     'api', |     'api', | ||||||
|     'activity', |     'activity', | ||||||
|     'food', |  | ||||||
|     'logs', |     'logs', | ||||||
|     'member', |     'member', | ||||||
|     'note', |     'note', | ||||||
| @@ -92,14 +90,12 @@ 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', | ||||||
|     'note_kfet.middlewares.ClacksMiddleware', |     'note_kfet.middlewares.ClacksMiddleware', | ||||||
| ] | ] | ||||||
| if "django_htcpcp_tea" in INSTALLED_APPS: |  | ||||||
|     MIDDLEWARE.append('django_htcpcp_tea.middleware.HTCPCPTeaMiddleware') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ROOT_URLCONF = 'note_kfet.urls' | ROOT_URLCONF = 'note_kfet.urls' | ||||||
|  |  | ||||||
| @@ -240,7 +236,7 @@ DEFAULT_FROM_EMAIL = "NoteKfet2020 <" + SERVER_EMAIL + ">" | |||||||
| cache_address = os.getenv("CACHE_ADDRESS", "127.0.0.1:11211") | cache_address = os.getenv("CACHE_ADDRESS", "127.0.0.1:11211") | ||||||
| CACHES = { | CACHES = { | ||||||
|     'default': { |     'default': { | ||||||
|         'BACKEND': 'django.core.cache.backends.memcached.PyMemcacheCache', |         'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache', | ||||||
|         'LOCATION': cache_address, |         'LOCATION': cache_address, | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -267,9 +263,6 @@ OAUTH2_PROVIDER = { | |||||||
|     'REFRESH_TOKEN_EXPIRE_SECONDS': timedelta(days=14), |     'REFRESH_TOKEN_EXPIRE_SECONDS': timedelta(days=14), | ||||||
| } | } | ||||||
|  |  | ||||||
| # PKCE (fix a breaking change of django-oauth-toolkit 2.0.0) |  | ||||||
| PKCE_REQUIRED = False |  | ||||||
|  |  | ||||||
| # Take control on how widget templates are sourced | # Take control on how widget templates are sourced | ||||||
| # See https://docs.djangoproject.com/en/2.2/ref/forms/renderers/#templatessetting | # See https://docs.djangoproject.com/en/2.2/ref/forms/renderers/#templatessetting | ||||||
| FORM_RENDERER = 'django.forms.renderers.TemplatesSetting' | FORM_RENDERER = 'django.forms.renderers.TemplatesSetting' | ||||||
| @@ -281,7 +274,6 @@ LOGIN_REDIRECT_URL = '/' | |||||||
| SESSION_COOKIE_AGE = 60 * 60 * 3 | SESSION_COOKIE_AGE = 60 * 60 * 3 | ||||||
|  |  | ||||||
| # Use Crispy Bootstrap4 theme | # Use Crispy Bootstrap4 theme | ||||||
| CRISPY_ALLOWED_TEMPLATE_PACKS = 'bootstrap4' |  | ||||||
| CRISPY_TEMPLATE_PACK = 'bootstrap4' | CRISPY_TEMPLATE_PACK = 'bootstrap4' | ||||||
|  |  | ||||||
| # Use Django Table2 Bootstrap4 theme | # Use Django Table2 Bootstrap4 theme | ||||||
| @@ -303,6 +295,3 @@ 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 %} | ||||||
|         {% if value is not False %}{{ name }}{% if value is not True %}="{{ value|stringformat:'s' }}"{% endif %}{% endif %} |         {% ifnotequal value False %}{{ name }}{% ifnotequal value True %}="{{ value|stringformat:'s' }}"{% endifnotequal %}{% endifnotequal %} | ||||||
|     {% endfor %} |     {% endfor %} | ||||||
|     aria-describedby="{{widget.attrs.id}}_tooltip"> |     aria-describedby="{{widget.attrs.id}}_tooltip"> | ||||||
|     {% if widget.resetable %} |     {% if widget.resetable %} | ||||||
|   | |||||||
| @@ -66,16 +66,10 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-coffee"></i> {% trans 'Consumptions' %}</a> |                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-coffee"></i> {% trans 'Consumptions' %}</a> | ||||||
|                         </li> |                         </li> | ||||||
|                     {% endif %} |                     {% endif %} | ||||||
| 		    {% if request.user.is_authenticated %} |  | ||||||
|                     	<li class="nav-item"> |  | ||||||
|                             {% url 'food:food_list' as url %} |  | ||||||
| 			    <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-cutlery"></i> {% trans 'Food' %}</a> |  | ||||||
|                     	</li> |  | ||||||
| 		    {% endif %}	 |  | ||||||
|                     {% if user.is_authenticated and user|is_member:"Kfet" %} |                     {% if user.is_authenticated and user|is_member:"Kfet" %} | ||||||
|                         <li class="nav-item"> |                         <li class="nav-item"> | ||||||
|                             {% url 'note:transfer' as url %} |                             {% url 'note:transfer' as url %} | ||||||
|                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-exchange"></i> {% trans 'Transfer' %}</a> |                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-exchange"></i> {% trans 'Transfer' %} </a> | ||||||
|                         </li> |                         </li> | ||||||
|                     {% endif %} |                     {% endif %} | ||||||
|                     {% if "auth.user"|model_list_length >= 2 %} |                     {% if "auth.user"|model_list_length >= 2 %} | ||||||
|   | |||||||
| @@ -21,7 +21,6 @@ urlpatterns = [ | |||||||
|     path('activity/', include('activity.urls')), |     path('activity/', include('activity.urls')), | ||||||
|     path('treasury/', include('treasury.urls')), |     path('treasury/', include('treasury.urls')), | ||||||
|     path('wei/', include('wei.urls')), |     path('wei/', include('wei.urls')), | ||||||
|     path('food/',include('food.urls')), |  | ||||||
|  |  | ||||||
|     # Include Django Contrib and Core routers |     # Include Django Contrib and Core routers | ||||||
|     path('i18n/', include('django.conf.urls.i18n')), |     path('i18n/', include('django.conf.urls.i18n')), | ||||||
| @@ -31,6 +30,9 @@ 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 | ||||||
| @@ -55,11 +57,6 @@ if "debug_toolbar" in settings.INSTALLED_APPS: | |||||||
|         path('__debug__/', include(debug_toolbar.urls)), |         path('__debug__/', include(debug_toolbar.urls)), | ||||||
|     ] + urlpatterns |     ] + urlpatterns | ||||||
|  |  | ||||||
| if "django_htcpcp_tea" in settings.INSTALLED_APPS: |  | ||||||
|     # Make coffee |  | ||||||
|     urlpatterns.append( |  | ||||||
|         path('coffee/', include('django_htcpcp_tea.urls')) |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
| handler400 = bad_request | handler400 = bad_request | ||||||
| handler403 = permission_denied | handler403 = permission_denied | ||||||
|   | |||||||
| @@ -1,20 +1,19 @@ | |||||||
| beautifulsoup4~=4.12.3 | beautifulsoup4~=4.7.1 | ||||||
| crispy-bootstrap4~=2023.1 | Django~=2.2.15 | ||||||
| Django~=4.2.9 | django-bootstrap-datepicker-plus~=3.0.5 | ||||||
| django-bootstrap-datepicker-plus~=5.0.5 | django-cas-server~=1.2.0 | ||||||
| #django-cas-server~=2.0.0 | django-colorfield~=0.3.2 | ||||||
| django-colorfield~=0.11.0 | django-crispy-forms~=1.7.2 | ||||||
| django-crispy-forms~=2.1.0 | django-extensions>=2.1.4 | ||||||
| django-extensions>=3.2.3 | django-filter~=2.1 | ||||||
| django-filter~=23.5 | django-htcpcp-tea~=0.3.1 | ||||||
| #django-htcpcp-tea~=0.8.1 | django-mailer~=2.0.1 | ||||||
| django-mailer~=2.3.1 | django-oauth-toolkit~=1.3.3 | ||||||
| django-oauth-toolkit~=2.3.0 | django-phonenumber-field~=5.0.0 | ||||||
| django-phonenumber-field~=7.3.0 | django-polymorphic>=2.0.3,<3.0.0 | ||||||
| django-polymorphic~=3.1.0 | djangorestframework>=3.9.0,<3.13.0 | ||||||
| djangorestframework~=3.14.0 | django-rest-polymorphic~=0.1.9 | ||||||
| django-rest-polymorphic~=0.1.10 | django-tables2~=2.3.1 | ||||||
| django-tables2~=2.7.0 | python-memcached~=1.59 | ||||||
| python-memcached~=1.62 | phonenumbers~=8.9.10 | ||||||
| phonenumbers~=8.13.28 | Pillow>=5.4.1 | ||||||
| Pillow>=10.2.0 |  | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | |||||||
| [tox] | [tox] | ||||||
| envlist = | envlist = | ||||||
|  |     # Debian Buster Python | ||||||
|  |     py37-django22 | ||||||
|  |  | ||||||
|  |     # Ubuntu 20.04 Python | ||||||
|  |     py38-django22 | ||||||
|  |  | ||||||
|     # Debian Bullseye Python |     # Debian Bullseye Python | ||||||
|     py39-django42 |     py39-django22 | ||||||
|  |  | ||||||
|     # Ubuntu 22.04 Python |  | ||||||
|     py310-django42 |  | ||||||
|  |  | ||||||
|     # Debian Bookworm Python |  | ||||||
|     py311-django42 |  | ||||||
|  |  | ||||||
|     linters |     linters | ||||||
| skipsdist = True | skipsdist = True | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user