2021-04-29 00:27:33 +02:00
|
|
|
import io
|
2015-03-17 19:03:51 +01:00
|
|
|
import os
|
2020-09-10 15:06:53 +02:00
|
|
|
import zipfile
|
2021-04-29 00:27:33 +02:00
|
|
|
|
2021-04-29 00:33:58 +02:00
|
|
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
2021-04-29 00:27:33 +02:00
|
|
|
from django.core.files import File
|
|
|
|
from django.db.models import Q
|
|
|
|
from django.http import Http404
|
|
|
|
from django.shortcuts import HttpResponse, get_object_or_404, redirect, render
|
2022-01-09 01:18:50 +01:00
|
|
|
from django.urls import reverse_lazy
|
2021-04-29 00:27:33 +02:00
|
|
|
from django.utils.safestring import mark_safe
|
|
|
|
from django.utils.text import slugify
|
2020-09-10 15:06:53 +02:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2022-01-09 01:18:50 +01:00
|
|
|
from django.views.generic import (CreateView, DeleteView, ListView,
|
|
|
|
TemplateView, UpdateView)
|
2015-03-29 17:50:31 +02:00
|
|
|
|
2021-04-29 00:33:58 +02:00
|
|
|
from gestion.mixins import ChefRequiredMixin
|
2021-04-29 00:27:33 +02:00
|
|
|
from gestion.models import Photo
|
|
|
|
from partitions.forms import UploadFileForm, UploadMorceauForm
|
2022-01-06 00:27:53 +01:00
|
|
|
from partitions.models import Category, Partition, PartitionSet, SetList
|
2021-04-29 00:27:33 +02:00
|
|
|
|
|
|
|
from .forms import ChefEditMorceauForm
|
2020-09-10 15:06:53 +02:00
|
|
|
|
2015-03-17 19:03:51 +01:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
def download_musecores(request):
|
2016-07-14 00:13:30 +02:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
p = Partition.objects.filter(
|
|
|
|
Q(part__contains=".mscz")
|
|
|
|
& Q(
|
|
|
|
Q(morceau__category__name="Partitions actives")
|
|
|
|
| Q(morceau__category__name="Partitions optionnelles")
|
|
|
|
)
|
|
|
|
)
|
2020-09-10 15:06:53 +02:00
|
|
|
|
|
|
|
zip_subdir = "Ernestophone_musescores"
|
|
|
|
zip_filename = "%s.zip" % zip_subdir
|
|
|
|
|
|
|
|
# Open StringIO to grab in-memory ZIP contents
|
|
|
|
s = io.BytesIO()
|
|
|
|
|
|
|
|
# The zip compressor
|
|
|
|
zf = zipfile.ZipFile(s, "w")
|
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
for part in p:
|
2020-09-10 15:06:53 +02:00
|
|
|
fpath = part.part.path
|
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
typ = ".mscz"
|
2020-09-10 15:06:53 +02:00
|
|
|
# Calculate path for file in zip
|
|
|
|
fdir, fname = os.path.split(fpath)
|
2021-04-29 00:27:33 +02:00
|
|
|
zip_path = os.path.join(
|
|
|
|
zip_subdir,
|
|
|
|
"%s_%s_%s.%s"
|
|
|
|
% (
|
|
|
|
slugify(part.morceau.nom),
|
|
|
|
slugify(part.morceau.auteur),
|
|
|
|
slugify(part.nom),
|
|
|
|
typ,
|
|
|
|
),
|
|
|
|
)
|
2020-09-10 15:06:53 +02:00
|
|
|
|
|
|
|
# Add file, at correct path
|
|
|
|
zf.write(fpath, zip_path)
|
|
|
|
|
|
|
|
# Must close zip for all contents to be written
|
|
|
|
zf.close()
|
|
|
|
# Grab ZIP file from in-memory, make response with correct MIME-type
|
|
|
|
resp = HttpResponse(s.getvalue())
|
|
|
|
# ..and correct content-disposition
|
2021-04-29 00:27:33 +02:00
|
|
|
resp["Content-Disposition"] = "attachment; filename=%s" % zip_filename
|
2020-09-10 15:06:53 +02:00
|
|
|
|
|
|
|
return resp
|
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
|
2022-01-09 01:18:50 +01:00
|
|
|
from datetime import date, timedelta
|
|
|
|
|
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
class Repertoire(TemplateView):
|
|
|
|
template_name = "partitions/repertoire.html"
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super().get_context_data(**kwargs)
|
2022-01-09 01:18:50 +01:00
|
|
|
context["setlists"] = (
|
|
|
|
SetList.objects.filter(
|
|
|
|
is_current="y", date__gt=(date.today() - timedelta(days=7))
|
|
|
|
)
|
|
|
|
.order_by("date")
|
|
|
|
.prefetch_related("morceaux")
|
|
|
|
)
|
2021-04-29 00:33:58 +02:00
|
|
|
context["categories"] = Category.objects.prefetch_related(
|
|
|
|
"partitionset_set"
|
|
|
|
).order_by("order")
|
|
|
|
context["photo"] = Photo.objects.filter(cat="part").order_by("?").first()
|
2021-04-29 00:27:33 +02:00
|
|
|
return context
|
|
|
|
|
|
|
|
|
|
|
|
class Morceau(LoginRequiredMixin, TemplateView):
|
|
|
|
template_name = "partitions/part.html"
|
|
|
|
form_class = ChefEditMorceauForm
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(Morceau, self).get_context_data(**kwargs)
|
2021-04-29 00:33:58 +02:00
|
|
|
p = get_object_or_404(
|
|
|
|
PartitionSet, nom=self.kwargs["nom"], auteur=self.kwargs["auteur"]
|
|
|
|
)
|
2021-04-29 00:27:33 +02:00
|
|
|
part = p.partition_set.all().order_by("nom")
|
|
|
|
form = self.form_class(instance=p)
|
|
|
|
infos = mark_safe(p.infos)
|
|
|
|
infos_en = mark_safe(p.infos_en)
|
|
|
|
context["p"] = p
|
|
|
|
context["infos"] = infos
|
|
|
|
context["infos_en"] = infos_en
|
|
|
|
context["form"] = form
|
|
|
|
context["part"] = part
|
2021-04-29 00:33:58 +02:00
|
|
|
context["nom"] = self.kwargs["nom"]
|
|
|
|
context["auteur"] = self.kwargs["auteur"]
|
2021-04-29 00:27:33 +02:00
|
|
|
|
|
|
|
return context
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
2021-04-29 00:33:58 +02:00
|
|
|
p = get_object_or_404(
|
|
|
|
PartitionSet, nom=self.kwargs["nom"], auteur=self.kwargs["auteur"]
|
|
|
|
)
|
2021-04-29 00:27:33 +02:00
|
|
|
if request.user.profile.is_chef:
|
|
|
|
form = self.form_class(request.POST, instance=p)
|
|
|
|
if form.is_valid():
|
|
|
|
form.save()
|
|
|
|
context = self.get_context_data()
|
|
|
|
return render(request, self.template_name, context)
|
|
|
|
|
|
|
|
|
|
|
|
class Upload(ChefRequiredMixin, TemplateView):
|
|
|
|
form_class = UploadFileForm
|
|
|
|
template_name = "partitions/upload.html"
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(Upload, self).get_context_data(**kwargs)
|
|
|
|
form = self.form_class()
|
|
|
|
|
2021-04-29 00:33:58 +02:00
|
|
|
context["nom"] = self.kwargs["nom"]
|
|
|
|
context["auteur"] = self.kwargs["auteur"]
|
2021-04-29 00:27:33 +02:00
|
|
|
context["form"] = form
|
|
|
|
return context
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
2015-03-17 19:03:51 +01:00
|
|
|
form = UploadFileForm(request.POST, request.FILES)
|
2021-04-29 01:00:17 +02:00
|
|
|
error = False
|
|
|
|
sauvegarde = False
|
2015-03-17 19:03:51 +01:00
|
|
|
if form.is_valid():
|
|
|
|
partition = Partition()
|
2021-04-29 00:27:33 +02:00
|
|
|
partition.part = form.cleaned_data["file"]
|
|
|
|
partition.nom = form.cleaned_data["title"]
|
|
|
|
if "/" in partition.nom:
|
2021-04-29 01:00:17 +02:00
|
|
|
error = _("Le caractère / n'est pas autorisé dans le nom")
|
2021-04-29 00:27:33 +02:00
|
|
|
context = self.get_context_data()
|
2021-04-29 01:00:17 +02:00
|
|
|
context["error"] = error
|
2021-04-29 00:27:33 +02:00
|
|
|
return render(request, self.template_name, context)
|
2021-04-29 00:33:58 +02:00
|
|
|
mor = get_object_or_404(
|
|
|
|
PartitionSet, nom=self.kwargs["nom"], auteur=self.kwargs["auteur"]
|
|
|
|
)
|
2015-03-17 19:03:51 +01:00
|
|
|
partition.morceau = mor
|
|
|
|
try:
|
|
|
|
mor.partition_set.get(nom=partition.nom)
|
2021-04-29 01:00:17 +02:00
|
|
|
error = _("Un morceau du même nom existe déjà")
|
2015-03-17 19:03:51 +01:00
|
|
|
except Partition.DoesNotExist:
|
|
|
|
partition.save()
|
2021-04-29 01:00:17 +02:00
|
|
|
sauvegarde = True
|
2021-04-29 00:27:33 +02:00
|
|
|
|
|
|
|
context = self.get_context_data()
|
2021-04-29 00:33:58 +02:00
|
|
|
context["form"] = form
|
2021-04-29 01:00:17 +02:00
|
|
|
context["error"] = error
|
|
|
|
context["sauvegarde"] = sauvegarde
|
2021-04-29 00:27:33 +02:00
|
|
|
return render(request, self.template_name, context)
|
2016-07-14 00:13:30 +02:00
|
|
|
|
2021-03-31 16:05:26 +02:00
|
|
|
|
2016-06-25 15:14:00 +02:00
|
|
|
def see(request, nom, auteur, partition_id):
|
|
|
|
partition = get_object_or_404(Partition, id=partition_id)
|
|
|
|
_, extension = os.path.splitext(partition.part.path)
|
2021-03-31 16:05:26 +02:00
|
|
|
download_unlogged = partition.morceau.download_unlogged
|
2021-04-29 00:27:33 +02:00
|
|
|
if download_unlogged == "o" or request.user.is_authenticated:
|
2021-03-31 16:05:26 +02:00
|
|
|
if ".pdf" == extension:
|
2021-04-29 00:27:33 +02:00
|
|
|
with open(partition.part.path, "rb") as f:
|
2021-03-31 16:05:26 +02:00
|
|
|
myfile = File(f)
|
|
|
|
response = HttpResponse(content=myfile.read())
|
|
|
|
response["Content-Type"] = "application/pdf"
|
|
|
|
response["Content-Disposition"] = "inline; filename=%s_%s_%s.pdf" % (
|
2021-04-29 00:27:33 +02:00
|
|
|
slugify(nom),
|
|
|
|
slugify(auteur),
|
|
|
|
slugify(partition.nom),
|
|
|
|
)
|
2021-03-31 16:05:26 +02:00
|
|
|
return response
|
|
|
|
elif ".mp3" == extension:
|
2021-04-29 00:27:33 +02:00
|
|
|
with open(partition.part.path, "rb") as f:
|
2021-03-31 16:05:26 +02:00
|
|
|
myfile = File(f)
|
|
|
|
response = HttpResponse()
|
|
|
|
response.write(myfile.read())
|
|
|
|
response["Content-Type"] = "audio/mp3"
|
|
|
|
response["Content-Length"] = myfile.size
|
|
|
|
return response
|
|
|
|
else:
|
|
|
|
p = get_object_or_404(PartitionSet, nom=nom, auteur=auteur)
|
|
|
|
part = p.partition_set.all()
|
2021-04-29 00:27:33 +02:00
|
|
|
return render(
|
|
|
|
request,
|
|
|
|
"partitions/part.html",
|
|
|
|
{"p": p, "part": part, "nom": nom, "auteur": auteur},
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return redirect("login")
|
|
|
|
|
|
|
|
|
|
|
|
class DeletePart(ChefRequiredMixin, TemplateView):
|
|
|
|
model = PartitionSet
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
2021-04-29 00:33:58 +02:00
|
|
|
p = get_object_or_404(
|
|
|
|
self.model, nom=self.kwargs["nom"], auteur=self.kwargs["auteur"]
|
|
|
|
)
|
2021-04-29 00:27:33 +02:00
|
|
|
try:
|
2021-04-29 00:33:58 +02:00
|
|
|
part = p.partition_set.get(id=self.kwargs["id"])
|
2021-04-29 00:27:33 +02:00
|
|
|
except Partition.DoesNotExist:
|
|
|
|
raise Http404
|
|
|
|
part.delete()
|
2021-04-29 00:33:58 +02:00
|
|
|
return redirect(
|
|
|
|
"partitions:listepart", nom=self.kwargs["nom"], auteur=self.kwargs["auteur"]
|
|
|
|
)
|
2021-04-29 00:27:33 +02:00
|
|
|
|
|
|
|
|
|
|
|
class CreateMorc(ChefRequiredMixin, TemplateView):
|
|
|
|
form_class = UploadMorceauForm
|
|
|
|
template_name = "partitions/new.html"
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(CreateMorc, self).get_context_data(**kwargs)
|
2021-04-29 00:33:58 +02:00
|
|
|
context["form"] = self.form_class()
|
2021-04-29 00:27:33 +02:00
|
|
|
return context
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
|
|
form = self.form_class(request.POST)
|
|
|
|
sauvegarde = False
|
|
|
|
error = False
|
2015-03-17 19:03:51 +01:00
|
|
|
if form.is_valid():
|
|
|
|
partitionset = PartitionSet()
|
2021-04-29 00:27:33 +02:00
|
|
|
partitionset.nom = form.cleaned_data["titre"]
|
|
|
|
partitionset.auteur = form.cleaned_data["auteur"]
|
|
|
|
if "/" in partitionset.auteur or "/" in partitionset.nom:
|
2020-09-10 15:06:53 +02:00
|
|
|
error = _("Le caractère / n'est pas autorisé")
|
2021-04-29 00:27:33 +02:00
|
|
|
context = self.get_context_data()
|
2021-04-29 00:33:58 +02:00
|
|
|
context["error"] = error
|
2021-04-29 00:27:33 +02:00
|
|
|
return render(request, self.template_name, context)
|
2015-03-17 19:03:51 +01:00
|
|
|
try:
|
2021-04-29 00:27:33 +02:00
|
|
|
PartitionSet.objects.get(
|
|
|
|
nom=partitionset.nom, auteur=partitionset.auteur
|
|
|
|
)
|
2020-09-10 15:06:53 +02:00
|
|
|
error = _("Un morceau du même nom existe déjà")
|
2015-03-17 19:03:51 +01:00
|
|
|
except PartitionSet.DoesNotExist:
|
2018-01-04 22:11:27 +01:00
|
|
|
# XXX. Hideous
|
|
|
|
cat = Category.objects.first()
|
|
|
|
try:
|
|
|
|
cat = Category.objects.get(name="Partitions à venir")
|
|
|
|
except Category.DoesNotExist:
|
|
|
|
pass
|
|
|
|
partitionset.category = cat
|
2015-03-17 19:03:51 +01:00
|
|
|
partitionset.save()
|
|
|
|
sauvegarde = True
|
2021-04-29 00:27:33 +02:00
|
|
|
return redirect("partitions:liste")
|
|
|
|
context = self.get_context_data()
|
2021-04-29 00:33:58 +02:00
|
|
|
context["sauvegarde"] = sauvegarde
|
|
|
|
context["error"] = error
|
|
|
|
context["form"] = form
|
2021-04-29 00:27:33 +02:00
|
|
|
return render(request, self.template_name, context)
|
2015-03-17 19:03:51 +01:00
|
|
|
|
2016-07-14 00:13:30 +02:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
class ConfDelete(ChefRequiredMixin, TemplateView):
|
|
|
|
template_name = "partitions/conf_delete.html"
|
2015-03-17 19:03:51 +01:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super().get_context_data(**kwargs)
|
2021-04-29 00:33:58 +02:00
|
|
|
context["nom"] = self.kwargs.get("nom")
|
|
|
|
context["auteur"] = self.kwargs.get("auteur")
|
|
|
|
context["id"] = self.kwargs.get("id")
|
2021-04-29 00:27:33 +02:00
|
|
|
return context
|
2016-07-14 00:13:30 +02:00
|
|
|
|
2015-03-17 19:03:51 +01:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
class DeleteMorc(ChefRequiredMixin, TemplateView):
|
|
|
|
model = PartitionSet
|
2015-03-17 19:03:51 +01:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
def get(self, request, *args, **kwargs):
|
2021-04-29 00:33:58 +02:00
|
|
|
p = get_object_or_404(
|
|
|
|
self.model, nom=self.kwargs["nom"], auteur=self.kwargs["auteur"]
|
|
|
|
)
|
2021-04-29 00:27:33 +02:00
|
|
|
part = p.partition_set.all()
|
|
|
|
for pa in part:
|
|
|
|
pa.delete()
|
|
|
|
p.delete()
|
|
|
|
return redirect("partitions:liste")
|
2016-07-14 00:13:30 +02:00
|
|
|
|
2015-03-17 19:03:51 +01:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
class ConfDeleteMorc(ChefRequiredMixin, TemplateView):
|
|
|
|
template_name = "partitions/conf_delete_morc.html"
|
2016-07-14 00:13:30 +02:00
|
|
|
|
2021-04-29 00:27:33 +02:00
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super().get_context_data(**kwargs)
|
2021-04-29 00:33:58 +02:00
|
|
|
context["nom"] = self.kwargs.get("nom")
|
|
|
|
context["auteur"] = self.kwargs.get("auteur")
|
2021-04-29 00:27:33 +02:00
|
|
|
return context
|
2020-09-10 15:06:53 +02:00
|
|
|
|
2021-03-31 16:05:26 +02:00
|
|
|
|
2020-09-10 15:06:53 +02:00
|
|
|
def download(request, nom, auteur, partition_id):
|
2021-03-31 16:05:26 +02:00
|
|
|
|
2020-09-10 15:06:53 +02:00
|
|
|
partition = get_object_or_404(Partition, id=partition_id)
|
2021-03-31 16:05:26 +02:00
|
|
|
download_unlogged = partition.morceau.download_unlogged
|
2021-04-29 00:27:33 +02:00
|
|
|
if download_unlogged == "o" or request.user.is_authenticated:
|
|
|
|
with open(partition.part.path, "rb") as f:
|
2021-03-31 16:05:26 +02:00
|
|
|
myfile = File(f)
|
|
|
|
response = HttpResponse(content=myfile.read())
|
|
|
|
typ = os.path.splitext(myfile.name)[1][1:]
|
2021-04-29 00:27:33 +02:00
|
|
|
response["Content-Type"] = "application/%s" % (typ,)
|
|
|
|
response["Content-Disposition"] = "attachment; filename=%s_%s_%s.%s" % (
|
|
|
|
slugify(nom),
|
|
|
|
slugify(auteur),
|
|
|
|
slugify(partition.nom),
|
|
|
|
typ,
|
|
|
|
)
|
2021-03-31 16:05:26 +02:00
|
|
|
return response
|
2021-04-29 00:27:33 +02:00
|
|
|
else:
|
|
|
|
return redirect("login")
|
2022-01-09 01:18:50 +01:00
|
|
|
|
|
|
|
|
|
|
|
class SetListListView(ChefRequiredMixin, ListView):
|
|
|
|
model = SetList
|
|
|
|
|
|
|
|
def get_queryset(self):
|
|
|
|
return SetList.objects.all().order_by("-date")
|
|
|
|
|
|
|
|
|
|
|
|
class SetListCreate(ChefRequiredMixin, CreateView):
|
|
|
|
model = SetList
|
|
|
|
fields = ["date", "morceaux", "is_current"]
|
|
|
|
success_url = reverse_lazy("partitions:list_setlist")
|
|
|
|
|
|
|
|
|
|
|
|
class SetListUpdate(ChefRequiredMixin, UpdateView):
|
|
|
|
model = SetList
|
|
|
|
fields = ["date", "morceaux", "is_current"]
|
|
|
|
success_url = reverse_lazy("partitions:list_setlist")
|
|
|
|
|
|
|
|
|
|
|
|
class SetListDelete(ChefRequiredMixin, DeleteView):
|
|
|
|
model = SetList
|
|
|
|
success_url = reverse_lazy("partitions:list_setlist")
|