page_models.py 49 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533
  1. """
  2. Base and abstract pages used in CodeRed CMS.
  3. """
  4. import json
  5. import os
  6. import geocoder
  7. from django.conf import settings
  8. from django.core.files.uploadedfile import InMemoryUploadedFile, TemporaryUploadedFile
  9. from django.core.files.storage import FileSystemStorage
  10. from django.core.mail import send_mail, EmailMessage
  11. from django.core.paginator import Paginator
  12. from django.core.serializers.json import DjangoJSONEncoder
  13. from django.core.validators import MaxValueValidator, MinValueValidator
  14. from django.db import models
  15. from django.http import JsonResponse
  16. from django.shortcuts import render, redirect
  17. from django.template import Context, Template
  18. from django.template.loader import render_to_string
  19. from django.utils import timezone
  20. from django.utils.html import strip_tags
  21. from django.utils.translation import ugettext_lazy as _
  22. from eventtools.models import BaseEvent, BaseOccurrence
  23. from icalendar import Event as ICalEvent
  24. from modelcluster.fields import ParentalKey
  25. from modelcluster.tags import ClusterTaggableManager
  26. from taggit.models import TaggedItemBase
  27. from wagtail.admin.edit_handlers import (
  28. HelpPanel,
  29. FieldPanel,
  30. FieldRowPanel,
  31. InlinePanel,
  32. MultiFieldPanel,
  33. ObjectList,
  34. PageChooserPanel,
  35. StreamFieldPanel,
  36. TabbedInterface)
  37. from wagtail.core import hooks
  38. from wagtail.core.fields import StreamField
  39. from wagtail.core.models import Orderable, PageBase, Page, Site
  40. from wagtail.core.utils import resolve_model_string
  41. from wagtail.contrib.forms.edit_handlers import FormSubmissionsPanel
  42. from wagtail.contrib.forms.forms import WagtailAdminFormPageForm
  43. from wagtail.images.edit_handlers import ImageChooserPanel
  44. from wagtail.contrib.forms.models import FormSubmission
  45. from wagtail.search import index
  46. from wagtailcache.cache import WagtailCacheMixin
  47. from coderedcms import schema, utils
  48. from coderedcms.blocks import (
  49. CONTENT_STREAMBLOCKS,
  50. LAYOUT_STREAMBLOCKS,
  51. ContentWallBlock,
  52. OpenHoursBlock,
  53. StructuredDataActionBlock)
  54. from coderedcms.fields import ColorField
  55. from coderedcms.forms import CoderedFormBuilder, CoderedSubmissionsListView
  56. from coderedcms.models.wagtailsettings_models import GeneralSettings, LayoutSettings, SeoSettings, GoogleApiSettings
  57. from coderedcms.settings import cr_settings
  58. CODERED_PAGE_MODELS = []
  59. def get_page_models():
  60. return CODERED_PAGE_MODELS
  61. class CoderedPageMeta(PageBase):
  62. def __init__(cls, name, bases, dct):
  63. super().__init__(name, bases, dct)
  64. if 'amp_template' not in dct:
  65. cls.amp_template = None
  66. if 'search_db_include' not in dct:
  67. cls.search_db_include = False
  68. if 'search_db_boost' not in dct:
  69. cls.search_db_boost = 0
  70. if 'search_filterable' not in dct:
  71. cls.search_filterable = False
  72. if 'search_name' not in dct:
  73. cls.search_name = cls._meta.verbose_name
  74. if 'search_name_plural' not in dct:
  75. cls.search_name_plural = cls._meta.verbose_name_plural
  76. if 'search_template' not in dct:
  77. cls.search_template = 'coderedcms/pages/search_result.html'
  78. if not cls._meta.abstract:
  79. CODERED_PAGE_MODELS.append(cls)
  80. class CoderedTag(TaggedItemBase):
  81. class Meta:
  82. verbose_name = _('CodeRed Tag')
  83. content_object = ParentalKey('coderedcms.CoderedPage', related_name='tagged_items')
  84. class CoderedPage(WagtailCacheMixin, Page, metaclass=CoderedPageMeta):
  85. """
  86. General use page with caching, templating, and SEO functionality.
  87. All pages should inherit from this.
  88. """
  89. class Meta:
  90. verbose_name = _('CodeRed Page')
  91. # Do not allow this page type to be created in wagtail admin
  92. is_creatable = False
  93. # Templates
  94. # The page will render the following templates under certain conditions:
  95. #
  96. # template = ''
  97. # amp_template = ''
  98. # ajax_template = ''
  99. # search_template = ''
  100. ###############
  101. # Content fields
  102. ###############
  103. cover_image = models.ForeignKey(
  104. 'wagtailimages.Image',
  105. null=True,
  106. blank=True,
  107. on_delete=models.SET_NULL,
  108. related_name='+',
  109. verbose_name=_('Cover image'),
  110. )
  111. ###############
  112. # Index fields
  113. ###############
  114. # Subclasses can override this to enabled index features by default.
  115. index_show_subpages_default = False
  116. # Subclasses can override this to query on a specific
  117. # page model, rather than the default wagtail Page.
  118. index_query_pagemodel = 'wagtailcore.Page'
  119. # Subclasses can override these fields to enable custom
  120. # ordering based on specific subpage fields.
  121. index_order_by_default = ''
  122. index_order_by_choices = (
  123. ('', _('Default Ordering')),
  124. ('-first_published_at', _('Date first published, newest to oldest')),
  125. ('first_published_at', _('Date first published, oldest to newest')),
  126. ('-last_published_at', _('Date updated, newest to oldest')),
  127. ('last_published_at', _('Date updated, oldest to newest')),
  128. ('title', _('Title, alphabetical')),
  129. ('-title', _('Title, reverse alphabetical')),
  130. )
  131. index_show_subpages = models.BooleanField(
  132. default=index_show_subpages_default,
  133. verbose_name=_('Show list of child pages')
  134. )
  135. index_order_by = models.CharField(
  136. max_length=255,
  137. choices=index_order_by_choices,
  138. default=index_order_by_default,
  139. verbose_name=_('Order child pages by'),
  140. blank=True,
  141. )
  142. index_num_per_page = models.PositiveIntegerField(
  143. default=10,
  144. verbose_name=_('Number per page'),
  145. )
  146. tags = ClusterTaggableManager(
  147. through=CoderedTag,
  148. verbose_name='Tags',
  149. blank=True,
  150. help_text=_('Used to categorize your pages.')
  151. )
  152. ###############
  153. # Layout fields
  154. ###############
  155. custom_template = models.CharField(
  156. blank=True,
  157. max_length=255,
  158. choices=None,
  159. verbose_name=_('Template')
  160. )
  161. ###############
  162. # SEO fields
  163. ###############
  164. og_image = models.ForeignKey(
  165. 'wagtailimages.Image',
  166. null=True,
  167. blank=True,
  168. on_delete=models.SET_NULL,
  169. related_name='+',
  170. verbose_name=_('Open Graph preview image'),
  171. help_text=_('The image shown when linking to this page on social media. If blank, defaults to article cover image, or logo in Settings > Layout > Logo')
  172. )
  173. struct_org_type = models.CharField(
  174. default='',
  175. blank=True,
  176. max_length=255,
  177. choices=schema.SCHEMA_ORG_CHOICES,
  178. verbose_name=_('Organization type'),
  179. help_text=_('If blank, no structured data will be used on this page.')
  180. )
  181. struct_org_name = models.CharField(
  182. default='',
  183. blank=True,
  184. max_length=255,
  185. verbose_name=_('Organization name'),
  186. help_text=_('Leave blank to use the site name in Settings > Sites')
  187. )
  188. struct_org_logo = models.ForeignKey(
  189. 'wagtailimages.Image',
  190. null=True,
  191. blank=True,
  192. on_delete=models.SET_NULL,
  193. related_name='+',
  194. verbose_name=_('Organization logo'),
  195. help_text=_('Leave blank to use the logo in Settings > Layout > Logo')
  196. )
  197. struct_org_image = models.ForeignKey(
  198. 'wagtailimages.Image',
  199. null=True,
  200. blank=True,
  201. on_delete=models.SET_NULL,
  202. related_name='+',
  203. verbose_name=_('Photo of Organization'),
  204. help_text=_('A photo of the facility. This photo will be cropped to 1:1, 4:3, and 16:9 aspect ratios automatically.')
  205. )
  206. struct_org_phone = models.CharField(
  207. blank=True,
  208. max_length=255,
  209. verbose_name=_('Telephone number'),
  210. help_text=_('Include country code for best results. For example: +1-216-555-8000')
  211. )
  212. struct_org_address_street = models.CharField(
  213. blank=True,
  214. max_length=255,
  215. verbose_name=_('Street address'),
  216. help_text=_('House number and street. For example, 55 Public Square Suite 1710')
  217. )
  218. struct_org_address_locality = models.CharField(
  219. blank=True,
  220. max_length=255,
  221. verbose_name=_('City'),
  222. help_text=_('City or locality. For example, Cleveland')
  223. )
  224. struct_org_address_region = models.CharField(
  225. blank=True,
  226. max_length=255,
  227. verbose_name=_('State'),
  228. help_text=_('State, province, county, or region. For example, OH')
  229. )
  230. struct_org_address_postal = models.CharField(
  231. blank=True,
  232. max_length=255,
  233. verbose_name=_('Postal code'),
  234. help_text=_('Zip or postal code. For example, 44113')
  235. )
  236. struct_org_address_country = models.CharField(
  237. blank=True,
  238. max_length=255,
  239. verbose_name=_('Country'),
  240. help_text=_('For example, USA. Two-letter ISO 3166-1 alpha-2 country code is also acceptible https://en.wikipedia.org/wiki/ISO_3166-1')
  241. )
  242. struct_org_geo_lat = models.DecimalField(
  243. blank=True,
  244. null=True,
  245. max_digits=10,
  246. decimal_places=8,
  247. verbose_name=_('Geographic latitude')
  248. )
  249. struct_org_geo_lng = models.DecimalField(
  250. blank=True,
  251. null=True,
  252. max_digits=10,
  253. decimal_places=8,
  254. verbose_name=_('Geographic longitude')
  255. )
  256. struct_org_hours = StreamField(
  257. [
  258. ('hours', OpenHoursBlock()),
  259. ],
  260. blank=True,
  261. verbose_name=_('Hours of operation')
  262. )
  263. struct_org_actions = StreamField(
  264. [
  265. ('actions', StructuredDataActionBlock())
  266. ],
  267. blank=True,
  268. verbose_name=_('Actions')
  269. )
  270. struct_org_extra_json = models.TextField(
  271. blank=True,
  272. verbose_name=_('Additional Organization markup'),
  273. help_text=_('Additional JSON-LD inserted into the Organization dictionary. Must be properties of https://schema.org/Organization or the selected organization type.')
  274. )
  275. ###############
  276. # Settings
  277. ###############
  278. content_walls = StreamField(
  279. [
  280. ('content_wall', ContentWallBlock())
  281. ],
  282. blank=True,
  283. verbose_name=_('Content Walls')
  284. )
  285. ###############
  286. # Search
  287. ###############
  288. search_fields = [
  289. index.SearchField('title', partial_match=True, boost=3),
  290. index.SearchField('seo_title', partial_match=True, boost=3),
  291. index.SearchField('search_description', boost=2),
  292. index.FilterField('title'),
  293. index.FilterField('id'),
  294. index.FilterField('live'),
  295. index.FilterField('owner'),
  296. index.FilterField('content_type'),
  297. index.FilterField('path'),
  298. index.FilterField('depth'),
  299. index.FilterField('locked'),
  300. index.FilterField('first_published_at'),
  301. index.FilterField('last_published_at'),
  302. index.FilterField('latest_revision_created_at'),
  303. index.FilterField('index_show_subpages'),
  304. index.FilterField('index_order_by'),
  305. index.FilterField('custom_template'),
  306. ]
  307. ###############
  308. # Panels
  309. ###############
  310. content_panels = Page.content_panels + [
  311. ImageChooserPanel('cover_image'),
  312. ]
  313. body_content_panels = []
  314. bottom_content_panels = [
  315. FieldPanel('tags'),
  316. ]
  317. layout_panels = [
  318. MultiFieldPanel(
  319. [
  320. FieldPanel('custom_template')
  321. ],
  322. heading=_('Visual Design')
  323. ),
  324. MultiFieldPanel(
  325. [
  326. FieldPanel('index_show_subpages'),
  327. FieldPanel('index_num_per_page'),
  328. FieldPanel('index_order_by'),
  329. ],
  330. heading=_('Show Child Pages')
  331. )
  332. ]
  333. promote_panels = [
  334. MultiFieldPanel(
  335. [
  336. FieldPanel('slug'),
  337. FieldPanel('seo_title'),
  338. FieldPanel('search_description'),
  339. ImageChooserPanel('og_image'),
  340. ],
  341. _('Page Meta Data')
  342. ),
  343. MultiFieldPanel(
  344. [
  345. HelpPanel(
  346. heading=_('About Organization Structured Data'),
  347. content=_("""The fields below help define brand, contact, and storefront
  348. information to search engines. This information should be filled out on
  349. the site’s root page (Home Page). If your organization has multiple locations,
  350. then also fill this info out on each location page using that particular
  351. location’s info."""),
  352. ),
  353. FieldPanel('struct_org_type'),
  354. FieldPanel('struct_org_name'),
  355. ImageChooserPanel('struct_org_logo'),
  356. ImageChooserPanel('struct_org_image'),
  357. FieldPanel('struct_org_phone'),
  358. FieldPanel('struct_org_address_street'),
  359. FieldPanel('struct_org_address_locality'),
  360. FieldPanel('struct_org_address_region'),
  361. FieldPanel('struct_org_address_postal'),
  362. FieldPanel('struct_org_address_country'),
  363. FieldPanel('struct_org_geo_lat'),
  364. FieldPanel('struct_org_geo_lng'),
  365. StreamFieldPanel('struct_org_hours'),
  366. StreamFieldPanel('struct_org_actions'),
  367. FieldPanel('struct_org_extra_json'),
  368. ],
  369. _('Structured Data - Organization')
  370. ),
  371. ]
  372. settings_panels = Page.settings_panels + [
  373. StreamFieldPanel('content_walls'),
  374. ]
  375. integration_panels = []
  376. def __init__(self, *args, **kwargs):
  377. """
  378. Inject custom choices and defalts into the form fields
  379. to enable customization by subclasses.
  380. """
  381. super().__init__(*args, **kwargs)
  382. klassname = self.__class__.__name__.lower()
  383. template_choices = cr_settings['FRONTEND_TEMPLATES_PAGES'].get('*', ()) + \
  384. cr_settings['FRONTEND_TEMPLATES_PAGES'].get(klassname, ())
  385. self._meta.get_field('index_order_by').choices = self.index_order_by_choices
  386. self._meta.get_field('custom_template').choices = template_choices
  387. if not self.id:
  388. self.index_order_by = self.index_order_by_default
  389. self.index_show_subpages = self.index_show_subpages_default
  390. @classmethod
  391. def get_edit_handler(cls):
  392. """
  393. Override to "lazy load" the panels overriden by subclasses.
  394. """
  395. panels = [
  396. ObjectList(cls.content_panels + cls.body_content_panels + cls.bottom_content_panels, heading='Content'),
  397. ObjectList(cls.layout_panels, heading='Layout'),
  398. ObjectList(cls.promote_panels, heading='SEO', classname="seo"),
  399. ObjectList(cls.settings_panels, heading='Settings', classname="settings"),
  400. ]
  401. if cls.integration_panels:
  402. panels.append(ObjectList(cls.integration_panels, heading='Integrations', classname='integrations'))
  403. return TabbedInterface(panels).bind_to_model(cls)
  404. def get_struct_org_name(self):
  405. """
  406. Gets org name for sturctured data using a fallback.
  407. """
  408. if self.struct_org_name:
  409. return self.struct_org_name
  410. return self.get_site().site_name
  411. def get_struct_org_logo(self):
  412. """
  413. Gets logo for structured data using a fallback.
  414. """
  415. if self.struct_org_logo:
  416. return self.struct_org_logo
  417. else:
  418. layout_settings = LayoutSettings.for_site(self.get_site())
  419. if layout_settings.logo:
  420. return layout_settings.logo
  421. return None
  422. def get_template(self, request, *args, **kwargs):
  423. """
  424. Override parent to serve different templates based on querystring.
  425. """
  426. if 'amp' in request.GET and hasattr(self, 'amp_template'):
  427. seo_settings = SeoSettings.for_site(request.site)
  428. if seo_settings.amp_pages:
  429. if request.is_ajax():
  430. return self.ajax_template or self.amp_template
  431. return self.amp_template
  432. if self.custom_template:
  433. return self.custom_template
  434. return super(CoderedPage, self).get_template(request, args, kwargs)
  435. def get_index_children(self):
  436. """
  437. Override to return query of subpages as defined by `index_` variables.
  438. """
  439. if self.index_query_pagemodel:
  440. querymodel = resolve_model_string(self.index_query_pagemodel, self._meta.app_label)
  441. query = querymodel.objects.child_of(self).live()
  442. else:
  443. query = super().get_children().live()
  444. if self.index_order_by:
  445. return query.order_by(self.index_order_by)
  446. return query
  447. def get_content_walls(self, check_child_setting=True):
  448. current_content_walls = []
  449. if check_child_setting:
  450. for wall in self.content_walls:
  451. content_wall = wall.value
  452. if wall.value['show_content_wall_on_children']:
  453. current_content_walls.append(wall.value)
  454. else:
  455. current_content_walls = self.content_walls
  456. try:
  457. return list(current_content_walls) + self.get_parent().specific.get_content_walls()
  458. except AttributeError:
  459. return list(current_content_walls)
  460. def get_context(self, request, *args, **kwargs):
  461. """
  462. Add child pages and paginated child pages to context.
  463. """
  464. context = super().get_context(request)
  465. if self.index_show_subpages:
  466. all_children = self.get_index_children()
  467. paginator = Paginator(all_children, self.index_num_per_page)
  468. page = request.GET.get('p', 1)
  469. try:
  470. paged_children = paginator.page(page)
  471. except:
  472. paged_children = paginator.page(1)
  473. context['index_paginated'] = paged_children
  474. context['index_children'] = all_children
  475. context['content_walls'] = self.get_content_walls(check_child_setting=False)
  476. return context
  477. ###############################################################################
  478. # Abstract pages providing pre-built common website functionality, suitable for subclassing.
  479. # These are abstract so subclasses can override fields if desired.
  480. ###############################################################################
  481. class CoderedWebPage(CoderedPage):
  482. """
  483. Provides a body and body-related functionality.
  484. This is abstract so that subclasses can override the body StreamField.
  485. """
  486. class Meta:
  487. verbose_name = _('CodeRed Web Page')
  488. abstract = True
  489. template = 'coderedcms/pages/web_page.html'
  490. # Child pages should override based on what blocks they want in the body.
  491. # Default is LAYOUT_STREAMBLOCKS which is the fullest editor experience.
  492. body = StreamField(LAYOUT_STREAMBLOCKS, null=True, blank=True)
  493. # Search fields
  494. search_fields = (
  495. CoderedPage.search_fields +
  496. [index.SearchField('body')]
  497. )
  498. # Panels
  499. body_content_panels = [
  500. StreamFieldPanel('body'),
  501. ]
  502. @property
  503. def body_preview(self):
  504. """
  505. A shortened, non-HTML version of the body.
  506. """
  507. # add spaces between tags for legibility
  508. body = str(self.body).replace('>', '> ')
  509. # strip tags
  510. body = strip_tags(body)
  511. # truncate and add ellipses
  512. return body[:200] + "..." if len(body) > 200 else body
  513. @property
  514. def page_ptr(self):
  515. """
  516. Overwrite of `page_ptr` to make it compatible with wagtailimportexport.
  517. """
  518. return self.base_page_ptr
  519. @page_ptr.setter
  520. def page_ptr(self, value):
  521. self.base_page_ptr = value
  522. class CoderedArticlePage(CoderedWebPage):
  523. """
  524. Article, suitable for news or blog content.
  525. """
  526. class Meta:
  527. verbose_name = _('CodeRed Article')
  528. abstract = True
  529. template = 'coderedcms/pages/article_page.html'
  530. amp_template = 'coderedcms/pages/article_page.amp.html'
  531. # Override body to provide simpler content
  532. body = StreamField(CONTENT_STREAMBLOCKS, null=True, blank=True)
  533. caption = models.CharField(
  534. max_length=255,
  535. blank=True,
  536. verbose_name=_('Caption'),
  537. )
  538. author = models.ForeignKey(
  539. settings.AUTH_USER_MODEL,
  540. null=True,
  541. blank=True,
  542. editable=True,
  543. on_delete=models.SET_NULL,
  544. verbose_name=_('Author'),
  545. )
  546. author_display = models.CharField(
  547. max_length=255,
  548. blank=True,
  549. verbose_name=_('Display author as'),
  550. help_text=_('Override how the author’s name displays on this article.'),
  551. )
  552. date_display = models.DateField(
  553. null=True,
  554. blank=True,
  555. verbose_name=_('Display publish date'),
  556. )
  557. def get_author_name(self):
  558. """
  559. Gets author name using a fallback.
  560. """
  561. if self.author_display:
  562. return self.author_display
  563. if self.author:
  564. return self.author.get_full_name()
  565. return ''
  566. def get_pub_date(self):
  567. """
  568. Gets published date.
  569. """
  570. if self.date_display:
  571. return self.date_display
  572. return ''
  573. def get_description(self):
  574. """
  575. Gets the description using a fallback.
  576. """
  577. if self.search_description:
  578. return self.search_description
  579. if self.caption:
  580. return self.caption
  581. if self.body_preview:
  582. return self.body_preview
  583. return ''
  584. search_fields = (
  585. CoderedWebPage.search_fields +
  586. [
  587. index.SearchField('caption', boost=2),
  588. index.FilterField('author'),
  589. index.FilterField('author_display'),
  590. index.FilterField('date_display'),
  591. ]
  592. )
  593. content_panels = CoderedWebPage.content_panels + [
  594. FieldPanel('caption'),
  595. MultiFieldPanel(
  596. [
  597. FieldPanel('author'),
  598. FieldPanel('author_display'),
  599. FieldPanel('date_display'),
  600. ],
  601. _('Publication Info')
  602. )
  603. ]
  604. class CoderedArticleIndexPage(CoderedWebPage):
  605. """
  606. Shows a list of article sub-pages.
  607. """
  608. class Meta:
  609. verbose_name = _('CodeRed Article Index Page')
  610. abstract = True
  611. template = 'coderedcms/pages/article_index_page.html'
  612. index_show_subpages_default = True
  613. index_order_by_default = '-date_display'
  614. index_order_by_choices = (('-date_display', 'Display publish date, newest first'),) + \
  615. CoderedWebPage.index_order_by_choices
  616. show_images = models.BooleanField(
  617. default=True,
  618. verbose_name=_('Show images'),
  619. )
  620. show_captions = models.BooleanField(
  621. default=True,
  622. )
  623. show_meta = models.BooleanField(
  624. default=True,
  625. verbose_name=_('Show author and date info'),
  626. )
  627. show_preview_text = models.BooleanField(
  628. default=True,
  629. verbose_name=_('Show preview text'),
  630. )
  631. layout_panels = CoderedWebPage.layout_panels + [
  632. MultiFieldPanel(
  633. [
  634. FieldPanel('show_images'),
  635. FieldPanel('show_captions'),
  636. FieldPanel('show_meta'),
  637. FieldPanel('show_preview_text'),
  638. ],
  639. heading=_('Child page display')
  640. ),
  641. ]
  642. class CoderedEventPage(CoderedWebPage, BaseEvent):
  643. class Meta:
  644. verbose_name = _('CodeRed Event')
  645. abstract = True
  646. calendar_color = ColorField(
  647. blank=True,
  648. help_text=_('The color that the event will use when displayed on a calendar.'),
  649. )
  650. address = models.TextField(
  651. blank=True,
  652. verbose_name=_("Address")
  653. )
  654. content_panels = CoderedWebPage.content_panels + [
  655. MultiFieldPanel(
  656. [
  657. FieldPanel('calendar_color'),
  658. FieldPanel('address'),
  659. ],
  660. heading=_('Event information')
  661. ),
  662. InlinePanel(
  663. 'occurrences',
  664. min_num=1,
  665. heading=_("Dates and times"),
  666. ),
  667. ]
  668. @property
  669. def upcoming_occurrences(self):
  670. """
  671. Returns the next x occurrences for this event.
  672. By default, it returns 10.
  673. """
  674. return self.query_occurrences(num_of_instances_to_return=10)
  675. @property
  676. def most_recent_occurrence(self):
  677. """
  678. Gets the next upcoming, or last occurrence if the event has no more occurrences.
  679. """
  680. try:
  681. noc = self.next_occurrence()
  682. if noc:
  683. return noc
  684. aoc = []
  685. for occurrence in self.occurrences.all():
  686. aoc += [instance for instance in occurrence.all_occurrences()]
  687. if len(aoc) > 0:
  688. return aoc[-1] # last one in the list
  689. except AttributeError:
  690. # Triggers when a preview is initiated on an EventPage because it uses a FakeQuerySet object.
  691. # Here we manually compute the next_occurrence
  692. occurrences = [e.next_occurrence() for e in self.occurrences.all()]
  693. if occurrences:
  694. return sorted(occurrences, key=lambda tup: tup[0])[0]
  695. def query_occurrences(self, num_of_instances_to_return=None, **kwargs):
  696. """
  697. Returns a list of all upcoming event instances for the specified query.
  698. For more information on what you can query with, visit
  699. https://github.com/gregplaysguitar/django-eventtools
  700. """
  701. event_instances = []
  702. occurrence_kwargs = {
  703. 'from_date': kwargs.get('from_date', timezone.now().date())
  704. }
  705. if 'limit' in kwargs:
  706. if kwargs['limit'] != None:
  707. # Limit the number of event instances that will be generated per occurrence rule to 10, if not otherwise specified.
  708. occurrence_kwargs['limit'] = kwargs.get('limit', 10)
  709. # For each occurrence rule in all of the occurrence rules for this event.
  710. for occurrence in self.occurrences.all():
  711. # Add the qualifying generated event instances to the list.
  712. event_instances += [instance for instance in occurrence.all_occurrences(**occurrence_kwargs)]
  713. # Sort all the events by the date that they start
  714. event_instances.sort(key=lambda d: d[0])
  715. # Return the event instances, possibly spliced if num_instances_to_return is set.
  716. return event_instances[:num_of_instances_to_return] if num_of_instances_to_return else event_instances
  717. def convert_to_ical_format(self, dt_start=None, dt_end=None, occurrence=None):
  718. ical_event = ICalEvent()
  719. ical_event.add('summary', self.title)
  720. if self.address:
  721. ical_event.add('location', self.address)
  722. if dt_start:
  723. ical_event.add('dtstart', dt_start)
  724. if dt_end:
  725. ical_event.add('dtend', dt_end)
  726. if occurrence:
  727. freq = occurrence.repeat.split(":")[1] if occurrence.repeat else None
  728. repeat_until = occurrence.repeat_until.strftime("%Y%m%dT000000Z") if occurrence.repeat_until else None
  729. ical_event.add('dtstart', occurrence.start)
  730. if occurrence.end:
  731. ical_event.add('dtend', occurrence.end)
  732. if freq:
  733. ical_event.add('RRULE', freq, encode=False)
  734. if repeat_until:
  735. ical_event.add('until', repeat_until)
  736. return ical_event
  737. def create_single_ical(self, dt_start, dt_end=None):
  738. return self.convert_to_ical_format(dt_start=dt_start, dt_end=dt_end)
  739. def create_recurring_ical(self):
  740. events = []
  741. for occurrence in self.occurrences.all():
  742. events.append(self.convert_to_ical_format(occurrence=occurrence))
  743. return events
  744. class DefaultCalendarViewChoices():
  745. MONTH = 'month'
  746. AGENDA_WEEK = 'agendaWeek'
  747. AGENDA_DAY = 'agendaDay'
  748. LIST_MONTH = 'listMonth'
  749. CHOICES = (
  750. ('', _('No calendar')),
  751. (MONTH, _('Monthly Calendar')),
  752. (AGENDA_WEEK, _('Weekly Calendar')),
  753. (AGENDA_DAY, _('Daily Calendar')),
  754. (LIST_MONTH, _('Calendar List View')),
  755. )
  756. class CoderedEventIndexPage(CoderedWebPage):
  757. """
  758. Shows a list of event sub-pages.
  759. """
  760. class Meta:
  761. verbose_name = _('CodeRed Event Index Page')
  762. abstract = True
  763. template = 'coderedcms/pages/event_index_page.html'
  764. index_show_subpages_default = True
  765. index_order_by_default = 'next_occurrence'
  766. index_order_by_choices = (
  767. ('next_occurrence', 'Display next occurrence, soonest first'),
  768. ) + \
  769. CoderedWebPage.index_order_by_choices
  770. default_calendar_view = models.CharField(
  771. blank=True,
  772. choices=DefaultCalendarViewChoices.CHOICES,
  773. max_length=255,
  774. verbose_name=_('Calendar Style'),
  775. help_text=_('The default look of the calendar on this page.')
  776. )
  777. layout_panels = CoderedWebPage.layout_panels + [
  778. FieldPanel('default_calendar_view'),
  779. ]
  780. def get_index_children(self):
  781. if self.index_query_pagemodel and self.index_order_by == 'next_occurrence':
  782. querymodel = resolve_model_string(self.index_query_pagemodel, self._meta.app_label)
  783. qs = querymodel.objects.child_of(self).live()
  784. # filter out events that don't have a next_occurrence
  785. upcoming = []
  786. for event in qs.all():
  787. if event.next_occurrence():
  788. upcoming.append(event)
  789. # sort the events by next_occurrence
  790. return sorted(upcoming, key=lambda e: e.next_occurrence())
  791. return super().get_index_children()
  792. def get_calendar_events(self, start, end):
  793. # start with all child events, regardless of get_index_children rules.
  794. querymodel = resolve_model_string(self.index_query_pagemodel, self._meta.app_label)
  795. qs = querymodel.objects.child_of(self).live()
  796. event_instances = []
  797. for event in qs:
  798. occurrences = event.query_occurrences(limit=None, from_date=start, to_date=end)
  799. for occurrence in occurrences:
  800. event_data = {
  801. 'title': event.title,
  802. 'start': occurrence[0].strftime('%Y-%m-%dT%H:%M:%S'),
  803. 'end' : occurrence[1].strftime('%Y-%m-%dT%H:%M:%S') if occurrence[1] else "",
  804. 'description': "",
  805. }
  806. if event.url:
  807. event_data['url'] = event.url
  808. if event.calendar_color:
  809. event_data['backgroundColor'] = event.calendar_color
  810. event_instances.append(event_data)
  811. return event_instances
  812. class CoderedEventOccurrence(Orderable, BaseOccurrence):
  813. class Meta:
  814. verbose_name = _('CodeRed Event Occurrence')
  815. abstract = True
  816. class CoderedFormPage(CoderedWebPage):
  817. """
  818. This is basically a clone of wagtail.contrib.forms.models.AbstractForm
  819. with changes in functionality and extending CoderedWebPage vs wagtailcore.Page.
  820. """
  821. class Meta:
  822. verbose_name = _('CodeRed Form Page')
  823. abstract = True
  824. template = 'coderedcms/pages/form_page.html'
  825. landing_page_template = 'coderedcms/pages/form_page_landing.html'
  826. base_form_class = WagtailAdminFormPageForm
  827. form_builder = CoderedFormBuilder
  828. submissions_list_view_class = CoderedSubmissionsListView
  829. ### Custom codered fields
  830. to_address = models.CharField(
  831. max_length=255,
  832. blank=True,
  833. verbose_name=_('Email form submissions to'),
  834. help_text=_('Optional - email form submissions to this address. Separate multiple addresses by comma.')
  835. )
  836. subject = models.CharField(
  837. max_length=255,
  838. blank=True,
  839. verbose_name=_('Subject'),
  840. )
  841. save_to_database = models.BooleanField(
  842. default=True,
  843. verbose_name=_('Save form submissions'),
  844. help_text=_('Submissions are saved to database and can be exported at any time.')
  845. )
  846. thank_you_page = models.ForeignKey(
  847. 'wagtailcore.Page',
  848. null=True,
  849. blank=True,
  850. on_delete=models.SET_NULL,
  851. related_name='+',
  852. verbose_name=_('Thank you page'),
  853. help_text=_('The page users are redirected to after submitting the form.'),
  854. )
  855. button_text = models.CharField(
  856. max_length=255,
  857. default=_('Submit'),
  858. verbose_name=_('Button text'),
  859. )
  860. button_style = models.CharField(
  861. blank=True,
  862. choices=cr_settings['FRONTEND_BTN_STYLE_CHOICES'],
  863. default=cr_settings["FRONTEND_BTN_STYLE_DEFAULT"],
  864. max_length=255,
  865. verbose_name=_('Button style'),
  866. )
  867. button_size = models.CharField(
  868. blank=True,
  869. choices=cr_settings['FRONTEND_BTN_SIZE_CHOICES'],
  870. default=cr_settings["FRONTEND_BTN_SIZE_DEFAULT"],
  871. max_length=255,
  872. verbose_name=_('Button Size'),
  873. )
  874. button_css_class = models.CharField(
  875. max_length=255,
  876. blank=True,
  877. verbose_name=_('Button CSS class'),
  878. help_text=_('Custom CSS class applied to the submit button.'),
  879. )
  880. form_css_class = models.CharField(
  881. max_length=255,
  882. blank=True,
  883. verbose_name=_('Form CSS Class'),
  884. help_text=_('Custom CSS class applied to <form> element.'),
  885. )
  886. form_id = models.CharField(
  887. max_length=255,
  888. blank=True,
  889. verbose_name=_('Form ID'),
  890. help_text=_('Custom ID applied to <form> element.'),
  891. )
  892. form_golive_at = models.DateTimeField(
  893. blank=True,
  894. null=True,
  895. verbose_name=_('Form go live date/time'),
  896. help_text=_('Date and time when the FORM goes live on the page.'),
  897. )
  898. form_expire_at = models.DateTimeField(
  899. blank=True,
  900. null=True,
  901. verbose_name=_('Form expiry date/time'),
  902. help_text=_('Date and time when the FORM will no longer be available on the page.'),
  903. )
  904. body_content_panels = CoderedWebPage.body_content_panels + [
  905. FormSubmissionsPanel(),
  906. InlinePanel('form_fields', label="Form fields"),
  907. MultiFieldPanel(
  908. [
  909. PageChooserPanel('thank_you_page'),
  910. FieldPanel('button_text'),
  911. FieldPanel('button_style'),
  912. FieldPanel('button_size'),
  913. FieldPanel('button_css_class'),
  914. FieldPanel('form_css_class'),
  915. FieldPanel('form_id'),
  916. ],
  917. _('Form Settings')
  918. ),
  919. MultiFieldPanel(
  920. [
  921. FieldPanel('save_to_database'),
  922. FieldPanel('to_address'),
  923. FieldPanel('subject'),
  924. ],
  925. _('Form Submissions')
  926. ),
  927. InlinePanel('confirmation_emails', label=_('Confirmation Emails'))
  928. ]
  929. settings_panels = CoderedPage.settings_panels + [
  930. MultiFieldPanel(
  931. [
  932. FieldRowPanel(
  933. [
  934. FieldPanel('form_golive_at'),
  935. FieldPanel('form_expire_at'),
  936. ],
  937. classname='label-above',
  938. ),
  939. ],
  940. _('Form Scheduled Publishing'),
  941. )
  942. ]
  943. @property
  944. def form_live(self):
  945. """
  946. A boolean on whether or not the <form> element should be shown on the page.
  947. """
  948. return (self.form_golive_at is None or self.form_golive_at <= timezone.now()) and \
  949. (self.form_expire_at is None or self.form_expire_at >= timezone.now())
  950. def __init__(self, *args, **kwargs):
  951. super().__init__(*args, **kwargs)
  952. if not hasattr(self, 'landing_page_template'):
  953. name, ext = os.path.splitext(self.template)
  954. self.landing_page_template = name + '_landing' + ext
  955. def get_form_fields(self):
  956. """
  957. Form page expects `form_fields` to be declared.
  958. If you want to change backwards relation name,
  959. you need to override this method.
  960. """
  961. return self.form_fields.all()
  962. def get_data_fields(self):
  963. """
  964. Returns a list of tuples with (field_name, field_label).
  965. """
  966. data_fields = [
  967. ('submit_time', _('Submission date')),
  968. ]
  969. data_fields += [
  970. (field.clean_name, field.label)
  971. for field in self.get_form_fields()
  972. ]
  973. return data_fields
  974. def get_form_class(self):
  975. fb = self.form_builder(self.get_form_fields())
  976. return fb.get_form_class()
  977. def get_form_parameters(self):
  978. return {}
  979. def get_form(self, *args, **kwargs):
  980. form_class = self.get_form_class()
  981. form_params = self.get_form_parameters()
  982. form_params.update(kwargs)
  983. return form_class(*args, **form_params)
  984. def get_landing_page_template(self, request, *args, **kwargs):
  985. return self.landing_page_template
  986. def get_submission_class(self):
  987. """
  988. Returns submission class.
  989. You can override this method to provide custom submission class.
  990. Your class must be inherited from AbstractFormSubmission.
  991. """
  992. return FormSubmission
  993. def process_form_submission(self, request, form):
  994. """
  995. Accepts form instance with submitted data, user and page.
  996. Creates submission instance.
  997. You can override this method if you want to have custom creation logic.
  998. For example, if you want to save reference to a user.
  999. """
  1000. processed_data = {}
  1001. # Handle file uploads
  1002. for key, val in form.cleaned_data.items():
  1003. if type(val) == InMemoryUploadedFile or type(val) == TemporaryUploadedFile:
  1004. # Save the file and get its URL
  1005. file_system = FileSystemStorage(
  1006. location=cr_settings['PROTECTED_MEDIA_ROOT'],
  1007. base_url=cr_settings['PROTECTED_MEDIA_URL']
  1008. )
  1009. filename = file_system.save(file_system.get_valid_name(val.name), val)
  1010. processed_data[key] = file_system.url(filename)
  1011. else:
  1012. processed_data[key] = val
  1013. # Get submission
  1014. form_submission = self.get_submission_class()(
  1015. form_data=json.dumps(processed_data, cls=DjangoJSONEncoder),
  1016. page=self,
  1017. )
  1018. # Save to database
  1019. if self.save_to_database:
  1020. form_submission.save()
  1021. # Send the mails
  1022. if self.to_address:
  1023. self.send_summary_mail(request, form, processed_data)
  1024. if self.confirmation_emails:
  1025. for email in self.confirmation_emails.all():
  1026. from_address = email.from_address
  1027. if from_address == '':
  1028. from_address = GeneralSettings.for_site(request.site).from_email_address
  1029. template_body = Template(email.body)
  1030. template_to = Template(email.to_address)
  1031. template_from_email = Template(from_address)
  1032. template_cc = Template(email.cc_address)
  1033. template_bcc = Template(email.bcc_address)
  1034. template_subject = Template(email.subject)
  1035. context = Context(self.data_to_dict(processed_data))
  1036. message = EmailMessage(
  1037. body=template_body.render(context),
  1038. to=template_to.render(context).split(','),
  1039. from_email=template_from_email.render(context),
  1040. cc=template_cc.render(context).split(','),
  1041. bcc=template_bcc.render(context).split(','),
  1042. subject=template_subject.render(context),
  1043. )
  1044. message.content_subtype = 'html'
  1045. message.send()
  1046. for fn in hooks.get_hooks('form_page_submit'):
  1047. fn(instance=self, form_submission=form_submission)
  1048. return processed_data
  1049. def send_summary_mail(self, request, form, processed_data):
  1050. """
  1051. Sends a form submission summary email.
  1052. """
  1053. addresses = [x.strip() for x in self.to_address.split(',')]
  1054. content = []
  1055. for field in form:
  1056. value = processed_data[field.name]
  1057. if isinstance(value, list):
  1058. value = ', '.join(value)
  1059. content.append('{0}: {1}'.format(field.label, utils.attempt_protected_media_value_conversion(request, value)))
  1060. content = '\n'.join(content)
  1061. send_mail(
  1062. self.subject,
  1063. content,
  1064. GeneralSettings.for_site(Site.objects.get(is_default_site=True)).from_email_address,
  1065. addresses
  1066. )
  1067. def data_to_dict(self, processed_data):
  1068. """
  1069. Converts processed form data into a dictionary suitable
  1070. for rendering in a context.
  1071. """
  1072. dictionary = {}
  1073. for key, value in processed_data.items():
  1074. dictionary[key.replace('-', '_')] = value
  1075. if isinstance(value, list):
  1076. dictionary[key] = ', '.join(value)
  1077. return dictionary
  1078. def render_landing_page(self, request, *args, form_submission=None, **kwargs):
  1079. """
  1080. Renders the landing page.
  1081. You can override this method to return a different HttpResponse as
  1082. landing page. E.g. you could return a redirect to a separate page.
  1083. """
  1084. if self.thank_you_page:
  1085. return redirect(self.thank_you_page.url)
  1086. context = self.get_context(request)
  1087. context['form_submission'] = form_submission
  1088. response = render(
  1089. request,
  1090. self.get_landing_page_template(request),
  1091. context
  1092. )
  1093. return response
  1094. def serve_submissions_list_view(self, request, *args, **kwargs):
  1095. """
  1096. Returns list submissions view for admin.
  1097. `list_submissions_view_class` can bse set to provide custom view class.
  1098. Your class must be inherited from SubmissionsListView.
  1099. """
  1100. view = self.submissions_list_view_class.as_view()
  1101. return view(request, form_page=self, *args, **kwargs)
  1102. def serve(self, request, *args, **kwargs):
  1103. if request.method == 'POST':
  1104. form = self.get_form(request.POST, request.FILES, page=self, user=request.user)
  1105. if form.is_valid():
  1106. form_submission = self.process_form_submission(request, form)
  1107. return self.render_landing_page(request, form_submission, *args, **kwargs)
  1108. else:
  1109. form = self.get_form(page=self, user=request.user)
  1110. context = self.get_context(request)
  1111. context['form'] = form
  1112. response = render(
  1113. request,
  1114. self.get_template(request),
  1115. context
  1116. )
  1117. return response
  1118. preview_modes = [
  1119. ('form', _('Form')),
  1120. ('landing', _('Thank you page')),
  1121. ]
  1122. def serve_preview(self, request, mode):
  1123. if mode == 'landing':
  1124. request.is_preview = True
  1125. return self.render_landing_page(request)
  1126. return super().serve_preview(request, mode)
  1127. class CoderedLocationPage(CoderedWebPage):
  1128. """
  1129. Location, suitable for store locations or help centers.
  1130. """
  1131. class Meta:
  1132. verbose_name = _('CodeRed Location')
  1133. abstract = True
  1134. template = 'coderedcms/pages/location_page.html'
  1135. # Override body to provide simpler content
  1136. body = StreamField(CONTENT_STREAMBLOCKS, null=True, blank=True)
  1137. address = models.TextField(
  1138. blank=True,
  1139. verbose_name=_("Address")
  1140. )
  1141. latitude = models.FloatField(
  1142. blank=True,
  1143. null=True,
  1144. verbose_name=_("Latitude")
  1145. )
  1146. longitude = models.FloatField(
  1147. blank=True,
  1148. null=True,
  1149. verbose_name=_("Longitude")
  1150. )
  1151. auto_update_latlng = models.BooleanField(
  1152. default=True,
  1153. verbose_name=_("Auto Update Latitude and Longitude"),
  1154. help_text=_("If checked, automatically update the latitude and longitude when the address is updated.")
  1155. )
  1156. map_title = models.CharField(
  1157. blank=True,
  1158. max_length=255,
  1159. verbose_name=_("Map Title"),
  1160. help_text=_("If this is filled out, this is the title that will be used on the map.")
  1161. )
  1162. map_description = models.CharField(
  1163. blank=True,
  1164. max_length=255,
  1165. verbose_name=_("Map Description"),
  1166. help_text=_("If this is filled out, this is the description that will be used on the map.")
  1167. )
  1168. website = models.TextField(
  1169. blank=True,
  1170. verbose_name=_("Website")
  1171. )
  1172. phone_number = models.CharField(
  1173. blank=True,
  1174. max_length=255,
  1175. verbose_name=_("Phone Number")
  1176. )
  1177. content_panels = CoderedWebPage.content_panels + [
  1178. FieldPanel('address'),
  1179. FieldPanel('website'),
  1180. FieldPanel('phone_number'),
  1181. ]
  1182. layout_panels = CoderedWebPage.layout_panels + [
  1183. MultiFieldPanel(
  1184. [
  1185. FieldPanel('map_title'),
  1186. FieldPanel('map_description'),
  1187. ],
  1188. heading=_('Map Layout')
  1189. ),
  1190. ]
  1191. settings_panels = CoderedWebPage.settings_panels + [
  1192. MultiFieldPanel(
  1193. [
  1194. FieldPanel('auto_update_latlng'),
  1195. FieldPanel('latitude'),
  1196. FieldPanel('longitude'),
  1197. ],
  1198. heading=_("Location Settings")
  1199. ),
  1200. ]
  1201. @property
  1202. def geojson_name(self):
  1203. return self.map_title or self.title
  1204. @property
  1205. def geojson_description(self):
  1206. return self.map_description
  1207. @property
  1208. def render_pin_description(self):
  1209. return render_to_string(
  1210. 'coderedcms/includes/map_pin_description.html',
  1211. {
  1212. 'page': self
  1213. }
  1214. )
  1215. @property
  1216. def render_list_description(self):
  1217. return render_to_string(
  1218. 'coderedcms/includes/map_list_description.html',
  1219. {
  1220. 'page': self
  1221. }
  1222. )
  1223. def to_geojson(self):
  1224. return {
  1225. "type": "Feature",
  1226. "geometry":{
  1227. "type": "Point",
  1228. "coordinates": [self.longitude, self.latitude]
  1229. },
  1230. "properties":{
  1231. "list_description": self.render_list_description,
  1232. "pin_description": self.render_pin_description
  1233. }
  1234. }
  1235. def save(self, *args, **kwargs):
  1236. if self.auto_update_latlng and GoogleApiSettings.for_site(Site.objects.get(is_default_site=True)).google_maps_api_key:
  1237. try:
  1238. g = geocoder.google(self.address, key=GoogleApiSettings.for_site(Site.objects.get(is_default_site=True)).google_maps_api_key)
  1239. self.latitude = g.latlng[0]
  1240. self.longitude = g.latlng[1]
  1241. except TypeError:
  1242. """Raised if google denied the request"""
  1243. pass
  1244. return super(CoderedLocationPage, self).save(*args, **kwargs)
  1245. def get_context(self, request, *args, **kwargs):
  1246. context = super().get_context(request)
  1247. context['google_api_key'] = GoogleApiSettings.for_site(Site.objects.get(is_default_site=True)).google_maps_api_key
  1248. return context
  1249. class CoderedLocationIndexPage(CoderedWebPage):
  1250. """
  1251. Shows a map view of the children CoderedLocationPage.
  1252. """
  1253. class Meta:
  1254. verbose_name = _('CodeRed Location Index Page')
  1255. abstract = True
  1256. template = 'coderedcms/pages/location_index_page.html'
  1257. index_show_subpages_default = True
  1258. center_latitude = models.FloatField(
  1259. null=True,
  1260. blank=True,
  1261. help_text=_('The default latitude you want the map set to.'),
  1262. default=0
  1263. )
  1264. center_longitude = models.FloatField(
  1265. null=True,
  1266. blank=True,
  1267. help_text=_('The default longitude you want the map set to.'),
  1268. default=0
  1269. )
  1270. zoom = models.IntegerField(
  1271. default=8,
  1272. validators=[
  1273. MaxValueValidator(20),
  1274. MinValueValidator(1),
  1275. ],
  1276. help_text=_('Requires API key to use zoom. 1: World, 5: Landmass/continent, 10: City, 15: Streets, 20: Buildings')
  1277. )
  1278. layout_panels = CoderedWebPage.layout_panels + [
  1279. MultiFieldPanel(
  1280. [
  1281. FieldPanel('center_latitude'),
  1282. FieldPanel('center_longitude'),
  1283. FieldPanel('zoom'),
  1284. ],
  1285. heading=_('Map Display')
  1286. ),
  1287. ]
  1288. def geojson_data(self, viewport=None):
  1289. """
  1290. function that will return all locations under this index as geoJSON compliant data.
  1291. It is filtered by a latitude/longitude viewport if given.
  1292. viewport is a string in the format of :
  1293. 'southwest.latitude,southwest.longitude|northeast.latitude,northeast.longitude'
  1294. An example viewport that covers Cleveland, OH would look like this:
  1295. '41.354912150983964,-81.95331736661791|41.663427748126935,-81.45206614591478'
  1296. """
  1297. qs = self.get_index_children().live()
  1298. if viewport:
  1299. southwest, northeast = viewport.split('|')
  1300. southwest = [float(x) for x in southwest.split(',')]
  1301. northeast = [float(x) for x in northeast.split(',')]
  1302. qs = qs.filter(latitude__gte=southwest[0], latitude__lte=northeast[0], longitude__gte=southwest[1], longitude__lte=northeast[1])
  1303. return {
  1304. "type": "FeatureCollection",
  1305. "features": [
  1306. location.to_geojson() for location in qs
  1307. ]
  1308. }
  1309. def serve(self, request, *args, **kwargs):
  1310. data_format = request.GET.get('data-format', None)
  1311. if data_format == 'geojson':
  1312. return self.serve_geojson(request, *args, **kwargs)
  1313. return super().serve(request, *args, **kwargs)
  1314. def serve_geojson(self, request, *args, **kwargs):
  1315. viewport = request.GET.get('viewport', None)
  1316. return JsonResponse(self.geojson_data(viewport=viewport))
  1317. def get_context(self, request, *args, **kwargs):
  1318. context = super().get_context(request)
  1319. context['google_api_key'] = GoogleApiSettings.for_site(Site.objects.get(is_default_site=True)).google_maps_api_key
  1320. return context