2021-01-19 19:25:39 +01:00
from django . views . generic import ListView , DetailView , FormView
2021-01-19 21:18:48 +01:00
from django . views . generic . list import MultipleObjectTemplateResponseMixin , MultipleObjectMixin
from django . views . generic . base import View
2021-01-19 19:25:39 +01:00
from django . views . generic . detail import SingleObjectMixin
2021-02-16 19:36:36 +01:00
from django . shortcuts import redirect , get_object_or_404
2021-01-19 21:18:48 +01:00
from django . urls import reverse
2021-02-16 22:59:45 +01:00
from django . db import transaction
2020-08-17 20:07:49 +02:00
from dataclasses import dataclass
import datetime
2021-04-06 20:30:48 +02:00
from itertools import groupby
2021-02-16 22:59:45 +01:00
import logging
2020-08-17 20:07:49 +02:00
import seminar . models as m
2021-01-19 19:25:39 +01:00
import seminar . forms as f
2021-03-02 22:46:43 +01:00
from seminar . forms import OdevzdavatkoTabulkaFiltrForm as FiltrForm
2021-04-06 20:30:48 +02:00
from seminar . utils import aktivniResitele , resi_v_rocniku , deadline
2020-08-17 20:07:49 +02:00
2021-02-16 22:59:45 +01:00
logger = logging . getLogger ( __name__ )
2020-08-17 20:07:49 +02:00
# Co chceme?
# - "Tabulku" aktuální řešitelé x zveřejněné problémy, v buňkách počet řešení
# - TabulkaOdevzdanychReseniView
# - Detail konkrétního problému a řešitele -- přehled všech řešení odevzdaných k tomuto problému
# - ReseniProblemuView
# - Detail konkrétního řešení -- všechny soubory, datum, ...
# - DetailReseniView
2021-04-06 23:33:10 +02:00
# - Pro řešitele: přehled jejich odevzdaných řešení
# - PrehledOdevzdanychReseni
2020-08-17 20:07:49 +02:00
#
# Taky se může hodit:
# - Tabulka všech řešitelů x všech problémů?
@dataclass
class SouhrnReseni :
""" Dataclass reprezentující data o odevzdaných řešeních pro zobrazení v tabulce. """
pocet_reseni : int
posledni_odevzdani : datetime . datetime
body : float
2020-11-18 00:35:18 +01:00
class TabulkaOdevzdanychReseniView ( ListView ) :
2020-10-27 23:54:35 +01:00
template_name = ' seminar/odevzdavatko/tabulka.html '
2020-11-18 00:35:18 +01:00
model = m . Hodnoceni
2021-01-26 19:12:01 +01:00
def inicializuj_osy_tabulky ( self ) :
""" Vyrobí prvotní querysety pro sloupce a řádky, tj. seznam všech řešitelů a problémů """
2021-03-02 22:46:43 +01:00
# FIXME: jméno metody není vypovídající...
2021-01-26 19:12:01 +01:00
# NOTE: Tenhle blok nemůže být přímo ve třídě, protože před vyrobením databáze neexistují ty objekty (?). TODO: Otestovat
# TODO: Prefetches, Select related, ...
self . resitele = m . Resitel . objects . all ( )
self . problemy = m . Problem . objects . all ( )
2021-03-02 22:46:43 +01:00
self . reseni = m . Reseni . objects . all ( )
form = FiltrForm ( self . request . GET )
if form . is_valid ( ) :
fcd = form . cleaned_data
resitele = fcd [ " resitele " ]
problemy = fcd [ " problemy " ]
reseni_od = fcd [ " reseni_od " ]
reseni_do = fcd [ " reseni_do " ]
else :
2021-03-09 21:33:53 +01:00
initial = FiltrForm . gen_initial ( )
resitele = initial [ ' resitele ' ]
problemy = initial [ ' problemy ' ]
2021-03-16 21:37:58 +01:00
reseni_od = initial [ ' reseni_od ' ] [ 0 ]
reseni_do = initial [ ' reseni_do ' ] [ 0 ]
2021-03-02 22:46:43 +01:00
# Filtrujeme!
aktualni_rocnik = m . Nastaveni . get_solo ( ) . aktualni_rocnik # .get_solo() vrátí tu jedinou instanci
2021-03-16 20:06:45 +01:00
self . chteni_resitele = resitele # Zapamatování pro get_context_data
2021-03-02 22:46:43 +01:00
if resitele == FiltrForm . RESITELE_RELEVANTNI :
2021-03-16 20:06:45 +01:00
# TODO: Zkontrolovat, že resi_v_rocniku vrací QuerySet (jinak asi bude žrát spoustu zdrojů zbytečně)
self . resitele = resi_v_rocniku ( aktualni_rocnik ) # Prvotní sada, pokud nebude mít body, odstraní se v get_context_data
2021-03-02 22:46:43 +01:00
elif resitele == FiltrForm . RESITELE_LETOSNI :
self . resitele = resi_v_rocniku ( aktualni_rocnik )
if problemy == FiltrForm . PROBLEMY_MOJE :
org = m . Organizator . objects . get ( osoba__user = self . request . user )
from django . db . models import Q
self . problemy = self . problemy . filter ( Q ( autor = org ) | Q ( garant = org ) | Q ( opravovatele = org ) , stav = m . Problem . STAV_ZADANY )
elif problemy == FiltrForm . PROBLEMY_LETOSNI :
self . problemy = self . problemy . filter ( stav = m . Problem . STAV_ZADANY )
#self.problemy = list(filter(lambda problem: problem.rocnik() == aktualni_rocnik, self.problemy)) # DB HOG? # FIXME: některé problémy nemají ročník....
# NOTE: Protože řešení odkazuje přímo na Problém a QuerySet na Hodnocení je nepolymorfní, musíme porovnávat taky s nepolymorfními Problémy.
self . problemy = self . problemy . non_polymorphic ( )
self . reseni = self . reseni . filter ( cas_doruceni__date__gte = reseni_od , cas_doruceni__date__lte = reseni_do )
2021-01-26 19:12:01 +01:00
2020-11-18 00:35:18 +01:00
def get_queryset ( self ) :
2021-01-26 19:12:01 +01:00
self . inicializuj_osy_tabulky ( )
2020-11-18 00:35:18 +01:00
qs = super ( ) . get_queryset ( )
2021-03-02 22:46:43 +01:00
qs = qs . filter ( problem__in = self . problemy , reseni__in = self . reseni , reseni__resitele__in = self . resitele ) . select_related ( ' reseni ' , ' problem ' ) . prefetch_related ( ' reseni__resitele__osoba ' )
2020-11-18 00:35:18 +01:00
return qs
2020-08-17 20:07:49 +02:00
def get_context_data ( self , * args , * * kwargs ) :
2021-03-02 22:46:43 +01:00
# self.resitele, self.reseni a self.problemy jsou již nastavené
2020-12-02 23:33:12 +01:00
2020-11-18 00:35:18 +01:00
ctx = super ( ) . get_context_data ( * args , * * kwargs )
2021-01-26 19:12:01 +01:00
ctx [ ' problemy ' ] = self . problemy
2020-11-18 00:35:18 +01:00
ctx [ ' resitele ' ] = self . resitele
tabulka = dict ( )
def pridej_reseni ( problem , resitel , body , cas ) :
if problem not in tabulka :
tabulka [ problem ] = dict ( )
if resitel not in tabulka [ problem ] :
tabulka [ problem ] [ resitel ] = SouhrnReseni ( pocet_reseni = 1 , posledni_odevzdani = cas , body = body )
else :
tabulka [ problem ] [ resitel ] . posledni_odevzdani = max ( tabulka [ problem ] [ resitel ] . posledni_odevzdani , cas )
2020-11-18 03:41:05 +01:00
tabulka [ problem ] [ resitel ] . body = max ( tabulka [ problem ] [ resitel ] . body , body ,
key = lambda x : x if x is not None else - 1 # None je malé číslo
# FIXME: Možná dává smysl i mít None jako velké číslo -- jakože "TODO: zadat body"
)
2020-11-18 00:35:18 +01:00
tabulka [ problem ] [ resitel ] . pocet_reseni + = 1
2020-12-01 23:52:15 +01:00
# Pro jednoduchost template si ještě poznamenáme ID problému a řešitele
tabulka [ problem ] [ resitel ] . problem_id = problem . id
tabulka [ problem ] [ resitel ] . resitel_id = resitel . id
2020-08-17 20:07:49 +02:00
2020-11-18 00:35:18 +01:00
for hodnoceni in self . get_queryset ( ) :
for resitel in hodnoceni . reseni . resitele . all ( ) :
pridej_reseni ( hodnoceni . problem , resitel , hodnoceni . body , hodnoceni . reseni . cas_doruceni )
hodnoty = [ ]
2021-03-16 20:06:45 +01:00
resitele_do_tabulky = [ ]
2020-11-18 00:35:18 +01:00
for resitel in self . resitele :
2021-03-16 20:06:45 +01:00
dostal_body = False
2020-11-18 00:35:18 +01:00
resiteluv_radek = [ ]
2021-01-26 19:12:01 +01:00
for problem in self . problemy :
2020-11-18 00:35:18 +01:00
if problem in tabulka and resitel in tabulka [ problem ] :
resiteluv_radek . append ( tabulka [ problem ] [ resitel ] )
2021-03-16 20:06:45 +01:00
dostal_body = True
2020-10-27 23:50:05 +01:00
else :
2020-11-18 00:35:18 +01:00
resiteluv_radek . append ( None )
2021-03-16 20:06:45 +01:00
if self . chteni_resitele != FiltrForm . RESITELE_RELEVANTNI or dostal_body :
hodnoty . append ( resiteluv_radek )
resitele_do_tabulky . append ( resitel )
ctx [ ' radky ' ] = list ( zip ( resitele_do_tabulky , hodnoty ) )
2021-03-02 22:46:43 +01:00
ctx [ ' filtr ' ] = FiltrForm ( initial = self . request . GET )
2021-02-23 22:13:14 +01:00
# Pro použití hacku na automatické {{form.media}} v template:
ctx [ ' form ' ] = ctx [ ' filtr ' ]
2020-08-17 20:07:49 +02:00
return ctx
2021-01-19 21:18:48 +01:00
# Velmi silně inspirováno zdrojáky, FIXME: Nedá se to udělat smysluplněji?
class ReseniProblemuView ( MultipleObjectTemplateResponseMixin , MultipleObjectMixin , View ) :
2020-08-17 20:07:49 +02:00
model = m . Reseni
2020-10-27 23:54:35 +01:00
template_name = ' seminar/odevzdavatko/seznam.html '
2020-08-17 20:07:49 +02:00
def get_queryset ( self ) :
qs = super ( ) . get_queryset ( )
2020-10-27 22:37:03 +01:00
resitel_id = self . kwargs [ ' resitel ' ]
if resitel_id is None :
raise ValueError ( " Nemám řešitele! " )
problem_id = self . kwargs [ ' problem ' ]
if problem_id is None :
raise ValueError ( " Nemám problém! (To je problém!) " )
resitel = m . Resitel . objects . get ( id = resitel_id )
2020-10-28 00:32:09 +01:00
problem = m . Problem . objects . get ( id = problem_id )
2020-10-27 22:37:03 +01:00
qs = qs . filter (
problem__in = [ problem ] ,
resitele__in = [ resitel ] ,
)
return qs
2021-01-19 21:18:48 +01:00
def get ( self , request , * args , * * kwargs ) :
self . object_list = self . get_queryset ( )
if self . object_list . count ( ) == 1 :
jedine_reseni = self . object_list . first ( )
return redirect ( reverse ( " odevzdavatko_detail_reseni " , kwargs = { " pk " : jedine_reseni . id } ) )
context = self . get_context_data ( )
return self . render_to_response ( context )
2021-04-06 20:30:48 +02:00
def get_context_data ( self , * args , * * kwargs ) :
ctx = super ( ) . get_context_data ( * args , * * kwargs )
2021-04-06 20:39:18 +02:00
# XXX: Předat groupby do template nejde: https://stackoverflow.com/questions/6906593/itertools-groupby-in-a-django-template
# Django má {% regroup %}, ale ten potřebuje, aby klíč byl atribut položky: https://docs.djangoproject.com/en/3.2/ref/templates/builtins/#regroup
# Takže rozbalíme groupby do slovníku klíč → seznam sami (dictionary comphrehension)
2021-04-06 20:30:48 +02:00
ctx [ ' reseni_podle_deadlinu ' ] = { k : list ( v ) for k , v in groupby ( ctx [ ' object_list ' ] , lambda r : deadline ( r . cas_doruceni ) ) }
return ctx
2020-08-17 20:07:49 +02:00
2021-01-19 19:25:39 +01:00
## XXX: https://docs.djangoproject.com/en/3.1/topics/class-based-views/mixins/#avoid-anything-more-complex
2020-10-27 22:37:03 +01:00
class DetailReseniView ( DetailView ) :
model = m . Reseni
2020-10-27 23:54:35 +01:00
template_name = ' seminar/odevzdavatko/detail.html '
2021-01-19 19:25:39 +01:00
def aktualni_hodnoceni ( self ) :
2021-02-16 19:36:36 +01:00
reseni = get_object_or_404 ( m . Reseni , id = self . kwargs [ ' pk ' ] )
result = [ ] # Slovníky s klíči problem, body, cislo_body -- initial data pro f.OhodnoceniReseniFormSet
for hodn in m . Hodnoceni . objects . filter ( reseni = reseni ) :
result . append (
{ " problem " : hodn . problem ,
" body " : hodn . body ,
" cislo_body " : hodn . cislo_body ,
} )
return result
2021-01-19 19:25:39 +01:00
def get_context_data ( self , * * kw ) :
ctx = super ( ) . get_context_data ( * * kw )
ctx [ ' form ' ] = f . OhodnoceniReseniFormSet (
initial = self . aktualni_hodnoceni ( )
)
return ctx
2021-02-16 22:59:45 +01:00
def hodnoceniReseniView ( request , pk , * args , * * kwargs ) :
reseni = get_object_or_404 ( m . Reseni , pk = pk )
2021-01-19 19:25:39 +01:00
template_name = ' seminar/odevzdavatko/detail.html '
form_class = f . OhodnoceniReseniFormSet
2021-02-16 22:59:45 +01:00
success_url = reverse ( ' odevzdavatko_detail_reseni ' , kwargs = { ' pk ' : pk } )
2021-02-16 23:37:59 +01:00
# FIXME: Použit initial i tady a nebastlit hodnocení tak nízkoúrovňově
# Also: https://docs.djangoproject.com/en/2.2/topics/forms/modelforms/#django.forms.ModelForm
2021-02-16 22:59:45 +01:00
formset = f . OhodnoceniReseniFormSet ( request . POST )
2021-02-16 23:49:42 +01:00
# TODO: Napsat validaci formuláře a formsetu
# TODO: Implementovat větev, kdy formulář validní není.
2021-02-16 22:59:45 +01:00
if formset . is_valid ( ) :
with transaction . atomic ( ) :
# Smažeme všechna dosavadní hodnocení tohoto řešení
qs = m . Hodnoceni . objects . filter ( reseni = reseni )
logger . info ( f " Will delete { qs . count ( ) } objects: { qs } " )
qs . delete ( )
# Vyrobíme nová podle formsetu
for form in formset :
problem = form . cleaned_data [ ' problem ' ]
body = form . cleaned_data [ ' body ' ]
cislo_body = form . cleaned_data [ ' cislo_body ' ]
hodnoceni = m . Hodnoceni (
problem = problem ,
body = body ,
cislo_body = cislo_body ,
reseni = reseni ,
)
logger . info ( f " Creating Hodnoceni: { hodnoceni } " )
hodnoceni . save ( )
2021-01-19 19:25:39 +01:00
2021-02-16 22:59:45 +01:00
return redirect ( success_url )
2021-01-19 19:25:39 +01:00
2020-08-17 20:07:49 +02:00
2021-04-06 23:33:10 +02:00
class PrehledOdevzdanychReseni ( ListView ) :
model = m . Hodnoceni
template_name = ' seminar/odevzdavatko/resitel_prehled.html '
def get_queryset ( self ) :
if not self . request . user . is_authenticated :
raise RuntimeError ( " Uživatel měl být přihlášený! " )
resitel = m . Resitel . objects . get ( osoba__user = self . request . user )
qs = super ( ) . get_queryset ( )
qs = qs . filter ( reseni__resitele__in = [ resitel ] )
return qs
def get_context_data ( self , * args , * * kwargs ) :
ctx = super ( ) . get_context_data ( * args , * * kwargs )
# Ročník určujeme podle čísla, do jehož deadlinu došlo řešení.
# Chceme to mít seřazené, takže místo comphrerehsion ručně postavíme pole polí. Django templates neumí použít OrderedDict :-/
podle_rocniku = [ ]
for rocnik , hodnoceni in groupby ( ctx [ ' object_list ' ] , lambda ho : deadline ( ho . reseni . cas_doruceni ) [ 1 ] . rocnik ) :
podle_rocniku . append ( ( rocnik , list ( hodnoceni ) ) )
ctx [ ' podle_rocniku ' ] = reversed ( podle_rocniku ) # Od nejnovějšího ročníku
# TODO: Umožnit stažení / zobrazení řešení
return ctx
2020-08-17 20:07:49 +02:00
# Přehled všech řešení kvůli debugování
class SeznamReseniView ( ListView ) :
model = m . Reseni
2020-10-27 23:54:35 +01:00
template_name = ' seminar/odevzdavatko/seznam.html '
2020-08-17 20:07:49 +02:00
class SeznamAktualnichReseniView ( SeznamReseniView ) :
def get_queryset ( self ) :
qs = super ( ) . get_queryset ( )
akt_rocnik = m . Nastaveni . get_solo ( ) . aktualni_rocnik # .get_solo() vrátí tu jedinou instanci, asi...
resitele = resi_v_rocniku ( akt_rocnik )
2020-10-28 00:32:20 +01:00
qs = qs . filter ( resitele__in = resitele ) # FIXME: Najde řešení i ze starých ročníků, která odevzdal alespoň jeden aktuální řešitel
2020-08-17 20:07:49 +02:00
return qs