375 lines
12 KiB
Python
375 lines
12 KiB
Python
|
|
from django.contrib.auth.decorators import login_required
|
|
from django.db.models import Q, F, Avg
|
|
from django.db.models.functions import TruncDay
|
|
from django.http import HttpResponseRedirect, JsonResponse
|
|
from django.shortcuts import render, get_object_or_404
|
|
from django.views.decorators.http import require_http_methods
|
|
|
|
import pendulum
|
|
|
|
from ultron.followup.models import Event
|
|
from ultron.followup.forms import GymnastHasRoutineForm
|
|
from ultron.followup.models import (
|
|
Chrono,
|
|
LearnedSkill,
|
|
MindState,
|
|
Skill,
|
|
Point,
|
|
Accident,
|
|
HeightWeight,
|
|
NumberOfRoutineDone,
|
|
)
|
|
|
|
from .models import Gymnast
|
|
from .forms import GymnastForm
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_lookup(request):
|
|
"""
|
|
Récupère la liste des gymnastes à la volée suivant des caractères de
|
|
recherche entrés. (min 3 caractères)
|
|
"""
|
|
|
|
results = []
|
|
pattern = request.GET.get("pattern", None)
|
|
|
|
# Ignore queries shorter than length 3
|
|
if pattern is not None and len(pattern) > 3:
|
|
model_results = Gymnast.objects.filter(
|
|
Q(last_name__icontains=pattern) | Q(first_name__icontains=pattern)
|
|
)
|
|
results = [{"ID": x.id, "Name": str(x)} for x in model_results]
|
|
|
|
# json = simplejson.dumps(results)
|
|
# return HttpResponse(json, content_type="application/json")
|
|
return JsonResponse(results, safe=False)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_listing(request):
|
|
"""
|
|
Liste tous les gymnasts connus
|
|
"""
|
|
gymnast_list = Gymnast.objects.all()
|
|
context = {"gymnast_list": gymnast_list}
|
|
return render(request, "people/gymnasts/list.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_details(request, gymnast_id, tab=None):
|
|
"""
|
|
Récupère toutes les informations d'un gymnaste.
|
|
"""
|
|
|
|
gymnast = get_object_or_404(Gymnast, pk=gymnast_id)
|
|
|
|
gymnast_nb_known_skills = gymnast.known_skills.distinct(
|
|
"skill"
|
|
).count() # devrait disparaitre
|
|
|
|
nb_skill = Skill.objects.all().count()
|
|
nb_known_skill = (
|
|
LearnedSkill.objects.filter(gymnast=gymnast_id).distinct("skill").count()
|
|
)
|
|
|
|
if nb_skill != 0:
|
|
percentage_known_skill = (nb_known_skill / nb_skill) * 100
|
|
else:
|
|
percentage_known_skill = 0
|
|
|
|
chronos_list = Chrono.objects.filter(gymnast=gymnast_id).order_by("-date")[:10]
|
|
straightjump_score = (
|
|
Chrono.objects.filter(gymnast=gymnast_id)
|
|
.filter(chrono_type=0)
|
|
.order_by("-date")
|
|
)
|
|
best_straightjump = (
|
|
Chrono.objects.filter(gymnast=gymnast_id)
|
|
.filter(chrono_type=0)
|
|
.order_by("-score")[:1]
|
|
)
|
|
best_routine = (
|
|
Chrono.objects.filter(gymnast=gymnast_id)
|
|
.filter(chrono_type=1)
|
|
.order_by("-score")[:1]
|
|
)
|
|
|
|
# nb_unknown_skill = nb_skill - gymnast_nb_known_skills
|
|
|
|
context = {
|
|
"gymnast": gymnast,
|
|
"gymnast_nb_known_skills": gymnast_nb_known_skills,
|
|
"chronos_list": chronos_list,
|
|
"straightjump_score": straightjump_score,
|
|
"best_routine": best_routine,
|
|
"best_straightjump": best_straightjump,
|
|
"nb_skill": nb_skill,
|
|
"nb_known_skill": nb_known_skill,
|
|
"percentage_known_skill": percentage_known_skill,
|
|
"tab": tab,
|
|
}
|
|
|
|
return render(request, "people/gymnasts/details.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_event(request, gymnast_id):
|
|
"""
|
|
Renvoie deux listes d'évènements : ceux à venir et ceux passés.
|
|
"""
|
|
today = pendulum.now().date()
|
|
next_event_list = Event.objects.filter(gymnasts=gymnast_id, datebegin__gte=today)
|
|
previous_event_list = Event.objects.filter(
|
|
gymnasts=gymnast_id, datebegin__lte=today
|
|
)
|
|
|
|
context = {
|
|
"next_event_list": next_event_list,
|
|
"previous_event_list": previous_event_list,
|
|
}
|
|
return render(request, "people/gymnasts/list_event.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_accident(request, gymnast_id):
|
|
"""
|
|
Renvoie deux listes d'évènements : ceux à venir et ceux passés.
|
|
"""
|
|
accident_list = Accident.objects.filter(gymnast=gymnast_id)
|
|
|
|
context = {"accident_list": accident_list, "gymnast_id": gymnast_id}
|
|
|
|
return render(request, "people/gymnasts/list_accident.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_physiological(request, gymnast_id):
|
|
"""
|
|
Renvoie les listes des tailles/poids, état d'esprit et accidents.
|
|
"""
|
|
accident_list = Accident.objects.filter(gymnast=gymnast_id).order_by("date")
|
|
mindstate_list = MindState.objects.filter(gymnast=gymnast_id).order_by("date")
|
|
height_weight_list = HeightWeight.objects.filter(gymnast=gymnast_id).order_by(
|
|
"date"
|
|
)
|
|
|
|
context = {
|
|
"accident_list": accident_list,
|
|
"mindstate_list": mindstate_list,
|
|
"height_weight_list": height_weight_list,
|
|
"gymnast_id": gymnast_id,
|
|
}
|
|
|
|
return render(request, "people/gymnasts/physiological_followup.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_scores_chrono(request, gymnast_id):
|
|
"""
|
|
Selectionne tous les scores réalisés par le gymnaste
|
|
"""
|
|
score_list = Point.objects.filter(gymnast=gymnast_id).order_by("-event__datebegin")
|
|
score_routine1_list = score_list.filter(routine_type=0)
|
|
score_routine2_list = score_list.filter(routine_type=1)
|
|
score_routine3_list = score_list.filter(routine_type=2)
|
|
|
|
chrono_list = Chrono.objects.filter(gymnast=gymnast_id).order_by("date")
|
|
chrono_10c = Chrono.objects \
|
|
.filter(gymnast=gymnast_id) \
|
|
.filter(chrono_type=0) \
|
|
.values('date') \
|
|
.annotate(score_avg=Avg('score')) \
|
|
.order_by('date')
|
|
chrono_r1 = Chrono.objects \
|
|
.filter(gymnast=gymnast_id) \
|
|
.filter(chrono_type=1) \
|
|
.values('date') \
|
|
.annotate(score_avg=Avg('score')) \
|
|
.order_by('date')
|
|
chrono_r2 = Chrono.objects \
|
|
.filter(gymnast=gymnast_id) \
|
|
.filter(chrono_type=2) \
|
|
.values('date') \
|
|
.annotate(score_avg=Avg('score')) \
|
|
.order_by('date')
|
|
chrono_rf = Chrono.objects \
|
|
.filter(gymnast=gymnast_id) \
|
|
.filter(chrono_type=3) \
|
|
.values('date') \
|
|
.annotate(score_avg=Avg('score')) \
|
|
.order_by('date')
|
|
|
|
context = {
|
|
"score_list": score_list,
|
|
"score_routine1_list": score_routine1_list,
|
|
"score_routine2_list": score_routine2_list,
|
|
"score_routine3_list": score_routine3_list,
|
|
"chrono_list": chrono_list,
|
|
"chrono_10c": chrono_10c,
|
|
"chrono_r1": chrono_r1,
|
|
"chrono_r2": chrono_r2,
|
|
"chrono_rf": chrono_rf,
|
|
"gymnast_id": gymnast_id,
|
|
}
|
|
|
|
return render(request, "people/gymnasts/list_scores_chronos.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_mindstate(request, gymnast_id):
|
|
"""
|
|
Selectionne tous les scores réalisés par le gymnaste
|
|
"""
|
|
mindstate_list = MindState.objects.filter(gymnast=gymnast_id).order_by("-date")
|
|
|
|
context = {
|
|
"mindstate_list": mindstate_list,
|
|
"gymnast_id": gymnast_id,
|
|
}
|
|
|
|
return render(request, "people/gymnasts/list_mindstate.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_routines(request, gymnast_id):
|
|
"""
|
|
Tag affichant les séries d'un gymnaste.
|
|
"""
|
|
gymnast = get_object_or_404(Gymnast, pk=gymnast_id)
|
|
ghr_list = gymnast.has_routine.prefetch_related("routine").filter(
|
|
dateend__isnull=True
|
|
)
|
|
routine_done_list = NumberOfRoutineDone.objects.filter(gymnast=gymnast_id).order_by(
|
|
"-date"
|
|
)[:8]
|
|
|
|
context = {
|
|
"ghr_list": ghr_list,
|
|
"routine_done_list": routine_done_list,
|
|
"gymnast_id": gymnast_id,
|
|
}
|
|
return render(request, "people/gymnasts/list_routine.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET", "POST"])
|
|
def link_routine_to_gymnast(request, gymnast_id=None):
|
|
""" """
|
|
|
|
if gymnast_id:
|
|
gymnast = get_object_or_404(Gymnast, pk=gymnast_id)
|
|
data = {"gymnast": gymnast_id, "gymnast_related": str(gymnast), "dateend": None}
|
|
else:
|
|
gymnast = None
|
|
data = {"dateend": None}
|
|
|
|
if request.method == "POST":
|
|
form = GymnastHasRoutineForm(request.POST)
|
|
|
|
if form.is_valid():
|
|
form.save()
|
|
if gymnast_id is not None:
|
|
return HttpResponseRedirect("/gymnast/details/" + str(gymnast_id) + "/")
|
|
|
|
return HttpResponseRedirect("/gymnast/")
|
|
else:
|
|
form = GymnastHasRoutineForm(instance=gymnast, initial=data)
|
|
|
|
context = {"form": form, "gymnast_id": gymnast_id}
|
|
return render(request, "people/gymnasts/link_to_routine.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET", "POST"])
|
|
def gymnast_create_or_update(request, gymnast_id=None):
|
|
"""
|
|
Formulaire de creation et modification d'un gymnaste.
|
|
"""
|
|
|
|
if gymnast_id:
|
|
gymnast = get_object_or_404(Gymnast, pk=gymnast_id)
|
|
data = {"club_related": gymnast.club}
|
|
else:
|
|
gymnast = None
|
|
data = {}
|
|
|
|
if request.method == "POST":
|
|
gymnast_form = GymnastForm(request.POST, instance=gymnast)
|
|
|
|
if gymnast_form.is_valid():
|
|
gymnast = gymnast_form.save()
|
|
return HttpResponseRedirect("/gymnast/details/" + str(gymnast.id))
|
|
|
|
print(gymnast_form.errors)
|
|
|
|
form = GymnastForm(instance=gymnast, initial=data)
|
|
|
|
context = {"form": form, "gymnast_id": gymnast_id}
|
|
return render(request, "people/gymnasts/create.html", context)
|
|
|
|
|
|
@login_required
|
|
@require_http_methods(["GET"])
|
|
def gymnast_display_skill(request, gymnast_id):
|
|
"""Tag affichant les statistiques de skill d'un gymnaste
|
|
|
|
Le nombre de saut qu'il sait faire (total, par niveau, par rank, …), calcule la complétude, …
|
|
|
|
.. todo:: Générer UNE fois la liste de skill que le gymnaste ne sait pas faire (1 query)
|
|
et les counts puis, dans le template on parcourt plusieurs fois cette même liste mais on
|
|
affiche conditionnellement (par age, par rank, ...)
|
|
"""
|
|
|
|
context = {}
|
|
gymnast = get_object_or_404(Gymnast, pk=gymnast_id)
|
|
gymnast_nb_known_skills = gymnast.known_skills.distinct("skill").count()
|
|
context = gymnast.get_informations_from_level()
|
|
context.update(gymnast.get_informations_from_rank())
|
|
|
|
planified_skill = (
|
|
Skill.objects.filter(plan__gymnast=gymnast.id)
|
|
.order_by("-plan__date")
|
|
.annotate(plan_date=F("plan__date"))
|
|
)
|
|
|
|
context["planified_skill"] = planified_skill
|
|
|
|
if gymnast.gender:
|
|
context["skill_by_age"] = Skill.objects.filter(
|
|
age_girl_masterised__lte=gymnast.age
|
|
).exclude(known_by__gymnast=gymnast.id)
|
|
else:
|
|
context["skill_by_age"] = Skill.objects.filter(
|
|
age_boy_masterised__lte=gymnast.age
|
|
).exclude(known_by__gymnast=gymnast.id)
|
|
|
|
skill_whith_help = (
|
|
Skill.objects.filter(known_by__gymnast=gymnast_id, known_by__cando=1)
|
|
.exclude(known_by__gymnast=gymnast.id, known_by__cando__gte=2)
|
|
.distinct()
|
|
)
|
|
|
|
skill_not_chained = (
|
|
Skill.objects.filter(known_by__gymnast=gymnast_id, known_by__cando=2)
|
|
.exclude(known_by__gymnast=gymnast.id, known_by__cando=3)
|
|
.distinct()
|
|
)
|
|
|
|
context["gymnast"] = gymnast
|
|
context["skill_whith_help"] = skill_whith_help
|
|
context["skill_not_chained"] = skill_not_chained
|
|
context["gymnast_nb_known_skills"] = gymnast_nb_known_skills
|
|
|
|
return render(request, "people/gymnasts/list_skill.html", context)
|