This repository has been archived on 2023-09-24. You can view files and clone it, but cannot push or open issues or pull requests.
takahe/activities/views/timelines.py

204 lines
6.7 KiB
Python
Raw Normal View History

2022-11-29 05:41:36 +01:00
from django.shortcuts import get_object_or_404, redirect
2022-11-14 00:14:38 +01:00
from django.template.defaultfilters import linebreaks_filter
from django.utils.decorators import method_decorator
2022-11-22 16:57:40 +01:00
from django.views.generic import FormView, ListView
2022-11-14 00:14:38 +01:00
2022-11-29 05:41:36 +01:00
from activities.models import Hashtag, Post, PostInteraction, TimelineEvent
from core.decorators import cache_page
2022-11-14 00:14:38 +01:00
from users.decorators import identity_required
2022-12-11 17:34:44 +01:00
from .compose import Compose
2022-11-14 00:14:38 +01:00
@method_decorator(identity_required, name="dispatch")
class Home(FormView):
template_name = "activities/home.html"
2022-12-11 17:34:44 +01:00
form_class = Compose.form_class
2022-11-14 00:14:38 +01:00
def get_context_data(self):
context = super().get_context_data()
context["events"] = list(
2022-11-14 02:42:47 +01:00
TimelineEvent.objects.filter(
2022-11-14 00:14:38 +01:00
identity=self.request.identity,
2022-11-14 02:42:47 +01:00
type__in=[TimelineEvent.Types.post, TimelineEvent.Types.boost],
2022-11-14 00:14:38 +01:00
)
.select_related("subject_post", "subject_post__author")
2022-11-17 07:00:10 +01:00
.prefetch_related("subject_post__attachments")
2022-11-18 01:43:00 +01:00
.order_by("-created")[:50]
2022-11-14 02:42:47 +01:00
)
context["interactions"] = PostInteraction.get_event_interactions(
context["events"], self.request.identity
)
2022-11-14 00:14:38 +01:00
context["current_page"] = "home"
context["allows_refresh"] = True
2022-11-14 00:14:38 +01:00
return context
def form_valid(self, form):
Post.create_local(
author=self.request.identity,
content=linebreaks_filter(form.cleaned_data["text"]),
summary=form.cleaned_data.get("content_warning"),
2022-11-26 03:33:46 +01:00
visibility=self.request.identity.config_identity.default_post_visibility,
2022-11-14 00:14:38 +01:00
)
return redirect(".")
2022-12-05 18:55:30 +01:00
@method_decorator(
cache_page("cache_timeout_page_timeline", public_only=True), name="dispatch"
2022-12-05 18:55:30 +01:00
)
2022-11-29 05:41:36 +01:00
class Tag(ListView):
template_name = "activities/tag.html"
extra_context = {
"current_page": "tag",
"allows_refresh": True,
}
paginate_by = 50
def get(self, request, hashtag, *args, **kwargs):
tag = hashtag.lower().lstrip("#")
if hashtag != tag:
# SEO sanitize
return redirect(f"/tags/{tag}/", permanent=True)
self.hashtag = get_object_or_404(Hashtag.objects.public(), hashtag=tag)
return super().get(request, *args, **kwargs)
def get_queryset(self):
return (
2022-11-29 06:14:52 +01:00
Post.objects.public()
2022-11-29 05:41:36 +01:00
.tagged_with(self.hashtag)
.select_related("author")
.prefetch_related("attachments")
.order_by("-created")[:50]
)
def get_context_data(self):
context = super().get_context_data()
context["hashtag"] = self.hashtag
context["interactions"] = PostInteraction.get_post_interactions(
context["page_obj"], self.request.identity
)
return context
2022-12-05 18:55:30 +01:00
@method_decorator(
cache_page("cache_timeout_page_timeline", public_only=True), name="dispatch"
2022-12-05 18:55:30 +01:00
)
2022-11-22 16:57:40 +01:00
class Local(ListView):
2022-11-14 02:42:47 +01:00
template_name = "activities/local.html"
extra_context = {
"current_page": "local",
"allows_refresh": True,
}
2022-11-22 16:57:40 +01:00
paginate_by = 50
2022-11-14 02:42:47 +01:00
2022-11-22 16:57:40 +01:00
def get_queryset(self):
return (
2022-11-29 05:41:36 +01:00
Post.objects.local_public()
2022-11-14 02:42:47 +01:00
.select_related("author")
2022-11-17 07:00:10 +01:00
.prefetch_related("attachments")
2022-11-18 01:43:00 +01:00
.order_by("-created")[:50]
2022-11-14 02:42:47 +01:00
)
def get_context_data(self):
context = super().get_context_data()
context["interactions"] = PostInteraction.get_post_interactions(
context["page_obj"], self.request.identity
)
return context
2022-11-14 02:42:47 +01:00
2022-11-14 00:14:38 +01:00
@method_decorator(identity_required, name="dispatch")
2022-11-22 16:57:40 +01:00
class Federated(ListView):
2022-11-14 00:14:38 +01:00
template_name = "activities/federated.html"
extra_context = {
"current_page": "federated",
"allows_refresh": True,
}
2022-11-22 16:57:40 +01:00
paginate_by = 50
2022-11-14 00:14:38 +01:00
2022-11-22 16:57:40 +01:00
def get_queryset(self):
return (
Post.objects.filter(
visibility=Post.Visibilities.public, in_reply_to__isnull=True
)
2022-11-14 00:14:38 +01:00
.select_related("author")
2022-11-17 07:00:10 +01:00
.prefetch_related("attachments")
2022-11-18 01:43:00 +01:00
.order_by("-created")[:50]
2022-11-14 00:14:38 +01:00
)
2022-11-14 02:42:47 +01:00
def get_context_data(self):
context = super().get_context_data()
context["interactions"] = PostInteraction.get_post_interactions(
context["page_obj"], self.request.identity
)
return context
2022-11-14 02:42:47 +01:00
@method_decorator(identity_required, name="dispatch")
2022-11-22 16:57:40 +01:00
class Notifications(ListView):
2022-11-14 02:42:47 +01:00
template_name = "activities/notifications.html"
extra_context = {
"current_page": "notifications",
"allows_refresh": True,
}
2022-11-22 16:57:40 +01:00
paginate_by = 50
notification_types = {
"followed": TimelineEvent.Types.followed,
"boosted": TimelineEvent.Types.boosted,
"mentioned": TimelineEvent.Types.mentioned,
"liked": TimelineEvent.Types.liked,
}
2022-11-14 02:42:47 +01:00
2022-11-22 16:57:40 +01:00
def get_queryset(self):
# Did they ask to change options?
notification_options = self.request.session.get("notification_options", {})
for type_name in self.notification_types:
notification_options.setdefault(type_name, True)
if self.request.GET.get(type_name) == "true":
notification_options[type_name] = True
elif self.request.GET.get(type_name) == "false":
notification_options[type_name] = False
self.request.session["notification_options"] = notification_options
# Return appropriate events
types = []
for type_name, type in self.notification_types.items():
if notification_options.get(type_name, True):
types.append(type)
2022-11-22 16:57:40 +01:00
return (
TimelineEvent.objects.filter(identity=self.request.identity, type__in=types)
2022-11-18 02:52:00 +01:00
.order_by("-created")[:50]
.select_related("subject_post", "subject_post__author", "subject_identity")
)
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
# Collapse similar notifications into one
events = []
for event in context["page_obj"]:
if (
events
and event.type
in [
TimelineEvent.Types.liked,
TimelineEvent.Types.boosted,
TimelineEvent.Types.mentioned,
]
and event.subject_post_id == events[-1].subject_post_id
):
events[-1].collapsed = True
events.append(event)
# Retrieve what kinds of things to show
context["events"] = events
context["notification_options"] = self.request.session["notification_options"]
context["interactions"] = PostInteraction.get_event_interactions(
context["page_obj"],
self.request.identity,
)
return context