Typo
[auf_savoirs_en_partage_django.git] / auf_savoirs_en_partage / chercheurs / forms.py
CommitLineData
932eef9a 1# -*- encoding: utf-8 -*-
a7b16ec9 2import hashlib
932eef9a 3from django import forms
3efbacbe 4from django.db.models import Q
ee8b3a49 5from django.forms.models import inlineformset_factory
fbcbdde0 6from itertools import chain
932eef9a 7from models import *
13146d99 8
c073c94d
EMS
9OUI_NON_CHOICES = (('1', 'Oui'), ('0', 'Non'))
10
932eef9a 11class ChercheurForm(forms.ModelForm):
a7b16ec9 12 """Formulaire d'édition d'un chercheur."""
13ec4813 13 genre = forms.ChoiceField(widget=forms.RadioSelect(), choices=GENRE_CHOICES)
a7b16ec9 14 membre_instance_auf = forms.ChoiceField(
a7b16ec9
EMS
15 label="Êtes-vous (ou avez-vous déjà été) membre d'une instance de l'AUF?",
16 help_text="e.g. conseil scientifique, conseil associatif, commission régionale d'experts",
c073c94d 17 choices=OUI_NON_CHOICES, widget=forms.RadioSelect()
a7b16ec9 18 )
614b3269 19 membre_instance_auf_details = forms.CharField(label="Préciser laquelle et votre fonction", required=False)
a7b16ec9 20 membre_instance_auf_dates = forms.CharField(label="Préciser les dates", required=False)
614b3269
EMS
21 expert_oif = forms.ChoiceField(label="Avez-vous déjà été sollicité par l'OIF?", choices=OUI_NON_CHOICES, widget=forms.RadioSelect())
22 expert_oif_details = forms.CharField(label="Préciser à quel titre", required=False,
23 help_text="Fonction dans l'organisation, participation à une étude ou à une action, etc.")
24 expert_oif_dates = forms.CharField(label="Préciser les dates", required=False)
c073c94d
EMS
25 membre_association_francophone = forms.ChoiceField(
26 label="Êtes-vous membre d'une association ou d'une société savante francophone?",
27 help_text="e.g. FIPF, Collège international de philosophie, AISLF, etc.",
28 choices=OUI_NON_CHOICES, widget=forms.RadioSelect()
29 )
30 membre_association_francophone_details = forms.CharField(label="Préciser laquelle", required=False)
31 membre_reseau_institutionnel = forms.ChoiceField(
32 label="Avez-vous fait partie des instances d'un réseau institutionnel de l'AUF?",
33 help_text="e.g. AFELSH, RIFFEF, CIDMEF, etc.",
34 choices=OUI_NON_CHOICES, widget=forms.RadioSelect()
35 )
36 membre_reseau_institutionnel_details = forms.CharField(required=False, label="Préciser lesquelles et votre fonction")
37 membre_reseau_institutionnel_dates = forms.CharField(required=False, label="Préciser les dates")
38
e836f6f7
EMS
39 pays_etablissement = forms.ModelChoiceField(label="Pays de l'établissement", queryset=Pays.objects.all(), required=True)
40 etablissement = forms.CharField(label="Nom de l'établissement", required=True)
a7b16ec9 41
cb591fb3
EMS
42 expertises_auf = forms.ChoiceField(
43 label="Êtes-vous disposé à réaliser des expertises pour l'AUF?",
44 choices=OUI_NON_CHOICES, widget=forms.RadioSelect()
45 )
46
332975c3
EMS
47 theme_recherche = forms.CharField(
48 max_length=1000, label='Thèmes de recherche', help_text='1000 signes maximum',
49 error_messages=dict(max_length="Veuillez entrer au maximum %(max)d signes (vous en avez entré %(length)d)."),
50 widget=forms.Textarea()
51 )
0874e7d1
EMS
52 attestation = forms.BooleanField(
53 required=True,
54 label="J'atteste sur l'honneur l'exactitude des renseignements fournis sur le formulaire d'inscription et j'accepte leur publication en ligne."
55 )
332975c3 56
932eef9a
AJ
57 class Meta:
58 model = Chercheur
13ec4813
EMS
59 fields = ('nom', 'prenom', 'courriel', 'genre',
60 'statut', 'diplome', 'discipline', 'theme_recherche',
219710da 61 'groupe_recherche', 'mots_cles', 'url_site_web',
0874e7d1
EMS
62 'url_blog', 'url_reseau_social', 'attestation',
63 'membre_instance_auf', 'membre_instance_auf_details',
219710da
EMS
64 'membre_instance_auf_dates', 'expert_oif',
65 'expert_oif_details', 'expert_oif_dates',
614b3269
EMS
66 'membre_association_francophone',
67 'membre_association_francophone_details',
219710da
EMS
68 'membre_reseau_institutionnel',
69 'membre_reseau_institutionnel_details',
cb591fb3 70 'membre_reseau_institutionnel_dates', 'expertises_auf')
7c596de2 71
219710da
EMS
72 def __init__(self, data=None, prefix=None, instance=None):
73 if instance is not None:
74 initial = {}
75 if instance.etablissement:
76 initial['etablissement'] = instance.etablissement.nom
77 initial['pays_etablissement'] = instance.etablissement.pays_id
78 else:
79 initial['etablissement'] = instance.etablissement_autre_nom
80 initial['pays_etablissement'] = instance.etablissement_autre_pays_id
81 else:
82 initial = None
83 super(ChercheurForm, self).__init__(data=data, prefix=prefix, instance=instance, initial=initial)
84
85 def save(self):
86 nom_etablissement = self.cleaned_data['etablissement']
87 pays_etablissement = self.cleaned_data['pays_etablissement']
88 try:
89 etablissement = Etablissement.objects.get(nom=nom_etablissement, pays=pays_etablissement)
90 self.instance.etablissement = etablissement
91 self.instance.etablissement_autre = ''
92 self.instance.etablissement_autre_pays = None
93 except Etablissement.DoesNotExist:
94 self.instance.etablissement = None
95 self.instance.etablissement_autre_nom = nom_etablissement
96 self.instance.etablissement_autre_pays = pays_etablissement
97 super(ChercheurForm, self).save()
98
13ec4813
EMS
99 def clean_courriel(self):
100 """On veut s'assurer qu'il n'y ait pas d'autre utilisateur actif
101 avec le même courriel."""
102 courriel = self.cleaned_data['courriel']
103 existing = Chercheur.objects.filter(courriel=courriel, actif=True)
104 if self.instance and self.instance.id:
105 existing = existing.exclude(id=self.instance.id)
106 if existing.count():
107 raise forms.ValidationError('Il existe déjà une fiche pour cette adresse électronique')
108 return courriel
109
c073c94d
EMS
110 def clean_membre_instance_auf(self):
111 return bool(int(self.cleaned_data['membre_instance_auf']))
112
614b3269
EMS
113 def clean_membre_instance_auf_details(self):
114 membre = self.cleaned_data.get('membre_instance_auf')
115 details = self.cleaned_data.get('membre_instance_auf_details')
116 if membre and not details:
117 raise forms.ValidationError('Veuillez préciser')
118 return details
119
c073c94d
EMS
120 def clean_membre_instance_auf_dates(self):
121 membre = self.cleaned_data.get('membre_instance_auf')
122 dates = self.cleaned_data.get('membre_instance_auf_dates')
123 if membre and not dates:
124 raise forms.ValidationError('Veuillez préciser les dates')
125 return dates
126
127 def clean_expert_oif(self):
128 return bool(int(self.cleaned_data['expert_oif']))
129
614b3269
EMS
130 def clean_expert_oif_details(self):
131 expert = self.cleaned_data.get('expert_oif')
132 details = self.cleaned_data.get('expert_oif_details')
133 if expert and not details:
134 raise forms.ValidationError('Veuillez préciser')
135 return details
136
137 def clean_expert_oif_dates(self):
138 expert = self.cleaned_data.get('expert_oif')
139 dates = self.cleaned_data.get('expert_oif_dates')
140 if expert and not dates:
141 raise forms.ValidationError('Veuillez préciser les dates')
142 return dates
143
c073c94d
EMS
144 def clean_membre_association_francophone(self):
145 return bool(int(self.cleaned_data['membre_association_francophone']))
146
147 def clean_membre_association_francophone_details(self):
148 membre = self.cleaned_data.get('membre_association_francophone')
149 details = self.cleaned_data.get('membre_association_francophone_details')
150 if membre and not details:
151 raise forms.ValidationError('Veuillez préciser')
152 return details
153
154 def clean_membre_reseau_institutionnel(self):
155 return bool(int(self.cleaned_data['membre_reseau_institutionnel']))
156
157 def clean_membre_reseau_institutionnel_details(self):
158 membre = self.cleaned_data.get('membre_reseau_institutionnel')
159 details = self.cleaned_data.get('membre_reseau_institutionnel_details')
160 if membre and not details:
161 raise forms.ValidationError('Veuillez préciser')
162 return details
163
164 def clean_membre_reseau_institutionnel_dates(self):
165 membre = self.cleaned_data.get('membre_reseau_institutionnel')
166 dates = self.cleaned_data.get('membre_reseau_institutionnel_dates')
167 if membre and not dates:
168 raise forms.ValidationError('Veuillez préciser les dates')
169 return dates
170
cb591fb3
EMS
171 def clean_expertises_auf(self):
172 return bool(int(self.cleaned_data['expertises_auf']))
173
13ec4813
EMS
174class ChercheurInscriptionForm(ChercheurForm):
175 password = forms.CharField(widget=forms.PasswordInput(), label="Mot de passe")
176 password_confirmation = forms.CharField(widget=forms.PasswordInput(), label="Confirmez votre mot de passe")
177
178 class Meta(ChercheurForm.Meta):
179 fields = ChercheurForm.Meta.fields + ('password',)
180
181 def clean_password_confirmation(self):
182 """S'assurer que le mot de passe et la confirmation sont identiques."""
183 password = self.cleaned_data.get('password')
184 confirmation = self.cleaned_data.get('password_confirmation')
185 if password != confirmation:
186 raise forms.ValidationError('Les deux mots de passe ne correspondent pas.')
187 return confirmation
188
189 def save(self):
518d0b44
EMS
190 super(ChercheurInscriptionForm, self).save()
191 self.instance.user.set_password(self.cleaned_data['password'])
192 self.instance.user.save()
13ec4813 193
a7b16ec9
EMS
194class GroupesForm(forms.Form):
195 """Formulaire qui associe des groupes à un chercheur."""
e4d01d1d
EMS
196 groupes = forms.ModelMultipleChoiceField(
197 queryset=Groupe.objects.all(),
198 label='Domaines de recherche', required=False,
199 help_text="Maintenez appuyé « Ctrl », ou « Commande (touche pomme) » sur un Mac, pour en sélectionner plusieurs."
200 )
a7b16ec9
EMS
201
202 def __init__(self, data=None, prefix=None, chercheur=None):
203 self.chercheur = chercheur
204 initial = {}
205 if chercheur:
206 initial['groupes'] = chercheur.groupes.values_list('id', flat=True)
207 super(GroupesForm, self).__init__(data=data, prefix=prefix, initial=initial)
208
209 def save(self):
210 if self.is_valid():
211 groupes = self.cleaned_data['groupes']
212 ChercheurGroupe.objects.filter(chercheur=self.chercheur).exclude(groupe__in=groupes).delete()
213 for g in groupes:
214 ChercheurGroupe.objects.get_or_create(chercheur=self.chercheur, groupe=g, actif=1)
215
00755d9b
AJ
216class PublicationForm(forms.ModelForm):
217 class Meta:
218 model = Publication
3eb41a6d 219 fields = ('auteurs', 'titre', 'revue', 'annee', 'editeur', 'lieu_edition', 'nb_pages', 'url')
f810842d 220
595ab4d6
EMS
221PublicationFormSet = inlineformset_factory(Chercheur, Publication, form=PublicationForm, extra=1)
222
223class TheseForm(forms.ModelForm):
e8e9e4fd 224 class Meta:
595ab4d6
EMS
225 model = These
226 fields = ('titre', 'annee', 'directeur', 'etablissement', 'nb_pages', 'url')
2a36714f 227
5b9abc81 228class ExpertiseForm(forms.ModelForm):
c073c94d
EMS
229 organisme_demandeur_visible = forms.ChoiceField(
230 label="Voulez-vous que l'organisme demandeur soit visible sur votre fiche?",
231 choices=OUI_NON_CHOICES, widget=forms.RadioSelect(), required=False
232 )
5b9abc81
AJ
233 class Meta:
234 model = Expertise
235 fields = ('nom', 'date', 'organisme_demandeur', 'organisme_demandeur_visible')
dab519fa 236
b16bcbaf 237 def clean_organisme_demandeur_visible(self):
6504a7e5
EMS
238 value = self.cleaned_data['organisme_demandeur_visible']
239 return bool(int(value)) if value else False
b16bcbaf 240
ee8b3a49
EMS
241ExpertiseFormSet = inlineformset_factory(Chercheur, Expertise, form=ExpertiseForm, extra=1)
242
a7b16ec9
EMS
243class ChercheurFormGroup(object):
244 """Groupe de formulaires nécessaires pour l'inscription et l'édition
245 d'un chercheur."""
00755d9b 246
a7b16ec9 247 def __init__(self, data=None, chercheur=None):
13ec4813 248 chercheur_form_class = ChercheurInscriptionForm if chercheur is None else ChercheurForm
e61eecfe
EMS
249 try:
250 these = chercheur and chercheur.these
251 except These.DoesNotExist:
252 these = These()
13ec4813 253 self.chercheur = chercheur_form_class(data=data, prefix='chercheur', instance=chercheur)
a7b16ec9 254 self.groupes = GroupesForm(data=data, prefix='chercheur', chercheur=chercheur)
ee8b3a49 255 self.expertises = ExpertiseFormSet(data=data, prefix='expertise', instance=chercheur)
e61eecfe 256 self.these = TheseForm(data=data, prefix='these', instance=these)
595ab4d6 257 self.publications = PublicationFormSet(data=data, prefix='publication', instance=chercheur)
a7b16ec9
EMS
258
259 @property
260 def has_errors(self):
13ec4813
EMS
261 return bool(self.chercheur.errors or self.groupes.errors or
262 self.these.errors or self.publications.errors or
263 self.expertises.errors)
a7b16ec9
EMS
264
265 def is_valid(self):
13ec4813
EMS
266 return self.chercheur.is_valid() and self.groupes.is_valid() and \
267 self.these.is_valid() and self.publications.is_valid() and \
268 self.expertises.is_valid()
a7b16ec9
EMS
269
270 def save(self):
271 if self.is_valid():
272
13ec4813 273 # Enregistrer d'abord le chercheur lui-même.
a7b16ec9
EMS
274 self.chercheur.save()
275
595ab4d6
EMS
276 # Puis les objets qui ont des clés étrangères vers nous
277 # puisqu'on a besoin d'un id.
13ec4813 278 chercheur = self.chercheur.instance
a7b16ec9
EMS
279 self.groupes.chercheur = chercheur
280 self.groupes.save()
595ab4d6
EMS
281 self.these.instance.chercheur = chercheur
282 self.these.save()
283 self.publications.instance = chercheur
284 self.publications.save()
3820b46e
EMS
285 self.expertises.instance = chercheur
286 self.expertises.save()
5b9abc81 287
13146d99 288class RepertoireSearchForm (forms.Form):
f0692c02 289 q = forms.CharField(required=False, label="Rechercher dans tous les champs")
3efbacbe 290 nom = forms.CharField(required=False, label="Nom")
0e9597af 291 domaine = forms.ModelChoiceField(queryset=Groupe.objects.all(), required=False, label="Domaine de recherche", empty_label="Tous")
bc415771
EMS
292 groupe_recherche = forms.CharField(required=False, label="Groupe de recherche",
293 help_text="ou Laboratoire, ou Groupement inter-universitaire")
6bd49ff1 294 statut = forms.ChoiceField(choices=(('','Tous'),)+STATUT_CHOICES+(('expert','Expert'),), required=False, label="Statut")
f0692c02 295 discipline = forms.ModelChoiceField(queryset=Discipline.objects.all(), required=False, label="Discipline", empty_label="Toutes")
3eba0476 296 pays = forms.ModelChoiceField(queryset=Pays.objects.all(), required=False, label="Pays", empty_label="Tous")
bc415771
EMS
297 region = forms.ModelChoiceField(queryset=Region.objects.all(), required=False, label="Région", empty_label="Toutes",
298 help_text="La région est ici définie au sens, non strictement géographique, du Bureau régional de l'AUF de référence.")
bc415771 299 nord_sud = forms.ChoiceField(choices=(('', 'Tous'), ('Nord', 'Nord'), ('Sud', 'Sud')), required=False, label="Nord/Sud",
5eecdf3e 300 help_text="Distinction d'ordre géopolitique et économique, non géographique, qui conditionne souvent l'attribution de soutiens par les agences internationales: on entend par Nord les pays développés, par Sud les pays en développement (pays les moins avancés, pays émergents et pays à économies en transition)")
c548d94a
EMS
301
302 def __init__(self, data=None, region=None):
303 super(RepertoireSearchForm, self).__init__(data)
304 if region:
305 pays = self.fields['pays']
306 pays.queryset = pays.queryset.filter(region=region)
307
3efbacbe 308 def get_query_set(self):
5212238e 309 chercheurs = Chercheur.objects
3efbacbe 310 if self.is_valid():
5212238e
EMS
311 q = self.cleaned_data["q"]
312 if q:
313 chercheurs = chercheurs.search(q)
3efbacbe
EMS
314 nom = self.cleaned_data['nom']
315 if nom:
5212238e 316 chercheurs = chercheurs.add_to_query('@(nom,prenom) ' + nom)
d087521a
EMS
317 groupe_recherche = self.cleaned_data['groupe_recherche']
318 if groupe_recherche:
5212238e 319 chercheurs = chercheurs.add_to_query('@groupe_recherche ' + groupe_recherche)
116db1fd
EMS
320 discipline = self.cleaned_data['discipline']
321 if discipline:
5212238e 322 chercheurs = chercheurs.filter_discipline(discipline)
116db1fd
EMS
323 region = self.cleaned_data['region']
324 if region:
5212238e
EMS
325 chercheurs = chercheurs.filter_region(region)
326 statut = self.cleaned_data["statut"]
327 if statut:
328 if statut == "expert":
329 chercheurs = chercheurs.filter_expert()
330 else:
331 chercheurs = chercheurs.filter_statut(statut)
332 domaine = self.cleaned_data["domaine"]
333 if domaine:
334 chercheurs = chercheurs.filter_groupe(domaine)
62eece34
EMS
335 pays = self.cleaned_data["pays"]
336 if pays:
5212238e 337 chercheurs = chercheurs.filter_pays(pays)
62eece34
EMS
338 nord_sud = self.cleaned_data['nord_sud']
339 if nord_sud:
5212238e
EMS
340 chercheurs = chercheurs.filter_nord_sud(nord_sud)
341 return chercheurs.all()
3efbacbe 342
0e9597af 343class SendPasswordForm(forms.Form):
89da853e 344 email = forms.EmailField(required=True, label="Adresse électronique")
0e9597af
AJ
345 def clean_email(self):
346 cleaned_data = self.cleaned_data
347 email = cleaned_data.get("email")
348 if email:
349 try:
13ec4813 350 Personne.objects.get(courriel=email)
0e9597af 351 except:
89da853e 352 raise forms.ValidationError("Cette adresse n'existe pas dans notre base de données.")
e427f068
AJ
353 return email
354
355class NewPasswordForm(forms.Form):
356 password = forms.CharField(widget=forms.PasswordInput(), required=True, label="Mot de passe")
357 password_repeat = forms.CharField(widget=forms.PasswordInput(), required=True, label="Confirmez mot de passe")
92990258 358
e427f068
AJ
359 def clean_password_repeat(self):
360 cleaned_data = self.cleaned_data
361 password = cleaned_data.get("password")
362 password_repeat = cleaned_data.get("password_repeat")
363 if password and password_repeat:
364 if password != password_repeat:
365 raise forms.ValidationError("Les mots de passe ne concordent pas")
366 return password_repeat
0e9597af 367