1 # -*- encoding: utf-8 -*-
3 from collections
import defaultdict
6 from django
.db
import models
7 from django
import forms
8 from django
.core
.urlresolvers
import reverse
9 from django
.contrib
import admin
10 from django
.contrib
.admin
.views
.main
import ChangeList
as DjangoChangeList
11 from django
.conf
import settings
12 from django
.db
.models
import Q
13 from django
.template
.defaultfilters
import date
14 from ajax_select
import make_ajax_form
15 from auf
.django
.metadata
.admin
import AUFMetadataAdminMixin
, AUFMetadataInlineAdminMixin
, AUF_METADATA_READONLY_FIELDS
16 from forms
import ContratForm
, AyantDroitForm
, EmployeAdminForm
, AjaxSelect
17 from dae
.utils
import get_employe_from_user
18 from groups
import grp_drh
24 class ChangeList(DjangoChangeList
):
26 def __init__(self
, *args
, **kwargs
):
27 super(ChangeList
, self
).__init__(*args
, **kwargs
)
29 def get_query_set(self
):
30 old
= self
.params
.copy()
33 for k
, v
in self
.params
.items():
34 if k
.startswith('date_debut'):
37 elif k
.startswith('date_fin'):
41 qs
= super(ChangeList
, self
).get_query_set()
43 if date_fin
is None and date_debut
is not None:
45 if date_debut
is None and date_fin
is not None:
48 if date_debut
is not None and date_fin
is not None:
49 q_left
= (Q(date_debut__isnull
=True) |
Q(date_debut__lte
=date_debut
)) & (Q(date_fin__gte
=date_debut
) & Q(date_fin__lte
=date_fin
))
50 q_right
= (Q(date_fin__isnull
=True) |
Q(date_fin__gte
=date_fin
)) & (Q(date_debut__gte
=date_debut
) & Q(date_debut__lte
=date_fin
))
51 q_both
= Q(date_fin__isnull
=True) |
Q(date_fin__lte
=date_fin
) & (Q(date_debut__isnull
=True) |
Q(date_debut__gte
=date_debut
))
52 qs
= qs
.filter(q_left | q_right | q_both
)
57 # Override of the InlineModelAdmin to support the link in the tabular inline
58 class LinkedInline(admin
.options
.InlineModelAdmin
):
59 template
= "admin/linked.html"
60 admin_model_path
= None
62 def __init__(self
, *args
):
63 super(LinkedInline
, self
).__init__(*args
)
64 if self
.admin_model_path
is None:
65 self
.admin_model_path
= self
.model
.__name__
.lower()
68 class ProtectRegionMixin(object):
70 def queryset(self
, request
):
71 from dae
.workflow
import grp_drh
, grp_correspondants_rh
72 qs
= super(ProtectRegionMixin
, self
).queryset(request
)
74 if request
.user
.is_superuser
:
77 user_groups
= request
.user
.groups
.all()
79 if grp_drh
in user_groups
:
82 if grp_correspondants_rh
in user_groups
:
83 employe
= get_employe_from_user(request
.user
)
84 q
= Q(**{self
.model
.prefix_implantation
: employe
.implantation
.region
})
85 qs
= qs
.filter(q
).distinct()
89 def has_change_permission(self
, request
, obj
=None):
90 user_groups
= request
.user
.groups
.all()
92 # Lock pour autoriser uniquement les DRH à utiliser RH
93 if not request
.user
.is_superuser
and not grp_drh
in user_groups
:
96 if len(user_groups
) == 0 and not request
.user
.is_superuser
:
101 ids
= [o
.id for o
in self
.queryset(request
)]
107 class ReadOnlyInlineMixin(object):
108 def get_readonly_fields(self
, request
, obj
=None):
109 return [f
.name
for f
in self
.model
._meta
.fields
if f
.name
not in AUF_METADATA_READONLY_FIELDS
]
112 class AyantDroitInline(AUFMetadataInlineAdminMixin
, admin
.StackedInline
):
113 model
= rh
.AyantDroit
114 form
= AyantDroitForm
119 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', 'lien_parente', )
124 class AyantDroitCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
125 readonly_fields
= ('owner', )
126 model
= rh
.AyantDroitCommentaire
130 class ContratInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
136 class DossierROInline(ReadOnlyInlineMixin
, LinkedInline
):
137 template
= "admin/rh/dossier/linked.html"
138 exclude
= AUF_METADATA_READONLY_FIELDS
143 def has_add_permission(self
, request
=None):
146 def has_change_permission(self
, request
, obj
=None):
149 def has_delete_permission(self
, request
, obj
=None):
153 class DossierCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
154 readonly_fields
= ('owner', )
155 model
= rh
.DossierCommentaire
159 class DossierPieceInline(admin
.TabularInline
):
160 model
= rh
.DossierPiece
164 class EmployeInline(admin
.TabularInline
):
167 class EmployeCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
168 readonly_fields
= ('owner', )
169 model
= rh
.EmployeCommentaire
173 class EmployePieceInline(admin
.TabularInline
):
174 model
= rh
.EmployePiece
178 class PosteCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
179 readonly_fields
= ('owner', )
180 model
= rh
.PosteCommentaire
184 class PosteFinancementInline(admin
.TabularInline
):
185 model
= rh
.PosteFinancement
188 class PostePieceInline(admin
.TabularInline
):
189 model
= rh
.PostePiece
192 class RemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
193 model
= rh
.Remuneration
197 class RemunerationROInline(ReadOnlyInlineMixin
, RemunerationInline
):
201 class TypePosteInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
205 class PosteComparaisonInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
206 model
= rh
.PosteComparaison
209 class ClassementAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
210 list_display
= ('_classement', 'date_modification', 'user_modification', )
211 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
213 'fields': ('type', 'echelon', 'degre', 'coefficient', )
217 def _classement(self
, obj
):
219 _classement
.short_description
= u
"Classement"
221 class CommentaireAdmin(admin
.ModelAdmin
):
225 class DeviseAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
226 list_display
= ('code', 'nom', 'date_modification', 'user_modification', )
227 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
229 'fields': ('code', 'nom', ),
234 class DossierAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
235 alphabet_filter
= 'employe__nom'
236 search_fields
= ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
247 'poste__implantation__region',
248 'poste__implantation',
250 'poste__type_poste__famille_emploi',
251 'rh_contrats__type_contrat',
255 inlines
= (DossierPieceInline
, ContratInline
,
257 DossierCommentaireInline
,
259 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
261 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
264 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
267 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
269 ('Occupation du Poste par cet Employe', {
270 'fields': (('date_debut', 'date_fin'), )
273 form
= make_ajax_form(rh
.Dossier
, {
274 'employe' : 'employes',
276 'remplacement_de' : 'dossiers',
279 def lookup_allowed(self
, key
, value
):
281 'employe__nom__istartswith',
282 'poste__implantation__region__id__exact',
283 'poste__implantation__id__exact',
284 'poste__type_poste__id__exact',
285 'poste__type_poste__famille_emploi__id__exact',
286 'rh_contrats__type_contrat__id__exact',
288 'date_debut__isnull',
294 def get_changelist(self
, request
, **kwargs
):
298 apercu_link
= u
"""<a title="Aperçu du dossier" onclick="return showAddAnotherPopup(this);" href='%s'><img src="%simg/loupe.png" /></a>""" % \
299 (reverse('dossier_apercu', args
=(d
.id,)),
302 link
= u
"""%s <a href="%s" title="Modifier le dossier"><strong>%s</strong></a>""" % \
304 reverse('admin:rh_dossier_change', args
=(d
.id,)),
308 _id
.allow_tags
= True
309 _id
.short_description
= u
"Dossier__#"
310 _id
.admin_order_field
= 'id'
313 def _date_debut(self
, obj
):
314 return date(obj
.date_debut
)
316 _date_debut
.short_description
= u
'Occupation début'
317 _date_debut
.admin_order_field
= 'date_debut'
319 def _date_fin(self
, obj
):
320 return date(obj
.date_fin
)
321 _date_fin
.short_description
= u
'Occupation fin'
322 _date_fin
.admin_order_field
= 'date_fin'
324 def _poste(self
, dossier
):
325 link
= u
"""<a title="Aperçu du poste" onclick="return showAddAnotherPopup(this);" href='%s'><img src="%simg/loupe.png" /></a> <a href="%s" title="Modifier le poste">%s</a>""" % \
326 (reverse('poste_apercu', args
=(dossier
.poste
.id,)),
328 reverse('admin:rh_poste_change', args
=(dossier
.poste
.id,)),
332 _poste
.allow_tags
= True
333 _poste
.short_description
= u
'Poste'
334 _poste
.admin_order_field
= 'poste__nom'
336 def _employe(self
, obj
):
337 employe
= obj
.employe
338 view_link
= reverse('employe_apercu', args
=(employe
.id,))
339 edit_link
= reverse('admin:rh_employe_change', args
=(employe
.id,))
342 view
= u
"""<a href="%s" title="Aperçu l'employé" onclick="return showAddAnotherPopup(this);"><img src="%simg/loupe.png" /></a>""" % (view_link
, settings
.STATIC_URL
,)
343 return u
"""%s<a href='%s' style="%s;">[%s] %s %s</a>""" % \
344 (view
, edit_link
, style
, employe
.id, employe
.nom
.upper(), employe
.prenom
.title())
345 _employe
.allow_tags
= True
346 _employe
.short_description
= u
"Employé ([code] NOM Prénom)"
347 _employe
.admin_order_field
= "employe__nom"
349 def save_formset(self
, request
, form
, formset
, change
):
350 instances
= formset
.save(commit
=False)
351 for instance
in instances
:
352 if instance
.__class__
== rh
.DossierCommentaire
:
353 instance
.owner
= request
.user
354 instance
.date_creation
= datetime
.datetime
.now()
358 class DossierPieceAdmin(admin
.ModelAdmin
):
362 class DossierCommentaireAdmin(admin
.ModelAdmin
):
366 class EmployeAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
367 alphabet_filter
= 'nom'
368 DEFAULT_ALPHABET
= u
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
369 search_fields
= ('id', 'nom', 'prenom', 'nom_affichage', )
371 form
= EmployeAdminForm
372 list_display
= ('_apercu', '_nom', '_dossiers', 'date_modification', 'user_modification', )
373 list_filter
= ('rh_dossiers__poste__implantation__region', 'rh_dossiers__poste__implantation', 'nb_postes', )
374 inlines
= (AyantDroitInline
,
377 EmployeCommentaireInline
)
378 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
380 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
382 ('Informations personnelles', {
383 'fields': ('situation_famille', 'date_entree', )
386 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
390 def _apercu(self
, obj
):
391 return u
"""<a title="Aperçu de l'employé" onclick="return showAddAnotherPopup(this);" href='%s'><img src="%simg/loupe.png" /></a>""" % \
392 (reverse('employe_apercu', args
=(obj
.id,)), settings
.STATIC_URL
)
393 _apercu
.allow_tags
= True
394 _apercu
.short_description
= u
""
395 _apercu
.admin_order_field
= ""
398 edit_link
= reverse('admin:rh_employe_change', args
=(obj
.id,))
399 return u
"""<a href='%s'><strong>[%s] %s %s</strong></a>""" % \
400 (edit_link
, obj
.id, obj
.nom
.upper(), obj
.prenom
.title(),)
401 _nom
.allow_tags
= True
402 _nom
.short_description
= u
"Employé ([code] NOM Prénom)"
403 _nom
.admin_order_field
= "nom"
405 def _dossiers(self
, obj
):
407 for d
in obj
.rh_dossiers
.all().order_by('-date_debut'):
408 apercu
= u
"""<a title="Aperçu du dossier" href="%s" onclick="return showAddAnotherPopup(this);" title="Aperçu du dossier"><img src="%simg/loupe.png" /></a>""" % \
409 (reverse('dossier_apercu', args
=(d
.id,)), settings
.STATIC_URL
,)
410 link
= u
"""<li>%s<a href='%s'>%s : %s</a></li>""" % \
412 reverse('admin:rh_dossier_change', args
=(d
.id,)),
417 # Dossier terminé en gris non cliquable
418 if d
.date_fin
is not None:
419 link
= u
"""<li style="color: grey">%s : %s</li>""" % \
425 return "<ul>%s</ul>" % "\n".join(l
)
426 _dossiers
.allow_tags
= True
427 _dossiers
.short_description
= u
"Dossiers"
429 def queryset(self
, request
):
430 qs
= super(EmployeAdmin
, self
).queryset(request
)
431 return qs
.select_related(depth
=1).order_by('nom')
433 def save_formset(self
, request
, form
, formset
, change
):
434 instances
= formset
.save(commit
=False)
435 for instance
in instances
:
436 if instance
.__class__
== rh
.EmployeCommentaire
:
437 instance
.owner
= request
.user
438 instance
.date_creation
= datetime
.datetime
.now()
443 class EmployeCommentaireAdmin(admin
.ModelAdmin
):
447 class EmployePieceAdmin(admin
.ModelAdmin
):
451 class FamilleEmploiAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
452 list_display
= ('nom', 'date_modification', 'user_modification', )
453 inlines
= (TypePosteInline
,)
454 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
461 class OrganismeBstgAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
462 search_fields
= ('nom',)
463 list_display
= ('nom', 'type', 'pays', 'date_modification', 'user_modification', )
464 list_filter
= ('type', )
465 inlines
= (DossierROInline
,)
466 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
468 'fields': ('nom', 'type', 'pays', )
473 class PosteAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
474 form
= make_ajax_form(rh
.Poste
, {
475 'implantation' : 'implantations',
476 'type_poste' : 'typepostes',
477 'responsable' : 'postes',
478 'valeur_point_min' : 'valeurpoints',
479 'valeur_point_max' : 'valeurpoints',
481 alphabet_filter
= 'nom'
482 search_fields
= ('nom',
483 'implantation__code',
485 'implantation__region__code',
486 'implantation__region__nom',
501 'implantation__region',
504 'type_poste__famille_emploi',
509 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
511 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
512 'service', 'responsable')
515 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
518 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
521 'fields': (('classement_min', 'valeur_point_min', 'devise_min', 'salaire_min', 'indemn_min', 'autre_min', ),
522 ('classement_max', 'valeur_point_max' ,'devise_max', 'salaire_max', 'indemn_max', 'autre_max', ),
525 ('Comparatifs de rémunération', {
526 'fields': ('devise_comparaison',
527 ('comp_locale_min', 'comp_locale_max'),
528 ('comp_universite_min', 'comp_universite_max'),
529 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
530 ('comp_ong_min', 'comp_ong_max'),
531 ('comp_autre_min', 'comp_autre_max'))
534 'fields': ('justification',)
536 ('Autres Méta-données', {
537 'fields': ('date_debut', 'date_fin')
541 inlines
= (PosteFinancementInline
,
544 PosteComparaisonInline
,
545 PosteCommentaireInline
, )
547 def lookup_allowed(self
, key
, value
):
550 'date_debut__isnull',
557 def _apercu(self
, poste
):
558 link
= u
"""<a onclick="return showAddAnotherPopup(this);" title="Aperçu du poste" href='%s'><img src="%simg/loupe.png" /> %s</a>""" % \
559 (reverse('poste_apercu', args
=(poste
.id,)),
564 _apercu
.allow_tags
= True
565 _apercu
.short_description
= 'Poste __#'
566 _apercu
.admin_order_field
= 'id'
568 def _service(self
, obj
):
571 def _nom(self
, poste
):
572 link
= u
"""<a href="%s" title="Modifier le poste"><strong>%s</strong></a>""" % \
573 (reverse('admin:rh_poste_change', args
=(poste
.id,)),
577 _nom
.allow_tags
= True
578 _nom
.short_description
= u
'Nom'
579 _nom
.admin_order_field
= 'nom'
581 def _occupe_par(self
, obj
):
582 """Formatte la méthode Poste.occupe_par() pour l'admin"""
584 if obj
.date_fin
is not None and obj
.date_fin
< datetime
.date
.now():
586 employes
= obj
.occupe_par()
590 link
= "<a href='%s' title='Aperçu de l\'employer' onclick='return showAddAnotherPopup(this)'><img src='%simg/loupe.png' /></a> <a href='%s'>%s</a>" % \
591 (reverse('employe_apercu', args
=(e
.id,)),
593 reverse('admin:rh_employe_change', args
=(e
.id,)),
597 output
= "\n<br />".join(l
)
599 _occupe_par
.allow_tags
= True
600 _occupe_par
.short_description
= "Occupé par"
602 def save_formset(self
, request
, form
, formset
, change
):
603 instances
= formset
.save(commit
=False)
604 for instance
in instances
:
605 if instance
.__class__
== rh
.PosteCommentaire
:
606 instance
.owner
= request
.user
607 instance
.date_creation
= datetime
.datetime
.now()
612 class PosteCommentaireAdmin(admin
.ModelAdmin
):
616 class PosteFinancementAdmin(admin
.ModelAdmin
):
620 class PostePieceAdmin(admin
.ModelAdmin
):
624 class RemunerationAdmin(admin
.ModelAdmin
):
628 class ResponsableImplantationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
629 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
631 'fields': ('employe', 'implantation', ),
636 class ServiceAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
637 list_display
= ('nom', 'date_modification', 'user_modification', )
638 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
644 class StatutAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
645 list_display
= ('code', 'nom', 'date_modification', 'user_modification', )
646 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
648 'fields': ('code', 'nom', ),
652 class TauxChangeAdmin(admin
.ModelAdmin
):
653 list_display
= ('taux', 'devise', 'annee', 'date_modification', 'user_modification', )
654 list_filter
= ('devise', )
655 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
657 'fields': ('taux', 'devise', 'annee', ),
661 class TypeContratAdmin(admin
.ModelAdmin
):
662 list_display
= ('nom', 'nom_long', 'date_modification', 'user_modification', )
663 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
665 'fields': ('nom', 'nom_long', ),
670 class TypePosteAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
671 search_fields
= ('nom', 'nom_feminin', )
672 list_display
= ('nom', 'famille_emploi', 'date_modification', 'user_modification', )
673 list_filter
= ('famille_emploi', )
674 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
676 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
681 class TypeRemunerationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
682 list_display
= ('nom', 'type_paiement', 'nature_remuneration', 'date_modification', 'user_modification', )
683 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
685 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
690 class TypeRevalorisationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
691 list_display
= ('nom', 'date_modification', 'user_modification', )
692 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
699 class ValeurPointAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
700 list_display
= ('_devise_code', '_devise_nom', 'annee', 'valeur', 'date_modification', 'user_modification', )
701 list_filter
= ('annee', 'devise', )
702 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
704 'fields': ('valeur', 'devise', 'implantation', 'annee', )
708 def _devise_code(self
, obj
):
709 return obj
.devise
.code
710 _devise_code
.short_description
= "Code de la devise"
712 def _devise_nom(self
, obj
):
713 return obj
.devise
.nom
714 _devise_nom
.short_description
= "Nom de la devise"
717 admin
.site
.register(rh
.Classement
, ClassementAdmin
)
718 admin
.site
.register(rh
.Devise
, DeviseAdmin
)
719 admin
.site
.register(rh
.Dossier
, DossierAdmin
)
720 admin
.site
.register(rh
.Employe
, EmployeAdmin
)
721 admin
.site
.register(rh
.FamilleEmploi
, FamilleEmploiAdmin
)
722 admin
.site
.register(rh
.OrganismeBstg
, OrganismeBstgAdmin
)
723 admin
.site
.register(rh
.Poste
, PosteAdmin
)
724 admin
.site
.register(rh
.ResponsableImplantation
, ResponsableImplantationAdmin
)
725 admin
.site
.register(rh
.Service
, ServiceAdmin
)
726 admin
.site
.register(rh
.Statut
, StatutAdmin
)
727 admin
.site
.register(rh
.TauxChange
, TauxChangeAdmin
)
728 admin
.site
.register(rh
.TypeContrat
, TypeContratAdmin
)
729 admin
.site
.register(rh
.TypePoste
, TypePosteAdmin
)
730 admin
.site
.register(rh
.TypeRemuneration
, TypeRemunerationAdmin
)
731 admin
.site
.register(rh
.TypeRevalorisation
, TypeRevalorisationAdmin
)
732 admin
.site
.register(rh
.ValeurPoint
, ValeurPointAdmin
)