mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-10-20 19:58:31 +00:00 
			
		
		
		
	Compare commits
	
		
			13 Commits
		
	
	
		
			dependabot
			...
			room-reser
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | e5a2d1b2db | ||
|  | f66c25cce2 | ||
|  | 96436570e0 | ||
|  | 1c4abb0fa6 | ||
|  | ed5fa13f00 | ||
|  | 2dc3007524 | ||
|  | 16e03f20d9 | ||
|  | 1118693816 | ||
|  | 88681cbe81 | ||
|  | e5f406b0f1 | ||
|  | 4a958481ce | ||
|  | ca3022b8ec | ||
| 0f99729a98 | 
| @@ -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: | ||||||
|   | |||||||
| @@ -1,25 +1,63 @@ | |||||||
| {% extends "core/base.jinja" %} | {% extends "core/base.jinja" %} | ||||||
|  | {% from "reservation/macros.jinja" import room_detail %} | ||||||
|  |  | ||||||
|  | {% block additional_css %} | ||||||
|  |   <link rel="stylesheet" href="{{ static("core/components/card.scss") }}"> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   <h3>{% trans %}Club tools{% endtrans %}</h3> |   <h3>{% trans %}Club tools{% endtrans %} ({{ club.name }})</h3> | ||||||
|   <div> |   <div> | ||||||
|     <h4>{% trans %}Communication:{% endtrans %}</h4> |     <h4>{% trans %}Communication:{% endtrans %}</h4> | ||||||
|     <ul> |     <ul> | ||||||
|       <li> <a href="{{ url('com:news_new') }}?club={{ object.id }}">{% trans %}Create a news{% endtrans %}</a></li> |       <li> | ||||||
|       <li> <a href="{{ url('com:weekmail_article') }}?club={{ object.id }}">{% trans %}Post in the Weekmail{% endtrans %}</a></li> |         <a href="{{ url('com:news_new') }}?club={{ object.id }}"> | ||||||
|  |           {% trans %}Create a news{% endtrans %} | ||||||
|  |         </a> | ||||||
|  |       </li> | ||||||
|  |       <li> | ||||||
|  |         <a href="{{ url('com:weekmail_article') }}?club={{ object.id }}"> | ||||||
|  |           {% trans %}Post in the Weekmail{% endtrans %} | ||||||
|  |         </a> | ||||||
|  |       </li> | ||||||
|       {% if object.trombi %} |       {% if object.trombi %} | ||||||
|         <li> <a href="{{ url('trombi:detail', trombi_id=object.trombi.id) }}">{% trans %}Edit Trombi{% endtrans %}</a></li> |         <li> | ||||||
|  |           <a href="{{ url('trombi:detail', trombi_id=object.trombi.id) }}"> | ||||||
|  |             {% trans %}Edit Trombi{% endtrans %}</a> | ||||||
|  |         </li> | ||||||
|       {% else %} |       {% else %} | ||||||
|         <li><a href="{{ url('trombi:create', club_id=object.id) }}">{% trans %}New Trombi{% endtrans %}</a></li> |         <li><a href="{{ url('trombi:create', club_id=object.id) }}">{% trans %}New Trombi{% endtrans %}</a></li> | ||||||
|         <li><a href="{{ url('club:poster_list', club_id=object.id) }}">{% trans %}Posters{% endtrans %}</a></li> |         <li><a href="{{ url('club:poster_list', club_id=object.id) }}">{% trans %}Posters{% endtrans %}</a></li> | ||||||
|       {% endif %} |       {% endif %} | ||||||
|     </ul> |     </ul> | ||||||
|  |     <h4>{% trans %}Reservable rooms{% endtrans %}</h4> | ||||||
|  |     <a | ||||||
|  |       href="{{ url("reservation:room_create") }}?club={{ object.id }}" | ||||||
|  |       class="btn btn-blue" | ||||||
|  |     > | ||||||
|  |       {% trans %}Add a room{% endtrans %} | ||||||
|  |     </a> | ||||||
|  |     {%- if reservable_rooms|length > 0 -%} | ||||||
|  |       <ul class="card-group"> | ||||||
|  |         {%- for room in reservable_rooms -%} | ||||||
|  |           {{ room_detail( | ||||||
|  |           room, | ||||||
|  |           can_edit=user.can_edit(room), | ||||||
|  |           can_delete=request.user.has_perm("reservation.delete_room") | ||||||
|  |           ) }} | ||||||
|  |         {%- endfor -%} | ||||||
|  |       </ul> | ||||||
|  |     {%- else -%} | ||||||
|  |       <p> | ||||||
|  |         {% trans %}This club manages no reservable room{% endtrans %} | ||||||
|  |       </p> | ||||||
|  |     {%- endif -%} | ||||||
|     <h4>{% trans %}Counters:{% endtrans %}</h4> |     <h4>{% trans %}Counters:{% endtrans %}</h4> | ||||||
|     <ul> |     <ul> | ||||||
|       {% for c in object.counters.filter(type="OFFICE") %} |       {% for counter in counters %} | ||||||
|         <li>{{ c }}: |         <li>{{ counter }}: | ||||||
|           <a href="{{ url('counter:details', counter_id=c.id) }}">View</a> |           <a href="{{ url('counter:details', counter_id=counter.id) }}">View</a> | ||||||
|           <a href="{{ url('counter:admin', counter_id=c.id) }}">Edit</a> |           <a href="{{ url('counter:admin', counter_id=counter.id) }}">Edit</a> | ||||||
|         </li> |         </li> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </ul> |     </ul> | ||||||
|   | |||||||
| @@ -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 | ||||||
| @@ -251,6 +262,12 @@ class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | |||||||
|     template_name = "club/club_tools.jinja" |     template_name = "club/club_tools.jinja" | ||||||
|     current_tab = "tools" |     current_tab = "tools" | ||||||
|  |  | ||||||
|  |     def get_context_data(self, **kwargs): | ||||||
|  |         return super().get_context_data(**kwargs) | { | ||||||
|  |             "reservable_rooms": list(self.object.reservable_rooms.all()), | ||||||
|  |             "counters": list(self.object.counters.filter(type="OFFICE")), | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class ClubAddMembersFragment( | class ClubAddMembersFragment( | ||||||
|     FragmentMixin, PermissionRequiredMixin, SuccessMessageMixin, CreateView |     FragmentMixin, PermissionRequiredMixin, SuccessMessageMixin, CreateView | ||||||
|   | |||||||
| @@ -81,7 +81,6 @@ | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     #links_content { |     #links_content { | ||||||
|       overflow: auto; |  | ||||||
|       box-shadow: $shadow-color 1px 1px 1px; |       box-shadow: $shadow-color 1px 1px 1px; | ||||||
|       min-height: 20em; |       min-height: 20em; | ||||||
|       padding-bottom: 1em; |       padding-bottom: 1em; | ||||||
|   | |||||||
| @@ -1,9 +1,11 @@ | |||||||
| {% extends "core/base.jinja" %} | {% extends "core/base.jinja" %} | ||||||
| {% from "com/macros.jinja" import news_moderation_alert %} | {% from "com/macros.jinja" import news_moderation_alert %} | ||||||
|  |  | ||||||
|  | {% block title %}AE UTBM{% endblock %} | ||||||
|  |  | ||||||
| {% block additional_css %} | {% block additional_css %} | ||||||
|   <link rel="stylesheet" href="{{ static('com/css/news-list.scss') }}"> |   <link rel="stylesheet" href="{{ static('com/css/news-list.scss') }}"> | ||||||
|   <link rel="stylesheet" href="{{ static('com/components/ics-calendar.scss') }}"> |   <link rel="stylesheet" href="{{ static('core/components/calendar.scss') }}"> | ||||||
|  |  | ||||||
|   {# Atom feed discovery, not really css but also goes there #} |   {# Atom feed discovery, not really css but also goes there #} | ||||||
|   <link rel="alternate" type="application/rss+xml" title="{% trans %}News feed{% endtrans %}" href="{{ url("com:news_feed") }}"> |   <link rel="alternate" type="application/rss+xml" title="{% trans %}News feed{% endtrans %}" href="{{ url("com:news_feed") }}"> | ||||||
| @@ -213,6 +215,12 @@ | |||||||
|               <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> | ||||||
|             </li> |             </li> | ||||||
|  |             {% if user.has_perm("reservation.view_reservationslot") %} | ||||||
|  |               <li> | ||||||
|  |                 <i class="fa-solid fa-thumbtack fa-xl"></i> | ||||||
|  |                 <a href="{{ url("reservation:main") }}">{% trans %}Room reservation{% endtrans %}</a> | ||||||
|  |               </li> | ||||||
|  |             {% endif %} | ||||||
|             <li> |             <li> | ||||||
|               <i class="fa-solid fa-check-to-slot fa-xl"></i> |               <i class="fa-solid fa-check-to-slot fa-xl"></i> | ||||||
|               <a href="{{ url("election:list") }}">{% trans %}Elections{% endtrans %}</a> |               <a href="{{ url("election:list") }}">{% trans %}Elections{% endtrans %}</a> | ||||||
|   | |||||||
| @@ -789,7 +789,11 @@ class Command(BaseCommand): | |||||||
|  |  | ||||||
|         subscribers = Group.objects.create(name="Cotisants") |         subscribers = Group.objects.create(name="Cotisants") | ||||||
|         subscribers.permissions.add( |         subscribers.permissions.add( | ||||||
|             *list(perms.filter(codename__in=["add_news", "add_uvcomment"])) |             *list( | ||||||
|  |                 perms.filter( | ||||||
|  |                     codename__in=["add_news", "add_uvcomment", "view_reservationslot"] | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|         ) |         ) | ||||||
|         old_subscribers = Group.objects.create(name="Anciens cotisants") |         old_subscribers = Group.objects.create(name="Anciens cotisants") | ||||||
|         old_subscribers.permissions.add( |         old_subscribers.permissions.add( | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import random | import random | ||||||
| from datetime import date, timedelta | from datetime import date, timedelta | ||||||
| from datetime import timezone as tz | from datetime import timezone as tz | ||||||
|  | from math import ceil | ||||||
| from typing import Iterator | from typing import Iterator | ||||||
|  |  | ||||||
| from dateutil.relativedelta import relativedelta | from dateutil.relativedelta import relativedelta | ||||||
| @@ -24,6 +25,7 @@ from counter.models import ( | |||||||
| ) | ) | ||||||
| from forum.models import Forum, ForumMessage, ForumTopic | from forum.models import Forum, ForumMessage, ForumTopic | ||||||
| from pedagogy.models import UV | from pedagogy.models import UV | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
| from subscription.models import Subscription | from subscription.models import Subscription | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -40,45 +42,20 @@ class Command(BaseCommand): | |||||||
|  |  | ||||||
|         self.stdout.write("Creating users...") |         self.stdout.write("Creating users...") | ||||||
|         users = self.create_users() |         users = self.create_users() | ||||||
|  |         # len(subscribers) is approximately 480 | ||||||
|         subscribers = random.sample(users, k=int(0.8 * len(users))) |         subscribers = random.sample(users, k=int(0.8 * len(users))) | ||||||
|         self.stdout.write("Creating subscriptions...") |         self.stdout.write("Creating subscriptions...") | ||||||
|         self.create_subscriptions(subscribers) |         self.create_subscriptions(subscribers) | ||||||
|         self.stdout.write("Creating club memberships...") |         self.stdout.write("Creating club memberships...") | ||||||
|         users_qs = User.objects.filter(id__in=[s.id for s in subscribers]) |         self.create_club_memberships(subscribers) | ||||||
|         subscribers_now = list( |         self.stdout.write("Creating rooms and reservation...") | ||||||
|             users_qs.annotate( |         self.create_resources_and_reservations(random.sample(subscribers, k=40)) | ||||||
|                 filter=Exists( |  | ||||||
|                     Subscription.objects.filter( |  | ||||||
|                         member_id=OuterRef("pk"), subscription_end__gte=now() |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
|         old_subscribers = list( |  | ||||||
|             users_qs.annotate( |  | ||||||
|                 filter=Exists( |  | ||||||
|                     Subscription.objects.filter( |  | ||||||
|                         member_id=OuterRef("pk"), subscription_end__lt=now() |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
|         self.make_club( |  | ||||||
|             Club.objects.get(id=settings.SITH_MAIN_CLUB_ID), |  | ||||||
|             random.sample(subscribers_now, k=min(30, len(subscribers_now))), |  | ||||||
|             random.sample(old_subscribers, k=min(60, len(old_subscribers))), |  | ||||||
|         ) |  | ||||||
|         self.make_club( |  | ||||||
|             Club.objects.get(name="Troll Penché"), |  | ||||||
|             random.sample(subscribers_now, k=min(20, len(subscribers_now))), |  | ||||||
|             random.sample(old_subscribers, k=min(80, len(old_subscribers))), |  | ||||||
|         ) |  | ||||||
|         self.stdout.write("Creating uvs...") |         self.stdout.write("Creating uvs...") | ||||||
|         self.create_uvs() |         self.create_uvs() | ||||||
|         self.stdout.write("Creating products...") |         self.stdout.write("Creating products...") | ||||||
|         self.create_products() |         self.create_products() | ||||||
|         self.stdout.write("Creating sales and refills...") |         self.stdout.write("Creating sales and refills...") | ||||||
|         sellers = random.sample(list(User.objects.all()), 100) |         sellers = list(User.objects.order_by("?")[:100]) | ||||||
|         self.create_sales(sellers) |         self.create_sales(sellers) | ||||||
|         self.stdout.write("Creating permanences...") |         self.stdout.write("Creating permanences...") | ||||||
|         self.create_permanences(sellers) |         self.create_permanences(sellers) | ||||||
| @@ -192,6 +169,97 @@ class Command(BaseCommand): | |||||||
|         memberships = Membership.objects.bulk_create(memberships) |         memberships = Membership.objects.bulk_create(memberships) | ||||||
|         Membership._add_club_groups(memberships) |         Membership._add_club_groups(memberships) | ||||||
|  |  | ||||||
|  |     def create_club_memberships(self, users: list[User]): | ||||||
|  |         users_qs = User.objects.filter(id__in=[s.id for s in users]) | ||||||
|  |         subscribers_now = list( | ||||||
|  |             users_qs.annotate( | ||||||
|  |                 filter=Exists( | ||||||
|  |                     Subscription.objects.filter( | ||||||
|  |                         member_id=OuterRef("pk"), subscription_end__gte=now() | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         old_subscribers = list( | ||||||
|  |             users_qs.annotate( | ||||||
|  |                 filter=Exists( | ||||||
|  |                     Subscription.objects.filter( | ||||||
|  |                         member_id=OuterRef("pk"), subscription_end__lt=now() | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         self.make_club( | ||||||
|  |             Club.objects.get(id=settings.SITH_MAIN_CLUB_ID), | ||||||
|  |             random.sample(subscribers_now, k=min(30, len(subscribers_now))), | ||||||
|  |             random.sample(old_subscribers, k=min(60, len(old_subscribers))), | ||||||
|  |         ) | ||||||
|  |         self.make_club( | ||||||
|  |             Club.objects.get(name="Troll Penché"), | ||||||
|  |             random.sample(subscribers_now, k=min(20, len(subscribers_now))), | ||||||
|  |             random.sample(old_subscribers, k=min(80, len(old_subscribers))), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def create_resources_and_reservations(self, users: list[User]): | ||||||
|  |         """Generate reservable rooms and reservations slots for those rooms. | ||||||
|  |  | ||||||
|  |         Contrary to the other data generator, | ||||||
|  |         this one generates more data than what is expected on the real db. | ||||||
|  |         """ | ||||||
|  |         ae = Club.objects.get(id=settings.SITH_MAIN_CLUB_ID) | ||||||
|  |         pdf = Club.objects.get(id=settings.SITH_PDF_CLUB_ID) | ||||||
|  |         troll = Club.objects.get(name="Troll Penché") | ||||||
|  |         rooms = [ | ||||||
|  |             Room( | ||||||
|  |                 name=name, | ||||||
|  |                 club=club, | ||||||
|  |                 location=location, | ||||||
|  |                 description=self.faker.text(100), | ||||||
|  |             ) | ||||||
|  |             for name, club, location in [ | ||||||
|  |                 ("Champi", ae, "BELFORT"), | ||||||
|  |                 ("Muzik", ae, "BELFORT"), | ||||||
|  |                 ("Pôle Tech", ae, "BELFORT"), | ||||||
|  |                 ("Jolly", troll, "BELFORT"), | ||||||
|  |                 ("Cookut", pdf, "BELFORT"), | ||||||
|  |                 ("Lucky", pdf, "BELFORT"), | ||||||
|  |                 ("Potards", pdf, "SEVENANS"), | ||||||
|  |                 ("Bureau AE", ae, "SEVENANS"), | ||||||
|  |             ] | ||||||
|  |         ] | ||||||
|  |         rooms = Room.objects.bulk_create(rooms) | ||||||
|  |         reservations = [] | ||||||
|  |         for room in rooms: | ||||||
|  |             # how much people use this room. | ||||||
|  |             # The higher the number, the more reservations exist, | ||||||
|  |             # the smaller the interval between two slot is, | ||||||
|  |             # and the more future reservations have already been made ahead of time | ||||||
|  |             affluence = random.randint(2, 6) | ||||||
|  |             slot_start = make_aware(self.faker.past_datetime("-5y").replace(minute=0)) | ||||||
|  |             generate_until = make_aware( | ||||||
|  |                 self.faker.future_datetime(timedelta(days=1) * affluence**2) | ||||||
|  |             ) | ||||||
|  |             while slot_start < generate_until: | ||||||
|  |                 if slot_start.hour < 8: | ||||||
|  |                     # if a reservation would start in the middle of the night | ||||||
|  |                     # make it start the next morning instead | ||||||
|  |                     slot_start += timedelta(hours=10 - slot_start.hour) | ||||||
|  |                 duration = timedelta(minutes=15) * (1 + int(random.gammavariate(3, 2))) | ||||||
|  |                 reservations.append( | ||||||
|  |                     ReservationSlot( | ||||||
|  |                         room=room, | ||||||
|  |                         author=random.choice(users), | ||||||
|  |                         start_at=slot_start, | ||||||
|  |                         end_at=slot_start + duration, | ||||||
|  |                         created_at=slot_start - self.faker.time_delta("+7d"), | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |                 slot_start += duration + ( | ||||||
|  |                     timedelta(minutes=15) * ceil(random.expovariate(affluence / 192)) | ||||||
|  |                 ) | ||||||
|  |         reservations.sort(key=lambda slot: slot.created_at) | ||||||
|  |         ReservationSlot.objects.bulk_create(reservations) | ||||||
|  |  | ||||||
|     def create_uvs(self): |     def create_uvs(self): | ||||||
|         root = User.objects.get(username="root") |         root = User.objects.get(username="root") | ||||||
|         categories = ["CS", "TM", "OM", "QC", "EC"] |         categories = ["CS", "TM", "OM", "QC", "EC"] | ||||||
| @@ -389,7 +457,7 @@ class Command(BaseCommand): | |||||||
|         Permanency.objects.bulk_create(perms) |         Permanency.objects.bulk_create(perms) | ||||||
|  |  | ||||||
|     def create_forums(self): |     def create_forums(self): | ||||||
|         forumers = random.sample(list(User.objects.all()), 100) |         forumers = list(User.objects.order_by("?")[:100]) | ||||||
|         most_actives = random.sample(forumers, 10) |         most_actives = random.sample(forumers, 10) | ||||||
|         categories = list(Forum.objects.filter(is_category=True)) |         categories = list(Forum.objects.filter(is_category=True)) | ||||||
|         new_forums = [ |         new_forums = [ | ||||||
| @@ -407,7 +475,7 @@ class Command(BaseCommand): | |||||||
|             for _ in range(100) |             for _ in range(100) | ||||||
|         ] |         ] | ||||||
|         ForumTopic.objects.bulk_create(new_topics) |         ForumTopic.objects.bulk_create(new_topics) | ||||||
|         topics = list(ForumTopic.objects.all()) |         topics = list(ForumTopic.objects.values_list("id", flat=True)) | ||||||
|  |  | ||||||
|         def get_author(): |         def get_author(): | ||||||
|             if random.random() > 0.5: |             if random.random() > 0.5: | ||||||
| @@ -415,7 +483,7 @@ class Command(BaseCommand): | |||||||
|             return random.choice(forumers) |             return random.choice(forumers) | ||||||
|  |  | ||||||
|         messages = [] |         messages = [] | ||||||
|         for t in topics: |         for topic_id in topics: | ||||||
|             nb_messages = max(1, int(random.normalvariate(mu=90, sigma=50))) |             nb_messages = max(1, int(random.normalvariate(mu=90, sigma=50))) | ||||||
|             dates = sorted( |             dates = sorted( | ||||||
|                 [ |                 [ | ||||||
| @@ -427,7 +495,7 @@ class Command(BaseCommand): | |||||||
|             messages.extend( |             messages.extend( | ||||||
|                 [ |                 [ | ||||||
|                     ForumMessage( |                     ForumMessage( | ||||||
|                         topic=t, |                         topic_id=topic_id, | ||||||
|                         author=get_author(), |                         author=get_author(), | ||||||
|                         date=d, |                         date=d, | ||||||
|                         message="\n\n".join( |                         message="\n\n".join( | ||||||
|   | |||||||
| @@ -1,9 +1,10 @@ | |||||||
| import { alpinePlugin as notificationPlugin } from "#core:utils/notifications"; | import { alpinePlugin } from "#core:utils/notifications"; | ||||||
|  | import { morph } from "@alpinejs/morph"; | ||||||
| import sort from "@alpinejs/sort"; | import sort from "@alpinejs/sort"; | ||||||
| import Alpine from "alpinejs"; | import Alpine from "alpinejs"; | ||||||
|  |  | ||||||
| Alpine.plugin(sort); | Alpine.plugin([sort, morph]); | ||||||
| Alpine.magic("notifications", notificationPlugin); | Alpine.magic("notifications", alpinePlugin); | ||||||
| window.Alpine = Alpine; | window.Alpine = Alpine; | ||||||
|  |  | ||||||
| window.addEventListener("DOMContentLoaded", () => { | window.addEventListener("DOMContentLoaded", () => { | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| import htmx from "htmx.org"; | import htmx from "htmx.org"; | ||||||
|  | import "htmx-ext-alpine-morph"; | ||||||
|  |  | ||||||
| document.body.addEventListener("htmx:beforeRequest", (event) => { | document.body.addEventListener("htmx:beforeRequest", (event) => { | ||||||
|   event.target.ariaBusy = true; |   event.target.ariaBusy = true; | ||||||
|   | |||||||
| @@ -16,14 +16,74 @@ | |||||||
|   --event-details-padding: 20px; |   --event-details-padding: 20px; | ||||||
|   --event-details-border: 1px solid #EEEEEE; |   --event-details-border: 1px solid #EEEEEE; | ||||||
|   --event-details-border-radius: 4px; |   --event-details-border-radius: 4px; | ||||||
|   --event-details-box-shadow: 0px 6px 20px 4px rgb(0 0 0 / 16%); |   --event-details-box-shadow: 0 6px 20px 4px rgb(0 0 0 / 16%); | ||||||
|   --event-details-max-width: 600px; |   --event-details-max-width: 600px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ics-calendar { | ics-calendar, | ||||||
|  | room-scheduler { | ||||||
|   border: none; |   border: none; | ||||||
|   box-shadow: none; |   box-shadow: none; | ||||||
| 
 | 
 | ||||||
|  |   a.fc-col-header-cell-cushion, | ||||||
|  |   a.fc-col-header-cell-cushion:hover { | ||||||
|  |     color: black; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   a.fc-daygrid-day-number, | ||||||
|  |   a.fc-daygrid-day-number:hover { | ||||||
|  |     color: rgb(34, 34, 34); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   td { | ||||||
|  |     overflow: visible; // Show events on multiple days | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   td, th { | ||||||
|  |     text-align: unset; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   //Reset from style.scss | ||||||
|  |   table { | ||||||
|  |     box-shadow: none; | ||||||
|  |     border-radius: 0; | ||||||
|  |     -moz-border-radius: 0; | ||||||
|  |     margin: 0; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // Reset from style.scss | ||||||
|  |   thead { | ||||||
|  |     background-color: white; | ||||||
|  |     color: black; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // Reset from style.scss | ||||||
|  |   tbody > tr { | ||||||
|  |     &:nth-child(even):not(.highlight) { | ||||||
|  |       background: white; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   .fc .fc-toolbar.fc-footer-toolbar { | ||||||
|  |     margin-bottom: 0.5em; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   button.text-copy, | ||||||
|  |   button.text-copy:focus, | ||||||
|  |   button.text-copy:hover { | ||||||
|  |     background-color: #67AE6E !important; | ||||||
|  |     transition: 500ms ease-in; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   button.text-copied, | ||||||
|  |   button.text-copied:focus, | ||||||
|  |   button.text-copied:hover { | ||||||
|  |     transition: 500ms ease-out; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ics-calendar { | ||||||
|   #event-details { |   #event-details { | ||||||
|     z-index: 10; |     z-index: 10; | ||||||
|     max-width: 1151px; |     max-width: 1151px; | ||||||
| @@ -60,31 +120,10 @@ ics-calendar { | |||||||
|       align-items: start; |       align-items: start; | ||||||
|       flex-direction: row; |       flex-direction: row; | ||||||
|       background-color: var(--event-details-background-color); |       background-color: var(--event-details-background-color); | ||||||
|       margin-top: 0px; |       margin-top: 0; | ||||||
|       margin-bottom: 4px; |       margin-bottom: 4px; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   a.fc-col-header-cell-cushion, |  | ||||||
|   a.fc-col-header-cell-cushion:hover { |  | ||||||
|     color: black; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   a.fc-daygrid-day-number, |  | ||||||
|   a.fc-daygrid-day-number:hover { |  | ||||||
|     color: rgb(34, 34, 34); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   td { |  | ||||||
|     overflow: visible; // Show events on multiple days |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   //Reset from style.scss |  | ||||||
|   table { |  | ||||||
|     box-shadow: none; |  | ||||||
|     border-radius: 0px; |  | ||||||
|     -moz-border-radius: 0px; |  | ||||||
|     margin: 0px; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Reset from style.scss | // Reset from style.scss | ||||||
| @@ -136,7 +175,6 @@ ics-calendar { | |||||||
| .fc .fc-helpButton-button:hover { | .fc .fc-helpButton-button:hover { | ||||||
|   background-color: rgba(20, 20, 20, 0.6); |   background-color: rgba(20, 20, 20, 0.6); | ||||||
| } | } | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| .tooltip.calendar-copy-tooltip { | .tooltip.calendar-copy-tooltip { | ||||||
|   opacity: 1; |   opacity: 1; | ||||||
| @@ -16,6 +16,13 @@ | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .card-group { | ||||||
|  |   display: flex; | ||||||
|  |   gap: 15px; | ||||||
|  |   margin-bottom: 30px; | ||||||
|  |   flex-wrap: wrap; | ||||||
|  | } | ||||||
|  |  | ||||||
| .card { | .card { | ||||||
|   background-color: $primary-neutral-light-color; |   background-color: $primary-neutral-light-color; | ||||||
|   border-radius: 5px; |   border-radius: 5px; | ||||||
| @@ -92,13 +99,23 @@ | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   @media screen and (max-width: 765px) { |   @media screen and (max-width: 765px) { | ||||||
|     @include row-layout |     @include row-layout; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // When combined with card, card-row display the card in a row layout, |   // When combined with card, card-row display the card in a row layout, | ||||||
|   // whatever the size of the screen. |   // whatever the size of the screen. | ||||||
|   &.card-row { |   &.card-row { | ||||||
|     @include row-layout |     @include row-layout; | ||||||
|  |  | ||||||
|  |     &.card-row-m { | ||||||
|  |       //width: 50%; | ||||||
|  |       max-width: 50%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     &.card-row-s { | ||||||
|  |       //width: 33%; | ||||||
|  |       max-width: 33%; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -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; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -10,10 +10,9 @@ | |||||||
|   border-radius: 5px; |   border-radius: 5px; | ||||||
|   padding: 5px 10px; |   padding: 5px 10px; | ||||||
|   position: absolute; |   position: absolute; | ||||||
|   white-space: nowrap; |  | ||||||
|   opacity: 0; |   opacity: 0; | ||||||
|   transition: opacity 500ms ease-out; |   transition: opacity 500ms ease-out; | ||||||
|  |   width: max-content; | ||||||
|   white-space: normal; |   white-space: normal; | ||||||
|  |  | ||||||
|   left: 0; |   left: 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, 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 %} |  | ||||||
|   | |||||||
| @@ -39,9 +39,8 @@ from django.forms import ( | |||||||
|     DateInput, |     DateInput, | ||||||
|     DateTimeInput, |     DateTimeInput, | ||||||
|     TextInput, |     TextInput, | ||||||
|     Widget, |  | ||||||
| ) | ) | ||||||
| from django.utils.timezone import now | from django.utils.timezone import localtime, now | ||||||
| from django.utils.translation import gettext | from django.utils.translation import gettext | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from phonenumber_field.widgets import RegionalPhoneNumberWidget | from phonenumber_field.widgets import RegionalPhoneNumberWidget | ||||||
| @@ -123,8 +122,8 @@ class FutureDateTimeField(forms.DateTimeField): | |||||||
|  |  | ||||||
|     default_validators = [validate_future_timestamp] |     default_validators = [validate_future_timestamp] | ||||||
|  |  | ||||||
|     def widget_attrs(self, widget: Widget) -> dict[str, str]: |     def widget_attrs(self, widget: forms.Widget) -> dict[str, str]: | ||||||
|         return {"min": widget.format_value(now())} |         return {"min": widget.format_value(localtime())} | ||||||
|  |  | ||||||
|  |  | ||||||
| # Forms | # Forms | ||||||
|   | |||||||
| @@ -78,7 +78,7 @@ class FragmentMixin(TemplateResponseMixin, ContextMixin): | |||||||
|             return render( |             return render( | ||||||
|                 request, |                 request, | ||||||
|                 "app/template.jinja", |                 "app/template.jinja", | ||||||
|                 context={"fragment": fragment(request) |                 context={"fragment": fragment(request)} | ||||||
|             } |             } | ||||||
|  |  | ||||||
|         # in urls.py |         # in urls.py | ||||||
|   | |||||||
| @@ -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. | ||||||
| @@ -1360,85 +1357,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}" |  | ||||||
|   | |||||||
| @@ -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) |  | ||||||
| @@ -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 22:40+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" | ||||||
|  |  | ||||||
| @@ -239,7 +239,7 @@ msgid "role" | |||||||
| msgstr "rôle" | msgstr "rôle" | ||||||
|  |  | ||||||
| #: club/models.py core/models.py counter/models.py election/models.py | #: club/models.py core/models.py counter/models.py election/models.py | ||||||
| #: forum/models.py | #: forum/models.py reservation/models.py | ||||||
| msgid "description" | msgid "description" | ||||||
| msgstr "description" | msgstr "description" | ||||||
|  |  | ||||||
| @@ -514,6 +514,18 @@ msgstr "Nouveau Trombi" | |||||||
| msgid "Posters" | msgid "Posters" | ||||||
| msgstr "Affiches" | msgstr "Affiches" | ||||||
|  |  | ||||||
|  | #: club/templates/club/club_tools.jinja | ||||||
|  | msgid "Reservable rooms" | ||||||
|  | msgstr "Salles réservables" | ||||||
|  |  | ||||||
|  | #: club/templates/club/club_tools.jinja | ||||||
|  | msgid "Add a room" | ||||||
|  | msgstr "Ajouter une salle" | ||||||
|  |  | ||||||
|  | #: club/templates/club/club_tools.jinja | ||||||
|  | msgid "This club manages no reservable room" | ||||||
|  | msgstr "Ce club ne gère pas de salle réservable" | ||||||
|  |  | ||||||
| #: club/templates/club/club_tools.jinja | #: club/templates/club/club_tools.jinja | ||||||
| msgid "Counters:" | msgid "Counters:" | ||||||
| msgstr "Comptoirs : " | msgstr "Comptoirs : " | ||||||
| @@ -556,8 +568,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 +700,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" | ||||||
| @@ -779,7 +789,7 @@ msgstr "Une description plus détaillée et exhaustive de l'évènement." | |||||||
| msgid "The club which organizes the event." | msgid "The club which organizes the event." | ||||||
| msgstr "Le club qui organise l'évènement." | msgstr "Le club qui organise l'évènement." | ||||||
|  |  | ||||||
| #: com/models.py pedagogy/models.py trombi/models.py | #: com/models.py pedagogy/models.py reservation/models.py trombi/models.py | ||||||
| msgid "author" | msgid "author" | ||||||
| msgstr "auteur" | msgstr "auteur" | ||||||
|  |  | ||||||
| @@ -1071,6 +1081,11 @@ msgstr "Emploi du temps" | |||||||
| msgid "Matmatronch" | msgid "Matmatronch" | ||||||
| msgstr "Matmatronch" | msgstr "Matmatronch" | ||||||
|  |  | ||||||
|  | #: com/templates/com/news_list.jinja | ||||||
|  | #: reservation/templates/reservation/schedule.jinja | ||||||
|  | msgid "Room reservation" | ||||||
|  | msgstr "Réservation de salle" | ||||||
|  |  | ||||||
| #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | ||||||
| #: core/templates/core/user_tools.jinja | #: core/templates/core/user_tools.jinja | ||||||
| msgid "Elections" | msgid "Elections" | ||||||
| @@ -1937,6 +1952,7 @@ msgstr "Confirmation" | |||||||
| #: core/templates/core/file_delete_confirm.jinja | #: core/templates/core/file_delete_confirm.jinja | ||||||
| #: counter/templates/counter/counter_click.jinja | #: counter/templates/counter/counter_click.jinja | ||||||
| #: counter/templates/counter/fragments/delete_student_card.jinja | #: counter/templates/counter/fragments/delete_student_card.jinja | ||||||
|  | #: reservation/templates/reservation/fragments/create_reservation.jinja | ||||||
| #: sas/templates/sas/ask_picture_removal.jinja | #: sas/templates/sas/ask_picture_removal.jinja | ||||||
| msgid "Cancel" | msgid "Cancel" | ||||||
| msgstr "Annuler" | msgstr "Annuler" | ||||||
| @@ -2957,18 +2973,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, " | ||||||
| @@ -3073,7 +3077,7 @@ msgstr "Mettre à True si le mail a reçu une erreur" | |||||||
| msgid "The operation that emptied the account." | msgid "The operation that emptied the account." | ||||||
| msgstr "L'opération qui a vidé le compte." | msgstr "L'opération qui a vidé le compte." | ||||||
|  |  | ||||||
| #: counter/models.py pedagogy/models.py | #: counter/models.py pedagogy/models.py reservation/models.py | ||||||
| msgid "comment" | msgid "comment" | ||||||
| msgstr "commentaire" | msgstr "commentaire" | ||||||
|  |  | ||||||
| @@ -3303,52 +3307,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 +3537,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 +3625,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 +3807,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" | ||||||
| @@ -4685,6 +4616,73 @@ msgstr "Signaler ce commentaire" | |||||||
| msgid "Edit UE" | msgid "Edit UE" | ||||||
| msgstr "Éditer l'UE" | msgstr "Éditer l'UE" | ||||||
|  |  | ||||||
|  | #: reservation/forms.py | ||||||
|  | msgid "The start must be set before the end" | ||||||
|  | msgstr "Le début doit être placé avant la fin" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "room name" | ||||||
|  | msgstr "Nom de la salle" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "room owner" | ||||||
|  | msgstr "propriétaire de la salle" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "The club which manages this room" | ||||||
|  | msgstr "Le club qui gère cette salle" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "site" | ||||||
|  | msgstr "site" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "reservable room" | ||||||
|  | msgstr "salle réservable" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "reservable rooms" | ||||||
|  | msgstr "salles réservables" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "reserved room" | ||||||
|  | msgstr "salle réservée" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "slot start" | ||||||
|  | msgstr "début du créneau" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "slot end" | ||||||
|  | msgstr "fin du créneau" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "reservation slot" | ||||||
|  | msgstr "créneau de réservation" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "reservation slots" | ||||||
|  | msgstr "créneaux de réservation" | ||||||
|  |  | ||||||
|  | #: reservation/models.py | ||||||
|  | msgid "There is already a reservation on this slot." | ||||||
|  | msgstr "Il y a déjà une réservation sur ce créneau." | ||||||
|  |  | ||||||
|  | #: reservation/templates/reservation/fragments/create_reservation.jinja | ||||||
|  | msgid "Book a room" | ||||||
|  | msgstr "Réserver une salle" | ||||||
|  |  | ||||||
|  | #: reservation/templates/reservation/schedule.jinja | ||||||
|  | msgid "You can book a room by selecting a free slot in the calendar." | ||||||
|  | msgstr "" | ||||||
|  | "Vous pouvez réserver une salle en sélectionnant un emplacement libre dans le " | ||||||
|  | "calendrier." | ||||||
|  |  | ||||||
|  | #: reservation/views.py | ||||||
|  | #, python-format | ||||||
|  | msgid "%(name)s was updated successfully" | ||||||
|  | msgstr "%(name)s a été mis à jour avec succès" | ||||||
|  |  | ||||||
| #: rootplace/forms.py | #: rootplace/forms.py | ||||||
| msgid "User that will be kept" | msgid "User that will be kept" | ||||||
| msgstr "Utilisateur qui sera conservé" | msgstr "Utilisateur qui sera conservé" | ||||||
|   | |||||||
| @@ -251,6 +251,14 @@ msgstr "Types de produits réordonnés !" | |||||||
| msgid "Product type reorganisation failed with status code : %d" | msgid "Product type reorganisation failed with status code : %d" | ||||||
| msgstr "La réorganisation des types de produit a échoué avec le code : %d" | msgstr "La réorganisation des types de produit a échoué avec le code : %d" | ||||||
|  |  | ||||||
|  | #: reservation/static/bundled/reservation/components/room-scheduler-index.ts | ||||||
|  | msgid "Rooms" | ||||||
|  | msgstr "Salles" | ||||||
|  |  | ||||||
|  | #: reservation/static/bundled/reservation/slot-reservation-index.ts | ||||||
|  | msgid "This slot has been successfully moved" | ||||||
|  | msgstr "Ce créneau a été bougé avec succès" | ||||||
|  |  | ||||||
| #: sas/static/bundled/sas/pictures-download-index.ts | #: sas/static/bundled/sas/pictures-download-index.ts | ||||||
| msgid "pictures.%(extension)s" | msgid "pictures.%(extension)s" | ||||||
| msgstr "photos.%(extension)s" | msgstr "photos.%(extension)s" | ||||||
|   | |||||||
							
								
								
									
										97
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										97
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -9,14 +9,18 @@ | |||||||
|       "version": "3", |       "version": "3", | ||||||
|       "license": "GPL-3.0-only", |       "license": "GPL-3.0-only", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|  |         "@alpinejs/morph": "^3.14.9", | ||||||
|         "@alpinejs/sort": "^3.14.7", |         "@alpinejs/sort": "^3.14.7", | ||||||
|         "@arendjr/text-clipper": "npm:@jsr/arendjr__text-clipper@^3.0.0", |         "@arendjr/text-clipper": "npm:@jsr/arendjr__text-clipper@^3.0.0", | ||||||
|         "@floating-ui/dom": "^1.6.13", |         "@floating-ui/dom": "^1.6.13", | ||||||
|         "@fortawesome/fontawesome-free": "^6.6.0", |         "@fortawesome/fontawesome-free": "^6.6.0", | ||||||
|         "@fullcalendar/core": "^6.1.15", |         "@fullcalendar/core": "^6.1.19", | ||||||
|         "@fullcalendar/daygrid": "^6.1.15", |         "@fullcalendar/daygrid": "^6.1.19", | ||||||
|         "@fullcalendar/icalendar": "^6.1.15", |         "@fullcalendar/icalendar": "^6.1.19", | ||||||
|         "@fullcalendar/list": "^6.1.15", |         "@fullcalendar/interaction": "^6.1.19", | ||||||
|  |         "@fullcalendar/list": "^6.1.19", | ||||||
|  |         "@fullcalendar/resource": "^6.1.19", | ||||||
|  |         "@fullcalendar/resource-timeline": "^6.1.19", | ||||||
|         "@sentry/browser": "^9.29.0", |         "@sentry/browser": "^9.29.0", | ||||||
|         "@zip.js/zip.js": "^2.7.52", |         "@zip.js/zip.js": "^2.7.52", | ||||||
|         "3d-force-graph": "^1.73.4", |         "3d-force-graph": "^1.73.4", | ||||||
| @@ -30,6 +34,7 @@ | |||||||
|         "easymde": "^2.19.0", |         "easymde": "^2.19.0", | ||||||
|         "glob": "^11.0.0", |         "glob": "^11.0.0", | ||||||
|         "html2canvas": "^1.4.1", |         "html2canvas": "^1.4.1", | ||||||
|  |         "htmx-ext-alpine-morph": "^2.0.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", | ||||||
| @@ -54,6 +59,12 @@ | |||||||
|         "vite-plugin-static-copy": "^3.1.2" |         "vite-plugin-static-copy": "^3.1.2" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/@alpinejs/morph": { | ||||||
|  |       "version": "3.14.9", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@alpinejs/morph/-/morph-3.14.9.tgz", | ||||||
|  |       "integrity": "sha512-i1mrH5Gza/egszxnCVwQWypRhsKGq28RFWHWuW7aI0+rWo1pvFw+aPhJLImbpt7hx44DtDOr5m4l9Ah+JPFmFw==", | ||||||
|  |       "license": "MIT" | ||||||
|  |     }, | ||||||
|     "node_modules/@alpinejs/sort": { |     "node_modules/@alpinejs/sort": { | ||||||
|       "version": "3.14.9", |       "version": "3.14.9", | ||||||
|       "resolved": "https://registry.npmjs.org/@alpinejs/sort/-/sort-3.14.9.tgz", |       "resolved": "https://registry.npmjs.org/@alpinejs/sort/-/sort-3.14.9.tgz", | ||||||
| @@ -2271,6 +2282,15 @@ | |||||||
|         "ical.js": "^1.4.0" |         "ical.js": "^1.4.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/@fullcalendar/interaction": { | ||||||
|  |       "version": "6.1.19", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@fullcalendar/interaction/-/interaction-6.1.19.tgz", | ||||||
|  |       "integrity": "sha512-GOciy79xe8JMVp+1evAU3ytdwN/7tv35t5i1vFkifiuWcQMLC/JnLg/RA2s4sYmQwoYhTw/p4GLcP0gO5B3X5w==", | ||||||
|  |       "license": "MIT", | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "@fullcalendar/core": "~6.1.19" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/@fullcalendar/list": { |     "node_modules/@fullcalendar/list": { | ||||||
|       "version": "6.1.19", |       "version": "6.1.19", | ||||||
|       "resolved": "https://registry.npmjs.org/@fullcalendar/list/-/list-6.1.19.tgz", |       "resolved": "https://registry.npmjs.org/@fullcalendar/list/-/list-6.1.19.tgz", | ||||||
| @@ -2280,6 +2300,67 @@ | |||||||
|         "@fullcalendar/core": "~6.1.19" |         "@fullcalendar/core": "~6.1.19" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/@fullcalendar/premium-common": { | ||||||
|  |       "version": "6.1.19", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@fullcalendar/premium-common/-/premium-common-6.1.19.tgz", | ||||||
|  |       "integrity": "sha512-bOWHm1u1dUy6M4fQ0hNK7qEI7SrVWrN1ovv/z4/FE/ybfM19ukz7SFs907Ur7KUBWLNKTQYXBtdrY/ginwWraw==", | ||||||
|  |       "license": "SEE LICENSE IN LICENSE.md", | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "@fullcalendar/core": "~6.1.19" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@fullcalendar/resource": { | ||||||
|  |       "version": "6.1.19", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@fullcalendar/resource/-/resource-6.1.19.tgz", | ||||||
|  |       "integrity": "sha512-br1ylX/aIOfd8m7Tzl2LpJBSI+N9Q6aS1qw7K9qnQjYXWQyHBlfLG6ZcPmmkjfaqTUJc8ARRbtNWj1ts5qOZgQ==", | ||||||
|  |       "license": "SEE LICENSE IN LICENSE.md", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@fullcalendar/premium-common": "~6.1.19" | ||||||
|  |       }, | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "@fullcalendar/core": "~6.1.19" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@fullcalendar/resource-timeline": { | ||||||
|  |       "version": "6.1.19", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@fullcalendar/resource-timeline/-/resource-timeline-6.1.19.tgz", | ||||||
|  |       "integrity": "sha512-oC3aVR++dLqJNeBwmLHq9sDgRDFfIG0qSteV7bgBekvNlqEMqXx8wPjUxnELrq8rrhMmK4iV3wO7AB/48IVgyg==", | ||||||
|  |       "license": "SEE LICENSE IN LICENSE.md", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@fullcalendar/premium-common": "~6.1.19", | ||||||
|  |         "@fullcalendar/scrollgrid": "~6.1.19", | ||||||
|  |         "@fullcalendar/timeline": "~6.1.19" | ||||||
|  |       }, | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "@fullcalendar/core": "~6.1.19", | ||||||
|  |         "@fullcalendar/resource": "~6.1.19" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@fullcalendar/scrollgrid": { | ||||||
|  |       "version": "6.1.19", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@fullcalendar/scrollgrid/-/scrollgrid-6.1.19.tgz", | ||||||
|  |       "integrity": "sha512-S1pbiYHvmV0ep6z5sWXJQfgW4Y/jrS5iLIAqSagDFPK0jr327nBxl7Ryi3Zb5UdMIP0/O4GXs8jwZabQPd8SOg==", | ||||||
|  |       "license": "SEE LICENSE IN LICENSE.md", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@fullcalendar/premium-common": "~6.1.19" | ||||||
|  |       }, | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "@fullcalendar/core": "~6.1.19" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/@fullcalendar/timeline": { | ||||||
|  |       "version": "6.1.19", | ||||||
|  |       "resolved": "https://registry.npmjs.org/@fullcalendar/timeline/-/timeline-6.1.19.tgz", | ||||||
|  |       "integrity": "sha512-d2P961mnUTXtJeWNmIq1neoDmZcrPUaK7nGFoc+jQAlnmG3aNSVWQmD1ia694AMqLWtcWkwipW9MuaJgx2QvrA==", | ||||||
|  |       "license": "SEE LICENSE IN LICENSE.md", | ||||||
|  |       "dependencies": { | ||||||
|  |         "@fullcalendar/premium-common": "~6.1.19", | ||||||
|  |         "@fullcalendar/scrollgrid": "~6.1.19" | ||||||
|  |       }, | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "@fullcalendar/core": "~6.1.19" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/@hey-api/json-schema-ref-parser": { |     "node_modules/@hey-api/json-schema-ref-parser": { | ||||||
|       "version": "1.0.6", |       "version": "1.0.6", | ||||||
|       "resolved": "https://registry.npmjs.org/@hey-api/json-schema-ref-parser/-/json-schema-ref-parser-1.0.6.tgz", |       "resolved": "https://registry.npmjs.org/@hey-api/json-schema-ref-parser/-/json-schema-ref-parser-1.0.6.tgz", | ||||||
| @@ -4197,6 +4278,14 @@ | |||||||
|         "node": ">=8.0.0" |         "node": ">=8.0.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/htmx-ext-alpine-morph": { | ||||||
|  |       "version": "2.0.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/htmx-ext-alpine-morph/-/htmx-ext-alpine-morph-2.0.1.tgz", | ||||||
|  |       "integrity": "sha512-teGpcVatx5IjDUYQs959x9FcePM1TIksjfW5tSe1KVQVEVSmbGxEoemneC7XV6RYpX+27i/xn1fPjduwvHDrAw==", | ||||||
|  |       "dependencies": { | ||||||
|  |         "htmx.org": "^2.0.2" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "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", | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								package.json
									
									
									
									
									
								
							| @@ -21,7 +21,8 @@ | |||||||
|     "#core:*": "./core/static/bundled/*", |     "#core:*": "./core/static/bundled/*", | ||||||
|     "#pedagogy:*": "./pedagogy/static/bundled/*", |     "#pedagogy:*": "./pedagogy/static/bundled/*", | ||||||
|     "#counter:*": "./counter/static/bundled/*", |     "#counter:*": "./counter/static/bundled/*", | ||||||
|     "#com:*": "./com/static/bundled/*" |     "#com:*": "./com/static/bundled/*", | ||||||
|  |     "#reservation:*": "./reservation/static/bundled/*" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@babel/core": "^7.25.2", |     "@babel/core": "^7.25.2", | ||||||
| @@ -39,14 +40,18 @@ | |||||||
|     "vite-plugin-static-copy": "^3.1.2" |     "vite-plugin-static-copy": "^3.1.2" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|  |     "@alpinejs/morph": "^3.14.9", | ||||||
|     "@alpinejs/sort": "^3.14.7", |     "@alpinejs/sort": "^3.14.7", | ||||||
|     "@arendjr/text-clipper": "npm:@jsr/arendjr__text-clipper@^3.0.0", |     "@arendjr/text-clipper": "npm:@jsr/arendjr__text-clipper@^3.0.0", | ||||||
|     "@floating-ui/dom": "^1.6.13", |     "@floating-ui/dom": "^1.6.13", | ||||||
|     "@fortawesome/fontawesome-free": "^6.6.0", |     "@fortawesome/fontawesome-free": "^6.6.0", | ||||||
|     "@fullcalendar/core": "^6.1.15", |     "@fullcalendar/core": "^6.1.19", | ||||||
|     "@fullcalendar/daygrid": "^6.1.15", |     "@fullcalendar/daygrid": "^6.1.19", | ||||||
|     "@fullcalendar/icalendar": "^6.1.15", |     "@fullcalendar/icalendar": "^6.1.19", | ||||||
|     "@fullcalendar/list": "^6.1.15", |     "@fullcalendar/interaction": "^6.1.19", | ||||||
|  |     "@fullcalendar/list": "^6.1.19", | ||||||
|  |     "@fullcalendar/resource": "^6.1.19", | ||||||
|  |     "@fullcalendar/resource-timeline": "^6.1.19", | ||||||
|     "@sentry/browser": "^9.29.0", |     "@sentry/browser": "^9.29.0", | ||||||
|     "@zip.js/zip.js": "^2.7.52", |     "@zip.js/zip.js": "^2.7.52", | ||||||
|     "3d-force-graph": "^1.73.4", |     "3d-force-graph": "^1.73.4", | ||||||
| @@ -60,6 +65,7 @@ | |||||||
|     "easymde": "^2.19.0", |     "easymde": "^2.19.0", | ||||||
|     "glob": "^11.0.0", |     "glob": "^11.0.0", | ||||||
|     "html2canvas": "^1.4.1", |     "html2canvas": "^1.4.1", | ||||||
|  |     "htmx-ext-alpine-morph": "^2.0.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", | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								reservation/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								reservation/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										19
									
								
								reservation/admin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								reservation/admin.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | from django.contrib import admin | ||||||
|  |  | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @admin.register(Room) | ||||||
|  | class RoomAdmin(admin.ModelAdmin): | ||||||
|  |     list_display = ("name", "club") | ||||||
|  |     list_filter = (("club", admin.RelatedOnlyFieldListFilter), "location") | ||||||
|  |     autocomplete_fields = ("club",) | ||||||
|  |     search_fields = ("name",) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @admin.register(ReservationSlot) | ||||||
|  | class ReservationSlotAdmin(admin.ModelAdmin): | ||||||
|  |     list_display = ("room", "start_at", "end_at", "author") | ||||||
|  |     autocomplete_fields = ("author",) | ||||||
|  |     list_filter = ("room",) | ||||||
|  |     date_hierarchy = "start_at" | ||||||
							
								
								
									
										64
									
								
								reservation/api.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								reservation/api.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | |||||||
|  | from typing import Any, Literal | ||||||
|  |  | ||||||
|  | from django.core.exceptions import ValidationError | ||||||
|  | from ninja import Query | ||||||
|  | from ninja_extra import ControllerBase, api_controller, paginate, route | ||||||
|  | from ninja_extra.pagination import PageNumberPaginationExtra | ||||||
|  | from ninja_extra.schemas import PaginatedResponseSchema | ||||||
|  |  | ||||||
|  | from api.permissions import HasPerm | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
|  | from reservation.schemas import ( | ||||||
|  |     RoomFilterSchema, | ||||||
|  |     RoomSchema, | ||||||
|  |     SlotFilterSchema, | ||||||
|  |     SlotSchema, | ||||||
|  |     UpdateReservationSlotSchema, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_controller("/reservation/room") | ||||||
|  | class ReservableRoomController(ControllerBase): | ||||||
|  |     @route.get( | ||||||
|  |         "", | ||||||
|  |         response=list[RoomSchema], | ||||||
|  |         permissions=[HasPerm("reservation.view_room")], | ||||||
|  |         url_name="fetch_reservable_rooms", | ||||||
|  |     ) | ||||||
|  |     def fetch_rooms(self, filters: Query[RoomFilterSchema]): | ||||||
|  |         return filters.filter(Room.objects.select_related("club")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_controller("/reservation/slot") | ||||||
|  | class ReservationSlotController(ControllerBase): | ||||||
|  |     @route.get( | ||||||
|  |         "", | ||||||
|  |         response=PaginatedResponseSchema[SlotSchema], | ||||||
|  |         permissions=[HasPerm("reservation.view_reservationslot")], | ||||||
|  |         url_name="fetch_reservation_slots", | ||||||
|  |     ) | ||||||
|  |     @paginate(PageNumberPaginationExtra) | ||||||
|  |     def fetch_slots(self, filters: Query[SlotFilterSchema]): | ||||||
|  |         return filters.filter( | ||||||
|  |             ReservationSlot.objects.select_related("author").order_by("start_at") | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @route.patch( | ||||||
|  |         "/reservation/slot/{int:slot_id}", | ||||||
|  |         permissions=[HasPerm("reservation.change_reservationslot")], | ||||||
|  |         response={ | ||||||
|  |             200: None, | ||||||
|  |             409: dict[Literal["detail"], dict[str, list[str]]], | ||||||
|  |             422: dict[Literal["detail"], list[dict[str, Any]]], | ||||||
|  |         }, | ||||||
|  |         url_name="change_reservation_slot", | ||||||
|  |     ) | ||||||
|  |     def update_slot(self, slot_id: int, params: UpdateReservationSlotSchema): | ||||||
|  |         slot = self.get_object_or_exception(ReservationSlot, id=slot_id) | ||||||
|  |         slot.start_at = params.start_at | ||||||
|  |         slot.end_at = params.end_at | ||||||
|  |         try: | ||||||
|  |             slot.full_clean() | ||||||
|  |             slot.save() | ||||||
|  |         except ValidationError as e: | ||||||
|  |             return self.create_response({"detail": dict(e)}, status_code=409) | ||||||
							
								
								
									
										6
									
								
								reservation/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								reservation/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | from django.apps import AppConfig | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ReservationConfig(AppConfig): | ||||||
|  |     default_auto_field = "django.db.models.BigAutoField" | ||||||
|  |     name = "reservation" | ||||||
							
								
								
									
										60
									
								
								reservation/forms.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								reservation/forms.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | from django import forms | ||||||
|  | from django.core.exceptions import NON_FIELD_ERRORS | ||||||
|  | from django.utils.translation import gettext_lazy as _ | ||||||
|  |  | ||||||
|  | from club.widgets.ajax_select import AutoCompleteSelectClub | ||||||
|  | from core.models import User | ||||||
|  | from core.views.forms import FutureDateTimeField, SelectDateTime | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomCreateForm(forms.ModelForm): | ||||||
|  |     required_css_class = "required" | ||||||
|  |     error_css_class = "error" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         model = Room | ||||||
|  |         fields = ["name", "club", "location", "description"] | ||||||
|  |         widgets = {"club": AutoCompleteSelectClub} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomUpdateForm(forms.ModelForm): | ||||||
|  |     required_css_class = "required" | ||||||
|  |     error_css_class = "error" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         model = Room | ||||||
|  |         fields = ["name", "club", "location", "description"] | ||||||
|  |         widgets = {"club": AutoCompleteSelectClub} | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, request_user: User, **kwargs): | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |         if not request_user.has_perm("reservation.change_room"): | ||||||
|  |             # if the user doesn't have the global edition permission | ||||||
|  |             # (i.e. it's a club board member, but not a sith admin) | ||||||
|  |             # some fields aren't editable | ||||||
|  |             del self.fields["club"] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ReservationForm(forms.ModelForm): | ||||||
|  |     required_css_class = "required" | ||||||
|  |     error_css_class = "error" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         model = ReservationSlot | ||||||
|  |         fields = ["room", "start_at", "end_at", "comment"] | ||||||
|  |         field_classes = {"start_at": FutureDateTimeField, "end_at": FutureDateTimeField} | ||||||
|  |         widgets = {"start_at": SelectDateTime(), "end_at": SelectDateTime()} | ||||||
|  |         error_messages = { | ||||||
|  |             NON_FIELD_ERRORS: { | ||||||
|  |                 "start_after_end": _("The start must be set before the end") | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, author: User, **kwargs): | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |         self.author = author | ||||||
|  |  | ||||||
|  |     def save(self, commit: bool = True):  # noqa FBT001 | ||||||
|  |         self.instance.author = self.author | ||||||
|  |         return super().save(commit) | ||||||
							
								
								
									
										117
									
								
								reservation/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								reservation/migrations/0001_initial.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | |||||||
|  | # Generated by Django 5.2.1 on 2025-06-05 10:44 | ||||||
|  |  | ||||||
|  | import django.core.validators | ||||||
|  | import django.db.models.deletion | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |     initial = True | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ("club", "0014_alter_club_options_rename_unix_name_club_slug_name_and_more"), | ||||||
|  |         migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name="Room", | ||||||
|  |             fields=[ | ||||||
|  |                 ( | ||||||
|  |                     "id", | ||||||
|  |                     models.BigAutoField( | ||||||
|  |                         auto_created=True, | ||||||
|  |                         primary_key=True, | ||||||
|  |                         serialize=False, | ||||||
|  |                         verbose_name="ID", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ("name", models.CharField(max_length=100, verbose_name="room name")), | ||||||
|  |                 ( | ||||||
|  |                     "description", | ||||||
|  |                     models.TextField( | ||||||
|  |                         blank=True, default="", verbose_name="description" | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "location", | ||||||
|  |                     models.CharField( | ||||||
|  |                         blank=True, | ||||||
|  |                         choices=[ | ||||||
|  |                             ("BELFORT", "Belfort"), | ||||||
|  |                             ("SEVENANS", "Sévenans"), | ||||||
|  |                             ("MONTBELIARD", "Montbéliard"), | ||||||
|  |                         ], | ||||||
|  |                         verbose_name="site", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "club", | ||||||
|  |                     models.ForeignKey( | ||||||
|  |                         help_text="The club which manages this room", | ||||||
|  |                         on_delete=django.db.models.deletion.CASCADE, | ||||||
|  |                         related_name="reservable_rooms", | ||||||
|  |                         to="club.club", | ||||||
|  |                         verbose_name="room owner", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 "verbose_name": "reservable room", | ||||||
|  |                 "verbose_name_plural": "reservable rooms", | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name="ReservationSlot", | ||||||
|  |             fields=[ | ||||||
|  |                 ( | ||||||
|  |                     "id", | ||||||
|  |                     models.BigAutoField( | ||||||
|  |                         auto_created=True, | ||||||
|  |                         primary_key=True, | ||||||
|  |                         serialize=False, | ||||||
|  |                         verbose_name="ID", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "comment", | ||||||
|  |                     models.TextField(blank=True, default="", verbose_name="comment"), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "start_at", | ||||||
|  |                     models.DateTimeField(db_index=True, verbose_name="slot start"), | ||||||
|  |                 ), | ||||||
|  |                 ("end_at", models.DateTimeField(verbose_name="slot end")), | ||||||
|  |                 ("created_at", models.DateTimeField(auto_now_add=True)), | ||||||
|  |                 ( | ||||||
|  |                     "author", | ||||||
|  |                     models.ForeignKey( | ||||||
|  |                         on_delete=django.db.models.deletion.CASCADE, | ||||||
|  |                         to=settings.AUTH_USER_MODEL, | ||||||
|  |                         verbose_name="author", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "room", | ||||||
|  |                     models.ForeignKey( | ||||||
|  |                         on_delete=django.db.models.deletion.CASCADE, | ||||||
|  |                         related_name="slots", | ||||||
|  |                         to="reservation.room", | ||||||
|  |                         verbose_name="reserved room", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 "verbose_name": "reservation slot", | ||||||
|  |                 "verbose_name_plural": "reservation slots", | ||||||
|  |                 "constraints": [ | ||||||
|  |                     models.CheckConstraint( | ||||||
|  |                         condition=models.Q(("end_at__gt", models.F("start_at"))), | ||||||
|  |                         name="reservation_slot_end_after_start", | ||||||
|  |                         violation_error_code="start_after_end", | ||||||
|  |                     ) | ||||||
|  |                 ], | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										0
									
								
								reservation/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								reservation/migrations/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										100
									
								
								reservation/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								reservation/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | |||||||
|  | from __future__ import annotations | ||||||
|  |  | ||||||
|  | from typing import Self | ||||||
|  |  | ||||||
|  | from django.core.exceptions import ValidationError | ||||||
|  | from django.db import models | ||||||
|  | from django.db.models import F, Q | ||||||
|  | from django.utils.translation import gettext_lazy as _ | ||||||
|  |  | ||||||
|  | from club.models import Club | ||||||
|  | from core.models import User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Room(models.Model): | ||||||
|  |     name = models.CharField(_("room name"), max_length=100) | ||||||
|  |     description = models.TextField(_("description"), blank=True, default="") | ||||||
|  |     club = models.ForeignKey( | ||||||
|  |         Club, | ||||||
|  |         on_delete=models.CASCADE, | ||||||
|  |         related_name="reservable_rooms", | ||||||
|  |         verbose_name=_("room owner"), | ||||||
|  |         help_text=_("The club which manages this room"), | ||||||
|  |     ) | ||||||
|  |     location = models.CharField( | ||||||
|  |         _("site"), | ||||||
|  |         blank=True, | ||||||
|  |         choices=[ | ||||||
|  |             ("BELFORT", "Belfort"), | ||||||
|  |             ("SEVENANS", "Sévenans"), | ||||||
|  |             ("MONTBELIARD", "Montbéliard"), | ||||||
|  |         ], | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         verbose_name = _("reservable room") | ||||||
|  |         verbose_name_plural = _("reservable rooms") | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return self.name | ||||||
|  |  | ||||||
|  |     def can_be_edited_by(self, user: User) -> bool: | ||||||
|  |         # a user may edit a room if it has the global perm | ||||||
|  |         # or is in the owner club board | ||||||
|  |         return user.has_perm("reservation.change_room") or self.club.board_group_id in [ | ||||||
|  |             g.id for g in user.cached_groups | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ReservationSlotQuerySet(models.QuerySet): | ||||||
|  |     def overlapping_with(self, slot: ReservationSlot) -> Self: | ||||||
|  |         return self.filter( | ||||||
|  |             Q(start_at__lt=slot.start_at, end_at__gt=slot.start_at) | ||||||
|  |             | Q(start_at__lt=slot.end_at, end_at__gt=slot.end_at) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ReservationSlot(models.Model): | ||||||
|  |     room = models.ForeignKey( | ||||||
|  |         Room, | ||||||
|  |         on_delete=models.CASCADE, | ||||||
|  |         related_name="slots", | ||||||
|  |         verbose_name=_("reserved room"), | ||||||
|  |     ) | ||||||
|  |     author = models.ForeignKey(User, on_delete=models.CASCADE, verbose_name=_("author")) | ||||||
|  |     comment = models.TextField(_("comment"), blank=True, default="") | ||||||
|  |     start_at = models.DateTimeField(_("slot start"), db_index=True) | ||||||
|  |     end_at = models.DateTimeField(_("slot end")) | ||||||
|  |     created_at = models.DateTimeField(auto_now_add=True) | ||||||
|  |  | ||||||
|  |     objects = ReservationSlotQuerySet.as_manager() | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         verbose_name = _("reservation slot") | ||||||
|  |         verbose_name_plural = _("reservation slots") | ||||||
|  |         constraints = [ | ||||||
|  |             models.CheckConstraint( | ||||||
|  |                 condition=Q(end_at__gt=F("start_at")), | ||||||
|  |                 name="reservation_slot_end_after_start", | ||||||
|  |                 violation_error_code="start_after_end", | ||||||
|  |             ) | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return f"{self.room.name} : {self.start_at} - {self.end_at}" | ||||||
|  |  | ||||||
|  |     def clean(self): | ||||||
|  |         super().clean() | ||||||
|  |         if self.end_at is None or self.start_at is None: | ||||||
|  |             # if there is no start or no end, then there is no | ||||||
|  |             # point to check if this perm overlap with another, | ||||||
|  |             # so in this case, don't do the overlap check and let | ||||||
|  |             # Django manage the non-null constraint error. | ||||||
|  |             return | ||||||
|  |         overlapping = ReservationSlot.objects.overlapping_with(self).filter( | ||||||
|  |             room_id=self.room_id | ||||||
|  |         ) | ||||||
|  |         if self.id is not None: | ||||||
|  |             overlapping = overlapping.exclude(id=self.id) | ||||||
|  |         if overlapping.exists(): | ||||||
|  |             raise ValidationError(_("There is already a reservation on this slot.")) | ||||||
							
								
								
									
										46
									
								
								reservation/schemas.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								reservation/schemas.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  | from ninja import FilterSchema, ModelSchema, Schema | ||||||
|  | from pydantic import Field, FutureDatetime | ||||||
|  |  | ||||||
|  | from club.schemas import SimpleClubSchema | ||||||
|  | from core.schemas import SimpleUserSchema | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomFilterSchema(FilterSchema): | ||||||
|  |     club: set[int] | None = Field(None, q="club_id__in") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomSchema(ModelSchema): | ||||||
|  |     class Meta: | ||||||
|  |         model = Room | ||||||
|  |         fields = ["id", "name", "description", "location"] | ||||||
|  |  | ||||||
|  |     club: SimpleClubSchema | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def resolve_location(obj: Room): | ||||||
|  |         return obj.get_location_display() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SlotFilterSchema(FilterSchema): | ||||||
|  |     after: datetime = Field(default=None, q="end_at__gt") | ||||||
|  |     before: datetime = Field(default=None, q="start_at__lt") | ||||||
|  |     room: set[int] | None = None | ||||||
|  |     club: set[int] | None = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SlotSchema(ModelSchema): | ||||||
|  |     class Meta: | ||||||
|  |         model = ReservationSlot | ||||||
|  |         fields = ["id", "room", "comment"] | ||||||
|  |  | ||||||
|  |     start: datetime = Field(alias="start_at") | ||||||
|  |     end: datetime = Field(alias="end_at") | ||||||
|  |     author: SimpleUserSchema | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UpdateReservationSlotSchema(Schema): | ||||||
|  |     start_at: FutureDatetime | ||||||
|  |     end_at: FutureDatetime | ||||||
| @@ -0,0 +1,138 @@ | |||||||
|  | import { inheritHtmlElement, registerComponent } from "#core:utils/web-components"; | ||||||
|  | import { | ||||||
|  |   Calendar, | ||||||
|  |   type DateSelectArg, | ||||||
|  |   type EventDropArg, | ||||||
|  |   type EventSourceFuncArg, | ||||||
|  | } from "@fullcalendar/core"; | ||||||
|  | import enLocale from "@fullcalendar/core/locales/en-gb"; | ||||||
|  | import frLocale from "@fullcalendar/core/locales/fr"; | ||||||
|  |  | ||||||
|  | import { | ||||||
|  |   type ReservationslotFetchSlotsData, | ||||||
|  |   type SlotSchema, | ||||||
|  |   reservableroomFetchRooms, | ||||||
|  |   reservationslotFetchSlots, | ||||||
|  |   reservationslotUpdateSlot, | ||||||
|  | } from "#openapi"; | ||||||
|  |  | ||||||
|  | import { paginated } from "#core:utils/api"; | ||||||
|  | import type { SlotSelectedEventArg } from "#reservation:reservation/types"; | ||||||
|  | import interactionPlugin, { type EventResizeDoneArg } from "@fullcalendar/interaction"; | ||||||
|  | import resourceTimelinePlugin from "@fullcalendar/resource-timeline"; | ||||||
|  |  | ||||||
|  | @registerComponent("room-scheduler") | ||||||
|  | export class RoomScheduler extends inheritHtmlElement("div") { | ||||||
|  |   static observedAttributes = ["locale", "can_edit_slot", "can_create_slot"]; | ||||||
|  |   private scheduler: Calendar; | ||||||
|  |   private locale = "en"; | ||||||
|  |   private canEditSlot = false; | ||||||
|  |   private canBookSlot = false; | ||||||
|  |   private canDeleteSlot = false; | ||||||
|  |  | ||||||
|  |   attributeChangedCallback(name: string, _oldValue?: string, newValue?: string) { | ||||||
|  |     if (name === "locale") { | ||||||
|  |       this.locale = newValue; | ||||||
|  |     } | ||||||
|  |     if (name === "can_edit_slot") { | ||||||
|  |       this.canEditSlot = newValue.toLowerCase() === "true"; | ||||||
|  |     } | ||||||
|  |     if (name === "can_create_slot") { | ||||||
|  |       this.canBookSlot = newValue.toLowerCase() === "true"; | ||||||
|  |     } | ||||||
|  |     if (name === "can_delete_slot") { | ||||||
|  |       this.canDeleteSlot = newValue.toLowerCase() === "true"; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Fetch the events displayed in the timeline. | ||||||
|  |    * cf https://fullcalendar.io/docs/events-function | ||||||
|  |    */ | ||||||
|  |   async fetchEvents(fetchInfo: EventSourceFuncArg) { | ||||||
|  |     const res: SlotSchema[] = await paginated(reservationslotFetchSlots, { | ||||||
|  |       query: { after: fetchInfo.startStr, before: fetchInfo.endStr }, | ||||||
|  |     } as ReservationslotFetchSlotsData); | ||||||
|  |     return res.map((i) => | ||||||
|  |       Object.assign(i, { | ||||||
|  |         title: `${i.author.first_name} ${i.author.last_name}`, | ||||||
|  |         resourceId: i.room, | ||||||
|  |         editable: new Date(i.start) > new Date(), | ||||||
|  |       }), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Fetch the resources which events are associated with. | ||||||
|  |    * cf https://fullcalendar.io/docs/resources-function | ||||||
|  |    */ | ||||||
|  |   async fetchResources() { | ||||||
|  |     const res = await reservableroomFetchRooms(); | ||||||
|  |     return res.data.map((i) => Object.assign(i, { title: i.name, group: i.location })); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Send a request to the API to change | ||||||
|  |    * the start and the duration of a reservation slot | ||||||
|  |    */ | ||||||
|  |   async changeReservation(args: EventDropArg | EventResizeDoneArg) { | ||||||
|  |     const response = await reservationslotUpdateSlot({ | ||||||
|  |       // biome-ignore lint/style/useNamingConvention: api is snake_case | ||||||
|  |       path: { slot_id: Number.parseInt(args.event.id) }, | ||||||
|  |       // biome-ignore lint/style/useNamingConvention: api is snake_case | ||||||
|  |       body: { start_at: args.event.startStr, end_at: args.event.endStr }, | ||||||
|  |     }); | ||||||
|  |     if (response.response.ok) { | ||||||
|  |       document.dispatchEvent(new CustomEvent("reservationSlotChanged")); | ||||||
|  |       this.scheduler.refetchEvents(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   selectFreeSlot(infos: DateSelectArg) { | ||||||
|  |     document.dispatchEvent( | ||||||
|  |       new CustomEvent<SlotSelectedEventArg>("timeSlotSelected", { | ||||||
|  |         detail: { | ||||||
|  |           ressource: Number.parseInt(infos.resource.id), | ||||||
|  |           start: infos.startStr, | ||||||
|  |           end: infos.endStr, | ||||||
|  |         }, | ||||||
|  |       }), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   connectedCallback() { | ||||||
|  |     super.connectedCallback(); | ||||||
|  |     this.scheduler = new Calendar(this.node, { | ||||||
|  |       schedulerLicenseKey: "GPL-My-Project-Is-Open-Source", | ||||||
|  |       initialView: "resourceTimelineDay", | ||||||
|  |       headerToolbar: { | ||||||
|  |         left: "prev,next today", | ||||||
|  |         center: "title", | ||||||
|  |         right: "resourceTimelineDay,resourceTimelineWeek", | ||||||
|  |       }, | ||||||
|  |       plugins: [resourceTimelinePlugin, interactionPlugin], | ||||||
|  |       locales: [frLocale, enLocale], | ||||||
|  |       height: "auto", | ||||||
|  |       locale: this.locale, | ||||||
|  |       resourceGroupField: "group", | ||||||
|  |       resourceAreaHeaderContent: gettext("Rooms"), | ||||||
|  |       editable: this.canEditSlot, | ||||||
|  |       snapDuration: "00:15", | ||||||
|  |       eventConstraint: { start: new Date() }, // forbid edition of past events | ||||||
|  |       eventOverlap: false, | ||||||
|  |       eventResourceEditable: false, | ||||||
|  |       refetchResourcesOnNavigate: true, | ||||||
|  |       resourceAreaWidth: "20%", | ||||||
|  |       resources: this.fetchResources, | ||||||
|  |       events: this.fetchEvents, | ||||||
|  |       select: this.selectFreeSlot, | ||||||
|  |       selectOverlap: false, | ||||||
|  |       selectable: this.canBookSlot, | ||||||
|  |       selectConstraint: { start: new Date() }, | ||||||
|  |       nowIndicator: true, | ||||||
|  |       eventDrop: this.changeReservation, | ||||||
|  |       eventResize: this.changeReservation, | ||||||
|  |     }); | ||||||
|  |     this.scheduler.render(); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -0,0 +1,39 @@ | |||||||
|  | import { AlertMessage } from "#core:utils/alert-message"; | ||||||
|  | import type { SlotSelectedEventArg } from "#reservation:reservation/types"; | ||||||
|  |  | ||||||
|  | document.addEventListener("alpine:init", () => { | ||||||
|  |   Alpine.data("slotReservation", () => ({ | ||||||
|  |     start: null as string, | ||||||
|  |     end: null as string, | ||||||
|  |     room: null as number, | ||||||
|  |     showForm: false, | ||||||
|  |  | ||||||
|  |     init() { | ||||||
|  |       document.addEventListener( | ||||||
|  |         "timeSlotSelected", | ||||||
|  |         (event: CustomEvent<SlotSelectedEventArg>) => { | ||||||
|  |           this.start = event.detail.start.split("+")[0]; | ||||||
|  |           this.end = event.detail.end.split("+")[0]; | ||||||
|  |           this.room = event.detail.ressource; | ||||||
|  |           this.showForm = true; | ||||||
|  |           this.$nextTick(() => this.$el.scrollIntoView({ behavior: "smooth" })).then(); | ||||||
|  |         }, | ||||||
|  |       ); | ||||||
|  |     }, | ||||||
|  |   })); | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Component that will catch events sent from the scheduler | ||||||
|  |    * to display success messages accordingly. | ||||||
|  |    */ | ||||||
|  |   Alpine.data("scheduleMessages", () => ({ | ||||||
|  |     alertMessage: new AlertMessage({ defaultDuration: 2000 }), | ||||||
|  |     init() { | ||||||
|  |       document.addEventListener("reservationSlotChanged", (_event: CustomEvent) => { | ||||||
|  |         this.alertMessage.display(gettext("This slot has been successfully moved"), { | ||||||
|  |           success: true, | ||||||
|  |         }); | ||||||
|  |       }); | ||||||
|  |     }, | ||||||
|  |   })); | ||||||
|  | }); | ||||||
							
								
								
									
										5
									
								
								reservation/static/bundled/reservation/types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								reservation/static/bundled/reservation/types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | export interface SlotSelectedEventArg { | ||||||
|  |   start: string; | ||||||
|  |   end: string; | ||||||
|  |   ressource: number; | ||||||
|  | } | ||||||
							
								
								
									
										39
									
								
								reservation/static/reservation/reservation.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								reservation/static/reservation/reservation.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | #slot-reservation { | ||||||
|  |   margin-top: 3em; | ||||||
|  |  | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  |   justify-content: center; | ||||||
|  |  | ||||||
|  |   h3 { | ||||||
|  |     display: block; | ||||||
|  |     margin: auto; | ||||||
|  |     text-align: left; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .alert, .error { | ||||||
|  |     display: block; | ||||||
|  |     margin: 1em auto auto; | ||||||
|  |     max-width: 400px; | ||||||
|  |     word-wrap: break-word; | ||||||
|  |     text-wrap: wrap; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   form { | ||||||
|  |     display: flex; | ||||||
|  |     flex-direction: column; | ||||||
|  |     gap: .5em; | ||||||
|  |     justify-content: center; | ||||||
|  |  | ||||||
|  |     .buttons-row { | ||||||
|  |       input[type="submit"], button { | ||||||
|  |         margin: 0; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     textarea { | ||||||
|  |       max-width: unset; | ||||||
|  |       width: 100%; | ||||||
|  |       margin-top: unset; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -0,0 +1,51 @@ | |||||||
|  | <section | ||||||
|  |   id="slot-reservation" | ||||||
|  |   x-data="slotReservation" | ||||||
|  |   x-show="showForm" | ||||||
|  |   hx-target="this" | ||||||
|  |   hx-ext="alpine-morph" | ||||||
|  |   hx-swap="morph" | ||||||
|  | > | ||||||
|  |   <h3>{% trans %}Book a room{% endtrans %}</h3> | ||||||
|  |   {% set non_field_errors = form.non_field_errors() %} | ||||||
|  |   {% if non_field_errors %} | ||||||
|  |     <div class="alert alert-red"> | ||||||
|  |       {% for error in non_field_errors %} | ||||||
|  |         <span>{{ error }}</span> | ||||||
|  |       {% endfor %} | ||||||
|  |     </div> | ||||||
|  |   {% endif %} | ||||||
|  |   <form | ||||||
|  |     id="slot-reservation-form" | ||||||
|  |     hx-post="{{ url("reservation:make_reservation") }}" | ||||||
|  |     hx-disabled-elt="find input[type='submit']" | ||||||
|  |   > | ||||||
|  |     {% csrf_token %} | ||||||
|  |     <div class="form-group"> | ||||||
|  |       {{ form.room.errors }} | ||||||
|  |       {{ form.room.label_tag() }} | ||||||
|  |       {{ form.room|add_attr("x-model=room") }} | ||||||
|  |     </div> | ||||||
|  |     <div class="form-group"> | ||||||
|  |       {{ form.start_at.errors }} | ||||||
|  |       {{ form.start_at.label_tag() }} | ||||||
|  |       {{ form.start_at|add_attr("x-model=start") }} | ||||||
|  |     </div> | ||||||
|  |     <div class="form-group"> | ||||||
|  |       {{ form.end_at.errors }} | ||||||
|  |       {{ form.end_at.label_tag() }} | ||||||
|  |       {{ form.end_at|add_attr("x-model=end") }} | ||||||
|  |     </div> | ||||||
|  |     <div class="form-group"> | ||||||
|  |       {{ form.comment.errors }} | ||||||
|  |       {{ form.comment.label_tag() }} | ||||||
|  |       {{ form.comment }} | ||||||
|  |     </div> | ||||||
|  |     <div class="row gap buttons-row"> | ||||||
|  |       <button class="btn btn-grey grow" @click.prevent="showForm = false"> | ||||||
|  |         {% trans %}Cancel{% endtrans %} | ||||||
|  |       </button> | ||||||
|  |       <input class="btn btn-blue grow" type="submit"> | ||||||
|  |     </div> | ||||||
|  |   </form> | ||||||
|  | </section> | ||||||
							
								
								
									
										27
									
								
								reservation/templates/reservation/macros.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								reservation/templates/reservation/macros.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | {% macro room_detail(room, can_edit, can_delete) %} | ||||||
|  |   <div class="card card-row card-row-m"> | ||||||
|  |     <div class="card-content"> | ||||||
|  |       <strong class="card-title">{{ room.name }}</strong> | ||||||
|  |       <em>{{ room.get_location_display() }}</em> | ||||||
|  |       <p>{{ room.description|truncate(250) }}</p> | ||||||
|  |     </div> | ||||||
|  |     <div class="card-top-left"> | ||||||
|  |       {% if can_edit %} | ||||||
|  |         <a | ||||||
|  |           class="btn btn-grey btn-no-text" | ||||||
|  |           href="{{ url("reservation:room_edit", room_id=room.id) }}" | ||||||
|  |         > | ||||||
|  |           <i class="fa fa-edit"></i> | ||||||
|  |         </a> | ||||||
|  |       {% endif %} | ||||||
|  |       {% if can_delete %} | ||||||
|  |         <a | ||||||
|  |           class="btn btn-red btn-no-text" | ||||||
|  |           href="{{ url("reservation:room_delete", room_id=room.id) }}" | ||||||
|  |         > | ||||||
|  |           <i class="fa fa-trash"></i> | ||||||
|  |         </a> | ||||||
|  |       {% endif %} | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | {% endmacro %} | ||||||
							
								
								
									
										33
									
								
								reservation/templates/reservation/schedule.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								reservation/templates/reservation/schedule.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | {% extends "core/base.jinja" %} | ||||||
|  |  | ||||||
|  | {% block additional_js %} | ||||||
|  |   <script type="module" src="{{ static("bundled/reservation/components/room-scheduler-index.ts") }}"></script> | ||||||
|  |   <script type="module" src="{{ static("bundled/reservation/slot-reservation-index.ts") }}"></script> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block additional_css %} | ||||||
|  |   <link rel="stylesheet" href="{{ static('core/components/calendar.scss') }}"> | ||||||
|  |   <link rel="stylesheet" href="{{ static('reservation/reservation.scss') }}"> | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |   <h2 class="margin-bottom">{% trans %}Room reservation{% endtrans %}</h2> | ||||||
|  |   <p | ||||||
|  |     x-data="scheduleMessages" | ||||||
|  |     class="alert snackbar" | ||||||
|  |     :class="alertMessage.success ? 'alert-green' : 'alert-red'" | ||||||
|  |     x-show="alertMessage.open" | ||||||
|  |     x-transition.duration.500ms | ||||||
|  |     x-text="alertMessage.content" | ||||||
|  |   ></p> | ||||||
|  |   <room-scheduler | ||||||
|  |     locale="{{ LANGUAGE_CODE }}" | ||||||
|  |     can_edit_slot="{{ user.has_perm("reservation.change_reservationslot") }}" | ||||||
|  |     can_create_slot="{{ user.has_perm("reservation.add_reservationslot") }}" | ||||||
|  |   ></room-scheduler> | ||||||
|  |   {% if user.has_perm("reservation.add_reservationslot") %} | ||||||
|  |     <p><em>{% trans %}You can book a room by selecting a free slot in the calendar.{% endtrans %}</em></p> | ||||||
|  |     {{ add_slot_fragment }} | ||||||
|  |   {% endif %} | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										0
									
								
								reservation/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								reservation/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										113
									
								
								reservation/tests/test_room.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								reservation/tests/test_room.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | |||||||
|  | import pytest | ||||||
|  | from django.contrib.auth.models import Permission | ||||||
|  | from django.test import Client | ||||||
|  | from django.urls import reverse | ||||||
|  | from model_bakery import baker | ||||||
|  | from pytest_django.asserts import assertNumQueries, assertRedirects | ||||||
|  |  | ||||||
|  | from club.models import Club | ||||||
|  | from core.models import User | ||||||
|  | from reservation.forms import RoomUpdateForm | ||||||
|  | from reservation.models import Room | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestFetchRoom: | ||||||
|  |     @pytest.fixture | ||||||
|  |     def user(self): | ||||||
|  |         return baker.make( | ||||||
|  |             User, | ||||||
|  |             user_permissions=[Permission.objects.get(codename="view_room")], | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_fetch_simple(self, client: Client, user: User): | ||||||
|  |         rooms = baker.make(Room, _quantity=3, _bulk_create=True) | ||||||
|  |         client.force_login(user) | ||||||
|  |         response = client.get(reverse("api:fetch_reservable_rooms")) | ||||||
|  |         assert response.status_code == 200 | ||||||
|  |         assert response.json() == [ | ||||||
|  |             { | ||||||
|  |                 "id": room.id, | ||||||
|  |                 "name": room.name, | ||||||
|  |                 "description": room.description, | ||||||
|  |                 "location": room.location, | ||||||
|  |                 "club": {"id": room.club.id, "name": room.club.name}, | ||||||
|  |             } | ||||||
|  |             for room in rooms | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def test_nb_queries(self, client: Client, user: User): | ||||||
|  |         client.force_login(user) | ||||||
|  |         with assertNumQueries(5): | ||||||
|  |             # 4 for authentication | ||||||
|  |             # 1 to fetch the actual data | ||||||
|  |             client.get(reverse("api:fetch_reservable_rooms")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestCreateRoom: | ||||||
|  |     def test_ok(self, client: Client): | ||||||
|  |         perm = Permission.objects.get(codename="add_room") | ||||||
|  |         club = baker.make(Club) | ||||||
|  |         client.force_login( | ||||||
|  |             baker.make(User, user_permissions=[perm], groups=[club.board_group]) | ||||||
|  |         ) | ||||||
|  |         response = client.post( | ||||||
|  |             reverse("reservation:room_create"), | ||||||
|  |             data={"club": club.id, "name": "test", "location": "BELFORT"}, | ||||||
|  |         ) | ||||||
|  |         assertRedirects(response, reverse("club:tools", kwargs={"club_id": club.id})) | ||||||
|  |         room = Room.objects.last() | ||||||
|  |         assert room is not None | ||||||
|  |         assert room.club == club | ||||||
|  |         assert room.name == "test" | ||||||
|  |         assert room.location == "BELFORT" | ||||||
|  |  | ||||||
|  |     def test_permission_denied(self, client: Client): | ||||||
|  |         club = baker.make(Club) | ||||||
|  |         client.force_login(baker.make(User)) | ||||||
|  |         response = client.get(reverse("reservation:room_create")) | ||||||
|  |         assert response.status_code == 403 | ||||||
|  |         response = client.post( | ||||||
|  |             reverse("reservation:room_create"), | ||||||
|  |             data={"club": club.id, "name": "test", "location": "BELFORT"}, | ||||||
|  |         ) | ||||||
|  |         assert response.status_code == 403 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestUpdateRoom: | ||||||
|  |     def test_ok(self, client: Client): | ||||||
|  |         club = baker.make(Club) | ||||||
|  |         room = baker.make(Room, club=club) | ||||||
|  |         client.force_login(baker.make(User, groups=[club.board_group])) | ||||||
|  |         url = reverse("reservation:room_edit", kwargs={"room_id": room.id}) | ||||||
|  |         response = client.post(url, data={"name": "test", "location": "BELFORT"}) | ||||||
|  |         assertRedirects(response, url) | ||||||
|  |         room.refresh_from_db() | ||||||
|  |         assert room.club == club | ||||||
|  |         assert room.name == "test" | ||||||
|  |         assert room.location == "BELFORT" | ||||||
|  |  | ||||||
|  |     def test_permission_denied(self, client: Client): | ||||||
|  |         club = baker.make(Club) | ||||||
|  |         room = baker.make(Room, club=club) | ||||||
|  |         client.force_login(baker.make(User)) | ||||||
|  |         url = reverse("reservation:room_edit", kwargs={"room_id": room.id}) | ||||||
|  |         response = client.get(url) | ||||||
|  |         assert response.status_code == 403 | ||||||
|  |         response = client.post(url, data={"name": "test", "location": "BELFORT"}) | ||||||
|  |         assert response.status_code == 403 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestUpdateRoomForm: | ||||||
|  |     def test_form_club_edition_rights(self): | ||||||
|  |         """The club field should appear only if the request user can edit it.""" | ||||||
|  |         room = baker.make(Room) | ||||||
|  |         perm = Permission.objects.get(codename="change_room") | ||||||
|  |         user_authorized = baker.make(User, user_permissions=[perm]) | ||||||
|  |         assert "club" in RoomUpdateForm(request_user=user_authorized).fields | ||||||
|  |  | ||||||
|  |         user_forbidden = baker.make(User, groups=[room.club.board_group]) | ||||||
|  |         assert "club" not in RoomUpdateForm(request_user=user_forbidden).fields | ||||||
							
								
								
									
										207
									
								
								reservation/tests/test_slot.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								reservation/tests/test_slot.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,207 @@ | |||||||
|  | from datetime import timedelta | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | from django.contrib.auth.models import Permission | ||||||
|  | from django.test import Client | ||||||
|  | from django.urls import reverse | ||||||
|  | from django.utils.timezone import now | ||||||
|  | from model_bakery import baker | ||||||
|  | from pytest_django.asserts import assertNumQueries | ||||||
|  |  | ||||||
|  | from core.models import User | ||||||
|  | from reservation.forms import ReservationForm | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestFetchReservationSlotsApi: | ||||||
|  |     @pytest.fixture | ||||||
|  |     def user(self): | ||||||
|  |         perm = Permission.objects.get(codename="view_reservationslot") | ||||||
|  |         return baker.make(User, user_permissions=[perm]) | ||||||
|  |  | ||||||
|  |     def test_fetch_simple(self, client: Client, user: User): | ||||||
|  |         slots = baker.make(ReservationSlot, _quantity=5, _bulk_create=True) | ||||||
|  |         client.force_login(user) | ||||||
|  |         response = client.get(reverse("api:fetch_reservation_slots")) | ||||||
|  |         assert response.json()["results"] == [ | ||||||
|  |             { | ||||||
|  |                 "id": slot.id, | ||||||
|  |                 "room": slot.room_id, | ||||||
|  |                 "comment": slot.comment, | ||||||
|  |                 "start": slot.start_at.isoformat(timespec="milliseconds").replace( | ||||||
|  |                     "+00:00", "Z" | ||||||
|  |                 ), | ||||||
|  |                 "end": slot.end_at.isoformat(timespec="milliseconds").replace( | ||||||
|  |                     "+00:00", "Z" | ||||||
|  |                 ), | ||||||
|  |                 "author": { | ||||||
|  |                     "id": slot.author.id, | ||||||
|  |                     "first_name": slot.author.first_name, | ||||||
|  |                     "last_name": slot.author.last_name, | ||||||
|  |                     "nick_name": slot.author.nick_name, | ||||||
|  |                 }, | ||||||
|  |             } | ||||||
|  |             for slot in slots | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def test_nb_queries(self, client: Client, user: User): | ||||||
|  |         client.force_login(user) | ||||||
|  |         with assertNumQueries(5): | ||||||
|  |             # 4 for authentication | ||||||
|  |             # 1 to fetch the actual data | ||||||
|  |             client.get(reverse("api:fetch_reservation_slots")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestUpdateReservationSlotApi: | ||||||
|  |     @pytest.fixture | ||||||
|  |     def user(self): | ||||||
|  |         perm = Permission.objects.get(codename="change_reservationslot") | ||||||
|  |         return baker.make(User, user_permissions=[perm]) | ||||||
|  |  | ||||||
|  |     @pytest.fixture | ||||||
|  |     def slot(self): | ||||||
|  |         return baker.make( | ||||||
|  |             ReservationSlot, | ||||||
|  |             start_at=now() + timedelta(hours=2), | ||||||
|  |             end_at=now() + timedelta(hours=4), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_ok(self, client: Client, user: User, slot: ReservationSlot): | ||||||
|  |         client.force_login(user) | ||||||
|  |         new_start = (slot.start_at + timedelta(hours=1)).replace(microsecond=0) | ||||||
|  |         response = client.patch( | ||||||
|  |             reverse("api:change_reservation_slot", kwargs={"slot_id": slot.id}), | ||||||
|  |             {"start_at": new_start, "end_at": new_start + timedelta(hours=2)}, | ||||||
|  |             content_type="application/json", | ||||||
|  |         ) | ||||||
|  |         assert response.status_code == 200 | ||||||
|  |         slot.refresh_from_db() | ||||||
|  |         assert slot.start_at.replace(microsecond=0) == new_start | ||||||
|  |         assert slot.end_at.replace(microsecond=0) == new_start + timedelta(hours=2) | ||||||
|  |  | ||||||
|  |     def test_change_past_event(self, client, user: User, slot: ReservationSlot): | ||||||
|  |         """Test that moving a slot that already began is impossible.""" | ||||||
|  |         client.force_login(user) | ||||||
|  |         new_start = now() - timedelta(hours=1) | ||||||
|  |         response = client.patch( | ||||||
|  |             reverse("api:change_reservation_slot", kwargs={"slot_id": slot.id}), | ||||||
|  |             {"start_at": new_start, "end_at": new_start + timedelta(hours=2)}, | ||||||
|  |             content_type="application/json", | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         assert response.status_code == 422 | ||||||
|  |  | ||||||
|  |     def test_move_event_to_occupied_slot( | ||||||
|  |         self, client: Client, user: User, slot: ReservationSlot | ||||||
|  |     ): | ||||||
|  |         client.force_login(user) | ||||||
|  |         other_slot = baker.make( | ||||||
|  |             ReservationSlot, | ||||||
|  |             room=slot.room, | ||||||
|  |             start_at=slot.end_at + timedelta(hours=1), | ||||||
|  |             end_at=slot.end_at + timedelta(hours=3), | ||||||
|  |         ) | ||||||
|  |         response = client.patch( | ||||||
|  |             reverse("api:change_reservation_slot", kwargs={"slot_id": slot.id}), | ||||||
|  |             { | ||||||
|  |                 "start_at": other_slot.start_at - timedelta(hours=1), | ||||||
|  |                 "end_at": other_slot.start_at + timedelta(hours=1), | ||||||
|  |             }, | ||||||
|  |             content_type="application/json", | ||||||
|  |         ) | ||||||
|  |         assert response.status_code == 409 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestReservationForm: | ||||||
|  |     def test_ok(self): | ||||||
|  |         start = now() + timedelta(hours=2) | ||||||
|  |         end = start + timedelta(hours=1) | ||||||
|  |         form = ReservationForm( | ||||||
|  |             author=baker.make(User), | ||||||
|  |             data={"room": baker.make(Room), "start_at": start, "end_at": end}, | ||||||
|  |         ) | ||||||
|  |         assert form.is_valid() | ||||||
|  |  | ||||||
|  |     @pytest.mark.parametrize( | ||||||
|  |         ("start_date", "end_date", "errors"), | ||||||
|  |         [ | ||||||
|  |             ( | ||||||
|  |                 now() - timedelta(hours=2), | ||||||
|  |                 now() + timedelta(hours=2), | ||||||
|  |                 {"start_at": ["Assurez-vous que cet horodatage est dans le futur"]}, | ||||||
|  |             ), | ||||||
|  |             ( | ||||||
|  |                 now() + timedelta(hours=3), | ||||||
|  |                 now() + timedelta(hours=2), | ||||||
|  |                 {"__all__": ["Le début doit être placé avant la fin"]}, | ||||||
|  |             ), | ||||||
|  |         ], | ||||||
|  |     ) | ||||||
|  |     def test_invalid_timedates(self, start_date, end_date, errors): | ||||||
|  |         form = ReservationForm( | ||||||
|  |             author=baker.make(User), | ||||||
|  |             data={"room": baker.make(Room), "start_at": start_date, "end_at": end_date}, | ||||||
|  |         ) | ||||||
|  |         assert not form.is_valid() | ||||||
|  |         assert form.errors == errors | ||||||
|  |  | ||||||
|  |     def test_unavailable_room(self): | ||||||
|  |         room = baker.make(Room) | ||||||
|  |         baker.make( | ||||||
|  |             ReservationSlot, | ||||||
|  |             room=room, | ||||||
|  |             start_at=now() + timedelta(hours=2), | ||||||
|  |             end_at=now() + timedelta(hours=4), | ||||||
|  |         ) | ||||||
|  |         form = ReservationForm( | ||||||
|  |             author=baker.make(User), | ||||||
|  |             data={ | ||||||
|  |                 "room": room, | ||||||
|  |                 "start_at": now() + timedelta(hours=1), | ||||||
|  |                 "end_at": now() + timedelta(hours=3), | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |         assert not form.is_valid() | ||||||
|  |         assert form.errors == { | ||||||
|  |             "__all__": ["Il y a déjà une réservation sur ce créneau."] | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestCreateReservationSlot: | ||||||
|  |     @pytest.fixture | ||||||
|  |     def user(self): | ||||||
|  |         perms = Permission.objects.filter( | ||||||
|  |             codename__in=["add_reservationslot", "view_reservationslot"] | ||||||
|  |         ) | ||||||
|  |         return baker.make(User, user_permissions=list(perms)) | ||||||
|  |  | ||||||
|  |     def test_ok(self, client: Client, user: User): | ||||||
|  |         client.force_login(user) | ||||||
|  |         start = now() + timedelta(hours=2) | ||||||
|  |         end = start + timedelta(hours=1) | ||||||
|  |         room = baker.make(Room) | ||||||
|  |         response = client.post( | ||||||
|  |             reverse("reservation:make_reservation"), | ||||||
|  |             {"room": room.id, "start_at": start, "end_at": end}, | ||||||
|  |         ) | ||||||
|  |         assert response.status_code == 200 | ||||||
|  |         assert response.headers.get("HX-Redirect", "") == reverse("reservation:main") | ||||||
|  |         slot = ReservationSlot.objects.filter(room=room).last() | ||||||
|  |         assert slot is not None | ||||||
|  |         assert slot.start_at == start | ||||||
|  |         assert slot.end_at == end | ||||||
|  |         assert slot.author == user | ||||||
|  |  | ||||||
|  |     def test_permissions_denied(self, client: Client): | ||||||
|  |         client.force_login(baker.make(User)) | ||||||
|  |         start = now() + timedelta(hours=2) | ||||||
|  |         end = start + timedelta(hours=1) | ||||||
|  |         response = client.post( | ||||||
|  |             reverse("reservation:make_reservation"), | ||||||
|  |             {"room": baker.make(Room), "start_at": start, "end_at": end}, | ||||||
|  |         ) | ||||||
|  |         assert response.status_code == 403 | ||||||
							
								
								
									
										19
									
								
								reservation/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								reservation/urls.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | from django.urls import path | ||||||
|  |  | ||||||
|  | from reservation.views import ( | ||||||
|  |     ReservationFragment, | ||||||
|  |     ReservationScheduleView, | ||||||
|  |     RoomCreateView, | ||||||
|  |     RoomDeleteView, | ||||||
|  |     RoomUpdateView, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | urlpatterns = [ | ||||||
|  |     path("", ReservationScheduleView.as_view(), name="main"), | ||||||
|  |     path("room/create/", RoomCreateView.as_view(), name="room_create"), | ||||||
|  |     path("room/<int:room_id>/edit", RoomUpdateView.as_view(), name="room_edit"), | ||||||
|  |     path("room/<int:room_id>/delete", RoomDeleteView.as_view(), name="room_delete"), | ||||||
|  |     path( | ||||||
|  |         "fragment/reservation", ReservationFragment.as_view(), name="make_reservation" | ||||||
|  |     ), | ||||||
|  | ] | ||||||
							
								
								
									
										72
									
								
								reservation/views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								reservation/views.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | |||||||
|  | # Create your views here. | ||||||
|  |  | ||||||
|  | from django.contrib.auth.mixins import PermissionRequiredMixin | ||||||
|  | from django.contrib.messages.views import SuccessMessageMixin | ||||||
|  | from django.urls import reverse, reverse_lazy | ||||||
|  | from django.utils.translation import gettext_lazy as _ | ||||||
|  | from django.views.generic import CreateView, DeleteView, TemplateView, UpdateView | ||||||
|  |  | ||||||
|  | from club.models import Club | ||||||
|  | from core.auth.mixins import CanEditMixin | ||||||
|  | from core.views import UseFragmentsMixin | ||||||
|  | from core.views.mixins import FragmentMixin | ||||||
|  | from reservation.forms import ReservationForm, RoomCreateForm, RoomUpdateForm | ||||||
|  | from reservation.models import ReservationSlot, Room | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ReservationFragment(PermissionRequiredMixin, FragmentMixin, CreateView): | ||||||
|  |     model = ReservationSlot | ||||||
|  |     form_class = ReservationForm | ||||||
|  |     permission_required = "reservation.add_reservationslot" | ||||||
|  |     template_name = "reservation/fragments/create_reservation.jinja" | ||||||
|  |     success_url = reverse_lazy("reservation:main") | ||||||
|  |     reload_on_redirect = True | ||||||
|  |     object = None | ||||||
|  |  | ||||||
|  |     def get_form_kwargs(self): | ||||||
|  |         return super().get_form_kwargs() | {"author": self.request.user} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ReservationScheduleView(PermissionRequiredMixin, UseFragmentsMixin, TemplateView): | ||||||
|  |     template_name = "reservation/schedule.jinja" | ||||||
|  |     permission_required = "reservation.view_reservationslot" | ||||||
|  |     fragments = {"add_slot_fragment": ReservationFragment} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomCreateView(PermissionRequiredMixin, CreateView): | ||||||
|  |     form_class = RoomCreateForm | ||||||
|  |     template_name = "core/create.jinja" | ||||||
|  |     permission_required = "reservation.add_room" | ||||||
|  |  | ||||||
|  |     def get_initial(self): | ||||||
|  |         init = super().get_initial() | ||||||
|  |         if "club" in self.request.GET: | ||||||
|  |             club_id = self.request.GET["club"] | ||||||
|  |             if club_id.isdigit() and int(club_id) > 0: | ||||||
|  |                 init["club"] = Club.objects.filter(id=int(club_id)).first() | ||||||
|  |         return init | ||||||
|  |  | ||||||
|  |     def get_success_url(self): | ||||||
|  |         return reverse("club:tools", kwargs={"club_id": self.object.club_id}) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomUpdateView(SuccessMessageMixin, CanEditMixin, UpdateView): | ||||||
|  |     model = Room | ||||||
|  |     pk_url_kwarg = "room_id" | ||||||
|  |     form_class = RoomUpdateForm | ||||||
|  |     template_name = "core/edit.jinja" | ||||||
|  |     success_message = _("%(name)s was updated successfully") | ||||||
|  |  | ||||||
|  |     def get_form_kwargs(self): | ||||||
|  |         return super().get_form_kwargs() | {"request_user": self.request.user} | ||||||
|  |  | ||||||
|  |     def get_success_url(self): | ||||||
|  |         return self.request.path | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RoomDeleteView(PermissionRequiredMixin, DeleteView): | ||||||
|  |     model = Room | ||||||
|  |     pk_url_kwarg = "room_id" | ||||||
|  |     template_name = "core/delete_confirm.jinja" | ||||||
|  |     success_url = reverse_lazy("reservation:room_list") | ||||||
|  |     permission_required = "reservation.delete_room" | ||||||
| @@ -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 | ||||||
|   | |||||||
| @@ -123,6 +123,7 @@ INSTALLED_APPS = ( | |||||||
|     "trombi", |     "trombi", | ||||||
|     "matmat", |     "matmat", | ||||||
|     "pedagogy", |     "pedagogy", | ||||||
|  |     "reservation", | ||||||
|     "galaxy", |     "galaxy", | ||||||
|     "antispam", |     "antispam", | ||||||
|     "timetable", |     "timetable", | ||||||
| @@ -275,7 +276,7 @@ LOGGING = { | |||||||
| # Internationalization | # Internationalization | ||||||
| # https://docs.djangoproject.com/en/1.8/topics/i18n/ | # https://docs.djangoproject.com/en/1.8/topics/i18n/ | ||||||
|  |  | ||||||
| LANGUAGE_CODE = "fr-FR" | LANGUAGE_CODE = "fr" | ||||||
|  |  | ||||||
| LANGUAGES = [("en", _("English")), ("fr", _("French"))] | LANGUAGES = [("en", _("English")), ("fr", _("French"))] | ||||||
|  |  | ||||||
|   | |||||||
| @@ -49,6 +49,10 @@ urlpatterns = [ | |||||||
|     path("trombi/", include(("trombi.urls", "trombi"), namespace="trombi")), |     path("trombi/", include(("trombi.urls", "trombi"), namespace="trombi")), | ||||||
|     path("matmatronch/", include(("matmat.urls", "matmat"), namespace="matmat")), |     path("matmatronch/", include(("matmat.urls", "matmat"), namespace="matmat")), | ||||||
|     path("pedagogy/", include(("pedagogy.urls", "pedagogy"), namespace="pedagogy")), |     path("pedagogy/", include(("pedagogy.urls", "pedagogy"), namespace="pedagogy")), | ||||||
|  |     path( | ||||||
|  |         "reservation/", | ||||||
|  |         include(("reservation.urls", "reservation"), namespace="reservation"), | ||||||
|  |     ), | ||||||
|     path("admin/", admin.site.urls), |     path("admin/", admin.site.urls), | ||||||
|     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"), | ||||||
|   | |||||||
| @@ -18,7 +18,8 @@ | |||||||
|       "#core:*": ["./core/static/bundled/*"], |       "#core:*": ["./core/static/bundled/*"], | ||||||
|       "#pedagogy:*": ["./pedagogy/static/bundled/*"], |       "#pedagogy:*": ["./pedagogy/static/bundled/*"], | ||||||
|       "#counter:*": ["./counter/static/bundled/*"], |       "#counter:*": ["./counter/static/bundled/*"], | ||||||
|       "#com:*": ["./com/static/bundled/*"] |       "#com:*": ["./com/static/bundled/*"], | ||||||
|  |       "#reservation:*": ["./reservation/static/bundled/*"] | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user