from __future__ import annotations
from datetime import datetime
import pytz
from sedate import replace_timezone
from onegov.api import ApiApp
from onegov.core import utils
from onegov.core.i18n import default_locale_negotiator
from onegov.core.utils import module_path
from onegov.foundation6.integration import FoundationApp
from onegov.org.app import OrgApp
from onegov.org.app import get_i18n_localedirs as get_org_i18n_localedirs
from onegov.town6.api import EventApiEndpoint, NewsApiEndpoint
from onegov.town6.custom import get_global_tools
from onegov.town6.initial_content import create_new_organisation
from onegov.town6.theme import TownTheme
from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
from collections.abc import Callable, Iterator, Sequence
from onegov.api import ApiEndpoint
from onegov.core.types import RenderData
from onegov.org.models import Organisation
from onegov.town6.request import TownRequest
[docs]
class TownApp(OrgApp, FoundationApp, ApiApp):
@property
[docs]
def font_family(self) -> str | None:
return self.theme_options.get('body-font-family-ui')
[docs]
def chat_open(self, request: TownRequest) -> bool:
if not request.app.org.specific_opening_hours:
return True
opening_hours = request.app.org.opening_hours_chat
tz = pytz.timezone('Europe/Zurich')
now = datetime.now(tz=tz)
if opening_hours:
for day, start, end in opening_hours:
if str(now.weekday()) == day:
open = replace_timezone(
datetime(now.year, now.month, now.day,
int(start.split(':')[0]),
int(start.split(':')[1])), tz)
close = replace_timezone(
datetime(now.year, now.month, now.day,
int(end.split(':')[0]),
int(end.split(':')[1])), tz)
if now > open and now < close:
return True
return False
@TownApp.webasset_path()
[docs]
def get_shared_assets_path() -> str:
return utils.module_path('onegov.shared', 'assets/js')
@TownApp.static_directory()
[docs]
def get_static_directory() -> str:
return 'static'
@TownApp.template_directory()
[docs]
def get_template_directory() -> str:
return 'templates'
@TownApp.template_variables()
[docs]
def get_template_variables(request: TownRequest) -> RenderData:
return {
'global_tools': tuple(get_global_tools(request))
}
@TownApp.setting(section='core', name='theme')
[docs]
def get_theme() -> TownTheme:
return TownTheme()
@TownApp.setting(section='i18n', name='locales')
[docs]
def get_i18n_used_locales() -> set[str]:
return {'de_CH', 'fr_CH'}
@TownApp.setting(section='i18n', name='localedirs')
[docs]
def get_i18n_localedirs() -> list[str]:
return [
module_path('onegov.town6', 'locale'),
*get_org_i18n_localedirs()
]
@TownApp.setting(section='i18n', name='default_locale')
[docs]
def get_i18n_default_locale() -> str:
return 'de_CH'
@TownApp.setting(section='i18n', name='locale_negotiator')
[docs]
def get_locale_negotiator(
) -> Callable[[Sequence[str], TownRequest], str | None]:
def locale_negotiator(
locales: Sequence[str],
request: TownRequest
) -> str | None:
if request.app.org:
locales = request.app.org.locales or get_i18n_default_locale()
if isinstance(locales, str):
locales = (locales, )
return default_locale_negotiator(locales, request) or locales[0]
else:
return default_locale_negotiator(locales, request)
return locale_negotiator
@TownApp.setting(section='org', name='create_new_organisation')
[docs]
def get_create_new_organisation_factory(
) -> Callable[[TownApp, str], Organisation]:
return create_new_organisation
@TownApp.setting(section='org', name='status_mail_roles')
[docs]
def get_status_mail_roles() -> tuple[str, ...]:
return ('admin', 'editor')
@TownApp.setting(section='org', name='ticket_manager_roles')
[docs]
def get_ticket_manager_roles() -> tuple[str, ...]:
return ('admin', 'editor')
@TownApp.setting(section='org', name='require_complete_userprofile')
[docs]
def get_require_complete_userprofile() -> bool:
return False
@TownApp.setting(section='org', name='is_complete_userprofile')
[docs]
def get_is_complete_userprofile_handler(
) -> Callable[[TownRequest, str], bool]:
def is_complete_userprofile(request: TownRequest, username: str) -> bool:
return True
return is_complete_userprofile
@TownApp.setting(section='org', name='default_directory_search_widget')
@TownApp.setting(section='org', name='default_event_search_widget')
@TownApp.setting(section='org', name='public_ticket_messages')
[docs]
def get_public_ticket_messages() -> tuple[str, ...]:
""" Returns a list of message types which are availble on the ticket
status page, visible to anyone that knows the unguessable url.
"""
# do *not* add ticket_note here, those are private!
return (
'directory',
'event',
'payment',
'reservation',
'submission',
'ticket',
'ticket_chat',
)
@TownApp.setting(section='api', name='endpoints')
[docs]
def get_api_endpoints() -> list[type[ApiEndpoint[Any]]]:
return [
EventApiEndpoint,
NewsApiEndpoint
]
@TownApp.webasset_path()
[docs]
def get_js_path() -> str:
return 'assets/js'
@TownApp.webasset_path()
[docs]
def get_css_path() -> str:
return 'assets/css'
@TownApp.webasset_output()
[docs]
def get_webasset_output() -> str:
return 'assets/bundles'
@TownApp.webasset('common')
[docs]
def get_common_asset() -> Iterator[str]:
yield 'global.js'
yield 'polyfills.js'
yield 'jquery.datetimepicker.css'
yield 'locale.js'
yield 'modernizr.js'
yield 'clipboard.js'
yield 'intercooler.js'
yield 'underscore.js'
yield 'react.js'
yield 'react-dom.js'
yield 'form_dependencies.js'
yield 'confirm.jsx'
yield 'typeahead.jsx'
yield 'many.jsx'
yield 'pay'
yield 'moment.js'
yield 'moment.de-ch.js'
yield 'moment.fr-ch.js'
yield 'jquery.datetimepicker.js'
yield 'jquery.mousewheel.js'
yield 'jquery.popupoverlay.js'
yield 'jquery.load.js'
yield 'videoframe.js'
yield 'datetimepicker.js'
yield 'url.js'
yield 'date-range-selector.js'
yield 'lazyalttext.js'
yield 'lazysizes.js'
yield 'common.js'
yield '_blank.js'
yield 'homepage_video_or_slider.js'
yield 'file-table-row-toggler.js'
yield 'animate.js'
yield 'forms.js'
yield 'internal_link_check.js'
yield 'tickets.js'
yield 'items_selectable.js'
yield 'aos.js'
yield 'aos-init.js'
yield 'aos.css'
yield 'notifications.js'
yield 'sidebar_mobile.js'
yield 'sidebar_fixed.js'
yield 'ResizeSensor.js'
yield 'theia-sticky-sidebar.js'
@TownApp.webasset('editor')
[docs]
def get_editor_asset() -> Iterator[str]:
yield 'bufferbuttons.js'
yield 'definedlinks.js'
yield 'filemanager.js'
yield 'imagemanager.js'
yield 'table.js'
yield 'redactor.de.js'
yield 'redactor.fr.js'
yield 'redactor.it.js'
yield 'input_with_button.js'
yield 'editor.js'
@TownApp.webasset('fullcalendar')
[docs]
def get_fullcalendar_asset() -> Iterator[str]:
yield 'fullcalendar.css'
yield 'fullcalendar.js'
yield 'fullcalendar.de.js'
yield 'reservationcalendar.jsx'
yield 'reservationcalendar_custom.js'
@TownApp.webasset('staff-chat')
[docs]
def get_staff_chat_asset() -> Iterator[str]:
yield 'chat-shared.js'
yield 'chat-staff.js'
@TownApp.webasset('client-chat')
[docs]
def get_staff_client_asset() -> Iterator[str]:
yield 'chat-shared.js'
yield 'chat-client.js'