Commit | Line | Data |
---|---|---|
df59fcab | 1 | # -*- encoding: utf-8 -*- |
2 | ||
6067184b | 3 | from django.core.urlresolvers import reverse |
4 | from django.http import HttpResponseRedirect | |
df59fcab | 5 | from django.contrib import admin |
38df74bb | 6 | from django.shortcuts import get_object_or_404 |
7 | ||
6067184b | 8 | from reversion.admin import VersionAdmin |
38df74bb | 9 | from datamaster_modeles.models import Employe, Implantation, Region |
6067184b | 10 | |
df59fcab | 11 | from recrutement.models import * |
f6724c20 NBV |
12 | from recrutement.workflow import grp_administrateurs_recrutement,\ |
13 | grp_evaluateurs_recrutement, grp_drh_recrutement | |
df59fcab | 14 | |
d2b30f5f | 15 | class OffreEmploiAdmin(VersionAdmin): |
7f9e891e | 16 | date_hierarchy = 'date_creation' |
514db699 NBV |
17 | list_display = ('nom', 'resume', 'date_limite', 'statut', 'est_affiche', |
18 | 'region', '_candidatsList') | |
19 | list_filter = ('statut', 'est_affiche', ) | |
c4874d66 NBV |
20 | actions = ['affecter_evaluateurs_offre_emploi', ] |
21 | # Affecter un évaluateurs à des offres d'emploi | |
22 | def affecter_evaluateurs_offre_emploi(modeladmin, obj, candidats): | |
23 | selected = obj.POST.getlist(admin.ACTION_CHECKBOX_NAME) | |
24 | ||
25 | return HttpResponseRedirect(reverse('affecter_evaluateurs_offre_emploi')+ | |
26 | "?ids=%s" % (",".join(selected))) | |
27 | affecter_evaluateurs_offre_emploi.short_description = u'Affecter évaluateur(s)' | |
28 | ||
8941aee7 | 29 | # Afficher la liste des candidats pour l'offre d'emploi |
596fe324 | 30 | def _candidatsList(self, obj): |
8ea41642 | 31 | return "<a href='%s?offre_emploi__id__exact=%s'>Voir les candidats \ |
32 | </a>" % (reverse('admin:recrutement_candidat_changelist'), obj.id) | |
2f78949d | 33 | _candidatsList.allow_tags = True |
f6724c20 | 34 | _candidatsList.short_description = "Afficher la liste des candidats" |
362a3534 | 35 | |
c4874d66 | 36 | |
2f78949d | 37 | def queryset(self, request): |
382501c1 | 38 | qs = self.model._default_manager.get_query_set() |
f6724c20 NBV |
39 | # Si user est superuser afficher toutes les offres d'emploi |
40 | user_groupes = request.user.groups.all() | |
41 | if not grp_drh_recrutement in user_groupes: | |
beef7690 | 42 | """ |
beef7690 NBV |
43 | Si le user n'est ni un évaluateur ni un administrateur régional, |
44 | retourner none | |
45 | Vérifier groupes | |
46 | """ | |
f6724c20 | 47 | if grp_evaluateurs_recrutement in user_groupes: |
ca73c3c6 NBV |
48 | try: |
49 | user = Evaluateur.objects.get(user=request.user) | |
50 | except Evaluateur.DoesNotExist: | |
51 | return qs.none() | |
f6724c20 | 52 | elif grp_administrateurs_recrutement in user_groupes: |
ca73c3c6 NBV |
53 | try: |
54 | user = AdministrateurRegional.objects.get(user=request.user) | |
55 | except AdministrateurRegional.DoesNotExist: | |
56 | return qs.none() | |
f6724c20 NBV |
57 | else: |
58 | return qs.none() | |
59 | ||
60 | if type(user) is AdministrateurRegional: | |
61 | region_ids = [g.id for g in user.regions.all()] | |
62 | return qs.select_related('offre_emploi').\ | |
63 | filter(region__in=region_ids) | |
beef7690 | 64 | if type(user) is Evaluateur: |
f6724c20 | 65 | candidats = [g for g in user.candidats.all()] |
beef7690 | 66 | offre_emploi_ids = [c.offre_emploi.id for c in candidats] |
f6724c20 | 67 | return qs.select_related('offre_emploi').\ |
beef7690 | 68 | filter(id__in=offre_emploi_ids) |
3bcef02d | 69 | return qs.none() |
382501c1 | 70 | return qs.select_related('offre_emploi') |
f6724c20 NBV |
71 | |
72 | def has_change_permission(self, request, obj=None): | |
73 | user_groupes = request.user.groups.all() | |
74 | if grp_drh_recrutement in user_groupes or \ | |
75 | grp_administrateurs_recrutement in user_groupes: | |
76 | return True | |
77 | return False | |
78 | ||
79 | class ProxyOffreEmploiAdmin(OffreEmploiAdmin): | |
beef7690 | 80 | list_display = ('nom', 'resume', 'date_limite', 'region', ) |
65c4cbd9 | 81 | readonly_fields = ('description', 'poste', 'bureau', |
f6724c20 NBV |
82 | 'duree_affectation', 'renumeration', |
83 | 'debut_affectation', 'lieu_affectation', 'nom', | |
84 | 'resume', 'date_limite', 'region') | |
85 | fieldsets = ( | |
720c3ad5 | 86 | ('Nom', { |
f6724c20 NBV |
87 | 'fields': ('nom', ) |
88 | }), | |
720c3ad5 | 89 | ('Description générale', { |
f6724c20 NBV |
90 | 'fields': ('poste', 'resume','description', 'date_limite', ) |
91 | }), | |
720c3ad5 | 92 | ('Coordonnées', { |
f6724c20 NBV |
93 | 'fields': ('lieu_affectation', 'bureau', 'region', ) |
94 | }), | |
720c3ad5 | 95 | ('Autre', { |
f6724c20 NBV |
96 | 'fields': ('debut_affectation', 'duree_affectation', |
97 | 'renumeration', ) | |
98 | }), | |
99 | ) | |
100 | def has_add_permission(self, request): | |
101 | return False | |
102 | ||
103 | def has_delete_permission(self, request, obj=None): | |
104 | return False | |
105 | ||
2d083449 NBV |
106 | def has_change_permission(self, request, obj=None): |
107 | user_groupes = request.user.groups.all() | |
beef7690 NBV |
108 | if grp_evaluateurs_recrutement in user_groupes or \ |
109 | grp_drh_recrutement in user_groupes: | |
2d083449 NBV |
110 | return True |
111 | return False | |
112 | ||
cced6a23 | 113 | class ProxyCandidatPiece(CandidatPiece): |
114 | """ | |
115 | Ce proxy sert uniquement dans l'admin à disposer d'un libellé | |
116 | plus ergonomique. | |
117 | """ | |
118 | class Meta: | |
119 | proxy = True | |
120 | verbose_name = "pièce jointe" | |
74cbc7a7 | 121 | verbose_name_plural = "pièces jointes" |
cced6a23 | 122 | |
170c9aa2 | 123 | class CandidatPieceInline(admin.TabularInline): |
cced6a23 | 124 | model = ProxyCandidatPiece |
125 | fields = ('candidat', 'nom', 'path', ) | |
170c9aa2 | 126 | extra = 1 |
127 | ||
27c81d11 | 128 | class ProxyEvaluateur(Evaluateur.candidats.through): |
cced6a23 | 129 | """ |
130 | Ce proxy sert uniquement dans l'admin à disposer d'un libellé | |
131 | plus ergonomique. | |
132 | """ | |
133 | class Meta: | |
134 | proxy = True | |
135 | verbose_name = "évaluateur" | |
136 | ||
eb579d40 | 137 | class EvaluateurInline(admin.TabularInline): |
cced6a23 | 138 | model = ProxyEvaluateur |
720c3ad5 | 139 | fields = ('evaluateur',) |
eb579d40 | 140 | extra = 1 |
141 | ||
d2b30f5f | 142 | class CandidatAdmin(VersionAdmin): |
7f9e891e | 143 | date_hierarchy = 'date_creation' |
0fd8a26d | 144 | list_display = ('nom', 'prenom', 'offre_emploi','statut', |
720c3ad5 | 145 | 'voir_offre_emploi', #'note_evaluateur', |
8941aee7 | 146 | 'calculer_moyenne', 'afficher_candidat',) |
8ea41642 | 147 | list_filter = ('offre_emploi', ) |
7f9e891e | 148 | fieldsets = ( |
4896b661 | 149 | ("Offre d'emploi", { |
150 | 'fields': ('offre_emploi', ) | |
151 | }), | |
7f9e891e | 152 | ('Informations personnelles', { |
153 | 'fields': ('prenom','nom','genre', 'nationalite', 'date_naissance', | |
154 | 'situation_famille', 'nombre_dependant',) | |
155 | }), | |
ec517164 | 156 | ('Coordonnées', { |
157 | 'fields': ('telephone', 'email', 'adresse', 'ville', | |
158 | 'etat_province', 'code_postal', 'pays', ) | |
7f9e891e | 159 | }), |
160 | ('Informations professionnelles', { | |
4896b661 | 161 | 'fields': ('niveau_diplome','employeur_actuel', |
8ea41642 | 162 | 'poste_actuel', 'domaine_professionnel',) |
7f9e891e | 163 | }), |
65c4cbd9 NBV |
164 | ('Traitement', { |
165 | 'fields': ('statut', ) | |
7f9e891e | 166 | }), |
167 | ) | |
170c9aa2 | 168 | inlines = [ |
169 | CandidatPieceInline, | |
eb579d40 | 170 | EvaluateurInline, |
170c9aa2 | 171 | ] |
f9983b5a | 172 | |
4e8340cf | 173 | actions = ['affecter_candidats_evaluateur', 'envoyer_courriel_candidats'] |
362a3534 | 174 | # Affecter un évaluateurs à des candidats |
7061f835 | 175 | def affecter_candidats_evaluateur(modeladmin, obj, candidats): |
596fe324 | 176 | selected = obj.POST.getlist(admin.ACTION_CHECKBOX_NAME) |
2adf9e0c | 177 | |
8ea41642 | 178 | return HttpResponseRedirect(reverse('affecter_evaluateurs_candidats')+ |
179 | "?ids=%s" % (",".join(selected))) | |
c4874d66 | 180 | affecter_candidats_evaluateur.short_description = u'Affecter évaluateur(s)' |
362a3534 | 181 | |
4e8340cf | 182 | # Envoyer un courriel à des candidats |
52765380 | 183 | def envoyer_courriel_candidats(modeladmin, obj, candidats): |
184 | selected = obj.POST.getlist(admin.ACTION_CHECKBOX_NAME) | |
185 | ||
186 | return HttpResponseRedirect(reverse('envoyer_courriel_candidats')+ | |
187 | "?ids=%s" % (",".join(selected))) | |
188 | envoyer_courriel_candidats.short_description = u'Envoyer courriel' | |
189 | ||
05503d56 | 190 | # Évaluer un candidat |
596fe324 | 191 | def evaluer_candidat(self, obj): |
beef7690 NBV |
192 | return "<a href='%s?candidat__id__exact=%s'>Évaluer le candidat </a>" % \ |
193 | (reverse('admin:recrutement_candidatevaluation_changelist'), | |
194 | obj.id) | |
596fe324 | 195 | evaluer_candidat.allow_tags = True |
beef7690 | 196 | evaluer_candidat.short_description = 'Évaluation' |
596fe324 | 197 | |
7d82fd33 | 198 | # Afficher un candidat |
199 | def afficher_candidat(self, obj): | |
382501c1 NBV |
200 | return "<a href='%s'>Voir le candidat</a>" % \ |
201 | (reverse('admin:recrutement_proxycandidat_change', args=(obj.id,))) | |
7d82fd33 | 202 | afficher_candidat.allow_tags = True |
f6724c20 | 203 | afficher_candidat.short_description = u'Afficher les détails du candidat' |
7d82fd33 | 204 | |
8941aee7 | 205 | # Voir l'offre d'emploi |
206 | def voir_offre_emploi(self, obj): | |
720c3ad5 NBV |
207 | return "<a href='%s'>Voir l'offre d'emploi</a>" % \ |
208 | (reverse('admin:recrutement_proxyoffreemploi_change', | |
209 | args=(obj.offre_emploi.id,))) | |
8941aee7 | 210 | voir_offre_emploi.allow_tags = True |
211 | voir_offre_emploi.short_description = "Afficher l'offre d'emploi" | |
212 | ||
8941aee7 | 213 | # Calculer la moyenne des notes |
214 | def calculer_moyenne(self, obj): | |
215 | evaluations = CandidatEvaluation.objects.filter(candidat=obj) | |
216 | offre_emploi = obj.offre_emploi | |
217 | ||
f6724c20 NBV |
218 | notes = [evaluation.note for evaluation in evaluations.all() \ |
219 | if evaluation.note is not None] | |
8941aee7 | 220 | |
221 | if len(notes) > 0 and offre_emploi.date_limite <= datetime.date.today(): | |
222 | moyenne_votes = float(sum(notes)) / len(notes) | |
223 | else: | |
224 | moyenne_votes = "Non disponible" | |
225 | return moyenne_votes | |
226 | calculer_moyenne.allow_tags = True | |
227 | calculer_moyenne.short_description = "Moyenne des notes" | |
228 | ||
f6724c20 NBV |
229 | def add_delete_permission(self, request, obj=None) : |
230 | user_groupes = request.user.groups.all() | |
231 | if grp_drh_recrutement in user_groupes or \ | |
232 | grp_administrateurs_recrutement in user_groupes: | |
233 | return True | |
234 | return False | |
4896b661 | 235 | |
f6724c20 NBV |
236 | def has_add_permission(self, request): |
237 | return self.add_delete_permission(request, request) | |
4896b661 | 238 | |
f6724c20 NBV |
239 | def has_delete_permission(self, request, obj=None): |
240 | return self.add_delete_permission(request, request) | |
241 | ||
242 | def has_change_permission(self, request, obj=None): | |
243 | user_groupes = request.user.groups.all() | |
244 | if grp_drh_recrutement in user_groupes or \ | |
2d083449 | 245 | grp_administrateurs_recrutement in user_groupes: |
f6724c20 NBV |
246 | return True |
247 | return False | |
4896b661 | 248 | |
596fe324 | 249 | def queryset(self, obj): |
f9983b5a | 250 | """ |
8ea41642 | 251 | Spécifie un queryset limité, autrement Django exécute un |
252 | select_related() sans paramètre, ce qui a pour effet de charger tous | |
253 | les objets FK, sans limite de profondeur. Dès qu'on arrive, dans les | |
254 | modèles de Region, il existe plusieurs boucles, ce qui conduit à la | |
255 | génération d'une requête infinie. | |
d835c9f3 | 256 | |
f9983b5a | 257 | """ |
f6724c20 NBV |
258 | qs = self.model._default_manager.get_query_set() |
259 | # Si user est superuser afficher tous les candidats | |
260 | user_groupes = obj.user.groups.all() | |
261 | if not grp_drh_recrutement in user_groupes: | |
262 | # Si le user n'est ni un évaluateur ni un administrateur régional, | |
263 | # retourner none | |
264 | ||
265 | # Vérifier groupes | |
266 | if grp_evaluateurs_recrutement in user_groupes: | |
ca73c3c6 NBV |
267 | try: |
268 | user = Evaluateur.objects.get(user=obj.user) | |
269 | except Evaluateur.DoesNotExist: | |
270 | return qs.none() | |
efedc086 NBV |
271 | """ |
272 | elif grp_administrateurs_recrutement in user_groupes: | |
273 | try: | |
274 | user = AdministrateurRegional.objects.get(user=obj.user) | |
275 | except AdministrateurRegional.DoesNotExist: | |
276 | return qs.none() | |
277 | """ | |
f6724c20 NBV |
278 | else: |
279 | return qs.none() | |
f6724c20 NBV |
280 | ids = [c.id for c in user.candidats.all()] |
281 | return qs.select_related('candidats').filter(id__in=ids) | |
282 | return qs.select_related('candidats') | |
283 | ||
284 | class ProxyCandidatAdmin(CandidatAdmin): | |
f6724c20 NBV |
285 | readonly_fields = ('statut', 'offre_emploi', 'prenom', 'nom', |
286 | 'genre', 'nationalite', 'date_naissance', | |
287 | 'situation_famille', 'nombre_dependant', 'telephone', | |
288 | 'email', 'adresse', 'ville', 'etat_province', | |
289 | 'code_postal', 'pays', 'niveau_diplome', | |
290 | 'employeur_actuel', 'poste_actuel', | |
291 | 'domaine_professionnel',) | |
2d083449 NBV |
292 | fieldsets = ( |
293 | ("Offre d'emploi", { | |
294 | 'fields': ('offre_emploi', ) | |
295 | }), | |
296 | ('Informations personnelles', { | |
297 | 'fields': ('prenom','nom','genre', 'nationalite', 'date_naissance', | |
298 | 'situation_famille', 'nombre_dependant',) | |
299 | }), | |
300 | ('Coordonnées', { | |
301 | 'fields': ('telephone', 'email', 'adresse', 'ville', | |
302 | 'etat_province', 'code_postal', 'pays', ) | |
303 | }), | |
304 | ('Informations professionnelles', { | |
305 | 'fields': ('niveau_diplome','employeur_actuel', | |
306 | 'poste_actuel', 'domaine_professionnel',) | |
307 | }), | |
308 | ) | |
f6724c20 | 309 | inlines = [] |
2d083449 | 310 | |
f6724c20 NBV |
311 | def has_add_permission(self, request): |
312 | return False | |
313 | ||
314 | def has_delete_permission(self, request, obj=None): | |
315 | return False | |
2adf9e0c | 316 | |
2d083449 NBV |
317 | def has_change_permission(self, request, obj=None): |
318 | user_groupes = request.user.groups.all() | |
beef7690 NBV |
319 | if grp_drh_recrutement in user_groupes or \ |
320 | grp_administrateurs_recrutement in user_groupes or \ | |
321 | grp_evaluateurs_recrutement in user_groupes: | |
2d083449 NBV |
322 | return True |
323 | return False | |
324 | ||
2e9ee615 | 325 | class CandidatPieceAdmin(admin.ModelAdmin): |
170c9aa2 | 326 | list_display = ('nom', 'candidat', ) |
327 | ||
328 | def queryset(self, request): | |
329 | """ | |
330 | Spécifie un queryset limité, autrement Django exécute un | |
331 | select_related() sans paramètre, ce qui a pour effet de charger tous | |
332 | les objets FK, sans limite de profondeur. Dès qu'on arrive, dans les | |
333 | modèles de Region, il existe plusieurs boucles, ce qui conduit à la | |
334 | génération d'une requête infinie. | |
d835c9f3 | 335 | Affiche la liste de candidats que si le user connecté |
27c81d11 | 336 | possède un Evaluateur |
170c9aa2 | 337 | """ |
338 | qs = self.model._default_manager.get_query_set() | |
339 | return qs.select_related('candidat') | |
2e9ee615 | 340 | |
d2b30f5f | 341 | class EvaluateurAdmin(VersionAdmin): |
eb579d40 | 342 | fieldsets = ( |
f6724c20 | 343 | (None, {'fields': ('user', )}), |
720c3ad5 | 344 | #(None, {'fields': ('candidats',)}), |
eb579d40 | 345 | ) |
4418c732 | 346 | |
b89fef74 | 347 | class AdministrateurRegionalAdmin(VersionAdmin): |
27c81d11 | 348 | pass |
349 | ||
d2b30f5f | 350 | class CandidatEvaluationAdmin(VersionAdmin): |
b903198b NBV |
351 | list_display = ('_offre_emploi', '_candidat', 'evaluateur', '_note', |
352 | '_commentaire', ) | |
beef7690 NBV |
353 | readonly_fields = ('candidat', 'evaluateur') |
354 | fieldsets = ( | |
355 | ('Évaluation du candidat', { | |
356 | 'fields': ('candidat', 'evaluateur', 'note', 'commentaire', ) | |
357 | }), | |
358 | ) | |
359 | ||
360 | def _note(self, obj): | |
361 | """ | |
362 | Si l'évaluateur n'a pas encore donné de note au candidat, indiquer | |
363 | un lien pour Évaluer le candidat. | |
364 | Sinon afficher la note. | |
365 | """ | |
b903198b NBV |
366 | evaluateur = obj.evaluateur |
367 | candidat = obj.candidat | |
368 | candidat_evaluation = CandidatEvaluation.objects.\ | |
369 | get(candidat=candidat, evaluateur=evaluateur) | |
beef7690 | 370 | if obj.note is None: |
b903198b | 371 | return "<a href='%s'>Candidat non évalué</a>" % \ |
beef7690 | 372 | (reverse('admin:recrutement_candidatevaluation_change', |
b903198b NBV |
373 | args=(candidat_evaluation.id,))) |
374 | return "<a href='%s'>%s</a>" % \ | |
375 | (reverse('admin:recrutement_candidatevaluation_change', | |
376 | args=(candidat_evaluation.id,)), obj.note) | |
377 | return | |
beef7690 NBV |
378 | _note.allow_tags = True |
379 | _note.short_description = "Votre note" | |
380 | _note.admin_order_field = 'note' | |
381 | ||
382 | def _candidat(self, obj): | |
383 | return "<a href='%s'>%s</a>" \ | |
384 | % (reverse('admin:recrutement_proxycandidat_change', | |
385 | args=(obj.candidat.id,)), obj.candidat) | |
386 | _candidat.allow_tags = True | |
387 | _candidat.short_description = 'Candidat' | |
388 | ||
389 | def _commentaire(self, obj): | |
390 | """ | |
391 | Si l'évaluateur n'a pas encore donné de note au candidat, indiquer | |
392 | dans le champ commentaire, Aucun au lieu de (None) | |
393 | Sinon afficher la note. | |
394 | """ | |
395 | if obj.commentaire is None: | |
396 | return "Aucun" | |
397 | return obj.commentaire | |
398 | _commentaire.allow_tags = True | |
399 | _commentaire.short_description = "Commentaire" | |
720c3ad5 | 400 | |
720c3ad5 | 401 | |
beef7690 NBV |
402 | def _offre_emploi(self, obj): |
403 | return "<a href='%s'>%s</a>" % \ | |
404 | (reverse('admin:recrutement_proxyoffreemploi_change', | |
405 | args=(obj.candidat.offre_emploi.id,)), obj.candidat.offre_emploi) | |
406 | _offre_emploi.allow_tags = True | |
407 | _offre_emploi.short_description = "Voir offre d'emploi" | |
408 | _offre_emploi.admin_order_field = 'offre_emploi' | |
409 | ||
21b02da5 NBV |
410 | def has_change_permission(self, request, obj=None): |
411 | """ | |
412 | Permettre la visualisation dans la changelist | |
413 | mais interdire l'accès à modifier l'objet si l'évaluateur n'est pas | |
414 | le request.user | |
415 | """ | |
416 | return obj is None or request.user == obj.evaluateur.user | |
417 | ||
720c3ad5 | 418 | def queryset(self, request): |
beef7690 NBV |
419 | """ |
420 | Afficher uniquement les évaluations de l'évaluateur, sauf si | |
421 | l'utilisateur est super admin. | |
422 | """ | |
720c3ad5 | 423 | qs = self.model._default_manager.get_query_set() |
beef7690 NBV |
424 | user_groupes = request.user.groups.all() |
425 | if grp_drh_recrutement in user_groupes: | |
426 | return qs.select_related('offre_emploi') | |
427 | ||
ca73c3c6 | 428 | try: |
d069cdf1 | 429 | evaluateur = Evaluateur.objects.get(user=request.user) |
ca73c3c6 NBV |
430 | except Evaluateur.DoesNotExist: |
431 | return qs.none() | |
432 | ||
433 | candidats_evaluations = CandidatEvaluation.objects.\ | |
434 | filter(evaluateur=evaluateur) | |
435 | candidats_evaluations_ids = [ce.id for ce in \ | |
436 | candidats_evaluations.all()] | |
437 | return qs.select_related('offre_emploi').\ | |
438 | filter(id__in=candidats_evaluations_ids) | |
596fe324 | 439 | |
4e8340cf | 440 | class CourrielTemplateAdmin(VersionAdmin): |
441 | pass | |
442 | ||
df59fcab | 443 | admin.site.register(OffreEmploi, OffreEmploiAdmin) |
382501c1 | 444 | admin.site.register(ProxyOffreEmploi, ProxyOffreEmploiAdmin) |
df59fcab | 445 | admin.site.register(Candidat, CandidatAdmin) |
382501c1 | 446 | admin.site.register(ProxyCandidat, ProxyCandidatAdmin) |
720c3ad5 | 447 | admin.site.register(CandidatEvaluation, CandidatEvaluationAdmin) |
27c81d11 | 448 | admin.site.register(Evaluateur, EvaluateurAdmin) |
beef7690 | 449 | #admin.site.register(AdministrateurRegional, AdministrateurRegionalAdmin) |
382501c1 | 450 | #admin.site.register(CourrielTemplate, CourrielTemplateAdmin) |