1 # -*- coding: utf-8 -*-
2 from django
.db
import models
3 from django
.db
.models
import Q
4 from django
.contrib
import admin
5 from django
.core
.urlresolvers
import reverse
as url
6 from django
.forms
.models
import BaseInlineFormSet
7 from django
.http
import HttpResponseRedirect
8 from django
.utils
.encoding
import smart_str
9 from django_exportateur
.exportateur
import exportateur
11 from chercheurs
.models
import Chercheur
, Publication
, GroupeChercheur
, DomaineRecherche
, ChercheurGroupe
, ChercheurQuerySet
, These
12 from savoirs
.models
import Search
14 class ChercheurAdmin(admin
.ModelAdmin
):
15 list_filter
= ['genre']
16 alphabet_filter
= 'nom'
17 alphabet_filter_table
= 'chercheurs_personne'
20 actions
= ('remove_from_group', 'export_as_ods', 'export_as_csv')
21 search_fields
= ('nom', 'prenom')
23 def has_change_permission(self
, request
, obj
=None):
24 if not obj
and request
.user
.has_perm('chercheurs.view_chercheur'):
27 return super(ChercheurAdmin
, self
).has_change_permission(request
, obj
)
29 def change_view(self
, request
, obj
=None):
30 if request
.user
.has_perm('chercheurs.view_chercheur') and \
31 not super(ChercheurAdmin
, self
).has_change_permission(request
, obj
):
32 return HttpResponseRedirect(url('admin:chercheurs_chercheur_changelist'))
34 return super(ChercheurAdmin
, self
).change_view(request
, obj
)
36 def lookup_allowed(self
, lookup
, value
):
37 return lookup
in ['genre', 'statut', 'membre_reseau_institutionnel',
38 'membre_instance_auf', 'discipline', 'region', 'pays',
39 'groupes', 'nord_sud'] or \
40 admin
.ModelAdmin
.lookup_allowed(self
, lookup
, value
)
42 def remove_from_group(self
, request
, queryset
):
43 groupe_id
= request
.GET
.get('groupes__id__exact')
44 chercheur_ids
= queryset
.values_list('id', flat
=True)
45 matches
= ChercheurGroupe
.objects
.filter(groupe
=groupe_id
, chercheur__in
=chercheur_ids
)
47 return HttpResponseRedirect(url('admin:chercheurs_chercheur_changelist') + '?groupes__id__exact=' + groupe_id
)
49 def get_actions(self
, request
):
50 actions
= super(ChercheurAdmin
, self
).get_actions(request
)
52 # Si on filtre par groupe de recherche, offrir d'en retirer les
53 # chercheurs sélectionnés.
54 groupe_id
= request
.GET
.get('groupes__id__exact')
56 groupe
= Groupe
.objects
.get(id=groupe_id
)
57 action_desc
= actions
['remove_from_group']
58 actions
['remove_from_group'] = (action_desc
[0], action_desc
[1], u
'Retirer du domaine de recherche « %s »' % groupe
.nom
)
60 del actions
['remove_from_group']
63 def queryset(self
, request
):
64 return ChercheurAdminQuerySet(Chercheur
)
66 def get_object(self
, request
, object_id
):
67 """On doit réimplémenter cette méthode à cause de ce qu'on fait avec "initial" dans la méthode queryset()."""
69 return Chercheur
.objects
.get(id=object_id
)
70 except Chercheur
.DoesNotExist
:
73 def export(self
, queryset
, type):
74 if queryset
.count() == 0:
77 headers
= ['Nom', 'Prénom', 'Genre', 'Courriel', 'Téléphone', 'Adresse postale',
78 'Statut', 'Diplôme', 'Établissement', 'Pays', 'Domaines de recherche',
79 'Thèse', 'Directeur', 'Discipline', 'Thèmes de recherche', 'Groupe de recherche', 'Mots-clés',
80 'Site web', 'Blog', 'Réseau social',
81 'Membre instance AUF', "Sollicité par l'OIF", 'Membre société francophone',
82 'Membre instance réseau institutionnel AUF', 'Expertises', 'Solliciter pour expertises',
89 row
.append(c
.get_genre_display())
90 row
.append(c
.courriel
)
91 row
.append(c
.telephone
)
92 row
.append(c
.adresse_postale
)
93 row
.append(c
.get_statut_display())
95 row
.append(c
.nom_etablissement
)
97 row
.append(', '.join(g
.nom
for g
in c
.groupes
.all()))
100 row
.append('%s, %s, %s, %s pages.' % (t
.titre
, t
.etablissement
, t
.annee
, t
.nb_pages
))
101 row
.append(t
.directeur
)
102 except These
.DoesNotExist
:
105 row
.append(c
.discipline
.nom
if c
.discipline
else '')
106 row
.append(c
.theme_recherche
)
107 row
.append(c
.groupe_recherche
)
108 row
.append(c
.mots_cles
)
109 row
.append(c
.url_site_web
)
110 row
.append(c
.url_blog
)
111 row
.append(c
.url_reseau_social
)
112 if c
.membre_instance_auf
:
113 row
.append(', '.join([c
.membre_instance_auf_nom
, c
.membre_instance_auf_fonction
, c
.membre_instance_auf_dates
]))
117 row
.append(', '.join([c
.expert_oif_details
, c
.expert_oif_dates
]))
120 if c
.membre_association_francophone
:
121 row
.append(c
.membre_association_francophone_details
)
124 if c
.membre_reseau_institutionnel
:
125 row
.append(', '.join([c
.membre_reseau_institutionnel_nom
, c
.membre_reseau_institutionnel_fonction
, c
.membre_reseau_institutionnel_dates
]))
128 row
.append('; '.join(', '.join([e
.nom
, e
.date
, e
.organisme_demandeur
]) for e
in c
.expertises
.all()))
129 if c
.expertises_auf
is None:
132 row
.append('Oui' if c
.expertises_auf
else 'Non')
133 row
.append('; '.join(p
.publication_affichage
if p
.publication_affichage
else
134 '%s, %s, %s, %s, %s, %s, %s pages.' %
135 (p
.auteurs
, p
.titre
, p
.revue
, p
.annee
, p
.editeur
, p
.lieu_edition
, p
.nb_pages
)
136 for p
in c
.publications
.all()))
137 data
.append([smart_str(x
) if x
else '' for x
in row
])
138 return exportateur(headers
, data
, type, filename
='chercheurs.%s' % type)
140 def export_as_csv(self
, request
, queryset
):
141 return self
.export(queryset
, 'csv')
142 export_as_csv
.short_description
= 'Export CSV'
144 def export_as_ods(self
, request
, queryset
):
145 return self
.export(queryset
, 'ods')
146 export_as_ods
.short_description
= 'Export ODS'
148 class ChercheurAdminQuerySet(ChercheurQuerySet
):
150 def filter(self
, *args
, **kwargs
):
151 """Gère des filtres supplémentaires pour l'admin.
153 C'est la seule façon que j'ai trouvée de contourner les mécanismes
154 de recherche de l'admin."""
155 pays
= kwargs
.pop('pays', None)
156 region
= kwargs
.pop('region', None)
157 nord_sud
= kwargs
.pop('nord_sud', None)
158 expert
= kwargs
.pop('expert', None)
161 qs
= qs
.filter(Q(etablissement__pays
=pays
) |
162 (Q(etablissement
=None) & Q(etablissement_autre_pays
=pays
)))
163 elif region
is not None:
164 qs
= qs
.filter(Q(etablissement__pays__region
=region
) |
165 (Q(etablissement
=None) & Q(etablissement_autre_pays__region
=region
)))
166 elif nord_sud
is not None:
167 qs
= qs
.filter(Q(etablissement__pays__nord_sud
=nord_sud
) |
168 (Q(etablissement
=None) & Q(etablissement_autre_pays__nord_sud
=nord_sud
)))
169 if expert
is not None:
170 if expert
in ['1', 1, True]:
171 qs
= qs
.exclude(expertises
=None)
173 qs
= qs
.filter(expertises
=None)
175 return super(ChercheurAdminQuerySet
, qs
).filter(*args
, **kwargs
)
178 class ChercheurGroupeAdmin(admin
.ModelAdmin
):
179 list_filter
= ('groupe',)
180 list_display
= ('groupe', 'chercheur', 'actif')
181 list_editable
= ('actif',)
183 def queryset(self
, request
):
184 qs
= super(ChercheurGroupeAdmin
, self
).queryset(request
)
186 if not request
.user
.is_superuser
and not request
.user
.has_perm('chercheurs.change_chercheurgroupe'):
187 qs
= qs
.filter(groupe__responsables
=request
.user
)
191 def has_change_permission(self
, request
, obj
=None):
194 if request
.user
.responsable_groupe
.count():
197 if request
.user
in obj
.groupe
.responsables
.all():
200 return super(ChercheurGroupeAdmin
, self
).has_change_permission(request
, obj
)
202 class MemberInline(admin
.TabularInline
):
203 model
= ChercheurGroupe
206 class BaseGroupeAdmin(admin
.ModelAdmin
):
207 filter_horizontal
= ('responsables',)
209 (('Options générales'), {'fields': ('nom', 'url', 'liste_diffusion',
210 'bulletin', 'page_accueil')}),
211 (('Responsables'), {'fields': ('responsables',)}),
212 (('Recherches prédéfinies'), {'fields': ('recherches',)}),
219 js
= ['js/tiny_mce/tiny_mce.js', 'js/tiny_mce_textareas.js']
221 def save_model(self
, request
, obj
, form
, change
):
222 responsables
= form
.cleaned_data
['responsables']
223 for user
in responsables
:
227 if not request
.user
.is_superuser
:
228 recherches
= Search
.objects
.exclude(user
=request
.user
)
229 form
.cleaned_data
['recherches'] = form
.cleaned_data
['recherches'] | recherches
231 super(BaseGroupeAdmin
, self
).save_model(request
, obj
, form
, change
)
233 def queryset(self
, request
):
234 qs
= super(BaseGroupeAdmin
, self
).queryset(request
)
236 if not request
.user
.is_superuser
and not request
.user
.has_perm('chercheurs.change_groupechercheur'):
237 qs
= qs
.filter(responsables
=request
.user
)
241 def has_change_permission(self
, request
, obj
=None, groupe_chercheur
=False):
244 if request
.user
.responsable_groupe
.filter(groupe_chercheur
=groupe_chercheur
).count():
247 if request
.user
in obj
.responsables
.all():
250 return super(BaseGroupeAdmin
, self
).has_change_permission(request
, obj
)
252 def formfield_for_manytomany(self
, db_field
, request
, **kwargs
):
253 if db_field
.name
== "recherches" and not request
.user
.is_superuser
:
254 kwargs
["queryset"] = Search
.objects
.filter(user
=request
.user
)
255 return db_field
.formfield(**kwargs
)
256 return super(BaseGroupeAdmin
, self
).formfield_for_manytomany(db_field
, request
, **kwargs
)
259 class GroupeChercheurAdmin(BaseGroupeAdmin
):
261 def has_change_permission(self
, request
, obj
=None):
262 return super(GroupeChercheurAdmin
, self
).has_change_permission(request
, obj
, groupe_chercheur
=True)
265 class DomaineRechercheAdmin(BaseGroupeAdmin
):
267 def has_change_permission(self
, request
, obj
=None):
268 return super(DomaineRechercheAdmin
, self
).has_change_permission(request
, obj
, groupe_chercheur
=False)
270 admin
.site
.register(Chercheur
, ChercheurAdmin
)
271 admin
.site
.register(Publication
)
272 admin
.site
.register(GroupeChercheur
, GroupeChercheurAdmin
)
273 admin
.site
.register(DomaineRecherche
, DomaineRechercheAdmin
)
274 admin
.site
.register(ChercheurGroupe
, ChercheurGroupeAdmin
)