Début du tableau de bord des évaluations pour les Évaluateurs
[auf_rh_dae.git] / project / recrutement / admin.py
1 # -*- encoding: utf-8 -*-
2
3 from django.core.urlresolvers import reverse
4 from django.http import HttpResponseRedirect
5 from django.contrib import admin
6 from django.shortcuts import get_object_or_404
7
8 from reversion.admin import VersionAdmin
9 from datamaster_modeles.models import Employe, Implantation, Region
10
11 from recrutement.models import *
12 from recrutement.workflow import grp_administrateurs_recrutement,\
13 grp_evaluateurs_recrutement, grp_drh_recrutement
14
15 class OffreEmploiAdmin(VersionAdmin):
16 date_hierarchy = 'date_creation'
17 list_display = ('nom', 'resume', 'date_limite', 'region', '_candidatsList')
18
19 # Afficher la liste des candidats pour l'offre d'emploi
20 def _candidatsList(self, obj):
21 return "<a href='%s?offre_emploi__id__exact=%s'>Voir les candidats \
22 </a>" % (reverse('admin:recrutement_candidat_changelist'), obj.id)
23 _candidatsList.allow_tags = True
24 _candidatsList.short_description = "Afficher la liste des candidats"
25
26 def queryset(self, request):
27 qs = self.model._default_manager.get_query_set()
28 # Si user est superuser afficher toutes les offres d'emploi
29 user_groupes = request.user.groups.all()
30 if not grp_drh_recrutement in user_groupes:
31 # Si le user n'est ni un évaluateur ni un administrateur régional,
32 # retourner none
33 # Vérifier groupes
34 if grp_evaluateurs_recrutement in user_groupes:
35 user = Evaluateur.objects.get(user=request.user)
36 elif grp_administrateurs_recrutement in user_groupes:
37 user = AdministrateurRegional.objects.get(user=request.user)
38 else:
39 return qs.none()
40
41 if type(user) is AdministrateurRegional:
42 region_ids = [g.id for g in user.regions.all()]
43 return qs.select_related('offre_emploi').\
44 filter(region__in=region_ids)
45 if type(user) is Evaluateur:
46
47 candidats = [g for g in user.candidats.all()]
48 offre_emploi_ids = [offre.id for offre in candidats]
49 return qs.select_related('offre_emploi').\
50 filter(id__in=offre_emploi_ids)
51
52 return qs.none()
53 return qs.select_related('offre_emploi')
54
55 def has_change_permission(self, request, obj=None):
56 user_groupes = request.user.groups.all()
57 if grp_drh_recrutement in user_groupes or \
58 grp_administrateurs_recrutement in user_groupes:
59 return True
60 return False
61
62 class ProxyOffreEmploiAdmin(OffreEmploiAdmin):
63 list_display = ('nom', 'resume', 'date_limite', 'region', '_candidatsList', )
64 readonly_fields = ('actif', 'description', 'poste', 'bureau',
65 'duree_affectation', 'renumeration',
66 'debut_affectation', 'lieu_affectation', 'nom',
67 'resume', 'date_limite', 'region')
68 fieldsets = (
69 ('Nom', {
70 'fields': ('nom', )
71 }),
72 ('Description générale', {
73 'fields': ('poste', 'resume','description', 'date_limite', )
74 }),
75 ('Coordonnées', {
76 'fields': ('lieu_affectation', 'bureau', 'region', )
77 }),
78 ('Autre', {
79 'fields': ('debut_affectation', 'duree_affectation',
80 'renumeration', )
81 }),
82 )
83 def has_add_permission(self, request):
84 return False
85
86 def has_delete_permission(self, request, obj=None):
87 return False
88
89 def has_change_permission(self, request, obj=None):
90 user_groupes = request.user.groups.all()
91 if grp_evaluateurs_recrutement in user_groupes:
92 return True
93 return False
94
95 def _candidatsList(self, obj):
96 return "<a href='%s?offre_emploi__id__exact=%s'>Voir les candidats \
97 </a>" % (reverse('admin:recrutement_proxycandidat_changelist'),
98 obj.id)
99 _candidatsList.allow_tags = True
100 _candidatsList.short_description = "Afficher la liste des candidats"
101
102 class ProxyCandidatPiece(CandidatPiece):
103 """
104 Ce proxy sert uniquement dans l'admin à disposer d'un libellé
105 plus ergonomique.
106 """
107 class Meta:
108 proxy = True
109 verbose_name = "pièce jointe"
110 verbose_name_plural = "pièces jointes"
111
112 class CandidatPieceInline(admin.TabularInline):
113 model = ProxyCandidatPiece
114 fields = ('candidat', 'nom', 'path', )
115 extra = 1
116
117 class ProxyEvaluateur(Evaluateur.candidats.through):
118 """
119 Ce proxy sert uniquement dans l'admin à disposer d'un libellé
120 plus ergonomique.
121 """
122 class Meta:
123 proxy = True
124 verbose_name = "évaluateur"
125
126 class EvaluateurInline(admin.TabularInline):
127 model = ProxyEvaluateur
128 fields = ('evaluateur',)
129 extra = 1
130
131 class CandidatAdmin(VersionAdmin):
132 date_hierarchy = 'date_creation'
133 list_display = ('nom', 'prenom', 'offre_emploi','statut',
134 'voir_offre_emploi', #'note_evaluateur',
135 'calculer_moyenne', 'afficher_candidat',)
136 list_filter = ('offre_emploi', )
137 fieldsets = (
138 ("Offre d'emploi", {
139 'fields': ('offre_emploi', )
140 }),
141 ('Informations personnelles', {
142 'fields': ('prenom','nom','genre', 'nationalite', 'date_naissance',
143 'situation_famille', 'nombre_dependant',)
144 }),
145 ('Coordonnées', {
146 'fields': ('telephone', 'email', 'adresse', 'ville',
147 'etat_province', 'code_postal', 'pays', )
148 }),
149 ('Informations professionnelles', {
150 'fields': ('niveau_diplome','employeur_actuel',
151 'poste_actuel', 'domaine_professionnel',)
152 }),
153 ('Options avancées', {
154 'classes': ('collapse',),
155 'fields': ('actif', 'statut', )
156 }),
157 )
158 inlines = [
159 CandidatPieceInline,
160 EvaluateurInline,
161 ]
162
163 actions = ['affecter_candidats_evaluateur', 'envoyer_courriel_candidats']
164 # Affecter un évaluateurs à des candidats
165 def affecter_candidats_evaluateur(modeladmin, obj, candidats):
166 selected = obj.POST.getlist(admin.ACTION_CHECKBOX_NAME)
167
168 return HttpResponseRedirect(reverse('affecter_evaluateurs_candidats')+
169 "?ids=%s" % (",".join(selected)))
170 affecter_candidats_evaluateur.short_description = u'Affecter évaluateur'
171
172 # Envoyer un courriel à des candidats
173 def envoyer_courriel_candidats(modeladmin, obj, candidats):
174 selected = obj.POST.getlist(admin.ACTION_CHECKBOX_NAME)
175
176 return HttpResponseRedirect(reverse('envoyer_courriel_candidats')+
177 "?ids=%s" % (",".join(selected)))
178 envoyer_courriel_candidats.short_description = u'Envoyer courriel'
179
180 # Évaluer un candidat
181 def evaluer_candidat(self, obj):
182 return "<a href='%s?id=%s'>Évaluer le candidat \
183 </a>" % (reverse('evaluer_candidat'), obj.id)
184 evaluer_candidat.allow_tags = True
185 evaluer_candidat.short_description = 'Note du candidat'
186
187 # Afficher un candidat
188 def afficher_candidat(self, obj):
189 return "<a href='%s'>Voir le candidat </a>" \
190 % (reverse('admin:recrutement_proxycandidat_change', args=(obj.id,)))
191 afficher_candidat.allow_tags = True
192 afficher_candidat.short_description = u'Afficher les détails du candidat'
193
194 # Voir l'offre d'emploi
195 def voir_offre_emploi(self, obj):
196 return "<a href='%s'>Voir l'offre d'emploi</a>" % \
197 (reverse('admin:recrutement_proxyoffreemploi_change',
198 args=(obj.offre_emploi.id,)))
199 voir_offre_emploi.allow_tags = True
200 voir_offre_emploi.short_description = "Afficher l'offre d'emploi"
201
202 """ # Afficher note poster par l'évaluateur
203 def note_evaluateur(self, obj):
204 import pdb;pdb.set_trace()
205 evaluateur = Evaluateur.objects.get(user=obj.user)
206 import pdb;pdb.set_trace()
207 note_eval = CandidatEvaluation.objects.filter(evaluateur=evaluateur,
208 candidat=obj.user)
209 return "blabla"
210 note_evaluateur.allow_tags = True
211 note_evaluateur.short_description = "Votre note"
212 """
213
214 # Calculer la moyenne des notes
215 def calculer_moyenne(self, obj):
216 evaluations = CandidatEvaluation.objects.filter(candidat=obj)
217 offre_emploi = obj.offre_emploi
218
219 notes = [evaluation.note for evaluation in evaluations.all() \
220 if evaluation.note is not None]
221
222 if len(notes) > 0 and offre_emploi.date_limite <= datetime.date.today():
223 moyenne_votes = float(sum(notes)) / len(notes)
224 else:
225 moyenne_votes = "Non disponible"
226 return moyenne_votes
227 calculer_moyenne.allow_tags = True
228 calculer_moyenne.short_description = "Moyenne des notes"
229
230 def add_delete_permission(self, request, obj=None) :
231 user_groupes = request.user.groups.all()
232 if grp_drh_recrutement in user_groupes or \
233 grp_administrateurs_recrutement in user_groupes:
234 return True
235 return False
236
237 def has_add_permission(self, request):
238 return self.add_delete_permission(request, request)
239
240 def has_delete_permission(self, request, obj=None):
241 return self.add_delete_permission(request, request)
242
243 def has_change_permission(self, request, obj=None):
244 user_groupes = request.user.groups.all()
245 if grp_drh_recrutement in user_groupes or \
246 grp_administrateurs_recrutement in user_groupes:
247 return True
248 return False
249
250 def queryset(self, obj):
251 """
252 Spécifie un queryset limité, autrement Django exécute un
253 select_related() sans paramètre, ce qui a pour effet de charger tous
254 les objets FK, sans limite de profondeur. Dès qu'on arrive, dans les
255 modèles de Region, il existe plusieurs boucles, ce qui conduit à la
256 génération d'une requête infinie.
257
258 """
259 qs = self.model._default_manager.get_query_set()
260 # Si user est superuser afficher tous les candidats
261 user_groupes = obj.user.groups.all()
262 if not grp_drh_recrutement in user_groupes:
263 # Si le user n'est ni un évaluateur ni un administrateur régional,
264 # retourner none
265
266 # Vérifier groupes
267 if grp_evaluateurs_recrutement in user_groupes:
268 user = Evaluateur.objects.get(user=obj.user)
269 #elif grp_administrateurs_recrutement in user_groupes:
270 # user = AdministrateurRegional.objects.get(user=obj.user)
271 else:
272 return qs.none()
273
274 ids = [c.id for c in user.candidats.all()]
275 return qs.select_related('candidats').filter(id__in=ids)
276 return qs.select_related('candidats')
277
278 class ProxyCandidatAdmin(CandidatAdmin):
279 #TODO: init boucler sur les fields pour le readonly
280 list_display = ('nom', 'prenom','statut', 'offre_emploi',
281 'voir_offre_emploi', 'evaluer_candidat', 'afficher_candidat')#'note_evaluateur',)
282 readonly_fields = ('statut', 'offre_emploi', 'prenom', 'nom',
283 'genre', 'nationalite', 'date_naissance',
284 'situation_famille', 'nombre_dependant', 'telephone',
285 'email', 'adresse', 'ville', 'etat_province',
286 'code_postal', 'pays', 'niveau_diplome',
287 'employeur_actuel', 'poste_actuel',
288 'domaine_professionnel',)
289 fieldsets = (
290 ("Offre d'emploi", {
291 'fields': ('offre_emploi', )
292 }),
293 ('Informations personnelles', {
294 'fields': ('prenom','nom','genre', 'nationalite', 'date_naissance',
295 'situation_famille', 'nombre_dependant',)
296 }),
297 ('Coordonnées', {
298 'fields': ('telephone', 'email', 'adresse', 'ville',
299 'etat_province', 'code_postal', 'pays', )
300 }),
301 ('Informations professionnelles', {
302 'fields': ('niveau_diplome','employeur_actuel',
303 'poste_actuel', 'domaine_professionnel',)
304 }),
305 )
306 inlines = []
307
308 def has_add_permission(self, request):
309 return False
310
311 def has_delete_permission(self, request, obj=None):
312 return False
313
314 def has_change_permission(self, request, obj=None):
315 user_groupes = request.user.groups.all()
316 if grp_evaluateurs_recrutement in user_groupes:
317 return True
318 return False
319
320 class CandidatPieceAdmin(admin.ModelAdmin):
321 list_display = ('nom', 'candidat', )
322
323 def queryset(self, request):
324 """
325 Spécifie un queryset limité, autrement Django exécute un
326 select_related() sans paramètre, ce qui a pour effet de charger tous
327 les objets FK, sans limite de profondeur. Dès qu'on arrive, dans les
328 modèles de Region, il existe plusieurs boucles, ce qui conduit à la
329 génération d'une requête infinie.
330 Affiche la liste de candidats que si le user connecté
331 possède un Evaluateur
332 """
333 qs = self.model._default_manager.get_query_set()
334 return qs.select_related('candidat')
335
336 class EvaluateurAdmin(VersionAdmin):
337 fieldsets = (
338 (None, {'fields': ('user', )}),
339 #(None, {'fields': ('candidats',)}),
340 )
341
342 # Peut-être qu'on peut enlever cette def, à cause des permissions
343 def queryset(self, request):
344 """
345 Affiche la liste d'évaluateur que si le user connecté
346 possède un Evaluateur
347 """
348 user_groupes = request.user.groups.all()
349 if grp_drh_recrutement in user_groupes:
350 return self.model._default_manager.get_query_set()
351
352 return qs.none()
353
354 class AdministrateurRegionalAdmin(VersionAdmin):
355 pass
356
357 class CandidatEvaluationAdmin(VersionAdmin):
358 list_display = ('candidat', 'offre_emploi_candidat', 'evaluateur', 'note',
359 'commentaire', 'date',)
360
361 def offre_emploi_candidat(self, obj):
362 return obj.candidat.offre_emploi
363
364 def queryset(self, request):
365 evaluateur = Evaluateur.objects.get(user=request.user)
366 candidats = Candidat.objects.filter(evaluateurs=evaluateur)
367 candidats_evaluations = CandidatEvaluation.objects.\
368 filter(evaluateur=evaluateur,
369 candidat__in=candidats)
370 qs = self.model._default_manager.get_query_set()
371 candidats_evaluations_ids = [ce.id for ce in \
372 candidats_evaluations.all()]
373 return qs.select_related('offre_emploi')#.\
374 #filter(id__in=candidats_evaluations_ids)
375
376 class CourrielTemplateAdmin(VersionAdmin):
377 pass
378
379 admin.site.register(OffreEmploi, OffreEmploiAdmin)
380 admin.site.register(Candidat, CandidatAdmin)
381 admin.site.register(CandidatEvaluation, CandidatEvaluationAdmin)
382 admin.site.register(CourrielTemplate, CourrielTemplateAdmin)
383 admin.site.register(Evaluateur, EvaluateurAdmin)
384 admin.site.register(AdministrateurRegional, AdministrateurRegionalAdmin)
385 admin.site.register(ProxyOffreEmploi, ProxyOffreEmploiAdmin)
386 admin.site.register(ProxyCandidat, ProxyCandidatAdmin)