models.py 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841
  1. from collections import OrderedDict
  2. from importlib import import_module
  3. from itertools import zip_longest
  4. import json
  5. import os
  6. from pathlib import Path
  7. from PIL import Image
  8. import datetime
  9. from django.apps import apps
  10. from django.conf import settings
  11. from django.contrib import messages
  12. from django.contrib.contenttypes.fields import GenericForeignKey
  13. from django.contrib.contenttypes.models import ContentType
  14. from django.contrib.humanize.templatetags.humanize import naturaltime
  15. from django.core.files.storage import default_storage
  16. from django.core.serializers.json import DjangoJSONEncoder
  17. from django.db.models import (
  18. CharField, TextField, DateTimeField, Model, ForeignKey, PROTECT, CASCADE,
  19. QuerySet,
  20. )
  21. from django.db.models.fields.files import FieldFile
  22. from django.db.models.signals import post_delete, post_save
  23. from django.dispatch import receiver
  24. from django.forms import Form, ImageField, FileField, URLField, EmailField
  25. from django.http import HttpResponseRedirect
  26. from django.template.response import TemplateResponse
  27. from django.utils.safestring import SafeData, mark_safe
  28. from django.utils.timezone import now
  29. from django.utils.translation import ugettext_lazy as _
  30. from wagtail.core.models import Page
  31. from wagtail.contrib.forms.models import (
  32. AbstractForm, AbstractEmailForm, AbstractFormSubmission)
  33. from .blocks import FormStepBlock, FormFieldBlock
  34. class Step:
  35. def __init__(self, steps, index, struct_child):
  36. self.steps = steps
  37. self.index = index
  38. block = getattr(struct_child, 'block', None)
  39. if block is None:
  40. struct_child = []
  41. if isinstance(block, FormStepBlock):
  42. self.name = struct_child.value['name']
  43. self.form_fields = struct_child.value['form_fields']
  44. else:
  45. self.name = ''
  46. self.form_fields = struct_child
  47. @property
  48. def index1(self):
  49. return self.index + 1
  50. @property
  51. def url(self):
  52. return '%s?step=%s' % (self.steps.page.url, self.index1)
  53. def get_form_fields(self):
  54. form_fields = OrderedDict()
  55. field_blocks = self.form_fields
  56. for struct_child in field_blocks:
  57. block = struct_child.block
  58. if isinstance(block, FormFieldBlock):
  59. struct_value = struct_child.value
  60. field_name = block.get_slug(struct_value)
  61. form_fields[field_name] = block.get_field(struct_value)
  62. return form_fields
  63. def get_form_class(self):
  64. return type('WagtailForm', self.steps.page.get_form_class_bases(),
  65. self.get_form_fields())
  66. def get_markups_and_bound_fields(self, form):
  67. for struct_child in self.form_fields:
  68. block = struct_child.block
  69. if isinstance(block, FormFieldBlock):
  70. struct_value = struct_child.value
  71. field_name = block.get_slug(struct_value)
  72. yield form[field_name], 'field'
  73. else:
  74. yield mark_safe(struct_child), 'markup'
  75. def __str__(self):
  76. if self.name:
  77. return self.name
  78. return _('Step %s') % self.index1
  79. @property
  80. def badge(self):
  81. return (mark_safe('<span class="badge">%s/%s</span>')
  82. % (self.index1, len(self.steps)))
  83. def __html__(self):
  84. return '%s %s' % (self, self.badge)
  85. @property
  86. def is_active(self):
  87. return self.index == self.steps.current_index
  88. @property
  89. def is_last(self):
  90. return self.index1 == len(self.steps)
  91. @property
  92. def has_prev(self):
  93. return self.index > 0
  94. @property
  95. def has_next(self):
  96. return self.index1 < len(self.steps)
  97. @property
  98. def prev(self):
  99. if self.has_prev:
  100. return self.steps[self.index-1]
  101. @property
  102. def next(self):
  103. if self.has_next:
  104. return self.steps[self.index+1]
  105. def get_existing_data(self, raw=False):
  106. data = self.steps.get_existing_data()[self.index]
  107. fields = self.get_form_fields()
  108. if not raw:
  109. class FakeField:
  110. storage = self.steps.get_storage()
  111. for field_name, value in data.items():
  112. if field_name in fields and isinstance(fields[field_name],
  113. FileField):
  114. data[field_name] = FieldFile(None, FakeField, value)
  115. return data
  116. @property
  117. def is_available(self):
  118. return self.prev is None or self.prev.get_existing_data(raw=True)
  119. class StreamFormJSONEncoder(DjangoJSONEncoder):
  120. def default(self, o):
  121. try:
  122. from phonenumber_field.phonenumber import PhoneNumber
  123. except ImportError:
  124. pass
  125. else:
  126. if isinstance(o, PhoneNumber):
  127. return str(o)
  128. return super().default(o)
  129. class Steps(list):
  130. def __init__(self, page, request=None):
  131. self.page = page
  132. # TODO: Make it possible to change the `form_fields` attribute.
  133. self.form_fields = page.form_fields
  134. self.request = request
  135. has_steps = any(isinstance(struct_child.block, FormStepBlock)
  136. for struct_child in self.form_fields)
  137. if has_steps:
  138. steps = [Step(self, i, form_field)
  139. for i, form_field in enumerate(self.form_fields)]
  140. else:
  141. steps = [Step(self, 0, self.form_fields)]
  142. super().__init__(steps)
  143. def clamp_index(self, index: int):
  144. if index < 0:
  145. index = 0
  146. if index >= len(self):
  147. index = len(self) - 1
  148. while not self[index].is_available:
  149. index -= 1
  150. return index
  151. @property
  152. def current_index(self):
  153. return self.request.session.get(self.page.current_step_session_key, 0)
  154. @property
  155. def current(self):
  156. return self[self.current_index]
  157. @current.setter
  158. def current(self, new_index: int):
  159. if not isinstance(new_index, int):
  160. raise TypeError('Use an integer to set the new current step.')
  161. self.request.session[self.page.current_step_session_key] = \
  162. self.clamp_index(new_index)
  163. def forward(self, increment: int = 1):
  164. self.current = self.current_index + increment
  165. def backward(self, increment: int = 1):
  166. self.current = self.current_index - increment
  167. def get_submission(self):
  168. return self.page.get_submission(self.request)
  169. def get_existing_data(self):
  170. submission = self.get_submission()
  171. data = [] if submission is None else json.loads(submission.form_data)
  172. length_difference = len(self) - len(data)
  173. if length_difference > 0:
  174. data.extend([{}] * length_difference)
  175. return data
  176. def get_current_form(self):
  177. request = self.request
  178. if request.method == 'POST':
  179. step_value = request.POST.get('step', 'next')
  180. if step_value == 'prev':
  181. self.backward()
  182. else:
  183. return self.current.get_form_class()(
  184. request.POST, request.FILES,
  185. initial=self.current.get_existing_data())
  186. return self.current.get_form_class()(
  187. initial=self.current.get_existing_data())
  188. def get_storage(self):
  189. return self.page.get_storage()
  190. def save_files(self, form):
  191. submission = self.get_submission()
  192. for name, field in form.fields.items():
  193. if isinstance(field, FileField):
  194. file = form.cleaned_data[name]
  195. if file == form.initial.get(name, ''): # Nothing submitted.
  196. form.cleaned_data[name] = file.name
  197. continue
  198. if submission is not None:
  199. submission.delete_file(name)
  200. if not file: # 'Clear' was checked.
  201. form.cleaned_data[name] = ''
  202. continue
  203. directory = self.request.session.session_key
  204. storage = self.get_storage()
  205. Path(storage.path(directory)).mkdir(parents=True,
  206. exist_ok=True)
  207. path = storage.get_available_name(
  208. str(Path(directory) / file.name))
  209. with storage.open(path, 'wb+') as destination:
  210. for chunk in file.chunks():
  211. destination.write(chunk)
  212. form.cleaned_data[name] = path
  213. def update_data(self):
  214. form = self.get_current_form()
  215. if form.is_valid():
  216. form_data = self.get_existing_data()
  217. self.save_files(form)
  218. form_data[self.current_index] = form.cleaned_data
  219. form_data = json.dumps(form_data, cls=StreamFormJSONEncoder)
  220. is_complete = self.current.is_last
  221. submission = self.get_submission()
  222. submission.form_data = form_data
  223. if not submission.is_complete and is_complete:
  224. submission.status = submission.COMPLETE
  225. submission.save()
  226. if is_complete:
  227. self.current = 0
  228. else:
  229. self.forward()
  230. return is_complete
  231. return False
  232. class SessionFormSubmission(AbstractFormSubmission):
  233. session_key = CharField(max_length=40, null=True, default=None)
  234. user = ForeignKey(settings.AUTH_USER_MODEL, null=True, blank=True,
  235. related_name='+', on_delete=PROTECT)
  236. thumbnails_by_path = TextField(default=json.dumps({}))
  237. last_modification = DateTimeField(_('last modification'), auto_now=True)
  238. INCOMPLETE = 'incomplete'
  239. COMPLETE = 'complete'
  240. REVIEWED = 'reviewed'
  241. APPROVED = 'approved'
  242. REJECTED = 'rejected'
  243. STATUSES = (
  244. (INCOMPLETE, _('Not submitted')),
  245. (COMPLETE, _('In progress')),
  246. (REVIEWED, _('Under consideration')),
  247. (APPROVED, _('Approved')),
  248. (REJECTED, _('Rejected')),
  249. )
  250. status = CharField(max_length=10, choices=STATUSES, default=INCOMPLETE)
  251. class Meta:
  252. verbose_name = _('form submission')
  253. verbose_name_plural = _('form submissions')
  254. unique_together = (('page', 'session_key'),
  255. ('page', 'user'))
  256. abstract=True
  257. @property
  258. def is_complete(self):
  259. return self.status != self.INCOMPLETE
  260. @property
  261. def form_page(self):
  262. return self.page.specific
  263. def get_session(self):
  264. return import_module(settings.SESSION_ENGINE).SessionStore(
  265. session_key=self.session_key)
  266. def reset_step(self):
  267. session = self.get_session()
  268. try:
  269. del session[self.form_page.current_step_session_key]
  270. except KeyError:
  271. pass
  272. else:
  273. session.save()
  274. def get_storage(self):
  275. return self.form_page.get_storage()
  276. def get_thumbnail_path(self, path, width=64, height=64):
  277. if not path:
  278. return ''
  279. variant = '%s×%s' % (width, height)
  280. thumbnails_by_path = json.loads(self.thumbnails_by_path)
  281. thumbnails_paths = thumbnails_by_path.get(path)
  282. if thumbnails_paths is None:
  283. thumbnails_by_path[path] = {}
  284. else:
  285. thumbnail_path = thumbnails_paths.get(variant)
  286. if thumbnail_path is not None:
  287. return thumbnail_path
  288. path = Path(path)
  289. thumbnail_path = str(path.with_suffix('.%s%s'
  290. % (variant, path.suffix)))
  291. storage = self.get_storage()
  292. thumbnail_path = storage.get_available_name(thumbnail_path)
  293. thumbnail = Image.open(storage.path(path))
  294. thumbnail.thumbnail((width, height))
  295. thumbnail.save(storage.path(thumbnail_path))
  296. thumbnails_by_path[str(path)][variant] = thumbnail_path
  297. self.thumbnails_by_path = json.dumps(thumbnails_by_path,
  298. cls=StreamFormJSONEncoder)
  299. self.save()
  300. return thumbnail_path
  301. def get_fields(self, by_step=False):
  302. return self.form_page.get_form_fields(by_step=by_step)
  303. def get_existing_thumbnails(self, path):
  304. thumbnails_paths = json.loads(self.thumbnails_by_path).get(path, {})
  305. for thumbnail_path in thumbnails_paths.values():
  306. yield thumbnail_path
  307. def get_files_by_field(self):
  308. data = self.get_data(raw=True)
  309. files = {}
  310. for name, field in self.get_fields().items():
  311. if isinstance(field, FileField):
  312. path = data.get(name)
  313. if path:
  314. files[name] = [path] + list(
  315. self.get_existing_thumbnails(path))
  316. return files
  317. def get_all_files(self):
  318. for paths in self.get_files_by_field().values():
  319. for path in paths:
  320. yield path
  321. def delete_file(self, field_name):
  322. thumbnails_by_path = json.loads(self.thumbnails_by_path)
  323. for path in self.get_files_by_field().get(field_name, ()):
  324. self.get_storage().delete(path)
  325. if path in thumbnails_by_path:
  326. del thumbnails_by_path[path]
  327. self.thumbnails_by_path = json.dumps(thumbnails_by_path,
  328. cls=StreamFormJSONEncoder)
  329. self.save()
  330. def render_email(self, value):
  331. return (mark_safe('<a href="mailto:%s" target="_blank">%s</a>')
  332. % (value, value))
  333. def render_link(self, value):
  334. return (mark_safe('<a href="%s" target="_blank">%s</a>')
  335. % (value, value))
  336. def render_image(self, value):
  337. storage = self.get_storage()
  338. return (mark_safe('<a href="%s" target="_blank"><img src="%s" /></a>')
  339. % (storage.url(value),
  340. storage.url(self.get_thumbnail_path(value))))
  341. def render_file(self, value):
  342. return mark_safe('<a href="%s" target="_blank">%s</a>') % (
  343. self.get_storage().url(value),
  344. Path(value).name
  345. )
  346. def format_value(self, field, value):
  347. if value is None or value == '':
  348. return '-'
  349. new_value = self.form_page.format_value(field, value)
  350. if new_value != value:
  351. return new_value
  352. if value is True:
  353. return 'Yes'
  354. if value is False:
  355. return 'No'
  356. if isinstance(value, (list, tuple)):
  357. return ', '.join([self.format_value(field, item)
  358. for item in value])
  359. if isinstance(value, datetime.date):
  360. return value
  361. if isinstance(field, EmailField):
  362. return self.render_email(value)
  363. if isinstance(field, URLField):
  364. return self.render_link(value)
  365. if isinstance(field, ImageField):
  366. return self.render_image(value)
  367. if isinstance(field, FileField):
  368. return self.render_file(value)
  369. if isinstance(value, SafeData) or hasattr(value, '__html__'):
  370. return value
  371. return str(value)
  372. def format_db_field(self, field_name, raw=False):
  373. method = getattr(self, 'get_%s_display' % field_name, None)
  374. if method is not None:
  375. return method()
  376. value = getattr(self, field_name)
  377. if raw:
  378. return value
  379. return self.format_value(self._meta.get_field(field_name).formfield(),
  380. value)
  381. def get_steps_data(self, raw=False):
  382. steps_data = json.loads(self.form_data)
  383. if raw:
  384. return steps_data
  385. fields_and_data_iterator = zip_longest(self.get_fields(by_step=True),
  386. steps_data, fillvalue={})
  387. return [
  388. OrderedDict([(name, self.format_value(field, step_data.get(name)))
  389. for name, field in step_fields.items()])
  390. for step_fields, step_data in fields_and_data_iterator]
  391. def get_extra_data(self, raw=False):
  392. return self.form_page.get_extra_data(self, raw=raw)
  393. def get_data(self, raw=False, add_metadata=True):
  394. steps_data = self.get_steps_data(raw=raw)
  395. form_data = {}
  396. form_data.update(self.get_extra_data(raw=raw))
  397. for step_data in steps_data:
  398. form_data.update(step_data)
  399. if add_metadata:
  400. form_data.update(
  401. status=self.format_db_field('status', raw=raw),
  402. user=self.format_db_field('user', raw=raw),
  403. submit_time=self.format_db_field('submit_time', raw=raw),
  404. last_modification=self.format_db_field('last_modification',
  405. raw=raw),
  406. )
  407. return form_data
  408. def steps_with_data_iterator(self, raw=False):
  409. for step, step_data_fields, step_data in zip(
  410. self.form_page.get_steps(),
  411. self.form_page.get_data_fields(by_step=True),
  412. self.get_steps_data(raw=raw)):
  413. yield step, [(field_name, field_label, step_data[field_name])
  414. for field_name, field_label in step_data_fields]
  415. @receiver(post_delete, sender=SessionFormSubmission)
  416. def delete_files(sender, **kwargs):
  417. instance = kwargs['instance']
  418. instance.reset_step()
  419. storage = instance.get_storage()
  420. for path in instance.get_all_files():
  421. storage.delete(path)
  422. # Automatically deletes ancestor folders if empty.
  423. directory = Path(path)
  424. while directory.parent != Path(directory.root):
  425. directory = directory.parent
  426. try:
  427. subdirectories, files = storage.listdir(directory)
  428. except FileNotFoundError:
  429. continue
  430. if not subdirectories and not files:
  431. Path(storage.path(directory)).rmdir()
  432. class SubmissionRevisionQuerySet(QuerySet):
  433. def for_submission(self, submission):
  434. return self.filter(**self.model.get_filters_for(submission))
  435. def created(self):
  436. return self.filter(type=self.model.CREATED)
  437. def changed(self):
  438. return self.filter(type=self.model.CHANGED)
  439. def deleted(self):
  440. return self.filter(type=self.model.DELETED)
  441. class SubmissionRevision(Model):
  442. CREATED = 'created'
  443. CHANGED = 'changed'
  444. DELETED = 'deleted'
  445. TYPES = (
  446. (CREATED, _('Created')),
  447. (CHANGED, _('Changed')),
  448. (DELETED, _('Deleted')),
  449. )
  450. type = CharField(max_length=7, choices=TYPES)
  451. created_at = DateTimeField(auto_now_add=True)
  452. submission_ct = ForeignKey('contenttypes.ContentType', on_delete=CASCADE)
  453. submission_id = TextField()
  454. submission = GenericForeignKey('submission_ct', 'submission_id')
  455. data = TextField()
  456. summary = TextField()
  457. objects = SubmissionRevisionQuerySet.as_manager()
  458. class Meta:
  459. ordering = ('-created_at',)
  460. abstract=True
  461. @staticmethod
  462. def get_filters_for(submission):
  463. return {
  464. 'submission_ct':
  465. ContentType.objects.get_for_model(submission._meta.model),
  466. 'submission_id': str(submission.pk),
  467. }
  468. @classmethod
  469. def diff_summary(cls, page, data1, data2):
  470. diff = []
  471. data_fields = page.get_data_fields()
  472. hidden_types = (tuple, list, dict)
  473. for k, label in data_fields:
  474. value1 = data1.get(k)
  475. value2 = data2.get(k)
  476. if value2 == value1 or not value1 and not value2:
  477. continue
  478. is_hidden = (isinstance(value1, hidden_types)
  479. or isinstance(value2, hidden_types))
  480. # Escapes newlines as they are used as separator inside summaries.
  481. if isinstance(value1, str):
  482. value1 = value1.replace('\n', r'\n')
  483. if isinstance(value2, str):
  484. value2 = value2.replace('\n', r'\n')
  485. if value2 and not value1:
  486. diff.append(
  487. ((_('“%s” set.') % label) if is_hidden
  488. else (_('“%s” set to “%s”.')) % (label, value2)))
  489. elif value1 and not value2:
  490. diff.append(_('“%s” unset.') % label)
  491. else:
  492. diff.append(((_('“%s” changed.') % label) if is_hidden
  493. else (_('“%s” changed from “%s” to “%s”.')
  494. % (label, value1, value2))))
  495. return '\n'.join(diff)
  496. @classmethod
  497. def create_from_submission(cls, submission, revision_type):
  498. page = submission.form_page
  499. try:
  500. previous = cls.objects.for_submission(
  501. submission).latest('created_at')
  502. except cls.DoesNotExist:
  503. previous_data = {}
  504. else:
  505. previous_data = previous.get_data()
  506. filters = cls.get_filters_for(submission)
  507. data = submission.get_data(raw=True, add_metadata=False)
  508. data['status'] = submission.status
  509. if revision_type == cls.CREATED:
  510. summary = _('Submission created.')
  511. elif revision_type == cls.DELETED:
  512. summary = _('Submission deleted.')
  513. else:
  514. summary = cls.diff_summary(page, previous_data, data)
  515. if not summary: # Nothing changed.
  516. return
  517. filters.update(
  518. type=revision_type,
  519. data=json.dumps(data, cls=StreamFormJSONEncoder),
  520. summary=summary,
  521. )
  522. return cls.objects.create(**filters)
  523. def get_data(self):
  524. return json.loads(self.data)
  525. # ORIGINAL NORIPYT CODE.
  526. # We don't want these receivers triggering.
  527. # @receiver(post_save)
  528. # def create_submission_changed_revision(sender, **kwargs):
  529. # if not issubclass(sender, SessionFormSubmission):
  530. # return
  531. # submission = kwargs['instance']
  532. # created = kwargs['created']
  533. # SubmissionRevision.create_from_submission(
  534. # submission, (SubmissionRevision.CREATED if created
  535. # else SubmissionRevision.CHANGED))
  536. # @receiver(post_delete)
  537. # def create_submission_deleted_revision(sender, **kwargs):
  538. # if not issubclass(sender, SessionFormSubmission):
  539. # return
  540. # submission = kwargs['instance']
  541. # SubmissionRevision.create_from_submission(submission,
  542. # SubmissionRevision.DELETED)
  543. class StreamFormMixin:
  544. preview_modes = Page.DEFAULT_PREVIEW_MODES
  545. @property
  546. def current_step_session_key(self):
  547. return '%s:step' % self.pk
  548. def get_steps(self, request=None):
  549. if not hasattr(self, 'steps'):
  550. steps = Steps(self, request=request)
  551. if request is None:
  552. return steps
  553. self.steps = steps
  554. return self.steps
  555. def get_form_fields(self, by_step=False):
  556. if by_step:
  557. return [step.get_form_fields() for step in self.get_steps()]
  558. form_fields = OrderedDict()
  559. for step_fields in self.get_form_fields(by_step=True):
  560. form_fields.update(step_fields)
  561. return form_fields
  562. def get_context(self, request, *args, **kwargs):
  563. context = super().get_context(request, *args, **kwargs)
  564. self.steps = self.get_steps(request)
  565. step_value = request.GET.get('step')
  566. if step_value is not None and step_value.isdigit():
  567. self.steps.current = int(step_value) - 1
  568. form = self.steps.get_current_form()
  569. context.update(
  570. steps=self.steps,
  571. step=self.steps.current,
  572. form=form,
  573. markups_and_bound_fields=list(
  574. self.steps.current.get_markups_and_bound_fields(form)),
  575. )
  576. return context
  577. def get_storage(self):
  578. return default_storage
  579. @staticmethod
  580. def get_form_class_bases():
  581. return Form,
  582. @staticmethod
  583. def get_submission_class():
  584. return SessionFormSubmission
  585. def get_submission(self, request):
  586. Submission = self.get_submission_class()
  587. if request.user.is_authenticated:
  588. user_submission = Submission.objects.filter(
  589. user=request.user, page=self).order_by('-pk').first()
  590. if user_submission is None:
  591. return Submission(user=request.user, page=self, form_data='[]')
  592. return user_submission
  593. user_submission = Submission.objects.filter(
  594. session_key=request.session.session_key, page=self
  595. ).order_by('-pk').first()
  596. if user_submission is None:
  597. return Submission(session_key=request.session.session_key,
  598. page=self, form_data='[]')
  599. return user_submission
  600. def get_success_url(self):
  601. form_complete_models = [model for model in apps.get_models()
  602. if issubclass(model, FormCompleteMixin)]
  603. cts = (ContentType.objects
  604. .get_for_models(*form_complete_models).values())
  605. first_child = self.get_children().filter(content_type__in=cts).first()
  606. if first_child is None:
  607. return self.url
  608. return first_child.url
  609. def serve_success(self, request, *args, **kwargs):
  610. url = self.get_success_url()
  611. if url == self.url:
  612. messages.success(request,
  613. _('Successfully submitted the form.'))
  614. return HttpResponseRedirect(url)
  615. def serve(self, request, *args, **kwargs):
  616. context = self.get_context(request)
  617. form = context['form']
  618. if request.method == 'POST' and form.is_valid():
  619. is_complete = self.steps.update_data()
  620. if is_complete:
  621. return self.serve_success(request, *args, **kwargs)
  622. return HttpResponseRedirect(self.url)
  623. return Page.serve(self, request, *args, **kwargs)
  624. def get_data_fields(self, by_step=False, add_metadata=True):
  625. if by_step:
  626. return [[(field_name, field.label)
  627. for field_name, field in step_fields.items()]
  628. for step_fields in self.get_form_fields(by_step=True)]
  629. data_fields = []
  630. data_fields.extend(self.get_extra_data_fields())
  631. if add_metadata:
  632. data_fields.extend((
  633. ('status', _('Status')),
  634. ('user', _('User')),
  635. ('submit_time', _('First modification')),
  636. ('last_modification', _('Last modification'))))
  637. data_fields.extend([
  638. (field_name, field_label)
  639. for step_data_fields in self.get_data_fields(by_step=True)
  640. for field_name, field_label in step_data_fields])
  641. return data_fields
  642. def get_extra_data_fields(self):
  643. return ()
  644. def get_extra_data(self, submission, raw=False):
  645. return {}
  646. def format_value(self, field, value):
  647. return value
  648. class ClosingFormMixin(Model):
  649. closing_at = DateTimeField()
  650. closed_template = None
  651. class Meta:
  652. abstract = True
  653. @property
  654. def is_closed(self):
  655. return now() > self.closing_at
  656. def get_closed_template(self, request, *args, **kwargs):
  657. if self.closed_template is None:
  658. template = self.get_template(request, *args, **kwargs)
  659. base, ext = os.path.splitext(template)
  660. return '%s_closed%s' % (base, ext)
  661. return self.closed_template
  662. def serve_closed(self, request, *args, **kwargs):
  663. return TemplateResponse(
  664. request,
  665. self.get_closed_template(request, *args, **kwargs),
  666. self.get_context(request, *args, **kwargs),
  667. )
  668. def serve(self, request, *args, **kwargs):
  669. if self.is_closed:
  670. return self.serve_closed(request, *args, **kwargs)
  671. return super().serve(request, *args, **kwargs)
  672. class FormCompleteMixin:
  673. def get_form_page(self):
  674. return self.get_parent().specific
  675. def serve(self, request, *args, **kwargs):
  676. form_page = self.get_form_page()
  677. if isinstance(form_page, LoginRequiredMixin) \
  678. and not request.user.is_authenticated():
  679. return HttpResponseRedirect(form_page.url)
  680. self.submission = form_page.get_submission(request)
  681. if self.submission is not None and self.submission.is_complete \
  682. or getattr(request, 'is_preview', False):
  683. return super().serve(request, *args, **kwargs)
  684. return HttpResponseRedirect(form_page.url)
  685. def get_context(self, *args, **kwargs):
  686. context = super().get_context(*args, **kwargs)
  687. if hasattr(self, 'submission'):
  688. context['submission'] = self.submission
  689. return context
  690. class LoginRequiredMixin:
  691. login_required_template = None
  692. def get_login_required_template(self, request, *args, **kwargs):
  693. if self.login_required_template is None:
  694. template = self.get_template(request, *args, **kwargs)
  695. base, ext = os.path.splitext(template)
  696. return '%s_login_required%s' % (base, ext)
  697. return self.login_required_template
  698. def serve_login_required(self, request, *args, **kwargs):
  699. return TemplateResponse(
  700. request,
  701. self.get_login_required_template(request, *args, **kwargs),
  702. self.get_context(request, *args, **kwargs),
  703. )
  704. def serve(self, request, *args, **kwargs):
  705. if not request.user.is_authenticated():
  706. return self.serve_login_required(request, *args, **kwargs)
  707. return super().serve(request, *args, **kwargs)
  708. class AbstractStreamForm(StreamFormMixin, AbstractForm):
  709. class Meta:
  710. abstract = True
  711. class AbstractEmailStreamForm(StreamFormMixin, AbstractEmailForm):
  712. class Meta:
  713. abstract = True