mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-10-20 19:58:31 +00:00 
			
		
		
		
	Compare commits
	
		
			107 Commits
		
	
	
		
			galaxy
			...
			dependabot
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 67f8543ac7 | ||
|  | 459edc1b6e | ||
| a760a0b75d | |||
|  | fc615e90b2 | ||
| 76eebaf54e | |||
|  | 9407f4b341 | ||
|  | 8bd82c9d7c | ||
|  | 957441ceb1 | ||
|  | 3bcd417ad0 | ||
|  | 453e13d54b | ||
|  | dbd86b66cc | ||
|  | dcf799b352 | ||
|  | d815f7da97 | ||
|  | dac52db434 | ||
|  | f398c9901c | ||
|  | 5b91fe2145 | ||
|  | abd905c24d | ||
|  | 42b53a39f3 | ||
|  | 5306001f6f | ||
|  | 83a4ac2a7e | ||
|  | 30fd4f6926 | ||
|  | 1b1ef18531 | ||
|  | bcf5d30d8f | ||
|  | 4b44e50780 | ||
|  | 40c3276c3c | ||
|  | 543a424258 | ||
|  | 8ff25e6034 | ||
| fa8772ede2 | |||
|  | 03f53e921b | ||
|  | 56f09fd739 | ||
|  | 19e3fc604d | ||
|  | 24e1ad6dc8 | ||
|  | 2a30f30a31 | ||
|  | 80545e682b | ||
|  | a7adb4bba3 | ||
|  | e75e7e697a | ||
|  | 9d99976bee | ||
|  | 4103dce1bb | ||
|  | 126fcbaaa1 | ||
|  | 8a27214801 | ||
|  | e82f3649e5 | ||
|  | d3444f6bea | ||
|  | 289ffe1109 | ||
|  | eadf74604c | ||
|  | cc58479a19 | ||
|  | c03b6e5d9d | ||
|  | 66cf2bd957 | ||
|  | 3e8f3b9275 | ||
|  | c7363de44f | ||
|  | 966fe0ec0e | ||
|  | fd0af3a804 | ||
|  | 7db66bb8f6 | ||
|  | ff5bb04af1 | ||
| ca50e5dc81 | |||
|  | f015bde768 | ||
| bb09fd0feb | |||
| 210278440a | |||
| e041da9cf4 | |||
| 54c1957776 | |||
| 30356d97f3 | |||
| 7eaf25a64f | |||
| c6e86841b3 | |||
| cbe9887efb | |||
|  | 980952807a | ||
|  | 0b7c516f18 | ||
|  | e186052283 | ||
|  | ec80b72a25 | ||
|  | 6cd3875b2b | ||
| ad8b003336 | |||
|  | b4f5a866e3 | ||
| d87b069769 | |||
|  | 9461b2e5d9 | ||
| 4701c0804b | |||
|  | acb6c6ce9c | ||
| 95e6fff98b | |||
|  | f1a5a0781c | ||
|  | 854dd2d9e7 | ||
|  | a7c96425c8 | ||
| dff23fae7f | |||
|  | 34b0dc3302 | ||
|  | 31aee01360 | ||
|  | ce2ef78a6d | ||
|  | f7c5088048 | ||
|  | 9bc6a447b9 | ||
|  | 08b16d6e74 | ||
|  | c6baab068a | ||
|  | 262281adda | ||
|  | b58eca3ed0 | ||
|  | c7fe8961ab | ||
|  | 18f77ef2cb | ||
|  | b58da0ea30 | ||
|  | 25cd877160 | ||
|  | 79297b7a75 | ||
|  | b767079c5a | ||
|  | 37961e437b | ||
|  | b97a1a2e56 | ||
|  | 3ad40b7383 | ||
|  | 3709b5c221 | ||
|  | 171a3f4d92 | ||
|  | 84e2f1b45a | ||
|  | e0702ce8be | ||
|  | f6683068ff | ||
|  | 81d1d1caca | ||
|  | 1cc2378476 | ||
|  | 61e370cf73 | ||
|  | 6377acfffa | ||
|  | 3c8933461a | 
							
								
								
									
										2
									
								
								.github/auto_assign.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/auto_assign.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ addAssignees: author | ||||
|  | ||||
| # A list of team reviewers to be added to pull requests (GitHub team slug) | ||||
| reviewers: | ||||
|   - ae-utbm/sith-3-developers | ||||
|   - ae-utbm/developpeurs | ||||
|  | ||||
| # Number of reviewers has no impact on GitHub teams | ||||
| # Set 0 to add all the reviewers (default: 0) | ||||
|   | ||||
							
								
								
									
										9
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,16 @@ multi-ecosystem-groups: | ||||
|  | ||||
| updates: | ||||
|   - package-ecosystem: "uv" | ||||
|     patterns: ["*"] | ||||
|     multi-ecosystem-group: "common" | ||||
|  | ||||
|   - package-ecosystem: "npm" | ||||
|     patterns: ["*"] | ||||
|     multi-ecosystem-group: "common" | ||||
|     groups: | ||||
|       # npm supports production and development groups, but not uv | ||||
|       # cf. https://docs.github.com/en/code-security/dependabot/working-with-dependabot/dependabot-options-reference#dependency-type-groups | ||||
|       main-deps: | ||||
|         dependency-type: "production" | ||||
|       dev-deps: | ||||
|         dependency-type: "development" | ||||
|   | ||||
							
								
								
									
										186
									
								
								club/forms.py
									
									
									
									
									
								
							
							
						
						
									
										186
									
								
								club/forms.py
									
									
									
									
									
								
							| @@ -26,12 +26,16 @@ from django import forms | ||||
| from django.conf import settings | ||||
| from django.db.models import Exists, OuterRef, Q | ||||
| from django.db.models.functions import Lower | ||||
| from django.utils.functional import cached_property | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| from club.models import Club, Mailing, MailingSubscription, Membership | ||||
| from core.models import User | ||||
| from core.views.forms import SelectDate, SelectDateTime | ||||
| from core.views.widgets.ajax_select import AutoCompleteSelectMultipleUser | ||||
| from core.views.forms import SelectDateTime | ||||
| from core.views.widgets.ajax_select import ( | ||||
|     AutoCompleteSelectMultipleUser, | ||||
|     AutoCompleteSelectUser, | ||||
| ) | ||||
| from counter.models import Counter, Selling | ||||
|  | ||||
|  | ||||
| @@ -188,105 +192,113 @@ class SellingsForm(forms.Form): | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class ClubMemberForm(forms.Form): | ||||
|     """Form handling the members of a club.""" | ||||
| class ClubOldMemberForm(forms.Form): | ||||
|     members_old = forms.ModelMultipleChoiceField( | ||||
|         Membership.objects.none(), | ||||
|         label=_("Mark as old"), | ||||
|         widget=forms.CheckboxSelectMultiple, | ||||
|         required=False, | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, *args, user: User, club: Club, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["members_old"].queryset = ( | ||||
|             Membership.objects.ongoing().filter(club=club).editable_by(user) | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class ClubMemberForm(forms.ModelForm): | ||||
|     """Form to add a member to the club, as a board member.""" | ||||
|  | ||||
|     error_css_class = "error" | ||||
|     required_css_class = "required" | ||||
|  | ||||
|     users = forms.ModelMultipleChoiceField( | ||||
|         label=_("Users to add"), | ||||
|         help_text=_("Search users to add (one or more)."), | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectMultipleUser, | ||||
|         queryset=User.objects.all(), | ||||
|     ) | ||||
|     class Meta: | ||||
|         model = Membership | ||||
|         fields = ["role", "description"] | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self.club = kwargs.pop("club") | ||||
|         self.request_user = kwargs.pop("request_user") | ||||
|         self.club_members = kwargs.pop("club_members", None) | ||||
|         if not self.club_members: | ||||
|             self.club_members = self.club.members.ongoing().order_by("-role").all() | ||||
|     def __init__(self, *args, club: Club, request_user: User, **kwargs): | ||||
|         self.club = club | ||||
|         self.request_user = request_user | ||||
|         self.request_user_membership = self.club.get_membership_for(self.request_user) | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.fields["role"].required = True | ||||
|         self.fields["role"].choices = [ | ||||
|             (value, name) | ||||
|             for value, name in settings.SITH_CLUB_ROLES.items() | ||||
|             if value <= self.max_available_role | ||||
|         ] | ||||
|         self.instance.club = club | ||||
|  | ||||
|         # Using a ModelForm binds too much the form with the model and we don't want that | ||||
|         # We want the view to process the model creation since they are multiple users | ||||
|         # We also want the form to handle bulk deletion | ||||
|         self.fields.update( | ||||
|             forms.fields_for_model( | ||||
|                 Membership, | ||||
|                 fields=("role", "start_date", "description"), | ||||
|                 widgets={"start_date": SelectDate}, | ||||
|             ) | ||||
|         ) | ||||
|     @property | ||||
|     def max_available_role(self): | ||||
|         """The greatest role that will be obtainable with this form.""" | ||||
|         # this is unreachable, because it will be overridden by subclasses | ||||
|         return -1  # pragma: no cover | ||||
|  | ||||
|         # Role is required only if users is specified | ||||
|         self.fields["role"].required = False | ||||
|  | ||||
|         # Start date and description are never really required | ||||
|         self.fields["start_date"].required = False | ||||
|         self.fields["description"].required = False | ||||
| class ClubAddMemberForm(ClubMemberForm): | ||||
|     """Form to add a member to the club, as a board member.""" | ||||
|  | ||||
|         self.fields["users_old"] = forms.ModelMultipleChoiceField( | ||||
|             User.objects.filter( | ||||
|                 id__in=[ | ||||
|                     ms.user.id | ||||
|                     for ms in self.club_members | ||||
|                     if ms.can_be_edited_by(self.request_user) | ||||
|                 ] | ||||
|             ).all(), | ||||
|             label=_("Mark as old"), | ||||
|             required=False, | ||||
|             widget=forms.CheckboxSelectMultiple, | ||||
|         ) | ||||
|         if not self.request_user.is_root: | ||||
|             self.fields.pop("start_date") | ||||
|     class Meta(ClubMemberForm.Meta): | ||||
|         fields = ["user", *ClubMemberForm.Meta.fields] | ||||
|         widgets = {"user": AutoCompleteSelectUser} | ||||
|  | ||||
|     def clean_users(self): | ||||
|         """Check that the user is not trying to add an user already in the club. | ||||
|     @cached_property | ||||
|     def max_available_role(self): | ||||
|         """The greatest role that will be obtainable with this form. | ||||
|  | ||||
|         Admins and the club president can attribute any role. | ||||
|         Board members can attribute roles lower than their own. | ||||
|         Other users cannot attribute roles with this form | ||||
|         """ | ||||
|         if self.request_user.has_perm("club.add_membership"): | ||||
|             return settings.SITH_CLUB_ROLES_ID["President"] | ||||
|         membership = self.request_user_membership | ||||
|         if membership is None or membership.role <= settings.SITH_MAXIMUM_FREE_ROLE: | ||||
|             return -1 | ||||
|         if membership.role == settings.SITH_CLUB_ROLES_ID["President"]: | ||||
|             return membership.role | ||||
|         return membership.role - 1 | ||||
|  | ||||
|     def clean_user(self): | ||||
|         """Check that the user is not trying to add a user already in the club. | ||||
|  | ||||
|         Also check that the user is valid and has a valid subscription. | ||||
|         """ | ||||
|         cleaned_data = super().clean() | ||||
|         users = [] | ||||
|         for user in cleaned_data["users"]: | ||||
|             if not user.is_subscribed: | ||||
|                 raise forms.ValidationError( | ||||
|                     _("User must be subscriber to take part to a club"), code="invalid" | ||||
|                 ) | ||||
|             if self.club.get_membership_for(user): | ||||
|                 raise forms.ValidationError( | ||||
|                     _("You can not add the same user twice"), code="invalid" | ||||
|                 ) | ||||
|             users.append(user) | ||||
|         return users | ||||
|         user = self.cleaned_data["user"] | ||||
|         if not user.is_subscribed: | ||||
|             raise forms.ValidationError( | ||||
|                 _("User must be subscriber to take part to a club"), code="invalid" | ||||
|             ) | ||||
|         if self.club.get_membership_for(user): | ||||
|             raise forms.ValidationError( | ||||
|                 _("You can not add the same user twice"), code="invalid" | ||||
|             ) | ||||
|         return user | ||||
|  | ||||
|  | ||||
| class JoinClubForm(ClubMemberForm): | ||||
|     """Form to join a club.""" | ||||
|  | ||||
|     def __init__(self, *args, club: Club, request_user: User, **kwargs): | ||||
|         super().__init__(*args, club=club, request_user=request_user, **kwargs) | ||||
|         # this form doesn't manage the user who will join the club, | ||||
|         # so we must set this here to avoid errors | ||||
|         self.instance.user = self.request_user | ||||
|  | ||||
|     @cached_property | ||||
|     def max_available_role(self): | ||||
|         return settings.SITH_MAXIMUM_FREE_ROLE | ||||
|  | ||||
|     def clean(self): | ||||
|         """Check user rights for adding an user.""" | ||||
|         cleaned_data = super().clean() | ||||
|  | ||||
|         if "start_date" in cleaned_data and not cleaned_data["start_date"]: | ||||
|             # Drop start_date if allowed to edition but not specified | ||||
|             cleaned_data.pop("start_date") | ||||
|  | ||||
|         if not cleaned_data.get("users"): | ||||
|             # No user to add equals no check needed | ||||
|             return cleaned_data | ||||
|  | ||||
|         if cleaned_data.get("role", "") == "": | ||||
|             # Role is required if users exists | ||||
|             self.add_error("role", _("You should specify a role")) | ||||
|             return cleaned_data | ||||
|  | ||||
|         request_user = self.request_user | ||||
|         membership = self.request_user_membership | ||||
|         if not ( | ||||
|             cleaned_data["role"] <= settings.SITH_MAXIMUM_FREE_ROLE | ||||
|             or (membership is not None and membership.role >= cleaned_data["role"]) | ||||
|             or request_user.is_board_member | ||||
|             or request_user.is_root | ||||
|         ): | ||||
|             raise forms.ValidationError(_("You do not have the permission to do that")) | ||||
|         return cleaned_data | ||||
|         """Check that the user is subscribed and isn't already in the club.""" | ||||
|         if not self.request_user.is_subscribed: | ||||
|             raise forms.ValidationError( | ||||
|                 _("You must be subscribed to join a club"), code="invalid" | ||||
|             ) | ||||
|         if self.club.get_membership_for(self.request_user): | ||||
|             raise forms.ValidationError( | ||||
|                 _("You are already a member of this club"), code="invalid" | ||||
|             ) | ||||
|         return super().clean() | ||||
|   | ||||
| @@ -34,12 +34,10 @@ def migrate_meta_groups(apps: StateApps, schema_editor): | ||||
|     clubs = list(Club.objects.all()) | ||||
|     for club in clubs: | ||||
|         club.board_group = meta_groups.get_or_create( | ||||
|             name=club.unix_name + settings.SITH_BOARD_SUFFIX, | ||||
|             defaults={"is_meta": True}, | ||||
|             name=f"{club.unix_name}-bureau", defaults={"is_meta": True} | ||||
|         )[0] | ||||
|         club.members_group = meta_groups.get_or_create( | ||||
|             name=club.unix_name + settings.SITH_MEMBER_SUFFIX, | ||||
|             defaults={"is_meta": True}, | ||||
|             name=f"{club.unix_name}-membres", defaults={"is_meta": True} | ||||
|         )[0] | ||||
|         club.save() | ||||
|         club.refresh_from_db() | ||||
|   | ||||
| @@ -30,7 +30,8 @@ from django.core.cache import cache | ||||
| from django.core.exceptions import ObjectDoesNotExist, ValidationError | ||||
| from django.core.validators import RegexValidator, validate_email | ||||
| from django.db import models, transaction | ||||
| from django.db.models import Exists, F, OuterRef, Q | ||||
| from django.db.models import Exists, F, OuterRef, Q, Value | ||||
| from django.db.models.functions import Greatest | ||||
| from django.urls import reverse | ||||
| from django.utils import timezone | ||||
| from django.utils.functional import cached_property | ||||
| @@ -42,6 +43,13 @@ from core.fields import ResizedImageField | ||||
| from core.models import Group, Notification, Page, SithFile, User | ||||
|  | ||||
|  | ||||
| class ClubQuerySet(models.QuerySet): | ||||
|     def having_board_member(self, user: User) -> Self: | ||||
|         """Filter all club in which the given user is a board member.""" | ||||
|         active_memberships = user.memberships.board().ongoing() | ||||
|         return self.filter(Exists(active_memberships.filter(club=OuterRef("pk")))) | ||||
|  | ||||
|  | ||||
| class Club(models.Model): | ||||
|     """The Club class, made as a tree to allow nice tidy organization.""" | ||||
|  | ||||
| @@ -91,6 +99,8 @@ class Club(models.Model): | ||||
|         Group, related_name="club_board", on_delete=models.PROTECT | ||||
|     ) | ||||
|  | ||||
|     objects = ClubQuerySet.as_manager() | ||||
|  | ||||
|     class Meta: | ||||
|         ordering = ["name"] | ||||
|  | ||||
| @@ -200,10 +210,6 @@ class Club(models.Model): | ||||
|         """Method to see if that object can be edited by the given user.""" | ||||
|         return self.has_rights_in_club(user) | ||||
|  | ||||
|     def can_be_viewed_by(self, user: User) -> bool: | ||||
|         """Method to see if that object can be seen by the given user.""" | ||||
|         return user.was_subscribed | ||||
|  | ||||
|     def get_membership_for(self, user: User) -> Membership | None: | ||||
|         """Return the current membership the given user. | ||||
|  | ||||
| @@ -243,6 +249,44 @@ class MembershipQuerySet(models.QuerySet): | ||||
|         """ | ||||
|         return self.filter(role__gt=settings.SITH_MAXIMUM_FREE_ROLE) | ||||
|  | ||||
|     def editable_by(self, user: User) -> Self: | ||||
|         """Filter Memberships that this user can edit. | ||||
|  | ||||
|         Users with the `club.change_membership` permission can edit all Membership. | ||||
|         The other users can edit : | ||||
|         - their own membership | ||||
|         - if they are board members, ongoing memberships with a role lower than their own | ||||
|  | ||||
|         For example, let's suppose the following users : | ||||
|         - A : board member | ||||
|         - B : board member | ||||
|         - C : simple member | ||||
|         - D : curious | ||||
|         - E : old member | ||||
|  | ||||
|         A will be able to edit the memberships of A, C and D ; | ||||
|         C and D will be able to edit only their own membership ; | ||||
|         nobody will be able to edit E's membership. | ||||
|         """ | ||||
|         if user.has_perm("club.change_membership"): | ||||
|             return self.all() | ||||
|         return self.filter( | ||||
|             Q(user=user) | ||||
|             | Exists( | ||||
|                 Membership.objects.filter( | ||||
|                     Q( | ||||
|                         role__gt=Greatest( | ||||
|                             OuterRef("role"), Value(settings.SITH_MAXIMUM_FREE_ROLE) | ||||
|                         ) | ||||
|                     ), | ||||
|                     user=user, | ||||
|                     end_date=None, | ||||
|                     club=OuterRef("club"), | ||||
|                 ) | ||||
|             ), | ||||
|             end_date=None, | ||||
|         ) | ||||
|  | ||||
|     def update(self, **kwargs) -> int: | ||||
|         """Refresh the cache and edit group ownership. | ||||
|  | ||||
| @@ -319,16 +363,12 @@ class Membership(models.Model): | ||||
|         User, | ||||
|         verbose_name=_("user"), | ||||
|         related_name="memberships", | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     club = models.ForeignKey( | ||||
|         Club, | ||||
|         verbose_name=_("club"), | ||||
|         related_name="members", | ||||
|         null=False, | ||||
|         blank=False, | ||||
|         on_delete=models.CASCADE, | ||||
|     ) | ||||
|     start_date = models.DateField(_("start date"), default=timezone.now) | ||||
|   | ||||
							
								
								
									
										24
									
								
								club/static/club/members.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								club/static/club/members.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| #club_members_table { | ||||
|   tbody label { | ||||
|     margin: 0; | ||||
|     padding: 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
| #add_club_members_form { | ||||
|   fieldset { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     column-gap: 2em; | ||||
|     row-gap: 1em; | ||||
|     flex-wrap: wrap; | ||||
|  | ||||
|     @media (max-width: 1100px) { | ||||
|       justify-content: space-evenly; | ||||
|     } | ||||
|  | ||||
|     .errorlist { | ||||
|       max-width: 300px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,15 +1,33 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
| {% from 'core/macros.jinja' import user_profile_link, select_all_checkbox %} | ||||
|  | ||||
| {% block additional_js %} | ||||
|   <script type="module" src="{{ static("bundled/core/components/ajax-select-index.ts") }}"></script> | ||||
| {% endblock %} | ||||
| {% block additional_css %} | ||||
|   <link rel="stylesheet" href="{{ static("bundled/core/components/ajax-select-index.css") }}"> | ||||
|   <link rel="stylesheet" href="{{ static("club/members.scss") }}"> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|   {% block notifications %} | ||||
|     {# Notifications are moved a little bit below #} | ||||
|   {% endblock %} | ||||
|  | ||||
|   <h2>{% trans %}Club members{% endtrans %}</h2> | ||||
|  | ||||
|   {% if add_member_fragment %} | ||||
|     <br /> | ||||
|     {{ add_member_fragment }} | ||||
|     <br /> | ||||
|   {% endif %} | ||||
|   {% include "core/base/notifications.jinja" %} | ||||
|   {% if members %} | ||||
|     <form action="{{ url('club:club_members', club_id=club.id) }}" id="users_old" method="post"> | ||||
|     <form action="{{ url('club:club_members', club_id=club.id) }}" id="members_old" method="post"> | ||||
|       {% csrf_token %} | ||||
|       {% set users_old = dict(form.users_old | groupby("choice_label")) %} | ||||
|       {% if users_old %} | ||||
|         {{ select_all_checkbox("users_old") }} | ||||
|         <p></p> | ||||
|       {% if can_end_membership %} | ||||
|         {{ select_all_checkbox("members_old") }} | ||||
|         <br /> | ||||
|       {% endif %} | ||||
|       <table id="club_members_table"> | ||||
|         <thead> | ||||
| @@ -18,7 +36,7 @@ | ||||
|             <td>{% trans %}Role{% endtrans %}</td> | ||||
|             <td>{% trans %}Description{% endtrans %}</td> | ||||
|             <td>{% trans %}Since{% endtrans %}</td> | ||||
|             {% if users_old %} | ||||
|             {% if can_end_membership %} | ||||
|               <td>{% trans %}Mark as old{% endtrans %}</td> | ||||
|             {% endif %} | ||||
|           </tr> | ||||
| @@ -30,20 +48,24 @@ | ||||
|               <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||
|               <td>{{ m.description }}</td> | ||||
|               <td>{{ m.start_date }}</td> | ||||
|               {% if users_old %} | ||||
|               {%- if can_end_membership -%} | ||||
|                 <td> | ||||
|                   {% set user_old = users_old[m.user.get_display_name()] %} | ||||
|                   {% if user_old %} | ||||
|                     {{ user_old[0].tag() }} | ||||
|                   {% endif %} | ||||
|                   {%- if m.is_editable -%} | ||||
|                     <label for="id_members_old_{{ loop.index }}"></label> | ||||
|                     <input | ||||
|                       type="checkbox" | ||||
|                       name="members_old" | ||||
|                       value="{{ m.id }}" | ||||
|                       id="id_members_old_{{ loop.index }}" | ||||
|                     > | ||||
|                   {%- endif -%} | ||||
|                 </td> | ||||
|               {% endif %} | ||||
|               {%- endif -%} | ||||
|             </tr> | ||||
|           {% endfor %} | ||||
|         </tbody> | ||||
|       </table> | ||||
|       {{ form.users_old.errors }} | ||||
|       {% if users_old %} | ||||
|       {% if can_end_membership %} | ||||
|         <p></p> | ||||
|         <input type="submit" name="submit" value="{% trans %}Mark as old{% endtrans %}"> | ||||
|       {% endif %} | ||||
| @@ -51,32 +73,4 @@ | ||||
|   {% else %} | ||||
|     <p>{% trans %}There are no members in this club.{% endtrans %}</p> | ||||
|   {% endif %} | ||||
|   <form action="{{ url('club:club_members', club_id=club.id) }}" id="add_users" method="post"> | ||||
|     {% csrf_token %} | ||||
|     {{ form.non_field_errors() }} | ||||
|     <p> | ||||
|       {{ form.users.errors }} | ||||
|       <label for="{{ form.users.id_for_label }}">{{ form.users.label }} :</label> | ||||
|       {{ form.users }} | ||||
|       <span class="helptext">{{ form.users.help_text }}</span> | ||||
|     </p> | ||||
|     <p> | ||||
|       {{ form.role.errors }} | ||||
|       <label for="{{ form.role.id_for_label }}">{{ form.role.label }} :</label> | ||||
|       {{ form.role }} | ||||
|     </p> | ||||
|     {% if form.start_date %} | ||||
|       <p> | ||||
|         {{ form.start_date.errors }} | ||||
|         <label for="{{ form.start_date.id_for_label }}">{{ form.start_date.label }} :</label> | ||||
|         {{ form.start_date }} | ||||
|       </p> | ||||
|     {% endif %} | ||||
|     <p> | ||||
|       {{ form.description.errors }} | ||||
|       <label for="{{ form.description.id_for_label }}">{{ form.description.label }} :</label> | ||||
|       {{ form.description }} | ||||
|     </p> | ||||
|     <p><input type="submit" value="{% trans %}Add{% endtrans %}" /></p> | ||||
|   </form> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -5,20 +5,22 @@ | ||||
|   <h2>{% trans %}Club old members{% endtrans %}</h2> | ||||
|   <table> | ||||
|     <thead> | ||||
|       <td>{% trans %}User{% endtrans %}</td> | ||||
|       <td>{% trans %}Role{% endtrans %}</td> | ||||
|       <td>{% trans %}Description{% endtrans %}</td> | ||||
|       <td>{% trans %}From{% endtrans %}</td> | ||||
|       <td>{% trans %}To{% endtrans %}</td> | ||||
|       <tr> | ||||
|         <td>{% trans %}User{% endtrans %}</td> | ||||
|         <td>{% trans %}Role{% endtrans %}</td> | ||||
|         <td>{% trans %}Description{% endtrans %}</td> | ||||
|         <td>{% trans %}From{% endtrans %}</td> | ||||
|         <td>{% trans %}To{% endtrans %}</td> | ||||
|       </tr> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|       {% for m in club.members.exclude(end_date=None).order_by('-role', 'description', '-end_date').all() %} | ||||
|       {% for member in old_members %} | ||||
|         <tr> | ||||
|           <td>{{ user_profile_link(m.user) }}</td> | ||||
|           <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||
|           <td>{{ m.description }}</td> | ||||
|           <td>{{ m.start_date }}</td> | ||||
|           <td>{{ m.end_date }}</td> | ||||
|           <td>{{ user_profile_link(member.user) }}</td> | ||||
|           <td>{{ settings.SITH_CLUB_ROLES[member.role] }}</td> | ||||
|           <td>{{ member.description }}</td> | ||||
|           <td>{{ member.start_date }}</td> | ||||
|           <td>{{ member.end_date }}</td> | ||||
|         </tr> | ||||
|       {% endfor %} | ||||
|     </tbody> | ||||
|   | ||||
| @@ -83,9 +83,10 @@ TODO : rewrite the pagination used in this template an Alpine one | ||||
|   </table> | ||||
|   <script type="text/javascript"> | ||||
|     function formPagination(link){ | ||||
|       $("form").attr("action", link.href); | ||||
|       const form = document.getElementById("form") | ||||
|       form.action = link.href; | ||||
|       link.href = "javascript:void(0)"; // block link action | ||||
|       $("form").submit(); | ||||
|       form.submit(); | ||||
|     } | ||||
|   </script> | ||||
|   {{ paginate(paginated_result, paginator, "formPagination(this)") }} | ||||
|   | ||||
							
								
								
									
										46
									
								
								club/templates/club/fragments/add_member.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								club/templates/club/fragments/add_member.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| <section id="member-fragment-container"> | ||||
|   {% if form.user %} | ||||
|     <h4>{% trans %}Add a new member{% endtrans %}</h4> | ||||
|   {% else %} | ||||
|     <h4>{% trans %}Join club{% endtrans %}</h4> | ||||
|   {% endif %} | ||||
|  | ||||
|   <form | ||||
|     hx-post="{{ url('club:club_new_members', club_id=club.id) }}" | ||||
|     hx-disabled-elt="find input[type='submit']" | ||||
|     hx-swap="outerHTML" | ||||
|     hx-target="#member-fragment-container" | ||||
|     id="add_club_members_form" | ||||
|   > | ||||
|     {% csrf_token %} | ||||
|     {{ form.non_field_errors() }} | ||||
|     <fieldset> | ||||
|       {% if form.user %} | ||||
|         <div> | ||||
|           {{ form.user.label_tag() }} | ||||
|           <span class="helptext">{{ form.user.help_text }}</span> | ||||
|           {{ form.user }} | ||||
|           {{ form.user.errors }} | ||||
|         </div> | ||||
|       {% endif %} | ||||
|       <div> | ||||
|         {{ form.role.label_tag() }} | ||||
|         {{ form.role }} | ||||
|         {{ form.role.errors }} | ||||
|       </div> | ||||
|       <div> | ||||
|         {{ form.description.label_tag() }} | ||||
|         {{ form.description }} | ||||
|         {{ form.description.errors }} | ||||
|       </div> | ||||
|     </fieldset> | ||||
|     <button type="submit" class="btn btn-blue"> | ||||
|       <i class="fa fa-user-plus"></i> | ||||
|       {%- if form.user -%} | ||||
|         {% trans %}Add{% endtrans %} | ||||
|       {%- else -%} | ||||
|         {% trans %}Join{% endtrans %} | ||||
|       {%- endif -%} | ||||
|     </button> | ||||
|   </form> | ||||
| </section> | ||||
| @@ -43,6 +43,9 @@ class TestClub(TestCase): | ||||
|  | ||||
|         cls.ae = Club.objects.get(pk=settings.SITH_MAIN_CLUB_ID) | ||||
|         cls.club = baker.make(Club) | ||||
|         cls.new_members_url = reverse( | ||||
|             "club:club_new_members", kwargs={"club_id": cls.club.id} | ||||
|         ) | ||||
|         cls.members_url = reverse("club:club_members", kwargs={"club_id": cls.club.id}) | ||||
|         a_month_ago = now() - timedelta(days=30) | ||||
|         yesterday = now() - timedelta(days=1) | ||||
|   | ||||
							
								
								
									
										27
									
								
								club/tests/test_club.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								club/tests/test_club.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| from datetime import timedelta | ||||
|  | ||||
| import pytest | ||||
| from django.utils.timezone import localdate | ||||
| from model_bakery import baker | ||||
| from model_bakery.recipe import Recipe | ||||
|  | ||||
| from club.models import Club, Membership | ||||
| from core.baker_recipes import subscriber_user | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_club_queryset_having_board_member(): | ||||
|     clubs = baker.make(Club, _quantity=5) | ||||
|     user = subscriber_user.make() | ||||
|     membership_recipe = Recipe( | ||||
|         Membership, user=user, start_date=localdate() - timedelta(days=3) | ||||
|     ) | ||||
|     membership_recipe.make(club=clubs[0], role=1) | ||||
|     membership_recipe.make(club=clubs[1], role=3) | ||||
|     membership_recipe.make(club=clubs[2], role=7) | ||||
|     membership_recipe.make( | ||||
|         club=clubs[3], role=3, end_date=localdate() - timedelta(days=1) | ||||
|     ) | ||||
|  | ||||
|     club_ids = Club.objects.having_board_member(user).values_list("id", flat=True) | ||||
|     assert set(club_ids) == {clubs[1].id, clubs[2].id} | ||||
| @@ -1,13 +1,20 @@ | ||||
| from collections.abc import Callable | ||||
| from datetime import timedelta | ||||
|  | ||||
| import pytest | ||||
| from bs4 import BeautifulSoup | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.models import Permission | ||||
| from django.core.cache import cache | ||||
| from django.db.models import Max | ||||
| from django.test import TestCase | ||||
| from django.urls import reverse | ||||
| from django.utils.timezone import localdate, localtime, now | ||||
| from model_bakery import baker | ||||
| from pytest_django.asserts import assertRedirects | ||||
|  | ||||
| from club.forms import ClubMemberForm | ||||
| from club.models import Membership | ||||
| from club.forms import ClubAddMemberForm, JoinClubForm | ||||
| from club.models import Club, Membership | ||||
| from club.tests.base import TestClub | ||||
| from core.baker_recipes import subscriber_user | ||||
| from core.models import AnonymousUser, User | ||||
| @@ -137,6 +144,38 @@ class TestMembershipQuerySet(TestClub): | ||||
|         assert set(user.groups.all()).isdisjoint(club_groups) | ||||
|  | ||||
|  | ||||
| class TestMembershipEditableBy(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|         Membership.objects.all().delete() | ||||
|         cls.club_a, cls.club_b = baker.make(Club, _quantity=2) | ||||
|         cls.memberships = [ | ||||
|             *baker.make( | ||||
|                 Membership, role=iter([7, 3, 3, 1]), club=cls.club_a, _quantity=4 | ||||
|             ), | ||||
|             *baker.make( | ||||
|                 Membership, role=iter([7, 3, 3, 1]), club=cls.club_b, _quantity=4 | ||||
|             ), | ||||
|         ] | ||||
|  | ||||
|     def test_admin_user(self): | ||||
|         perm = Permission.objects.get(codename="change_membership") | ||||
|         user = baker.make(User, user_permissions=[perm]) | ||||
|         qs = Membership.objects.editable_by(user).values_list("id", flat=True) | ||||
|         assert set(qs) == set(Membership.objects.values_list("id", flat=True)) | ||||
|  | ||||
|     def test_simple_subscriber_user(self): | ||||
|         user = subscriber_user.make() | ||||
|         assert not Membership.objects.editable_by(user).exists() | ||||
|  | ||||
|     def test_board_member(self): | ||||
|         # a board member can end lower memberships and its own one | ||||
|         user = self.memberships[2].user | ||||
|         qs = Membership.objects.editable_by(user).values_list("id", flat=True) | ||||
|         expected = {self.memberships[2].id, self.memberships[3].id} | ||||
|         assert set(qs) == expected | ||||
|  | ||||
|  | ||||
| class TestMembership(TestClub): | ||||
|     def assert_membership_started_today(self, user: User, role: int): | ||||
|         """Assert that the given membership is active and started today.""" | ||||
| @@ -151,7 +190,7 @@ class TestMembership(TestClub): | ||||
|  | ||||
|     def assert_membership_ended_today(self, user: User): | ||||
|         """Assert that the given user have a membership which ended today.""" | ||||
|         today = localtime(now()).date() | ||||
|         today = localdate() | ||||
|         assert user.memberships.filter(club=self.club, end_date=today).exists() | ||||
|         assert self.club.get_membership_for(user) is None | ||||
|  | ||||
| @@ -160,7 +199,9 @@ class TestMembership(TestClub): | ||||
|         cannot see the page. | ||||
|         """ | ||||
|         response = self.client.post(self.members_url) | ||||
|         assert response.status_code == 403 | ||||
|         assertRedirects( | ||||
|             response, reverse("core:login", query={"next": self.members_url}) | ||||
|         ) | ||||
|  | ||||
|         self.client.force_login(self.public) | ||||
|         response = self.client.post(self.members_url) | ||||
| @@ -171,7 +212,9 @@ class TestMembership(TestClub): | ||||
|         information are displayed. | ||||
|         """ | ||||
|         self.client.force_login(self.simple_board_member) | ||||
|         response = self.client.get(self.members_url) | ||||
|         response = self.client.get( | ||||
|             reverse("club:club_members", kwargs={"club_id": self.club.id}) | ||||
|         ) | ||||
|         assert response.status_code == 200 | ||||
|         soup = BeautifulSoup(response.text, "lxml") | ||||
|         table = soup.find("table", id="club_members_table") | ||||
| @@ -197,59 +240,45 @@ class TestMembership(TestClub): | ||||
|             assert cols[2].text == membership.description | ||||
|             assert cols[3].text == str(membership.start_date) | ||||
|  | ||||
|             if membership.role <= 3:  # 3 is the role of simple_board_member | ||||
|             if membership.role < 3 or membership.user_id == self.simple_board_member.id: | ||||
|                 # 3 is the role of simple_board_member | ||||
|                 form_input = cols[4].find("input") | ||||
|                 expected_attrs = { | ||||
|                     "type": "checkbox", | ||||
|                     "name": "users_old", | ||||
|                     "value": str(user.id), | ||||
|                     "name": "members_old", | ||||
|                     "value": str(membership.id), | ||||
|                 } | ||||
|                 assert form_input.attrs.items() >= expected_attrs.items() | ||||
|             else: | ||||
|                 assert cols[4].find_all() == [] | ||||
|  | ||||
|     def test_root_add_one_club_member(self): | ||||
|         """Test that root users can add members to clubs, one at a time.""" | ||||
|         """Test that root users can add members to clubs""" | ||||
|         self.client.force_login(self.root) | ||||
|         response = self.client.post( | ||||
|             self.members_url, | ||||
|             {"users": [self.subscriber.id], "role": 3}, | ||||
|             self.new_members_url, {"user": self.subscriber.id, "role": 3} | ||||
|         ) | ||||
|         assert response.status_code == 200 | ||||
|         assert response.headers.get("HX-Redirect", "") == reverse( | ||||
|             "club:club_members", kwargs={"club_id": self.club.id} | ||||
|         ) | ||||
|         self.assertRedirects(response, self.members_url) | ||||
|         self.subscriber.refresh_from_db() | ||||
|         self.assert_membership_started_today(self.subscriber, role=3) | ||||
|  | ||||
|     def test_root_add_multiple_club_member(self): | ||||
|         """Test that root users can add multiple members at once to clubs.""" | ||||
|         self.client.force_login(self.root) | ||||
|         response = self.client.post( | ||||
|             self.members_url, | ||||
|             { | ||||
|                 "users": (self.subscriber.id, self.krophil.id), | ||||
|                 "role": 3, | ||||
|             }, | ||||
|         ) | ||||
|         self.assertRedirects(response, self.members_url) | ||||
|         self.subscriber.refresh_from_db() | ||||
|         self.assert_membership_started_today(self.subscriber, role=3) | ||||
|         self.assert_membership_started_today(self.krophil, role=3) | ||||
|  | ||||
|     def test_add_unauthorized_members(self): | ||||
|         """Test that users who are not currently subscribed | ||||
|         cannot be members of clubs. | ||||
|         """ | ||||
|         for user in self.public, self.old_subscriber: | ||||
|             form = ClubMemberForm( | ||||
|                 data={"users": [user.id], "role": 1}, | ||||
|             form = ClubAddMemberForm( | ||||
|                 data={"user": user.id, "role": 1}, | ||||
|                 request_user=self.root, | ||||
|                 club=self.club, | ||||
|             ) | ||||
|  | ||||
|             assert not form.is_valid() | ||||
|             assert form.errors == { | ||||
|                 "users": [ | ||||
|                     "L'utilisateur doit être cotisant pour faire partie d'un club" | ||||
|                 ] | ||||
|                 "user": ["L'utilisateur doit être cotisant pour faire partie d'un club"] | ||||
|             } | ||||
|  | ||||
|     def test_add_members_already_members(self): | ||||
| @@ -281,16 +310,16 @@ class TestMembership(TestClub): | ||||
|         nb_memberships = self.club.members.count() | ||||
|         max_id = User.objects.aggregate(id=Max("id"))["id"] | ||||
|         for members in [max_id + 1], [max_id + 1, self.subscriber.id]: | ||||
|             form = ClubMemberForm( | ||||
|                 data={"users": members, "role": 1}, | ||||
|             form = ClubAddMemberForm( | ||||
|                 data={"user": members, "role": 1}, | ||||
|                 request_user=self.root, | ||||
|                 club=self.club, | ||||
|             ) | ||||
|             assert not form.is_valid() | ||||
|             assert form.errors == { | ||||
|                 "users": [ | ||||
|                 "user": [ | ||||
|                     "Sélectionnez un choix valide. " | ||||
|                     f"{max_id + 1} n\u2019en fait pas partie." | ||||
|                     "Ce choix ne fait pas partie de ceux disponibles." | ||||
|                 ] | ||||
|             } | ||||
|         self.club.refresh_from_db() | ||||
| @@ -303,10 +332,12 @@ class TestMembership(TestClub): | ||||
|         nb_subscriber_memberships = self.subscriber.memberships.count() | ||||
|         self.client.force_login(president) | ||||
|         response = self.client.post( | ||||
|             self.members_url, | ||||
|             {"users": self.subscriber.id, "role": 9}, | ||||
|             self.new_members_url, {"user": self.subscriber.id, "role": 9} | ||||
|         ) | ||||
|         assert response.status_code == 200 | ||||
|         assert response.headers.get("HX-Redirect", "") == reverse( | ||||
|             "club:club_members", kwargs={"club_id": self.club.id} | ||||
|         ) | ||||
|         self.assertRedirects(response, self.members_url) | ||||
|         self.club.refresh_from_db() | ||||
|         self.subscriber.refresh_from_db() | ||||
|         assert self.club.members.count() == nb_club_membership + 1 | ||||
| @@ -317,8 +348,8 @@ class TestMembership(TestClub): | ||||
|         """Test that a member of the club member cannot create | ||||
|         a membership with a greater role than its own. | ||||
|         """ | ||||
|         form = ClubMemberForm( | ||||
|             data={"users": [self.subscriber.id], "role": 10}, | ||||
|         form = ClubAddMemberForm( | ||||
|             data={"user": self.subscriber.id, "role": 10}, | ||||
|             request_user=self.simple_board_member, | ||||
|             club=self.club, | ||||
|         ) | ||||
| @@ -326,7 +357,7 @@ class TestMembership(TestClub): | ||||
|  | ||||
|         assert not form.is_valid() | ||||
|         assert form.errors == { | ||||
|             "__all__": ["Vous n'avez pas la permission de faire cela"] | ||||
|             "role": ["Sélectionnez un choix valide. 10 n\u2019en fait pas partie."] | ||||
|         } | ||||
|         self.club.refresh_from_db() | ||||
|         assert nb_memberships == self.club.members.count() | ||||
| @@ -334,23 +365,53 @@ class TestMembership(TestClub): | ||||
|  | ||||
|     def test_add_member_without_role(self): | ||||
|         """Test that trying to add members without specifying their role fails.""" | ||||
|         self.client.force_login(self.root) | ||||
|         form = ClubMemberForm( | ||||
|             data={"users": [self.subscriber.id]}, | ||||
|             request_user=self.simple_board_member, | ||||
|             club=self.club, | ||||
|         form = ClubAddMemberForm( | ||||
|             data={"user": self.subscriber.id}, request_user=self.root, club=self.club | ||||
|         ) | ||||
|  | ||||
|         assert not form.is_valid() | ||||
|         assert form.errors == {"role": ["Vous devez choisir un rôle"]} | ||||
|         assert form.errors == {"role": ["Ce champ est obligatoire."]} | ||||
|  | ||||
|     def test_add_member_already_there(self): | ||||
|         form = ClubAddMemberForm( | ||||
|             data={"user": self.simple_board_member, "role": 3}, | ||||
|             request_user=self.root, | ||||
|             club=self.club, | ||||
|         ) | ||||
|         assert not form.is_valid() | ||||
|         assert form.errors == { | ||||
|             "user": ["Vous ne pouvez pas ajouter deux fois le même utilisateur"] | ||||
|         } | ||||
|  | ||||
|     def test_add_other_member_forbidden(self): | ||||
|         non_member = subscriber_user.make() | ||||
|         simple_member = baker.make(Membership, club=self.club, role=1).user | ||||
|         for user in non_member, simple_member: | ||||
|             form = ClubAddMemberForm( | ||||
|                 data={"user": subscriber_user.make(), "role": 1}, | ||||
|                 request_user=user, | ||||
|                 club=self.club, | ||||
|             ) | ||||
|             assert not form.is_valid() | ||||
|             assert form.errors == { | ||||
|                 "role": ["Sélectionnez un choix valide. 1 n\u2019en fait pas partie."] | ||||
|             } | ||||
|  | ||||
|     def test_simple_members_dont_see_form_anymore(self): | ||||
|         """Test that simple club members don't see the form to add members""" | ||||
|         user = subscriber_user.make() | ||||
|         baker.make(Membership, club=self.club, user=user, role=1) | ||||
|         self.client.force_login(user) | ||||
|         res = self.client.get(self.members_url) | ||||
|         assert res.status_code == 200 | ||||
|         soup = BeautifulSoup(res.text, "lxml") | ||||
|         assert not soup.find(id="add_club_members_form") | ||||
|  | ||||
|     def test_end_membership_self(self): | ||||
|         """Test that a member can end its own membership.""" | ||||
|         self.client.force_login(self.simple_board_member) | ||||
|         self.client.post( | ||||
|             self.members_url, | ||||
|             {"users_old": self.simple_board_member.id}, | ||||
|         ) | ||||
|         membership = self.club.members.get(end_date=None, user=self.simple_board_member) | ||||
|         self.client.post(self.members_url, {"members_old": [membership.id]}) | ||||
|         self.simple_board_member.refresh_from_db() | ||||
|         self.assert_membership_ended_today(self.simple_board_member) | ||||
|  | ||||
| @@ -358,15 +419,13 @@ class TestMembership(TestClub): | ||||
|         """Test that board members of the club can end memberships | ||||
|         of users with lower roles. | ||||
|         """ | ||||
|         # remainder : simple_board_member has role 3, president has role 10, richard has role 1 | ||||
|         # reminder : simple_board_member has role 3 | ||||
|         self.client.force_login(self.simple_board_member) | ||||
|         response = self.client.post( | ||||
|             self.members_url, | ||||
|             {"users_old": self.richard.id}, | ||||
|         ) | ||||
|         membership = baker.make(Membership, club=self.club, role=2, end_date=None) | ||||
|         response = self.client.post(self.members_url, {"members_old": [membership.id]}) | ||||
|         self.assertRedirects(response, self.members_url) | ||||
|         self.club.refresh_from_db() | ||||
|         self.assert_membership_ended_today(self.richard) | ||||
|         self.assert_membership_ended_today(membership.user) | ||||
|  | ||||
|     def test_end_membership_higher_role(self): | ||||
|         """Test that board members of the club cannot end memberships | ||||
| @@ -374,46 +433,30 @@ class TestMembership(TestClub): | ||||
|         """ | ||||
|         membership = self.president.memberships.filter(club=self.club).first() | ||||
|         self.client.force_login(self.simple_board_member) | ||||
|         self.client.post( | ||||
|             self.members_url, | ||||
|             {"users_old": self.president.id}, | ||||
|         ) | ||||
|         self.client.post(self.members_url, {"members_old": [membership.id]}) | ||||
|         self.club.refresh_from_db() | ||||
|         new_membership = self.club.get_membership_for(self.president) | ||||
|         assert new_membership is not None | ||||
|         assert new_membership == membership | ||||
|  | ||||
|         membership = self.president.memberships.filter(club=self.club).first() | ||||
|         membership.refresh_from_db() | ||||
|         assert membership.end_date is None | ||||
|  | ||||
|     def test_end_membership_as_main_club_board(self): | ||||
|         """Test that board members of the main club can end the membership | ||||
|         of anyone. | ||||
|         """ | ||||
|     def test_end_membership_with_permission(self): | ||||
|         """Test that users with permission can end any membership.""" | ||||
|         # make subscriber a board member | ||||
|         subscriber = subscriber_user.make() | ||||
|         Membership.objects.create(club=self.ae, user=subscriber, role=3) | ||||
|  | ||||
|         nb_memberships = self.club.members.ongoing().count() | ||||
|         self.client.force_login(subscriber) | ||||
|         self.client.force_login( | ||||
|             subscriber_user.make( | ||||
|                 user_permissions=[Permission.objects.get(codename="change_membership")] | ||||
|             ) | ||||
|         ) | ||||
|         president_membership = self.club.president | ||||
|         response = self.client.post( | ||||
|             self.members_url, | ||||
|             {"users_old": self.president.id}, | ||||
|             self.members_url, {"members_old": [president_membership.id]} | ||||
|         ) | ||||
|         self.assertRedirects(response, self.members_url) | ||||
|         self.assert_membership_ended_today(self.president) | ||||
|         assert self.club.members.ongoing().count() == nb_memberships - 1 | ||||
|  | ||||
|     def test_end_membership_as_root(self): | ||||
|         """Test that root users can end the membership of anyone.""" | ||||
|         nb_memberships = self.club.members.ongoing().count() | ||||
|         self.client.force_login(self.root) | ||||
|         response = self.client.post( | ||||
|             self.members_url, | ||||
|             {"users_old": [self.president.id]}, | ||||
|         ) | ||||
|         self.assertRedirects(response, self.members_url) | ||||
|         self.assert_membership_ended_today(self.president) | ||||
|         self.assert_membership_ended_today(president_membership.user) | ||||
|         assert self.club.members.ongoing().count() == nb_memberships - 1 | ||||
|  | ||||
|     def test_end_membership_as_foreigner(self): | ||||
| @@ -421,14 +464,11 @@ class TestMembership(TestClub): | ||||
|         nb_memberships = self.club.members.count() | ||||
|         membership = self.richard.memberships.filter(club=self.club).first() | ||||
|         self.client.force_login(self.subscriber) | ||||
|         self.client.post( | ||||
|             self.members_url, | ||||
|             {"users_old": [self.richard.id]}, | ||||
|         ) | ||||
|         self.client.post(self.members_url, {"members_old": [self.richard.id]}) | ||||
|         # nothing should have changed | ||||
|         new_mem = self.club.get_membership_for(self.richard) | ||||
|         membership.refresh_from_db() | ||||
|         assert self.club.members.count() == nb_memberships | ||||
|         assert membership == new_mem | ||||
|         assert membership.end_date is None | ||||
|  | ||||
|     def test_remove_from_club_group(self): | ||||
|         """Test that when a membership ends, the user is removed from club groups.""" | ||||
| @@ -490,3 +530,85 @@ class TestMembership(TestClub): | ||||
|         new_board = set(self.club.board_group.users.values_list("id", flat=True)) | ||||
|         assert new_members == initial_members | ||||
|         assert new_board == initial_board | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| class TestJoinClub: | ||||
|     @pytest.fixture(autouse=True) | ||||
|     def clear_cache(self): | ||||
|         cache.clear() | ||||
|  | ||||
|     @pytest.mark.parametrize( | ||||
|         ("user_factory", "role", "errors"), | ||||
|         [ | ||||
|             ( | ||||
|                 subscriber_user.make, | ||||
|                 2, | ||||
|                 { | ||||
|                     "role": [ | ||||
|                         "Sélectionnez un choix valide. 2 n\u2019en fait pas partie." | ||||
|                     ] | ||||
|                 }, | ||||
|             ), | ||||
|             ( | ||||
|                 lambda: baker.make(User), | ||||
|                 1, | ||||
|                 {"__all__": ["Vous devez être cotisant pour faire partie d'un club"]}, | ||||
|             ), | ||||
|         ], | ||||
|     ) | ||||
|     def test_join_club_errors( | ||||
|         self, user_factory: Callable[[], User], role: int, errors: dict | ||||
|     ): | ||||
|         club = baker.make(Club) | ||||
|         user = user_factory() | ||||
|         form = JoinClubForm(club=club, request_user=user, data={"role": role}) | ||||
|         assert not form.is_valid() | ||||
|         assert form.errors == errors | ||||
|  | ||||
|     def test_user_already_in_club(self): | ||||
|         club = baker.make(Club) | ||||
|         user = subscriber_user.make() | ||||
|         baker.make(Membership, user=user, club=club) | ||||
|         form = JoinClubForm(club=club, request_user=user, data={"role": 1}) | ||||
|         assert not form.is_valid() | ||||
|         assert form.errors == {"__all__": ["Vous êtes déjà membre de ce club."]} | ||||
|  | ||||
|     def test_ok(self): | ||||
|         club = baker.make(Club) | ||||
|         user = subscriber_user.make() | ||||
|         form = JoinClubForm(club=club, request_user=user, data={"role": 1}) | ||||
|         assert form.is_valid() | ||||
|         form.save() | ||||
|         assert Membership.objects.ongoing().filter(user=user, club=club).exists() | ||||
|  | ||||
|  | ||||
| class TestOldMembersView(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|         club = baker.make(Club) | ||||
|         roles = [1, 1, 1, 2, 2, 4, 4, 5, 7, 9, 10] | ||||
|         cls.memberships = baker.make( | ||||
|             Membership, | ||||
|             role=iter(roles), | ||||
|             club=club, | ||||
|             start_date=now() - timedelta(days=14), | ||||
|             end_date=now() - timedelta(days=7), | ||||
|             _quantity=len(roles), | ||||
|             _bulk_create=True, | ||||
|         ) | ||||
|         cls.url = reverse("club:club_old_members", kwargs={"club_id": club.id}) | ||||
|  | ||||
|     def test_ok(self): | ||||
|         user = subscriber_user.make() | ||||
|         self.client.force_login(user) | ||||
|         res = self.client.get(self.url) | ||||
|         assert res.status_code == 200 | ||||
|  | ||||
|     def test_access_forbidden(self): | ||||
|         res = self.client.get(self.url) | ||||
|         assertRedirects(res, reverse("core:login", query={"next": self.url})) | ||||
|  | ||||
|         self.client.force_login(baker.make(User)) | ||||
|         res = self.client.get(self.url) | ||||
|         assert res.status_code == 403 | ||||
|   | ||||
							
								
								
									
										35
									
								
								club/tests/test_posters.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								club/tests/test_posters.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| import pytest | ||||
| from django.test import Client | ||||
| from django.urls import reverse | ||||
| from model_bakery import baker | ||||
|  | ||||
| from club.models import Club | ||||
| from com.models import Poster | ||||
| from core.baker_recipes import subscriber_user | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| @pytest.mark.parametrize("route_url", ["club:poster_list", "club:poster_create"]) | ||||
| def test_access(client: Client, route_url): | ||||
|     club = baker.make(Club) | ||||
|     user = subscriber_user.make() | ||||
|     url = reverse(route_url, kwargs={"club_id": club.id}) | ||||
|  | ||||
|     client.force_login(user) | ||||
|     assert client.get(url).status_code == 403 | ||||
|     club.board_group.users.add(user) | ||||
|     assert client.get(url).status_code == 200 | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| @pytest.mark.parametrize("route_url", ["club:poster_edit", "club:poster_delete"]) | ||||
| def test_access_specific_poster(client: Client, route_url): | ||||
|     club = baker.make(Club) | ||||
|     user = subscriber_user.make() | ||||
|     poster = baker.make(Poster) | ||||
|     url = reverse(route_url, kwargs={"club_id": club.id, "poster_id": poster.id}) | ||||
|  | ||||
|     client.force_login(user) | ||||
|     assert client.get(url).status_code == 403 | ||||
|     club.board_group.users.add(user) | ||||
|     assert client.get(url).status_code == 200 | ||||
| @@ -25,6 +25,7 @@ | ||||
| from django.urls import path | ||||
|  | ||||
| from club.views import ( | ||||
|     ClubAddMembersFragment, | ||||
|     ClubCreateView, | ||||
|     ClubEditView, | ||||
|     ClubListView, | ||||
| @@ -60,6 +61,11 @@ urlpatterns = [ | ||||
|     path("<int:club_id>/edit/", ClubEditView.as_view(), name="club_edit"), | ||||
|     path("<int:club_id>/edit/page/", ClubPageEditView.as_view(), name="club_edit_page"), | ||||
|     path("<int:club_id>/members/", ClubMembersView.as_view(), name="club_members"), | ||||
|     path( | ||||
|         "fragment/<int:club_id>/members/", | ||||
|         ClubAddMembersFragment.as_view(), | ||||
|         name="club_new_members", | ||||
|     ), | ||||
|     path( | ||||
|         "<int:club_id>/elderlies/", | ||||
|         ClubOldMembersView.as_view(), | ||||
|   | ||||
							
								
								
									
										219
									
								
								club/views.py
									
									
									
									
									
								
							
							
						
						
									
										219
									
								
								club/views.py
									
									
									
									
									
								
							| @@ -23,52 +23,57 @@ | ||||
| # | ||||
|  | ||||
| import csv | ||||
| from typing import Any | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.mixins import PermissionRequiredMixin | ||||
| from django.contrib.messages.views import SuccessMessageMixin | ||||
| from django.core.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError | ||||
| from django.core.paginator import InvalidPage, Paginator | ||||
| from django.db.models import Sum | ||||
| from django.http import ( | ||||
|     Http404, | ||||
|     HttpResponseRedirect, | ||||
|     StreamingHttpResponse, | ||||
| ) | ||||
| from django.db.models import Q, Sum | ||||
| from django.http import Http404, HttpResponseRedirect, StreamingHttpResponse | ||||
| from django.shortcuts import get_object_or_404, redirect | ||||
| from django.urls import reverse, reverse_lazy | ||||
| from django.utils import timezone | ||||
| from django.utils.functional import cached_property | ||||
| from django.utils.safestring import SafeString | ||||
| from django.utils.timezone import now | ||||
| from django.utils.translation import gettext as _t | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django.views.generic import DetailView, ListView, View | ||||
| from django.views.generic.edit import CreateView, DeleteView, UpdateView | ||||
|  | ||||
| from club.forms import ( | ||||
|     ClubAddMemberForm, | ||||
|     ClubAdminEditForm, | ||||
|     ClubEditForm, | ||||
|     ClubMemberForm, | ||||
|     ClubOldMemberForm, | ||||
|     JoinClubForm, | ||||
|     MailingForm, | ||||
|     SellingsForm, | ||||
| ) | ||||
| from club.models import Club, Mailing, MailingSubscription, Membership | ||||
| from com.models import Poster | ||||
| from com.views import ( | ||||
|     PosterCreateBaseView, | ||||
|     PosterDeleteBaseView, | ||||
|     PosterEditBaseView, | ||||
|     PosterListBaseView, | ||||
| ) | ||||
| from core.auth.mixins import CanCreateMixin, CanEditMixin, CanViewMixin | ||||
| from core.auth.mixins import CanEditMixin | ||||
| from core.models import PageRev | ||||
| from core.views import DetailFormView, PageEditViewBase | ||||
| from core.views.mixins import TabedViewMixin | ||||
| from core.views import DetailFormView, PageEditViewBase, UseFragmentsMixin | ||||
| from core.views.mixins import FragmentMixin, FragmentRenderer, TabedViewMixin | ||||
| from counter.models import Selling | ||||
|  | ||||
|  | ||||
| class ClubTabsMixin(TabedViewMixin): | ||||
|     def get_tabs_title(self): | ||||
|         obj = self.get_object() | ||||
|         if isinstance(obj, PageRev): | ||||
|             self.object = obj.page.club | ||||
|         if not hasattr(self, "object") or not self.object: | ||||
|             self.object = self.get_object() | ||||
|         if isinstance(self.object, PageRev): | ||||
|             self.object = self.object.page.club | ||||
|         elif isinstance(self.object, Poster): | ||||
|             self.object = self.object.club | ||||
|         return self.object.get_display_name() | ||||
|  | ||||
|     def get_list_of_tabs(self): | ||||
| @@ -79,7 +84,7 @@ class ClubTabsMixin(TabedViewMixin): | ||||
|                 "name": _("Infos"), | ||||
|             } | ||||
|         ] | ||||
|         if self.request.user.can_view(self.object): | ||||
|         if self.request.user.has_perm("club.view_club"): | ||||
|             tab_list.extend( | ||||
|                 [ | ||||
|                     { | ||||
| @@ -98,16 +103,16 @@ class ClubTabsMixin(TabedViewMixin): | ||||
|                     }, | ||||
|                 ] | ||||
|             ) | ||||
|         if self.object.page: | ||||
|             tab_list.append( | ||||
|                 { | ||||
|                     "url": reverse( | ||||
|                         "club:club_hist", kwargs={"club_id": self.object.id} | ||||
|                     ), | ||||
|                     "slug": "history", | ||||
|                     "name": _("History"), | ||||
|                 } | ||||
|             ) | ||||
|             if self.object.page: | ||||
|                 tab_list.append( | ||||
|                     { | ||||
|                         "url": reverse( | ||||
|                             "club:club_hist", kwargs={"club_id": self.object.id} | ||||
|                         ), | ||||
|                         "slug": "history", | ||||
|                         "name": _("History"), | ||||
|                     } | ||||
|                 ) | ||||
|         if self.request.user.can_edit(self.object): | ||||
|             tab_list.extend( | ||||
|                 [ | ||||
| @@ -159,7 +164,7 @@ class ClubTabsMixin(TabedViewMixin): | ||||
|                             "club:poster_list", kwargs={"club_id": self.object.id} | ||||
|                         ), | ||||
|                         "slug": "posters", | ||||
|                         "name": _("Posters list"), | ||||
|                         "name": _("Posters"), | ||||
|                     }, | ||||
|                 ] | ||||
|             ) | ||||
| @@ -228,13 +233,14 @@ class ClubPageEditView(ClubTabsMixin, PageEditViewBase): | ||||
|         return reverse_lazy("club:club_view", kwargs={"club_id": self.club.id}) | ||||
|  | ||||
|  | ||||
| class ClubPageHistView(ClubTabsMixin, CanViewMixin, DetailView): | ||||
| class ClubPageHistView(ClubTabsMixin, PermissionRequiredMixin, DetailView): | ||||
|     """Modification hostory of the page.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
|     template_name = "club/page_history.jinja" | ||||
|     current_tab = "history" | ||||
|     permission_required = "club.view_club" | ||||
|  | ||||
|  | ||||
| class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | ||||
| @@ -246,57 +252,121 @@ class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | ||||
|     current_tab = "tools" | ||||
|  | ||||
|  | ||||
| class ClubMembersView(ClubTabsMixin, CanViewMixin, DetailFormView): | ||||
| class ClubAddMembersFragment( | ||||
|     FragmentMixin, PermissionRequiredMixin, SuccessMessageMixin, CreateView | ||||
| ): | ||||
|     template_name = "club/fragments/add_member.jinja" | ||||
|     model = Membership | ||||
|     object = None | ||||
|     reload_on_redirect = True | ||||
|     permission_required = "club.view_club" | ||||
|  | ||||
|     def dispatch(self, *args, **kwargs): | ||||
|         self.club = get_object_or_404(Club, pk=kwargs.get("club_id")) | ||||
|         return super().dispatch(*args, **kwargs) | ||||
|  | ||||
|     def get_form_class(self): | ||||
|         user = self.request.user | ||||
|         if user.has_perm("club.add_membership") or self.club.get_membership_for(user): | ||||
|             return ClubAddMemberForm | ||||
|         return JoinClubForm | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         return super().get_form_kwargs() | { | ||||
|             "request_user": self.request.user, | ||||
|             "club": self.club, | ||||
|         } | ||||
|  | ||||
|     def render_fragment(self, request, **kwargs) -> SafeString: | ||||
|         self.club = kwargs.get("club") | ||||
|         return super().render_fragment(request, **kwargs) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse("club:club_members", kwargs={"club_id": self.club.id}) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         return super().get_context_data(**kwargs) | {"club": self.club} | ||||
|  | ||||
|     def get_success_message(self, cleaned_data): | ||||
|         if "user" not in cleaned_data or cleaned_data["user"] == self.request.user: | ||||
|             return _("You are now a member of this club.") | ||||
|         return _("%(user)s has been added to club.") % cleaned_data | ||||
|  | ||||
|  | ||||
| class ClubMembersView( | ||||
|     ClubTabsMixin, UseFragmentsMixin, PermissionRequiredMixin, DetailFormView | ||||
| ): | ||||
|     """View of a club's members.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
|     form_class = ClubMemberForm | ||||
|     form_class = ClubOldMemberForm | ||||
|     template_name = "club/club_members.jinja" | ||||
|     current_tab = "members" | ||||
|     permission_required = "club.view_club" | ||||
|  | ||||
|     @cached_property | ||||
|     def members(self) -> list[Membership]: | ||||
|         return list(self.object.members.ongoing().order_by("-role")) | ||||
|     def get_fragments(self) -> dict[str, type[FragmentMixin] | FragmentRenderer]: | ||||
|         membership = self.object.get_membership_for(self.request.user) | ||||
|         if ( | ||||
|             membership | ||||
|             and membership.role <= settings.SITH_MAXIMUM_FREE_ROLE | ||||
|             and not self.request.user.has_perm("club.add_membership") | ||||
|         ): | ||||
|             # Simple club members won't see the form anymore. | ||||
|             # Even if they saw it, they couldn't add anyone to the club anyway | ||||
|             return {} | ||||
|         return {"add_member_fragment": ClubAddMembersFragment} | ||||
|  | ||||
|     def get_fragment_data(self) -> dict[str, Any]: | ||||
|         return {"add_member_fragment": {"club": self.object}} | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         kwargs = super().get_form_kwargs() | ||||
|         kwargs["request_user"] = self.request.user | ||||
|         kwargs["club"] = self.object | ||||
|         kwargs["club_members"] = self.members | ||||
|         return kwargs | ||||
|         return super().get_form_kwargs() | { | ||||
|             "user": self.request.user, | ||||
|             "club": self.object, | ||||
|         } | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["members"] = self.members | ||||
|         editable = list( | ||||
|             kwargs["form"].fields["members_old"].queryset.values_list("id", flat=True) | ||||
|         ) | ||||
|         kwargs["members"] = list( | ||||
|             self.object.members.ongoing() | ||||
|             .annotate(is_editable=Q(id__in=editable)) | ||||
|             .order_by("-role") | ||||
|             .select_related("user") | ||||
|         ) | ||||
|         kwargs["can_end_membership"] = len(editable) > 0 | ||||
|         return kwargs | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         """Check user rights.""" | ||||
|         resp = super().form_valid(form) | ||||
|  | ||||
|         data = form.clean() | ||||
|         users = data.pop("users", []) | ||||
|         users_old = data.pop("users_old", []) | ||||
|         for user in users: | ||||
|             Membership(club=self.object, user=user, **data).save() | ||||
|         for user in users_old: | ||||
|             membership = self.object.get_membership_for(user) | ||||
|             membership.end_date = timezone.now() | ||||
|         for membership in form.cleaned_data.get("members_old"): | ||||
|             membership.end_date = now() | ||||
|             membership.save() | ||||
|         return resp | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|     def get_success_url(self, **kwargs): | ||||
|         return self.request.path | ||||
|  | ||||
|  | ||||
| class ClubOldMembersView(ClubTabsMixin, CanViewMixin, DetailView): | ||||
| class ClubOldMembersView(ClubTabsMixin, PermissionRequiredMixin, DetailView): | ||||
|     """Old members of a club.""" | ||||
|  | ||||
|     model = Club | ||||
|     pk_url_kwarg = "club_id" | ||||
|     template_name = "club/club_old_members.jinja" | ||||
|     current_tab = "elderlies" | ||||
|     permission_required = "club.view_club" | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         return super().get_context_data(**kwargs) | { | ||||
|             "old_members": ( | ||||
|                 self.object.members.exclude(end_date=None) | ||||
|                 .order_by("-role", "description", "-end_date") | ||||
|                 .select_related("user") | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|  | ||||
| class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
| @@ -337,7 +407,7 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|         form = self.get_form() | ||||
|         if form.is_valid(): | ||||
|             if not len([v for v in form.cleaned_data.values() if v is not None]): | ||||
|                 qs = Selling.objects.filter(id=-1) | ||||
|                 qs = Selling.objects.none() | ||||
|             if form.cleaned_data["begin_date"]: | ||||
|                 qs = qs.filter(date__gte=form.cleaned_data["begin_date"]) | ||||
|             if form.cleaned_data["end_date"]: | ||||
| @@ -355,7 +425,9 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||
|             if len(selected_products) > 0: | ||||
|                 qs = qs.filter(product__in=selected_products) | ||||
|  | ||||
|             kwargs["result"] = qs.all().order_by("-id") | ||||
|             kwargs["result"] = qs.select_related( | ||||
|                 "counter", "counter__club", "customer", "customer__user", "seller" | ||||
|             ).order_by("-id") | ||||
|             kwargs["total"] = sum([s.quantity * s.unit_price for s in kwargs["result"]]) | ||||
|             total_quantity = qs.all().aggregate(Sum("quantity")) | ||||
|             if total_quantity["quantity__sum"]: | ||||
| @@ -686,48 +758,45 @@ class MailingAutoGenerationView(View): | ||||
|         return redirect("club:mailing", club_id=club.id) | ||||
|  | ||||
|  | ||||
| class PosterListView(ClubTabsMixin, PosterListBaseView, CanViewMixin): | ||||
| class PosterListView(ClubTabsMixin, PosterListBaseView): | ||||
|     """List communication posters.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     extra_context = {"app": "club"} | ||||
|  | ||||
|     def get_queryset(self): | ||||
|         return super().get_queryset().filter(club=self.club.id) | ||||
|  | ||||
|     def get_object(self): | ||||
|         return self.club | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "club" | ||||
|         kwargs["club"] = self.club | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class PosterCreateView(PosterCreateBaseView, CanCreateMixin): | ||||
| class PosterCreateView(ClubTabsMixin, PosterCreateBaseView): | ||||
|     """Create communication poster.""" | ||||
|  | ||||
|     pk_url_kwarg = "club_id" | ||||
|  | ||||
|     def get_object(self): | ||||
|         obj = super().get_object() | ||||
|         if not obj: | ||||
|             return self.club | ||||
|         return obj | ||||
|     current_tab = "posters" | ||||
|  | ||||
|     def get_success_url(self, **kwargs): | ||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) | ||||
|  | ||||
|     def get_object(self, *args, **kwargs): | ||||
|         return self.club | ||||
|  | ||||
| class PosterEditView(ClubTabsMixin, PosterEditBaseView, CanEditMixin): | ||||
|  | ||||
| class PosterEditView(ClubTabsMixin, PosterEditBaseView): | ||||
|     """Edit communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     extra_context = {"app": "club"} | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "club" | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class PosterDeleteView(PosterDeleteBaseView, ClubTabsMixin, CanEditMixin): | ||||
| class PosterDeleteView(ClubTabsMixin, PosterDeleteBaseView): | ||||
|     """Delete communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) | ||||
|   | ||||
							
								
								
									
										26
									
								
								com/forms.py
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								com/forms.py
									
									
									
									
									
								
							| @@ -2,7 +2,6 @@ from datetime import date | ||||
|  | ||||
| from dateutil.relativedelta import relativedelta | ||||
| from django import forms | ||||
| from django.db.models import Exists, OuterRef | ||||
| from django.forms import CheckboxInput | ||||
| from django.utils import timezone | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| @@ -35,20 +34,18 @@ class PosterForm(forms.ModelForm): | ||||
|         label=_("Start date"), | ||||
|         widget=SelectDateTime, | ||||
|         required=True, | ||||
|         initial=timezone.now().strftime("%Y-%m-%d %H:%M:%S"), | ||||
|         initial=timezone.now(), | ||||
|     ) | ||||
|     date_end = forms.DateTimeField( | ||||
|         label=_("End date"), widget=SelectDateTime, required=False | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self.user = kwargs.pop("user", None) | ||||
|     def __init__(self, *args, user: User, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         if self.user and not self.user.is_com_admin: | ||||
|             self.fields["club"].queryset = Club.objects.filter( | ||||
|                 id__in=self.user.clubs_with_rights | ||||
|             ) | ||||
|             self.fields.pop("display_time") | ||||
|         if user.is_root or user.is_com_admin: | ||||
|             self.fields["club"].widget = AutoCompleteSelectClub() | ||||
|         else: | ||||
|             self.fields["club"].queryset = Club.objects.having_board_member(user) | ||||
|  | ||||
|  | ||||
| class NewsDateForm(forms.ModelForm): | ||||
| @@ -161,16 +158,9 @@ class NewsForm(forms.ModelForm): | ||||
|         # if the author is an admin, he/she can choose any club, | ||||
|         # otherwise, only clubs for which he/she is a board member can be selected | ||||
|         if author.is_root or author.is_com_admin: | ||||
|             self.fields["club"] = forms.ModelChoiceField( | ||||
|                 queryset=Club.objects.all(), widget=AutoCompleteSelectClub | ||||
|             ) | ||||
|             self.fields["club"].widget = AutoCompleteSelectClub() | ||||
|         else: | ||||
|             active_memberships = author.memberships.board().ongoing() | ||||
|             self.fields["club"] = forms.ModelChoiceField( | ||||
|                 queryset=Club.objects.filter( | ||||
|                     Exists(active_memberships.filter(club=OuterRef("pk"))) | ||||
|                 ) | ||||
|             ) | ||||
|             self.fields["club"].queryset = Club.objects.having_board_member(author) | ||||
|  | ||||
|     def is_valid(self): | ||||
|         return super().is_valid() and self.date_form.is_valid() | ||||
|   | ||||
| @@ -412,17 +412,5 @@ class Poster(models.Model): | ||||
|         if self.date_end and self.date_begin > self.date_end: | ||||
|             raise ValidationError(_("Begin date should be before end date")) | ||||
|  | ||||
|     def is_owned_by(self, user): | ||||
|         if user.is_anonymous: | ||||
|             return False | ||||
|         return user.is_com_admin or len(user.clubs_with_rights) > 0 | ||||
|  | ||||
|     def can_be_moderated_by(self, user): | ||||
|         return user.is_com_admin | ||||
|  | ||||
|     def get_display_name(self): | ||||
|         return self.club.get_display_name() | ||||
|  | ||||
|     @property | ||||
|     def page(self): | ||||
|         return self.club.page | ||||
|   | ||||
							
								
								
									
										49
									
								
								com/static/bundled/com/slideshow-index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								com/static/bundled/com/slideshow-index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| const INTERVAL = 10; | ||||
|  | ||||
| interface Poster { | ||||
|   url: string; // URL of the poster | ||||
|   displayTime: number; // Number of seconds to display that poster | ||||
| } | ||||
|  | ||||
| document.addEventListener("alpine:init", () => { | ||||
|   Alpine.data("slideshow", (posters: Poster[]) => ({ | ||||
|     posters: posters, | ||||
|     progress: 0, | ||||
|     elapsed: 0, | ||||
|  | ||||
|     current: 0, | ||||
|     previous: 0, | ||||
|  | ||||
|     init() { | ||||
|       this.$watch("elapsed", () => { | ||||
|         const displayTime = this.posters[this.current].displayTime * 1000; | ||||
|         if (this.elapsed > displayTime) { | ||||
|           this.previous = this.current; | ||||
|           this.current = this.getNext(); | ||||
|           this.elapsed = 0; | ||||
|         } | ||||
|         if (displayTime === 0) { | ||||
|           this.progress = 100; | ||||
|         } else { | ||||
|           this.progress = (100 * this.elapsed) / displayTime; | ||||
|         } | ||||
|       }); | ||||
|       setInterval(() => { | ||||
|         this.elapsed += INTERVAL; | ||||
|       }, INTERVAL); | ||||
|     }, | ||||
|  | ||||
|     getNext() { | ||||
|       return (this.current + 1) % this.posters.length; | ||||
|     }, | ||||
|  | ||||
|     async toggleFullScreen(event: Event) { | ||||
|       if (document.fullscreenElement) { | ||||
|         await document.exitFullscreen(); | ||||
|         return; | ||||
|       } | ||||
|       const target = event.target as HTMLElement; | ||||
|       await target.requestFullscreen(); | ||||
|     }, | ||||
|   })); | ||||
| }); | ||||
| @@ -83,7 +83,8 @@ | ||||
|     #links_content { | ||||
|       overflow: auto; | ||||
|       box-shadow: $shadow-color 1px 1px 1px; | ||||
|       height: 20em; | ||||
|       min-height: 20em; | ||||
|       padding-bottom: 1em; | ||||
|  | ||||
|       h4 { | ||||
|         margin-left: 5px; | ||||
|   | ||||
| @@ -111,7 +111,7 @@ | ||||
|             top: 0; | ||||
|             left: 0; | ||||
|             z-index: 10; | ||||
|             content: "Click to expand"; | ||||
|             content: attr(hover); | ||||
|             color: white; | ||||
|             background-color: rgba(black, 0.5); | ||||
|           } | ||||
|   | ||||
| @@ -1,23 +0,0 @@ | ||||
| $(document).ready(() => { | ||||
|   $("#poster_list #view").click(() => { | ||||
|     $("#view").removeClass("active"); | ||||
|   }); | ||||
|  | ||||
|   $("#poster_list .poster .image").click((e) => { | ||||
|     let el = $(e.target); | ||||
|     if (el.hasClass("image")) { | ||||
|       el = el.find("img"); | ||||
|     } | ||||
|     $("#poster_list #view #placeholder").html(el.clone()); | ||||
|  | ||||
|     $("#view").addClass("active"); | ||||
|   }); | ||||
|  | ||||
|   $(document).keyup((e) => { | ||||
|     if (e.keyCode === 27) { | ||||
|       // escape key maps to keycode `27` | ||||
|       e.preventDefault(); | ||||
|       $("#view").removeClass("active"); | ||||
|     } | ||||
|   }); | ||||
| }); | ||||
| @@ -1,98 +0,0 @@ | ||||
| $(document).ready(() => { | ||||
|   const transitionTime = 1000; | ||||
|  | ||||
|   let i = 0; | ||||
|   const max = $("#slideshow .slide").length; | ||||
|  | ||||
|   function enterFullscreen() { | ||||
|     const element = document.getElementById("slideshow"); | ||||
|     $(element).addClass("fullscreen"); | ||||
|     if (element.requestFullscreen) { | ||||
|       element.requestFullscreen(); | ||||
|     } else if (element.mozRequestFullScreen) { | ||||
|       element.mozRequestFullScreen(); | ||||
|     } else if (element.webkitRequestFullscreen) { | ||||
|       element.webkitRequestFullscreen(); | ||||
|     } else if (element.msRequestFullscreen) { | ||||
|       element.msRequestFullscreen(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   function exitFullscreen() { | ||||
|     const element = document.getElementById("slideshow"); | ||||
|     $(element).removeClass("fullscreen"); | ||||
|     if (document.exitFullscreen) { | ||||
|       document.exitFullscreen(); | ||||
|     } else if (document.webkitExitFullscreen) { | ||||
|       document.webkitExitFullscreen(); | ||||
|     } else if (document.mozCancelFullScreen) { | ||||
|       document.mozCancelFullScreen(); | ||||
|     } else if (document.msExitFullscreen) { | ||||
|       document.msExitFullscreen(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   function initProgressBar() { | ||||
|     $("#slideshow #progress_bar").css("transition", "none"); | ||||
|     $("#slideshow #progress_bar").removeClass("progress"); | ||||
|     $("#slideshow #progress_bar").addClass("init"); | ||||
|   } | ||||
|  | ||||
|   function startProgressBar(displayTime) { | ||||
|     $("#slideshow #progress_bar").removeClass("init"); | ||||
|     $("#slideshow #progress_bar").addClass("progress"); | ||||
|     $("#slideshow #progress_bar").css("transition", `width ${displayTime}s linear`); | ||||
|   } | ||||
|  | ||||
|   function next() { | ||||
|     initProgressBar(); | ||||
|     const slide = $($("#slideshow .slide").get(i % max)); | ||||
|     slide.removeClass("center"); | ||||
|     slide.addClass("left"); | ||||
|  | ||||
|     const nextSlide = $($("#slideshow .slide").get((i + 1) % max)); | ||||
|     nextSlide.removeClass("right"); | ||||
|     nextSlide.addClass("center"); | ||||
|     const displayTime = nextSlide.attr("display_time") || 2; | ||||
|  | ||||
|     $("#slideshow .bullet").removeClass("active"); | ||||
|     const bullet = $("#slideshow .bullet")[(i + 1) % max]; | ||||
|     $(bullet).addClass("active"); | ||||
|  | ||||
|     i = (i + 1) % max; | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       const othersLeft = $("#slideshow .slide.left"); | ||||
|       othersLeft.removeClass("left"); | ||||
|       othersLeft.addClass("right"); | ||||
|  | ||||
|       startProgressBar(displayTime); | ||||
|       setTimeout(next, displayTime * 1000); | ||||
|     }, transitionTime); | ||||
|   } | ||||
|  | ||||
|   const displayTime = $("#slideshow .center").attr("display_time"); | ||||
|   initProgressBar(); | ||||
|   setTimeout(() => { | ||||
|     if (max > 1) { | ||||
|       startProgressBar(displayTime); | ||||
|       setTimeout(next, displayTime * 1000); | ||||
|     } | ||||
|   }, 10); | ||||
|  | ||||
|   $("#slideshow").click(() => { | ||||
|     if ($("#slideshow").hasClass("fullscreen")) { | ||||
|       exitFullscreen(); | ||||
|     } else { | ||||
|       enterFullscreen(); | ||||
|     } | ||||
|   }); | ||||
|  | ||||
|   $(document).keyup((e) => { | ||||
|     if (e.keyCode === 27) { | ||||
|       // escape key maps to keycode `27` | ||||
|       e.preventDefault(); | ||||
|       exitFullscreen(); | ||||
|     } | ||||
|   }); | ||||
| }); | ||||
| @@ -1,4 +1,4 @@ | ||||
| body{ | ||||
| body { | ||||
|   position: absolute; | ||||
|   width: 100vw; | ||||
|   height: 100vh; | ||||
| @@ -7,22 +7,22 @@ body{ | ||||
|   margin: 0; | ||||
| } | ||||
|  | ||||
| #slideshow{ | ||||
| #slideshow { | ||||
|   position: relative; | ||||
|   background-color: lightgrey; | ||||
|  | ||||
|   height: 100%; | ||||
|  | ||||
|   *{ | ||||
|   * { | ||||
|     -webkit-user-select: none; | ||||
|     -moz-user-select: none; | ||||
|     -ms-user-select: none; | ||||
|     user-select: none; | ||||
|   } | ||||
|  | ||||
|   &:hover{ | ||||
|   &:hover { | ||||
|  | ||||
|     &::before{ | ||||
|     &::before { | ||||
|  | ||||
|       position: absolute; | ||||
|       width: 100%; | ||||
| @@ -34,7 +34,7 @@ body{ | ||||
|  | ||||
|       z-index: 10; | ||||
|  | ||||
|       content: "Click to expand"; | ||||
|       content: attr(hover); | ||||
|  | ||||
|       color: white; | ||||
|       background-color: rgba(black, 0.5); | ||||
| @@ -43,7 +43,7 @@ body{ | ||||
|  | ||||
|   } | ||||
|  | ||||
|   &.fullscreen{ | ||||
|   &:fullscreen { | ||||
|     position: fixed; | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
| @@ -51,57 +51,78 @@ body{ | ||||
|     left: 0; | ||||
|     background: none; | ||||
|  | ||||
|     &:before{ | ||||
|       display:none; | ||||
|     &:before { | ||||
|       display: none; | ||||
|     } | ||||
|  | ||||
|     #slides{ | ||||
|     #slides { | ||||
|       height: 100vh; | ||||
|     } | ||||
|  | ||||
|   } | ||||
|  | ||||
|   #slides{ | ||||
|   #slides { | ||||
|     position: relative; | ||||
|     height: 100%; | ||||
|     overflow: hidden; | ||||
|     background-color: grey; | ||||
|  | ||||
|     .slide{ | ||||
|     .slide { | ||||
|       position: absolute; | ||||
|       width: 100%; | ||||
|       height: 100%; | ||||
|  | ||||
|       display: inline-flex; | ||||
|       display: none; | ||||
|       justify-content: center; | ||||
|  | ||||
|       top: 0px; | ||||
|       left: 0%; | ||||
|  | ||||
|       background-color: grey; | ||||
|       transition: left 1s ease-out; | ||||
|  | ||||
|       img{ | ||||
|       img { | ||||
|         max-width: 100%; | ||||
|         max-height: 100%; | ||||
|         object-fit: contain; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .slide.left{ | ||||
|       left: -100%; | ||||
|     } | ||||
|       &.current { | ||||
|         display: inline-flex; | ||||
|         left: 0%; | ||||
|         animation: scrolling-in 1s linear; | ||||
|       } | ||||
|  | ||||
|     .slide.center{ | ||||
|       left: 0px; | ||||
|     } | ||||
|       &.previous { | ||||
|         display: inline-flex; | ||||
|         animation: scrolling-out 1s linear; | ||||
|         opacity: 0; | ||||
|         transition: opacity 0.1s; | ||||
|         transition-delay: 0.9s; | ||||
|       } | ||||
|  | ||||
|       @keyframes scrolling-in { | ||||
|         0% { | ||||
|           transform: translateX(100%); | ||||
|         } | ||||
|  | ||||
|         100% { | ||||
|           transform: translateX(0%); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       @keyframes scrolling-out { | ||||
|         0% { | ||||
|           transform: translateX(0%); | ||||
|         } | ||||
|  | ||||
|         100% { | ||||
|           transform: translateX(-100%); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|     .slide.right{ | ||||
|       left: 100%; | ||||
|       transition: none; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  | ||||
|   #progress_bullets{ | ||||
|   #progress_bullets { | ||||
|     position: absolute; | ||||
|     bottom: 10px; | ||||
|     width: 100%; | ||||
| @@ -112,7 +133,7 @@ body{ | ||||
|  | ||||
|     margin-bottom: 10px; | ||||
|  | ||||
|     .bullet{ | ||||
|     .bullet { | ||||
|       height: 10px; | ||||
|       width: 10px; | ||||
|  | ||||
| @@ -123,27 +144,33 @@ body{ | ||||
|  | ||||
|       background-color: grey; | ||||
|  | ||||
|       &.active{ | ||||
|       &.active { | ||||
|         background-color: #c99836; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   #progress_bar{ | ||||
|   progress { | ||||
|     --color: #304c83; | ||||
|  | ||||
|     position: absolute; | ||||
|     bottom: 0px; | ||||
|     height: 10px; | ||||
|     background-color: #304c83; | ||||
|     color: var(--color); | ||||
|     width: 100%; | ||||
|     margin-bottom: 0px; | ||||
|     border: none; | ||||
|  | ||||
|     &.init{ | ||||
|       width: 0px; | ||||
|       transition: none; | ||||
|     &::-moz-progress-bar { | ||||
|       background: var(--color); | ||||
|     } | ||||
|  | ||||
|     &.progress{ | ||||
|       width: 100%; | ||||
|       transition: width 10s linear; | ||||
|     &::-webkit-progress-value { | ||||
|       background: var(--color); | ||||
|     } | ||||
|  | ||||
|     &[value] { | ||||
|       background-color: transparent; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -76,18 +76,20 @@ | ||||
|               It will stay hidden for other users until it has been published. | ||||
|             {% endtrans %} | ||||
|           </p> | ||||
|           {% if user.has_perm("com.moderate_news") %} | ||||
|           {%- if user.has_perm("com.moderate_news") -%} | ||||
|             {# This is an additional query for each non-moderated news, | ||||
|             but it will be executed only for admin users, and only one time | ||||
|             (if they do their job and moderated news as soon as they see them), | ||||
|             (if they do their job and moderate news as soon as they see them), | ||||
|             so it's still reasonable #} | ||||
|             <div | ||||
|               {% if news is integer or news is string %} | ||||
|               {% if news is integer or news is string -%} | ||||
|                 x-data="{ nbEvents: 0 }" | ||||
|                 x-init="nbEvents = await nbToPublish()" | ||||
|               {% else %} | ||||
|               {%- elif news.is_published -%} | ||||
|                 x-data="{ nbEvents: 0 }" | ||||
|               {%- else -%} | ||||
|                 x-data="{ nbEvents: {{ news.dates.count() }} }" | ||||
|               {% endif %} | ||||
|               {%- endif -%} | ||||
|             > | ||||
|               <template x-if="nbEvents > 1"> | ||||
|                 <div> | ||||
|   | ||||
| @@ -205,6 +205,10 @@ | ||||
|               <i class="fa-solid fa-graduation-cap fa-xl"></i> | ||||
|               <a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a> | ||||
|             </li> | ||||
|             <li> | ||||
|               <i class="fa-solid fa-calendar-days fa-xl"></i> | ||||
|               <a href="{{ url("timetable:generator") }}">{% trans %}Timetable{% endtrans %}</a> | ||||
|             </li> | ||||
|             <li> | ||||
|               <i class="fa-solid fa-magnifying-glass fa-xl"></i> | ||||
|               <a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a> | ||||
|   | ||||
| @@ -1,11 +1,5 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block script %} | ||||
|   {{ super() }} | ||||
|   <script src="{{ static('com/js/poster_list.js') }}"></script> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
| {% block title %} | ||||
|   {% trans %}Poster{% endtrans %} | ||||
| {% endblock %} | ||||
| @@ -15,7 +9,7 @@ | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|   <div id="poster_list"> | ||||
|   <div id="poster_list" x-data="{ active: null }"> | ||||
|  | ||||
|     <div id="title"> | ||||
|       <h3>{% trans %}Posters{% endtrans %}</h3> | ||||
| @@ -38,7 +32,13 @@ | ||||
|         {% for poster in poster_list %} | ||||
|           <div class="poster{% if not poster.is_moderated %} not_moderated{% endif %}"> | ||||
|             <div class="name">{{ poster.name }}</div> | ||||
|             <div class="image"><img src="{{ poster.file.url }}"></img></div> | ||||
|             <div | ||||
|               class="image" | ||||
|               hover="{% trans %}Click to expand{% endtrans %}" | ||||
|               @click="active = $el.firstElementChild" | ||||
|             > | ||||
|               <img src="{{ poster.file.url }}"></img> | ||||
|             </div> | ||||
|             <div class="dates"> | ||||
|               <div class="begin">{{ poster.date_begin | localtime | date("d/M/Y H:m") }}</div> | ||||
|               <div class="end">{{ poster.date_end | localtime | date("d/M/Y H:m") }}</div> | ||||
| @@ -62,7 +62,14 @@ | ||||
|  | ||||
|     </div> | ||||
|  | ||||
|     <div id="view"><div id="placeholder"></div></div> | ||||
|     <div | ||||
|       id="view" | ||||
|       @keyup.escape.window="active = null" | ||||
|       @click="active = null" | ||||
|       :class="{active: active !== null}" | ||||
|     > | ||||
|       <div id="placeholder"><img :src="active?.src"></div> | ||||
|     </div> | ||||
|  | ||||
|   </div> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -2,28 +2,44 @@ | ||||
| <html lang="fr"> | ||||
|   <head> | ||||
|     <title>{% trans %}Slideshow{% endtrans %}</title> | ||||
|     <link rel="shortcut icon" href="{{ static('core/img/favicon.ico') }}"> | ||||
|     <link href="{{ static('css/slideshow.scss') }}" rel="stylesheet" type="text/css" /> | ||||
|     <script src="{{ static('bundled/vendored/jquery.min.js') }}"></script> | ||||
|     <script src="{{ static('com/js/slideshow.js') }}"></script> | ||||
|     <script type="module" src="{{ static('bundled/alpine-index.js') }}"></script> | ||||
|     <script type="module" src="{{ static('bundled/com/slideshow-index.ts') }}"></script> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="slideshow"> | ||||
|   <body x-data="slideshow([ | ||||
|                 {% for poster in posters %} | ||||
|                   { | ||||
|                   url: '{{ poster.file.url }}', | ||||
|                   displayTime: {{ poster.display_time }} | ||||
|                   }, | ||||
|                 {% endfor %} | ||||
|                 ])"> | ||||
|     <div | ||||
|       id="slideshow" | ||||
|       @click="toggleFullScreen" | ||||
|       hover="{% trans %}Click to expand{% endtrans %}" | ||||
|       @keyup.f.window="toggleFullScreen" | ||||
|     > | ||||
|  | ||||
|       <div id="slides"> | ||||
|         {% for poster in posters %} | ||||
|           <div class="slide {% if loop.first %}center{% else %}right{% endif %}" display_time="{{ poster.display_time }}"> | ||||
|             <img src="{{ poster.file.url }}"> | ||||
|         <template x-for="(poster, index) in posters"> | ||||
|           <div class="slide" :class="{ | ||||
|                                      current: index === current, | ||||
|                                      previous: index !== current && index === previous, | ||||
|                                      }"> | ||||
|             <img :src="poster.url"> | ||||
|           </div> | ||||
|         {% endfor %} | ||||
|         </template> | ||||
|       </div> | ||||
|  | ||||
|       <div id="progress_bullets"> | ||||
|         {% for poster in posters %} | ||||
|           <div class="bullet {% if loop.first %}active{% endif %}"></div> | ||||
|         {% endfor %} | ||||
|         <template x-for="(poster, index) in posters"> | ||||
|           <div class="bullet" :class="{active: current === index}"></div> | ||||
|         </template> | ||||
|       </div> | ||||
|  | ||||
|       <div id="progress_bar"></div> | ||||
|       <progress :value="progress" max="100" x-show="posters.length > 1 && progress > 0"></progress> | ||||
|  | ||||
|     </div> | ||||
|   </body> | ||||
|   | ||||
| @@ -31,9 +31,7 @@ | ||||
|           <td> | ||||
|             <a href="{{ url('com:weekmail_article_edit', article_id=a.id) }}">{% trans %}Edit{% endtrans %}</a> | | ||||
|             <a href="{{ url('com:weekmail_article_delete', article_id=a.id) }}">{% trans %}Delete{% endtrans %}</a> | | ||||
|             <a href="?add_article={{ a.id }}">{% trans %}Add to weekmail{% endtrans %}</a> | | ||||
|             <a href="?up_article={{ a.id }}">{% trans %}Up{% endtrans %}</a> | | ||||
|             <a href="?down_article={{ a.id }}">{% trans %}Down{% endtrans %}</a> | ||||
|             <a href="?add_article={{ a.id }}">{% trans %}Add to weekmail{% endtrans %}</a> | ||||
|           </td> | ||||
|         </tr> | ||||
|       {% endfor %} | ||||
|   | ||||
| @@ -18,17 +18,16 @@ from unittest.mock import patch | ||||
| import pytest | ||||
| from django.conf import settings | ||||
| from django.contrib.sites.models import Site | ||||
| from django.core.files.uploadedfile import SimpleUploadedFile | ||||
| from django.test import Client, TestCase | ||||
| from django.urls import reverse | ||||
| from django.utils import html | ||||
| from django.utils.timezone import localtime, now | ||||
| from django.utils.timezone import now | ||||
| from django.utils.translation import gettext as _ | ||||
| from model_bakery import baker | ||||
| from pytest_django.asserts import assertNumQueries, assertRedirects | ||||
|  | ||||
| from club.models import Club, Membership | ||||
| from com.models import News, NewsDate, Poster, Sith, Weekmail, WeekmailArticle | ||||
| from com.models import News, NewsDate, Sith, Weekmail, WeekmailArticle | ||||
| from core.baker_recipes import subscriber_user | ||||
| from core.models import AnonymousUser, Group, User | ||||
|  | ||||
| @@ -207,31 +206,6 @@ class TestWeekmailArticle(TestCase): | ||||
|         assert not self.article.is_owned_by(self.sli) | ||||
|  | ||||
|  | ||||
| class TestPoster(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|         cls.com_admin = User.objects.get(username="comunity") | ||||
|         cls.poster = Poster.objects.create( | ||||
|             name="dummy", | ||||
|             file=SimpleUploadedFile("dummy.jpg", b"azertyuiop"), | ||||
|             club=Club.objects.first(), | ||||
|             date_begin=localtime(now()), | ||||
|         ) | ||||
|         cls.sli = User.objects.get(username="sli") | ||||
|         cls.sli.memberships.all().delete() | ||||
|         Membership(user=cls.sli, club=Club.objects.first(), role=5).save() | ||||
|         cls.susbcriber = User.objects.get(username="subscriber") | ||||
|         cls.anonymous = AnonymousUser() | ||||
|  | ||||
|     def test_poster_owner(self): | ||||
|         """Test that poster are owned by com admins and board members in clubs.""" | ||||
|         assert self.poster.is_owned_by(self.com_admin) | ||||
|         assert not self.poster.is_owned_by(self.anonymous) | ||||
|  | ||||
|         assert not self.poster.is_owned_by(self.susbcriber) | ||||
|         assert self.poster.is_owned_by(self.sli) | ||||
|  | ||||
|  | ||||
| class TestNewsCreation(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|   | ||||
							
								
								
									
										225
									
								
								com/views.py
									
									
									
									
									
								
							
							
						
						
									
										225
									
								
								com/views.py
									
									
									
									
									
								
							| @@ -28,7 +28,10 @@ from typing import Any | ||||
|  | ||||
| from dateutil.relativedelta import relativedelta | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.mixins import AccessMixin, PermissionRequiredMixin | ||||
| from django.contrib import messages | ||||
| from django.contrib.auth.mixins import ( | ||||
|     PermissionRequiredMixin, | ||||
| ) | ||||
| from django.contrib.syndication.views import Feed | ||||
| from django.core.exceptions import PermissionDenied, ValidationError | ||||
| from django.db.models import Max | ||||
| @@ -50,9 +53,10 @@ from core.auth.mixins import ( | ||||
|     CanEditPropMixin, | ||||
|     CanViewMixin, | ||||
|     PermissionOrAuthorRequiredMixin, | ||||
|     PermissionOrClubBoardRequiredMixin, | ||||
| ) | ||||
| from core.models import User | ||||
| from core.views.mixins import QuickNotifMixin, TabedViewMixin | ||||
| from core.views.mixins import TabedViewMixin | ||||
| from core.views.widgets.markdown import MarkdownInput | ||||
|  | ||||
| # Sith object | ||||
| @@ -99,13 +103,6 @@ class ComTabsMixin(TabedViewMixin): | ||||
|         ] | ||||
|  | ||||
|  | ||||
| class IsComAdminMixin(AccessMixin): | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         if not request.user.is_com_admin: | ||||
|             raise PermissionDenied | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|  | ||||
| class ComEditView(ComTabsMixin, CanEditPropMixin, UpdateView): | ||||
|     model = Sith | ||||
|     template_name = "core/edit.jinja" | ||||
| @@ -337,7 +334,7 @@ class NewsFeed(Feed): | ||||
| # Weekmail | ||||
|  | ||||
|  | ||||
| class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, DetailView): | ||||
| class WeekmailPreviewView(ComTabsMixin, CanEditPropMixin, DetailView): | ||||
|     model = Weekmail | ||||
|     template_name = "com/weekmail_preview.jinja" | ||||
|     success_url = reverse_lazy("com:weekmail") | ||||
| @@ -349,12 +346,11 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai | ||||
|  | ||||
|     def post(self, request, *args, **kwargs): | ||||
|         self.object = self.get_object() | ||||
|         messages.success(self.request, _("Weekmail sent successfully")) | ||||
|         if request.POST["send"] == "validate": | ||||
|             try: | ||||
|                 self.object.send() | ||||
|                 return HttpResponseRedirect( | ||||
|                     reverse("com:weekmail") + "?qn_weekmail_send_success" | ||||
|                 ) | ||||
|                 return HttpResponseRedirect(reverse("com:weekmail")) | ||||
|             except SMTPRecipientsRefused as e: | ||||
|                 self.bad_recipients = e.recipients | ||||
|         elif request.POST["send"] == "clean": | ||||
| @@ -365,7 +361,6 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai | ||||
|                 for u in users: | ||||
|                     u.preferences.receive_weekmail = False | ||||
|                     u.preferences.save() | ||||
|                 self.quick_notif_list += ["qn_success"] | ||||
|         return super().get(request, *args, **kwargs) | ||||
|  | ||||
|     def get_object(self, queryset=None): | ||||
| @@ -379,7 +374,7 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView): | ||||
| class WeekmailEditView(ComTabsMixin, CanEditPropMixin, UpdateView): | ||||
|     model = Weekmail | ||||
|     template_name = "com/weekmail.jinja" | ||||
|     form_class = modelform_factory( | ||||
| @@ -419,7 +414,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | ||||
|                 art.rank, prev_art.rank = prev_art.rank, art.rank | ||||
|                 art.save() | ||||
|                 prev_art.save() | ||||
|                 self.quick_notif_list += ["qn_success"] | ||||
|                 messages.success( | ||||
|                     self.request, | ||||
|                     _("%(title)s moved up in the Weekmail") % {"title": art.title}, | ||||
|                 ) | ||||
|         if "down_article" in request.GET: | ||||
|             art = get_object_or_404( | ||||
|                 WeekmailArticle, id=request.GET["down_article"], weekmail=self.object | ||||
| @@ -431,7 +429,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | ||||
|                 art.rank, next_art.rank = next_art.rank, art.rank | ||||
|                 art.save() | ||||
|                 next_art.save() | ||||
|                 self.quick_notif_list += ["qn_success"] | ||||
|                 messages.success( | ||||
|                     self.request, | ||||
|                     _("%(title)s moved down in the Weekmail") % {"title": art.title}, | ||||
|                 ) | ||||
|         if "add_article" in request.GET: | ||||
|             art = get_object_or_404( | ||||
|                 WeekmailArticle, id=request.GET["add_article"], weekmail=None | ||||
| @@ -440,7 +441,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | ||||
|             art.rank = self.object.articles.aggregate(Max("rank"))["rank__max"] or 0 | ||||
|             art.rank += 1 | ||||
|             art.save() | ||||
|             self.quick_notif_list += ["qn_success"] | ||||
|             messages.success( | ||||
|                 self.request, | ||||
|                 _("%(title)s added to the Weekmail") % {"title": art.title}, | ||||
|             ) | ||||
|         if "del_article" in request.GET: | ||||
|             art = get_object_or_404( | ||||
|                 WeekmailArticle, id=request.GET["del_article"], weekmail=self.object | ||||
| @@ -448,7 +452,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | ||||
|             art.weekmail = None | ||||
|             art.rank = -1 | ||||
|             art.save() | ||||
|             self.quick_notif_list += ["qn_success"] | ||||
|             messages.success( | ||||
|                 self.request, | ||||
|                 _("%(title)s removed from the Weekmail") % {"title": art.title}, | ||||
|             ) | ||||
|         return super().get(request, *args, **kwargs) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
| @@ -458,9 +465,7 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class WeekmailArticleEditView( | ||||
|     ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView | ||||
| ): | ||||
| class WeekmailArticleEditView(ComTabsMixin, CanEditPropMixin, UpdateView): | ||||
|     """Edit an article.""" | ||||
|  | ||||
|     model = WeekmailArticle | ||||
| @@ -472,11 +477,10 @@ class WeekmailArticleEditView( | ||||
|     pk_url_kwarg = "article_id" | ||||
|     template_name = "core/edit.jinja" | ||||
|     success_url = reverse_lazy("com:weekmail") | ||||
|     quick_notif_url_arg = "qn_weekmail_article_edit" | ||||
|     current_tab = "weekmail" | ||||
|  | ||||
|  | ||||
| class WeekmailArticleCreateView(QuickNotifMixin, CreateView): | ||||
| class WeekmailArticleCreateView(CreateView): | ||||
|     """Post an article.""" | ||||
|  | ||||
|     model = WeekmailArticle | ||||
| @@ -487,7 +491,6 @@ class WeekmailArticleCreateView(QuickNotifMixin, CreateView): | ||||
|     ) | ||||
|     template_name = "core/create.jinja" | ||||
|     success_url = reverse_lazy("core:user_tools") | ||||
|     quick_notif_url_arg = "qn_weekmail_new_article" | ||||
|  | ||||
|     def get_initial(self): | ||||
|         if "club" not in self.request.GET: | ||||
| @@ -558,161 +561,109 @@ class MailingModerateView(View): | ||||
|         raise PermissionDenied | ||||
|  | ||||
|  | ||||
| class PosterAdminViewMixin(IsComAdminMixin, ComTabsMixin): | ||||
|     current_tab = "posters" | ||||
|  | ||||
|  | ||||
| class PosterListBaseView(PosterAdminViewMixin, ListView): | ||||
| class PosterListBaseView(PermissionOrClubBoardRequiredMixin, ListView): | ||||
|     """List communication posters.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     model = Poster | ||||
|     template_name = "com/poster_list.jinja" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         club_id = kwargs.pop("club_id", None) | ||||
|         self.club = None | ||||
|         if club_id: | ||||
|             self.club = get_object_or_404(Club, pk=club_id) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_queryset(self): | ||||
|         if self.request.user.is_com_admin: | ||||
|             return Poster.objects.all().order_by("-date_begin") | ||||
|         else: | ||||
|             return Poster.objects.filter(club=self.club.id) | ||||
|     permission_required = "com.view_poster" | ||||
|     ordering = ["-date_begin"] | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         if not self.request.user.is_com_admin: | ||||
|             kwargs["club"] = self.club | ||||
|         return kwargs | ||||
|         return super().get_context_data(**kwargs) | {"club": self.club} | ||||
|  | ||||
|  | ||||
| class PosterCreateBaseView(PosterAdminViewMixin, CreateView): | ||||
| class PosterCreateBaseView(PermissionOrClubBoardRequiredMixin, CreateView): | ||||
|     """Create communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     form_class = PosterForm | ||||
|     template_name = "core/create.jinja" | ||||
|     permission_required = "com.add_poster" | ||||
|  | ||||
|     def get_queryset(self): | ||||
|         return Poster.objects.all() | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         if "club_id" in kwargs: | ||||
|             self.club = get_object_or_404(Club, pk=kwargs["club_id"]) | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         kwargs = super().get_form_kwargs() | ||||
|         kwargs.update({"user": self.request.user}) | ||||
|         return kwargs | ||||
|         return super().get_form_kwargs() | {"user": self.request.user} | ||||
|  | ||||
|     def get_initial(self): | ||||
|         return {"club": self.club} | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         if not self.request.user.is_com_admin: | ||||
|             kwargs["club"] = self.club | ||||
|         return kwargs | ||||
|         return super().get_context_data(**kwargs) | {"club": self.club} | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         if self.request.user.is_com_admin: | ||||
|         if self.request.user.has_perm("com.moderate_poster"): | ||||
|             form.instance.is_moderated = True | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|  | ||||
| class PosterEditBaseView(PosterAdminViewMixin, UpdateView): | ||||
| class PosterEditBaseView(PermissionOrClubBoardRequiredMixin, UpdateView): | ||||
|     """Edit communication poster.""" | ||||
|  | ||||
|     pk_url_kwarg = "poster_id" | ||||
|     current_tab = "posters" | ||||
|     form_class = PosterForm | ||||
|     template_name = "com/poster_edit.jinja" | ||||
|  | ||||
|     def get_initial(self): | ||||
|         return { | ||||
|             "date_begin": self.object.date_begin.strftime("%Y-%m-%d %H:%M:%S") | ||||
|             if self.object.date_begin | ||||
|             else None, | ||||
|             "date_end": self.object.date_end.strftime("%Y-%m-%d %H:%M:%S") | ||||
|             if self.object.date_end | ||||
|             else None, | ||||
|         } | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         if kwargs.get("club_id"): | ||||
|             try: | ||||
|                 self.club = Club.objects.get(pk=kwargs["club_id"]) | ||||
|             except Club.DoesNotExist as e: | ||||
|                 raise PermissionDenied from e | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|     permission_required = "com.change_poster" | ||||
|  | ||||
|     def get_queryset(self): | ||||
|         return Poster.objects.all() | ||||
|  | ||||
|     def get_form_kwargs(self): | ||||
|         kwargs = super().get_form_kwargs() | ||||
|         kwargs.update({"user": self.request.user}) | ||||
|         return kwargs | ||||
|         return super().get_form_kwargs() | {"user": self.request.user} | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         if hasattr(self, "club"): | ||||
|             kwargs["club"] = self.club | ||||
|         return kwargs | ||||
|         return super().get_context_data(**kwargs) | {"club": self.club} | ||||
|  | ||||
|     def form_valid(self, form): | ||||
|         if self.request.user.is_com_admin: | ||||
|         if not self.request.user.has_perm("com.moderate_poster"): | ||||
|             form.instance.is_moderated = False | ||||
|         return super().form_valid(form) | ||||
|  | ||||
|  | ||||
| class PosterDeleteBaseView(PosterAdminViewMixin, DeleteView): | ||||
| class PosterDeleteBaseView( | ||||
|     PermissionOrClubBoardRequiredMixin, ComTabsMixin, DeleteView | ||||
| ): | ||||
|     """Edit communication poster.""" | ||||
|  | ||||
|     pk_url_kwarg = "poster_id" | ||||
|     current_tab = "posters" | ||||
|     model = Poster | ||||
|     template_name = "core/delete_confirm.jinja" | ||||
|  | ||||
|     def dispatch(self, request, *args, **kwargs): | ||||
|         if kwargs.get("club_id"): | ||||
|             try: | ||||
|                 self.club = Club.objects.get(pk=kwargs["club_id"]) | ||||
|             except Club.DoesNotExist as e: | ||||
|                 raise PermissionDenied from e | ||||
|         return super().dispatch(request, *args, **kwargs) | ||||
|     permission_required = "com.delete_poster" | ||||
|  | ||||
|  | ||||
| class PosterListView(PosterListBaseView): | ||||
| class PosterListView(ComTabsMixin, PosterListBaseView): | ||||
|     """List communication posters.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|  | ||||
|     def get_queryset(self): | ||||
|         qs = super().get_queryset() | ||||
|         if self.request.user.has_perm("com.view_poster"): | ||||
|             return qs | ||||
|         return qs.filter(club=self.club.id) | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "com" | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class PosterCreateView(PosterCreateBaseView): | ||||
| class PosterCreateView(ComTabsMixin, PosterCreateBaseView): | ||||
|     """Create communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     success_url = reverse_lazy("com:poster_list") | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "com" | ||||
|         return kwargs | ||||
|     extra_context = {"app": "com"} | ||||
|  | ||||
|  | ||||
| class PosterEditView(PosterEditBaseView): | ||||
| class PosterEditView(ComTabsMixin, PosterEditBaseView): | ||||
|     """Edit communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     success_url = reverse_lazy("com:poster_list") | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "com" | ||||
|         return kwargs | ||||
|     extra_context = {"app": "com"} | ||||
|  | ||||
|  | ||||
| class PosterDeleteView(PosterDeleteBaseView): | ||||
| @@ -721,44 +672,39 @@ class PosterDeleteView(PosterDeleteBaseView): | ||||
|     success_url = reverse_lazy("com:poster_list") | ||||
|  | ||||
|  | ||||
| class PosterModerateListView(PosterAdminViewMixin, ListView): | ||||
| class PosterModerateListView(PermissionRequiredMixin, ComTabsMixin, ListView): | ||||
|     """Moderate list communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     model = Poster | ||||
|     template_name = "com/poster_moderate.jinja" | ||||
|     queryset = Poster.objects.filter(is_moderated=False).all() | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["app"] = "com" | ||||
|         return kwargs | ||||
|     permission_required = "com.moderate_poster" | ||||
|     extra_context = {"app": "com"} | ||||
|  | ||||
|  | ||||
| class PosterModerateView(PosterAdminViewMixin, View): | ||||
| class PosterModerateView(PermissionRequiredMixin, ComTabsMixin, View): | ||||
|     """Moderate communication poster.""" | ||||
|  | ||||
|     current_tab = "posters" | ||||
|     permission_required = "com.moderate_poster" | ||||
|     extra_context = {"app": "com"} | ||||
|  | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         obj = get_object_or_404(Poster, pk=kwargs["object_id"]) | ||||
|         if obj.can_be_moderated_by(request.user): | ||||
|             obj.is_moderated = True | ||||
|             obj.moderator = request.user | ||||
|             obj.save() | ||||
|             return redirect("com:poster_moderate_list") | ||||
|         raise PermissionDenied | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super(PosterModerateListView, self).get_context_data(**kwargs) | ||||
|         kwargs["app"] = "com" | ||||
|         return kwargs | ||||
|         obj.is_moderated = True | ||||
|         obj.moderator = request.user | ||||
|         obj.save() | ||||
|         return redirect("com:poster_moderate_list") | ||||
|  | ||||
|  | ||||
| class ScreenListView(IsComAdminMixin, ComTabsMixin, ListView): | ||||
| class ScreenListView(PermissionRequiredMixin, ComTabsMixin, ListView): | ||||
|     """List communication screens.""" | ||||
|  | ||||
|     current_tab = "screens" | ||||
|     model = Screen | ||||
|     template_name = "com/screen_list.jinja" | ||||
|     permission_required = "com.view_screen" | ||||
|  | ||||
|  | ||||
| class ScreenSlideshowView(DetailView): | ||||
| @@ -769,12 +715,12 @@ class ScreenSlideshowView(DetailView): | ||||
|     template_name = "com/screen_slideshow.jinja" | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["posters"] = self.object.active_posters() | ||||
|         return kwargs | ||||
|         return super().get_context_data(**kwargs) | { | ||||
|             "posters": self.object.active_posters() | ||||
|         } | ||||
|  | ||||
|  | ||||
| class ScreenCreateView(IsComAdminMixin, ComTabsMixin, CreateView): | ||||
| class ScreenCreateView(PermissionRequiredMixin, ComTabsMixin, CreateView): | ||||
|     """Create communication screen.""" | ||||
|  | ||||
|     current_tab = "screens" | ||||
| @@ -782,9 +728,10 @@ class ScreenCreateView(IsComAdminMixin, ComTabsMixin, CreateView): | ||||
|     fields = ["name"] | ||||
|     template_name = "core/create.jinja" | ||||
|     success_url = reverse_lazy("com:screen_list") | ||||
|     permission_required = "com.add_screen" | ||||
|  | ||||
|  | ||||
| class ScreenEditView(IsComAdminMixin, ComTabsMixin, UpdateView): | ||||
| class ScreenEditView(PermissionRequiredMixin, ComTabsMixin, UpdateView): | ||||
|     """Edit communication screen.""" | ||||
|  | ||||
|     pk_url_kwarg = "screen_id" | ||||
| @@ -793,9 +740,10 @@ class ScreenEditView(IsComAdminMixin, ComTabsMixin, UpdateView): | ||||
|     fields = ["name"] | ||||
|     template_name = "com/screen_edit.jinja" | ||||
|     success_url = reverse_lazy("com:screen_list") | ||||
|     permission_required = "com.change_screen" | ||||
|  | ||||
|  | ||||
| class ScreenDeleteView(IsComAdminMixin, ComTabsMixin, DeleteView): | ||||
| class ScreenDeleteView(PermissionRequiredMixin, ComTabsMixin, DeleteView): | ||||
|     """Delete communication screen.""" | ||||
|  | ||||
|     pk_url_kwarg = "screen_id" | ||||
| @@ -803,3 +751,4 @@ class ScreenDeleteView(IsComAdminMixin, ComTabsMixin, DeleteView): | ||||
|     model = Screen | ||||
|     template_name = "core/delete_confirm.jinja" | ||||
|     success_url = reverse_lazy("com:screen_list") | ||||
|     permission_required = "com.delete_screen" | ||||
|   | ||||
							
								
								
									
										11
									
								
								core/api.py
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								core/api.py
									
									
									
									
									
								
							| @@ -25,6 +25,7 @@ from core.schemas import ( | ||||
|     UserFamilySchema, | ||||
|     UserFilterSchema, | ||||
|     UserProfileSchema, | ||||
|     UserSchema, | ||||
| ) | ||||
| from core.templatetags.renderer import markdown | ||||
|  | ||||
| @@ -69,16 +70,22 @@ class MailingListController(ControllerBase): | ||||
|         return data | ||||
|  | ||||
|  | ||||
| @api_controller("/user", permissions=[CanAccessLookup]) | ||||
| @api_controller("/user") | ||||
| class UserController(ControllerBase): | ||||
|     @route.get("", response=list[UserProfileSchema]) | ||||
|     @route.get("", response=list[UserProfileSchema], permissions=[CanAccessLookup]) | ||||
|     def fetch_profiles(self, pks: Query[set[int]]): | ||||
|         return User.objects.filter(pk__in=pks) | ||||
|  | ||||
|     @route.get("/{int:user_id}", response=UserSchema, permissions=[CanView]) | ||||
|     def fetch_user(self, user_id: int): | ||||
|         """Fetch a single user""" | ||||
|         return self.get_object_or_exception(User, id=user_id) | ||||
|  | ||||
|     @route.get( | ||||
|         "/search", | ||||
|         response=PaginatedResponseSchema[UserProfileSchema], | ||||
|         url_name="search_users", | ||||
|         permissions=[CanAccessLookup], | ||||
|     ) | ||||
|     @paginate(PageNumberPaginationExtra, page_size=20) | ||||
|     def search_users(self, filters: Query[UserFilterSchema]): | ||||
|   | ||||
| @@ -29,8 +29,14 @@ from typing import TYPE_CHECKING, Any, LiteralString | ||||
|  | ||||
| from django.contrib.auth.mixins import AccessMixin, PermissionRequiredMixin | ||||
| from django.core.exceptions import ImproperlyConfigured, PermissionDenied | ||||
| from django.http import Http404 | ||||
| from django.shortcuts import get_object_or_404 | ||||
| from django.utils.functional import cached_property | ||||
| from django.utils.translation import gettext as _ | ||||
| from django.views.generic.base import View | ||||
|  | ||||
| from club.models import Club | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from django.db.models import Model | ||||
|  | ||||
| @@ -297,3 +303,50 @@ class PermissionOrAuthorRequiredMixin(PermissionRequiredMixin): | ||||
|             self.author_field += "_id" | ||||
|         author_id = getattr(obj, self.author_field, None) | ||||
|         return author_id == self.request.user.id | ||||
|  | ||||
|  | ||||
| class PermissionOrClubBoardRequiredMixin(PermissionRequiredMixin): | ||||
|     """Require that the user has the required perm or is the board of the club. | ||||
|  | ||||
|     This mixin can be used in any view that is called from a url | ||||
|     having a `club_id` kwarg. | ||||
|  | ||||
|     Example: | ||||
|  | ||||
|         In `urls.py` : | ||||
|         ```python | ||||
|         urlpatterns = [ | ||||
|             path("foo/<int:club_id>/bar/", FooView.as_view()) | ||||
|         ] | ||||
|         ``` | ||||
|  | ||||
|         In `views.py` : | ||||
|  | ||||
|         ```python | ||||
|         # this view is available to users that either have the | ||||
|         # "foo.view_foo" permission or are in the board of the club | ||||
|         # which id was given in the url | ||||
|         class FooView(PermissionOrClubBoardRequiredMixin, View): | ||||
|             permission_required = "foo.view_foo" | ||||
|         ``` | ||||
|     """ | ||||
|  | ||||
|     club_pk_url_kwarg = "club_id" | ||||
|  | ||||
|     @cached_property | ||||
|     def club(self): | ||||
|         club_id: str | int = self.kwargs.pop(self.club_pk_url_kwarg, None) | ||||
|         if club_id is None: | ||||
|             return None | ||||
|         if isinstance(club_id, int) or club_id.isdigit(): | ||||
|             return get_object_or_404(Club, pk=club_id) | ||||
|         raise Http404(_("No club found with id %(id)s") % {"id": club_id}) | ||||
|  | ||||
|     def has_permission(self): | ||||
|         if self.request.user.is_anonymous: | ||||
|             return False | ||||
|         if super().has_permission(): | ||||
|             return True | ||||
|         return self.club is not None and any( | ||||
|             g.id == self.club.board_group_id for g in self.request.user.cached_groups | ||||
|         ) | ||||
|   | ||||
| @@ -768,7 +768,7 @@ class Command(BaseCommand): | ||||
|         s = Subscription( | ||||
|             member=user, | ||||
|             subscription_type=subscription_type, | ||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0][0], | ||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1][0], | ||||
|         ) | ||||
|         s.subscription_start = s.compute_start(start) | ||||
|         s.subscription_end = s.compute_end( | ||||
|   | ||||
| @@ -94,7 +94,11 @@ class Command(BaseCommand): | ||||
|                 username=self.faker.user_name(), | ||||
|                 first_name=self.faker.first_name(), | ||||
|                 last_name=self.faker.last_name(), | ||||
|                 date_of_birth=self.faker.date_of_birth(minimum_age=15, maximum_age=25), | ||||
|                 date_of_birth=( | ||||
|                     None | ||||
|                     if random.random() < 0.2 | ||||
|                     else self.faker.date_of_birth(minimum_age=15, maximum_age=25) | ||||
|                 ), | ||||
|                 email=self.faker.email(), | ||||
|                 phone=self.faker.phone_number(), | ||||
|                 address=self.faker.address(), | ||||
|   | ||||
| @@ -560,7 +560,7 @@ class User(AbstractUser): | ||||
|         """Determine if the object is owned by the user.""" | ||||
|         if hasattr(obj, "is_owned_by") and obj.is_owned_by(self): | ||||
|             return True | ||||
|         if hasattr(obj, "owner_group") and self.is_in_group(pk=obj.owner_group.id): | ||||
|         if hasattr(obj, "owner_group") and self.is_in_group(pk=obj.owner_group_id): | ||||
|             return True | ||||
|         return self.is_root | ||||
|  | ||||
| @@ -569,9 +569,15 @@ class User(AbstractUser): | ||||
|         if hasattr(obj, "can_be_edited_by") and obj.can_be_edited_by(self): | ||||
|             return True | ||||
|         if hasattr(obj, "edit_groups"): | ||||
|             for pk in obj.edit_groups.values_list("pk", flat=True): | ||||
|                 if self.is_in_group(pk=pk): | ||||
|                     return True | ||||
|             if ( | ||||
|                 hasattr(obj, "_prefetched_objects_cache") | ||||
|                 and "edit_groups" in obj._prefetched_objects_cache | ||||
|             ): | ||||
|                 pks = [g.id for g in obj.edit_groups.all()] | ||||
|             else: | ||||
|                 pks = list(obj.edit_groups.values_list("id", flat=True)) | ||||
|             if any(self.is_in_group(pk=pk) for pk in pks): | ||||
|                 return True | ||||
|         if isinstance(obj, User) and obj == self: | ||||
|             return True | ||||
|         return self.is_owner(obj) | ||||
| @@ -581,9 +587,18 @@ class User(AbstractUser): | ||||
|         if hasattr(obj, "can_be_viewed_by") and obj.can_be_viewed_by(self): | ||||
|             return True | ||||
|         if hasattr(obj, "view_groups"): | ||||
|             for pk in obj.view_groups.values_list("pk", flat=True): | ||||
|                 if self.is_in_group(pk=pk): | ||||
|                     return True | ||||
|             # if "view_groups" has already been prefetched, use | ||||
|             # the prefetch cache, else fetch only the ids, to make | ||||
|             # the query lighter. | ||||
|             if ( | ||||
|                 hasattr(obj, "_prefetched_objects_cache") | ||||
|                 and "view_groups" in obj._prefetched_objects_cache | ||||
|             ): | ||||
|                 pks = [g.id for g in obj.view_groups.all()] | ||||
|             else: | ||||
|                 pks = list(obj.view_groups.values_list("id", flat=True)) | ||||
|             if any(self.is_in_group(pk=pk) for pk in pks): | ||||
|                 return True | ||||
|         return self.can_edit(obj) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
| @@ -636,9 +651,6 @@ class User(AbstractUser): | ||||
|  | ||||
|  | ||||
| class AnonymousUser(AuthAnonymousUser): | ||||
|     def __init__(self): | ||||
|         super().__init__() | ||||
|  | ||||
|     @property | ||||
|     def was_subscribed(self): | ||||
|         return False | ||||
| @@ -647,10 +659,6 @@ class AnonymousUser(AuthAnonymousUser): | ||||
|     def is_subscribed(self): | ||||
|         return False | ||||
|  | ||||
|     @property | ||||
|     def subscribed(self): | ||||
|         return False | ||||
|  | ||||
|     @property | ||||
|     def is_root(self): | ||||
|         return False | ||||
| @@ -1182,6 +1190,18 @@ class NotLocked(LockError): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class PageQuerySet(models.QuerySet): | ||||
|     def viewable_by(self, user: User) -> Self: | ||||
|         if user.is_anonymous: | ||||
|             return self.filter(view_groups=settings.SITH_GROUP_PUBLIC_ID) | ||||
|         if user.has_perm("core.view_page"): | ||||
|             return self.all() | ||||
|         groups_ids = [g.id for g in user.cached_groups] | ||||
|         if user.is_subscribed: | ||||
|             groups_ids.append(settings.SITH_GROUP_SUBSCRIBERS_ID) | ||||
|         return self.filter(view_groups__in=groups_ids) | ||||
|  | ||||
|  | ||||
| # This function prevents generating migration upon settings change | ||||
| def get_default_owner_group(): | ||||
|     return settings.SITH_GROUP_ROOT_ID | ||||
| @@ -1251,6 +1271,8 @@ class Page(models.Model): | ||||
|         _("lock_timeout"), null=True, blank=True, default=None | ||||
|     ) | ||||
|  | ||||
|     objects = PageQuerySet.as_manager() | ||||
|  | ||||
|     class Meta: | ||||
|         unique_together = ("name", "parent") | ||||
|         permissions = ( | ||||
| @@ -1260,12 +1282,9 @@ class Page(models.Model): | ||||
|     def __str__(self): | ||||
|         return self.get_full_name() | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|     def save(self, *args, force_lock: bool = False, **kwargs): | ||||
|         """Performs some needed actions before and after saving a page in database.""" | ||||
|         locked = kwargs.pop("force_lock", False) | ||||
|         if not locked: | ||||
|             locked = self.is_locked() | ||||
|         if not locked: | ||||
|         if not force_lock and not self.is_locked(): | ||||
|             raise NotLocked("The page is not locked and thus can not be saved") | ||||
|         self.full_clean() | ||||
|         if not self.id: | ||||
| @@ -1277,7 +1296,7 @@ class Page(models.Model): | ||||
|         # It also update all the children to maintain correct names | ||||
|         self._full_name = self.get_full_name() | ||||
|         for c in self.children.all(): | ||||
|             c.save() | ||||
|             c.save(force_lock=force_lock) | ||||
|         super().save(*args, **kwargs) | ||||
|         self.unset_lock() | ||||
|  | ||||
| @@ -1384,23 +1403,23 @@ class Page(models.Model): | ||||
|  | ||||
|     @cached_property | ||||
|     def is_club_page(self): | ||||
|         club_root_page = Page.objects.filter(name=settings.SITH_CLUB_ROOT_PAGE).first() | ||||
|         return club_root_page is not None and ( | ||||
|             self == club_root_page or club_root_page in self.get_parent_list() | ||||
|         return ( | ||||
|             self.name == settings.SITH_CLUB_ROOT_PAGE | ||||
|             or settings.SITH_CLUB_ROOT_PAGE in [p.name for p in self.get_parent_list()] | ||||
|         ) | ||||
|  | ||||
|     @cached_property | ||||
|     def need_club_redirection(self): | ||||
|         return self.is_club_page and self.name != settings.SITH_CLUB_ROOT_PAGE | ||||
|  | ||||
|     def delete(self): | ||||
|     def delete(self, *args, **kwargs): | ||||
|         self.unset_lock_recursive() | ||||
|         self.set_lock_recursive(User.objects.get(id=0)) | ||||
|         for child in self.children.all(): | ||||
|             child.parent = self.parent | ||||
|             child.save() | ||||
|             child.unset_lock_recursive() | ||||
|         super().delete() | ||||
|         return super().delete(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| class PageRev(models.Model): | ||||
| @@ -1447,9 +1466,12 @@ class PageRev(models.Model): | ||||
|     def get_absolute_url(self): | ||||
|         return reverse("core:page", kwargs={"page_name": self.page._full_name}) | ||||
|  | ||||
|     def can_be_edited_by(self, user): | ||||
|     def can_be_edited_by(self, user: User) -> bool: | ||||
|         return self.page.can_be_edited_by(user) | ||||
|  | ||||
|     def is_owned_by(self, user: User) -> bool: | ||||
|         return any(g.id == self.page.owner_group_id for g in user.cached_groups) | ||||
|  | ||||
|  | ||||
| def get_notification_types(): | ||||
|     return settings.SITH_NOTIFICATIONS | ||||
|   | ||||
| @@ -34,6 +34,22 @@ class SimpleUserSchema(ModelSchema): | ||||
|         fields = ["id", "nick_name", "first_name", "last_name"] | ||||
|  | ||||
|  | ||||
| class UserSchema(ModelSchema): | ||||
|     class Meta: | ||||
|         model = User | ||||
|         fields = [ | ||||
|             "id", | ||||
|             "nick_name", | ||||
|             "first_name", | ||||
|             "last_name", | ||||
|             "date_of_birth", | ||||
|             "email", | ||||
|             "role", | ||||
|             "quote", | ||||
|             "promo", | ||||
|         ] | ||||
|  | ||||
|  | ||||
| class UserProfileSchema(ModelSchema): | ||||
|     """The necessary information to show a user profile""" | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,9 @@ | ||||
| import { alpinePlugin as notificationPlugin } from "#core:utils/notifications"; | ||||
| import sort from "@alpinejs/sort"; | ||||
| import Alpine from "alpinejs"; | ||||
|  | ||||
| Alpine.plugin(sort); | ||||
| Alpine.magic("notifications", notificationPlugin); | ||||
| window.Alpine = Alpine; | ||||
|  | ||||
| window.addEventListener("DOMContentLoaded", () => { | ||||
|   | ||||
							
								
								
									
										36
									
								
								core/static/bundled/utils/notifications.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								core/static/bundled/utils/notifications.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| export enum NotificationLevel { | ||||
|   Error = "error", | ||||
|   Warning = "warning", | ||||
|   Success = "success", | ||||
| } | ||||
|  | ||||
| export function createNotification(message: string, level: NotificationLevel) { | ||||
|   const element = document.getElementById("quick-notifications"); | ||||
|   if (element === null) { | ||||
|     return false; | ||||
|   } | ||||
|   return element.dispatchEvent( | ||||
|     new CustomEvent("quick-notification-add", { | ||||
|       detail: { text: message, tag: level }, | ||||
|     }), | ||||
|   ); | ||||
| } | ||||
|  | ||||
| export function deleteNotifications() { | ||||
|   const element = document.getElementById("quick-notifications"); | ||||
|   if (element === null) { | ||||
|     return false; | ||||
|   } | ||||
|   return element.dispatchEvent(new CustomEvent("quick-notification-delete")); | ||||
| } | ||||
|  | ||||
| export function alpinePlugin() { | ||||
|   return { | ||||
|     error: (message: string) => createNotification(message, NotificationLevel.Error), | ||||
|     warning: (message: string) => | ||||
|       createNotification(message, NotificationLevel.Warning), | ||||
|     success: (message: string) => | ||||
|       createNotification(message, NotificationLevel.Success), | ||||
|     clear: () => deleteNotifications(), | ||||
|   }; | ||||
| } | ||||
| @@ -36,6 +36,7 @@ | ||||
|   > .ts-control { | ||||
|     box-shadow: none; | ||||
|     max-width: 300px; | ||||
|     width: 300px; | ||||
|     background-color: var(--nf-input-background-color); | ||||
|  | ||||
|     &::after { | ||||
|   | ||||
| @@ -47,6 +47,7 @@ | ||||
|   } | ||||
|  | ||||
|   input, | ||||
|   select, | ||||
|   textarea[type="text"], | ||||
|   [type="number"], | ||||
|   .ts-control { | ||||
| @@ -153,10 +154,8 @@ form { | ||||
|     margin-bottom: 1rem; | ||||
|   } | ||||
|  | ||||
|   .row { | ||||
|     label { | ||||
|       margin: unset; | ||||
|     } | ||||
|   .row > label { | ||||
|     margin: unset; | ||||
|   } | ||||
|  | ||||
|   // ------------- LABEL | ||||
| @@ -240,6 +239,23 @@ form { | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   input[type="text"], | ||||
|   input[type="email"], | ||||
|   input[type="tel"], | ||||
|   input[type="url"], | ||||
|   input[type="password"], | ||||
|   input[type="number"], | ||||
|   input[type="date"], | ||||
|   input[type="datetime-local"], | ||||
|   input[type="week"], | ||||
|   input[type="time"], | ||||
|   input[type="month"], | ||||
|   input[type="search"], | ||||
|   textarea, | ||||
|   select, | ||||
|   .ts-control { | ||||
|     min-height: calc(var(--nf-input-size) * 2.5); | ||||
|   } | ||||
|  | ||||
|   input[type="text"], | ||||
|   input[type="checkbox"], | ||||
|   | ||||
| @@ -321,7 +321,6 @@ $hovered-red-text-color: #ff4d4d; | ||||
|  | ||||
|         >#header_notif { | ||||
|           box-sizing: border-box; | ||||
|           display: none; | ||||
|           position: absolute; | ||||
|           margin: 0; | ||||
|           background-color: whitesmoke; | ||||
|   | ||||
| @@ -1,38 +0,0 @@ | ||||
| $(() => { | ||||
|   $("#quick_notif li").click(function () { | ||||
|     $(this).hide(); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
| // biome-ignore lint/correctness/noUnusedVariables: used in other scripts | ||||
| function createQuickNotif(msg) { | ||||
|   const el = document.createElement("li"); | ||||
|   el.textContent = msg; | ||||
|   el.addEventListener("click", () => el.parentNode.removeChild(el)); | ||||
|   document.getElementById("quick_notif").appendChild(el); | ||||
| } | ||||
|  | ||||
| // biome-ignore lint/correctness/noUnusedVariables: used in other scripts | ||||
| function deleteQuickNotifs() { | ||||
|   const el = document.getElementById("quick_notif"); | ||||
|   while (el.firstChild) { | ||||
|     el.removeChild(el.firstChild); | ||||
|   } | ||||
| } | ||||
|  | ||||
| // biome-ignore lint/correctness/noUnusedVariables: used in other scripts | ||||
| function displayNotif() { | ||||
|   $("#header_notif").toggle().parent().toggleClass("white"); | ||||
| } | ||||
|  | ||||
| // You can't get the csrf token from the template in a widget | ||||
| // We get it from a cookie as a workaround, see this link | ||||
| // https://docs.djangoproject.com/en/2.0/ref/csrf/#ajax | ||||
| // Sadly, getting the cookie is not possible with CSRF_COOKIE_HTTPONLY or CSRF_USE_SESSIONS is True | ||||
| // So, the true workaround is to get the token from the dom | ||||
| // https://docs.djangoproject.com/en/2.0/ref/csrf/#acquiring-the-token-if-csrf-use-sessions-is-true | ||||
| // biome-ignore lint/style/useNamingConvention: can't find it used anywhere but I will not play with the devil | ||||
| // biome-ignore lint/correctness/noUnusedVariables: used in other scripts | ||||
| function getCSRFToken() { | ||||
|   return $("[name=csrfmiddlewaretoken]").val(); | ||||
| } | ||||
| @@ -270,17 +270,6 @@ body { | ||||
|   } | ||||
|  | ||||
|   /*--------------------------------CONTENT------------------------------*/ | ||||
|   #quick_notif { | ||||
|     width: 100%; | ||||
|     margin: 0 auto; | ||||
|     list-style-type: none; | ||||
|     background: $second-color; | ||||
|  | ||||
|     li { | ||||
|       padding: 10px; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   #content { | ||||
|     padding: 1em 1%; | ||||
|     box-shadow: $shadow-color 0 5px 10px; | ||||
| @@ -514,9 +503,17 @@ th { | ||||
|   text-align: center; | ||||
|   padding: 5px 10px; | ||||
|  | ||||
|   >input[type="checkbox"] { | ||||
|     padding: unset; | ||||
|   } | ||||
|  | ||||
|   >ul { | ||||
|     margin-top: 0; | ||||
|   } | ||||
|  | ||||
|   >input[type="checkbox"] { | ||||
|     padding: unset; | ||||
|   } | ||||
| } | ||||
|  | ||||
| td { | ||||
|   | ||||
| @@ -32,10 +32,6 @@ | ||||
|       <script type="module" src="{{ static('bundled/country-flags-index.ts') }}"></script> | ||||
|       <script type="module" src="{{ static('bundled/core/tooltips-index.ts') }}"></script> | ||||
|  | ||||
|       <!-- Jquery declared here to be accessible in every django widgets --> | ||||
|       <script src="{{ static('bundled/vendored/jquery.min.js') }}"></script> | ||||
|       <script src="{{ static('core/js/script.js') }}"></script> | ||||
|  | ||||
|       {% block additional_css %}{% endblock %} | ||||
|       {% block additional_js %}{% endblock %} | ||||
|     {% endblock %} | ||||
| @@ -74,17 +70,15 @@ | ||||
|  | ||||
|     <div id="page"> | ||||
|  | ||||
|       <ul id="quick_notif"> | ||||
|         {% for n in quick_notifs %} | ||||
|           <li>{{ n }}</li> | ||||
|         {% endfor %} | ||||
|       </ul> | ||||
|  | ||||
|       <div id="content"> | ||||
|         {%- block tabs -%} | ||||
|           {% include "core/base/tabs.jinja" %} | ||||
|         {%- endblock -%} | ||||
|  | ||||
|         {% block notifications %} | ||||
|           {% include "core/base/notifications.jinja" %} | ||||
|         {% endblock %} | ||||
|  | ||||
|         {%- block errors -%} | ||||
|           {% if error %} | ||||
|             {{ error }} | ||||
| @@ -101,16 +95,6 @@ | ||||
|     {% endblock %} | ||||
|  | ||||
|     {% block script %} | ||||
|       <script> | ||||
|         document.addEventListener("keydown", (e) => { | ||||
|           // Looking at the `s` key when not typing in a form | ||||
|           if (e.keyCode !== 83 || ["INPUT", "TEXTAREA", "SELECT"].includes(e.target.nodeName)) { | ||||
|             return; | ||||
|           } | ||||
|           document.getElementById("search").focus(); | ||||
|           e.preventDefault(); // Don't type the character in the focused search input | ||||
|         }) | ||||
|       </script> | ||||
|     {% endblock %} | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -74,25 +74,25 @@ | ||||
|             {% endif %} | ||||
|           ></a> | ||||
|         </div> | ||||
|         <div class="notification"> | ||||
|           <a href="#" onclick="displayNotif()"> | ||||
|             <i class="fa-regular fa-bell"></i> | ||||
|             {% set notification_count = user.notifications.filter(viewed=False).count() %} | ||||
|         <div class="notification" x-data="{display: false}" :class="{white: display}"> | ||||
|           <a href="#" @click.prevent="display = !display"> | ||||
|             <i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i> | ||||
|             {% set notifications = user.notifications.filter(viewed=False).order_by("-date")|list %} | ||||
|  | ||||
|             {% if notification_count > 0 %} | ||||
|             {%- if notifications|length > 0 -%} | ||||
|               <span> | ||||
|                 {% if notification_count < 100 %} | ||||
|                   {{ notification_count }} | ||||
|                 {% else %} | ||||
|                     | ||||
|                 {% endif %} | ||||
|                 {% if notifications|length < 100 %} | ||||
|                   {{ notifications|length }} | ||||
|                 {%- else -%} | ||||
|                   99+ | ||||
|                 {%- endif -%} | ||||
|               </span> | ||||
|             {% endif %} | ||||
|           </a> | ||||
|           <div id="header_notif"> | ||||
|           <div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false"> | ||||
|             <ul> | ||||
|               {% if user.notifications.filter(viewed=False).count() > 0 %} | ||||
|                 {% for n in user.notifications.filter(viewed=False).order_by('-date') %} | ||||
|               {%- if notifications|length > 0 -%} | ||||
|                 {%- for n in notifications -%} | ||||
|                   <li> | ||||
|                     <a href="{{ url("core:notification", notif_id=n.id) }}"> | ||||
|                       <div class="datetime"> | ||||
| @@ -108,10 +108,10 @@ | ||||
|                       </div> | ||||
|                     </a> | ||||
|                   </li> | ||||
|                 {% endfor %} | ||||
|               {% else %} | ||||
|                 {%- endfor -%} | ||||
|               {%- else -%} | ||||
|                 <li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li> | ||||
|               {% endif %} | ||||
|               {%- endif -%} | ||||
|             </ul> | ||||
|             <div class="options"> | ||||
|               <a href="{{ url('core:notification_list') }}"> | ||||
|   | ||||
							
								
								
									
										24
									
								
								core/templates/core/base/notifications.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								core/templates/core/base/notifications.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| <div id="quick-notifications" | ||||
|      x-data="{ | ||||
|              messages: [ | ||||
|              {% if messages %} | ||||
|                {% for message in messages %} | ||||
|                  { | ||||
|                  tag: '{{ message.tags }}', | ||||
|                  text: '{{ message }}', | ||||
|                  }, | ||||
|                {% endfor %} | ||||
|              {% endif %} | ||||
|              ] | ||||
|              }" | ||||
|      @quick-notification-add="(e) => messages.push(e?.detail)" | ||||
|      @quick-notification-delete="messages = []"> | ||||
|   <template x-for="(message, index) in messages"> | ||||
|     <div class="alert" :class="`alert-${message.tag}`" x-transition> | ||||
|       <span class="alert-main" x-text="message.text"></span> | ||||
|       <span class="clickable" @click="messages = messages.filter((item, i) => i !== index)"> | ||||
|         <i class="fa fa-close"></i> | ||||
|       </span> | ||||
|     </div> | ||||
|   </template> | ||||
| </div> | ||||
| @@ -15,6 +15,7 @@ | ||||
|       {{ select_all_checkbox("add_users") }} | ||||
|       <hr> | ||||
|       {% csrf_token %} | ||||
|       {{ form.non_field_errors() }} | ||||
|       <label for="{{ form.users_removed.id_for_label }}">{{ form.users_removed.label }} :</label> | ||||
|       {{ form.users_removed.errors }} | ||||
|       {% for user in form.users_removed %} | ||||
|   | ||||
| @@ -245,3 +245,26 @@ | ||||
|   <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> | ||||
| {% 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 %} | ||||
|   | ||||
| @@ -5,16 +5,12 @@ | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|   {% if page_list %} | ||||
|     <h3>{% trans %}Page list{% endtrans %}</h3> | ||||
|     <ul> | ||||
|       {% for p in page_list %} | ||||
|         <li><a href="{{ p.get_absolute_url() }}">{{ p.get_display_name() }}</a></li> | ||||
|       {% endfor %} | ||||
|     </ul> | ||||
|   {% else %} | ||||
|     {% trans %}There is no page in this website.{% endtrans %} | ||||
|   {% endif %} | ||||
|   <h3>{% trans %}Page list{% endtrans %}</h3> | ||||
|   <ul> | ||||
|     {% for p in page_list %} | ||||
|       <li><a href="{{ p.get_absolute_url() }}">{{ p.display_name }}</a></li> | ||||
|     {% endfor %} | ||||
|   </ul> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -30,7 +30,11 @@ | ||||
|               - {{ purchase.date|localtime|time(DATETIME_FORMAT) }} | ||||
|             </td> | ||||
|             <td>{{ purchase.counter }}</td> | ||||
|             <td><a href="{{ purchase.seller.get_absolute_url() }}">{{ purchase.seller.get_display_name() }}</a></td> | ||||
|             {% if not purchase.seller %} | ||||
|               <td>{% trans %}Deleted user{% endtrans %}</td> | ||||
|             {% else %} | ||||
|               <td><a href="{{ purchase.seller.get_absolute_url() }}">{{ purchase.seller.get_display_name() }}</a></td> | ||||
|             {% endif %} | ||||
|             <td>{{ purchase.label }}</td> | ||||
|             <td>{{ purchase.quantity }}</td> | ||||
|             <td>{{ purchase.quantity * purchase.unit_price }} €</td> | ||||
|   | ||||
| @@ -1,23 +1,25 @@ | ||||
| {% for js in statics.js %} | ||||
|   <script-once type="module" src="{{ js }}"></script-once> | ||||
| {% endfor %} | ||||
| {% for css in statics.css %} | ||||
|   <link-once rel="stylesheet" type="text/css" href="{{ css }}" defer></link-once> | ||||
| {% endfor %} | ||||
|  | ||||
| <{{ component }} name="{{ widget.name }}" {% include "django/forms/widgets/attrs.html" %}> | ||||
| {% for group_name, group_choices, group_index in widget.optgroups %} | ||||
|   {% if group_name %} | ||||
|     <optgroup label="{{ group_name }}"> | ||||
|   {% endif %} | ||||
|   {% for widget in group_choices %} | ||||
|     {% include widget.template_name %} | ||||
| {% spaceless %} | ||||
|   {% for js in statics.js %} | ||||
|     <script-once type="module" src="{{ js }}"></script-once> | ||||
|   {% endfor %} | ||||
|   {% if group_name %} | ||||
|     </optgroup> | ||||
|   {% for css in statics.css %} | ||||
|     <link-once rel="stylesheet" type="text/css" href="{{ css }}" defer></link-once> | ||||
|   {% endfor %} | ||||
|  | ||||
|   <{{ component }} name="{{ widget.name }}" {% include "django/forms/widgets/attrs.html" %}> | ||||
|   {% for group_name, group_choices, group_index in widget.optgroups %} | ||||
|     {% if group_name %} | ||||
|       <optgroup label="{{ group_name }}"> | ||||
|     {% endif %} | ||||
|     {% for widget in group_choices %} | ||||
|       {% include widget.template_name %} | ||||
|     {% endfor %} | ||||
|     {% if group_name %} | ||||
|       </optgroup> | ||||
|     {% endif %} | ||||
|   {% endfor %} | ||||
|   {% if initial %} | ||||
|     <slot style="display:none" name="initial">{{ initial }}</slot> | ||||
|   {% endif %} | ||||
| {% endfor %} | ||||
| {% if initial %} | ||||
|   <slot style="display:none" name="initial">{{ initial }}</slot> | ||||
| {% endif %} | ||||
| </{{ component }}> | ||||
|   </{{ component }}> | ||||
| {% endspaceless %} | ||||
							
								
								
									
										58
									
								
								core/tests/test_page.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								core/tests/test_page.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| import pytest | ||||
| from django.conf import settings | ||||
| 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 assertRedirects | ||||
|  | ||||
| from core.baker_recipes import board_user, subscriber_user | ||||
| from core.models import AnonymousUser, Page, User | ||||
| from sith.settings import SITH_GROUP_OLD_SUBSCRIBERS_ID, SITH_GROUP_SUBSCRIBERS_ID | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_edit_page(client: Client): | ||||
|     user = board_user.make() | ||||
|     page = baker.prepare(Page) | ||||
|     page.save(force_lock=True) | ||||
|     page.view_groups.add(user.groups.first()) | ||||
|     client.force_login(user) | ||||
|  | ||||
|     url = reverse("core:page_edit", kwargs={"page_name": page._full_name}) | ||||
|     res = client.get(url) | ||||
|     assert res.status_code == 200 | ||||
|  | ||||
|     res = client.post(url, data={"content": "Hello World"}) | ||||
|     assertRedirects(res, reverse("core:page", kwargs={"page_name": page._full_name})) | ||||
|     revision = page.revisions.last() | ||||
|     assert revision.content == "Hello World" | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_viewable_by(): | ||||
|     # remove existing pages to prevent side effect | ||||
|     Page.objects.all().delete() | ||||
|     view_groups = [ | ||||
|         [settings.SITH_GROUP_PUBLIC_ID], | ||||
|         [settings.SITH_GROUP_PUBLIC_ID, SITH_GROUP_SUBSCRIBERS_ID], | ||||
|         [SITH_GROUP_SUBSCRIBERS_ID], | ||||
|         [SITH_GROUP_SUBSCRIBERS_ID, SITH_GROUP_OLD_SUBSCRIBERS_ID], | ||||
|         [], | ||||
|     ] | ||||
|     pages = baker.make(Page, _quantity=len(view_groups), _bulk_create=True) | ||||
|     for page, groups in zip(pages, view_groups, strict=True): | ||||
|         page.view_groups.set(groups) | ||||
|  | ||||
|     viewable = Page.objects.viewable_by(AnonymousUser()).values_list("id", flat=True) | ||||
|     assert set(viewable) == {pages[0].id, pages[1].id} | ||||
|  | ||||
|     subscriber = subscriber_user.make() | ||||
|     viewable = Page.objects.viewable_by(subscriber).values_list("id", flat=True) | ||||
|     assert set(viewable) == {p.id for p in pages[0:4]} | ||||
|  | ||||
|     root_user = baker.make( | ||||
|         User, user_permissions=[Permission.objects.get(codename="view_page")] | ||||
|     ) | ||||
|     viewable = Page.objects.viewable_by(root_user).values_list("id", flat=True) | ||||
|     assert set(viewable) == {p.id for p in pages} | ||||
| @@ -20,7 +20,8 @@ from core.baker_recipes import ( | ||||
| ) | ||||
| from core.models import Group, User | ||||
| from core.views import UserTabsMixin | ||||
| from counter.models import Counter, Refilling, Selling | ||||
| from counter.baker_recipes import sale_recipe | ||||
| from counter.models import Counter, Customer, Refilling, Selling | ||||
| from eboutic.models import Invoice, InvoiceItem | ||||
|  | ||||
|  | ||||
| @@ -129,6 +130,31 @@ def test_user_account_not_found(client: Client): | ||||
|     assert res.status_code == 404 | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_is_deleted_barman_shown_as_deleted(client: Client): | ||||
|     customer = baker.make(Customer) | ||||
|     date = now() | ||||
|     sale_recipe.make( | ||||
|         seller=iter([None, baker.make(User)]), | ||||
|         customer=customer, | ||||
|         date=date, | ||||
|         _quantity=2, | ||||
|         _bulk_create=True, | ||||
|     ) | ||||
|     client.force_login(customer.user) | ||||
|     res = client.get( | ||||
|         reverse( | ||||
|             "core:user_account_detail", | ||||
|             kwargs={ | ||||
|                 "user_id": customer.user.id, | ||||
|                 "year": date.year, | ||||
|                 "month": date.month, | ||||
|             }, | ||||
|         ) | ||||
|     ) | ||||
|     assert res.status_code == 200 | ||||
|  | ||||
|  | ||||
| class TestFilterInactive(TestCase): | ||||
|     @classmethod | ||||
|     def setUpTestData(cls): | ||||
|   | ||||
| @@ -115,7 +115,7 @@ class SelectUser(TextInput): | ||||
|  | ||||
| def validate_future_timestamp(value: date | datetime): | ||||
|     if value <= now(): | ||||
|         raise ValueError(_("Ensure this timestamp is set in the future")) | ||||
|         raise ValidationError(_("Ensure this timestamp is set in the future")) | ||||
|  | ||||
|  | ||||
| class FutureDateTimeField(forms.DateTimeField): | ||||
|   | ||||
| @@ -2,7 +2,6 @@ import copy | ||||
| import inspect | ||||
| from typing import Any, ClassVar, LiteralString, Protocol, Unpack | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.core.exceptions import ImproperlyConfigured | ||||
| from django.http import HttpRequest, HttpResponse | ||||
| from django.template.loader import render_to_string | ||||
| @@ -41,36 +40,6 @@ class TabedViewMixin(View): | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class QuickNotifMixin: | ||||
|     quick_notif_list = [] | ||||
|  | ||||
|     def dispatch(self, request, *arg, **kwargs): | ||||
|         # In some cases, the class can stay instanciated, so we need to reset the list | ||||
|         self.quick_notif_list = [] | ||||
|         return super().dispatch(request, *arg, **kwargs) | ||||
|  | ||||
|     def get_success_url(self): | ||||
|         ret = super().get_success_url() | ||||
|         if hasattr(self, "quick_notif_url_arg"): | ||||
|             if "?" in ret: | ||||
|                 ret += "&" + self.quick_notif_url_arg | ||||
|             else: | ||||
|                 ret += "?" + self.quick_notif_url_arg | ||||
|         return ret | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         """Add quick notifications to context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["quick_notifs"] = [] | ||||
|         for n in self.quick_notif_list: | ||||
|             kwargs["quick_notifs"].append(settings.SITH_QUICK_NOTIF[n]) | ||||
|         for key, val in settings.SITH_QUICK_NOTIF.items(): | ||||
|             for gk in self.request.GET: | ||||
|                 if key == gk: | ||||
|                     kwargs["quick_notifs"].append(val) | ||||
|         return kwargs | ||||
|  | ||||
|  | ||||
| class AllowFragment: | ||||
|     """Add `is_fragment` to templates. It's only True if the request is emitted by htmx""" | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,10 @@ | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
|  | ||||
| from django.contrib.auth.mixins import PermissionRequiredMixin | ||||
| from django.db.models import F, OuterRef, Subquery | ||||
| from django.db.models.functions import Coalesce | ||||
|  | ||||
| # This file contains all the views that concern the page model | ||||
| from django.forms.models import modelform_factory | ||||
| @@ -40,10 +43,26 @@ class CanEditPagePropMixin(CanEditPropMixin): | ||||
|         return res | ||||
|  | ||||
|  | ||||
| class PageListView(CanViewMixin, ListView): | ||||
| class PageListView(ListView): | ||||
|     model = Page | ||||
|     template_name = "core/page_list.jinja" | ||||
|  | ||||
|     def get_queryset(self): | ||||
|         return ( | ||||
|             Page.objects.viewable_by(self.request.user) | ||||
|             .annotate( | ||||
|                 display_name=Coalesce( | ||||
|                     Subquery( | ||||
|                         PageRev.objects.filter(page=OuterRef("id")) | ||||
|                         .order_by("-date") | ||||
|                         .values("title")[:1] | ||||
|                     ), | ||||
|                     F("name"), | ||||
|                 ) | ||||
|             ) | ||||
|             .select_related("parent") | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class PageView(CanViewMixin, DetailView): | ||||
|     model = Page | ||||
| @@ -167,7 +186,7 @@ class PageEditViewBase(CanEditMixin, UpdateView): | ||||
|     ) | ||||
|     template_name = "core/pagerev_edit.jinja" | ||||
|  | ||||
|     def get_object(self): | ||||
|     def get_object(self, *args, **kwargs): | ||||
|         self.page = Page.get_page_by_full_name(self.kwargs["page_name"]) | ||||
|         return self._get_revision() | ||||
|  | ||||
|   | ||||
| @@ -65,7 +65,7 @@ from core.views.forms import ( | ||||
|     UserGroupsForm, | ||||
|     UserProfileForm, | ||||
| ) | ||||
| from core.views.mixins import QuickNotifMixin, TabedViewMixin, UseFragmentsMixin | ||||
| from core.views.mixins import TabedViewMixin, UseFragmentsMixin | ||||
| from counter.models import Counter, Refilling, Selling | ||||
| from eboutic.models import Invoice | ||||
| from subscription.models import Subscription | ||||
| @@ -564,7 +564,7 @@ class UserUpdateGroupView(UserTabsMixin, CanEditPropMixin, UpdateView): | ||||
|     current_tab = "groups" | ||||
|  | ||||
|  | ||||
| class UserToolsView(LoginRequiredMixin, QuickNotifMixin, UserTabsMixin, TemplateView): | ||||
| class UserToolsView(LoginRequiredMixin, UserTabsMixin, TemplateView): | ||||
|     """Displays the logged user's tools.""" | ||||
|  | ||||
|     template_name = "core/user_tools.jinja" | ||||
|   | ||||
| @@ -22,6 +22,7 @@ from counter.models import ( | ||||
|     Counter, | ||||
|     Customer, | ||||
|     Eticket, | ||||
|     InvoiceCall, | ||||
|     Permanency, | ||||
|     Product, | ||||
|     ProductType, | ||||
| @@ -160,3 +161,11 @@ class CashRegisterSummaryAdmin(SearchModelAdmin): | ||||
| class EticketAdmin(SearchModelAdmin): | ||||
|     list_display = ("product", "event_date", "event_title") | ||||
|     search_fields = ("product__name", "event_title") | ||||
|  | ||||
|  | ||||
| @admin.register(InvoiceCall) | ||||
| class InvoiceCallAdmin(SearchModelAdmin): | ||||
|     list_display = ("club", "month", "is_validated") | ||||
|     search_fields = ("club__name",) | ||||
|     list_filter = (("club", admin.RelatedOnlyFieldListFilter),) | ||||
|     date_hierarchy = "month" | ||||
|   | ||||
							
								
								
									
										187
									
								
								counter/forms.py
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								counter/forms.py
									
									
									
									
									
								
							| @@ -1,13 +1,26 @@ | ||||
| import json | ||||
| import math | ||||
| import uuid | ||||
| from datetime import date | ||||
|  | ||||
| from dateutil.relativedelta import relativedelta | ||||
| from django import forms | ||||
| from django.db.models import Q | ||||
| from django.db.models import Exists, OuterRef, Q | ||||
| from django.forms import BaseModelFormSet | ||||
| from django.utils.timezone import now | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django_celery_beat.models import ClockedSchedule | ||||
| from phonenumber_field.widgets import RegionalPhoneNumberWidget | ||||
|  | ||||
| from club.models import Club | ||||
| from club.widgets.ajax_select import AutoCompleteSelectClub | ||||
| from core.models import User | ||||
| from core.views.forms import NFCTextInput, SelectDate, SelectDateTime | ||||
| from core.views.forms import ( | ||||
|     FutureDateTimeField, | ||||
|     NFCTextInput, | ||||
|     SelectDate, | ||||
|     SelectDateTime, | ||||
| ) | ||||
| from core.views.widgets.ajax_select import ( | ||||
|     AutoCompleteSelect, | ||||
|     AutoCompleteSelectMultipleGroup, | ||||
| @@ -19,10 +32,14 @@ from counter.models import ( | ||||
|     Counter, | ||||
|     Customer, | ||||
|     Eticket, | ||||
|     InvoiceCall, | ||||
|     Product, | ||||
|     Refilling, | ||||
|     ReturnableProduct, | ||||
|     ScheduledProductAction, | ||||
|     Selling, | ||||
|     StudentCard, | ||||
|     get_product_actions, | ||||
| ) | ||||
| from counter.widgets.ajax_select import ( | ||||
|     AutoCompleteSelectMultipleCounter, | ||||
| @@ -158,7 +175,101 @@ class CounterEditForm(forms.ModelForm): | ||||
|         } | ||||
|  | ||||
|  | ||||
| class ProductEditForm(forms.ModelForm): | ||||
| class ScheduledProductActionForm(forms.ModelForm): | ||||
|     """Form for automatic product archiving. | ||||
|  | ||||
|     The `save` method will update or create tasks using celery-beat. | ||||
|     """ | ||||
|  | ||||
|     required_css_class = "required" | ||||
|     prefix = "scheduled" | ||||
|  | ||||
|     class Meta: | ||||
|         model = ScheduledProductAction | ||||
|         fields = ["task"] | ||||
|         widgets = {"task": forms.RadioSelect(choices=get_product_actions)} | ||||
|         labels = {"task": _("Action")} | ||||
|         help_texts = {"task": ""} | ||||
|  | ||||
|     trigger_at = FutureDateTimeField( | ||||
|         label=_("Date and time of action"), widget=SelectDateTime | ||||
|     ) | ||||
|     counters = forms.ModelMultipleChoiceField( | ||||
|         label=_("New counters"), | ||||
|         help_text=_("The selected counters will replace the current ones"), | ||||
|         required=False, | ||||
|         widget=AutoCompleteSelectMultipleCounter, | ||||
|         queryset=Counter.objects.all(), | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, *args, product: Product, **kwargs): | ||||
|         self.product = product | ||||
|         super().__init__(*args, **kwargs) | ||||
|         if not self.instance._state.adding: | ||||
|             self.fields["trigger_at"].initial = self.instance.clocked.clocked_time | ||||
|             self.fields["counters"].initial = json.loads(self.instance.kwargs).get( | ||||
|                 "counters" | ||||
|             ) | ||||
|  | ||||
|     def clean(self): | ||||
|         if not self.changed_data or "trigger_at" in self.errors: | ||||
|             return super().clean() | ||||
|         if "trigger_at" in self.changed_data: | ||||
|             if not self.instance.clocked_id: | ||||
|                 self.instance.clocked = ClockedSchedule( | ||||
|                     clocked_time=self.cleaned_data["trigger_at"] | ||||
|                 ) | ||||
|             else: | ||||
|                 self.instance.clocked.clocked_time = self.cleaned_data["trigger_at"] | ||||
|             self.instance.clocked.save() | ||||
|         task_kwargs = {"product_id": self.product.id} | ||||
|         if ( | ||||
|             self.cleaned_data["task"] == "counter.tasks.change_counters" | ||||
|             and "counters" in self.changed_data | ||||
|         ): | ||||
|             task_kwargs["counters"] = [c.id for c in self.cleaned_data["counters"]] | ||||
|         self.instance.product = self.product | ||||
|         self.instance.kwargs = json.dumps(task_kwargs) | ||||
|         self.instance.name = ( | ||||
|             f"{self.cleaned_data['task']} - {self.product} - {uuid.uuid4()}" | ||||
|         ) | ||||
|         return super().clean() | ||||
|  | ||||
|  | ||||
| class BaseScheduledProductActionFormSet(BaseModelFormSet): | ||||
|     def __init__(self, *args, product: Product, **kwargs): | ||||
|         if product.id: | ||||
|             queryset = ( | ||||
|                 product.scheduled_actions.filter( | ||||
|                     enabled=True, clocked__clocked_time__gt=now() | ||||
|                 ) | ||||
|                 .order_by("clocked__clocked_time") | ||||
|                 .select_related("clocked") | ||||
|             ) | ||||
|         else: | ||||
|             queryset = ScheduledProductAction.objects.none() | ||||
|         form_kwargs = {"product": product} | ||||
|         super().__init__(*args, queryset=queryset, form_kwargs=form_kwargs, **kwargs) | ||||
|  | ||||
|     def delete_existing(self, obj: ScheduledProductAction, commit: bool = True):  # noqa FBT001 | ||||
|         clocked = obj.clocked | ||||
|         super().delete_existing(obj, commit=commit) | ||||
|         if commit: | ||||
|             clocked.delete() | ||||
|  | ||||
|  | ||||
| ScheduledProductActionFormSet = forms.modelformset_factory( | ||||
|     ScheduledProductAction, | ||||
|     ScheduledProductActionForm, | ||||
|     formset=BaseScheduledProductActionFormSet, | ||||
|     absolute_max=None, | ||||
|     can_delete=True, | ||||
|     can_delete_extra=False, | ||||
|     extra=2, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class ProductForm(forms.ModelForm): | ||||
|     error_css_class = "error" | ||||
|     required_css_class = "required" | ||||
|  | ||||
| @@ -199,22 +310,21 @@ class ProductEditForm(forms.ModelForm): | ||||
|         queryset=Counter.objects.all(), | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|     def __init__(self, *args, instance=None, **kwargs): | ||||
|         super().__init__(*args, instance=instance, **kwargs) | ||||
|         if self.instance.id: | ||||
|             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): | ||||
|         ret = super().save(*args, **kwargs) | ||||
|         if self.fields["counters"].initial: | ||||
|             # 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() | ||||
|         self.instance.counters.set(self.cleaned_data["counters"]) | ||||
|         self.action_formset.save() | ||||
|         return ret | ||||
|  | ||||
|  | ||||
| @@ -266,7 +376,7 @@ class CloseCustomerAccountForm(forms.Form): | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class ProductForm(forms.Form): | ||||
| class BasketProductForm(forms.Form): | ||||
|     quantity = forms.IntegerField(min_value=1, required=True) | ||||
|     id = forms.IntegerField(min_value=0, required=True) | ||||
|  | ||||
| @@ -371,5 +481,50 @@ class BaseBasketForm(forms.BaseFormSet): | ||||
|  | ||||
|  | ||||
| BasketForm = forms.formset_factory( | ||||
|     ProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 | ||||
|     BasketProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 | ||||
| ) | ||||
|  | ||||
|  | ||||
| class InvoiceCallForm(forms.Form): | ||||
|     def __init__(self, *args, month: date, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.month = month | ||||
|         self.clubs = list( | ||||
|             Club.objects.filter( | ||||
|                 Exists( | ||||
|                     Selling.objects.filter( | ||||
|                         club=OuterRef("pk"), | ||||
|                         date__gte=month, | ||||
|                         date__lte=month + relativedelta(months=1), | ||||
|                     ) | ||||
|                 ) | ||||
|             ).annotate( | ||||
|                 validated_invoice=Exists( | ||||
|                     InvoiceCall.objects.filter( | ||||
|                         club=OuterRef("pk"), month=month, is_validated=True | ||||
|                     ) | ||||
|                 ) | ||||
|             ) | ||||
|         ) | ||||
|         self.fields = { | ||||
|             str(club.id): forms.BooleanField( | ||||
|                 required=False, initial=club.validated_invoice | ||||
|             ) | ||||
|             for club in self.clubs | ||||
|         } | ||||
|  | ||||
|     def save(self): | ||||
|         invoice_calls = [ | ||||
|             InvoiceCall( | ||||
|                 month=self.month, | ||||
|                 club_id=club.id, | ||||
|                 is_validated=self.cleaned_data.get(str(club.id), False), | ||||
|             ) | ||||
|             for club in self.clubs | ||||
|         ] | ||||
|         InvoiceCall.objects.bulk_create( | ||||
|             invoice_calls, | ||||
|             update_conflicts=True, | ||||
|             update_fields=["is_validated"], | ||||
|             unique_fields=["month", "club"], | ||||
|         ) | ||||
|   | ||||
							
								
								
									
										40
									
								
								counter/migrations/0032_scheduledproductaction.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								counter/migrations/0032_scheduledproductaction.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| # Generated by Django 5.2.3 on 2025-09-14 11:29 | ||||
|  | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|     dependencies = [ | ||||
|         ("counter", "0031_alter_counter_options"), | ||||
|         ("django_celery_beat", "0019_alter_periodictasks_options"), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name="ScheduledProductAction", | ||||
|             fields=[ | ||||
|                 ( | ||||
|                     "periodictask_ptr", | ||||
|                     models.OneToOneField( | ||||
|                         auto_created=True, | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                         parent_link=True, | ||||
|                         primary_key=True, | ||||
|                         serialize=False, | ||||
|                         to="django_celery_beat.periodictask", | ||||
|                     ), | ||||
|                 ), | ||||
|                 ( | ||||
|                     "product", | ||||
|                     models.ForeignKey( | ||||
|                         on_delete=django.db.models.deletion.CASCADE, | ||||
|                         related_name="scheduled_actions", | ||||
|                         to="counter.product", | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
|             options={"verbose_name": "Product scheduled action"}, | ||||
|             bases=("django_celery_beat.periodictask",), | ||||
|         ), | ||||
|     ] | ||||
							
								
								
									
										51
									
								
								counter/migrations/0033_invoicecall.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								counter/migrations/0033_invoicecall.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | ||||
| # Generated by Django 5.2.3 on 2025-10-15 21:54 | ||||
|  | ||||
| import django.db.models.deletion | ||||
| from django.db import migrations, models | ||||
|  | ||||
| import counter.models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|     dependencies = [ | ||||
|         ("club", "0014_alter_club_options_rename_unix_name_club_slug_name_and_more"), | ||||
|         ("counter", "0032_scheduledproductaction"), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name="InvoiceCall", | ||||
|             fields=[ | ||||
|                 ( | ||||
|                     "id", | ||||
|                     models.AutoField( | ||||
|                         auto_created=True, | ||||
|                         primary_key=True, | ||||
|                         serialize=False, | ||||
|                         verbose_name="ID", | ||||
|                     ), | ||||
|                 ), | ||||
|                 ( | ||||
|                     "is_validated", | ||||
|                     models.BooleanField(default=False, verbose_name="is validated"), | ||||
|                 ), | ||||
|                 ("month", counter.models.MonthField(verbose_name="invoice date")), | ||||
|                 ( | ||||
|                     "club", | ||||
|                     models.ForeignKey( | ||||
|                         on_delete=django.db.models.deletion.CASCADE, to="club.club" | ||||
|                     ), | ||||
|                 ), | ||||
|             ], | ||||
|             options={ | ||||
|                 "verbose_name": "Invoice call", | ||||
|                 "verbose_name_plural": "Invoice calls", | ||||
|                 "constraints": [ | ||||
|                     models.UniqueConstraint( | ||||
|                         fields=("club", "month"), | ||||
|                         name="counter_invoicecall_unique_club_month", | ||||
|                     ) | ||||
|                 ], | ||||
|             }, | ||||
|         ), | ||||
|     ] | ||||
| @@ -15,6 +15,7 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| import base64 | ||||
| import contextlib | ||||
| import os | ||||
| import random | ||||
| import string | ||||
| @@ -34,6 +35,7 @@ from django.urls import reverse | ||||
| from django.utils import timezone | ||||
| from django.utils.functional import cached_property | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django_celery_beat.models import PeriodicTask | ||||
| from django_countries.fields import CountryField | ||||
| from ordered_model.models import OrderedModel | ||||
| from phonenumber_field.modelfields import PhoneNumberField | ||||
| @@ -445,7 +447,8 @@ class Product(models.Model): | ||||
|         buying_groups = list(self.buying_groups.all()) | ||||
|         if not buying_groups: | ||||
|             return True | ||||
|         return any(user.is_in_group(pk=group.id) for group in buying_groups) | ||||
|         res = any(user.is_in_group(pk=group.id) for group in buying_groups) | ||||
|         return res | ||||
|  | ||||
|     @property | ||||
|     def profit(self): | ||||
| @@ -479,7 +482,7 @@ class CounterQuerySet(models.QuerySet): | ||||
|         return self.annotate(has_annotated_barman=Exists(subquery)) | ||||
|  | ||||
|     def annotate_is_open(self) -> Self: | ||||
|         """Annotate tue queryset with the `is_open` field. | ||||
|         """Annotate the queryset with the `is_open` field. | ||||
|  | ||||
|         For each counter, if `is_open=True`, then the counter is currently opened. | ||||
|         Else the counter is closed. | ||||
| @@ -535,13 +538,6 @@ class Counter(models.Model): | ||||
|     def __str__(self): | ||||
|         return self.name | ||||
|  | ||||
|     def __getattribute__(self, name: str): | ||||
|         if name == "edit_groups": | ||||
|             return Group.objects.filter( | ||||
|                 name=self.club.unix_name + settings.SITH_BOARD_SUFFIX | ||||
|             ).all() | ||||
|         return object.__getattribute__(self, name) | ||||
|  | ||||
|     def get_absolute_url(self) -> str: | ||||
|         if self.type == "EBOUTIC": | ||||
|             return reverse("eboutic:main") | ||||
| @@ -690,8 +686,10 @@ class Counter(models.Model): | ||||
|         Prices will be annotated | ||||
|         """ | ||||
|  | ||||
|         products = self.products.select_related("product_type").prefetch_related( | ||||
|             "buying_groups" | ||||
|         products = ( | ||||
|             self.products.filter(archived=False) | ||||
|             .select_related("product_type") | ||||
|             .prefetch_related("buying_groups") | ||||
|         ) | ||||
|  | ||||
|         # Only include age appropriate products | ||||
| @@ -1362,3 +1360,85 @@ class ReturnableProductBalance(models.Model): | ||||
|             f"return balance of {self.customer} " | ||||
|             f"for {self.returnable.product_id} : {self.balance}" | ||||
|         ) | ||||
|  | ||||
|  | ||||
| def get_product_actions(): | ||||
|     return [ | ||||
|         ("counter.tasks.archive_product", _("Archiving")), | ||||
|         ("counter.tasks.change_counters", _("Counters change")), | ||||
|     ] | ||||
|  | ||||
|  | ||||
| class ScheduledProductAction(PeriodicTask): | ||||
|     """Extension of celery-beat tasks dedicated to perform actions on Product.""" | ||||
|  | ||||
|     product = models.ForeignKey( | ||||
|         Product, related_name="scheduled_actions", on_delete=models.CASCADE | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("Product scheduled action") | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self._meta.get_field("task").choices = get_product_actions() | ||||
|         super().__init__(*args, **kwargs) | ||||
|  | ||||
|     def full_clean(self, *args, **kwargs): | ||||
|         self.one_off = True  # A product action should occur one time only | ||||
|         return super().full_clean(*args, **kwargs) | ||||
|  | ||||
|     def clean_clocked(self): | ||||
|         if not self.clocked: | ||||
|             raise ValidationError(_("Product actions must declare a clocked schedule.")) | ||||
|  | ||||
|     def validate_unique(self, *args, **kwargs): | ||||
|         # The checks done in PeriodicTask.validate_unique aren't | ||||
|         # adapted in the case of scheduled product action, | ||||
|         # so we skip it and execute directly Model.validate_unique | ||||
|         return super(PeriodicTask, self).validate_unique(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| class MonthField(models.DateField): | ||||
|     description = _("Year + month field (day forced to 1)") | ||||
|     default_error_messages = { | ||||
|         "invalid": _( | ||||
|             "“%(value)s” value has an invalid date format. It must be " | ||||
|             "in YYYY-MM format." | ||||
|         ), | ||||
|         "invalid_date": _( | ||||
|             "“%(value)s” value has the correct format (YYYY-MM) " | ||||
|             "but it is an invalid date." | ||||
|         ), | ||||
|     } | ||||
|  | ||||
|     def to_python(self, value): | ||||
|         if isinstance(value, str): | ||||
|             with contextlib.suppress(ValueError): | ||||
|                 # If the string is given as YYYY-mm, try to parse it. | ||||
|                 # If it fails, it means that the string may be in the form YYYY-mm-dd | ||||
|                 # or in an invalid format. | ||||
|                 # Whatever the case, we let Django deal with it | ||||
|                 # and raise an error if needed | ||||
|                 value = datetime.strptime(value, "%Y-%m") | ||||
|         value = super().to_python(value) | ||||
|         if value is None: | ||||
|             return None | ||||
|         return value.replace(day=1) | ||||
|  | ||||
|  | ||||
| class InvoiceCall(models.Model): | ||||
|     is_validated = models.BooleanField(verbose_name=_("is validated"), default=False) | ||||
|     club = models.ForeignKey(Club, on_delete=models.CASCADE) | ||||
|     month = MonthField(verbose_name=_("invoice date")) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("Invoice call") | ||||
|         verbose_name_plural = _("Invoice calls") | ||||
|         constraints = [ | ||||
|             models.UniqueConstraint( | ||||
|                 fields=["club", "month"], name="counter_invoicecall_unique_club_month" | ||||
|             ) | ||||
|         ] | ||||
|  | ||||
|     def __str__(self): | ||||
|         return f"invoice call of {self.month} made by {self.club}" | ||||
|   | ||||
| @@ -39,6 +39,7 @@ | ||||
|   flex: auto; | ||||
|   margin: 0.2em; | ||||
|   width: 20%; | ||||
|   min-width: 350px; | ||||
|  | ||||
|   ul { | ||||
|     list-style-type: none; | ||||
|   | ||||
							
								
								
									
										19
									
								
								counter/tasks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								counter/tasks.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # Create your tasks here | ||||
|  | ||||
| from celery import shared_task | ||||
|  | ||||
| from counter.models import Counter, Product | ||||
|  | ||||
|  | ||||
| @shared_task | ||||
| def archive_product(*, product_id: int, **kwargs): | ||||
|     product = Product.objects.get(id=product_id) | ||||
|     product.archived = True | ||||
|     product.save() | ||||
|  | ||||
|  | ||||
| @shared_task | ||||
| def change_counters(*, product_id: int, counters: list[int], **kwargs): | ||||
|     product = Product.objects.get(id=product_id) | ||||
|     counters = Counter.objects.filter(id__in=counters) | ||||
|     product.counters.set(counters) | ||||
| @@ -67,13 +67,13 @@ | ||||
|                 <option value="FIN">{% trans %}Confirm (FIN){% endtrans %}</option> | ||||
|                 <option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option> | ||||
|               </optgroup> | ||||
|               {% for category in categories.keys() %} | ||||
|               {%- for category in categories.keys() -%} | ||||
|                 <optgroup label="{{ category }}"> | ||||
|                   {% for product in categories[category] %} | ||||
|                   {%- for product in categories[category] -%} | ||||
|                     <option value="{{ product.id }}">{{ product }}</option> | ||||
|                   {% endfor %} | ||||
|                   {%- endfor -%} | ||||
|                 </optgroup> | ||||
|               {% endfor %} | ||||
|               {%- endfor -%} | ||||
|             </counter-product-select> | ||||
|  | ||||
|             <input type="submit" value="{% trans %}Go{% endtrans %}"/> | ||||
|   | ||||
| @@ -4,35 +4,49 @@ | ||||
|   {% trans %}Invoices call{% endtrans %} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block notifications %}{# Notifications are moved below #}{% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|   <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=""> | ||||
|     <select name="month"> | ||||
|     <label for="id_form_other_month">{% trans %}Choose another month: {% endtrans %}</label> | ||||
|     <select name="month" id="id_form_other_month"> | ||||
|       {% for m in months %} | ||||
|         <option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option> | ||||
|       {% endfor %} | ||||
|     </select> | ||||
|     <input type="submit" value="{% trans %}Go{% endtrans %}" /> | ||||
|   </form> | ||||
|  | ||||
|   <br> | ||||
|   <p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p> | ||||
|   <br> | ||||
|   <table> | ||||
|     <thead> | ||||
|       <td>{% trans %}Club{% endtrans %}</td> | ||||
|       <td>{% trans %}Sum{% endtrans %}</td> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|       {% for i in sums %} | ||||
|  | ||||
|   {% include "core/base/notifications.jinja" %} | ||||
|  | ||||
|   <form method="post" action=""> | ||||
|     {% csrf_token %} | ||||
|     <table> | ||||
|       <thead> | ||||
|         <tr> | ||||
|           <td>{{ i['club__name'] }}</td> | ||||
|           <td>{{ i['selling_sum'] }} €</td> | ||||
|           <td>{% trans %}Club{% endtrans %}</td> | ||||
|           <td>{% trans %}Sum{% endtrans %}</td> | ||||
|           <td>{% trans %}Validated{% endtrans %}</td> | ||||
|         </tr> | ||||
|       {% endfor %} | ||||
|     </tbody> | ||||
|   </table> | ||||
|       </thead> | ||||
|       <tbody> | ||||
|         {% for invoice in invoices %} | ||||
|           <tr> | ||||
|             <td>{{ invoice.club__name }}</td> | ||||
|             <td>{{ "%.2f"|format(invoice.selling_sum) }} €</td> | ||||
|             <td> | ||||
|               {{ form[invoice.club_id|string] }} | ||||
|             </td> | ||||
|           </tr> | ||||
|         {% endfor %} | ||||
|       </tbody> | ||||
|     </table> | ||||
|     <input type="hidden" name="month" value="{{ start_date|date('Y-m') }}"> | ||||
|     <button type="submit">{% trans %}Save{% endtrans %}</button> | ||||
|   </form> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										56
									
								
								counter/templates/counter/product_form.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								counter/templates/counter/product_form.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block content %} | ||||
|   {% if object %} | ||||
|     <h2>{% trans name=object %}Edit product {{ name }}{% endtrans %}</h2> | ||||
|   {% else %} | ||||
|     <h2>{% trans %}Product creation{% endtrans %}</h2> | ||||
|   {% endif %} | ||||
|   <form method="post"> | ||||
|     {% csrf_token %} | ||||
|     {{ form.as_p() }} | ||||
|  | ||||
|     <br /> | ||||
|  | ||||
|     <h3>{% trans %}Automatic actions{% endtrans %}</h3> | ||||
|  | ||||
|     <p class="margin-bottom"> | ||||
|       <em> | ||||
|         {%- trans trimmed -%} | ||||
|           Automatic actions allows to schedule product changes | ||||
|           ahead of time. | ||||
|         {%- endtrans -%} | ||||
|       </em> | ||||
|     </p> | ||||
|  | ||||
|     {{ form.action_formset.management_form }} | ||||
|     {%- for action_form in form.action_formset.forms -%} | ||||
|       <fieldset x-data="{action: '{{ action_form.task.initial }}'}"> | ||||
|         {{ action_form.non_field_errors() }} | ||||
|         <div class="row gap-2x margin-bottom"> | ||||
|           <div> | ||||
|             {{ action_form.task.errors }} | ||||
|             {{ action_form.task.label_tag() }} | ||||
|             {{ action_form.task|add_attr("x-model=action") }} | ||||
|           </div> | ||||
|           <div>{{ action_form.trigger_at.as_field_group() }}</div> | ||||
|         </div> | ||||
|         <div x-show="action==='counter.tasks.change_counters'" class="margin-bottom"> | ||||
|           {{ action_form.counters.as_field_group() }} | ||||
|         </div> | ||||
|         {%- if action_form.DELETE -%} | ||||
|           <div class="row gap"> | ||||
|             {{ action_form.DELETE.as_field_group() }} | ||||
|           </div> | ||||
|         {%- endif -%} | ||||
|         {%- for field in action_form.hidden_fields() -%} | ||||
|           {{ field }} | ||||
|         {%- endfor -%} | ||||
|       </fieldset> | ||||
|       {%- if not loop.last -%} | ||||
|         <hr class="margin-bottom"> | ||||
|       {%- endif -%} | ||||
|     {%- endfor -%} | ||||
|     <p><input type="submit" value="{% trans %}Save{% endtrans %}" /></p> | ||||
|   </form> | ||||
| {% endblock %} | ||||
							
								
								
									
										116
									
								
								counter/tests/test_auto_actions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								counter/tests/test_auto_actions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,116 @@ | ||||
| import json | ||||
| from datetime import timedelta | ||||
|  | ||||
| import pytest | ||||
| from django.conf import settings | ||||
| from django.test import Client | ||||
| from django.urls import reverse | ||||
| from django.utils.timezone import now | ||||
| from django_celery_beat.models import ClockedSchedule | ||||
| from model_bakery import baker | ||||
|  | ||||
| from core.models import Group, User | ||||
| from counter.baker_recipes import counter_recipe, product_recipe | ||||
| from counter.forms import ScheduledProductActionForm, ScheduledProductActionFormSet | ||||
| from counter.models import ScheduledProductAction | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_edit_product(client: Client): | ||||
|     client.force_login( | ||||
|         baker.make( | ||||
|             User, groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)] | ||||
|         ) | ||||
|     ) | ||||
|     product = product_recipe.make() | ||||
|     url = reverse("counter:product_edit", kwargs={"product_id": product.id}) | ||||
|     res = client.get(url) | ||||
|     assert res.status_code == 200 | ||||
|  | ||||
|     res = client.post(url, data={}) | ||||
|     # This is actually a failure, but we just want to check that | ||||
|     # we don't have a 403 or a 500. | ||||
|     # The actual behaviour will be tested directly on the form. | ||||
|     assert res.status_code == 200 | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| class TestProductActionForm: | ||||
|     def test_single_form_archive(self): | ||||
|         product = product_recipe.make() | ||||
|         trigger_at = now() + timedelta(minutes=10) | ||||
|         form = ScheduledProductActionForm( | ||||
|             product=product, | ||||
|             data={ | ||||
|                 "scheduled-task": "counter.tasks.archive_product", | ||||
|                 "scheduled-trigger_at": trigger_at, | ||||
|             }, | ||||
|         ) | ||||
|         assert form.is_valid() | ||||
|         instance = form.save() | ||||
|         assert instance.clocked.clocked_time == trigger_at | ||||
|         assert instance.enabled is True | ||||
|         assert instance.one_off is True | ||||
|         assert instance.task == "counter.tasks.archive_product" | ||||
|         assert instance.kwargs == json.dumps({"product_id": product.id}) | ||||
|  | ||||
|     def test_single_form_change_counters(self): | ||||
|         product = product_recipe.make() | ||||
|         counter = counter_recipe.make() | ||||
|         trigger_at = now() + timedelta(minutes=10) | ||||
|         form = ScheduledProductActionForm( | ||||
|             product=product, | ||||
|             data={ | ||||
|                 "scheduled-task": "counter.tasks.change_counters", | ||||
|                 "scheduled-trigger_at": trigger_at, | ||||
|                 "scheduled-counters": [counter.id], | ||||
|             }, | ||||
|         ) | ||||
|         assert form.is_valid() | ||||
|         instance = form.save() | ||||
|         instance.refresh_from_db() | ||||
|         assert instance.clocked.clocked_time == trigger_at | ||||
|         assert instance.enabled is True | ||||
|         assert instance.one_off is True | ||||
|         assert instance.task == "counter.tasks.change_counters" | ||||
|         assert instance.kwargs == json.dumps( | ||||
|             {"product_id": product.id, "counters": [counter.id]} | ||||
|         ) | ||||
|  | ||||
|     def test_delete(self): | ||||
|         product = product_recipe.make() | ||||
|         clocked = baker.make(ClockedSchedule, clocked_time=now() + timedelta(minutes=2)) | ||||
|         task = baker.make( | ||||
|             ScheduledProductAction, | ||||
|             product=product, | ||||
|             one_off=True, | ||||
|             clocked=clocked, | ||||
|             task="counter.tasks.archive_product", | ||||
|         ) | ||||
|         formset = ScheduledProductActionFormSet(product=product) | ||||
|         formset.delete_existing(task) | ||||
|         assert not ScheduledProductAction.objects.filter(id=task.id).exists() | ||||
|         assert not ClockedSchedule.objects.filter(id=clocked.id).exists() | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| class TestProductActionFormSet: | ||||
|     def test_ok(self): | ||||
|         product = product_recipe.make() | ||||
|         counter = counter_recipe.make() | ||||
|         trigger_at = now() + timedelta(minutes=10) | ||||
|         formset = ScheduledProductActionFormSet( | ||||
|             product=product, | ||||
|             data={ | ||||
|                 "form-TOTAL_FORMS": "2", | ||||
|                 "form-INITIAL_FORMS": "0", | ||||
|                 "form-0-task": "counter.tasks.archive_product", | ||||
|                 "form-0-trigger_at": trigger_at, | ||||
|                 "form-1-task": "counter.tasks.change_counters", | ||||
|                 "form-1-trigger_at": trigger_at, | ||||
|                 "form-1-counters": [counter.id], | ||||
|             }, | ||||
|         ) | ||||
|         assert formset.is_valid() | ||||
|         formset.save() | ||||
|         assert ScheduledProductAction.objects.filter(product=product).count() == 2 | ||||
| @@ -583,6 +583,16 @@ class TestCounterClick(TestFullClickBase): | ||||
|             - self.beer.selling_price | ||||
|         ) | ||||
|  | ||||
|     def test_no_fetch_archived_product(self): | ||||
|         counter = baker.make(Counter) | ||||
|         customer = baker.make(Customer) | ||||
|         product_recipe.make(archived=True, counters=[counter]) | ||||
|         unarchived_products = product_recipe.make( | ||||
|             archived=False, counters=[counter], _quantity=3 | ||||
|         ) | ||||
|         customer_products = counter.get_products_for(customer) | ||||
|         assert unarchived_products == customer_products | ||||
|  | ||||
|  | ||||
| class TestCounterStats(TestCase): | ||||
|     @classmethod | ||||
|   | ||||
							
								
								
									
										76
									
								
								counter/tests/test_invoices.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								counter/tests/test_invoices.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| from datetime import date, datetime | ||||
|  | ||||
| import pytest | ||||
| from dateutil.relativedelta import relativedelta | ||||
| from django.contrib.auth.models import Permission | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.test import Client | ||||
| from django.urls import reverse | ||||
| from django.utils.timezone import localdate | ||||
| from model_bakery import baker | ||||
| from pytest_django.asserts import assertRedirects | ||||
|  | ||||
| from club.models import Club | ||||
| from core.models import User | ||||
| from counter.baker_recipes import sale_recipe | ||||
| from counter.forms import InvoiceCallForm | ||||
| from counter.models import Customer, InvoiceCall, Selling | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| @pytest.mark.parametrize( | ||||
|     "month", [date(2025, 10, 20), "2025-10", datetime(2025, 10, 15, 12, 30)] | ||||
| ) | ||||
| def test_invoice_date_with_date(month: date | datetime | str): | ||||
|     club = baker.make(Club) | ||||
|     invoice = InvoiceCall.objects.create(club=club, month=month) | ||||
|     invoice.refresh_from_db() | ||||
|     assert not invoice.is_validated | ||||
|     assert invoice.month == date(2025, 10, 1) | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_invoice_call_invalid_month_string(): | ||||
|     club = baker.make(Club) | ||||
|  | ||||
|     with pytest.raises(ValidationError): | ||||
|         InvoiceCall.objects.create(club=club, month="2025-13") | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| @pytest.mark.parametrize("query", [None, {"month": "2025-08"}]) | ||||
| def test_invoice_call_view(client: Client, query: dict | None): | ||||
|     user = baker.make( | ||||
|         User, | ||||
|         user_permissions=[ | ||||
|             *Permission.objects.filter( | ||||
|                 codename__in=["view_invoicecall", "change_invoicecall"] | ||||
|             ) | ||||
|         ], | ||||
|     ) | ||||
|     client.force_login(user) | ||||
|     url = reverse("counter:invoices_call", query=query) | ||||
|     assert client.get(url).status_code == 200 | ||||
|     assertRedirects(client.post(url), url) | ||||
|  | ||||
|  | ||||
| @pytest.mark.django_db | ||||
| def test_invoice_call_form(): | ||||
|     Selling.objects.all().delete() | ||||
|     month = localdate() - relativedelta(months=1) | ||||
|     clubs = baker.make(Club, _quantity=2) | ||||
|     recipe = sale_recipe.extend(date=month, customer=baker.make(Customer, amount=10000)) | ||||
|     recipe.make(club=clubs[0], quantity=2, unit_price=200) | ||||
|     recipe.make(club=clubs[0], quantity=3, unit_price=5) | ||||
|     recipe.make(club=clubs[1], quantity=20, unit_price=10) | ||||
|     form = InvoiceCallForm( | ||||
|         month=month, data={str(clubs[0].id): True, str(clubs[1].id): False} | ||||
|     ) | ||||
|     assert form.is_valid() | ||||
|     form.save() | ||||
|     assert InvoiceCall.objects.filter( | ||||
|         club=clubs[0], month=month, is_validated=True | ||||
|     ).exists() | ||||
|     assert InvoiceCall.objects.filter( | ||||
|         club=clubs[1], month=month, is_validated=False | ||||
|     ).exists() | ||||
| @@ -6,14 +6,16 @@ import pytest | ||||
| from django.conf import settings | ||||
| from django.core.cache import cache | ||||
| from django.core.files.uploadedfile import SimpleUploadedFile | ||||
| from django.test import Client | ||||
| from django.test import Client, TestCase | ||||
| from django.urls import reverse | ||||
| from model_bakery import baker | ||||
| from PIL import Image | ||||
| from pytest_django.asserts import assertNumQueries | ||||
| from pytest_django.asserts import assertNumQueries, assertRedirects | ||||
|  | ||||
| from club.models import Club | ||||
| from core.baker_recipes import board_user, subscriber_user | ||||
| from core.models import Group, User | ||||
| from counter.forms import ProductForm | ||||
| from counter.models import Product, ProductType | ||||
|  | ||||
|  | ||||
| @@ -84,3 +86,49 @@ def test_fetch_product_nb_queries(client: Client): | ||||
|         # - 1 for the actual request | ||||
|         # - 1 to prefetch the related buying_groups | ||||
|         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 ( | ||||
|     CloseCustomerAccountForm, | ||||
|     CounterEditForm, | ||||
|     ProductEditForm, | ||||
|     ProductForm, | ||||
|     ReturnableProductForm, | ||||
| ) | ||||
| from counter.models import ( | ||||
| @@ -146,8 +146,8 @@ class ProductCreateView(CounterAdminTabsMixin, CounterAdminMixin, CreateView): | ||||
|     """A create view for the admins.""" | ||||
|  | ||||
|     model = Product | ||||
|     form_class = ProductEditForm | ||||
|     template_name = "core/create.jinja" | ||||
|     form_class = ProductForm | ||||
|     template_name = "counter/product_form.jinja" | ||||
|     current_tab = "products" | ||||
|  | ||||
|  | ||||
| @@ -155,9 +155,9 @@ class ProductEditView(CounterAdminTabsMixin, CounterAdminMixin, UpdateView): | ||||
|     """An edit view for the admins.""" | ||||
|  | ||||
|     model = Product | ||||
|     form_class = ProductEditForm | ||||
|     form_class = ProductForm | ||||
|     pk_url_kwarg = "product_id" | ||||
|     template_name = "core/edit.jinja" | ||||
|     template_name = "counter/product_form.jinja" | ||||
|     current_tab = "products" | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -12,77 +12,81 @@ | ||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | ||||
| # | ||||
| # | ||||
| from datetime import datetime, timedelta | ||||
| from datetime import timezone as tz | ||||
| from datetime import datetime | ||||
| from urllib.parse import urlencode | ||||
|  | ||||
| from django.db.models import F | ||||
| from django.utils import timezone | ||||
| from django.views.generic import TemplateView | ||||
| from dateutil.relativedelta import relativedelta | ||||
| from django.contrib.auth.mixins import PermissionRequiredMixin | ||||
| from django.contrib.messages.views import SuccessMessageMixin | ||||
| from django.db.models import F, Sum | ||||
| from django.utils.timezone import localdate, make_aware | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django.views.generic import FormView | ||||
|  | ||||
| from counter.fields import CurrencyField | ||||
| from counter.forms import InvoiceCallForm | ||||
| from counter.models import Refilling, Selling | ||||
| from counter.views.mixins import CounterAdminMixin, CounterAdminTabsMixin | ||||
| from counter.views.mixins import CounterAdminTabsMixin | ||||
|  | ||||
|  | ||||
| class InvoiceCallView(CounterAdminTabsMixin, CounterAdminMixin, TemplateView): | ||||
| class InvoiceCallView( | ||||
|     CounterAdminTabsMixin, PermissionRequiredMixin, SuccessMessageMixin, FormView | ||||
| ): | ||||
|     template_name = "counter/invoices_call.jinja" | ||||
|     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): | ||||
|         """Add sums to the context.""" | ||||
|         kwargs = super().get_context_data(**kwargs) | ||||
|         kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC") | ||||
|         if "month" in self.request.GET: | ||||
|             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 | ||||
|         start_date = self.get_month() | ||||
|         end_date = start_date + relativedelta(months=1) | ||||
|  | ||||
|         kwargs["sum_cb"] = sum( | ||||
|             [ | ||||
|                 r.amount | ||||
|                 for r in Refilling.objects.filter( | ||||
|                     payment_method="CARD", | ||||
|                     is_validated=True, | ||||
|                     date__gte=start_date, | ||||
|                     date__lte=end_date, | ||||
|                 ) | ||||
|             ] | ||||
|         ) | ||||
|         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["sum_cb"] = Refilling.objects.filter( | ||||
|             payment_method="CARD", | ||||
|             is_validated=True, | ||||
|             date__gte=start_date, | ||||
|             date__lte=end_date, | ||||
|         ).aggregate(res=Sum("amount", default=0))["res"] | ||||
|         kwargs["sum_cb"] += ( | ||||
|             Selling.objects.filter( | ||||
|                 payment_method="CARD", | ||||
|                 is_validated=True, | ||||
|                 date__gte=start_date, | ||||
|                 date__lte=end_date, | ||||
|             ) | ||||
|             .annotate(amount=F("unit_price") * F("quantity")) | ||||
|             .aggregate(res=Sum("amount", default=0))["res"] | ||||
|         ) | ||||
|         kwargs["start_date"] = start_date | ||||
|         kwargs["sums"] = ( | ||||
|             Selling.objects.values("club__name") | ||||
|             .annotate( | ||||
|                 selling_sum=Sum( | ||||
|                     Case( | ||||
|                         When( | ||||
|                             date__gte=start_date, | ||||
|                             date__lt=end_date, | ||||
|                             then=F("unit_price") * F("quantity"), | ||||
|                         ), | ||||
|                         output_field=CurrencyField(), | ||||
|                     ) | ||||
|                 ) | ||||
|             ) | ||||
|         kwargs["invoices"] = ( | ||||
|             Selling.objects.filter(date__gte=start_date, date__lt=end_date) | ||||
|             .values("club_id", "club__name") | ||||
|             .annotate(selling_sum=Sum(F("unit_price") * F("quantity"))) | ||||
|             .exclude(selling_sum=None) | ||||
|             .order_by("-selling_sum") | ||||
|         ) | ||||
|   | ||||
| @@ -4,7 +4,6 @@ | ||||
|         heading_level: 3 | ||||
|         members: | ||||
|             - TabedViewMixin | ||||
|             - QuickNotifMixin | ||||
|             - AllowFragment | ||||
|             - FragmentMixin | ||||
|             - UseFragmentsMixin | ||||
| @@ -17,7 +17,6 @@ document.addEventListener("alpine:init", () => { | ||||
|       this.$watch("basket", () => { | ||||
|         this.saveBasket(); | ||||
|       }); | ||||
|  | ||||
|       // Invalidate basket if a purchase was made | ||||
|       if (lastPurchaseTime !== null && localStorage.basketTimestamp !== undefined) { | ||||
|         if ( | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| {% from 'core/macros.jinja' import update_notifications %} | ||||
|  | ||||
| <div id=billing-infos-fragment> | ||||
|   <div | ||||
|     class="collapse" | ||||
| @@ -29,14 +31,6 @@ | ||||
|       > | ||||
|     </form> | ||||
|   </div> | ||||
|  | ||||
|   <br> | ||||
|  | ||||
|   {% if messages %} | ||||
|     {% for message in messages %} | ||||
|       <div class="alert alert-{{ message.tags }}"> | ||||
|         {{ message }} | ||||
|       </div> | ||||
|     {% endfor %} | ||||
|   {% endif %} | ||||
|   {{ update_notifications(messages) }} | ||||
| </div> | ||||
|   | ||||
| @@ -1,5 +1,9 @@ | ||||
| {% extends "core/base.jinja" %} | ||||
|  | ||||
| {% block notifications %} | ||||
|   {# Notifications are moved under the billing form #} | ||||
| {% endblock %} | ||||
|  | ||||
| {% block title %} | ||||
|   {% trans %}Basket state{% endtrans %} | ||||
| {% endblock %} | ||||
| @@ -56,6 +60,7 @@ | ||||
|       <div @htmx:after-request="fill"> | ||||
|         {{ billing_infos_form }} | ||||
|       </div> | ||||
|       {% include "core/base/notifications.jinja" %} | ||||
|       <form | ||||
|         method="post" | ||||
|         action="{{ settings.SITH_EBOUTIC_ET_URL }}" | ||||
|   | ||||
| @@ -22,14 +22,6 @@ | ||||
| {% block content %} | ||||
|   <h1 id="eboutic-title">{% trans %}Eboutic{% endtrans %}</h1> | ||||
|  | ||||
|   {% if messages %} | ||||
|     {% for message in messages %} | ||||
|       <div class="alert alert-{{ message.tags }}"> | ||||
|         {{ message }} | ||||
|       </div> | ||||
|     {% endfor %} | ||||
|   {% endif %} | ||||
|  | ||||
|   <div id="eboutic" x-data="basket({{ last_purchase_time }})"> | ||||
|     <div id="basket"> | ||||
|       <h3>Panier</h3> | ||||
|   | ||||
| @@ -4,14 +4,6 @@ | ||||
|   <h3>{% trans %}Eboutic{% endtrans %}</h3> | ||||
|  | ||||
|   <div> | ||||
|     {% if messages %} | ||||
|       {% for message in messages %} | ||||
|         <div class="alert alert-{{ message.tags }}"> | ||||
|           {{ message }} | ||||
|         </div> | ||||
|       {% endfor %} | ||||
|     {% endif %} | ||||
|  | ||||
|     {% if success %} | ||||
|       {% trans %}Payment successful{% endtrans %} | ||||
|     {% else %} | ||||
|   | ||||
| @@ -48,7 +48,7 @@ from django_countries.fields import Country | ||||
|  | ||||
| from core.auth.mixins import CanViewMixin | ||||
| from core.views.mixins import FragmentMixin, UseFragmentsMixin | ||||
| from counter.forms import BaseBasketForm, BillingInfoForm, ProductForm | ||||
| from counter.forms import BaseBasketForm, BasketProductForm, BillingInfoForm | ||||
| from counter.models import ( | ||||
|     BillingInfo, | ||||
|     Customer, | ||||
| @@ -78,7 +78,7 @@ class BaseEbouticBasketForm(BaseBasketForm): | ||||
|  | ||||
|  | ||||
| EbouticBasketForm = forms.formset_factory( | ||||
|     ProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 | ||||
|     BasketProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 | ||||
| ) | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										138
									
								
								galaxy/static/bundled/galaxy/galaxy-index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										138
									
								
								galaxy/static/bundled/galaxy/galaxy-index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,138 @@ | ||||
| import { default as ForceGraph3D } from "3d-force-graph"; | ||||
| import { forceX, forceY, forceZ } from "d3-force-3d"; | ||||
| // biome-ignore lint/style/noNamespaceImport: This is how it should be imported | ||||
| import * as Three from "three"; | ||||
| import SpriteText from "three-spritetext"; | ||||
|  | ||||
| /** | ||||
|  * @typedef GalaxyConfig | ||||
|  * @property {number} nodeId id of the current user node | ||||
|  * @property {string} dataUrl url to fetch the galaxy data from | ||||
|  **/ | ||||
|  | ||||
| /** | ||||
|  * Load the galaxy of an user | ||||
|  * @param {GalaxyConfig} config | ||||
|  **/ | ||||
| window.loadGalaxy = (config) => { | ||||
|   window.getNodeFromId = (id) => { | ||||
|     return Graph.graphData().nodes.find((n) => n.id === id); | ||||
|   }; | ||||
|  | ||||
|   window.getLinksFromNodeId = (id) => { | ||||
|     return Graph.graphData().links.filter( | ||||
|       (l) => l.source.id === id || l.target.id === id, | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   window.focusNode = (node) => { | ||||
|     highlightNodes.clear(); | ||||
|     highlightLinks.clear(); | ||||
|  | ||||
|     hoverNode = node || null; | ||||
|     if (node) { | ||||
|       // collect neighbors and links for highlighting | ||||
|       for (const link of window.getLinksFromNodeId(node.id)) { | ||||
|         highlightLinks.add(link); | ||||
|         highlightNodes.add(link.source); | ||||
|         highlightNodes.add(link.target); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     // refresh node and link display | ||||
|     Graph.nodeThreeObject(Graph.nodeThreeObject()) | ||||
|       .linkWidth(Graph.linkWidth()) | ||||
|       .linkDirectionalParticles(Graph.linkDirectionalParticles()); | ||||
|  | ||||
|     // Aim at node from outside it | ||||
|     const distance = 42; | ||||
|     const distRatio = 1 + distance / Math.hypot(node.x, node.y, node.z); | ||||
|  | ||||
|     const newPos = | ||||
|       node.x || node.y || node.z | ||||
|         ? { x: node.x * distRatio, y: node.y * distRatio, z: node.z * distRatio } | ||||
|         : { x: 0, y: 0, z: distance }; // special case if node is in (0,0,0) | ||||
|  | ||||
|     Graph.cameraPosition( | ||||
|       newPos, // new position | ||||
|       node, // lookAt ({ x, y, z }) | ||||
|       3000, // ms transition duration | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const highlightNodes = new Set(); | ||||
|   const highlightLinks = new Set(); | ||||
|   let hoverNode = null; | ||||
|  | ||||
|   const grpahDiv = document.getElementById("3d-graph"); | ||||
|   const Graph = ForceGraph3D(); | ||||
|   Graph(grpahDiv); | ||||
|   Graph.jsonUrl(config.dataUrl) | ||||
|     .width( | ||||
|       grpahDiv.parentElement.clientWidth > 1200 | ||||
|         ? 1200 | ||||
|         : grpahDiv.parentElement.clientWidth, | ||||
|     ) // Not perfect at all. JS-fu master from the future, please fix this :-) | ||||
|     .height(1000) | ||||
|     .enableNodeDrag(false) // allow easier navigation | ||||
|     .onNodeClick((node) => { | ||||
|       const camera = Graph.cameraPosition(); | ||||
|       const distance = Math.sqrt( | ||||
|         (node.x - camera.x) ** 2 + (node.y - camera.y) ** 2 + (node.z - camera.z) ** 2, | ||||
|       ); | ||||
|       if (distance < 120 || highlightNodes.has(node)) { | ||||
|         window.focusNode(node); | ||||
|       } | ||||
|     }) | ||||
|     .linkWidth((link) => (highlightLinks.has(link) ? 0.4 : 0.0)) | ||||
|     .linkColor((link) => | ||||
|       highlightLinks.has(link) ? "rgba(255,160,0,1)" : "rgba(128,255,255,0.6)", | ||||
|     ) | ||||
|     .linkVisibility((link) => highlightLinks.has(link)) | ||||
|     .nodeVisibility((node) => highlightNodes.has(node) || node.mass > 4) | ||||
|     // .linkDirectionalParticles(link => highlightLinks.has(link) ? 3 : 1) // kinda buggy for now, and slows this a bit, but would be great to help visualize lanes | ||||
|     .linkDirectionalParticleWidth(0.2) | ||||
|     .linkDirectionalParticleSpeed(-0.006) | ||||
|     .nodeThreeObject((node) => { | ||||
|       const sprite = new SpriteText(node.name); | ||||
|       sprite.material.depthWrite = false; // make sprite background transparent | ||||
|       sprite.color = highlightNodes.has(node) | ||||
|         ? node === hoverNode | ||||
|           ? "rgba(200,0,0,1)" | ||||
|           : "rgba(255,160,0,0.8)" | ||||
|         : "rgba(0,255,255,0.2)"; | ||||
|       sprite.textHeight = 2; | ||||
|       sprite.center = new Three.Vector2(1.2, 0.5); | ||||
|       return sprite; | ||||
|     }) | ||||
|     .onEngineStop(() => { | ||||
|       window.focusNode(window.getNodeFromId(config.nodeId)); | ||||
|       Graph.onEngineStop(() => { | ||||
|         /* nope */ | ||||
|       }); // don't call ourselves in a loop while moving the focus | ||||
|     }); | ||||
|  | ||||
|   // Set distance between stars | ||||
|   Graph.d3Force("link").distance((link) => link.value); | ||||
|  | ||||
|   // Set high masses nearer the center of the galaxy | ||||
|   // TODO: quick and dirty strength computation, this will need tuning. | ||||
|   Graph.d3Force( | ||||
|     "positionX", | ||||
|     forceX().strength((node) => { | ||||
|       return 1 - 1 / node.mass; | ||||
|     }), | ||||
|   ); | ||||
|   Graph.d3Force( | ||||
|     "positionY", | ||||
|     forceY().strength((node) => { | ||||
|       return 1 - 1 / node.mass; | ||||
|     }), | ||||
|   ); | ||||
|   Graph.d3Force( | ||||
|     "positionZ", | ||||
|     forceZ().strength((node) => { | ||||
|       return 1 - 1 / node.mass; | ||||
|     }), | ||||
|   ); | ||||
| }; | ||||
| @@ -1,137 +0,0 @@ | ||||
| import { exportToHtml } from "#core:utils/globals"; | ||||
|  | ||||
| import cytoscape from "cytoscape"; | ||||
| import d3Force, { type D3ForceLayoutOptions } from "cytoscape-d3-force"; | ||||
|  | ||||
| cytoscape.use(d3Force); | ||||
|  | ||||
| interface GalaxyConfig { | ||||
|   nodeId: number; | ||||
|   dataUrl: string; | ||||
| } | ||||
|  | ||||
| async function getGraphData(dataUrl: string) { | ||||
|   const response = await fetch(dataUrl); | ||||
|   if (!response.ok) { | ||||
|     return []; | ||||
|   } | ||||
|  | ||||
|   const content = await response.json(); | ||||
|   const nodes = content.nodes.map((node, i) => { | ||||
|     return { | ||||
|       group: "nodes", | ||||
|       data: { | ||||
|         id: node.id, | ||||
|         name: node.name, | ||||
|         mass: node.mass, | ||||
|       }, | ||||
|     }; | ||||
|   }); | ||||
|  | ||||
|   const edges = content.links.map((link) => { | ||||
|     return { | ||||
|       group: "edges", | ||||
|       data: { | ||||
|         id: `edge_${link.source}_${link.value}`, | ||||
|         source: link.source, | ||||
|         target: link.target, | ||||
|         value: link.value, | ||||
|       }, | ||||
|     }; | ||||
|   }); | ||||
|  | ||||
|   return { nodes: nodes, edges: edges }; | ||||
| } | ||||
|  | ||||
| exportToHtml("loadGalaxy", async (config: GalaxyConfig) => { | ||||
|   const graphDiv = document.getElementById("3d-graph"); | ||||
|   const elements = await getGraphData(config.dataUrl); | ||||
|   const cy = cytoscape({ | ||||
|     container: graphDiv, | ||||
|     elements: elements, | ||||
|     style: [ | ||||
|       { | ||||
|         selector: "node", | ||||
|         style: { | ||||
|           label: "data(name)", | ||||
|           "background-color": "red", | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         selector: ".focused", | ||||
|         style: { | ||||
|           "border-width": "5px", | ||||
|           "border-style": "solid", | ||||
|           "border-color": "black", | ||||
|           "target-arrow-color": "black", | ||||
|           "line-color": "black", | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         selector: "edge", | ||||
|         style: { | ||||
|           width: 0.1, | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         selector: ".direct", | ||||
|         style: { | ||||
|           width: "5px", | ||||
|           "line-color": "red", | ||||
|         }, | ||||
|       }, | ||||
|     ], | ||||
|     layout: { | ||||
|       name: "d3-force", | ||||
|       animate: true, | ||||
|       fit: false, | ||||
|       ungrabifyWhileSimulating: true, | ||||
|       fixedAfterDragging: true, | ||||
|  | ||||
|       linkId: (node) => { | ||||
|         return node.id; | ||||
|       }, | ||||
|  | ||||
|       linkDistance: (link) => { | ||||
|         return elements.nodes.length * 10; | ||||
|       }, | ||||
|  | ||||
|       linkStrength: (link) => { | ||||
|         return 1 / Math.max(1, link?.value); | ||||
|       }, | ||||
|  | ||||
|       linkIterations: 10, | ||||
|  | ||||
|       manyBodyStrength: (node) => { | ||||
|         return node?.mass; | ||||
|       }, | ||||
|  | ||||
|       // manyBodyDistanceMin: 500, | ||||
|       collideRadius: () => { | ||||
|         return 50; | ||||
|       }, | ||||
|  | ||||
|       ready: (e) => { | ||||
|         // Center on current user node at the start of the simulation | ||||
|         // Color all direct paths from that citizen to it's neighbor | ||||
|         const citizen = e.cy.nodes(`#${config.nodeId}`)[0]; | ||||
|         citizen.addClass("focused"); | ||||
|         citizen.connectedEdges().addClass("direct"); | ||||
|         e.cy.center(citizen); | ||||
|       }, | ||||
|  | ||||
|       tick: () => { | ||||
|         // Center on current user node during simulation | ||||
|         const citizen = cy.nodes(`#${config.nodeId}`)[0]; | ||||
|         cy.center(citizen); | ||||
|       }, | ||||
|  | ||||
|       stop: (e) => { | ||||
|         // Disable user grabbing of nodes | ||||
|         // This has to be disabled after the simulation is done | ||||
|         // Otherwise the simulation can't move nodes | ||||
|         e.cy.autolock(true); | ||||
|       }, | ||||
|     } as D3ForceLayoutOptions, | ||||
|   }); | ||||
| }); | ||||
| @@ -5,14 +5,14 @@ | ||||
| {% endblock %} | ||||
|  | ||||
| {% block additional_js %} | ||||
|   <script type="module" src="{{ static('bundled/galaxy/galaxy-index.ts') }}"></script> | ||||
|   <script type="module" src="{{ static('bundled/galaxy/galaxy-index.js') }}"></script> | ||||
| {% endblock %} | ||||
|  | ||||
|  | ||||
| {% block content %} | ||||
|   {% if object.current_star %} | ||||
|     <div style="display: flex; flex-wrap: wrap;"> | ||||
|       <div style="width: 100%; height: 70vh; display: block" id="3d-graph"></div> | ||||
|       <div id="3d-graph"></div> | ||||
|  | ||||
|       <div style="margin: 1em;"> | ||||
|         <p><a onclick="window.focusNode(window.getNodeFromId({{ object.id }}))">Reset on {{ object.get_display_name() }}</a></p> | ||||
|   | ||||
| @@ -6,7 +6,7 @@ | ||||
| msgid "" | ||||
| msgstr "" | ||||
| "Report-Msgid-Bugs-To: \n" | ||||
| "POT-Creation-Date: 2025-09-01 18:18+0200\n" | ||||
| "POT-Creation-Date: 2025-10-17 13:41+0200\n" | ||||
| "PO-Revision-Date: 2016-07-18\n" | ||||
| "Last-Translator: Maréchal <thomas.girod@utbm.fr\n" | ||||
| "Language-Team: AE info <ae.info@utbm.fr>\n" | ||||
| @@ -117,7 +117,7 @@ msgstr "S'abonner" | ||||
| msgid "Remove" | ||||
| msgstr "Retirer" | ||||
|  | ||||
| #: club/forms.py pedagogy/templates/pedagogy/moderation.jinja | ||||
| #: club/forms.py counter/forms.py pedagogy/templates/pedagogy/moderation.jinja | ||||
| msgid "Action" | ||||
| msgstr "Action" | ||||
|  | ||||
| @@ -174,12 +174,12 @@ msgid "You can not add the same user twice" | ||||
| msgstr "Vous ne pouvez pas ajouter deux fois le même utilisateur" | ||||
|  | ||||
| #: club/forms.py | ||||
| msgid "You should specify a role" | ||||
| msgstr "Vous devez choisir un rôle" | ||||
| msgid "You must be subscribed to join a club" | ||||
| msgstr "Vous devez être cotisant pour faire partie d'un club" | ||||
|  | ||||
| #: club/forms.py sas/forms.py | ||||
| msgid "You do not have the permission to do that" | ||||
| msgstr "Vous n'avez pas la permission de faire cela" | ||||
| #: club/forms.py | ||||
| msgid "You are already a member of this club" | ||||
| msgstr "Vous êtes déjà membre de ce club." | ||||
|  | ||||
| #: club/models.py | ||||
| msgid "slug name" | ||||
| @@ -350,11 +350,6 @@ msgstr "Depuis" | ||||
| msgid "There are no members in this club." | ||||
| msgstr "Il n'y a pas de membres dans ce club." | ||||
|  | ||||
| #: club/templates/club/club_members.jinja core/templates/core/file_detail.jinja | ||||
| #: core/views/forms.py trombi/templates/trombi/detail.jinja | ||||
| msgid "Add" | ||||
| msgstr "Ajouter" | ||||
|  | ||||
| #: club/templates/club/club_old_members.jinja | ||||
| msgid "Club old members" | ||||
| msgstr "Anciens membres du club" | ||||
| @@ -514,8 +509,8 @@ msgstr "Éditer le Trombi" | ||||
| msgid "New Trombi" | ||||
| msgstr "Nouveau Trombi" | ||||
|  | ||||
| #: club/templates/club/club_tools.jinja com/templates/com/poster_list.jinja | ||||
| #: core/templates/core/user_tools.jinja | ||||
| #: club/templates/club/club_tools.jinja club/views.py | ||||
| #: com/templates/com/poster_list.jinja core/templates/core/user_tools.jinja | ||||
| msgid "Posters" | ||||
| msgstr "Affiches" | ||||
|  | ||||
| @@ -561,6 +556,8 @@ msgstr "" | ||||
| #: core/templates/core/user_godfathers_tree.jinja | ||||
| #: core/templates/core/user_preferences.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 | ||||
| #: subscription/templates/subscription/fragments/creation_form.jinja | ||||
| #: trombi/templates/trombi/comment.jinja | ||||
| @@ -569,6 +566,24 @@ msgstr "" | ||||
| msgid "Save" | ||||
| msgstr "Sauver" | ||||
|  | ||||
| #: club/templates/club/fragments/add_member.jinja | ||||
| msgid "Add a new member" | ||||
| msgstr "Ajouter un nouveau membre" | ||||
|  | ||||
| #: club/templates/club/fragments/add_member.jinja | ||||
| msgid "Join club" | ||||
| msgstr "Rejoindre le club" | ||||
|  | ||||
| #: club/templates/club/fragments/add_member.jinja | ||||
| #: core/templates/core/file_detail.jinja core/views/forms.py | ||||
| #: trombi/templates/trombi/detail.jinja | ||||
| msgid "Add" | ||||
| msgstr "Ajouter" | ||||
|  | ||||
| #: club/templates/club/fragments/add_member.jinja | ||||
| msgid "Join" | ||||
| msgstr "Rejoindre" | ||||
|  | ||||
| #: club/templates/club/mailing.jinja | ||||
| msgid "Mailing lists" | ||||
| msgstr "Mailing listes" | ||||
| @@ -675,9 +690,14 @@ msgstr "Vente" | ||||
| msgid "Mailing list" | ||||
| msgstr "Listes de diffusion" | ||||
|  | ||||
| #: club/views.py com/views.py | ||||
| msgid "Posters list" | ||||
| msgstr "Liste d'affiches" | ||||
| #: club/views.py | ||||
| msgid "You are now a member of this club." | ||||
| msgstr "Vous êtes maintenant membre de ce club." | ||||
|  | ||||
| #: club/views.py | ||||
| #, python-format | ||||
| msgid "%(user)s has been added to club." | ||||
| msgstr "%(user)s a été ajouté au club." | ||||
|  | ||||
| #: com/forms.py | ||||
| msgid "Format: 16:9 | Resolution: 1920x1080" | ||||
| @@ -1043,6 +1063,10 @@ msgstr "Nos services" | ||||
| msgid "UV Guide" | ||||
| msgstr "Guide des UVs" | ||||
|  | ||||
| #: com/templates/com/news_list.jinja | ||||
| msgid "Timetable" | ||||
| msgstr "Emploi du temps" | ||||
|  | ||||
| #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | ||||
| msgid "Matmatronch" | ||||
| msgstr "Matmatronch" | ||||
| @@ -1107,6 +1131,10 @@ msgstr "Modération" | ||||
| msgid "No posters" | ||||
| msgstr "Aucune affiche" | ||||
|  | ||||
| #: com/templates/com/poster_list.jinja com/templates/com/screen_slideshow.jinja | ||||
| msgid "Click to expand" | ||||
| msgstr "Cliquez pour agrandir" | ||||
|  | ||||
| #: com/templates/com/poster_moderate.jinja | ||||
| msgid "Posters - moderation" | ||||
| msgstr "Affiches - modération" | ||||
| @@ -1164,14 +1192,6 @@ msgstr "Contenu" | ||||
| msgid "Add to weekmail" | ||||
| msgstr "Ajouter au Weekmail" | ||||
|  | ||||
| #: com/templates/com/weekmail.jinja | ||||
| msgid "Up" | ||||
| msgstr "Monter" | ||||
|  | ||||
| #: com/templates/com/weekmail.jinja | ||||
| msgid "Down" | ||||
| msgstr "Descendre" | ||||
|  | ||||
| #: com/templates/com/weekmail.jinja | ||||
| msgid "Articles included the next weekmail" | ||||
| msgstr "Article inclus dans le prochain Weekmail" | ||||
| @@ -1180,6 +1200,14 @@ msgstr "Article inclus dans le prochain Weekmail" | ||||
| msgid "Delete from weekmail" | ||||
| msgstr "Supprimer du Weekmail" | ||||
|  | ||||
| #: com/templates/com/weekmail.jinja | ||||
| msgid "Up" | ||||
| msgstr "Monter" | ||||
|  | ||||
| #: com/templates/com/weekmail.jinja | ||||
| msgid "Down" | ||||
| msgstr "Descendre" | ||||
|  | ||||
| #: com/templates/com/weekmail_preview.jinja | ||||
| #: core/templates/core/user_account_detail.jinja | ||||
| #: pedagogy/templates/pedagogy/uv_detail.jinja | ||||
| @@ -1249,6 +1277,10 @@ msgstr "Message d'info" | ||||
| msgid "Alert message" | ||||
| msgstr "Message d'alerte" | ||||
|  | ||||
| #: com/views.py | ||||
| msgid "Posters list" | ||||
| msgstr "Liste d'affiches" | ||||
|  | ||||
| #: com/views.py | ||||
| msgid "Screens list" | ||||
| msgstr "Liste d'écrans" | ||||
| @@ -1257,6 +1289,10 @@ msgstr "Liste d'écrans" | ||||
| msgid "All incoming events" | ||||
| msgstr "Tous les événements à venir" | ||||
|  | ||||
| #: com/views.py | ||||
| msgid "Weekmail sent successfully" | ||||
| msgstr "Weekmail envoyé avec succès" | ||||
|  | ||||
| #: com/views.py | ||||
| msgid "Delete and save to regenerate" | ||||
| msgstr "Supprimer et sauver pour régénérer" | ||||
| @@ -1265,6 +1301,26 @@ msgstr "Supprimer et sauver pour régénérer" | ||||
| msgid "Weekmail of the " | ||||
| msgstr "Weekmail du " | ||||
|  | ||||
| #: com/views.py | ||||
| #, python-format | ||||
| msgid "%(title)s moved up in the Weekmail" | ||||
| msgstr "%(title)s monté dans le Weekmail" | ||||
|  | ||||
| #: com/views.py | ||||
| #, python-format | ||||
| msgid "%(title)s moved down in the Weekmail" | ||||
| msgstr "%(title)s descendu dans le Weekmail" | ||||
|  | ||||
| #: com/views.py | ||||
| #, python-format | ||||
| msgid "%(title)s added to the Weekmail" | ||||
| msgstr "%(title)s ajouté dans Weekmail" | ||||
|  | ||||
| #: com/views.py | ||||
| #, python-format | ||||
| msgid "%(title)s removed from the Weekmail" | ||||
| msgstr "%(title)s retiré du Weekmail" | ||||
|  | ||||
| #: com/views.py | ||||
| msgid "" | ||||
| "You must be a board member of the selected club to post in the Weekmail." | ||||
| @@ -1272,6 +1328,11 @@ msgstr "" | ||||
| "Vous devez êtres un membre du bureau du club sélectionné pour poster dans le " | ||||
| "Weekmail." | ||||
|  | ||||
| #: core/auth/mixins.py | ||||
| #, python-format | ||||
| msgid "No club found with id %(id)s" | ||||
| msgstr "Pas de club avec l'id %(id)s trouvé" | ||||
|  | ||||
| #: core/models.py | ||||
| msgid "Is manually manageable" | ||||
| msgstr "Est gérable manuellement" | ||||
| @@ -1713,8 +1774,8 @@ msgid "" | ||||
| "AE UTBM is a voluntary organisation run by UTBM students. It organises " | ||||
| "student life at UTBM and manages its student facilities." | ||||
| msgstr "" | ||||
| "L'AE UTBM est une association bénévole gérée par les étudiants de " | ||||
| "l'UTBM. Elle organise la vie étudiante de l'UTBM et gère ses lieux de vie." | ||||
| "L'AE UTBM est une association bénévole gérée par les étudiants de l'UTBM. " | ||||
| "Elle organise la vie étudiante de l'UTBM et gère ses lieux de vie." | ||||
|  | ||||
| #: core/templates/core/base/footer.jinja core/templates/core/base/navbar.jinja | ||||
| msgid "Contacts" | ||||
| @@ -2157,10 +2218,6 @@ msgstr "" | ||||
| msgid "Page history" | ||||
| msgstr "Historique de la page" | ||||
|  | ||||
| #: core/templates/core/page_list.jinja | ||||
| msgid "There is no page in this website." | ||||
| msgstr "Il n'y a pas de page sur ce site web." | ||||
|  | ||||
| #: core/templates/core/page_prop.jinja | ||||
| msgid "Page properties" | ||||
| msgstr "Propriétés de la page" | ||||
| @@ -2339,6 +2396,10 @@ msgstr "Etickets" | ||||
| msgid "User has no account" | ||||
| msgstr "L'utilisateur n'a pas de compte" | ||||
|  | ||||
| #: core/templates/core/user_account_detail.jinja | ||||
| msgid "Deleted user" | ||||
| msgstr "Utilisateur supprimé" | ||||
|  | ||||
| #: core/templates/core/user_account_detail.jinja | ||||
| #: counter/templates/counter/last_ops.jinja | ||||
| #: counter/templates/counter/refilling_list.jinja | ||||
| @@ -2896,6 +2957,18 @@ msgstr "Cet UID est invalide" | ||||
| msgid "User not found" | ||||
| 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 | ||||
| msgid "" | ||||
| "Describe the product. If it's an event's click, give some insights about it, " | ||||
| @@ -3230,6 +3303,52 @@ msgid "The returnable product cannot be the same as the returned one" | ||||
| msgstr "" | ||||
| "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 | ||||
| #, python-format | ||||
| msgid "%(counter_name)s activity" | ||||
| @@ -3460,6 +3579,10 @@ msgstr "Payements en Carte Bancaire" | ||||
| msgid "Sum" | ||||
| msgstr "Somme" | ||||
|  | ||||
| #: counter/templates/counter/invoices_call.jinja | ||||
| msgid "Validated" | ||||
| msgstr "Validé" | ||||
|  | ||||
| #: counter/templates/counter/last_ops.jinja | ||||
| #, python-format | ||||
| msgid "%(counter_name)s last operations" | ||||
| @@ -3548,6 +3671,25 @@ msgstr "" | ||||
| "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." | ||||
|  | ||||
| #: 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 | ||||
| msgid "Product list" | ||||
| msgstr "Liste des produits" | ||||
| @@ -3730,6 +3872,10 @@ msgstr "L'utilisateur n'est pas barman." | ||||
| msgid "Bad location, someone is already logged in somewhere else" | ||||
| 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 | ||||
| msgid "Cash summary" | ||||
| msgstr "Relevé de caisse" | ||||
| @@ -4539,22 +4685,6 @@ msgstr "Signaler ce commentaire" | ||||
| msgid "Edit UE" | ||||
| msgstr "Éditer l'UE" | ||||
|  | ||||
| #: pedagogy/templates/pedagogy/uv_edit.jinja | ||||
| msgid "Import from UTBM" | ||||
| msgstr "Importer depuis l'UTBM" | ||||
|  | ||||
| #: pedagogy/templates/pedagogy/uv_edit.jinja | ||||
| msgid "Unknown UE code" | ||||
| msgstr "Code d'UE inconnu" | ||||
|  | ||||
| #: pedagogy/templates/pedagogy/uv_edit.jinja | ||||
| msgid "Successful autocomplete" | ||||
| msgstr "Autocomplétion réussite" | ||||
|  | ||||
| #: pedagogy/templates/pedagogy/uv_edit.jinja | ||||
| msgid "An error occurred: " | ||||
| msgstr "Une erreur est survenue : " | ||||
|  | ||||
| #: rootplace/forms.py | ||||
| msgid "User that will be kept" | ||||
| msgstr "Utilisateur qui sera conservé" | ||||
| @@ -4628,6 +4758,10 @@ msgstr "Pas de ban actif" | ||||
| msgid "Add a new album" | ||||
| msgstr "Ajouter un nouvel album" | ||||
|  | ||||
| #: sas/forms.py | ||||
| msgid "You do not have the permission to do that" | ||||
| msgstr "Vous n'avez pas la permission de faire cela" | ||||
|  | ||||
| #: sas/forms.py | ||||
| msgid "Upload images" | ||||
| msgstr "Envoyer les images" | ||||
| @@ -4818,8 +4952,8 @@ msgid "N/A" | ||||
| msgstr "N/A" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "Transfert" | ||||
| msgstr "Virement" | ||||
| msgid "AE account" | ||||
| msgstr "Compte AE" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "Belfort" | ||||
| @@ -5107,26 +5241,6 @@ msgstr "Vous avez acheté %s" | ||||
| msgid "You have a notification" | ||||
| msgstr "Vous avez une notification" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "Success!" | ||||
| msgstr "Succès !" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "Fail!" | ||||
| msgstr "Échec !" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "You successfully posted an article in the Weekmail" | ||||
| msgstr "Article posté avec succès dans le Weekmail" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "You successfully edited an article in the Weekmail" | ||||
| msgstr "Article édité avec succès dans le Weekmail" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "You successfully sent the Weekmail" | ||||
| msgstr "Weekmail envoyé avec succès" | ||||
|  | ||||
| #: sith/settings.py | ||||
| msgid "AE tee-shirt" | ||||
| msgstr "Tee-shirt AE" | ||||
| @@ -5135,6 +5249,10 @@ msgstr "Tee-shirt AE" | ||||
| msgid "A user with that email address already exists" | ||||
| msgstr "Un utilisateur avec cette adresse email existe déjà" | ||||
|  | ||||
| #: subscription/forms.py | ||||
| msgid "This user didn't fill its birthdate yet." | ||||
| msgstr "Cet utilisateur n'a pas encore renseigné sa date de naissance" | ||||
|  | ||||
| #: subscription/models.py | ||||
| msgid "Bad subscription type" | ||||
| msgstr "Mauvais type de cotisation" | ||||
| @@ -5163,6 +5281,14 @@ msgstr "lieu" | ||||
| msgid "You can not subscribe many time for the same period" | ||||
| msgstr "Vous ne pouvez pas cotiser plusieurs fois pour la même période" | ||||
|  | ||||
| #: subscription/templates/subscription/forms/create_existing_user.jinja | ||||
| msgid "" | ||||
| "If the subscription is done using the AE account, you must also click it on " | ||||
| "the AE counter." | ||||
| msgstr "" | ||||
| "Si la cotisation est faite en utilisant le compte AE, vous devez également " | ||||
| "la cliquer sur le comptoir AE." | ||||
|  | ||||
| #: subscription/templates/subscription/fragments/creation_success.jinja | ||||
| #, python-format | ||||
| msgid "Subscription created for %(user)s" | ||||
| @@ -5174,7 +5300,7 @@ msgid "" | ||||
| "%(user)s received its new %(type)s subscription. It will be active until " | ||||
| "%(end)s included." | ||||
| msgstr "" | ||||
| "%(user)s a reçu sa nouvelle cotisaton %(type)s. Elle sert active jusqu'au " | ||||
| "%(user)s a reçu sa nouvelle cotisaton %(type)s. Elle sera active jusqu'au " | ||||
| "%(end)s inclu." | ||||
|  | ||||
| #: subscription/templates/subscription/fragments/creation_success.jinja | ||||
| @@ -5201,6 +5327,18 @@ msgstr "Membre existant" | ||||
| msgid "the groups that can create subscriptions" | ||||
| msgstr "les groupes pouvant créer des cotisations" | ||||
|  | ||||
| #: timetable/templates/timetable/generator.jinja | ||||
| msgid "Timetable generator" | ||||
| msgstr "Générateur d'emploi du temps" | ||||
|  | ||||
| #: timetable/templates/timetable/generator.jinja | ||||
| msgid "Generate" | ||||
| msgstr "Générer" | ||||
|  | ||||
| #: timetable/templates/timetable/generator.jinja | ||||
| msgid "Save to PNG" | ||||
| msgstr "Sauver en PNG" | ||||
|  | ||||
| #: trombi/models.py | ||||
| msgid "subscription deadline" | ||||
| msgstr "fin des inscriptions" | ||||
| @@ -5426,10 +5564,38 @@ msgstr "Mes photos" | ||||
| msgid "Admin tools" | ||||
| msgstr "Admin Trombi" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Trombi modified" | ||||
| msgstr "Trombi modifié" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "User added to the trombi" | ||||
| msgstr "Utilisateur ajouté au trombi" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "User couldn't be added to the trombi" | ||||
| msgstr "L'utilisateur n'a pas pu être ajouté au trombi" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "User removed from the trombi" | ||||
| msgstr "Utilisateur retiré du trombi" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Explain why you rejected the comment" | ||||
| msgstr "Expliquez pourquoi vous refusez le commentaire" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Comment accepted" | ||||
| msgstr "Commentaire accepté" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Comment rejected" | ||||
| msgstr "Commentaire rejeté" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Comment removed" | ||||
| msgstr "Commentaire retiré" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Rejected comment" | ||||
| msgstr "Commentaire rejeté" | ||||
| @@ -5470,6 +5636,10 @@ msgstr "" | ||||
| "pouvez vous inscrire qu'à un seul Trombi, donc ne jouez pas avec cet option " | ||||
| "ou vous encourerez la colère des admins!" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "User modified" | ||||
| msgstr "Utilisateur modifié" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Personal email (not UTBM)" | ||||
| msgstr "Email personnel (pas UTBM)" | ||||
| @@ -5482,6 +5652,14 @@ msgstr "Téléphone" | ||||
| msgid "Native town" | ||||
| msgstr "Ville d'origine" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "User removed from trombi" | ||||
| msgstr "Utilisateur retiré du trombi" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "Comment added" | ||||
| msgstr "Commentaire ajouté" | ||||
|  | ||||
| #: trombi/views.py | ||||
| msgid "" | ||||
| "You can not yet write comment, you must wait for the subscription deadline " | ||||
|   | ||||
							
								
								
									
										130
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										130
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -25,13 +25,12 @@ | ||||
|         "country-flag-emoji-polyfill": "^0.1.8", | ||||
|         "cytoscape": "^3.30.2", | ||||
|         "cytoscape-cxtmenu": "^3.5.0", | ||||
|         "cytoscape-d3-force": "^1.1.4", | ||||
|         "cytoscape-klay": "^3.1.4", | ||||
|         "d3-force-3d": "^3.0.5", | ||||
|         "easymde": "^2.19.0", | ||||
|         "glob": "^11.0.0", | ||||
|         "html2canvas": "^1.4.1", | ||||
|         "htmx.org": "^2.0.3", | ||||
|         "jquery": "^3.7.1", | ||||
|         "js-cookie": "^3.0.5", | ||||
|         "lit-html": "^3.3.0", | ||||
|         "native-file-system-adapter": "^3.0.1", | ||||
| @@ -47,12 +46,10 @@ | ||||
|         "@rollup/plugin-inject": "^5.0.5", | ||||
|         "@types/alpinejs": "^3.13.10", | ||||
|         "@types/cytoscape-cxtmenu": "^3.4.4", | ||||
|         "@types/cytoscape-d3-force": "^1.0.0", | ||||
|         "@types/cytoscape-klay": "^3.1.4", | ||||
|         "@types/jquery": "^3.5.31", | ||||
|         "@types/js-cookie": "^3.0.6", | ||||
|         "typescript": "^5.8.3", | ||||
|         "vite": "^6.2.6", | ||||
|         "vite": "^6.3.6", | ||||
|         "vite-bundle-visualizer": "^1.2.1", | ||||
|         "vite-plugin-static-copy": "^3.1.2" | ||||
|       } | ||||
| @@ -2875,16 +2872,6 @@ | ||||
|         "@types/cytoscape": "*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@types/cytoscape-d3-force": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/@types/cytoscape-d3-force/-/cytoscape-d3-force-1.0.0.tgz", | ||||
|       "integrity": "sha512-1eRd9xr/DvJ4MIA5lCEG8DMX2Ha87qAbpP7irpuKZun0ZCBQPpoOBo9mPl0WrkJbXH+hHwG8s3E2CpUz3HxLrw==", | ||||
|       "dev": true, | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@types/cytoscape": "^3.0.9" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@types/cytoscape-klay": { | ||||
|       "version": "3.1.4", | ||||
|       "resolved": "https://registry.npmjs.org/@types/cytoscape-klay/-/cytoscape-klay-3.1.4.tgz", | ||||
| @@ -2901,16 +2888,6 @@ | ||||
|       "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/@types/jquery": { | ||||
|       "version": "3.5.33", | ||||
|       "resolved": "https://registry.npmjs.org/@types/jquery/-/jquery-3.5.33.tgz", | ||||
|       "integrity": "sha512-SeyVJXlCZpEki5F0ghuYe+L+PprQta6nRZqhONt9F13dWBtR/ftoaIbdRQ7cis7womE+X2LKhsDdDtkkDhJS6g==", | ||||
|       "dev": true, | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@types/sizzle": "*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@types/js-cookie": { | ||||
|       "version": "3.0.6", | ||||
|       "resolved": "https://registry.npmjs.org/@types/js-cookie/-/js-cookie-3.0.6.tgz", | ||||
| @@ -2931,13 +2908,6 @@ | ||||
|       "integrity": "sha512-a79Yc3TOk6dGdituy8hmTTJXjOkZ7zsFYV10L337ttq/rec8lRMDBpV7fL3uLx6TgbFCa5DU/h8FmIBQPSbU0w==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/@types/sizzle": { | ||||
|       "version": "2.3.10", | ||||
|       "resolved": "https://registry.npmjs.org/@types/sizzle/-/sizzle-2.3.10.tgz", | ||||
|       "integrity": "sha512-TC0dmN0K8YcWEAEfiPi5gJP14eJe30TTGjkvek3iM/1NdHHsdCA/Td6GvNndMOo/iSnIsZ4HuuhrYPDAmbxzww==", | ||||
|       "dev": true, | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/@types/tern": { | ||||
|       "version": "0.23.9", | ||||
|       "resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.9.tgz", | ||||
| @@ -3136,6 +3106,15 @@ | ||||
|         "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/base64-arraybuffer": { | ||||
|       "version": "1.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz", | ||||
|       "integrity": "sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">= 0.6.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/binary-extensions": { | ||||
|       "version": "2.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", | ||||
| @@ -3524,6 +3503,15 @@ | ||||
|         "node": ">= 8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/css-line-break": { | ||||
|       "version": "2.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/css-line-break/-/css-line-break-2.1.0.tgz", | ||||
|       "integrity": "sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "utrie": "^1.0.2" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/cytoscape": { | ||||
|       "version": "3.33.1", | ||||
|       "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.33.1.tgz", | ||||
| @@ -3542,18 +3530,6 @@ | ||||
|         "cytoscape": "^3.2.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/cytoscape-d3-force": { | ||||
|       "version": "1.1.4", | ||||
|       "resolved": "https://registry.npmjs.org/cytoscape-d3-force/-/cytoscape-d3-force-1.1.4.tgz", | ||||
|       "integrity": "sha512-8NjI/yEoB3YqVsdf7ud7Oh8Kyi+C9Lhh1fICmtemIo6EC1ZUtm8KcPNLkQySYO8nRS2mQKj5eVdCr7W0L8ONoQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "d3-force": "^2.0.1" | ||||
|       }, | ||||
|       "peerDependencies": { | ||||
|         "cytoscape": "^3.2.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/cytoscape-klay": { | ||||
|       "version": "3.1.4", | ||||
|       "resolved": "https://registry.npmjs.org/cytoscape-klay/-/cytoscape-klay-3.1.4.tgz", | ||||
| @@ -3602,17 +3578,6 @@ | ||||
|         "node": ">=12" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/d3-force": { | ||||
|       "version": "2.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-2.1.1.tgz", | ||||
|       "integrity": "sha512-nAuHEzBqMvpFVMf9OX75d00OxvOXdxY+xECIXjW6Gv8BRrXu6gAWbv/9XKrvfJ5i5DCokDW7RYE50LRoK092ew==", | ||||
|       "license": "BSD-3-Clause", | ||||
|       "dependencies": { | ||||
|         "d3-dispatch": "1 - 2", | ||||
|         "d3-quadtree": "1 - 2", | ||||
|         "d3-timer": "1 - 2" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/d3-force-3d": { | ||||
|       "version": "3.0.6", | ||||
|       "resolved": "https://registry.npmjs.org/d3-force-3d/-/d3-force-3d-3.0.6.tgz", | ||||
| @@ -3629,24 +3594,6 @@ | ||||
|         "node": ">=12" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/d3-force/node_modules/d3-dispatch": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-2.0.0.tgz", | ||||
|       "integrity": "sha512-S/m2VsXI7gAti2pBoLClFFTMOO1HTtT0j99AuXLoGFKO6deHDdnv6ZGTxSTTUTgO1zVcv82fCOtDjYK4EECmWA==", | ||||
|       "license": "BSD-3-Clause" | ||||
|     }, | ||||
|     "node_modules/d3-force/node_modules/d3-quadtree": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-2.0.0.tgz", | ||||
|       "integrity": "sha512-b0Ed2t1UUalJpc3qXzKi+cPGxeXRr4KU9YSlocN74aTzp6R/Ud43t79yLLqxHRWZfsvWXmbDWPpoENK1K539xw==", | ||||
|       "license": "BSD-3-Clause" | ||||
|     }, | ||||
|     "node_modules/d3-force/node_modules/d3-timer": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-2.0.0.tgz", | ||||
|       "integrity": "sha512-TO4VLh0/420Y/9dO3+f9abDEFYeCUr2WZRlxJvbp4HPTQcSylXNiL6yZa9FIUvV1yRiFufl1bszTCLDqv9PWNA==", | ||||
|       "license": "BSD-3-Clause" | ||||
|     }, | ||||
|     "node_modules/d3-format": { | ||||
|       "version": "3.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.0.tgz", | ||||
| @@ -4237,6 +4184,19 @@ | ||||
|         "node": ">= 0.4" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/html2canvas": { | ||||
|       "version": "1.4.1", | ||||
|       "resolved": "https://registry.npmjs.org/html2canvas/-/html2canvas-1.4.1.tgz", | ||||
|       "integrity": "sha512-fPU6BHNpsyIhr8yyMpTLLxAbkaK8ArIBcmZIRiBLiDhjeqvXolaEmDGmELFuX9I4xDcaKKcJl+TKZLqruBbmWA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "css-line-break": "^2.1.0", | ||||
|         "text-segmentation": "^1.0.3" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/htmx.org": { | ||||
|       "version": "2.0.6", | ||||
|       "resolved": "https://registry.npmjs.org/htmx.org/-/htmx.org-2.0.6.tgz", | ||||
| @@ -4437,12 +4397,6 @@ | ||||
|         "jiti": "lib/jiti-cli.mjs" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/jquery": { | ||||
|       "version": "3.7.1", | ||||
|       "resolved": "https://registry.npmjs.org/jquery/-/jquery-3.7.1.tgz", | ||||
|       "integrity": "sha512-m4avr8yL8kmFN8psrbFFFmB/If14iN5o9nw/NgnnM+kybDJpRsAynV2BsfpTYrTRysYUdADVD7CkUUizgkpLfg==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/js-cookie": { | ||||
|       "version": "3.0.5", | ||||
|       "resolved": "https://registry.npmjs.org/js-cookie/-/js-cookie-3.0.5.tgz", | ||||
| @@ -5532,6 +5486,15 @@ | ||||
|       "dev": true, | ||||
|       "license": "ISC" | ||||
|     }, | ||||
|     "node_modules/text-segmentation": { | ||||
|       "version": "1.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/text-segmentation/-/text-segmentation-1.0.3.tgz", | ||||
|       "integrity": "sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "utrie": "^1.0.2" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/three": { | ||||
|       "version": "0.177.0", | ||||
|       "resolved": "https://registry.npmjs.org/three/-/three-0.177.0.tgz", | ||||
| @@ -5789,6 +5752,15 @@ | ||||
|         "browserslist": ">= 4.21.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/utrie": { | ||||
|       "version": "1.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/utrie/-/utrie-1.0.2.tgz", | ||||
|       "integrity": "sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "base64-arraybuffer": "^1.0.2" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/vite": { | ||||
|       "version": "6.3.6", | ||||
|       "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.6.tgz", | ||||
|   | ||||
| @@ -31,12 +31,10 @@ | ||||
|     "@rollup/plugin-inject": "^5.0.5", | ||||
|     "@types/alpinejs": "^3.13.10", | ||||
|     "@types/cytoscape-cxtmenu": "^3.4.4", | ||||
|     "@types/cytoscape-d3-force": "^1.0.0", | ||||
|     "@types/cytoscape-klay": "^3.1.4", | ||||
|     "@types/jquery": "^3.5.31", | ||||
|     "@types/js-cookie": "^3.0.6", | ||||
|     "typescript": "^5.8.3", | ||||
|     "vite": "^6.2.6", | ||||
|     "vite": "^6.3.6", | ||||
|     "vite-bundle-visualizer": "^1.2.1", | ||||
|     "vite-plugin-static-copy": "^3.1.2" | ||||
|   }, | ||||
| @@ -57,13 +55,12 @@ | ||||
|     "country-flag-emoji-polyfill": "^0.1.8", | ||||
|     "cytoscape": "^3.30.2", | ||||
|     "cytoscape-cxtmenu": "^3.5.0", | ||||
|     "cytoscape-d3-force": "^1.1.4", | ||||
|     "cytoscape-klay": "^3.1.4", | ||||
|     "d3-force-3d": "^3.0.5", | ||||
|     "easymde": "^2.19.0", | ||||
|     "glob": "^11.0.0", | ||||
|     "html2canvas": "^1.4.1", | ||||
|     "htmx.org": "^2.0.3", | ||||
|     "jquery": "^3.7.1", | ||||
|     "js-cookie": "^3.0.5", | ||||
|     "lit-html": "^3.3.0", | ||||
|     "native-file-system-adapter": "^3.0.1", | ||||
|   | ||||
| @@ -13,16 +13,15 @@ | ||||
| {% block content %} | ||||
|   <div class="pedagogy"> | ||||
|     <div id="uv_detail"> | ||||
|       <p id="return_noscript"><a href="{{ url('pedagogy:guide') }}">{% trans %}Back{% endtrans %}</a></p> | ||||
|       <button id="return_js" onclick='(function(){ | ||||
|                                       // If comes from the guide page, go back with history | ||||
|                                       if (document.referrer.replace(/\?(.+)/gm,"").endsWith(`{{ url("pedagogy:guide") }}`)){ | ||||
|                                       window.history.back(); | ||||
|                                       return; | ||||
|                                       } | ||||
|                                       // Simply goes to the guide page | ||||
|                                       window.location.href = `{{ url("pedagogy:guide") }}`; | ||||
|                                       })()' hidden>{% trans %}Back{% endtrans %}</button> | ||||
|       <button onclick='(function(){ | ||||
|                        // If comes from the guide page, go back with history | ||||
|                        if (document.referrer.replace(/\?(.+)/gm,"").endsWith(`{{ url("pedagogy:guide") }}`)){ | ||||
|                        window.history.back(); | ||||
|                        return; | ||||
|                        } | ||||
|                        // Simply goes to the guide page | ||||
|                        window.location.href = `{{ url("pedagogy:guide") }}`; | ||||
|                        })()' hidden>{% trans %}Back{% endtrans %}</button> | ||||
|  | ||||
|       <h1>{{ object.code }} - {{ object.title }}</h1> | ||||
|       <br> | ||||
| @@ -217,9 +216,4 @@ | ||||
|  | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <script type="text/javascript"> | ||||
|     $("#return_noscript").hide(); | ||||
|     $("#return_js").show(); | ||||
|   </script> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -21,11 +21,6 @@ | ||||
|           {{ field.errors }} | ||||
|           <label for="{{ field.name }}">{{ field.label }}</label> | ||||
|           {{ field }} | ||||
|  | ||||
|  | ||||
|           {% if field.name == 'code' %} | ||||
|             <button type="button" id="autofill">{% trans %}Import from UTBM{% endtrans %}</button> | ||||
|           {% endif %} | ||||
|         </p> | ||||
|       {% endif %} | ||||
|  | ||||
| @@ -36,48 +31,3 @@ | ||||
|     <p><input type="submit" value="{% trans %}Update{% endtrans %}" /></p> | ||||
|   </form> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block script %} | ||||
|   {{ super() }} | ||||
|  | ||||
|   <script type="text/javascript"> | ||||
|     document.addEventListener('DOMContentLoaded', function() { | ||||
|       const autofillBtn = document.getElementById('autofill') | ||||
|       const codeInput = document.querySelector('input[name="code"]') | ||||
|  | ||||
|       autofillBtn.addEventListener('click', () => { | ||||
|         const url = `/api/uv/${codeInput.value}`; | ||||
|         deleteQuickNotifs() | ||||
|  | ||||
|         $.ajax({ | ||||
|           dataType: "json", | ||||
|           url: url, | ||||
|           success: function(data, _, xhr) { | ||||
|             if (xhr.status !== 200) { | ||||
|               createQuickNotif("{% trans %}Unknown UE code{% endtrans %}") | ||||
|               return | ||||
|             } | ||||
|             Object.entries(data) | ||||
|               .filter(([_, val]) => !!val)  // skip entries with null or undefined value | ||||
|               .map(([key, val]) => {  // convert keys to DOM elements | ||||
|                 return [document.querySelector('[name="' + key + '"]'), val]; | ||||
|               }) | ||||
|               .filter(([elem, _]) => !!elem)  // skip non-existing DOM elements | ||||
|               .forEach(([elem, val]) => {  // write the value in the form field | ||||
|                 if (elem.tagName === 'TEXTAREA') { | ||||
|                   // MD editor text input | ||||
|                   elem.parentNode.querySelector('.CodeMirror').CodeMirror.setValue(val); | ||||
|                 } else { | ||||
|                   elem.value = val; | ||||
|                 } | ||||
|               }); | ||||
|             createQuickNotif('{% trans %}Successful autocomplete{% endtrans %}') | ||||
|           }, | ||||
|           error: function(_, _, statusMessage) { | ||||
|             createQuickNotif('{% trans %}An error occurred: {% endtrans %}' + statusMessage) | ||||
|           }, | ||||
|         }) | ||||
|       }) | ||||
|     }) | ||||
|   </script> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -22,7 +22,7 @@ dependencies = [ | ||||
|     "django>=5.2.1,<6.0.0", | ||||
|     "django-ninja<2.0.0,>=1.4.0", | ||||
|     "django-ninja-extra<1.0.0,>=0.22.9", | ||||
|     "Pillow<12.0.0,>=11.1.0", | ||||
|     "Pillow>=11.1.0,<13.0.0", | ||||
|     "mistune<4.0.0,>=3.1.3", | ||||
|     "django-jinja<3.0.0,>=2.11.0", | ||||
|     "cryptography>=45.0.3,<46.0.0", | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import { | ||||
|  | ||||
| interface PagePictureConfig { | ||||
|   userId: number; | ||||
|   nbPictures?: number; | ||||
| } | ||||
|  | ||||
| interface Album { | ||||
| @@ -20,11 +21,27 @@ document.addEventListener("alpine:init", () => { | ||||
|     loading: true, | ||||
|     albums: [] as Album[], | ||||
|  | ||||
|     async init() { | ||||
|     async fetchPictures(): Promise<PictureSchema[]> { | ||||
|       const localStorageKey = `user${config.userId}Pictures`; | ||||
|       const localStorageInvalidationKey = `user${config.userId}PicturesNumber`; | ||||
|       const lastCachedNumber = localStorage.getItem(localStorageInvalidationKey); | ||||
|       if ( | ||||
|         lastCachedNumber !== null && | ||||
|         Number.parseInt(lastCachedNumber) === config.nbPictures | ||||
|       ) { | ||||
|         return JSON.parse(localStorage.getItem(localStorageKey)); | ||||
|       } | ||||
|       const pictures = await paginated(picturesFetchPictures, { | ||||
|         // biome-ignore lint/style/useNamingConvention: from python api | ||||
|         query: { users_identified: [config.userId] }, | ||||
|       } 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); | ||||
|       this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => { | ||||
|         return { | ||||
|   | ||||
| @@ -309,6 +309,7 @@ exportToHtml("loadViewer", (config: ViewerConfig) => { | ||||
|         // Clear selection and cache of retrieved user so they can be filtered again | ||||
|         widget.clear(false); | ||||
|         widget.clearOptions(); | ||||
|         widget.setTextboxValue(""); | ||||
|       }, | ||||
|  | ||||
|       /** | ||||
|   | ||||
| @@ -15,7 +15,7 @@ | ||||
| {% endblock %} | ||||
|  | ||||
| {% block content %} | ||||
|   <main x-data="user_pictures({ userId: {{ object.id }} })"> | ||||
|   <main x-data="user_pictures({ userId: {{ object.id }}, nbPictures: {{ object.nb_pictures }} })"> | ||||
|     {% if user.id == object.id %} | ||||
|       {{ download_button(_("Download all my pictures")) }} | ||||
|     {% endif %} | ||||
|   | ||||
							
								
								
									
										10
									
								
								sas/views.py
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								sas/views.py
									
									
									
									
									
								
							| @@ -16,6 +16,7 @@ from typing import Any | ||||
|  | ||||
| from django.conf import settings | ||||
| from django.core.exceptions import PermissionDenied | ||||
| from django.db.models import Count, OuterRef, Subquery | ||||
| from django.http import Http404, HttpResponseRedirect | ||||
| from django.shortcuts import get_object_or_404 | ||||
| from django.urls import reverse | ||||
| @@ -36,7 +37,7 @@ from sas.forms import ( | ||||
|     PictureModerationRequestForm, | ||||
|     PictureUploadForm, | ||||
| ) | ||||
| from sas.models import Album, Picture | ||||
| from sas.models import Album, PeoplePictureRelation, Picture | ||||
|  | ||||
|  | ||||
| class AlbumCreateFragment(FragmentMixin, CreateView): | ||||
| @@ -178,6 +179,13 @@ class UserPicturesView(UserTabsMixin, CanViewMixin, DetailView): | ||||
|     context_object_name = "profile" | ||||
|     template_name = "sas/user_pictures.jinja" | ||||
|     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 | ||||
|   | ||||
| @@ -125,6 +125,7 @@ INSTALLED_APPS = ( | ||||
|     "pedagogy", | ||||
|     "galaxy", | ||||
|     "antispam", | ||||
|     "timetable", | ||||
|     "api", | ||||
| ) | ||||
|  | ||||
| @@ -405,9 +406,6 @@ SITH_FORUM_PAGE_LENGTH = 30 | ||||
| SITH_SAS_ROOT_DIR_ID = env.int("SITH_SAS_ROOT_DIR_ID", default=4) | ||||
| SITH_SAS_IMAGES_PER_PAGE = 60 | ||||
|  | ||||
| SITH_BOARD_SUFFIX = "-bureau" | ||||
| SITH_MEMBER_SUFFIX = "-membres" | ||||
|  | ||||
| SITH_PROFILE_DEPARTMENTS = [ | ||||
|     ("TC", _("TC")), | ||||
|     ("IMSI", _("IMSI")), | ||||
| @@ -424,18 +422,11 @@ SITH_PROFILE_DEPARTMENTS = [ | ||||
|     ("NA", _("N/A")), | ||||
| ] | ||||
|  | ||||
| SITH_ACCOUNTING_PAYMENT_METHOD = [ | ||||
|     ("CHECK", _("Check")), | ||||
|     ("CASH", _("Cash")), | ||||
|     ("TRANSFERT", _("Transfert")), | ||||
|     ("CARD", _("Credit card")), | ||||
| ] | ||||
|  | ||||
| SITH_SUBSCRIPTION_PAYMENT_METHOD = [ | ||||
|     ("CHECK", _("Check")), | ||||
|     ("CARD", _("Credit card")), | ||||
|     ("CASH", _("Cash")), | ||||
|     ("EBOUTIC", _("Eboutic")), | ||||
|     ("AE_ACCOUNT", _("AE account")), | ||||
|     ("OTHER", _("Other")), | ||||
| ] | ||||
|  | ||||
| @@ -444,6 +435,7 @@ SITH_SUBSCRIPTION_LOCATIONS = [ | ||||
|     ("SEVENANS", _("Sevenans")), | ||||
|     ("MONTBELIARD", _("Montbéliard")), | ||||
|     ("EBOUTIC", _("Eboutic")), | ||||
|     ("OTHER", _("Other")), | ||||
| ] | ||||
|  | ||||
| SITH_COUNTER_BARS = [(1, "MDE"), (2, "Foyer"), (35, "La Gommette")] | ||||
| @@ -694,14 +686,6 @@ SITH_PERMANENT_NOTIFICATIONS = { | ||||
|     "SAS_MODERATION": "sas.models.sas_notification_callback", | ||||
| } | ||||
|  | ||||
| SITH_QUICK_NOTIF = { | ||||
|     "qn_success": _("Success!"), | ||||
|     "qn_fail": _("Fail!"), | ||||
|     "qn_weekmail_new_article": _("You successfully posted an article in the Weekmail"), | ||||
|     "qn_weekmail_article_edit": _("You successfully edited an article in the Weekmail"), | ||||
|     "qn_weekmail_send_success": _("You successfully sent the Weekmail"), | ||||
| } | ||||
|  | ||||
| # Mailing related settings | ||||
|  | ||||
| SITH_MAILING_DOMAIN = "utbm.fr" | ||||
|   | ||||
| @@ -53,6 +53,7 @@ urlpatterns = [ | ||||
|     path("i18n/", include("django.conf.urls.i18n")), | ||||
|     path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"), | ||||
|     path("captcha/", include("captcha.urls")), | ||||
|     path("edt/", include(("timetable.urls", "timetable"), namespace="timetable")), | ||||
| ] | ||||
|  | ||||
| if settings.DEBUG: | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import secrets | ||||
| from typing import Any | ||||
|  | ||||
| from django import forms | ||||
| from django.conf import settings | ||||
| from django.core.exceptions import ValidationError | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
|  | ||||
| @@ -23,13 +24,28 @@ class SelectionDateForm(forms.Form): | ||||
|  | ||||
|  | ||||
| class SubscriptionForm(forms.ModelForm): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         initial = kwargs.pop("initial", {}) | ||||
|     allowed_payment_methods = ["CARD", "CASH", "AE_ACCOUNT"] | ||||
|  | ||||
|     class Meta: | ||||
|         model = Subscription | ||||
|         fields = ["subscription_type", "payment_method", "location"] | ||||
|         widgets = {"payment_method": forms.RadioSelect} | ||||
|  | ||||
|     def __init__(self, *args, initial=None, **kwargs): | ||||
|         initial = initial or {} | ||||
|         if "subscription_type" not in initial: | ||||
|             initial["subscription_type"] = "deux-semestres" | ||||
|         if "payment_method" not in initial: | ||||
|             initial["payment_method"] = "CARD" | ||||
|         super().__init__(*args, initial=initial, **kwargs) | ||||
|         self.fields["payment_method"].choices = [ | ||||
|             m | ||||
|             for m in settings.SITH_SUBSCRIPTION_PAYMENT_METHOD | ||||
|             if m[0] in self.allowed_payment_methods | ||||
|         ] | ||||
|         self.fields["location"].choices = [ | ||||
|             m for m in settings.SITH_SUBSCRIPTION_LOCATIONS if m[0] != "EBOUTIC" | ||||
|         ] | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|         if self.errors: | ||||
| @@ -61,7 +77,8 @@ class SubscriptionNewUserForm(SubscriptionForm): | ||||
|         assert user.is_subscribed | ||||
|     """ | ||||
|  | ||||
|     template_name = "subscription/forms/create_new_user.html" | ||||
|     allowed_payment_methods = ["CARD", "CASH"] | ||||
|     template_name = "subscription/forms/create_new_user.jinja" | ||||
|  | ||||
|     __user_fields = forms.fields_for_model( | ||||
|         User, | ||||
| @@ -73,10 +90,6 @@ class SubscriptionNewUserForm(SubscriptionForm): | ||||
|     email = __user_fields["email"] | ||||
|     date_of_birth = __user_fields["date_of_birth"] | ||||
|  | ||||
|     class Meta: | ||||
|         model = Subscription | ||||
|         fields = ["subscription_type", "payment_method", "location"] | ||||
|  | ||||
|     field_order = [ | ||||
|         "first_name", | ||||
|         "last_name", | ||||
| @@ -130,9 +143,57 @@ class SubscriptionNewUserForm(SubscriptionForm): | ||||
| class SubscriptionExistingUserForm(SubscriptionForm): | ||||
|     """Form to add a subscription to an existing user.""" | ||||
|  | ||||
|     template_name = "subscription/forms/create_existing_user.html" | ||||
|     template_name = "subscription/forms/create_existing_user.jinja" | ||||
|     required_css_class = "required" | ||||
|  | ||||
|     class Meta: | ||||
|         model = Subscription | ||||
|         fields = ["member", "subscription_type", "payment_method", "location"] | ||||
|         widgets = {"member": AutoCompleteSelectUser} | ||||
|     birthdate = forms.fields_for_model( | ||||
|         User, | ||||
|         ["date_of_birth"], | ||||
|         widgets={"date_of_birth": SelectDate(attrs={"hidden": True})}, | ||||
|         help_texts={"date_of_birth": _("This user didn't fill its birthdate yet.")}, | ||||
|     )["date_of_birth"] | ||||
|  | ||||
|     class Meta(SubscriptionForm.Meta): | ||||
|         fields = ["member", *SubscriptionForm.Meta.fields] | ||||
|         widgets = SubscriptionForm.Meta.widgets | {"member": AutoCompleteSelectUser} | ||||
|  | ||||
|     field_order = [ | ||||
|         "member", | ||||
|         "birthdate", | ||||
|         "subscription_type", | ||||
|         "payment_method", | ||||
|         "location", | ||||
|     ] | ||||
|  | ||||
|     def __init__(self, *args, initial=None, **kwargs): | ||||
|         super().__init__(*args, initial=initial, **kwargs) | ||||
|         self.fields["birthdate"].required = True | ||||
|         if not initial: | ||||
|             return | ||||
|         member: str | None = initial.get("member") | ||||
|         if member and member.isdigit(): | ||||
|             member: User | None = User.objects.filter(id=int(member)).first() | ||||
|         else: | ||||
|             member = None | ||||
|         if member and member.date_of_birth: | ||||
|             # if there is an initial member with a birthdate, | ||||
|             # there is no need to ask this to the user | ||||
|             self.fields["birthdate"].initial = member.date_of_birth | ||||
|         elif member: | ||||
|             # if there is an initial member without a birthdate, | ||||
|             # then the field must be displayed | ||||
|             self.fields["birthdate"].widget.attrs.update({"hidden": False}) | ||||
|         # if there is no initial member, it means that it will be | ||||
|         # dynamically selected using the AutoCompleteSelectUser widget. | ||||
|         # JS will take care of un-hiding the field if necessary | ||||
|  | ||||
|     def save(self, *args, **kwargs): | ||||
|         if self.errors: | ||||
|             return super().save(*args, **kwargs) | ||||
|         if ( | ||||
|             self.cleaned_data["birthdate"] is not None | ||||
|             and self.instance.member.date_of_birth is None | ||||
|         ): | ||||
|             self.instance.member.date_of_birth = self.cleaned_data["birthdate"] | ||||
|             self.instance.member.save() | ||||
|         return super().save(*args, **kwargs) | ||||
|   | ||||
| @@ -0,0 +1,56 @@ | ||||
| # Generated by Django 5.2.3 on 2025-09-08 05:38 | ||||
|  | ||||
| from django.db import migrations, models | ||||
| from django.db.migrations.state import StateApps | ||||
|  | ||||
|  | ||||
| def rename_enums(apps: StateApps, schema_editor): | ||||
|     Subscription = apps.get_model("subscription", "Subscription") | ||||
|     Subscription.objects.filter(subscription_type="EBOUTIC").update( | ||||
|         subscription_type="AE_ACCOUNT" | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def rename_enums_reverse(apps: StateApps, schema_editor): | ||||
|     Subscription = apps.get_model("subscription", "Subscription") | ||||
|     Subscription.objects.filter(subscription_type="AE_ACCOUNT").update( | ||||
|         subscription_type="EBOUTIC" | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|     dependencies = [("subscription", "0014_auto_20201207_2323")] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.AlterField( | ||||
|             model_name="subscription", | ||||
|             name="location", | ||||
|             field=models.CharField( | ||||
|                 choices=[ | ||||
|                     ("BELFORT", "Belfort"), | ||||
|                     ("SEVENANS", "Sevenans"), | ||||
|                     ("MONTBELIARD", "Montbéliard"), | ||||
|                     ("EBOUTIC", "Eboutic"), | ||||
|                     ("OTHER", "Other"), | ||||
|                 ], | ||||
|                 max_length=20, | ||||
|                 verbose_name="location", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.AlterField( | ||||
|             model_name="subscription", | ||||
|             name="payment_method", | ||||
|             field=models.CharField( | ||||
|                 choices=[ | ||||
|                     ("CHECK", "Check"), | ||||
|                     ("CARD", "Credit card"), | ||||
|                     ("CASH", "Cash"), | ||||
|                     ("AE_ACCOUNT", "AE account"), | ||||
|                     ("OTHER", "Other"), | ||||
|                 ], | ||||
|                 max_length=255, | ||||
|                 verbose_name="payment method", | ||||
|             ), | ||||
|         ), | ||||
|         migrations.RunPython(rename_enums, reverse_code=rename_enums_reverse), | ||||
|     ] | ||||
| @@ -1,3 +1,5 @@ | ||||
| import { userFetchUser } from "#openapi"; | ||||
|  | ||||
| document.addEventListener("alpine:init", () => { | ||||
|   Alpine.data("existing_user_subscription_form", () => ({ | ||||
|     loading: false, | ||||
| @@ -12,13 +14,24 @@ document.addEventListener("alpine:init", () => { | ||||
|     }, | ||||
|  | ||||
|     async loadProfile(userId: number) { | ||||
|       const birthdayInput = document.getElementById("id_birthdate") as HTMLInputElement; | ||||
|       if (!Number.isInteger(userId)) { | ||||
|         this.profileFragment = ""; | ||||
|         birthdayInput.hidden = true; | ||||
|         return; | ||||
|       } | ||||
|       this.loading = true; | ||||
|       const response = await fetch(`/user/${userId}/mini/`); | ||||
|       this.profileFragment = await response.text(); | ||||
|       const [miniProfile, userInfos] = await Promise.all([ | ||||
|         fetch(`/user/${userId}/mini/`), | ||||
|         // biome-ignore lint/style/useNamingConvention: api is snake_case | ||||
|         userFetchUser({ path: { user_id: userId } }), | ||||
|       ]); | ||||
|       this.profileFragment = await miniProfile.text(); | ||||
|       // If the user has no birthdate yet, show the form input | ||||
|       // to fill this info. | ||||
|       // Else keep the input hidden and change its value to the user birthdate | ||||
|       birthdayInput.value = userInfos.data.date_of_birth; | ||||
|       birthdayInput.hidden = userInfos.data.date_of_birth !== null; | ||||
|       this.loading = false; | ||||
|     }, | ||||
|   })); | ||||
|   | ||||
| @@ -1,4 +1,14 @@ | ||||
| #subscription-form form { | ||||
|   margin-top: 0; | ||||
|  | ||||
|   .form-content { | ||||
|     margin-top: 0; | ||||
|   } | ||||
|  | ||||
|   fieldset p:first-of-type, & > p:first-of-type { | ||||
|     margin-top: 0; | ||||
|   } | ||||
|  | ||||
|   .form-content.existing-user { | ||||
|     max-height: 100%; | ||||
|     display: flex; | ||||
| @@ -13,6 +23,11 @@ | ||||
|      * then display the user profile right in the middle of the remaining space. */ | ||||
|     fieldset { | ||||
|       flex: 0 1 auto; | ||||
|  | ||||
|       p:has(input[hidden]) { | ||||
|         // when the input is hidden, hide the whole label+input+help text group | ||||
|         display: none; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     #subscription-form-user-mini-profile { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user