mirror of
https://github.com/spiral-project/ihatemoney.git
synced 2025-04-29 01:42:37 +02:00
Makes the computation working.
This commit is contained in:
parent
548101d8bb
commit
769bcbf3f1
4 changed files with 63 additions and 30 deletions
|
@ -1,3 +1,5 @@
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from flaskext.sqlalchemy import SQLAlchemy
|
from flaskext.sqlalchemy import SQLAlchemy
|
||||||
|
|
||||||
|
@ -12,6 +14,28 @@ class Project(db.Model):
|
||||||
contact_email = db.Column(db.String)
|
contact_email = db.Column(db.String)
|
||||||
members = db.relationship("Person", backref="project")
|
members = db.relationship("Person", backref="project")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def active_members(self):
|
||||||
|
return [m for m in self.members if m.activated]
|
||||||
|
|
||||||
|
def get_balance(self):
|
||||||
|
|
||||||
|
balances, should_pay, should_receive = defaultdict(int), defaultdict(int), defaultdict(int)
|
||||||
|
|
||||||
|
# for each person
|
||||||
|
for person in self.members:
|
||||||
|
# get the list of bills he has to pay
|
||||||
|
bills = Bill.query.filter(Bill.owers.contains(person))
|
||||||
|
for bill in bills.all():
|
||||||
|
if person != bill.payer:
|
||||||
|
should_pay[person] += bill.pay_each()
|
||||||
|
should_receive[bill.payer] += bill.pay_each()
|
||||||
|
|
||||||
|
for person in self.members:
|
||||||
|
balances[person] = should_receive[person] - should_pay[person]
|
||||||
|
|
||||||
|
return balances
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "<Project %s>" % self.name
|
return "<Project %s>" % self.name
|
||||||
|
|
||||||
|
@ -22,7 +46,7 @@ class Person(db.Model):
|
||||||
bills = db.relationship("Bill", backref="payer")
|
bills = db.relationship("Bill", backref="payer")
|
||||||
|
|
||||||
name = db.Column(db.UnicodeText)
|
name = db.Column(db.UnicodeText)
|
||||||
# activated = db.Column(db.Boolean, default=True)
|
activated = db.Column(db.Boolean, default=True)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
|
@ -1,4 +1,9 @@
|
||||||
{% extends "layout.html" %}
|
{% extends "layout.html" %}
|
||||||
|
|
||||||
|
{% block top_menu %}
|
||||||
|
<a href="{{ url_for('list_bills', project_id=project.id) }}">Back to the list</a>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h2>Computations</h2>
|
<h2>Computations</h2>
|
||||||
|
|
||||||
|
@ -12,14 +17,14 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="container span-6 last">
|
<div class="container span-6 last">
|
||||||
<a class="awesome large green button" onclick = "if (! confirm('Are you sure ?')) return false;" href="{{ url_for("reset_bills") }}">Mark this as payed / processed</a>
|
<a class="awesome large green button" onclick = "if (! confirm('Are you sure ?')) return false;" href="">Mark this as payed / processed</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
<thead><tr><th>Name</th><th>Balance</th></tr></thead>
|
<thead><tr><th>Name</th><th>Balance</th></tr></thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for name, balance in balances.items() %}
|
{% for person, balance in balances.items() %}
|
||||||
<tr><td>{{ name }}</td><td>{{ balance }}</td></tr>
|
<tr><td>{{ person.name }}</td><td>{{ balance }}</td></tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
|
@ -7,16 +7,22 @@
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div id="leftmenu" class="span-6">
|
<div id="leftmenu" class="span-4">
|
||||||
<ul>
|
<ul>
|
||||||
{% for member in project.members %}
|
{% set balance = project.get_balance() %}
|
||||||
<li>{{ member.name }}</li>
|
{% for member in project.active_members %}
|
||||||
|
<li> {{ member.name }} {{ balance[member] }} <a href="{{ url_for("remove_member", project_id=project.id, member_id=member.id) }}">x</a></li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
{% set form=member_form %}
|
<form action="{{ url_for("add_member", project_id=project.id) }}" method="post">
|
||||||
{% include "member_form.html" %}
|
{{ forms.add_member(member_form) }}
|
||||||
|
</form>
|
||||||
</div>
|
</div>
|
||||||
<div id="content" class="span-18 last">
|
<div id="content" class="uniForm span-20 last">
|
||||||
|
<form id="add_bill" action="{{ url_for('add_bill', project_id=project.id) }}" method="post" style="width: 400px; display: none">
|
||||||
|
{{ forms.add_bill(bill_form) }}
|
||||||
|
</form>
|
||||||
|
|
||||||
{% if bills.count() > 0 %}
|
{% if bills.count() > 0 %}
|
||||||
<table>
|
<table>
|
||||||
<thead><tr><th>When ?</th><th>Who paid?</th><th>for what ?</th><th>Owers</th><th>How much ?</th><th>Actions</th></tr></thead>
|
<thead><tr><th>When ?</th><th>Who paid?</th><th>for what ?</th><th>Owers</th><th>How much ?</th><th>Actions</th></tr></thead>
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
from flask import (Flask, session, request, redirect, url_for, render_template,
|
from flask import (Flask, session, request, redirect, url_for, render_template,
|
||||||
flash)
|
flash)
|
||||||
from flaskext.mail import Mail, Message
|
from flaskext.mail import Mail, Message
|
||||||
|
@ -107,7 +109,10 @@ def list_bills(project):
|
||||||
# FIXME filter to only get the bills for this particular project
|
# FIXME filter to only get the bills for this particular project
|
||||||
bills = Bill.query.order_by(Bill.id.asc())
|
bills = Bill.query.order_by(Bill.id.asc())
|
||||||
return render_template("list_bills.html",
|
return render_template("list_bills.html",
|
||||||
bills=bills, project=project, member_form=MemberForm(project))
|
bills=bills, project=project,
|
||||||
|
member_form=MemberForm(project),
|
||||||
|
bill_form=get_billform_for(project)
|
||||||
|
)
|
||||||
|
|
||||||
@app.route("/<string:project_id>/members/add", methods=["GET", "POST"])
|
@app.route("/<string:project_id>/members/add", methods=["GET", "POST"])
|
||||||
@requires_auth
|
@requires_auth
|
||||||
|
@ -121,10 +126,20 @@ def add_member(project):
|
||||||
return redirect(url_for("list_bills", project_id=project.id))
|
return redirect(url_for("list_bills", project_id=project.id))
|
||||||
return render_template("add_member.html", form=form, project=project)
|
return render_template("add_member.html", form=form, project=project)
|
||||||
|
|
||||||
|
@app.route("/<string:project_id>/members/<int:member_id>/delete", methods=["GET", "POST"])
|
||||||
|
@requires_auth
|
||||||
|
def remove_member(project, member_id):
|
||||||
|
person = Person.query.get_or_404(member_id)
|
||||||
|
if person.project == project:
|
||||||
|
person.activated = False
|
||||||
|
db.session.commit()
|
||||||
|
flash("%s has been removed" % person.name)
|
||||||
|
return redirect(url_for("list_bills", project_id=project.id))
|
||||||
|
|
||||||
@app.route("/<string:project_id>/add", methods=["GET", "POST"])
|
@app.route("/<string:project_id>/add", methods=["GET", "POST"])
|
||||||
@requires_auth
|
@requires_auth
|
||||||
def add_bill(project):
|
def add_bill(project):
|
||||||
form = get_billform_for(project.id)
|
form = get_billform_for(project)
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
if form.validate():
|
if form.validate():
|
||||||
db.session.add(form.save())
|
db.session.add(form.save())
|
||||||
|
@ -140,24 +155,7 @@ def add_bill(project):
|
||||||
@requires_auth
|
@requires_auth
|
||||||
def compute_bills(project):
|
def compute_bills(project):
|
||||||
"""Compute the sum each one have to pay to each other and display it"""
|
"""Compute the sum each one have to pay to each other and display it"""
|
||||||
# FIXME make it work
|
return render_template("compute_bills.html", project=project)
|
||||||
|
|
||||||
balances, should_pay, should_receive = {}, {}, {}
|
|
||||||
# for each person, get the list of should_pay other have for him
|
|
||||||
for name, void in PAYER_CHOICES:
|
|
||||||
bills = Bill.query.join(BillOwer).filter(Bill.processed==False)\
|
|
||||||
.filter(BillOwer.name==name)
|
|
||||||
for bill in bills.all():
|
|
||||||
if name != bill.payer:
|
|
||||||
should_pay.setdefault(name, 0)
|
|
||||||
should_pay[name] += bill.pay_each()
|
|
||||||
should_receive.setdefault(bill.payer, 0)
|
|
||||||
should_receive[bill.payer] += bill.pay_each()
|
|
||||||
|
|
||||||
for name, void in PAYER_CHOICES:
|
|
||||||
balances[name] = should_receive.get(name, 0) - should_pay.get(name, 0)
|
|
||||||
|
|
||||||
return render_template("compute_bills.html", balances=balances, project=project)
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/<string:project_id>/reset")
|
@app.route("/<string:project_id>/reset")
|
||||||
|
|
Loading…
Reference in a new issue