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
# include all fields used below so we can filter them out
# when we merge this form with the custom form definition
[docs]
class FindYourSpotForm(Form):
if TYPE_CHECKING:
[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 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
)