Source code for org.forms.reservation

from __future__ import annotations

from datetime import date, time, timedelta
from functools import cached_property
from uuid import UUID
from wtforms.fields import DateField
from wtforms.fields import EmailField
from wtforms.fields import RadioField
from wtforms.validators import DataRequired, Email, InputRequired

from onegov.core.csv import convert_list_of_list_of_dicts_to_xlsx
from onegov.form import Form
from onegov.form.fields import MultiCheckboxField, TimeField
from onegov.org import _
from onegov.org.forms.util import WEEKDAYS


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from _typeshed import SupportsRichComparison
    from collections.abc import Callable, Iterable, Sequence
    from onegov.org.request import OrgRequest
    from onegov.reservation import Resource
    from typing import TypeAlias
    from .allocation import DateContainer

[docs] StrKeyFunc: TypeAlias = Callable[[str], SupportsRichComparison]
# include all fields used below so we can filter them out # when we merge this form with the custom form definition
[docs] RESERVED_FIELDS: list[str] = ['email']
[docs] class ReservationForm(Form):
[docs] reserved_fields = RESERVED_FIELDS
[docs] email = EmailField( label=_('E-Mail'), validators=[InputRequired(), Email()] )
[docs] class FindYourSpotForm(Form): if TYPE_CHECKING:
[docs] request: OrgRequest
[docs] start = DateField( label=_('From'), validators=[InputRequired()])
[docs] end = DateField( label=_('Until'), validators=[InputRequired()])
[docs] start_time = TimeField( label=_('Start'), description=_('HH:MM'), default=time(7), validators=[DataRequired()])
[docs] end_time = TimeField( label=_('End'), description=_('HH:MM'), default=time(22), validators=[DataRequired()])
[docs] rooms = MultiCheckboxField( label=_('Rooms'), choices=(), coerce=lambda v: UUID(v) if isinstance(v, str) else v, validators=[InputRequired()], render_kw={ 'prefix_label': False, 'class_': 'oneline-checkboxes' })
[docs] weekdays = MultiCheckboxField( label=_('Weekdays'), choices=WEEKDAYS, coerce=int, default=[v for v, l in WEEKDAYS[:5]], validators=[InputRequired()], render_kw={ 'prefix_label': False, 'class_': 'oneline-checkboxes' })
[docs] on_holidays = RadioField( label=_('On holidays'), choices=( ('yes', _('Yes')), ('no', _('No')) ), default='no')
[docs] during_school_holidays = RadioField( label=_('During school holidays'), choices=( ('yes', _('Yes')), ('no', _('No')) ), default='no')
[docs] def on_request(self) -> None: if not self.request.app.org.holidays: self.delete_field('on_holidays') if not self.request.app.org.has_school_holidays: self.delete_field('during_school_holidays') if not self.request.POST: # by default search one week from now self.start.data = date.today() self.end.data = self.start.data + timedelta(days=7)
[docs] def apply_rooms(self, rooms: Sequence[Resource]) -> None: if len(rooms) < 2: # no need to filter self.delete_field('rooms') return self.rooms.choices = [(room.id, room.title) for room in rooms] if not self.request.POST: # select all rooms by default self.rooms.data = [room.id for room in rooms]
[docs] def ensure_future_start(self) -> bool | None: if self.start.data: if self.start.data < date.today(): assert isinstance(self.start.errors, list) self.start.errors.append(_('Start date in past')) return False return None
[docs] def ensure_start_before_end(self) -> bool | None: if self.start.data and self.end.data: if self.start.data > self.end.data: assert isinstance(self.start.errors, list) self.start.errors.append(_('Start date before end date')) return False return None
[docs] def ensure_start_time_before_end_time(self) -> bool | None: start = self.start_time.data end = self.end_time.data if start and end: if ( start.hour > end.hour or ( start.hour == end.hour and start.minute >= end.minute ) ): assert isinstance(self.start_time.errors, list) self.start_time.errors.append(_('Start time before end time')) return False return None
@cached_property
[docs] def exceptions(self) -> DateContainer: if not hasattr(self, 'request'): return () if not self.on_holidays: return () if self.on_holidays.data == 'yes': return () return self.request.app.org.holidays
@cached_property
[docs] def ranged_exceptions(self) -> Sequence[tuple[date, date]]: if not hasattr(self, 'request'): return () if not self.during_school_holidays: return () if self.during_school_holidays.data == 'yes': return () return tuple(self.request.app.org.school_holidays)
[docs] def is_excluded(self, date: date) -> bool: if date in self.exceptions: return True if date.weekday() not in (self.weekdays.data or ()): return True for start, end in self.ranged_exceptions: if start <= date <= end: return True return False
[docs] class ExportToExcelWorksheets(Form): """ A form providing the export of multiple reservations into Worksheets """ @property
[docs] def format(self) -> str: return 'xlsx'
[docs] def as_multiple_export_response( self, keys: Sequence[StrKeyFunc | None] | None, results: Sequence[Iterable[dict[str, Any]]], titles: Sequence[str] ) -> bytes: return convert_list_of_list_of_dicts_to_xlsx( results, titles_list=titles, key_list=keys )