models.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. from __future__ import unicode_literals
  2. from django.contrib import messages
  3. from django.db import models
  4. from django.shortcuts import redirect, render
  5. from modelcluster.contrib.taggit import ClusterTaggableManager
  6. from modelcluster.fields import ParentalKey
  7. from taggit.models import Tag, TaggedItemBase
  8. from wagtail.contrib.wagtailroutablepage.models import RoutablePageMixin, route
  9. from wagtail.wagtailadmin.edit_handlers import FieldPanel, InlinePanel, StreamFieldPanel
  10. from wagtail.wagtailcore.fields import StreamField
  11. from wagtail.wagtailcore.models import Page, Orderable
  12. from wagtail.wagtailimages.edit_handlers import ImageChooserPanel
  13. from wagtail.wagtailsearch import index
  14. from wagtail.wagtailsnippets.edit_handlers import SnippetChooserPanel
  15. from bakerydemo.base.blocks import BaseStreamBlock
  16. class BlogPeopleRelationship(Orderable, models.Model):
  17. """
  18. This defines the relationship between the `People` within the `base`
  19. app and the BlogPage below allowing us to add people to a BlogPage.
  20. """
  21. page = ParentalKey(
  22. 'BlogPage', related_name='blog_person_relationship'
  23. )
  24. people = models.ForeignKey(
  25. 'base.People', related_name='person_blog_relationship'
  26. )
  27. panels = [
  28. SnippetChooserPanel('people')
  29. ]
  30. class BlogPageTag(TaggedItemBase):
  31. content_object = ParentalKey('BlogPage', related_name='tagged_items')
  32. class BlogPage(Page):
  33. """
  34. A Blog Page (Post)
  35. """
  36. introduction = models.TextField(
  37. help_text='Text to describe the page',
  38. blank=True)
  39. image = models.ForeignKey(
  40. 'wagtailimages.Image',
  41. null=True,
  42. blank=True,
  43. on_delete=models.SET_NULL,
  44. related_name='+',
  45. help_text='Landscape mode only; horizontal width between 1000px and 3000px.'
  46. )
  47. body = StreamField(
  48. BaseStreamBlock(), verbose_name="Page body", blank=True
  49. )
  50. subtitle = models.CharField(blank=True, max_length=255)
  51. tags = ClusterTaggableManager(through=BlogPageTag, blank=True)
  52. date_published = models.DateField("Date article published", blank=True, null=True)
  53. content_panels = Page.content_panels + [
  54. FieldPanel('subtitle', classname="full"),
  55. FieldPanel('introduction', classname="full"),
  56. ImageChooserPanel('image'),
  57. StreamFieldPanel('body'),
  58. FieldPanel('date_published'),
  59. InlinePanel(
  60. 'blog_person_relationship', label="Author(s)",
  61. panels=None, min_num=1),
  62. FieldPanel('tags'),
  63. ]
  64. search_fields = Page.search_fields + [
  65. index.SearchField('title'),
  66. index.SearchField('body'),
  67. ]
  68. def authors(self):
  69. """
  70. Returns the BlogPage's related People
  71. """
  72. authors = [
  73. n.people for n in self.blog_person_relationship.all()
  74. ]
  75. return authors
  76. @property
  77. def get_tags(self):
  78. """
  79. Returns the BlogPage's related list of Tags.
  80. Each Tag is modified to include a url attribute
  81. """
  82. tags = self.tags.all()
  83. for tag in tags:
  84. tag.url = '/'+'/'.join(s.strip('/') for s in [
  85. self.get_parent().url,
  86. 'tags',
  87. tag.slug
  88. ])
  89. return tags
  90. parent_page_types = ['BlogIndexPage']
  91. # Define what content types can exist as children of BlogPage.
  92. # Empty list means that no child content types are allowed.
  93. subpage_types = []
  94. class BlogIndexPage(RoutablePageMixin, Page):
  95. """
  96. Index page for blogs.
  97. We need to alter the page model's context to return the child page objects - the
  98. BlogPage - so that it works as an index page
  99. RoutablePageMixin is used to allow for a custom sub-URL for tag views.
  100. """
  101. introduction = models.TextField(
  102. help_text='Text to describe the page',
  103. blank=True)
  104. image = models.ForeignKey(
  105. 'wagtailimages.Image',
  106. null=True,
  107. blank=True,
  108. on_delete=models.SET_NULL,
  109. related_name='+',
  110. help_text='Landscape mode only; horizontal width between 1000px and 3000px.'
  111. )
  112. # What pages types can live under this page type?
  113. subpage_types = ['BlogPage']
  114. def children(self):
  115. return self.get_children().specific().live()
  116. def get_context(self, request):
  117. context = super(BlogIndexPage, self).get_context(request)
  118. context['posts'] = BlogPage.objects.descendant_of(
  119. self).live().order_by(
  120. '-date_published')
  121. return context
  122. @route('^tags/$', name='tag_archive')
  123. @route('^tags/(\w+)/$', name='tag_archive')
  124. def tag_archive(self, request, tag=None):
  125. """
  126. A Custom view that utilizes Tags.
  127. This view will return all related BlogPages for a given Tag or
  128. redirect back to the BlogIndexPage.
  129. """
  130. try:
  131. tag = Tag.objects.get(slug=tag)
  132. except Tag.DoesNotExist:
  133. if tag:
  134. msg = 'There are no blog posts tagged with "{}"'.format(tag)
  135. messages.add_message(request, messages.INFO, msg)
  136. return redirect(self.url)
  137. posts = self.get_posts(tag=tag)
  138. context = {
  139. 'tag': tag,
  140. 'posts': posts
  141. }
  142. return render(request, 'blog/blog_index_page.html', context)
  143. def get_posts(self, tag=None):
  144. """
  145. Return the child BlogPage objects for this BlogPageIndex.
  146. Optional filter by tag.
  147. """
  148. posts = BlogPage.objects.live().descendant_of(self)
  149. if tag:
  150. posts = posts.filter(tags=tag)
  151. return posts
  152. def get_child_tags(self):
  153. """
  154. Returns the list of Tags for all child posts of this BlogPage.
  155. """
  156. tags = []
  157. for post in self.get_posts():
  158. tags += post.get_tags # Not tags.append() because we don't want a list of lists
  159. tags = sorted(set(tags))
  160. return tags
  161. content_panels = Page.content_panels + [
  162. FieldPanel('introduction', classname="full"),
  163. ImageChooserPanel('image'),
  164. ]