Source code for feriennet.forms.billing
from __future__ import annotations
from functools import cached_property
from onegov.activity import Invoice
from onegov.feriennet import _
from onegov.form import Form
from onegov.form.fields import MultiCheckboxField
from onegov.user import User, UserCollection
from sqlalchemy import func
from wtforms.fields import BooleanField
from wtforms.fields import DateField
from wtforms.fields import DecimalField
from wtforms.fields import RadioField
from wtforms.fields import SelectField
from wtforms.fields import StringField
from wtforms.validators import InputRequired
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from decimal import Decimal
from sqlalchemy.orm import Query
[docs]
class BillingForm(Form):
[docs]
confirm = RadioField(
label=_('Confirm billing:'),
default='no',
choices=[
('no', _('No, preview only')),
('yes', _('Yes, confirm billing'))
]
)
[docs]
sure = BooleanField(
label=_(
'I know that after confirmation, bills are made visible to users.'
),
default=False,
depends_on=('confirm', 'yes')
)
@property
[docs]
def finalize_period(self) -> bool:
return self.confirm.data == 'yes' and self.sure.data is True
[docs]
class ManualBookingForm(Form):
[docs]
tags = MultiCheckboxField(
label=_('Tags'),
validators=(InputRequired(), ),
depends_on=('target', 'for-users-with-tags'),
choices=()
)
[docs]
username = SelectField(
label=_('User'),
validators=(InputRequired(), ),
depends_on=('target', 'for-user'),
)
[docs]
kind = RadioField(
label=_('Kind'),
default='discount',
choices=(
('discount', _('Discount')),
('surcharge', _('Surcharge'))
)
)
[docs]
discount = DecimalField(
label=_('Discount'),
validators=(InputRequired(), ),
depends_on=('kind', 'discount')
)
[docs]
surcharge = DecimalField(
label=_('Surcharge'),
validators=(InputRequired(), ),
depends_on=('kind', 'surcharge')
)
@property
[docs]
def amount(self) -> Decimal:
if self.kind.data == 'discount':
assert self.discount.data is not None
return -self.discount.data
elif self.kind.data == 'surcharge':
assert self.surcharge.data is not None
return self.surcharge.data
else:
raise NotImplementedError
@property
@property
[docs]
def available_usernames(self) -> Query[tuple[str, str]]:
return (
self.usercollection.query()
.with_entities(User.username, User.realname)
.filter(func.trim(func.coalesce(User.realname, '')) != '')
.filter(User.active == True)
.order_by(func.unaccent(func.lower(User.realname)))
)
@property
[docs]
def users(self) -> tuple[str, ...]:
if self.target.data == 'all':
return tuple(username for username, _ in self.available_usernames)
elif self.target.data == 'for-user':
return (self.username.data, )
elif self.target.data == 'for-users-with-tags':
assert self.tags.data is not None
return self.usercollection.usernames_by_tags(self.tags.data)
else:
raise NotImplementedError
[docs]
def on_request(self) -> None:
self.target.choices = [
('all', _('All')),
('for-user', _('For a specific user'))
]
self.load_usernames()
self.load_user_tags()
if self.tags.choices:
self.target.choices.append(
('for-users-with-tags', _('For users with tags')))
if (self.request.params.get('for-user')
and not self.target.data):
self.target.data = 'for-user'
if not self.username.data:
self.username.data = self.request.params['for-user']
@property
[docs]
def load_user_tags(self) -> None:
self.tags.choices = [(t, t) for t in self.usercollection.tags]
[docs]
class PaymentWithDateForm(Form):
[docs]
target = RadioField(
validators=(InputRequired(), ),
label=_('Target'),
choices=(
('all', _('Whole invoice')),
('specific', _('Only for specific items'))
),
)
[docs]
items = MultiCheckboxField(
label=_('Items'),
validators=(InputRequired(), ),
depends_on=('target', 'specific'),
choices=()
)
[docs]
def on_request(self) -> None:
self.items.choices = [
(i.id.hex,
f'{i.group} - {i.text} ({round(i.amount, 2)})')
for i in self.invoice.items if not i.paid]
if self.request.params['item-id'] != 'all':
# Set defaults according to parameters
if not self.target.data:
self.target.data = 'specific'
if not self.items.data:
self.items.data = [self.request.params['item-id']]
# Check all unpaid items if 'all' is selected
if self.target.data == 'all':
self.items.data = [i.id.hex for i in self.invoice.items
if not i.paid]
else:
# Set defaults according to parameters
if not self.target.data:
self.target.data = 'all'
if not self.items.data:
self.items.data = [i.id.hex for i in self.invoice.items
if not i.paid]
# Check all unpaid items if 'all' is selected
if self.target.data == 'all':
self.items.data = [i.id.hex for i in self.invoice.items]
@cached_property
[docs]
def invoice(self) -> Invoice:
return self.request.session.query(
Invoice).filter_by(id=self.request.params['invoice-id']).one()