2020-06-07 21:01:54 +02:00
|
|
|
from django.contrib.auth.mixins import PermissionRequiredMixin
|
2020-07-21 14:47:07 +02:00
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
|
|
|
from django.http import HttpResponseRedirect
|
|
|
|
from django.views.generic.base import ContextMixin, TemplateResponseMixin, View
|
2020-06-07 21:01:54 +02:00
|
|
|
|
|
|
|
|
|
|
|
class StaffRequiredMixin(PermissionRequiredMixin):
|
2020-07-10 22:33:24 +02:00
|
|
|
permission_required = "bds.is_team"
|
2020-07-21 14:47:07 +02:00
|
|
|
|
|
|
|
|
|
|
|
class MultipleFormMixin(ContextMixin):
|
2020-08-28 18:00:54 +02:00
|
|
|
"""Mixin pour gérer plusieurs formulaires dans la même vue.
|
2020-07-21 14:47:07 +02:00
|
|
|
Le fonctionnement est relativement identique à celui de
|
2020-07-26 22:02:14 +02:00
|
|
|
FormMixin, dont la documentation est disponible ici :
|
|
|
|
https://docs.djangoproject.com/en/3.0/ref/class-based-views/mixins-editing/
|
|
|
|
|
|
|
|
Les principales différences sont :
|
|
|
|
- au lieu de form_class, il faut donner comme attribut un dict de la forme
|
|
|
|
{<form_name>: <form_class>}, avec tous les formulaires à instancier. On
|
|
|
|
peut aussi redéfinir `get_form_classes`
|
|
|
|
|
|
|
|
- les données initiales se récupèrent pour chaque form via l'attribut
|
|
|
|
`<form_name>_initial` ou la fonction `get_<form_name>_initial`. De même,
|
|
|
|
si certaines forms sont des `ModelForm`s, on peut définir la fonction
|
|
|
|
`get_<form_name>_instance`.
|
|
|
|
|
|
|
|
- chaque form a un préfixe rajouté, par défaut <form_name>, mais qui peut
|
|
|
|
être customisé via `prefixes` ou `get_prefixes`.
|
2020-07-21 14:47:07 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
form_classes = {}
|
|
|
|
prefixes = {}
|
|
|
|
initial = {}
|
|
|
|
|
|
|
|
success_url = None
|
|
|
|
|
|
|
|
def get_form_classes(self):
|
|
|
|
return self.form_classes
|
|
|
|
|
|
|
|
def get_initial(self, form_name):
|
|
|
|
initial_attr = "%s_initial" % form_name
|
|
|
|
|
|
|
|
initial_method = "get_%s_initial" % form_name
|
|
|
|
initial_method = getattr(self, initial_method, None)
|
|
|
|
|
|
|
|
if hasattr(self, initial_attr):
|
|
|
|
return getattr(self, initial_attr)
|
|
|
|
elif callable(initial_method):
|
|
|
|
return initial_method()
|
|
|
|
else:
|
|
|
|
return self.initial.copy()
|
|
|
|
|
|
|
|
def get_prefix(self, form_name):
|
|
|
|
return self.prefixes.get(form_name, form_name)
|
|
|
|
|
|
|
|
def get_instance(self, form_name):
|
|
|
|
# Au cas où certaines des forms soient des ModelForms
|
|
|
|
instance_method = "get_%s_instance" % form_name
|
|
|
|
instance_method = getattr(self, instance_method, None)
|
|
|
|
|
|
|
|
if callable(instance_method):
|
|
|
|
return instance_method()
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def get_form_kwargs(self, form_name):
|
|
|
|
kwargs = {
|
|
|
|
"initial": self.get_initial(form_name),
|
|
|
|
"prefix": self.get_prefix(form_name),
|
|
|
|
"instance": self.get_instance(form_name),
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.request.method in ("POST", "PUT"):
|
|
|
|
kwargs.update({"data": self.request.POST, "files": self.request.FILES})
|
|
|
|
|
|
|
|
return kwargs
|
|
|
|
|
|
|
|
def get_forms(self):
|
|
|
|
form_classes = self.get_form_classes()
|
|
|
|
return {
|
|
|
|
form_name: form_class(**self.get_form_kwargs(form_name))
|
|
|
|
for form_name, form_class in form_classes.items()
|
|
|
|
}
|
|
|
|
|
|
|
|
def get_success_url(self):
|
|
|
|
if not self.success_url:
|
|
|
|
raise ImproperlyConfigured("No URL to redirect to. Provide a success_url.")
|
|
|
|
return str(self.success_url)
|
|
|
|
|
|
|
|
def form_valid(self, forms):
|
|
|
|
# on garde le nom form_valid pour l'interface avec SuccessMessageMixin
|
|
|
|
return HttpResponseRedirect(self.get_success_url())
|
|
|
|
|
|
|
|
def form_invalid(self, forms):
|
|
|
|
"""If the form is invalid, render the invalid form."""
|
|
|
|
return self.render_to_response(self.get_context_data(forms=forms))
|
|
|
|
|
|
|
|
|
|
|
|
class ProcessMultipleFormView(View):
|
2020-08-28 18:00:54 +02:00
|
|
|
"""Équivalent de `ProcessFormView` pour plusieurs forms.
|
2020-07-26 22:09:52 +02:00
|
|
|
Note : il faut que *tous* les formulaires soient valides pour
|
|
|
|
qu'ils soient sauvegardés !
|
|
|
|
"""
|
|
|
|
|
2020-07-21 14:47:07 +02:00
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
forms = self.get_forms()
|
|
|
|
return self.render_to_response(self.get_context_data(forms=forms))
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
forms = self.get_forms()
|
|
|
|
if all(form.is_valid() for form in forms.values()):
|
|
|
|
return self.form_valid(forms)
|
|
|
|
else:
|
|
|
|
return self.form_invalid(forms)
|
|
|
|
|
|
|
|
|
|
|
|
class BaseMultipleFormView(MultipleFormMixin, ProcessMultipleFormView):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class MultipleFormView(TemplateResponseMixin, BaseMultipleFormView):
|
|
|
|
pass
|