Source code for landsgemeinde.forms.agenda
from __future__ import annotations
from onegov.form.fields import TagsField
from onegov.form.fields import TimeField
from onegov.form.fields import UploadField
from onegov.form.forms import NamedFileForm
from onegov.form.validators import FileSizeLimit
from onegov.form.validators import WhitelistedMimeType
from onegov.landsgemeinde import _
from onegov.landsgemeinde.layouts import DefaultLayout
from onegov.landsgemeinde.models import AgendaItem
from onegov.landsgemeinde.models.agenda import STATES
from onegov.landsgemeinde.utils import timestamp_to_seconds
from onegov.org.forms.fields import HtmlField
from sqlalchemy import func
from wtforms.fields import BooleanField
from wtforms.fields import IntegerField
from wtforms.fields import RadioField
from wtforms.fields import StringField
from wtforms.fields import TextAreaField
from wtforms.validators import InputRequired
from wtforms.validators import NumberRange
from wtforms.validators import Optional
from wtforms.validators import ValidationError
from typing import Any
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from onegov.landsgemeinde.request import LandsgemeindeRequest
[docs]
class AgendaItemForm(NamedFileForm):
[docs]
state = RadioField(
_('State'),
fieldset=_('General'),
choices=list(STATES.items()),
validators=[
InputRequired()
],
default=next(iter(STATES.keys()))
)
[docs]
memorial_pdf = UploadField(
label=_('Excerpt from the Memorial (PDF)'),
fieldset=_('Memorial'),
validators=[
WhitelistedMimeType({'application/pdf'}),
FileSizeLimit(100 * 1024 * 1024)
]
)
[docs]
memorial_page = IntegerField(
description=_(
'Links to the whole memorial (if there is one linked to the '
'assembly), but opens it on the chosen page number'
),
label=_('Alternatively: Page from the Memorial'),
fieldset=_('Memorial'),
validators=[
NumberRange(min=1),
Optional()
],
)
[docs]
start_time = TimeField(
label=_('Start'),
fieldset=_('Progress'),
render_kw={
'long_description': _(
'Automatically updated when agenda item changed to ongoing.'
),
'step': 1
},
format='%H:%M:%S',
validators=[
Optional()
],
)
[docs]
calculated_timestamp = StringField(
label=_('Calculated video timestamp'),
fieldset=_('Progress'),
render_kw={
'long_description': _(
'Calculated automatically based on the start time of the '
'agenda item and the start time of of the livestream of the '
'assembly .'
),
'readonly': True,
'step': 1
},
validators=[
Optional()
],
)
[docs]
video_timestamp = StringField(
label=_('Manual video timestamp'),
fieldset=_('Progress'),
description='1h2m1s',
render_kw={
'long_description': _('Overrides the calculated video timestamp.'),
'step': 1
},
validators=[
Optional()
],
)
@property
[docs]
def next_number(self) -> int:
query = self.request.session.query(func.max(AgendaItem.number))
query = query.filter(AgendaItem.assembly_id == self.model.assembly.id)
return (query.scalar() or 0) + 1
[docs]
def on_request(self) -> None:
DefaultLayout(self.model, self.request)
self.request.include('redactor')
self.request.include('editor')
self.request.include('tags-input')
self.request.include('start_time')
[docs]
def get_useful_data(self) -> dict[str, Any]: # type:ignore[override]
data = super().get_useful_data(exclude={'calculated_timestamp'})
data['assembly_id'] = self.model.assembly.id
return data
[docs]
def validate_number(self, field: IntegerField) -> None:
if field.data:
session = self.request.session
query = session.query(AgendaItem)
query = query.filter(
AgendaItem.assembly_id == self.model.assembly.id,
AgendaItem.number == field.data
)
if isinstance(self.model, AgendaItem):
query = query.filter(AgendaItem.id != self.model.id)
if session.query(query.exists()).scalar():
raise ValidationError(_('Number already used.'))
[docs]
def validate_video_timestamp(self, field: StringField) -> None:
if field.data and timestamp_to_seconds(field.data) is None:
raise ValidationError(_('Invalid timestamp.'))
[docs]
def populate_obj(self, obj: AgendaItem) -> None: # type:ignore[override]
super().populate_obj(obj, exclude={'calculated_timestamp'})