2017-04-11 00:20:14 +02:00
|
|
|
# coding: utf-8
|
|
|
|
|
2017-04-20 01:53:29 +02:00
|
|
|
from django.shortcuts import render, redirect, get_object_or_404
|
2017-04-04 00:31:50 +02:00
|
|
|
|
2017-04-27 21:42:13 +02:00
|
|
|
from django.views.generic import DetailView, ListView
|
2017-04-07 03:01:27 +02:00
|
|
|
from django.views.generic.edit import UpdateView, CreateView
|
|
|
|
from django import forms
|
|
|
|
from django.urls import reverse
|
2017-06-27 23:14:24 +02:00
|
|
|
from django.contrib.admin.views.decorators import staff_member_required
|
2017-04-07 03:01:27 +02:00
|
|
|
from django.contrib.auth.decorators import login_required
|
|
|
|
from braces.views import LoginRequiredMixin
|
2018-12-28 23:46:24 +01:00
|
|
|
from django.http import JsonResponse, HttpResponseForbidden, Http404
|
2017-04-20 23:04:07 +02:00
|
|
|
from django.core.mail import send_mail
|
2017-06-27 23:14:24 +02:00
|
|
|
from django.db.models import Q, Count
|
2018-09-09 00:17:11 +02:00
|
|
|
from collections import Counter, defaultdict
|
2017-04-07 03:01:27 +02:00
|
|
|
|
2018-12-28 23:46:24 +01:00
|
|
|
from .models import Normalien, Stage, Lieu, AvisLieu, AvisStage
|
|
|
|
from .forms import StageForm, LieuForm, AvisStageForm, AvisLieuForm, FeedbackForm
|
|
|
|
from .utils import en_scolarite
|
|
|
|
|
|
|
|
from .views_search import *
|
2017-04-07 03:01:27 +02:00
|
|
|
|
2017-05-16 00:22:59 +02:00
|
|
|
import random, math
|
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
#
|
|
|
|
# LECTURE
|
|
|
|
#
|
|
|
|
|
2017-04-07 03:01:27 +02:00
|
|
|
# Page d'accueil
|
2017-04-05 00:23:35 +02:00
|
|
|
def index(request):
|
2017-06-19 00:04:13 +02:00
|
|
|
num_stages = Stage.objects.filter(public=True).count()
|
|
|
|
return render(request, 'avisstage/index.html',
|
|
|
|
{"num_stages": num_stages})
|
2017-04-05 00:23:35 +02:00
|
|
|
|
2017-04-07 03:01:27 +02:00
|
|
|
# Espace personnel
|
|
|
|
@login_required
|
2017-04-05 00:23:35 +02:00
|
|
|
def perso(request):
|
|
|
|
return render(request, 'avisstage/perso.html')
|
|
|
|
|
2018-12-28 23:46:24 +01:00
|
|
|
# 403 Archicubes
|
|
|
|
@login_required
|
|
|
|
def archicubes_interdits(request):
|
|
|
|
return render(request, 'avisstage/403-archicubes.html')
|
|
|
|
|
2017-04-07 03:01:27 +02:00
|
|
|
# Profil
|
2017-05-02 23:23:26 +02:00
|
|
|
#login_required
|
2017-05-16 22:56:19 +02:00
|
|
|
class ProfilView(LoginRequiredMixin, DetailView):
|
2017-05-02 23:23:26 +02:00
|
|
|
model = Normalien
|
|
|
|
template_name = 'avisstage/detail/profil.html'
|
|
|
|
|
2017-05-16 22:56:19 +02:00
|
|
|
# Récupération du profil
|
2017-05-02 23:23:26 +02:00
|
|
|
def get_object(self):
|
2018-12-28 23:46:24 +01:00
|
|
|
|
|
|
|
# Restriction d'accès pour les archicubes
|
|
|
|
if (en_scolarite(self.request.user) or
|
|
|
|
self.kwargs.get('username') == self.request.user.username):
|
2019-01-07 22:11:51 +01:00
|
|
|
return get_object_or_404(
|
|
|
|
Normalien, user__username=self.kwargs.get('username'))
|
2018-12-28 23:46:24 +01:00
|
|
|
else:
|
|
|
|
raise Http404
|
2017-05-02 23:23:26 +02:00
|
|
|
|
|
|
|
# Stage
|
|
|
|
#login_required
|
2017-05-16 22:56:19 +02:00
|
|
|
class StageView(LoginRequiredMixin, DetailView):
|
2017-05-02 23:23:26 +02:00
|
|
|
model = Stage
|
|
|
|
template_name = 'avisstage/detail/stage.html'
|
|
|
|
|
|
|
|
# Restriction aux stages publics ou personnels
|
|
|
|
def get_queryset(self):
|
2018-12-28 23:46:24 +01:00
|
|
|
filtre = Q(auteur__user_id=self.request.user.id)
|
|
|
|
|
|
|
|
# Restriction d'accès pour les archicubes
|
|
|
|
if en_scolarite(self.request.user):
|
|
|
|
filtre |= Q(public=True)
|
2017-05-02 23:23:26 +02:00
|
|
|
|
2018-12-28 23:46:24 +01:00
|
|
|
return Stage.objects.filter(filtre)
|
2017-06-26 22:40:35 +02:00
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
|
2017-05-12 23:29:29 +02:00
|
|
|
# FAQ
|
|
|
|
def faq(request):
|
|
|
|
return render(request, 'avisstage/faq.html')
|
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
#
|
|
|
|
# EDITION
|
|
|
|
#
|
|
|
|
|
|
|
|
# Profil
|
|
|
|
#login_required
|
2017-05-16 22:56:19 +02:00
|
|
|
class ProfilEdit(LoginRequiredMixin, UpdateView):
|
2017-04-07 03:01:27 +02:00
|
|
|
model = Normalien
|
2018-12-28 23:46:24 +01:00
|
|
|
fields = ['nom', 'promotion', 'contactez_moi', 'bio']
|
2017-04-07 03:01:27 +02:00
|
|
|
template_name = 'avisstage/formulaires/profil.html'
|
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Limitation à son propre profil
|
2017-04-07 03:01:27 +02:00
|
|
|
def get_object(self):
|
|
|
|
return self.request.user.profil
|
|
|
|
|
|
|
|
def get_success_url(self):
|
|
|
|
return reverse('avisstage:perso')
|
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Stage
|
2017-04-13 22:50:00 +02:00
|
|
|
@login_required
|
2017-04-20 01:53:29 +02:00
|
|
|
def manage_stage(request, pk=None):
|
2017-05-02 23:23:26 +02:00
|
|
|
# Objet de base
|
2018-03-21 13:14:14 +01:00
|
|
|
last_maj = None
|
2017-04-20 01:53:29 +02:00
|
|
|
if pk is None:
|
2017-04-13 22:50:00 +02:00
|
|
|
stage = Stage(auteur=request.user.profil)
|
|
|
|
avis_stage = AvisStage(stage=stage)
|
|
|
|
c_del = False
|
2018-03-21 13:14:14 +01:00
|
|
|
last_creation = Stage.objects.filter(auteur=request.user.profil)\
|
|
|
|
.order_by("-date_creation")[:1]
|
|
|
|
if len(last_creation) != 0:
|
|
|
|
last_maj = last_creation[0].date_creation
|
2017-04-13 22:50:00 +02:00
|
|
|
else:
|
2017-05-16 23:53:30 +02:00
|
|
|
try:
|
|
|
|
stage = Stage.objects.filter(auteur=request.user.profil).get(pk=pk)
|
|
|
|
except Stage.DoesNotExist:
|
|
|
|
return HttpResponseForbidden()
|
2018-03-21 13:14:14 +01:00
|
|
|
last_maj = stage.date_maj
|
2017-04-13 22:50:00 +02:00
|
|
|
avis_stage, _ = AvisStage.objects.get_or_create(stage=stage)
|
|
|
|
c_del = True
|
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Formset pour les avis des lieux
|
2017-04-13 22:50:00 +02:00
|
|
|
AvisLieuFormSet = forms.inlineformset_factory(
|
2017-04-15 20:03:08 +02:00
|
|
|
Stage, AvisLieu, form=AvisLieuForm, can_delete=c_del, extra=0)
|
2017-04-13 22:50:00 +02:00
|
|
|
|
|
|
|
if request.method == "POST":
|
2017-05-02 23:23:26 +02:00
|
|
|
# Lecture des données
|
2017-04-13 22:50:00 +02:00
|
|
|
form = StageForm(request.POST, request=request, instance=stage, prefix="stage")
|
|
|
|
avis_stage_form = AvisStageForm(request.POST,
|
|
|
|
instance=avis_stage, prefix="avis")
|
|
|
|
avis_lieu_formset = AvisLieuFormSet(request.POST, instance=stage,
|
|
|
|
prefix="lieux")
|
2017-05-02 23:23:26 +02:00
|
|
|
|
|
|
|
# Validation et enregistrement
|
2017-04-13 22:50:00 +02:00
|
|
|
if (form.is_valid() and
|
|
|
|
avis_stage_form.is_valid() and
|
|
|
|
avis_lieu_formset.is_valid()):
|
|
|
|
stage = form.save()
|
|
|
|
avis_stage_form.instance.stage = stage
|
|
|
|
avis_stage_form.save()
|
|
|
|
avis_lieu_formset.save()
|
2018-12-30 16:29:36 +01:00
|
|
|
#print(request.POST)
|
2017-05-16 23:53:30 +02:00
|
|
|
if "continuer" in request.POST:
|
2017-05-17 13:14:43 +02:00
|
|
|
if pk is None:
|
|
|
|
return redirect(reverse('avisstage:stage_edit',kwargs={'pk':stage.id}))
|
2017-05-16 23:53:30 +02:00
|
|
|
else:
|
|
|
|
return redirect(reverse('avisstage:stage',
|
|
|
|
kwargs={'pk':stage.id}))
|
2017-04-13 22:50:00 +02:00
|
|
|
else:
|
|
|
|
form = StageForm(instance=stage, prefix="stage")
|
|
|
|
avis_stage_form = AvisStageForm(instance=avis_stage, prefix="avis")
|
|
|
|
avis_lieu_formset = AvisLieuFormSet(instance=stage, prefix="lieux")
|
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Affichage du formulaire
|
2017-04-13 22:50:00 +02:00
|
|
|
return render(request, "avisstage/formulaires/stage.html",
|
|
|
|
{'form': form, 'avis_stage_form': avis_stage_form,
|
2017-04-15 20:03:08 +02:00
|
|
|
'avis_lieu_formset': avis_lieu_formset,
|
2018-03-21 13:14:14 +01:00
|
|
|
'creation': pk is None, "last_maj": last_maj})
|
2017-06-16 00:17:25 +02:00
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Ajout d'un lieu de stage
|
|
|
|
#login_required
|
2017-05-16 00:22:59 +02:00
|
|
|
|
|
|
|
# Stage
|
|
|
|
@login_required
|
|
|
|
def save_lieu(request):
|
|
|
|
normalien = request.user.profil
|
|
|
|
|
|
|
|
if request.method == "POST":
|
|
|
|
pk = request.POST.get("id", None)
|
2018-12-30 16:29:36 +01:00
|
|
|
#print(request.POST)
|
2017-05-16 00:22:59 +02:00
|
|
|
jitter = False
|
|
|
|
if pk is None or pk == '':
|
|
|
|
lieu = Lieu()
|
|
|
|
else:
|
|
|
|
# Modification du lieu
|
|
|
|
lieu = get_object_or_404(Lieu, pk=pk)
|
|
|
|
|
|
|
|
# On regarde si les stages associés à ce lieu "appartiennent" tous à l'utilisateur
|
|
|
|
not_same_user = lieu.stages.exclude(auteur=normalien).count()
|
|
|
|
|
|
|
|
# Si d'autres personnes ont un stage à cet endroit, on crée un nouveau lieu, un peu à côté
|
|
|
|
if not_same_user > 0:
|
|
|
|
lieu = Lieu()
|
|
|
|
# Servira à bouger un peu le lieu
|
|
|
|
jitter = True
|
|
|
|
|
|
|
|
# Lecture des données
|
|
|
|
form = LieuForm(request.POST, instance=lieu)
|
|
|
|
|
|
|
|
# Validation et enregistrement
|
|
|
|
if form.is_valid():
|
2017-10-06 10:29:00 +02:00
|
|
|
lieu = form.save(commit=False)
|
2017-05-16 00:22:59 +02:00
|
|
|
if jitter:
|
|
|
|
cdx, cdy = lieu.coord.get_coords()
|
|
|
|
ang = random.random() * 6.29;
|
|
|
|
rad = (random.random() + 0.5) * 3e-4
|
|
|
|
cdx += math.cos(ang) * rad;
|
|
|
|
cdy += math.sin(ang) * rad;
|
|
|
|
lieu.coord.set_coords((cdx, cdy))
|
|
|
|
lieu.save()
|
2017-10-06 10:29:00 +02:00
|
|
|
|
|
|
|
# Élimination des doublons
|
|
|
|
if pk is None or pk == "":
|
|
|
|
olieux = Lieu.objects.filter(nom=lieu.nom, coord__distance_lte=(lieu.coord, 10))
|
|
|
|
for olieu in olieux:
|
|
|
|
if olieu.type_lieu == lieu.type_lieu and \
|
|
|
|
olieu.ville == lieu.ville and \
|
|
|
|
olieu.pays == lieu.pays:
|
|
|
|
return JsonResponse({"success": True, "id": olieu.id})
|
|
|
|
|
|
|
|
lieu.save()
|
2017-05-16 00:22:59 +02:00
|
|
|
return JsonResponse({"success": True, "id": lieu.id})
|
|
|
|
else:
|
|
|
|
return JsonResponse({"success": False,
|
|
|
|
"errors": form.errors})
|
|
|
|
else:
|
|
|
|
return JsonResponse({"erreur": "Aucune donnée POST"})
|
|
|
|
|
2017-05-16 22:56:19 +02:00
|
|
|
class LieuAjout(LoginRequiredMixin, CreateView):
|
2017-04-11 00:20:14 +02:00
|
|
|
model = Lieu
|
|
|
|
form_class = LieuForm
|
|
|
|
template_name = 'avisstage/formulaires/lieu.html'
|
2017-04-18 02:43:05 +02:00
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Retourne d'un JSON si requête AJAX
|
2017-04-18 02:43:05 +02:00
|
|
|
def form_valid(self, form):
|
|
|
|
if self.request.GET.get("format", "") == "json":
|
|
|
|
self.object = form.save()
|
|
|
|
return JsonResponse({"success": True,
|
|
|
|
"id": self.object.id})
|
|
|
|
else:
|
|
|
|
super(LieuAjout, self).form_valid(form)
|
|
|
|
|
|
|
|
def form_invalid(self, form):
|
|
|
|
if self.request.GET.get("format", "") == "json":
|
|
|
|
return JsonResponse({"success": False,
|
|
|
|
"errors": form.errors})
|
|
|
|
else:
|
|
|
|
super(LieuAjout, self).form_valid(form)
|
2017-04-20 01:53:29 +02:00
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
# Passage d'un stage en mode public
|
2017-04-20 01:53:29 +02:00
|
|
|
@login_required
|
|
|
|
def publier_stage(request, pk):
|
|
|
|
if request.method != "POST":
|
|
|
|
return HttpResponseForbidden()
|
|
|
|
stage = get_object_or_404(Stage, pk=pk)
|
2017-05-02 23:23:26 +02:00
|
|
|
|
|
|
|
# Stage non possédé par l'utilisateur
|
2017-04-20 01:53:29 +02:00
|
|
|
if stage.auteur != request.user.profil:
|
|
|
|
return HttpResponseForbidden()
|
2017-05-02 23:23:26 +02:00
|
|
|
|
|
|
|
# Mise à jour du statut
|
2017-04-20 01:53:29 +02:00
|
|
|
if "publier" in request.POST:
|
|
|
|
stage.public = True
|
|
|
|
else:
|
|
|
|
stage.public = False
|
2017-05-02 23:23:26 +02:00
|
|
|
|
2017-04-20 01:53:29 +02:00
|
|
|
stage.save()
|
2017-05-02 23:23:26 +02:00
|
|
|
|
2017-04-20 01:53:29 +02:00
|
|
|
return redirect(reverse("avisstage:stage", kwargs={"pk": pk}))
|
2017-04-20 23:04:07 +02:00
|
|
|
|
2017-05-02 23:23:26 +02:00
|
|
|
#
|
2017-04-20 23:04:07 +02:00
|
|
|
# FEEDBACK
|
2017-05-02 23:23:26 +02:00
|
|
|
#
|
|
|
|
|
2017-04-20 23:04:07 +02:00
|
|
|
@login_required
|
2017-04-05 00:23:35 +02:00
|
|
|
def feedback(request):
|
2017-04-20 23:04:07 +02:00
|
|
|
if request.method == "POST":
|
|
|
|
form = FeedbackForm(request.POST)
|
|
|
|
if form.is_valid():
|
|
|
|
objet = form.cleaned_data['objet']
|
|
|
|
message = form.cleaned_data['message']
|
|
|
|
send_mail(
|
|
|
|
"[experiENS] "+ objet,
|
|
|
|
message,
|
|
|
|
request.user.username + "@clipper.ens.fr",
|
|
|
|
['champeno@clipper.ens.fr'],
|
|
|
|
fail_silently=False,
|
|
|
|
)
|
|
|
|
if request.GET.get("format", None) == "json":
|
|
|
|
return JsonResponse({"success": True})
|
|
|
|
return redirect(reverse("avisstage:index"))
|
|
|
|
else:
|
|
|
|
if request.GET.get("format", None) == "json":
|
|
|
|
return JsonResponse({"success": False,
|
|
|
|
"errors": form.errors})
|
|
|
|
else:
|
|
|
|
form = FeedbackForm()
|
2018-12-29 23:23:36 +01:00
|
|
|
raise Http404()
|
2017-06-27 23:14:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# STATISTIQUES
|
|
|
|
#
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
@staff_member_required
|
|
|
|
def statistiques(request):
|
|
|
|
nstages = Stage.objects.count()
|
|
|
|
npubstages = Stage.objects.filter(public=True).count()
|
2018-09-09 00:17:11 +02:00
|
|
|
nbymatiere_raw = Stage.objects.values('matieres__nom', 'public').annotate(scount=Count('matieres__nom'))
|
|
|
|
nbymatiere = defaultdict(dict)
|
|
|
|
for npm in nbymatiere_raw:
|
|
|
|
nbymatiere[npm["matieres__nom"]]["publics" if npm["public"] else "drafts"] = npm["scount"]
|
|
|
|
for mat, npm in nbymatiere.items():
|
|
|
|
npm["matiere"] = mat
|
2018-09-09 00:22:28 +02:00
|
|
|
nbymatiere = sorted(list(nbymatiere.values()), key=lambda npm: -npm.get("publics", 0))
|
2018-09-09 00:17:11 +02:00
|
|
|
nbylength = [("Vide", Stage.objects.filter(len_avis_stage__lt=5).count(),
|
|
|
|
Stage.objects.filter(len_avis_lieux__lt=5).count()),
|
|
|
|
("Court", Stage.objects.filter(len_avis_stage__lt=30, len_avis_stage__gt=4).count(),
|
|
|
|
Stage.objects.filter(len_avis_lieux__lt=30, len_avis_lieux__gt=4).count()),
|
|
|
|
("Moyen", Stage.objects.filter(len_avis_stage__lt=100, len_avis_stage__gt=29).count(),
|
|
|
|
Stage.objects.filter(len_avis_lieux__lt=100, len_avis_lieux__gt=29).count()),
|
|
|
|
("Long", Stage.objects.filter(len_avis_stage__gt=99).count(),
|
|
|
|
Stage.objects.filter(len_avis_lieux__gt=99).count())]
|
2017-06-27 23:14:24 +02:00
|
|
|
nusers = Normalien.objects.count()
|
|
|
|
nauts = Normalien.objects.filter(stages__isnull=False).distinct().count()
|
|
|
|
nbyaut = Counter(Normalien.objects.filter(stages__isnull=False).annotate(scount=Count('stages')).values_list('scount', flat="True")).items()
|
|
|
|
nlieux = Lieu.objects.filter(stages__isnull=False).distinct().count()
|
|
|
|
return render(request, 'avisstage/moderation/statistiques.html',
|
|
|
|
{'num_stages': nstages,
|
|
|
|
'num_stages_pub': npubstages,
|
|
|
|
'num_par_matiere': nbymatiere,
|
|
|
|
'num_users': nusers,
|
|
|
|
'num_auteurs': nauts,
|
|
|
|
'num_par_auteur': nbyaut,
|
2018-09-09 00:17:11 +02:00
|
|
|
'num_lieux_utiles': nlieux,
|
|
|
|
'num_par_longueur': nbylength,
|
|
|
|
})
|