Source code for swissvotes.views.vote

from __future__ import annotations

from morepath import redirect
from onegov.core.crypto import random_token
from onegov.core.security import Private
from onegov.core.security import Public
from onegov.core.static import StaticFile
from onegov.core.utils import normalize_for_url
from onegov.file.integration import render_depot_file
from onegov.file.utils import as_fileintent
from onegov.form import Form
from onegov.swissvotes import _
from onegov.swissvotes import SwissvotesApp
from onegov.swissvotes.forms import AttachmentsForm
from onegov.swissvotes.forms import AttachmentsSearchForm
from onegov.swissvotes.layouts import DeleteVoteAttachmentLayout
from onegov.swissvotes.layouts import DeleteVoteLayout
from onegov.swissvotes.layouts import ManageCampaingMaterialLayout
from onegov.swissvotes.layouts import ManageCampaingMaterialNayLayout
from onegov.swissvotes.layouts import ManageCampaingMaterialYeaLayout
from onegov.swissvotes.layouts import UploadVoteAttachemtsLayout
from onegov.swissvotes.layouts import VoteCampaignMaterialLayout
from onegov.swissvotes.layouts import VoteLayout
from onegov.swissvotes.layouts import VoteStrengthsLayout
from onegov.swissvotes.models import Actor
from onegov.swissvotes.models import SwissVote
from onegov.swissvotes.models import SwissVoteFile
from webob.exc import HTTPBadRequest
from webob.exc import HTTPNotFound
from webob.exc import HTTPUnsupportedMediaType


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Callable
    from decimal import Decimal
    from depot.io.interfaces import StoredFile
    from onegov.core.types import JSON_ro
    from onegov.core.types import RenderData
    from onegov.swissvotes.request import SwissvotesRequest
    from webob import Response


@SwissvotesApp.form(
    model=SwissVote,
    permission=Public,
    template='vote.pt',
    form=AttachmentsSearchForm
)
[docs] def view_vote( self: SwissVote, request: SwissvotesRequest, form: AttachmentsSearchForm ) -> RenderData: layout = VoteLayout(self, request) query = request.session.query(SwissVote) prev_vote = ( query.order_by(SwissVote.bfs_number.desc()) .filter(SwissVote.bfs_number < self.bfs_number).first() ) next_vote = ( query.order_by(SwissVote.bfs_number.asc()) .filter(SwissVote.bfs_number > self.bfs_number).first() ) if self.bfs_map_host: request.content_security_policy.default_src |= {self.bfs_map_host} form.action += '#search' if not form.errors: form.apply_model(self) return { 'layout': layout, 'prev': prev_vote, 'next': next_vote, 'map_preview': request.link(StaticFile('images/map-preview.png')), 'posters': self.posters(request), 'form': form, 'search_results': layout.search_results }
@SwissvotesApp.json( model=SwissVote, permission=Public, name='percentages' )
[docs] def view_vote_percentages( self: SwissVote, request: SwissvotesRequest ) -> JSON_ro: def create( text: str, text_label: str | None = None, percentage: Decimal | None = None, yeas_p: Decimal | None = None, nays_p: Decimal | None = None, yeas: Decimal | int | None = None, nays: Decimal | int | None = None, code: str | None = None, yea_label: str | None = None, nay_label: str | None = None, none_label: str | None = None, empty: bool = False ) -> dict[str, JSON_ro]: translate = request.translate result: dict[str, JSON_ro] = { 'text': translate(text), 'text_label': translate(text_label) if text_label else '', 'yea': 0.0, 'nay': 0.0, 'none': 0.0, 'yea_label': '', 'nay_label': '', 'none_label': '', 'empty': empty } default_yea_label = _('${x}% yea') default_nay_label = _('${x}% nay') yea_label_no_perc = _('${x} yea') nay_label_no_per = _('${x} nay') if self._legal_form == 5: default_yea_label = _('${x}% for the popular initiative') default_nay_label = _('${x}% for the counter-proposal') yea_label_no_perc = _('${x} for the popular initiative') nay_label_no_per = _('${x} for the counter-proposal') if percentage is not None: yea = round(float(percentage), 1) nay = round(float(100 - percentage), 1) yea_label = yea_label or default_yea_label nay_label = nay_label or default_nay_label result.update({ 'yea': yea, 'nay': nay, 'yea_label': translate(_(yea_label, mapping={'x': yea})), 'nay_label': translate(_(nay_label, mapping={'x': nay})), }) elif yeas_p is not None and nays_p is not None: yea = round(float(yeas_p), 1) nay = round(float(nays_p), 1) none = round(float(100 - yeas_p - nays_p), 1) yea_label = yea_label or default_yea_label nay_label = nay_label or default_nay_label none_label = none_label or _('${x}% none') result.update({ 'yea': yea, 'nay': nay, 'none': none, 'yea_label': translate(_(yea_label, mapping={'x': yea})), 'nay_label': translate(_(nay_label, mapping={'x': nay})), 'none_label': translate(_(none_label, mapping={'x': none})) }) elif yeas is not None and nays is not None: yea = round(float(100 * (yeas / (yeas + nays))), 1) nay = round(float(100 * (nays / (yeas + nays))), 1) yea_label = yea_label or yea_label_no_perc nay_label = nay_label or nay_label_no_per result.update({ 'yea': yea, 'nay': nay, 'yea_label': translate(_(yea_label, mapping={'x': yeas})), 'nay_label': translate(_(nay_label, mapping={'x': nays})), }) elif code is not None: value = getattr(self, f'_{code}') label = getattr(self, code) if value in (1, 9): result.update({ 'yea': True, 'yea_label': translate(label) }) if value in (0, 2, 8): result.update({ 'nay': True, 'nay_label': translate(label) }) if value == 3: result.update({ 'none': True, 'none_label': translate(label) }) return result results = [ create( _('People'), percentage=self.result_people_yeas_p, code='result_people_accepted' ), create( _('Cantons'), yeas=self.result_cantons_yeas, nays=self.result_cantons_nays, code='result_cantons_accepted' ), create('', empty=True), create( _('Federal Council'), text_label=_('Position of the Federal Council'), code='position_federal_council' ), create( _('National Council'), yeas=self.position_national_council_yeas, nays=self.position_national_council_nays, code='position_national_council' ), create( _('Council of States'), yeas=self.position_council_of_states_yeas, nays=self.position_council_of_states_nays, code='position_council_of_states' ), create( _('Party slogans'), text_label=_('Recommendations by political parties'), yeas_p=self.national_council_share_yeas, nays_p=self.national_council_share_nays, yea_label=_( 'Electoral shares of parties: ' 'Parties recommending Yes ${x}%' ) if self._legal_form != 5 else _( 'Electoral shares of parties: ' 'Parties preferring the initiative ${x}%' ), nay_label=_( 'Electoral shares of parties: ' 'Parties recommending No ${x}%' ) if self._legal_form != 5 else _( 'Electoral shares of parties: ' 'Parties preferring the counter-proposal ${x}%' ), none_label=_( 'Electoral shares of parties: neutral/unknown ${x}%' ) ) ] results = [ result for result in results if any(( result['yea'], result['nay'], result['none'], result['empty'] )) ] return { 'results': results, 'title': self.title }
@SwissvotesApp.html( model=SwissVote, permission=Public, template='strengths.pt', name='strengths' )
[docs] def view_vote_strengths( self: SwissVote, request: SwissvotesRequest ) -> RenderData: return { 'layout': VoteStrengthsLayout(self, request), 'Actor': Actor }
@SwissvotesApp.html( model=SwissVote, permission=Public, template='campaign_material.pt', name='campaign-material' )
[docs] def view_vote_campaign_material( self: SwissVote, request: SwissvotesRequest ) -> RenderData: layout = VoteCampaignMaterialLayout(self, request) files = sorted( ( (file, layout.metadata(file.filename)) for file in self.campaign_material_other ), key=lambda it: (it[1].get('order', 999), it[1].get('title', '')) ) return { 'layout': layout, 'files': files }
@SwissvotesApp.form( model=SwissVote, permission=Private, form=AttachmentsForm, template='form.pt', name='upload' )
[docs] def upload_vote_attachments( self: SwissVote, request: SwissvotesRequest, form: AttachmentsForm ) -> RenderData | Response: if form.submitted(request): form.update_model(self) request.message(_('Attachments updated'), 'success') return request.redirect(request.link(self)) if not form.errors: form.apply_model(self) layout = UploadVoteAttachemtsLayout(self, request) return { 'layout': layout, 'form': form, 'button_text': _('Upload'), 'cancel': request.link(self) }
@SwissvotesApp.form( model=SwissVote, permission=Private, form=Form, template='form.pt', name='delete' )
[docs] def delete_vote( self: SwissVote, request: SwissvotesRequest, form: Form ) -> RenderData | Response: layout = DeleteVoteLayout(self, request) if form.submitted(request): request.session.delete(self) request.message(_('Vote deleted'), 'success') return request.redirect(layout.votes_url) return { 'layout': layout, 'form': form, 'subtitle': self.title, 'message': _( 'Do you really want to delete "${item}"?', mapping={'item': self.title} ), 'button_text': _('Delete'), 'button_class': 'alert', 'cancel': request.link(self) }
@SwissvotesApp.view( model=SwissVoteFile, render=render_depot_file, permission=Public )
[docs] def view_file( self: SwissVoteFile, request: SwissvotesRequest ) -> StoredFile: @request.after def set_filename(response: Response) -> None: attribute = SwissVote.localized_files().get(self.name.split('-')[0]) if attribute: bfs_number = self.linked_swissvotes[0].bfs_number extension = attribute.extension title = normalize_for_url(request.translate(attribute.label)) response.headers['Content-Disposition'] = ( f'inline; filename={bfs_number}-{title}.{extension}' ) return self.reference.file
[docs] def create_static_file_view( attribute: str, locale: str ) -> Callable[[SwissVote, SwissvotesRequest], Response]: def static_view( self: SwissVote, request: SwissvotesRequest ) -> Response: file = self.get_file(attribute, locale=locale, fallback=False) if not file: raise HTTPNotFound() return request.redirect(request.link(file)) return static_view
for attribute_name, attribute in SwissVote.localized_files().items(): for locale, view_name in attribute.static_views.items(): SwissvotesApp.view( model=SwissVote, permission=Public, name=view_name )(create_static_file_view(attribute_name, locale)) @SwissvotesApp.html( model=SwissVote, template='attachments.pt', name='manage-campaign-material', permission=Private )
[docs] def view_manage_campaign_material( self: SwissVote, request: SwissvotesRequest ) -> RenderData: layout = ManageCampaingMaterialLayout(self, request) return { 'layout': layout, 'title': self.title, 'upload_url': layout.csrf_protected_url( request.link(self, name='upload-campaign-material') ), 'files': self.campaign_material_other, 'notice': self, }
@SwissvotesApp.view( model=SwissVote, name='upload-campaign-material', permission=Private, request_method='POST' )
[docs] def upload_manage_campaign_material( self: SwissVote, request: SwissvotesRequest ) -> None: request.assert_valid_csrf_token() fs = request.params.get('file', '') if isinstance(fs, str): # malformed formdata raise HTTPBadRequest() attachment = SwissVoteFile(id=random_token()) attachment.name = f'campaign_material_other-{fs.filename}' attachment.reference = as_fileintent(fs.file, fs.filename) if attachment.reference.content_type != 'application/pdf': raise HTTPUnsupportedMediaType() self.files.append(attachment)
@SwissvotesApp.html( model=SwissVote, template='attachments.pt', name='manage-campaign-material-yea', permission=Private )
[docs] def view_manage_campaign_material_yea( self: SwissVote, request: SwissvotesRequest ) -> RenderData: layout = ManageCampaingMaterialYeaLayout(self, request) return { 'layout': layout, 'title': self.title, 'upload_url': layout.csrf_protected_url( request.link(self, name='upload-campaign-material-yea') ), 'files': self.campaign_material_yea, 'notice': self, }
@SwissvotesApp.view( model=SwissVote, name='upload-campaign-material-yea', permission=Private, request_method='POST' )
[docs] def upload_manage_campaign_material_yea( self: SwissVote, request: SwissvotesRequest ) -> None: request.assert_valid_csrf_token() fs = request.params.get('file', '') if isinstance(fs, str): # malformed formdata raise HTTPBadRequest() attachment = SwissVoteFile(id=random_token()) attachment.name = f'campaign_material_yea-{fs.filename}' attachment.reference = as_fileintent(fs.file, fs.filename) if attachment.reference.content_type not in ('image/jpeg', 'image/png'): raise HTTPUnsupportedMediaType() self.files.append(attachment)
@SwissvotesApp.html( model=SwissVote, template='attachments.pt', name='manage-campaign-material-nay', permission=Private )
[docs] def view_manage_campaign_material_nay( self: SwissVote, request: SwissvotesRequest ) -> RenderData: layout = ManageCampaingMaterialNayLayout(self, request) return { 'layout': layout, 'title': self.title, 'upload_url': layout.csrf_protected_url( request.link(self, name='upload-campaign-material-nay') ), 'files': self.campaign_material_nay, 'notice': self, }
@SwissvotesApp.view( model=SwissVote, name='upload-campaign-material-nay', permission=Private, request_method='POST' )
[docs] def upload_manage_campaign_material_nay( self: SwissVote, request: SwissvotesRequest ) -> None: request.assert_valid_csrf_token() fs = request.params.get('file', '') if isinstance(fs, str): # malformed formdata raise HTTPBadRequest() attachment = SwissVoteFile(id=random_token()) attachment.name = f'campaign_material_nay-{fs.filename}' attachment.reference = as_fileintent(fs.file, fs.filename) if attachment.reference.content_type not in ('image/jpeg', 'image/png'): raise HTTPUnsupportedMediaType() self.files.append(attachment)
@SwissvotesApp.form( model=SwissVoteFile, name='delete', template='form.pt', permission=Private, form=Form )
[docs] def delete_vote_attachment( self: SwissVoteFile, request: SwissvotesRequest, form: Form ) -> RenderData | Response: layout = DeleteVoteAttachmentLayout(self, request) name = 'manage-campaign-material' if 'yea' in self.name: name += '-yea' if 'nay' in self.name: name += '-nay' url = request.link(layout.parent, name) if form.submitted(request): request.session.delete(self) request.message(_('Attachment deleted.'), 'success') return redirect(url) return { 'message': _( 'Do you really want to delete "${item}"?', mapping={'item': self.filename} ), 'layout': layout, 'form': form, 'title': self.filename, 'subtitle': _('Delete'), 'button_text': _('Delete'), 'button_class': 'alert', 'cancel': url }