8
0
Fork 0
mirror of https://gitlab2.federez.net/re2o/re2o synced 2024-12-22 23:13:46 +00:00

feat: Add statistics for deposits app

This commit is contained in:
Jean-Romain Garnier 2021-08-07 23:19:14 +02:00
parent 1861baeb12
commit 8138aa9af1
6 changed files with 254 additions and 21 deletions

View file

@ -21,7 +21,7 @@ msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2021-08-07 18:27+0200\n"
"POT-Creation-Date: 2021-08-07 23:10+0200\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language-Team: LANGUAGE <LL@li.org>\n"
@ -32,10 +32,12 @@ msgstr ""
"Plural-Forms: nplurals=2; plural=(n > 1);\n"
#: deposits/forms.py:44 deposits/templates/deposits/aff_deposit_item.html:31
#: deposits/views.py:301
msgid "Deposit item"
msgstr "Article (caution)"
#: deposits/forms.py:45 deposits/templates/deposits/aff_deposits.html:48
#: deposits/views.py:272
msgid "Payment method"
msgstr "Moyen de paiement"
@ -59,7 +61,7 @@ msgstr "rendue"
msgid "deposit amount"
msgstr "montant de la caution"
#: deposits/models.py:60 deposits/views.py:114
#: deposits/models.py:60 deposits/views.py:116
msgid "deposit"
msgstr "caution"
@ -98,7 +100,8 @@ msgid "deposit items"
msgstr "articles (cautions)"
#: deposits/templates/deposits/aff_deposit_item.html:32
#: deposits/templates/deposits/aff_deposits.html:44
#: deposits/templates/deposits/aff_deposits.html:44 deposits/views.py:276
#: deposits/views.py:305 deposits/views.py:338
msgid "Amount"
msgstr "Montant"
@ -144,7 +147,7 @@ msgstr ""
"Attention: voulez-vous vraiment supprimer cet objet %(objet_name)s "
"( %(objet)s ) ?"
#: deposits/templates/deposits/delete.html:36 deposits/views.py:70
#: deposits/templates/deposits/delete.html:36 deposits/views.py:72
msgid "Confirm"
msgstr "Confirmer"
@ -152,7 +155,7 @@ msgstr "Confirmer"
msgid "Create or edit deposit"
msgstr "Créer ou modifier une caution"
#: deposits/templates/deposits/deposit.html:35 deposits/views.py:173
#: deposits/templates/deposits/deposit.html:35 deposits/views.py:179
msgid "Add"
msgstr "Ajouter"
@ -189,66 +192,102 @@ msgid_plural "Unreturned deposits"
msgstr[0] "Caution non rendue"
msgstr[1] "Cautions non rendues"
#: deposits/templates/deposits/index_deposits.html:46
#: deposits/templates/deposits/index_deposits.html:45 deposits/views.py:274
#: deposits/views.py:303
msgid "Unreturned deposits"
msgstr "Cautions non rendues"
#: deposits/templates/deposits/index_deposits.html:57
#: deposits/templates/deposits/index_deposits.html:56 deposits/views.py:273
#: deposits/views.py:302
msgid "Returned deposits"
msgstr "Cautions rendues"
#: deposits/templates/deposits/index_stats.html:28
#: deposits/templates/deposits/index_stats.html:31
#: deposits/templates/deposits/navbar.html:36
msgid "Deposits statistics"
msgstr "Statistiques sur les cautions"
#: deposits/templates/deposits/navbar.html:30
msgid "View deposits"
msgstr "Voir les cautions"
#: deposits/views.py:64
#: deposits/views.py:66
msgid "The deposit was created."
msgstr "La caution a été créée."
#: deposits/views.py:71
#: deposits/views.py:73
msgid "New deposit"
msgstr "Nouvelle caution"
#: deposits/views.py:89
#: deposits/views.py:91
msgid "The deposit was edited."
msgstr "La caution a été modifiée."
#: deposits/views.py:95 deposits/views.py:196
#: deposits/views.py:97 deposits/views.py:202
msgid "Edit"
msgstr "Modifier"
#: deposits/views.py:96
#: deposits/views.py:98
msgid "Edit deposit"
msgstr "Modifier la caution"
#: deposits/views.py:111
#: deposits/views.py:113
msgid "The deposit was deleted."
msgstr "La caution a été supprimée."
#: deposits/views.py:168
#: deposits/views.py:174
msgid "The item was created."
msgstr "L'article a été créé."
#: deposits/views.py:174
#: deposits/views.py:180
msgid "New deposit item"
msgstr "Nouvel article (caution)"
#: deposits/views.py:191
#: deposits/views.py:197
msgid "The item was edited."
msgstr "L'article a été modifié."
#: deposits/views.py:197
#: deposits/views.py:203
msgid "Edit deposit item"
msgstr "Modifier l'article"
#: deposits/views.py:214
#: deposits/views.py:220
msgid "The items were deleted."
msgstr "Les articles ont été supprimés."
#: deposits/views.py:219
#: deposits/views.py:225
msgid "Delete"
msgstr "Supprimer"
#: deposits/views.py:220
#: deposits/views.py:226
msgid "Delete deposit item"
msgstr "Supprimer l'article"
#: deposits/views.py:270
msgid "Deposits by payment method"
msgstr "Cautions par moyen de paiement"
#: deposits/views.py:275 deposits/views.py:304 deposits/views.py:337
msgid "Total"
msgstr "Total"
#: deposits/views.py:299
msgid "Deposits by item type"
msgstr "Cautions par type d'article"
#: deposits/views.py:321
msgid "Not yet returned"
msgstr "Pas encore rendues"
#: deposits/views.py:326
msgid "Already returned"
msgstr "Déjà rendues"
#: deposits/views.py:334
msgid "Deposits amounts"
msgstr "Montants de cautions"
#: deposits/views.py:336
msgid "Category"
msgstr "Catégorie"

View file

@ -0,0 +1,40 @@
{% comment %}
Re2o est un logiciel d'administration développé initiallement au Rézo Metz. Il
se veut agnostique au réseau considéré, de manière à être installable en
quelques clics.
Copyright © 2021 Jean-Romain Garnier
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
{% endcomment %}
<h3>{{ stats_dict.title }}</h3>
<table class="table table-striped">
<thead>
<tr>
{% for header in stats_dict.headers %}
<th>{{ header }}</th>
{% endfor %}
</tr>
</thead>
{% for stats in stats_dict.data %}
<tr>
{% for item in stats %}
<td>{{ item }}</td>
{% endfor %}
</tr>
{% endfor %}
</table>

View file

@ -0,0 +1,36 @@
{% extends 'deposits/sidebar.html' %}
{% comment %}
Re2o est un logiciel d'administration développé initiallement au Rézo Metz. Il
se veut agnostique au réseau considéré, de manière à être installable en
quelques clics.
Copyright © 2021 Jean-Romain Garnier
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
{% endcomment %}
{% load i18n %}
{% block title %}{% trans "Deposits statistics" %}{% endblock %}
{% block content %}
<h2>{% trans "Deposits statistics" %}</h2>
{% for stats_dict in stats_list %}
{% include 'deposits/aff_stats.html' with stats_dict=stats_dict %}
<br/>
{% endfor %}
{% endblock %}

View file

@ -21,6 +21,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
{% endcomment %}
{% load i18n %}
<li><a class="dropdown-item" href="#"><i class="fa fa-balance-scale"></i>
{% trans "Deposits" %}
&raquo </a>
@ -31,5 +32,8 @@ with this program; if not, write to the Free Software Foundation, Inc.,
<li><a class="dropdown-item" href="{% url 'deposits:index-deposit-item' %}"><i
class="fa fa-barcode"></i>
{% trans "Deposit items" %}</a></li>
<li><a class="dropdown-item" href="{% url 'deposits:index-stats' %}"><i
class="fa fa-area-chart"></i>
{% trans "Deposits statistics" %}</a></li>
</ul>
</li>

View file

@ -41,4 +41,5 @@ urlpatterns = [
),
path("del_deposit_item", views.del_deposit_item, name="del-deposit-item"),
path("index_deposit_item", views.index_deposit_item, name="index-deposit-item"),
path("index_stats", views.index_stats, name="index-stats"),
]

View file

@ -28,6 +28,7 @@ from django.shortcuts import redirect, render
from django.template.loader import render_to_string
from django.urls import reverse
from django.utils.translation import ugettext as _
from django.db.models import Sum
from preferences.models import GeneralOption
from re2o.acl import (
@ -40,6 +41,7 @@ from re2o.acl import (
from re2o.base import re2o_paginator
from re2o.views import form
from users.models import User
from cotisations.models import Paiement
from .forms import DepositForm, DepositItemForm, DelDepositItemForm
from .models import Deposit, DepositItem
@ -118,7 +120,7 @@ def del_deposit(request, deposit, **_kwargs):
@login_required
@can_view_all(Deposit)
@can_view_all(Deposit, DepositItem, Paiement, User)
def index_deposits(request):
"""
View used to display the list of all deposits.
@ -239,6 +241,117 @@ def index_deposit_item(request):
return render(request, "deposits/index_deposit_item.html", {"item_list": item_list})
@login_required
@can_view_all(Deposit, DepositItem, Paiement, User)
def index_stats(request):
"""
View used to display general statistics about deposits
"""
# We want to build a list of tables for statistics
stats = []
# Statistics for payment methods
payment_data = []
for method in Paiement.objects.order_by("moyen"):
deposits = Deposit.objects.filter(payment_method=method)
amount = deposits.aggregate(Sum("deposit_amount")).get(
"deposit_amount__sum", None
)
payment_data.append(
(
method.moyen,
deposits.filter(returned=False).count(),
deposits.filter(returned=True).count(),
deposits.count(),
"{}".format(amount or 0),
)
)
stats.append(
{
"title": _("Deposits by payment method"),
"headers": [
_("Payment method"),
_("Returned deposits"),
_("Unreturned deposits"),
_("Total"),
_("Amount"),
],
"data": payment_data,
}
)
# Statistics for deposit items
items_data = []
for item in DepositItem.objects.order_by("name"):
deposits = Deposit.objects.filter(item=item)
amount = deposits.aggregate(Sum("deposit_amount")).get(
"deposit_amount__sum", None
)
items_data.append(
(
item.name,
deposits.filter(returned=False).count(),
deposits.filter(returned=True).count(),
deposits.count(),
"{}".format(amount or 0),
)
)
stats.append(
{
"title": _("Deposits by item type"),
"headers": [
_("Deposit item"),
_("Returned deposits"),
_("Unreturned deposits"),
_("Total"),
_("Amount"),
],
"data": items_data,
}
)
# Statistics for amounts
pending_amount = (
Deposit.objects.filter(returned=False)
.aggregate(Sum("deposit_amount"))
.get("deposit_amount__sum", None)
)
reimbursed_amount = (
Deposit.objects.filter(returned=True)
.aggregate(Sum("deposit_amount"))
.get("deposit_amount__sum", None)
)
amounts_data = [
(
_("Not yet returned"),
Deposit.objects.filter(returned=False).count(),
"{}".format(pending_amount or 0),
),
(
_("Already returned"),
Deposit.objects.filter(returned=True).count(),
"{}".format(reimbursed_amount or 0),
),
]
stats.append(
{
"title": _("Deposits amounts"),
"headers": [
_("Category"),
_("Total"),
_("Amount"),
],
"data": amounts_data,
}
)
return render(request, "deposits/index_stats.html", {"stats_list": stats})
# Canonic views for optional apps
def aff_profil(request, user):
"""View used to display the deposits on a user's profile."""