mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-31 15:50:03 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			123 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			123 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
 | |
| # SPDX-License-Identifier: GPL-3.0-or-later
 | |
| 
 | |
| from datetime import timedelta
 | |
| from random import shuffle
 | |
| 
 | |
| from bootstrap_datepicker_plus.widgets import DateTimePickerInput
 | |
| from django import forms
 | |
| from django.contrib.contenttypes.models import ContentType
 | |
| from django.utils import timezone
 | |
| from django.utils.translation import gettext_lazy as _
 | |
| from member.models import Club
 | |
| from note.models import Note, NoteUser
 | |
| from note_kfet.inputs import Autocomplete
 | |
| from note_kfet.middlewares import get_current_request
 | |
| from permission.backends import PermissionBackend
 | |
| 
 | |
| from .models import Activity, Guest
 | |
| 
 | |
| 
 | |
| class ActivityForm(forms.ModelForm):
 | |
|     def __init__(self, *args, **kwargs):
 | |
|         super().__init__(*args, **kwargs)
 | |
|         # By default, the Kfet club is attended
 | |
|         self.fields["attendees_club"].initial = Club.objects.get(name="Kfet")
 | |
|         self.fields["attendees_club"].widget.attrs["placeholder"] = "Kfet"
 | |
|         clubs = list(Club.objects.filter(PermissionBackend
 | |
|                                          .filter_queryset(get_current_request(), Club, "view")).all())
 | |
|         shuffle(clubs)
 | |
|         self.fields["organizer"].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
 | |
| 
 | |
|     def clean_organizer(self):
 | |
|         organizer = self.cleaned_data['organizer']
 | |
|         if not organizer.note.is_active:
 | |
|             self.add_error('organizer', _('The note of this club is inactive.'))
 | |
|         return organizer
 | |
| 
 | |
|     def clean_date_end(self):
 | |
|         date_end = self.cleaned_data["date_end"]
 | |
|         date_start = self.cleaned_data["date_start"]
 | |
|         if date_end < date_start:
 | |
|             self.add_error("date_end", _("The end date must be after the start date."))
 | |
|         return date_end
 | |
| 
 | |
|     class Meta:
 | |
|         model = Activity
 | |
|         exclude = ('creater', 'valid', 'open', 'opener', )
 | |
|         widgets = {
 | |
|             "organizer": Autocomplete(
 | |
|                 model=Club,
 | |
|                 attrs={"api_url": "/api/members/club/"},
 | |
|             ),
 | |
|             "note": Autocomplete(
 | |
|                 model=Note,
 | |
|                 attrs={
 | |
|                     "api_url": "/api/note/note/",
 | |
|                     'placeholder': 'Note de l\'événement sur laquelle envoyer les crédits d\'invitation ...'
 | |
|                 },
 | |
|             ),
 | |
|             "attendees_club": Autocomplete(
 | |
|                 model=Club,
 | |
|                 attrs={"api_url": "/api/members/club/"},
 | |
|             ),
 | |
|             "date_start": DateTimePickerInput(),
 | |
|             "date_end": DateTimePickerInput(),
 | |
|         }
 | |
| 
 | |
| 
 | |
| class GuestForm(forms.ModelForm):
 | |
|     def clean(self):
 | |
|         """
 | |
|         Someone can be invited as a Guest to an Activity if:
 | |
|         - the activity has not already started.
 | |
|         - the activity is validated.
 | |
|         - the Guest has not already been invited more than 5 times.
 | |
|         - the Guest is already invited.
 | |
|         - the inviter already invited 3 peoples.
 | |
|         """
 | |
| 
 | |
|         cleaned_data = super().clean()
 | |
| 
 | |
|         if timezone.now() > timezone.localtime(self.activity.date_start):
 | |
|             self.add_error("inviter", _("You can't invite someone once the activity is started."))
 | |
| 
 | |
|         if not self.activity.valid:
 | |
|             self.add_error("inviter", _("This activity is not validated yet."))
 | |
| 
 | |
|         one_year = timedelta(days=365)
 | |
| 
 | |
|         qs = Guest.objects.filter(
 | |
|             first_name__iexact=cleaned_data["first_name"],
 | |
|             last_name__iexact=cleaned_data["last_name"],
 | |
|             activity__date_start__gte=self.activity.date_start - one_year,
 | |
|         )
 | |
|         if qs.filter(entry__isnull=False).count() >= 5:
 | |
|             self.add_error("last_name", _("This person has been already invited 5 times this year."))
 | |
| 
 | |
|         qs = qs.filter(activity=self.activity)
 | |
|         if qs.exists():
 | |
|             self.add_error("last_name", _("This person is already invited."))
 | |
| 
 | |
|         if "inviter" in cleaned_data:
 | |
|             if Guest.objects.filter(inviter=cleaned_data["inviter"], activity=self.activity).count() >= 3:
 | |
|                 self.add_error("inviter", _("You can't invite more than 3 people to this activity."))
 | |
| 
 | |
|         return cleaned_data
 | |
| 
 | |
|     class Meta:
 | |
|         model = Guest
 | |
|         fields = ('last_name', 'first_name', 'school', 'inviter', )
 | |
|         widgets = {
 | |
|             "inviter": Autocomplete(
 | |
|                 NoteUser,
 | |
|                 attrs={
 | |
|                     'api_url': '/api/note/note/',
 | |
|                     # We don't evaluate the content type at launch because the DB might be not initialized
 | |
|                     'api_url_suffix':
 | |
|                         lambda: '&polymorphic_ctype=' + str(ContentType.objects.get_for_model(NoteUser).pk),
 | |
|                     'placeholder': 'Note ...',
 | |
|                 },
 | |
|             ),
 | |
|         }
 |