2020-03-01 17:16:38 +01:00
|
|
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-03-31 04:16:30 +02:00
|
|
|
|
2020-03-30 17:27:02 +02:00
|
|
|
from datetime import datetime, timezone
|
2020-03-01 17:16:38 +01:00
|
|
|
|
2020-04-10 00:02:22 +02:00
|
|
|
from django.conf import settings
|
2020-03-27 13:50:02 +01:00
|
|
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
2020-03-28 01:45:13 +01:00
|
|
|
from django.contrib.contenttypes.models import ContentType
|
|
|
|
from django.db.models import F, Q
|
2020-03-27 16:19:33 +01:00
|
|
|
from django.urls import reverse_lazy
|
2020-03-27 00:40:35 +01:00
|
|
|
from django.views.generic import CreateView, DetailView, UpdateView, TemplateView
|
2020-03-27 01:31:54 +01:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2020-03-01 17:16:38 +01:00
|
|
|
from django_tables2.views import SingleTableView
|
2020-03-28 16:52:58 +01:00
|
|
|
from note.models import NoteUser, Alias, NoteSpecial
|
2020-03-27 21:18:27 +01:00
|
|
|
from permission.backends import PermissionBackend
|
2020-03-31 04:16:30 +02:00
|
|
|
from permission.views import ProtectQuerysetMixin
|
2020-03-28 01:45:13 +01:00
|
|
|
|
2020-03-27 18:02:22 +01:00
|
|
|
from .forms import ActivityForm, GuestForm
|
2020-03-28 16:52:58 +01:00
|
|
|
from .models import Activity, Guest, Entry
|
2020-03-28 01:45:13 +01:00
|
|
|
from .tables import ActivityTable, GuestTable, EntryTable
|
2020-03-27 00:40:35 +01:00
|
|
|
|
2020-03-01 17:16:38 +01:00
|
|
|
|
2020-03-31 04:16:30 +02:00
|
|
|
class ActivityCreateView(ProtectQuerysetMixin, LoginRequiredMixin, CreateView):
|
2020-03-01 17:16:38 +01:00
|
|
|
model = Activity
|
2020-03-27 01:31:54 +01:00
|
|
|
form_class = ActivityForm
|
2020-03-27 21:18:27 +01:00
|
|
|
|
2020-03-28 19:05:21 +01:00
|
|
|
def form_valid(self, form):
|
|
|
|
form.instance.creater = self.request.user
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
2020-03-27 21:18:27 +01:00
|
|
|
def get_success_url(self, **kwargs):
|
2020-03-28 19:05:21 +01:00
|
|
|
self.object.refresh_from_db()
|
|
|
|
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.object.pk})
|
2020-03-27 00:40:35 +01:00
|
|
|
|
|
|
|
|
2020-03-31 04:16:30 +02:00
|
|
|
class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView):
|
2020-03-01 17:16:38 +01:00
|
|
|
model = Activity
|
2020-03-27 18:02:22 +01:00
|
|
|
table_class = ActivityTable
|
2020-03-27 01:31:54 +01:00
|
|
|
|
2020-03-28 19:05:21 +01:00
|
|
|
def get_queryset(self):
|
2020-03-31 04:16:30 +02:00
|
|
|
return super().get_queryset().reverse()
|
2020-03-28 19:05:21 +01:00
|
|
|
|
2020-03-27 01:31:54 +01:00
|
|
|
def get_context_data(self, **kwargs):
|
2020-04-06 12:13:12 +02:00
|
|
|
context = super().get_context_data(**kwargs)
|
2020-03-27 01:31:54 +01:00
|
|
|
|
2020-04-06 12:13:12 +02:00
|
|
|
context['title'] = _("Activities")
|
2020-03-28 16:52:58 +01:00
|
|
|
|
|
|
|
upcoming_activities = Activity.objects.filter(date_end__gt=datetime.now())
|
2020-04-11 03:37:06 +02:00
|
|
|
context['upcoming'] = ActivityTable(
|
|
|
|
data=upcoming_activities.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view")))
|
2020-03-27 01:31:54 +01:00
|
|
|
|
2020-04-06 12:13:12 +02:00
|
|
|
return context
|
2020-03-01 17:16:38 +01:00
|
|
|
|
2020-03-27 00:40:35 +01:00
|
|
|
|
2020-03-31 04:16:30 +02:00
|
|
|
class ActivityDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
|
2020-03-01 17:16:38 +01:00
|
|
|
model = Activity
|
2020-03-27 19:47:43 +01:00
|
|
|
context_object_name = "activity"
|
2020-03-01 17:16:38 +01:00
|
|
|
|
2020-03-27 21:18:27 +01:00
|
|
|
def get_context_data(self, **kwargs):
|
2020-04-06 12:13:12 +02:00
|
|
|
context = super().get_context_data()
|
2020-03-27 21:18:27 +01:00
|
|
|
|
|
|
|
table = GuestTable(data=Guest.objects.filter(activity=self.object)
|
|
|
|
.filter(PermissionBackend.filter_queryset(self.request.user, Guest, "view")))
|
2020-04-06 12:13:12 +02:00
|
|
|
context["guests"] = table
|
2020-03-27 21:18:27 +01:00
|
|
|
|
2020-04-06 12:13:12 +02:00
|
|
|
context["activity_started"] = datetime.now(timezone.utc) > self.object.date_start
|
2020-03-30 17:27:02 +02:00
|
|
|
|
2020-04-06 12:13:12 +02:00
|
|
|
return context
|
2020-03-27 21:18:27 +01:00
|
|
|
|
2020-03-27 00:40:35 +01:00
|
|
|
|
2020-03-31 04:16:30 +02:00
|
|
|
class ActivityUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView):
|
2020-03-01 17:16:38 +01:00
|
|
|
model = Activity
|
2020-03-27 01:31:54 +01:00
|
|
|
form_class = ActivityForm
|
2020-03-27 21:18:27 +01:00
|
|
|
|
|
|
|
def get_success_url(self, **kwargs):
|
|
|
|
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.kwargs["pk"]})
|
2020-03-27 00:40:35 +01:00
|
|
|
|
|
|
|
|
2020-03-31 04:16:30 +02:00
|
|
|
class ActivityInviteView(ProtectQuerysetMixin, LoginRequiredMixin, CreateView):
|
2020-03-27 18:02:22 +01:00
|
|
|
model = Guest
|
|
|
|
form_class = GuestForm
|
|
|
|
template_name = "activity/activity_invite.html"
|
|
|
|
|
2020-03-30 00:42:32 +02:00
|
|
|
def get_form(self, form_class=None):
|
|
|
|
form = super().get_form(form_class)
|
2020-03-31 04:16:30 +02:00
|
|
|
form.activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view"))\
|
|
|
|
.get(pk=self.kwargs["pk"])
|
2020-03-30 00:42:32 +02:00
|
|
|
return form
|
|
|
|
|
2020-03-27 19:16:38 +01:00
|
|
|
def form_valid(self, form):
|
2020-03-31 04:16:30 +02:00
|
|
|
form.instance.activity = Activity.objects\
|
|
|
|
.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view")).get(pk=self.kwargs["pk"])
|
2020-03-27 19:16:38 +01:00
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
|
|
def get_success_url(self, **kwargs):
|
|
|
|
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.kwargs["pk"]})
|
|
|
|
|
2020-03-27 18:02:22 +01:00
|
|
|
|
2020-03-27 13:50:02 +01:00
|
|
|
class ActivityEntryView(LoginRequiredMixin, TemplateView):
|
2020-03-28 01:45:13 +01:00
|
|
|
template_name = "activity/activity_entry.html"
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
2020-04-06 12:13:12 +02:00
|
|
|
context = super().get_context_data(**kwargs)
|
2020-03-28 01:45:13 +01:00
|
|
|
|
2020-03-31 04:16:30 +02:00
|
|
|
activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view"))\
|
|
|
|
.get(pk=self.kwargs["pk"])
|
2020-04-06 12:13:12 +02:00
|
|
|
context["activity"] = activity
|
2020-03-28 01:45:13 +01:00
|
|
|
|
|
|
|
matched = []
|
|
|
|
|
|
|
|
pattern = "^$"
|
|
|
|
if "search" in self.request.GET:
|
|
|
|
pattern = self.request.GET["search"]
|
|
|
|
|
2020-03-28 13:38:31 +01:00
|
|
|
if not pattern:
|
|
|
|
pattern = "^$"
|
|
|
|
|
|
|
|
if pattern[0] != "^":
|
|
|
|
pattern = "^" + pattern
|
2020-03-28 01:45:13 +01:00
|
|
|
|
|
|
|
guest_qs = Guest.objects\
|
|
|
|
.annotate(balance=F("inviter__balance"), note_name=F("inviter__user__username"))\
|
|
|
|
.filter(Q(first_name__regex=pattern) | Q(last_name__regex=pattern)
|
|
|
|
| Q(inviter__alias__name__regex=pattern)
|
2020-03-28 19:05:21 +01:00
|
|
|
| Q(inviter__alias__normalized_name__regex=Alias.normalize(pattern))) \
|
|
|
|
.filter(PermissionBackend.filter_queryset(self.request.user, Guest, "view"))\
|
2020-03-28 01:45:13 +01:00
|
|
|
.distinct()[:20]
|
|
|
|
for guest in guest_qs:
|
|
|
|
guest.type = "Invité"
|
|
|
|
matched.append(guest)
|
|
|
|
|
|
|
|
note_qs = Alias.objects.annotate(last_name=F("note__noteuser__user__last_name"),
|
|
|
|
first_name=F("note__noteuser__user__first_name"),
|
|
|
|
username=F("note__noteuser__user__username"),
|
|
|
|
note_name=F("name"),
|
|
|
|
balance=F("note__balance"))\
|
|
|
|
.filter(Q(note__polymorphic_ctype__model="noteuser")
|
|
|
|
& (Q(note__noteuser__user__first_name__regex=pattern)
|
|
|
|
| Q(note__noteuser__user__last_name__regex=pattern)
|
2020-03-28 13:38:31 +01:00
|
|
|
| Q(name__regex=pattern)
|
2020-03-28 19:05:21 +01:00
|
|
|
| Q(normalized_name__regex=Alias.normalize(pattern)))) \
|
2020-04-10 00:02:22 +02:00
|
|
|
.filter(PermissionBackend.filter_queryset(self.request.user, Alias, "view"))
|
|
|
|
if settings.DATABASES[note_qs.db]["ENGINE"] == 'django.db.backends.postgresql_psycopg2':
|
|
|
|
note_qs = note_qs.distinct('note__pk')[:20]
|
|
|
|
else:
|
|
|
|
# SQLite doesn't support distinct fields. For compatibility reason (in dev mode), the note list will only
|
|
|
|
# have distinct aliases rather than distinct notes with a SQLite DB, but it can fill the result page.
|
|
|
|
# In production mode, please use PostgreSQL.
|
|
|
|
note_qs = note_qs.distinct()[:20]
|
2020-03-28 01:45:13 +01:00
|
|
|
for note in note_qs:
|
|
|
|
note.type = "Adhérent"
|
2020-03-28 16:52:58 +01:00
|
|
|
note.activity = activity
|
2020-03-28 01:45:13 +01:00
|
|
|
matched.append(note)
|
|
|
|
|
|
|
|
table = EntryTable(data=matched)
|
2020-04-06 12:13:12 +02:00
|
|
|
context["table"] = table
|
2020-03-28 01:45:13 +01:00
|
|
|
|
2020-04-06 12:13:12 +02:00
|
|
|
context["entries"] = Entry.objects.filter(activity=activity)
|
2020-03-28 16:52:58 +01:00
|
|
|
|
2020-04-06 12:13:12 +02:00
|
|
|
context["title"] = _('Entry for activity "{}"').format(activity.name)
|
|
|
|
context["noteuser_ctype"] = ContentType.objects.get_for_model(NoteUser).pk
|
|
|
|
context["notespecial_ctype"] = ContentType.objects.get_for_model(NoteSpecial).pk
|
2020-04-10 00:02:22 +02:00
|
|
|
|
2020-04-09 22:31:54 +02:00
|
|
|
context["activities_open"] = Activity.objects.filter(open=True).filter(
|
2020-04-06 07:06:52 +02:00
|
|
|
PermissionBackend.filter_queryset(self.request.user, Activity, "view")).filter(
|
|
|
|
PermissionBackend.filter_queryset(self.request.user, Activity, "change")).all()
|
|
|
|
|
2020-04-10 00:02:22 +02:00
|
|
|
return context
|