2024-02-07 02:26:49 +01:00
|
|
|
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
2019-07-08 13:59:31 +02:00
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-02-18 21:30:26 +01:00
|
|
|
|
2020-09-06 12:04:54 +02:00
|
|
|
import io
|
|
|
|
|
2024-02-12 21:25:07 +01:00
|
|
|
from bootstrap_datepicker_plus.widgets import DatePickerInput
|
2020-03-07 22:28:59 +01:00
|
|
|
from django import forms
|
2020-09-06 12:04:54 +02:00
|
|
|
from django.conf import settings
|
2020-04-05 05:17:28 +02:00
|
|
|
from django.contrib.auth.forms import AuthenticationForm
|
2019-08-11 16:22:52 +02:00
|
|
|
from django.contrib.auth.models import User
|
2020-09-11 22:52:16 +02:00
|
|
|
from django.db import transaction
|
2020-08-03 13:33:25 +02:00
|
|
|
from django.forms import CheckboxSelectMultiple
|
2020-08-15 21:30:08 +02:00
|
|
|
from django.utils import timezone
|
2020-04-05 18:37:04 +02:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2020-07-28 23:16:38 +02:00
|
|
|
from note.models import NoteSpecial, Alias
|
2024-02-12 21:25:07 +01:00
|
|
|
from note_kfet.inputs import Autocomplete, AmountInput
|
2020-07-25 19:40:30 +02:00
|
|
|
from permission.models import PermissionMask, Role
|
2024-02-12 21:25:07 +01:00
|
|
|
from PIL import Image, ImageSequence
|
2020-03-20 02:14:43 +01:00
|
|
|
|
2020-07-25 19:40:30 +02:00
|
|
|
from .models import Profile, Club, Membership
|
2019-08-14 18:47:46 +02:00
|
|
|
|
2019-07-08 13:59:31 +02:00
|
|
|
|
2020-03-19 16:12:52 +01:00
|
|
|
class CustomAuthenticationForm(AuthenticationForm):
|
|
|
|
permission_mask = forms.ModelChoiceField(
|
2020-09-06 20:21:31 +02:00
|
|
|
label=_("Permission mask"),
|
2020-03-19 16:12:52 +01:00
|
|
|
queryset=PermissionMask.objects.order_by("rank"),
|
|
|
|
empty_label=None,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2020-07-29 19:37:40 +02:00
|
|
|
class UserForm(forms.ModelForm):
|
|
|
|
def _get_validation_exclusions(self):
|
|
|
|
# Django usernames can only contain letters, numbers, @, ., +, - and _.
|
|
|
|
# We want to allow users to have uncommon and unpractical usernames:
|
|
|
|
# That is their problem, and we have normalized aliases for us.
|
2024-02-12 22:56:43 +01:00
|
|
|
return super()._get_validation_exclusions() | {"username"}
|
2020-07-29 19:37:40 +02:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = User
|
|
|
|
fields = ('first_name', 'last_name', 'username', 'email',)
|
|
|
|
|
|
|
|
|
2019-08-11 16:22:52 +02:00
|
|
|
class ProfileForm(forms.ModelForm):
|
2019-08-11 17:52:41 +02:00
|
|
|
"""
|
2020-02-02 15:42:39 +01:00
|
|
|
A form for the extras field provided by the :model:`member.Profile` model.
|
2019-08-11 17:52:41 +02:00
|
|
|
"""
|
2020-08-06 19:56:37 +02:00
|
|
|
report_frequency = forms.IntegerField(required=False, initial=0, label=_("Report frequency"))
|
|
|
|
|
2020-08-08 15:30:33 +02:00
|
|
|
last_report = forms.DateTimeField(required=False, disabled=True, label=_("Last report date"))
|
2020-03-07 22:28:59 +01:00
|
|
|
|
2023-08-31 12:21:38 +02:00
|
|
|
VSS_charter_read = forms.BooleanField(
|
|
|
|
required=True,
|
2023-08-31 13:40:53 +02:00
|
|
|
label=_("Anti-VSS (<em>Violences Sexistes et Sexuelles</em>) charter read and approved"),
|
|
|
|
help_text=_("Tick after having read and accepted the anti-VSS charter \
|
|
|
|
<a href=https://perso.crans.org/club-bde/Charte-anti-VSS.pdf target=_blank> available here in pdf</a>")
|
2023-08-31 12:21:38 +02:00
|
|
|
)
|
|
|
|
|
2020-08-15 21:30:08 +02:00
|
|
|
def clean_promotion(self):
|
|
|
|
promotion = self.cleaned_data["promotion"]
|
|
|
|
if promotion > timezone.now().year:
|
|
|
|
self.add_error("promotion", _("You can't register to the note if you come from the future."))
|
|
|
|
return promotion
|
|
|
|
|
2020-08-10 12:09:05 +02:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.fields['address'].widget.attrs.update({"placeholder": "4 avenue des Sciences, 91190 GIF-SUR-YVETTE"})
|
2020-08-15 21:30:08 +02:00
|
|
|
self.fields['promotion'].widget.attrs.update({"max": timezone.now().year})
|
2020-08-10 12:09:05 +02:00
|
|
|
|
2020-09-11 22:52:16 +02:00
|
|
|
@transaction.atomic
|
2020-04-22 16:25:09 +02:00
|
|
|
def save(self, commit=True):
|
|
|
|
if not self.instance.section or (("department" in self.changed_data
|
|
|
|
or "promotion" in self.changed_data) and "section" not in self.changed_data):
|
|
|
|
self.instance.section = self.instance.section_generated
|
|
|
|
return super().save(commit)
|
|
|
|
|
2019-08-11 16:22:52 +02:00
|
|
|
class Meta:
|
|
|
|
model = Profile
|
|
|
|
fields = '__all__'
|
2020-08-09 16:38:37 +02:00
|
|
|
exclude = ('user', 'email_confirmed', 'registration_valid', )
|
2019-08-11 23:25:27 +02:00
|
|
|
|
2020-02-18 12:31:15 +01:00
|
|
|
|
2020-08-18 18:19:39 +02:00
|
|
|
class ImageForm(forms.Form):
|
|
|
|
"""
|
|
|
|
Form used for the js interface for profile picture
|
|
|
|
"""
|
|
|
|
image = forms.ImageField(required=False,
|
|
|
|
label=_('select an image'),
|
|
|
|
help_text=_('Maximal size: 2MB'))
|
|
|
|
x = forms.FloatField(widget=forms.HiddenInput())
|
|
|
|
y = forms.FloatField(widget=forms.HiddenInput())
|
|
|
|
width = forms.FloatField(widget=forms.HiddenInput())
|
|
|
|
height = forms.FloatField(widget=forms.HiddenInput())
|
|
|
|
|
2020-09-06 12:04:54 +02:00
|
|
|
def clean(self):
|
2020-09-06 18:54:21 +02:00
|
|
|
"""
|
|
|
|
Load image and crop
|
|
|
|
|
|
|
|
In the future, when Pillow will support APNG we will be able to
|
|
|
|
simplify this code to save only PNG/APNG.
|
|
|
|
"""
|
2020-09-06 12:04:54 +02:00
|
|
|
cleaned_data = super().clean()
|
|
|
|
|
|
|
|
# Image size is limited by Django DATA_UPLOAD_MAX_MEMORY_SIZE
|
|
|
|
image = cleaned_data.get('image')
|
|
|
|
if image:
|
|
|
|
# Let Pillow detect and load image
|
2020-09-06 18:54:21 +02:00
|
|
|
# If it is an animation, then there will be multiple frames
|
2020-09-06 12:04:54 +02:00
|
|
|
try:
|
|
|
|
im = Image.open(image)
|
|
|
|
except OSError:
|
|
|
|
# Rare case in which Django consider the upload file as an image
|
|
|
|
# but Pil is unable to load it
|
|
|
|
raise forms.ValidationError(_('This image cannot be loaded.'))
|
|
|
|
|
2020-09-06 18:54:21 +02:00
|
|
|
# Crop each frame
|
2020-09-06 12:04:54 +02:00
|
|
|
x = cleaned_data.get('x', 0)
|
|
|
|
y = cleaned_data.get('y', 0)
|
|
|
|
w = cleaned_data.get('width', 200)
|
|
|
|
h = cleaned_data.get('height', 200)
|
2020-09-06 18:54:21 +02:00
|
|
|
frames = []
|
|
|
|
for frame in ImageSequence.Iterator(im):
|
|
|
|
frame = frame.crop((x, y, x + w, y + h))
|
|
|
|
frame = frame.resize(
|
|
|
|
(settings.PIC_WIDTH, settings.PIC_RATIO * settings.PIC_WIDTH),
|
2024-02-12 22:56:43 +01:00
|
|
|
Image.LANCZOS,
|
2020-09-06 18:54:21 +02:00
|
|
|
)
|
|
|
|
frames.append(frame)
|
2020-09-06 12:04:54 +02:00
|
|
|
|
|
|
|
# Save
|
2020-09-06 19:16:35 +02:00
|
|
|
om = frames.pop(0) # Get first frame
|
|
|
|
om.info = im.info # Copy metadata
|
2020-09-06 12:04:54 +02:00
|
|
|
image.file = io.BytesIO()
|
2020-09-06 18:54:21 +02:00
|
|
|
if len(frames) > 1:
|
|
|
|
# Save as GIF
|
|
|
|
om.save(image.file, "GIF", save_all=True, append_images=list(frames), loop=0)
|
|
|
|
else:
|
|
|
|
# Save as PNG
|
|
|
|
om.save(image.file, "PNG")
|
2020-09-06 12:04:54 +02:00
|
|
|
|
|
|
|
return cleaned_data
|
|
|
|
|
2024-07-13 17:37:19 +02:00
|
|
|
def is_valid(self):
|
|
|
|
return super().is_valid() or super().clean().get('image') is None
|
|
|
|
|
2020-08-30 11:59:10 +02:00
|
|
|
|
2019-08-11 23:25:27 +02:00
|
|
|
class ClubForm(forms.ModelForm):
|
2020-07-28 23:16:38 +02:00
|
|
|
def clean(self):
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
|
|
|
|
if not self.instance.pk: # Creating a club
|
|
|
|
if Alias.objects.filter(normalized_name=Alias.normalize(self.cleaned_data["name"])).exists():
|
|
|
|
self.add_error('name', _("An alias with a similar name already exists."))
|
|
|
|
|
|
|
|
return cleaned_data
|
|
|
|
|
2019-08-11 23:25:27 +02:00
|
|
|
class Meta:
|
|
|
|
model = Club
|
2024-07-15 14:27:55 +02:00
|
|
|
exclude = ("add_registration_form",)
|
2020-03-31 01:03:30 +02:00
|
|
|
widgets = {
|
2020-04-01 04:07:55 +02:00
|
|
|
"membership_fee_paid": AmountInput(),
|
|
|
|
"membership_fee_unpaid": AmountInput(),
|
2020-03-31 04:16:30 +02:00
|
|
|
"parent_club": Autocomplete(
|
|
|
|
Club,
|
2020-10-07 09:48:21 +02:00
|
|
|
resetable=True,
|
2020-03-31 04:16:30 +02:00
|
|
|
attrs={
|
|
|
|
'api_url': '/api/members/club/',
|
|
|
|
}
|
|
|
|
),
|
2020-03-31 23:54:14 +02:00
|
|
|
"membership_start": DatePickerInput(),
|
|
|
|
"membership_end": DatePickerInput(),
|
2020-03-31 01:03:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-14 18:47:46 +02:00
|
|
|
class MembershipForm(forms.ModelForm):
|
2020-04-05 22:35:56 +02:00
|
|
|
soge = forms.BooleanField(
|
|
|
|
label=_("Inscription paid by Société Générale"),
|
|
|
|
required=False,
|
2020-09-13 12:40:10 +02:00
|
|
|
help_text=_("Check this case if the Société Générale paid the inscription."),
|
2020-04-05 22:35:56 +02:00
|
|
|
)
|
|
|
|
|
2020-04-05 18:37:04 +02:00
|
|
|
credit_type = forms.ModelChoiceField(
|
|
|
|
queryset=NoteSpecial.objects,
|
|
|
|
label=_("Credit type"),
|
|
|
|
empty_label=_("No credit"),
|
|
|
|
required=False,
|
|
|
|
help_text=_("You can credit the note of the user."),
|
|
|
|
)
|
|
|
|
|
|
|
|
credit_amount = forms.IntegerField(
|
|
|
|
label=_("Credit amount"),
|
|
|
|
required=False,
|
|
|
|
initial=0,
|
|
|
|
widget=AmountInput(),
|
|
|
|
)
|
|
|
|
|
|
|
|
last_name = forms.CharField(
|
|
|
|
label=_("Last name"),
|
|
|
|
required=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
first_name = forms.CharField(
|
|
|
|
label=_("First name"),
|
|
|
|
required=False,
|
|
|
|
)
|
|
|
|
|
|
|
|
bank = forms.CharField(
|
|
|
|
label=_("Bank"),
|
|
|
|
required=False,
|
|
|
|
)
|
|
|
|
|
2019-08-14 18:47:46 +02:00
|
|
|
class Meta:
|
|
|
|
model = Membership
|
2020-07-31 09:41:22 +02:00
|
|
|
fields = ('user', 'date_start')
|
2022-08-29 13:19:19 +02:00
|
|
|
# Le champ d'utilisateur⋅rice est remplacé par un champ d'auto-complétion.
|
2020-02-08 21:40:32 +01:00
|
|
|
# Quand des lettres sont tapées, une requête est envoyée sur l'API d'auto-complétion
|
2022-08-29 13:19:19 +02:00
|
|
|
# et récupère les noms d'utilisateur⋅rices valides
|
2020-02-08 20:39:37 +01:00
|
|
|
widgets = {
|
2020-02-18 12:31:15 +01:00
|
|
|
'user':
|
2020-03-30 00:42:32 +02:00
|
|
|
Autocomplete(
|
2020-03-27 16:19:33 +01:00
|
|
|
User,
|
2020-03-07 22:28:59 +01:00
|
|
|
attrs={
|
2020-03-27 16:19:33 +01:00
|
|
|
'api_url': '/api/user/',
|
|
|
|
'name_field': 'username',
|
|
|
|
'placeholder': 'Nom ...',
|
2020-03-07 22:28:59 +01:00
|
|
|
},
|
|
|
|
),
|
2020-03-31 23:54:14 +02:00
|
|
|
'date_start': DatePickerInput(),
|
2020-02-08 20:39:37 +01:00
|
|
|
}
|
2020-07-31 09:41:22 +02:00
|
|
|
|
2020-08-01 16:07:47 +02:00
|
|
|
|
2020-07-31 09:41:22 +02:00
|
|
|
class MembershipRolesForm(forms.ModelForm):
|
|
|
|
user = forms.ModelChoiceField(
|
|
|
|
queryset=User.objects,
|
|
|
|
label=_("User"),
|
|
|
|
disabled=True,
|
|
|
|
widget=Autocomplete(
|
2020-08-01 16:07:47 +02:00
|
|
|
User,
|
|
|
|
attrs={
|
|
|
|
'api_url': '/api/user/',
|
|
|
|
'name_field': 'username',
|
|
|
|
'placeholder': 'Nom ...',
|
|
|
|
},
|
|
|
|
),
|
2020-07-31 09:41:22 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
roles = forms.ModelMultipleChoiceField(
|
|
|
|
queryset=Role.objects.filter(weirole=None).all(),
|
|
|
|
label=_("Roles"),
|
2020-08-03 13:33:25 +02:00
|
|
|
widget=CheckboxSelectMultiple(),
|
2020-07-31 09:41:22 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
model = Membership
|
2020-08-01 16:07:47 +02:00
|
|
|
fields = ('user', 'roles')
|