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
.contrib
.admin
.views
.main
import ChangeList
as DjangoChangeList
11 from django
.conf
import settings
12 from django
.db
.models
import Q
13 from django
.template
.defaultfilters
import date
14 from ajax_select
import make_ajax_form
15 from auf
.django
.metadata
.admin
import AUFMetadataAdminMixin
, AUFMetadataInlineAdminMixin
, AUF_METADATA_READONLY_FIELDS
16 from forms
import ContratForm
, AyantDroitForm
, EmployeAdminForm
, AjaxSelect
17 from dae
.utils
import get_employe_from_user
18 from groups
import grp_drh
24 class ChangeList(DjangoChangeList
):
26 def __init__(self
, *args
, **kwargs
):
27 super(ChangeList
, self
).__init__(*args
, **kwargs
)
29 def get_query_set(self
):
30 old
= self
.params
.copy()
33 for k
, v
in self
.params
.items():
35 prefix_debut
= "".join(k
.split('date_debut')[0:-1]) + 'date_debut'
39 prefix_fin
= "".join(k
.split('date_fin')[0:-1]) + 'date_fin'
42 qs
= super(ChangeList
, self
).get_query_set()
44 if date_fin
is None and date_debut
is not None:
46 prefix_fin
= prefix_debut
.replace('debut', 'fin')
47 if date_debut
is None and date_fin
is not None:
49 prefix_debut
= prefix_fin
.replace('fin', 'debut')
51 if date_debut
is not None and date_fin
is not None:
52 q_left
= (Q(**{'%s__isnull' % prefix_debut
: True}) |
Q(**{'%s__lte' % prefix_debut
: date_debut
})) & (Q(**{'%s__gte' % prefix_fin
: date_debut
}) & Q(**{'%s__lte' % prefix_fin
: date_fin
}))
53 q_right
= (Q(**{'%s__isnull' % prefix_fin
: True}) |
Q(**{'%s__gte' % prefix_fin
: date_fin
})) & (Q(**{'%s__gte' % prefix_debut
: date_debut
}) & Q(**{'%s__lte' % prefix_debut
: date_fin
}))
54 q_both
= Q(**{'%s__isnull' % prefix_fin
: True}) |
Q(**{'%s__lte' % prefix_fin
: date_fin
}) & (Q(**{'%s__isnull' % prefix_debut
: True}) |
Q(**{'%s__gte' % prefix_debut
: date_debut
}))
55 qs
= qs
.filter(q_left | q_right | q_both
)
60 # Override of the InlineModelAdmin to support the link in the tabular inline
61 class LinkedInline(admin
.options
.InlineModelAdmin
):
62 template
= "admin/linked.html"
63 admin_model_path
= None
65 def __init__(self
, *args
):
66 super(LinkedInline
, self
).__init__(*args
)
67 if self
.admin_model_path
is None:
68 self
.admin_model_path
= self
.model
.__name__
.lower()
71 class ProtectRegionMixin(object):
73 def queryset(self
, request
):
74 from dae
.workflow
import grp_drh
, grp_correspondants_rh
75 qs
= super(ProtectRegionMixin
, self
).queryset(request
)
77 if request
.user
.is_superuser
:
80 user_groups
= request
.user
.groups
.all()
82 if grp_drh
in user_groups
:
85 if grp_correspondants_rh
in user_groups
:
86 employe
= get_employe_from_user(request
.user
)
87 q
= Q(**{self
.model
.prefix_implantation
: employe
.implantation
.region
})
88 qs
= qs
.filter(q
).distinct()
92 def has_change_permission(self
, request
, obj
=None):
93 user_groups
= request
.user
.groups
.all()
95 # Lock pour autoriser uniquement les DRH à utiliser RH
96 if not request
.user
.is_superuser
and not grp_drh
in user_groups
:
99 if len(user_groups
) == 0 and not request
.user
.is_superuser
:
104 ids
= [o
.id for o
in self
.queryset(request
)]
110 class ReadOnlyInlineMixin(object):
111 def get_readonly_fields(self
, request
, obj
=None):
112 return [f
.name
for f
in self
.model
._meta
.fields
if f
.name
not in AUF_METADATA_READONLY_FIELDS
]
115 class AyantDroitInline(AUFMetadataInlineAdminMixin
, admin
.StackedInline
):
116 model
= rh
.AyantDroit
117 form
= AyantDroitForm
122 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', 'lien_parente', )
127 class AyantDroitCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
128 readonly_fields
= ('owner', )
129 model
= rh
.AyantDroitCommentaire
133 class ContratInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
139 class DossierROInline(ReadOnlyInlineMixin
, LinkedInline
):
140 template
= "admin/rh/dossier/linked.html"
141 exclude
= AUF_METADATA_READONLY_FIELDS
146 def has_add_permission(self
, request
=None):
149 def has_change_permission(self
, request
, obj
=None):
152 def has_delete_permission(self
, request
, obj
=None):
156 class DossierCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
157 readonly_fields
= ('owner', )
158 model
= rh
.DossierCommentaire
162 class DossierPieceInline(admin
.TabularInline
):
163 model
= rh
.DossierPiece
167 class EmployeInline(admin
.TabularInline
):
170 class EmployeCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
171 readonly_fields
= ('owner', )
172 model
= rh
.EmployeCommentaire
176 class EmployePieceInline(admin
.TabularInline
):
177 model
= rh
.EmployePiece
181 class PosteCommentaireInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
182 readonly_fields
= ('owner', )
183 model
= rh
.PosteCommentaire
187 class PosteFinancementInline(admin
.TabularInline
):
188 model
= rh
.PosteFinancement
191 class PostePieceInline(admin
.TabularInline
):
192 model
= rh
.PostePiece
195 class RemunerationInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
196 model
= rh
.Remuneration
200 class RemunerationROInline(ReadOnlyInlineMixin
, RemunerationInline
):
204 class TypePosteInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
208 class PosteComparaisonInline(AUFMetadataInlineAdminMixin
, admin
.TabularInline
):
209 model
= rh
.PosteComparaison
212 class ClassementAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
213 list_display
= ('_classement', 'date_modification', 'user_modification', )
214 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
216 'fields': ('type', 'echelon', 'degre', 'coefficient', )
220 def _classement(self
, obj
):
222 _classement
.short_description
= u
"Classement"
224 class CommentaireAdmin(admin
.ModelAdmin
):
228 class DeviseAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
229 list_display
= ('code', 'nom', 'date_modification', 'user_modification', )
230 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
232 'fields': ('code', 'nom', ),
237 class DossierAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
238 alphabet_filter
= 'employe__nom'
239 search_fields
= ('employe__nom', 'employe__prenom', 'poste__nom', 'poste__nom_feminin')
250 'poste__implantation__region',
251 'poste__implantation',
253 'poste__type_poste__famille_emploi',
254 'rh_contrats__type_contrat',
258 inlines
= (DossierPieceInline
, ContratInline
,
260 DossierCommentaireInline
,
262 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
264 'fields': ('employe', 'poste', 'statut', 'organisme_bstg',)
267 'fields': ('statut_residence', 'remplacement', 'remplacement_de', )
270 'fields': ('classement', ('regime_travail', 'regime_travail_nb_heure_semaine'),)
272 ('Occupation du Poste par cet Employe', {
273 'fields': (('date_debut', 'date_fin'), )
276 form
= make_ajax_form(rh
.Dossier
, {
277 'employe' : 'employes',
279 'remplacement_de' : 'dossiers',
282 def lookup_allowed(self
, key
, value
):
284 'employe__nom__istartswith',
285 'poste__implantation__region__id__exact',
286 'poste__implantation__id__exact',
287 'poste__type_poste__id__exact',
288 'poste__type_poste__famille_emploi__id__exact',
289 'rh_contrats__type_contrat__id__exact',
291 'date_debut__isnull',
297 def get_changelist(self
, request
, **kwargs
):
301 apercu_link
= u
"""<a title="Aperçu du dossier" onclick="return showAddAnotherPopup(this);" href='%s'><img src="%simg/loupe.png" /></a>""" % \
302 (reverse('dossier_apercu', args
=(d
.id,)),
305 link
= u
"""%s <a href="%s" title="Modifier le dossier"><strong>%s</strong></a>""" % \
307 reverse('admin:rh_dossier_change', args
=(d
.id,)),
311 _id
.allow_tags
= True
312 _id
.short_description
= u
"Dossier__#"
313 _id
.admin_order_field
= 'id'
316 def _date_debut(self
, obj
):
317 return date(obj
.date_debut
)
319 _date_debut
.short_description
= u
'Occupation début'
320 _date_debut
.admin_order_field
= 'date_debut'
322 def _date_fin(self
, obj
):
323 return date(obj
.date_fin
)
324 _date_fin
.short_description
= u
'Occupation fin'
325 _date_fin
.admin_order_field
= 'date_fin'
327 def _poste(self
, dossier
):
328 link
= u
"""<a title="Aperçu du poste" onclick="return showAddAnotherPopup(this);" href='%s'><img src="%simg/loupe.png" /></a> <a href="%s" title="Modifier le poste">%s</a>""" % \
329 (reverse('poste_apercu', args
=(dossier
.poste
.id,)),
331 reverse('admin:rh_poste_change', args
=(dossier
.poste
.id,)),
335 _poste
.allow_tags
= True
336 _poste
.short_description
= u
'Poste'
337 _poste
.admin_order_field
= 'poste__nom'
339 def _employe(self
, obj
):
340 employe
= obj
.employe
341 view_link
= reverse('employe_apercu', args
=(employe
.id,))
342 edit_link
= reverse('admin:rh_employe_change', args
=(employe
.id,))
345 view
= u
"""<a href="%s" title="Aperçu l'employé" onclick="return showAddAnotherPopup(this);"><img src="%simg/loupe.png" /></a>""" % (view_link
, settings
.STATIC_URL
,)
346 return u
"""%s<a href='%s' style="%s;">[%s] %s %s</a>""" % \
347 (view
, edit_link
, style
, employe
.id, employe
.nom
.upper(), employe
.prenom
.title())
348 _employe
.allow_tags
= True
349 _employe
.short_description
= u
"Employé ([code] NOM Prénom)"
350 _employe
.admin_order_field
= "employe__nom"
352 def save_formset(self
, request
, form
, formset
, change
):
353 instances
= formset
.save(commit
=False)
354 for instance
in instances
:
355 if instance
.__class__
== rh
.DossierCommentaire
:
356 instance
.owner
= request
.user
357 instance
.date_creation
= datetime
.datetime
.now()
361 class DossierPieceAdmin(admin
.ModelAdmin
):
365 class DossierCommentaireAdmin(admin
.ModelAdmin
):
369 class EmployeAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
):
370 alphabet_filter
= 'nom'
371 DEFAULT_ALPHABET
= u
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
372 search_fields
= ('id', 'nom', 'prenom', 'nom_affichage', )
374 form
= EmployeAdminForm
375 list_display
= ('_apercu', '_nom', '_dossiers', 'date_modification', 'user_modification', )
376 list_filter
= ('rh_dossiers__poste__implantation__region', 'rh_dossiers__poste__implantation', 'nb_postes', 'rh_dossiers__date_debut', 'rh_dossiers__date_fin')
377 inlines
= (AyantDroitInline
,
380 EmployeCommentaireInline
)
381 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
383 'fields': (('nom', 'prenom'), ('nom_affichage', 'genre'), 'nationalite', 'date_naissance', )
385 ('Informations personnelles', {
386 'fields': ('situation_famille', 'date_entree', )
389 'fields': (('tel_domicile', 'tel_cellulaire'), ('adresse', 'ville'), ('code_postal', 'province'), 'pays', )
393 def get_changelist(self
, request
, **kwargs
):
397 def _apercu(self
, obj
):
398 return u
"""<a title="Aperçu de l'employé" onclick="return showAddAnotherPopup(this);" href='%s'><img src="%simg/loupe.png" /></a>""" % \
399 (reverse('employe_apercu', args
=(obj
.id,)), settings
.STATIC_URL
)
400 _apercu
.allow_tags
= True
401 _apercu
.short_description
= u
""
402 _apercu
.admin_order_field
= ""
405 edit_link
= reverse('admin:rh_employe_change', args
=(obj
.id,))
406 return u
"""<a href='%s'><strong>[%s] %s %s</strong></a>""" % \
407 (edit_link
, obj
.id, obj
.nom
.upper(), obj
.prenom
.title(),)
408 _nom
.allow_tags
= True
409 _nom
.short_description
= u
"Employé ([code] NOM Prénom)"
410 _nom
.admin_order_field
= "nom"
412 def _dossiers(self
, obj
):
414 for d
in obj
.rh_dossiers
.all().order_by('-date_debut'):
415 apercu
= u
"""<a title="Aperçu du dossier" href="%s" onclick="return showAddAnotherPopup(this);" title="Aperçu du dossier"><img src="%simg/loupe.png" /></a>""" % \
416 (reverse('dossier_apercu', args
=(d
.id,)), settings
.STATIC_URL
,)
417 link
= u
"""<li>%s<a href='%s'>%s : %s</a></li>""" % \
419 reverse('admin:rh_dossier_change', args
=(d
.id,)),
424 # Dossier terminé en gris non cliquable
425 if d
.date_fin
is not None:
426 link
= u
"""<li style="color: grey">%s : %s</li>""" % \
432 return "<ul>%s</ul>" % "\n".join(l
)
433 _dossiers
.allow_tags
= True
434 _dossiers
.short_description
= u
"Dossiers"
436 def queryset(self
, request
):
437 qs
= super(EmployeAdmin
, self
).queryset(request
)
438 return qs
.select_related(depth
=1).order_by('nom')
440 def save_formset(self
, request
, form
, formset
, change
):
441 instances
= formset
.save(commit
=False)
442 for instance
in instances
:
443 if instance
.__class__
== rh
.EmployeCommentaire
:
444 instance
.owner
= request
.user
445 instance
.date_creation
= datetime
.datetime
.now()
450 class EmployeCommentaireAdmin(admin
.ModelAdmin
):
454 class EmployePieceAdmin(admin
.ModelAdmin
):
458 class FamilleEmploiAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
459 list_display
= ('nom', 'date_modification', 'user_modification', )
460 inlines
= (TypePosteInline
,)
461 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
468 class OrganismeBstgAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
469 search_fields
= ('nom',)
470 list_display
= ('nom', 'type', 'pays', 'date_modification', 'user_modification', )
471 list_filter
= ('type', )
472 inlines
= (DossierROInline
,)
473 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
475 'fields': ('nom', 'type', 'pays', )
480 class PosteAdmin(AUFMetadataAdminMixin
, ProtectRegionMixin
, admin
.ModelAdmin
, AjaxSelect
):
481 form
= make_ajax_form(rh
.Poste
, {
482 'implantation' : 'implantations',
483 'type_poste' : 'typepostes',
484 'responsable' : 'postes',
485 'valeur_point_min' : 'valeurpoints',
486 'valeur_point_max' : 'valeurpoints',
488 alphabet_filter
= 'nom'
489 search_fields
= ('nom',
490 'implantation__code',
492 'implantation__region__code',
493 'implantation__region__nom',
508 'implantation__region',
511 'type_poste__famille_emploi',
516 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
518 'fields': (('nom', 'nom_feminin'), 'implantation', 'type_poste',
519 'service', 'responsable')
522 'fields': (('regime_travail', 'regime_travail_nb_heure_semaine'), )
525 'fields': (('local', 'expatrie', 'mise_a_disposition', 'appel'),)
528 'fields': (('classement_min', 'valeur_point_min', 'devise_min', 'salaire_min', 'indemn_min', 'autre_min', ),
529 ('classement_max', 'valeur_point_max' ,'devise_max', 'salaire_max', 'indemn_max', 'autre_max', ),
532 ('Comparatifs de rémunération', {
533 'fields': ('devise_comparaison',
534 ('comp_locale_min', 'comp_locale_max'),
535 ('comp_universite_min', 'comp_universite_max'),
536 ('comp_fonctionpub_min', 'comp_fonctionpub_max'),
537 ('comp_ong_min', 'comp_ong_max'),
538 ('comp_autre_min', 'comp_autre_max'))
541 'fields': ('justification',)
543 ('Autres Méta-données', {
544 'fields': ('date_debut', 'date_fin')
548 inlines
= (PosteFinancementInline
,
551 PosteComparaisonInline
,
552 PosteCommentaireInline
, )
555 def get_changelist(self
, request
, **kwargs
):
559 def lookup_allowed(self
, key
, value
):
562 'date_debut__isnull',
569 def _apercu(self
, poste
):
570 link
= u
"""<a onclick="return showAddAnotherPopup(this);" title="Aperçu du poste" href='%s'><img src="%simg/loupe.png" /> %s</a>""" % \
571 (reverse('poste_apercu', args
=(poste
.id,)),
576 _apercu
.allow_tags
= True
577 _apercu
.short_description
= 'Poste __#'
578 _apercu
.admin_order_field
= 'id'
580 def _service(self
, obj
):
583 def _nom(self
, poste
):
584 link
= u
"""<a href="%s" title="Modifier le poste"><strong>%s</strong></a>""" % \
585 (reverse('admin:rh_poste_change', args
=(poste
.id,)),
589 _nom
.allow_tags
= True
590 _nom
.short_description
= u
'Nom'
591 _nom
.admin_order_field
= 'nom'
593 def _occupe_par(self
, obj
):
594 """Formatte la méthode Poste.occupe_par() pour l'admin"""
596 if obj
.date_fin
is not None and obj
.date_fin
< datetime
.date
.now():
598 employes
= obj
.occupe_par()
602 link
= "<a href='%s' title='Aperçu de l\'employer' onclick='return showAddAnotherPopup(this)'><img src='%simg/loupe.png' /></a> <a href='%s'>%s</a>" % \
603 (reverse('employe_apercu', args
=(e
.id,)),
605 reverse('admin:rh_employe_change', args
=(e
.id,)),
609 output
= "\n<br />".join(l
)
611 _occupe_par
.allow_tags
= True
612 _occupe_par
.short_description
= "Occupé par"
614 def save_formset(self
, request
, form
, formset
, change
):
615 instances
= formset
.save(commit
=False)
616 for instance
in instances
:
617 if instance
.__class__
== rh
.PosteCommentaire
:
618 instance
.owner
= request
.user
619 instance
.date_creation
= datetime
.datetime
.now()
624 class PosteCommentaireAdmin(admin
.ModelAdmin
):
628 class PosteFinancementAdmin(admin
.ModelAdmin
):
632 class PostePieceAdmin(admin
.ModelAdmin
):
636 class RemunerationAdmin(admin
.ModelAdmin
):
640 class ResponsableImplantationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
641 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
643 'fields': ('employe', 'implantation', ),
648 class ServiceAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
649 list_display
= ('nom', 'date_modification', 'user_modification', )
650 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
656 class StatutAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
657 list_display
= ('code', 'nom', 'date_modification', 'user_modification', )
658 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
660 'fields': ('code', 'nom', ),
664 class TauxChangeAdmin(admin
.ModelAdmin
):
665 list_display
= ('taux', 'devise', 'annee', 'date_modification', 'user_modification', )
666 list_filter
= ('devise', )
667 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
669 'fields': ('taux', 'devise', 'annee', ),
673 class TypeContratAdmin(admin
.ModelAdmin
):
674 list_display
= ('nom', 'nom_long', 'date_modification', 'user_modification', )
675 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
677 'fields': ('nom', 'nom_long', ),
682 class TypePosteAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
683 search_fields
= ('nom', 'nom_feminin', )
684 list_display
= ('nom', 'famille_emploi', 'date_modification', 'user_modification', )
685 list_filter
= ('famille_emploi', )
686 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
688 'fields': ('nom', 'nom_feminin', 'is_responsable', 'famille_emploi', )
693 class TypeRemunerationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
694 list_display
= ('nom', 'type_paiement', 'nature_remuneration', 'date_modification', 'user_modification', )
695 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
697 'fields': ('nom', 'type_paiement', 'nature_remuneration', )
702 class TypeRevalorisationAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
703 list_display
= ('nom', 'date_modification', 'user_modification', )
704 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
711 class ValeurPointAdmin(AUFMetadataAdminMixin
, admin
.ModelAdmin
):
712 list_display
= ('_devise_code', '_devise_nom', 'annee', 'valeur', 'date_modification', 'user_modification', )
713 list_filter
= ('annee', 'devise', )
714 fieldsets
= AUFMetadataAdminMixin
.fieldsets
+ (
716 'fields': ('valeur', 'devise', 'implantation', 'annee', )
720 def _devise_code(self
, obj
):
721 return obj
.devise
.code
722 _devise_code
.short_description
= "Code de la devise"
724 def _devise_nom(self
, obj
):
725 return obj
.devise
.nom
726 _devise_nom
.short_description
= "Nom de la devise"
729 admin
.site
.register(rh
.Classement
, ClassementAdmin
)
730 admin
.site
.register(rh
.Devise
, DeviseAdmin
)
731 admin
.site
.register(rh
.Dossier
, DossierAdmin
)
732 admin
.site
.register(rh
.Employe
, EmployeAdmin
)
733 admin
.site
.register(rh
.FamilleEmploi
, FamilleEmploiAdmin
)
734 admin
.site
.register(rh
.OrganismeBstg
, OrganismeBstgAdmin
)
735 admin
.site
.register(rh
.Poste
, PosteAdmin
)
736 admin
.site
.register(rh
.ResponsableImplantation
, ResponsableImplantationAdmin
)
737 admin
.site
.register(rh
.Service
, ServiceAdmin
)
738 admin
.site
.register(rh
.Statut
, StatutAdmin
)
739 admin
.site
.register(rh
.TauxChange
, TauxChangeAdmin
)
740 admin
.site
.register(rh
.TypeContrat
, TypeContratAdmin
)
741 admin
.site
.register(rh
.TypePoste
, TypePosteAdmin
)
742 admin
.site
.register(rh
.TypeRemuneration
, TypeRemunerationAdmin
)
743 admin
.site
.register(rh
.TypeRevalorisation
, TypeRevalorisationAdmin
)
744 admin
.site
.register(rh
.ValeurPoint
, ValeurPointAdmin
)