Source code for org.views.form_definition

from __future__ import annotations

import morepath

from onegov.core.security import Private, Public
from onegov.core.utils import normalize_for_url
from onegov.form import FormCollection, FormDefinition
from onegov.form import FormRegistrationWindow
from onegov.gis import Coordinates
from onegov.org import _, OrgApp
from onegov.org.cli import close_ticket
from onegov.org.elements import Link
from onegov.org.forms import FormDefinitionForm
from onegov.org.forms.form_definition import FormDefinitionUrlForm
from onegov.org.layout import FormEditorLayout, FormSubmissionLayout
from onegov.org.models import BuiltinFormDefinition, CustomFormDefinition
from onegov.org.models.form import submission_deletable
from webob import exc


from typing import TypeVar, TYPE_CHECKING

if TYPE_CHECKING:
    from collections.abc import Iterable, Iterator
    from onegov.core.layout import Layout
    from onegov.core.types import RenderData
    from onegov.form import Form, FormSubmission
    from onegov.org.request import OrgRequest
    from sqlalchemy.orm import Session
    from webob import Response

[docs] FormDefinitionT = TypeVar('FormDefinitionT', bound=FormDefinition)
[docs] def get_form_class( model: BuiltinFormDefinition | CustomFormDefinition | FormCollection, request: OrgRequest ) -> type[FormDefinitionForm]: if isinstance(model, FormCollection): model = CustomFormDefinition() form_classes = { 'builtin': FormDefinitionForm, 'custom': FormDefinitionForm } return model.with_content_extensions(form_classes[model.type], request)
# FIXME: format_date should really be a utility function on the request # first and on the layout second, passing around layouts in contexts # where we don't actually always have one is weird
[docs] def get_hints( layout: Layout, window: FormRegistrationWindow | None ) -> Iterator[tuple[str, str]]: if not window: return if window.in_the_past: yield 'stop', _('The registration has ended') elif not window.enabled: yield 'stop', _('The registration is closed') if window.enabled and window.in_the_future: yield 'date', _('The registration opens on ${day}, ${date}', mapping={ 'day': layout.format_date(window.start, 'weekday_long'), 'date': layout.format_date(window.start, 'date_long') }) if window.enabled and window.in_the_present: yield 'date', _('The registration closes on ${day}, ${date}', mapping={ 'day': layout.format_date(window.end, 'weekday_long'), 'date': layout.format_date(window.end, 'date_long') }) if window.limit and window.overflow: yield 'count', _("There's a limit of ${count} attendees", mapping={ 'count': window.limit }) if window.limit and not window.overflow: spots = window.available_spots if spots == 0: yield 'stop', _('There are no spots left') elif spots == 1: yield 'count', _('There is one spot left') else: yield 'count', _('There are ${count} spots left', mapping={ 'count': spots })
[docs] def handle_form_change_name( form: FormDefinitionT, session: Session, new_name: str ) -> FormDefinitionT: new_form = form.for_new_name(new_name) session.add(new_form) session.flush() submissions = form.submissions windows = form.registration_windows with session.no_autoflush: # This placed elsewhere will not work form.submissions = [] form.registration_windows = [] # assigning the whole list directly will not work for s in submissions: s.name = new_name new_form.submissions.append(s) for w in windows: w.name = new_name new_form.registration_windows.append(w) session.flush() assert not form.submissions assert not form.registration_windows return new_form
@OrgApp.form( model=FormDefinition, form=FormDefinitionUrlForm, template='form.pt', permission=Private, name='change-url' )
[docs] def handle_change_form_name( self: FormDefinition, request: OrgRequest, form: FormDefinitionUrlForm, layout: FormEditorLayout | None = None ) -> RenderData | Response: """Since the name used for the url is the primary key, we create a new FormDefinition to make our live easier """ site_title = _('Change URL') if form.submitted(request): assert form.name.data is not None new_form = handle_form_change_name( self, request.session, form.name.data ) request.session.delete(self) return request.redirect(request.link(new_form)) elif not request.POST: form.process(obj=self) return { 'layout': layout or FormEditorLayout(self, request), 'form': form, 'title': site_title, }
@OrgApp.form( model=FormDefinition, template='form.pt', permission=Public, form=lambda self, request: self.form_class )
[docs] def handle_defined_form( self: FormDefinition, request: OrgRequest, form: Form, layout: FormSubmissionLayout | None = None ) -> RenderData | Response: """ Renders the empty form and takes input, even if it's not valid, stores it as a pending submission and redirects the user to the view that handles pending submissions. """ collection = FormCollection(request.session) if not self.current_registration_window: spots = 0 enabled = True else: spots = 1 enabled = self.current_registration_window.accepts_submissions(spots) if enabled and request.POST: submission = collection.submissions.add( self.name, form, state='pending', spots=spots) return morepath.redirect(request.link(submission)) layout = layout or FormSubmissionLayout(self, request) return { 'layout': layout, 'title': self.title, 'form': enabled and form, 'definition': self, 'form_width': 'small', 'lead': layout.linkify(self.meta.get('lead')), 'text': self.text, 'people': getattr(self, 'people', None), 'files': getattr(self, 'files', None), 'contact': getattr(self, 'contact_html', None), 'coordinates': getattr(self, 'coordinates', Coordinates()), 'hints': tuple(get_hints(layout, self.current_registration_window)), 'hints_callout': not enabled, 'button_text': _('Continue') }
@OrgApp.form( model=FormCollection, name='new', template='form.pt', permission=Private, form=get_form_class )
[docs] def handle_new_definition( self: FormCollection, request: OrgRequest, form: FormDefinitionForm, layout: FormEditorLayout | None = None ) -> RenderData | Response: if form.submitted(request): assert form.title.data is not None assert form.definition.data is not None if self.definitions.by_name(normalize_for_url(form.title.data)): request.alert(_('A form with this name already exists')) else: definition = self.definitions.add( title=form.title.data, definition=form.definition.data, type='custom' ) form.populate_obj(definition) request.success(_('Added a new form')) return morepath.redirect(request.link(definition)) layout = layout or FormEditorLayout(self, request) layout.breadcrumbs = [ Link(_('Homepage'), layout.homepage_url), Link(_('Forms'), request.link(self)), Link(_('New Form'), request.link(self, name='new')) ] layout.edit_mode = True return { 'layout': layout, 'title': _('New Form'), 'form': form, 'form_width': 'large', }
@OrgApp.form( model=FormDefinition, template='form.pt', permission=Private, form=get_form_class, name='edit' )
[docs] def handle_edit_definition( self: FormDefinition, request: OrgRequest, form: FormDefinitionForm, layout: FormEditorLayout | None = None ) -> RenderData | Response: if form.submitted(request): assert form.definition.data is not None # why do we exclude definition here? we set it normally right after # which is also what populate_obj should be doing form.populate_obj(self, exclude={'definition'}) self.definition = form.definition.data request.success(_('Your changes were saved')) return morepath.redirect(request.link(self)) elif not request.POST: form.process(obj=self) collection = FormCollection(request.session) layout = layout or FormEditorLayout(self, request) layout.breadcrumbs = [ Link(_('Homepage'), layout.homepage_url), Link(_('Forms'), request.link(collection)), Link(self.title, request.link(self)), Link(_('Edit'), request.link(self, name='edit')) ] layout.edit_mode = True return { 'layout': layout, 'title': self.title, 'form': form, 'form_width': 'large', }
@OrgApp.view( model=FormDefinition, request_method='DELETE', permission=Private )
[docs] def delete_form_definition( self: FormDefinition, request: OrgRequest ) -> None: """ With introduction of cancelling submissions over the registration window, we encourage the user to use this functionality to cancel all form submissions through the ticket system. This ensures all submissions are cancelled/denied and the tickets are closed.In that case the ticket itself attached to the submission is deletable. If the customer wants to delete the form directly, we allow it now even when there are completed submissions. In each case there is a ticket associated with it we might take a snapshot before deleting it. """ request.assert_valid_csrf_token() if self.type != 'custom': raise exc.HTTPMethodNotAllowed() def handle_ticket(submission: FormSubmission) -> None: ticket = submission_deletable(submission, request.session) if ticket is False: raise exc.HTTPMethodNotAllowed() if ticket is not True: assert request.current_user is not None close_ticket(ticket, request.current_user, request) ticket.create_snapshot(request) def handle_submissions(submissions: Iterable[FormSubmission]) -> None: for s in submissions: handle_ticket(s) FormCollection(request.session).definitions.delete( self.name, with_submissions=True, with_registration_windows=True, handle_submissions=handle_submissions )