models.py 12 KB

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