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 DossierForm
, 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
,):
263 alphabet_filter
= 'employe__nom'
264 search_fields
= ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
265 list_display
= ('_id', '_employe', '_actif', '_poste', 'date_debut', 'date_fin', 'date_modification')
266 list_filter
= ('poste__implantation__region', 'poste__implantation', )
267 inlines
= (DossierPieceInline
, ContratInline
,
270 DossierCommentaireInline
,
272 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
274 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
277 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
280 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
282 ('Occupation du Poste par cet Employe', {
283 'fields': (('date_debut', 'date_fin'), )
288 js
= ('js/dossier.js',)
290 def lookup_allowed(self
, key
, value
):
292 'employe__nom__istartswith',
294 'poste__implantation__region__id__exact',
295 'poste__implantation__id__exact',
300 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>""" % \
301 (reverse('dossier_apercu', args
=(d
.id,)),
303 reverse('admin:rh_dossier_change', args
=(d
.id,)),
307 _id
.allow_tags
= True
308 _id
.short_description
= u
'Numéro de dossier'
309 _id
.admin_order_field
= 'id'
312 def _actif(self
, dossier
):
313 if dossier
.employe
.actif
:
314 html
= """<img alt="True" src="%simg/admin/icon-yes.gif">"""
316 html
= """<img alt="False" src="%simg/admin/icon-no.gif">"""
317 return html
% settings
.ADMIN_MEDIA_PREFIX
318 _actif
.allow_tags
= True
319 _actif
.short_description
= u
'Employé actif'
320 _actif
.admin_order_field
= 'employe__actif'
322 def _poste(self
, dossier
):
323 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>""" % \
324 (reverse('poste_apercu', args
=(dossier
.poste
.id,)),
326 reverse('admin:rh_poste_change', args
=(dossier
.poste
.id,)),
330 _poste
.allow_tags
= True
331 _poste
.short_description
= u
'Poste'
332 _poste
.admin_order_field
= 'poste__nom'
334 def _employe(self
, obj
):
335 employe
= obj
.employe
336 view_link
= reverse('employe_apercu', args
=(employe
.id,))
337 edit_link
= reverse('admin:rh_employe_change', args
=(employe
.id,))
338 return u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>[%s] %s %s</a>
339 <a href="%s" title="Modifier l'employé"><img src="%simg/user_edit.png" /></a>""" % \
340 (view_link
, employe
.id, employe
.nom
.upper(), employe
.prenom
.title(), edit_link
, settings
.MEDIA_URL
,)
341 _employe
.allow_tags
= True
342 _employe
.short_description
= u
"Employé ([code] NOM Prénom)"
343 _employe
.admin_order_field
= "employe__nom"
345 def save_formset(self
, request
, form
, formset
, change
):
346 instances
= formset
.save(commit
=False)
347 for instance
in instances
:
348 if instance
.__class__
== rh
.DossierCommentaire
:
349 instance
.owner
= request
.user
353 class DossierPieceAdmin(admin
.ModelAdmin
):
357 class DossierCommentaireAdmin(admin
.ModelAdmin
):
361 class EmployeAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
362 alphabet_filter
= 'nom'
363 DEFAULT_ALPHABET
= u
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
364 search_fields
= ('id', 'nom', 'prenom', 'nom_affichage', )
366 form
= EmployeAdminForm
367 list_display
= ('_nom', '_dossiers', 'date_modification', 'user_modification',)
368 list_filter
= ('rh_dossiers__poste__implantation__region', 'rh_dossiers__poste__implantation', 'actif', )
369 inlines
= (AyantDroitInline
,
372 EmployeCommentaireInline
)
373 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
375 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
377 ('Informations personnelles', {
378 'fields': ('situation_famille', 'date_entree', )
381 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
386 view_link
= reverse('employe_apercu', args
=(obj
.id,))
387 edit_link
= reverse('admin:rh_employe_change', args
=(obj
.id,))
388 return u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>[%s] %s %s</a>
389 <a href="%s" title="Modifier l'employé"><img src="%simg/user_edit.png" /></a>""" % \
390 (view_link
, obj
.id, obj
.nom
.upper(), obj
.prenom
.title(), edit_link
, settings
.MEDIA_URL
,)
391 _nom
.allow_tags
= True
392 _nom
.short_description
= u
"Employé ([code] NOM Prénom)"
393 _nom
.admin_order_field
= "nom"
395 def _dossiers(self
, obj
):
397 for d
in obj
.dossiers
.all().order_by('-date_debut'):
399 if d
.date_fin
is not None:
400 style
= "color: grey";
401 link
= u
"""<li><a style="%s;" onclick="return showAddAnotherPopup(this);" href='%s'>%s : %s</a> <a href="%s" title="Modifier le dossier"><img src="%simg/page_edit.png" /></a> </li>""" % \
403 reverse('dossier_apercu', args
=(d
.id,)),
406 reverse('admin:rh_dossier_change', args
=(d
.id,)),
410 return "<ul>%s</ul>" % "\n".join(l
)
411 _dossiers
.allow_tags
= True
413 def queryset(self
, request
):
414 qs
= super(EmployeAdmin
, self
).queryset(request
)
415 return qs
.filter(actif
=True).select_related(depth
=1).order_by('nom')
417 def save_formset(self
, request
, form
, formset
, change
):
418 instances
= formset
.save(commit
=False)
419 for instance
in instances
:
420 if instance
.__class__
== rh
.EmployeCommentaire
:
421 instance
.owner
= request
.user
426 class EmployeCommentaireAdmin(admin
.ModelAdmin
):
430 class EmployePieceAdmin(admin
.ModelAdmin
):
434 class EvenementAdmin(admin
.ModelAdmin
):
435 inlines
= (EvenementRemunerationInline
,)
438 class EvenementRemunerationAdmin(admin
.ModelAdmin
):
442 class FamilleEmploiAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
443 inlines
= (TypePosteInline
,)
444 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
451 class OrganismeBstgAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
452 search_fields
= ('nom', )
453 list_display
= ('nom', 'type', 'pays', )
454 inlines
= (DossierROInline
,)
455 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
457 'fields': ('nom', 'type', 'pays', )
462 class PosteAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
463 form
= make_ajax_form(rh
.Poste
, {
464 'implantation' : 'implantations',
465 'type_poste' : 'typepostes',
466 'responsable' : 'postes',
467 'valeur_point_min' : 'valeurpoints',
468 'valeur_point_max' : 'valeurpoints',
470 alphabet_filter
= 'nom'
471 search_fields
= ('nom',
472 'implantation__code',
474 'implantation__region__code',
475 'implantation__region__nom',
487 list_filter
= ('service',
488 'implantation__region',
491 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
493 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
494 'service', 'responsable')
497 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
500 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
503 'fields': (('classement_min', 'classement_max'),
504 ('valeur_point_min', 'valeur_point_max'),
505 ('devise_min', 'devise_max'),
506 ('salaire_min', 'salaire_max'),
507 ('indemn_min', 'indemn_max'),
508 ('autre_min', 'autre_max'))
510 ('Comparatifs de rémunération', {
511 'fields': ('devise_comparaison',
512 ('comp_locale_min', 'comp_locale_max'),
513 ('comp_universite_min', 'comp_universite_max'),
514 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
515 ('comp_ong_min', 'comp_ong_max'),
516 ('comp_autre_min', 'comp_autre_max'))
519 'fields': ('justification',)
521 ('Autres Metadata', {
522 'fields': ('date_debut', 'date_fin')
526 inlines
= (PosteFinancementInline
,
529 PosteCommentaireInline
, )
532 def _nom(self
, poste
):
533 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>""" % \
534 (reverse('poste_apercu', args
=(poste
.id,)),
536 reverse('admin:rh_poste_change', args
=(poste
.id,)),
540 _nom
.allow_tags
= True
541 _nom
.short_description
= u
'Nom'
542 _nom
.admin_order_field
= 'nom'
544 def _occupe_par(self
, obj
):
545 """Formatte la méthode Poste.occupe_par() pour l'admin"""
547 employes
= obj
.occupe_par()
551 link
= "<a href='%s'>%s</a>" % \
552 (reverse('admin:rh_employe_change', args
=(e
.id,)),
555 output
= "\n<br />".join(l
)
557 _occupe_par
.allow_tags
= True
558 _occupe_par
.short_description
= "Occupé par"
560 def save_formset(self
, request
, form
, formset
, change
):
561 instances
= formset
.save(commit
=False)
562 for instance
in instances
:
563 if instance
.__class__
== rh
.PosteCommentaire
:
564 instance
.owner
= request
.user
569 class PosteCommentaireAdmin(admin
.ModelAdmin
):
573 class PosteFinancementAdmin(admin
.ModelAdmin
):
577 class PostePieceAdmin(admin
.ModelAdmin
):
581 class RemunerationAdmin(admin
.ModelAdmin
):
585 class ResponsableImplantationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
586 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
588 'fields': ('employe', 'implantation', ),
593 class ServiceAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
594 list_display
= ('nom', 'actif', )
595 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
601 class StatutAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
602 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
604 'fields': ('code', 'nom', ),
608 class TauxChangeAdmin(admin
.ModelAdmin
):
609 list_display
= ('taux', 'devise', 'annee', )
610 list_filter
= ('devise', )
611 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
613 'fields': ('taux', 'devise', 'annee', ),
617 class TypeContratAdmin(admin
.ModelAdmin
):
618 inlines
= (ContratInline
,)
619 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
621 'fields': ('nom', 'nom_long', ),
626 class TypePosteAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
627 search_fields
= ('nom', 'nom_feminin', )
628 list_display
= ('nom', 'famille_emploi', )
629 list_filter
= ('famille_emploi', )
630 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
632 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
637 class TypeRemunerationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
638 list_display
= ('nom', 'type_paiement', 'nature_remuneration', )
639 #inlines = (RemunerationROInline,) utilité?
640 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
642 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
647 class TypeRevalorisationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
648 #inlines = (RemunerationROInline,) utilité?
649 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
656 class ValeurPointAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
657 list_display
= ('_devise_code', '_devise_nom', 'annee', 'valeur', )
658 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
660 'fields': ('valeur', 'devise', 'implantation', 'annee', )
664 def _devise_code(self
, obj
):
665 return obj
.devise
.code
666 _devise_code
.short_description
= "Code de la devise"
668 def _devise_nom(self
, obj
):
669 return obj
.devise
.nom
670 _devise_nom
.short_description
= "Nom de la devise"
673 def calc_remun(dossier
):
674 thisyear
= datetime
.date
.today().year
675 thisyearfilter
= Q(date_debut__year
=thisyear
) |
Q(date_fin__year
=thisyear
)
677 remunnow
= dossier
.rh_remuneration_remunerations
.filter(thisyearfilter
)
681 sums
= defaultdict(int)
682 sums_euro
= defaultdict(int)
684 nature
= r
.type.nature_remuneration
685 sums
[nature
] += r
.montant
686 sums_euro
[nature
] += r
.montant_euro()
687 remun_sum
+= r
.montant
688 remun_sum_euro
+= r
.montant_euro()
692 for n
, s
in sums
.iteritems():
693 remun
[n
] = [sums
[n
], sums_euro
[n
]]
695 return remun
, remun_sum
, remun_sum_euro