mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-10-22 04:38:29 +00:00 
			
		
		
		
	Compare commits
	
		
			24 Commits
		
	
	
		
			room-reser
			...
			dependabot
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | c625db4316 | ||
|  | 710b4aa942 | ||
|  | 459edc1b6e | ||
| a760a0b75d | |||
|  | fc615e90b2 | ||
| 76eebaf54e | |||
|  | 9407f4b341 | ||
|  | 8bd82c9d7c | ||
|  | 957441ceb1 | ||
|  | 3bcd417ad0 | ||
|  | 453e13d54b | ||
|  | dbd86b66cc | ||
|  | dcf799b352 | ||
|  | d815f7da97 | ||
|  | dac52db434 | ||
|  | f398c9901c | ||
|  | 5b91fe2145 | ||
|  | abd905c24d | ||
|  | 42b53a39f3 | ||
|  | 5306001f6f | ||
|  | 83a4ac2a7e | ||
| fa8772ede2 | |||
|  | 2a30f30a31 | ||
|  | 5fee2e4720 | 
| @@ -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_subscription"): |         if self.request_user.has_perm("club.add_membership"): | ||||||
|             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,11 +31,7 @@ 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 ( | from django.http import Http404, HttpResponseRedirect, StreamingHttpResponse | ||||||
|     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 | ||||||
| @@ -55,12 +51,7 @@ from club.forms import ( | |||||||
|     MailingForm, |     MailingForm, | ||||||
|     SellingsForm, |     SellingsForm, | ||||||
| ) | ) | ||||||
| from club.models import ( | from club.models import Club, Mailing, MailingSubscription, Membership | ||||||
|     Club, |  | ||||||
|     Mailing, |  | ||||||
|     MailingSubscription, |  | ||||||
|     Membership, |  | ||||||
| ) |  | ||||||
| from com.models import Poster | from com.models import Poster | ||||||
| from com.views import ( | from com.views import ( | ||||||
|     PosterCreateBaseView, |     PosterCreateBaseView, | ||||||
| @@ -68,9 +59,7 @@ from com.views import ( | |||||||
|     PosterEditBaseView, |     PosterEditBaseView, | ||||||
|     PosterListBaseView, |     PosterListBaseView, | ||||||
| ) | ) | ||||||
| from core.auth.mixins import ( | from core.auth.mixins import CanEditMixin | ||||||
|     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 | ||||||
|   | |||||||
| @@ -1,9 +1,9 @@ | |||||||
| import { alpinePlugin } from "#core:utils/notifications"; | import { alpinePlugin as notificationPlugin } 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", alpinePlugin); | Alpine.magic("notifications", notificationPlugin); | ||||||
| window.Alpine = Alpine; | window.Alpine = Alpine; | ||||||
|  |  | ||||||
| window.addEventListener("DOMContentLoaded", () => { | window.addEventListener("DOMContentLoaded", () => { | ||||||
|   | |||||||
| @@ -154,10 +154,8 @@ form { | |||||||
|     margin-bottom: 1rem; |     margin-bottom: 1rem; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .row { |   .row > label { | ||||||
|     label { |     margin: unset; | ||||||
|       margin: unset; |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // ------------- LABEL |   // ------------- LABEL | ||||||
|   | |||||||
| @@ -503,6 +503,10 @@ 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; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -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 in messages"> |   <template x-for="(message, index) in messages"> | ||||||
|     <div x-data="{show: true}" class="alert" :class="`alert-${message.tag}`" x-show="show" x-transition> |     <div class="alert" :class="`alert-${message.tag}`" x-transition> | ||||||
|       <span class="alert-main" x-text="message.text"></span> |       <span class="alert-main" x-text="message.text"></span> | ||||||
|       <span class="clickable" @click="show = false"> |       <span class="clickable" @click="messages = messages.filter((item, i) => i !== index)"> | ||||||
|         <i class="fa fa-close"></i> |         <i class="fa fa-close"></i> | ||||||
|       </span> |       </span> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
| @@ -245,3 +245,26 @@ | |||||||
|   <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 %} | ||||||
|   | |||||||
| @@ -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 ValueError(_("Ensure this timestamp is set in the future")) |         raise ValidationError(_("Ensure this timestamp is set in the future")) | ||||||
|  |  | ||||||
|  |  | ||||||
| class FutureDateTimeField(forms.DateTimeField): | class FutureDateTimeField(forms.DateTimeField): | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ from counter.models import ( | |||||||
|     Counter, |     Counter, | ||||||
|     Customer, |     Customer, | ||||||
|     Eticket, |     Eticket, | ||||||
|  |     InvoiceCall, | ||||||
|     Permanency, |     Permanency, | ||||||
|     Product, |     Product, | ||||||
|     ProductType, |     ProductType, | ||||||
| @@ -160,3 +161,11 @@ 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,13 +1,26 @@ | |||||||
|  | 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 Q | from django.db.models import Exists, OuterRef, 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 NFCTextInput, SelectDate, SelectDateTime | from core.views.forms import ( | ||||||
|  |     FutureDateTimeField, | ||||||
|  |     NFCTextInput, | ||||||
|  |     SelectDate, | ||||||
|  |     SelectDateTime, | ||||||
|  | ) | ||||||
| from core.views.widgets.ajax_select import ( | from core.views.widgets.ajax_select import ( | ||||||
|     AutoCompleteSelect, |     AutoCompleteSelect, | ||||||
|     AutoCompleteSelectMultipleGroup, |     AutoCompleteSelectMultipleGroup, | ||||||
| @@ -19,10 +32,14 @@ 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, | ||||||
| @@ -158,7 +175,101 @@ class CounterEditForm(forms.ModelForm): | |||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProductEditForm(forms.ModelForm): | class ScheduledProductActionForm(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" | ||||||
|  |  | ||||||
| @@ -199,22 +310,21 @@ class ProductEditForm(forms.ModelForm): | |||||||
|         queryset=Counter.objects.all(), |         queryset=Counter.objects.all(), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, instance=None, **kwargs): | ||||||
|         super().__init__(*args, **kwargs) |         super().__init__(*args, instance=instance, **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) | ||||||
|         if self.fields["counters"].initial: |         self.instance.counters.set(self.cleaned_data["counters"]) | ||||||
|             # Remove the product from all counter it was added to |         self.action_formset.save() | ||||||
|             # 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 | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -266,7 +376,7 @@ class CloseCustomerAccountForm(forms.Form): | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProductForm(forms.Form): | class BasketProductForm(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) | ||||||
|  |  | ||||||
| @@ -371,5 +481,50 @@ class BaseBasketForm(forms.BaseFormSet): | |||||||
|  |  | ||||||
|  |  | ||||||
| BasketForm = forms.formset_factory( | BasketForm = forms.formset_factory( | ||||||
|     ProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 |     BasketProductForm, 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"], | ||||||
|  |         ) | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								counter/migrations/0032_scheduledproductaction.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								counter/migrations/0032_scheduledproductaction.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | # 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",), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										51
									
								
								counter/migrations/0033_invoicecall.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								counter/migrations/0033_invoicecall.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | |||||||
|  | # 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,6 +15,7 @@ | |||||||
| 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 | ||||||
| @@ -34,6 +35,7 @@ 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 | ||||||
| @@ -445,7 +447,8 @@ 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 | ||||||
|         return any(user.is_in_group(pk=group.id) for group in buying_groups) |         res = any(user.is_in_group(pk=group.id) for group in buying_groups) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def profit(self): |     def profit(self): | ||||||
| @@ -479,7 +482,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 tue queryset with the `is_open` field. |         """Annotate the 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. | ||||||
| @@ -1357,3 +1360,85 @@ 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}" | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								counter/tasks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								counter/tasks.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | # 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) | ||||||
| @@ -4,35 +4,49 @@ | |||||||
|   {% 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=""> | ||||||
|     <select name="month"> |     <label for="id_form_other_month">{% trans %}Choose another month: {% endtrans %}</label> | ||||||
|  |     <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> | ||||||
|   <table> |  | ||||||
|     <thead> |   {% include "core/base/notifications.jinja" %} | ||||||
|       <td>{% trans %}Club{% endtrans %}</td> |  | ||||||
|       <td>{% trans %}Sum{% endtrans %}</td> |   <form method="post" action=""> | ||||||
|     </thead> |     {% csrf_token %} | ||||||
|     <tbody> |     <table> | ||||||
|       {% for i in sums %} |       <thead> | ||||||
|         <tr> |         <tr> | ||||||
|           <td>{{ i['club__name'] }}</td> |           <td>{% trans %}Club{% endtrans %}</td> | ||||||
|           <td>{{ i['selling_sum'] }} €</td> |           <td>{% trans %}Sum{% endtrans %}</td> | ||||||
|  |           <td>{% trans %}Validated{% endtrans %}</td> | ||||||
|         </tr> |         </tr> | ||||||
|       {% endfor %} |       </thead> | ||||||
|     </tbody> |       <tbody> | ||||||
|   </table> |         {% for invoice in invoices %} | ||||||
| {% endblock %} |           <tr> | ||||||
|  |             <td>{{ invoice.club__name }}</td> | ||||||
|  |             <td>{{ "%.2f"|format(invoice.selling_sum) }} €</td> | ||||||
|  |             <td> | ||||||
|  |               {{ form[invoice.club_id|string] }} | ||||||
|  |             </td> | ||||||
|  |           </tr> | ||||||
|  |         {% endfor %} | ||||||
|  |       </tbody> | ||||||
|  |     </table> | ||||||
|  |     <input type="hidden" name="month" value="{{ start_date|date('Y-m') }}"> | ||||||
|  |     <button type="submit">{% trans %}Save{% endtrans %}</button> | ||||||
|  |   </form> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										56
									
								
								counter/templates/counter/product_form.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								counter/templates/counter/product_form.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | |||||||
|  | {% 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 %} | ||||||
							
								
								
									
										116
									
								
								counter/tests/test_auto_actions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								counter/tests/test_auto_actions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,116 @@ | |||||||
|  | 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 | ||||||
							
								
								
									
										76
									
								
								counter/tests/test_invoices.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								counter/tests/test_invoices.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | |||||||
|  | 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,14 +6,16 @@ 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 | from django.test import Client, TestCase | ||||||
| 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 | from pytest_django.asserts import assertNumQueries, assertRedirects | ||||||
|  |  | ||||||
|  | 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 | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -84,3 +86,49 @@ 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, | ||||||
|     ProductEditForm, |     ProductForm, | ||||||
|     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 = ProductEditForm |     form_class = ProductForm | ||||||
|     template_name = "core/create.jinja" |     template_name = "counter/product_form.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 = ProductEditForm |     form_class = ProductForm | ||||||
|     pk_url_kwarg = "product_id" |     pk_url_kwarg = "product_id" | ||||||
|     template_name = "core/edit.jinja" |     template_name = "counter/product_form.jinja" | ||||||
|     current_tab = "products" |     current_tab = "products" | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,77 +12,81 @@ | |||||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | # OR WITHIN THE LOCAL FILE "LICENSE" | ||||||
| # | # | ||||||
| # | # | ||||||
| from datetime import datetime, timedelta | from datetime import datetime | ||||||
| from datetime import timezone as tz | from urllib.parse import urlencode | ||||||
|  |  | ||||||
| from django.db.models import F | from dateutil.relativedelta import relativedelta | ||||||
| from django.utils import timezone | from django.contrib.auth.mixins import PermissionRequiredMixin | ||||||
| from django.views.generic import TemplateView | from django.contrib.messages.views import SuccessMessageMixin | ||||||
|  | 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.fields import CurrencyField | from counter.forms import InvoiceCallForm | ||||||
| from counter.models import Refilling, Selling | from counter.models import Refilling, Selling | ||||||
| from counter.views.mixins import CounterAdminMixin, CounterAdminTabsMixin | from counter.views.mixins import CounterAdminTabsMixin | ||||||
|  |  | ||||||
|  |  | ||||||
| class InvoiceCallView(CounterAdminTabsMixin, CounterAdminMixin, TemplateView): | class InvoiceCallView( | ||||||
|  |     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") | ||||||
|         if "month" in self.request.GET: |         start_date = self.get_month() | ||||||
|             start_date = datetime.strptime(self.request.GET["month"], "%Y-%m") |         end_date = start_date + relativedelta(months=1) | ||||||
|         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"] = sum( |         kwargs["sum_cb"] = Refilling.objects.filter( | ||||||
|             [ |             payment_method="CARD", | ||||||
|                 r.amount |             is_validated=True, | ||||||
|                 for r in Refilling.objects.filter( |             date__gte=start_date, | ||||||
|                     payment_method="CARD", |             date__lte=end_date, | ||||||
|                     is_validated=True, |         ).aggregate(res=Sum("amount", default=0))["res"] | ||||||
|                     date__gte=start_date, |         kwargs["sum_cb"] += ( | ||||||
|                     date__lte=end_date, |             Selling.objects.filter( | ||||||
|                 ) |                 payment_method="CARD", | ||||||
|             ] |                 is_validated=True, | ||||||
|         ) |                 date__gte=start_date, | ||||||
|         kwargs["sum_cb"] += sum( |                 date__lte=end_date, | ||||||
|             [ |             ) | ||||||
|                 s.quantity * s.unit_price |             .annotate(amount=F("unit_price") * F("quantity")) | ||||||
|                 for s in Selling.objects.filter( |             .aggregate(res=Sum("amount", default=0))["res"] | ||||||
|                     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["sums"] = ( |         kwargs["invoices"] = ( | ||||||
|             Selling.objects.values("club__name") |             Selling.objects.filter(date__gte=start_date, date__lt=end_date) | ||||||
|             .annotate( |             .values("club_id", "club__name") | ||||||
|                 selling_sum=Sum( |             .annotate(selling_sum=Sum(F("unit_price") * F("quantity"))) | ||||||
|                     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,3 +1,5 @@ | |||||||
|  | {% from 'core/macros.jinja' import update_notifications %} | ||||||
|  |  | ||||||
| <div id=billing-infos-fragment> | <div id=billing-infos-fragment> | ||||||
|   <div |   <div | ||||||
|     class="collapse" |     class="collapse" | ||||||
| @@ -29,7 +31,6 @@ | |||||||
|       > |       > | ||||||
|     </form> |     </form> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|   <br> |   <br> | ||||||
|   {% include "core/base/notifications.jinja" %} |   {{ update_notifications(messages) }} | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| {% extends "core/base.jinja" %} | {% extends "core/base.jinja" %} | ||||||
|  |  | ||||||
| {% block notifications %} | {% block notifications %} | ||||||
|   {# Notifications are moved inside the billing info fragment #} |   {# Notifications are moved under the billing form #} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block title %} | {% block title %} | ||||||
| @@ -60,6 +60,7 @@ | |||||||
|       <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, BillingInfoForm, ProductForm | from counter.forms import BaseBasketForm, BasketProductForm, BillingInfoForm | ||||||
| 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( | ||||||
|     ProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 |     BasketProductForm, 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-09-26 17:36+0200\n" | "POT-Creation-Date: 2025-10-17 13:41+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 pedagogy/templates/pedagogy/moderation.jinja | #: club/forms.py counter/forms.py pedagogy/templates/pedagogy/moderation.jinja | ||||||
| msgid "Action" | msgid "Action" | ||||||
| msgstr "Action" | msgstr "Action" | ||||||
|  |  | ||||||
| @@ -556,6 +556,8 @@ 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 | ||||||
| @@ -688,15 +690,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" | ||||||
| @@ -2955,6 +2957,18 @@ 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, " | ||||||
| @@ -3289,6 +3303,52 @@ 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" | ||||||
| @@ -3519,6 +3579,10 @@ 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" | ||||||
| @@ -3607,6 +3671,25 @@ 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" | ||||||
| @@ -3789,6 +3872,10 @@ 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" | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										8
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -49,7 +49,7 @@ | |||||||
|         "@types/cytoscape-klay": "^3.1.4", |         "@types/cytoscape-klay": "^3.1.4", | ||||||
|         "@types/js-cookie": "^3.0.6", |         "@types/js-cookie": "^3.0.6", | ||||||
|         "typescript": "^5.8.3", |         "typescript": "^5.8.3", | ||||||
|         "vite": "^6.3.6", |         "vite": "^6.4.1", | ||||||
|         "vite-bundle-visualizer": "^1.2.1", |         "vite-bundle-visualizer": "^1.2.1", | ||||||
|         "vite-plugin-static-copy": "^3.1.2" |         "vite-plugin-static-copy": "^3.1.2" | ||||||
|       } |       } | ||||||
| @@ -5762,9 +5762,9 @@ | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "node_modules/vite": { |     "node_modules/vite": { | ||||||
|       "version": "6.3.6", |       "version": "6.4.1", | ||||||
|       "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.6.tgz", |       "resolved": "https://registry.npmjs.org/vite/-/vite-6.4.1.tgz", | ||||||
|       "integrity": "sha512-0msEVHJEScQbhkbVTb/4iHZdJ6SXp/AvxL2sjwYQFfBqleHtnCqv1J3sa9zbWz/6kW1m9Tfzn92vW+kZ1WV6QA==", |       "integrity": "sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|   | |||||||
| @@ -34,7 +34,7 @@ | |||||||
|     "@types/cytoscape-klay": "^3.1.4", |     "@types/cytoscape-klay": "^3.1.4", | ||||||
|     "@types/js-cookie": "^3.0.6", |     "@types/js-cookie": "^3.0.6", | ||||||
|     "typescript": "^5.8.3", |     "typescript": "^5.8.3", | ||||||
|     "vite": "^6.3.6", |     "vite": "^6.4.1", | ||||||
|     "vite-bundle-visualizer": "^1.2.1", |     "vite-bundle-visualizer": "^1.2.1", | ||||||
|     "vite-plugin-static-copy": "^3.1.2" |     "vite-plugin-static-copy": "^3.1.2" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import { | |||||||
|  |  | ||||||
| interface PagePictureConfig { | interface PagePictureConfig { | ||||||
|   userId: number; |   userId: number; | ||||||
|  |   nbPictures?: number; | ||||||
| } | } | ||||||
|  |  | ||||||
| interface Album { | interface Album { | ||||||
| @@ -20,11 +21,27 @@ document.addEventListener("alpine:init", () => { | |||||||
|     loading: true, |     loading: true, | ||||||
|     albums: [] as Album[], |     albums: [] as Album[], | ||||||
|  |  | ||||||
|     async init() { |     async fetchPictures(): Promise<PictureSchema[]> { | ||||||
|  |       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 }} })"> |   <main x-data="user_pictures({ userId: {{ object.id }}, nbPictures: {{ object.nb_pictures }} })"> | ||||||
|     {% 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,6 +16,7 @@ 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 | ||||||
| @@ -36,7 +37,7 @@ from sas.forms import ( | |||||||
|     PictureModerationRequestForm, |     PictureModerationRequestForm, | ||||||
|     PictureUploadForm, |     PictureUploadForm, | ||||||
| ) | ) | ||||||
| from sas.models import Album, Picture | from sas.models import Album, PeoplePictureRelation, Picture | ||||||
|  |  | ||||||
|  |  | ||||||
| class AlbumCreateFragment(FragmentMixin, CreateView): | class AlbumCreateFragment(FragmentMixin, CreateView): | ||||||
| @@ -178,6 +179,13 @@ 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 | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user