Source code for org.forms.form_registration

from __future__ import annotations

from functools import cached_property
from onegov.form import Form, FormDefinition
from onegov.form.fields import MultiCheckboxField
from onegov.org import _
from wtforms.fields import BooleanField
from wtforms.fields import DateField
from wtforms.fields import IntegerField
from wtforms.fields import RadioField
from wtforms.fields import TextAreaField
from wtforms.validators import InputRequired
from wtforms.validators import NumberRange
from wtforms.validators import ValidationError


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from onegov.form import FormRegistrationWindow, FormSubmission
    from onegov.org.request import OrgRequest


[docs] class FormRegistrationMessageForm(Form):
[docs] message = TextAreaField( label=_('Your message'), render_kw={'rows': 12}, validators=[InputRequired()] )
[docs] registration_state = MultiCheckboxField( label=_('Send to attendees with status'), choices=[ ('open', _('Open')), ('confirmed', _('Confirmed')), ('cancelled', _('Cancelled')), ], default=['confirmed'], validators=[InputRequired()] )
[docs] def ensure_receivers(self) -> bool | None: receivers = self.receivers if not receivers: assert isinstance(self.registration_state.errors, list) self.registration_state.errors.append( _('No email receivers found for the selection') ) return False return None
@property
[docs] def receivers(self) -> dict[str, FormSubmission]: registration_state = self.registration_state.data assert registration_state is not None return { subm.email: subm for subm in self.model.submissions if subm.registration_state in registration_state }
[docs] class FormRegistrationWindowForm(Form): """ Form to edit registration windows. """ if TYPE_CHECKING:
[docs] model: FormDefinition | FormRegistrationWindow
request: OrgRequest
[docs] start = DateField( label=_('Start'), validators=[InputRequired()] )
[docs] end = DateField( label=_('End'), validators=[InputRequired()] )
[docs] limit_attendees = RadioField( label=_('Limit the number of attendees'), fieldset=_('Attendees'), choices=[ ('yes', _('Yes, limit the number of attendees')), ('no', _('No, allow an unlimited number of attendees')) ], default='yes' )
[docs] limit = IntegerField( label=_('Number of attendees'), fieldset=_('Attendees'), depends_on=('limit_attendees', 'yes'), validators=(InputRequired(), NumberRange(min=1, max=None), ) )
[docs] waitinglist = RadioField( label=_('Waitinglist'), fieldset=_('Attendees'), depends_on=('limit_attendees', 'yes'), choices=[ ('yes', _('Yes, allow for more submissions than available spots')), ('no', _('No, ensure that all submissions can be confirmed')) ], default='yes' )
[docs] stop = BooleanField( label=_('Do not accept any submissions'), fieldset=_('Advanced'), default=False )
[docs] def process_obj( self, obj: FormRegistrationWindow # type:ignore[override] ) -> None: super().process_obj(obj) self.waitinglist.data = obj.overflow and 'yes' or 'no' self.limit_attendees.data = obj.limit and 'yes' or 'no' self.limit.data = obj.limit self.stop.data = not obj.enabled
[docs] def populate_obj( # type:ignore[override] self, obj: FormRegistrationWindow, # type:ignore[override] *args: Any, **kwargs: Any, ) -> None: super().populate_obj(obj, *args, **kwargs) obj.overflow = self.waitinglist.data == 'yes' obj.limit = self.limit_attendees.data == 'yes' and self.limit.data or 0 obj.enabled = not self.stop.data
@cached_property
[docs] def claimed_spots(self) -> int: assert not isinstance(self.model, FormDefinition) return self.model.claimed_spots
@cached_property
[docs] def requested_spots(self) -> int: assert not isinstance(self.model, FormDefinition) return self.model.requested_spots
[docs] def ensure_start_before_end(self) -> bool | None: """ Validate start and end for proper error message. def ensure_start_end(self) would also be run in side the validate function, but the error is not clear. """ if not self.start.data or not self.end.data: return None if self.start.data >= self.end.data: assert isinstance(self.end.errors, list) self.end.errors.append(_('Please use a stop date after the start')) return False return None
[docs] def ensure_no_overlapping_windows(self) -> bool | None: """ Ensure that this registration window does not overlap with other already defined registration windows. """ if not self.start.data or not self.end.data: return None # FIXME: An isinstance check would be nicer but we would need to # stop using Bunch in the tests form: FormDefinition form = getattr(self.model, 'form', self.model) # type:ignore for existing in form.registration_windows: if existing == self.model: continue latest_start = max(self.start.data, existing.start) earliest_end = min(self.end.data, existing.end) delta = (earliest_end - latest_start).days + 1 if delta > 0: # circular from onegov.org.layout import DefaultLayout layout = DefaultLayout(self.model, self.request) msg = _( 'The date range overlaps with an existing registration ' 'window (${range}).', mapping={ 'range': layout.format_date_range( existing.start, existing.end ) } ) assert isinstance(self.start.errors, list) self.start.errors.append(msg) assert isinstance(self.end.errors, list) self.end.errors.append(msg) return False return None
[docs] def validate_limit(self, field: IntegerField) -> None: # new registration windows do not need to enforce limits if isinstance(self.model, FormDefinition): return # nor do unlimited registration windows if not self.limit_attendees.data: return assert self.limit.data is not None # but may be 0 / limit inactive if 0 < self.limit.data < self.claimed_spots: raise ValidationError(_( 'The limit cannot be lower than the already confirmed ' 'number of attendees (${claimed_spots})', mapping={ 'claimed_spots': self.claimed_spots } )) # without a waitinglist we guarantee that all people who sent in # an attendance request are getting a spot (unless they are maunally # cancelled), so we need to take requested spots into account if self.waitinglist.data == 'yes': return if 0 < self.limit.data < (self.requested_spots + self.claimed_spots): raise ValidationError(_( 'The limit cannot be lower than the already confirmed ' 'number attendees (${claimed_spots}) and the number of ' 'pending requests (${pending_requests}). Either enable the ' 'waiting list, process the pending requests or increase the ' 'limit. ', mapping={ 'claimed_spots': self.claimed_spots, 'pending_requests': self.requested_spots } ))