mirror of
				https://github.com/ae-utbm/sith.git
				synced 2025-10-21 20:28:29 +00:00 
			
		
		
		
	Compare commits
	
		
			109 Commits
		
	
	
		
			galaxy
			...
			dependabot
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | c625db4316 | ||
|  | 710b4aa942 | ||
|  | 459edc1b6e | ||
| a760a0b75d | |||
|  | fc615e90b2 | ||
| 76eebaf54e | |||
|  | 9407f4b341 | ||
|  | 8bd82c9d7c | ||
|  | 957441ceb1 | ||
|  | 3bcd417ad0 | ||
|  | 453e13d54b | ||
|  | dbd86b66cc | ||
|  | dcf799b352 | ||
|  | d815f7da97 | ||
|  | dac52db434 | ||
|  | f398c9901c | ||
|  | 5b91fe2145 | ||
|  | abd905c24d | ||
|  | 42b53a39f3 | ||
|  | 5306001f6f | ||
|  | 83a4ac2a7e | ||
|  | 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 | ||
|  | 5fee2e4720 | ||
|  | 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) | # A list of team reviewers to be added to pull requests (GitHub team slug) | ||||||
| reviewers: | reviewers: | ||||||
|   - ae-utbm/sith-3-developers |   - ae-utbm/developpeurs | ||||||
|  |  | ||||||
| # Number of reviewers has no impact on GitHub teams | # Number of reviewers has no impact on GitHub teams | ||||||
| # Set 0 to add all the reviewers (default: 0) | # 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: | updates: | ||||||
|   - package-ecosystem: "uv" |   - package-ecosystem: "uv" | ||||||
|  |     patterns: ["*"] | ||||||
|     multi-ecosystem-group: "common" |     multi-ecosystem-group: "common" | ||||||
|  |  | ||||||
|   - package-ecosystem: "npm" |   - package-ecosystem: "npm" | ||||||
|  |     patterns: ["*"] | ||||||
|     multi-ecosystem-group: "common" |     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.conf import settings | ||||||
| from django.db.models import Exists, OuterRef, Q | from django.db.models import Exists, OuterRef, Q | ||||||
| from django.db.models.functions import Lower | from django.db.models.functions import Lower | ||||||
|  | from django.utils.functional import cached_property | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  |  | ||||||
| from club.models import Club, Mailing, MailingSubscription, Membership | from club.models import Club, Mailing, MailingSubscription, Membership | ||||||
| from core.models import User | from core.models import User | ||||||
| from core.views.forms import SelectDate, SelectDateTime | from core.views.forms import SelectDateTime | ||||||
| from core.views.widgets.ajax_select import AutoCompleteSelectMultipleUser | from core.views.widgets.ajax_select import ( | ||||||
|  |     AutoCompleteSelectMultipleUser, | ||||||
|  |     AutoCompleteSelectUser, | ||||||
|  | ) | ||||||
| from counter.models import Counter, Selling | from counter.models import Counter, Selling | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -188,105 +192,113 @@ class SellingsForm(forms.Form): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ClubMemberForm(forms.Form): | class ClubOldMemberForm(forms.Form): | ||||||
|     """Form handling the members of a club.""" |     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" |     error_css_class = "error" | ||||||
|     required_css_class = "required" |     required_css_class = "required" | ||||||
|  |  | ||||||
|     users = forms.ModelMultipleChoiceField( |     class Meta: | ||||||
|         label=_("Users to add"), |         model = Membership | ||||||
|         help_text=_("Search users to add (one or more)."), |         fields = ["role", "description"] | ||||||
|         required=False, |  | ||||||
|         widget=AutoCompleteSelectMultipleUser, |  | ||||||
|         queryset=User.objects.all(), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, club: Club, request_user: User, **kwargs): | ||||||
|         self.club = kwargs.pop("club") |         self.club = club | ||||||
|         self.request_user = kwargs.pop("request_user") |         self.request_user = 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() |  | ||||||
|         self.request_user_membership = self.club.get_membership_for(self.request_user) |         self.request_user_membership = self.club.get_membership_for(self.request_user) | ||||||
|         super().__init__(*args, **kwargs) |         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 |     @property | ||||||
|         # We want the view to process the model creation since they are multiple users |     def max_available_role(self): | ||||||
|         # We also want the form to handle bulk deletion |         """The greatest role that will be obtainable with this form.""" | ||||||
|         self.fields.update( |         # this is unreachable, because it will be overridden by subclasses | ||||||
|             forms.fields_for_model( |         return -1  # pragma: no cover | ||||||
|                 Membership, |  | ||||||
|                 fields=("role", "start_date", "description"), |  | ||||||
|                 widgets={"start_date": SelectDate}, |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         # Role is required only if users is specified |  | ||||||
|         self.fields["role"].required = False |  | ||||||
|  |  | ||||||
|         # Start date and description are never really required | class ClubAddMemberForm(ClubMemberForm): | ||||||
|         self.fields["start_date"].required = False |     """Form to add a member to the club, as a board member.""" | ||||||
|         self.fields["description"].required = False |  | ||||||
|  |  | ||||||
|         self.fields["users_old"] = forms.ModelMultipleChoiceField( |     class Meta(ClubMemberForm.Meta): | ||||||
|             User.objects.filter( |         fields = ["user", *ClubMemberForm.Meta.fields] | ||||||
|                 id__in=[ |         widgets = {"user": AutoCompleteSelectUser} | ||||||
|                     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") |  | ||||||
|  |  | ||||||
|     def clean_users(self): |     @cached_property | ||||||
|         """Check that the user is not trying to add an user already in the club. |     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. |         Also check that the user is valid and has a valid subscription. | ||||||
|         """ |         """ | ||||||
|         cleaned_data = super().clean() |         user = self.cleaned_data["user"] | ||||||
|         users = [] |         if not user.is_subscribed: | ||||||
|         for user in cleaned_data["users"]: |             raise forms.ValidationError( | ||||||
|             if not user.is_subscribed: |                 _("User must be subscriber to take part to a club"), code="invalid" | ||||||
|                 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( | ||||||
|             if self.club.get_membership_for(user): |                 _("You can not add the same user twice"), code="invalid" | ||||||
|                 raise forms.ValidationError( |             ) | ||||||
|                     _("You can not add the same user twice"), code="invalid" |         return user | ||||||
|                 ) |  | ||||||
|             users.append(user) |  | ||||||
|         return users | 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): |     def clean(self): | ||||||
|         """Check user rights for adding an user.""" |         """Check that the user is subscribed and isn't already in the club.""" | ||||||
|         cleaned_data = super().clean() |         if not self.request_user.is_subscribed: | ||||||
|  |             raise forms.ValidationError( | ||||||
|         if "start_date" in cleaned_data and not cleaned_data["start_date"]: |                 _("You must be subscribed to join a club"), code="invalid" | ||||||
|             # Drop start_date if allowed to edition but not specified |             ) | ||||||
|             cleaned_data.pop("start_date") |         if self.club.get_membership_for(self.request_user): | ||||||
|  |             raise forms.ValidationError( | ||||||
|         if not cleaned_data.get("users"): |                 _("You are already a member of this club"), code="invalid" | ||||||
|             # No user to add equals no check needed |             ) | ||||||
|             return cleaned_data |         return super().clean() | ||||||
|  |  | ||||||
|         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 |  | ||||||
|   | |||||||
| @@ -34,12 +34,10 @@ def migrate_meta_groups(apps: StateApps, schema_editor): | |||||||
|     clubs = list(Club.objects.all()) |     clubs = list(Club.objects.all()) | ||||||
|     for club in clubs: |     for club in clubs: | ||||||
|         club.board_group = meta_groups.get_or_create( |         club.board_group = meta_groups.get_or_create( | ||||||
|             name=club.unix_name + settings.SITH_BOARD_SUFFIX, |             name=f"{club.unix_name}-bureau", defaults={"is_meta": True} | ||||||
|             defaults={"is_meta": True}, |  | ||||||
|         )[0] |         )[0] | ||||||
|         club.members_group = meta_groups.get_or_create( |         club.members_group = meta_groups.get_or_create( | ||||||
|             name=club.unix_name + settings.SITH_MEMBER_SUFFIX, |             name=f"{club.unix_name}-membres", defaults={"is_meta": True} | ||||||
|             defaults={"is_meta": True}, |  | ||||||
|         )[0] |         )[0] | ||||||
|         club.save() |         club.save() | ||||||
|         club.refresh_from_db() |         club.refresh_from_db() | ||||||
|   | |||||||
| @@ -30,7 +30,8 @@ from django.core.cache import cache | |||||||
| from django.core.exceptions import ObjectDoesNotExist, ValidationError | from django.core.exceptions import ObjectDoesNotExist, ValidationError | ||||||
| from django.core.validators import RegexValidator, validate_email | from django.core.validators import RegexValidator, validate_email | ||||||
| from django.db import models, transaction | 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.urls import reverse | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
| @@ -42,6 +43,13 @@ from core.fields import ResizedImageField | |||||||
| from core.models import Group, Notification, Page, SithFile, User | 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): | class Club(models.Model): | ||||||
|     """The Club class, made as a tree to allow nice tidy organization.""" |     """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 |         Group, related_name="club_board", on_delete=models.PROTECT | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |     objects = ClubQuerySet.as_manager() | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         ordering = ["name"] |         ordering = ["name"] | ||||||
|  |  | ||||||
| @@ -200,10 +210,6 @@ class Club(models.Model): | |||||||
|         """Method to see if that object can be edited by the given user.""" |         """Method to see if that object can be edited by the given user.""" | ||||||
|         return self.has_rights_in_club(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: |     def get_membership_for(self, user: User) -> Membership | None: | ||||||
|         """Return the current membership the given user. |         """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) |         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: |     def update(self, **kwargs) -> int: | ||||||
|         """Refresh the cache and edit group ownership. |         """Refresh the cache and edit group ownership. | ||||||
|  |  | ||||||
| @@ -319,16 +363,12 @@ class Membership(models.Model): | |||||||
|         User, |         User, | ||||||
|         verbose_name=_("user"), |         verbose_name=_("user"), | ||||||
|         related_name="memberships", |         related_name="memberships", | ||||||
|         null=False, |  | ||||||
|         blank=False, |  | ||||||
|         on_delete=models.CASCADE, |         on_delete=models.CASCADE, | ||||||
|     ) |     ) | ||||||
|     club = models.ForeignKey( |     club = models.ForeignKey( | ||||||
|         Club, |         Club, | ||||||
|         verbose_name=_("club"), |         verbose_name=_("club"), | ||||||
|         related_name="members", |         related_name="members", | ||||||
|         null=False, |  | ||||||
|         blank=False, |  | ||||||
|         on_delete=models.CASCADE, |         on_delete=models.CASCADE, | ||||||
|     ) |     ) | ||||||
|     start_date = models.DateField(_("start date"), default=timezone.now) |     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" %} | {% extends "core/base.jinja" %} | ||||||
| {% from 'core/macros.jinja' import user_profile_link, select_all_checkbox %} | {% 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 content %} | ||||||
|  |   {% block notifications %} | ||||||
|  |     {# Notifications are moved a little bit below #} | ||||||
|  |   {% endblock %} | ||||||
|  |  | ||||||
|   <h2>{% trans %}Club members{% endtrans %}</h2> |   <h2>{% trans %}Club members{% endtrans %}</h2> | ||||||
|  |  | ||||||
|  |   {% if add_member_fragment %} | ||||||
|  |     <br /> | ||||||
|  |     {{ add_member_fragment }} | ||||||
|  |     <br /> | ||||||
|  |   {% endif %} | ||||||
|  |   {% include "core/base/notifications.jinja" %} | ||||||
|   {% if members %} |   {% 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 %} |       {% csrf_token %} | ||||||
|       {% set users_old = dict(form.users_old | groupby("choice_label")) %} |       {% if can_end_membership %} | ||||||
|       {% if users_old %} |         {{ select_all_checkbox("members_old") }} | ||||||
|         {{ select_all_checkbox("users_old") }} |         <br /> | ||||||
|         <p></p> |  | ||||||
|       {% endif %} |       {% endif %} | ||||||
|       <table id="club_members_table"> |       <table id="club_members_table"> | ||||||
|         <thead> |         <thead> | ||||||
| @@ -18,7 +36,7 @@ | |||||||
|             <td>{% trans %}Role{% endtrans %}</td> |             <td>{% trans %}Role{% endtrans %}</td> | ||||||
|             <td>{% trans %}Description{% endtrans %}</td> |             <td>{% trans %}Description{% endtrans %}</td> | ||||||
|             <td>{% trans %}Since{% endtrans %}</td> |             <td>{% trans %}Since{% endtrans %}</td> | ||||||
|             {% if users_old %} |             {% if can_end_membership %} | ||||||
|               <td>{% trans %}Mark as old{% endtrans %}</td> |               <td>{% trans %}Mark as old{% endtrans %}</td> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|           </tr> |           </tr> | ||||||
| @@ -30,20 +48,24 @@ | |||||||
|               <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> |               <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> | ||||||
|               <td>{{ m.description }}</td> |               <td>{{ m.description }}</td> | ||||||
|               <td>{{ m.start_date }}</td> |               <td>{{ m.start_date }}</td> | ||||||
|               {% if users_old %} |               {%- if can_end_membership -%} | ||||||
|                 <td> |                 <td> | ||||||
|                   {% set user_old = users_old[m.user.get_display_name()] %} |                   {%- if m.is_editable -%} | ||||||
|                   {% if user_old %} |                     <label for="id_members_old_{{ loop.index }}"></label> | ||||||
|                     {{ user_old[0].tag() }} |                     <input | ||||||
|                   {% endif %} |                       type="checkbox" | ||||||
|  |                       name="members_old" | ||||||
|  |                       value="{{ m.id }}" | ||||||
|  |                       id="id_members_old_{{ loop.index }}" | ||||||
|  |                     > | ||||||
|  |                   {%- endif -%} | ||||||
|                 </td> |                 </td> | ||||||
|               {% endif %} |               {%- endif -%} | ||||||
|             </tr> |             </tr> | ||||||
|           {% endfor %} |           {% endfor %} | ||||||
|         </tbody> |         </tbody> | ||||||
|       </table> |       </table> | ||||||
|       {{ form.users_old.errors }} |       {% if can_end_membership %} | ||||||
|       {% if users_old %} |  | ||||||
|         <p></p> |         <p></p> | ||||||
|         <input type="submit" name="submit" value="{% trans %}Mark as old{% endtrans %}"> |         <input type="submit" name="submit" value="{% trans %}Mark as old{% endtrans %}"> | ||||||
|       {% endif %} |       {% endif %} | ||||||
| @@ -51,32 +73,4 @@ | |||||||
|   {% else %} |   {% else %} | ||||||
|     <p>{% trans %}There are no members in this club.{% endtrans %}</p> |     <p>{% trans %}There are no members in this club.{% endtrans %}</p> | ||||||
|   {% endif %} |   {% 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 %} | {% endblock %} | ||||||
|   | |||||||
| @@ -5,20 +5,22 @@ | |||||||
|   <h2>{% trans %}Club old members{% endtrans %}</h2> |   <h2>{% trans %}Club old members{% endtrans %}</h2> | ||||||
|   <table> |   <table> | ||||||
|     <thead> |     <thead> | ||||||
|       <td>{% trans %}User{% endtrans %}</td> |       <tr> | ||||||
|       <td>{% trans %}Role{% endtrans %}</td> |         <td>{% trans %}User{% endtrans %}</td> | ||||||
|       <td>{% trans %}Description{% endtrans %}</td> |         <td>{% trans %}Role{% endtrans %}</td> | ||||||
|       <td>{% trans %}From{% endtrans %}</td> |         <td>{% trans %}Description{% endtrans %}</td> | ||||||
|       <td>{% trans %}To{% endtrans %}</td> |         <td>{% trans %}From{% endtrans %}</td> | ||||||
|  |         <td>{% trans %}To{% endtrans %}</td> | ||||||
|  |       </tr> | ||||||
|     </thead> |     </thead> | ||||||
|     <tbody> |     <tbody> | ||||||
|       {% for m in club.members.exclude(end_date=None).order_by('-role', 'description', '-end_date').all() %} |       {% for member in old_members %} | ||||||
|         <tr> |         <tr> | ||||||
|           <td>{{ user_profile_link(m.user) }}</td> |           <td>{{ user_profile_link(member.user) }}</td> | ||||||
|           <td>{{ settings.SITH_CLUB_ROLES[m.role] }}</td> |           <td>{{ settings.SITH_CLUB_ROLES[member.role] }}</td> | ||||||
|           <td>{{ m.description }}</td> |           <td>{{ member.description }}</td> | ||||||
|           <td>{{ m.start_date }}</td> |           <td>{{ member.start_date }}</td> | ||||||
|           <td>{{ m.end_date }}</td> |           <td>{{ member.end_date }}</td> | ||||||
|         </tr> |         </tr> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </tbody> |     </tbody> | ||||||
|   | |||||||
| @@ -83,9 +83,10 @@ TODO : rewrite the pagination used in this template an Alpine one | |||||||
|   </table> |   </table> | ||||||
|   <script type="text/javascript"> |   <script type="text/javascript"> | ||||||
|     function formPagination(link){ |     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 |       link.href = "javascript:void(0)"; // block link action | ||||||
|       $("form").submit(); |       form.submit(); | ||||||
|     } |     } | ||||||
|   </script> |   </script> | ||||||
|   {{ paginate(paginated_result, paginator, "formPagination(this)") }} |   {{ 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.ae = Club.objects.get(pk=settings.SITH_MAIN_CLUB_ID) | ||||||
|         cls.club = baker.make(Club) |         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}) |         cls.members_url = reverse("club:club_members", kwargs={"club_id": cls.club.id}) | ||||||
|         a_month_ago = now() - timedelta(days=30) |         a_month_ago = now() - timedelta(days=30) | ||||||
|         yesterday = now() - timedelta(days=1) |         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 bs4 import BeautifulSoup | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.contrib.auth.models import Permission | ||||||
| from django.core.cache import cache | from django.core.cache import cache | ||||||
| from django.db.models import Max | from django.db.models import Max | ||||||
|  | from django.test import TestCase | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| from django.utils.timezone import localdate, localtime, now | from django.utils.timezone import localdate, localtime, now | ||||||
| from model_bakery import baker | from model_bakery import baker | ||||||
|  | from pytest_django.asserts import assertRedirects | ||||||
|  |  | ||||||
| from club.forms import ClubMemberForm | from club.forms import ClubAddMemberForm, JoinClubForm | ||||||
| from club.models import Membership | from club.models import Club, Membership | ||||||
| from club.tests.base import TestClub | from club.tests.base import TestClub | ||||||
| from core.baker_recipes import subscriber_user | from core.baker_recipes import subscriber_user | ||||||
| from core.models import AnonymousUser, User | from core.models import AnonymousUser, User | ||||||
| @@ -137,6 +144,38 @@ class TestMembershipQuerySet(TestClub): | |||||||
|         assert set(user.groups.all()).isdisjoint(club_groups) |         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): | class TestMembership(TestClub): | ||||||
|     def assert_membership_started_today(self, user: User, role: int): |     def assert_membership_started_today(self, user: User, role: int): | ||||||
|         """Assert that the given membership is active and started today.""" |         """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): |     def assert_membership_ended_today(self, user: User): | ||||||
|         """Assert that the given user have a membership which ended today.""" |         """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 user.memberships.filter(club=self.club, end_date=today).exists() | ||||||
|         assert self.club.get_membership_for(user) is None |         assert self.club.get_membership_for(user) is None | ||||||
|  |  | ||||||
| @@ -160,7 +199,9 @@ class TestMembership(TestClub): | |||||||
|         cannot see the page. |         cannot see the page. | ||||||
|         """ |         """ | ||||||
|         response = self.client.post(self.members_url) |         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) |         self.client.force_login(self.public) | ||||||
|         response = self.client.post(self.members_url) |         response = self.client.post(self.members_url) | ||||||
| @@ -171,7 +212,9 @@ class TestMembership(TestClub): | |||||||
|         information are displayed. |         information are displayed. | ||||||
|         """ |         """ | ||||||
|         self.client.force_login(self.simple_board_member) |         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 |         assert response.status_code == 200 | ||||||
|         soup = BeautifulSoup(response.text, "lxml") |         soup = BeautifulSoup(response.text, "lxml") | ||||||
|         table = soup.find("table", id="club_members_table") |         table = soup.find("table", id="club_members_table") | ||||||
| @@ -197,59 +240,45 @@ class TestMembership(TestClub): | |||||||
|             assert cols[2].text == membership.description |             assert cols[2].text == membership.description | ||||||
|             assert cols[3].text == str(membership.start_date) |             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") |                 form_input = cols[4].find("input") | ||||||
|                 expected_attrs = { |                 expected_attrs = { | ||||||
|                     "type": "checkbox", |                     "type": "checkbox", | ||||||
|                     "name": "users_old", |                     "name": "members_old", | ||||||
|                     "value": str(user.id), |                     "value": str(membership.id), | ||||||
|                 } |                 } | ||||||
|                 assert form_input.attrs.items() >= expected_attrs.items() |                 assert form_input.attrs.items() >= expected_attrs.items() | ||||||
|             else: |             else: | ||||||
|                 assert cols[4].find_all() == [] |                 assert cols[4].find_all() == [] | ||||||
|  |  | ||||||
|     def test_root_add_one_club_member(self): |     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) |         self.client.force_login(self.root) | ||||||
|         response = self.client.post( |         response = self.client.post( | ||||||
|             self.members_url, |             self.new_members_url, {"user": self.subscriber.id, "role": 3} | ||||||
|             {"users": [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.subscriber.refresh_from_db() | ||||||
|         self.assert_membership_started_today(self.subscriber, role=3) |         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): |     def test_add_unauthorized_members(self): | ||||||
|         """Test that users who are not currently subscribed |         """Test that users who are not currently subscribed | ||||||
|         cannot be members of clubs. |         cannot be members of clubs. | ||||||
|         """ |         """ | ||||||
|         for user in self.public, self.old_subscriber: |         for user in self.public, self.old_subscriber: | ||||||
|             form = ClubMemberForm( |             form = ClubAddMemberForm( | ||||||
|                 data={"users": [user.id], "role": 1}, |                 data={"user": user.id, "role": 1}, | ||||||
|                 request_user=self.root, |                 request_user=self.root, | ||||||
|                 club=self.club, |                 club=self.club, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|             assert not form.is_valid() |             assert not form.is_valid() | ||||||
|             assert form.errors == { |             assert form.errors == { | ||||||
|                 "users": [ |                 "user": ["L'utilisateur doit être cotisant pour faire partie d'un club"] | ||||||
|                     "L'utilisateur doit être cotisant pour faire partie d'un club" |  | ||||||
|                 ] |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|     def test_add_members_already_members(self): |     def test_add_members_already_members(self): | ||||||
| @@ -281,16 +310,16 @@ class TestMembership(TestClub): | |||||||
|         nb_memberships = self.club.members.count() |         nb_memberships = self.club.members.count() | ||||||
|         max_id = User.objects.aggregate(id=Max("id"))["id"] |         max_id = User.objects.aggregate(id=Max("id"))["id"] | ||||||
|         for members in [max_id + 1], [max_id + 1, self.subscriber.id]: |         for members in [max_id + 1], [max_id + 1, self.subscriber.id]: | ||||||
|             form = ClubMemberForm( |             form = ClubAddMemberForm( | ||||||
|                 data={"users": members, "role": 1}, |                 data={"user": members, "role": 1}, | ||||||
|                 request_user=self.root, |                 request_user=self.root, | ||||||
|                 club=self.club, |                 club=self.club, | ||||||
|             ) |             ) | ||||||
|             assert not form.is_valid() |             assert not form.is_valid() | ||||||
|             assert form.errors == { |             assert form.errors == { | ||||||
|                 "users": [ |                 "user": [ | ||||||
|                     "Sélectionnez un choix valide. " |                     "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() |         self.club.refresh_from_db() | ||||||
| @@ -303,10 +332,12 @@ class TestMembership(TestClub): | |||||||
|         nb_subscriber_memberships = self.subscriber.memberships.count() |         nb_subscriber_memberships = self.subscriber.memberships.count() | ||||||
|         self.client.force_login(president) |         self.client.force_login(president) | ||||||
|         response = self.client.post( |         response = self.client.post( | ||||||
|             self.members_url, |             self.new_members_url, {"user": self.subscriber.id, "role": 9} | ||||||
|             {"users": 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.club.refresh_from_db() | ||||||
|         self.subscriber.refresh_from_db() |         self.subscriber.refresh_from_db() | ||||||
|         assert self.club.members.count() == nb_club_membership + 1 |         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 |         """Test that a member of the club member cannot create | ||||||
|         a membership with a greater role than its own. |         a membership with a greater role than its own. | ||||||
|         """ |         """ | ||||||
|         form = ClubMemberForm( |         form = ClubAddMemberForm( | ||||||
|             data={"users": [self.subscriber.id], "role": 10}, |             data={"user": self.subscriber.id, "role": 10}, | ||||||
|             request_user=self.simple_board_member, |             request_user=self.simple_board_member, | ||||||
|             club=self.club, |             club=self.club, | ||||||
|         ) |         ) | ||||||
| @@ -326,7 +357,7 @@ class TestMembership(TestClub): | |||||||
|  |  | ||||||
|         assert not form.is_valid() |         assert not form.is_valid() | ||||||
|         assert form.errors == { |         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() |         self.club.refresh_from_db() | ||||||
|         assert nb_memberships == self.club.members.count() |         assert nb_memberships == self.club.members.count() | ||||||
| @@ -334,23 +365,53 @@ class TestMembership(TestClub): | |||||||
|  |  | ||||||
|     def test_add_member_without_role(self): |     def test_add_member_without_role(self): | ||||||
|         """Test that trying to add members without specifying their role fails.""" |         """Test that trying to add members without specifying their role fails.""" | ||||||
|         self.client.force_login(self.root) |         form = ClubAddMemberForm( | ||||||
|         form = ClubMemberForm( |             data={"user": self.subscriber.id}, request_user=self.root, club=self.club | ||||||
|             data={"users": [self.subscriber.id]}, |  | ||||||
|             request_user=self.simple_board_member, |  | ||||||
|             club=self.club, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         assert not form.is_valid() |         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): |     def test_end_membership_self(self): | ||||||
|         """Test that a member can end its own membership.""" |         """Test that a member can end its own membership.""" | ||||||
|         self.client.force_login(self.simple_board_member) |         self.client.force_login(self.simple_board_member) | ||||||
|         self.client.post( |         membership = self.club.members.get(end_date=None, user=self.simple_board_member) | ||||||
|             self.members_url, |         self.client.post(self.members_url, {"members_old": [membership.id]}) | ||||||
|             {"users_old": self.simple_board_member.id}, |  | ||||||
|         ) |  | ||||||
|         self.simple_board_member.refresh_from_db() |         self.simple_board_member.refresh_from_db() | ||||||
|         self.assert_membership_ended_today(self.simple_board_member) |         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 |         """Test that board members of the club can end memberships | ||||||
|         of users with lower roles. |         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) |         self.client.force_login(self.simple_board_member) | ||||||
|         response = self.client.post( |         membership = baker.make(Membership, club=self.club, role=2, end_date=None) | ||||||
|             self.members_url, |         response = self.client.post(self.members_url, {"members_old": [membership.id]}) | ||||||
|             {"users_old": self.richard.id}, |  | ||||||
|         ) |  | ||||||
|         self.assertRedirects(response, self.members_url) |         self.assertRedirects(response, self.members_url) | ||||||
|         self.club.refresh_from_db() |         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): |     def test_end_membership_higher_role(self): | ||||||
|         """Test that board members of the club cannot end memberships |         """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() |         membership = self.president.memberships.filter(club=self.club).first() | ||||||
|         self.client.force_login(self.simple_board_member) |         self.client.force_login(self.simple_board_member) | ||||||
|         self.client.post( |         self.client.post(self.members_url, {"members_old": [membership.id]}) | ||||||
|             self.members_url, |  | ||||||
|             {"users_old": self.president.id}, |  | ||||||
|         ) |  | ||||||
|         self.club.refresh_from_db() |         self.club.refresh_from_db() | ||||||
|         new_membership = self.club.get_membership_for(self.president) |         new_membership = self.club.get_membership_for(self.president) | ||||||
|         assert new_membership is not None |         assert new_membership is not None | ||||||
|         assert new_membership == membership |         assert new_membership == membership | ||||||
|  |  | ||||||
|         membership = self.president.memberships.filter(club=self.club).first() |         membership.refresh_from_db() | ||||||
|         assert membership.end_date is None |         assert membership.end_date is None | ||||||
|  |  | ||||||
|     def test_end_membership_as_main_club_board(self): |     def test_end_membership_with_permission(self): | ||||||
|         """Test that board members of the main club can end the membership |         """Test that users with permission can end any membership.""" | ||||||
|         of anyone. |  | ||||||
|         """ |  | ||||||
|         # make subscriber a board member |         # 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() |         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( |         response = self.client.post( | ||||||
|             self.members_url, |             self.members_url, {"members_old": [president_membership.id]} | ||||||
|             {"users_old": self.president.id}, |  | ||||||
|         ) |         ) | ||||||
|         self.assertRedirects(response, self.members_url) |         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_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) |  | ||||||
|         assert self.club.members.ongoing().count() == nb_memberships - 1 |         assert self.club.members.ongoing().count() == nb_memberships - 1 | ||||||
|  |  | ||||||
|     def test_end_membership_as_foreigner(self): |     def test_end_membership_as_foreigner(self): | ||||||
| @@ -421,14 +464,11 @@ class TestMembership(TestClub): | |||||||
|         nb_memberships = self.club.members.count() |         nb_memberships = self.club.members.count() | ||||||
|         membership = self.richard.memberships.filter(club=self.club).first() |         membership = self.richard.memberships.filter(club=self.club).first() | ||||||
|         self.client.force_login(self.subscriber) |         self.client.force_login(self.subscriber) | ||||||
|         self.client.post( |         self.client.post(self.members_url, {"members_old": [self.richard.id]}) | ||||||
|             self.members_url, |  | ||||||
|             {"users_old": [self.richard.id]}, |  | ||||||
|         ) |  | ||||||
|         # nothing should have changed |         # 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 self.club.members.count() == nb_memberships | ||||||
|         assert membership == new_mem |         assert membership.end_date is None | ||||||
|  |  | ||||||
|     def test_remove_from_club_group(self): |     def test_remove_from_club_group(self): | ||||||
|         """Test that when a membership ends, the user is removed from club groups.""" |         """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)) |         new_board = set(self.club.board_group.users.values_list("id", flat=True)) | ||||||
|         assert new_members == initial_members |         assert new_members == initial_members | ||||||
|         assert new_board == initial_board |         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 django.urls import path | ||||||
|  |  | ||||||
| from club.views import ( | from club.views import ( | ||||||
|  |     ClubAddMembersFragment, | ||||||
|     ClubCreateView, |     ClubCreateView, | ||||||
|     ClubEditView, |     ClubEditView, | ||||||
|     ClubListView, |     ClubListView, | ||||||
| @@ -60,6 +61,11 @@ urlpatterns = [ | |||||||
|     path("<int:club_id>/edit/", ClubEditView.as_view(), name="club_edit"), |     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>/edit/page/", ClubPageEditView.as_view(), name="club_edit_page"), | ||||||
|     path("<int:club_id>/members/", ClubMembersView.as_view(), name="club_members"), |     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( |     path( | ||||||
|         "<int:club_id>/elderlies/", |         "<int:club_id>/elderlies/", | ||||||
|         ClubOldMembersView.as_view(), |         ClubOldMembersView.as_view(), | ||||||
|   | |||||||
							
								
								
									
										219
									
								
								club/views.py
									
									
									
									
									
								
							
							
						
						
									
										219
									
								
								club/views.py
									
									
									
									
									
								
							| @@ -23,52 +23,57 @@ | |||||||
| # | # | ||||||
|  |  | ||||||
| import csv | import csv | ||||||
|  | from typing import Any | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.auth.mixins import PermissionRequiredMixin | 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.exceptions import NON_FIELD_ERRORS, PermissionDenied, ValidationError | ||||||
| from django.core.paginator import InvalidPage, Paginator | from django.core.paginator import InvalidPage, Paginator | ||||||
| from django.db.models import Sum | from django.db.models import Q, Sum | ||||||
| from django.http import ( | from django.http import Http404, HttpResponseRedirect, StreamingHttpResponse | ||||||
|     Http404, |  | ||||||
|     HttpResponseRedirect, |  | ||||||
|     StreamingHttpResponse, |  | ||||||
| ) |  | ||||||
| from django.shortcuts import get_object_or_404, redirect | from django.shortcuts import get_object_or_404, redirect | ||||||
| from django.urls import reverse, reverse_lazy | from django.urls import reverse, reverse_lazy | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| 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 as _t | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from django.views.generic import DetailView, ListView, View | from django.views.generic import DetailView, ListView, View | ||||||
| from django.views.generic.edit import CreateView, DeleteView, UpdateView | from django.views.generic.edit import CreateView, DeleteView, UpdateView | ||||||
|  |  | ||||||
| from club.forms import ( | from club.forms import ( | ||||||
|  |     ClubAddMemberForm, | ||||||
|     ClubAdminEditForm, |     ClubAdminEditForm, | ||||||
|     ClubEditForm, |     ClubEditForm, | ||||||
|     ClubMemberForm, |     ClubOldMemberForm, | ||||||
|  |     JoinClubForm, | ||||||
|     MailingForm, |     MailingForm, | ||||||
|     SellingsForm, |     SellingsForm, | ||||||
| ) | ) | ||||||
| from club.models import Club, Mailing, MailingSubscription, Membership | from club.models import Club, Mailing, MailingSubscription, Membership | ||||||
|  | from com.models import Poster | ||||||
| from com.views import ( | from com.views import ( | ||||||
|     PosterCreateBaseView, |     PosterCreateBaseView, | ||||||
|     PosterDeleteBaseView, |     PosterDeleteBaseView, | ||||||
|     PosterEditBaseView, |     PosterEditBaseView, | ||||||
|     PosterListBaseView, |     PosterListBaseView, | ||||||
| ) | ) | ||||||
| from core.auth.mixins import CanCreateMixin, CanEditMixin, CanViewMixin | from core.auth.mixins import CanEditMixin | ||||||
| from core.models import PageRev | from core.models import PageRev | ||||||
| from core.views import DetailFormView, PageEditViewBase | from core.views import DetailFormView, PageEditViewBase, UseFragmentsMixin | ||||||
| from core.views.mixins import TabedViewMixin | from core.views.mixins import FragmentMixin, FragmentRenderer, TabedViewMixin | ||||||
| from counter.models import Selling | from counter.models import Selling | ||||||
|  |  | ||||||
|  |  | ||||||
| class ClubTabsMixin(TabedViewMixin): | class ClubTabsMixin(TabedViewMixin): | ||||||
|     def get_tabs_title(self): |     def get_tabs_title(self): | ||||||
|         obj = self.get_object() |         if not hasattr(self, "object") or not self.object: | ||||||
|         if isinstance(obj, PageRev): |             self.object = self.get_object() | ||||||
|             self.object = obj.page.club |         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() |         return self.object.get_display_name() | ||||||
|  |  | ||||||
|     def get_list_of_tabs(self): |     def get_list_of_tabs(self): | ||||||
| @@ -79,7 +84,7 @@ class ClubTabsMixin(TabedViewMixin): | |||||||
|                 "name": _("Infos"), |                 "name": _("Infos"), | ||||||
|             } |             } | ||||||
|         ] |         ] | ||||||
|         if self.request.user.can_view(self.object): |         if self.request.user.has_perm("club.view_club"): | ||||||
|             tab_list.extend( |             tab_list.extend( | ||||||
|                 [ |                 [ | ||||||
|                     { |                     { | ||||||
| @@ -98,16 +103,16 @@ class ClubTabsMixin(TabedViewMixin): | |||||||
|                     }, |                     }, | ||||||
|                 ] |                 ] | ||||||
|             ) |             ) | ||||||
|         if self.object.page: |             if self.object.page: | ||||||
|             tab_list.append( |                 tab_list.append( | ||||||
|                 { |                     { | ||||||
|                     "url": reverse( |                         "url": reverse( | ||||||
|                         "club:club_hist", kwargs={"club_id": self.object.id} |                             "club:club_hist", kwargs={"club_id": self.object.id} | ||||||
|                     ), |                         ), | ||||||
|                     "slug": "history", |                         "slug": "history", | ||||||
|                     "name": _("History"), |                         "name": _("History"), | ||||||
|                 } |                     } | ||||||
|             ) |                 ) | ||||||
|         if self.request.user.can_edit(self.object): |         if self.request.user.can_edit(self.object): | ||||||
|             tab_list.extend( |             tab_list.extend( | ||||||
|                 [ |                 [ | ||||||
| @@ -159,7 +164,7 @@ class ClubTabsMixin(TabedViewMixin): | |||||||
|                             "club:poster_list", kwargs={"club_id": self.object.id} |                             "club:poster_list", kwargs={"club_id": self.object.id} | ||||||
|                         ), |                         ), | ||||||
|                         "slug": "posters", |                         "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}) |         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.""" |     """Modification hostory of the page.""" | ||||||
|  |  | ||||||
|     model = Club |     model = Club | ||||||
|     pk_url_kwarg = "club_id" |     pk_url_kwarg = "club_id" | ||||||
|     template_name = "club/page_history.jinja" |     template_name = "club/page_history.jinja" | ||||||
|     current_tab = "history" |     current_tab = "history" | ||||||
|  |     permission_required = "club.view_club" | ||||||
|  |  | ||||||
|  |  | ||||||
| class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | ||||||
| @@ -246,57 +252,121 @@ class ClubToolsView(ClubTabsMixin, CanEditMixin, DetailView): | |||||||
|     current_tab = "tools" |     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.""" |     """View of a club's members.""" | ||||||
|  |  | ||||||
|     model = Club |     model = Club | ||||||
|     pk_url_kwarg = "club_id" |     pk_url_kwarg = "club_id" | ||||||
|     form_class = ClubMemberForm |     form_class = ClubOldMemberForm | ||||||
|     template_name = "club/club_members.jinja" |     template_name = "club/club_members.jinja" | ||||||
|     current_tab = "members" |     current_tab = "members" | ||||||
|  |     permission_required = "club.view_club" | ||||||
|  |  | ||||||
|     @cached_property |     def get_fragments(self) -> dict[str, type[FragmentMixin] | FragmentRenderer]: | ||||||
|     def members(self) -> list[Membership]: |         membership = self.object.get_membership_for(self.request.user) | ||||||
|         return list(self.object.members.ongoing().order_by("-role")) |         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): |     def get_form_kwargs(self): | ||||||
|         kwargs = super().get_form_kwargs() |         return super().get_form_kwargs() | { | ||||||
|         kwargs["request_user"] = self.request.user |             "user": self.request.user, | ||||||
|         kwargs["club"] = self.object |             "club": self.object, | ||||||
|         kwargs["club_members"] = self.members |         } | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs = super().get_context_data(**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 |         return kwargs | ||||||
|  |  | ||||||
|     def form_valid(self, form): |     def form_valid(self, form): | ||||||
|         """Check user rights.""" |         for membership in form.cleaned_data.get("members_old"): | ||||||
|         resp = super().form_valid(form) |             membership.end_date = now() | ||||||
|  |  | ||||||
|         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() |  | ||||||
|             membership.save() |             membership.save() | ||||||
|         return resp |         return super().form_valid(form) | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |     def get_success_url(self, **kwargs): | ||||||
|         return self.request.path |         return self.request.path | ||||||
|  |  | ||||||
|  |  | ||||||
| class ClubOldMembersView(ClubTabsMixin, CanViewMixin, DetailView): | class ClubOldMembersView(ClubTabsMixin, PermissionRequiredMixin, DetailView): | ||||||
|     """Old members of a club.""" |     """Old members of a club.""" | ||||||
|  |  | ||||||
|     model = Club |     model = Club | ||||||
|     pk_url_kwarg = "club_id" |     pk_url_kwarg = "club_id" | ||||||
|     template_name = "club/club_old_members.jinja" |     template_name = "club/club_old_members.jinja" | ||||||
|     current_tab = "elderlies" |     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): | class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | ||||||
| @@ -337,7 +407,7 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | |||||||
|         form = self.get_form() |         form = self.get_form() | ||||||
|         if form.is_valid(): |         if form.is_valid(): | ||||||
|             if not len([v for v in form.cleaned_data.values() if v is not None]): |             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"]: |             if form.cleaned_data["begin_date"]: | ||||||
|                 qs = qs.filter(date__gte=form.cleaned_data["begin_date"]) |                 qs = qs.filter(date__gte=form.cleaned_data["begin_date"]) | ||||||
|             if form.cleaned_data["end_date"]: |             if form.cleaned_data["end_date"]: | ||||||
| @@ -355,7 +425,9 @@ class ClubSellingView(ClubTabsMixin, CanEditMixin, DetailFormView): | |||||||
|             if len(selected_products) > 0: |             if len(selected_products) > 0: | ||||||
|                 qs = qs.filter(product__in=selected_products) |                 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"]]) |             kwargs["total"] = sum([s.quantity * s.unit_price for s in kwargs["result"]]) | ||||||
|             total_quantity = qs.all().aggregate(Sum("quantity")) |             total_quantity = qs.all().aggregate(Sum("quantity")) | ||||||
|             if total_quantity["quantity__sum"]: |             if total_quantity["quantity__sum"]: | ||||||
| @@ -686,48 +758,45 @@ class MailingAutoGenerationView(View): | |||||||
|         return redirect("club:mailing", club_id=club.id) |         return redirect("club:mailing", club_id=club.id) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterListView(ClubTabsMixin, PosterListBaseView, CanViewMixin): | class PosterListView(ClubTabsMixin, PosterListBaseView): | ||||||
|     """List communication posters.""" |     """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): |     def get_object(self): | ||||||
|         return self.club |         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(ClubTabsMixin, PosterCreateBaseView): | ||||||
| class PosterCreateView(PosterCreateBaseView, CanCreateMixin): |  | ||||||
|     """Create communication poster.""" |     """Create communication poster.""" | ||||||
|  |  | ||||||
|     pk_url_kwarg = "club_id" |     current_tab = "posters" | ||||||
|  |  | ||||||
|     def get_object(self): |  | ||||||
|         obj = super().get_object() |  | ||||||
|         if not obj: |  | ||||||
|             return self.club |  | ||||||
|         return obj |  | ||||||
|  |  | ||||||
|     def get_success_url(self, **kwargs): |     def get_success_url(self, **kwargs): | ||||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) |         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.""" |     """Edit communication poster.""" | ||||||
|  |  | ||||||
|  |     current_tab = "posters" | ||||||
|  |     extra_context = {"app": "club"} | ||||||
|  |  | ||||||
|     def get_success_url(self): |     def get_success_url(self): | ||||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) |         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(ClubTabsMixin, PosterDeleteBaseView): | ||||||
| class PosterDeleteView(PosterDeleteBaseView, ClubTabsMixin, CanEditMixin): |  | ||||||
|     """Delete communication poster.""" |     """Delete communication poster.""" | ||||||
|  |  | ||||||
|  |     current_tab = "posters" | ||||||
|  |  | ||||||
|     def get_success_url(self): |     def get_success_url(self): | ||||||
|         return reverse_lazy("club:poster_list", kwargs={"club_id": self.club.id}) |         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 dateutil.relativedelta import relativedelta | ||||||
| from django import forms | from django import forms | ||||||
| from django.db.models import Exists, OuterRef |  | ||||||
| from django.forms import CheckboxInput | from django.forms import CheckboxInput | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| @@ -35,20 +34,18 @@ class PosterForm(forms.ModelForm): | |||||||
|         label=_("Start date"), |         label=_("Start date"), | ||||||
|         widget=SelectDateTime, |         widget=SelectDateTime, | ||||||
|         required=True, |         required=True, | ||||||
|         initial=timezone.now().strftime("%Y-%m-%d %H:%M:%S"), |         initial=timezone.now(), | ||||||
|     ) |     ) | ||||||
|     date_end = forms.DateTimeField( |     date_end = forms.DateTimeField( | ||||||
|         label=_("End date"), widget=SelectDateTime, required=False |         label=_("End date"), widget=SelectDateTime, required=False | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, user: User, **kwargs): | ||||||
|         self.user = kwargs.pop("user", None) |  | ||||||
|         super().__init__(*args, **kwargs) |         super().__init__(*args, **kwargs) | ||||||
|         if self.user and not self.user.is_com_admin: |         if user.is_root or user.is_com_admin: | ||||||
|             self.fields["club"].queryset = Club.objects.filter( |             self.fields["club"].widget = AutoCompleteSelectClub() | ||||||
|                 id__in=self.user.clubs_with_rights |         else: | ||||||
|             ) |             self.fields["club"].queryset = Club.objects.having_board_member(user) | ||||||
|             self.fields.pop("display_time") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class NewsDateForm(forms.ModelForm): | class NewsDateForm(forms.ModelForm): | ||||||
| @@ -161,16 +158,9 @@ class NewsForm(forms.ModelForm): | |||||||
|         # if the author is an admin, he/she can choose any club, |         # 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 |         # otherwise, only clubs for which he/she is a board member can be selected | ||||||
|         if author.is_root or author.is_com_admin: |         if author.is_root or author.is_com_admin: | ||||||
|             self.fields["club"] = forms.ModelChoiceField( |             self.fields["club"].widget = AutoCompleteSelectClub() | ||||||
|                 queryset=Club.objects.all(), widget=AutoCompleteSelectClub |  | ||||||
|             ) |  | ||||||
|         else: |         else: | ||||||
|             active_memberships = author.memberships.board().ongoing() |             self.fields["club"].queryset = Club.objects.having_board_member(author) | ||||||
|             self.fields["club"] = forms.ModelChoiceField( |  | ||||||
|                 queryset=Club.objects.filter( |  | ||||||
|                     Exists(active_memberships.filter(club=OuterRef("pk"))) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|     def is_valid(self): |     def is_valid(self): | ||||||
|         return super().is_valid() and self.date_form.is_valid() |         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: |         if self.date_end and self.date_begin > self.date_end: | ||||||
|             raise ValidationError(_("Begin date should be before end date")) |             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): |     def get_display_name(self): | ||||||
|         return self.club.get_display_name() |         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 { |     #links_content { | ||||||
|       overflow: auto; |       overflow: auto; | ||||||
|       box-shadow: $shadow-color 1px 1px 1px; |       box-shadow: $shadow-color 1px 1px 1px; | ||||||
|       height: 20em; |       min-height: 20em; | ||||||
|  |       padding-bottom: 1em; | ||||||
|  |  | ||||||
|       h4 { |       h4 { | ||||||
|         margin-left: 5px; |         margin-left: 5px; | ||||||
|   | |||||||
| @@ -111,7 +111,7 @@ | |||||||
|             top: 0; |             top: 0; | ||||||
|             left: 0; |             left: 0; | ||||||
|             z-index: 10; |             z-index: 10; | ||||||
|             content: "Click to expand"; |             content: attr(hover); | ||||||
|             color: white; |             color: white; | ||||||
|             background-color: rgba(black, 0.5); |             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; |   position: absolute; | ||||||
|   width: 100vw; |   width: 100vw; | ||||||
|   height: 100vh; |   height: 100vh; | ||||||
| @@ -7,22 +7,22 @@ body{ | |||||||
|   margin: 0; |   margin: 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| #slideshow{ | #slideshow { | ||||||
|   position: relative; |   position: relative; | ||||||
|   background-color: lightgrey; |   background-color: lightgrey; | ||||||
|  |  | ||||||
|   height: 100%; |   height: 100%; | ||||||
|  |  | ||||||
|   *{ |   * { | ||||||
|     -webkit-user-select: none; |     -webkit-user-select: none; | ||||||
|     -moz-user-select: none; |     -moz-user-select: none; | ||||||
|     -ms-user-select: none; |     -ms-user-select: none; | ||||||
|     user-select: none; |     user-select: none; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   &:hover{ |   &:hover { | ||||||
|  |  | ||||||
|     &::before{ |     &::before { | ||||||
|  |  | ||||||
|       position: absolute; |       position: absolute; | ||||||
|       width: 100%; |       width: 100%; | ||||||
| @@ -34,7 +34,7 @@ body{ | |||||||
|  |  | ||||||
|       z-index: 10; |       z-index: 10; | ||||||
|  |  | ||||||
|       content: "Click to expand"; |       content: attr(hover); | ||||||
|  |  | ||||||
|       color: white; |       color: white; | ||||||
|       background-color: rgba(black, 0.5); |       background-color: rgba(black, 0.5); | ||||||
| @@ -43,7 +43,7 @@ body{ | |||||||
|  |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   &.fullscreen{ |   &:fullscreen { | ||||||
|     position: fixed; |     position: fixed; | ||||||
|     width: 100%; |     width: 100%; | ||||||
|     height: 100%; |     height: 100%; | ||||||
| @@ -51,57 +51,78 @@ body{ | |||||||
|     left: 0; |     left: 0; | ||||||
|     background: none; |     background: none; | ||||||
|  |  | ||||||
|     &:before{ |     &:before { | ||||||
|       display:none; |       display: none; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #slides{ |     #slides { | ||||||
|       height: 100vh; |       height: 100vh; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   #slides{ |   #slides { | ||||||
|     position: relative; |     position: relative; | ||||||
|     height: 100%; |     height: 100%; | ||||||
|     overflow: hidden; |     overflow: hidden; | ||||||
|  |     background-color: grey; | ||||||
|  |  | ||||||
|     .slide{ |     .slide { | ||||||
|       position: absolute; |       position: absolute; | ||||||
|       width: 100%; |       width: 100%; | ||||||
|       height: 100%; |       height: 100%; | ||||||
|  |  | ||||||
|       display: inline-flex; |       display: none; | ||||||
|       justify-content: center; |       justify-content: center; | ||||||
|  |  | ||||||
|       top: 0px; |       top: 0px; | ||||||
|  |       left: 0%; | ||||||
|  |  | ||||||
|       background-color: grey; |       img { | ||||||
|       transition: left 1s ease-out; |  | ||||||
|  |  | ||||||
|       img{ |  | ||||||
|         max-width: 100%; |         max-width: 100%; | ||||||
|         max-height: 100%; |         max-height: 100%; | ||||||
|         object-fit: contain; |         object-fit: contain; | ||||||
|       } |       } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .slide.left{ |       &.current { | ||||||
|       left: -100%; |         display: inline-flex; | ||||||
|     } |         left: 0%; | ||||||
|  |         animation: scrolling-in 1s linear; | ||||||
|  |       } | ||||||
|  |  | ||||||
|     .slide.center{ |       &.previous { | ||||||
|       left: 0px; |         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; |     position: absolute; | ||||||
|     bottom: 10px; |     bottom: 10px; | ||||||
|     width: 100%; |     width: 100%; | ||||||
| @@ -112,7 +133,7 @@ body{ | |||||||
|  |  | ||||||
|     margin-bottom: 10px; |     margin-bottom: 10px; | ||||||
|  |  | ||||||
|     .bullet{ |     .bullet { | ||||||
|       height: 10px; |       height: 10px; | ||||||
|       width: 10px; |       width: 10px; | ||||||
|  |  | ||||||
| @@ -123,27 +144,33 @@ body{ | |||||||
|  |  | ||||||
|       background-color: grey; |       background-color: grey; | ||||||
|  |  | ||||||
|       &.active{ |       &.active { | ||||||
|         background-color: #c99836; |         background-color: #c99836; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   #progress_bar{ |   progress { | ||||||
|  |     --color: #304c83; | ||||||
|  |  | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     bottom: 0px; |     bottom: 0px; | ||||||
|     height: 10px; |     height: 10px; | ||||||
|     background-color: #304c83; |     color: var(--color); | ||||||
|  |     width: 100%; | ||||||
|  |     margin-bottom: 0px; | ||||||
|  |     border: none; | ||||||
|  |  | ||||||
|     &.init{ |     &::-moz-progress-bar { | ||||||
|       width: 0px; |       background: var(--color); | ||||||
|       transition: none; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     &.progress{ |     &::-webkit-progress-value { | ||||||
|       width: 100%; |       background: var(--color); | ||||||
|       transition: width 10s linear; |     } | ||||||
|  |  | ||||||
|  |     &[value] { | ||||||
|  |       background-color: transparent; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -76,18 +76,20 @@ | |||||||
|               It will stay hidden for other users until it has been published. |               It will stay hidden for other users until it has been published. | ||||||
|             {% endtrans %} |             {% endtrans %} | ||||||
|           </p> |           </p> | ||||||
|           {% if user.has_perm("com.moderate_news") %} |           {%- if user.has_perm("com.moderate_news") -%} | ||||||
|             {# This is an additional query for each non-moderated news, |             {# This is an additional query for each non-moderated news, | ||||||
|             but it will be executed only for admin users, and only one time |             but it will be executed only for admin users, and only one time | ||||||
|             (if they do their job and 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 #} |             so it's still reasonable #} | ||||||
|             <div |             <div | ||||||
|               {% if news is integer or news is string %} |               {% if news is integer or news is string -%} | ||||||
|                 x-data="{ nbEvents: 0 }" |                 x-data="{ nbEvents: 0 }" | ||||||
|                 x-init="nbEvents = await nbToPublish()" |                 x-init="nbEvents = await nbToPublish()" | ||||||
|               {% else %} |               {%- elif news.is_published -%} | ||||||
|  |                 x-data="{ nbEvents: 0 }" | ||||||
|  |               {%- else -%} | ||||||
|                 x-data="{ nbEvents: {{ news.dates.count() }} }" |                 x-data="{ nbEvents: {{ news.dates.count() }} }" | ||||||
|               {% endif %} |               {%- endif -%} | ||||||
|             > |             > | ||||||
|               <template x-if="nbEvents > 1"> |               <template x-if="nbEvents > 1"> | ||||||
|                 <div> |                 <div> | ||||||
|   | |||||||
| @@ -205,6 +205,10 @@ | |||||||
|               <i class="fa-solid fa-graduation-cap fa-xl"></i> |               <i class="fa-solid fa-graduation-cap fa-xl"></i> | ||||||
|               <a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a> |               <a href="{{ url("pedagogy:guide") }}">{% trans %}UV Guide{% endtrans %}</a> | ||||||
|             </li> |             </li> | ||||||
|  |             <li> | ||||||
|  |               <i class="fa-solid fa-calendar-days fa-xl"></i> | ||||||
|  |               <a href="{{ url("timetable:generator") }}">{% trans %}Timetable{% endtrans %}</a> | ||||||
|  |             </li> | ||||||
|             <li> |             <li> | ||||||
|               <i class="fa-solid fa-magnifying-glass fa-xl"></i> |               <i class="fa-solid fa-magnifying-glass fa-xl"></i> | ||||||
|               <a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a> |               <a href="{{ url("matmat:search_clear") }}">{% trans %}Matmatronch{% endtrans %}</a> | ||||||
|   | |||||||
| @@ -1,11 +1,5 @@ | |||||||
| {% extends "core/base.jinja" %} | {% extends "core/base.jinja" %} | ||||||
|  |  | ||||||
| {% block script %} |  | ||||||
|   {{ super() }} |  | ||||||
|   <script src="{{ static('com/js/poster_list.js') }}"></script> |  | ||||||
| {% endblock %} |  | ||||||
|  |  | ||||||
|  |  | ||||||
| {% block title %} | {% block title %} | ||||||
|   {% trans %}Poster{% endtrans %} |   {% trans %}Poster{% endtrans %} | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -15,7 +9,7 @@ | |||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   <div id="poster_list"> |   <div id="poster_list" x-data="{ active: null }"> | ||||||
|  |  | ||||||
|     <div id="title"> |     <div id="title"> | ||||||
|       <h3>{% trans %}Posters{% endtrans %}</h3> |       <h3>{% trans %}Posters{% endtrans %}</h3> | ||||||
| @@ -38,7 +32,13 @@ | |||||||
|         {% for poster in poster_list %} |         {% for poster in poster_list %} | ||||||
|           <div class="poster{% if not poster.is_moderated %} not_moderated{% endif %}"> |           <div class="poster{% if not poster.is_moderated %} not_moderated{% endif %}"> | ||||||
|             <div class="name">{{ poster.name }}</div> |             <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="dates"> | ||||||
|               <div class="begin">{{ poster.date_begin | localtime | date("d/M/Y H:m") }}</div> |               <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> |               <div class="end">{{ poster.date_end | localtime | date("d/M/Y H:m") }}</div> | ||||||
| @@ -62,7 +62,14 @@ | |||||||
|  |  | ||||||
|     </div> |     </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> |   </div> | ||||||
| {% endblock %} | {% endblock %} | ||||||
|   | |||||||
| @@ -2,28 +2,44 @@ | |||||||
| <html lang="fr"> | <html lang="fr"> | ||||||
|   <head> |   <head> | ||||||
|     <title>{% trans %}Slideshow{% endtrans %}</title> |     <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" /> |     <link href="{{ static('css/slideshow.scss') }}" rel="stylesheet" type="text/css" /> | ||||||
|     <script src="{{ static('bundled/vendored/jquery.min.js') }}"></script> |     <script type="module" src="{{ static('bundled/alpine-index.js') }}"></script> | ||||||
|     <script src="{{ static('com/js/slideshow.js') }}"></script> |     <script type="module" src="{{ static('bundled/com/slideshow-index.ts') }}"></script> | ||||||
|   </head> |   </head> | ||||||
|   <body> |   <body x-data="slideshow([ | ||||||
|     <div id="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"> |       <div id="slides"> | ||||||
|         {% for poster in posters %} |         <template x-for="(poster, index) in posters"> | ||||||
|           <div class="slide {% if loop.first %}center{% else %}right{% endif %}" display_time="{{ poster.display_time }}"> |           <div class="slide" :class="{ | ||||||
|             <img src="{{ poster.file.url }}"> |                                      current: index === current, | ||||||
|  |                                      previous: index !== current && index === previous, | ||||||
|  |                                      }"> | ||||||
|  |             <img :src="poster.url"> | ||||||
|           </div> |           </div> | ||||||
|         {% endfor %} |         </template> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|       <div id="progress_bullets"> |       <div id="progress_bullets"> | ||||||
|         {% for poster in posters %} |         <template x-for="(poster, index) in posters"> | ||||||
|           <div class="bullet {% if loop.first %}active{% endif %}"></div> |           <div class="bullet" :class="{active: current === index}"></div> | ||||||
|         {% endfor %} |         </template> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
|       <div id="progress_bar"></div> |       <progress :value="progress" max="100" x-show="posters.length > 1 && progress > 0"></progress> | ||||||
|  |  | ||||||
|     </div> |     </div> | ||||||
|   </body> |   </body> | ||||||
|   | |||||||
| @@ -31,9 +31,7 @@ | |||||||
|           <td> |           <td> | ||||||
|             <a href="{{ url('com:weekmail_article_edit', article_id=a.id) }}">{% trans %}Edit{% endtrans %}</a> | |             <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="{{ 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="?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> |  | ||||||
|           </td> |           </td> | ||||||
|         </tr> |         </tr> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|   | |||||||
| @@ -18,17 +18,16 @@ from unittest.mock import patch | |||||||
| import pytest | import pytest | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.sites.models import Site | from django.contrib.sites.models import Site | ||||||
| from django.core.files.uploadedfile import SimpleUploadedFile |  | ||||||
| from django.test import Client, TestCase | from django.test import Client, TestCase | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| from django.utils import html | 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 django.utils.translation import gettext as _ | ||||||
| from model_bakery import baker | from model_bakery import baker | ||||||
| from pytest_django.asserts import assertNumQueries, assertRedirects | from pytest_django.asserts import assertNumQueries, assertRedirects | ||||||
|  |  | ||||||
| from club.models import Club, Membership | 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.baker_recipes import subscriber_user | ||||||
| from core.models import AnonymousUser, Group, User | from core.models import AnonymousUser, Group, User | ||||||
|  |  | ||||||
| @@ -207,31 +206,6 @@ class TestWeekmailArticle(TestCase): | |||||||
|         assert not self.article.is_owned_by(self.sli) |         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): | class TestNewsCreation(TestCase): | ||||||
|     @classmethod |     @classmethod | ||||||
|     def setUpTestData(cls): |     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 dateutil.relativedelta import relativedelta | ||||||
| from django.conf import settings | 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.contrib.syndication.views import Feed | ||||||
| from django.core.exceptions import PermissionDenied, ValidationError | from django.core.exceptions import PermissionDenied, ValidationError | ||||||
| from django.db.models import Max | from django.db.models import Max | ||||||
| @@ -50,9 +53,10 @@ from core.auth.mixins import ( | |||||||
|     CanEditPropMixin, |     CanEditPropMixin, | ||||||
|     CanViewMixin, |     CanViewMixin, | ||||||
|     PermissionOrAuthorRequiredMixin, |     PermissionOrAuthorRequiredMixin, | ||||||
|  |     PermissionOrClubBoardRequiredMixin, | ||||||
| ) | ) | ||||||
| from core.models import User | 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 | from core.views.widgets.markdown import MarkdownInput | ||||||
|  |  | ||||||
| # Sith object | # 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): | class ComEditView(ComTabsMixin, CanEditPropMixin, UpdateView): | ||||||
|     model = Sith |     model = Sith | ||||||
|     template_name = "core/edit.jinja" |     template_name = "core/edit.jinja" | ||||||
| @@ -337,7 +334,7 @@ class NewsFeed(Feed): | |||||||
| # Weekmail | # Weekmail | ||||||
|  |  | ||||||
|  |  | ||||||
| class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, DetailView): | class WeekmailPreviewView(ComTabsMixin, CanEditPropMixin, DetailView): | ||||||
|     model = Weekmail |     model = Weekmail | ||||||
|     template_name = "com/weekmail_preview.jinja" |     template_name = "com/weekmail_preview.jinja" | ||||||
|     success_url = reverse_lazy("com:weekmail") |     success_url = reverse_lazy("com:weekmail") | ||||||
| @@ -349,12 +346,11 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai | |||||||
|  |  | ||||||
|     def post(self, request, *args, **kwargs): |     def post(self, request, *args, **kwargs): | ||||||
|         self.object = self.get_object() |         self.object = self.get_object() | ||||||
|  |         messages.success(self.request, _("Weekmail sent successfully")) | ||||||
|         if request.POST["send"] == "validate": |         if request.POST["send"] == "validate": | ||||||
|             try: |             try: | ||||||
|                 self.object.send() |                 self.object.send() | ||||||
|                 return HttpResponseRedirect( |                 return HttpResponseRedirect(reverse("com:weekmail")) | ||||||
|                     reverse("com:weekmail") + "?qn_weekmail_send_success" |  | ||||||
|                 ) |  | ||||||
|             except SMTPRecipientsRefused as e: |             except SMTPRecipientsRefused as e: | ||||||
|                 self.bad_recipients = e.recipients |                 self.bad_recipients = e.recipients | ||||||
|         elif request.POST["send"] == "clean": |         elif request.POST["send"] == "clean": | ||||||
| @@ -365,7 +361,6 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai | |||||||
|                 for u in users: |                 for u in users: | ||||||
|                     u.preferences.receive_weekmail = False |                     u.preferences.receive_weekmail = False | ||||||
|                     u.preferences.save() |                     u.preferences.save() | ||||||
|                 self.quick_notif_list += ["qn_success"] |  | ||||||
|         return super().get(request, *args, **kwargs) |         return super().get(request, *args, **kwargs) | ||||||
|  |  | ||||||
|     def get_object(self, queryset=None): |     def get_object(self, queryset=None): | ||||||
| @@ -379,7 +374,7 @@ class WeekmailPreviewView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, Detai | |||||||
|         return kwargs |         return kwargs | ||||||
|  |  | ||||||
|  |  | ||||||
| class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView): | class WeekmailEditView(ComTabsMixin, CanEditPropMixin, UpdateView): | ||||||
|     model = Weekmail |     model = Weekmail | ||||||
|     template_name = "com/weekmail.jinja" |     template_name = "com/weekmail.jinja" | ||||||
|     form_class = modelform_factory( |     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.rank, prev_art.rank = prev_art.rank, art.rank | ||||||
|                 art.save() |                 art.save() | ||||||
|                 prev_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: |         if "down_article" in request.GET: | ||||||
|             art = get_object_or_404( |             art = get_object_or_404( | ||||||
|                 WeekmailArticle, id=request.GET["down_article"], weekmail=self.object |                 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.rank, next_art.rank = next_art.rank, art.rank | ||||||
|                 art.save() |                 art.save() | ||||||
|                 next_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: |         if "add_article" in request.GET: | ||||||
|             art = get_object_or_404( |             art = get_object_or_404( | ||||||
|                 WeekmailArticle, id=request.GET["add_article"], weekmail=None |                 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 = self.object.articles.aggregate(Max("rank"))["rank__max"] or 0 | ||||||
|             art.rank += 1 |             art.rank += 1 | ||||||
|             art.save() |             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: |         if "del_article" in request.GET: | ||||||
|             art = get_object_or_404( |             art = get_object_or_404( | ||||||
|                 WeekmailArticle, id=request.GET["del_article"], weekmail=self.object |                 WeekmailArticle, id=request.GET["del_article"], weekmail=self.object | ||||||
| @@ -448,7 +452,10 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | |||||||
|             art.weekmail = None |             art.weekmail = None | ||||||
|             art.rank = -1 |             art.rank = -1 | ||||||
|             art.save() |             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) |         return super().get(request, *args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
| @@ -458,9 +465,7 @@ class WeekmailEditView(ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateVi | |||||||
|         return kwargs |         return kwargs | ||||||
|  |  | ||||||
|  |  | ||||||
| class WeekmailArticleEditView( | class WeekmailArticleEditView(ComTabsMixin, CanEditPropMixin, UpdateView): | ||||||
|     ComTabsMixin, QuickNotifMixin, CanEditPropMixin, UpdateView |  | ||||||
| ): |  | ||||||
|     """Edit an article.""" |     """Edit an article.""" | ||||||
|  |  | ||||||
|     model = WeekmailArticle |     model = WeekmailArticle | ||||||
| @@ -472,11 +477,10 @@ class WeekmailArticleEditView( | |||||||
|     pk_url_kwarg = "article_id" |     pk_url_kwarg = "article_id" | ||||||
|     template_name = "core/edit.jinja" |     template_name = "core/edit.jinja" | ||||||
|     success_url = reverse_lazy("com:weekmail") |     success_url = reverse_lazy("com:weekmail") | ||||||
|     quick_notif_url_arg = "qn_weekmail_article_edit" |  | ||||||
|     current_tab = "weekmail" |     current_tab = "weekmail" | ||||||
|  |  | ||||||
|  |  | ||||||
| class WeekmailArticleCreateView(QuickNotifMixin, CreateView): | class WeekmailArticleCreateView(CreateView): | ||||||
|     """Post an article.""" |     """Post an article.""" | ||||||
|  |  | ||||||
|     model = WeekmailArticle |     model = WeekmailArticle | ||||||
| @@ -487,7 +491,6 @@ class WeekmailArticleCreateView(QuickNotifMixin, CreateView): | |||||||
|     ) |     ) | ||||||
|     template_name = "core/create.jinja" |     template_name = "core/create.jinja" | ||||||
|     success_url = reverse_lazy("core:user_tools") |     success_url = reverse_lazy("core:user_tools") | ||||||
|     quick_notif_url_arg = "qn_weekmail_new_article" |  | ||||||
|  |  | ||||||
|     def get_initial(self): |     def get_initial(self): | ||||||
|         if "club" not in self.request.GET: |         if "club" not in self.request.GET: | ||||||
| @@ -558,161 +561,109 @@ class MailingModerateView(View): | |||||||
|         raise PermissionDenied |         raise PermissionDenied | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterAdminViewMixin(IsComAdminMixin, ComTabsMixin): | class PosterListBaseView(PermissionOrClubBoardRequiredMixin, ListView): | ||||||
|     current_tab = "posters" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterListBaseView(PosterAdminViewMixin, ListView): |  | ||||||
|     """List communication posters.""" |     """List communication posters.""" | ||||||
|  |  | ||||||
|     current_tab = "posters" |  | ||||||
|     model = Poster |     model = Poster | ||||||
|     template_name = "com/poster_list.jinja" |     template_name = "com/poster_list.jinja" | ||||||
|  |     permission_required = "com.view_poster" | ||||||
|     def dispatch(self, request, *args, **kwargs): |     ordering = ["-date_begin"] | ||||||
|         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) |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs = super().get_context_data(**kwargs) |         return super().get_context_data(**kwargs) | {"club": self.club} | ||||||
|         if not self.request.user.is_com_admin: |  | ||||||
|             kwargs["club"] = self.club |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterCreateBaseView(PosterAdminViewMixin, CreateView): | class PosterCreateBaseView(PermissionOrClubBoardRequiredMixin, CreateView): | ||||||
|     """Create communication poster.""" |     """Create communication poster.""" | ||||||
|  |  | ||||||
|     current_tab = "posters" |  | ||||||
|     form_class = PosterForm |     form_class = PosterForm | ||||||
|     template_name = "core/create.jinja" |     template_name = "core/create.jinja" | ||||||
|  |     permission_required = "com.add_poster" | ||||||
|  |  | ||||||
|     def get_queryset(self): |     def get_queryset(self): | ||||||
|         return Poster.objects.all() |         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): |     def get_form_kwargs(self): | ||||||
|         kwargs = super().get_form_kwargs() |         return super().get_form_kwargs() | {"user": self.request.user} | ||||||
|         kwargs.update({"user": self.request.user}) |  | ||||||
|         return kwargs |     def get_initial(self): | ||||||
|  |         return {"club": self.club} | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs = super().get_context_data(**kwargs) |         return super().get_context_data(**kwargs) | {"club": self.club} | ||||||
|         if not self.request.user.is_com_admin: |  | ||||||
|             kwargs["club"] = self.club |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|     def form_valid(self, form): |     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 |             form.instance.is_moderated = True | ||||||
|         return super().form_valid(form) |         return super().form_valid(form) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterEditBaseView(PosterAdminViewMixin, UpdateView): | class PosterEditBaseView(PermissionOrClubBoardRequiredMixin, UpdateView): | ||||||
|     """Edit communication poster.""" |     """Edit communication poster.""" | ||||||
|  |  | ||||||
|     pk_url_kwarg = "poster_id" |     pk_url_kwarg = "poster_id" | ||||||
|     current_tab = "posters" |  | ||||||
|     form_class = PosterForm |     form_class = PosterForm | ||||||
|     template_name = "com/poster_edit.jinja" |     template_name = "com/poster_edit.jinja" | ||||||
|  |     permission_required = "com.change_poster" | ||||||
|     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) |  | ||||||
|  |  | ||||||
|     def get_queryset(self): |     def get_queryset(self): | ||||||
|         return Poster.objects.all() |         return Poster.objects.all() | ||||||
|  |  | ||||||
|     def get_form_kwargs(self): |     def get_form_kwargs(self): | ||||||
|         kwargs = super().get_form_kwargs() |         return super().get_form_kwargs() | {"user": self.request.user} | ||||||
|         kwargs.update({"user": self.request.user}) |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs = super().get_context_data(**kwargs) |         return super().get_context_data(**kwargs) | {"club": self.club} | ||||||
|         if hasattr(self, "club"): |  | ||||||
|             kwargs["club"] = self.club |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|     def form_valid(self, form): |     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 |             form.instance.is_moderated = False | ||||||
|         return super().form_valid(form) |         return super().form_valid(form) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterDeleteBaseView(PosterAdminViewMixin, DeleteView): | class PosterDeleteBaseView( | ||||||
|  |     PermissionOrClubBoardRequiredMixin, ComTabsMixin, DeleteView | ||||||
|  | ): | ||||||
|     """Edit communication poster.""" |     """Edit communication poster.""" | ||||||
|  |  | ||||||
|     pk_url_kwarg = "poster_id" |     pk_url_kwarg = "poster_id" | ||||||
|     current_tab = "posters" |     current_tab = "posters" | ||||||
|     model = Poster |     model = Poster | ||||||
|     template_name = "core/delete_confirm.jinja" |     template_name = "core/delete_confirm.jinja" | ||||||
|  |     permission_required = "com.delete_poster" | ||||||
|     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) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterListView(PosterListBaseView): | class PosterListView(ComTabsMixin, PosterListBaseView): | ||||||
|     """List communication posters.""" |     """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): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs = super().get_context_data(**kwargs) |         kwargs = super().get_context_data(**kwargs) | ||||||
|         kwargs["app"] = "com" |         kwargs["app"] = "com" | ||||||
|         return kwargs |         return kwargs | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterCreateView(PosterCreateBaseView): | class PosterCreateView(ComTabsMixin, PosterCreateBaseView): | ||||||
|     """Create communication poster.""" |     """Create communication poster.""" | ||||||
|  |  | ||||||
|  |     current_tab = "posters" | ||||||
|     success_url = reverse_lazy("com:poster_list") |     success_url = reverse_lazy("com:poster_list") | ||||||
|  |     extra_context = {"app": "com"} | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         kwargs = super().get_context_data(**kwargs) |  | ||||||
|         kwargs["app"] = "com" |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterEditView(PosterEditBaseView): | class PosterEditView(ComTabsMixin, PosterEditBaseView): | ||||||
|     """Edit communication poster.""" |     """Edit communication poster.""" | ||||||
|  |  | ||||||
|  |     current_tab = "posters" | ||||||
|     success_url = reverse_lazy("com:poster_list") |     success_url = reverse_lazy("com:poster_list") | ||||||
|  |     extra_context = {"app": "com"} | ||||||
|     def get_context_data(self, **kwargs): |  | ||||||
|         kwargs = super().get_context_data(**kwargs) |  | ||||||
|         kwargs["app"] = "com" |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterDeleteView(PosterDeleteBaseView): | class PosterDeleteView(PosterDeleteBaseView): | ||||||
| @@ -721,44 +672,39 @@ class PosterDeleteView(PosterDeleteBaseView): | |||||||
|     success_url = reverse_lazy("com:poster_list") |     success_url = reverse_lazy("com:poster_list") | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterModerateListView(PosterAdminViewMixin, ListView): | class PosterModerateListView(PermissionRequiredMixin, ComTabsMixin, ListView): | ||||||
|     """Moderate list communication poster.""" |     """Moderate list communication poster.""" | ||||||
|  |  | ||||||
|     current_tab = "posters" |     current_tab = "posters" | ||||||
|     model = Poster |     model = Poster | ||||||
|     template_name = "com/poster_moderate.jinja" |     template_name = "com/poster_moderate.jinja" | ||||||
|     queryset = Poster.objects.filter(is_moderated=False).all() |     queryset = Poster.objects.filter(is_moderated=False).all() | ||||||
|  |     permission_required = "com.moderate_poster" | ||||||
|     def get_context_data(self, **kwargs): |     extra_context = {"app": "com"} | ||||||
|         kwargs = super().get_context_data(**kwargs) |  | ||||||
|         kwargs["app"] = "com" |  | ||||||
|         return kwargs |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PosterModerateView(PosterAdminViewMixin, View): | class PosterModerateView(PermissionRequiredMixin, ComTabsMixin, View): | ||||||
|     """Moderate communication poster.""" |     """Moderate communication poster.""" | ||||||
|  |  | ||||||
|  |     current_tab = "posters" | ||||||
|  |     permission_required = "com.moderate_poster" | ||||||
|  |     extra_context = {"app": "com"} | ||||||
|  |  | ||||||
|     def get(self, request, *args, **kwargs): |     def get(self, request, *args, **kwargs): | ||||||
|         obj = get_object_or_404(Poster, pk=kwargs["object_id"]) |         obj = get_object_or_404(Poster, pk=kwargs["object_id"]) | ||||||
|         if obj.can_be_moderated_by(request.user): |         obj.is_moderated = True | ||||||
|             obj.is_moderated = True |         obj.moderator = request.user | ||||||
|             obj.moderator = request.user |         obj.save() | ||||||
|             obj.save() |         return redirect("com:poster_moderate_list") | ||||||
|             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 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ScreenListView(IsComAdminMixin, ComTabsMixin, ListView): | class ScreenListView(PermissionRequiredMixin, ComTabsMixin, ListView): | ||||||
|     """List communication screens.""" |     """List communication screens.""" | ||||||
|  |  | ||||||
|     current_tab = "screens" |     current_tab = "screens" | ||||||
|     model = Screen |     model = Screen | ||||||
|     template_name = "com/screen_list.jinja" |     template_name = "com/screen_list.jinja" | ||||||
|  |     permission_required = "com.view_screen" | ||||||
|  |  | ||||||
|  |  | ||||||
| class ScreenSlideshowView(DetailView): | class ScreenSlideshowView(DetailView): | ||||||
| @@ -769,12 +715,12 @@ class ScreenSlideshowView(DetailView): | |||||||
|     template_name = "com/screen_slideshow.jinja" |     template_name = "com/screen_slideshow.jinja" | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         kwargs = super().get_context_data(**kwargs) |         return super().get_context_data(**kwargs) | { | ||||||
|         kwargs["posters"] = self.object.active_posters() |             "posters": self.object.active_posters() | ||||||
|         return kwargs |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class ScreenCreateView(IsComAdminMixin, ComTabsMixin, CreateView): | class ScreenCreateView(PermissionRequiredMixin, ComTabsMixin, CreateView): | ||||||
|     """Create communication screen.""" |     """Create communication screen.""" | ||||||
|  |  | ||||||
|     current_tab = "screens" |     current_tab = "screens" | ||||||
| @@ -782,9 +728,10 @@ class ScreenCreateView(IsComAdminMixin, ComTabsMixin, CreateView): | |||||||
|     fields = ["name"] |     fields = ["name"] | ||||||
|     template_name = "core/create.jinja" |     template_name = "core/create.jinja" | ||||||
|     success_url = reverse_lazy("com:screen_list") |     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.""" |     """Edit communication screen.""" | ||||||
|  |  | ||||||
|     pk_url_kwarg = "screen_id" |     pk_url_kwarg = "screen_id" | ||||||
| @@ -793,9 +740,10 @@ class ScreenEditView(IsComAdminMixin, ComTabsMixin, UpdateView): | |||||||
|     fields = ["name"] |     fields = ["name"] | ||||||
|     template_name = "com/screen_edit.jinja" |     template_name = "com/screen_edit.jinja" | ||||||
|     success_url = reverse_lazy("com:screen_list") |     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.""" |     """Delete communication screen.""" | ||||||
|  |  | ||||||
|     pk_url_kwarg = "screen_id" |     pk_url_kwarg = "screen_id" | ||||||
| @@ -803,3 +751,4 @@ class ScreenDeleteView(IsComAdminMixin, ComTabsMixin, DeleteView): | |||||||
|     model = Screen |     model = Screen | ||||||
|     template_name = "core/delete_confirm.jinja" |     template_name = "core/delete_confirm.jinja" | ||||||
|     success_url = reverse_lazy("com:screen_list") |     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, |     UserFamilySchema, | ||||||
|     UserFilterSchema, |     UserFilterSchema, | ||||||
|     UserProfileSchema, |     UserProfileSchema, | ||||||
|  |     UserSchema, | ||||||
| ) | ) | ||||||
| from core.templatetags.renderer import markdown | from core.templatetags.renderer import markdown | ||||||
|  |  | ||||||
| @@ -69,16 +70,22 @@ class MailingListController(ControllerBase): | |||||||
|         return data |         return data | ||||||
|  |  | ||||||
|  |  | ||||||
| @api_controller("/user", permissions=[CanAccessLookup]) | @api_controller("/user") | ||||||
| class UserController(ControllerBase): | class UserController(ControllerBase): | ||||||
|     @route.get("", response=list[UserProfileSchema]) |     @route.get("", response=list[UserProfileSchema], permissions=[CanAccessLookup]) | ||||||
|     def fetch_profiles(self, pks: Query[set[int]]): |     def fetch_profiles(self, pks: Query[set[int]]): | ||||||
|         return User.objects.filter(pk__in=pks) |         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( |     @route.get( | ||||||
|         "/search", |         "/search", | ||||||
|         response=PaginatedResponseSchema[UserProfileSchema], |         response=PaginatedResponseSchema[UserProfileSchema], | ||||||
|         url_name="search_users", |         url_name="search_users", | ||||||
|  |         permissions=[CanAccessLookup], | ||||||
|     ) |     ) | ||||||
|     @paginate(PageNumberPaginationExtra, page_size=20) |     @paginate(PageNumberPaginationExtra, page_size=20) | ||||||
|     def search_users(self, filters: Query[UserFilterSchema]): |     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.contrib.auth.mixins import AccessMixin, PermissionRequiredMixin | ||||||
| from django.core.exceptions import ImproperlyConfigured, PermissionDenied | 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 django.views.generic.base import View | ||||||
|  |  | ||||||
|  | from club.models import Club | ||||||
|  |  | ||||||
| if TYPE_CHECKING: | if TYPE_CHECKING: | ||||||
|     from django.db.models import Model |     from django.db.models import Model | ||||||
|  |  | ||||||
| @@ -297,3 +303,50 @@ class PermissionOrAuthorRequiredMixin(PermissionRequiredMixin): | |||||||
|             self.author_field += "_id" |             self.author_field += "_id" | ||||||
|         author_id = getattr(obj, self.author_field, None) |         author_id = getattr(obj, self.author_field, None) | ||||||
|         return author_id == self.request.user.id |         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( |         s = Subscription( | ||||||
|             member=user, |             member=user, | ||||||
|             subscription_type=subscription_type, |             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_start = s.compute_start(start) | ||||||
|         s.subscription_end = s.compute_end( |         s.subscription_end = s.compute_end( | ||||||
|   | |||||||
| @@ -94,7 +94,11 @@ class Command(BaseCommand): | |||||||
|                 username=self.faker.user_name(), |                 username=self.faker.user_name(), | ||||||
|                 first_name=self.faker.first_name(), |                 first_name=self.faker.first_name(), | ||||||
|                 last_name=self.faker.last_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(), |                 email=self.faker.email(), | ||||||
|                 phone=self.faker.phone_number(), |                 phone=self.faker.phone_number(), | ||||||
|                 address=self.faker.address(), |                 address=self.faker.address(), | ||||||
|   | |||||||
| @@ -560,7 +560,7 @@ class User(AbstractUser): | |||||||
|         """Determine if the object is owned by the user.""" |         """Determine if the object is owned by the user.""" | ||||||
|         if hasattr(obj, "is_owned_by") and obj.is_owned_by(self): |         if hasattr(obj, "is_owned_by") and obj.is_owned_by(self): | ||||||
|             return True |             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 True | ||||||
|         return self.is_root |         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): |         if hasattr(obj, "can_be_edited_by") and obj.can_be_edited_by(self): | ||||||
|             return True |             return True | ||||||
|         if hasattr(obj, "edit_groups"): |         if hasattr(obj, "edit_groups"): | ||||||
|             for pk in obj.edit_groups.values_list("pk", flat=True): |             if ( | ||||||
|                 if self.is_in_group(pk=pk): |                 hasattr(obj, "_prefetched_objects_cache") | ||||||
|                     return True |                 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: |         if isinstance(obj, User) and obj == self: | ||||||
|             return True |             return True | ||||||
|         return self.is_owner(obj) |         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): |         if hasattr(obj, "can_be_viewed_by") and obj.can_be_viewed_by(self): | ||||||
|             return True |             return True | ||||||
|         if hasattr(obj, "view_groups"): |         if hasattr(obj, "view_groups"): | ||||||
|             for pk in obj.view_groups.values_list("pk", flat=True): |             # if "view_groups" has already been prefetched, use | ||||||
|                 if self.is_in_group(pk=pk): |             # the prefetch cache, else fetch only the ids, to make | ||||||
|                     return True |             # 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) |         return self.can_edit(obj) | ||||||
|  |  | ||||||
|     def can_be_edited_by(self, user): |     def can_be_edited_by(self, user): | ||||||
| @@ -636,9 +651,6 @@ class User(AbstractUser): | |||||||
|  |  | ||||||
|  |  | ||||||
| class AnonymousUser(AuthAnonymousUser): | class AnonymousUser(AuthAnonymousUser): | ||||||
|     def __init__(self): |  | ||||||
|         super().__init__() |  | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def was_subscribed(self): |     def was_subscribed(self): | ||||||
|         return False |         return False | ||||||
| @@ -647,10 +659,6 @@ class AnonymousUser(AuthAnonymousUser): | |||||||
|     def is_subscribed(self): |     def is_subscribed(self): | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|     @property |  | ||||||
|     def subscribed(self): |  | ||||||
|         return False |  | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def is_root(self): |     def is_root(self): | ||||||
|         return False |         return False | ||||||
| @@ -1182,6 +1190,18 @@ class NotLocked(LockError): | |||||||
|     pass |     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 | # This function prevents generating migration upon settings change | ||||||
| def get_default_owner_group(): | def get_default_owner_group(): | ||||||
|     return settings.SITH_GROUP_ROOT_ID |     return settings.SITH_GROUP_ROOT_ID | ||||||
| @@ -1251,6 +1271,8 @@ class Page(models.Model): | |||||||
|         _("lock_timeout"), null=True, blank=True, default=None |         _("lock_timeout"), null=True, blank=True, default=None | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |     objects = PageQuerySet.as_manager() | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         unique_together = ("name", "parent") |         unique_together = ("name", "parent") | ||||||
|         permissions = ( |         permissions = ( | ||||||
| @@ -1260,12 +1282,9 @@ class Page(models.Model): | |||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return self.get_full_name() |         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.""" |         """Performs some needed actions before and after saving a page in database.""" | ||||||
|         locked = kwargs.pop("force_lock", False) |         if not force_lock and not self.is_locked(): | ||||||
|         if not locked: |  | ||||||
|             locked = self.is_locked() |  | ||||||
|         if not locked: |  | ||||||
|             raise NotLocked("The page is not locked and thus can not be saved") |             raise NotLocked("The page is not locked and thus can not be saved") | ||||||
|         self.full_clean() |         self.full_clean() | ||||||
|         if not self.id: |         if not self.id: | ||||||
| @@ -1277,7 +1296,7 @@ class Page(models.Model): | |||||||
|         # It also update all the children to maintain correct names |         # It also update all the children to maintain correct names | ||||||
|         self._full_name = self.get_full_name() |         self._full_name = self.get_full_name() | ||||||
|         for c in self.children.all(): |         for c in self.children.all(): | ||||||
|             c.save() |             c.save(force_lock=force_lock) | ||||||
|         super().save(*args, **kwargs) |         super().save(*args, **kwargs) | ||||||
|         self.unset_lock() |         self.unset_lock() | ||||||
|  |  | ||||||
| @@ -1384,23 +1403,23 @@ class Page(models.Model): | |||||||
|  |  | ||||||
|     @cached_property |     @cached_property | ||||||
|     def is_club_page(self): |     def is_club_page(self): | ||||||
|         club_root_page = Page.objects.filter(name=settings.SITH_CLUB_ROOT_PAGE).first() |         return ( | ||||||
|         return club_root_page is not None and ( |             self.name == settings.SITH_CLUB_ROOT_PAGE | ||||||
|             self == club_root_page or club_root_page in self.get_parent_list() |             or settings.SITH_CLUB_ROOT_PAGE in [p.name for p in self.get_parent_list()] | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     @cached_property |     @cached_property | ||||||
|     def need_club_redirection(self): |     def need_club_redirection(self): | ||||||
|         return self.is_club_page and self.name != settings.SITH_CLUB_ROOT_PAGE |         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.unset_lock_recursive() | ||||||
|         self.set_lock_recursive(User.objects.get(id=0)) |         self.set_lock_recursive(User.objects.get(id=0)) | ||||||
|         for child in self.children.all(): |         for child in self.children.all(): | ||||||
|             child.parent = self.parent |             child.parent = self.parent | ||||||
|             child.save() |             child.save() | ||||||
|             child.unset_lock_recursive() |             child.unset_lock_recursive() | ||||||
|         super().delete() |         return super().delete(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PageRev(models.Model): | class PageRev(models.Model): | ||||||
| @@ -1447,9 +1466,12 @@ class PageRev(models.Model): | |||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         return reverse("core:page", kwargs={"page_name": self.page._full_name}) |         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) |         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(): | def get_notification_types(): | ||||||
|     return settings.SITH_NOTIFICATIONS |     return settings.SITH_NOTIFICATIONS | ||||||
|   | |||||||
| @@ -34,6 +34,22 @@ class SimpleUserSchema(ModelSchema): | |||||||
|         fields = ["id", "nick_name", "first_name", "last_name"] |         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): | class UserProfileSchema(ModelSchema): | ||||||
|     """The necessary information to show a user profile""" |     """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 sort from "@alpinejs/sort"; | ||||||
| import Alpine from "alpinejs"; | import Alpine from "alpinejs"; | ||||||
|  |  | ||||||
| Alpine.plugin(sort); | Alpine.plugin(sort); | ||||||
|  | Alpine.magic("notifications", notificationPlugin); | ||||||
| window.Alpine = Alpine; | window.Alpine = Alpine; | ||||||
|  |  | ||||||
| window.addEventListener("DOMContentLoaded", () => { | 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 { |   > .ts-control { | ||||||
|     box-shadow: none; |     box-shadow: none; | ||||||
|     max-width: 300px; |     max-width: 300px; | ||||||
|  |     width: 300px; | ||||||
|     background-color: var(--nf-input-background-color); |     background-color: var(--nf-input-background-color); | ||||||
|  |  | ||||||
|     &::after { |     &::after { | ||||||
|   | |||||||
| @@ -47,6 +47,7 @@ | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   input, |   input, | ||||||
|  |   select, | ||||||
|   textarea[type="text"], |   textarea[type="text"], | ||||||
|   [type="number"], |   [type="number"], | ||||||
|   .ts-control { |   .ts-control { | ||||||
| @@ -153,10 +154,8 @@ form { | |||||||
|     margin-bottom: 1rem; |     margin-bottom: 1rem; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .row { |   .row > label { | ||||||
|     label { |     margin: unset; | ||||||
|       margin: unset; |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // ------------- LABEL |   // ------------- 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="text"], | ||||||
|   input[type="checkbox"], |   input[type="checkbox"], | ||||||
|   | |||||||
| @@ -321,7 +321,6 @@ $hovered-red-text-color: #ff4d4d; | |||||||
|  |  | ||||||
|         >#header_notif { |         >#header_notif { | ||||||
|           box-sizing: border-box; |           box-sizing: border-box; | ||||||
|           display: none; |  | ||||||
|           position: absolute; |           position: absolute; | ||||||
|           margin: 0; |           margin: 0; | ||||||
|           background-color: whitesmoke; |           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------------------------------*/ |   /*--------------------------------CONTENT------------------------------*/ | ||||||
|   #quick_notif { |  | ||||||
|     width: 100%; |  | ||||||
|     margin: 0 auto; |  | ||||||
|     list-style-type: none; |  | ||||||
|     background: $second-color; |  | ||||||
|  |  | ||||||
|     li { |  | ||||||
|       padding: 10px; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   #content { |   #content { | ||||||
|     padding: 1em 1%; |     padding: 1em 1%; | ||||||
|     box-shadow: $shadow-color 0 5px 10px; |     box-shadow: $shadow-color 0 5px 10px; | ||||||
| @@ -514,9 +503,17 @@ th { | |||||||
|   text-align: center; |   text-align: center; | ||||||
|   padding: 5px 10px; |   padding: 5px 10px; | ||||||
|  |  | ||||||
|  |   >input[type="checkbox"] { | ||||||
|  |     padding: unset; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   >ul { |   >ul { | ||||||
|     margin-top: 0; |     margin-top: 0; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   >input[type="checkbox"] { | ||||||
|  |     padding: unset; | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| td { | td { | ||||||
|   | |||||||
| @@ -32,10 +32,6 @@ | |||||||
|       <script type="module" src="{{ static('bundled/country-flags-index.ts') }}"></script> |       <script type="module" src="{{ static('bundled/country-flags-index.ts') }}"></script> | ||||||
|       <script type="module" src="{{ static('bundled/core/tooltips-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_css %}{% endblock %} | ||||||
|       {% block additional_js %}{% endblock %} |       {% block additional_js %}{% endblock %} | ||||||
|     {% endblock %} |     {% endblock %} | ||||||
| @@ -74,17 +70,15 @@ | |||||||
|  |  | ||||||
|     <div id="page"> |     <div id="page"> | ||||||
|  |  | ||||||
|       <ul id="quick_notif"> |  | ||||||
|         {% for n in quick_notifs %} |  | ||||||
|           <li>{{ n }}</li> |  | ||||||
|         {% endfor %} |  | ||||||
|       </ul> |  | ||||||
|  |  | ||||||
|       <div id="content"> |       <div id="content"> | ||||||
|         {%- block tabs -%} |         {%- block tabs -%} | ||||||
|           {% include "core/base/tabs.jinja" %} |           {% include "core/base/tabs.jinja" %} | ||||||
|         {%- endblock -%} |         {%- endblock -%} | ||||||
|  |  | ||||||
|  |         {% block notifications %} | ||||||
|  |           {% include "core/base/notifications.jinja" %} | ||||||
|  |         {% endblock %} | ||||||
|  |  | ||||||
|         {%- block errors -%} |         {%- block errors -%} | ||||||
|           {% if error %} |           {% if error %} | ||||||
|             {{ error }} |             {{ error }} | ||||||
| @@ -101,16 +95,6 @@ | |||||||
|     {% endblock %} |     {% endblock %} | ||||||
|  |  | ||||||
|     {% block script %} |     {% 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 %} |     {% endblock %} | ||||||
|   </body> |   </body> | ||||||
| </html> | </html> | ||||||
|   | |||||||
| @@ -74,25 +74,25 @@ | |||||||
|             {% endif %} |             {% endif %} | ||||||
|           ></a> |           ></a> | ||||||
|         </div> |         </div> | ||||||
|         <div class="notification"> |         <div class="notification" x-data="{display: false}" :class="{white: display}"> | ||||||
|           <a href="#" onclick="displayNotif()"> |           <a href="#" @click.prevent="display = !display"> | ||||||
|             <i class="fa-regular fa-bell"></i> |             <i :class="`fa-${display ? 'solid': 'regular'} fa-bell`" x-transition></i> | ||||||
|             {% set notification_count = user.notifications.filter(viewed=False).count() %} |             {% set notifications = user.notifications.filter(viewed=False).order_by("-date")|list %} | ||||||
|  |  | ||||||
|             {% if notification_count > 0 %} |             {%- if notifications|length > 0 -%} | ||||||
|               <span> |               <span> | ||||||
|                 {% if notification_count < 100 %} |                 {% if notifications|length < 100 %} | ||||||
|                   {{ notification_count }} |                   {{ notifications|length }} | ||||||
|                 {% else %} |                 {%- else -%} | ||||||
|                     |                   99+ | ||||||
|                 {% endif %} |                 {%- endif -%} | ||||||
|               </span> |               </span> | ||||||
|             {% endif %} |             {% endif %} | ||||||
|           </a> |           </a> | ||||||
|           <div id="header_notif"> |           <div id="header_notif" x-show="display" x-cloak x-transition @click.outside="display = false"> | ||||||
|             <ul> |             <ul> | ||||||
|               {% if user.notifications.filter(viewed=False).count() > 0 %} |               {%- if notifications|length > 0 -%} | ||||||
|                 {% for n in user.notifications.filter(viewed=False).order_by('-date') %} |                 {%- for n in notifications -%} | ||||||
|                   <li> |                   <li> | ||||||
|                     <a href="{{ url("core:notification", notif_id=n.id) }}"> |                     <a href="{{ url("core:notification", notif_id=n.id) }}"> | ||||||
|                       <div class="datetime"> |                       <div class="datetime"> | ||||||
| @@ -108,10 +108,10 @@ | |||||||
|                       </div> |                       </div> | ||||||
|                     </a> |                     </a> | ||||||
|                   </li> |                   </li> | ||||||
|                 {% endfor %} |                 {%- endfor -%} | ||||||
|               {% else %} |               {%- else -%} | ||||||
|                 <li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li> |                 <li class="empty-notification">{% trans %}You do not have any unread notification{% endtrans %}</li> | ||||||
|               {% endif %} |               {%- endif -%} | ||||||
|             </ul> |             </ul> | ||||||
|             <div class="options"> |             <div class="options"> | ||||||
|               <a href="{{ url('core:notification_list') }}"> |               <a href="{{ url('core:notification_list') }}"> | ||||||
|   | |||||||
							
								
								
									
										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") }} |       {{ select_all_checkbox("add_users") }} | ||||||
|       <hr> |       <hr> | ||||||
|       {% csrf_token %} |       {% csrf_token %} | ||||||
|  |       {{ form.non_field_errors() }} | ||||||
|       <label for="{{ form.users_removed.id_for_label }}">{{ form.users_removed.label }} :</label> |       <label for="{{ form.users_removed.id_for_label }}">{{ form.users_removed.label }} :</label> | ||||||
|       {{ form.users_removed.errors }} |       {{ form.users_removed.errors }} | ||||||
|       {% for user in form.users_removed %} |       {% 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}}(true);">{% trans %}Select All{% endtrans %}</button> | ||||||
|   <button type="button" onclick="checkbox_{{form_id}}(false);">{% trans %}Unselect All{% endtrans %}</button> |   <button type="button" onclick="checkbox_{{form_id}}(false);">{% trans %}Unselect All{% endtrans %}</button> | ||||||
| {% endmacro %} | {% endmacro %} | ||||||
|  |  | ||||||
|  | {% macro update_notifications(messages, clear) %} | ||||||
|  |   {# Update notification area from new messages sent by django backend | ||||||
|  |      This is useful when performing fragment swaps to keep messages up to date | ||||||
|  |      Without this, the fragment would need to take control of the notification area and | ||||||
|  |      this would be an issue when having more than one fragment | ||||||
|  |  | ||||||
|  |      Parameters: | ||||||
|  |       messages: messages from django.contrib | ||||||
|  |       clear   : optional boolean that controls if notifications should be cleared first. True is the default | ||||||
|  |   #} | ||||||
|  |   {% set clear = clear|default(true) %} | ||||||
|  |   {% if messages %} | ||||||
|  |     <div x-init="() => { | ||||||
|  |                  {% if clear %} | ||||||
|  |                    $notifications.clear() | ||||||
|  |                  {% endif %} | ||||||
|  |                  {% for message in messages %} | ||||||
|  |                    $notifications.{{ message.tags }}('{{ message }}') | ||||||
|  |                  {% endfor %} | ||||||
|  |                  }"></div> | ||||||
|  |   {% endif %} | ||||||
|  | {% endmacro %} | ||||||
|   | |||||||
| @@ -5,16 +5,12 @@ | |||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   {% if page_list %} |   <h3>{% trans %}Page list{% endtrans %}</h3> | ||||||
|     <h3>{% trans %}Page list{% endtrans %}</h3> |   <ul> | ||||||
|     <ul> |     {% for p in page_list %} | ||||||
|       {% for p in page_list %} |       <li><a href="{{ p.get_absolute_url() }}">{{ p.display_name }}</a></li> | ||||||
|         <li><a href="{{ p.get_absolute_url() }}">{{ p.get_display_name() }}</a></li> |     {% endfor %} | ||||||
|       {% endfor %} |   </ul> | ||||||
|     </ul> |  | ||||||
|   {% else %} |  | ||||||
|     {% trans %}There is no page in this website.{% endtrans %} |  | ||||||
|   {% endif %} |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -30,7 +30,11 @@ | |||||||
|               - {{ purchase.date|localtime|time(DATETIME_FORMAT) }} |               - {{ purchase.date|localtime|time(DATETIME_FORMAT) }} | ||||||
|             </td> |             </td> | ||||||
|             <td>{{ purchase.counter }}</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.label }}</td> | ||||||
|             <td>{{ purchase.quantity }}</td> |             <td>{{ purchase.quantity }}</td> | ||||||
|             <td>{{ purchase.quantity * purchase.unit_price }} €</td> |             <td>{{ purchase.quantity * purchase.unit_price }} €</td> | ||||||
|   | |||||||
| @@ -1,23 +1,25 @@ | |||||||
| {% for js in statics.js %} | {% spaceless %} | ||||||
|   <script-once type="module" src="{{ js }}"></script-once> |   {% for js in statics.js %} | ||||||
| {% endfor %} |     <script-once type="module" src="{{ js }}"></script-once> | ||||||
| {% 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 %} |   {% endfor %} | ||||||
|   {% if group_name %} |   {% for css in statics.css %} | ||||||
|     </optgroup> |     <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 %} |   {% endif %} | ||||||
| {% endfor %} |   </{{ component }}> | ||||||
| {% if initial %} | {% endspaceless %} | ||||||
|   <slot style="display:none" name="initial">{{ initial }}</slot> |  | ||||||
| {% endif %} |  | ||||||
| </{{ component }}> |  | ||||||
							
								
								
									
										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.models import Group, User | ||||||
| from core.views import UserTabsMixin | 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 | from eboutic.models import Invoice, InvoiceItem | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -129,6 +130,31 @@ def test_user_account_not_found(client: Client): | |||||||
|     assert res.status_code == 404 |     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): | class TestFilterInactive(TestCase): | ||||||
|     @classmethod |     @classmethod | ||||||
|     def setUpTestData(cls): |     def setUpTestData(cls): | ||||||
|   | |||||||
| @@ -115,7 +115,7 @@ class SelectUser(TextInput): | |||||||
|  |  | ||||||
| def validate_future_timestamp(value: date | datetime): | def validate_future_timestamp(value: date | datetime): | ||||||
|     if value <= now(): |     if value <= now(): | ||||||
|         raise ValueError(_("Ensure this timestamp is set in the future")) |         raise ValidationError(_("Ensure this timestamp is set in the future")) | ||||||
|  |  | ||||||
|  |  | ||||||
| class FutureDateTimeField(forms.DateTimeField): | class FutureDateTimeField(forms.DateTimeField): | ||||||
|   | |||||||
| @@ -2,7 +2,6 @@ import copy | |||||||
| import inspect | import inspect | ||||||
| from typing import Any, ClassVar, LiteralString, Protocol, Unpack | from typing import Any, ClassVar, LiteralString, Protocol, Unpack | ||||||
|  |  | ||||||
| from django.conf import settings |  | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.http import HttpRequest, HttpResponse | from django.http import HttpRequest, HttpResponse | ||||||
| from django.template.loader import render_to_string | from django.template.loader import render_to_string | ||||||
| @@ -41,36 +40,6 @@ class TabedViewMixin(View): | |||||||
|         return kwargs |         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: | class AllowFragment: | ||||||
|     """Add `is_fragment` to templates. It's only True if the request is emitted by htmx""" |     """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" | # OR WITHIN THE LOCAL FILE "LICENSE" | ||||||
| # | # | ||||||
| # | # | ||||||
|  |  | ||||||
| from django.contrib.auth.mixins import PermissionRequiredMixin | 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 | # This file contains all the views that concern the page model | ||||||
| from django.forms.models import modelform_factory | from django.forms.models import modelform_factory | ||||||
| @@ -40,10 +43,26 @@ class CanEditPagePropMixin(CanEditPropMixin): | |||||||
|         return res |         return res | ||||||
|  |  | ||||||
|  |  | ||||||
| class PageListView(CanViewMixin, ListView): | class PageListView(ListView): | ||||||
|     model = Page |     model = Page | ||||||
|     template_name = "core/page_list.jinja" |     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): | class PageView(CanViewMixin, DetailView): | ||||||
|     model = Page |     model = Page | ||||||
| @@ -167,7 +186,7 @@ class PageEditViewBase(CanEditMixin, UpdateView): | |||||||
|     ) |     ) | ||||||
|     template_name = "core/pagerev_edit.jinja" |     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"]) |         self.page = Page.get_page_by_full_name(self.kwargs["page_name"]) | ||||||
|         return self._get_revision() |         return self._get_revision() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -65,7 +65,7 @@ from core.views.forms import ( | |||||||
|     UserGroupsForm, |     UserGroupsForm, | ||||||
|     UserProfileForm, |     UserProfileForm, | ||||||
| ) | ) | ||||||
| from core.views.mixins import QuickNotifMixin, TabedViewMixin, UseFragmentsMixin | from core.views.mixins import TabedViewMixin, UseFragmentsMixin | ||||||
| from counter.models import Counter, Refilling, Selling | from counter.models import Counter, Refilling, Selling | ||||||
| from eboutic.models import Invoice | from eboutic.models import Invoice | ||||||
| from subscription.models import Subscription | from subscription.models import Subscription | ||||||
| @@ -564,7 +564,7 @@ class UserUpdateGroupView(UserTabsMixin, CanEditPropMixin, UpdateView): | |||||||
|     current_tab = "groups" |     current_tab = "groups" | ||||||
|  |  | ||||||
|  |  | ||||||
| class UserToolsView(LoginRequiredMixin, QuickNotifMixin, UserTabsMixin, TemplateView): | class UserToolsView(LoginRequiredMixin, UserTabsMixin, TemplateView): | ||||||
|     """Displays the logged user's tools.""" |     """Displays the logged user's tools.""" | ||||||
|  |  | ||||||
|     template_name = "core/user_tools.jinja" |     template_name = "core/user_tools.jinja" | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ from counter.models import ( | |||||||
|     Counter, |     Counter, | ||||||
|     Customer, |     Customer, | ||||||
|     Eticket, |     Eticket, | ||||||
|  |     InvoiceCall, | ||||||
|     Permanency, |     Permanency, | ||||||
|     Product, |     Product, | ||||||
|     ProductType, |     ProductType, | ||||||
| @@ -160,3 +161,11 @@ class CashRegisterSummaryAdmin(SearchModelAdmin): | |||||||
| class EticketAdmin(SearchModelAdmin): | class EticketAdmin(SearchModelAdmin): | ||||||
|     list_display = ("product", "event_date", "event_title") |     list_display = ("product", "event_date", "event_title") | ||||||
|     search_fields = ("product__name", "event_title") |     search_fields = ("product__name", "event_title") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @admin.register(InvoiceCall) | ||||||
|  | class InvoiceCallAdmin(SearchModelAdmin): | ||||||
|  |     list_display = ("club", "month", "is_validated") | ||||||
|  |     search_fields = ("club__name",) | ||||||
|  |     list_filter = (("club", admin.RelatedOnlyFieldListFilter),) | ||||||
|  |     date_hierarchy = "month" | ||||||
|   | |||||||
							
								
								
									
										187
									
								
								counter/forms.py
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								counter/forms.py
									
									
									
									
									
								
							| @@ -1,13 +1,26 @@ | |||||||
|  | import json | ||||||
| import math | import math | ||||||
|  | import uuid | ||||||
|  | from datetime import date | ||||||
|  |  | ||||||
|  | from dateutil.relativedelta import relativedelta | ||||||
| from django import forms | from django import forms | ||||||
| from django.db.models import Q | from django.db.models import Exists, OuterRef, Q | ||||||
|  | from django.forms import BaseModelFormSet | ||||||
|  | from django.utils.timezone import now | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  | from django_celery_beat.models import ClockedSchedule | ||||||
| from phonenumber_field.widgets import RegionalPhoneNumberWidget | from phonenumber_field.widgets import RegionalPhoneNumberWidget | ||||||
|  |  | ||||||
|  | from club.models import Club | ||||||
| from club.widgets.ajax_select import AutoCompleteSelectClub | from club.widgets.ajax_select import AutoCompleteSelectClub | ||||||
| from core.models import User | from core.models import User | ||||||
| from core.views.forms import NFCTextInput, SelectDate, SelectDateTime | from core.views.forms import ( | ||||||
|  |     FutureDateTimeField, | ||||||
|  |     NFCTextInput, | ||||||
|  |     SelectDate, | ||||||
|  |     SelectDateTime, | ||||||
|  | ) | ||||||
| from core.views.widgets.ajax_select import ( | from core.views.widgets.ajax_select import ( | ||||||
|     AutoCompleteSelect, |     AutoCompleteSelect, | ||||||
|     AutoCompleteSelectMultipleGroup, |     AutoCompleteSelectMultipleGroup, | ||||||
| @@ -19,10 +32,14 @@ from counter.models import ( | |||||||
|     Counter, |     Counter, | ||||||
|     Customer, |     Customer, | ||||||
|     Eticket, |     Eticket, | ||||||
|  |     InvoiceCall, | ||||||
|     Product, |     Product, | ||||||
|     Refilling, |     Refilling, | ||||||
|     ReturnableProduct, |     ReturnableProduct, | ||||||
|  |     ScheduledProductAction, | ||||||
|  |     Selling, | ||||||
|     StudentCard, |     StudentCard, | ||||||
|  |     get_product_actions, | ||||||
| ) | ) | ||||||
| from counter.widgets.ajax_select import ( | from counter.widgets.ajax_select import ( | ||||||
|     AutoCompleteSelectMultipleCounter, |     AutoCompleteSelectMultipleCounter, | ||||||
| @@ -158,7 +175,101 @@ class CounterEditForm(forms.ModelForm): | |||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProductEditForm(forms.ModelForm): | class ScheduledProductActionForm(forms.ModelForm): | ||||||
|  |     """Form for automatic product archiving. | ||||||
|  |  | ||||||
|  |     The `save` method will update or create tasks using celery-beat. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     required_css_class = "required" | ||||||
|  |     prefix = "scheduled" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         model = ScheduledProductAction | ||||||
|  |         fields = ["task"] | ||||||
|  |         widgets = {"task": forms.RadioSelect(choices=get_product_actions)} | ||||||
|  |         labels = {"task": _("Action")} | ||||||
|  |         help_texts = {"task": ""} | ||||||
|  |  | ||||||
|  |     trigger_at = FutureDateTimeField( | ||||||
|  |         label=_("Date and time of action"), widget=SelectDateTime | ||||||
|  |     ) | ||||||
|  |     counters = forms.ModelMultipleChoiceField( | ||||||
|  |         label=_("New counters"), | ||||||
|  |         help_text=_("The selected counters will replace the current ones"), | ||||||
|  |         required=False, | ||||||
|  |         widget=AutoCompleteSelectMultipleCounter, | ||||||
|  |         queryset=Counter.objects.all(), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, product: Product, **kwargs): | ||||||
|  |         self.product = product | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |         if not self.instance._state.adding: | ||||||
|  |             self.fields["trigger_at"].initial = self.instance.clocked.clocked_time | ||||||
|  |             self.fields["counters"].initial = json.loads(self.instance.kwargs).get( | ||||||
|  |                 "counters" | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     def clean(self): | ||||||
|  |         if not self.changed_data or "trigger_at" in self.errors: | ||||||
|  |             return super().clean() | ||||||
|  |         if "trigger_at" in self.changed_data: | ||||||
|  |             if not self.instance.clocked_id: | ||||||
|  |                 self.instance.clocked = ClockedSchedule( | ||||||
|  |                     clocked_time=self.cleaned_data["trigger_at"] | ||||||
|  |                 ) | ||||||
|  |             else: | ||||||
|  |                 self.instance.clocked.clocked_time = self.cleaned_data["trigger_at"] | ||||||
|  |             self.instance.clocked.save() | ||||||
|  |         task_kwargs = {"product_id": self.product.id} | ||||||
|  |         if ( | ||||||
|  |             self.cleaned_data["task"] == "counter.tasks.change_counters" | ||||||
|  |             and "counters" in self.changed_data | ||||||
|  |         ): | ||||||
|  |             task_kwargs["counters"] = [c.id for c in self.cleaned_data["counters"]] | ||||||
|  |         self.instance.product = self.product | ||||||
|  |         self.instance.kwargs = json.dumps(task_kwargs) | ||||||
|  |         self.instance.name = ( | ||||||
|  |             f"{self.cleaned_data['task']} - {self.product} - {uuid.uuid4()}" | ||||||
|  |         ) | ||||||
|  |         return super().clean() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseScheduledProductActionFormSet(BaseModelFormSet): | ||||||
|  |     def __init__(self, *args, product: Product, **kwargs): | ||||||
|  |         if product.id: | ||||||
|  |             queryset = ( | ||||||
|  |                 product.scheduled_actions.filter( | ||||||
|  |                     enabled=True, clocked__clocked_time__gt=now() | ||||||
|  |                 ) | ||||||
|  |                 .order_by("clocked__clocked_time") | ||||||
|  |                 .select_related("clocked") | ||||||
|  |             ) | ||||||
|  |         else: | ||||||
|  |             queryset = ScheduledProductAction.objects.none() | ||||||
|  |         form_kwargs = {"product": product} | ||||||
|  |         super().__init__(*args, queryset=queryset, form_kwargs=form_kwargs, **kwargs) | ||||||
|  |  | ||||||
|  |     def delete_existing(self, obj: ScheduledProductAction, commit: bool = True):  # noqa FBT001 | ||||||
|  |         clocked = obj.clocked | ||||||
|  |         super().delete_existing(obj, commit=commit) | ||||||
|  |         if commit: | ||||||
|  |             clocked.delete() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ScheduledProductActionFormSet = forms.modelformset_factory( | ||||||
|  |     ScheduledProductAction, | ||||||
|  |     ScheduledProductActionForm, | ||||||
|  |     formset=BaseScheduledProductActionFormSet, | ||||||
|  |     absolute_max=None, | ||||||
|  |     can_delete=True, | ||||||
|  |     can_delete_extra=False, | ||||||
|  |     extra=2, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ProductForm(forms.ModelForm): | ||||||
|     error_css_class = "error" |     error_css_class = "error" | ||||||
|     required_css_class = "required" |     required_css_class = "required" | ||||||
|  |  | ||||||
| @@ -199,22 +310,21 @@ class ProductEditForm(forms.ModelForm): | |||||||
|         queryset=Counter.objects.all(), |         queryset=Counter.objects.all(), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, instance=None, **kwargs): | ||||||
|         super().__init__(*args, **kwargs) |         super().__init__(*args, instance=instance, **kwargs) | ||||||
|         if self.instance.id: |         if self.instance.id: | ||||||
|             self.fields["counters"].initial = self.instance.counters.all() |             self.fields["counters"].initial = self.instance.counters.all() | ||||||
|  |         self.action_formset = ScheduledProductActionFormSet( | ||||||
|  |             *args, product=self.instance, **kwargs | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def is_valid(self): | ||||||
|  |         return super().is_valid() and self.action_formset.is_valid() | ||||||
|  |  | ||||||
|     def save(self, *args, **kwargs): |     def save(self, *args, **kwargs): | ||||||
|         ret = super().save(*args, **kwargs) |         ret = super().save(*args, **kwargs) | ||||||
|         if self.fields["counters"].initial: |         self.instance.counters.set(self.cleaned_data["counters"]) | ||||||
|             # Remove the product from all counter it was added to |         self.action_formset.save() | ||||||
|             # It will then only be added to selected counters |  | ||||||
|             for counter in self.fields["counters"].initial: |  | ||||||
|                 counter.products.remove(self.instance) |  | ||||||
|                 counter.save() |  | ||||||
|         for counter in self.cleaned_data["counters"]: |  | ||||||
|             counter.products.add(self.instance) |  | ||||||
|             counter.save() |  | ||||||
|         return ret |         return ret | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -266,7 +376,7 @@ class CloseCustomerAccountForm(forms.Form): | |||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ProductForm(forms.Form): | class BasketProductForm(forms.Form): | ||||||
|     quantity = forms.IntegerField(min_value=1, required=True) |     quantity = forms.IntegerField(min_value=1, required=True) | ||||||
|     id = forms.IntegerField(min_value=0, required=True) |     id = forms.IntegerField(min_value=0, required=True) | ||||||
|  |  | ||||||
| @@ -371,5 +481,50 @@ class BaseBasketForm(forms.BaseFormSet): | |||||||
|  |  | ||||||
|  |  | ||||||
| BasketForm = forms.formset_factory( | BasketForm = forms.formset_factory( | ||||||
|     ProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 |     BasketProductForm, formset=BaseBasketForm, absolute_max=None, min_num=1 | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class InvoiceCallForm(forms.Form): | ||||||
|  |     def __init__(self, *args, month: date, **kwargs): | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |         self.month = month | ||||||
|  |         self.clubs = list( | ||||||
|  |             Club.objects.filter( | ||||||
|  |                 Exists( | ||||||
|  |                     Selling.objects.filter( | ||||||
|  |                         club=OuterRef("pk"), | ||||||
|  |                         date__gte=month, | ||||||
|  |                         date__lte=month + relativedelta(months=1), | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ).annotate( | ||||||
|  |                 validated_invoice=Exists( | ||||||
|  |                     InvoiceCall.objects.filter( | ||||||
|  |                         club=OuterRef("pk"), month=month, is_validated=True | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         self.fields = { | ||||||
|  |             str(club.id): forms.BooleanField( | ||||||
|  |                 required=False, initial=club.validated_invoice | ||||||
|  |             ) | ||||||
|  |             for club in self.clubs | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def save(self): | ||||||
|  |         invoice_calls = [ | ||||||
|  |             InvoiceCall( | ||||||
|  |                 month=self.month, | ||||||
|  |                 club_id=club.id, | ||||||
|  |                 is_validated=self.cleaned_data.get(str(club.id), False), | ||||||
|  |             ) | ||||||
|  |             for club in self.clubs | ||||||
|  |         ] | ||||||
|  |         InvoiceCall.objects.bulk_create( | ||||||
|  |             invoice_calls, | ||||||
|  |             update_conflicts=True, | ||||||
|  |             update_fields=["is_validated"], | ||||||
|  |             unique_fields=["month", "club"], | ||||||
|  |         ) | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								counter/migrations/0032_scheduledproductaction.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								counter/migrations/0032_scheduledproductaction.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | # Generated by Django 5.2.3 on 2025-09-14 11:29 | ||||||
|  |  | ||||||
|  | import django.db.models.deletion | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |     dependencies = [ | ||||||
|  |         ("counter", "0031_alter_counter_options"), | ||||||
|  |         ("django_celery_beat", "0019_alter_periodictasks_options"), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name="ScheduledProductAction", | ||||||
|  |             fields=[ | ||||||
|  |                 ( | ||||||
|  |                     "periodictask_ptr", | ||||||
|  |                     models.OneToOneField( | ||||||
|  |                         auto_created=True, | ||||||
|  |                         on_delete=django.db.models.deletion.CASCADE, | ||||||
|  |                         parent_link=True, | ||||||
|  |                         primary_key=True, | ||||||
|  |                         serialize=False, | ||||||
|  |                         to="django_celery_beat.periodictask", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "product", | ||||||
|  |                     models.ForeignKey( | ||||||
|  |                         on_delete=django.db.models.deletion.CASCADE, | ||||||
|  |                         related_name="scheduled_actions", | ||||||
|  |                         to="counter.product", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |             ], | ||||||
|  |             options={"verbose_name": "Product scheduled action"}, | ||||||
|  |             bases=("django_celery_beat.periodictask",), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										51
									
								
								counter/migrations/0033_invoicecall.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								counter/migrations/0033_invoicecall.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,51 @@ | |||||||
|  | # Generated by Django 5.2.3 on 2025-10-15 21:54 | ||||||
|  |  | ||||||
|  | import django.db.models.deletion | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  | import counter.models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |     dependencies = [ | ||||||
|  |         ("club", "0014_alter_club_options_rename_unix_name_club_slug_name_and_more"), | ||||||
|  |         ("counter", "0032_scheduledproductaction"), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name="InvoiceCall", | ||||||
|  |             fields=[ | ||||||
|  |                 ( | ||||||
|  |                     "id", | ||||||
|  |                     models.AutoField( | ||||||
|  |                         auto_created=True, | ||||||
|  |                         primary_key=True, | ||||||
|  |                         serialize=False, | ||||||
|  |                         verbose_name="ID", | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|  |                     "is_validated", | ||||||
|  |                     models.BooleanField(default=False, verbose_name="is validated"), | ||||||
|  |                 ), | ||||||
|  |                 ("month", counter.models.MonthField(verbose_name="invoice date")), | ||||||
|  |                 ( | ||||||
|  |                     "club", | ||||||
|  |                     models.ForeignKey( | ||||||
|  |                         on_delete=django.db.models.deletion.CASCADE, to="club.club" | ||||||
|  |                     ), | ||||||
|  |                 ), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 "verbose_name": "Invoice call", | ||||||
|  |                 "verbose_name_plural": "Invoice calls", | ||||||
|  |                 "constraints": [ | ||||||
|  |                     models.UniqueConstraint( | ||||||
|  |                         fields=("club", "month"), | ||||||
|  |                         name="counter_invoicecall_unique_club_month", | ||||||
|  |                     ) | ||||||
|  |                 ], | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
| @@ -15,6 +15,7 @@ | |||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| import base64 | import base64 | ||||||
|  | import contextlib | ||||||
| import os | import os | ||||||
| import random | import random | ||||||
| import string | import string | ||||||
| @@ -34,6 +35,7 @@ from django.urls import reverse | |||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  | from django_celery_beat.models import PeriodicTask | ||||||
| from django_countries.fields import CountryField | from django_countries.fields import CountryField | ||||||
| from ordered_model.models import OrderedModel | from ordered_model.models import OrderedModel | ||||||
| from phonenumber_field.modelfields import PhoneNumberField | from phonenumber_field.modelfields import PhoneNumberField | ||||||
| @@ -445,7 +447,8 @@ class Product(models.Model): | |||||||
|         buying_groups = list(self.buying_groups.all()) |         buying_groups = list(self.buying_groups.all()) | ||||||
|         if not buying_groups: |         if not buying_groups: | ||||||
|             return True |             return True | ||||||
|         return any(user.is_in_group(pk=group.id) for group in buying_groups) |         res = any(user.is_in_group(pk=group.id) for group in buying_groups) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def profit(self): |     def profit(self): | ||||||
| @@ -479,7 +482,7 @@ class CounterQuerySet(models.QuerySet): | |||||||
|         return self.annotate(has_annotated_barman=Exists(subquery)) |         return self.annotate(has_annotated_barman=Exists(subquery)) | ||||||
|  |  | ||||||
|     def annotate_is_open(self) -> Self: |     def annotate_is_open(self) -> Self: | ||||||
|         """Annotate tue queryset with the `is_open` field. |         """Annotate the queryset with the `is_open` field. | ||||||
|  |  | ||||||
|         For each counter, if `is_open=True`, then the counter is currently opened. |         For each counter, if `is_open=True`, then the counter is currently opened. | ||||||
|         Else the counter is closed. |         Else the counter is closed. | ||||||
| @@ -535,13 +538,6 @@ class Counter(models.Model): | |||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return self.name |         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: |     def get_absolute_url(self) -> str: | ||||||
|         if self.type == "EBOUTIC": |         if self.type == "EBOUTIC": | ||||||
|             return reverse("eboutic:main") |             return reverse("eboutic:main") | ||||||
| @@ -690,8 +686,10 @@ class Counter(models.Model): | |||||||
|         Prices will be annotated |         Prices will be annotated | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         products = self.products.select_related("product_type").prefetch_related( |         products = ( | ||||||
|             "buying_groups" |             self.products.filter(archived=False) | ||||||
|  |             .select_related("product_type") | ||||||
|  |             .prefetch_related("buying_groups") | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         # Only include age appropriate products |         # Only include age appropriate products | ||||||
| @@ -1362,3 +1360,85 @@ class ReturnableProductBalance(models.Model): | |||||||
|             f"return balance of {self.customer} " |             f"return balance of {self.customer} " | ||||||
|             f"for {self.returnable.product_id} : {self.balance}" |             f"for {self.returnable.product_id} : {self.balance}" | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_product_actions(): | ||||||
|  |     return [ | ||||||
|  |         ("counter.tasks.archive_product", _("Archiving")), | ||||||
|  |         ("counter.tasks.change_counters", _("Counters change")), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ScheduledProductAction(PeriodicTask): | ||||||
|  |     """Extension of celery-beat tasks dedicated to perform actions on Product.""" | ||||||
|  |  | ||||||
|  |     product = models.ForeignKey( | ||||||
|  |         Product, related_name="scheduled_actions", on_delete=models.CASCADE | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         verbose_name = _("Product scheduled action") | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         self._meta.get_field("task").choices = get_product_actions() | ||||||
|  |         super().__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def full_clean(self, *args, **kwargs): | ||||||
|  |         self.one_off = True  # A product action should occur one time only | ||||||
|  |         return super().full_clean(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def clean_clocked(self): | ||||||
|  |         if not self.clocked: | ||||||
|  |             raise ValidationError(_("Product actions must declare a clocked schedule.")) | ||||||
|  |  | ||||||
|  |     def validate_unique(self, *args, **kwargs): | ||||||
|  |         # The checks done in PeriodicTask.validate_unique aren't | ||||||
|  |         # adapted in the case of scheduled product action, | ||||||
|  |         # so we skip it and execute directly Model.validate_unique | ||||||
|  |         return super(PeriodicTask, self).validate_unique(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MonthField(models.DateField): | ||||||
|  |     description = _("Year + month field (day forced to 1)") | ||||||
|  |     default_error_messages = { | ||||||
|  |         "invalid": _( | ||||||
|  |             "“%(value)s” value has an invalid date format. It must be " | ||||||
|  |             "in YYYY-MM format." | ||||||
|  |         ), | ||||||
|  |         "invalid_date": _( | ||||||
|  |             "“%(value)s” value has the correct format (YYYY-MM) " | ||||||
|  |             "but it is an invalid date." | ||||||
|  |         ), | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def to_python(self, value): | ||||||
|  |         if isinstance(value, str): | ||||||
|  |             with contextlib.suppress(ValueError): | ||||||
|  |                 # If the string is given as YYYY-mm, try to parse it. | ||||||
|  |                 # If it fails, it means that the string may be in the form YYYY-mm-dd | ||||||
|  |                 # or in an invalid format. | ||||||
|  |                 # Whatever the case, we let Django deal with it | ||||||
|  |                 # and raise an error if needed | ||||||
|  |                 value = datetime.strptime(value, "%Y-%m") | ||||||
|  |         value = super().to_python(value) | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |         return value.replace(day=1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class InvoiceCall(models.Model): | ||||||
|  |     is_validated = models.BooleanField(verbose_name=_("is validated"), default=False) | ||||||
|  |     club = models.ForeignKey(Club, on_delete=models.CASCADE) | ||||||
|  |     month = MonthField(verbose_name=_("invoice date")) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         verbose_name = _("Invoice call") | ||||||
|  |         verbose_name_plural = _("Invoice calls") | ||||||
|  |         constraints = [ | ||||||
|  |             models.UniqueConstraint( | ||||||
|  |                 fields=["club", "month"], name="counter_invoicecall_unique_club_month" | ||||||
|  |             ) | ||||||
|  |         ] | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return f"invoice call of {self.month} made by {self.club}" | ||||||
|   | |||||||
| @@ -39,6 +39,7 @@ | |||||||
|   flex: auto; |   flex: auto; | ||||||
|   margin: 0.2em; |   margin: 0.2em; | ||||||
|   width: 20%; |   width: 20%; | ||||||
|  |   min-width: 350px; | ||||||
|  |  | ||||||
|   ul { |   ul { | ||||||
|     list-style-type: none; |     list-style-type: none; | ||||||
|   | |||||||
							
								
								
									
										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="FIN">{% trans %}Confirm (FIN){% endtrans %}</option> | ||||||
|                 <option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option> |                 <option value="ANN">{% trans %}Cancel (ANN){% endtrans %}</option> | ||||||
|               </optgroup> |               </optgroup> | ||||||
|               {% for category in categories.keys() %} |               {%- for category in categories.keys() -%} | ||||||
|                 <optgroup label="{{ category }}"> |                 <optgroup label="{{ category }}"> | ||||||
|                   {% for product in categories[category] %} |                   {%- for product in categories[category] -%} | ||||||
|                     <option value="{{ product.id }}">{{ product }}</option> |                     <option value="{{ product.id }}">{{ product }}</option> | ||||||
|                   {% endfor %} |                   {%- endfor -%} | ||||||
|                 </optgroup> |                 </optgroup> | ||||||
|               {% endfor %} |               {%- endfor -%} | ||||||
|             </counter-product-select> |             </counter-product-select> | ||||||
|  |  | ||||||
|             <input type="submit" value="{% trans %}Go{% endtrans %}"/> |             <input type="submit" value="{% trans %}Go{% endtrans %}"/> | ||||||
|   | |||||||
| @@ -4,35 +4,49 @@ | |||||||
|   {% trans %}Invoices call{% endtrans %} |   {% trans %}Invoices call{% endtrans %} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
|  | {% block notifications %}{# Notifications are moved below #}{% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   <h3>{% trans date=start_date|date("F Y") %}Invoices call for {{ date }}{% endtrans %}</h3> |   <h3>{% trans date=start_date|date("F Y") %}Invoices call for {{ date }}{% endtrans %}</h3> | ||||||
|   <p>{% trans %}Choose another month: {% endtrans %}</p> |  | ||||||
|   <form method="get" action=""> |   <form method="get" action=""> | ||||||
|     <select name="month"> |     <label for="id_form_other_month">{% trans %}Choose another month: {% endtrans %}</label> | ||||||
|  |     <select name="month" id="id_form_other_month"> | ||||||
|       {% for m in months %} |       {% for m in months %} | ||||||
|         <option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option> |         <option value="{{ m|date("Y-m") }}">{{ m|date("Y-m") }}</option> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </select> |     </select> | ||||||
|     <input type="submit" value="{% trans %}Go{% endtrans %}" /> |     <input type="submit" value="{% trans %}Go{% endtrans %}" /> | ||||||
|   </form> |   </form> | ||||||
|  |  | ||||||
|   <br> |   <br> | ||||||
|   <p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p> |   <p>{% trans %}CB Payments{% endtrans %} : {{ sum_cb }} €</p> | ||||||
|   <br> |   <br> | ||||||
|   <table> |  | ||||||
|     <thead> |   {% include "core/base/notifications.jinja" %} | ||||||
|       <td>{% trans %}Club{% endtrans %}</td> |  | ||||||
|       <td>{% trans %}Sum{% endtrans %}</td> |   <form method="post" action=""> | ||||||
|     </thead> |     {% csrf_token %} | ||||||
|     <tbody> |     <table> | ||||||
|       {% for i in sums %} |       <thead> | ||||||
|         <tr> |         <tr> | ||||||
|           <td>{{ i['club__name'] }}</td> |           <td>{% trans %}Club{% endtrans %}</td> | ||||||
|           <td>{{ i['selling_sum'] }} €</td> |           <td>{% trans %}Sum{% endtrans %}</td> | ||||||
|  |           <td>{% trans %}Validated{% endtrans %}</td> | ||||||
|         </tr> |         </tr> | ||||||
|       {% endfor %} |       </thead> | ||||||
|     </tbody> |       <tbody> | ||||||
|   </table> |         {% for invoice in invoices %} | ||||||
| {% endblock %} |           <tr> | ||||||
|  |             <td>{{ invoice.club__name }}</td> | ||||||
|  |             <td>{{ "%.2f"|format(invoice.selling_sum) }} €</td> | ||||||
|  |             <td> | ||||||
|  |               {{ form[invoice.club_id|string] }} | ||||||
|  |             </td> | ||||||
|  |           </tr> | ||||||
|  |         {% endfor %} | ||||||
|  |       </tbody> | ||||||
|  |     </table> | ||||||
|  |     <input type="hidden" name="month" value="{{ start_date|date('Y-m') }}"> | ||||||
|  |     <button type="submit">{% trans %}Save{% endtrans %}</button> | ||||||
|  |   </form> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										56
									
								
								counter/templates/counter/product_form.jinja
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								counter/templates/counter/product_form.jinja
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | |||||||
|  | {% extends "core/base.jinja" %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |   {% if object %} | ||||||
|  |     <h2>{% trans name=object %}Edit product {{ name }}{% endtrans %}</h2> | ||||||
|  |   {% else %} | ||||||
|  |     <h2>{% trans %}Product creation{% endtrans %}</h2> | ||||||
|  |   {% endif %} | ||||||
|  |   <form method="post"> | ||||||
|  |     {% csrf_token %} | ||||||
|  |     {{ form.as_p() }} | ||||||
|  |  | ||||||
|  |     <br /> | ||||||
|  |  | ||||||
|  |     <h3>{% trans %}Automatic actions{% endtrans %}</h3> | ||||||
|  |  | ||||||
|  |     <p class="margin-bottom"> | ||||||
|  |       <em> | ||||||
|  |         {%- trans trimmed -%} | ||||||
|  |           Automatic actions allows to schedule product changes | ||||||
|  |           ahead of time. | ||||||
|  |         {%- endtrans -%} | ||||||
|  |       </em> | ||||||
|  |     </p> | ||||||
|  |  | ||||||
|  |     {{ form.action_formset.management_form }} | ||||||
|  |     {%- for action_form in form.action_formset.forms -%} | ||||||
|  |       <fieldset x-data="{action: '{{ action_form.task.initial }}'}"> | ||||||
|  |         {{ action_form.non_field_errors() }} | ||||||
|  |         <div class="row gap-2x margin-bottom"> | ||||||
|  |           <div> | ||||||
|  |             {{ action_form.task.errors }} | ||||||
|  |             {{ action_form.task.label_tag() }} | ||||||
|  |             {{ action_form.task|add_attr("x-model=action") }} | ||||||
|  |           </div> | ||||||
|  |           <div>{{ action_form.trigger_at.as_field_group() }}</div> | ||||||
|  |         </div> | ||||||
|  |         <div x-show="action==='counter.tasks.change_counters'" class="margin-bottom"> | ||||||
|  |           {{ action_form.counters.as_field_group() }} | ||||||
|  |         </div> | ||||||
|  |         {%- if action_form.DELETE -%} | ||||||
|  |           <div class="row gap"> | ||||||
|  |             {{ action_form.DELETE.as_field_group() }} | ||||||
|  |           </div> | ||||||
|  |         {%- endif -%} | ||||||
|  |         {%- for field in action_form.hidden_fields() -%} | ||||||
|  |           {{ field }} | ||||||
|  |         {%- endfor -%} | ||||||
|  |       </fieldset> | ||||||
|  |       {%- if not loop.last -%} | ||||||
|  |         <hr class="margin-bottom"> | ||||||
|  |       {%- endif -%} | ||||||
|  |     {%- endfor -%} | ||||||
|  |     <p><input type="submit" value="{% trans %}Save{% endtrans %}" /></p> | ||||||
|  |   </form> | ||||||
|  | {% endblock %} | ||||||
							
								
								
									
										116
									
								
								counter/tests/test_auto_actions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								counter/tests/test_auto_actions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,116 @@ | |||||||
|  | import json | ||||||
|  | from datetime import timedelta | ||||||
|  |  | ||||||
|  | import pytest | ||||||
|  | from django.conf import settings | ||||||
|  | from django.test import Client | ||||||
|  | from django.urls import reverse | ||||||
|  | from django.utils.timezone import now | ||||||
|  | from django_celery_beat.models import ClockedSchedule | ||||||
|  | from model_bakery import baker | ||||||
|  |  | ||||||
|  | from core.models import Group, User | ||||||
|  | from counter.baker_recipes import counter_recipe, product_recipe | ||||||
|  | from counter.forms import ScheduledProductActionForm, ScheduledProductActionFormSet | ||||||
|  | from counter.models import ScheduledProductAction | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | def test_edit_product(client: Client): | ||||||
|  |     client.force_login( | ||||||
|  |         baker.make( | ||||||
|  |             User, groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)] | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |     product = product_recipe.make() | ||||||
|  |     url = reverse("counter:product_edit", kwargs={"product_id": product.id}) | ||||||
|  |     res = client.get(url) | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |  | ||||||
|  |     res = client.post(url, data={}) | ||||||
|  |     # This is actually a failure, but we just want to check that | ||||||
|  |     # we don't have a 403 or a 500. | ||||||
|  |     # The actual behaviour will be tested directly on the form. | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestProductActionForm: | ||||||
|  |     def test_single_form_archive(self): | ||||||
|  |         product = product_recipe.make() | ||||||
|  |         trigger_at = now() + timedelta(minutes=10) | ||||||
|  |         form = ScheduledProductActionForm( | ||||||
|  |             product=product, | ||||||
|  |             data={ | ||||||
|  |                 "scheduled-task": "counter.tasks.archive_product", | ||||||
|  |                 "scheduled-trigger_at": trigger_at, | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |         assert form.is_valid() | ||||||
|  |         instance = form.save() | ||||||
|  |         assert instance.clocked.clocked_time == trigger_at | ||||||
|  |         assert instance.enabled is True | ||||||
|  |         assert instance.one_off is True | ||||||
|  |         assert instance.task == "counter.tasks.archive_product" | ||||||
|  |         assert instance.kwargs == json.dumps({"product_id": product.id}) | ||||||
|  |  | ||||||
|  |     def test_single_form_change_counters(self): | ||||||
|  |         product = product_recipe.make() | ||||||
|  |         counter = counter_recipe.make() | ||||||
|  |         trigger_at = now() + timedelta(minutes=10) | ||||||
|  |         form = ScheduledProductActionForm( | ||||||
|  |             product=product, | ||||||
|  |             data={ | ||||||
|  |                 "scheduled-task": "counter.tasks.change_counters", | ||||||
|  |                 "scheduled-trigger_at": trigger_at, | ||||||
|  |                 "scheduled-counters": [counter.id], | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |         assert form.is_valid() | ||||||
|  |         instance = form.save() | ||||||
|  |         instance.refresh_from_db() | ||||||
|  |         assert instance.clocked.clocked_time == trigger_at | ||||||
|  |         assert instance.enabled is True | ||||||
|  |         assert instance.one_off is True | ||||||
|  |         assert instance.task == "counter.tasks.change_counters" | ||||||
|  |         assert instance.kwargs == json.dumps( | ||||||
|  |             {"product_id": product.id, "counters": [counter.id]} | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_delete(self): | ||||||
|  |         product = product_recipe.make() | ||||||
|  |         clocked = baker.make(ClockedSchedule, clocked_time=now() + timedelta(minutes=2)) | ||||||
|  |         task = baker.make( | ||||||
|  |             ScheduledProductAction, | ||||||
|  |             product=product, | ||||||
|  |             one_off=True, | ||||||
|  |             clocked=clocked, | ||||||
|  |             task="counter.tasks.archive_product", | ||||||
|  |         ) | ||||||
|  |         formset = ScheduledProductActionFormSet(product=product) | ||||||
|  |         formset.delete_existing(task) | ||||||
|  |         assert not ScheduledProductAction.objects.filter(id=task.id).exists() | ||||||
|  |         assert not ClockedSchedule.objects.filter(id=clocked.id).exists() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.mark.django_db | ||||||
|  | class TestProductActionFormSet: | ||||||
|  |     def test_ok(self): | ||||||
|  |         product = product_recipe.make() | ||||||
|  |         counter = counter_recipe.make() | ||||||
|  |         trigger_at = now() + timedelta(minutes=10) | ||||||
|  |         formset = ScheduledProductActionFormSet( | ||||||
|  |             product=product, | ||||||
|  |             data={ | ||||||
|  |                 "form-TOTAL_FORMS": "2", | ||||||
|  |                 "form-INITIAL_FORMS": "0", | ||||||
|  |                 "form-0-task": "counter.tasks.archive_product", | ||||||
|  |                 "form-0-trigger_at": trigger_at, | ||||||
|  |                 "form-1-task": "counter.tasks.change_counters", | ||||||
|  |                 "form-1-trigger_at": trigger_at, | ||||||
|  |                 "form-1-counters": [counter.id], | ||||||
|  |             }, | ||||||
|  |         ) | ||||||
|  |         assert formset.is_valid() | ||||||
|  |         formset.save() | ||||||
|  |         assert ScheduledProductAction.objects.filter(product=product).count() == 2 | ||||||
| @@ -583,6 +583,16 @@ class TestCounterClick(TestFullClickBase): | |||||||
|             - self.beer.selling_price |             - 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): | class TestCounterStats(TestCase): | ||||||
|     @classmethod |     @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.conf import settings | ||||||
| from django.core.cache import cache | from django.core.cache import cache | ||||||
| from django.core.files.uploadedfile import SimpleUploadedFile | from django.core.files.uploadedfile import SimpleUploadedFile | ||||||
| from django.test import Client | from django.test import Client, TestCase | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| from model_bakery import baker | from model_bakery import baker | ||||||
| from PIL import Image | from PIL import Image | ||||||
| from pytest_django.asserts import assertNumQueries | from pytest_django.asserts import assertNumQueries, assertRedirects | ||||||
|  |  | ||||||
|  | from club.models import Club | ||||||
| from core.baker_recipes import board_user, subscriber_user | from core.baker_recipes import board_user, subscriber_user | ||||||
| from core.models import Group, User | from core.models import Group, User | ||||||
|  | from counter.forms import ProductForm | ||||||
| from counter.models import Product, ProductType | from counter.models import Product, ProductType | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -84,3 +86,49 @@ def test_fetch_product_nb_queries(client: Client): | |||||||
|         # - 1 for the actual request |         # - 1 for the actual request | ||||||
|         # - 1 to prefetch the related buying_groups |         # - 1 to prefetch the related buying_groups | ||||||
|         client.get(reverse("api:search_products_detailed")) |         client.get(reverse("api:search_products_detailed")) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestCreateProduct(TestCase): | ||||||
|  |     @classmethod | ||||||
|  |     def setUpTestData(cls): | ||||||
|  |         cls.product_type = baker.make(ProductType) | ||||||
|  |         cls.club = baker.make(Club) | ||||||
|  |         cls.data = { | ||||||
|  |             "name": "foo", | ||||||
|  |             "description": "bar", | ||||||
|  |             "product_type": cls.product_type.id, | ||||||
|  |             "club": cls.club.id, | ||||||
|  |             "code": "FOO", | ||||||
|  |             "purchase_price": 1.0, | ||||||
|  |             "selling_price": 1.0, | ||||||
|  |             "special_selling_price": 1.0, | ||||||
|  |             "limit_age": 0, | ||||||
|  |             "form-TOTAL_FORMS": 0, | ||||||
|  |             "form-INITIAL_FORMS": 0, | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def test_form(self): | ||||||
|  |         form = ProductForm(data=self.data) | ||||||
|  |         assert form.is_valid() | ||||||
|  |         instance = form.save() | ||||||
|  |         assert instance.club == self.club | ||||||
|  |         assert instance.product_type == self.product_type | ||||||
|  |         assert instance.name == "foo" | ||||||
|  |         assert instance.selling_price == 1.0 | ||||||
|  |  | ||||||
|  |     def test_view(self): | ||||||
|  |         self.client.force_login( | ||||||
|  |             baker.make( | ||||||
|  |                 User, | ||||||
|  |                 groups=[Group.objects.get(id=settings.SITH_GROUP_COUNTER_ADMIN_ID)], | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         url = reverse("counter:new_product") | ||||||
|  |         response = self.client.get(url) | ||||||
|  |         assert response.status_code == 200 | ||||||
|  |         response = self.client.post(url, data=self.data) | ||||||
|  |         assertRedirects(response, reverse("counter:product_list")) | ||||||
|  |         product = Product.objects.last() | ||||||
|  |         assert product.name == "foo" | ||||||
|  |         assert product.club == self.club | ||||||
|  |         assert product.product_type == self.product_type | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ from core.utils import get_semester_code, get_start_of_semester | |||||||
| from counter.forms import ( | from counter.forms import ( | ||||||
|     CloseCustomerAccountForm, |     CloseCustomerAccountForm, | ||||||
|     CounterEditForm, |     CounterEditForm, | ||||||
|     ProductEditForm, |     ProductForm, | ||||||
|     ReturnableProductForm, |     ReturnableProductForm, | ||||||
| ) | ) | ||||||
| from counter.models import ( | from counter.models import ( | ||||||
| @@ -146,8 +146,8 @@ class ProductCreateView(CounterAdminTabsMixin, CounterAdminMixin, CreateView): | |||||||
|     """A create view for the admins.""" |     """A create view for the admins.""" | ||||||
|  |  | ||||||
|     model = Product |     model = Product | ||||||
|     form_class = ProductEditForm |     form_class = ProductForm | ||||||
|     template_name = "core/create.jinja" |     template_name = "counter/product_form.jinja" | ||||||
|     current_tab = "products" |     current_tab = "products" | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -155,9 +155,9 @@ class ProductEditView(CounterAdminTabsMixin, CounterAdminMixin, UpdateView): | |||||||
|     """An edit view for the admins.""" |     """An edit view for the admins.""" | ||||||
|  |  | ||||||
|     model = Product |     model = Product | ||||||
|     form_class = ProductEditForm |     form_class = ProductForm | ||||||
|     pk_url_kwarg = "product_id" |     pk_url_kwarg = "product_id" | ||||||
|     template_name = "core/edit.jinja" |     template_name = "counter/product_form.jinja" | ||||||
|     current_tab = "products" |     current_tab = "products" | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,77 +12,81 @@ | |||||||
| # OR WITHIN THE LOCAL FILE "LICENSE" | # OR WITHIN THE LOCAL FILE "LICENSE" | ||||||
| # | # | ||||||
| # | # | ||||||
| from datetime import datetime, timedelta | from datetime import datetime | ||||||
| from datetime import timezone as tz | from urllib.parse import urlencode | ||||||
|  |  | ||||||
| from django.db.models import F | from dateutil.relativedelta import relativedelta | ||||||
| from django.utils import timezone | from django.contrib.auth.mixins import PermissionRequiredMixin | ||||||
| from django.views.generic import TemplateView | from django.contrib.messages.views import SuccessMessageMixin | ||||||
|  | from django.db.models import F, Sum | ||||||
|  | from django.utils.timezone import localdate, make_aware | ||||||
|  | from django.utils.translation import gettext_lazy as _ | ||||||
|  | from django.views.generic import FormView | ||||||
|  |  | ||||||
| from counter.fields import CurrencyField | from counter.forms import InvoiceCallForm | ||||||
| from counter.models import Refilling, Selling | from counter.models import Refilling, Selling | ||||||
| from counter.views.mixins import CounterAdminMixin, CounterAdminTabsMixin | from counter.views.mixins import CounterAdminTabsMixin | ||||||
|  |  | ||||||
|  |  | ||||||
| class InvoiceCallView(CounterAdminTabsMixin, CounterAdminMixin, TemplateView): | class InvoiceCallView( | ||||||
|  |     CounterAdminTabsMixin, PermissionRequiredMixin, SuccessMessageMixin, FormView | ||||||
|  | ): | ||||||
|     template_name = "counter/invoices_call.jinja" |     template_name = "counter/invoices_call.jinja" | ||||||
|     current_tab = "invoices_call" |     current_tab = "invoices_call" | ||||||
|  |     permission_required = ["counter.view_invoicecall", "counter.change_invoicecall"] | ||||||
|  |     form_class = InvoiceCallForm | ||||||
|  |     success_message = _("Invoice calls status has been updated.") | ||||||
|  |  | ||||||
|  |     def get_month(self): | ||||||
|  |         kwargs = self.request.GET or self.request.POST | ||||||
|  |         if "month" in kwargs: | ||||||
|  |             return make_aware(datetime.strptime(kwargs["month"], "%Y-%m")) | ||||||
|  |         return localdate().replace(day=1) - relativedelta(months=1) | ||||||
|  |  | ||||||
|  |     def get_form_kwargs(self): | ||||||
|  |         return super().get_form_kwargs() | {"month": self.get_month()} | ||||||
|  |  | ||||||
|  |     def form_valid(self, form): | ||||||
|  |         form.save() | ||||||
|  |         return super().form_valid(form) | ||||||
|  |  | ||||||
|  |     def get_success_url(self): | ||||||
|  |         # redirect to the month from which the request is originated | ||||||
|  |         url = self.request.path | ||||||
|  |         kwargs = self.request.GET or self.request.POST | ||||||
|  |         if "month" in kwargs: | ||||||
|  |             query = urlencode({"month": kwargs["month"]}) | ||||||
|  |             url += f"?{query}" | ||||||
|  |         return url | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         """Add sums to the context.""" |         """Add sums to the context.""" | ||||||
|         kwargs = super().get_context_data(**kwargs) |         kwargs = super().get_context_data(**kwargs) | ||||||
|         kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC") |         kwargs["months"] = Selling.objects.datetimes("date", "month", order="DESC") | ||||||
|         if "month" in self.request.GET: |         start_date = self.get_month() | ||||||
|             start_date = datetime.strptime(self.request.GET["month"], "%Y-%m") |         end_date = start_date + relativedelta(months=1) | ||||||
|         else: |  | ||||||
|             start_date = datetime( |  | ||||||
|                 year=timezone.now().year, |  | ||||||
|                 month=(timezone.now().month + 10) % 12 + 1, |  | ||||||
|                 day=1, |  | ||||||
|             ) |  | ||||||
|         start_date = start_date.replace(tzinfo=tz.utc) |  | ||||||
|         end_date = (start_date + timedelta(days=32)).replace( |  | ||||||
|             day=1, hour=0, minute=0, microsecond=0 |  | ||||||
|         ) |  | ||||||
|         from django.db.models import Case, Sum, When |  | ||||||
|  |  | ||||||
|         kwargs["sum_cb"] = sum( |         kwargs["sum_cb"] = Refilling.objects.filter( | ||||||
|             [ |             payment_method="CARD", | ||||||
|                 r.amount |             is_validated=True, | ||||||
|                 for r in Refilling.objects.filter( |             date__gte=start_date, | ||||||
|                     payment_method="CARD", |             date__lte=end_date, | ||||||
|                     is_validated=True, |         ).aggregate(res=Sum("amount", default=0))["res"] | ||||||
|                     date__gte=start_date, |         kwargs["sum_cb"] += ( | ||||||
|                     date__lte=end_date, |             Selling.objects.filter( | ||||||
|                 ) |                 payment_method="CARD", | ||||||
|             ] |                 is_validated=True, | ||||||
|         ) |                 date__gte=start_date, | ||||||
|         kwargs["sum_cb"] += sum( |                 date__lte=end_date, | ||||||
|             [ |             ) | ||||||
|                 s.quantity * s.unit_price |             .annotate(amount=F("unit_price") * F("quantity")) | ||||||
|                 for s in Selling.objects.filter( |             .aggregate(res=Sum("amount", default=0))["res"] | ||||||
|                     payment_method="CARD", |  | ||||||
|                     is_validated=True, |  | ||||||
|                     date__gte=start_date, |  | ||||||
|                     date__lte=end_date, |  | ||||||
|                 ) |  | ||||||
|             ] |  | ||||||
|         ) |         ) | ||||||
|         kwargs["start_date"] = start_date |         kwargs["start_date"] = start_date | ||||||
|         kwargs["sums"] = ( |         kwargs["invoices"] = ( | ||||||
|             Selling.objects.values("club__name") |             Selling.objects.filter(date__gte=start_date, date__lt=end_date) | ||||||
|             .annotate( |             .values("club_id", "club__name") | ||||||
|                 selling_sum=Sum( |             .annotate(selling_sum=Sum(F("unit_price") * F("quantity"))) | ||||||
|                     Case( |  | ||||||
|                         When( |  | ||||||
|                             date__gte=start_date, |  | ||||||
|                             date__lt=end_date, |  | ||||||
|                             then=F("unit_price") * F("quantity"), |  | ||||||
|                         ), |  | ||||||
|                         output_field=CurrencyField(), |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|             .exclude(selling_sum=None) |             .exclude(selling_sum=None) | ||||||
|             .order_by("-selling_sum") |             .order_by("-selling_sum") | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ | |||||||
|         heading_level: 3 |         heading_level: 3 | ||||||
|         members: |         members: | ||||||
|             - TabedViewMixin |             - TabedViewMixin | ||||||
|             - QuickNotifMixin |  | ||||||
|             - AllowFragment |             - AllowFragment | ||||||
|             - FragmentMixin |             - FragmentMixin | ||||||
|             - UseFragmentsMixin |             - UseFragmentsMixin | ||||||
| @@ -17,7 +17,6 @@ document.addEventListener("alpine:init", () => { | |||||||
|       this.$watch("basket", () => { |       this.$watch("basket", () => { | ||||||
|         this.saveBasket(); |         this.saveBasket(); | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       // Invalidate basket if a purchase was made |       // Invalidate basket if a purchase was made | ||||||
|       if (lastPurchaseTime !== null && localStorage.basketTimestamp !== undefined) { |       if (lastPurchaseTime !== null && localStorage.basketTimestamp !== undefined) { | ||||||
|         if ( |         if ( | ||||||
|   | |||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | {% from 'core/macros.jinja' import update_notifications %} | ||||||
|  |  | ||||||
| <div id=billing-infos-fragment> | <div id=billing-infos-fragment> | ||||||
|   <div |   <div | ||||||
|     class="collapse" |     class="collapse" | ||||||
| @@ -29,14 +31,6 @@ | |||||||
|       > |       > | ||||||
|     </form> |     </form> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|   <br> |   <br> | ||||||
|  |   {{ update_notifications(messages) }} | ||||||
|   {% if messages %} |  | ||||||
|     {% for message in messages %} |  | ||||||
|       <div class="alert alert-{{ message.tags }}"> |  | ||||||
|         {{ message }} |  | ||||||
|       </div> |  | ||||||
|     {% endfor %} |  | ||||||
|   {% endif %} |  | ||||||
| </div> | </div> | ||||||
|   | |||||||
| @@ -1,5 +1,9 @@ | |||||||
| {% extends "core/base.jinja" %} | {% extends "core/base.jinja" %} | ||||||
|  |  | ||||||
|  | {% block notifications %} | ||||||
|  |   {# Notifications are moved under the billing form #} | ||||||
|  | {% endblock %} | ||||||
|  |  | ||||||
| {% block title %} | {% block title %} | ||||||
|   {% trans %}Basket state{% endtrans %} |   {% trans %}Basket state{% endtrans %} | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -56,6 +60,7 @@ | |||||||
|       <div @htmx:after-request="fill"> |       <div @htmx:after-request="fill"> | ||||||
|         {{ billing_infos_form }} |         {{ billing_infos_form }} | ||||||
|       </div> |       </div> | ||||||
|  |       {% include "core/base/notifications.jinja" %} | ||||||
|       <form |       <form | ||||||
|         method="post" |         method="post" | ||||||
|         action="{{ settings.SITH_EBOUTIC_ET_URL }}" |         action="{{ settings.SITH_EBOUTIC_ET_URL }}" | ||||||
|   | |||||||
| @@ -22,14 +22,6 @@ | |||||||
| {% block content %} | {% block content %} | ||||||
|   <h1 id="eboutic-title">{% trans %}Eboutic{% endtrans %}</h1> |   <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="eboutic" x-data="basket({{ last_purchase_time }})"> | ||||||
|     <div id="basket"> |     <div id="basket"> | ||||||
|       <h3>Panier</h3> |       <h3>Panier</h3> | ||||||
|   | |||||||
| @@ -4,14 +4,6 @@ | |||||||
|   <h3>{% trans %}Eboutic{% endtrans %}</h3> |   <h3>{% trans %}Eboutic{% endtrans %}</h3> | ||||||
|  |  | ||||||
|   <div> |   <div> | ||||||
|     {% if messages %} |  | ||||||
|       {% for message in messages %} |  | ||||||
|         <div class="alert alert-{{ message.tags }}"> |  | ||||||
|           {{ message }} |  | ||||||
|         </div> |  | ||||||
|       {% endfor %} |  | ||||||
|     {% endif %} |  | ||||||
|  |  | ||||||
|     {% if success %} |     {% if success %} | ||||||
|       {% trans %}Payment successful{% endtrans %} |       {% trans %}Payment successful{% endtrans %} | ||||||
|     {% else %} |     {% else %} | ||||||
|   | |||||||
| @@ -48,7 +48,7 @@ from django_countries.fields import Country | |||||||
|  |  | ||||||
| from core.auth.mixins import CanViewMixin | from core.auth.mixins import CanViewMixin | ||||||
| from core.views.mixins import FragmentMixin, UseFragmentsMixin | from core.views.mixins import FragmentMixin, UseFragmentsMixin | ||||||
| from counter.forms import BaseBasketForm, BillingInfoForm, ProductForm | from counter.forms import BaseBasketForm, BasketProductForm, BillingInfoForm | ||||||
| from counter.models import ( | from counter.models import ( | ||||||
|     BillingInfo, |     BillingInfo, | ||||||
|     Customer, |     Customer, | ||||||
| @@ -78,7 +78,7 @@ class BaseEbouticBasketForm(BaseBasketForm): | |||||||
|  |  | ||||||
|  |  | ||||||
| EbouticBasketForm = forms.formset_factory( | EbouticBasketForm = forms.formset_factory( | ||||||
|     ProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 |     BasketProductForm, formset=BaseEbouticBasketForm, absolute_max=None, min_num=1 | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
| msgid "" | msgid "" | ||||||
| msgstr "" | msgstr "" | ||||||
| "Report-Msgid-Bugs-To: \n" | "Report-Msgid-Bugs-To: \n" | ||||||
| "POT-Creation-Date: 2025-09-01 18:18+0200\n" | "POT-Creation-Date: 2025-10-17 13:41+0200\n" | ||||||
| "PO-Revision-Date: 2016-07-18\n" | "PO-Revision-Date: 2016-07-18\n" | ||||||
| "Last-Translator: Maréchal <thomas.girod@utbm.fr\n" | "Last-Translator: Maréchal <thomas.girod@utbm.fr\n" | ||||||
| "Language-Team: AE info <ae.info@utbm.fr>\n" | "Language-Team: AE info <ae.info@utbm.fr>\n" | ||||||
| @@ -117,7 +117,7 @@ msgstr "S'abonner" | |||||||
| msgid "Remove" | msgid "Remove" | ||||||
| msgstr "Retirer" | msgstr "Retirer" | ||||||
|  |  | ||||||
| #: club/forms.py pedagogy/templates/pedagogy/moderation.jinja | #: club/forms.py counter/forms.py pedagogy/templates/pedagogy/moderation.jinja | ||||||
| msgid "Action" | msgid "Action" | ||||||
| msgstr "Action" | msgstr "Action" | ||||||
|  |  | ||||||
| @@ -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" | msgstr "Vous ne pouvez pas ajouter deux fois le même utilisateur" | ||||||
|  |  | ||||||
| #: club/forms.py | #: club/forms.py | ||||||
| msgid "You should specify a role" | msgid "You must be subscribed to join a club" | ||||||
| msgstr "Vous devez choisir un rôle" | msgstr "Vous devez être cotisant pour faire partie d'un club" | ||||||
|  |  | ||||||
| #: club/forms.py sas/forms.py | #: club/forms.py | ||||||
| msgid "You do not have the permission to do that" | msgid "You are already a member of this club" | ||||||
| msgstr "Vous n'avez pas la permission de faire cela" | msgstr "Vous êtes déjà membre de ce club." | ||||||
|  |  | ||||||
| #: club/models.py | #: club/models.py | ||||||
| msgid "slug name" | msgid "slug name" | ||||||
| @@ -350,11 +350,6 @@ msgstr "Depuis" | |||||||
| msgid "There are no members in this club." | msgid "There are no members in this club." | ||||||
| msgstr "Il n'y a pas de membres dans ce 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 | #: club/templates/club/club_old_members.jinja | ||||||
| msgid "Club old members" | msgid "Club old members" | ||||||
| msgstr "Anciens membres du club" | msgstr "Anciens membres du club" | ||||||
| @@ -514,8 +509,8 @@ msgstr "Éditer le Trombi" | |||||||
| msgid "New Trombi" | msgid "New Trombi" | ||||||
| msgstr "Nouveau Trombi" | msgstr "Nouveau Trombi" | ||||||
|  |  | ||||||
| #: club/templates/club/club_tools.jinja com/templates/com/poster_list.jinja | #: club/templates/club/club_tools.jinja club/views.py | ||||||
| #: core/templates/core/user_tools.jinja | #: com/templates/com/poster_list.jinja core/templates/core/user_tools.jinja | ||||||
| msgid "Posters" | msgid "Posters" | ||||||
| msgstr "Affiches" | msgstr "Affiches" | ||||||
|  |  | ||||||
| @@ -561,6 +556,8 @@ msgstr "" | |||||||
| #: core/templates/core/user_godfathers_tree.jinja | #: core/templates/core/user_godfathers_tree.jinja | ||||||
| #: core/templates/core/user_preferences.jinja | #: core/templates/core/user_preferences.jinja | ||||||
| #: counter/templates/counter/cash_register_summary.jinja | #: counter/templates/counter/cash_register_summary.jinja | ||||||
|  | #: counter/templates/counter/invoices_call.jinja | ||||||
|  | #: counter/templates/counter/product_form.jinja | ||||||
| #: forum/templates/forum/reply.jinja | #: forum/templates/forum/reply.jinja | ||||||
| #: subscription/templates/subscription/fragments/creation_form.jinja | #: subscription/templates/subscription/fragments/creation_form.jinja | ||||||
| #: trombi/templates/trombi/comment.jinja | #: trombi/templates/trombi/comment.jinja | ||||||
| @@ -569,6 +566,24 @@ msgstr "" | |||||||
| msgid "Save" | msgid "Save" | ||||||
| msgstr "Sauver" | 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 | #: club/templates/club/mailing.jinja | ||||||
| msgid "Mailing lists" | msgid "Mailing lists" | ||||||
| msgstr "Mailing listes" | msgstr "Mailing listes" | ||||||
| @@ -675,9 +690,14 @@ msgstr "Vente" | |||||||
| msgid "Mailing list" | msgid "Mailing list" | ||||||
| msgstr "Listes de diffusion" | msgstr "Listes de diffusion" | ||||||
|  |  | ||||||
| #: club/views.py com/views.py | #: club/views.py | ||||||
| msgid "Posters list" | msgid "You are now a member of this club." | ||||||
| msgstr "Liste d'affiches" | 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 | #: com/forms.py | ||||||
| msgid "Format: 16:9 | Resolution: 1920x1080" | msgid "Format: 16:9 | Resolution: 1920x1080" | ||||||
| @@ -1043,6 +1063,10 @@ msgstr "Nos services" | |||||||
| msgid "UV Guide" | msgid "UV Guide" | ||||||
| msgstr "Guide des UVs" | msgstr "Guide des UVs" | ||||||
|  |  | ||||||
|  | #: com/templates/com/news_list.jinja | ||||||
|  | msgid "Timetable" | ||||||
|  | msgstr "Emploi du temps" | ||||||
|  |  | ||||||
| #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | #: com/templates/com/news_list.jinja core/templates/core/base/navbar.jinja | ||||||
| msgid "Matmatronch" | msgid "Matmatronch" | ||||||
| msgstr "Matmatronch" | msgstr "Matmatronch" | ||||||
| @@ -1107,6 +1131,10 @@ msgstr "Modération" | |||||||
| msgid "No posters" | msgid "No posters" | ||||||
| msgstr "Aucune affiche" | 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 | #: com/templates/com/poster_moderate.jinja | ||||||
| msgid "Posters - moderation" | msgid "Posters - moderation" | ||||||
| msgstr "Affiches - modération" | msgstr "Affiches - modération" | ||||||
| @@ -1164,14 +1192,6 @@ msgstr "Contenu" | |||||||
| msgid "Add to weekmail" | msgid "Add to weekmail" | ||||||
| msgstr "Ajouter au 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 | #: com/templates/com/weekmail.jinja | ||||||
| msgid "Articles included the next weekmail" | msgid "Articles included the next weekmail" | ||||||
| msgstr "Article inclus dans le prochain Weekmail" | msgstr "Article inclus dans le prochain Weekmail" | ||||||
| @@ -1180,6 +1200,14 @@ msgstr "Article inclus dans le prochain Weekmail" | |||||||
| msgid "Delete from weekmail" | msgid "Delete from weekmail" | ||||||
| msgstr "Supprimer du 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 | #: com/templates/com/weekmail_preview.jinja | ||||||
| #: core/templates/core/user_account_detail.jinja | #: core/templates/core/user_account_detail.jinja | ||||||
| #: pedagogy/templates/pedagogy/uv_detail.jinja | #: pedagogy/templates/pedagogy/uv_detail.jinja | ||||||
| @@ -1249,6 +1277,10 @@ msgstr "Message d'info" | |||||||
| msgid "Alert message" | msgid "Alert message" | ||||||
| msgstr "Message d'alerte" | msgstr "Message d'alerte" | ||||||
|  |  | ||||||
|  | #: com/views.py | ||||||
|  | msgid "Posters list" | ||||||
|  | msgstr "Liste d'affiches" | ||||||
|  |  | ||||||
| #: com/views.py | #: com/views.py | ||||||
| msgid "Screens list" | msgid "Screens list" | ||||||
| msgstr "Liste d'écrans" | msgstr "Liste d'écrans" | ||||||
| @@ -1257,6 +1289,10 @@ msgstr "Liste d'écrans" | |||||||
| msgid "All incoming events" | msgid "All incoming events" | ||||||
| msgstr "Tous les événements à venir" | msgstr "Tous les événements à venir" | ||||||
|  |  | ||||||
|  | #: com/views.py | ||||||
|  | msgid "Weekmail sent successfully" | ||||||
|  | msgstr "Weekmail envoyé avec succès" | ||||||
|  |  | ||||||
| #: com/views.py | #: com/views.py | ||||||
| msgid "Delete and save to regenerate" | msgid "Delete and save to regenerate" | ||||||
| msgstr "Supprimer et sauver pour régénérer" | 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 " | msgid "Weekmail of the " | ||||||
| msgstr "Weekmail du " | 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 | #: com/views.py | ||||||
| msgid "" | msgid "" | ||||||
| "You must be a board member of the selected club to post in the Weekmail." | "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 " | "Vous devez êtres un membre du bureau du club sélectionné pour poster dans le " | ||||||
| "Weekmail." | "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 | #: core/models.py | ||||||
| msgid "Is manually manageable" | msgid "Is manually manageable" | ||||||
| msgstr "Est gérable manuellement" | msgstr "Est gérable manuellement" | ||||||
| @@ -1713,8 +1774,8 @@ msgid "" | |||||||
| "AE UTBM is a voluntary organisation run by UTBM students. It organises " | "AE UTBM is a voluntary organisation run by UTBM students. It organises " | ||||||
| "student life at UTBM and manages its student facilities." | "student life at UTBM and manages its student facilities." | ||||||
| msgstr "" | msgstr "" | ||||||
| "L'AE UTBM est une association bénévole gérée par les étudiants de " | "L'AE UTBM est une association bénévole gérée par les étudiants de l'UTBM. " | ||||||
| "l'UTBM. Elle organise la vie étudiante de l'UTBM et gère ses lieux de vie." | "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 | #: core/templates/core/base/footer.jinja core/templates/core/base/navbar.jinja | ||||||
| msgid "Contacts" | msgid "Contacts" | ||||||
| @@ -2157,10 +2218,6 @@ msgstr "" | |||||||
| msgid "Page history" | msgid "Page history" | ||||||
| msgstr "Historique de la page" | 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 | #: core/templates/core/page_prop.jinja | ||||||
| msgid "Page properties" | msgid "Page properties" | ||||||
| msgstr "Propriétés de la page" | msgstr "Propriétés de la page" | ||||||
| @@ -2339,6 +2396,10 @@ msgstr "Etickets" | |||||||
| msgid "User has no account" | msgid "User has no account" | ||||||
| msgstr "L'utilisateur n'a pas de compte" | 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 | #: core/templates/core/user_account_detail.jinja | ||||||
| #: counter/templates/counter/last_ops.jinja | #: counter/templates/counter/last_ops.jinja | ||||||
| #: counter/templates/counter/refilling_list.jinja | #: counter/templates/counter/refilling_list.jinja | ||||||
| @@ -2896,6 +2957,18 @@ msgstr "Cet UID est invalide" | |||||||
| msgid "User not found" | msgid "User not found" | ||||||
| msgstr "Utilisateur non trouvé" | msgstr "Utilisateur non trouvé" | ||||||
|  |  | ||||||
|  | #: counter/forms.py | ||||||
|  | msgid "Date and time of action" | ||||||
|  | msgstr "Date et heure de l'action" | ||||||
|  |  | ||||||
|  | #: counter/forms.py | ||||||
|  | msgid "New counters" | ||||||
|  | msgstr "Nouveaux comptoirs" | ||||||
|  |  | ||||||
|  | #: counter/forms.py | ||||||
|  | msgid "The selected counters will replace the current ones" | ||||||
|  | msgstr "Les comptoirs sélectionnés remplaceront les comptoirs actuels" | ||||||
|  |  | ||||||
| #: counter/forms.py | #: counter/forms.py | ||||||
| msgid "" | msgid "" | ||||||
| "Describe the product. If it's an event's click, give some insights about it, " | "Describe the product. If it's an event's click, give some insights about it, " | ||||||
| @@ -3230,6 +3303,52 @@ msgid "The returnable product cannot be the same as the returned one" | |||||||
| msgstr "" | msgstr "" | ||||||
| "Le produit consigné ne peut pas être le même que le produit de déconsigne" | "Le produit consigné ne peut pas être le même que le produit de déconsigne" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Archiving" | ||||||
|  | msgstr "Archivage" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Counters change" | ||||||
|  | msgstr "Changement des comptoirs" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Product scheduled action" | ||||||
|  | msgstr "Actions sur produit planifiées" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Product actions must declare a clocked schedule." | ||||||
|  | msgstr "Les actions sur les produits doivent avoir un horaire planifié." | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Year + month field (day forced to 1)" | ||||||
|  | msgstr "Champ Année + mois (jour forcé à 1)" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | #, python-format | ||||||
|  | msgid "" | ||||||
|  | "“%(value)s” value has an invalid date format. It must be in YYYY-MM format." | ||||||
|  | msgstr "" | ||||||
|  | "La valeur « %(value)s » a un format de date invalide. Ce doit être au format " | ||||||
|  | "YYYY-MM." | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | #, python-format | ||||||
|  | msgid "" | ||||||
|  | "“%(value)s” value has the correct format (YYYY-MM) but it is an invalid date." | ||||||
|  | msgstr "La valeur « %(value)s » a le bon format, mais est une date invalide." | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "invoice date" | ||||||
|  | msgstr "date de la facture" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Invoice call" | ||||||
|  | msgstr "Appel à facture" | ||||||
|  |  | ||||||
|  | #: counter/models.py | ||||||
|  | msgid "Invoice calls" | ||||||
|  | msgstr "Appels à facture" | ||||||
|  |  | ||||||
| #: counter/templates/counter/activity.jinja | #: counter/templates/counter/activity.jinja | ||||||
| #, python-format | #, python-format | ||||||
| msgid "%(counter_name)s activity" | msgid "%(counter_name)s activity" | ||||||
| @@ -3460,6 +3579,10 @@ msgstr "Payements en Carte Bancaire" | |||||||
| msgid "Sum" | msgid "Sum" | ||||||
| msgstr "Somme" | msgstr "Somme" | ||||||
|  |  | ||||||
|  | #: counter/templates/counter/invoices_call.jinja | ||||||
|  | msgid "Validated" | ||||||
|  | msgstr "Validé" | ||||||
|  |  | ||||||
| #: counter/templates/counter/last_ops.jinja | #: counter/templates/counter/last_ops.jinja | ||||||
| #, python-format | #, python-format | ||||||
| msgid "%(counter_name)s last operations" | msgid "%(counter_name)s last operations" | ||||||
| @@ -3548,6 +3671,25 @@ msgstr "" | |||||||
| "votre cotisation. Si vous ne renouvelez pas votre cotisation, il n'y aura " | "votre cotisation. Si vous ne renouvelez pas votre cotisation, il n'y aura " | ||||||
| "aucune conséquence autre que le retrait de l'argent de votre compte." | "aucune conséquence autre que le retrait de l'argent de votre compte." | ||||||
|  |  | ||||||
|  | #: counter/templates/counter/product_form.jinja | ||||||
|  | #, python-format | ||||||
|  | msgid "Edit product %(name)s" | ||||||
|  | msgstr "Édition du produit %(name)s" | ||||||
|  |  | ||||||
|  | #: counter/templates/counter/product_form.jinja | ||||||
|  | msgid "Product creation" | ||||||
|  | msgstr "Création de produit" | ||||||
|  |  | ||||||
|  | #: counter/templates/counter/product_form.jinja | ||||||
|  | msgid "Automatic actions" | ||||||
|  | msgstr "Actions automatiques" | ||||||
|  |  | ||||||
|  | #: counter/templates/counter/product_form.jinja | ||||||
|  | msgid "Automatic actions allows to schedule product changes ahead of time." | ||||||
|  | msgstr "" | ||||||
|  | "Les actions automatiques vous permettent de planifier des modifications du " | ||||||
|  | "produit à l'avance." | ||||||
|  |  | ||||||
| #: counter/templates/counter/product_list.jinja | #: counter/templates/counter/product_list.jinja | ||||||
| msgid "Product list" | msgid "Product list" | ||||||
| msgstr "Liste des produits" | msgstr "Liste des produits" | ||||||
| @@ -3730,6 +3872,10 @@ msgstr "L'utilisateur n'est pas barman." | |||||||
| msgid "Bad location, someone is already logged in somewhere else" | msgid "Bad location, someone is already logged in somewhere else" | ||||||
| msgstr "Mauvais comptoir, quelqu'un est déjà connecté ailleurs" | msgstr "Mauvais comptoir, quelqu'un est déjà connecté ailleurs" | ||||||
|  |  | ||||||
|  | #: counter/views/invoice.py | ||||||
|  | msgid "Invoice calls status has been updated." | ||||||
|  | msgstr "Le statut des appels à facture a été mis à jour." | ||||||
|  |  | ||||||
| #: counter/views/mixins.py | #: counter/views/mixins.py | ||||||
| msgid "Cash summary" | msgid "Cash summary" | ||||||
| msgstr "Relevé de caisse" | msgstr "Relevé de caisse" | ||||||
| @@ -4539,22 +4685,6 @@ msgstr "Signaler ce commentaire" | |||||||
| msgid "Edit UE" | msgid "Edit UE" | ||||||
| msgstr "Éditer l'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 | #: rootplace/forms.py | ||||||
| msgid "User that will be kept" | msgid "User that will be kept" | ||||||
| msgstr "Utilisateur qui sera conservé" | msgstr "Utilisateur qui sera conservé" | ||||||
| @@ -4628,6 +4758,10 @@ msgstr "Pas de ban actif" | |||||||
| msgid "Add a new album" | msgid "Add a new album" | ||||||
| msgstr "Ajouter un nouvel 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 | #: sas/forms.py | ||||||
| msgid "Upload images" | msgid "Upload images" | ||||||
| msgstr "Envoyer les images" | msgstr "Envoyer les images" | ||||||
| @@ -4818,8 +4952,8 @@ msgid "N/A" | |||||||
| msgstr "N/A" | msgstr "N/A" | ||||||
|  |  | ||||||
| #: sith/settings.py | #: sith/settings.py | ||||||
| msgid "Transfert" | msgid "AE account" | ||||||
| msgstr "Virement" | msgstr "Compte AE" | ||||||
|  |  | ||||||
| #: sith/settings.py | #: sith/settings.py | ||||||
| msgid "Belfort" | msgid "Belfort" | ||||||
| @@ -5107,26 +5241,6 @@ msgstr "Vous avez acheté %s" | |||||||
| msgid "You have a notification" | msgid "You have a notification" | ||||||
| msgstr "Vous avez une 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 | #: sith/settings.py | ||||||
| msgid "AE tee-shirt" | msgid "AE tee-shirt" | ||||||
| msgstr "Tee-shirt AE" | msgstr "Tee-shirt AE" | ||||||
| @@ -5135,6 +5249,10 @@ msgstr "Tee-shirt AE" | |||||||
| msgid "A user with that email address already exists" | msgid "A user with that email address already exists" | ||||||
| msgstr "Un utilisateur avec cette adresse email existe déjà" | 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 | #: subscription/models.py | ||||||
| msgid "Bad subscription type" | msgid "Bad subscription type" | ||||||
| msgstr "Mauvais type de cotisation" | msgstr "Mauvais type de cotisation" | ||||||
| @@ -5163,6 +5281,14 @@ msgstr "lieu" | |||||||
| msgid "You can not subscribe many time for the same period" | 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" | 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 | #: subscription/templates/subscription/fragments/creation_success.jinja | ||||||
| #, python-format | #, python-format | ||||||
| msgid "Subscription created for %(user)s" | msgid "Subscription created for %(user)s" | ||||||
| @@ -5174,7 +5300,7 @@ msgid "" | |||||||
| "%(user)s received its new %(type)s subscription. It will be active until " | "%(user)s received its new %(type)s subscription. It will be active until " | ||||||
| "%(end)s included." | "%(end)s included." | ||||||
| msgstr "" | 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." | "%(end)s inclu." | ||||||
|  |  | ||||||
| #: subscription/templates/subscription/fragments/creation_success.jinja | #: subscription/templates/subscription/fragments/creation_success.jinja | ||||||
| @@ -5201,6 +5327,18 @@ msgstr "Membre existant" | |||||||
| msgid "the groups that can create subscriptions" | msgid "the groups that can create subscriptions" | ||||||
| msgstr "les groupes pouvant créer des cotisations" | msgstr "les groupes pouvant créer des cotisations" | ||||||
|  |  | ||||||
|  | #: timetable/templates/timetable/generator.jinja | ||||||
|  | msgid "Timetable generator" | ||||||
|  | msgstr "Générateur d'emploi du temps" | ||||||
|  |  | ||||||
|  | #: timetable/templates/timetable/generator.jinja | ||||||
|  | msgid "Generate" | ||||||
|  | msgstr "Générer" | ||||||
|  |  | ||||||
|  | #: timetable/templates/timetable/generator.jinja | ||||||
|  | msgid "Save to PNG" | ||||||
|  | msgstr "Sauver en PNG" | ||||||
|  |  | ||||||
| #: trombi/models.py | #: trombi/models.py | ||||||
| msgid "subscription deadline" | msgid "subscription deadline" | ||||||
| msgstr "fin des inscriptions" | msgstr "fin des inscriptions" | ||||||
| @@ -5426,10 +5564,38 @@ msgstr "Mes photos" | |||||||
| msgid "Admin tools" | msgid "Admin tools" | ||||||
| msgstr "Admin Trombi" | 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 | #: trombi/views.py | ||||||
| msgid "Explain why you rejected the comment" | msgid "Explain why you rejected the comment" | ||||||
| msgstr "Expliquez pourquoi vous refusez le commentaire" | 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 | #: trombi/views.py | ||||||
| msgid "Rejected comment" | msgid "Rejected comment" | ||||||
| msgstr "Commentaire rejeté" | msgstr "Commentaire rejeté" | ||||||
| @@ -5470,6 +5636,10 @@ msgstr "" | |||||||
| "pouvez vous inscrire qu'à un seul Trombi, donc ne jouez pas avec cet option " | "pouvez vous inscrire qu'à un seul Trombi, donc ne jouez pas avec cet option " | ||||||
| "ou vous encourerez la colère des admins!" | "ou vous encourerez la colère des admins!" | ||||||
|  |  | ||||||
|  | #: trombi/views.py | ||||||
|  | msgid "User modified" | ||||||
|  | msgstr "Utilisateur modifié" | ||||||
|  |  | ||||||
| #: trombi/views.py | #: trombi/views.py | ||||||
| msgid "Personal email (not UTBM)" | msgid "Personal email (not UTBM)" | ||||||
| msgstr "Email personnel (pas UTBM)" | msgstr "Email personnel (pas UTBM)" | ||||||
| @@ -5482,6 +5652,14 @@ msgstr "Téléphone" | |||||||
| msgid "Native town" | msgid "Native town" | ||||||
| msgstr "Ville d'origine" | 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 | #: trombi/views.py | ||||||
| msgid "" | msgid "" | ||||||
| "You can not yet write comment, you must wait for the subscription deadline " | "You can not yet write comment, you must wait for the subscription deadline " | ||||||
|   | |||||||
							
								
								
									
										83
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										83
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -29,8 +29,8 @@ | |||||||
|         "d3-force-3d": "^3.0.5", |         "d3-force-3d": "^3.0.5", | ||||||
|         "easymde": "^2.19.0", |         "easymde": "^2.19.0", | ||||||
|         "glob": "^11.0.0", |         "glob": "^11.0.0", | ||||||
|  |         "html2canvas": "^1.4.1", | ||||||
|         "htmx.org": "^2.0.3", |         "htmx.org": "^2.0.3", | ||||||
|         "jquery": "^3.7.1", |  | ||||||
|         "js-cookie": "^3.0.5", |         "js-cookie": "^3.0.5", | ||||||
|         "lit-html": "^3.3.0", |         "lit-html": "^3.3.0", | ||||||
|         "native-file-system-adapter": "^3.0.1", |         "native-file-system-adapter": "^3.0.1", | ||||||
| @@ -47,10 +47,9 @@ | |||||||
|         "@types/alpinejs": "^3.13.10", |         "@types/alpinejs": "^3.13.10", | ||||||
|         "@types/cytoscape-cxtmenu": "^3.4.4", |         "@types/cytoscape-cxtmenu": "^3.4.4", | ||||||
|         "@types/cytoscape-klay": "^3.1.4", |         "@types/cytoscape-klay": "^3.1.4", | ||||||
|         "@types/jquery": "^3.5.31", |  | ||||||
|         "@types/js-cookie": "^3.0.6", |         "@types/js-cookie": "^3.0.6", | ||||||
|         "typescript": "^5.8.3", |         "typescript": "^5.8.3", | ||||||
|         "vite": "^6.2.6", |         "vite": "^6.4.1", | ||||||
|         "vite-bundle-visualizer": "^1.2.1", |         "vite-bundle-visualizer": "^1.2.1", | ||||||
|         "vite-plugin-static-copy": "^3.1.2" |         "vite-plugin-static-copy": "^3.1.2" | ||||||
|       } |       } | ||||||
| @@ -2889,16 +2888,6 @@ | |||||||
|       "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", |       "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", | ||||||
|       "license": "MIT" |       "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": { |     "node_modules/@types/js-cookie": { | ||||||
|       "version": "3.0.6", |       "version": "3.0.6", | ||||||
|       "resolved": "https://registry.npmjs.org/@types/js-cookie/-/js-cookie-3.0.6.tgz", |       "resolved": "https://registry.npmjs.org/@types/js-cookie/-/js-cookie-3.0.6.tgz", | ||||||
| @@ -2919,13 +2908,6 @@ | |||||||
|       "integrity": "sha512-a79Yc3TOk6dGdituy8hmTTJXjOkZ7zsFYV10L337ttq/rec8lRMDBpV7fL3uLx6TgbFCa5DU/h8FmIBQPSbU0w==", |       "integrity": "sha512-a79Yc3TOk6dGdituy8hmTTJXjOkZ7zsFYV10L337ttq/rec8lRMDBpV7fL3uLx6TgbFCa5DU/h8FmIBQPSbU0w==", | ||||||
|       "license": "MIT" |       "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": { |     "node_modules/@types/tern": { | ||||||
|       "version": "0.23.9", |       "version": "0.23.9", | ||||||
|       "resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.9.tgz", |       "resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.9.tgz", | ||||||
| @@ -3124,6 +3106,15 @@ | |||||||
|         "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" |         "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/base64-arraybuffer": { | ||||||
|  |       "version": "1.0.2", | ||||||
|  |       "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz", | ||||||
|  |       "integrity": "sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ==", | ||||||
|  |       "license": "MIT", | ||||||
|  |       "engines": { | ||||||
|  |         "node": ">= 0.6.0" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/binary-extensions": { |     "node_modules/binary-extensions": { | ||||||
|       "version": "2.3.0", |       "version": "2.3.0", | ||||||
|       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", |       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", | ||||||
| @@ -3512,6 +3503,15 @@ | |||||||
|         "node": ">= 8" |         "node": ">= 8" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/css-line-break": { | ||||||
|  |       "version": "2.1.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/css-line-break/-/css-line-break-2.1.0.tgz", | ||||||
|  |       "integrity": "sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w==", | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "utrie": "^1.0.2" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/cytoscape": { |     "node_modules/cytoscape": { | ||||||
|       "version": "3.33.1", |       "version": "3.33.1", | ||||||
|       "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.33.1.tgz", |       "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.33.1.tgz", | ||||||
| @@ -4184,6 +4184,19 @@ | |||||||
|         "node": ">= 0.4" |         "node": ">= 0.4" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/html2canvas": { | ||||||
|  |       "version": "1.4.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/html2canvas/-/html2canvas-1.4.1.tgz", | ||||||
|  |       "integrity": "sha512-fPU6BHNpsyIhr8yyMpTLLxAbkaK8ArIBcmZIRiBLiDhjeqvXolaEmDGmELFuX9I4xDcaKKcJl+TKZLqruBbmWA==", | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "css-line-break": "^2.1.0", | ||||||
|  |         "text-segmentation": "^1.0.3" | ||||||
|  |       }, | ||||||
|  |       "engines": { | ||||||
|  |         "node": ">=8.0.0" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/htmx.org": { |     "node_modules/htmx.org": { | ||||||
|       "version": "2.0.6", |       "version": "2.0.6", | ||||||
|       "resolved": "https://registry.npmjs.org/htmx.org/-/htmx.org-2.0.6.tgz", |       "resolved": "https://registry.npmjs.org/htmx.org/-/htmx.org-2.0.6.tgz", | ||||||
| @@ -4384,12 +4397,6 @@ | |||||||
|         "jiti": "lib/jiti-cli.mjs" |         "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": { |     "node_modules/js-cookie": { | ||||||
|       "version": "3.0.5", |       "version": "3.0.5", | ||||||
|       "resolved": "https://registry.npmjs.org/js-cookie/-/js-cookie-3.0.5.tgz", |       "resolved": "https://registry.npmjs.org/js-cookie/-/js-cookie-3.0.5.tgz", | ||||||
| @@ -5479,6 +5486,15 @@ | |||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "ISC" |       "license": "ISC" | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/text-segmentation": { | ||||||
|  |       "version": "1.0.3", | ||||||
|  |       "resolved": "https://registry.npmjs.org/text-segmentation/-/text-segmentation-1.0.3.tgz", | ||||||
|  |       "integrity": "sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw==", | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "utrie": "^1.0.2" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/three": { |     "node_modules/three": { | ||||||
|       "version": "0.177.0", |       "version": "0.177.0", | ||||||
|       "resolved": "https://registry.npmjs.org/three/-/three-0.177.0.tgz", |       "resolved": "https://registry.npmjs.org/three/-/three-0.177.0.tgz", | ||||||
| @@ -5736,10 +5752,19 @@ | |||||||
|         "browserslist": ">= 4.21.0" |         "browserslist": ">= 4.21.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/utrie": { | ||||||
|  |       "version": "1.0.2", | ||||||
|  |       "resolved": "https://registry.npmjs.org/utrie/-/utrie-1.0.2.tgz", | ||||||
|  |       "integrity": "sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw==", | ||||||
|  |       "license": "MIT", | ||||||
|  |       "dependencies": { | ||||||
|  |         "base64-arraybuffer": "^1.0.2" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/vite": { |     "node_modules/vite": { | ||||||
|       "version": "6.3.5", |       "version": "6.4.1", | ||||||
|       "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.5.tgz", |       "resolved": "https://registry.npmjs.org/vite/-/vite-6.4.1.tgz", | ||||||
|       "integrity": "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ==", |       "integrity": "sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g==", | ||||||
|       "dev": true, |       "dev": true, | ||||||
|       "license": "MIT", |       "license": "MIT", | ||||||
|       "dependencies": { |       "dependencies": { | ||||||
|   | |||||||
| @@ -32,10 +32,9 @@ | |||||||
|     "@types/alpinejs": "^3.13.10", |     "@types/alpinejs": "^3.13.10", | ||||||
|     "@types/cytoscape-cxtmenu": "^3.4.4", |     "@types/cytoscape-cxtmenu": "^3.4.4", | ||||||
|     "@types/cytoscape-klay": "^3.1.4", |     "@types/cytoscape-klay": "^3.1.4", | ||||||
|     "@types/jquery": "^3.5.31", |  | ||||||
|     "@types/js-cookie": "^3.0.6", |     "@types/js-cookie": "^3.0.6", | ||||||
|     "typescript": "^5.8.3", |     "typescript": "^5.8.3", | ||||||
|     "vite": "^6.2.6", |     "vite": "^6.4.1", | ||||||
|     "vite-bundle-visualizer": "^1.2.1", |     "vite-bundle-visualizer": "^1.2.1", | ||||||
|     "vite-plugin-static-copy": "^3.1.2" |     "vite-plugin-static-copy": "^3.1.2" | ||||||
|   }, |   }, | ||||||
| @@ -60,8 +59,8 @@ | |||||||
|     "d3-force-3d": "^3.0.5", |     "d3-force-3d": "^3.0.5", | ||||||
|     "easymde": "^2.19.0", |     "easymde": "^2.19.0", | ||||||
|     "glob": "^11.0.0", |     "glob": "^11.0.0", | ||||||
|  |     "html2canvas": "^1.4.1", | ||||||
|     "htmx.org": "^2.0.3", |     "htmx.org": "^2.0.3", | ||||||
|     "jquery": "^3.7.1", |  | ||||||
|     "js-cookie": "^3.0.5", |     "js-cookie": "^3.0.5", | ||||||
|     "lit-html": "^3.3.0", |     "lit-html": "^3.3.0", | ||||||
|     "native-file-system-adapter": "^3.0.1", |     "native-file-system-adapter": "^3.0.1", | ||||||
|   | |||||||
| @@ -13,16 +13,15 @@ | |||||||
| {% block content %} | {% block content %} | ||||||
|   <div class="pedagogy"> |   <div class="pedagogy"> | ||||||
|     <div id="uv_detail"> |     <div id="uv_detail"> | ||||||
|       <p id="return_noscript"><a href="{{ url('pedagogy:guide') }}">{% trans %}Back{% endtrans %}</a></p> |       <button onclick='(function(){ | ||||||
|       <button id="return_js" onclick='(function(){ |                        // If comes from the guide page, go back with history | ||||||
|                                       // If comes from the guide page, go back with history |                        if (document.referrer.replace(/\?(.+)/gm,"").endsWith(`{{ url("pedagogy:guide") }}`)){ | ||||||
|                                       if (document.referrer.replace(/\?(.+)/gm,"").endsWith(`{{ url("pedagogy:guide") }}`)){ |                        window.history.back(); | ||||||
|                                       window.history.back(); |                        return; | ||||||
|                                       return; |                        } | ||||||
|                                       } |                        // Simply goes to the guide page | ||||||
|                                       // Simply goes to the guide page |                        window.location.href = `{{ url("pedagogy:guide") }}`; | ||||||
|                                       window.location.href = `{{ url("pedagogy:guide") }}`; |                        })()' hidden>{% trans %}Back{% endtrans %}</button> | ||||||
|                                       })()' hidden>{% trans %}Back{% endtrans %}</button> |  | ||||||
|  |  | ||||||
|       <h1>{{ object.code }} - {{ object.title }}</h1> |       <h1>{{ object.code }} - {{ object.title }}</h1> | ||||||
|       <br> |       <br> | ||||||
| @@ -217,9 +216,4 @@ | |||||||
|  |  | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|   <script type="text/javascript"> |  | ||||||
|     $("#return_noscript").hide(); |  | ||||||
|     $("#return_js").show(); |  | ||||||
|   </script> |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
|   | |||||||
| @@ -21,11 +21,6 @@ | |||||||
|           {{ field.errors }} |           {{ field.errors }} | ||||||
|           <label for="{{ field.name }}">{{ field.label }}</label> |           <label for="{{ field.name }}">{{ field.label }}</label> | ||||||
|           {{ field }} |           {{ field }} | ||||||
|  |  | ||||||
|  |  | ||||||
|           {% if field.name == 'code' %} |  | ||||||
|             <button type="button" id="autofill">{% trans %}Import from UTBM{% endtrans %}</button> |  | ||||||
|           {% endif %} |  | ||||||
|         </p> |         </p> | ||||||
|       {% endif %} |       {% endif %} | ||||||
|  |  | ||||||
| @@ -36,48 +31,3 @@ | |||||||
|     <p><input type="submit" value="{% trans %}Update{% endtrans %}" /></p> |     <p><input type="submit" value="{% trans %}Update{% endtrans %}" /></p> | ||||||
|   </form> |   </form> | ||||||
| {% endblock %} | {% 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 %} |  | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import { | |||||||
|  |  | ||||||
| interface PagePictureConfig { | interface PagePictureConfig { | ||||||
|   userId: number; |   userId: number; | ||||||
|  |   nbPictures?: number; | ||||||
| } | } | ||||||
|  |  | ||||||
| interface Album { | interface Album { | ||||||
| @@ -20,11 +21,27 @@ document.addEventListener("alpine:init", () => { | |||||||
|     loading: true, |     loading: true, | ||||||
|     albums: [] as Album[], |     albums: [] as Album[], | ||||||
|  |  | ||||||
|     async init() { |     async fetchPictures(): Promise<PictureSchema[]> { | ||||||
|  |       const localStorageKey = `user${config.userId}Pictures`; | ||||||
|  |       const localStorageInvalidationKey = `user${config.userId}PicturesNumber`; | ||||||
|  |       const lastCachedNumber = localStorage.getItem(localStorageInvalidationKey); | ||||||
|  |       if ( | ||||||
|  |         lastCachedNumber !== null && | ||||||
|  |         Number.parseInt(lastCachedNumber) === config.nbPictures | ||||||
|  |       ) { | ||||||
|  |         return JSON.parse(localStorage.getItem(localStorageKey)); | ||||||
|  |       } | ||||||
|       const pictures = await paginated(picturesFetchPictures, { |       const pictures = await paginated(picturesFetchPictures, { | ||||||
|         // biome-ignore lint/style/useNamingConvention: from python api |         // biome-ignore lint/style/useNamingConvention: from python api | ||||||
|         query: { users_identified: [config.userId] }, |         query: { users_identified: [config.userId] }, | ||||||
|       } as PicturesFetchPicturesData); |       } as PicturesFetchPicturesData); | ||||||
|  |       localStorage.setItem(localStorageInvalidationKey, config.nbPictures.toString()); | ||||||
|  |       localStorage.setItem(localStorageKey, JSON.stringify(pictures)); | ||||||
|  |       return pictures; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     async init() { | ||||||
|  |       const pictures = await this.fetchPictures(); | ||||||
|       const groupedAlbums = Object.groupBy(pictures, (i: PictureSchema) => i.album.id); |       const groupedAlbums = Object.groupBy(pictures, (i: PictureSchema) => i.album.id); | ||||||
|       this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => { |       this.albums = Object.values(groupedAlbums).map((pictures: PictureSchema[]) => { | ||||||
|         return { |         return { | ||||||
|   | |||||||
| @@ -309,6 +309,7 @@ exportToHtml("loadViewer", (config: ViewerConfig) => { | |||||||
|         // Clear selection and cache of retrieved user so they can be filtered again |         // Clear selection and cache of retrieved user so they can be filtered again | ||||||
|         widget.clear(false); |         widget.clear(false); | ||||||
|         widget.clearOptions(); |         widget.clearOptions(); | ||||||
|  |         widget.setTextboxValue(""); | ||||||
|       }, |       }, | ||||||
|  |  | ||||||
|       /** |       /** | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ | |||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|   <main x-data="user_pictures({ userId: {{ object.id }} })"> |   <main x-data="user_pictures({ userId: {{ object.id }}, nbPictures: {{ object.nb_pictures }} })"> | ||||||
|     {% if user.id == object.id %} |     {% if user.id == object.id %} | ||||||
|       {{ download_button(_("Download all my pictures")) }} |       {{ download_button(_("Download all my pictures")) }} | ||||||
|     {% endif %} |     {% endif %} | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								sas/views.py
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								sas/views.py
									
									
									
									
									
								
							| @@ -16,6 +16,7 @@ from typing import Any | |||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.core.exceptions import PermissionDenied | from django.core.exceptions import PermissionDenied | ||||||
|  | from django.db.models import Count, OuterRef, Subquery | ||||||
| from django.http import Http404, HttpResponseRedirect | from django.http import Http404, HttpResponseRedirect | ||||||
| from django.shortcuts import get_object_or_404 | from django.shortcuts import get_object_or_404 | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| @@ -36,7 +37,7 @@ from sas.forms import ( | |||||||
|     PictureModerationRequestForm, |     PictureModerationRequestForm, | ||||||
|     PictureUploadForm, |     PictureUploadForm, | ||||||
| ) | ) | ||||||
| from sas.models import Album, Picture | from sas.models import Album, PeoplePictureRelation, Picture | ||||||
|  |  | ||||||
|  |  | ||||||
| class AlbumCreateFragment(FragmentMixin, CreateView): | class AlbumCreateFragment(FragmentMixin, CreateView): | ||||||
| @@ -178,6 +179,13 @@ class UserPicturesView(UserTabsMixin, CanViewMixin, DetailView): | |||||||
|     context_object_name = "profile" |     context_object_name = "profile" | ||||||
|     template_name = "sas/user_pictures.jinja" |     template_name = "sas/user_pictures.jinja" | ||||||
|     current_tab = "pictures" |     current_tab = "pictures" | ||||||
|  |     queryset = User.objects.annotate( | ||||||
|  |         nb_pictures=Subquery( | ||||||
|  |             PeoplePictureRelation.objects.filter(user=OuterRef("id")) | ||||||
|  |             .values("user_id") | ||||||
|  |             .values(count=Count("*")) | ||||||
|  |         ) | ||||||
|  |     ).all() | ||||||
|  |  | ||||||
|  |  | ||||||
| # Admin views | # Admin views | ||||||
|   | |||||||
| @@ -125,6 +125,7 @@ INSTALLED_APPS = ( | |||||||
|     "pedagogy", |     "pedagogy", | ||||||
|     "galaxy", |     "galaxy", | ||||||
|     "antispam", |     "antispam", | ||||||
|  |     "timetable", | ||||||
|     "api", |     "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_ROOT_DIR_ID = env.int("SITH_SAS_ROOT_DIR_ID", default=4) | ||||||
| SITH_SAS_IMAGES_PER_PAGE = 60 | SITH_SAS_IMAGES_PER_PAGE = 60 | ||||||
|  |  | ||||||
| SITH_BOARD_SUFFIX = "-bureau" |  | ||||||
| SITH_MEMBER_SUFFIX = "-membres" |  | ||||||
|  |  | ||||||
| SITH_PROFILE_DEPARTMENTS = [ | SITH_PROFILE_DEPARTMENTS = [ | ||||||
|     ("TC", _("TC")), |     ("TC", _("TC")), | ||||||
|     ("IMSI", _("IMSI")), |     ("IMSI", _("IMSI")), | ||||||
| @@ -424,18 +422,11 @@ SITH_PROFILE_DEPARTMENTS = [ | |||||||
|     ("NA", _("N/A")), |     ("NA", _("N/A")), | ||||||
| ] | ] | ||||||
|  |  | ||||||
| SITH_ACCOUNTING_PAYMENT_METHOD = [ |  | ||||||
|     ("CHECK", _("Check")), |  | ||||||
|     ("CASH", _("Cash")), |  | ||||||
|     ("TRANSFERT", _("Transfert")), |  | ||||||
|     ("CARD", _("Credit card")), |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| SITH_SUBSCRIPTION_PAYMENT_METHOD = [ | SITH_SUBSCRIPTION_PAYMENT_METHOD = [ | ||||||
|     ("CHECK", _("Check")), |     ("CHECK", _("Check")), | ||||||
|     ("CARD", _("Credit card")), |     ("CARD", _("Credit card")), | ||||||
|     ("CASH", _("Cash")), |     ("CASH", _("Cash")), | ||||||
|     ("EBOUTIC", _("Eboutic")), |     ("AE_ACCOUNT", _("AE account")), | ||||||
|     ("OTHER", _("Other")), |     ("OTHER", _("Other")), | ||||||
| ] | ] | ||||||
|  |  | ||||||
| @@ -444,6 +435,7 @@ SITH_SUBSCRIPTION_LOCATIONS = [ | |||||||
|     ("SEVENANS", _("Sevenans")), |     ("SEVENANS", _("Sevenans")), | ||||||
|     ("MONTBELIARD", _("Montbéliard")), |     ("MONTBELIARD", _("Montbéliard")), | ||||||
|     ("EBOUTIC", _("Eboutic")), |     ("EBOUTIC", _("Eboutic")), | ||||||
|  |     ("OTHER", _("Other")), | ||||||
| ] | ] | ||||||
|  |  | ||||||
| SITH_COUNTER_BARS = [(1, "MDE"), (2, "Foyer"), (35, "La Gommette")] | SITH_COUNTER_BARS = [(1, "MDE"), (2, "Foyer"), (35, "La Gommette")] | ||||||
| @@ -694,14 +686,6 @@ SITH_PERMANENT_NOTIFICATIONS = { | |||||||
|     "SAS_MODERATION": "sas.models.sas_notification_callback", |     "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 | # Mailing related settings | ||||||
|  |  | ||||||
| SITH_MAILING_DOMAIN = "utbm.fr" | SITH_MAILING_DOMAIN = "utbm.fr" | ||||||
|   | |||||||
| @@ -53,6 +53,7 @@ urlpatterns = [ | |||||||
|     path("i18n/", include("django.conf.urls.i18n")), |     path("i18n/", include("django.conf.urls.i18n")), | ||||||
|     path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"), |     path("jsi18n/", JavaScriptCatalog.as_view(), name="javascript-catalog"), | ||||||
|     path("captcha/", include("captcha.urls")), |     path("captcha/", include("captcha.urls")), | ||||||
|  |     path("edt/", include(("timetable.urls", "timetable"), namespace="timetable")), | ||||||
| ] | ] | ||||||
|  |  | ||||||
| if settings.DEBUG: | if settings.DEBUG: | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import secrets | |||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from django import forms | from django import forms | ||||||
|  | from django.conf import settings | ||||||
| from django.core.exceptions import ValidationError | from django.core.exceptions import ValidationError | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
|  |  | ||||||
| @@ -23,13 +24,28 @@ class SelectionDateForm(forms.Form): | |||||||
|  |  | ||||||
|  |  | ||||||
| class SubscriptionForm(forms.ModelForm): | class SubscriptionForm(forms.ModelForm): | ||||||
|     def __init__(self, *args, **kwargs): |     allowed_payment_methods = ["CARD", "CASH", "AE_ACCOUNT"] | ||||||
|         initial = kwargs.pop("initial", {}) |  | ||||||
|  |     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: |         if "subscription_type" not in initial: | ||||||
|             initial["subscription_type"] = "deux-semestres" |             initial["subscription_type"] = "deux-semestres" | ||||||
|         if "payment_method" not in initial: |         if "payment_method" not in initial: | ||||||
|             initial["payment_method"] = "CARD" |             initial["payment_method"] = "CARD" | ||||||
|         super().__init__(*args, initial=initial, **kwargs) |         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): |     def save(self, *args, **kwargs): | ||||||
|         if self.errors: |         if self.errors: | ||||||
| @@ -61,7 +77,8 @@ class SubscriptionNewUserForm(SubscriptionForm): | |||||||
|         assert user.is_subscribed |         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_fields = forms.fields_for_model( | ||||||
|         User, |         User, | ||||||
| @@ -73,10 +90,6 @@ class SubscriptionNewUserForm(SubscriptionForm): | |||||||
|     email = __user_fields["email"] |     email = __user_fields["email"] | ||||||
|     date_of_birth = __user_fields["date_of_birth"] |     date_of_birth = __user_fields["date_of_birth"] | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         model = Subscription |  | ||||||
|         fields = ["subscription_type", "payment_method", "location"] |  | ||||||
|  |  | ||||||
|     field_order = [ |     field_order = [ | ||||||
|         "first_name", |         "first_name", | ||||||
|         "last_name", |         "last_name", | ||||||
| @@ -130,9 +143,57 @@ class SubscriptionNewUserForm(SubscriptionForm): | |||||||
| class SubscriptionExistingUserForm(SubscriptionForm): | class SubscriptionExistingUserForm(SubscriptionForm): | ||||||
|     """Form to add a subscription to an existing user.""" |     """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: |     birthdate = forms.fields_for_model( | ||||||
|         model = Subscription |         User, | ||||||
|         fields = ["member", "subscription_type", "payment_method", "location"] |         ["date_of_birth"], | ||||||
|         widgets = {"member": AutoCompleteSelectUser} |         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", () => { | document.addEventListener("alpine:init", () => { | ||||||
|   Alpine.data("existing_user_subscription_form", () => ({ |   Alpine.data("existing_user_subscription_form", () => ({ | ||||||
|     loading: false, |     loading: false, | ||||||
| @@ -12,13 +14,24 @@ document.addEventListener("alpine:init", () => { | |||||||
|     }, |     }, | ||||||
|  |  | ||||||
|     async loadProfile(userId: number) { |     async loadProfile(userId: number) { | ||||||
|  |       const birthdayInput = document.getElementById("id_birthdate") as HTMLInputElement; | ||||||
|       if (!Number.isInteger(userId)) { |       if (!Number.isInteger(userId)) { | ||||||
|         this.profileFragment = ""; |         this.profileFragment = ""; | ||||||
|  |         birthdayInput.hidden = true; | ||||||
|         return; |         return; | ||||||
|       } |       } | ||||||
|       this.loading = true; |       this.loading = true; | ||||||
|       const response = await fetch(`/user/${userId}/mini/`); |       const [miniProfile, userInfos] = await Promise.all([ | ||||||
|       this.profileFragment = await response.text(); |         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; |       this.loading = false; | ||||||
|     }, |     }, | ||||||
|   })); |   })); | ||||||
|   | |||||||
| @@ -1,4 +1,14 @@ | |||||||
| #subscription-form form { | #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 { |   .form-content.existing-user { | ||||||
|     max-height: 100%; |     max-height: 100%; | ||||||
|     display: flex; |     display: flex; | ||||||
| @@ -13,6 +23,11 @@ | |||||||
|      * then display the user profile right in the middle of the remaining space. */ |      * then display the user profile right in the middle of the remaining space. */ | ||||||
|     fieldset { |     fieldset { | ||||||
|       flex: 0 1 auto; |       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 { |     #subscription-form-user-mini-profile { | ||||||
|   | |||||||
| @@ -1,14 +0,0 @@ | |||||||
| {% load static %} |  | ||||||
| {% load i18n %} |  | ||||||
|  |  | ||||||
|  |  | ||||||
| <div x-data="existing_user_subscription_form" class="form-content existing-user"> |  | ||||||
|   <fieldset> |  | ||||||
|     {{ form.as_p }} |  | ||||||
|   </fieldset> |  | ||||||
|   <div |  | ||||||
|     id="subscription-form-user-mini-profile" |  | ||||||
|     x-html="profileFragment" |  | ||||||
|     :aria-busy="loading" |  | ||||||
|   ></div> |  | ||||||
| </div> |  | ||||||
| @@ -0,0 +1,28 @@ | |||||||
|  | {% load static %} | ||||||
|  | {% load i18n %} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <div x-data="existing_user_subscription_form" class="form-content existing-user"> | ||||||
|  |   <fieldset> | ||||||
|  |     {{ errors }} | ||||||
|  |     {% for field, errors in fields %} | ||||||
|  |       <p{% with classes=field.css_classes %}{% if classes %} class="{{ classes }}"{% endif %}{% endwith %}> | ||||||
|  |         {{ field.label_tag }} | ||||||
|  |         {{ field }} | ||||||
|  |         {% if field.help_text %} | ||||||
|  |           <span class="helptext">{{ field.help_text }}</span> | ||||||
|  |         {% endif %} | ||||||
|  |       </p> | ||||||
|  |       {% if field.name == "payment_method" %} | ||||||
|  |         <i> | ||||||
|  |           {% blocktranslate %}If the subscription is done using the AE account, you must also click it on the AE counter.{% endblocktranslate %} | ||||||
|  |         </i> | ||||||
|  |       {% endif %} | ||||||
|  |     {% endfor %} | ||||||
|  |   </fieldset> | ||||||
|  |   <div | ||||||
|  |     id="subscription-form-user-mini-profile" | ||||||
|  |     x-html="profileFragment" | ||||||
|  |     :aria-busy="loading" | ||||||
|  |   ></div> | ||||||
|  | </div> | ||||||
| @@ -90,7 +90,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=self.user, |             member=self.user, | ||||||
|             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], |             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2017, 8, 29) |         s.subscription_start = date(2017, 8, 29) | ||||||
|         s.subscription_end = s.compute_end(duration=0.166, start=s.subscription_start) |         s.subscription_end = s.compute_end(duration=0.166, start=s.subscription_start) | ||||||
| @@ -101,7 +101,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=self.user, |             member=self.user, | ||||||
|             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], |             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2017, 8, 29) |         s.subscription_start = date(2017, 8, 29) | ||||||
|         s.subscription_end = s.compute_end(duration=0.333, start=s.subscription_start) |         s.subscription_end = s.compute_end(duration=0.333, start=s.subscription_start) | ||||||
| @@ -112,7 +112,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=self.user, |             member=self.user, | ||||||
|             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], |             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2017, 8, 29) |         s.subscription_start = date(2017, 8, 29) | ||||||
|         s.subscription_end = s.compute_end( |         s.subscription_end = s.compute_end( | ||||||
| @@ -126,7 +126,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=self.user, |             member=self.user, | ||||||
|             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], |             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2017, 8, 29) |         s.subscription_start = date(2017, 8, 29) | ||||||
|         s.subscription_end = s.compute_end(duration=0.5, start=s.subscription_start) |         s.subscription_end = s.compute_end(duration=0.5, start=s.subscription_start) | ||||||
| @@ -137,7 +137,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=self.user, |             member=self.user, | ||||||
|             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], |             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2017, 8, 29) |         s.subscription_start = date(2017, 8, 29) | ||||||
|         s.subscription_end = s.compute_end(duration=0.67, start=s.subscription_start) |         s.subscription_end = s.compute_end(duration=0.67, start=s.subscription_start) | ||||||
| @@ -148,7 +148,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=self.user, |             member=self.user, | ||||||
|             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], |             subscription_type=list(settings.SITH_SUBSCRIPTIONS.keys())[3], | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2018, 9, 1) |         s.subscription_start = date(2018, 9, 1) | ||||||
|         s.subscription_end = s.compute_end(duration=0.23, start=s.subscription_start) |         s.subscription_end = s.compute_end(duration=0.23, start=s.subscription_start) | ||||||
| @@ -160,7 +160,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=user, |             member=user, | ||||||
|             subscription_type="deux-semestres", |             subscription_type="deux-semestres", | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2015, 8, 29) |         s.subscription_start = date(2015, 8, 29) | ||||||
|         s.subscription_end = s.compute_end( |         s.subscription_end = s.compute_end( | ||||||
| @@ -181,7 +181,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=user, |             member=user, | ||||||
|             subscription_type="deux-mois-essai", |             subscription_type="deux-mois-essai", | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2015, 8, 29) |         s.subscription_start = date(2015, 8, 29) | ||||||
|         s.subscription_end = s.compute_end( |         s.subscription_end = s.compute_end( | ||||||
| @@ -202,7 +202,7 @@ class TestSubscriptionIntegration(TestCase): | |||||||
|         s = Subscription( |         s = Subscription( | ||||||
|             member=user, |             member=user, | ||||||
|             subscription_type="deux-mois-essai", |             subscription_type="deux-mois-essai", | ||||||
|             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[0], |             payment_method=settings.SITH_SUBSCRIPTION_PAYMENT_METHOD[1], | ||||||
|         ) |         ) | ||||||
|         s.subscription_start = date(2015, 8, 29) |         s.subscription_start = date(2015, 8, 29) | ||||||
|         s.subscription_end = s.compute_end( |         s.subscription_end = s.compute_end( | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user