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
.conf
import settings
11 from django
.db
.models
import Q
12 from ajax_select
import make_ajax_form
13 from auf
.django
.metadata
.admin
import AUFMetadataAdminMixin
, AUFMetadataInlineAdminMixin
, AUF_METADATA_READONLY_FIELDS
14 from project
.rh
import models
as rh
15 from forms
import ContratForm
, AyantDroitForm
, EmployeAdminForm
, AjaxSelect
16 from dae
.utils
import get_employe_from_user
19 # Desactivation des bactch action a cause des managers qui delete
20 AUFMetadataAdminMixin
.actions
= None
22 # Override of the InlineModelAdmin to support the link in the tabular inline
23 class LinkedInline(admin
.options
.InlineModelAdmin
):
24 template
= "admin/linked.html"
25 admin_model_path
= None
27 def __init__(self
, *args
):
28 super(LinkedInline
, self
).__init__(*args
)
29 if self
.admin_model_path
is None:
30 self
.admin_model_path
= self
.model
.__name__
.lower()
33 class ProtectRegionMixin(object):
35 def queryset(self
, request
):
36 from dae
.workflow
import grp_drh
, grp_correspondants_rh
37 qs
= super(ProtectRegionMixin
, self
).queryset(request
)
39 if request
.user
.is_superuser
:
42 user_groups
= request
.user
.groups
.all()
44 if grp_drh
in user_groups
:
47 if grp_correspondants_rh
in user_groups
:
48 employe
= get_employe_from_user(request
.user
)
49 q
= Q(**{self
.model
.prefix_implantation
: employe
.implantation
.region
})
50 qs
= qs
.filter(q
).distinct()
54 def has_change_permission(self
, request
, obj
=None):
57 ids
= [o
.id for o
in self
.queryset(request
)]
63 class ReadOnlyInlineMixin(object):
64 def get_readonly_fields(self
, request
, obj
=None):
65 return [f
.name
for f
in self
.model
._meta
.fields
if f
.name
not in AUF_METADATA_READONLY_FIELDS
]
68 class AyantDroitInline(AUFMetadataInlineAdminMixin
, admin
.StackedInline
):
69 model
= models
.Model
# à remplacer dans admin.py
75 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', 'lien_parente', )
80 class AyantDroitCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
81 readonly_fields
= ('owner', )
82 model
= models
.Model
# à remplacer dans admin.py
86 class ContratInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
88 model
= models
.Model
# à remplacer dans admin.py
92 class DossierROInline(ReadOnlyInlineMixin
, LinkedInline
):
93 template
= "admin/rh/dossier/linked.html"
94 exclude
= AUF_METADATA_READONLY_FIELDS
95 model
= models
.Model
# à remplacer dans admin.py
99 def has_add_permission(self
, request
=None):
102 def has_change_permission(self
, request
, obj
=None):
105 def has_delete_permission(self
, request
, obj
=None):
109 class DossierCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
110 readonly_fields
= ('owner', )
111 model
= models
.Model
# à remplacer dans admin.py
115 class DossierPieceInline(admin
.TabularInline
):
116 model
= models
.Model
# à remplacer dans admin.py
120 class EmployeInline(admin
.TabularInline
):
121 model
= models
.Model
# à remplacer dans admin.py
123 class EmployeCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
124 readonly_fields
= ('owner', )
125 model
= models
.Model
# à remplacer dans admin.py
129 class EmployePieceInline(admin
.TabularInline
):
130 model
= models
.Model
# à remplacer dans admin.py
134 class EvenementInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
135 model
= models
.Model
# à remplacer dans admin.py
139 class EvenementRemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
140 model
= models
.Model
# à remplacer dans admin.py
144 class PosteCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
145 readonly_fields
= ('owner', )
146 model
= models
.Model
# à remplacer dans admin.py
150 class PosteFinancementInline(admin
.TabularInline
):
151 model
= models
.Model
# à remplacer dans admin.py
154 class PostePieceInline(admin
.TabularInline
):
155 model
= models
.Model
# à remplacer dans admin.py
158 class RemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
159 model
= models
.Model
# à remplacer dans admin.py
163 class RemunerationROInline(ReadOnlyInlineMixin
, RemunerationInline
):
167 class TypePosteInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
168 model
= models
.Model
# à remplacer dans admin.py
173 class AyantDroitAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
175 L'ajout d'un nouvel ayantdroit se fait dans l'admin de l'employé.
177 alphabet_filter
= 'nom'
178 search_fields
= ('nom', 'prenom', 'employe__nom', 'employe__prenom', )
179 list_display
= ('_employe', 'lien_parente', '_ayantdroit', )
180 inlines
= (AyantDroitCommentaireInline
,)
181 readonly_fields
= AUFMetadataAdminMixin
.readonly_fields
+ ('employe',)
182 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
183 ("Lien avec l'employé", {
184 'fields': (('employe', 'lien_parente'), )
188 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
192 def save_formset(self
, request
, form
, formset
, change
):
193 instances
= formset
.save(commit
=False)
194 for instance
in instances
:
195 if instance
.__class__
== rh
.AyantDroitCommentaire
:
196 instance
.owner
= request
.user
199 def _ayantdroit(self
, obj
):
201 _ayantdroit
.short_description
= u
'Ayant droit'
203 def _employe(self
, obj
):
204 return unicode(obj
.employe
)
205 _employe
.short_description
= u
'Employé'
207 def has_add_permission(self
, request
):
210 class AyantDroitCommentaireAdmin(admin
.ModelAdmin
):
214 class ClassementAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
215 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
217 'fields': ('type', 'echelon', 'degre', 'coefficient', )
222 class CommentaireAdmin(admin
.ModelAdmin
):
226 #class ContratAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
228 # alphabet_filter = 'dossier__employe__nom'
229 # search_fields = ('dossier__employe__nom', 'dossier__employe__prenom', 'dossier__poste__nom', 'dossier__poste__nom_feminin', )
230 # list_display = ('id', '_employe', '_poste', 'date_debut', 'date_fin', '_implantation', )
231 # fieldsets = AUFMetadataAdminMixin.fieldsets + (
233 # 'fields': ('dossier', 'type_contrat', 'date_debut', 'date_fin', )
237 # def lookup_allowed(self, key, value):
238 # if key in ('dossier__employe__nom__istartswith', ):
241 # def _employe(self, obj):
242 # return unicode(obj.dossier.employe)
243 # _employe.short_description = "Employé"
245 # def _poste(self, obj):
246 # return obj.dossier.poste.nom
247 # _poste.short_description = "Poste"
249 # def _implantation(self, obj):
250 # return obj.dossier.poste.implantation
251 # _poste.short_description = "Implantation"
253 class DeviseAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
254 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
256 'fields': ('code', 'nom', ),
261 class DossierAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
262 alphabet_filter
= 'employe__nom'
263 search_fields
= ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
273 'poste__implantation__region',
274 'poste__implantation',
276 'poste__type_poste__famille_emploi',
278 inlines
= (DossierPieceInline
, ContratInline
,
281 DossierCommentaireInline
,
283 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
285 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
288 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
291 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
293 ('Occupation du Poste par cet Employe', {
294 'fields': (('date_debut', 'date_fin'), )
297 form
= make_ajax_form(rh
.Dossier
, {
298 'employe' : 'employes',
300 'remplacement_de' : 'dossiers',
303 def lookup_allowed(self
, key
, value
):
305 'employe__nom__istartswith',
307 'poste__implantation__region__id__exact',
308 'poste__implantation__id__exact',
309 'poste__type_poste__id__exact',
310 'poste__type_poste__famille_emploi__id__exact',
315 link
= u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>%s</a> <a href="%s" title="Modifier le dossier"><img src="%simg/page_edit.png" /></a>""" % \
316 (reverse('dossier_apercu', args
=(d
.id,)),
318 reverse('admin:rh_dossier_change', args
=(d
.id,)),
322 _id
.allow_tags
= True
323 _id
.short_description
= u
'#'
324 _id
.admin_order_field
= 'id'
327 def _actif(self
, dossier
):
328 if dossier
.employe
.actif
:
329 html
= """<img alt="True" src="%simg/admin/icon-yes.gif">"""
331 html
= """<img alt="False" src="%simg/admin/icon-no.gif">"""
332 return html
% settings
.ADMIN_MEDIA_PREFIX
333 _actif
.allow_tags
= True
334 _actif
.short_description
= u
'Employé actif'
335 _actif
.admin_order_field
= 'employe__actif'
337 def _date_debut(self
, obj
):
338 return obj
.date_debut
339 _date_debut
.short_description
= u
'Occupation début'
340 _date_debut
.admin_order_field
= 'date_debut'
342 def _date_fin(self
, obj
):
344 _date_fin
.short_description
= u
'Occupation fin'
345 _date_fin
.admin_order_field
= 'date_fin'
347 def _poste(self
, dossier
):
348 link
= u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>%s</a> <a href="%s" title="Modifier le poste"><img src="%simg/page_edit.png" /></a>""" % \
349 (reverse('poste_apercu', args
=(dossier
.poste
.id,)),
351 reverse('admin:rh_poste_change', args
=(dossier
.poste
.id,)),
355 _poste
.allow_tags
= True
356 _poste
.short_description
= u
'Poste'
357 _poste
.admin_order_field
= 'poste__nom'
359 def _employe(self
, obj
):
360 employe
= obj
.employe
361 view_link
= reverse('employe_apercu', args
=(employe
.id,))
362 edit_link
= reverse('admin:rh_employe_change', args
=(employe
.id,))
364 if employe
.actif
== False:
365 style
= "color: grey";
369 edit
= u
"""<a href="%s" title="Modifier l'employé"><img src="%simg/user_edit.png" /></a>""" % (edit_link
, settings
.MEDIA_URL
,)
370 return u
"""<a onclick="return showAddAnotherPopup(this);" href='%s' style="%s;">[%s] %s %s</a>%s
372 (view_link
, style
, employe
.id, employe
.nom
.upper(), employe
.prenom
.title(), edit
)
373 _employe
.allow_tags
= True
374 _employe
.short_description
= u
"Employé ([code] NOM Prénom)"
375 _employe
.admin_order_field
= "employe__nom"
377 def save_formset(self
, request
, form
, formset
, change
):
378 instances
= formset
.save(commit
=False)
379 for instance
in instances
:
380 if instance
.__class__
== rh
.DossierCommentaire
:
381 instance
.owner
= request
.user
385 class DossierPieceAdmin(admin
.ModelAdmin
):
389 class DossierCommentaireAdmin(admin
.ModelAdmin
):
393 class EmployeAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
394 alphabet_filter
= 'nom'
395 DEFAULT_ALPHABET
= u
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
396 search_fields
= ('id', 'nom', 'prenom', 'nom_affichage', )
398 form
= EmployeAdminForm
399 list_display
= ('_nom', '_dossiers', 'date_modification', 'user_modification',)
400 list_filter
= ('rh_dossiers__poste__implantation__region', 'rh_dossiers__poste__implantation', 'actif', )
401 inlines
= (AyantDroitInline
,
404 EmployeCommentaireInline
)
405 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
407 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
409 ('Informations personnelles', {
410 'fields': ('situation_famille', 'date_entree', )
413 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
418 view_link
= reverse('employe_apercu', args
=(obj
.id,))
419 edit_link
= reverse('admin:rh_employe_change', args
=(obj
.id,))
420 return u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>[%s] %s %s</a>
421 <a href="%s" title="Modifier l'employé"><img src="%simg/user_edit.png" /></a>""" % \
422 (view_link
, obj
.id, obj
.nom
.upper(), obj
.prenom
.title(), edit_link
, settings
.MEDIA_URL
,)
423 _nom
.allow_tags
= True
424 _nom
.short_description
= u
"Employé ([code] NOM Prénom)"
425 _nom
.admin_order_field
= "nom"
427 def _dossiers(self
, obj
):
429 for d
in obj
.rh_dossiers
.all().order_by('-date_debut'):
431 edit
= u
"""<a href="%s" title="Modifier le dossier"><img src="%simg/page_edit.png" /></a>""" % (reverse('admin:rh_dossier_change', args
=(d
.id,)), settings
.MEDIA_URL
,)
432 if d
.date_fin
is not None:
434 style
= u
"color: grey";
435 link
= u
"""<li><a style="%s;" onclick="return showAddAnotherPopup(this);" href='%s'>%s : %s</a>%s</li>""" % \
437 reverse('dossier_apercu', args
=(d
.id,)),
443 return "<ul>%s</ul>" % "\n".join(l
)
444 _dossiers
.allow_tags
= True
446 def queryset(self
, request
):
447 qs
= super(EmployeAdmin
, self
).queryset(request
)
448 return qs
.filter(actif
=True).select_related(depth
=1).order_by('nom')
450 def save_formset(self
, request
, form
, formset
, change
):
451 instances
= formset
.save(commit
=False)
452 for instance
in instances
:
453 if instance
.__class__
== rh
.EmployeCommentaire
:
454 instance
.owner
= request
.user
459 class EmployeCommentaireAdmin(admin
.ModelAdmin
):
463 class EmployePieceAdmin(admin
.ModelAdmin
):
467 class EvenementAdmin(admin
.ModelAdmin
):
468 inlines
= (EvenementRemunerationInline
,)
471 class EvenementRemunerationAdmin(admin
.ModelAdmin
):
475 class FamilleEmploiAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
476 inlines
= (TypePosteInline
,)
477 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
484 class OrganismeBstgAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
485 search_fields
= ('nom', )
486 list_display
= ('nom', 'type', 'pays', )
487 inlines
= (DossierROInline
,)
488 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
490 'fields': ('nom', 'type', 'pays', )
495 class PosteAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
496 form
= make_ajax_form(rh
.Poste
, {
497 'implantation' : 'implantations',
498 'type_poste' : 'typepostes',
499 'responsable' : 'postes',
500 'valeur_point_min' : 'valeurpoints',
501 'valeur_point_max' : 'valeurpoints',
503 alphabet_filter
= 'nom'
504 search_fields
= ('nom',
505 'implantation__code',
507 'implantation__region__code',
508 'implantation__region__nom',
520 list_filter
= ('service',
521 'implantation__region',
524 'type_poste__famille_emploi',
526 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
528 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
529 'service', 'responsable')
532 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
535 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
538 'fields': (('classement_min', 'valeur_point_min', 'devise_min', 'salaire_min', 'indemn_min', 'autre_min', ),
539 ('classement_max', 'valeur_point_max' ,'devise_max', 'salaire_max', 'indemn_max', 'autre_max', ),
542 ('Comparatifs de rémunération', {
543 'fields': ('devise_comparaison',
544 ('comp_locale_min', 'comp_locale_max'),
545 ('comp_universite_min', 'comp_universite_max'),
546 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
547 ('comp_ong_min', 'comp_ong_max'),
548 ('comp_autre_min', 'comp_autre_max'))
551 'fields': ('justification',)
553 ('Autres Metadata', {
554 'fields': ('date_debut', 'date_fin')
558 inlines
= (PosteFinancementInline
,
561 PosteCommentaireInline
, )
564 def _nom(self
, poste
):
565 link
= u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>%s</a> <a href="%s" title="Modifier le poste"><img src="%simg/page_edit.png" /></a>""" % \
566 (reverse('poste_apercu', args
=(poste
.id,)),
568 reverse('admin:rh_poste_change', args
=(poste
.id,)),
572 _nom
.allow_tags
= True
573 _nom
.short_description
= u
'Nom'
574 _nom
.admin_order_field
= 'nom'
576 def _occupe_par(self
, obj
):
577 """Formatte la méthode Poste.occupe_par() pour l'admin"""
579 employes
= obj
.occupe_par()
583 link
= "<a href='%s'>%s</a>" % \
584 (reverse('admin:rh_employe_change', args
=(e
.id,)),
587 output
= "\n<br />".join(l
)
589 _occupe_par
.allow_tags
= True
590 _occupe_par
.short_description
= "Occupé par"
592 def save_formset(self
, request
, form
, formset
, change
):
593 instances
= formset
.save(commit
=False)
594 for instance
in instances
:
595 if instance
.__class__
== rh
.PosteCommentaire
:
596 instance
.owner
= request
.user
601 class PosteCommentaireAdmin(admin
.ModelAdmin
):
605 class PosteFinancementAdmin(admin
.ModelAdmin
):
609 class PostePieceAdmin(admin
.ModelAdmin
):
613 class RemunerationAdmin(admin
.ModelAdmin
):
617 class ResponsableImplantationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
618 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
620 'fields': ('employe', 'implantation', ),
625 class ServiceAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
626 list_display
= ('nom', 'actif', )
627 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
633 class StatutAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
634 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
636 'fields': ('code', 'nom', ),
640 class TauxChangeAdmin(admin
.ModelAdmin
):
641 list_display
= ('taux', 'devise', 'annee', )
642 list_filter
= ('devise', )
643 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
645 'fields': ('taux', 'devise', 'annee', ),
649 class TypeContratAdmin(admin
.ModelAdmin
):
650 inlines
= (ContratInline
,)
651 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
653 'fields': ('nom', 'nom_long', ),
658 class TypePosteAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
659 search_fields
= ('nom', 'nom_feminin', )
660 list_display
= ('nom', 'famille_emploi', )
661 list_filter
= ('famille_emploi', )
662 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
664 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
669 class TypeRemunerationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
670 list_display
= ('nom', 'type_paiement', 'nature_remuneration', )
671 #inlines = (RemunerationROInline,) utilité?
672 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
674 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
679 class TypeRevalorisationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
680 #inlines = (RemunerationROInline,) utilité?
681 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
688 class ValeurPointAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
689 list_display
= ('_devise_code', '_devise_nom', 'annee', 'valeur', )
690 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
692 'fields': ('valeur', 'devise', 'implantation', 'annee', )
696 def _devise_code(self
, obj
):
697 return obj
.devise
.code
698 _devise_code
.short_description
= "Code de la devise"
700 def _devise_nom(self
, obj
):
701 return obj
.devise
.nom
702 _devise_nom
.short_description
= "Nom de la devise"
705 def calc_remun(dossier
):
706 thisyear
= datetime
.date
.today().year
707 thisyearfilter
= Q(date_debut__year
=thisyear
) |
Q(date_fin__year
=thisyear
)
709 remunnow
= dossier
.rh_remunerations
.filter(thisyearfilter
)
713 sums
= defaultdict(int)
714 sums_euro
= defaultdict(int)
716 nature
= r
.type.nature_remuneration
717 sums
[nature
] += r
.montant
718 sums_euro
[nature
] += r
.montant_euro()
719 remun_sum
+= r
.montant
720 remun_sum_euro
+= r
.montant_euro()
724 for n
, s
in sums
.iteritems():
725 remun
[n
] = [sums
[n
], sums_euro
[n
]]
727 return remun
, remun_sum
, remun_sum_euro