1 # -*- encoding: utf-8 -*-
3 from collections
import defaultdict
6 from django
.db
import models
7 from django
import forms
8 from django
.contrib
import admin
9 from django
.conf
import settings
10 from django
.db
.models
import Q
11 from auf
.django
.metadata
.admin
import AUFMetadataAdminMixin
, AUFMetadataInlineAdminMixin
, AUF_METADATA_READONLY_FIELDS
12 from project
.rh
import models
as rh
13 from forms
import DossierForm
, ContratForm
14 from dae
.utils
import get_employe_from_user
18 # Override of the InlineModelAdmin to support the link in the tabular inline
19 class LinkedInline(admin
.options
.InlineModelAdmin
):
20 template
= "admin/linked.html"
21 admin_model_path
= None
23 def __init__(self
, *args
):
24 super(LinkedInline
, self
).__init__(*args
)
25 if self
.admin_model_path
is None:
26 self
.admin_model_path
= self
.model
.__name__
.lower()
29 class ProtectRegionMixin(object):
31 def queryset(self
, request
):
32 qs
= super(ProtectRegionMixin
, self
).queryset(request
)
34 if request
.user
.is_superuser
:
37 employe
= get_employe_from_user(request
.user
)
39 q
= Q(**{self
.model
.prefix_implantation
: employe
.implantation
.region
})
40 qs
= qs
.filter(q
).distinct()
43 def has_change_permission(self
, request
, obj
=None):
44 if request
.user
.is_superuser
:
48 employe
= get_employe_from_user(request
.user
)
49 if employe
.implantation
.region
in obj
.get_regions():
59 class ReadOnlyInlineMixin(object):
60 def get_readonly_fields(self
, request
, obj
=None):
61 return [f
.name
for f
in self
.model
._meta
.fields
if f
.name
not in AUF_METADATA_READONLY_FIELDS
]
64 class AyantDroitInline(AUFMetadataInlineAdminMixin
, admin
.StackedInline
):
65 model
= models
.Model
# à remplacer dans admin.py
70 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', 'lien_parente', )
75 class AyantDroitCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
76 readonly_fields
= ('owner', )
77 model
= models
.Model
# à remplacer dans admin.py
81 class ContratInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
83 model
= models
.Model
# à remplacer dans admin.py
87 class DossierROInline(ReadOnlyInlineMixin
, LinkedInline
):
88 exclude
= AUF_METADATA_READONLY_FIELDS
89 model
= models
.Model
# à remplacer dans admin.py
93 def has_add_permission(self
, request
):
96 def has_change_permission(self
, request
, obj
=None):
99 def has_delete_permission(self
, request
, obj
=None):
103 class DossierCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
104 readonly_fields
= ('owner', )
105 model
= models
.Model
# à remplacer dans admin.py
109 class DossierPieceInline(admin
.TabularInline
):
110 model
= models
.Model
# à remplacer dans admin.py
114 class EmployeInline(admin
.TabularInline
):
115 model
= models
.Model
# à remplacer dans admin.py
117 class EmployeCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
118 readonly_fields
= ('owner', )
119 model
= models
.Model
# à remplacer dans admin.py
123 class EmployePieceInline(admin
.TabularInline
):
124 model
= models
.Model
# à remplacer dans admin.py
128 class EvenementInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
129 model
= models
.Model
# à remplacer dans admin.py
133 class EvenementRemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
134 model
= models
.Model
# à remplacer dans admin.py
138 class PosteCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
139 readonly_fields
= ('owner', )
140 model
= models
.Model
# à remplacer dans admin.py
144 class PosteFinancementInline(admin
.TabularInline
):
145 model
= models
.Model
# à remplacer dans admin.py
148 class PostePieceInline(admin
.TabularInline
):
149 model
= models
.Model
# à remplacer dans admin.py
152 class RemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
153 model
= models
.Model
# à remplacer dans admin.py
157 class RemunerationROInline(ReadOnlyInlineMixin
, RemunerationInline
):
161 class TypePosteInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
162 model
= models
.Model
# à remplacer dans admin.py
167 class AyantDroitAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
169 L'ajout d'un nouvel ayantdroit se fait dans l'admin de l'employé.
171 alphabet_filter
= 'nom'
172 search_fields
= ('nom', 'prenom', 'employe__nom', 'employe__prenom', )
173 list_display
= ('_employe', 'lien_parente', '_ayantdroit', )
174 inlines
= (AyantDroitCommentaireInline
,)
175 readonly_fields
= AUFMetadataAdminMixin
.readonly_fields
+ ('employe',)
176 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
177 ("Lien avec l'employé", {
178 'fields': (('employe', 'lien_parente'), )
182 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
186 def save_formset(self
, request
, form
, formset
, change
):
187 instances
= formset
.save(commit
=False)
188 for instance
in instances
:
189 if instance
.__class__
== rh
.AyantDroitCommentaire
:
190 instance
.owner
= request
.user
193 def _ayantdroit(self
, obj
):
195 _ayantdroit
.short_description
= u
'Ayant droit'
197 def _employe(self
, obj
):
198 return unicode(obj
.employe
)
199 _employe
.short_description
= u
'Employé'
201 def has_add_permission(self
, request
):
204 class AyantDroitCommentaireAdmin(admin
.ModelAdmin
):
208 class ClassementAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
209 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
211 'fields': ('type', 'echelon', 'degre', 'coefficient', )
216 class CommentaireAdmin(admin
.ModelAdmin
):
220 #class ContratAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
222 # alphabet_filter = 'dossier__employe__nom'
223 # search_fields = ('dossier__employe__nom', 'dossier__employe__prenom', 'dossier__poste__nom', 'dossier__poste__nom_feminin', )
224 # list_display = ('id', '_employe', '_poste', 'date_debut', 'date_fin', '_implantation', )
225 # fieldsets = AUFMetadataAdminMixin.fieldsets + (
227 # 'fields': ('dossier', 'type_contrat', 'date_debut', 'date_fin', )
231 # def lookup_allowed(self, key, value):
232 # if key in ('dossier__employe__nom__istartswith', ):
235 # def _employe(self, obj):
236 # return unicode(obj.dossier.employe)
237 # _employe.short_description = "Employé"
239 # def _poste(self, obj):
240 # return obj.dossier.poste.nom
241 # _poste.short_description = "Poste"
243 # def _implantation(self, obj):
244 # return obj.dossier.poste.implantation
245 # _poste.short_description = "Implantation"
247 class DeviseAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
248 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
250 'fields': ('code', 'nom', ),
255 class DossierAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
,):
257 alphabet_filter
= 'employe__nom'
258 search_fields
= ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
259 list_display
= ('_employe', '_poste', 'date_debut', 'date_fin', 'date_modification', '_actif')
260 inlines
= (DossierPieceInline
, ContratInline
,
263 DossierCommentaireInline
,
265 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
267 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
270 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
273 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
275 ('Occupation du Poste par cet Employe', {
276 'fields': (('date_debut', 'date_fin'), )
280 add_form_template
= 'admin/change_form.html'
282 def queryset(self
, request
):
283 return self
.model
.actifs
.all()
286 js
= ('js/dossier.js',)
288 def lookup_allowed(self
, key
, value
):
289 if key
in ('employe__nom__istartswith', 'actif__exact', ):
292 def _actif(self
, dossier
):
293 if dossier
.employe
.actif
:
294 html
= """<img alt="True" src="%simg/admin/icon-yes.gif">"""
296 html
= """<img alt="False" src="%simg/admin/icon-no.gif">"""
297 return html
% settings
.ADMIN_MEDIA_PREFIX
298 _actif
.allow_tags
= u
'Employé actif'
299 _actif
.short_description
= u
'Employé actif'
300 _actif
.admin_order_field
= 'employe__actif'
302 def _poste(self
, dossier
):
303 return unicode(dossier
.poste
.nom
)
304 _poste
.short_description
= u
'Poste'
305 _poste
.admin_order_field
= 'poste__nom'
307 def _employe(self
, dossier
):
308 return unicode(dossier
.employe
)
309 _employe
.short_description
= u
'Employé'
310 _employe
.admin_order_field
= 'employe__nom'
312 def save_formset(self
, request
, form
, formset
, change
):
313 instances
= formset
.save(commit
=False)
314 for instance
in instances
:
315 if instance
.__class__
== rh
.DossierCommentaire
:
316 instance
.owner
= request
.user
319 def render_change_form(self
, request
, context
, *args
, **kwargs
):
320 obj
= kwargs
.get('obj', None)
323 return super(DossierAdmin
, self
).render_change_form(request
, context
, *args
, **kwargs
)
326 remun
, remun_sum
, remun_sum_euro
= calc_remun(obj
)
330 'remun_sum': remun_sum
,
331 'remun_sum_euro': remun_sum_euro
,
332 'employe': obj
.employe
,
333 'poste_nom': obj
.poste
.nom
,
334 'poste_service': obj
.poste
.service
,
335 'poste_implantation': obj
.poste
.implantation
,
338 context
.update(extra
)
340 return super(DossierAdmin
, self
).render_change_form(request
, context
, *args
, **kwargs
)
343 class DossierInactifAdmin(DossierAdmin
):
344 def queryset(self
, request
):
345 return self
.model
.inactifs
.all()
348 class DossierPieceAdmin(admin
.ModelAdmin
):
352 class DossierCommentaireAdmin(admin
.ModelAdmin
):
356 class EmployeAdminForm(forms
.ModelForm
):
360 def __init__(self
, *args
, **kwargs
):
361 super(EmployeAdminForm
, self
).__init__(*args
, **kwargs
)
362 self
.fields
['date_naissance'].widget
= forms
.widgets
.DateInput()
365 class EmployeAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
366 alphabet_filter
= 'nom'
367 search_fields
= ('id', 'nom', 'prenom', 'nom_affichage', )
369 actions
= ('desactiver', )
370 form
= EmployeAdminForm
371 list_display
= ('id', 'nom', 'prenom', )
372 list_display_links
= ('id', 'nom',)
373 inlines
= (AyantDroitInline
,
376 EmployeCommentaireInline
)
377 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
379 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
381 ('Informations personnelles', {
382 'fields': ('situation_famille', 'date_entree', )
385 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
389 add_form_template
= 'admin/change_form.html'
391 def queryset(self
, request
):
392 return self
.model
.actifs
.all()
394 def save_formset(self
, request
, form
, formset
, change
):
395 instances
= formset
.save(commit
=False)
396 for instance
in instances
:
397 if instance
.__class__
== rh
.EmployeCommentaire
:
398 instance
.owner
= request
.user
401 def render_change_form(self
, request
, context
, *args
, **kwargs
):
402 obj
= kwargs
.get('obj', None)
405 return super(EmployeAdmin
, self
).render_change_form(request
, context
, *args
, **kwargs
)
410 dossiers
= obj
.dossiers
.all()
412 for dossier
in dossiers
:
413 this_remun
, this_remun_sum
, this_remun_sum_euro
= calc_remun(dossier
)
415 for item
in this_remun
:
416 if item
not in remun
:
417 remun
[item
] = this_remun
[item
]
419 remun
[item
][0] += this_remun
[item
][0]
420 remun
[item
][1] += this_remun
[item
][1]
422 remun_sum
+= this_remun_sum
423 remun_sum_euro
+= this_remun_sum_euro
427 'remun_sum': remun_sum
,
428 'remun_sum_euro': remun_sum_euro
,
431 context
.update(extra
)
433 return super(EmployeAdmin
, self
).render_change_form(request
, context
, *args
, **kwargs
)
436 class EmployeInactifAdmin(EmployeAdmin
):
437 def queryset(self
, request
):
438 return self
.model
.inactifs
.all()
441 class EmployeCommentaireAdmin(admin
.ModelAdmin
):
445 class EmployePieceAdmin(admin
.ModelAdmin
):
449 class EvenementAdmin(admin
.ModelAdmin
):
450 inlines
= (EvenementRemunerationInline
,)
453 class EvenementRemunerationAdmin(admin
.ModelAdmin
):
457 class FamilleEmploiAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
458 inlines
= (TypePosteInline
,)
459 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
466 class OrganismeBstgAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
467 search_fields
= ('nom', )
468 list_display
= ('nom', 'type', 'pays', )
469 inlines
= (DossierROInline
,)
470 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
472 'fields': ('nom', 'type', 'pays', )
477 class PosteAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
478 alphabet_filter
= 'nom'
479 search_fields
= ('nom', 'implantation__code', 'implantation__nom', 'implantation__region__code', 'implantation__region__nom', )
480 list_display
= ('nom', 'implantation', 'service', 'type_poste', 'date_debut', 'date_fin', )
481 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
483 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
484 'service', 'responsable')
487 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
490 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
493 'fields': (('classement_min', 'classement_max'),
494 ('valeur_point_min', 'valeur_point_max'),
495 ('devise_min', 'devise_max'),
496 ('salaire_min', 'salaire_max'),
497 ('indemn_min', 'indemn_max'),
498 ('autre_min', 'autre_max'))
500 ('Comparatifs de rémunération', {
501 'fields': ('devise_comparaison',
502 ('comp_locale_min', 'comp_locale_max'),
503 ('comp_universite_min', 'comp_universite_max'),
504 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
505 ('comp_ong_min', 'comp_ong_max'),
506 ('comp_autre_min', 'comp_autre_max'))
509 'fields': ('justification',)
511 ('Autres Metadata', {
512 'fields': ('date_validation', ('date_debut', 'date_fin'))
516 inlines
= (PosteFinancementInline
,
519 PosteCommentaireInline
, )
521 def save_formset(self
, request
, form
, formset
, change
):
522 instances
= formset
.save(commit
=False)
523 for instance
in instances
:
524 if instance
.__class__
== rh
.PosteCommentaire
:
525 instance
.owner
= request
.user
530 class PosteCommentaireAdmin(admin
.ModelAdmin
):
534 class PosteFinancementAdmin(admin
.ModelAdmin
):
538 class PostePieceAdmin(admin
.ModelAdmin
):
542 class RemunerationAdmin(admin
.ModelAdmin
):
546 class ResponsableImplantationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
547 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
549 'fields': ('employe', 'implantation', ),
554 class ServiceAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
555 list_display
= ('nom', 'actif', )
556 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
562 class StatutAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
563 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
565 'fields': ('code', 'nom', ),
569 class TauxChangeAdmin(admin
.ModelAdmin
):
570 list_display
= ('taux', 'devise', 'annee', )
571 list_filter
= ('devise', )
572 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
574 'fields': ('taux', 'devise', 'annee', ),
578 class TypeContratAdmin(admin
.ModelAdmin
):
579 inlines
= (ContratInline
,)
580 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
582 'fields': ('nom', 'nom_long', ),
587 class TypePosteAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
588 search_fields
= ('nom', 'nom_feminin', )
589 list_display
= ('nom', 'famille_emploi', )
590 list_filter
= ('famille_emploi', )
591 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
593 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
598 class TypeRemunerationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
599 list_display
= ('nom', 'type_paiement', 'nature_remuneration', )
600 #inlines = (RemunerationROInline,) utilité?
601 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
603 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
608 class TypeRevalorisationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
609 #inlines = (RemunerationROInline,) utilité?
610 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
617 class ValeurPointAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
618 list_display
= ('_devise_code', '_devise_nom', 'annee', 'valeur', )
619 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
621 'fields': ('valeur', 'devise', 'implantation', 'annee', )
625 def _devise_code(self
, obj
):
626 return obj
.devise
.code
627 _devise_code
.short_description
= "Code de la devise"
629 def _devise_nom(self
, obj
):
630 return obj
.devise
.nom
631 _devise_nom
.short_description
= "Nom de la devise"
634 def calc_remun(dossier
):
635 thisyear
= datetime
.date
.today().year
636 thisyearfilter
= Q(date_debut__year
=thisyear
) |
Q(date_fin__year
=thisyear
)
638 remunnow
= dossier
.rh_remuneration_remunerations
.filter(thisyearfilter
)
642 sums
= defaultdict(int)
643 sums_euro
= defaultdict(int)
645 nature
= r
.type.nature_remuneration
646 sums
[nature
] += r
.montant
647 sums_euro
[nature
] += r
.montant_euro()
648 remun_sum
+= r
.montant
649 remun_sum_euro
+= r
.montant_euro()
653 for n
, s
in sums
.iteritems():
654 remun
[n
] = [sums
[n
], sums_euro
[n
]]
656 return remun
, remun_sum
, remun_sum_euro