123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224 |
- from __future__ import unicode_literals
- from django.contrib import messages
- from django.db import models
- from django.shortcuts import redirect, render
- from modelcluster.contrib.taggit import ClusterTaggableManager
- from modelcluster.fields import ParentalKey
- from taggit.models import Tag, TaggedItemBase
- from wagtail.admin.panels import FieldPanel, InlinePanel
- from wagtail.contrib.routable_page.models import RoutablePageMixin, route
- from wagtail.fields import StreamField
- from wagtail.models import Orderable, Page
- from wagtail.search import index
- from wagtail_editable_help.models import HelpText
- from bakerydemo.base.blocks import BaseStreamBlock
- class BlogPeopleRelationship(Orderable, models.Model):
- """
- This defines the relationship between the `People` within the `base`
- app and the BlogPage below. This allows People to be added to a BlogPage.
- We have created a two way relationship between BlogPage and People using
- the ParentalKey and ForeignKey
- """
- page = ParentalKey(
- "BlogPage", related_name="blog_person_relationship", on_delete=models.CASCADE
- )
- people = models.ForeignKey(
- "base.People", related_name="person_blog_relationship", on_delete=models.CASCADE
- )
- panels = [FieldPanel("people")]
- class BlogPageTag(TaggedItemBase):
- """
- This model allows us to create a many-to-many relationship between
- the BlogPage object and tags. There's a longer guide on using it at
- https://docs.wagtail.org/en/stable/reference/pages/model_recipes.html#tagging
- """
- content_object = ParentalKey(
- "BlogPage", related_name="tagged_items", on_delete=models.CASCADE
- )
- class BlogPage(Page):
- """
- A Blog Page
- We access the People object with an inline panel that references the
- ParentalKey's related_name in BlogPeopleRelationship. More docs:
- https://docs.wagtail.org/en/stable/topics/pages.html#inline-models
- """
- introduction = models.TextField(
- help_text=HelpText(
- "Blog page", "introduction", default="Text to describe the page"
- ),
- blank=True,
- )
- image = models.ForeignKey(
- "wagtailimages.Image",
- null=True,
- blank=True,
- on_delete=models.SET_NULL,
- related_name="+",
- help_text=HelpText(
- "Common",
- "hero image",
- default="Landscape mode only; horizontal width between 1000px and 3000px.",
- ),
- )
- body = StreamField(
- BaseStreamBlock(), verbose_name="Page body", blank=True, use_json_field=True
- )
- subtitle = models.CharField(blank=True, max_length=255)
- tags = ClusterTaggableManager(through=BlogPageTag, blank=True)
- date_published = models.DateField("Date article published", blank=True, null=True)
- content_panels = Page.content_panels + [
- FieldPanel("subtitle", classname="full"),
- FieldPanel("introduction", classname="full"),
- FieldPanel("image"),
- FieldPanel("body"),
- FieldPanel("date_published"),
- InlinePanel(
- "blog_person_relationship", label="Author(s)", panels=None, min_num=1
- ),
- FieldPanel("tags"),
- ]
- search_fields = Page.search_fields + [
- index.SearchField("body"),
- ]
- def authors(self):
- """
- Returns the BlogPage's related People. Again note that we are using
- the ParentalKey's related_name from the BlogPeopleRelationship model
- to access these objects. This allows us to access the People objects
- with a loop on the template. If we tried to access the blog_person_
- relationship directly we'd print `blog.BlogPeopleRelationship.None`
- """
- authors = [n.people for n in self.blog_person_relationship.all()]
- return authors
- @property
- def get_tags(self):
- """
- Similar to the authors function above we're returning all the tags that
- are related to the blog post into a list we can access on the template.
- We're additionally adding a URL to access BlogPage objects with that tag
- """
- tags = self.tags.all()
- for tag in tags:
- tag.url = "/" + "/".join(
- s.strip("/") for s in [self.get_parent().url, "tags", tag.slug]
- )
- return tags
- # Specifies parent to BlogPage as being BlogIndexPages
- parent_page_types = ["BlogIndexPage"]
- # Specifies what content types can exist as children of BlogPage.
- # Empty list means that no child content types are allowed.
- subpage_types = []
- class BlogIndexPage(RoutablePageMixin, Page):
- """
- Index page for blogs.
- We need to alter the page model's context to return the child page objects,
- the BlogPage objects, so that it works as an index page
- RoutablePageMixin is used to allow for a custom sub-URL for the tag views
- defined above.
- """
- introduction = models.TextField(
- help_text=HelpText(
- "Blog index page", "introduction", default="Text to describe the page"
- ),
- blank=True,
- )
- image = models.ForeignKey(
- "wagtailimages.Image",
- null=True,
- blank=True,
- on_delete=models.SET_NULL,
- related_name="+",
- help_text=HelpText(
- "Common",
- "hero image",
- default="Landscape mode only; horizontal width between 1000px and 3000px.",
- ),
- )
- content_panels = Page.content_panels + [
- FieldPanel("introduction", classname="full"),
- FieldPanel("image"),
- ]
- # Speficies that only BlogPage objects can live under this index page
- subpage_types = ["BlogPage"]
- # Defines a method to access the children of the page (e.g. BlogPage
- # objects). On the demo site we use this on the HomePage
- def children(self):
- return self.get_children().specific().live()
- # Overrides the context to list all child items, that are live, by the
- # date that they were published
- # https://docs.wagtail.org/en/stable/getting_started/tutorial.html#overriding-context
- def get_context(self, request):
- context = super(BlogIndexPage, self).get_context(request)
- context["posts"] = (
- BlogPage.objects.descendant_of(self).live().order_by("-date_published")
- )
- return context
- # This defines a Custom view that utilizes Tags. This view will return all
- # related BlogPages for a given Tag or redirect back to the BlogIndexPage.
- # More information on RoutablePages is at
- # https://docs.wagtail.org/en/stable/reference/contrib/routablepage.html
- @route(r"^tags/$", name="tag_archive")
- @route(r"^tags/([\w-]+)/$", name="tag_archive")
- def tag_archive(self, request, tag=None):
- try:
- tag = Tag.objects.get(slug=tag)
- except Tag.DoesNotExist:
- if tag:
- msg = 'There are no blog posts tagged with "{}"'.format(tag)
- messages.add_message(request, messages.INFO, msg)
- return redirect(self.url)
- posts = self.get_posts(tag=tag)
- context = {"tag": tag, "posts": posts}
- return render(request, "blog/blog_index_page.html", context)
- def serve_preview(self, request, mode_name):
- # Needed for previews to work
- return self.serve(request)
- # Returns the child BlogPage objects for this BlogPageIndex.
- # If a tag is used then it will filter the posts by tag.
- def get_posts(self, tag=None):
- posts = BlogPage.objects.live().descendant_of(self)
- if tag:
- posts = posts.filter(tags=tag)
- return posts
- # Returns the list of Tags for all child posts of this BlogPage.
- def get_child_tags(self):
- tags = []
- for post in self.get_posts():
- # Not tags.append() because we don't want a list of lists
- tags += post.get_tags
- tags = sorted(set(tags))
- return tags
|