mirror of
https://github.com/YunoHost-Apps/ihatemoney_ynh.git
synced 2024-09-03 19:26:15 +02:00
2862130269
- Adapt to BS4 class names and hierarchy - Redesign some forms to have the label above the input (mobile-friendly) - For the remaining inline form (add bill), use the grid, as BS no longer provides inline form alignment helpers
198 lines
7.3 KiB
Python
198 lines
7.3 KiB
Python
from flask_wtf import DateField, DecimalField, Email, Form, PasswordField, \
|
|
Required, SelectField, SelectMultipleField, SubmitField, TextAreaField, \
|
|
TextField, ValidationError
|
|
from flask_babel import lazy_gettext as _
|
|
from flask import request
|
|
|
|
from wtforms.widgets import html_params
|
|
from models import Project, Person
|
|
from datetime import datetime
|
|
from jinja2 import Markup
|
|
from utils import slugify
|
|
|
|
|
|
def select_multi_checkbox(field, ul_class='', **kwargs):
|
|
kwargs.setdefault('type', 'checkbox')
|
|
field_id = kwargs.pop('id', field.id)
|
|
html = [u'<ul %s>' % html_params(id=field_id, class_="inputs-list")]
|
|
|
|
choice_id = u'toggleField'
|
|
js_function = u'toggle();'
|
|
options = dict(kwargs, id=choice_id, onclick=js_function)
|
|
html.append(u'<p><a href="#" id="selectall" onclick="selectall()">%s</a> | <a href="#" id="selectnone" onclick="selectnone()">%s</a></p>'% (_("Select all"), _("Select none")))
|
|
|
|
for value, label, checked in field.iter_choices():
|
|
choice_id = u'%s-%s' % (field_id, value)
|
|
options = dict(kwargs, name=field.name, value=value, id=choice_id)
|
|
if checked:
|
|
options['checked'] = 'checked'
|
|
html.append(u'<p class="form-check"><label for="%s" class="form-check-label">%s<span>%s</span></label></p>'
|
|
% (choice_id, '<input %s /> ' % html_params(**options), label))
|
|
html.append(u'</ul>')
|
|
return u''.join(html)
|
|
|
|
|
|
def get_billform_for(project, set_default=True, **kwargs):
|
|
"""Return an instance of BillForm configured for a particular project.
|
|
|
|
:set_default: if set to True, on GET methods (usually when we want to
|
|
display the default form, it will call set_default on it.
|
|
|
|
"""
|
|
form = BillForm(**kwargs)
|
|
form.payed_for.choices = form.payer.choices = [(m.id, m.name)
|
|
for m in project.active_members]
|
|
form.payed_for.default = [m.id for m in project.active_members]
|
|
|
|
if set_default and request.method == "GET":
|
|
form.set_default()
|
|
return form
|
|
|
|
|
|
class CommaDecimalField(DecimalField):
|
|
"""A class to deal with comma in Decimal Field"""
|
|
def process_formdata(self, value):
|
|
if value:
|
|
value[0] = str(value[0]).replace(',', '.')
|
|
return super(CommaDecimalField, self).process_formdata(value)
|
|
|
|
|
|
class EditProjectForm(Form):
|
|
name = TextField(_("Project name"), validators=[Required()])
|
|
password = TextField(_("Private code"), validators=[Required()])
|
|
contact_email = TextField(_("Email"), validators=[Required(), Email()])
|
|
|
|
def save(self):
|
|
"""Create a new project with the information given by this form.
|
|
|
|
Returns the created instance
|
|
"""
|
|
project = Project(name=self.name.data, id=self.id.data,
|
|
password=self.password.data,
|
|
contact_email=self.contact_email.data)
|
|
return project
|
|
|
|
def update(self, project):
|
|
"""Update the project with the information from the form"""
|
|
project.name = self.name.data
|
|
project.password = self.password.data
|
|
project.contact_email = self.contact_email.data
|
|
|
|
return project
|
|
|
|
|
|
class ProjectForm(EditProjectForm):
|
|
id = TextField(_("Project identifier"), validators=[Required()])
|
|
password = PasswordField(_("Private code"), validators=[Required()])
|
|
submit = SubmitField(_("Create the project"))
|
|
|
|
def validate_id(form, field):
|
|
form.id.data = slugify(field.data)
|
|
if (form.id.data == "dashboard") or Project.query.get(form.id.data):
|
|
raise ValidationError(Markup(_("The project identifier is used "
|
|
"to log in and for the URL of the project. "
|
|
"We tried to generate an identifier for you but a project "
|
|
"with this identifier already exists. "
|
|
"Please create a new identifier "
|
|
"that you will be able to remember.")))
|
|
|
|
|
|
class AuthenticationForm(Form):
|
|
id = TextField(_("Project identifier"), validators=[Required()])
|
|
password = PasswordField(_("Private code"), validators=[Required()])
|
|
submit = SubmitField(_("Get in"))
|
|
|
|
|
|
class PasswordReminder(Form):
|
|
id = TextField(_("Project identifier"), validators=[Required()])
|
|
submit = SubmitField(_("Send me the code by email"))
|
|
|
|
def validate_id(form, field):
|
|
if not Project.query.get(field.data):
|
|
raise ValidationError(_("This project does not exists"))
|
|
|
|
|
|
class BillForm(Form):
|
|
date = DateField(_("Date"), validators=[Required()], default=datetime.now)
|
|
what = TextField(_("What?"), validators=[Required()])
|
|
payer = SelectField(_("Payer"), validators=[Required()], coerce=int)
|
|
amount = CommaDecimalField(_("Amount paid"), validators=[Required()])
|
|
payed_for = SelectMultipleField(_("For whom?"),
|
|
validators=[Required()], widget=select_multi_checkbox, coerce=int)
|
|
submit = SubmitField(_("Submit"))
|
|
submit2 = SubmitField(_("Submit and add a new one"))
|
|
|
|
def save(self, bill, project):
|
|
bill.payer_id = self.payer.data
|
|
bill.amount = self.amount.data
|
|
bill.what = self.what.data
|
|
bill.date = self.date.data
|
|
bill.owers = [Person.query.get(ower, project)
|
|
for ower in self.payed_for.data]
|
|
|
|
return bill
|
|
|
|
def fill(self, bill):
|
|
self.payer.data = bill.payer_id
|
|
self.amount.data = bill.amount
|
|
self.what.data = bill.what
|
|
self.date.data = bill.date
|
|
self.payed_for.data = [int(ower.id) for ower in bill.owers]
|
|
|
|
def set_default(self):
|
|
self.payed_for.data = self.payed_for.default
|
|
|
|
def validate_amount(self, field):
|
|
if field.data == 0:
|
|
raise ValidationError(_("Bills can't be null"))
|
|
|
|
|
|
class MemberForm(Form):
|
|
|
|
name = TextField(_("Name"), validators=[Required()])
|
|
weight = CommaDecimalField(_("Weight"), default=1)
|
|
submit = SubmitField(_("Add"))
|
|
|
|
def __init__(self, project, edit=False, *args, **kwargs):
|
|
super(MemberForm, self).__init__(*args, **kwargs)
|
|
self.project = project
|
|
self.edit = edit
|
|
|
|
def validate_name(form, field):
|
|
if field.data == form.name.default:
|
|
raise ValidationError(_("User name incorrect"))
|
|
if (not form.edit and Person.query.filter(
|
|
Person.name == field.data,
|
|
Person.project == form.project,
|
|
Person.activated == True).all()):
|
|
raise ValidationError(_("This project already have this member"))
|
|
|
|
def save(self, project, person):
|
|
# if the user is already bound to the project, just reactivate him
|
|
person.name = self.name.data
|
|
person.project = project
|
|
person.weight = self.weight.data
|
|
|
|
return person
|
|
|
|
def fill(self, member):
|
|
self.name.data = member.name
|
|
self.weight.data = member.weight
|
|
|
|
|
|
class InviteForm(Form):
|
|
emails = TextAreaField(_("People to notify"))
|
|
submit = SubmitField(_("Send invites"))
|
|
|
|
def validate_emails(form, field):
|
|
validator = Email()
|
|
for email in [email.strip() for email in form.emails.data.split(",")]:
|
|
if not validator.regex.match(email):
|
|
raise ValidationError(_("The email %(email)s is not valid",
|
|
email=email))
|
|
|
|
|
|
class CreateArchiveForm(Form):
|
|
name = TextField(_("Name for this archive (optional)"), validators=[])
|
|
start_date = DateField(_("Start date"), validators=[Required()])
|
|
end_date = DateField(_("End date"), validators=[Required()], default=datetime.now)
|