2017-01-15 23:01:18 +00:00
|
|
|
# Re2o est un logiciel d'administration développé initiallement au rezometz. Il
|
|
|
|
# se veut agnostique au réseau considéré, de manière à être installable en
|
|
|
|
# quelques clics.
|
|
|
|
#
|
|
|
|
# Copyright © 2017 Gabriel Détraz
|
|
|
|
# Copyright © 2017 Goulven Kermarec
|
|
|
|
# Copyright © 2017 Augustin Lemesle
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2016-07-02 13:58:50 +00:00
|
|
|
# App de gestion des users pour re2o
|
|
|
|
# Goulven Kermarec, Gabriel Détraz
|
|
|
|
# Gplv2
|
2017-09-10 23:29:24 +00:00
|
|
|
from __future__ import unicode_literals
|
2017-10-13 20:47:32 +00:00
|
|
|
import os
|
2016-07-02 13:58:50 +00:00
|
|
|
from django.shortcuts import render, redirect
|
2016-07-22 01:25:04 +00:00
|
|
|
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
|
2016-07-09 02:12:09 +00:00
|
|
|
from django.contrib.auth.decorators import login_required, permission_required
|
2016-07-02 13:58:50 +00:00
|
|
|
from django.contrib import messages
|
2017-10-13 20:47:32 +00:00
|
|
|
from django.db.models import ProtectedError
|
2016-07-21 14:58:12 +00:00
|
|
|
from django.db import transaction
|
2016-07-13 23:54:06 +00:00
|
|
|
from django.forms import modelformset_factory, formset_factory
|
2017-10-13 20:47:32 +00:00
|
|
|
from django.utils import timezone
|
2016-07-21 14:58:12 +00:00
|
|
|
from reversion import revisions as reversion
|
2016-12-11 16:24:04 +00:00
|
|
|
from reversion.models import Version
|
2017-10-13 20:47:32 +00:00
|
|
|
# Import des models, forms et fonctions re2o
|
|
|
|
from users.models import User
|
|
|
|
from re2o.settings import LOGO_PATH
|
|
|
|
from re2o import settings
|
|
|
|
from re2o.views import form
|
|
|
|
from preferences.models import OptionalUser, AssoOption, GeneralOption
|
|
|
|
from .models import Facture, Article, Vente, Paiement, Banque
|
2017-10-13 03:08:30 +00:00
|
|
|
from .forms import NewFactureForm, TrezEditFactureForm, EditFactureForm
|
|
|
|
from .forms import ArticleForm, DelArticleForm, PaiementForm, DelPaiementForm
|
|
|
|
from .forms import BanqueForm, DelBanqueForm, NewFactureFormPdf
|
|
|
|
from .forms import SelectArticleForm, CreditSoldeForm
|
2016-07-08 01:27:02 +00:00
|
|
|
from .tex import render_tex
|
2016-07-02 13:58:50 +00:00
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-02 13:58:50 +00:00
|
|
|
def new_facture(request, userid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Creation d'une facture pour un user. Renvoie la liste des articles
|
|
|
|
et crée des factures dans un formset. Utilise un peu de js coté template
|
|
|
|
pour ajouter des articles.
|
|
|
|
Parse les article et boucle dans le formset puis save les ventes,
|
|
|
|
enfin sauve la facture parente.
|
|
|
|
TODO : simplifier cette fonction, déplacer l'intelligence coté models
|
|
|
|
Facture et Vente."""
|
2016-07-02 15:58:04 +00:00
|
|
|
try:
|
|
|
|
user = User.objects.get(pk=userid)
|
|
|
|
except User.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Utilisateur inexistant")
|
2016-07-02 15:58:04 +00:00
|
|
|
return redirect("/cotisations/")
|
|
|
|
facture = Facture(user=user)
|
2016-07-14 18:51:45 +00:00
|
|
|
# Le template a besoin de connaitre les articles pour le js
|
|
|
|
article_list = Article.objects.all()
|
|
|
|
# On envoie la form fature et un formset d'articles
|
2016-07-02 13:58:50 +00:00
|
|
|
facture_form = NewFactureForm(request.POST or None, instance=facture)
|
2016-07-14 18:51:45 +00:00
|
|
|
article_formset = formset_factory(SelectArticleForm)(request.POST or None)
|
2016-07-14 11:55:46 +00:00
|
|
|
if facture_form.is_valid() and article_formset.is_valid():
|
2017-10-13 20:47:32 +00:00
|
|
|
new_facture_instance = facture_form.save(commit=False)
|
2016-07-14 11:55:46 +00:00
|
|
|
articles = article_formset
|
2016-07-14 18:51:45 +00:00
|
|
|
# Si au moins un article est rempli
|
2016-07-14 11:55:46 +00:00
|
|
|
if any(art.cleaned_data for art in articles):
|
2017-10-13 20:47:32 +00:00
|
|
|
options, _created = OptionalUser.objects.get_or_create()
|
2017-06-26 17:23:01 +00:00
|
|
|
user_solde = options.user_solde
|
|
|
|
solde_negatif = options.solde_negatif
|
2017-10-13 03:08:30 +00:00
|
|
|
# Si on paye par solde, que l'option est activée,
|
|
|
|
# on vérifie que le négatif n'est pas atteint
|
2017-06-26 17:23:01 +00:00
|
|
|
if user_solde:
|
2017-10-13 20:47:32 +00:00
|
|
|
if new_facture_instance.paiement == Paiement.objects.get_or_create(
|
|
|
|
moyen='solde'
|
2017-10-13 03:08:30 +00:00
|
|
|
)[0]:
|
2017-06-26 17:23:01 +00:00
|
|
|
prix_total = 0
|
|
|
|
for art_item in articles:
|
|
|
|
if art_item.cleaned_data:
|
2017-10-13 03:08:30 +00:00
|
|
|
prix_total += art_item.cleaned_data['article']\
|
|
|
|
.prix*art_item.cleaned_data['quantity']
|
2017-06-26 17:23:01 +00:00
|
|
|
if float(user.solde) - float(prix_total) < solde_negatif:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Le solde est insuffisant pour\
|
|
|
|
effectuer l'opération")
|
2017-06-26 17:23:01 +00:00
|
|
|
return redirect("/users/profil/" + userid)
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
2017-10-13 20:47:32 +00:00
|
|
|
new_facture_instance.save()
|
2016-07-21 14:58:12 +00:00
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Création")
|
2016-07-14 11:55:46 +00:00
|
|
|
for art_item in articles:
|
|
|
|
if art_item.cleaned_data:
|
|
|
|
article = art_item.cleaned_data['article']
|
|
|
|
quantity = art_item.cleaned_data['quantity']
|
2017-10-13 03:08:30 +00:00
|
|
|
new_vente = Vente.objects.create(
|
2017-10-13 20:47:32 +00:00
|
|
|
facture=new_facture_instance,
|
|
|
|
name=article.name,
|
|
|
|
prix=article.prix,
|
|
|
|
iscotisation=article.iscotisation,
|
|
|
|
duration=article.duration,
|
|
|
|
number=quantity
|
|
|
|
)
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
new_vente.save()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Création")
|
2017-10-13 03:08:30 +00:00
|
|
|
if any(art_item.cleaned_data['article'].iscotisation
|
2017-10-13 20:47:32 +00:00
|
|
|
for art_item in articles if art_item.cleaned_data):
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.success(
|
|
|
|
request,
|
|
|
|
"La cotisation a été prolongée\
|
|
|
|
pour l'adhérent %s jusqu'au %s" % (
|
|
|
|
user.pseudo, user.end_adhesion()
|
|
|
|
)
|
|
|
|
)
|
2016-07-14 11:55:46 +00:00
|
|
|
else:
|
|
|
|
messages.success(request, "La facture a été crée")
|
|
|
|
return redirect("/users/profil/" + userid)
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(
|
2017-10-13 20:47:32 +00:00
|
|
|
request,
|
|
|
|
u"Il faut au moins un article valide pour créer une facture"
|
2017-10-13 03:08:30 +00:00
|
|
|
)
|
|
|
|
return form({
|
|
|
|
'factureform': facture_form,
|
|
|
|
'venteform': article_formset,
|
|
|
|
'articlelist': article_list
|
|
|
|
}, 'cotisations/new_facture.html', request)
|
|
|
|
|
2016-07-02 13:58:50 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-08 01:27:02 +00:00
|
|
|
def new_facture_pdf(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Permet de générer un pdf d'une facture. Réservée
|
|
|
|
au trésorier, permet d'emettre des factures sans objet
|
|
|
|
Vente ou Facture correspondant en bdd"""
|
2016-07-09 21:26:17 +00:00
|
|
|
facture_form = NewFactureFormPdf(request.POST or None)
|
|
|
|
if facture_form.is_valid():
|
2017-10-13 20:47:32 +00:00
|
|
|
options, _created = AssoOption.objects.get_or_create()
|
2016-07-09 21:26:17 +00:00
|
|
|
tbl = []
|
|
|
|
article = facture_form.cleaned_data['article']
|
|
|
|
quantite = facture_form.cleaned_data['number']
|
|
|
|
paid = facture_form.cleaned_data['paid']
|
|
|
|
destinataire = facture_form.cleaned_data['dest']
|
2016-07-13 23:54:06 +00:00
|
|
|
chambre = facture_form.cleaned_data['chambre']
|
|
|
|
fid = facture_form.cleaned_data['fid']
|
2017-10-13 20:47:32 +00:00
|
|
|
for art in article:
|
|
|
|
tbl.append([art, quantite, art.prix * quantite])
|
2016-07-09 21:26:17 +00:00
|
|
|
prix_total = sum(a[2] for a in tbl)
|
2017-10-13 03:08:30 +00:00
|
|
|
user = {'name': destinataire, 'room': chambre}
|
|
|
|
return render_tex(request, 'cotisations/factures.tex', {
|
|
|
|
'DATE': timezone.now(),
|
|
|
|
'dest': user,
|
|
|
|
'fid': fid,
|
|
|
|
'article': tbl,
|
|
|
|
'total': prix_total,
|
|
|
|
'paid': paid,
|
|
|
|
'asso_name': options.name,
|
|
|
|
'line1': options.adresse1,
|
|
|
|
'line2': options.adresse2,
|
|
|
|
'siret': options.siret,
|
|
|
|
'email': options.contact,
|
|
|
|
'phone': options.telephone,
|
|
|
|
'tpl_path': os.path.join(settings.BASE_DIR, LOGO_PATH)
|
|
|
|
})
|
|
|
|
return form({
|
|
|
|
'factureform': facture_form
|
|
|
|
}, 'cotisations/facture.html', request)
|
|
|
|
|
2016-07-08 01:27:02 +00:00
|
|
|
|
2016-07-11 22:05:07 +00:00
|
|
|
@login_required
|
|
|
|
def facture_pdf(request, factureid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Affiche en pdf une facture. Cree une ligne par Vente de la facture,
|
|
|
|
et génére une facture avec le total, le moyen de paiement, l'adresse
|
|
|
|
de l'adhérent, etc. Réservée à self pour un user sans droits,
|
|
|
|
les droits cableurs permettent d'afficher toute facture"""
|
2016-07-11 22:05:07 +00:00
|
|
|
try:
|
|
|
|
facture = Facture.objects.get(pk=factureid)
|
|
|
|
except Facture.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Facture inexistante")
|
2016-07-11 22:05:07 +00:00
|
|
|
return redirect("/cotisations/")
|
2017-10-13 03:08:30 +00:00
|
|
|
if not request.user.has_perms(('cableur',))\
|
|
|
|
and facture.user != request.user:
|
|
|
|
messages.error(request, "Vous ne pouvez pas afficher une facture ne vous\
|
|
|
|
appartenant pas sans droit cableur")
|
2016-07-11 22:05:07 +00:00
|
|
|
return redirect("/users/profil/" + str(request.user.id))
|
2016-07-14 20:29:30 +00:00
|
|
|
if not facture.valid:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Vous ne pouvez pas afficher\
|
|
|
|
une facture non valide")
|
2016-07-14 20:29:30 +00:00
|
|
|
return redirect("/users/profil/" + str(request.user.id))
|
2017-10-13 20:47:32 +00:00
|
|
|
ventes_objects = Vente.objects.all().filter(facture=facture)
|
2016-07-11 22:05:07 +00:00
|
|
|
ventes = []
|
2017-10-13 20:47:32 +00:00
|
|
|
options, _created = AssoOption.objects.get_or_create()
|
|
|
|
for vente in ventes_objects:
|
|
|
|
ventes.append([vente, vente.number, vente.prix_total])
|
2017-10-13 03:08:30 +00:00
|
|
|
return render_tex(request, 'cotisations/factures.tex', {
|
|
|
|
'paid': True,
|
|
|
|
'fid': facture.id,
|
|
|
|
'DATE': facture.date,
|
|
|
|
'dest': facture.user,
|
|
|
|
'article': ventes,
|
|
|
|
'total': facture.prix_total(),
|
|
|
|
'asso_name': options.name,
|
|
|
|
'line1': options.adresse1,
|
|
|
|
'line2': options.adresse2,
|
|
|
|
'siret': options.siret,
|
|
|
|
'email': options.contact,
|
|
|
|
'phone': options.telephone,
|
|
|
|
'tpl_path': os.path.join(settings.BASE_DIR, LOGO_PATH)
|
|
|
|
})
|
|
|
|
|
2016-07-11 22:05:07 +00:00
|
|
|
|
2016-07-13 23:54:06 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-02 13:58:50 +00:00
|
|
|
def edit_facture(request, factureid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Permet l'édition d'une facture. On peut y éditer les ventes
|
|
|
|
déjà effectuer, ou rendre une facture invalide (non payées, chèque
|
|
|
|
en bois etc). Mets à jour les durée de cotisation attenantes"""
|
2016-07-02 13:58:50 +00:00
|
|
|
try:
|
|
|
|
facture = Facture.objects.get(pk=factureid)
|
|
|
|
except Facture.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Facture inexistante")
|
2016-07-02 13:58:50 +00:00
|
|
|
return redirect("/cotisations/")
|
2017-10-03 03:51:14 +00:00
|
|
|
if request.user.has_perms(['tresorier']):
|
2017-10-13 03:08:30 +00:00
|
|
|
facture_form = TrezEditFactureForm(
|
|
|
|
request.POST or None,
|
|
|
|
instance=facture
|
|
|
|
)
|
2016-07-14 20:29:30 +00:00
|
|
|
elif facture.control or not facture.valid:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Vous ne pouvez pas editer une facture\
|
|
|
|
controlée ou invalidée par le trésorier")
|
2016-07-14 20:29:30 +00:00
|
|
|
return redirect("/cotisations/")
|
|
|
|
else:
|
|
|
|
facture_form = EditFactureForm(request.POST or None, instance=facture)
|
2016-07-13 23:54:06 +00:00
|
|
|
ventes_objects = Vente.objects.filter(facture=facture)
|
2017-10-13 03:08:30 +00:00
|
|
|
vente_form_set = modelformset_factory(
|
2017-10-13 20:47:32 +00:00
|
|
|
Vente,
|
|
|
|
fields=('name', 'number'),
|
|
|
|
extra=0,
|
|
|
|
max_num=len(ventes_objects)
|
|
|
|
)
|
2016-07-13 23:54:06 +00:00
|
|
|
vente_form = vente_form_set(request.POST or None, queryset=ventes_objects)
|
|
|
|
if facture_form.is_valid() and vente_form.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
facture_form.save()
|
|
|
|
vente_form.save()
|
|
|
|
reversion.set_user(request.user)
|
2017-10-13 03:08:30 +00:00
|
|
|
reversion.set_comment("Champs modifié(s) : %s" % ', '.join(
|
|
|
|
field for form in vente_form for field
|
2017-10-13 20:47:32 +00:00
|
|
|
in facture_form.changed_data + form.changed_data))
|
2016-07-02 13:58:50 +00:00
|
|
|
messages.success(request, "La facture a bien été modifiée")
|
|
|
|
return redirect("/cotisations/")
|
2017-10-13 03:08:30 +00:00
|
|
|
return form({
|
|
|
|
'factureform': facture_form,
|
|
|
|
'venteform': vente_form
|
|
|
|
}, 'cotisations/edit_facture.html', request)
|
|
|
|
|
2016-07-02 13:58:50 +00:00
|
|
|
|
2016-07-14 20:29:30 +00:00
|
|
|
@login_required
|
|
|
|
@permission_required('cableur')
|
|
|
|
def del_facture(request, factureid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Suppression d'une facture. Supprime en cascade les ventes
|
|
|
|
et cotisations filles"""
|
2016-07-14 20:29:30 +00:00
|
|
|
try:
|
|
|
|
facture = Facture.objects.get(pk=factureid)
|
|
|
|
except Facture.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Facture inexistante")
|
2016-07-14 20:29:30 +00:00
|
|
|
return redirect("/cotisations/")
|
2017-10-13 20:47:32 +00:00
|
|
|
if facture.control or not facture.valid:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Vous ne pouvez pas editer une facture\
|
|
|
|
controlée ou invalidée par le trésorier")
|
2016-07-14 20:29:30 +00:00
|
|
|
return redirect("/cotisations/")
|
2016-07-18 16:37:52 +00:00
|
|
|
if request.method == "POST":
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
facture.delete()
|
|
|
|
reversion.set_user(request.user)
|
2016-07-18 16:37:52 +00:00
|
|
|
messages.success(request, "La facture a été détruite")
|
|
|
|
return redirect("/cotisations/")
|
2017-10-13 03:08:30 +00:00
|
|
|
return form({
|
|
|
|
'objet': facture,
|
|
|
|
'objet_name': 'facture'
|
|
|
|
}, 'cotisations/delete.html', request)
|
|
|
|
|
2016-07-14 20:29:30 +00:00
|
|
|
|
2017-06-26 17:23:01 +00:00
|
|
|
@login_required
|
|
|
|
@permission_required('cableur')
|
|
|
|
def credit_solde(request, userid):
|
|
|
|
""" Credit ou débit de solde """
|
|
|
|
try:
|
|
|
|
user = User.objects.get(pk=userid)
|
|
|
|
except User.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Utilisateur inexistant")
|
2017-06-26 17:23:01 +00:00
|
|
|
return redirect("/cotisations/")
|
|
|
|
facture = CreditSoldeForm(request.POST or None)
|
|
|
|
if facture.is_valid():
|
|
|
|
facture_instance = facture.save(commit=False)
|
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
facture_instance.user = user
|
|
|
|
facture_instance.save()
|
|
|
|
reversion.set_user(request.user)
|
2017-10-13 03:08:30 +00:00
|
|
|
reversion.set_comment("Création")
|
|
|
|
new_vente = Vente.objects.create(
|
2017-10-13 20:47:32 +00:00
|
|
|
facture=facture_instance,
|
|
|
|
name="solde",
|
|
|
|
prix=facture.cleaned_data['montant'],
|
|
|
|
iscotisation=False,
|
|
|
|
duration=0,
|
|
|
|
number=1
|
|
|
|
)
|
2017-06-26 17:23:01 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
new_vente.save()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Création")
|
|
|
|
messages.success(request, "Solde modifié")
|
|
|
|
return redirect("/cotisations/")
|
|
|
|
return form({'factureform': facture}, 'cotisations/facture.html', request)
|
|
|
|
|
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-06 18:57:31 +00:00
|
|
|
def add_article(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Ajoute un article. Champs : désignation,
|
|
|
|
prix, est-ce une cotisation et si oui sa durée
|
|
|
|
Réservé au trésorier
|
|
|
|
Nota bene : les ventes déjà effectuées ne sont pas reliées
|
|
|
|
aux articles en vente. La désignation, le prix... sont
|
|
|
|
copiés à la création de la facture. Un changement de prix n'a
|
|
|
|
PAS de conséquence sur les ventes déjà faites"""
|
2016-07-06 18:57:31 +00:00
|
|
|
article = ArticleForm(request.POST or None)
|
|
|
|
if article.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
article.save()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Création")
|
2016-07-06 18:57:31 +00:00
|
|
|
messages.success(request, "L'article a été ajouté")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_article/")
|
|
|
|
return form({'factureform': article}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-08 01:27:02 +00:00
|
|
|
def edit_article(request, articleid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Edition d'un article (designation, prix, etc)
|
|
|
|
Réservé au trésorier"""
|
2016-07-08 01:27:02 +00:00
|
|
|
try:
|
|
|
|
article_instance = Article.objects.get(pk=articleid)
|
|
|
|
except Article.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Entrée inexistante")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_article/")
|
|
|
|
article = ArticleForm(request.POST or None, instance=article_instance)
|
|
|
|
if article.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
article.save()
|
|
|
|
reversion.set_user(request.user)
|
2017-10-13 03:08:30 +00:00
|
|
|
reversion.set_comment(
|
|
|
|
"Champs modifié(s) : %s" % ', '.join(
|
|
|
|
field for field in article.changed_data
|
|
|
|
)
|
|
|
|
)
|
2016-07-08 01:27:02 +00:00
|
|
|
messages.success(request, "Type d'article modifié")
|
|
|
|
return redirect("/cotisations/index_article/")
|
2016-07-06 18:57:31 +00:00
|
|
|
return form({'factureform': article}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-06 18:57:31 +00:00
|
|
|
def del_article(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Suppression d'un article en vente"""
|
2016-07-06 18:57:31 +00:00
|
|
|
article = DelArticleForm(request.POST or None)
|
|
|
|
if article.is_valid():
|
|
|
|
article_del = article.cleaned_data['articles']
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
article_del.delete()
|
|
|
|
reversion.set_user(request.user)
|
2016-07-06 18:57:31 +00:00
|
|
|
messages.success(request, "Le/les articles ont été supprimé")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_article")
|
2016-07-06 18:57:31 +00:00
|
|
|
return form({'factureform': article}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-06 19:43:39 +00:00
|
|
|
def add_paiement(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Ajoute un moyen de paiement. Relié aux factures
|
|
|
|
via foreign key"""
|
2016-07-06 19:43:39 +00:00
|
|
|
paiement = PaiementForm(request.POST or None)
|
|
|
|
if paiement.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
paiement.save()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Création")
|
2016-07-06 19:43:39 +00:00
|
|
|
messages.success(request, "Le moyen de paiement a été ajouté")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_paiement/")
|
|
|
|
return form({'factureform': paiement}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-08 01:27:02 +00:00
|
|
|
def edit_paiement(request, paiementid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Edition d'un moyen de paiement"""
|
2016-07-08 01:27:02 +00:00
|
|
|
try:
|
|
|
|
paiement_instance = Paiement.objects.get(pk=paiementid)
|
|
|
|
except Paiement.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Entrée inexistante")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_paiement/")
|
|
|
|
paiement = PaiementForm(request.POST or None, instance=paiement_instance)
|
|
|
|
if paiement.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
paiement.save()
|
|
|
|
reversion.set_user(request.user)
|
2017-10-13 03:08:30 +00:00
|
|
|
reversion.set_comment(
|
|
|
|
"Champs modifié(s) : %s" % ', '.join(
|
2017-10-13 20:47:32 +00:00
|
|
|
field for field in paiement.changed_data
|
2017-10-13 03:08:30 +00:00
|
|
|
)
|
|
|
|
)
|
2016-07-08 01:27:02 +00:00
|
|
|
messages.success(request, "Type de paiement modifié")
|
|
|
|
return redirect("/cotisations/index_paiement/")
|
2016-07-06 19:43:39 +00:00
|
|
|
return form({'factureform': paiement}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-06 19:43:39 +00:00
|
|
|
def del_paiement(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Suppression d'un moyen de paiement"""
|
2016-07-06 19:43:39 +00:00
|
|
|
paiement = DelPaiementForm(request.POST or None)
|
|
|
|
if paiement.is_valid():
|
|
|
|
paiement_dels = paiement.cleaned_data['paiements']
|
|
|
|
for paiement_del in paiement_dels:
|
|
|
|
try:
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
paiement_del.delete()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Destruction")
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.success(
|
2017-10-13 20:47:32 +00:00
|
|
|
request,
|
|
|
|
"Le moyen de paiement a été supprimé"
|
2017-10-13 03:08:30 +00:00
|
|
|
)
|
2016-07-06 19:43:39 +00:00
|
|
|
except ProtectedError:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(
|
|
|
|
request,
|
|
|
|
"Le moyen de paiement %s est affecté à au moins une\
|
|
|
|
facture, vous ne pouvez pas le supprimer" % paiement_del
|
|
|
|
)
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_paiement/")
|
2016-07-06 19:43:39 +00:00
|
|
|
return form({'factureform': paiement}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-06 20:20:49 +00:00
|
|
|
def add_banque(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Ajoute une banque à la liste des banques"""
|
2016-07-06 20:20:49 +00:00
|
|
|
banque = BanqueForm(request.POST or None)
|
|
|
|
if banque.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
banque.save()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Création")
|
2016-07-06 20:20:49 +00:00
|
|
|
messages.success(request, "La banque a été ajoutée")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_banque/")
|
|
|
|
return form({'factureform': banque}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-08 01:27:02 +00:00
|
|
|
def edit_banque(request, banqueid):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Edite le nom d'une banque"""
|
2016-07-08 01:27:02 +00:00
|
|
|
try:
|
2016-07-17 19:20:43 +00:00
|
|
|
banque_instance = Banque.objects.get(pk=banqueid)
|
2016-07-08 01:27:02 +00:00
|
|
|
except Banque.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, u"Entrée inexistante")
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_banque/")
|
|
|
|
banque = BanqueForm(request.POST or None, instance=banque_instance)
|
|
|
|
if banque.is_valid():
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
banque.save()
|
|
|
|
reversion.set_user(request.user)
|
2017-10-13 03:08:30 +00:00
|
|
|
reversion.set_comment(
|
|
|
|
"Champs modifié(s) : %s" % ', '.join(
|
|
|
|
field for field in banque.changed_data
|
|
|
|
)
|
|
|
|
)
|
2016-07-08 01:27:02 +00:00
|
|
|
messages.success(request, "Banque modifiée")
|
|
|
|
return redirect("/cotisations/index_banque/")
|
2016-07-06 20:20:49 +00:00
|
|
|
return form({'factureform': banque}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-06 20:20:49 +00:00
|
|
|
def del_banque(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Supprime une banque"""
|
2016-07-06 20:20:49 +00:00
|
|
|
banque = DelBanqueForm(request.POST or None)
|
|
|
|
if banque.is_valid():
|
|
|
|
banque_dels = banque.cleaned_data['banques']
|
|
|
|
for banque_del in banque_dels:
|
|
|
|
try:
|
2016-07-21 14:58:12 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
banque_del.delete()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Destruction")
|
2016-07-06 20:20:49 +00:00
|
|
|
messages.success(request, "La banque a été supprimée")
|
|
|
|
except ProtectedError:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "La banque %s est affectée à au moins\
|
|
|
|
une facture, vous ne pouvez pas la supprimer" % banque_del)
|
2016-07-08 01:27:02 +00:00
|
|
|
return redirect("/cotisations/index_banque/")
|
2016-07-06 20:20:49 +00:00
|
|
|
return form({'factureform': banque}, 'cotisations/facture.html', request)
|
|
|
|
|
2017-10-13 03:08:30 +00:00
|
|
|
|
2016-07-17 18:08:56 +00:00
|
|
|
@login_required
|
2017-10-03 03:51:14 +00:00
|
|
|
@permission_required('tresorier')
|
2016-07-17 18:08:56 +00:00
|
|
|
def control(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Pour le trésorier, vue pour controler en masse les
|
|
|
|
factures.Case à cocher, pratique"""
|
2017-10-13 20:47:32 +00:00
|
|
|
options, _created = GeneralOption.objects.get_or_create()
|
2017-06-25 23:29:34 +00:00
|
|
|
pagination_number = options.pagination_number
|
2016-07-17 18:08:56 +00:00
|
|
|
facture_list = Facture.objects.order_by('date').reverse()
|
2017-10-13 03:08:30 +00:00
|
|
|
controlform_set = modelformset_factory(
|
2017-10-13 20:47:32 +00:00
|
|
|
Facture,
|
|
|
|
fields=('control', 'valid'),
|
|
|
|
extra=0
|
|
|
|
)
|
2017-06-25 23:29:34 +00:00
|
|
|
paginator = Paginator(facture_list, pagination_number)
|
2016-11-18 10:31:33 +00:00
|
|
|
page = request.GET.get('page')
|
|
|
|
try:
|
|
|
|
facture_list = paginator.page(page)
|
|
|
|
except PageNotAnInteger:
|
|
|
|
facture_list = paginator.page(1)
|
|
|
|
except EmptyPage:
|
|
|
|
facture_list = paginator.page(paginator.num.pages)
|
2017-10-13 03:08:30 +00:00
|
|
|
page_query = Facture.objects.order_by('date').reverse().filter(
|
2017-10-13 20:47:32 +00:00
|
|
|
id__in=[facture.id for facture in facture_list]
|
|
|
|
)
|
2016-11-18 10:31:33 +00:00
|
|
|
controlform = controlform_set(request.POST or None, queryset=page_query)
|
2016-07-17 18:08:56 +00:00
|
|
|
if controlform.is_valid():
|
2016-12-14 02:08:57 +00:00
|
|
|
with transaction.atomic(), reversion.create_revision():
|
|
|
|
controlform.save()
|
|
|
|
reversion.set_user(request.user)
|
|
|
|
reversion.set_comment("Controle trésorier")
|
2016-07-17 18:08:56 +00:00
|
|
|
return redirect("/cotisations/control/")
|
2017-10-13 03:08:30 +00:00
|
|
|
return render(request, 'cotisations/control.html', {
|
|
|
|
'facture_list': facture_list,
|
|
|
|
'controlform': controlform
|
|
|
|
})
|
|
|
|
|
2016-07-17 18:08:56 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-08 01:27:02 +00:00
|
|
|
def index_article(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Affiche l'ensemble des articles en vente"""
|
2016-07-08 01:27:02 +00:00
|
|
|
article_list = Article.objects.order_by('name')
|
2017-10-13 03:08:30 +00:00
|
|
|
return render(request, 'cotisations/index_article.html', {
|
|
|
|
'article_list': article_list
|
|
|
|
})
|
|
|
|
|
2016-07-08 01:27:02 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-08 01:27:02 +00:00
|
|
|
def index_paiement(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Affiche l'ensemble des moyens de paiement en vente"""
|
2016-07-08 01:27:02 +00:00
|
|
|
paiement_list = Paiement.objects.order_by('moyen')
|
2017-10-13 03:08:30 +00:00
|
|
|
return render(request, 'cotisations/index_paiement.html', {
|
|
|
|
'paiement_list': paiement_list
|
|
|
|
})
|
|
|
|
|
2016-07-08 01:27:02 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-08 01:27:02 +00:00
|
|
|
def index_banque(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Affiche l'ensemble des banques"""
|
2016-07-08 01:27:02 +00:00
|
|
|
banque_list = Banque.objects.order_by('name')
|
2017-10-13 03:08:30 +00:00
|
|
|
return render(request, 'cotisations/index_banque.html', {
|
|
|
|
'banque_list': banque_list
|
|
|
|
})
|
|
|
|
|
2016-07-08 01:27:02 +00:00
|
|
|
|
2016-07-08 10:35:53 +00:00
|
|
|
@login_required
|
2016-07-09 15:16:44 +00:00
|
|
|
@permission_required('cableur')
|
2016-07-02 13:58:50 +00:00
|
|
|
def index(request):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Affiche l'ensemble des factures, pour les cableurs et +"""
|
2017-10-13 20:47:32 +00:00
|
|
|
options, _created = GeneralOption.objects.get_or_create()
|
2017-06-25 23:29:34 +00:00
|
|
|
pagination_number = options.pagination_number
|
2017-10-13 03:08:30 +00:00
|
|
|
facture_list = Facture.objects.order_by('date').select_related('user')\
|
|
|
|
.select_related('paiement').prefetch_related('vente_set').reverse()
|
2017-06-25 23:29:34 +00:00
|
|
|
paginator = Paginator(facture_list, pagination_number)
|
2016-07-22 01:25:04 +00:00
|
|
|
page = request.GET.get('page')
|
|
|
|
try:
|
|
|
|
facture_list = paginator.page(page)
|
|
|
|
except PageNotAnInteger:
|
|
|
|
# If page is not an integer, deliver first page.
|
|
|
|
facture_list = paginator.page(1)
|
|
|
|
except EmptyPage:
|
|
|
|
# If page is out of range (e.g. 9999), deliver last page of results.
|
|
|
|
facture_list = paginator.page(paginator.num_pages)
|
2017-10-13 03:08:30 +00:00
|
|
|
return render(request, 'cotisations/index.html', {
|
|
|
|
'facture_list': facture_list
|
|
|
|
})
|
|
|
|
|
2016-07-21 14:58:12 +00:00
|
|
|
|
|
|
|
@login_required
|
2017-10-13 20:47:32 +00:00
|
|
|
def history(request, object, object_id):
|
2017-10-13 03:08:30 +00:00
|
|
|
"""Affiche l'historique de chaque objet"""
|
2016-07-21 14:58:12 +00:00
|
|
|
if object == 'facture':
|
|
|
|
try:
|
2017-10-13 20:47:32 +00:00
|
|
|
object_instance = Facture.objects.get(pk=object_id)
|
2016-07-21 14:58:12 +00:00
|
|
|
except Facture.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Facture inexistante")
|
|
|
|
return redirect("/cotisations/")
|
|
|
|
if not request.user.has_perms(('cableur',))\
|
|
|
|
and object_instance.user != request.user:
|
|
|
|
messages.error(request, "Vous ne pouvez pas afficher l'historique\
|
|
|
|
d'une facture d'un autre user que vous sans droit cableur")
|
|
|
|
return redirect("/users/profil/" + str(request.user.id))
|
2016-07-21 14:58:12 +00:00
|
|
|
elif object == 'paiement' and request.user.has_perms(('cableur',)):
|
|
|
|
try:
|
2017-10-13 20:47:32 +00:00
|
|
|
object_instance = Paiement.objects.get(pk=object_id)
|
2016-07-21 14:58:12 +00:00
|
|
|
except Paiement.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Paiement inexistant")
|
|
|
|
return redirect("/cotisations/")
|
2016-07-21 14:58:12 +00:00
|
|
|
elif object == 'article' and request.user.has_perms(('cableur',)):
|
|
|
|
try:
|
2017-10-13 20:47:32 +00:00
|
|
|
object_instance = Article.objects.get(pk=object_id)
|
2016-07-21 14:58:12 +00:00
|
|
|
except Article.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Article inexistante")
|
|
|
|
return redirect("/cotisations/")
|
2016-07-21 14:58:12 +00:00
|
|
|
elif object == 'banque' and request.user.has_perms(('cableur',)):
|
|
|
|
try:
|
2017-10-13 20:47:32 +00:00
|
|
|
object_instance = Banque.objects.get(pk=object_id)
|
2016-07-21 14:58:12 +00:00
|
|
|
except Banque.DoesNotExist:
|
2017-10-13 03:08:30 +00:00
|
|
|
messages.error(request, "Banque inexistante")
|
|
|
|
return redirect("/cotisations/")
|
2016-07-21 14:58:12 +00:00
|
|
|
else:
|
|
|
|
messages.error(request, "Objet inconnu")
|
|
|
|
return redirect("/cotisations/")
|
2017-10-13 20:47:32 +00:00
|
|
|
options, _created = GeneralOption.objects.get_or_create()
|
2017-06-25 23:29:34 +00:00
|
|
|
pagination_number = options.pagination_number
|
2016-12-11 16:24:04 +00:00
|
|
|
reversions = Version.objects.get_for_object(object_instance)
|
2017-06-25 23:29:34 +00:00
|
|
|
paginator = Paginator(reversions, pagination_number)
|
2016-07-27 01:36:28 +00:00
|
|
|
page = request.GET.get('page')
|
|
|
|
try:
|
|
|
|
reversions = paginator.page(page)
|
|
|
|
except PageNotAnInteger:
|
|
|
|
# If page is not an integer, deliver first page.
|
|
|
|
reversions = paginator.page(1)
|
|
|
|
except EmptyPage:
|
|
|
|
# If page is out of range (e.g. 9999), deliver last page of results.
|
|
|
|
reversions = paginator.page(paginator.num_pages)
|
2017-10-13 03:08:30 +00:00
|
|
|
return render(request, 're2o/history.html', {
|
|
|
|
'reversions': reversions,
|
|
|
|
'object': object_instance
|
|
|
|
})
|