1677: Changer les informations concernant le SMTP + composer message. 1655: Envois...
[auf_rh_dae.git] / project / recrutement / admin.py
CommitLineData
df59fcab 1# -*- encoding: utf-8 -*-
2
6067184b 3from django.core.urlresolvers import reverse
4from django.http import HttpResponseRedirect
df59fcab 5from django.contrib import admin
38df74bb 6from django.shortcuts import get_object_or_404
7
6067184b 8from reversion.admin import VersionAdmin
38df74bb 9from datamaster_modeles.models import Employe, Implantation, Region
6067184b 10
df59fcab 11from recrutement.models import *
f6724c20
NBV
12from recrutement.workflow import grp_administrateurs_recrutement,\
13 grp_evaluateurs_recrutement, grp_drh_recrutement
df59fcab 14
d2b30f5f 15class 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
79class 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 113class 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 123class CandidatPieceInline(admin.TabularInline):
cced6a23 124 model = ProxyCandidatPiece
125 fields = ('candidat', 'nom', 'path', )
170c9aa2 126 extra = 1
127
27c81d11 128class 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 137class EvaluateurInline(admin.TabularInline):
cced6a23 138 model = ProxyEvaluateur
720c3ad5 139 fields = ('evaluateur',)
eb579d40 140 extra = 1
141
d2b30f5f 142class 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
32834000 186 return HttpResponseRedirect(reverse('selectionner_template')+
52765380 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
284class 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 325class 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 341class EvaluateurAdmin(VersionAdmin):
eb579d40 342 fieldsets = (
f6724c20 343 (None, {'fields': ('user', )}),
720c3ad5 344 #(None, {'fields': ('candidats',)}),
eb579d40 345 )
4418c732 346
b89fef74 347class AdministrateurRegionalAdmin(VersionAdmin):
27c81d11 348 pass
349
d2b30f5f 350class 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 440class CourrielTemplateAdmin(VersionAdmin):
441 pass
442
df59fcab 443admin.site.register(OffreEmploi, OffreEmploiAdmin)
382501c1 444admin.site.register(ProxyOffreEmploi, ProxyOffreEmploiAdmin)
df59fcab 445admin.site.register(Candidat, CandidatAdmin)
382501c1 446admin.site.register(ProxyCandidat, ProxyCandidatAdmin)
720c3ad5 447admin.site.register(CandidatEvaluation, CandidatEvaluationAdmin)
27c81d11 448admin.site.register(Evaluateur, EvaluateurAdmin)
beef7690 449#admin.site.register(AdministrateurRegional, AdministrateurRegionalAdmin)
32834000 450admin.site.register(CourrielTemplate, CourrielTemplateAdmin)