models.py 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. from django import forms
  2. from django.db import models
  3. from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
  4. from modelcluster.fields import ParentalManyToManyField
  5. from wagtail.admin.edit_handlers import (
  6. FieldPanel, MultiFieldPanel, StreamFieldPanel
  7. )
  8. from wagtail.core.fields import StreamField
  9. from wagtail.core.models import Page
  10. from wagtail.search import index
  11. from wagtail.snippets.models import register_snippet
  12. from wagtail.images.edit_handlers import ImageChooserPanel
  13. from bakerydemo.base.blocks import BaseStreamBlock
  14. @register_snippet
  15. class Country(models.Model):
  16. """
  17. A Django model to store set of countries of origin.
  18. It uses the `@register_snippet` decorator to allow it to be accessible
  19. via the Snippets UI (e.g. /admin/snippets/breads/country/) In the BreadPage
  20. model you'll see we use a ForeignKey to create the relationship between
  21. Country and BreadPage. This allows a single relationship (e.g only one
  22. Country can be added) that is one-way (e.g. Country will have no way to
  23. access related BreadPage objects).
  24. """
  25. title = models.CharField(max_length=100)
  26. def __str__(self):
  27. return self.title
  28. class Meta:
  29. verbose_name_plural = "Countries of Origin"
  30. @register_snippet
  31. class BreadIngredient(models.Model):
  32. """
  33. Standard Django model that is displayed as a snippet within the admin due
  34. to the `@register_snippet` decorator. We use a new piece of functionality
  35. available to Wagtail called the ParentalManyToManyField on the BreadPage
  36. model to display this. The Wagtail Docs give a slightly more detailed example
  37. http://docs.wagtail.io/en/latest/getting_started/tutorial.html#categories
  38. """
  39. name = models.CharField(max_length=255)
  40. panels = [
  41. FieldPanel('name'),
  42. ]
  43. def __str__(self):
  44. return self.name
  45. class Meta:
  46. verbose_name_plural = 'Bread ingredients'
  47. @register_snippet
  48. class BreadType(models.Model):
  49. """
  50. A Django model to define the bread type
  51. It uses the `@register_snippet` decorator to allow it to be accessible
  52. via the Snippets UI. In the BreadPage model you'll see we use a ForeignKey
  53. to create the relationship between BreadType and BreadPage. This allows a
  54. single relationship (e.g only one BreadType can be added) that is one-way
  55. (e.g. BreadType will have no way to access related BreadPage objects)
  56. """
  57. title = models.CharField(max_length=255)
  58. panels = [
  59. FieldPanel('title'),
  60. ]
  61. def __str__(self):
  62. return self.title
  63. class Meta:
  64. verbose_name_plural = "Bread types"
  65. class BreadPage(Page):
  66. """
  67. Detail view for a specific bread
  68. """
  69. introduction = models.TextField(
  70. help_text='Text to describe the page',
  71. blank=True)
  72. image = models.ForeignKey(
  73. 'wagtailimages.Image',
  74. null=True,
  75. blank=True,
  76. on_delete=models.SET_NULL,
  77. related_name='+',
  78. help_text='Landscape mode only; horizontal width between 1000px and 3000px.'
  79. )
  80. body = StreamField(
  81. BaseStreamBlock(), verbose_name="Page body", blank=True
  82. )
  83. origin = models.ForeignKey(
  84. Country,
  85. on_delete=models.SET_NULL,
  86. null=True,
  87. blank=True,
  88. )
  89. # We include related_name='+' to avoid name collisions on relationships.
  90. # e.g. there are two FooPage models in two different apps,
  91. # and they both have a FK to bread_type, they'll both try to create a
  92. # relationship called `foopage_objects` that will throw a valueError on
  93. # collision.
  94. bread_type = models.ForeignKey(
  95. 'breads.BreadType',
  96. null=True,
  97. blank=True,
  98. on_delete=models.SET_NULL,
  99. related_name='+'
  100. )
  101. ingredients = ParentalManyToManyField('BreadIngredient', blank=True)
  102. content_panels = Page.content_panels + [
  103. FieldPanel('introduction', classname="full"),
  104. ImageChooserPanel('image'),
  105. StreamFieldPanel('body'),
  106. FieldPanel('origin'),
  107. FieldPanel('bread_type'),
  108. MultiFieldPanel(
  109. [
  110. FieldPanel(
  111. 'ingredients',
  112. widget=forms.CheckboxSelectMultiple,
  113. ),
  114. ],
  115. heading="Additional Metadata",
  116. classname="collapsible collapsed"
  117. ),
  118. ]
  119. search_fields = Page.search_fields + [
  120. index.SearchField('body'),
  121. ]
  122. parent_page_types = ['BreadsIndexPage']
  123. class BreadsIndexPage(Page):
  124. """
  125. Index page for breads.
  126. This is more complex than other index pages on the bakery demo site as we've
  127. included pagination. We've separated the different aspects of the index page
  128. to be discrete functions to make it easier to follow
  129. """
  130. introduction = models.TextField(
  131. help_text='Text to describe the page',
  132. blank=True)
  133. image = models.ForeignKey(
  134. 'wagtailimages.Image',
  135. null=True,
  136. blank=True,
  137. on_delete=models.SET_NULL,
  138. related_name='+',
  139. help_text='Landscape mode only; horizontal width between 1000px and '
  140. '3000px.'
  141. )
  142. content_panels = Page.content_panels + [
  143. FieldPanel('introduction', classname="full"),
  144. ImageChooserPanel('image'),
  145. ]
  146. # Can only have BreadPage children
  147. subpage_types = ['BreadPage']
  148. # Returns a queryset of BreadPage objects that are live, that are direct
  149. # descendants of this index page with most recent first
  150. def get_breads(self):
  151. return BreadPage.objects.live().descendant_of(
  152. self).order_by('-first_published_at')
  153. # Allows child objects (e.g. BreadPage objects) to be accessible via the
  154. # template. We use this on the HomePage to display child items of featured
  155. # content
  156. def children(self):
  157. return self.get_children().specific().live()
  158. # Pagination for the index page. We use the `django.core.paginator` as any
  159. # standard Django app would, but the difference here being we have it as a
  160. # method on the model rather than within a view function
  161. def paginate(self, request, *args):
  162. page = request.GET.get('page')
  163. paginator = Paginator(self.get_breads(), 12)
  164. try:
  165. pages = paginator.page(page)
  166. except PageNotAnInteger:
  167. pages = paginator.page(1)
  168. except EmptyPage:
  169. pages = paginator.page(paginator.num_pages)
  170. return pages
  171. # Returns the above to the get_context method that is used to populate the
  172. # template
  173. def get_context(self, request):
  174. context = super(BreadsIndexPage, self).get_context(request)
  175. # BreadPage objects (get_breads) are passed through pagination
  176. breads = self.paginate(request, self.get_breads())
  177. context['breads'] = breads
  178. return context