models.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. from __future__ import unicode_literals
  2. from django.db import models
  3. from modelcluster.fields import ParentalKey
  4. from modelcluster.models import ClusterableModel
  5. from wagtail.admin.panels import FieldPanel, FieldRowPanel, InlinePanel, MultiFieldPanel
  6. from wagtail.contrib.forms.models import AbstractEmailForm, AbstractFormField
  7. from wagtail.fields import RichTextField, StreamField
  8. from wagtail.models import Collection, Page
  9. from wagtail.search import index
  10. from wagtail.snippets.models import register_snippet
  11. from .blocks import BaseStreamBlock
  12. @register_snippet
  13. class People(index.Indexed, ClusterableModel):
  14. """
  15. A Django model to store People objects.
  16. It uses the `@register_snippet` decorator to allow it to be accessible
  17. via the Snippets UI (e.g. /admin/snippets/base/people/)
  18. `People` uses the `ClusterableModel`, which allows the relationship with
  19. another model to be stored locally to the 'parent' model (e.g. a PageModel)
  20. until the parent is explicitly saved. This allows the editor to use the
  21. 'Preview' button, to preview the content, without saving the relationships
  22. to the database.
  23. https://github.com/wagtail/django-modelcluster
  24. """
  25. first_name = models.CharField("First name", max_length=254)
  26. last_name = models.CharField("Last name", max_length=254)
  27. job_title = models.CharField("Job title", max_length=254)
  28. image = models.ForeignKey(
  29. "wagtailimages.Image",
  30. null=True,
  31. blank=True,
  32. on_delete=models.SET_NULL,
  33. related_name="+",
  34. )
  35. panels = [
  36. MultiFieldPanel(
  37. [
  38. FieldRowPanel(
  39. [
  40. FieldPanel("first_name", classname="col6"),
  41. FieldPanel("last_name", classname="col6"),
  42. ]
  43. )
  44. ],
  45. "Name",
  46. ),
  47. FieldPanel("job_title"),
  48. FieldPanel("image"),
  49. ]
  50. search_fields = [
  51. index.SearchField("first_name"),
  52. index.SearchField("last_name"),
  53. ]
  54. @property
  55. def thumb_image(self):
  56. # Returns an empty string if there is no profile pic or the rendition
  57. # file can't be found.
  58. try:
  59. return self.image.get_rendition("fill-50x50").img_tag()
  60. except: # noqa: E722 FIXME: remove bare 'except:'
  61. return ""
  62. def __str__(self):
  63. return "{} {}".format(self.first_name, self.last_name)
  64. class Meta:
  65. verbose_name = "Person"
  66. verbose_name_plural = "People"
  67. @register_snippet
  68. class FooterText(models.Model):
  69. """
  70. This provides editable text for the site footer. Again it uses the decorator
  71. `register_snippet` to allow it to be accessible via the admin. It is made
  72. accessible on the template via a template tag defined in base/templatetags/
  73. navigation_tags.py
  74. """
  75. body = RichTextField()
  76. panels = [
  77. FieldPanel("body"),
  78. ]
  79. def __str__(self):
  80. return "Footer text"
  81. class Meta:
  82. verbose_name_plural = "Footer Text"
  83. class StandardPage(Page):
  84. """
  85. A generic content page. On this demo site we use it for an about page but
  86. it could be used for any type of page content that only needs a title,
  87. image, introduction and body field
  88. """
  89. introduction = models.TextField(help_text="Text to describe the page", blank=True)
  90. image = models.ForeignKey(
  91. "wagtailimages.Image",
  92. null=True,
  93. blank=True,
  94. on_delete=models.SET_NULL,
  95. related_name="+",
  96. help_text="Landscape mode only; horizontal width between 1000px and 3000px.",
  97. )
  98. body = StreamField(
  99. BaseStreamBlock(), verbose_name="Page body", blank=True, use_json_field=True
  100. )
  101. content_panels = Page.content_panels + [
  102. FieldPanel("introduction", classname="full"),
  103. FieldPanel("body"),
  104. FieldPanel("image"),
  105. ]
  106. class HomePage(Page):
  107. """
  108. The Home Page. This looks slightly more complicated than it is. You can
  109. see if you visit your site and edit the homepage that it is split between
  110. a:
  111. - Hero area
  112. - Body area
  113. - A promotional area
  114. - Moveable featured site sections
  115. """
  116. # Hero section of HomePage
  117. image = models.ForeignKey(
  118. "wagtailimages.Image",
  119. null=True,
  120. blank=True,
  121. on_delete=models.SET_NULL,
  122. related_name="+",
  123. help_text="Homepage image",
  124. )
  125. hero_text = models.CharField(
  126. max_length=255, help_text="Write an introduction for the bakery"
  127. )
  128. hero_cta = models.CharField(
  129. verbose_name="Hero CTA",
  130. max_length=255,
  131. help_text="Text to display on Call to Action",
  132. )
  133. hero_cta_link = models.ForeignKey(
  134. "wagtailcore.Page",
  135. null=True,
  136. blank=True,
  137. on_delete=models.SET_NULL,
  138. related_name="+",
  139. verbose_name="Hero CTA link",
  140. help_text="Choose a page to link to for the Call to Action",
  141. )
  142. # Body section of the HomePage
  143. body = StreamField(
  144. BaseStreamBlock(),
  145. verbose_name="Home content block",
  146. blank=True,
  147. use_json_field=True,
  148. )
  149. # Promo section of the HomePage
  150. promo_image = models.ForeignKey(
  151. "wagtailimages.Image",
  152. null=True,
  153. blank=True,
  154. on_delete=models.SET_NULL,
  155. related_name="+",
  156. help_text="Promo image",
  157. )
  158. promo_title = models.CharField(
  159. blank=True, max_length=255, help_text="Title to display above the promo copy"
  160. )
  161. promo_text = RichTextField(
  162. null=True, blank=True, help_text="Write some promotional copy"
  163. )
  164. # Featured sections on the HomePage
  165. # You will see on templates/base/home_page.html that these are treated
  166. # in different ways, and displayed in different areas of the page.
  167. # Each list their children items that we access via the children function
  168. # that we define on the individual Page models e.g. BlogIndexPage
  169. featured_section_1_title = models.CharField(
  170. blank=True, max_length=255, help_text="Title to display above the promo copy"
  171. )
  172. featured_section_1 = models.ForeignKey(
  173. "wagtailcore.Page",
  174. null=True,
  175. blank=True,
  176. on_delete=models.SET_NULL,
  177. related_name="+",
  178. help_text="First featured section for the homepage. Will display up to "
  179. "three child items.",
  180. verbose_name="Featured section 1",
  181. )
  182. featured_section_2_title = models.CharField(
  183. blank=True, max_length=255, help_text="Title to display above the promo copy"
  184. )
  185. featured_section_2 = models.ForeignKey(
  186. "wagtailcore.Page",
  187. null=True,
  188. blank=True,
  189. on_delete=models.SET_NULL,
  190. related_name="+",
  191. help_text="Second featured section for the homepage. Will display up to "
  192. "three child items.",
  193. verbose_name="Featured section 2",
  194. )
  195. featured_section_3_title = models.CharField(
  196. blank=True, max_length=255, help_text="Title to display above the promo copy"
  197. )
  198. featured_section_3 = models.ForeignKey(
  199. "wagtailcore.Page",
  200. null=True,
  201. blank=True,
  202. on_delete=models.SET_NULL,
  203. related_name="+",
  204. help_text="Third featured section for the homepage. Will display up to "
  205. "six child items.",
  206. verbose_name="Featured section 3",
  207. )
  208. content_panels = Page.content_panels + [
  209. MultiFieldPanel(
  210. [
  211. FieldPanel("image"),
  212. FieldPanel("hero_text", classname="full"),
  213. MultiFieldPanel(
  214. [
  215. FieldPanel("hero_cta"),
  216. FieldPanel("hero_cta_link"),
  217. ]
  218. ),
  219. ],
  220. heading="Hero section",
  221. ),
  222. MultiFieldPanel(
  223. [
  224. FieldPanel("promo_image"),
  225. FieldPanel("promo_title"),
  226. FieldPanel("promo_text"),
  227. ],
  228. heading="Promo section",
  229. ),
  230. FieldPanel("body"),
  231. MultiFieldPanel(
  232. [
  233. MultiFieldPanel(
  234. [
  235. FieldPanel("featured_section_1_title"),
  236. FieldPanel("featured_section_1"),
  237. ]
  238. ),
  239. MultiFieldPanel(
  240. [
  241. FieldPanel("featured_section_2_title"),
  242. FieldPanel("featured_section_2"),
  243. ]
  244. ),
  245. MultiFieldPanel(
  246. [
  247. FieldPanel("featured_section_3_title"),
  248. FieldPanel("featured_section_3"),
  249. ]
  250. ),
  251. ],
  252. heading="Featured homepage sections",
  253. classname="collapsible",
  254. ),
  255. ]
  256. def __str__(self):
  257. return self.title
  258. class GalleryPage(Page):
  259. """
  260. This is a page to list locations from the selected Collection. We use a Q
  261. object to list any Collection created (/admin/collections/) even if they
  262. contain no items. In this demo we use it for a GalleryPage,
  263. and is intended to show the extensibility of this aspect of Wagtail
  264. """
  265. introduction = models.TextField(help_text="Text to describe the page", blank=True)
  266. image = models.ForeignKey(
  267. "wagtailimages.Image",
  268. null=True,
  269. blank=True,
  270. on_delete=models.SET_NULL,
  271. related_name="+",
  272. help_text="Landscape mode only; horizontal width between 1000px and " "3000px.",
  273. )
  274. body = StreamField(
  275. BaseStreamBlock(), verbose_name="Page body", blank=True, use_json_field=True
  276. )
  277. collection = models.ForeignKey(
  278. Collection,
  279. limit_choices_to=~models.Q(name__in=["Root"]),
  280. null=True,
  281. blank=True,
  282. on_delete=models.SET_NULL,
  283. help_text="Select the image collection for this gallery.",
  284. )
  285. content_panels = Page.content_panels + [
  286. FieldPanel("introduction", classname="full"),
  287. FieldPanel("body"),
  288. FieldPanel("image"),
  289. FieldPanel("collection"),
  290. ]
  291. # Defining what content type can sit under the parent. Since it's a blank
  292. # array no subpage can be added
  293. subpage_types = []
  294. class FormField(AbstractFormField):
  295. """
  296. Wagtailforms is a module to introduce simple forms on a Wagtail site. It
  297. isn't intended as a replacement to Django's form support but as a quick way
  298. to generate a general purpose data-collection form or contact form
  299. without having to write code. We use it on the site for a contact form. You
  300. can read more about Wagtail forms at:
  301. https://docs.wagtail.org/en/stable/reference/contrib/forms/index.html
  302. """
  303. page = ParentalKey("FormPage", related_name="form_fields", on_delete=models.CASCADE)
  304. class FormPage(AbstractEmailForm):
  305. image = models.ForeignKey(
  306. "wagtailimages.Image",
  307. null=True,
  308. blank=True,
  309. on_delete=models.SET_NULL,
  310. related_name="+",
  311. )
  312. body = StreamField(BaseStreamBlock(), use_json_field=True)
  313. thank_you_text = RichTextField(blank=True)
  314. # Note how we include the FormField object via an InlinePanel using the
  315. # related_name value
  316. content_panels = AbstractEmailForm.content_panels + [
  317. FieldPanel("image"),
  318. FieldPanel("body"),
  319. InlinePanel("form_fields", label="Form fields"),
  320. FieldPanel("thank_you_text", classname="full"),
  321. MultiFieldPanel(
  322. [
  323. FieldRowPanel(
  324. [
  325. FieldPanel("from_address", classname="col6"),
  326. FieldPanel("to_address", classname="col6"),
  327. ]
  328. ),
  329. FieldPanel("subject"),
  330. ],
  331. "Email",
  332. ),
  333. ]