Sith/forum/views.py

441 lines
14 KiB
Python
Raw Normal View History

#
# Copyright 2016,2017,2018
# - Skia <skia@libskia.so>
# - Sli <antoine@bartuccio.fr>
#
# Ce fichier fait partie du site de l'Association des Étudiants de l'UTBM,
# http://ae.utbm.fr.
#
# This program is free software; you can redistribute it and/or modify it under
# the terms of the GNU General Public License a published by the Free Software
# Foundation; either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# this program; if not, write to the Free Sofware Foundation, Inc., 59 Temple
# Place - Suite 330, Boston, MA 02111-1307, USA.
#
#
2024-08-06 09:42:10 +00:00
import logging
2024-06-27 12:46:43 +00:00
import math
2024-07-22 09:40:11 +00:00
from functools import partial
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
2024-07-18 18:23:30 +00:00
from django.contrib.auth.mixins import LoginRequiredMixin
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
2024-06-27 12:46:43 +00:00
from django.db import IntegrityError
2024-06-24 11:07:36 +00:00
from django.shortcuts import get_object_or_404
from django.urls import reverse_lazy
from django.utils import html, timezone
2024-07-22 09:40:11 +00:00
from django.utils.decorators import method_decorator
2024-06-24 11:07:36 +00:00
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
2024-07-22 09:40:11 +00:00
from honeypot.decorators import check_honeypot
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,
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):
2024-06-27 12:46:43 +00:00
fi = request.user.forum_infos
fi.last_read_date = timezone.now()
fi.save()
2017-01-28 23:16:41 +00:00
try:
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
2024-06-27 12:46:43 +00:00
except IntegrityError:
2017-06-12 07:58:24 +00:00
pass
2024-06-27 12:46:43 +00:00
return super().get(request, *args, **kwargs)
2017-01-28 23:16:41 +00:00
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):
2024-06-27 12:46:43 +00:00
init = super().get_initial()
parent = Forum.objects.filter(id=self.request.GET["parent"]).first()
if parent is not None:
2018-10-04 19:29:19 +00:00
init["parent"] = parent
init["owner_club"] = parent.owner_club
init["edit_groups"] = parent.edit_groups.all()
init["view_groups"] = parent.view_groups.all()
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):
2024-06-27 12:46:43 +00:00
ret = super().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):
2024-06-27 12:46:43 +00:00
kwargs = super().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
2024-07-22 09:40:11 +00:00
@method_decorator(
partial(check_honeypot, field_name=settings.HONEYPOT_FIELD_NAME_FORUM), name="post"
)
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
2024-06-27 12:46:43 +00:00
return super().dispatch(request, *args, **kwargs)
2017-01-21 02:42:06 +00:00
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
2024-06-27 12:46:43 +00:00
return super().form_valid(form)
2017-01-21 02:42:06 +00:00
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(
2024-07-18 18:23:30 +00:00
LoginRequiredMixin, CanViewMixin, 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):
2024-08-16 20:43:54 +00:00
topic: ForumTopic = self.object
2024-06-27 12:46:43 +00:00
kwargs = super().get_context_data(**kwargs)
2024-08-16 20:43:54 +00:00
msg = topic.get_first_unread_message(self.request.user)
2024-06-27 12:46:43 +00:00
if msg is None:
kwargs["first_unread_message_id"] = math.inf
else:
2018-10-04 19:29:19 +00:00
kwargs["first_unread_message_id"] = msg.id
paginator = Paginator(
2024-08-16 20:43:54 +00:00
topic.messages.select_related("author__avatar_pict", "topic__forum")
2018-10-04 19:29:19 +00:00
.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
2024-07-22 09:40:11 +00:00
@method_decorator(
partial(check_honeypot, field_name=settings.HONEYPOT_FIELD_NAME_FORUM), name="post"
)
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()
2024-06-27 12:46:43 +00:00
return super().form_valid(form)
2017-02-24 00:50:05 +00:00
def get_context_data(self, **kwargs):
2024-06-27 12:46:43 +00:00
kwargs = super().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
2024-07-22 09:40:11 +00:00
@method_decorator(
partial(check_honeypot, field_name=settings.HONEYPOT_FIELD_NAME_FORUM), name="post"
)
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
2024-06-27 12:46:43 +00:00
return super().dispatch(request, *args, **kwargs)
2017-01-21 02:42:06 +00:00
def get_initial(self):
2024-06-27 12:46:43 +00:00
init = super().get_initial()
2017-01-21 02:42:06 +00:00
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:
2024-08-06 09:42:10 +00:00
logging.error(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
2024-06-27 12:46:43 +00:00
return super().form_valid(form)
2017-01-21 02:42:06 +00:00
2017-02-24 16:22:13 +00:00
def get_context_data(self, **kwargs):
2024-06-27 12:46:43 +00:00
kwargs = super().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