snippet_models.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. """
  2. Snippets are for content that is re-usable in nature.
  3. """
  4. from django.db import models
  5. from django.utils.text import slugify
  6. from django.utils.translation import gettext_lazy as _
  7. from modelcluster.fields import ParentalKey
  8. from modelcluster.models import ClusterableModel
  9. from wagtail.admin.panels import (
  10. FieldPanel,
  11. InlinePanel,
  12. MultiFieldPanel,
  13. )
  14. from wagtail.models import Orderable
  15. from wagtail.snippets.models import register_snippet
  16. from wagtail.images import get_image_model_string
  17. from wagtailcrx.blocks import HTML_STREAMBLOCKS, LAYOUT_STREAMBLOCKS, NAVIGATION_STREAMBLOCKS
  18. from wagtailcrx.fields import CoderedStreamField
  19. from wagtailcrx.settings import crx_settings
  20. @register_snippet
  21. class Carousel(ClusterableModel):
  22. """
  23. Model that represents a Carousel. Can be modified through the snippets UI.
  24. Selected through Page StreamField bodies by the CarouselSnippetChooser in
  25. snippet_choosers.py
  26. """
  27. class Meta:
  28. verbose_name = _('Carousel')
  29. name = models.CharField(
  30. max_length=255,
  31. verbose_name=_('Name'),
  32. )
  33. show_controls = models.BooleanField(
  34. default=True,
  35. verbose_name=_('Show controls'),
  36. help_text=_('Shows arrows on the left and right of the carousel to advance next or previous slides.'), # noqa
  37. )
  38. show_indicators = models.BooleanField(
  39. default=True,
  40. verbose_name=_('Show indicators'),
  41. help_text=_('Shows small indicators at the bottom of the carousel based on the number of slides.'), # noqa
  42. )
  43. animation = models.CharField(
  44. blank=True,
  45. max_length=20,
  46. choices=None,
  47. default='',
  48. verbose_name=_('Animation'),
  49. help_text=_('The animation when transitioning between slides.'),
  50. )
  51. panels = (
  52. [
  53. MultiFieldPanel(
  54. heading=_('Slider'),
  55. children=[
  56. FieldPanel('name'),
  57. FieldPanel('show_controls'),
  58. FieldPanel('show_indicators'),
  59. FieldPanel('animation'),
  60. ]
  61. ),
  62. InlinePanel('carousel_slides', label=_('Slides'))
  63. ]
  64. )
  65. def __str__(self):
  66. return self.name
  67. def __init__(self, *args, **kwargs):
  68. """
  69. Inject custom choices and defaults into the form fields
  70. to enable customization of settings without causing migration issues.
  71. """
  72. super().__init__(*args, **kwargs)
  73. # Set choices dynamically.
  74. self._meta.get_field('animation').choices = (
  75. crx_settings.CRX_FRONTEND_CAROUSEL_FX_CHOICES
  76. )
  77. # Set default dynamically.
  78. if not self.id:
  79. self.animation = crx_settings.CRX_FRONTEND_CAROUSEL_FX_DEFAULT
  80. class CarouselSlide(Orderable, models.Model):
  81. """
  82. Represents a slide for the Carousel model. Can be modified through the
  83. snippets UI.
  84. """
  85. class Meta(Orderable.Meta):
  86. verbose_name = _('Carousel Slide')
  87. carousel = ParentalKey(
  88. Carousel,
  89. related_name='carousel_slides',
  90. verbose_name=_('Carousel'),
  91. )
  92. image = models.ForeignKey(
  93. get_image_model_string(),
  94. null=True,
  95. blank=True,
  96. on_delete=models.SET_NULL,
  97. related_name='+',
  98. verbose_name=_('Image'),
  99. )
  100. background_color = models.CharField(
  101. max_length=255,
  102. blank=True,
  103. verbose_name=_('Background color'),
  104. help_text=_('Hexadecimal, rgba, or CSS color notation (e.g. #ff0011)'),
  105. )
  106. custom_css_class = models.CharField(
  107. max_length=255,
  108. blank=True,
  109. verbose_name=_('Custom CSS class'),
  110. )
  111. custom_id = models.CharField(
  112. max_length=255,
  113. blank=True,
  114. verbose_name=_('Custom ID'),
  115. )
  116. content = CoderedStreamField(
  117. HTML_STREAMBLOCKS,
  118. blank=True,
  119. use_json_field=True,
  120. )
  121. panels = (
  122. [
  123. FieldPanel('image'),
  124. FieldPanel('background_color'),
  125. FieldPanel('custom_css_class'),
  126. FieldPanel('custom_id'),
  127. FieldPanel('content'),
  128. ]
  129. )
  130. @register_snippet
  131. class Classifier(ClusterableModel):
  132. """
  133. Simple and generic model to organize/categorize/group pages.
  134. """
  135. class Meta:
  136. verbose_name = _('Classifier')
  137. verbose_name_plural = _('Classifiers')
  138. ordering = ['name']
  139. slug = models.SlugField(
  140. allow_unicode=True,
  141. unique=True,
  142. verbose_name=_('Slug'),
  143. )
  144. name = models.CharField(
  145. max_length=255,
  146. verbose_name=_('Name'),
  147. )
  148. panels = [
  149. FieldPanel('name'),
  150. InlinePanel('terms', label=_('Classifier Terms'))
  151. ]
  152. def save(self, *args, **kwargs):
  153. if not self.slug:
  154. # Make a slug and suffix a number if it already exists to ensure uniqueness
  155. newslug = slugify(self.name, allow_unicode=True)
  156. tmpslug = newslug
  157. suffix = 1
  158. while True:
  159. if not Classifier.objects.filter(slug=tmpslug).exists():
  160. self.slug = tmpslug
  161. break
  162. tmpslug = newslug + "-" + str(suffix)
  163. suffix += 1
  164. return super().save(*args, **kwargs)
  165. def __str__(self):
  166. return self.name
  167. class ClassifierTerm(Orderable, models.Model):
  168. """
  169. Term used to categorize a page.
  170. """
  171. class Meta(Orderable.Meta):
  172. verbose_name = _('Classifier Term')
  173. verbose_name_plural = _('Classifier Terms')
  174. classifier = ParentalKey(
  175. Classifier,
  176. related_name='terms',
  177. verbose_name=_('Classifier'),
  178. )
  179. slug = models.SlugField(
  180. allow_unicode=True,
  181. unique=True,
  182. verbose_name=_('Slug'),
  183. )
  184. name = models.CharField(
  185. max_length=255,
  186. verbose_name=_('Name'),
  187. )
  188. panels = [
  189. FieldPanel('name'),
  190. ]
  191. def save(self, *args, **kwargs):
  192. if not self.slug:
  193. # Make a slug and suffix a number if it already exists to ensure uniqueness
  194. newslug = slugify(self.name, allow_unicode=True)
  195. tmpslug = newslug
  196. suffix = 1
  197. while True:
  198. if not ClassifierTerm.objects.filter(slug=tmpslug).exists():
  199. self.slug = tmpslug
  200. break
  201. tmpslug = newslug + "-" + str(suffix)
  202. suffix += 1
  203. return super().save(*args, **kwargs)
  204. def __str__(self):
  205. return "{0} > {1}".format(self.classifier.name, self.name)
  206. @register_snippet
  207. class Navbar(models.Model):
  208. """
  209. Snippet for site navigation bars (header, main menu, etc.)
  210. """
  211. class Meta:
  212. verbose_name = _('Navigation Bar')
  213. name = models.CharField(
  214. max_length=255,
  215. verbose_name=_('Name'),
  216. )
  217. custom_css_class = models.CharField(
  218. max_length=255,
  219. blank=True,
  220. verbose_name=_('Custom CSS Class'),
  221. )
  222. custom_id = models.CharField(
  223. max_length=255,
  224. blank=True,
  225. verbose_name=_('Custom ID'),
  226. )
  227. menu_items = CoderedStreamField(
  228. NAVIGATION_STREAMBLOCKS,
  229. verbose_name=_('Navigation links'),
  230. blank=True,
  231. use_json_field=True,
  232. )
  233. panels = [
  234. FieldPanel('name'),
  235. MultiFieldPanel(
  236. [
  237. FieldPanel('custom_css_class'),
  238. FieldPanel('custom_id'),
  239. ],
  240. heading=_('Attributes')
  241. ),
  242. FieldPanel('menu_items')
  243. ]
  244. def __str__(self):
  245. return self.name
  246. @register_snippet
  247. class Footer(models.Model):
  248. """
  249. Snippet for website footer content.
  250. """
  251. class Meta:
  252. verbose_name = _('Footer')
  253. name = models.CharField(
  254. max_length=255,
  255. verbose_name=_('Name'),
  256. )
  257. custom_css_class = models.CharField(
  258. max_length=255,
  259. blank=True,
  260. verbose_name=_('Custom CSS Class'),
  261. )
  262. custom_id = models.CharField(
  263. max_length=255,
  264. blank=True,
  265. verbose_name=_('Custom ID'),
  266. )
  267. content = CoderedStreamField(
  268. LAYOUT_STREAMBLOCKS,
  269. verbose_name=_('Content'),
  270. blank=True,
  271. use_json_field=True,
  272. )
  273. panels = [
  274. FieldPanel('name'),
  275. MultiFieldPanel(
  276. [
  277. FieldPanel('custom_css_class'),
  278. FieldPanel('custom_id'),
  279. ],
  280. heading=_('Attributes')
  281. ),
  282. FieldPanel('content')
  283. ]
  284. def __str__(self):
  285. return self.name
  286. @register_snippet
  287. class ReusableContent(models.Model):
  288. """
  289. Snippet for resusable content in streamfields.
  290. """
  291. class Meta:
  292. verbose_name = _('Reusable Content')
  293. verbose_name_plural = _('Reusable Content')
  294. name = models.CharField(
  295. max_length=255,
  296. verbose_name=_('Name'),
  297. )
  298. content = CoderedStreamField(
  299. LAYOUT_STREAMBLOCKS,
  300. verbose_name=_('content'),
  301. blank=True,
  302. use_json_field=True,
  303. )
  304. panels = [
  305. FieldPanel('name'),
  306. FieldPanel('content')
  307. ]
  308. def __str__(self):
  309. return self.name
  310. @register_snippet
  311. class ContentWall(models.Model):
  312. """
  313. Snippet that restricts access to a page with a modal.
  314. """
  315. class Meta:
  316. verbose_name = _('Content Wall')
  317. name = models.CharField(
  318. max_length=255,
  319. verbose_name=_('Name'),
  320. )
  321. content = CoderedStreamField(
  322. LAYOUT_STREAMBLOCKS,
  323. verbose_name=_('Content'),
  324. blank=True,
  325. use_json_field=True,
  326. )
  327. is_dismissible = models.BooleanField(
  328. default=True,
  329. verbose_name=_('Dismissible'),
  330. )
  331. show_once = models.BooleanField(
  332. default=True,
  333. verbose_name=_('Show once'),
  334. help_text=_('Do not show the content wall to the same user again after it has been closed.')
  335. )
  336. panels = [
  337. MultiFieldPanel(
  338. [
  339. FieldPanel('name'),
  340. FieldPanel('is_dismissible'),
  341. FieldPanel('show_once'),
  342. ],
  343. heading=_('Content Wall')
  344. ),
  345. FieldPanel('content'),
  346. ]
  347. def __str__(self):
  348. return self.name
  349. class CoderedEmail(ClusterableModel):
  350. """
  351. General purpose abstract clusterable model used for holding email information.
  352. Most likely this should be subclassed with addition of a ParentalKey.
  353. """
  354. class Meta:
  355. abstract = True
  356. verbose_name = _('CodeRed Email')
  357. to_address = models.CharField(
  358. max_length=255,
  359. blank=True,
  360. verbose_name=_('To Addresses'),
  361. help_text=_('Separate multiple email addresses with commas.')
  362. )
  363. from_address = models.CharField(
  364. max_length=255,
  365. blank=True,
  366. verbose_name=_('From Address'),
  367. help_text=_('For example: "sender@example.com" or "Sender Name <sender@example.com>" (without quotes).') # noqa
  368. )
  369. reply_address = models.CharField(
  370. max_length=255,
  371. blank=True,
  372. verbose_name=_('Reply-To Address'),
  373. help_text=_('Separate multiple email addresses with commas.')
  374. )
  375. cc_address = models.CharField(
  376. max_length=255,
  377. blank=True,
  378. verbose_name=_('CC'),
  379. help_text=_('Separate multiple email addresses with commas.')
  380. )
  381. bcc_address = models.CharField(
  382. max_length=255,
  383. blank=True,
  384. verbose_name=_('BCC'),
  385. help_text=_('Separate multiple email addresses with commas.')
  386. )
  387. subject = models.CharField(max_length=255, blank=True, verbose_name=_('Subject'))
  388. body = models.TextField(blank=True, verbose_name=_('Body'))
  389. panels = (
  390. [
  391. MultiFieldPanel(
  392. [
  393. FieldPanel('to_address'),
  394. FieldPanel('from_address'),
  395. FieldPanel('cc_address'),
  396. FieldPanel('bcc_address'),
  397. FieldPanel('subject'),
  398. FieldPanel('body'),
  399. ],
  400. _('Email Message')
  401. ),
  402. ])
  403. def __str__(self):
  404. return self.subject