573 lines
20 KiB
Python
573 lines
20 KiB
Python
##############################################################################
|
|
# ScoDoc
|
|
# Copyright (c) 1999 - 2023 Emmanuel Viennet. All rights reserved.
|
|
# See LICENSE
|
|
##############################################################################
|
|
|
|
"""Jury BUT: affichage/formulaire
|
|
"""
|
|
|
|
import re
|
|
import numpy as np
|
|
|
|
import flask
|
|
from flask import flash, render_template, url_for
|
|
from flask import g, request
|
|
|
|
from app import db
|
|
from app.but import jury_but
|
|
from app.but.jury_but import (
|
|
DecisionsProposeesAnnee,
|
|
DecisionsProposeesRCUE,
|
|
DecisionsProposeesUE,
|
|
)
|
|
from app.comp import res_sem
|
|
from app.comp.res_but import ResultatsSemestreBUT
|
|
from app.models import (
|
|
ApcNiveau,
|
|
FormSemestre,
|
|
FormSemestreInscription,
|
|
Identite,
|
|
UniteEns,
|
|
ScolarAutorisationInscription,
|
|
ScolarFormSemestreValidation,
|
|
ScolarNews,
|
|
)
|
|
from app.models.config import ScoDocSiteConfig
|
|
from app.scodoc import html_sco_header
|
|
from app.scodoc import codes_cursus as sco_codes
|
|
from app.scodoc.sco_exceptions import ScoValueError
|
|
from app.scodoc import sco_preferences
|
|
from app.scodoc import sco_utils as scu
|
|
|
|
|
|
def show_etud(deca: DecisionsProposeesAnnee, read_only: bool = True) -> str:
|
|
"""Affichage des décisions annuelles BUT
|
|
Si pas read_only, menus sélection codes jury.
|
|
"""
|
|
H = []
|
|
|
|
if deca.jury_annuel:
|
|
H.append(
|
|
f"""
|
|
<div class="but_section_annee">
|
|
<div>
|
|
<b>Décision de jury pour l'année :</b> {
|
|
_gen_but_select("code_annee", deca.codes, deca.code_valide,
|
|
disabled=True, klass="manual")
|
|
}
|
|
<span>({deca.code_valide or 'non'} enregistrée)</span>
|
|
</div>
|
|
</div>
|
|
"""
|
|
)
|
|
|
|
formsemestre_1 = deca.formsemestre_impair
|
|
formsemestre_2 = deca.formsemestre_pair
|
|
# Ordonne selon les dates des 2 semestres considérés (pour les redoublants à cheval):
|
|
reverse_semestre = (
|
|
deca.formsemestre_pair
|
|
and deca.formsemestre_impair
|
|
and deca.formsemestre_pair.date_debut < deca.formsemestre_impair.date_debut
|
|
)
|
|
if reverse_semestre:
|
|
formsemestre_1, formsemestre_2 = formsemestre_2, formsemestre_1
|
|
H.append(
|
|
f"""
|
|
<div class="titre_niveaux">
|
|
<b>Niveaux de compétences et unités d'enseignement du BUT{deca.annee_but}</b>
|
|
</div>
|
|
<div class="but_explanation">{deca.explanation}</div>
|
|
<div class="but_annee">
|
|
<div class="titre"></div>
|
|
<div class="titre">{"S" +str(formsemestre_1.semestre_id)
|
|
if formsemestre_1 else "-"}
|
|
<span class="avertissement_redoublement">{formsemestre_1.annee_scolaire_str()
|
|
if formsemestre_1 else ""}</span>
|
|
</div>
|
|
<div class="titre">{"S"+str(formsemestre_2.semestre_id)
|
|
if formsemestre_2 else "-"}
|
|
<span class="avertissement_redoublement">{formsemestre_2.annee_scolaire_str()
|
|
if formsemestre_2 else ""}</span>
|
|
</div>
|
|
<div class="titre">RCUE</div>
|
|
"""
|
|
)
|
|
for dec_rcue in deca.get_decisions_rcues_annee():
|
|
rcue = dec_rcue.rcue
|
|
niveau = rcue.niveau
|
|
H.append(
|
|
f"""<div class="but_niveau_titre">
|
|
<div title="{niveau.competence.titre_long}">{niveau.competence.titre}</div>
|
|
</div>"""
|
|
)
|
|
ue_impair, ue_pair = rcue.ue_1, rcue.ue_2
|
|
# Les UEs à afficher,
|
|
# qui
|
|
ues_ro = [
|
|
(
|
|
ue_impair,
|
|
rcue.ue_cur_impair is None,
|
|
),
|
|
(
|
|
ue_pair,
|
|
rcue.ue_cur_pair is None,
|
|
),
|
|
]
|
|
# Ordonne selon les dates des 2 semestres considérés:
|
|
if reverse_semestre:
|
|
ues_ro[0], ues_ro[1] = ues_ro[1], ues_ro[0]
|
|
# Colonnes d'UE:
|
|
for ue, ue_read_only in ues_ro:
|
|
if ue:
|
|
H.append(
|
|
_gen_but_niveau_ue(
|
|
ue,
|
|
deca.decisions_ues[ue.id],
|
|
disabled=read_only or ue_read_only,
|
|
annee_prec=ue_read_only,
|
|
niveau_id=ue.niveau_competence.id,
|
|
)
|
|
)
|
|
else:
|
|
H.append("""<div class="niveau_vide"></div>""")
|
|
|
|
# Colonne RCUE
|
|
H.append(_gen_but_rcue(dec_rcue, niveau))
|
|
|
|
H.append("</div>") # but_annee
|
|
return "\n".join(H)
|
|
|
|
|
|
def _gen_but_select(
|
|
name: str,
|
|
codes: list[str],
|
|
code_valide: str,
|
|
disabled: bool = False,
|
|
klass: str = "",
|
|
data: dict = None,
|
|
code_valide_label: str = "",
|
|
) -> str:
|
|
"Le menu html select avec les codes"
|
|
# if disabled: # mauvaise idée car le disabled est traité en JS
|
|
# return f"""<div class="but_code {klass}">{code_valide}</div>"""
|
|
data = data or {}
|
|
options_htm = "\n".join(
|
|
[
|
|
f"""<option value="{code}"
|
|
{'selected' if code == code_valide else ''}
|
|
class="{'recorded' if code == code_valide else ''}"
|
|
>{code
|
|
if ((code != code_valide) or not code_valide_label)
|
|
else code_valide_label
|
|
}</option>"""
|
|
for code in codes
|
|
]
|
|
)
|
|
return f"""<select required name="{name}"
|
|
class="but_code {klass}"
|
|
data-orig_code="{code_valide or (codes[0] if codes else '')}"
|
|
data-orig_recorded="{code_valide or ''}"
|
|
onchange="change_menu_code(this);"
|
|
{"disabled" if disabled else ""}
|
|
{" ".join( f'data-{k}="{v}"' for (k,v) in data.items() )}
|
|
>{options_htm}</select>
|
|
"""
|
|
|
|
|
|
def _gen_but_niveau_ue(
|
|
ue: UniteEns,
|
|
dec_ue: DecisionsProposeesUE,
|
|
disabled: bool = False,
|
|
annee_prec: bool = False,
|
|
niveau_id: int = None,
|
|
) -> str:
|
|
if dec_ue.ue_status and dec_ue.ue_status["is_capitalized"]:
|
|
moy_ue_str = f"""<span class="ue_cap">{
|
|
scu.fmt_note(dec_ue.moy_ue_with_cap)}</span>"""
|
|
scoplement = f"""<div class="scoplement">
|
|
<div>
|
|
<b>UE {ue.acronyme} capitalisée </b>
|
|
<span>le {dec_ue.ue_status["event_date"].strftime("%d/%m/%Y")}
|
|
</span>
|
|
</div>
|
|
<div>UE en cours
|
|
{ "sans notes" if np.isnan(dec_ue.moy_ue)
|
|
else
|
|
("avec moyenne <b>" + scu.fmt_note(dec_ue.moy_ue) + "</b>")
|
|
}
|
|
</div>
|
|
</div>
|
|
"""
|
|
else:
|
|
moy_ue_str = f"""<span>{scu.fmt_note(dec_ue.moy_ue)}</span>"""
|
|
if dec_ue.code_valide:
|
|
scoplement = f"""<div class="scoplement">
|
|
<div>Code {dec_ue.code_valide} enregistré le {dec_ue.validation.event_date.strftime("%d/%m/%Y")}
|
|
à {dec_ue.validation.event_date.strftime("%Hh%M")}
|
|
</div>
|
|
</div>
|
|
"""
|
|
else:
|
|
scoplement = ""
|
|
|
|
ue_class = "" # 'recorded' if dec_ue.code_valide is not None else ''
|
|
if dec_ue.code_valide is not None and dec_ue.codes:
|
|
if dec_ue.code_valide == dec_ue.codes[0]:
|
|
ue_class = "recorded"
|
|
else:
|
|
ue_class = "recorded_different"
|
|
|
|
return f"""<div class="but_niveau_ue {ue_class}
|
|
{'annee_prec' if annee_prec else ''}
|
|
">
|
|
<div title="{ue.titre}">{ue.acronyme}</div>
|
|
<div class="but_note with_scoplement">
|
|
<div>{moy_ue_str}</div>
|
|
{scoplement}
|
|
</div>
|
|
<div class="but_code">{
|
|
_gen_but_select("code_ue_"+str(ue.id),
|
|
dec_ue.codes,
|
|
dec_ue.code_valide,
|
|
disabled=disabled,
|
|
klass=f"code_ue ue_rcue_{niveau_id}" if not disabled else ""
|
|
)
|
|
}</div>
|
|
|
|
</div>"""
|
|
|
|
|
|
def _gen_but_rcue(dec_rcue: DecisionsProposeesRCUE, niveau: ApcNiveau) -> str:
|
|
if dec_rcue is None or not dec_rcue.rcue.complete:
|
|
return """
|
|
<div class="but_niveau_rcue niveau_vide with_scoplement">
|
|
<div></div>
|
|
<div class="scoplement">Pas de RCUE (UE non capitalisée ?)</div>
|
|
</div>
|
|
"""
|
|
|
|
code_propose_menu = dec_rcue.code_valide # le code enregistré
|
|
code_valide_label = code_propose_menu
|
|
if dec_rcue.validation:
|
|
if dec_rcue.code_valide == dec_rcue.codes[0]:
|
|
descr_validation = dec_rcue.validation.html()
|
|
else: # on une validation enregistrée différence de celle proposée
|
|
descr_validation = f"""Décision recommandée: <b>{dec_rcue.codes[0]}.</b>
|
|
Il y avait {dec_rcue.validation.html()}"""
|
|
if (
|
|
sco_codes.BUT_CODES_ORDER[dec_rcue.codes[0]]
|
|
> sco_codes.BUT_CODES_ORDER[dec_rcue.code_valide]
|
|
):
|
|
code_propose_menu = dec_rcue.codes[0]
|
|
code_valide_label = (
|
|
f"{dec_rcue.codes[0]} (actuel {dec_rcue.code_valide})"
|
|
)
|
|
scoplement = f"""<div class="scoplement">{descr_validation}</div>"""
|
|
else:
|
|
scoplement = "" # "pas de validation"
|
|
|
|
# Déjà enregistré ?
|
|
niveau_rcue_class = ""
|
|
if dec_rcue.code_valide is not None and dec_rcue.codes:
|
|
if dec_rcue.code_valide == dec_rcue.codes[0]:
|
|
niveau_rcue_class = "recorded"
|
|
else:
|
|
niveau_rcue_class = "recorded_different"
|
|
|
|
return f"""
|
|
<div class="but_niveau_rcue {niveau_rcue_class}
|
|
">
|
|
<div class="but_note with_scoplement">
|
|
<div>{scu.fmt_note(dec_rcue.rcue.moy_rcue)}</div>
|
|
{scoplement}
|
|
</div>
|
|
<div class="but_code">
|
|
{_gen_but_select("code_rcue_"+str(niveau.id),
|
|
dec_rcue.codes,
|
|
code_propose_menu,
|
|
disabled=True,
|
|
klass="manual code_rcue",
|
|
data = { "niveau_id" : str(niveau.id)},
|
|
code_valide_label = code_valide_label,
|
|
)}
|
|
</div>
|
|
</div>
|
|
"""
|
|
|
|
|
|
def jury_but_semestriel(
|
|
formsemestre: FormSemestre,
|
|
etud: Identite,
|
|
read_only: bool,
|
|
navigation_div: str = "",
|
|
) -> str:
|
|
"""Page: formulaire saisie décision d'UE d'un semestre BUT isolé (pas jury annuel)."""
|
|
res: ResultatsSemestreBUT = res_sem.load_formsemestre_results(formsemestre)
|
|
parcour, ues = jury_but.list_ue_parcour_etud(formsemestre, etud, res)
|
|
inscription_etat = etud.inscription_etat(formsemestre.id)
|
|
semestre_terminal = (
|
|
formsemestre.semestre_id >= formsemestre.formation.get_cursus().NB_SEM
|
|
)
|
|
autorisations_passage = ScolarAutorisationInscription.query.filter_by(
|
|
etudid=etud.id,
|
|
origin_formsemestre_id=formsemestre.id,
|
|
).all()
|
|
# Par défaut: autorisé à passer dans le semestre suivant si sem. impair,
|
|
# ou si décision déjà enregistrée:
|
|
est_autorise_a_passer = (formsemestre.semestre_id % 2) or (
|
|
formsemestre.semestre_id + 1
|
|
) in (a.semestre_id for a in autorisations_passage)
|
|
decisions_ues = {
|
|
ue.id: DecisionsProposeesUE(etud, formsemestre, ue, inscription_etat)
|
|
for ue in ues
|
|
}
|
|
for dec_ue in decisions_ues.values():
|
|
dec_ue.compute_codes()
|
|
|
|
if request.method == "POST":
|
|
if not read_only:
|
|
for key in request.form:
|
|
code = request.form[key]
|
|
# Codes d'UE
|
|
code_match = re.match(r"^code_ue_(\d+)$", key)
|
|
if code_match:
|
|
ue_id = int(code_match.group(1))
|
|
dec_ue = decisions_ues.get(ue_id)
|
|
if not dec_ue:
|
|
raise ScoValueError(f"UE invalide ue_id={ue_id}")
|
|
dec_ue.record(code)
|
|
db.session.commit()
|
|
flash("codes enregistrés")
|
|
if not semestre_terminal:
|
|
if request.form.get("autorisation_passage"):
|
|
if not formsemestre.semestre_id + 1 in (
|
|
a.semestre_id for a in autorisations_passage
|
|
):
|
|
ScolarAutorisationInscription.delete_autorisation_etud(
|
|
etud.id, formsemestre.id
|
|
)
|
|
ScolarAutorisationInscription.autorise_etud(
|
|
etud.id,
|
|
formsemestre.formation.formation_code,
|
|
formsemestre.id,
|
|
formsemestre.semestre_id + 1,
|
|
)
|
|
db.session.commit()
|
|
flash(
|
|
f"""autorisation de passage en S{formsemestre.semestre_id + 1
|
|
} enregistrée"""
|
|
)
|
|
else:
|
|
if est_autorise_a_passer:
|
|
ScolarAutorisationInscription.delete_autorisation_etud(
|
|
etud.id, formsemestre.id
|
|
)
|
|
db.session.commit()
|
|
flash(
|
|
f"autorisation de passage en S{formsemestre.semestre_id + 1} annulée"
|
|
)
|
|
ScolarNews.add(
|
|
typ=ScolarNews.NEWS_JURY,
|
|
obj=formsemestre.id,
|
|
text=f"""Saisie décision jury dans {formsemestre.html_link_status()}""",
|
|
url=url_for(
|
|
"notes.formsemestre_status",
|
|
scodoc_dept=g.scodoc_dept,
|
|
formsemestre_id=formsemestre.id,
|
|
),
|
|
)
|
|
return flask.redirect(
|
|
url_for(
|
|
"notes.formsemestre_validation_but",
|
|
scodoc_dept=g.scodoc_dept,
|
|
formsemestre_id=formsemestre.id,
|
|
etudid=etud.id,
|
|
)
|
|
)
|
|
# GET
|
|
if formsemestre.semestre_id % 2 == 0:
|
|
warning = f"""<div class="warning">
|
|
Cet étudiant de S{formsemestre.semestre_id} ne peut pas passer
|
|
en jury BUT annuel car il lui manque le semestre précédent.
|
|
</div>"""
|
|
else:
|
|
warning = ""
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title=f"Validation BUT S{formsemestre.semestre_id}",
|
|
formsemestre_id=formsemestre.id,
|
|
etudid=etud.id,
|
|
cssstyles=("css/jury_but.css",),
|
|
javascripts=("js/jury_but.js",),
|
|
),
|
|
f"""
|
|
<div class="jury_but">
|
|
<div>
|
|
<div class="bull_head">
|
|
<div>
|
|
<div class="titre_parcours">Jury BUT S{formsemestre.id}
|
|
- Parcours {(parcour.libelle if parcour else False) or "non spécifié"}
|
|
</div>
|
|
<div class="nom_etud">{etud.nomprenom}</div>
|
|
</div>
|
|
<div class="bull_photo"><a href="{
|
|
url_for("scolar.ficheEtud", scodoc_dept=g.scodoc_dept, etudid=etud.id)
|
|
}">{etud.photo_html(title="fiche de " + etud.nomprenom)}</a>
|
|
</div>
|
|
</div>
|
|
<h3>Jury sur un semestre BUT isolé (ne concerne que les UEs)</h3>
|
|
{warning}
|
|
</div>
|
|
|
|
<form method="post" class="jury_but_box" id="jury_but">
|
|
""",
|
|
]
|
|
|
|
erase_span = ""
|
|
if not read_only:
|
|
# Requête toutes les validations (pas seulement celles du deca courant),
|
|
# au cas où: changement d'architecture, saisie en mode classique, ...
|
|
validations = ScolarFormSemestreValidation.query.filter_by(
|
|
etudid=etud.id, formsemestre_id=formsemestre.id
|
|
).all()
|
|
if validations:
|
|
erase_span = f"""<a href="{
|
|
url_for("notes.formsemestre_jury_but_erase",
|
|
scodoc_dept=g.scodoc_dept, formsemestre_id=formsemestre.id,
|
|
etudid=etud.id, only_one_sem=1)
|
|
}" class="stdlink">effacer les décisions enregistrées</a>"""
|
|
else:
|
|
erase_span = (
|
|
"Cet étudiant n'a aucune décision enregistrée pour ce semestre."
|
|
)
|
|
|
|
H.append(
|
|
f"""
|
|
<div class="but_section_annee">
|
|
</div>
|
|
<div><b>Unités d'enseignement de S{formsemestre.semestre_id}:</b></div>
|
|
"""
|
|
)
|
|
if not ues:
|
|
H.append(
|
|
"""<div class="warning">Aucune UE ! Vérifiez votre programme de
|
|
formation, et l'association UEs / Niveaux de compétences</div>"""
|
|
)
|
|
else:
|
|
H.append(
|
|
"""
|
|
<div class="but_annee">
|
|
<div class="titre"></div>
|
|
<div class="titre"></div>
|
|
<div class="titre"></div>
|
|
<div class="titre"></div>
|
|
"""
|
|
)
|
|
for ue in ues:
|
|
dec_ue = decisions_ues[ue.id]
|
|
H.append("""<div class="but_niveau_titre"><div></div></div>""")
|
|
H.append(
|
|
_gen_but_niveau_ue(
|
|
ue,
|
|
dec_ue,
|
|
disabled=read_only,
|
|
)
|
|
)
|
|
H.append(
|
|
"""<div style=""></div>
|
|
<div class=""></div>"""
|
|
)
|
|
H.append("</div>") # but_annee
|
|
|
|
div_autorisations_passage = (
|
|
f"""
|
|
<div class="but_autorisations_passage">
|
|
<span>Autorisé à passer en :</span>
|
|
{ ", ".join( ["S" + str(a.semestre_id or '') for a in autorisations_passage ] )}
|
|
</div>
|
|
"""
|
|
if autorisations_passage
|
|
else """<div class="but_autorisations_passage but_explanation">pas d'autorisations de passage enregistrées.</div>"""
|
|
)
|
|
H.append(div_autorisations_passage)
|
|
|
|
if read_only:
|
|
H.append(
|
|
f"""<div class="but_explanation">
|
|
{"Vous n'avez pas la permission de modifier ces décisions."
|
|
if formsemestre.etat
|
|
else "Semestre verrouillé."}
|
|
Les champs entourés en vert sont enregistrés.
|
|
</div>
|
|
"""
|
|
)
|
|
else:
|
|
if formsemestre.semestre_id < formsemestre.formation.get_cursus().NB_SEM:
|
|
H.append(
|
|
f"""
|
|
<div class="but_settings">
|
|
<input type="checkbox" name="autorisation_passage" value="1" {
|
|
"checked" if est_autorise_a_passer else ""}>
|
|
<em>autoriser à passer dans le semestre S{formsemestre.semestre_id+1}</em>
|
|
</input>
|
|
</div>
|
|
"""
|
|
)
|
|
else:
|
|
H.append("""<div class="help">dernier semestre de la formation.</div>""")
|
|
H.append(
|
|
f"""
|
|
<div class="but_buttons">
|
|
<span><input type="submit" value="Enregistrer ces décisions"></span>
|
|
<span>{erase_span}</span>
|
|
</div>
|
|
"""
|
|
)
|
|
|
|
H.append(navigation_div)
|
|
H.append("</div>")
|
|
H.append(
|
|
render_template(
|
|
"but/documentation_codes_jury.j2",
|
|
nom_univ=f"""Export {sco_preferences.get_preference("InstituteName")
|
|
or sco_preferences.get_preference("UnivName")
|
|
or "Apogée"}""",
|
|
codes=ScoDocSiteConfig.get_codes_apo_dict(),
|
|
)
|
|
)
|
|
|
|
return "\n".join(H)
|
|
|
|
|
|
# -------------
|
|
def infos_fiche_etud_html(etudid: int) -> str:
|
|
"""Section html pour fiche etudiant
|
|
provisoire pour BUT 2022
|
|
"""
|
|
etud = Identite.get_etud(etudid)
|
|
inscriptions = (
|
|
FormSemestreInscription.query.join(FormSemestreInscription.formsemestre)
|
|
.filter(
|
|
FormSemestreInscription.etudid == etud.id,
|
|
)
|
|
.order_by(FormSemestre.date_debut)
|
|
)
|
|
formsemestres_but = [
|
|
i.formsemestre for i in inscriptions if i.formsemestre.formation.is_apc()
|
|
]
|
|
if len(formsemestres_but) == 0:
|
|
return ""
|
|
|
|
# temporaire quick & dirty: affiche le dernier
|
|
try:
|
|
deca = DecisionsProposeesAnnee(etud, formsemestres_but[-1])
|
|
return f"""<div class="infos_but">
|
|
{show_etud(deca, read_only=True)}
|
|
</div>
|
|
"""
|
|
except ScoValueError:
|
|
pass
|
|
|
|
return ""
|