mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-10-20 19:58:31 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			dependabot
			...
			subscripti
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 500af2f73a | 
| @@ -252,7 +252,7 @@ class ClubAddMemberForm(ClubMemberForm): | |||||||
|         Board members can attribute roles lower than their own. |         Board members can attribute roles lower than their own. | ||||||
|         Other users cannot attribute roles with this form |         Other users cannot attribute roles with this form | ||||||
|         """ |         """ | ||||||
|         if self.request_user.has_perm("club.add_membership"): |         if self.request_user.has_perm("club.add_subscription"): | ||||||
|             return settings.SITH_CLUB_ROLES_ID["President"] |             return settings.SITH_CLUB_ROLES_ID["President"] | ||||||
|         membership = self.request_user_membership |         membership = self.request_user_membership | ||||||
|         if membership is None or membership.role <= settings.SITH_MAXIMUM_FREE_ROLE: |         if membership is None or membership.role <= settings.SITH_MAXIMUM_FREE_ROLE: | ||||||
|   | |||||||
| @@ -31,7 +31,11 @@ from django.contrib.messages.views import SuccessMessageMixin | |||||||
| from django.core.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError | from django.core.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError | ||||||
| from django.core.paginator import InvalidPage, Paginator | from django.core.paginator import InvalidPage, Paginator | ||||||
| from django.db.models import Q, Sum | from django.db.models import Q, Sum | ||||||
| from django.http import Http404, HttpResponseRedirect, StreamingHttpResponse | from django.http import ( | ||||||
|  |     Http404, | ||||||
|  |     HttpResponseRedirect, | ||||||
|  |     StreamingHttpResponse, | ||||||
|  | ) | ||||||
| from django.shortcuts import get_object_or_404, redirect | from django.shortcuts import get_object_or_404, redirect | ||||||
| from django.urls import reverse, reverse_lazy | from django.urls import reverse, reverse_lazy | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| @@ -51,7 +55,12 @@ from club.forms import ( | |||||||
|     MailingForm, |     MailingForm, | ||||||
|     SellingsForm, |     SellingsForm, | ||||||
| ) | ) | ||||||
| from club.models import Club, Mailing, MailingSubscription, Membership | from club.models import ( | ||||||
|  |     Club, | ||||||
|  |     Mailing, | ||||||
|  |     MailingSubscription, | ||||||
|  |     Membership, | ||||||
|  | ) | ||||||
| from com.models import Poster | from com.models import Poster | ||||||
| from com.views import ( | from com.views import ( | ||||||
|     PosterCreateBaseView, |     PosterCreateBaseView, | ||||||
| @@ -59,7 +68,9 @@ from com.views import ( | |||||||
|     PosterEditBaseView, |     PosterEditBaseView, | ||||||
|     PosterListBaseView, |     PosterListBaseView, | ||||||
| ) | ) | ||||||
| from core.auth.mixins import CanEditMixin | from core.auth.mixins import ( | ||||||
|  |     CanEditMixin, | ||||||
|  | ) | ||||||
| from core.models import PageRev | from core.models import PageRev | ||||||
| from core.views import DetailFormView, PageEditViewBase, UseFragmentsMixin | from core.views import DetailFormView, PageEditViewBase, UseFragmentsMixin | ||||||
| from core.views.mixins import FragmentMixin, FragmentRenderer, TabedViewMixin | from core.views.mixins import FragmentMixin, FragmentRenderer, TabedViewMixin | ||||||
|   | |||||||
| @@ -83,8 +83,7 @@ | |||||||
|     #links_content { |     #links_content { | ||||||
|       overflow: auto; |       overflow: auto; | ||||||
|       box-shadow: $shadow-color 1px 1px 1px; |       box-shadow: $shadow-color 1px 1px 1px; | ||||||
|       min-height: 20em; |       height: 20em; | ||||||
|       padding-bottom: 1em; |  | ||||||
|  |  | ||||||
|       h4 { |       h4 { | ||||||
|         margin-left: 5px; |         margin-left: 5px; | ||||||
|   | |||||||
| @@ -76,20 +76,18 @@ | |||||||
|               It will stay hidden for other users until it has been published. |               It will stay hidden for other users until it has been published. | ||||||
|             {% endtrans %} |             {% endtrans %} | ||||||
|           </p> |           </p> | ||||||
|           {%- if user.has_perm("com.moderate_news") -%} |           {% if user.has_perm("com.moderate_news") %} | ||||||
|             {# This is an additional query for each non-moderated news, |             {# This is an additional query for each non-moderated news, | ||||||
|             but it will be executed only for admin users, and only one time |             but it will be executed only for admin users, and only one time | ||||||
|             (if they do their job and moderate news as soon as they see them), |             (if they do their job and moderated news as soon as they see them), | ||||||
|             so it's still reasonable #} |             so it's still reasonable #} | ||||||
|             <div |             <div | ||||||
|               {% if news is integer or news is string -%} |               {% if news is integer or news is string %} | ||||||
|                 x-data="{ nbEvents: 0 }" |                 x-data="{ nbEvents: 0 }" | ||||||
|                 x-init="nbEvents = await nbToPublish()" |                 x-init="nbEvents = await nbToPublish()" | ||||||
|               {%- elif news.is_published -%} |               {% else %} | ||||||
|                 x-data="{ nbEvents: 0 }" |  | ||||||
|               {%- else -%} |  | ||||||
|                 x-data="{ nbEvents: {{ news.dates.count() }} }" |                 x-data="{ nbEvents: {{ news.dates.count() }} }" | ||||||
|               {%- endif -%} |               {% endif %} | ||||||
|             > |             > | ||||||
|               <template x-if="nbEvents > 1"> |               <template x-if="nbEvents > 1"> | ||||||
|                 <div> |                 <div> | ||||||
|   | |||||||
| @@ -205,10 +205,6 @@ | |||||||
|               <i class="fa-solid fa-graduation-cap fa-xl"></i> |               <i class="fa-solid fa-graduation-cap fa-xl"></i> | ||||||
|               <a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a> |               <a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a> | ||||||
|             </li> |             </li> | ||||||
|             <li> |  | ||||||
|               <i class="fa-solid fa-calendar-days fa-xl"></i> |  | ||||||
|               <a href="{{ url("timetable:generator") }}">{% trans %}Timetable{% endtrans %}</a> |  | ||||||
|             </li> |  | ||||||
|             <li> |             <li> | ||||||
|               <i class="fa-solid fa-magnifying-glass fa-xl"></i> |               <i class="fa-solid fa-magnifying-glass fa-xl"></i> | ||||||
|               <a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a> |               <a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a> | ||||||
|   | |||||||
| @@ -651,6 +651,9 @@ class User(AbstractUser): | |||||||
|  |  | ||||||
|  |  | ||||||
| class AnonymousUser(AuthAnonymousUser): | class AnonymousUser(AuthAnonymousUser): | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__() | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def was_subscribed(self): |     def was_subscribed(self): | ||||||
|         return False |         return False | ||||||
| @@ -659,6 +662,10 @@ class AnonymousUser(AuthAnonymousUser): | |||||||
|     def is_subscribed(self): |     def is_subscribed(self): | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def subscribed(self): | ||||||
|  |         return False | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def is_root(self): |     def is_root(self): | ||||||
|         return False |         return False | ||||||
|   | |||||||
| @@ -1,9 +1,9 @@ | |||||||
| import { alpinePlugin as notificationPlugin } from "#core:utils/notifications"; | import { alpinePlugin } from "#core:utils/notifications"; | ||||||
| import sort from "@alpinejs/sort"; | import sort from "@alpinejs/sort"; | ||||||
| import Alpine from "alpinejs"; | import Alpine from "alpinejs"; | ||||||
|  |  | ||||||
| Alpine.plugin(sort); | Alpine.plugin(sort); | ||||||
| Alpine.magic("notifications", notificationPlugin); | Alpine.magic("notifications", alpinePlugin); | ||||||
| window.Alpine = Alpine; | window.Alpine = Alpine; | ||||||
|  |  | ||||||
| window.addEventListener("DOMContentLoaded", () => { | window.addEventListener("DOMContentLoaded", () => { | ||||||
|   | |||||||
| @@ -154,9 +154,11 @@ form { | |||||||
|     margin-bottom: 1rem; |     margin-bottom: 1rem; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .row > label { |   .row { | ||||||
|  |     label { | ||||||
|       margin: unset; |       margin: unset; | ||||||
|     } |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // ------------- LABEL |   // ------------- LABEL | ||||||
|   label, legend { |   label, legend { | ||||||
|   | |||||||
| @@ -503,10 +503,6 @@ th { | |||||||
|   text-align: center; |   text-align: center; | ||||||
|   padding: 5px 10px; |   padding: 5px 10px; | ||||||
|  |  | ||||||
|   >input[type="checkbox"] { |  | ||||||
|     padding: unset; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   >ul { |   >ul { | ||||||
|     margin-top: 0; |     margin-top: 0; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -77,22 +77,22 @@ | |||||||
|         <div class="notification" x-data="{display: false}" :class="{white: display}"> |         <div class="notification" x-data="{display: false}" :class="{white: display}"> | ||||||
|           <a href="#" @click.prevent="display = !display"> |           <a href="#" @click.prevent="display = !display"> | ||||||
|             <i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i> |             <i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i> | ||||||
|             {% set notifications = user.notifications.filter(viewed=False).order_by("-date")|list %} |             {% set notification_count = user.notifications.filter(viewed=False).count() %} | ||||||
|  |  | ||||||
|             {%- if notifications|length > 0 -%} |             {% if notification_count > 0 %} | ||||||
|               <span> |               <span> | ||||||
|                 {% if notifications|length < 100 %} |                 {% if notification_count < 100 %} | ||||||
|                   {{ notifications|length }} |                   {{ notification_count }} | ||||||
|                 {%- else -%} |                 {% else %} | ||||||
|                   99+ |                     | ||||||
|                 {%- endif -%} |                 {% endif %} | ||||||
|               </span> |               </span> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|           </a> |           </a> | ||||||
|           <div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false"> |           <div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false"> | ||||||
|             <ul> |             <ul> | ||||||
|               {%- if notifications|length > 0 -%} |               {% if user.notifications.filter(viewed=False).count() > 0 %} | ||||||
|                 {%- for n in notifications -%} |                 {% for n in user.notifications.filter(viewed=False).order_by('-date') %} | ||||||
|                   <li> |                   <li> | ||||||
|                     <a href="{{ url("core:notification", notif_id=n.id) }}"> |                     <a href="{{ url("core:notification", notif_id=n.id) }}"> | ||||||
|                       <div class="datetime"> |                       <div class="datetime"> | ||||||
| @@ -108,10 +108,10 @@ | |||||||
|                       </div> |                       </div> | ||||||
|                     </a> |                     </a> | ||||||
|                   </li> |                   </li> | ||||||
|                 {%- endfor -%} |                 {% endfor %} | ||||||
|               {%- else -%} |               {% else %} | ||||||
|                 <li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li> |                 <li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li> | ||||||
|               {%- endif -%} |               {% endif %} | ||||||
|             </ul> |             </ul> | ||||||
|             <div class="options"> |             <div class="options"> | ||||||
|               <a href="{{ url('core:notification_list') }}"> |               <a href="{{ url('core:notification_list') }}"> | ||||||
|   | |||||||
| @@ -13,10 +13,10 @@ | |||||||
|              }" |              }" | ||||||
|      @quick-notification-add="(e) => messages.push(e?.detail)" |      @quick-notification-add="(e) => messages.push(e?.detail)" | ||||||
|      @quick-notification-delete="messages = []"> |      @quick-notification-delete="messages = []"> | ||||||
|   <template x-for="(message, index) in messages"> |   <template x-for="message in messages"> | ||||||
|     <div class="alert" :class="`alert-${message.tag}`" x-transition> |     <div x-data="{show: true}" class="alert" :class="`alert-${message.tag}`" x-show="show" x-transition> | ||||||
|       <span class="alert-main" x-text="message.text"></span> |       <span class="alert-main" x-text="message.text"></span> | ||||||
|       <span class="clickable" @click="messages = messages.filter((item, i) => i !== index)"> |       <span class="clickable" @click="show = false"> | ||||||
|         <i class="fa fa-close"></i> |         <i class="fa fa-close"></i> | ||||||
|       </span> |       </span> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -245,26 +245,3 @@ | |||||||
|   <button type="button" onclick="checkbox_{{form_id}}(true);">{% trans %}Select All{% endtrans %}</button> |   <button type="button" onclick="checkbox_{{form_id}}(true);">{% trans %}Select All{% endtrans %}</button> | ||||||
|   <button type="button" onclick="checkbox_{{form_id}}(false);">{% trans %}Unselect All{% endtrans %}</button> |   <button type="button" onclick="checkbox_{{form_id}}(false);">{% trans %}Unselect All{% endtrans %}</button> | ||||||
| {% endmacro %} | {% endmacro %} | ||||||
|  |  | ||||||
| {% macro update_notifications(messages, clear) %} |  | ||||||
|   {# Update notification area from new messages sent by django backend |  | ||||||
|      This is useful when performing fragment swaps to keep messages up to date |  | ||||||
|      Without this, the fragment would need to take control of the notification area and |  | ||||||
|      this would be an issue when having more than one fragment |  | ||||||
|  |  | ||||||
|      Parameters: |  | ||||||
|       messages: messages from django.contrib |  | ||||||
|       clear   : optional boolean that controls if notifications should be cleared first. True is the default |  | ||||||
|   #} |  | ||||||
|   {% set clear = clear|default(true) %} |  | ||||||
|   {% if messages %} |  | ||||||
|     <div x-init="() => { |  | ||||||
|                  {% if clear %} |  | ||||||
|                    $notifications.clear() |  | ||||||
|                  {% endif %} |  | ||||||
|                  {% for message in messages %} |  | ||||||
|                    $notifications.{{ message.tags }}('{{ message }}') |  | ||||||
|                  {% endfor %} |  | ||||||
|                  }"></div> |  | ||||||
|   {% endif %} |  | ||||||
| {% endmacro %} |  | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| {% spaceless %} |  | ||||||
| {% for js in statics.js %} | {% for js in statics.js %} | ||||||
|   <script-once type="module" src="{{ js }}"></script-once> |   <script-once type="module" src="{{ js }}"></script-once> | ||||||
| {% endfor %} | {% endfor %} | ||||||
| @@ -22,4 +21,3 @@ | |||||||
|   <slot style="display:none" name="initial">{{ initial }}</slot> |   <slot style="display:none" name="initial">{{ initial }}</slot> | ||||||
| {% endif %} | {% endif %} | ||||||
| </{{ component }}> | </{{ component }}> | ||||||
| {% endspaceless %} |  | ||||||
| @@ -115,7 +115,7 @@ class SelectUser(TextInput): | |||||||
|  |  | ||||||
| def validate_future_timestamp(value: date | datetime): | def validate_future_timestamp(value: date | datetime): | ||||||
|     if value <= now(): |     if value <= now(): | ||||||
|         raise ValidationError(_("Ensure this timestamp is set in the future")) |         raise ValueError(_("Ensure this timestamp is set in the future")) | ||||||
|  |  | ||||||
|  |  | ||||||
| class FutureDateTimeField(forms.DateTimeField): | class FutureDateTimeField(forms.DateTimeField): | ||||||
|   | |||||||
| @@ -22,7 +22,6 @@ from counter.models import ( | |||||||
|     Counter, |     Counter, | ||||||
|     Customer, |     Customer, | ||||||
|     Eticket, |     Eticket, | ||||||
|     InvoiceCall, |  | ||||||
|     Permanency, |     Permanency, | ||||||
|     Product, |     Product, | ||||||
|     ProductType, |     ProductType, | ||||||
| @@ -161,11 +160,3 @@ class CashRegisterSummaryAdmin(SearchModelAdmin): | |||||||
| class EticketAdmin(SearchModelAdmin): | class EticketAdmin(SearchModelAdmin): | ||||||
|     list_display = ("product", "event_date", "event_title") |     list_display = ("product", "event_date", "event_title") | ||||||
|     search_fields = ("product__name", "event_title") |     search_fields = ("product__name", "event_title") | ||||||
|  |  | ||||||
|  |  | ||||||
| @admin.register(InvoiceCall) |  | ||||||
| class InvoiceCallAdmin(SearchModelAdmin): |  | ||||||
|     list_display = ("club", "month", "is_validated") |  | ||||||
|     search_fields = ("club__name",) |  | ||||||
|     list_filter = (("club", admin.RelatedOnlyFieldListFilter),) |  | ||||||
|     date_hierarchy = "month" |  | ||||||
|   | |||||||
							
								
								
									
										187
									
								
								counter/forms.py
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								counter/forms.py
									
									
									
									
									
								
							| @@ -1,26 +1,13 @@ | |||||||
| import json |  | ||||||
| import math | import math | ||||||
| import uuid |  | ||||||
| from datetime import date |  | ||||||
|  |  | ||||||
| from dateutil.relativedelta import relativedelta |  | ||||||
| from django import forms | from django import forms | ||||||
| from django.db.models import Exists, OuterRef, Q | from django.db.models import Q | ||||||
| from django.forms import BaseModelFormSet |  | ||||||
| from django.utils.timezone import now |  | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from django_celery_beat.models import ClockedSchedule |  | ||||||
| from phonenumber_field.widgets import RegionalPhoneNumberWidget | from phonenumber_field.widgets import RegionalPhoneNumberWidget | ||||||
|  |  | ||||||
| from club.models import Club |  | ||||||
| from club.widgets.ajax_select import AutoCompleteSelectClub | from club.widgets.ajax_select import AutoCompleteSelectClub | ||||||
| from core.models import User | from core.models import User | ||||||
| from core.views.forms import ( | from core.views.forms import NFCTextInput, SelectDate, SelectDateTime | ||||||
|     FutureDateTimeField, |  | ||||||
|     NFCTextInput, |  | ||||||
|     SelectDate, |  | ||||||
|     SelectDateTime, |  | ||||||
| ) |  | ||||||
| from core.views.widgets.ajax_select import ( | from core.views.widgets.ajax_select import ( | ||||||
|     AutoCompleteSelect, |     AutoCompleteSelect, | ||||||
|     AutoCompleteSelectMultipleGroup, |     AutoCompleteSelectMultipleGroup, | ||||||
| @@ -32,14 +19,10 @@ from counter.models import ( | |||||||
|     Counter, |     Counter, | ||||||
|     Customer, |     Customer, | ||||||
|     Eticket, |     Eticket, | ||||||
|     InvoiceCall, |  | ||||||
|     Product, |     Product, | ||||||
|     Refilling, |     Refilling, | ||||||
|     ReturnableProduct, |     ReturnableProduct, | ||||||
|     ScheduledProductAction, |  | ||||||
|     Selling, |  | ||||||
|     StudentCard, |     StudentCard, | ||||||
|     get_product_actions, |  | ||||||
| ) | ) | ||||||
| from counter.widgets.ajax_select import ( | from counter.widgets.ajax_select import ( | ||||||
|     AutoCompleteSelectMultipleCounter, |     AutoCompleteSelectMultipleCounter, | ||||||
| @@ -175,101 +158,7 @@ class CounterEditForm(forms.ModelForm): | |||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class ScheduledProductActionForm(forms.ModelForm): | class ProductEditForm(forms.ModelForm): | ||||||
|     """Form for automatic product archiving. |  | ||||||
|  |  | ||||||
|     The `save` method will update or create tasks using celery-beat. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     required_css_class = "required" |  | ||||||
|     prefix = "scheduled" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = ScheduledProductAction |  | ||||||
|         fields = ["task"] |  | ||||||
|         widgets = {"task": forms.RadioSelect(choices=get_product_actions)} |  | ||||||
|         labels = {"task": _("Action")} |  | ||||||
|         help_texts = {"task": ""} |  | ||||||
|  |  | ||||||
|     trigger_at = FutureDateTimeField( |  | ||||||
|         label=_("Date and time of action"), widget=SelectDateTime |  | ||||||
|     ) |  | ||||||
|     counters = forms.ModelMultipleChoiceField( |  | ||||||
|         label=_("New counters"), |  | ||||||
|         help_text=_("The selected counters will replace the current ones"), |  | ||||||
|         required=False, |  | ||||||
|         widget=AutoCompleteSelectMultipleCounter, |  | ||||||
|         queryset=Counter.objects.all(), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def __init__(self, *args, product: Product, **kwargs): |  | ||||||
|         self.product = product |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         if not self.instance._state.adding: |  | ||||||
|             self.fields["trigger_at"].initial = self.instance.clocked.clocked_time |  | ||||||
|             self.fields["counters"].initial = json.loads(self.instance.kwargs).get( |  | ||||||
|                 "counters" |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|     def clean(self): |  | ||||||
|         if not self.changed_data or "trigger_at" in self.errors: |  | ||||||
|             return super().clean() |  | ||||||
|         if "trigger_at" in self.changed_data: |  | ||||||
|             if not self.instance.clocked_id: |  | ||||||
|                 self.instance.clocked = ClockedSchedule( |  | ||||||
|                     clocked_time=self.cleaned_data["trigger_at"] |  | ||||||
|                 ) |  | ||||||
|             else: |  | ||||||
|                 self.instance.clocked.clocked_time = self.cleaned_data["trigger_at"] |  | ||||||
|             self.instance.clocked.save() |  | ||||||
|         task_kwargs = {"product_id": self.product.id} |  | ||||||
|         if ( |  | ||||||
|             self.cleaned_data["task"] == "counter.tasks.change_counters" |  | ||||||
|             and "counters" in self.changed_data |  | ||||||
|         ): |  | ||||||
|             task_kwargs["counters"] = [c.id for c in self.cleaned_data["counters"]] |  | ||||||
|         self.instance.product = self.product |  | ||||||
|         self.instance.kwargs = json.dumps(task_kwargs) |  | ||||||
|         self.instance.name = ( |  | ||||||
|             f"{self.cleaned_data['task']} - {self.product} - {uuid.uuid4()}" |  | ||||||
|         ) |  | ||||||
|         return super().clean() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BaseScheduledProductActionFormSet(BaseModelFormSet): |  | ||||||
|     def __init__(self, *args, product: Product, **kwargs): |  | ||||||
|         if product.id: |  | ||||||
|             queryset = ( |  | ||||||
|                 product.scheduled_actions.filter( |  | ||||||
|                     enabled=True, clocked__clocked_time__gt=now() |  | ||||||
|                 ) |  | ||||||
|                 .order_by("clocked__clocked_time") |  | ||||||
|                 .select_related("clocked") |  | ||||||
|             ) |  | ||||||
|         else: |  | ||||||
|             queryset = ScheduledProductAction.objects.none() |  | ||||||
|         form_kwargs = {"product": product} |  | ||||||
|         super().__init__(*args, queryset=queryset, form_kwargs=form_kwargs, **kwargs) |  | ||||||
|  |  | ||||||
|     def delete_existing(self, obj: ScheduledProductAction, commit: bool = True):  # noqa FBT001 |  | ||||||
|         clocked = obj.clocked |  | ||||||
|         super().delete_existing(obj, commit=commit) |  | ||||||
|         if commit: |  | ||||||
|             clocked.delete() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ScheduledProductActionFormSet = forms.modelformset_factory( |  | ||||||
|     ScheduledProductAction, |  | ||||||
|     ScheduledProductActionForm, |  | ||||||
|     formset=BaseScheduledProductActionFormSet, |  | ||||||
|     absolute_max=None, |  | ||||||
|     can_delete=True, |  | ||||||
|     can_delete_extra=False, |  | ||||||
|     extra=2, |  | ||||||
| ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProductForm(forms.ModelForm): |  | ||||||
|     error_css_class = "error" |     error_css_class = "error" | ||||||
|     required_css_class = "required" |     required_css_class = "required" | ||||||
|  |  | ||||||
| @@ -310,21 +199,22 @@ class ProductForm(forms.ModelForm): | |||||||
|         queryset=Counter.objects.all(), |         queryset=Counter.objects.all(), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def __init__(self, *args, instance=None, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super().__init__(*args, instance=instance, **kwargs) |         super().__init__(*args, **kwargs) | ||||||
|         if self.instance.id: |         if self.instance.id: | ||||||
|             self.fields["counters"].initial = self.instance.counters.all() |             self.fields["counters"].initial = self.instance.counters.all() | ||||||
|         self.action_formset = ScheduledProductActionFormSet( |  | ||||||
|             *args, product=self.instance, **kwargs |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def is_valid(self): |  | ||||||
|         return super().is_valid() and self.action_formset.is_valid() |  | ||||||
|  |  | ||||||
|     def save(self, *args, **kwargs): |     def save(self, *args, **kwargs): | ||||||
|         ret = super().save(*args, **kwargs) |         ret = super().save(*args, **kwargs) | ||||||
|         self.instance.counters.set(self.cleaned_data["counters"]) |         if self.fields["counters"].initial: | ||||||
|         self.action_formset.save() |             # Remove the product from all counter it was added to | ||||||
|  |             # It will then only be added to selected counters | ||||||
|  |             for counter in self.fields["counters"].initial: | ||||||
|  |                 counter.products.remove(self.instance) | ||||||
|  |                 counter.save() | ||||||
|  |         for counter in self.cleaned_data["counters"]: | ||||||
|  |             counter.products.add(self.instance) | ||||||
|  |             counter.save() | ||||||
|         return ret |         return ret | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -376,7 +266,7 @@ class CloseCustomerAccountForm(forms.Form): | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class BasketProductForm(forms.Form): | class ProductForm(forms.Form): | ||||||
|     quantity = forms.IntegerField(min_value=1, required=True) |     quantity = forms.IntegerField(min_value=1, required=True) | ||||||
|     id = forms.IntegerField(min_value=0, required=True) |     id = forms.IntegerField(min_value=0, required=True) | ||||||
|  |  | ||||||
| @@ -481,50 +371,5 @@ class BaseBasketForm(forms.BaseFormSet): | |||||||
|  |  | ||||||
|  |  | ||||||
| BasketForm = forms.formset_factory( | BasketForm = forms.formset_factory( | ||||||
|     BasketProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 |     ProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 | ||||||
| ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class InvoiceCallForm(forms.Form): |  | ||||||
|     def __init__(self, *args, month: date, **kwargs): |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|         self.month = month |  | ||||||
|         self.clubs = list( |  | ||||||
|             Club.objects.filter( |  | ||||||
|                 Exists( |  | ||||||
|                     Selling.objects.filter( |  | ||||||
|                         club=OuterRef("pk"), |  | ||||||
|                         date__gte=month, |  | ||||||
|                         date__lte=month + relativedelta(months=1), |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             ).annotate( |  | ||||||
|                 validated_invoice=Exists( |  | ||||||
|                     InvoiceCall.objects.filter( |  | ||||||
|                         club=OuterRef("pk"), month=month, is_validated=True |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
|         self.fields = { |  | ||||||
|             str(club.id): forms.BooleanField( |  | ||||||
|                 required=False, initial=club.validated_invoice |  | ||||||
|             ) |  | ||||||
|             for club in self.clubs |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def save(self): |  | ||||||
|         invoice_calls = [ |  | ||||||
|             InvoiceCall( |  | ||||||
|                 month=self.month, |  | ||||||
|                 club_id=club.id, |  | ||||||
|                 is_validated=self.cleaned_data.get(str(club.id), False), |  | ||||||
|             ) |  | ||||||
|             for club in self.clubs |  | ||||||
|         ] |  | ||||||
|         InvoiceCall.objects.bulk_create( |  | ||||||
|             invoice_calls, |  | ||||||
|             update_conflicts=True, |  | ||||||
|             update_fields=["is_validated"], |  | ||||||
|             unique_fields=["month", "club"], |  | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -1,40 +0,0 @@ | |||||||
| # Generated by Django 5.2.3 on 2025-09-14 11:29 |  | ||||||
|  |  | ||||||
| import django.db.models.deletion |  | ||||||
| from django.db import migrations, models |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|     dependencies = [ |  | ||||||
|         ("counter", "0031_alter_counter_options"), |  | ||||||
|         ("django_celery_beat", "0019_alter_periodictasks_options"), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name="ScheduledProductAction", |  | ||||||
|             fields=[ |  | ||||||
|                 ( |  | ||||||
|                     "periodictask_ptr", |  | ||||||
|                     models.OneToOneField( |  | ||||||
|                         auto_created=True, |  | ||||||
|                         on_delete=django.db.models.deletion.CASCADE, |  | ||||||
|                         parent_link=True, |  | ||||||
|                         primary_key=True, |  | ||||||
|                         serialize=False, |  | ||||||
|                         to="django_celery_beat.periodictask", |  | ||||||
|                     ), |  | ||||||
|                 ), |  | ||||||
|                 ( |  | ||||||
|                     "product", |  | ||||||
|                     models.ForeignKey( |  | ||||||
|                         on_delete=django.db.models.deletion.CASCADE, |  | ||||||
|                         related_name="scheduled_actions", |  | ||||||
|                         to="counter.product", |  | ||||||
|                     ), |  | ||||||
|                 ), |  | ||||||
|             ], |  | ||||||
|             options={"verbose_name": "Product scheduled action"}, |  | ||||||
|             bases=("django_celery_beat.periodictask",), |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -1,51 +0,0 @@ | |||||||
| # Generated by Django 5.2.3 on 2025-10-15 21:54 |  | ||||||
|  |  | ||||||
| import django.db.models.deletion |  | ||||||
| from django.db import migrations, models |  | ||||||
|  |  | ||||||
| import counter.models |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|     dependencies = [ |  | ||||||
|         ("club", "0014_alter_club_options_rename_unix_name_club_slug_name_and_more"), |  | ||||||
|         ("counter", "0032_scheduledproductaction"), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.CreateModel( |  | ||||||
|             name="InvoiceCall", |  | ||||||
|             fields=[ |  | ||||||
|                 ( |  | ||||||
|                     "id", |  | ||||||
|                     models.AutoField( |  | ||||||
|                         auto_created=True, |  | ||||||
|                         primary_key=True, |  | ||||||
|                         serialize=False, |  | ||||||
|                         verbose_name="ID", |  | ||||||
|                     ), |  | ||||||
|                 ), |  | ||||||
|                 ( |  | ||||||
|                     "is_validated", |  | ||||||
|                     models.BooleanField(default=False, verbose_name="is validated"), |  | ||||||
|                 ), |  | ||||||
|                 ("month", counter.models.MonthField(verbose_name="invoice date")), |  | ||||||
|                 ( |  | ||||||
|                     "club", |  | ||||||
|                     models.ForeignKey( |  | ||||||
|                         on_delete=django.db.models.deletion.CASCADE, to="club.club" |  | ||||||
|                     ), |  | ||||||
|                 ), |  | ||||||
|             ], |  | ||||||
|             options={ |  | ||||||
|                 "verbose_name": "Invoice call", |  | ||||||
|                 "verbose_name_plural": "Invoice calls", |  | ||||||
|                 "constraints": [ |  | ||||||
|                     models.UniqueConstraint( |  | ||||||
|                         fields=("club", "month"), |  | ||||||
|                         name="counter_invoicecall_unique_club_month", |  | ||||||
|                     ) |  | ||||||
|                 ], |  | ||||||
|             }, |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @@ -15,7 +15,6 @@ | |||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| import base64 | import base64 | ||||||
| import contextlib |  | ||||||
| import os | import os | ||||||
| import random | import random | ||||||
| import string | import string | ||||||
| @@ -35,7 +34,6 @@ from django.urls import reverse | |||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from django_celery_beat.models import PeriodicTask |  | ||||||
| from django_countries.fields import CountryField | from django_countries.fields import CountryField | ||||||
| from ordered_model.models import OrderedModel | from ordered_model.models import OrderedModel | ||||||
| from phonenumber_field.modelfields import PhoneNumberField | from phonenumber_field.modelfields import PhoneNumberField | ||||||
| @@ -447,8 +445,7 @@ class Product(models.Model): | |||||||
|         buying_groups = list(self.buying_groups.all()) |         buying_groups = list(self.buying_groups.all()) | ||||||
|         if not buying_groups: |         if not buying_groups: | ||||||
|             return True |             return True | ||||||
|         res = any(user.is_in_group(pk=group.id) for group in buying_groups) |         return any(user.is_in_group(pk=group.id) for group in buying_groups) | ||||||
|         return res |  | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def profit(self): |     def profit(self): | ||||||
| @@ -482,7 +479,7 @@ class CounterQuerySet(models.QuerySet): | |||||||
|         return self.annotate(has_annotated_barman=Exists(subquery)) |         return self.annotate(has_annotated_barman=Exists(subquery)) | ||||||
|  |  | ||||||
|     def annotate_is_open(self) -> Self: |     def annotate_is_open(self) -> Self: | ||||||
|         """Annotate the queryset with the `is_open` field. |         """Annotate tue queryset with the `is_open` field. | ||||||
|  |  | ||||||
|         For each counter, if `is_open=True`, then the counter is currently opened. |         For each counter, if `is_open=True`, then the counter is currently opened. | ||||||
|         Else the counter is closed. |         Else the counter is closed. | ||||||
| @@ -884,6 +881,7 @@ class Selling(models.Model): | |||||||
|             if ( |             if ( | ||||||
|                 self.product |                 self.product | ||||||
|                 and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_ONE_SEMESTER |                 and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_ONE_SEMESTER | ||||||
|  |                 and self.counter.type == "EBOUTIC" | ||||||
|             ): |             ): | ||||||
|                 sub = Subscription( |                 sub = Subscription( | ||||||
|                     member=user, |                     member=user, | ||||||
| @@ -907,6 +905,7 @@ class Selling(models.Model): | |||||||
|             elif ( |             elif ( | ||||||
|                 self.product |                 self.product | ||||||
|                 and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_TWO_SEMESTERS |                 and self.product.id == settings.SITH_PRODUCT_SUBSCRIPTION_TWO_SEMESTERS | ||||||
|  |                 and self.counter.type == "EBOUTIC" | ||||||
|             ): |             ): | ||||||
|                 sub = Subscription( |                 sub = Subscription( | ||||||
|                     member=user, |                     member=user, | ||||||
| @@ -1360,85 +1359,3 @@ class ReturnableProductBalance(models.Model): | |||||||
|             f"return balance of {self.customer} " |             f"return balance of {self.customer} " | ||||||
|             f"for {self.returnable.product_id} : {self.balance}" |             f"for {self.returnable.product_id} : {self.balance}" | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_product_actions(): |  | ||||||
|     return [ |  | ||||||
|         ("counter.tasks.archive_product", _("Archiving")), |  | ||||||
|         ("counter.tasks.change_counters", _("Counters change")), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ScheduledProductAction(PeriodicTask): |  | ||||||
|     """Extension of celery-beat tasks dedicated to perform actions on Product.""" |  | ||||||
|  |  | ||||||
|     product = models.ForeignKey( |  | ||||||
|         Product, related_name="scheduled_actions", on_delete=models.CASCADE |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _("Product scheduled action") |  | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |  | ||||||
|         self._meta.get_field("task").choices = get_product_actions() |  | ||||||
|         super().__init__(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     def full_clean(self, *args, **kwargs): |  | ||||||
|         self.one_off = True  # A product action should occur one time only |  | ||||||
|         return super().full_clean(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     def clean_clocked(self): |  | ||||||
|         if not self.clocked: |  | ||||||
|             raise ValidationError(_("Product actions must declare a clocked schedule.")) |  | ||||||
|  |  | ||||||
|     def validate_unique(self, *args, **kwargs): |  | ||||||
|         # The checks done in PeriodicTask.validate_unique aren't |  | ||||||
|         # adapted in the case of scheduled product action, |  | ||||||
|         # so we skip it and execute directly Model.validate_unique |  | ||||||
|         return super(PeriodicTask, self).validate_unique(*args, **kwargs) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class MonthField(models.DateField): |  | ||||||
|     description = _("Year + month field (day forced to 1)") |  | ||||||
|     default_error_messages = { |  | ||||||
|         "invalid": _( |  | ||||||
|             "“%(value)s” value has an invalid date format. It must be " |  | ||||||
|             "in YYYY-MM format." |  | ||||||
|         ), |  | ||||||
|         "invalid_date": _( |  | ||||||
|             "“%(value)s” value has the correct format (YYYY-MM) " |  | ||||||
|             "but it is an invalid date." |  | ||||||
|         ), |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def to_python(self, value): |  | ||||||
|         if isinstance(value, str): |  | ||||||
|             with contextlib.suppress(ValueError): |  | ||||||
|                 # If the string is given as YYYY-mm, try to parse it. |  | ||||||
|                 # If it fails, it means that the string may be in the form YYYY-mm-dd |  | ||||||
|                 # or in an invalid format. |  | ||||||
|                 # Whatever the case, we let Django deal with it |  | ||||||
|                 # and raise an error if needed |  | ||||||
|                 value = datetime.strptime(value, "%Y-%m") |  | ||||||
|         value = super().to_python(value) |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|         return value.replace(day=1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class InvoiceCall(models.Model): |  | ||||||
|     is_validated = models.BooleanField(verbose_name=_("is validated"), default=False) |  | ||||||
|     club = models.ForeignKey(Club, on_delete=models.CASCADE) |  | ||||||
|     month = MonthField(verbose_name=_("invoice date")) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _("Invoice call") |  | ||||||
|         verbose_name_plural = _("Invoice calls") |  | ||||||
|         constraints = [ |  | ||||||
|             models.UniqueConstraint( |  | ||||||
|                 fields=["club", "month"], name="counter_invoicecall_unique_club_month" |  | ||||||
|             ) |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|     def __str__(self): |  | ||||||
|         return f"invoice call of {self.month} made by {self.club}" |  | ||||||
|   | |||||||
| @@ -39,7 +39,6 @@ | |||||||
|   flex: auto; |   flex: auto; | ||||||
|   margin: 0.2em; |   margin: 0.2em; | ||||||
|   width: 20%; |   width: 20%; | ||||||
|   min-width: 350px; |  | ||||||
|  |  | ||||||
|   ul { |   ul { | ||||||
|     list-style-type: none; |     list-style-type: none; | ||||||
|   | |||||||
| @@ -1,19 +0,0 @@ | |||||||
| # Create your tasks here |  | ||||||
|  |  | ||||||
| from celery import shared_task |  | ||||||
|  |  | ||||||
| from counter.models import Counter, Product |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @shared_task |  | ||||||
| def archive_product(*, product_id: int, **kwargs): |  | ||||||
|     product = Product.objects.get(id=product_id) |  | ||||||
|     product.archived = True |  | ||||||
|     product.save() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @shared_task |  | ||||||
| def change_counters(*, product_id: int, counters: list[int], **kwargs): |  | ||||||
|     product = Product.objects.get(id=product_id) |  | ||||||
|     counters = Counter.objects.filter(id__in=counters) |  | ||||||
|     product.counters.set(counters) |  | ||||||
| @@ -67,13 +67,13 @@ | |||||||
|                 <option value="FIN">{% trans %}Confirm (FIN){% endtrans %}</option> |                 <option value="FIN">{% trans %}Confirm (FIN){% endtrans %}</option> | ||||||
|                 <option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option> |                 <option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option> | ||||||
|               </optgroup> |               </optgroup> | ||||||
|               {%- for category in categories.keys() -%} |               {% for category in categories.keys() %} | ||||||
|                 <optgroup label="{{ category }}"> |                 <optgroup label="{{ category }}"> | ||||||
|                   {%- for product in categories[category] -%} |                   {% for product in categories[category] %} | ||||||
|                     <option value="{{ product.id }}">{{ product }}</option> |                     <option value="{{ product.id }}">{{ product }}</option> | ||||||
|                   {%- endfor -%} |                   {% endfor %} | ||||||
|                 </optgroup> |                 </optgroup> | ||||||
|               {%- endfor -%} |               {% endfor %} | ||||||
|             </counter-product-select> |             </counter-product-select> | ||||||
|  |  | ||||||
|             <input type="submit" value="{% trans %}Go{% endtrans %}"/> |             <input type="submit" value="{% trans %}Go{% endtrans %}"/> | ||||||
|   | |||||||
| @@ -4,49 +4,35 @@ | |||||||
|   {% trans %}Invoices call{% endtrans %} |   {% trans %}Invoices call{% endtrans %} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block notifications %}{# Notifications are moved below #}{% endblock %} |  | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   <h3>{% trans date=start_date|date("F Y") %}Invoices call for {{ date }}{% endtrans %}</h3> |   <h3>{% trans date=start_date|date("F Y") %}Invoices call for {{ date }}{% endtrans %}</h3> | ||||||
|  |   <p>{% trans %}Choose another month: {% endtrans %}</p> | ||||||
|   <form method="get" action=""> |   <form method="get" action=""> | ||||||
|     <label for="id_form_other_month">{% trans %}Choose another month: {% endtrans %}</label> |     <select name="month"> | ||||||
|     <select name="month" id="id_form_other_month"> |  | ||||||
|       {% for m in months %} |       {% for m in months %} | ||||||
|         <option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option> |         <option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </select> |     </select> | ||||||
|     <input type="submit" value="{% trans %}Go{% endtrans %}" /> |     <input type="submit" value="{% trans %}Go{% endtrans %}" /> | ||||||
|   </form> |   </form> | ||||||
|  |  | ||||||
|   <br> |   <br> | ||||||
|   <p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p> |   <p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p> | ||||||
|   <br> |   <br> | ||||||
|  |  | ||||||
|   {% include "core/base/notifications.jinja" %} |  | ||||||
|  |  | ||||||
|   <form method="post" action=""> |  | ||||||
|     {% csrf_token %} |  | ||||||
|   <table> |   <table> | ||||||
|     <thead> |     <thead> | ||||||
|         <tr> |  | ||||||
|       <td>{% trans %}Club{% endtrans %}</td> |       <td>{% trans %}Club{% endtrans %}</td> | ||||||
|       <td>{% trans %}Sum{% endtrans %}</td> |       <td>{% trans %}Sum{% endtrans %}</td> | ||||||
|           <td>{% trans %}Validated{% endtrans %}</td> |  | ||||||
|         </tr> |  | ||||||
|     </thead> |     </thead> | ||||||
|     <tbody> |     <tbody> | ||||||
|         {% for invoice in invoices %} |       {% for i in sums %} | ||||||
|         <tr> |         <tr> | ||||||
|             <td>{{ invoice.club__name }}</td> |           <td>{{ i['club__name'] }}</td> | ||||||
|             <td>{{ "%.2f"|format(invoice.selling_sum) }} €</td> |           <td>{{ i['selling_sum'] }} €</td> | ||||||
|             <td> |  | ||||||
|               {{ form[invoice.club_id|string] }} |  | ||||||
|             </td> |  | ||||||
|         </tr> |         </tr> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </tbody> |     </tbody> | ||||||
|   </table> |   </table> | ||||||
|     <input type="hidden" name="month" value="{{ start_date|date('Y-m') }}"> |  | ||||||
|     <button type="submit">{% trans %}Save{% endtrans %}</button> |  | ||||||
|   </form> |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,56 +0,0 @@ | |||||||
| {% extends "core/base.jinja" %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
|   {% if object %} |  | ||||||
|     <h2>{% trans name=object %}Edit product {{ name }}{% endtrans %}</h2> |  | ||||||
|   {% else %} |  | ||||||
|     <h2>{% trans %}Product creation{% endtrans %}</h2> |  | ||||||
|   {% endif %} |  | ||||||
|   <form method="post"> |  | ||||||
|     {% csrf_token %} |  | ||||||
|     {{ form.as_p() }} |  | ||||||
|  |  | ||||||
|     <br /> |  | ||||||
|  |  | ||||||
|     <h3>{% trans %}Automatic actions{% endtrans %}</h3> |  | ||||||
|  |  | ||||||
|     <p class="margin-bottom"> |  | ||||||
|       <em> |  | ||||||
|         {%- trans trimmed -%} |  | ||||||
|           Automatic actions allows to schedule product changes |  | ||||||
|           ahead of time. |  | ||||||
|         {%- endtrans -%} |  | ||||||
|       </em> |  | ||||||
|     </p> |  | ||||||
|  |  | ||||||
|     {{ form.action_formset.management_form }} |  | ||||||
|     {%- for action_form in form.action_formset.forms -%} |  | ||||||
|       <fieldset x-data="{action: '{{ action_form.task.initial }}'}"> |  | ||||||
|         {{ action_form.non_field_errors() }} |  | ||||||
|         <div class="row gap-2x margin-bottom"> |  | ||||||
|           <div> |  | ||||||
|             {{ action_form.task.errors }} |  | ||||||
|             {{ action_form.task.label_tag() }} |  | ||||||
|             {{ action_form.task|add_attr("x-model=action") }} |  | ||||||
|           </div> |  | ||||||
|           <div>{{ action_form.trigger_at.as_field_group() }}</div> |  | ||||||
|         </div> |  | ||||||
|         <div x-show="action==='counter.tasks.change_counters'" class="margin-bottom"> |  | ||||||
|           {{ action_form.counters.as_field_group() }} |  | ||||||
|         </div> |  | ||||||
|         {%- if action_form.DELETE -%} |  | ||||||
|           <div class="row gap"> |  | ||||||
|             {{ action_form.DELETE.as_field_group() }} |  | ||||||
|           </div> |  | ||||||
|         {%- endif -%} |  | ||||||
|         {%- for field in action_form.hidden_fields() -%} |  | ||||||
|           {{ field }} |  | ||||||
|         {%- endfor -%} |  | ||||||
|       </fieldset> |  | ||||||
|       {%- if not loop.last -%} |  | ||||||
|         <hr class="margin-bottom"> |  | ||||||
|       {%- endif -%} |  | ||||||
|     {%- endfor -%} |  | ||||||
|     <p><input type="submit" value="{% trans %}Save{% endtrans %}" /></p> |  | ||||||
|   </form> |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,116 +0,0 @@ | |||||||
| import json |  | ||||||
| from datetime import timedelta |  | ||||||
|  |  | ||||||
| import pytest |  | ||||||
| from django.conf import settings |  | ||||||
| from django.test import Client |  | ||||||
| from django.urls import reverse |  | ||||||
| from django.utils.timezone import now |  | ||||||
| from django_celery_beat.models import ClockedSchedule |  | ||||||
| from model_bakery import baker |  | ||||||
|  |  | ||||||
| from core.models import Group, User |  | ||||||
| from counter.baker_recipes import counter_recipe, product_recipe |  | ||||||
| from counter.forms import ScheduledProductActionForm, ScheduledProductActionFormSet |  | ||||||
| from counter.models import ScheduledProductAction |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| def test_edit_product(client: Client): |  | ||||||
|     client.force_login( |  | ||||||
|         baker.make( |  | ||||||
|             User, groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)] |  | ||||||
|         ) |  | ||||||
|     ) |  | ||||||
|     product = product_recipe.make() |  | ||||||
|     url = reverse("counter:product_edit", kwargs={"product_id": product.id}) |  | ||||||
|     res = client.get(url) |  | ||||||
|     assert res.status_code == 200 |  | ||||||
|  |  | ||||||
|     res = client.post(url, data={}) |  | ||||||
|     # This is actually a failure, but we just want to check that |  | ||||||
|     # we don't have a 403 or a 500. |  | ||||||
|     # The actual behaviour will be tested directly on the form. |  | ||||||
|     assert res.status_code == 200 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| class TestProductActionForm: |  | ||||||
|     def test_single_form_archive(self): |  | ||||||
|         product = product_recipe.make() |  | ||||||
|         trigger_at = now() + timedelta(minutes=10) |  | ||||||
|         form = ScheduledProductActionForm( |  | ||||||
|             product=product, |  | ||||||
|             data={ |  | ||||||
|                 "scheduled-task": "counter.tasks.archive_product", |  | ||||||
|                 "scheduled-trigger_at": trigger_at, |  | ||||||
|             }, |  | ||||||
|         ) |  | ||||||
|         assert form.is_valid() |  | ||||||
|         instance = form.save() |  | ||||||
|         assert instance.clocked.clocked_time == trigger_at |  | ||||||
|         assert instance.enabled is True |  | ||||||
|         assert instance.one_off is True |  | ||||||
|         assert instance.task == "counter.tasks.archive_product" |  | ||||||
|         assert instance.kwargs == json.dumps({"product_id": product.id}) |  | ||||||
|  |  | ||||||
|     def test_single_form_change_counters(self): |  | ||||||
|         product = product_recipe.make() |  | ||||||
|         counter = counter_recipe.make() |  | ||||||
|         trigger_at = now() + timedelta(minutes=10) |  | ||||||
|         form = ScheduledProductActionForm( |  | ||||||
|             product=product, |  | ||||||
|             data={ |  | ||||||
|                 "scheduled-task": "counter.tasks.change_counters", |  | ||||||
|                 "scheduled-trigger_at": trigger_at, |  | ||||||
|                 "scheduled-counters": [counter.id], |  | ||||||
|             }, |  | ||||||
|         ) |  | ||||||
|         assert form.is_valid() |  | ||||||
|         instance = form.save() |  | ||||||
|         instance.refresh_from_db() |  | ||||||
|         assert instance.clocked.clocked_time == trigger_at |  | ||||||
|         assert instance.enabled is True |  | ||||||
|         assert instance.one_off is True |  | ||||||
|         assert instance.task == "counter.tasks.change_counters" |  | ||||||
|         assert instance.kwargs == json.dumps( |  | ||||||
|             {"product_id": product.id, "counters": [counter.id]} |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def test_delete(self): |  | ||||||
|         product = product_recipe.make() |  | ||||||
|         clocked = baker.make(ClockedSchedule, clocked_time=now() + timedelta(minutes=2)) |  | ||||||
|         task = baker.make( |  | ||||||
|             ScheduledProductAction, |  | ||||||
|             product=product, |  | ||||||
|             one_off=True, |  | ||||||
|             clocked=clocked, |  | ||||||
|             task="counter.tasks.archive_product", |  | ||||||
|         ) |  | ||||||
|         formset = ScheduledProductActionFormSet(product=product) |  | ||||||
|         formset.delete_existing(task) |  | ||||||
|         assert not ScheduledProductAction.objects.filter(id=task.id).exists() |  | ||||||
|         assert not ClockedSchedule.objects.filter(id=clocked.id).exists() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| class TestProductActionFormSet: |  | ||||||
|     def test_ok(self): |  | ||||||
|         product = product_recipe.make() |  | ||||||
|         counter = counter_recipe.make() |  | ||||||
|         trigger_at = now() + timedelta(minutes=10) |  | ||||||
|         formset = ScheduledProductActionFormSet( |  | ||||||
|             product=product, |  | ||||||
|             data={ |  | ||||||
|                 "form-TOTAL_FORMS": "2", |  | ||||||
|                 "form-INITIAL_FORMS": "0", |  | ||||||
|                 "form-0-task": "counter.tasks.archive_product", |  | ||||||
|                 "form-0-trigger_at": trigger_at, |  | ||||||
|                 "form-1-task": "counter.tasks.change_counters", |  | ||||||
|                 "form-1-trigger_at": trigger_at, |  | ||||||
|                 "form-1-counters": [counter.id], |  | ||||||
|             }, |  | ||||||
|         ) |  | ||||||
|         assert formset.is_valid() |  | ||||||
|         formset.save() |  | ||||||
|         assert ScheduledProductAction.objects.filter(product=product).count() == 2 |  | ||||||
| @@ -1,76 +0,0 @@ | |||||||
| from datetime import date, datetime |  | ||||||
|  |  | ||||||
| import pytest |  | ||||||
| from dateutil.relativedelta import relativedelta |  | ||||||
| from django.contrib.auth.models import Permission |  | ||||||
| from django.core.exceptions import ValidationError |  | ||||||
| from django.test import Client |  | ||||||
| from django.urls import reverse |  | ||||||
| from django.utils.timezone import localdate |  | ||||||
| from model_bakery import baker |  | ||||||
| from pytest_django.asserts import assertRedirects |  | ||||||
|  |  | ||||||
| from club.models import Club |  | ||||||
| from core.models import User |  | ||||||
| from counter.baker_recipes import sale_recipe |  | ||||||
| from counter.forms import InvoiceCallForm |  | ||||||
| from counter.models import Customer, InvoiceCall, Selling |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| @pytest.mark.parametrize( |  | ||||||
|     "month", [date(2025, 10, 20), "2025-10", datetime(2025, 10, 15, 12, 30)] |  | ||||||
| ) |  | ||||||
| def test_invoice_date_with_date(month: date | datetime | str): |  | ||||||
|     club = baker.make(Club) |  | ||||||
|     invoice = InvoiceCall.objects.create(club=club, month=month) |  | ||||||
|     invoice.refresh_from_db() |  | ||||||
|     assert not invoice.is_validated |  | ||||||
|     assert invoice.month == date(2025, 10, 1) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| def test_invoice_call_invalid_month_string(): |  | ||||||
|     club = baker.make(Club) |  | ||||||
|  |  | ||||||
|     with pytest.raises(ValidationError): |  | ||||||
|         InvoiceCall.objects.create(club=club, month="2025-13") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| @pytest.mark.parametrize("query", [None, {"month": "2025-08"}]) |  | ||||||
| def test_invoice_call_view(client: Client, query: dict | None): |  | ||||||
|     user = baker.make( |  | ||||||
|         User, |  | ||||||
|         user_permissions=[ |  | ||||||
|             *Permission.objects.filter( |  | ||||||
|                 codename__in=["view_invoicecall", "change_invoicecall"] |  | ||||||
|             ) |  | ||||||
|         ], |  | ||||||
|     ) |  | ||||||
|     client.force_login(user) |  | ||||||
|     url = reverse("counter:invoices_call", query=query) |  | ||||||
|     assert client.get(url).status_code == 200 |  | ||||||
|     assertRedirects(client.post(url), url) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @pytest.mark.django_db |  | ||||||
| def test_invoice_call_form(): |  | ||||||
|     Selling.objects.all().delete() |  | ||||||
|     month = localdate() - relativedelta(months=1) |  | ||||||
|     clubs = baker.make(Club, _quantity=2) |  | ||||||
|     recipe = sale_recipe.extend(date=month, customer=baker.make(Customer, amount=10000)) |  | ||||||
|     recipe.make(club=clubs[0], quantity=2, unit_price=200) |  | ||||||
|     recipe.make(club=clubs[0], quantity=3, unit_price=5) |  | ||||||
|     recipe.make(club=clubs[1], quantity=20, unit_price=10) |  | ||||||
|     form = InvoiceCallForm( |  | ||||||
|         month=month, data={str(clubs[0].id): True, str(clubs[1].id): False} |  | ||||||
|     ) |  | ||||||
|     assert form.is_valid() |  | ||||||
|     form.save() |  | ||||||
|     assert InvoiceCall.objects.filter( |  | ||||||
|         club=clubs[0], month=month, is_validated=True |  | ||||||
|     ).exists() |  | ||||||
|     assert InvoiceCall.objects.filter( |  | ||||||
|         club=clubs[1], month=month, is_validated=False |  | ||||||
|     ).exists() |  | ||||||
| @@ -6,16 +6,14 @@ import pytest | |||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.core.cache import cache | from django.core.cache import cache | ||||||
| from django.core.files.uploadedfile import SimpleUploadedFile | from django.core.files.uploadedfile import SimpleUploadedFile | ||||||
| from django.test import Client, TestCase | from django.test import Client | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| from model_bakery import baker | from model_bakery import baker | ||||||
| from PIL import Image | from PIL import Image | ||||||
| from pytest_django.asserts import assertNumQueries, assertRedirects | from pytest_django.asserts import assertNumQueries | ||||||
|  |  | ||||||
| from club.models import Club |  | ||||||
| from core.baker_recipes import board_user, subscriber_user | from core.baker_recipes import board_user, subscriber_user | ||||||
| from core.models import Group, User | from core.models import Group, User | ||||||
| from counter.forms import ProductForm |  | ||||||
| from counter.models import Product, ProductType | from counter.models import Product, ProductType | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -86,49 +84,3 @@ def test_fetch_product_nb_queries(client: Client): | |||||||
|         # - 1 for the actual request |         # - 1 for the actual request | ||||||
|         # - 1 to prefetch the related buying_groups |         # - 1 to prefetch the related buying_groups | ||||||
|         client.get(reverse("api:search_products_detailed")) |         client.get(reverse("api:search_products_detailed")) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestCreateProduct(TestCase): |  | ||||||
|     @classmethod |  | ||||||
|     def setUpTestData(cls): |  | ||||||
|         cls.product_type = baker.make(ProductType) |  | ||||||
|         cls.club = baker.make(Club) |  | ||||||
|         cls.data = { |  | ||||||
|             "name": "foo", |  | ||||||
|             "description": "bar", |  | ||||||
|             "product_type": cls.product_type.id, |  | ||||||
|             "club": cls.club.id, |  | ||||||
|             "code": "FOO", |  | ||||||
|             "purchase_price": 1.0, |  | ||||||
|             "selling_price": 1.0, |  | ||||||
|             "special_selling_price": 1.0, |  | ||||||
|             "limit_age": 0, |  | ||||||
|             "form-TOTAL_FORMS": 0, |  | ||||||
|             "form-INITIAL_FORMS": 0, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     def test_form(self): |  | ||||||
|         form = ProductForm(data=self.data) |  | ||||||
|         assert form.is_valid() |  | ||||||
|         instance = form.save() |  | ||||||
|         assert instance.club == self.club |  | ||||||
|         assert instance.product_type == self.product_type |  | ||||||
|         assert instance.name == "foo" |  | ||||||
|         assert instance.selling_price == 1.0 |  | ||||||
|  |  | ||||||
|     def test_view(self): |  | ||||||
|         self.client.force_login( |  | ||||||
|             baker.make( |  | ||||||
|                 User, |  | ||||||
|                 groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)], |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
|         url = reverse("counter:new_product") |  | ||||||
|         response = self.client.get(url) |  | ||||||
|         assert response.status_code == 200 |  | ||||||
|         response = self.client.post(url, data=self.data) |  | ||||||
|         assertRedirects(response, reverse("counter:product_list")) |  | ||||||
|         product = Product.objects.last() |  | ||||||
|         assert product.name == "foo" |  | ||||||
|         assert product.club == self.club |  | ||||||
|         assert product.product_type == self.product_type |  | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ from core.utils import get_semester_code, get_start_of_semester | |||||||
| from counter.forms import ( | from counter.forms import ( | ||||||
|     CloseCustomerAccountForm, |     CloseCustomerAccountForm, | ||||||
|     CounterEditForm, |     CounterEditForm, | ||||||
|     ProductForm, |     ProductEditForm, | ||||||
|     ReturnableProductForm, |     ReturnableProductForm, | ||||||
| ) | ) | ||||||
| from counter.models import ( | from counter.models import ( | ||||||
| @@ -146,8 +146,8 @@ class ProductCreateView(CounterAdminTabsMixin, CounterAdminMixin, CreateView): | |||||||
|     """A create view for the admins.""" |     """A create view for the admins.""" | ||||||
|  |  | ||||||
|     model = Product |     model = Product | ||||||
|     form_class = ProductForm |     form_class = ProductEditForm | ||||||
|     template_name = "counter/product_form.jinja" |     template_name = "core/create.jinja" | ||||||
|     current_tab = "products" |     current_tab = "products" | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -155,9 +155,9 @@ class ProductEditView(CounterAdminTabsMixin, CounterAdminMixin, UpdateView): | |||||||
|     """An edit view for the admins.""" |     """An edit view for the admins.""" | ||||||
|  |  | ||||||
|     model = Product |     model = Product | ||||||
|     form_class = ProductForm |     form_class = ProductEditForm | ||||||
|     pk_url_kwarg = "product_id" |     pk_url_kwarg = "product_id" | ||||||
|     template_name = "counter/product_form.jinja" |     template_name = "core/edit.jinja" | ||||||
|     current_tab = "products" |     current_tab = "products" | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,81 +12,77 @@ | |||||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | # OR WITHIN THE LOCAL FILE "LICENSE" | ||||||
| # | # | ||||||
| # | # | ||||||
| from datetime import datetime | from datetime import datetime, timedelta | ||||||
| from urllib.parse import urlencode | from datetime import timezone as tz | ||||||
|  |  | ||||||
| from dateutil.relativedelta import relativedelta | from django.db.models import F | ||||||
| from django.contrib.auth.mixins import PermissionRequiredMixin | from django.utils import timezone | ||||||
| from django.contrib.messages.views import SuccessMessageMixin | from django.views.generic import TemplateView | ||||||
| from django.db.models import F, Sum |  | ||||||
| from django.utils.timezone import localdate, make_aware |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
| from django.views.generic import FormView |  | ||||||
|  |  | ||||||
| from counter.forms import InvoiceCallForm | from counter.fields import CurrencyField | ||||||
| from counter.models import Refilling, Selling | from counter.models import Refilling, Selling | ||||||
| from counter.views.mixins import CounterAdminTabsMixin | from counter.views.mixins import CounterAdminMixin, CounterAdminTabsMixin | ||||||
|  |  | ||||||
|  |  | ||||||
| class InvoiceCallView( | class InvoiceCallView(CounterAdminTabsMixin, CounterAdminMixin, TemplateView): | ||||||
|     CounterAdminTabsMixin, PermissionRequiredMixin, SuccessMessageMixin, FormView |  | ||||||
| ): |  | ||||||
|     template_name = "counter/invoices_call.jinja" |     template_name = "counter/invoices_call.jinja" | ||||||
|     current_tab = "invoices_call" |     current_tab = "invoices_call" | ||||||
|     permission_required = ["counter.view_invoicecall", "counter.change_invoicecall"] |  | ||||||
|     form_class = InvoiceCallForm |  | ||||||
|     success_message = _("Invoice calls status has been updated.") |  | ||||||
|  |  | ||||||
|     def get_month(self): |  | ||||||
|         kwargs = self.request.GET or self.request.POST |  | ||||||
|         if "month" in kwargs: |  | ||||||
|             return make_aware(datetime.strptime(kwargs["month"], "%Y-%m")) |  | ||||||
|         return localdate().replace(day=1) - relativedelta(months=1) |  | ||||||
|  |  | ||||||
|     def get_form_kwargs(self): |  | ||||||
|         return super().get_form_kwargs() | {"month": self.get_month()} |  | ||||||
|  |  | ||||||
|     def form_valid(self, form): |  | ||||||
|         form.save() |  | ||||||
|         return super().form_valid(form) |  | ||||||
|  |  | ||||||
|     def get_success_url(self): |  | ||||||
|         # redirect to the month from which the request is originated |  | ||||||
|         url = self.request.path |  | ||||||
|         kwargs = self.request.GET or self.request.POST |  | ||||||
|         if "month" in kwargs: |  | ||||||
|             query = urlencode({"month": kwargs["month"]}) |  | ||||||
|             url += f"?{query}" |  | ||||||
|         return url |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         """Add sums to the context.""" |         """Add sums to the context.""" | ||||||
|         kwargs = super().get_context_data(**kwargs) |         kwargs = super().get_context_data(**kwargs) | ||||||
|         kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC") |         kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC") | ||||||
|         start_date = self.get_month() |         if "month" in self.request.GET: | ||||||
|         end_date = start_date + relativedelta(months=1) |             start_date = datetime.strptime(self.request.GET["month"], "%Y-%m") | ||||||
|  |         else: | ||||||
|  |             start_date = datetime( | ||||||
|  |                 year=timezone.now().year, | ||||||
|  |                 month=(timezone.now().month + 10) % 12 + 1, | ||||||
|  |                 day=1, | ||||||
|  |             ) | ||||||
|  |         start_date = start_date.replace(tzinfo=tz.utc) | ||||||
|  |         end_date = (start_date + timedelta(days=32)).replace( | ||||||
|  |             day=1, hour=0, minute=0, microsecond=0 | ||||||
|  |         ) | ||||||
|  |         from django.db.models import Case, Sum, When | ||||||
|  |  | ||||||
|         kwargs["sum_cb"] = Refilling.objects.filter( |         kwargs["sum_cb"] = sum( | ||||||
|             payment_method="CARD", |             [ | ||||||
|             is_validated=True, |                 r.amount | ||||||
|             date__gte=start_date, |                 for r in Refilling.objects.filter( | ||||||
|             date__lte=end_date, |  | ||||||
|         ).aggregate(res=Sum("amount", default=0))["res"] |  | ||||||
|         kwargs["sum_cb"] += ( |  | ||||||
|             Selling.objects.filter( |  | ||||||
|                     payment_method="CARD", |                     payment_method="CARD", | ||||||
|                     is_validated=True, |                     is_validated=True, | ||||||
|                     date__gte=start_date, |                     date__gte=start_date, | ||||||
|                     date__lte=end_date, |                     date__lte=end_date, | ||||||
|                 ) |                 ) | ||||||
|             .annotate(amount=F("unit_price") * F("quantity")) |             ] | ||||||
|             .aggregate(res=Sum("amount", default=0))["res"] |         ) | ||||||
|  |         kwargs["sum_cb"] += sum( | ||||||
|  |             [ | ||||||
|  |                 s.quantity * s.unit_price | ||||||
|  |                 for s in Selling.objects.filter( | ||||||
|  |                     payment_method="CARD", | ||||||
|  |                     is_validated=True, | ||||||
|  |                     date__gte=start_date, | ||||||
|  |                     date__lte=end_date, | ||||||
|  |                 ) | ||||||
|  |             ] | ||||||
|         ) |         ) | ||||||
|         kwargs["start_date"] = start_date |         kwargs["start_date"] = start_date | ||||||
|         kwargs["invoices"] = ( |         kwargs["sums"] = ( | ||||||
|             Selling.objects.filter(date__gte=start_date, date__lt=end_date) |             Selling.objects.values("club__name") | ||||||
|             .values("club_id", "club__name") |             .annotate( | ||||||
|             .annotate(selling_sum=Sum(F("unit_price") * F("quantity"))) |                 selling_sum=Sum( | ||||||
|  |                     Case( | ||||||
|  |                         When( | ||||||
|  |                             date__gte=start_date, | ||||||
|  |                             date__lt=end_date, | ||||||
|  |                             then=F("unit_price") * F("quantity"), | ||||||
|  |                         ), | ||||||
|  |                         output_field=CurrencyField(), | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|             .exclude(selling_sum=None) |             .exclude(selling_sum=None) | ||||||
|             .order_by("-selling_sum") |             .order_by("-selling_sum") | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
| {% from 'core/macros.jinja' import update_notifications %} |  | ||||||
|  |  | ||||||
| <div id=billing-infos-fragment> | <div id=billing-infos-fragment> | ||||||
|   <div |   <div | ||||||
|     class="collapse" |     class="collapse" | ||||||
| @@ -31,6 +29,7 @@ | |||||||
|       > |       > | ||||||
|     </form> |     </form> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|   <br> |   <br> | ||||||
|   {{ update_notifications(messages) }} |   {% include "core/base/notifications.jinja" %} | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| {% extends "core/base.jinja" %} | {% extends "core/base.jinja" %} | ||||||
|  |  | ||||||
| {% block notifications %} | {% block notifications %} | ||||||
|   {# Notifications are moved under the billing form #} |   {# Notifications are moved inside the billing info fragment #} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block title %} | {% block title %} | ||||||
| @@ -60,7 +60,6 @@ | |||||||
|       <div @htmx:after-request="fill"> |       <div @htmx:after-request="fill"> | ||||||
|         {{ billing_infos_form }} |         {{ billing_infos_form }} | ||||||
|       </div> |       </div> | ||||||
|       {% include "core/base/notifications.jinja" %} |  | ||||||
|       <form |       <form | ||||||
|         method="post" |         method="post" | ||||||
|         action="{{ settings.SITH_EBOUTIC_ET_URL }}" |         action="{{ settings.SITH_EBOUTIC_ET_URL }}" | ||||||
|   | |||||||
| @@ -48,7 +48,7 @@ from django_countries.fields import Country | |||||||
|  |  | ||||||
| from core.auth.mixins import CanViewMixin | from core.auth.mixins import CanViewMixin | ||||||
| from core.views.mixins import FragmentMixin, UseFragmentsMixin | from core.views.mixins import FragmentMixin, UseFragmentsMixin | ||||||
| from counter.forms import BaseBasketForm, BasketProductForm, BillingInfoForm | from counter.forms import BaseBasketForm, BillingInfoForm, ProductForm | ||||||
| from counter.models import ( | from counter.models import ( | ||||||
|     BillingInfo, |     BillingInfo, | ||||||
|     Customer, |     Customer, | ||||||
| @@ -78,7 +78,7 @@ class BaseEbouticBasketForm(BaseBasketForm): | |||||||
|  |  | ||||||
|  |  | ||||||
| EbouticBasketForm = forms.formset_factory( | EbouticBasketForm = forms.formset_factory( | ||||||
|     BasketProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 |     ProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
| msgid "" | msgid "" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Report-Msgid-Bugs-To: \n" | "Report-Msgid-Bugs-To: \n" | ||||||
| "POT-Creation-Date: 2025-10-17 13:41+0200\n" | "POT-Creation-Date: 2025-09-26 17:36+0200\n" | ||||||
| "PO-Revision-Date: 2016-07-18\n" | "PO-Revision-Date: 2016-07-18\n" | ||||||
| "Last-Translator: Maréchal <thomas.girod@utbm.fr\n" | "Last-Translator: Maréchal <thomas.girod@utbm.fr\n" | ||||||
| "Language-Team: AE info <ae.info@utbm.fr>\n" | "Language-Team: AE info <ae.info@utbm.fr>\n" | ||||||
| @@ -117,7 +117,7 @@ msgstr "S'abonner" | |||||||
| msgid "Remove" | msgid "Remove" | ||||||
| msgstr "Retirer" | msgstr "Retirer" | ||||||
|  |  | ||||||
| #: club/forms.py counter/forms.py pedagogy/templates/pedagogy/moderation.jinja | #: club/forms.py pedagogy/templates/pedagogy/moderation.jinja | ||||||
| msgid "Action" | msgid "Action" | ||||||
| msgstr "Action" | msgstr "Action" | ||||||
|  |  | ||||||
| @@ -556,8 +556,6 @@ msgstr "" | |||||||
| #: core/templates/core/user_godfathers_tree.jinja | #: core/templates/core/user_godfathers_tree.jinja | ||||||
| #: core/templates/core/user_preferences.jinja | #: core/templates/core/user_preferences.jinja | ||||||
| #: counter/templates/counter/cash_register_summary.jinja | #: counter/templates/counter/cash_register_summary.jinja | ||||||
| #: counter/templates/counter/invoices_call.jinja |  | ||||||
| #: counter/templates/counter/product_form.jinja |  | ||||||
| #: forum/templates/forum/reply.jinja | #: forum/templates/forum/reply.jinja | ||||||
| #: subscription/templates/subscription/fragments/creation_form.jinja | #: subscription/templates/subscription/fragments/creation_form.jinja | ||||||
| #: trombi/templates/trombi/comment.jinja | #: trombi/templates/trombi/comment.jinja | ||||||
| @@ -690,15 +688,15 @@ msgstr "Vente" | |||||||
| msgid "Mailing list" | msgid "Mailing list" | ||||||
| msgstr "Listes de diffusion" | msgstr "Listes de diffusion" | ||||||
|  |  | ||||||
| #: club/views.py |  | ||||||
| msgid "You are now a member of this club." |  | ||||||
| msgstr "Vous êtes maintenant membre de ce club." |  | ||||||
|  |  | ||||||
| #: club/views.py | #: club/views.py | ||||||
| #, python-format | #, python-format | ||||||
| msgid "%(user)s has been added to club." | msgid "%(user)s has been added to club." | ||||||
| msgstr "%(user)s a été ajouté au club." | msgstr "%(user)s a été ajouté au club." | ||||||
|  |  | ||||||
|  | #: club/views.py | ||||||
|  | msgid "You are now a member of this club." | ||||||
|  | msgstr "Vous êtes maintenant membre de ce club." | ||||||
|  |  | ||||||
| #: com/forms.py | #: com/forms.py | ||||||
| msgid "Format: 16:9 | Resolution: 1920x1080" | msgid "Format: 16:9 | Resolution: 1920x1080" | ||||||
| msgstr "Format : 16:9 | Résolution : 1920x1080" | msgstr "Format : 16:9 | Résolution : 1920x1080" | ||||||
| @@ -1063,10 +1061,6 @@ msgstr "Nos services" | |||||||
| msgid "UV Guide" | msgid "UV Guide" | ||||||
| msgstr "Guide des UVs" | msgstr "Guide des UVs" | ||||||
|  |  | ||||||
| #: com/templates/com/news_list.jinja |  | ||||||
| msgid "Timetable" |  | ||||||
| msgstr "Emploi du temps" |  | ||||||
|  |  | ||||||
| #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | ||||||
| msgid "Matmatronch" | msgid "Matmatronch" | ||||||
| msgstr "Matmatronch" | msgstr "Matmatronch" | ||||||
| @@ -2957,18 +2951,6 @@ msgstr "Cet UID est invalide" | |||||||
| msgid "User not found" | msgid "User not found" | ||||||
| msgstr "Utilisateur non trouvé" | msgstr "Utilisateur non trouvé" | ||||||
|  |  | ||||||
| #: counter/forms.py |  | ||||||
| msgid "Date and time of action" |  | ||||||
| msgstr "Date et heure de l'action" |  | ||||||
|  |  | ||||||
| #: counter/forms.py |  | ||||||
| msgid "New counters" |  | ||||||
| msgstr "Nouveaux comptoirs" |  | ||||||
|  |  | ||||||
| #: counter/forms.py |  | ||||||
| msgid "The selected counters will replace the current ones" |  | ||||||
| msgstr "Les comptoirs sélectionnés remplaceront les comptoirs actuels" |  | ||||||
|  |  | ||||||
| #: counter/forms.py | #: counter/forms.py | ||||||
| msgid "" | msgid "" | ||||||
| "Describe the product. If it's an event's click, give some insights about it, " | "Describe the product. If it's an event's click, give some insights about it, " | ||||||
| @@ -3303,52 +3285,6 @@ msgid "The returnable product cannot be the same as the returned one" | |||||||
| msgstr "" | msgstr "" | ||||||
| "Le produit consigné ne peut pas être le même que le produit de déconsigne" | "Le produit consigné ne peut pas être le même que le produit de déconsigne" | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Archiving" |  | ||||||
| msgstr "Archivage" |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Counters change" |  | ||||||
| msgstr "Changement des comptoirs" |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Product scheduled action" |  | ||||||
| msgstr "Actions sur produit planifiées" |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Product actions must declare a clocked schedule." |  | ||||||
| msgstr "Les actions sur les produits doivent avoir un horaire planifié." |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Year + month field (day forced to 1)" |  | ||||||
| msgstr "Champ Année + mois (jour forcé à 1)" |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| #, python-format |  | ||||||
| msgid "" |  | ||||||
| "“%(value)s” value has an invalid date format. It must be in YYYY-MM format." |  | ||||||
| msgstr "" |  | ||||||
| "La valeur « %(value)s » a un format de date invalide. Ce doit être au format " |  | ||||||
| "YYYY-MM." |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| #, python-format |  | ||||||
| msgid "" |  | ||||||
| "“%(value)s” value has the correct format (YYYY-MM) but it is an invalid date." |  | ||||||
| msgstr "La valeur « %(value)s » a le bon format, mais est une date invalide." |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "invoice date" |  | ||||||
| msgstr "date de la facture" |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Invoice call" |  | ||||||
| msgstr "Appel à facture" |  | ||||||
|  |  | ||||||
| #: counter/models.py |  | ||||||
| msgid "Invoice calls" |  | ||||||
| msgstr "Appels à facture" |  | ||||||
|  |  | ||||||
| #: counter/templates/counter/activity.jinja | #: counter/templates/counter/activity.jinja | ||||||
| #, python-format | #, python-format | ||||||
| msgid "%(counter_name)s activity" | msgid "%(counter_name)s activity" | ||||||
| @@ -3579,10 +3515,6 @@ msgstr "Payements en Carte Bancaire" | |||||||
| msgid "Sum" | msgid "Sum" | ||||||
| msgstr "Somme" | msgstr "Somme" | ||||||
|  |  | ||||||
| #: counter/templates/counter/invoices_call.jinja |  | ||||||
| msgid "Validated" |  | ||||||
| msgstr "Validé" |  | ||||||
|  |  | ||||||
| #: counter/templates/counter/last_ops.jinja | #: counter/templates/counter/last_ops.jinja | ||||||
| #, python-format | #, python-format | ||||||
| msgid "%(counter_name)s last operations" | msgid "%(counter_name)s last operations" | ||||||
| @@ -3671,25 +3603,6 @@ msgstr "" | |||||||
| "votre cotisation. Si vous ne renouvelez pas votre cotisation, il n'y aura " | "votre cotisation. Si vous ne renouvelez pas votre cotisation, il n'y aura " | ||||||
| "aucune conséquence autre que le retrait de l'argent de votre compte." | "aucune conséquence autre que le retrait de l'argent de votre compte." | ||||||
|  |  | ||||||
| #: counter/templates/counter/product_form.jinja |  | ||||||
| #, python-format |  | ||||||
| msgid "Edit product %(name)s" |  | ||||||
| msgstr "Édition du produit %(name)s" |  | ||||||
|  |  | ||||||
| #: counter/templates/counter/product_form.jinja |  | ||||||
| msgid "Product creation" |  | ||||||
| msgstr "Création de produit" |  | ||||||
|  |  | ||||||
| #: counter/templates/counter/product_form.jinja |  | ||||||
| msgid "Automatic actions" |  | ||||||
| msgstr "Actions automatiques" |  | ||||||
|  |  | ||||||
| #: counter/templates/counter/product_form.jinja |  | ||||||
| msgid "Automatic actions allows to schedule product changes ahead of time." |  | ||||||
| msgstr "" |  | ||||||
| "Les actions automatiques vous permettent de planifier des modifications du " |  | ||||||
| "produit à l'avance." |  | ||||||
|  |  | ||||||
| #: counter/templates/counter/product_list.jinja | #: counter/templates/counter/product_list.jinja | ||||||
| msgid "Product list" | msgid "Product list" | ||||||
| msgstr "Liste des produits" | msgstr "Liste des produits" | ||||||
| @@ -3872,10 +3785,6 @@ msgstr "L'utilisateur n'est pas barman." | |||||||
| msgid "Bad location, someone is already logged in somewhere else" | msgid "Bad location, someone is already logged in somewhere else" | ||||||
| msgstr "Mauvais comptoir, quelqu'un est déjà connecté ailleurs" | msgstr "Mauvais comptoir, quelqu'un est déjà connecté ailleurs" | ||||||
|  |  | ||||||
| #: counter/views/invoice.py |  | ||||||
| msgid "Invoice calls status has been updated." |  | ||||||
| msgstr "Le statut des appels à facture a été mis à jour." |  | ||||||
|  |  | ||||||
| #: counter/views/mixins.py | #: counter/views/mixins.py | ||||||
| msgid "Cash summary" | msgid "Cash summary" | ||||||
| msgstr "Relevé de caisse" | msgstr "Relevé de caisse" | ||||||
| @@ -5327,18 +5236,6 @@ msgstr "Membre existant" | |||||||
| msgid "the groups that can create subscriptions" | msgid "the groups that can create subscriptions" | ||||||
| msgstr "les groupes pouvant créer des cotisations" | msgstr "les groupes pouvant créer des cotisations" | ||||||
|  |  | ||||||
| #: timetable/templates/timetable/generator.jinja |  | ||||||
| msgid "Timetable generator" |  | ||||||
| msgstr "Générateur d'emploi du temps" |  | ||||||
|  |  | ||||||
| #: timetable/templates/timetable/generator.jinja |  | ||||||
| msgid "Generate" |  | ||||||
| msgstr "Générer" |  | ||||||
|  |  | ||||||
| #: timetable/templates/timetable/generator.jinja |  | ||||||
| msgid "Save to PNG" |  | ||||||
| msgstr "Sauver en PNG" |  | ||||||
|  |  | ||||||
| #: trombi/models.py | #: trombi/models.py | ||||||
| msgid "subscription deadline" | msgid "subscription deadline" | ||||||
| msgstr "fin des inscriptions" | msgstr "fin des inscriptions" | ||||||
|   | |||||||
							
								
								
									
										50
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										50
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -29,7 +29,6 @@ | |||||||
|         "d3-force-3d": "^3.0.5", |         "d3-force-3d": "^3.0.5", | ||||||
|         "easymde": "^2.19.0", |         "easymde": "^2.19.0", | ||||||
|         "glob": "^11.0.0", |         "glob": "^11.0.0", | ||||||
|         "html2canvas": "^1.4.1", |  | ||||||
|         "htmx.org": "^2.0.3", |         "htmx.org": "^2.0.3", | ||||||
|         "js-cookie": "^3.0.5", |         "js-cookie": "^3.0.5", | ||||||
|         "lit-html": "^3.3.0", |         "lit-html": "^3.3.0", | ||||||
| @@ -3106,15 +3105,6 @@ | |||||||
|         "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" |         "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/base64-arraybuffer": { |  | ||||||
|       "version": "1.0.2", |  | ||||||
|       "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz", |  | ||||||
|       "integrity": "sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ==", |  | ||||||
|       "license": "MIT", |  | ||||||
|       "engines": { |  | ||||||
|         "node": ">= 0.6.0" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "node_modules/binary-extensions": { |     "node_modules/binary-extensions": { | ||||||
|       "version": "2.3.0", |       "version": "2.3.0", | ||||||
|       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", |       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", | ||||||
| @@ -3503,15 +3493,6 @@ | |||||||
|         "node": ">= 8" |         "node": ">= 8" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/css-line-break": { |  | ||||||
|       "version": "2.1.0", |  | ||||||
|       "resolved": "https://registry.npmjs.org/css-line-break/-/css-line-break-2.1.0.tgz", |  | ||||||
|       "integrity": "sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w==", |  | ||||||
|       "license": "MIT", |  | ||||||
|       "dependencies": { |  | ||||||
|         "utrie": "^1.0.2" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "node_modules/cytoscape": { |     "node_modules/cytoscape": { | ||||||
|       "version": "3.33.1", |       "version": "3.33.1", | ||||||
|       "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.33.1.tgz", |       "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.33.1.tgz", | ||||||
| @@ -4184,19 +4165,6 @@ | |||||||
|         "node": ">= 0.4" |         "node": ">= 0.4" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/html2canvas": { |  | ||||||
|       "version": "1.4.1", |  | ||||||
|       "resolved": "https://registry.npmjs.org/html2canvas/-/html2canvas-1.4.1.tgz", |  | ||||||
|       "integrity": "sha512-fPU6BHNpsyIhr8yyMpTLLxAbkaK8ArIBcmZIRiBLiDhjeqvXolaEmDGmELFuX9I4xDcaKKcJl+TKZLqruBbmWA==", |  | ||||||
|       "license": "MIT", |  | ||||||
|       "dependencies": { |  | ||||||
|         "css-line-break": "^2.1.0", |  | ||||||
|         "text-segmentation": "^1.0.3" |  | ||||||
|       }, |  | ||||||
|       "engines": { |  | ||||||
|         "node": ">=8.0.0" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "node_modules/htmx.org": { |     "node_modules/htmx.org": { | ||||||
|       "version": "2.0.6", |       "version": "2.0.6", | ||||||
|       "resolved": "https://registry.npmjs.org/htmx.org/-/htmx.org-2.0.6.tgz", |       "resolved": "https://registry.npmjs.org/htmx.org/-/htmx.org-2.0.6.tgz", | ||||||
| @@ -5486,15 +5454,6 @@ | |||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "ISC" |       "license": "ISC" | ||||||
|     }, |     }, | ||||||
|     "node_modules/text-segmentation": { |  | ||||||
|       "version": "1.0.3", |  | ||||||
|       "resolved": "https://registry.npmjs.org/text-segmentation/-/text-segmentation-1.0.3.tgz", |  | ||||||
|       "integrity": "sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw==", |  | ||||||
|       "license": "MIT", |  | ||||||
|       "dependencies": { |  | ||||||
|         "utrie": "^1.0.2" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "node_modules/three": { |     "node_modules/three": { | ||||||
|       "version": "0.177.0", |       "version": "0.177.0", | ||||||
|       "resolved": "https://registry.npmjs.org/three/-/three-0.177.0.tgz", |       "resolved": "https://registry.npmjs.org/three/-/three-0.177.0.tgz", | ||||||
| @@ -5752,15 +5711,6 @@ | |||||||
|         "browserslist": ">= 4.21.0" |         "browserslist": ">= 4.21.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/utrie": { |  | ||||||
|       "version": "1.0.2", |  | ||||||
|       "resolved": "https://registry.npmjs.org/utrie/-/utrie-1.0.2.tgz", |  | ||||||
|       "integrity": "sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw==", |  | ||||||
|       "license": "MIT", |  | ||||||
|       "dependencies": { |  | ||||||
|         "base64-arraybuffer": "^1.0.2" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "node_modules/vite": { |     "node_modules/vite": { | ||||||
|       "version": "6.3.6", |       "version": "6.3.6", | ||||||
|       "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.6.tgz", |       "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.6.tgz", | ||||||
|   | |||||||
| @@ -59,7 +59,6 @@ | |||||||
|     "d3-force-3d": "^3.0.5", |     "d3-force-3d": "^3.0.5", | ||||||
|     "easymde": "^2.19.0", |     "easymde": "^2.19.0", | ||||||
|     "glob": "^11.0.0", |     "glob": "^11.0.0", | ||||||
|     "html2canvas": "^1.4.1", |  | ||||||
|     "htmx.org": "^2.0.3", |     "htmx.org": "^2.0.3", | ||||||
|     "js-cookie": "^3.0.5", |     "js-cookie": "^3.0.5", | ||||||
|     "lit-html": "^3.3.0", |     "lit-html": "^3.3.0", | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ dependencies = [ | |||||||
|     "django>=5.2.1,<6.0.0", |     "django>=5.2.1,<6.0.0", | ||||||
|     "django-ninja<2.0.0,>=1.4.0", |     "django-ninja<2.0.0,>=1.4.0", | ||||||
|     "django-ninja-extra<1.0.0,>=0.22.9", |     "django-ninja-extra<1.0.0,>=0.22.9", | ||||||
|     "Pillow>=11.1.0,<13.0.0", |     "Pillow<12.0.0,>=11.1.0", | ||||||
|     "mistune<4.0.0,>=3.1.3", |     "mistune<4.0.0,>=3.1.3", | ||||||
|     "django-jinja<3.0.0,>=2.11.0", |     "django-jinja<3.0.0,>=2.11.0", | ||||||
|     "cryptography>=45.0.3,<46.0.0", |     "cryptography>=45.0.3,<46.0.0", | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ import { | |||||||
|  |  | ||||||
| interface PagePictureConfig { | interface PagePictureConfig { | ||||||
|   userId: number; |   userId: number; | ||||||
|   nbPictures?: number; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| interface Album { | interface Album { | ||||||
| @@ -21,27 +20,11 @@ document.addEventListener("alpine:init", () => { | |||||||
|     loading: true, |     loading: true, | ||||||
|     albums: [] as Album[], |     albums: [] as Album[], | ||||||
|  |  | ||||||
|     async fetchPictures(): Promise<PictureSchema[]> { |     async init() { | ||||||
|       const localStorageKey = `user${config.userId}Pictures`; |  | ||||||
|       const localStorageInvalidationKey = `user${config.userId}PicturesNumber`; |  | ||||||
|       const lastCachedNumber = localStorage.getItem(localStorageInvalidationKey); |  | ||||||
|       if ( |  | ||||||
|         lastCachedNumber !== null && |  | ||||||
|         Number.parseInt(lastCachedNumber) === config.nbPictures |  | ||||||
|       ) { |  | ||||||
|         return JSON.parse(localStorage.getItem(localStorageKey)); |  | ||||||
|       } |  | ||||||
|       const pictures = await paginated(picturesFetchPictures, { |       const pictures = await paginated(picturesFetchPictures, { | ||||||
|         // biome-ignore lint/style/useNamingConvention: from python api |         // biome-ignore lint/style/useNamingConvention: from python api | ||||||
|         query: { users_identified: [config.userId] }, |         query: { users_identified: [config.userId] }, | ||||||
|       } as PicturesFetchPicturesData); |       } as PicturesFetchPicturesData); | ||||||
|       localStorage.setItem(localStorageInvalidationKey, config.nbPictures.toString()); |  | ||||||
|       localStorage.setItem(localStorageKey, JSON.stringify(pictures)); |  | ||||||
|       return pictures; |  | ||||||
|     }, |  | ||||||
|  |  | ||||||
|     async init() { |  | ||||||
|       const pictures = await this.fetchPictures(); |  | ||||||
|       const groupedAlbums = Object.groupBy(pictures, (i: PictureSchema) => i.album.id); |       const groupedAlbums = Object.groupBy(pictures, (i: PictureSchema) => i.album.id); | ||||||
|       this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => { |       this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => { | ||||||
|         return { |         return { | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ | |||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   <main x-data="user_pictures({ userId: {{ object.id }}, nbPictures: {{ object.nb_pictures }} })"> |   <main x-data="user_pictures({ userId: {{ object.id }} })"> | ||||||
|     {% if user.id == object.id %} |     {% if user.id == object.id %} | ||||||
|       {{ download_button(_("Download all my pictures")) }} |       {{ download_button(_("Download all my pictures")) }} | ||||||
|     {% endif %} |     {% endif %} | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								sas/views.py
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								sas/views.py
									
									
									
									
									
								
							| @@ -16,7 +16,6 @@ from typing import Any | |||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.core.exceptions import PermissionDenied | from django.core.exceptions import PermissionDenied | ||||||
| from django.db.models import Count, OuterRef, Subquery |  | ||||||
| from django.http import Http404, HttpResponseRedirect | from django.http import Http404, HttpResponseRedirect | ||||||
| from django.shortcuts import get_object_or_404 | from django.shortcuts import get_object_or_404 | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| @@ -37,7 +36,7 @@ from sas.forms import ( | |||||||
|     PictureModerationRequestForm, |     PictureModerationRequestForm, | ||||||
|     PictureUploadForm, |     PictureUploadForm, | ||||||
| ) | ) | ||||||
| from sas.models import Album, PeoplePictureRelation, Picture | from sas.models import Album, Picture | ||||||
|  |  | ||||||
|  |  | ||||||
| class AlbumCreateFragment(FragmentMixin, CreateView): | class AlbumCreateFragment(FragmentMixin, CreateView): | ||||||
| @@ -179,13 +178,6 @@ class UserPicturesView(UserTabsMixin, CanViewMixin, DetailView): | |||||||
|     context_object_name = "profile" |     context_object_name = "profile" | ||||||
|     template_name = "sas/user_pictures.jinja" |     template_name = "sas/user_pictures.jinja" | ||||||
|     current_tab = "pictures" |     current_tab = "pictures" | ||||||
|     queryset = User.objects.annotate( |  | ||||||
|         nb_pictures=Subquery( |  | ||||||
|             PeoplePictureRelation.objects.filter(user=OuterRef("id")) |  | ||||||
|             .values("user_id") |  | ||||||
|             .values(count=Count("*")) |  | ||||||
|         ) |  | ||||||
|     ).all() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Admin views | # Admin views | ||||||
|   | |||||||
| @@ -125,7 +125,6 @@ INSTALLED_APPS = ( | |||||||
|     "pedagogy", |     "pedagogy", | ||||||
|     "galaxy", |     "galaxy", | ||||||
|     "antispam", |     "antispam", | ||||||
|     "timetable", |  | ||||||
|     "api", |     "api", | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -53,7 +53,6 @@ urlpatterns = [ | |||||||
|     path("i18n/", include("django.conf.urls.i18n")), |     path("i18n/", include("django.conf.urls.i18n")), | ||||||
|     path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"), |     path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"), | ||||||
|     path("captcha/", include("captcha.urls")), |     path("captcha/", include("captcha.urls")), | ||||||
|     path("edt/", include(("timetable.urls", "timetable"), namespace="timetable")), |  | ||||||
| ] | ] | ||||||
|  |  | ||||||
| if settings.DEBUG: | if settings.DEBUG: | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| # Register your models here. |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| from django.apps import AppConfig |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TimetableConfig(AppConfig): |  | ||||||
|     default_auto_field = "django.db.models.BigAutoField" |  | ||||||
|     name = "timetable" |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| # Create your models here. |  | ||||||
| @@ -1,184 +0,0 @@ | |||||||
| import html2canvas from "html2canvas"; |  | ||||||
|  |  | ||||||
| // see https://regex101.com/r/QHSaPM/2 |  | ||||||
| const TIMETABLE_ROW_RE: RegExp = |  | ||||||
|   /^(?<ueCode>\w.+\w)\s+(?<courseType>[A-Z]{2}\d)\s+((?<weekGroup>[AB])\s+)?(?<weekday>(lundi)|(mardi)|(mercredi)|(jeudi)|(vendredi)|(samedi)|(dimanche))\s+(?<startHour>\d{2}:\d{2})\s+(?<endHour>\d{2}:\d{2})\s+[\dA-B]\s+((?<attendance>[\wé]*)\s+)?(?<room>\w+(?:, \w+)?)$/; |  | ||||||
|  |  | ||||||
| const DEFAULT_TIMETABLE: string = `DS52\t\tCM1\t\tlundi\t08:00\t10:00\t1\tPrésentiel\tA113 |  | ||||||
| DS53\t\tCM1\t\tlundi\t10:15\t12:15\t1\tPrésentiel\tA101 |  | ||||||
| DS53\t\tTP1\t\tlundi\t13:00\t16:00\t1\tPrésentiel\tH010 |  | ||||||
| SO03\t\tCM1\t\tlundi\t16:15\t17:45\t1\tPrésentiel\tA103 |  | ||||||
| SO03\t\tTD1\t\tlundi\t17:45\t19:45\t1\tPrésentiel\tA103 |  | ||||||
| DS50\t\tTP1\t\tmardi\t08:00\t10:00\t1\tPrésentiel\tA216 |  | ||||||
| DS51\t\tCM1\t\tmardi\t10:15\t12:15\t1\tPrésentiel\tA216 |  | ||||||
| DS51\t\tTP1\t\tmardi\t14:00\t18:00\t1\tPrésentiel\tH010 |  | ||||||
| DS52\t\tTP2\tA\tjeudi\t08:00\t10:00\tA\tPrésentiel\tA110a, A110b |  | ||||||
| DS52\t\tTD1\t\tjeudi\t10:15\t12:15\t1\tPrésentiel\tA110a, A110b |  | ||||||
| LC02\t\tTP1\t\tjeudi\t15:00\t16:00\t1\tPrésentiel\tA209 |  | ||||||
| LC02\t\tTD1\t\tjeudi\t16:15\t18:15\t1\tPrésentiel\tA206`; |  | ||||||
|  |  | ||||||
| type WeekDay = |  | ||||||
|   | "lundi" |  | ||||||
|   | "mardi" |  | ||||||
|   | "mercredi" |  | ||||||
|   | "jeudi" |  | ||||||
|   | "vendredi" |  | ||||||
|   | "samedi" |  | ||||||
|   | "dimanche"; |  | ||||||
|  |  | ||||||
| const WEEKDAYS = [ |  | ||||||
|   "lundi", |  | ||||||
|   "mardi", |  | ||||||
|   "mercredi", |  | ||||||
|   "jeudi", |  | ||||||
|   "vendredi", |  | ||||||
|   "samedi", |  | ||||||
|   "dimanche", |  | ||||||
| ] as const; |  | ||||||
|  |  | ||||||
| const SLOT_HEIGHT = 20 as const; // Each 15min has a height of 20px in the timetable |  | ||||||
| const SLOT_WIDTH = 250 as const; // Each weekday ha a width of 400px in the timetable |  | ||||||
| const MINUTES_PER_SLOT = 15 as const; |  | ||||||
|  |  | ||||||
| interface TimetableSlot { |  | ||||||
|   courseType: string; |  | ||||||
|   room: string; |  | ||||||
|   startHour: string; |  | ||||||
|   endHour: string; |  | ||||||
|   startSlot: number; |  | ||||||
|   endSlot: number; |  | ||||||
|   ueCode: string; |  | ||||||
|   weekGroup?: string; |  | ||||||
|   weekday: WeekDay; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function parseSlots(s: string): TimetableSlot[] { |  | ||||||
|   return s |  | ||||||
|     .split("\n") |  | ||||||
|     .filter((s: string) => s.length > 0) |  | ||||||
|     .map((row: string) => { |  | ||||||
|       const parsed = TIMETABLE_ROW_RE.exec(row); |  | ||||||
|       if (!parsed) { |  | ||||||
|         throw new Error(`Couldn't parse row ${row}`); |  | ||||||
|       } |  | ||||||
|       const [startHour, startMin] = parsed.groups.startHour |  | ||||||
|         .split(":") |  | ||||||
|         .map((i) => Number.parseInt(i)); |  | ||||||
|       const [endHour, endMin] = parsed.groups.endHour |  | ||||||
|         .split(":") |  | ||||||
|         .map((i) => Number.parseInt(i)); |  | ||||||
|       return { |  | ||||||
|         ...parsed.groups, |  | ||||||
|         startSlot: Math.floor((startHour * 60 + startMin) / MINUTES_PER_SLOT), |  | ||||||
|         endSlot: Math.floor((endHour * 60 + endMin) / MINUTES_PER_SLOT), |  | ||||||
|       } as unknown as TimetableSlot; |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| document.addEventListener("alpine:init", () => { |  | ||||||
|   Alpine.data("timetableGenerator", () => ({ |  | ||||||
|     content: DEFAULT_TIMETABLE, |  | ||||||
|     error: "", |  | ||||||
|     displayedWeekdays: [] as WeekDay[], |  | ||||||
|     courses: [] as TimetableSlot[], |  | ||||||
|     startSlot: 0, |  | ||||||
|     endSlot: 0, |  | ||||||
|     table: { |  | ||||||
|       height: 0, |  | ||||||
|       width: 0, |  | ||||||
|     }, |  | ||||||
|  |  | ||||||
|     colors: {} as Record<string, string>, |  | ||||||
|     colorPalette: [ |  | ||||||
|       "#27ae60", |  | ||||||
|       "#2980b9", |  | ||||||
|       "#c0392b", |  | ||||||
|       "#7f8c8d", |  | ||||||
|       "#f1c40f", |  | ||||||
|       "#1abc9c", |  | ||||||
|       "#95a5a6", |  | ||||||
|       "#26C6DA", |  | ||||||
|       "#c2185b", |  | ||||||
|       "#e64a19", |  | ||||||
|       "#1b5e20", |  | ||||||
|     ], |  | ||||||
|  |  | ||||||
|     generate() { |  | ||||||
|       try { |  | ||||||
|         this.courses = parseSlots(this.content); |  | ||||||
|       } catch { |  | ||||||
|         this.error = gettext( |  | ||||||
|           "Wrong timetable format. Make sure you copied if from your student folder.", |  | ||||||
|         ); |  | ||||||
|         return; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       // color each UE |  | ||||||
|       let colorIndex = 0; |  | ||||||
|       for (const slot of this.courses) { |  | ||||||
|         if (!this.colors[slot.ueCode]) { |  | ||||||
|           this.colors[slot.ueCode] = |  | ||||||
|             this.colorPalette[colorIndex % this.colorPalette.length]; |  | ||||||
|           colorIndex++; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       this.displayedWeekdays = WEEKDAYS.filter((day) => |  | ||||||
|         this.courses.some((slot: TimetableSlot) => slot.weekday === day), |  | ||||||
|       ); |  | ||||||
|       this.startSlot = this.courses.reduce( |  | ||||||
|         (acc: number, curr: TimetableSlot) => Math.min(acc, curr.startSlot), |  | ||||||
|         25 * 4, |  | ||||||
|       ); |  | ||||||
|       this.endSlot = this.courses.reduce( |  | ||||||
|         (acc: number, curr: TimetableSlot) => Math.max(acc, curr.endSlot), |  | ||||||
|         1, |  | ||||||
|       ); |  | ||||||
|       this.table.height = SLOT_HEIGHT * (this.endSlot - this.startSlot); |  | ||||||
|       this.table.width = SLOT_WIDTH * this.displayedWeekdays.length; |  | ||||||
|     }, |  | ||||||
|  |  | ||||||
|     getStyle(slot: TimetableSlot) { |  | ||||||
|       const hasWeekGroup = slot.weekGroup !== undefined; |  | ||||||
|       const width = hasWeekGroup ? SLOT_WIDTH / 2 : SLOT_WIDTH; |  | ||||||
|       const leftOffset = slot.weekGroup === "B" ? SLOT_WIDTH / 2 : 0; |  | ||||||
|       return { |  | ||||||
|         height: `${(slot.endSlot - slot.startSlot) * SLOT_HEIGHT}px`, |  | ||||||
|         width: `${width}px`, |  | ||||||
|         top: `${(slot.startSlot - this.startSlot) * SLOT_HEIGHT}px`, |  | ||||||
|         left: `${this.displayedWeekdays.indexOf(slot.weekday) * SLOT_WIDTH + leftOffset}px`, |  | ||||||
|         backgroundColor: this.colors[slot.ueCode], |  | ||||||
|       }; |  | ||||||
|     }, |  | ||||||
|  |  | ||||||
|     getHours(): [string, object][] { |  | ||||||
|       let hour: number = Number.parseInt( |  | ||||||
|         this.courses |  | ||||||
|           .map((c: TimetableSlot) => c.startHour) |  | ||||||
|           .reduce((res: string, hour: string) => (hour < res ? hour : res), "24:00") |  | ||||||
|           .split(":")[0], |  | ||||||
|       ); |  | ||||||
|       const res: [string, object][] = []; |  | ||||||
|       for (let i = 0; i <= this.endSlot - this.startSlot; i += 60 / MINUTES_PER_SLOT) { |  | ||||||
|         res.push([`${hour}:00`, { top: `${i * SLOT_HEIGHT}px` }]); |  | ||||||
|         hour += 1; |  | ||||||
|       } |  | ||||||
|       return res; |  | ||||||
|     }, |  | ||||||
|  |  | ||||||
|     getWidth() { |  | ||||||
|       return this.displayedWeekdays.length * SLOT_WIDTH + 20; |  | ||||||
|     }, |  | ||||||
|  |  | ||||||
|     async savePng() { |  | ||||||
|       const elem = document.getElementById("timetable"); |  | ||||||
|       const img = (await html2canvas(elem)).toDataURL(); |  | ||||||
|       const downloadLink = document.createElement("a"); |  | ||||||
|       downloadLink.href = img; |  | ||||||
|       downloadLink.download = "edt.png"; |  | ||||||
|       document.body.appendChild(downloadLink); |  | ||||||
|       downloadLink.click(); |  | ||||||
|       downloadLink.remove(); |  | ||||||
|     }, |  | ||||||
|   })); |  | ||||||
| }); |  | ||||||
| @@ -1,67 +0,0 @@ | |||||||
| @import "core/static/core/colors"; |  | ||||||
|  |  | ||||||
| #timetable { |  | ||||||
|   --hour-side-width: 60px; |  | ||||||
|  |  | ||||||
|   display: block; |  | ||||||
|   margin: 2em auto; |  | ||||||
|   .header { |  | ||||||
|     background-color: $white-color; |  | ||||||
|     font-weight: bold; |  | ||||||
|     box-shadow: none; |  | ||||||
|     width: calc(100% - var(--hour-side-width) - 10px); |  | ||||||
|     margin-left: var(--hour-side-width); |  | ||||||
|     padding-left: 0; |  | ||||||
|     display: flex; |  | ||||||
|     flex-direction: row; |  | ||||||
|     gap: 0; |  | ||||||
|     span { |  | ||||||
|       flex: 1; |  | ||||||
|       text-align: center; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .content { |  | ||||||
|     position: relative; |  | ||||||
|   } |  | ||||||
|   .hours { |  | ||||||
|     position: absolute; |  | ||||||
|     width: 40px; |  | ||||||
|     left: 0; |  | ||||||
|     top: -.5em; |  | ||||||
|  |  | ||||||
|     .hour { |  | ||||||
|       position: absolute; |  | ||||||
|  |  | ||||||
|       .hour-bar { |  | ||||||
|         content: ""; |  | ||||||
|         position: absolute; |  | ||||||
|         height: 1px; |  | ||||||
|         background: lightgray; |  | ||||||
|         top: 50%; |  | ||||||
|         left: 100%; |  | ||||||
|         margin-left: 10px; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   .courses { |  | ||||||
|     position: absolute; |  | ||||||
|     text-align: center; |  | ||||||
|     top: 0; |  | ||||||
|     left: var(--hour-side-width); |  | ||||||
|  |  | ||||||
|     .slot { |  | ||||||
|       background-color: cadetblue; |  | ||||||
|       position: absolute; |  | ||||||
|       display: flex; |  | ||||||
|       flex-direction: column; |  | ||||||
|       justify-content: center; |  | ||||||
|  |  | ||||||
|       .course-type { |  | ||||||
|         position: absolute; |  | ||||||
|         top: 0; |  | ||||||
|         right: 0; |  | ||||||
|         padding: 10px; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,68 +0,0 @@ | |||||||
| {% extends 'core/base.jinja' %} |  | ||||||
|  |  | ||||||
| {%- block additional_css -%} |  | ||||||
|   <link rel="stylesheet" href="{{ static('timetable/css/generator.scss') }}"> |  | ||||||
| {%- endblock -%} |  | ||||||
|  |  | ||||||
| {%- block additional_js -%} |  | ||||||
|   <script type="module" src="{{ static('bundled/timetable/generator-index.ts') }}"></script> |  | ||||||
| {%- endblock -%} |  | ||||||
|  |  | ||||||
| {% block title %} |  | ||||||
|   {% trans %}Timetable generator{% endtrans %} |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
| {% block content %} |  | ||||||
|   <div x-data="timetableGenerator"> |  | ||||||
|     <form @submit.prevent="generate()"> |  | ||||||
|       <h1>Générateur d'emploi du temps</h1> |  | ||||||
|       <div class="alert alert-red" x-show="!!error" x-cloak> |  | ||||||
|         <span class="alert-main" x-text="error"></span> |  | ||||||
|       </div> |  | ||||||
|       <div class="form-group"> |  | ||||||
|         <label for="timetable-input">Colle ton emploi du temps (sans l'entête)</label> |  | ||||||
|         <textarea id="timetable-input" cols="30" rows="15" x-model="content"></textarea> |  | ||||||
|       </div> |  | ||||||
|       <input type="submit" class="btn btn-blue" value="{% trans %}Generate{% endtrans %}"> |  | ||||||
|     </form> |  | ||||||
|     <div |  | ||||||
|       id="timetable" |  | ||||||
|       x-show="table.height > 0 && table.width > 0" |  | ||||||
|       :style="{width: `${table.width+80}px`, height: `${table.height+40}px`}" |  | ||||||
|     > |  | ||||||
|       <div class="header"> |  | ||||||
|         <template x-for="weekday in displayedWeekdays"> |  | ||||||
|           <span x-text="weekday"></span> |  | ||||||
|         </template> |  | ||||||
|       </div> |  | ||||||
|       <div class="content"> |  | ||||||
|         <div class="hours" :height="(endSlot - endSlot%4) - (startSlot - startSlot%4)"> |  | ||||||
|           <template x-for="[hour, style] in getHours()"> |  | ||||||
|             <div class="hour" :style="style"> |  | ||||||
|               <div x-text="hour"></div> |  | ||||||
|               <div class="hour-bar" :style="{width: `${getWidth()}px`}"></div> |  | ||||||
|             </div> |  | ||||||
|           </template> |  | ||||||
|         </div> |  | ||||||
|         <div class="courses"> |  | ||||||
|           <template x-for="course in courses"> |  | ||||||
|             <div class="slot" :style="getStyle(course)"> |  | ||||||
|               <span class="course-type" x-text="course.courseType"></span> |  | ||||||
|               <span x-text="course.ueCode"></span> |  | ||||||
|               <span x-text="`${course.startHour} - ${course.endHour}`"></span> |  | ||||||
|               <span x-text="(course.weekGroup ? `\nGroupe ${course.weekGroup}` : '')"></span> |  | ||||||
|               <span x-text="course.room"></span> |  | ||||||
|             </div> |  | ||||||
|           </template> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |  | ||||||
|     <button |  | ||||||
|       class="margin-bottom btn btn-blue" |  | ||||||
|       @click="savePng" |  | ||||||
|       x-show="table.height > 0 && table.width > 0" |  | ||||||
|     > |  | ||||||
|       {% trans %}Save to PNG{% endtrans %} |  | ||||||
|     </button> |  | ||||||
|   </div> |  | ||||||
| {% endblock content %} |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| # Create your tests here. |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| from django.urls import path |  | ||||||
|  |  | ||||||
| from timetable.views import GeneratorView |  | ||||||
|  |  | ||||||
| urlpatterns = [path("", GeneratorView.as_view(), name="generator")] |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| # Create your views here. |  | ||||||
| from django.views.generic import TemplateView |  | ||||||
|  |  | ||||||
| from core.auth.mixins import FormerSubscriberMixin |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class GeneratorView(FormerSubscriberMixin, TemplateView): |  | ||||||
|     template_name = "timetable/generator.jinja" |  | ||||||
		Reference in New Issue
	
	Block a user