from django import forms from django.shortcuts import get_object_or_404, redirect from django.template.defaultfilters import linebreaks_filter from django.utils.decorators import method_decorator from django.views.generic import FormView, ListView from activities.models import Hashtag, Post, PostInteraction, TimelineEvent from core.decorators import cache_page from core.models import Config from users.decorators import identity_required @method_decorator(identity_required, name="dispatch") class Home(FormView): template_name = "activities/home.html" class form_class(forms.Form): text = forms.CharField( widget=forms.Textarea( attrs={ "placeholder": "What's on your mind?", "_": f""" on load or input set characters to my.value.trim().length put { Config.system.post_length } - characters into #character-counter if characters > { Config.system.post_length } then set #character-counter's style.color to 'var(--color-text-error)' add [@disabled=] to #post-button else set #character-counter's style.color to '' remove @disabled from #post-button end """, }, ) ) content_warning = forms.CharField( required=False, label=Config.lazy_system_value("content_warning_text"), widget=forms.TextInput( attrs={ "class": "hidden", "placeholder": Config.lazy_system_value("content_warning_text"), }, ), ) def get_context_data(self): context = super().get_context_data() context["events"] = list( TimelineEvent.objects.filter( identity=self.request.identity, type__in=[TimelineEvent.Types.post, TimelineEvent.Types.boost], ) .select_related("subject_post", "subject_post__author") .prefetch_related("subject_post__attachments") .order_by("-created")[:50] ) context["interactions"] = PostInteraction.get_event_interactions( context["events"], self.request.identity ) context["current_page"] = "home" context["allows_refresh"] = True 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"), visibility=self.request.identity.config_identity.default_post_visibility, ) return redirect(".") @method_decorator( cache_page("cache_timeout_page_timeline", public_only=True), name="dispatch" ) 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 ( Post.objects.public() .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 @method_decorator( cache_page("cache_timeout_page_timeline", public_only=True), name="dispatch" ) class Local(ListView): template_name = "activities/local.html" extra_context = { "current_page": "local", "allows_refresh": True, } paginate_by = 50 def get_queryset(self): return ( Post.objects.local_public() .select_related("author") .prefetch_related("attachments") .order_by("-created")[:50] ) def get_context_data(self): context = super().get_context_data() context["interactions"] = PostInteraction.get_post_interactions( context["page_obj"], self.request.identity ) return context @method_decorator(identity_required, name="dispatch") class Federated(ListView): template_name = "activities/federated.html" extra_context = { "current_page": "federated", "allows_refresh": True, } paginate_by = 50 def get_queryset(self): return ( Post.objects.filter( visibility=Post.Visibilities.public, in_reply_to__isnull=True ) .select_related("author") .prefetch_related("attachments") .order_by("-created")[:50] ) def get_context_data(self): context = super().get_context_data() context["interactions"] = PostInteraction.get_post_interactions( context["page_obj"], self.request.identity ) return context @method_decorator(identity_required, name="dispatch") class Notifications(ListView): template_name = "activities/notifications.html" extra_context = { "current_page": "notifications", "allows_refresh": True, } paginate_by = 50 notification_types = { "followed": TimelineEvent.Types.followed, "boosted": TimelineEvent.Types.boosted, "mentioned": TimelineEvent.Types.mentioned, "liked": TimelineEvent.Types.liked, } 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) return ( TimelineEvent.objects.filter(identity=self.request.identity, type__in=types) .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