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 auf
.django
.metadata
.admin
import AUFMetadataAdminMixin
, AUFMetadataInlineAdminMixin
, AUF_METADATA_READONLY_FIELDS
13 from project
.rh
import models
as rh
14 from forms
import DossierForm
, ContratForm
15 from dae
.utils
import get_employe_from_user
19 # Override of the InlineModelAdmin to support the link in the tabular inline
20 class LinkedInline(admin
.options
.InlineModelAdmin
):
21 template
= "admin/linked.html"
22 admin_model_path
= None
24 def __init__(self
, *args
):
25 super(LinkedInline
, self
).__init__(*args
)
26 if self
.admin_model_path
is None:
27 self
.admin_model_path
= self
.model
.__name__
.lower()
30 class ProtectRegionMixin(object):
32 def queryset(self
, request
):
33 from dae
.workflow
import grp_drh
, grp_correspondants_rh
34 qs
= super(ProtectRegionMixin
, self
).queryset(request
)
36 if request
.user
.is_superuser
:
39 user_groups
= request
.user
.groups
.all()
41 if grp_drh
in user_groups
:
44 if grp_correspondants_rh
in user_groups
:
45 employe
= get_employe_from_user(request
.user
)
46 q
= Q(**{self
.model
.prefix_implantation
: employe
.implantation
.region
})
47 qs
= qs
.filter(q
).distinct()
51 def has_change_permission(self
, request
, obj
=None):
52 if request
.user
.is_superuser
:
56 employe
= get_employe_from_user(request
.user
)
57 if employe
.implantation
.region
in obj
.get_regions():
67 class ReadOnlyInlineMixin(object):
68 def get_readonly_fields(self
, request
, obj
=None):
69 return [f
.name
for f
in self
.model
._meta
.fields
if f
.name
not in AUF_METADATA_READONLY_FIELDS
]
72 class AyantDroitInline(AUFMetadataInlineAdminMixin
, admin
.StackedInline
):
73 model
= models
.Model
# à remplacer dans admin.py
78 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', 'lien_parente', )
83 class AyantDroitCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
84 readonly_fields
= ('owner', )
85 model
= models
.Model
# à remplacer dans admin.py
89 class ContratInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
91 model
= models
.Model
# à remplacer dans admin.py
95 class DossierROInline(ReadOnlyInlineMixin
, LinkedInline
):
96 template
= "admin/rh/dossier/linked.html"
97 exclude
= AUF_METADATA_READONLY_FIELDS
98 model
= models
.Model
# à remplacer dans admin.py
102 def has_add_permission(self
, request
=None):
105 def has_change_permission(self
, request
, obj
=None):
108 def has_delete_permission(self
, request
, obj
=None):
112 class DossierCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
113 readonly_fields
= ('owner', )
114 model
= models
.Model
# à remplacer dans admin.py
118 class DossierPieceInline(admin
.TabularInline
):
119 model
= models
.Model
# à remplacer dans admin.py
123 class EmployeInline(admin
.TabularInline
):
124 model
= models
.Model
# à remplacer dans admin.py
126 class EmployeCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
127 readonly_fields
= ('owner', )
128 model
= models
.Model
# à remplacer dans admin.py
132 class EmployePieceInline(admin
.TabularInline
):
133 model
= models
.Model
# à remplacer dans admin.py
137 class EvenementInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
138 model
= models
.Model
# à remplacer dans admin.py
142 class EvenementRemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
143 model
= models
.Model
# à remplacer dans admin.py
147 class PosteCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
148 readonly_fields
= ('owner', )
149 model
= models
.Model
# à remplacer dans admin.py
153 class PosteFinancementInline(admin
.TabularInline
):
154 model
= models
.Model
# à remplacer dans admin.py
157 class PostePieceInline(admin
.TabularInline
):
158 model
= models
.Model
# à remplacer dans admin.py
161 class RemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
162 model
= models
.Model
# à remplacer dans admin.py
166 class RemunerationROInline(ReadOnlyInlineMixin
, RemunerationInline
):
170 class TypePosteInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
171 model
= models
.Model
# à remplacer dans admin.py
176 class AyantDroitAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
178 L'ajout d'un nouvel ayantdroit se fait dans l'admin de l'employé.
180 alphabet_filter
= 'nom'
181 search_fields
= ('nom', 'prenom', 'employe__nom', 'employe__prenom', )
182 list_display
= ('_employe', 'lien_parente', '_ayantdroit', )
183 inlines
= (AyantDroitCommentaireInline
,)
184 readonly_fields
= AUFMetadataAdminMixin
.readonly_fields
+ ('employe',)
185 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
186 ("Lien avec l'employé", {
187 'fields': (('employe', 'lien_parente'), )
191 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
195 def save_formset(self
, request
, form
, formset
, change
):
196 instances
= formset
.save(commit
=False)
197 for instance
in instances
:
198 if instance
.__class__
== rh
.AyantDroitCommentaire
:
199 instance
.owner
= request
.user
202 def _ayantdroit(self
, obj
):
204 _ayantdroit
.short_description
= u
'Ayant droit'
206 def _employe(self
, obj
):
207 return unicode(obj
.employe
)
208 _employe
.short_description
= u
'Employé'
210 def has_add_permission(self
, request
):
213 class AyantDroitCommentaireAdmin(admin
.ModelAdmin
):
217 class ClassementAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
218 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
220 'fields': ('type', 'echelon', 'degre', 'coefficient', )
225 class CommentaireAdmin(admin
.ModelAdmin
):
229 #class ContratAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
231 # alphabet_filter = 'dossier__employe__nom'
232 # search_fields = ('dossier__employe__nom', 'dossier__employe__prenom', 'dossier__poste__nom', 'dossier__poste__nom_feminin', )
233 # list_display = ('id', '_employe', '_poste', 'date_debut', 'date_fin', '_implantation', )
234 # fieldsets = AUFMetadataAdminMixin.fieldsets + (
236 # 'fields': ('dossier', 'type_contrat', 'date_debut', 'date_fin', )
240 # def lookup_allowed(self, key, value):
241 # if key in ('dossier__employe__nom__istartswith', ):
244 # def _employe(self, obj):
245 # return unicode(obj.dossier.employe)
246 # _employe.short_description = "Employé"
248 # def _poste(self, obj):
249 # return obj.dossier.poste.nom
250 # _poste.short_description = "Poste"
252 # def _implantation(self, obj):
253 # return obj.dossier.poste.implantation
254 # _poste.short_description = "Implantation"
256 class DeviseAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
257 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
259 'fields': ('code', 'nom', ),
264 class DossierAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
,):
266 alphabet_filter
= 'employe__nom'
267 search_fields
= ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
268 list_display
= ('_id', '_employe', '_actif', '_poste', 'date_debut', 'date_fin', 'date_modification')
269 list_filter
= ('poste__implantation__region', 'poste__implantation', )
270 inlines
= (DossierPieceInline
, ContratInline
,
273 DossierCommentaireInline
,
275 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
277 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
280 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
283 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
285 ('Occupation du Poste par cet Employe', {
286 'fields': (('date_debut', 'date_fin'), )
291 js
= ('js/dossier.js',)
293 def lookup_allowed(self
, key
, value
):
295 'employe__nom__istartswith',
297 'poste__implantation__region__id__exact',
298 'poste__implantation__id__exact',
303 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>""" % \
304 (reverse('dossier_apercu', args
=(d
.id,)),
306 reverse('admin:rh_dossier_change', args
=(d
.id,)),
310 _id
.allow_tags
= True
311 _id
.short_description
= u
'Numéro de dossier'
312 _id
.admin_order_field
= 'id'
315 def _actif(self
, dossier
):
316 if dossier
.employe
.actif
:
317 html
= """<img alt="True" src="%simg/admin/icon-yes.gif">"""
319 html
= """<img alt="False" src="%simg/admin/icon-no.gif">"""
320 return html
% settings
.ADMIN_MEDIA_PREFIX
321 _actif
.allow_tags
= True
322 _actif
.short_description
= u
'Employé actif'
323 _actif
.admin_order_field
= 'employe__actif'
325 def _poste(self
, dossier
):
326 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>""" % \
327 (reverse('poste_apercu', args
=(dossier
.poste
.id,)),
329 reverse('admin:rh_poste_change', args
=(dossier
.poste
.id,)),
333 _poste
.allow_tags
= True
334 _poste
.short_description
= u
'Poste'
335 _poste
.admin_order_field
= 'poste__nom'
337 def _employe(self
, obj
):
338 employe
= obj
.employe
339 view_link
= reverse('employe_apercu', args
=(employe
.id,))
340 edit_link
= reverse('admin:rh_employe_change', args
=(employe
.id,))
341 return u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>[%s] %s %s</a>
342 <a href="%s" title="Modifier l'employé"><img src="%simg/user_edit.png" /></a>""" % \
343 (view_link
, employe
.id, employe
.nom
.upper(), employe
.prenom
.title(), edit_link
, settings
.MEDIA_URL
,)
344 _employe
.allow_tags
= True
345 _employe
.short_description
= u
"Employé ([code] NOM Prénom)"
346 _employe
.admin_order_field
= "employe__nom"
348 def save_formset(self
, request
, form
, formset
, change
):
349 instances
= formset
.save(commit
=False)
350 for instance
in instances
:
351 if instance
.__class__
== rh
.DossierCommentaire
:
352 instance
.owner
= request
.user
356 class DossierPieceAdmin(admin
.ModelAdmin
):
360 class DossierCommentaireAdmin(admin
.ModelAdmin
):
364 class EmployeAdminForm(forms
.ModelForm
):
368 def __init__(self
, *args
, **kwargs
):
369 super(EmployeAdminForm
, self
).__init__(*args
, **kwargs
)
370 self
.fields
['date_naissance'].widget
= forms
.widgets
.DateInput()
373 class EmployeAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
374 alphabet_filter
= 'nom'
375 DEFAULT_ALPHABET
= u
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
376 search_fields
= ('id', 'nom', 'prenom', 'nom_affichage', )
378 actions
= ('desactiver', )
379 form
= EmployeAdminForm
380 list_display
= ('_nom', '_dossiers', 'date_modification', 'user_modification',)
381 list_filter
= ('rh_dossiers__poste__implantation__region', 'rh_dossiers__poste__implantation', 'actif', )
382 inlines
= (AyantDroitInline
,
385 EmployeCommentaireInline
)
386 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
388 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
390 ('Informations personnelles', {
391 'fields': ('situation_famille', 'date_entree', )
394 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
399 view_link
= reverse('employe_apercu', args
=(obj
.id,))
400 edit_link
= reverse('admin:rh_employe_change', args
=(obj
.id,))
401 return u
"""<a onclick="return showAddAnotherPopup(this);" href='%s'>[%s] %s %s</a>
402 <a href="%s" title="Modifier l'employé"><img src="%simg/user_edit.png" /></a>""" % \
403 (view_link
, obj
.id, obj
.nom
.upper(), obj
.prenom
.title(), edit_link
, settings
.MEDIA_URL
,)
404 _nom
.allow_tags
= True
405 _nom
.short_description
= u
"Employé ([code] NOM Prénom)"
406 _nom
.admin_order_field
= "nom"
408 def _dossiers(self
, obj
):
410 for d
in obj
.dossiers
.all().order_by('-date_debut'):
412 if d
.date_fin
is not None:
413 style
= "color: grey";
414 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>""" % \
416 reverse('dossier_apercu', args
=(d
.id,)),
419 reverse('admin:rh_dossier_change', args
=(d
.id,)),
423 return "<ul>%s</ul>" % "\n".join(l
)
424 _dossiers
.allow_tags
= True
426 def queryset(self
, request
):
427 qs
= super(EmployeAdmin
, self
).queryset(request
)
428 return qs
.filter(actif
=True).select_related(depth
=1).order_by('nom')
430 def save_formset(self
, request
, form
, formset
, change
):
431 instances
= formset
.save(commit
=False)
432 for instance
in instances
:
433 if instance
.__class__
== rh
.EmployeCommentaire
:
434 instance
.owner
= request
.user
439 class EmployeCommentaireAdmin(admin
.ModelAdmin
):
443 class EmployePieceAdmin(admin
.ModelAdmin
):
447 class EvenementAdmin(admin
.ModelAdmin
):
448 inlines
= (EvenementRemunerationInline
,)
451 class EvenementRemunerationAdmin(admin
.ModelAdmin
):
455 class FamilleEmploiAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
456 inlines
= (TypePosteInline
,)
457 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
464 class OrganismeBstgAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
465 search_fields
= ('nom', )
466 list_display
= ('nom', 'type', 'pays', )
467 inlines
= (DossierROInline
,)
468 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
470 'fields': ('nom', 'type', 'pays', )
475 class PosteAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
476 alphabet_filter
= 'nom'
477 search_fields
= ('nom',
478 'implantation__code',
480 'implantation__region__code',
481 'implantation__region__nom',
493 list_filter
= ('service',
494 'implantation__region',
497 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
499 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
500 'service', 'responsable')
503 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
506 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
509 'fields': (('classement_min', 'classement_max'),
510 ('valeur_point_min', 'valeur_point_max'),
511 ('devise_min', 'devise_max'),
512 ('salaire_min', 'salaire_max'),
513 ('indemn_min', 'indemn_max'),
514 ('autre_min', 'autre_max'))
516 ('Comparatifs de rémunération', {
517 'fields': ('devise_comparaison',
518 ('comp_locale_min', 'comp_locale_max'),
519 ('comp_universite_min', 'comp_universite_max'),
520 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
521 ('comp_ong_min', 'comp_ong_max'),
522 ('comp_autre_min', 'comp_autre_max'))
525 'fields': ('justification',)
527 ('Autres Metadata', {
528 'fields': ('date_validation', ('date_debut', 'date_fin'))
532 inlines
= (PosteFinancementInline
,
535 PosteCommentaireInline
, )
537 def _nom(self
, poste
):
538 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>""" % \
539 (reverse('poste_apercu', args
=(poste
.id,)),
541 reverse('admin:rh_poste_change', args
=(poste
.id,)),
545 _nom
.allow_tags
= True
546 _nom
.short_description
= u
'Nom'
547 _nom
.admin_order_field
= 'nom'
549 def _occupe_par(self
, obj
):
550 """Formatte la méthode Poste.occupe_par() pour l'admin"""
552 employes
= obj
.occupe_par()
556 link
= "<a href='%s'>%s</a>" % \
557 (reverse('admin:rh_employe_change', args
=(e
.id,)),
560 output
= "\n<br />".join(l
)
562 _occupe_par
.allow_tags
= True
563 _occupe_par
.short_description
= "Occupé par"
565 def save_formset(self
, request
, form
, formset
, change
):
566 instances
= formset
.save(commit
=False)
567 for instance
in instances
:
568 if instance
.__class__
== rh
.PosteCommentaire
:
569 instance
.owner
= request
.user
574 class PosteCommentaireAdmin(admin
.ModelAdmin
):
578 class PosteFinancementAdmin(admin
.ModelAdmin
):
582 class PostePieceAdmin(admin
.ModelAdmin
):
586 class RemunerationAdmin(admin
.ModelAdmin
):
590 class ResponsableImplantationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
591 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
593 'fields': ('employe', 'implantation', ),
598 class ServiceAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
599 list_display
= ('nom', 'actif', )
600 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
606 class StatutAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
607 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
609 'fields': ('code', 'nom', ),
613 class TauxChangeAdmin(admin
.ModelAdmin
):
614 list_display
= ('taux', 'devise', 'annee', )
615 list_filter
= ('devise', )
616 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
618 'fields': ('taux', 'devise', 'annee', ),
622 class TypeContratAdmin(admin
.ModelAdmin
):
623 inlines
= (ContratInline
,)
624 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
626 'fields': ('nom', 'nom_long', ),
631 class TypePosteAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
632 search_fields
= ('nom', 'nom_feminin', )
633 list_display
= ('nom', 'famille_emploi', )
634 list_filter
= ('famille_emploi', )
635 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
637 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
642 class TypeRemunerationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
643 list_display
= ('nom', 'type_paiement', 'nature_remuneration', )
644 #inlines = (RemunerationROInline,) utilité?
645 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
647 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
652 class TypeRevalorisationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
653 #inlines = (RemunerationROInline,) utilité?
654 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
661 class ValeurPointAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
662 list_display
= ('_devise_code', '_devise_nom', 'annee', 'valeur', )
663 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
665 'fields': ('valeur', 'devise', 'implantation', 'annee', )
669 def _devise_code(self
, obj
):
670 return obj
.devise
.code
671 _devise_code
.short_description
= "Code de la devise"
673 def _devise_nom(self
, obj
):
674 return obj
.devise
.nom
675 _devise_nom
.short_description
= "Nom de la devise"
678 def calc_remun(dossier
):
679 thisyear
= datetime
.date
.today().year
680 thisyearfilter
= Q(date_debut__year
=thisyear
) |
Q(date_fin__year
=thisyear
)
682 remunnow
= dossier
.rh_remuneration_remunerations
.filter(thisyearfilter
)
686 sums
= defaultdict(int)
687 sums_euro
= defaultdict(int)
689 nature
= r
.type.nature_remuneration
690 sums
[nature
] += r
.montant
691 sums_euro
[nature
] += r
.montant_euro()
692 remun_sum
+= r
.montant
693 remun_sum_euro
+= r
.montant_euro()
697 for n
, s
in sums
.iteritems():
698 remun
[n
] = [sums
[n
], sums_euro
[n
]]
700 return remun
, remun_sum
, remun_sum_euro