Source code for org.forms.extensions

from __future__ import annotations

from functools import cached_property
from sedate import utcnow, to_timezone

from onegov.core.html_diff import render_html_diff
from onegov.form.extensions import FormExtension
from onegov.form.fields import HoneyPotField
from onegov.form.fields import TimezoneDateTimeField
from onegov.form.fields import UploadField
from onegov.form.submissions import prepare_for_submission
from onegov.form.validators import StrictOptional, ValidPhoneNumber
from onegov.gis import CoordinatesField
from onegov.org import _
from wtforms.fields import EmailField
from wtforms.fields import StringField
from wtforms.fields import TextAreaField
from wtforms.validators import DataRequired, InputRequired


from typing import TypeVar, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Collection
    from markupsafe import Markup
    from onegov.form import Form
    from wtforms import Field


[docs] FormT = TypeVar('FormT', bound='Form')
[docs] class CoordinatesFormExtension(FormExtension[FormT], name='coordinates'):
[docs] def create(self) -> type[FormT]: class CoordinatesForm(self.form_class): # type:ignore coordinates = CoordinatesField( label=_('Coordinates'), description=_( 'The marker can be moved by dragging it with the mouse' ), fieldset=_('Map'), render_kw={'data-map-type': 'marker'} ) return CoordinatesForm
[docs] class SubmitterFormExtension(FormExtension[FormT], name='submitter'):
[docs] def create(self) -> type[FormT]: class SubmitterForm(self.form_class): # type:ignore submitter = EmailField( label=_('E-Mail'), fieldset=_('Submitter'), validators=[DataRequired()] ) submitter_name = StringField( label=_('Name'), fieldset=_('Submitter'), validators=[InputRequired()] ) submitter_address = StringField( label=_('Address'), fieldset=_('Submitter'), validators=[InputRequired()] ) submitter_phone = StringField( label=_('Phone'), fieldset=_('Submitter'), validators=[InputRequired(), ValidPhoneNumber()] ) def on_request(self) -> None: """ This is not an optimal solution defining this on a form extension. However, this is the first of it's kind. Don't forget to call super for the next one. =) """ if hasattr(super(), 'on_request'): super().on_request() if not hasattr(self.model, 'directory'): fields: Collection[str] = [] else: fields = self.model.directory.submitter_meta_fields or [] for field in ('name', 'address', 'phone'): if f'submitter_{field}' not in fields: self.delete_field(f'submitter_{field}') @property def submitter_meta(self) -> dict[str, str | None]: def field_data(name: str) -> str | None: field = getattr(self, name) return field and field.data or None return { 'submitter_name': field_data('submitter_name'), 'submitter_phone': field_data('submitter_phone'), 'submitter_address': field_data('submitter_address') } return SubmitterForm
[docs] class CommentFormExtension(FormExtension[FormT], name='comment'):
[docs] def create(self) -> type[FormT]: class CommentForm(self.form_class): # type:ignore comment = TextAreaField( label=_('Comment'), fieldset=_('Submitter'), render_kw={'rows': 7} ) return CommentForm
[docs] class ChangeRequestFormExtension(FormExtension[FormT], name='change-request'):
[docs] def create(self) -> type[FormT]: # XXX circular import from onegov.org.models.directory import ExtendedDirectoryEntry prepare_for_submission(self.form_class, for_change_request=True) class ChangeRequestForm(self.form_class): # type:ignore @cached_property def target(self) -> ExtendedDirectoryEntry | None: # not all steps have this information set, for example, towards # the end, the onegov.form submission code runs an extra # validation, which we ignore, trusting that it all worked # out earlier if not getattr(self, 'model', None): return None return ( self.request.session.query(ExtendedDirectoryEntry) .filter_by(id=self.model.meta['directory_entry']) .first() ) def is_different(self, field: Field) -> bool: # if the target has been removed, stop if not self.target: return True # after the changes have been applied, use the list of changes if self.model.meta.get('changed'): return field.id in self.model.meta['changed'] # ignore CSRF token if field.id == 'csrf_token': return False # coordinates fields are provided through extension if field.id == 'coordinates': return field.data != self.target.coordinates # upload fields differ if they are not empty if isinstance(field, UploadField): return field.data and True or False # like coordinates, provided through extension if field.id in ('publication_start', 'publication_end'): if not field.data: return False return ( to_timezone(field.data, 'UTC') != getattr(self.target, field.id) ) stored = self.target.values.get(field.id) or None field_data = field.data or None return stored != field_data def render_original( self, field: Field, from_model: bool = False ) -> Markup: prev = field.data try: model = self.target if model is not None: field.data = ( model.values.get(field.id) if not from_model else getattr(model, field.id) ) else: field.data = None return super().render_display(field) finally: field.data = prev def render_display(self, field: Field) -> Markup | None: if self.is_different(field): proposed = super().render_display(field) if not self.target: return proposed if field.id in ('csrf_token', 'coordinates'): return proposed if field.id in ('publication_start', 'publication_end'): original = self.render_original(field, from_model=True) return render_html_diff(original, proposed) if field.id not in self.target.values: return proposed if isinstance(field, UploadField): return proposed original = self.render_original(field) return render_html_diff(original, proposed) return None def ensure_changes(self) -> bool | None: if not self.target: return None for name, field in self._fields.items(): if self.is_different(field): return None for name, field in self._fields.items(): if name == 'csrf_token': continue field.errors.append( _('Please provide at least one change') ) return False return ChangeRequestForm
[docs] class PublicationFormExtension(FormExtension[FormT], name='publication'): """Can be used with TimezonePublicationMixin or UTCDateTime type decorator. """
[docs] def create(self, timezone: str = 'Europe/Zurich') -> type[FormT]: tz = timezone class PublicationForm(self.form_class): # type:ignore publication_start = TimezoneDateTimeField( label=_('Start'), timezone=tz, fieldset=_('Publication'), validators=[StrictOptional()] ) publication_end = TimezoneDateTimeField( label=_('End'), timezone=tz, fieldset=_('Publication'), validators=[StrictOptional()] ) def ensure_publication_start_end(self) -> bool | None: start = self.publication_start end = self.publication_end if not start or not end: return None if end.data and to_timezone(end.data, 'UTC') <= utcnow(): assert isinstance(self.publication_end.errors, list) self.publication_end.errors.append( _('Publication end must be in the future')) return False if not start.data or not end.data: return None if end.data <= start.data: for field_name in ('publication_start', 'publication_end'): field = getattr(self, field_name) field.errors.append( _('Publication start must be prior to end')) return False return None return PublicationForm
[docs] class HoneyPotFormExtension(FormExtension[FormT], name='honeypot'):
[docs] def create(self) -> type[FormT]: class HoneyPotForm(self.form_class): # type:ignore duplicate_of = HoneyPotField() def on_request(self) -> None: if self.model and not getattr(self.model, 'honeypot', False): self.delete_field('duplicate_of') return HoneyPotForm