Source code for swissvotes.views.page

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.file.utils import as_fileintent
from onegov.form import Form
from onegov.swissvotes import _
from onegov.swissvotes import SwissvotesApp
from onegov.swissvotes.collections import TranslatablePageCollection
from onegov.swissvotes.forms import PageForm
from onegov.swissvotes.layouts import AddPageLayout
from onegov.swissvotes.layouts import DeletePageAttachmentLayout
from onegov.swissvotes.layouts import DeletePageLayout
from onegov.swissvotes.layouts import EditPageLayout
from onegov.swissvotes.layouts import ManagePageAttachmentsLayout
from onegov.swissvotes.layouts import ManagePageSliderImagesLayout
from onegov.swissvotes.layouts import PageLayout
from onegov.swissvotes.models import TranslatablePage
from onegov.swissvotes.models import TranslatablePageFile
from onegov.swissvotes.models import TranslatablePageMove
from random import sample
from webob.exc import HTTPBadRequest
from webob.exc import HTTPNotFound
from webob.exc import HTTPUnsupportedMediaType


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from onegov.core.types import RenderData
    from onegov.swissvotes.request import SwissvotesRequest
    from webob import Response


@SwissvotesApp.html(
    model=TranslatablePage,
    permission=Public,
    name='kurzbeschreibung-de.pdf'
)
[docs] def brief_desc_static_de( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file_by_locale('QUELLEN', 'de_CH') if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='kurzbeschreibung-fr.pdf' )
[docs] def brief_desc_static_fr( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file_by_locale('REFERENCES des descriptifs', 'fr_CH') if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='kurzbeschreibung-en.pdf' )
[docs] def brief_desc_static_en( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file_by_locale('REFERENCES for descriptions', 'en_US') if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='swissvotes_dataset.csv' )
[docs] def dataset_csv_static( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file('DATASET CSV', request) if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='swissvotes_dataset.xlsx' )
[docs] def dataset_xlsx_static( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file('DATASET XLSX', request) if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='codebook-de.pdf' )
[docs] def codebook_de_static( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file_by_locale('CODEBOOK', 'de_CH') if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='codebook-fr.pdf' )
[docs] def codebook_fr_static( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file_by_locale('CODEBOOK', 'fr_CH') if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, permission=Public, name='codebook-en.pdf' )
[docs] def codebook_us_static( self: TranslatablePage, request: SwissvotesRequest ) -> Response: file = self.get_file_by_locale('CODEBOOK', 'en_US') if not file: raise HTTPNotFound() return request.redirect(request.link(file))
@SwissvotesApp.html( model=TranslatablePage, template='page.pt', permission=Public )
[docs] def view_page( self: TranslatablePage, request: SwissvotesRequest ) -> RenderData: layout = PageLayout(self, request) files = [file for file in self.files if file.locale == request.locale] if not any('DATASET' in f.filename for f in files): files.extend( file for file in self.files if 'DATASET' in file.filename and file.locale == request.default_locale ) file_urls = sorted((f.filename, layout.get_file_url(f)) for f in files) if self.show_timeline: request.include('mastodon') return { 'layout': layout, 'files': file_urls, 'slides': sample(layout.slides, len(layout.slides)) }
@SwissvotesApp.form( model=TranslatablePageCollection, permission=Private, form=PageForm, template='form.pt', name='add' )
[docs] def add_page( self: TranslatablePageCollection, request: SwissvotesRequest, form: PageForm ) -> RenderData | Response: if form.submitted(request): page = TranslatablePage() form.update_model(page) request.session.add(page) request.message(_('Page added.'), 'success') return request.redirect(request.link(page)) request.include('quill') return { 'layout': AddPageLayout(self, request), 'form': form, 'button_text': _('Add') }
@SwissvotesApp.form( model=TranslatablePage, permission=Private, form=PageForm, template='form.pt', name='edit' )
[docs] def edit_page( self: TranslatablePage, request: SwissvotesRequest, form: PageForm ) -> RenderData | Response: if form.submitted(request): form.update_model(self) request.message(_('Page modified.'), 'success') return request.redirect(request.link(self)) if not form.errors: form.apply_model(self) request.include('quill') return { 'layout': EditPageLayout(self, request), 'form': form, 'button_text': _('Update'), }
@SwissvotesApp.form( model=TranslatablePage, permission=Private, form=Form, template='form.pt', name='delete' )
[docs] def delete_page( self: TranslatablePage, request: SwissvotesRequest, form: Form ) -> RenderData | Response: layout = DeletePageLayout(self, request) if form.submitted(request): request.session.delete(self) request.message(_('Page deleted'), 'success') return request.redirect(layout.homepage_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=TranslatablePageMove, permission=Private, request_method='PUT' )
[docs] def move_page( self: TranslatablePageMove, request: SwissvotesRequest ) -> None: request.assert_valid_csrf_token() self.execute()
@SwissvotesApp.html( model=TranslatablePage, template='attachments.pt', name='attachments', permission=Private )
[docs] def view_page_attachments( self: TranslatablePage, request: SwissvotesRequest ) -> RenderData: layout = ManagePageAttachmentsLayout(self, request) files = [file for file in self.files if file.locale == request.locale] return { 'layout': layout, 'title': self.title, 'upload_url': layout.csrf_protected_url( request.link(self, name='upload-attachment') ), 'files': files, 'notice': self, }
@SwissvotesApp.view( model=TranslatablePage, name='upload-attachment', permission=Private, request_method='POST' )
[docs] def upload_page_attachment( self: TranslatablePage, request: SwissvotesRequest ) -> None: request.assert_valid_csrf_token() fs = request.params.get('file', '') if isinstance(fs, str): # malformed formdata raise HTTPBadRequest() attachment = TranslatablePageFile(id=random_token()) attachment.name = f'{request.locale}-{fs.filename}' attachment.reference = as_fileintent(fs.file, fs.filename) self.files.append(attachment)
@SwissvotesApp.html( model=TranslatablePage, template='attachments.pt', name='slider-images', permission=Private )
[docs] def view_page_slider_images( self: TranslatablePage, request: SwissvotesRequest ) -> RenderData: layout = ManagePageSliderImagesLayout(self, request) return { 'layout': layout, 'title': self.title, 'callout': _( 'To associate a slider image with a vote, start the filename with ' 'the BFS number followed by a hyphen, for example, "501-1.jpg".' ), 'upload_url': layout.csrf_protected_url( request.link(self, name='upload-slider-image') ), 'files': self.slider_images, 'notice': self, }
@SwissvotesApp.view( model=TranslatablePage, name='upload-slider-image', permission=Private, request_method='POST' )
[docs] def upload_page_slider_image( self: TranslatablePage, request: SwissvotesRequest ) -> None: request.assert_valid_csrf_token() fs = request.params.get('file', '') if isinstance(fs, str): # malformed formdata raise HTTPBadRequest() attachment = TranslatablePageFile(id=random_token()) attachment.name = f'slider_images-{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=TranslatablePageFile, name='delete', template='form.pt', permission=Private, form=Form )
[docs] def delete_page_attachment( self: TranslatablePageFile, request: SwissvotesRequest, form: Form ) -> RenderData | Response: layout = DeletePageAttachmentLayout(self, request) url = request.link( layout.parent, 'slider-images' if self in layout.parent.slider_images else 'attachments' ) 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 }