Sith/forum/views.py

423 lines
14 KiB
Python
Raw Permalink Normal View History

2023-04-06 11:08:42 +00:00
## -*- coding:utf-8 -*-
#
2023-04-06 11:08:42 +00:00
# Copyright 2023 © AE UTBM
# ae@utbm.fr / ae.info@utbm.fr
# All contributors are listed in the CONTRIBUTORS file.
#
2023-04-06 11:08:42 +00:00
# This file is part of the website of the UTBM Student Association (AE UTBM),
# https://ae.utbm.fr.
#
2023-04-06 11:08:42 +00:00
# You can find the whole source code at https://github.com/ae-utbm/sith3
#
2023-04-06 11:08:42 +00:00
# LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3)
# SEE : https://raw.githubusercontent.com/ae-utbm/sith3/master/LICENSE
# OR WITHIN THE LOCAL FILE "LICENSE"
#
2023-04-06 11:08:42 +00:00
# PREVIOUSLY LICENSED UNDER THE MIT LICENSE,
# SEE : https://raw.githubusercontent.com/ae-utbm/sith3/master/LICENSE.old
# OR WITHIN THE LOCAL FILE "LICENSE.old"
#
2024-06-24 11:07:36 +00:00
from ajax_select import make_ajax_field
2017-01-21 02:42:06 +00:00
from django import forms
2024-06-24 11:07:36 +00:00
from django.conf import settings
2017-01-21 02:42:06 +00:00
from django.core.exceptions import PermissionDenied
2024-06-24 11:07:36 +00:00
from django.core.paginator import EmptyPage, PageNotAnInteger, Paginator
from django.shortcuts import get_object_or_404
from django.urls import reverse_lazy
from django.utils import html, timezone
from django.utils.translation import gettext_lazy as _
from django.views.generic import DetailView, ListView, RedirectView
from django.views.generic.detail import SingleObjectMixin
from django.views.generic.edit import CreateView, DeleteView, UpdateView
from haystack.query import RelatedSearchQuerySet
2017-02-24 00:50:09 +00:00
from core.views import (
2024-06-24 11:07:36 +00:00
CanCreateMixin,
CanEditMixin,
CanEditPropMixin,
2024-06-24 11:07:36 +00:00
CanViewMixin,
UserIsLoggedMixin,
can_view,
)
from core.views.forms import MarkdownInput
2024-06-24 11:07:36 +00:00
from forum.models import Forum, ForumMessage, ForumMessageMeta, ForumTopic
class ForumSearchView(ListView):
template_name = "forum/search.jinja"
def get_queryset(self):
query = self.request.GET.get("query", "")
order_by = self.request.GET.get("order", "")
try:
queryset = (
RelatedSearchQuerySet()
.models(ForumMessage)
.autocomplete(auto=html.escape(query))
)
except TypeError:
return []
if order_by == "date":
2018-12-13 09:12:20 +00:00
queryset = queryset.order_by("-date")
queryset = queryset.load_all()
queryset = queryset.load_all_queryset(
ForumMessage,
ForumMessage.objects.all()
.prefetch_related("topic__forum__edit_groups")
.prefetch_related("topic__forum__view_groups")
.prefetch_related("topic__forum__owner_club"),
)
# Filter unauthorized responses
resp = []
count = 0
max_count = 30
for r in queryset:
if count >= max_count:
return resp
if can_view(r.object, self.request.user) and can_view(
r.object.topic, self.request.user
):
resp.append(r.object)
count += 1
return resp
2017-01-21 02:42:06 +00:00
2017-06-12 07:58:24 +00:00
2017-02-24 00:50:00 +00:00
class ForumMainView(ListView):
2018-10-04 19:29:19 +00:00
queryset = Forum.objects.filter(parent=None).prefetch_related(
"children___last_message__author", "children___last_message__topic"
)
2017-01-21 02:42:06 +00:00
template_name = "forum/main.jinja"
2017-06-12 07:58:24 +00:00
2017-01-28 23:16:41 +00:00
class ForumMarkAllAsRead(RedirectView):
permanent = False
2018-10-04 19:29:19 +00:00
url = reverse_lazy("forum:last_unread")
2017-01-28 23:16:41 +00:00
def get(self, request, *args, **kwargs):
try:
fi = request.user.forum_infos
fi.last_read_date = timezone.now()
fi.save()
for m in request.user.read_messages.filter(date__lt=fi.last_read_date):
2017-06-12 07:58:24 +00:00
m.readers.remove(request.user) # Clean up to keep table low in data
except:
pass
2017-01-28 23:16:41 +00:00
return super(ForumMarkAllAsRead, self).get(request, *args, **kwargs)
2017-06-12 07:58:24 +00:00
class ForumFavoriteTopics(ListView):
model = ForumTopic
template_name = "forum/favorite_topics.jinja"
paginate_by = settings.SITH_FORUM_PAGE_LENGTH / 2
def get_queryset(self):
topic_list = self.request.user.favorite_topics.all()
return topic_list
2018-04-26 13:20:45 +00:00
2017-01-28 23:16:41 +00:00
class ForumLastUnread(ListView):
model = ForumTopic
template_name = "forum/last_unread.jinja"
2017-05-20 10:40:30 +00:00
paginate_by = settings.SITH_FORUM_PAGE_LENGTH / 2
2017-01-28 23:16:41 +00:00
def get_queryset(self):
2018-10-04 19:29:19 +00:00
topic_list = (
self.model.objects.filter(
_last_message__date__gt=self.request.user.forum_infos.last_read_date
)
.exclude(_last_message__readers=self.request.user)
.order_by("-_last_message__date")
.select_related("_last_message__author", "author")
.prefetch_related("forum__edit_groups")
)
return topic_list
2017-01-28 23:16:41 +00:00
2017-06-12 07:58:24 +00:00
2018-04-26 13:20:45 +00:00
class ForumNameField(forms.ModelChoiceField):
def label_from_instance(self, obj):
return obj.get_full_name()
class ForumForm(forms.ModelForm):
class Meta:
model = Forum
2018-10-04 19:29:19 +00:00
fields = [
"name",
"parent",
"number",
"owner_club",
"is_category",
"edit_groups",
"view_groups",
]
edit_groups = make_ajax_field(Forum, "edit_groups", "groups", help_text="")
view_groups = make_ajax_field(Forum, "view_groups", "groups", help_text="")
2018-04-26 13:20:45 +00:00
parent = ForumNameField(Forum.objects.all())
2017-06-12 07:58:24 +00:00
class ForumCreateView(CanCreateMixin, CreateView):
2017-01-21 02:42:06 +00:00
model = Forum
form_class = ForumForm
2017-01-21 02:42:06 +00:00
template_name = "core/create.jinja"
def get_initial(self):
init = super(ForumCreateView, self).get_initial()
2017-01-21 21:47:30 +00:00
try:
2018-10-04 19:29:19 +00:00
parent = Forum.objects.filter(id=self.request.GET["parent"]).first()
init["parent"] = parent
init["owner_club"] = parent.owner_club
init["edit_groups"] = parent.edit_groups.all()
init["view_groups"] = parent.view_groups.all()
2017-06-12 07:58:24 +00:00
except:
pass
2017-01-21 02:42:06 +00:00
return init
2017-06-12 07:58:24 +00:00
class ForumEditForm(ForumForm):
2018-10-04 19:29:19 +00:00
recursive = forms.BooleanField(
label=_("Apply rights and club owner recursively"), required=False
)
2017-06-12 07:58:24 +00:00
class ForumEditView(CanEditPropMixin, UpdateView):
2017-01-21 02:42:06 +00:00
model = Forum
pk_url_kwarg = "forum_id"
form_class = ForumEditForm
2017-01-21 02:42:06 +00:00
template_name = "core/edit.jinja"
2018-10-04 19:29:19 +00:00
success_url = reverse_lazy("forum:main")
2017-01-21 02:42:06 +00:00
def form_valid(self, form):
ret = super(ForumEditView, self).form_valid(form)
2018-10-04 19:29:19 +00:00
if form.cleaned_data["recursive"]:
self.object.apply_rights_recursively()
return ret
2017-06-12 07:58:24 +00:00
class ForumDeleteView(CanEditPropMixin, DeleteView):
model = Forum
pk_url_kwarg = "forum_id"
template_name = "core/delete_confirm.jinja"
2018-10-04 19:29:19 +00:00
success_url = reverse_lazy("forum:main")
2017-06-12 07:58:24 +00:00
class ForumDetailView(CanViewMixin, DetailView):
2017-01-21 02:42:06 +00:00
model = Forum
template_name = "forum/forum.jinja"
pk_url_kwarg = "forum_id"
2017-01-21 11:28:32 +00:00
def get_context_data(self, **kwargs):
kwargs = super(ForumDetailView, self).get_context_data(**kwargs)
2018-10-04 19:29:19 +00:00
qs = (
self.object.topics.order_by("-_last_message__date")
.select_related("_last_message__author", "author")
2017-06-12 07:58:24 +00:00
.prefetch_related("forum__edit_groups")
2018-10-04 19:29:19 +00:00
)
paginator = Paginator(qs, settings.SITH_FORUM_PAGE_LENGTH)
page = self.request.GET.get("topic_page")
2017-05-20 10:40:30 +00:00
try:
kwargs["topics"] = paginator.page(page)
except PageNotAnInteger:
kwargs["topics"] = paginator.page(1)
except EmptyPage:
kwargs["topics"] = paginator.page(paginator.num_pages)
2017-01-21 11:28:32 +00:00
return kwargs
2017-06-12 07:58:24 +00:00
class TopicForm(forms.ModelForm):
class Meta:
model = ForumMessage
2018-10-04 19:29:19 +00:00
fields = ["title", "message"]
widgets = {"message": MarkdownInput}
2017-04-12 17:19:25 +00:00
title = forms.CharField(required=True, label=_("Title"))
2017-06-12 07:58:24 +00:00
class ForumTopicCreateView(CanCreateMixin, CreateView):
2017-01-21 03:19:15 +00:00
model = ForumMessage
form_class = TopicForm
2017-04-12 17:19:25 +00:00
template_name = "forum/reply.jinja"
2017-01-21 02:42:06 +00:00
def dispatch(self, request, *args, **kwargs):
2018-10-04 19:29:19 +00:00
self.forum = get_object_or_404(
Forum, id=self.kwargs["forum_id"], is_category=False
)
2017-01-21 02:42:06 +00:00
if not request.user.can_view(self.forum):
raise PermissionDenied
return super(ForumTopicCreateView, self).dispatch(request, *args, **kwargs)
def form_valid(self, form):
2018-10-04 19:29:19 +00:00
topic = ForumTopic(
_title=form.instance.title, author=self.request.user, forum=self.forum
)
2017-01-21 03:19:15 +00:00
topic.save()
form.instance.topic = topic
2017-01-21 02:42:06 +00:00
form.instance.author = self.request.user
return super(ForumTopicCreateView, self).form_valid(form)
2017-06-12 07:58:24 +00:00
2017-01-21 21:47:30 +00:00
class ForumTopicEditView(CanEditMixin, UpdateView):
2017-01-21 02:42:06 +00:00
model = ForumTopic
2018-10-04 19:29:19 +00:00
fields = ["forum"]
2017-01-21 02:42:06 +00:00
pk_url_kwarg = "topic_id"
template_name = "core/edit.jinja"
2018-10-04 19:29:19 +00:00
class ForumTopicSubscribeView(
CanViewMixin, UserIsLoggedMixin, SingleObjectMixin, RedirectView
):
model = ForumTopic
pk_url_kwarg = "topic_id"
permanent = False
def get(self, request, *args, **kwargs):
self.object = self.get_object()
if self.object.subscribed_users.filter(id=request.user.id).exists():
self.object.subscribed_users.remove(request.user)
else:
self.object.subscribed_users.add(request.user)
return super().get(request, *args, **kwargs)
def get_redirect_url(self, *args, **kwargs):
return self.object.get_absolute_url()
2017-06-12 07:58:24 +00:00
class ForumTopicDetailView(CanViewMixin, DetailView):
2017-01-21 02:42:06 +00:00
model = ForumTopic
pk_url_kwarg = "topic_id"
template_name = "forum/topic.jinja"
context_object_name = "topic"
2018-10-04 19:29:19 +00:00
queryset = ForumTopic.objects.select_related("forum__parent")
2017-01-21 02:42:06 +00:00
2017-01-28 23:16:41 +00:00
def get_context_data(self, **kwargs):
kwargs = super(ForumTopicDetailView, self).get_context_data(**kwargs)
try:
msg = self.object.get_first_unread_message(self.request.user)
2018-10-04 19:29:19 +00:00
kwargs["first_unread_message_id"] = msg.id
2017-01-28 23:16:41 +00:00
except:
2018-10-04 19:29:19 +00:00
kwargs["first_unread_message_id"] = float("inf")
paginator = Paginator(
self.object.messages.select_related("author__avatar_pict")
.prefetch_related("topic__forum__edit_groups", "readers")
.order_by("date"),
settings.SITH_FORUM_PAGE_LENGTH,
)
page = self.request.GET.get("page")
2017-03-12 19:24:16 +00:00
try:
kwargs["msgs"] = paginator.page(page)
except PageNotAnInteger:
kwargs["msgs"] = paginator.page(1)
except EmptyPage:
kwargs["msgs"] = paginator.page(paginator.num_pages)
2017-01-28 23:16:41 +00:00
return kwargs
2017-06-12 07:58:24 +00:00
class ForumMessageView(SingleObjectMixin, RedirectView):
model = ForumMessage
pk_url_kwarg = "message_id"
permanent = False
def get_redirect_url(self, *args, **kwargs):
self.object = self.get_object()
return self.object.get_url()
2017-06-12 07:58:24 +00:00
2017-01-21 11:28:32 +00:00
class ForumMessageEditView(CanEditMixin, UpdateView):
model = ForumMessage
2018-10-04 19:29:19 +00:00
form_class = forms.modelform_factory(
model=ForumMessage,
fields=["title", "message"],
widgets={"message": MarkdownInput},
)
template_name = "forum/reply.jinja"
2017-01-21 11:28:32 +00:00
pk_url_kwarg = "message_id"
2017-02-24 00:50:05 +00:00
def form_valid(self, form):
2018-10-04 19:29:19 +00:00
ForumMessageMeta(
message=self.object, user=self.request.user, action="EDIT"
).save()
2017-02-24 00:50:05 +00:00
return super(ForumMessageEditView, self).form_valid(form)
def get_context_data(self, **kwargs):
kwargs = super(ForumMessageEditView, self).get_context_data(**kwargs)
2018-10-04 19:29:19 +00:00
kwargs["topic"] = self.object.topic
return kwargs
2017-06-12 07:58:24 +00:00
2017-02-24 00:50:05 +00:00
class ForumMessageDeleteView(SingleObjectMixin, RedirectView):
model = ForumMessage
pk_url_kwarg = "message_id"
permanent = False
def get_redirect_url(self, *args, **kwargs):
2017-02-24 00:50:09 +00:00
self.object = self.get_object()
2017-02-24 00:50:05 +00:00
if self.object.can_be_moderated_by(self.request.user):
2018-10-04 19:29:19 +00:00
ForumMessageMeta(
message=self.object, user=self.request.user, action="DELETE"
).save()
2017-02-24 00:50:05 +00:00
return self.object.get_absolute_url()
2017-06-12 07:58:24 +00:00
2017-02-24 00:50:05 +00:00
class ForumMessageUndeleteView(SingleObjectMixin, RedirectView):
model = ForumMessage
pk_url_kwarg = "message_id"
permanent = False
def get_redirect_url(self, *args, **kwargs):
2017-02-24 00:50:09 +00:00
self.object = self.get_object()
2017-02-24 00:50:05 +00:00
if self.object.can_be_moderated_by(self.request.user):
2018-10-04 19:29:19 +00:00
ForumMessageMeta(
message=self.object, user=self.request.user, action="UNDELETE"
).save()
2017-02-24 00:50:05 +00:00
return self.object.get_absolute_url()
2017-06-12 07:58:24 +00:00
class ForumMessageCreateView(CanCreateMixin, CreateView):
2017-01-21 02:42:06 +00:00
model = ForumMessage
2018-10-04 19:29:19 +00:00
form_class = forms.modelform_factory(
model=ForumMessage,
fields=["title", "message"],
widgets={"message": MarkdownInput},
)
2017-02-24 16:22:13 +00:00
template_name = "forum/reply.jinja"
2017-01-21 02:42:06 +00:00
def dispatch(self, request, *args, **kwargs):
2018-10-04 19:29:19 +00:00
self.topic = get_object_or_404(ForumTopic, id=self.kwargs["topic_id"])
2017-01-21 02:42:06 +00:00
if not request.user.can_view(self.topic):
raise PermissionDenied
return super(ForumMessageCreateView, self).dispatch(request, *args, **kwargs)
def get_initial(self):
init = super(ForumMessageCreateView, self).get_initial()
try:
2018-10-04 19:29:19 +00:00
message = (
ForumMessage.objects.select_related("author")
.filter(id=self.request.GET["quote_id"])
.first()
)
init["message"] = "> ##### %s\n" % (
_("%(author)s said") % {"author": message.author.get_short_name()}
)
init["message"] += "\n".join(
["> " + line for line in message.message.split("\n")]
)
init["message"] += "\n\n"
2017-02-24 00:50:09 +00:00
except Exception as e:
print(repr(e))
2017-01-21 02:42:06 +00:00
return init
def form_valid(self, form):
form.instance.topic = self.topic
form.instance.author = self.request.user
return super(ForumMessageCreateView, self).form_valid(form)
2017-02-24 16:22:13 +00:00
def get_context_data(self, **kwargs):
kwargs = super(ForumMessageCreateView, self).get_context_data(**kwargs)
2018-10-04 19:29:19 +00:00
kwargs["topic"] = self.topic
2017-02-24 16:22:13 +00:00
return kwargs