page_models.py 69 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125
  1. """
  2. Base and abstract pages used in Wagtail CRX.
  3. """
  4. import json
  5. import logging
  6. import os
  7. import warnings
  8. from datetime import date
  9. from datetime import datetime
  10. from datetime import timezone as dttimezone
  11. from pathlib import Path
  12. from typing import TYPE_CHECKING
  13. from typing import Dict
  14. from typing import List
  15. from typing import Optional
  16. from typing import Tuple
  17. from typing import Union
  18. import geocoder
  19. from django import forms
  20. from django.conf import settings
  21. from django.contrib import messages
  22. from django.core.files.storage import FileSystemStorage
  23. from django.core.files.uploadedfile import InMemoryUploadedFile
  24. from django.core.files.uploadedfile import TemporaryUploadedFile
  25. from django.core.mail import EmailMessage
  26. from django.core.paginator import EmptyPage
  27. from django.core.paginator import InvalidPage
  28. from django.core.paginator import PageNotAnInteger
  29. from django.core.paginator import Paginator
  30. from django.core.serializers.json import DjangoJSONEncoder
  31. from django.core.validators import MaxValueValidator
  32. from django.core.validators import MinValueValidator
  33. from django.db import models
  34. from django.http import HttpResponseRedirect
  35. from django.http import JsonResponse
  36. from django.shortcuts import redirect
  37. from django.shortcuts import render
  38. from django.template import Context
  39. from django.template import Template
  40. from django.template.loader import render_to_string
  41. from django.utils import timezone
  42. from django.utils.html import strip_tags
  43. from django.utils.safestring import mark_safe
  44. from django.utils.translation import gettext_lazy as _
  45. from eventtools.models import BaseEvent
  46. from eventtools.models import BaseOccurrence
  47. from icalendar import Alarm
  48. from icalendar import Event as ICalEvent
  49. from modelcluster.contrib.taggit import ClusterTaggableManager
  50. from modelcluster.fields import ParentalKey
  51. from modelcluster.fields import ParentalManyToManyField
  52. from taggit.models import TaggedItemBase
  53. from wagtail import hooks
  54. from wagtail.admin.panels import FieldPanel
  55. from wagtail.admin.panels import FieldRowPanel
  56. from wagtail.admin.panels import InlinePanel
  57. from wagtail.admin.panels import MultiFieldPanel
  58. from wagtail.admin.panels import ObjectList
  59. from wagtail.admin.panels import TabbedInterface
  60. from wagtail.contrib.forms.forms import WagtailAdminFormPageForm
  61. from wagtail.contrib.forms.models import FormSubmission
  62. from wagtail.contrib.forms.panels import FormSubmissionsPanel
  63. from wagtail.coreutils import resolve_model_string
  64. from wagtail.fields import StreamField
  65. from wagtail.images import get_image_model_string
  66. from wagtail.models import Orderable
  67. from wagtail.models import Page
  68. from wagtail.models import PageBase
  69. from wagtail.models import Site
  70. from wagtail.search import index
  71. from wagtail.utils.decorators import cached_classmethod
  72. from wagtail_flexible_forms.models import AbstractSessionFormSubmission
  73. from wagtail_flexible_forms.models import AbstractSubmissionRevision
  74. from wagtail_flexible_forms.models import StreamFormJSONEncoder
  75. from wagtail_flexible_forms.models import StreamFormMixin
  76. from wagtailcache.cache import WagtailCacheMixin
  77. from wagtailseo.models import SeoMixin
  78. from wagtailseo.models import TwitterCard
  79. from wagtailseo.utils import StructDataEncoder
  80. from wagtailseo.utils import get_struct_data_images
  81. from coderedcms import utils
  82. from coderedcms.blocks import CONTENT_STREAMBLOCKS
  83. from coderedcms.blocks import LAYOUT_STREAMBLOCKS
  84. from coderedcms.blocks import STREAMFORM_BLOCKS
  85. from coderedcms.blocks import ContentWallBlock
  86. from coderedcms.fields import CoderedStreamField
  87. from coderedcms.fields import ColorField
  88. from coderedcms.forms import CoderedFormBuilder
  89. from coderedcms.forms import CoderedSubmissionsListView
  90. from coderedcms.models.snippet_models import ClassifierTerm
  91. from coderedcms.models.wagtailsettings_models import LayoutSettings
  92. from coderedcms.recaptcha import verify_response
  93. from coderedcms.settings import crx_settings
  94. from coderedcms.widgets import ClassifierSelectWidget
  95. if TYPE_CHECKING:
  96. from wagtail.images.models import AbstractImage
  97. logger = logging.getLogger("coderedcms")
  98. CRX_PAGE_MODELS = []
  99. def get_page_models():
  100. return CRX_PAGE_MODELS
  101. class CoderedPageMeta(PageBase):
  102. def __init__(cls, name, bases, dct):
  103. super().__init__(name, bases, dct)
  104. if not cls._meta.abstract:
  105. CRX_PAGE_MODELS.append(cls)
  106. class CoderedTag(TaggedItemBase):
  107. class Meta:
  108. verbose_name = _("CodeRed Tag")
  109. content_object = ParentalKey(
  110. "coderedcms.CoderedPage", related_name="tagged_items"
  111. )
  112. class CoderedPage(WagtailCacheMixin, SeoMixin, Page, metaclass=CoderedPageMeta):
  113. """
  114. General use page with caching, templating, and SEO functionality.
  115. All pages should inherit from this.
  116. """
  117. class Meta:
  118. verbose_name = _("CodeRed Page")
  119. # Do not allow this page type to be created in wagtail admin
  120. is_creatable = False
  121. # Templates
  122. # The page will render the following templates under certain conditions:
  123. #
  124. # template = ''
  125. # ajax_template = ''
  126. # search_template = ''
  127. # Template used in site search results.
  128. search_template = "coderedcms/pages/search_result.html"
  129. # Template used for related pages, Latest Pages block, and Page Preview block.
  130. miniview_template = "coderedcms/pages/page.mini.html"
  131. ###############
  132. # Content fields
  133. ###############
  134. cover_image = models.ForeignKey(
  135. get_image_model_string(),
  136. null=True,
  137. blank=True,
  138. on_delete=models.SET_NULL,
  139. related_name="+",
  140. verbose_name=_("Cover image"),
  141. )
  142. ###############
  143. # Index fields
  144. ###############
  145. # Subclasses can override this to enabled index features by default.
  146. index_show_subpages_default = False
  147. # Subclasses can override this to query on a specific
  148. # page model, rather than the default wagtail Page.
  149. index_query_pagemodel = "coderedcms.CoderedPage"
  150. # Subclasses can override these fields to enable custom
  151. # ordering based on specific subpage fields.
  152. index_order_by_default = ""
  153. index_order_by_choices = (
  154. ("", _("Default Ordering")),
  155. ("-first_published_at", _("Date first published, newest to oldest")),
  156. ("first_published_at", _("Date first published, oldest to newest")),
  157. ("-last_published_at", _("Date updated, newest to oldest")),
  158. ("last_published_at", _("Date updated, oldest to newest")),
  159. ("title", _("Title, alphabetical")),
  160. ("-title", _("Title, reverse alphabetical")),
  161. )
  162. index_show_subpages = models.BooleanField(
  163. default=index_show_subpages_default,
  164. verbose_name=_("Show list of child pages"),
  165. )
  166. index_order_by_classifier = models.ForeignKey(
  167. "coderedcms.Classifier",
  168. blank=True,
  169. null=True,
  170. on_delete=models.SET_NULL,
  171. related_name="+",
  172. verbose_name=_("Order child pages by classifier"),
  173. help_text=_(
  174. "Child pages will first be sorted following the order of this "
  175. "classifier’s terms (from Snippets > Classifiers)."
  176. ),
  177. )
  178. index_order_by = models.CharField(
  179. max_length=255,
  180. choices=index_order_by_choices,
  181. default=index_order_by_default,
  182. blank=True,
  183. verbose_name=_("Order child pages by"),
  184. help_text=_("Child pages will then be sorted by this attribute."),
  185. )
  186. index_num_per_page = models.PositiveIntegerField(
  187. default=10,
  188. verbose_name=_("Number per page"),
  189. )
  190. index_classifiers = ParentalManyToManyField(
  191. "coderedcms.Classifier",
  192. blank=True,
  193. verbose_name=_("Filter child pages by"),
  194. help_text=_("Enable filtering child pages by these classifiers."),
  195. )
  196. #####################
  197. # Related Page Fields
  198. #####################
  199. # Subclasses can override this to query on a specific page model, in the
  200. # format "appname.Model". By default sibling pages are used.
  201. related_query_pagemodel = None
  202. # Subclasses can override this to enabled related pages by default.
  203. related_show_default = False
  204. related_show = models.BooleanField(
  205. default=related_show_default,
  206. verbose_name=_("Show list of related pages"),
  207. )
  208. related_num = models.PositiveIntegerField(
  209. default=3,
  210. verbose_name=_("Number of related pages to show"),
  211. )
  212. related_classifier_term = models.ForeignKey(
  213. "coderedcms.ClassifierTerm",
  214. blank=True,
  215. null=True,
  216. on_delete=models.SET_NULL,
  217. related_name="+",
  218. verbose_name=_("Preferred related classifier term"),
  219. help_text=_(
  220. "When getting related pages, pages with this term will be "
  221. "weighted over other classifier terms. By default, pages with "
  222. "the greatest number of classifiers in common are ranked highest."
  223. ),
  224. )
  225. ###############
  226. # Layout fields
  227. ###############
  228. custom_template = models.CharField(
  229. blank=True, max_length=255, choices=None, verbose_name=_("Template")
  230. )
  231. ###############
  232. # SEO overrides
  233. ###############
  234. seo_image_sources = [
  235. "og_image",
  236. "cover_image",
  237. ]
  238. ###############
  239. # Classify
  240. ###############
  241. classifier_terms = ParentalManyToManyField(
  242. "coderedcms.ClassifierTerm",
  243. blank=True,
  244. verbose_name=_("Classifiers"),
  245. help_text=_(
  246. "Categorize and group pages together with classifiers. "
  247. "Used to organize and filter pages across the site."
  248. ),
  249. )
  250. tags = ClusterTaggableManager(
  251. through=CoderedTag,
  252. blank=True,
  253. verbose_name=_("Tags"),
  254. help_text=_("Used to organize pages across the site."),
  255. )
  256. ###############
  257. # Settings
  258. ###############
  259. content_walls = CoderedStreamField(
  260. [("content_wall", ContentWallBlock())],
  261. blank=True,
  262. use_json_field=True,
  263. verbose_name=_("Content Walls"),
  264. )
  265. ###############
  266. # Search
  267. ###############
  268. search_fields = Page.search_fields + [
  269. index.SearchField("seo_title", boost=2),
  270. index.SearchField("search_description", boost=2),
  271. index.FilterField("index_show_subpages"),
  272. index.FilterField("index_order_by"),
  273. index.FilterField("custom_template"),
  274. index.FilterField("classifier_terms"),
  275. ]
  276. ###############
  277. # Panels
  278. ###############
  279. content_panels = Page.content_panels + [
  280. FieldPanel("cover_image"),
  281. ]
  282. body_content_panels = []
  283. bottom_content_panels = []
  284. classify_panels = [
  285. FieldPanel("classifier_terms", widget=ClassifierSelectWidget()),
  286. FieldPanel("tags"),
  287. ]
  288. layout_panels = [
  289. MultiFieldPanel(
  290. [FieldPanel("custom_template")], heading=_("Visual Design")
  291. ),
  292. MultiFieldPanel(
  293. [
  294. FieldPanel("index_show_subpages"),
  295. FieldPanel("index_num_per_page"),
  296. FieldPanel("index_order_by_classifier"),
  297. FieldPanel("index_order_by"),
  298. FieldPanel(
  299. "index_classifiers", widget=forms.CheckboxSelectMultiple()
  300. ),
  301. ],
  302. heading=_("Show Child Pages"),
  303. ),
  304. MultiFieldPanel(
  305. [
  306. FieldPanel("related_show"),
  307. FieldPanel("related_num"),
  308. FieldPanel("related_classifier_term"),
  309. ],
  310. heading=_("Related Pages"),
  311. ),
  312. ]
  313. promote_panels = SeoMixin.seo_meta_panels
  314. settings_panels = Page.settings_panels + [
  315. FieldPanel("content_walls"),
  316. ]
  317. integration_panels = []
  318. def __init__(self, *args, **kwargs):
  319. """
  320. Inject custom choices and defaults into the form fields
  321. to enable customization by subclasses.
  322. """
  323. super().__init__(*args, **kwargs)
  324. klassname = self.__class__.__name__.lower()
  325. template_choices = crx_settings.CRX_FRONTEND_TEMPLATES_PAGES.get(
  326. "*", []
  327. ) + crx_settings.CRX_FRONTEND_TEMPLATES_PAGES.get(klassname, [])
  328. self._meta.get_field(
  329. "index_order_by"
  330. ).choices = self.index_order_by_choices
  331. self._meta.get_field("custom_template").choices = template_choices
  332. if not self.id:
  333. self.index_order_by = self.index_order_by_default
  334. self.index_show_subpages = self.index_show_subpages_default
  335. self.related_show = self.related_show_default
  336. @cached_classmethod
  337. def get_edit_handler(cls):
  338. """
  339. Override to "lazy load" the panels overridden by subclasses.
  340. """
  341. panels = [
  342. ObjectList(
  343. cls.content_panels
  344. + cls.body_content_panels
  345. + cls.bottom_content_panels,
  346. heading=_("Content"),
  347. ),
  348. ObjectList(cls.classify_panels, heading=_("Classify")),
  349. ObjectList(cls.layout_panels, heading=_("Layout")),
  350. ObjectList(cls.promote_panels, heading=_("SEO"), classname="seo"),
  351. ObjectList(
  352. cls.settings_panels, heading=_("Settings"), classname="settings"
  353. ),
  354. ]
  355. if cls.integration_panels:
  356. panels.append(
  357. ObjectList(
  358. cls.integration_panels,
  359. heading="Integrations",
  360. classname="integrations",
  361. )
  362. )
  363. edit_handler = TabbedInterface(panels)
  364. return edit_handler.bind_to_model(cls)
  365. @property
  366. def seo_logo(self) -> "Optional[AbstractImage]":
  367. """
  368. Override method in SeoMixin.
  369. Gets the primary logo of the organization.
  370. """
  371. logo = super().seo_logo
  372. if logo:
  373. return logo
  374. else:
  375. layout_settings = LayoutSettings.for_site(self.get_site())
  376. if layout_settings.logo:
  377. return layout_settings.logo
  378. return None
  379. @property
  380. def seo_image(self) -> "Optional[AbstractImage]":
  381. """
  382. Override method in SeoMixin.
  383. Fallback to logo if opengraph image is not specified.
  384. """
  385. img = super().seo_image
  386. if img is None:
  387. return self.seo_logo
  388. return img
  389. @property
  390. def seo_twitter_card_content(self) -> str:
  391. """
  392. Override of method in SeoMixin.
  393. Show a large twitter card if the page has an image set.
  394. """
  395. if self.seo_image:
  396. return TwitterCard.LARGE.value
  397. return self.seo_twitter_card.value
  398. def get_template(self, request, *args, **kwargs):
  399. """
  400. Override parent to serve different templates based on querystring.
  401. """
  402. if self.custom_template:
  403. return self.custom_template
  404. return super().get_template(request, args, kwargs)
  405. def get_index_children(self):
  406. """
  407. Returns query of subpages as defined by `index_` variables.
  408. """
  409. if self.index_query_pagemodel:
  410. querymodel = resolve_model_string(
  411. self.index_query_pagemodel, self._meta.app_label
  412. )
  413. query = querymodel.objects.child_of(self).live()
  414. else:
  415. query = self.get_children().live()
  416. # Determine query sorting order.
  417. order = []
  418. # To sort by term order of a specific classifier, annotate the child
  419. # pages with the `sort_order` of its ClassifierTerms.
  420. if self.index_order_by_classifier:
  421. terms = ClassifierTerm.objects.filter(
  422. classifier=self.index_order_by_classifier,
  423. # Reverse ManyToMany of `coderedpage.classifier_terms`.
  424. coderedpage=models.OuterRef("pk"),
  425. )[:1]
  426. query = query.annotate(
  427. term_sort_order=models.Subquery(terms.values("sort_order"))
  428. )
  429. order.append("term_sort_order")
  430. # Second, order by the specified model attribute.
  431. if self.index_order_by:
  432. order.append(self.index_order_by)
  433. # Order the query.
  434. if order:
  435. query = query.order_by(*order)
  436. return query
  437. def get_related_pages(
  438. self, pagetype: str = None, num: int = None
  439. ) -> models.QuerySet:
  440. """
  441. Returns a queryset of sibling pages, or the model type
  442. defined by ``pagetype`` or ``self.related_query_pagemodel``,
  443. ordered by number of shared classifier terms.
  444. :param str pagetype: The model type to query on. This should
  445. be a string in the format "appname.Model".
  446. Overrides ``self.related_query_pagemodel``
  447. :param int num: The number of results to return.
  448. Overrides ``self.related_num``.
  449. """
  450. if pagetype is None:
  451. pagetype = self.related_query_pagemodel
  452. if num is None:
  453. num = self.related_num
  454. # Get our related query model, and queryset.
  455. if pagetype:
  456. querymodel = resolve_model_string(pagetype, self._meta.app_label)
  457. r_qs = querymodel.objects.all().live()
  458. else:
  459. r_qs = self.get_parent().specific.get_index_children()
  460. # Exclude self to avoid infinite recursion.
  461. r_qs = r_qs.exclude(pk=self.pk)
  462. order_by = []
  463. # If we have a preferred classifier term, order by that.
  464. if self.related_classifier_term:
  465. p_ct_q = models.Q(classifier_terms=self.related_classifier_term)
  466. r_qs = r_qs.annotate(p_ct=p_ct_q)
  467. order_by.append("-p_ct")
  468. # If this page has a classifier, then order by number of
  469. # shared classifier terms.
  470. if self.classifier_terms.exists():
  471. r_ct_q = models.Q(classifier_terms__in=self.classifier_terms.all())
  472. r_qs = r_qs.annotate(r_ct=models.Count("classifier_terms", r_ct_q))
  473. order_by.append("-r_ct")
  474. # Order the related pages, then add distinct to deal with
  475. # annotating based on a many to many.
  476. if order_by:
  477. r_qs = r_qs.order_by(*order_by).distinct()
  478. return r_qs[:num]
  479. def get_content_walls(self, check_child_setting=True):
  480. current_content_walls = []
  481. if check_child_setting:
  482. for wall in self.content_walls:
  483. if wall.value["show_content_wall_on_children"]:
  484. current_content_walls.append(wall.value)
  485. else:
  486. current_content_walls = self.content_walls
  487. try:
  488. return (
  489. list(current_content_walls)
  490. + self.get_parent().specific.get_content_walls()
  491. )
  492. except AttributeError:
  493. return list(current_content_walls)
  494. def get_context(self, request, *args, **kwargs):
  495. """
  496. Add child pages and paginated child pages to context.
  497. """
  498. context = super().get_context(request)
  499. # Show list of child pages.
  500. if self.index_show_subpages:
  501. # Get child pages
  502. all_children = self.get_index_children()
  503. # Filter by classifier terms if applicable
  504. if len(request.GET) > 0 and self.index_classifiers.exists():
  505. # Look up comma separated ClassifierTerm slugs
  506. # i.e. `/?c=term1-slug,term2-slug`
  507. terms = []
  508. get_c = request.GET.get("c", None)
  509. if get_c:
  510. terms = get_c.split(",")
  511. # Else look up individual querystrings
  512. # i.e. `/?classifier-slug=term1-slug`
  513. else:
  514. for classifier in self.index_classifiers.all().only("slug"):
  515. get_term = request.GET.get(classifier.slug, None)
  516. if get_term:
  517. terms.append(get_term)
  518. if len(terms) > 0:
  519. selected_terms = ClassifierTerm.objects.filter(
  520. slug__in=terms
  521. )
  522. context["selected_terms"] = selected_terms
  523. if len(selected_terms) > 0:
  524. try:
  525. for term in selected_terms:
  526. all_children = all_children.filter(
  527. classifier_terms=term
  528. )
  529. except AttributeError:
  530. logger.warning(
  531. (
  532. "Tried to filter by ClassifierTerm, "
  533. "but <%s.%s ('%s')>.get_index_children() "
  534. "did not return a queryset or is not a "
  535. "queryset of CoderedPage models."
  536. ),
  537. self._meta.app_label,
  538. self.__class__.__name__,
  539. self.title,
  540. )
  541. paginator = Paginator(all_children, self.index_num_per_page)
  542. pagenum = request.GET.get("p", 1)
  543. try:
  544. paged_children = paginator.page(pagenum)
  545. except (PageNotAnInteger, EmptyPage, InvalidPage):
  546. paged_children = paginator.page(1)
  547. context["index_paginated"] = paged_children
  548. context["index_children"] = all_children
  549. # Show a list of related pages.
  550. if self.related_show:
  551. context["related_pages"] = self.get_related_pages()
  552. # Content walls.
  553. context["content_walls"] = self.get_content_walls(
  554. check_child_setting=False
  555. )
  556. return context
  557. ###############################################################################
  558. # Abstract pages providing pre-built common website functionality, suitable for subclassing.
  559. # These are abstract so subclasses can override fields if desired.
  560. ###############################################################################
  561. class CoderedWebPage(CoderedPage):
  562. """
  563. Provides a body and body-related functionality.
  564. This is abstract so that subclasses can override the body StreamField.
  565. """
  566. class Meta:
  567. verbose_name = _("CodeRed Web Page")
  568. abstract = True
  569. template = "coderedcms/pages/web_page.html"
  570. # Child pages should override based on what blocks they want in the body.
  571. # Default is LAYOUT_STREAMBLOCKS which is the fullest editor experience.
  572. body = StreamField(
  573. LAYOUT_STREAMBLOCKS,
  574. null=True,
  575. blank=True,
  576. use_json_field=True,
  577. )
  578. # Search fields
  579. search_fields = CoderedPage.search_fields + [index.SearchField("body")]
  580. # Panels
  581. body_content_panels = [
  582. FieldPanel("body"),
  583. ]
  584. @property
  585. def body_preview(self):
  586. """
  587. A shortened version of the body without HTML tags.
  588. """
  589. # add spaces between tags for legibility
  590. body = str(self.body).replace(">", "> ")
  591. # strip tags
  592. body = strip_tags(body)
  593. # truncate and add ellipses
  594. preview = body[:200] + "..." if len(body) > 200 else body
  595. return mark_safe(preview)
  596. class CoderedArticlePage(CoderedWebPage):
  597. """
  598. Article, suitable for news or blog content.
  599. """
  600. class Meta:
  601. verbose_name = _("CodeRed Article")
  602. abstract = True
  603. template = "coderedcms/pages/article_page.html"
  604. miniview_template = "coderedcms/pages/article_page.mini.html"
  605. search_template = "coderedcms/pages/article_page.search.html"
  606. related_show_default = True
  607. # Override body to provide simpler content
  608. body = StreamField(
  609. CONTENT_STREAMBLOCKS,
  610. null=True,
  611. blank=True,
  612. use_json_field=True,
  613. )
  614. caption = models.CharField(
  615. max_length=255,
  616. blank=True,
  617. verbose_name=_("Caption"),
  618. )
  619. author = models.ForeignKey(
  620. settings.AUTH_USER_MODEL,
  621. null=True,
  622. blank=True,
  623. editable=True,
  624. on_delete=models.SET_NULL,
  625. verbose_name=_("Author"),
  626. )
  627. author_display = models.CharField(
  628. max_length=255,
  629. blank=True,
  630. verbose_name=_("Display author as"),
  631. help_text=_("Override how the author’s name displays on this article."),
  632. )
  633. date_display = models.DateField(
  634. null=True,
  635. blank=True,
  636. verbose_name=_("Display publish date"),
  637. )
  638. def get_author_name(self):
  639. warnings.warn(
  640. (
  641. "CoderedArticlePage.get_author_name has been replaced with "
  642. "CoderedArticlePage.seo_author"
  643. ),
  644. DeprecationWarning,
  645. )
  646. return self.seo_author
  647. @property
  648. def seo_author(self) -> str:
  649. """
  650. Override of method in SeoMixin.
  651. Gets author name using a fallback.
  652. """
  653. if self.author_display:
  654. return self.author_display
  655. if self.author:
  656. return self.author.get_full_name()
  657. if self.owner:
  658. return self.owner.get_full_name()
  659. return ""
  660. def get_pub_date(self):
  661. warnings.warn(
  662. (
  663. "CoderedArticlePage.get_pub_date has been replaced with "
  664. "CoderedArticlePage.seo_published_at"
  665. ),
  666. DeprecationWarning,
  667. )
  668. return self.seo_published_at
  669. @property
  670. def seo_published_at(self) -> datetime:
  671. """
  672. Override of method in SeoMixin.
  673. Gets published date.
  674. """
  675. if self.date_display:
  676. return self.date_display
  677. return self.first_published_at
  678. def get_description(self):
  679. warnings.warn(
  680. (
  681. "CoderedArticlePage.get_description has been replaced with "
  682. "CoderedArticlePage.seo_description"
  683. ),
  684. DeprecationWarning,
  685. )
  686. return self.seo_description
  687. @property
  688. def seo_description(self) -> str:
  689. """
  690. Override of method in SeoMixin.
  691. Gets the description using a fallback.
  692. """
  693. if self.search_description:
  694. return self.search_description
  695. if self.caption:
  696. return self.caption
  697. if self.body_preview:
  698. return self.body_preview
  699. return ""
  700. search_fields = CoderedWebPage.search_fields + [
  701. index.SearchField("caption", boost=2),
  702. index.FilterField("author"),
  703. index.FilterField("author_display"),
  704. index.FilterField("date_display"),
  705. ]
  706. content_panels = CoderedWebPage.content_panels + [
  707. FieldPanel("caption"),
  708. MultiFieldPanel(
  709. [
  710. FieldPanel("author"),
  711. FieldPanel("author_display"),
  712. FieldPanel("date_display"),
  713. ],
  714. _("Publication Info"),
  715. ),
  716. ]
  717. class CoderedArticleIndexPage(CoderedWebPage):
  718. """
  719. Shows a list of article sub-pages.
  720. """
  721. class Meta:
  722. verbose_name = _("CodeRed Article Index Page")
  723. abstract = True
  724. template = "coderedcms/pages/article_index_page.html"
  725. index_show_subpages_default = True
  726. index_order_by_default = "-date_display"
  727. index_order_by_choices = (
  728. ("-date_display", "Display publish date, newest first"),
  729. ) + CoderedWebPage.index_order_by_choices
  730. class CoderedEventPage(CoderedWebPage, BaseEvent):
  731. class Meta:
  732. verbose_name = _("CodeRed Event")
  733. abstract = True
  734. template = "coderedcms/pages/event_page.html"
  735. miniview_template = "coderedcms/pages/event_page.mini.html"
  736. search_template = "coderedcms/pages/event_page.search.html"
  737. calendar_color = ColorField(
  738. blank=True,
  739. help_text=_(
  740. "The color that the event will use when displayed on a calendar."
  741. ),
  742. )
  743. address = models.TextField(blank=True, verbose_name=_("Address"))
  744. content_panels = CoderedWebPage.content_panels + [
  745. MultiFieldPanel(
  746. [
  747. FieldPanel("calendar_color"),
  748. FieldPanel("address"),
  749. ],
  750. heading=_("Event information"),
  751. ),
  752. InlinePanel(
  753. "occurrences",
  754. min_num=1,
  755. heading=_("Dates and times"),
  756. ),
  757. ]
  758. @property
  759. def upcoming_occurrences(self):
  760. """
  761. Returns the next x occurrences for this event.
  762. By default, it returns 10.
  763. """
  764. return self.query_occurrences(num_of_instances_to_return=10)
  765. @property
  766. def most_recent_occurrence(
  767. self,
  768. ) -> Tuple[datetime, datetime, BaseOccurrence]:
  769. """
  770. Gets the next upcoming, or last occurrence if the event has no more occurrences.
  771. """
  772. try:
  773. noc = self.next_occurrence()
  774. if noc:
  775. return noc
  776. except AttributeError:
  777. # Triggers when a preview is initiated on an
  778. # EventPage because it uses a FakeQuerySet object.
  779. # Here we manually compute the next_occurrence
  780. occurrences = [e.next_occurrence() for e in self.occurrences.all()]
  781. # If there are no more occurrences, we find the last one instead
  782. if occurrences and None not in occurrences:
  783. return sorted(occurrences, key=lambda tup: tup[0])[0]
  784. # If both of the above methods fail to find a future occurrence,
  785. # instead return the last occurrence.
  786. aoc = []
  787. for occurrence in self.occurrences.all():
  788. aoc += [instance for instance in occurrence.all_occurrences()]
  789. if len(aoc) > 0:
  790. return aoc[-1] # last one in the list
  791. @property
  792. def seo_struct_event_dict(self) -> dict:
  793. next_occ = self.most_recent_occurrence
  794. if not next_occ:
  795. return {}
  796. # The method returns a tuple of the start, end, and object. We only care about
  797. # the object, so take it out of the tuple.
  798. next_occ = next_occ[2]
  799. sd_dict = {
  800. "@context": "https://schema.org/",
  801. "@type": "Event",
  802. "name": self.title,
  803. "description": self.seo_description,
  804. "startDate": next_occ.start,
  805. "endDate": next_occ.end,
  806. "mainEntityOfPage": {
  807. "@type": "WebPage",
  808. "@id": self.get_full_url(),
  809. },
  810. }
  811. if self.seo_image:
  812. sd_dict.update(
  813. {
  814. "image": get_struct_data_images(
  815. self.get_site(), self.seo_image
  816. )
  817. }
  818. )
  819. if self.address:
  820. sd_dict.update(
  821. {
  822. "location": {
  823. "@type": "Place",
  824. "name": self.title,
  825. "address": {
  826. "@type": "PostalAddress",
  827. "streetAddress": self.address,
  828. },
  829. },
  830. }
  831. )
  832. return sd_dict
  833. @property
  834. def seo_struct_event_json(self) -> str:
  835. return json.dumps(self.seo_struct_event_dict, cls=StructDataEncoder)
  836. def query_occurrences(self, num_of_instances_to_return=None, **kwargs):
  837. """
  838. Returns a list of all upcoming event instances for the specified query.
  839. For more information on what you can query with, visit
  840. https://github.com/gregplaysguitar/django-eventtools
  841. """
  842. event_instances = []
  843. occurrence_kwargs = {
  844. "from_date": kwargs.get("from_date", timezone.now().date())
  845. }
  846. if "limit" in kwargs:
  847. if kwargs["limit"] is not None:
  848. # Limit the number of event instances that will be
  849. # generated per occurrence rule to 10, if not otherwise specified.
  850. occurrence_kwargs["limit"] = kwargs.get("limit", 10)
  851. # For each occurrence rule in all of the occurrence rules for this event.
  852. for occurrence in self.occurrences.all():
  853. # Add the qualifying generated event instances to the list.
  854. event_instances += [
  855. instance
  856. for instance in occurrence.all_occurrences(**occurrence_kwargs)
  857. ]
  858. # Sort all the events by the date that they start
  859. event_instances.sort(key=lambda d: d[0])
  860. # Return the event instances, possibly spliced if num_instances_to_return is set.
  861. return (
  862. event_instances[:num_of_instances_to_return]
  863. if num_of_instances_to_return
  864. else event_instances
  865. )
  866. def convert_to_ical_format(
  867. self,
  868. dt_start: datetime = None,
  869. dt_end: datetime = None,
  870. occurrence=None,
  871. ):
  872. ical_event = ICalEvent()
  873. ical_event.add("summary", self.title)
  874. # needs to get full page url, not just slug
  875. desc_str = _("Details")
  876. ical_event.add("dtstamp", timezone.now())
  877. ical_event.add("description", f"{desc_str}: {self.full_url}")
  878. # NOTE: The use of the url for the id is technically breaking the iCal standard,
  879. # which recommends against use of identifiable info:
  880. # https://icalendar.org/New-Properties-for-iCalendar-RFC-7986/5-3-uid-property.html
  881. # If this breaks in the future,
  882. # implementing a uuid field on the object is probably necessary.
  883. ical_event.add("uid", self.get_full_url())
  884. if self.address:
  885. ical_event.add("location", self.address)
  886. if dt_start:
  887. # Convert to UTC to remove timezone confusion in ical
  888. # clients (i.e. Outlook).
  889. dt_start = dt_start.astimezone(dttimezone.utc)
  890. ical_event.add("dtstart", dt_start)
  891. if dt_end:
  892. # Convert to UTC to remove timezone confusion in ical
  893. # clients (i.e. Outlook).
  894. dt_end = dt_end.astimezone(dttimezone.utc)
  895. ical_event.add("dtend", dt_end)
  896. # Add a reminder alarm
  897. reminder_mins = 15
  898. alarm = Alarm()
  899. alarm.add("action", "DISPLAY")
  900. alarm.add("description", "Reminder")
  901. # Sets the reminder alarm
  902. alarm.add("TRIGGER;RELATED=START", "-PT{0}M".format(reminder_mins))
  903. ical_event.add_component(alarm)
  904. if occurrence:
  905. freq = (
  906. occurrence.repeat.split(":")[1] if occurrence.repeat else None
  907. )
  908. repeat_until = (
  909. occurrence.repeat_until.strftime("%Y%m%dT000000Z")
  910. if occurrence.repeat_until
  911. else None
  912. )
  913. ical_event.add("dtstart", occurrence.start)
  914. if occurrence.end:
  915. ical_event.add("dtend", occurrence.end)
  916. if freq:
  917. ical_event.add("RRULE", freq, encode=False)
  918. if repeat_until:
  919. ical_event.add("until", repeat_until)
  920. return ical_event
  921. def create_single_ical(self, dt_start, dt_end=None):
  922. return self.convert_to_ical_format(dt_start=dt_start, dt_end=dt_end)
  923. def create_recurring_ical(self):
  924. events = []
  925. for occurrence in self.occurrences.all():
  926. events.append(self.convert_to_ical_format(occurrence=occurrence))
  927. return events
  928. class CoderedEventIndexPage(CoderedWebPage):
  929. """
  930. Shows a list of event sub-pages.
  931. """
  932. class Meta:
  933. verbose_name = _("CodeRed Event Index Page")
  934. abstract = True
  935. class CalendarViews(models.TextChoices):
  936. NONE = "", _("No calendar")
  937. MONTH = "month", _("Month")
  938. AGENDA_WEEK = "agendaWeek", _("Week")
  939. AGENDA_DAY = "agendaDay", _("Day")
  940. LIST_MONTH = "listMonth", _("List of events")
  941. class EventStyles(models.TextChoices):
  942. DEFAULT = "", _("Default")
  943. BLOCK = "block", _("Solid rectangles")
  944. DOT = "list-item", _("Dots with labels")
  945. template = "coderedcms/pages/event_index_page.html"
  946. index_show_subpages_default = True
  947. index_order_by_default = "next_occurrence"
  948. index_order_by_choices = (
  949. ("next_occurrence", "Display next occurrence, soonest first"),
  950. ) + CoderedWebPage.index_order_by_choices
  951. default_calendar_view = models.CharField(
  952. blank=True,
  953. choices=CalendarViews.choices,
  954. default=CalendarViews.MONTH,
  955. max_length=255,
  956. verbose_name=_("Calendar Style"),
  957. help_text=_("The default look of the calendar on this page."),
  958. )
  959. event_style = models.CharField(
  960. blank=True,
  961. choices=EventStyles.choices,
  962. default=EventStyles.DEFAULT,
  963. max_length=255,
  964. verbose_name=_("Event Style"),
  965. help_text=_("How events look on the calendar."),
  966. )
  967. layout_panels = CoderedWebPage.layout_panels + [
  968. MultiFieldPanel(
  969. [
  970. FieldPanel("default_calendar_view"),
  971. FieldPanel("event_style"),
  972. ],
  973. heading=_("Calendar Style"),
  974. )
  975. ]
  976. @property
  977. def fullcalendar_view(self) -> str:
  978. """
  979. Translate calendar views to fullcalendar.js identifiers.
  980. """
  981. return {
  982. self.CalendarViews.NONE: "",
  983. self.CalendarViews.MONTH: "dayGridMonth",
  984. self.CalendarViews.AGENDA_WEEK: "timeGridWeek",
  985. self.CalendarViews.AGENDA_DAY: "timeGridDay",
  986. self.CalendarViews.LIST_MONTH: "listMonth",
  987. }[self.default_calendar_view]
  988. @property
  989. def fullcalendar_event_display(self) -> str:
  990. """
  991. Translate event display styles to fullcalendar.js identifiers.
  992. """
  993. return {
  994. self.EventStyles.DEFAULT: "auto",
  995. self.EventStyles.BLOCK: "block",
  996. self.EventStyles.DOT: "list-item",
  997. }[self.event_style]
  998. def get_index_children(self):
  999. if (
  1000. self.index_query_pagemodel
  1001. and self.index_order_by == "next_occurrence"
  1002. ):
  1003. querymodel = resolve_model_string(
  1004. self.index_query_pagemodel, self._meta.app_label
  1005. )
  1006. qs = querymodel.objects.child_of(self).live()
  1007. # filter out events that don't have a next_occurrence
  1008. upcoming = []
  1009. for event in qs.all():
  1010. if event.next_occurrence():
  1011. upcoming.append(event)
  1012. # Sort the events by next_occurrence start date.
  1013. return sorted(upcoming, key=lambda e: e.next_occurrence()[0])
  1014. return super().get_index_children()
  1015. def get_calendar_events(
  1016. self, start: Union[datetime, date], end: Union[datetime, date]
  1017. ) -> List[Dict[str, str]]:
  1018. """
  1019. Returns a list of event occurrences as dictionaries with times
  1020. converted to Django TIME_ZONE settings.
  1021. """
  1022. # start with all child events, regardless of get_index_children rules.
  1023. querymodel = resolve_model_string(
  1024. self.index_query_pagemodel, self._meta.app_label
  1025. )
  1026. qs = querymodel.objects.child_of(self).live()
  1027. event_instances = []
  1028. for event in qs:
  1029. occurrences = event.query_occurrences(
  1030. limit=None, from_date=start, to_date=end
  1031. )
  1032. for occurrence in occurrences:
  1033. local_start = timezone.localtime(value=occurrence[0])
  1034. local_end = None
  1035. if occurrence[1]:
  1036. local_end = timezone.localtime(value=occurrence[1])
  1037. event_data = {
  1038. "title": event.title,
  1039. "start": local_start.strftime("%Y-%m-%dT%H:%M:%S%z"),
  1040. "end": (
  1041. local_end.strftime("%Y-%m-%dT%H:%M:%S%z")
  1042. if local_end
  1043. else ""
  1044. ),
  1045. "description": "",
  1046. }
  1047. if event.url:
  1048. event_data["url"] = event.url
  1049. if event.calendar_color:
  1050. event_data["color"] = event.calendar_color
  1051. event_instances.append(event_data)
  1052. return event_instances
  1053. class CoderedEventOccurrence(Orderable, BaseOccurrence):
  1054. class Meta(Orderable.Meta):
  1055. verbose_name = _("CodeRed Event Occurrence")
  1056. abstract = True
  1057. class CoderedFormMixin(models.Model):
  1058. class Meta:
  1059. abstract = True
  1060. submissions_list_view_class = CoderedSubmissionsListView
  1061. encoder = DjangoJSONEncoder
  1062. # Custom codered fields
  1063. to_address = models.CharField(
  1064. max_length=255,
  1065. blank=True,
  1066. verbose_name=_("Email form submissions to"),
  1067. help_text=_(
  1068. "Optional - email form submissions to this address. "
  1069. "Separate multiple addresses by comma."
  1070. ),
  1071. )
  1072. reply_address = models.CharField(
  1073. max_length=255,
  1074. blank=True,
  1075. verbose_name=_("Reply-to address"),
  1076. help_text=_(
  1077. "Optional - to reply to the submitter, specify the email field here. "
  1078. "For example, if a form field above is labeled "
  1079. '"Your Email", enter: {{ your_email }}'
  1080. ),
  1081. )
  1082. subject = models.CharField(
  1083. max_length=255,
  1084. blank=True,
  1085. verbose_name=_("Subject"),
  1086. )
  1087. save_to_database = models.BooleanField(
  1088. default=True,
  1089. verbose_name=_("Save form submissions"),
  1090. help_text=_(
  1091. "Submissions are saved to database and can be exported at any time."
  1092. ),
  1093. )
  1094. thank_you_page = models.ForeignKey(
  1095. "wagtailcore.Page",
  1096. null=True,
  1097. blank=True,
  1098. on_delete=models.SET_NULL,
  1099. related_name="+",
  1100. verbose_name=_("Thank you page"),
  1101. help_text=_(
  1102. "The page users are redirected to after submitting the form."
  1103. ),
  1104. )
  1105. button_text = models.CharField(
  1106. max_length=255,
  1107. default=_("Submit"),
  1108. verbose_name=_("Button text"),
  1109. )
  1110. button_style = models.CharField(
  1111. blank=True,
  1112. choices=crx_settings.CRX_FRONTEND_BTN_STYLE_CHOICES,
  1113. default=crx_settings.CRX_FRONTEND_BTN_STYLE_DEFAULT,
  1114. max_length=255,
  1115. verbose_name=_("Button style"),
  1116. )
  1117. button_size = models.CharField(
  1118. blank=True,
  1119. choices=crx_settings.CRX_FRONTEND_BTN_SIZE_CHOICES,
  1120. default=crx_settings.CRX_FRONTEND_BTN_SIZE_DEFAULT,
  1121. max_length=255,
  1122. verbose_name=_("Button Size"),
  1123. )
  1124. button_css_class = models.CharField(
  1125. max_length=255,
  1126. blank=True,
  1127. verbose_name=_("Button CSS class"),
  1128. help_text=_("Custom CSS class applied to the submit button."),
  1129. )
  1130. form_css_class = models.CharField(
  1131. max_length=255,
  1132. blank=True,
  1133. verbose_name=_("Form CSS Class"),
  1134. help_text=_("Custom CSS class applied to <form> element."),
  1135. )
  1136. form_id = models.CharField(
  1137. max_length=255,
  1138. blank=True,
  1139. verbose_name=_("Form ID"),
  1140. help_text=_("Custom ID applied to <form> element."),
  1141. )
  1142. form_golive_at = models.DateTimeField(
  1143. blank=True,
  1144. null=True,
  1145. verbose_name=_("Form go live date/time"),
  1146. help_text=_("Date and time when the FORM goes live on the page."),
  1147. )
  1148. form_expire_at = models.DateTimeField(
  1149. blank=True,
  1150. null=True,
  1151. verbose_name=_("Form expiry date/time"),
  1152. help_text=_(
  1153. "Date and time when the FORM will no longer be available on the page."
  1154. ),
  1155. )
  1156. spam_protection = models.BooleanField(
  1157. default=True,
  1158. verbose_name=_("Spam Protection"),
  1159. help_text=_(
  1160. "When enabled, the CMS will filter out spam form submissions for this page."
  1161. ),
  1162. )
  1163. body_content_panels = [
  1164. MultiFieldPanel(
  1165. [
  1166. FieldPanel("thank_you_page"),
  1167. FieldPanel("button_text"),
  1168. FieldPanel("button_style"),
  1169. FieldPanel("button_size"),
  1170. FieldPanel("button_css_class"),
  1171. FieldPanel("form_css_class"),
  1172. FieldPanel("form_id"),
  1173. ],
  1174. _("Form Settings"),
  1175. ),
  1176. MultiFieldPanel(
  1177. [
  1178. FieldPanel("save_to_database"),
  1179. FieldPanel("to_address"),
  1180. FieldPanel("reply_address"),
  1181. FieldPanel("subject"),
  1182. ],
  1183. _("Form Submissions"),
  1184. ),
  1185. ]
  1186. settings_panels = [
  1187. MultiFieldPanel(
  1188. [
  1189. FieldRowPanel(
  1190. [
  1191. FieldPanel("form_golive_at"),
  1192. FieldPanel("form_expire_at"),
  1193. ],
  1194. classname="label-above",
  1195. ),
  1196. ],
  1197. _("Form Scheduled Publishing"),
  1198. ),
  1199. FieldPanel("spam_protection"),
  1200. ]
  1201. @property
  1202. def get_form_id(self):
  1203. """
  1204. Returns a suitable HTML element ID.
  1205. """
  1206. if self.form_id:
  1207. return self.form_id.strip()
  1208. return f"crx-form-{self.pk}"
  1209. @property
  1210. def form_live(self):
  1211. """
  1212. A boolean on whether or not the <form> element should be shown on the page.
  1213. """
  1214. return (
  1215. self.form_golive_at is None or self.form_golive_at <= timezone.now()
  1216. ) and (
  1217. self.form_expire_at is None or self.form_expire_at >= timezone.now()
  1218. )
  1219. def get_landing_page_template(self, request, *args, **kwargs):
  1220. return self.landing_page_template
  1221. def process_data(self, form, request):
  1222. processed_data = {}
  1223. # Handle file uploads
  1224. for key, val in form.cleaned_data.items():
  1225. if (
  1226. type(val) is InMemoryUploadedFile
  1227. or type(val) is TemporaryUploadedFile
  1228. ):
  1229. # Save the file and get its URL
  1230. # Custom code to ensure that anonymous users get a session key.
  1231. if not request.session.session_key:
  1232. request.session.create()
  1233. directory = request.session.session_key
  1234. storage = self.get_storage()
  1235. Path(storage.path(directory)).mkdir(parents=True, exist_ok=True)
  1236. path = storage.get_available_name(
  1237. str(Path(directory) / val.name)
  1238. )
  1239. with storage.open(path, "wb+") as destination:
  1240. for chunk in val.chunks():
  1241. destination.write(chunk)
  1242. processed_data[key] = "{0}{1}".format(
  1243. crx_settings.CRX_PROTECTED_MEDIA_URL, path
  1244. )
  1245. else:
  1246. processed_data[key] = val
  1247. return processed_data
  1248. def get_storage(self):
  1249. return FileSystemStorage(
  1250. location=crx_settings.CRX_PROTECTED_MEDIA_ROOT,
  1251. base_url=crx_settings.CRX_PROTECTED_MEDIA_URL,
  1252. )
  1253. def process_form_submission(
  1254. self, request, form, form_submission, processed_data
  1255. ):
  1256. # Save to database
  1257. if self.save_to_database:
  1258. form_submission.save()
  1259. # Send the mails
  1260. if self.to_address:
  1261. self.send_summary_mail(request, form, processed_data)
  1262. if self.confirmation_emails:
  1263. # Convert form data into a context.
  1264. context = Context(self.data_to_dict(processed_data, request))
  1265. # Render emails as if they are django templates.
  1266. for email in self.confirmation_emails.all():
  1267. # Build email message parameters.
  1268. message_args = {}
  1269. # From
  1270. if email.from_address:
  1271. template_from_email = Template(email.from_address)
  1272. message_args["from_email"] = template_from_email.render(
  1273. context
  1274. )
  1275. else:
  1276. genemail = LayoutSettings.for_request(
  1277. request
  1278. ).from_email_address
  1279. if genemail:
  1280. message_args["from_email"] = genemail
  1281. # Reply-to
  1282. if email.reply_address:
  1283. template_reply_to = Template(email.reply_address)
  1284. message_args["reply_to"] = template_reply_to.render(
  1285. context
  1286. ).split(",")
  1287. # CC
  1288. if email.cc_address:
  1289. template_cc = Template(email.cc_address)
  1290. message_args["cc"] = template_cc.render(context).split(",")
  1291. # BCC
  1292. if email.bcc_address:
  1293. template_bcc = Template(email.bcc_address)
  1294. message_args["bcc"] = template_bcc.render(context).split(
  1295. ","
  1296. )
  1297. # Subject
  1298. if email.subject:
  1299. template_subject = Template(email.subject)
  1300. message_args["subject"] = template_subject.render(context)
  1301. else:
  1302. message_args["subject"] = self.title
  1303. # Body
  1304. template_body = Template(email.body)
  1305. message_args["body"] = template_body.render(context)
  1306. # To
  1307. template_to = Template(email.to_address)
  1308. message_args["to"] = template_to.render(context).split(",")
  1309. # Send email
  1310. self.send_mail(request, message_args, "html")
  1311. for fn in hooks.get_hooks("form_page_submit"):
  1312. fn(instance=self, form_submission=form_submission)
  1313. def send_summary_mail(self, request, form, processed_data):
  1314. """
  1315. Sends a form submission summary email.
  1316. """
  1317. addresses = [x.strip() for x in self.to_address.split(",")]
  1318. content = []
  1319. data = self.data_to_dict(processed_data, request)
  1320. for field in form:
  1321. # Get key from form, transform same as data_to_dict() does.
  1322. key = field.html_name.replace("-", "_")
  1323. label = field.label
  1324. value = data.get(key)
  1325. content.append("{0}: {1}".format(label, value))
  1326. content_str = "\n-------------------- \n".join(content) + "\n"
  1327. # Build email message parameters
  1328. message_args = {
  1329. "body": content_str,
  1330. "to": addresses,
  1331. }
  1332. if self.subject:
  1333. message_args["subject"] = self.subject
  1334. else:
  1335. message_args["subject"] = self.title
  1336. genemail = LayoutSettings.for_request(request).from_email_address
  1337. if genemail:
  1338. message_args["from_email"] = genemail
  1339. if self.reply_address:
  1340. # Render reply-to field using form submission as context.
  1341. context = Context(data)
  1342. template_reply_to = Template(self.reply_address)
  1343. message_args["reply_to"] = template_reply_to.render(context).split(
  1344. ","
  1345. )
  1346. # Send email
  1347. self.send_mail(request, message_args)
  1348. def send_mail(
  1349. self, resuest, message_args: dict, content_subtype: Optional[str] = None
  1350. ):
  1351. """
  1352. Utility to send email messages from form submissions.
  1353. You can override this method to have the form send mail in a different
  1354. way or using a different backend as needed.
  1355. """
  1356. message = EmailMessage(**message_args)
  1357. if content_subtype:
  1358. message.content_subtype = content_subtype
  1359. message.send()
  1360. def render_landing_page(self, request, form_submission=None):
  1361. """
  1362. Renders the landing page.
  1363. You can override this method to return a different HttpResponse as
  1364. landing page. E.g. you could return a redirect to a separate page.
  1365. """
  1366. if self.thank_you_page:
  1367. return redirect(self.thank_you_page.url)
  1368. context = self.get_context(request)
  1369. context["form_submission"] = form_submission
  1370. response = render(
  1371. request, self.get_landing_page_template(request), context
  1372. )
  1373. return response
  1374. def data_to_dict(self, processed_data, request):
  1375. """
  1376. Converts processed form data into a dictionary suitable
  1377. for rendering in a context.
  1378. """
  1379. dictionary = {}
  1380. for key, value in processed_data.items():
  1381. new_key = key.replace("-", "_")
  1382. if isinstance(value, list):
  1383. dictionary[new_key] = ", ".join(value)
  1384. else:
  1385. dictionary[new_key] = (
  1386. utils.attempt_protected_media_value_conversion(
  1387. request, value
  1388. )
  1389. )
  1390. return dictionary
  1391. preview_modes = [
  1392. ("form", _("Form")),
  1393. ("landing", _("Thank you page")),
  1394. ("wagtail-seo", _("SEO Preview")),
  1395. ]
  1396. def serve_preview(self, request, mode):
  1397. if mode == "landing":
  1398. request.is_preview = True
  1399. return self.render_landing_page(request)
  1400. return super().serve_preview(request, mode)
  1401. def serve_submissions_list_view(self, request, *args, **kwargs):
  1402. """
  1403. Returns list submissions view for admin.
  1404. `list_submissions_view_class` can be set to provide custom view class.
  1405. Your class must be inherited from SubmissionsListView.
  1406. """
  1407. view = self.submissions_list_view_class.as_view()
  1408. return view(request, form_page=self, *args, **kwargs)
  1409. def get_form(self, request, *args, **kwargs):
  1410. form_class = self.get_form_class()
  1411. form_params = self.get_form_parameters()
  1412. form_params.update(kwargs)
  1413. if request.method == "POST":
  1414. return form_class(request.POST, request.FILES, *args, **form_params)
  1415. return form_class(*args, **form_params)
  1416. def contains_spam(self, request) -> bool:
  1417. """
  1418. Checks if the site's spam_service identifies spam.
  1419. """
  1420. if not self.spam_protection:
  1421. return False
  1422. ls = LayoutSettings.for_site(self.get_site())
  1423. if ls.spam_service == ls.SpamService.HONEYPOT:
  1424. if request.POST.get("cr-decoy-comments", None):
  1425. return True
  1426. elif ls.spam_service == ls.SpamService.RECAPTCHA_V3:
  1427. rr = verify_response(
  1428. request.POST.get("g-recaptcha-response", ""),
  1429. ls.recaptcha_secret_key,
  1430. utils.get_ip(request),
  1431. )
  1432. # Score ranges from 0 (likely spam) to 1 (likely good).
  1433. if rr.success and rr.score >= 0:
  1434. return rr.score < ls.recaptcha_threshold
  1435. else:
  1436. return True
  1437. elif ls.spam_service == ls.SpamService.RECAPTCHA_V2:
  1438. rr = verify_response(
  1439. request.POST.get("g-recaptcha-response", ""),
  1440. ls.recaptcha_secret_key,
  1441. utils.get_ip(request),
  1442. )
  1443. return not rr.success
  1444. return False
  1445. def process_spam_request(self, form, request):
  1446. """
  1447. Called when spam is found in the request.
  1448. """
  1449. messages.error(request, self.get_spam_message())
  1450. logger.info(
  1451. "Detected spam submission on page: {0}\n{1}".format(
  1452. self.title, vars(request)
  1453. )
  1454. )
  1455. return self.process_form_get(form, request)
  1456. def get_spam_message(self):
  1457. return _(
  1458. "There was an error while processing your submission. Please try again."
  1459. )
  1460. def process_form_post(self, form, request):
  1461. if form.is_valid():
  1462. processed_data = self.process_data(form, request)
  1463. form_submission = self.get_submission_class()(
  1464. form_data=processed_data,
  1465. page=self,
  1466. )
  1467. self.process_form_submission(
  1468. request=request,
  1469. form=form,
  1470. form_submission=form_submission,
  1471. processed_data=processed_data,
  1472. )
  1473. return self.render_landing_page(request, form_submission)
  1474. return self.process_form_get(form, request)
  1475. def process_form_get(self, form, request):
  1476. context = self.get_context(request)
  1477. context["form"] = form
  1478. response = render(request, self.get_template(request), context)
  1479. return response
  1480. def serve(self, request, *args, **kwargs):
  1481. form = self.get_form(request, page=self, user=request.user)
  1482. if request.method == "POST":
  1483. if self.contains_spam(request):
  1484. return self.process_spam_request(form, request)
  1485. return self.process_form_post(form, request)
  1486. return self.process_form_get(form, request)
  1487. def get_preview_context(self, request, *args, **kwargs):
  1488. """
  1489. In preview mode, simlpy show a blank form but do not handle submissions.
  1490. """
  1491. ctx = super().get_preview_context(request, *args, **kwargs)
  1492. ctx["form"] = self.get_form(request, page=self, user=request.user)
  1493. return ctx
  1494. class CoderedFormPage(CoderedFormMixin, CoderedWebPage):
  1495. """
  1496. This is basically a clone of wagtail.contrib.forms.models.AbstractForm
  1497. with changes in functionality and extending CoderedWebPage vs wagtailcore.Page.
  1498. """
  1499. class Meta:
  1500. verbose_name = _("CodeRed Form Page")
  1501. abstract = True
  1502. template = "coderedcms/pages/form_page.html"
  1503. miniview_template = "coderedcms/pages/form_page.mini.html"
  1504. landing_page_template = "coderedcms/pages/form_page_landing.html"
  1505. base_form_class = WagtailAdminFormPageForm
  1506. form_builder = CoderedFormBuilder
  1507. body_content_panels = (
  1508. CoderedWebPage.body_content_panels
  1509. + [
  1510. InlinePanel("form_fields", label="Form fields"),
  1511. ]
  1512. + CoderedFormMixin.body_content_panels
  1513. + [
  1514. FormSubmissionsPanel(),
  1515. InlinePanel("confirmation_emails", label=_("Confirmation Emails")),
  1516. ]
  1517. )
  1518. settings_panels = (
  1519. CoderedPage.settings_panels + CoderedFormMixin.settings_panels
  1520. )
  1521. def __init__(self, *args, **kwargs):
  1522. super().__init__(*args, **kwargs)
  1523. if not hasattr(self, "landing_page_template"):
  1524. name, ext = os.path.splitext(self.template)
  1525. self.landing_page_template = name + "_landing" + ext
  1526. def get_form_fields(self):
  1527. """
  1528. Form page expects `form_fields` to be declared.
  1529. If you want to change backwards relation name,
  1530. you need to override this method.
  1531. """
  1532. return self.form_fields.all()
  1533. def get_data_fields(self):
  1534. """
  1535. Returns a list of tuples with (field_name, field_label).
  1536. """
  1537. data_fields = [
  1538. ("submit_time", _("Submission date")),
  1539. ]
  1540. data_fields += [
  1541. (field.clean_name, field.label) for field in self.get_form_fields()
  1542. ]
  1543. return data_fields
  1544. def get_form_class(self):
  1545. fb = self.form_builder(self.get_form_fields())
  1546. return fb.get_form_class()
  1547. def get_form_parameters(self):
  1548. return {}
  1549. def get_submission_class(self):
  1550. """
  1551. Returns submission class.
  1552. You can override this method to provide custom submission class.
  1553. Your class must be inherited from AbstractFormSubmission.
  1554. """
  1555. return FormSubmission
  1556. class CoderedSubmissionRevision(AbstractSubmissionRevision):
  1557. pass
  1558. class CoderedSessionFormSubmission(AbstractSessionFormSubmission):
  1559. """
  1560. Customize how certain fields are rendered.
  1561. """
  1562. @staticmethod
  1563. def get_revision_class():
  1564. return CoderedSubmissionRevision
  1565. def render_link(self, value):
  1566. return "{0}{1}".format(crx_settings.CRX_PROTECTED_MEDIA_URL, value)
  1567. def render_image(self, value):
  1568. return "{0}{1}".format(crx_settings.CRX_PROTECTED_MEDIA_URL, value)
  1569. def render_file(self, value):
  1570. return "{0}{1}".format(crx_settings.CRX_PROTECTED_MEDIA_URL, value)
  1571. class CoderedStreamFormMixin(StreamFormMixin):
  1572. """
  1573. Customize the classes used to store submissions.
  1574. """
  1575. @staticmethod
  1576. def get_submission_class():
  1577. return FormSubmission
  1578. @staticmethod
  1579. def get_session_submission_class():
  1580. return CoderedSessionFormSubmission
  1581. class CoderedStreamFormPage(
  1582. CoderedFormMixin, CoderedStreamFormMixin, CoderedWebPage
  1583. ):
  1584. class Meta:
  1585. verbose_name = _("CodeRed Advanced Form Page")
  1586. abstract = True
  1587. template = "coderedcms/pages/stream_form_page.html"
  1588. landing_page_template = "coderedcms/pages/form_page_landing.html"
  1589. form_fields = StreamField(
  1590. STREAMFORM_BLOCKS,
  1591. use_json_field=True,
  1592. )
  1593. encoder = StreamFormJSONEncoder
  1594. body_content_panels = (
  1595. [FieldPanel("form_fields")]
  1596. + CoderedFormMixin.body_content_panels
  1597. + [InlinePanel("confirmation_emails", label=_("Confirmation Emails"))]
  1598. )
  1599. def process_form_post(self, form, request):
  1600. if form.is_valid():
  1601. is_complete = self.steps.update_data()
  1602. if is_complete:
  1603. # NOTE: normally this happens in
  1604. # ``wagtail_flexible_forms.StreamFormMixin.serve()``
  1605. # but we are overriding it in ``CoderedFormMixin``.
  1606. # This is a potentially confusing architecture.
  1607. form_submission = self.create_final_submission(
  1608. request, delete_session=True
  1609. )
  1610. self.process_form_submission(
  1611. request=request,
  1612. form=form,
  1613. form_submission=form_submission,
  1614. processed_data=form_submission.get_data(),
  1615. )
  1616. return self.render_landing_page(request, form_submission)
  1617. return HttpResponseRedirect(self.url)
  1618. return self.process_form_get(form, request)
  1619. def process_form_get(self, form, request):
  1620. return CoderedWebPage.serve(self, request)
  1621. def get_form(self, request, *args, **kwargs):
  1622. return self.get_context(request)["form"]
  1623. def get_storage(self):
  1624. return FileSystemStorage(
  1625. location=crx_settings.CRX_PROTECTED_MEDIA_ROOT,
  1626. base_url=crx_settings.CRX_PROTECTED_MEDIA_URL,
  1627. )
  1628. class CoderedLocationPage(CoderedWebPage):
  1629. """
  1630. Location, suitable for store locations or help centers.
  1631. """
  1632. class Meta:
  1633. verbose_name = _("CodeRed Location")
  1634. abstract = True
  1635. template = "coderedcms/pages/location_page.html"
  1636. # Override body to provide simpler content
  1637. body = StreamField(
  1638. CONTENT_STREAMBLOCKS,
  1639. null=True,
  1640. blank=True,
  1641. use_json_field=True,
  1642. )
  1643. address = models.TextField(blank=True, verbose_name=_("Address"))
  1644. latitude = models.FloatField(
  1645. blank=True, null=True, verbose_name=_("Latitude")
  1646. )
  1647. longitude = models.FloatField(
  1648. blank=True, null=True, verbose_name=_("Longitude")
  1649. )
  1650. auto_update_latlng = models.BooleanField(
  1651. default=True,
  1652. verbose_name=_("Auto Update Latitude and Longitude"),
  1653. help_text=_(
  1654. "If checked, automatically update the latitude and longitude "
  1655. "when the address is updated."
  1656. ),
  1657. )
  1658. map_title = models.CharField(
  1659. blank=True,
  1660. max_length=255,
  1661. verbose_name=_("Map Title"),
  1662. help_text=_(
  1663. "If this is filled out, this is the title that will be used on the map."
  1664. ),
  1665. )
  1666. map_description = models.CharField(
  1667. blank=True,
  1668. max_length=255,
  1669. verbose_name=_("Map Description"),
  1670. help_text=_(
  1671. "If this is filled out, this is the description that will be used on the map."
  1672. ),
  1673. )
  1674. website = models.TextField(blank=True, verbose_name=_("Website"))
  1675. phone_number = models.CharField(
  1676. blank=True, max_length=255, verbose_name=_("Phone Number")
  1677. )
  1678. content_panels = CoderedWebPage.content_panels + [
  1679. FieldPanel("address"),
  1680. FieldPanel("website"),
  1681. FieldPanel("phone_number"),
  1682. ]
  1683. layout_panels = CoderedWebPage.layout_panels + [
  1684. MultiFieldPanel(
  1685. [
  1686. FieldPanel("map_title"),
  1687. FieldPanel("map_description"),
  1688. ],
  1689. heading=_("Map Layout"),
  1690. ),
  1691. ]
  1692. settings_panels = CoderedWebPage.settings_panels + [
  1693. MultiFieldPanel(
  1694. [
  1695. FieldPanel("auto_update_latlng"),
  1696. FieldPanel("latitude"),
  1697. FieldPanel("longitude"),
  1698. ],
  1699. heading=_("Location Settings"),
  1700. ),
  1701. ]
  1702. @property
  1703. def geojson_name(self):
  1704. return self.map_title or self.title
  1705. @property
  1706. def geojson_description(self):
  1707. return self.map_description
  1708. @property
  1709. def render_pin_description(self):
  1710. return render_to_string(
  1711. "coderedcms/includes/map_pin_description.html", {"page": self}
  1712. )
  1713. @property
  1714. def render_list_description(self):
  1715. return render_to_string(
  1716. "coderedcms/includes/map_list_description.html", {"page": self}
  1717. )
  1718. def to_geojson(self):
  1719. return {
  1720. "type": "Feature",
  1721. "geometry": {
  1722. "type": "Point",
  1723. "coordinates": [self.longitude, self.latitude],
  1724. },
  1725. "properties": {
  1726. "list_description": self.render_list_description,
  1727. "pin_description": self.render_pin_description,
  1728. },
  1729. }
  1730. def save(self, *args, **kwargs):
  1731. if (
  1732. self.auto_update_latlng
  1733. and LayoutSettings.for_site(
  1734. Site.objects.get(is_default_site=True)
  1735. ).google_maps_api_key
  1736. ):
  1737. try:
  1738. g = geocoder.google(
  1739. self.address,
  1740. key=LayoutSettings.for_site(
  1741. Site.objects.get(is_default_site=True)
  1742. ).google_maps_api_key,
  1743. )
  1744. self.latitude = g.latlng[0]
  1745. self.longitude = g.latlng[1]
  1746. except TypeError:
  1747. # Raised if google denied the request
  1748. pass
  1749. return super(CoderedLocationPage, self).save(*args, **kwargs)
  1750. def get_context(self, request, *args, **kwargs):
  1751. context = super().get_context(request)
  1752. context["google_api_key"] = LayoutSettings.for_site(
  1753. Site.objects.get(is_default_site=True)
  1754. ).google_maps_api_key
  1755. return context
  1756. class CoderedLocationIndexPage(CoderedWebPage):
  1757. """
  1758. Shows a map view of the children CoderedLocationPage.
  1759. """
  1760. class Meta:
  1761. verbose_name = _("CodeRed Location Index Page")
  1762. abstract = True
  1763. template = "coderedcms/pages/location_index_page.html"
  1764. index_show_subpages_default = True
  1765. center_latitude = models.FloatField(
  1766. null=True,
  1767. blank=True,
  1768. help_text=_("The default latitude you want the map set to."),
  1769. default=0,
  1770. )
  1771. center_longitude = models.FloatField(
  1772. null=True,
  1773. blank=True,
  1774. help_text=_("The default longitude you want the map set to."),
  1775. default=0,
  1776. )
  1777. zoom = models.IntegerField(
  1778. default=8,
  1779. validators=[
  1780. MaxValueValidator(20),
  1781. MinValueValidator(1),
  1782. ],
  1783. help_text=_(
  1784. "Requires API key to use zoom. "
  1785. "1: World, 5: Landmass/continent, 10: City, 15: Streets, 20: Buildings"
  1786. ),
  1787. )
  1788. layout_panels = CoderedWebPage.layout_panels + [
  1789. MultiFieldPanel(
  1790. [
  1791. FieldPanel("center_latitude"),
  1792. FieldPanel("center_longitude"),
  1793. FieldPanel("zoom"),
  1794. ],
  1795. heading=_("Map Display"),
  1796. ),
  1797. ]
  1798. def geojson_data(self, viewport=None):
  1799. """
  1800. function that will return all locations under this index as geoJSON compliant data.
  1801. It is filtered by a latitude/longitude viewport if given.
  1802. viewport is a string in the format of :
  1803. 'southwest.latitude,southwest.longitude|northeast.latitude,northeast.longitude'
  1804. An example viewport that covers Cleveland, OH would look like this:
  1805. '41.354912150983964,-81.95331736661791|41.663427748126935,-81.45206614591478'
  1806. """
  1807. qs = self.get_index_children().live()
  1808. if viewport:
  1809. southwest, northeast = viewport.split("|")
  1810. southwest = [float(x) for x in southwest.split(",")]
  1811. northeast = [float(x) for x in northeast.split(",")]
  1812. qs = qs.filter(
  1813. latitude__gte=southwest[0],
  1814. latitude__lte=northeast[0],
  1815. longitude__gte=southwest[1],
  1816. longitude__lte=northeast[1],
  1817. )
  1818. return {
  1819. "type": "FeatureCollection",
  1820. "features": [location.to_geojson() for location in qs],
  1821. }
  1822. def serve(self, request, *args, **kwargs):
  1823. data_format = request.GET.get("data-format", None)
  1824. if data_format == "geojson":
  1825. return self.serve_geojson(request, *args, **kwargs)
  1826. return super().serve(request, *args, **kwargs)
  1827. def serve_geojson(self, request, *args, **kwargs):
  1828. viewport = request.GET.get("viewport", None)
  1829. return JsonResponse(self.geojson_data(viewport=viewport))
  1830. def get_context(self, request, *args, **kwargs):
  1831. context = super().get_context(request)
  1832. context["google_api_key"] = LayoutSettings.for_site(
  1833. Site.objects.get(is_default_site=True)
  1834. ).google_maps_api_key
  1835. return context