e915ed880e360de5a5d3598af57fa0f99a37c3f8
[auf_rh_dae.git] / project / rh / lib.py
1 # -*- encoding: utf-8 -*-
2
3 from collections import defaultdict
4 import datetime
5
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
16
17
18
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
23
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()
28
29
30 class ProtectRegionMixin(object):
31
32 def queryset(self, request):
33 qs = super(ProtectRegionMixin, self).queryset(request)
34
35 if request.user.is_superuser:
36 return qs
37
38 employe = get_employe_from_user(request.user)
39
40 q = Q(**{self.model.prefix_implantation: employe.implantation.region})
41 qs = qs.filter(q).distinct()
42 return qs
43
44 def has_change_permission(self, request, obj=None):
45 if request.user.is_superuser:
46 return True
47
48 if obj:
49 employe = get_employe_from_user(request.user)
50 if employe.implantation.region in obj.get_regions():
51 return True
52 else:
53 return False
54
55 return True
56
57
58 # Inlines
59
60 class ReadOnlyInlineMixin(object):
61 def get_readonly_fields(self, request, obj=None):
62 return [f.name for f in self.model._meta.fields if f.name not in AUF_METADATA_READONLY_FIELDS]
63
64
65 class AyantDroitInline(AUFMetadataInlineAdminMixin, admin.StackedInline):
66 model = models.Model # à remplacer dans admin.py
67 extra = 0
68
69 fieldsets = (
70 (None, {
71 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', 'lien_parente', )
72 }),
73 )
74
75
76 class AyantDroitCommentaireInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
77 readonly_fields = ('owner', )
78 model = models.Model # à remplacer dans admin.py
79 extra = 1
80
81
82 class ContratInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
83 form = ContratForm
84 model = models.Model # à remplacer dans admin.py
85 extra = 1
86
87
88 class DossierROInline(ReadOnlyInlineMixin, LinkedInline):
89 template = "admin/rh/dossier/linked.html"
90 exclude = AUF_METADATA_READONLY_FIELDS
91 model = models.Model # à remplacer dans admin.py
92 extra = 0
93 can_delete = False
94
95 def has_add_permission(self, request=None):
96 return False
97
98 def has_change_permission(self, request, obj=None):
99 return False
100
101 def has_delete_permission(self, request, obj=None):
102 return False
103
104
105 class DossierCommentaireInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
106 readonly_fields = ('owner', )
107 model = models.Model # à remplacer dans admin.py
108 extra = 1
109
110
111 class DossierPieceInline(admin.TabularInline):
112 model = models.Model # à remplacer dans admin.py
113 extra = 4
114
115
116 class EmployeInline(admin.TabularInline):
117 model = models.Model # à remplacer dans admin.py
118
119 class EmployeCommentaireInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
120 readonly_fields = ('owner', )
121 model = models.Model # à remplacer dans admin.py
122 extra = 1
123
124
125 class EmployePieceInline(admin.TabularInline):
126 model = models.Model # à remplacer dans admin.py
127 extra = 4
128
129
130 class EvenementInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
131 model = models.Model # à remplacer dans admin.py
132 extra = 1
133
134
135 class EvenementRemunerationInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
136 model = models.Model # à remplacer dans admin.py
137 extra = 1
138
139
140 class PosteCommentaireInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
141 readonly_fields = ('owner', )
142 model = models.Model # à remplacer dans admin.py
143 extra = 1
144
145
146 class PosteFinancementInline(admin.TabularInline):
147 model = models.Model # à remplacer dans admin.py
148
149
150 class PostePieceInline(admin.TabularInline):
151 model = models.Model # à remplacer dans admin.py
152
153
154 class RemunerationInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
155 model = models.Model # à remplacer dans admin.py
156 extra = 1
157
158
159 class RemunerationROInline(ReadOnlyInlineMixin, RemunerationInline):
160 pass
161
162
163 class TypePosteInline(AUFMetadataInlineAdminMixin, admin.TabularInline):
164 model = models.Model # à remplacer dans admin.py
165
166
167 # Admins
168
169 class AyantDroitAdmin(AUFMetadataAdminMixin, ProtectRegionMixin, admin.ModelAdmin):
170 """
171 L'ajout d'un nouvel ayantdroit se fait dans l'admin de l'employé.
172 """
173 alphabet_filter = 'nom'
174 search_fields = ('nom', 'prenom', 'employe__nom', 'employe__prenom', )
175 list_display = ('_employe', 'lien_parente', '_ayantdroit', )
176 inlines = (AyantDroitCommentaireInline,)
177 readonly_fields = AUFMetadataAdminMixin.readonly_fields + ('employe',)
178 fieldsets = AUFMetadataAdminMixin.fieldsets + (
179 ("Lien avec l'employé", {
180 'fields': (('employe', 'lien_parente'), )
181 }),
182
183 ('Identification', {
184 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
185 }),
186 )
187
188 def save_formset(self, request, form, formset, change):
189 instances = formset.save(commit=False)
190 for instance in instances:
191 if instance.__class__ == rh.AyantDroitCommentaire:
192 instance.owner = request.user
193 instance.save()
194
195 def _ayantdroit(self, obj):
196 return unicode(obj)
197 _ayantdroit.short_description = u'Ayant droit'
198
199 def _employe(self, obj):
200 return unicode(obj.employe)
201 _employe.short_description = u'Employé'
202
203 def has_add_permission(self, request):
204 return False
205
206 class AyantDroitCommentaireAdmin(admin.ModelAdmin):
207 pass
208
209
210 class ClassementAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
211 fieldsets = AUFMetadataAdminMixin.fieldsets + (
212 (None, {
213 'fields': ('type', 'echelon', 'degre', 'coefficient', )
214 }),
215 )
216
217
218 class CommentaireAdmin(admin.ModelAdmin):
219 pass
220
221
222 #class ContratAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
223 # form = ContratForm
224 # alphabet_filter = 'dossier__employe__nom'
225 # search_fields = ('dossier__employe__nom', 'dossier__employe__prenom', 'dossier__poste__nom', 'dossier__poste__nom_feminin', )
226 # list_display = ('id', '_employe', '_poste', 'date_debut', 'date_fin', '_implantation', )
227 # fieldsets = AUFMetadataAdminMixin.fieldsets + (
228 # (None, {
229 # 'fields': ('dossier', 'type_contrat', 'date_debut', 'date_fin', )
230 # }),
231 # )
232 #
233 # def lookup_allowed(self, key, value):
234 # if key in ('dossier__employe__nom__istartswith', ):
235 # return True
236 #
237 # def _employe(self, obj):
238 # return unicode(obj.dossier.employe)
239 # _employe.short_description = "Employé"
240 #
241 # def _poste(self, obj):
242 # return obj.dossier.poste.nom
243 # _poste.short_description = "Poste"
244 #
245 # def _implantation(self, obj):
246 # return obj.dossier.poste.implantation
247 # _poste.short_description = "Implantation"
248
249 class DeviseAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
250 fieldsets = AUFMetadataAdminMixin.fieldsets + (
251 (None, {
252 'fields': ('code', 'nom', ),
253 }),
254 )
255
256
257 class DossierAdmin(AUFMetadataAdminMixin, ProtectRegionMixin, admin.ModelAdmin,):
258 form = DossierForm
259 alphabet_filter = 'employe__nom'
260 search_fields = ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
261 list_display = ('_id', '_employe', '_actif', '_poste', 'date_debut', 'date_fin', 'date_modification')
262 list_filter = ('poste__implantation__region', 'poste__implantation', 'date_debut', 'date_fin', )
263 inlines = (DossierPieceInline, ContratInline,
264 RemunerationInline,
265 #EvenementInline,
266 DossierCommentaireInline,
267 )
268 fieldsets = AUFMetadataAdminMixin.fieldsets + (
269 (None, {
270 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
271 }),
272 ('Recrutement', {
273 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
274 }),
275 ('Rémunération', {
276 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
277 }),
278 ('Occupation du Poste par cet Employe', {
279 'fields': (('date_debut', 'date_fin'), )
280 }),
281 )
282
283 def queryset(self, request):
284 return self.model.actifs.all()
285
286 class Media:
287 js = ('js/dossier.js',)
288
289 def lookup_allowed(self, key, value):
290 if key in (
291 'employe__nom__istartswith',
292 'actif__exact',
293 'date_fin__year',
294 'date_fin__month',
295 'date_fin__day',
296 'date_debut__year',
297 'date_debut__month',
298 'date_debut__day',
299 ):
300 return True
301
302 def _id(self, d):
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,)),
305 d.id,
306 reverse('admin:rh_dossier_change', args=(d.id,)),
307 settings.MEDIA_URL,
308 )
309 return link
310 _id.allow_tags = True
311 _id.short_description = u'Numéro de dossier'
312 _id.admin_order_field = 'id'
313
314
315 def _actif(self, dossier):
316 if dossier.employe.actif:
317 html = """<img alt="True" src="%simg/admin/icon-yes.gif">"""
318 else:
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'
324
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,)),
328 dossier.poste,
329 reverse('admin:rh_poste_change', args=(dossier.poste.id,)),
330 settings.MEDIA_URL,
331 )
332 return link
333 _poste.allow_tags = True
334 _poste.short_description = u'Poste'
335 _poste.admin_order_field = 'poste__nom'
336
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"
347
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
353 instance.save()
354
355
356 class DossierPieceAdmin(admin.ModelAdmin):
357 pass
358
359
360 class DossierCommentaireAdmin(admin.ModelAdmin):
361 pass
362
363
364 class EmployeAdminForm(forms.ModelForm):
365 class Meta:
366 model = rh.Employe
367
368 def __init__(self, *args, **kwargs):
369 super(EmployeAdminForm, self).__init__(*args, **kwargs)
370 self.fields['date_naissance'].widget = forms.widgets.DateInput()
371
372
373 class EmployeAdmin(AUFMetadataAdminMixin, ProtectRegionMixin, admin.ModelAdmin):
374 alphabet_filter = 'nom'
375 DEFAULT_ALPHABET = u'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
376 search_fields = ('id', 'nom', 'prenom', 'nom_affichage', )
377 ordering = ('nom', )
378 actions = ('desactiver', )
379 form = EmployeAdminForm
380 list_display = ('_nom', '_dossiers', 'date_modification', 'user_modification',)
381 list_filter = ('dossiers__poste__implantation__region', 'dossiers__poste__implantation', 'actif', )
382 inlines = (AyantDroitInline,
383 DossierROInline,
384 EmployePieceInline,
385 EmployeCommentaireInline)
386 fieldsets = AUFMetadataAdminMixin.fieldsets + (
387 ('Identification', {
388 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
389 }),
390 ('Informations personnelles', {
391 'fields': ('situation_famille', 'date_entree', )
392 }),
393 ('Coordonnées', {
394 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
395 }),
396 )
397
398 def _nom(self, obj):
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"
407
408 def _dossiers(self, obj):
409 l = []
410 for d in obj.dossiers.all().order_by('-date_debut'):
411 style = ""
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>""" % \
415 (style,
416 reverse('dossier_apercu', args=(d.id,)),
417 d.date_debut.year,
418 d.poste,
419 reverse('admin:rh_dossier_change', args=(d.id,)),
420 settings.MEDIA_URL,
421 )
422 l.append(link)
423 return "<ul>%s</ul>" % "\n".join(l)
424 _dossiers.allow_tags = True
425
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')
429
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
435 instance.save()
436
437
438
439 class EmployeCommentaireAdmin(admin.ModelAdmin):
440 pass
441
442
443 class EmployePieceAdmin(admin.ModelAdmin):
444 pass
445
446
447 class EvenementAdmin(admin.ModelAdmin):
448 inlines = (EvenementRemunerationInline,)
449
450
451 class EvenementRemunerationAdmin(admin.ModelAdmin):
452 pass
453
454
455 class FamilleEmploiAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
456 inlines = (TypePosteInline,)
457 fieldsets = AUFMetadataAdminMixin.fieldsets + (
458 (None, {
459 'fields': ('nom', )
460 }),
461 )
462
463
464 class OrganismeBstgAdmin(AUFMetadataAdminMixin, ProtectRegionMixin, admin.ModelAdmin):
465 search_fields = ('nom', )
466 list_display = ('nom', 'type', 'pays', )
467 inlines = (DossierROInline,)
468 fieldsets = AUFMetadataAdminMixin.fieldsets + (
469 (None, {
470 'fields': ('nom', 'type', 'pays', )
471 }),
472 )
473
474
475 class PosteAdmin(AUFMetadataAdminMixin, ProtectRegionMixin, admin.ModelAdmin):
476 alphabet_filter = 'nom'
477 search_fields = ('nom',
478 'implantation__code',
479 'implantation__nom',
480 'implantation__region__code',
481 'implantation__region__nom',
482 )
483 list_display = (
484 '_nom',
485 '_occupe_par',
486 'implantation',
487 'service',
488 'type_poste',
489 'date_debut',
490 'date_fin',
491 'date_modification',
492 'user_modification',
493 )
494 list_filter = ('service',
495 'implantation__region',
496 'implantation__adresse_physique_pays',
497 'implantation',
498 )
499 fieldsets = AUFMetadataAdminMixin.fieldsets + (
500 (None, {
501 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
502 'service', 'responsable')
503 }),
504 ('Contrat', {
505 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
506 }),
507 ('Recrutement', {
508 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
509 }),
510 ('Rémunération', {
511 'fields': (('classement_min', 'classement_max'),
512 ('valeur_point_min', 'valeur_point_max'),
513 ('devise_min', 'devise_max'),
514 ('salaire_min', 'salaire_max'),
515 ('indemn_min', 'indemn_max'),
516 ('autre_min', 'autre_max'))
517 }),
518 ('Comparatifs de rémunération', {
519 'fields': ('devise_comparaison',
520 ('comp_locale_min', 'comp_locale_max'),
521 ('comp_universite_min', 'comp_universite_max'),
522 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
523 ('comp_ong_min', 'comp_ong_max'),
524 ('comp_autre_min', 'comp_autre_max'))
525 }),
526 ('Justification', {
527 'fields': ('justification',)
528 }),
529 ('Autres Metadata', {
530 'fields': ('date_validation', ('date_debut', 'date_fin'))
531 }),
532 )
533
534 inlines = (PosteFinancementInline,
535 PostePieceInline,
536 DossierROInline,
537 PosteCommentaireInline, )
538
539 def _nom(self, poste):
540 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>""" % \
541 (reverse('poste_apercu', args=(poste.id,)),
542 poste.nom,
543 reverse('admin:rh_poste_change', args=(poste.id,)),
544 settings.MEDIA_URL,
545 )
546 return link
547 _nom.allow_tags = True
548 _nom.short_description = u'Nom'
549 _nom.admin_order_field = 'nom'
550
551 def _occupe_par(self, obj):
552 """Formatte la méthode Poste.occupe_par() pour l'admin"""
553 output = "VACANT"
554 employes = obj.occupe_par()
555 if employes:
556 l = []
557 for e in employes:
558 link = "<a href='%s'>%s</a>" % \
559 (reverse('admin:rh_employe_change', args=(e.id,)),
560 e)
561 l.append(link)
562 output = "\n<br />".join(l)
563 return output
564 _occupe_par.allow_tags = True
565 _occupe_par.short_description = "Occupé par"
566
567 def save_formset(self, request, form, formset, change):
568 instances = formset.save(commit=False)
569 for instance in instances:
570 if instance.__class__ == rh.PosteCommentaire:
571 instance.owner = request.user
572 instance.save()
573 formset.save_m2m()
574
575
576 class PosteCommentaireAdmin(admin.ModelAdmin):
577 pass
578
579
580 class PosteFinancementAdmin(admin.ModelAdmin):
581 pass
582
583
584 class PostePieceAdmin(admin.ModelAdmin):
585 pass
586
587
588 class RemunerationAdmin(admin.ModelAdmin):
589 pass
590
591
592 class ResponsableImplantationAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
593 fieldsets = AUFMetadataAdminMixin.fieldsets + (
594 (None, {
595 'fields': ('employe', 'implantation', ),
596 }),
597 )
598
599
600 class ServiceAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
601 list_display = ('nom', 'actif', )
602 fieldsets = AUFMetadataAdminMixin.fieldsets + (
603 (None, {
604 'fields': ('nom', ),
605 }),
606 )
607
608 class StatutAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
609 fieldsets = AUFMetadataAdminMixin.fieldsets + (
610 (None, {
611 'fields': ('code', 'nom', ),
612 }),
613 )
614
615 class TauxChangeAdmin(admin.ModelAdmin):
616 list_display = ('taux', 'devise', 'annee', )
617 list_filter = ('devise', )
618 fieldsets = AUFMetadataAdminMixin.fieldsets + (
619 (None, {
620 'fields': ('taux', 'devise', 'annee', ),
621 }),
622 )
623
624 class TypeContratAdmin(admin.ModelAdmin):
625 inlines = (ContratInline,)
626 fieldsets = AUFMetadataAdminMixin.fieldsets + (
627 (None, {
628 'fields': ('nom', 'nom_long', ),
629 }),
630 )
631
632
633 class TypePosteAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
634 search_fields = ('nom', 'nom_feminin', )
635 list_display = ('nom', 'famille_emploi', )
636 list_filter = ('famille_emploi', )
637 fieldsets = AUFMetadataAdminMixin.fieldsets + (
638 (None, {
639 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
640 }),
641 )
642
643
644 class TypeRemunerationAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
645 list_display = ('nom', 'type_paiement', 'nature_remuneration', )
646 #inlines = (RemunerationROInline,) utilité?
647 fieldsets = AUFMetadataAdminMixin.fieldsets + (
648 (None, {
649 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
650 }),
651 )
652
653
654 class TypeRevalorisationAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
655 #inlines = (RemunerationROInline,) utilité?
656 fieldsets = AUFMetadataAdminMixin.fieldsets + (
657 (None, {
658 'fields': ('nom', )
659 }),
660 )
661
662
663 class ValeurPointAdmin(AUFMetadataAdminMixin, admin.ModelAdmin):
664 list_display = ('_devise_code', '_devise_nom', 'annee', 'valeur', )
665 fieldsets = AUFMetadataAdminMixin.fieldsets + (
666 (None, {
667 'fields': ('valeur', 'devise', 'implantation', 'annee', )
668 }),
669 )
670
671 def _devise_code(self, obj):
672 return obj.devise.code
673 _devise_code.short_description = "Code de la devise"
674
675 def _devise_nom(self, obj):
676 return obj.devise.nom
677 _devise_nom.short_description = "Nom de la devise"
678
679
680 def calc_remun(dossier):
681 thisyear = datetime.date.today().year
682 thisyearfilter = Q(date_debut__year=thisyear) | Q(date_fin__year=thisyear)
683
684 remunnow = dossier.rh_remuneration_remunerations.filter(thisyearfilter)
685
686 remun_sum = 0
687 remun_sum_euro = 0
688 sums = defaultdict(int)
689 sums_euro = defaultdict(int)
690 for r in remunnow:
691 nature = r.type.nature_remuneration
692 sums[nature] += r.montant
693 sums_euro[nature] += r.montant_euro()
694 remun_sum += r.montant
695 remun_sum_euro += r.montant_euro()
696
697 remun = {}
698 sums = dict(sums)
699 for n, s in sums.iteritems():
700 remun[n] = [sums[n], sums_euro[n]]
701
702 return remun, remun_sum, remun_sum_euro