mirror of
https://github.com/spiral-project/ihatemoney.git
synced 2025-05-01 02:32:23 +02:00
feat: Optional field 'external link' in bill form.
An optional field has been added to the bill form to add a link to a real bill. A new action button allow user to see this bill. Breaking change with Bill model update for database, a migration is needed. See issue #429.
This commit is contained in:
parent
1f539b5c0d
commit
afc9353180
7 changed files with 50 additions and 11 deletions
|
@ -1,8 +1,8 @@
|
||||||
from flask_wtf.form import FlaskForm
|
from flask_wtf.form import FlaskForm
|
||||||
from wtforms.fields.core import SelectField, SelectMultipleField
|
from wtforms.fields.core import SelectField, SelectMultipleField
|
||||||
from wtforms.fields.html5 import DateField, DecimalField
|
from wtforms.fields.html5 import DateField, DecimalField, URLField
|
||||||
from wtforms.fields.simple import PasswordField, SubmitField, TextAreaField, StringField
|
from wtforms.fields.simple import PasswordField, SubmitField, TextAreaField, StringField
|
||||||
from wtforms.validators import Email, DataRequired, ValidationError, EqualTo, NumberRange
|
from wtforms.validators import Email, DataRequired, ValidationError, EqualTo, NumberRange, Optional
|
||||||
from flask_babel import lazy_gettext as _
|
from flask_babel import lazy_gettext as _
|
||||||
from flask import request
|
from flask import request
|
||||||
from werkzeug.security import generate_password_hash
|
from werkzeug.security import generate_password_hash
|
||||||
|
@ -146,6 +146,8 @@ class BillForm(FlaskForm):
|
||||||
what = StringField(_("What?"), validators=[DataRequired()])
|
what = StringField(_("What?"), validators=[DataRequired()])
|
||||||
payer = SelectField(_("Payer"), validators=[DataRequired()], coerce=int)
|
payer = SelectField(_("Payer"), validators=[DataRequired()], coerce=int)
|
||||||
amount = CalculatorStringField(_("Amount paid"), validators=[DataRequired()])
|
amount = CalculatorStringField(_("Amount paid"), validators=[DataRequired()])
|
||||||
|
external_link = URLField(_("External link"), validators=[Optional(
|
||||||
|
)], description="A link to an external document, related to this bill")
|
||||||
payed_for = SelectMultipleField(_("For whom?"),
|
payed_for = SelectMultipleField(_("For whom?"),
|
||||||
validators=[DataRequired()], coerce=int)
|
validators=[DataRequired()], coerce=int)
|
||||||
submit = SubmitField(_("Submit"))
|
submit = SubmitField(_("Submit"))
|
||||||
|
@ -155,6 +157,7 @@ class BillForm(FlaskForm):
|
||||||
bill.payer_id = self.payer.data
|
bill.payer_id = self.payer.data
|
||||||
bill.amount = self.amount.data
|
bill.amount = self.amount.data
|
||||||
bill.what = self.what.data
|
bill.what = self.what.data
|
||||||
|
bill.external_link = self.external_link.data
|
||||||
bill.date = self.date.data
|
bill.date = self.date.data
|
||||||
bill.owers = [Person.query.get(ower, project)
|
bill.owers = [Person.query.get(ower, project)
|
||||||
for ower in self.payed_for.data]
|
for ower in self.payed_for.data]
|
||||||
|
@ -165,6 +168,7 @@ class BillForm(FlaskForm):
|
||||||
self.payer.data = bill.payer_id
|
self.payer.data = bill.payer_id
|
||||||
self.amount.data = bill.amount
|
self.amount.data = bill.amount
|
||||||
self.what.data = bill.what
|
self.what.data = bill.what
|
||||||
|
self.external_link.data = bill.external_link
|
||||||
self.date.data = bill.date
|
self.date.data = bill.date
|
||||||
self.payed_for.data = [int(ower.id) for ower in bill.owers]
|
self.payed_for.data = [int(ower.id) for ower in bill.owers]
|
||||||
|
|
||||||
|
|
26
ihatemoney/migrations/versions/6c6fb2b7f229_.py
Normal file
26
ihatemoney/migrations/versions/6c6fb2b7f229_.py
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
"""empty message
|
||||||
|
|
||||||
|
Revision ID: 6c6fb2b7f229
|
||||||
|
Revises: a67119aa3ee5
|
||||||
|
Create Date: 2019-09-28 13:38:09.550747
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '6c6fb2b7f229'
|
||||||
|
down_revision = 'a67119aa3ee5'
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.add_column('bill', sa.Column('external_link', sa.UnicodeText(), nullable=True))
|
||||||
|
# ### end Alembic commands ###
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.drop_column('bill', 'external_link')
|
||||||
|
# ### end Alembic commands ###
|
|
@ -327,6 +327,7 @@ class Bill(db.Model):
|
||||||
date = db.Column(db.Date, default=datetime.now)
|
date = db.Column(db.Date, default=datetime.now)
|
||||||
creation_date = db.Column(db.Date, default=datetime.now)
|
creation_date = db.Column(db.Date, default=datetime.now)
|
||||||
what = db.Column(db.UnicodeText)
|
what = db.Column(db.UnicodeText)
|
||||||
|
external_link = db.Column(db.UnicodeText)
|
||||||
|
|
||||||
archive = db.Column(db.Integer, db.ForeignKey("archive.id"))
|
archive = db.Column(db.Integer, db.ForeignKey("archive.id"))
|
||||||
|
|
||||||
|
@ -340,6 +341,7 @@ class Bill(db.Model):
|
||||||
"date": self.date,
|
"date": self.date,
|
||||||
"creation_date": self.creation_date,
|
"creation_date": self.creation_date,
|
||||||
"what": self.what,
|
"what": self.what,
|
||||||
|
"external_link": self.external_link,
|
||||||
}
|
}
|
||||||
|
|
||||||
def pay_each(self):
|
def pay_each(self):
|
||||||
|
|
|
@ -32,7 +32,7 @@ body {
|
||||||
|
|
||||||
@media (min-width: 992px) {
|
@media (min-width: 992px) {
|
||||||
.projects-item {
|
.projects-item {
|
||||||
margin-left: auto!important;
|
margin-left: auto !important;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -108,7 +108,6 @@ body {
|
||||||
.identifier {
|
.identifier {
|
||||||
margin-bottom: 0;
|
margin-bottom: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#add-member-form {
|
#add-member-form {
|
||||||
|
@ -132,7 +131,7 @@ body {
|
||||||
|
|
||||||
/* Home */
|
/* Home */
|
||||||
.home-container {
|
.home-container {
|
||||||
background: linear-gradient(150deg, #abe128 0%, #43CA61 100%);
|
background: linear-gradient(150deg, #abe128 0%, #43ca61 100%);
|
||||||
}
|
}
|
||||||
.home {
|
.home {
|
||||||
padding-top: 1em;
|
padding-top: 1em;
|
||||||
|
@ -146,7 +145,7 @@ body {
|
||||||
min-width: 25em;
|
min-width: 25em;
|
||||||
border: 0;
|
border: 0;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
box-shadow: 0px 0px 10px rgba(83, 88, 93, .40);
|
box-shadow: 0px 0px 10px rgba(83, 88, 93, 0.4);
|
||||||
margin-right: 25px;
|
margin-right: 25px;
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
margin-left: 25px;
|
margin-left: 25px;
|
||||||
|
@ -159,7 +158,6 @@ body {
|
||||||
|
|
||||||
.empty-bill {
|
.empty-bill {
|
||||||
margin-top: 5rem !important;
|
margin-top: 5rem !important;
|
||||||
|
|
||||||
}
|
}
|
||||||
.empty-bill .card {
|
.empty-bill .card {
|
||||||
border: 0;
|
border: 0;
|
||||||
|
@ -240,7 +238,7 @@ footer .footer-right a {
|
||||||
@-moz-document url-prefix() {
|
@-moz-document url-prefix() {
|
||||||
/** Firefox style fix **/
|
/** Firefox style fix **/
|
||||||
footer .footer-right a {
|
footer .footer-right a {
|
||||||
padding-top: 2px
|
padding-top: 2px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
footer .footer-right a:hover {
|
footer .footer-right a:hover {
|
||||||
|
@ -290,7 +288,8 @@ footer .footer-left {
|
||||||
}
|
}
|
||||||
|
|
||||||
.bill-actions > .delete,
|
.bill-actions > .delete,
|
||||||
.bill-actions > .edit {
|
.bill-actions > .edit,
|
||||||
|
.bill-actions > .see {
|
||||||
font-size: 0px;
|
font-size: 0px;
|
||||||
display: block;
|
display: block;
|
||||||
width: 16px;
|
width: 16px;
|
||||||
|
@ -308,6 +307,10 @@ footer .footer-left {
|
||||||
background: url("../images/edit.png") no-repeat right;
|
background: url("../images/edit.png") no-repeat right;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bill-actions > .see {
|
||||||
|
background: url("../images/see.png") no-repeat right;
|
||||||
|
}
|
||||||
|
|
||||||
#bill_table {
|
#bill_table {
|
||||||
margin-top: 30px;
|
margin-top: 30px;
|
||||||
}
|
}
|
||||||
|
|
BIN
ihatemoney/static/images/see.png
Normal file
BIN
ihatemoney/static/images/see.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 448 B |
|
@ -14,7 +14,7 @@
|
||||||
{{ field(class=class, placeholder=placeholder) | safe }}
|
{{ field(class=class, placeholder=placeholder) | safe }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if field.description %}
|
{% if field.description %}
|
||||||
<p class="help-inline">{{ field.description }}</p>
|
<small id="{{field.name}}_description"" class="form-text text-muted">{{ field.description }}</small>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -95,6 +95,7 @@
|
||||||
{{ input(form.what, inline=True) }}
|
{{ input(form.what, inline=True) }}
|
||||||
{{ input(form.payer, inline=True, class="form-control custom-select") }}
|
{{ input(form.payer, inline=True, class="form-control custom-select") }}
|
||||||
{{ input(form.amount, inline=True) }}
|
{{ input(form.amount, inline=True) }}
|
||||||
|
{{ input(form.external_link, inline=True) }}
|
||||||
|
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-3" for="payed_for">{{ _("For whom?") }}</label>
|
<label class="col-3" for="payed_for">{{ _("For whom?") }}</label>
|
||||||
|
|
|
@ -134,6 +134,9 @@
|
||||||
<td class="bill-actions">
|
<td class="bill-actions">
|
||||||
<a class="edit" href="{{ url_for(".edit_bill", bill_id=bill.id) }}" title="{{ _("edit") }}">{{ _('edit') }}</a>
|
<a class="edit" href="{{ url_for(".edit_bill", bill_id=bill.id) }}" title="{{ _("edit") }}">{{ _('edit') }}</a>
|
||||||
<a class="delete" href="{{ url_for(".delete_bill", bill_id=bill.id) }}" title="{{ _("delete") }}">{{ _('delete') }}</a>
|
<a class="delete" href="{{ url_for(".delete_bill", bill_id=bill.id) }}" title="{{ _("delete") }}">{{ _('delete') }}</a>
|
||||||
|
{% if bill.external_link %}
|
||||||
|
<a class="see" href="{{ bill.external_link }}" ref="noopener" target="_blank" title="{{ _("see") }}">{{ _('see') }} </a>
|
||||||
|
{% endif %}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
Loading…
Reference in a new issue