Source code for translator_directory.forms.translator

from __future__ import annotations

import re

from functools import cached_property

from onegov.form import Form
from onegov.form.fields import ChosenSelectMultipleField, ChosenSelectField
from onegov.form.fields import MultiCheckboxField
from onegov.form.fields import TagsField
from onegov.form.validators import Stdnum
from onegov.form.validators import StrictOptional
from onegov.form.validators import ValidPhoneNumber
from onegov.form.validators import ValidSwissSocialSecurityNumber
from onegov.gis import CoordinatesField
from onegov.translator_directory import _
from onegov.translator_directory.collections.certificate import (
    LanguageCertificateCollection)
from onegov.translator_directory.collections.language import LanguageCollection
from onegov.translator_directory.collections.translator import order_cols
from onegov.translator_directory.collections.translator import (
    TranslatorCollection)
from onegov.translator_directory.constants import ADMISSIONS
from onegov.translator_directory.constants import full_text_max_chars
from onegov.translator_directory.constants import GENDERS
from onegov.translator_directory.constants import INTERPRETING_TYPES
from onegov.translator_directory.constants import PROFESSIONAL_GUILDS
from onegov.translator_directory.forms.mixins import DrivingDistanceMixin
from onegov.translator_directory.models.translator import (
    certificate_association_table)
from onegov.translator_directory.models.translator import (
    monitoring_association_table)
from onegov.translator_directory.models.translator import (
    mother_tongue_association_table)
from onegov.translator_directory.models.translator import (
    spoken_association_table)
from onegov.translator_directory.models.translator import Translator
from onegov.translator_directory.models.translator import (
    written_association_table)
from onegov.translator_directory.utils import nationality_choices
from wtforms.fields import BooleanField
from wtforms.fields import DateField
from wtforms.fields import EmailField
from wtforms.fields import FloatField
from wtforms.fields import IntegerField
from wtforms.fields import RadioField
from wtforms.fields import SelectField
from wtforms.fields import StringField
from wtforms.fields import TextAreaField
from wtforms.validators import Email
from wtforms.validators import InputRequired
from wtforms.validators import Length
from wtforms.validators import Optional
from wtforms.validators import ValidationError


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from onegov.translator_directory.models.certificate import (
        LanguageCertificate)
    from onegov.translator_directory.models.language import Language
    from onegov.translator_directory.request import TranslatorAppRequest
    from sqlalchemy.orm import Query
    from wtforms.fields.choices import _Choice


[docs] class FormChoicesMixin:
[docs] request: TranslatorAppRequest
@property
[docs] def available_languages(self) -> Query[Language]: return LanguageCollection(self.request.session).query()
@property
[docs] def available_certificates(self) -> Query[LanguageCertificate]: return LanguageCertificateCollection(self.request.session).query()
@property
[docs] def available_additional_guilds(self) -> list[str]: translators = TranslatorCollection(self.request.app) return translators.available_additional_professional_guilds
@cached_property
[docs] def language_choices(self) -> list[_Choice]: return [ (str(lang.id), lang.name) for lang in self.available_languages ]
@cached_property
[docs] def certificate_choices(self) -> list[_Choice]: return [ (str(cert.id), cert.name) for cert in self.available_certificates ]
@cached_property
[docs] def gender_choices(self) -> list[_Choice]: return [ (id_, self.request.translate(choice)) for id_, choice in GENDERS.items() ]
@staticmethod
[docs] def get_ids(model: object, attr: str) -> list[str]: return [ str(item.id) for item in getattr(model, attr) ]
@cached_property
[docs] def interpret_types_choices(self) -> list[_Choice]: return [ (k, self.request.translate(v)) for k, v in INTERPRETING_TYPES.items() ]
@cached_property
[docs] def guilds_choices(self) -> list[_Choice]: result: list[_Choice] = [ (k, self.request.translate(v)) for k, v in PROFESSIONAL_GUILDS.items() ] result.extend((k, k) for k in self.available_additional_guilds) return sorted(result, key=lambda x: x[1].upper())
@cached_property
[docs] def admission_choices(self) -> list[_Choice]: admissions = tuple( (k, self.request.translate(v)) for k, v in ADMISSIONS.items() ) return sorted(admissions, key=lambda x: x[1].upper())
@property
[docs] def lang_collection(self) -> LanguageCollection: return LanguageCollection(self.request.session)
[docs] class EditorTranslatorForm(Form, FormChoicesMixin):
[docs] request: TranslatorAppRequest
[docs] pers_id = IntegerField( label=_('Personal ID'), validators=[Optional()] )
[docs] def update_model(self, model: Translator) -> None: model.pers_id = self.pers_id.data or None
[docs] class TranslatorForm(Form, FormChoicesMixin, DrivingDistanceMixin):
[docs] request: TranslatorAppRequest
[docs] pers_id = IntegerField( label=_('Personal ID'), validators=[Optional()] )
[docs] admission = RadioField( label=_('Admission'), choices=tuple( (id_, label) for id_, label in ADMISSIONS.items() ), default=next(iter(ADMISSIONS)) )
[docs] withholding_tax = BooleanField( label=_('Withholding tax'), default=False )
[docs] self_employed = BooleanField( label=_('Self-employed'), default=False )
[docs] last_name = StringField( label=_('Last name'), validators=[InputRequired()], fieldset=_('Personal Information') )
[docs] first_name = StringField( label=_('First name'), validators=[InputRequired()], fieldset=_('Personal Information') )
[docs] gender = SelectField( label=_('Gender'), validators=[StrictOptional()], choices=[], fieldset=_('Personal Information') )
[docs] date_of_birth = DateField( label=_('Date of birth'), validators=[Optional()], fieldset=_('Personal Information') )
[docs] nationalities = ChosenSelectMultipleField( label=_('Nationality(ies)'), validators=[Optional()], choices=[], # will be filled in on_request fieldset=_('Personal Information') )
[docs] hometown = StringField( label=_('Hometown'), fieldset=_('Personal Information'), validators=[Optional()], )
[docs] coordinates = CoordinatesField( label=_('Location'), description=_( 'Search for the exact address to set a marker. The address fields ' 'beneath are filled out automatically.' ), fieldset=_('Address'), render_kw={'data-map-type': 'marker'} )
[docs] address = StringField( label=_('Street and house number'), fieldset=_('Address') )
[docs] zip_code = StringField( label=_('Zip Code'), fieldset=_('Address') )
[docs] city = StringField( label=_('City'), fieldset=_('Address') )
[docs] drive_distance = FloatField( label=_('Drive distance (km)'), validators=[Optional()], fieldset=_('Address'), )
[docs] social_sec_number = StringField( label=_('Swiss social security number'), validators=[ValidSwissSocialSecurityNumber(), InputRequired()], fieldset=_('Identification / bank account') )
[docs] bank_name = StringField( label=_('Bank name') )
[docs] bank_address = StringField( label=_('Bank address') )
[docs] account_owner = StringField( label=_('Account owner') )
[docs] iban = StringField( label=_('IBAN'), validators=[Optional(), Stdnum(format='iban')] )
[docs] email = EmailField( label=_('Email'), validators=[InputRequired(), Email()], fieldset=_('Contact information') )
[docs] tel_mobile = StringField( label=_('Mobile Number'), validators=[InputRequired(), ValidPhoneNumber()], )
[docs] tel_private = StringField( label=_('Private Phone Number'), validators=[Optional(), ValidPhoneNumber()], )
[docs] tel_office = StringField( label=_('Office Phone Number'), validators=[Optional(), ValidPhoneNumber()], )
[docs] availability = StringField( label=_('Availability'), )
[docs] operation_comments = TextAreaField( label=_('Comments on possible field of application'), render_kw={'rows': 3} )
[docs] confirm_name_reveal = BooleanField( label=_('Name revealing confirmation'), fieldset=_('Legal information'), description=_('Consent to the disclosure of the name ' 'to other persons and authorities') )
[docs] date_of_application = DateField( label=_('Date of application'), validators=[Optional()], fieldset=_('Admission to the directory') )
[docs] date_of_decision = DateField( label=_('Date of decision'), validators=[Optional()], )
[docs] mother_tongues_ids = ChosenSelectMultipleField( label=_('Mother tongues'), validators=[InputRequired()], choices=[], fieldset=_('Language training and expertise') )
[docs] spoken_languages_ids = ChosenSelectMultipleField( label=_('Spoken languages'), validators=[StrictOptional()], choices=[] )
[docs] written_languages_ids = ChosenSelectMultipleField( label=_('Written languages'), validators=[StrictOptional()], choices=[] )
[docs] monitoring_languages_ids = ChosenSelectMultipleField( label=_('Monitoring languages'), validators=[StrictOptional()], choices=[] )
[docs] profession = StringField( label=_('Learned profession') )
[docs] occupation = StringField( label=_('Current professional activity') )
[docs] expertise_professional_guilds = MultiCheckboxField( label=_('Expertise by professional guild'), choices=list(PROFESSIONAL_GUILDS.items()) )
[docs] expertise_professional_guilds_other = TagsField( label=_('Expertise by professional guild: other') )
[docs] expertise_interpreting_types = MultiCheckboxField( label=_('Expertise by interpreting type'), choices=list(INTERPRETING_TYPES.items()) )
[docs] proof_of_preconditions = StringField( label=_('Proof of preconditions') )
[docs] agency_references = TextAreaField( label=_('Agency references'), validators=[Optional()], render_kw={'rows': 3} )
[docs] education_as_interpreter = BooleanField( label=_('Education as interpreter'), default=False )
[docs] certificates_ids = ChosenSelectMultipleField( label=_('Language Certificates'), validators=[Optional()], choices=[] )
[docs] comments = TextAreaField( label=_('Comments') )
[docs] for_admins_only = BooleanField( label=_('Hidden'), default=False )
@property
[docs] def cert_collection(self) -> LanguageCertificateCollection: return LanguageCertificateCollection(self.request.session)
@property
[docs] def certificates(self) -> list[LanguageCertificate]: if not self.certificates_ids.data: return [] return self.cert_collection.by_ids(self.certificates_ids.data)
@property
[docs] def mother_tongues(self) -> list[Language]: if not self.mother_tongues_ids.data: return [] return self.lang_collection.by_ids(self.mother_tongues_ids.data)
@property
[docs] def spoken_languages(self) -> list[Language]: if not self.spoken_languages_ids.data: return [] return self.lang_collection.by_ids(self.spoken_languages_ids.data)
@property
[docs] def written_languages(self) -> list[Language]: if not self.written_languages_ids.data: return [] return self.lang_collection.by_ids(self.written_languages_ids.data)
@property
[docs] def monitoring_languages(self) -> list[Language]: if not self.monitoring_languages_ids.data: return [] return self.lang_collection.by_ids(self.monitoring_languages_ids.data)
[docs] special_fields = { 'mother_tongues_ids': mother_tongue_association_table, 'spoken_languages_ids': spoken_association_table, 'written_languages_ids': written_association_table, 'monitoring_languages_ids': monitoring_association_table, 'certificates_ids': certificate_association_table }
# Here come the actual file fields to upload stuff
[docs] def on_request(self) -> None: self.request.include('tags-input') self.gender.choices = self.gender_choices self.nationalities.choices = nationality_choices(self.request.locale) self.mother_tongues_ids.choices = self.language_choices self.spoken_languages_ids.choices = self.language_choices self.written_languages_ids.choices = self.language_choices self.monitoring_languages_ids.choices = self.language_choices self.certificates_ids.choices = self.certificate_choices self.hide(self.drive_distance)
[docs] def get_useful_data(self) -> dict[str, Any]: # type:ignore[override] """Do not use to update and instance of a translator.""" data = super().get_useful_data( exclude={'csrf_token', *self.special_fields.keys()} ) data['email'] = data['email'] or None data['mother_tongues'] = self.mother_tongues data['spoken_languages'] = self.spoken_languages data['written_languages'] = self.written_languages data['monitoring_languages'] = self.monitoring_languages data['certificates'] = self.certificates return data
[docs] def validate_zip_code(self, field: StringField) -> None: if field.data and not re.match(r'\d{4}', field.data): raise ValidationError(_('Zip code must consist of 4 digits'))
[docs] def validate_email(self, field: EmailField) -> None: if field.data: field.data = field.data.lower() if isinstance(self.model, Translator): if str(self.model.email) == field.data: return query = self.request.session.query trs = query(Translator).filter_by(email=field.data).first() if trs: raise ValidationError( _('A translator with this email already exists'))
[docs] def update_association( self, model: Translator, db_field: str, suffix: str ) -> None: field = f'{db_field}{suffix}' setattr(model, db_field, []) if not getattr(self, field).data: return for item in getattr(self, db_field): getattr(model, db_field).append(item)
[docs] def update_model(self, model: Translator) -> None: translators = TranslatorCollection(self.request.app) translators.update_user(model, self.email.data) assert self.first_name.data is not None assert self.last_name.data is not None model.first_name = self.first_name.data model.last_name = self.last_name.data model.iban = self.iban.data model.pers_id = self.pers_id.data or None model.admission = self.admission.data model.withholding_tax = self.withholding_tax.data model.self_employed = self.self_employed.data model.gender = self.gender.data model.date_of_birth = self.date_of_birth.data or None model.nationalities = self.nationalities.data or [] model.address = self.address.data or None model.zip_code = self.zip_code.data or None model.city = self.city.data or None model.hometown = self.hometown.data or None model.drive_distance = self.drive_distance.data or None model.social_sec_number = self.social_sec_number.data or None model.bank_name = self.bank_name.data or None model.bank_address = self.bank_address.data or None model.account_owner = self.account_owner.data or None model.email = self.email.data or None model.tel_mobile = self.tel_mobile.data or None model.tel_private = self.tel_private.data or None model.tel_office = self.tel_office.data or None model.confirm_name_reveal = self.confirm_name_reveal.data model.availability = self.availability.data or None model.date_of_application = self.date_of_application.data or None model.date_of_decision = self.date_of_decision.data or None model.proof_of_preconditions = self.proof_of_preconditions.data or None model.agency_references = self.agency_references.data or None model.education_as_interpreter = self.education_as_interpreter.data model.comments = self.comments.data or None model.for_admins_only = self.for_admins_only.data model.profession = self.profession.data model.occupation = self.occupation.data model.operation_comments = self.operation_comments.data or None model.coordinates = self.coordinates.data self.update_association(model, 'mother_tongues', '_ids') self.update_association(model, 'spoken_languages', '_ids') self.update_association(model, 'written_languages', '_ids') self.update_association(model, 'monitoring_languages', '_ids') self.update_association(model, 'certificates', '_ids') model.expertise_professional_guilds = ( self.expertise_professional_guilds.data or []) model.expertise_professional_guilds_other = ( self.expertise_professional_guilds_other.data) model.expertise_interpreting_types = ( self.expertise_interpreting_types.data or [])
[docs] class TranslatorSearchForm(Form, FormChoicesMixin):
[docs] request: TranslatorAppRequest
[docs] spoken_langs = ChosenSelectMultipleField( label=_('Spoken languages'), choices=[] )
[docs] written_langs = ChosenSelectMultipleField( label=_('Written languages'), choices=[] )
[docs] monitoring_languages_ids = ChosenSelectMultipleField( label=_('Monitoring languages'), choices=[] )
[docs] interpret_types = ChosenSelectMultipleField( label=_('Expertise by interpreting type'), choices=[] )
[docs] guilds = ChosenSelectMultipleField( label=_('Expertise by professional guild'), choices=[] )
[docs] genders = ChosenSelectMultipleField( label=_('Gender'), choices=[], )
[docs] admission = ChosenSelectMultipleField( label=_('Admission'), choices=[] )
[docs] search = StringField( label=_('Search in first and last name'), validators=[Optional(), Length(max=full_text_max_chars)] )
[docs] order_by = RadioField( label=_('Order by'), choices=( (order_cols[0], _('Last name')), (order_cols[1], _('Drive distance')), ), default=order_cols[0] )
[docs] order_desc = RadioField( label=_('Order direction'), choices=( ('0', _('Ascending')), ('1', _('Descending')) ), default='0' )
@property
[docs] def monitoring_languages(self) -> list[Language]: if not self.monitoring_languages_ids.data: return [] return self.lang_collection.by_ids(self.monitoring_languages_ids.data)
[docs] def apply_model(self, model: TranslatorCollection) -> None: if model.spoken_langs: self.spoken_langs.data = model.spoken_langs if model.genders: self.genders.data = model.genders if model.written_langs: self.written_langs.data = model.written_langs if model.monitor_langs: self.monitoring_languages_ids.data = model.monitor_langs self.order_by.data = model.order_by self.order_desc.data = model.order_desc and '1' or '0' self.search.data = model.search self.interpret_types.data = model.interpret_types or [] self.guilds.data = model.guilds or [] self.admission.data = model.admissions or []
[docs] def update_model(self, model: TranslatorCollection) -> None: model.spoken_langs = self.spoken_langs.data model.written_langs = self.written_langs.data model.monitor_langs = self.monitoring_languages_ids.data model.order_by = self.order_by.data model.order_desc = self.order_desc.data == '1' and True or False model.search = self.search.data model.interpret_types = self.interpret_types.data or [] model.guilds = self.guilds.data or [] model.admissions = self.admission.data or [] model.genders = self.genders.data or []
[docs] def on_request(self) -> None: self.spoken_langs.choices = self.language_choices self.written_langs.choices = self.language_choices self.guilds.choices = self.guilds_choices self.interpret_types.choices = self.interpret_types_choices self.monitoring_languages_ids.choices = self.language_choices self.admission.choices = self.admission_choices self.genders.choices = self.gender_choices
[docs] class MailTemplatesForm(Form): """ Defines the form for generating mail templates. """
[docs] request: TranslatorAppRequest
[docs] templates = ChosenSelectField( label=_('Chose a mail template to generate:'), choices=[] )
[docs] def on_request(self) -> None: self.templates.choices = [ (t, t) for t in self.request.app.mail_templates ]