Source code for election_day.forms.election_compound
from __future__ import annotations
from datetime import date
from onegov.core.utils import Bunch
from onegov.core.utils import normalize_for_url
from onegov.election_day import _
from onegov.election_day.layouts import DefaultLayout
from onegov.election_day.models import Election
from onegov.election_day.models import ElectionCompound
from onegov.election_day.models import ElectionCompoundRelationship
from onegov.form import Form
from onegov.form.fields import ChosenSelectMultipleField
from onegov.form.fields import PanelField
from onegov.form.fields import UploadField
from onegov.form.validators import FileSizeLimit
from onegov.form.validators import WhitelistedMimeType
from re import findall
from sqlalchemy import or_
from wtforms.fields import BooleanField
from wtforms.fields import DateField
from wtforms.fields import RadioField
from wtforms.fields import StringField
from wtforms.fields import TextAreaField
from wtforms.fields import URLField
from wtforms.validators import InputRequired
from wtforms.validators import Optional
from wtforms.validators import URL
from wtforms.validators import ValidationError
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from onegov.election_day.request import ElectionDayRequest
from onegov.file import File
from wtforms.fields.choices import _Choice
[docs]
class ElectionCompoundForm(Form):
[docs]
id_hint = PanelField(
label=_('Identifier'),
fieldset=_('Identifier'),
kind='callout',
text=_(
'The ID is used in the URL and might be used somewhere. '
'Changing the ID might break links on external sites!'
)
)
[docs]
id = StringField(
label=_('Identifier'),
fieldset=_('Identifier'),
validators=[
InputRequired()
],
)
[docs]
external_id = StringField(
label=_('External ID'),
fieldset=_('Identifier'),
render_kw={'long_description': _('Used for import if set.')},
)
[docs]
domain = RadioField(
label=_('Domain'),
fieldset=_('Properties'),
choices=[
('canton', _('Cantonal'))
],
default='canton',
validators=[
InputRequired()
]
)
[docs]
domain_elections = RadioField(
label=_('Domain of the elections'),
fieldset=_('Properties'),
validators=[
InputRequired()
]
)
[docs]
date = DateField(
label=_('Date'),
fieldset=_('Properties'),
validators=[
InputRequired()
],
default=date.today
)
[docs]
shortcode = StringField(
label=_('Shortcode'),
fieldset=_('Properties'),
render_kw={'long_description': _('Used for sorting.')}
)
[docs]
region_elections = ChosenSelectMultipleField(
label=_('Elections'),
fieldset=_('Properties'),
choices=[],
validators=[
InputRequired()
],
depends_on=('domain_elections', 'region'),
)
[docs]
district_elections = ChosenSelectMultipleField(
label=_('Elections'),
fieldset=_('Properties'),
choices=[],
validators=[
InputRequired()
],
depends_on=('domain_elections', 'district'),
)
[docs]
municipality_elections = ChosenSelectMultipleField(
label=_('Elections'),
fieldset=_('Properties'),
choices=[],
validators=[
InputRequired()
],
depends_on=('domain_elections', 'municipality'),
)
[docs]
completes_manually = BooleanField(
label=_('Completes manually'),
description=_(
'Enables manual completion of the election compound. '
'All elections are completed only once the election compound '
'is completed.'
),
fieldset=_('Completion'),
render_kw={'force_simple': True}
)
[docs]
manually_completed = BooleanField(
label=_('Completed'),
fieldset=_('Completion'),
depends_on=('completes_manually', 'y'),
render_kw={'force_simple': True}
)
[docs]
title_de = StringField(
label=_('German'),
fieldset=_('Title of the election'),
render_kw={'lang': 'de'}
)
[docs]
title_fr = StringField(
label=_('French'),
fieldset=_('Title of the election'),
render_kw={'lang': 'fr'}
)
[docs]
title_it = StringField(
label=_('Italian'),
fieldset=_('Title of the election'),
render_kw={'lang': 'it'}
)
[docs]
title_rm = StringField(
label=_('Romansh'),
fieldset=_('Title of the election'),
render_kw={'lang': 'rm'}
)
[docs]
short_title_de = StringField(
label=_('German'),
fieldset=_('Short title of the election'),
render_kw={'lang': 'de'}
)
[docs]
short_title_fr = StringField(
label=_('French'),
fieldset=_('Short title of the election'),
render_kw={'lang': 'fr'}
)
[docs]
short_title_it = StringField(
label=_('Italian'),
fieldset=_('Short title of the election'),
render_kw={'lang': 'it'}
)
[docs]
short_title_rm = StringField(
label=_('Romansh'),
fieldset=_('Short title of the election'),
render_kw={'lang': 'rm'}
)
[docs]
explanations_pdf = UploadField(
label=_('Explanations (PDF)'),
validators=[
WhitelistedMimeType({'application/pdf'}),
FileSizeLimit(100 * 1024 * 1024)
],
fieldset=_('Related link')
)
[docs]
upper_apportionment_pdf = UploadField(
label=_('Upper apportionment (PDF)'),
validators=[
WhitelistedMimeType({'application/pdf'}),
FileSizeLimit(100 * 1024 * 1024)
],
fieldset=_('Related link'),
depends_on=('pukelsheim', 'y'),
)
[docs]
lower_apportionment_pdf = UploadField(
label=_('Lower apportionment (PDF)'),
validators=[
WhitelistedMimeType({'application/pdf'}),
FileSizeLimit(100 * 1024 * 1024)
],
fieldset=_('Related link'),
depends_on=('pukelsheim', 'y'),
)
[docs]
pukelsheim = BooleanField(
label=_('Doppelter Pukelsheim'),
fieldset=_('View options'),
description=_('Allows to show the list groups and lists views.'),
render_kw={'force_simple': True}
)
[docs]
voters_counts = BooleanField(
label=_('Voters counts'),
fieldset=_('View options'),
description=_(
'Shows voters counts instead of votes in the party strengths '
'view.'
),
)
[docs]
exact_voters_counts = BooleanField(
label=_('Exact voters counts'),
fieldset=_('View options'),
description=_(
'Shows exact voters counts instead of rounded values.'
),
render_kw={'force_simple': True}
)
[docs]
horizontal_party_strengths = BooleanField(
label=_('Horizonal party strengths chart'),
fieldset=_('View options'),
description=_(
'Shows a horizontal bar chart instead of a vertical bar chart.'
),
depends_on=('show_party_strengths', 'y'),
render_kw={'force_simple': True}
)
[docs]
use_historical_party_results = BooleanField(
label=_('Use party results from the last legislative period'),
fieldset=_('View options'),
description=_(
'Requires party results. Requires a related election from another '
'legislative period with party results. Requires that both '
'elections use the same party IDs.'
),
render_kw={'force_simple': True}
)
[docs]
show_seat_allocation = BooleanField(
label=_('Seat allocation'),
description=_(
'Shows a tab with the comparison of seat allocation as a bar '
'chart. Requires party results.'
),
fieldset=_('Views'),
render_kw={'force_simple': True},
)
[docs]
show_list_groups = BooleanField(
label=_('List groups'),
description=_(
'Shows a tab with list group results. Requires party results with '
'voters counts. Only if Doppelter Pukelsheim.'
),
fieldset=_('Views'),
render_kw={'force_simple': True},
depends_on=('pukelsheim', 'y'),
)
[docs]
show_party_strengths = BooleanField(
label=_('Party strengths'),
description=_(
'Shows a tab with the comparison of party strengths as a bar '
'chart. Requires party results.'
),
fieldset=_('Views'),
render_kw={'force_simple': True}
)
[docs]
show_party_panachage = BooleanField(
label=_('Panachage'),
description=_(
'Shows a tab with the panachage. Requires party results.'
),
fieldset=_('Views'),
render_kw={'force_simple': True}
)
[docs]
color_hint = PanelField(
label=_('Color suggestions'),
hide_label=False,
fieldset=_('Colors'),
text=(
'AL #a74c97\n'
'BDP #a9cf00\n'
'Die Mitte #d28b00\n'
'EDU #7f6b65\n'
'EVP #e3c700\n'
'FDP #0084c7\n'
'GLP #aeca00\n'
'GRÜNE #54ba00\n'
'Piraten #333333\n'
'SP #c31906\n'
'SVP #408b3d\n'
),
kind='',
)
[docs]
def parse_colors(self, text: str | None) -> dict[str, str]:
if not text:
return {}
result = {
key.strip(): value
for key, value in findall(r'(.+)\s+(\#[0-9a-fA-F]{6})', text)
}
if len(text.strip().splitlines()) != len(result):
raise ValueError('Could not parse colors')
return result
[docs]
def validate_colors(self, field: TextAreaField) -> None:
try:
self.parse_colors(field.data)
except Exception as exception:
raise ValidationError(
_('Invalid color definitions')
) from exception
[docs]
def validate_id(self, field: StringField) -> None:
if normalize_for_url(field.data or '') != field.data:
raise ValidationError(_('Invalid ID'))
if self.model.id != field.data:
query = self.request.session.query(ElectionCompound.id)
query = query.filter_by(id=field.data)
if query.first():
raise ValidationError(_('ID already exists'))
[docs]
def validate_external_id(self, field: StringField) -> None:
if field.data:
if (
not hasattr(self.model, 'external_id')
or self.model.external_id != field.data
):
for cls in (Election, ElectionCompound):
query = self.request.session.query(cls)
query = query.filter_by(external_id=field.data)
if query.first():
raise ValidationError(_('ID already exists'))
[docs]
def on_request(self) -> None:
principal = self.request.app.principal
self.domain_elections.choices = []
for domain in ('region', 'district', 'municipality'):
if domain in principal.domains_election:
self.domain_elections.choices.append((
domain,
self.request.translate(principal.domains_election[domain])
))
self.title_de.validators = []
self.title_fr.validators = []
self.title_it.validators = []
self.title_rm.validators = []
default_locale = self.request.default_locale or ''
if default_locale.startswith('de'):
self.title_de.validators.append(InputRequired())
if default_locale.startswith('fr'):
self.title_fr.validators.append(InputRequired())
if default_locale.startswith('it'):
self.title_de.validators.append(InputRequired())
if default_locale.startswith('rm'):
self.title_de.validators.append(InputRequired())
layout = DefaultLayout(None, self.request)
query = self.request.session.query(Election)
query = query.order_by(Election.date.desc(), Election.shortcode)
query = query.filter(Election.type == 'proporz')
self.region_elections.choices = [
(
item.id,
'{} {} {}'.format(
layout.format_date(item.date, 'date'),
item.shortcode or '',
item.title,
).replace(' ', ' ')
) for item in query.filter(Election.domain == 'region')
]
self.district_elections.choices = [
(
item.id,
'{} {} {}'.format(
layout.format_date(item.date, 'date'),
item.shortcode or '',
item.title,
).replace(' ', ' ')
) for item in query.filter(Election.domain == 'district')
]
self.municipality_elections.choices = [
(
item.id,
'{} {} {}'.format(
layout.format_date(item.date, 'date'),
item.shortcode or '',
item.title,
).replace(' ', ' ')
) for item in query.filter(Election.domain == 'municipality')
]
query = self.request.session.query(ElectionCompound)
query = query.order_by(
ElectionCompound.date.desc(),
ElectionCompound.shortcode
)
choices: list[_Choice] = [
(
compound.id,
'{} {} {}'.format(
layout.format_date(compound.date, 'date'),
compound.shortcode or '',
compound.title,
).strip().replace(' ', ' ')
) for compound in query
]
self.related_compounds_historical.choices = choices
self.related_compounds_other.choices = choices
[docs]
def update_realtionships(
self,
model: ElectionCompound,
type_: str
) -> None:
# use symetric relationships
session = self.request.session
query = session.query(ElectionCompoundRelationship)
query = query.filter(
or_(
ElectionCompoundRelationship.source_id == model.id,
ElectionCompoundRelationship.target_id == model.id,
),
ElectionCompoundRelationship.type == type_
)
for relationship in query:
session.delete(relationship)
data = getattr(self, f'related_compounds_{type_}', Bunch(data=[])).data
for id_ in data:
if not model.id:
model.id = model.id_from_title(session)
session.add(
ElectionCompoundRelationship(
source_id=model.id, target_id=id_, type=type_
)
)
session.add(
ElectionCompoundRelationship(
source_id=id_, target_id=model.id, type=type_
)
)
[docs]
def update_model(self, model: ElectionCompound) -> None:
if self.id and self.id.data:
model.id = self.id.data
model.external_id = self.external_id.data
model.domain = self.domain.data
model.domain_elections = self.domain_elections.data
assert self.date.data is not None
model.date = self.date.data
model.shortcode = self.shortcode.data
model.related_link = self.related_link.data
model.show_seat_allocation = self.show_seat_allocation.data
model.show_list_groups = self.show_list_groups.data
model.show_party_strengths = self.show_party_strengths.data
model.show_party_panachage = self.show_party_panachage.data
model.pukelsheim = self.pukelsheim.data
model.completes_manually = self.completes_manually.data
model.manually_completed = self.manually_completed.data
model.voters_counts = self.voters_counts.data
model.exact_voters_counts = self.exact_voters_counts.data
model.horizontal_party_strengths = self.horizontal_party_strengths.data
model.use_historical_party_results = (
self.use_historical_party_results.data)
model.elections = []
query = self.request.session.query(Election)
if self.domain_elections.data == 'region':
if self.region_elections.data:
model.elections = query.filter(
Election.id.in_(self.region_elections.data)
).all()
if self.domain_elections.data == 'district':
if self.district_elections.data:
model.elections = query.filter(
Election.id.in_(self.district_elections.data)
).all()
if self.domain_elections.data == 'municipality':
if self.municipality_elections.data:
model.elections = query.filter(
Election.id.in_(self.municipality_elections.data)
).all()
titles = {}
if self.title_de.data:
titles['de_CH'] = self.title_de.data
if self.title_fr.data:
titles['fr_CH'] = self.title_fr.data
if self.title_it.data:
titles['it_CH'] = self.title_it.data
if self.title_rm.data:
titles['rm_CH'] = self.title_rm.data
model.title_translations = titles
titles = {}
if self.short_title_de.data:
titles['de_CH'] = self.short_title_de.data
if self.short_title_fr.data:
titles['fr_CH'] = self.short_title_fr.data
if self.short_title_it.data:
titles['it_CH'] = self.short_title_it.data
if self.short_title_rm.data:
titles['rm_CH'] = self.short_title_rm.data
model.short_title_translations = titles
link_labels = {}
if self.related_link_label_de.data:
link_labels['de_CH'] = self.related_link_label_de.data
if self.related_link_label_fr.data:
link_labels['fr_CH'] = self.related_link_label_fr.data
if self.related_link_label_it.data:
link_labels['it_CH'] = self.related_link_label_it.data
if self.related_link_label_rm.data:
link_labels['rm_CH'] = self.related_link_label_rm.data
model.related_link_label = link_labels
for file in (
'explanations_pdf',
'upper_apportionment_pdf',
'lower_apportionment_pdf'
):
field = getattr(self, file)
action = getattr(field, 'action', '')
if action == 'delete':
delattr(model, file)
if action == 'replace' and field.data:
setattr(model, file, (field.file, field.filename,))
model.colors = self.parse_colors(self.colors.data)
with self.request.session.no_autoflush:
self.update_realtionships(model, 'historical')
self.update_realtionships(model, 'other')
[docs]
def apply_model(self, model: ElectionCompound) -> None:
self.id.data = model.id
self.external_id.data = model.external_id
titles = model.title_translations or {}
self.title_de.data = titles.get('de_CH')
self.title_fr.data = titles.get('fr_CH')
self.title_it.data = titles.get('it_CH')
self.title_rm.data = titles.get('rm_CH')
titles = model.short_title_translations or {}
self.short_title_de.data = titles.get('de_CH')
self.short_title_fr.data = titles.get('fr_CH')
self.short_title_it.data = titles.get('it_CH')
self.short_title_rm.data = titles.get('rm_CH')
link_labels = model.related_link_label or {}
self.related_link_label_de.data = link_labels.get('de_CH', '')
self.related_link_label_fr.data = link_labels.get('fr_CH', '')
self.related_link_label_it.data = link_labels.get('it_CH', '')
self.related_link_label_rm.data = link_labels.get('rm_CH', '')
for file_attr in (
'explanations_pdf',
'upper_apportionment_pdf',
'lower_apportionment_pdf'
):
field = getattr(self, file_attr)
file: File = getattr(model, file_attr)
if file:
field.data = {
'filename': file.reference.filename,
'size': file.reference.file.content_length,
'mimetype': file.reference.content_type
}
self.domain.data = model.domain
self.domain_elections.data = model.domain_elections
self.date.data = model.date
self.shortcode.data = model.shortcode
self.related_link.data = model.related_link
self.pukelsheim.data = model.pukelsheim
self.completes_manually.data = model.completes_manually
self.manually_completed.data = model.manually_completed
self.voters_counts.data = model.voters_counts
self.exact_voters_counts.data = model.exact_voters_counts
self.horizontal_party_strengths.data = model.horizontal_party_strengths
self.use_historical_party_results.data = (
model.use_historical_party_results)
self.show_seat_allocation.data = model.show_seat_allocation
self.show_list_groups.data = model.show_list_groups
self.show_party_strengths.data = model.show_party_strengths
self.show_party_panachage.data = model.show_party_panachage
self.region_elections.data = []
if model.domain_elections == 'region':
self.region_elections.data = [e.id for e in model.elections]
self.district_elections.data = []
if model.domain_elections == 'district':
self.district_elections.data = [e.id for e in model.elections]
self.municipality_elections.data = []
if model.domain_elections == 'municipality':
self.municipality_elections.data = [e.id for e in model.elections]
self.colors.data = '\n'.join(
f'{name} {model.colors[name]}' for name in sorted(model.colors)
)
self.related_compounds_historical.choices = [
choice for choice in self.related_compounds_historical.choices
if choice[0] != model.id
]
self.related_compounds_other.choices = [
choice for choice in self.related_compounds_other.choices
if choice[0] != model.id
]
self.related_compounds_historical.data = [
association.target_id for association in model.related_compounds
if association.type == 'historical'
]
self.related_compounds_other.data = [
association.target_id for association in model.related_compounds
if association.type == 'other'
]