Source code for fsi.layouts.audit
from __future__ import annotations
from datetime import timedelta
from functools import cached_property
from onegov.core.elements import Link, LinkGroup
from onegov.fsi.layout import DefaultLayout
from onegov.fsi import _
from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from datetime import datetime
    from onegov.fsi.collections.audit import AuditCollection
[docs]
class AuditLayout(DefaultLayout):
    @property
    @cached_property
[docs]
    def breadcrumbs(self) -> list[Link]:
        links = super().breadcrumbs
        assert isinstance(links, list)
        links.append(
            Link(_('Audit'), self.request.link(self.model))
        )
        return links
    @cached_property
[docs]
    def editbar_links(self) -> list[Link | LinkGroup]:
        if not self.model.course_id or not self.model.subset_count:
            return []
        return [
            Link(
                text=_('PDF'),
                url=self.request.link(self.model, name='pdf'),
                attrs={'class': 'print-icon', 'target': '_blank'}
            ),
        ]
[docs]
    def render_start_end(
        self,
        start: datetime | None,
        end: datetime | None
    ) -> str:
        if not start:
            return '-'
        date_str = self.format_date(start, 'date')
        start_str = self.format_date(start, 'time')
        end_str = self.format_date(end, 'time')
        return f'{date_str} {start_str} - {end_str}'
[docs]
    def format_refresh_interval(self, num_years: int) -> str:
        assert isinstance(num_years, int)
        return self.format_timedelta(
            # NOTE: Even though 365.25 would be more accurate, babel itself
            #       uses 365 to count the number of years, so this is the
            #       most reliable way to not encounter any drift (not that
            #       we'd expect to see num_years high enough for this to
            #       matter)
            timedelta(days=num_years * 365)
        )
    @staticmethod
[docs]
    def next_event_date(
        start: datetime | None,
        refresh_interval: int | None
    ) -> datetime | None:
        if not start:
            return None
        if refresh_interval is None:
            return None
        assert isinstance(refresh_interval, int)
        return start.replace(year=start.year + refresh_interval)
    @cached_property
[docs]
    def audit_table_headers(self) -> list[str]:
        if not self.model.course:
            # FIXME: Maybe it would be better to assert this? It doesn't
            #        look like we expect to be able to deal with this case
            #        anywhere
            return []
        # FIXME: Does this mean this should actually not be nullable?
        assert self.model.course.refresh_interval is not None
        due_in = self.format_refresh_interval(
            self.model.course.refresh_interval)
        titles = (
            _('Name'),
            _('Shortcode'),
            _('Last Event'),
            _('Registered'),
            _('Due by (every ${refresh_interval})', mapping={
                'refresh_interval': due_in}
              )
        )
        return [self.request.translate(head) for head in titles]