forked from ScoDoc/ScoDoc
241 lines
9.2 KiB
Python
241 lines
9.2 KiB
Python
# -*- mode: python -*-
|
|
# -*- coding: utf-8 -*-
|
|
|
|
##############################################################################
|
|
#
|
|
# Gestion scolarite IUT
|
|
#
|
|
# Copyright (c) 1999 - 2024 Emmanuel Viennet. All rights reserved.
|
|
#
|
|
# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
# Emmanuel Viennet emmanuel.viennet@gmail.com
|
|
#
|
|
##############################################################################
|
|
|
|
"""Extraction de données pour poursuites d'études
|
|
|
|
Recapitule tous les semestres validés dans une feuille excel.
|
|
"""
|
|
import collections
|
|
|
|
from flask import url_for, g, request
|
|
|
|
from app.comp import res_sem
|
|
from app.comp.res_compat import NotesTableCompat
|
|
from app.models import FormSemestre
|
|
import app.scodoc.sco_utils as scu
|
|
from app.scodoc import sco_assiduites
|
|
from app.scodoc import sco_formsemestre
|
|
from app.scodoc import sco_groups
|
|
from app.scodoc import sco_preferences
|
|
from app.scodoc import sco_etud
|
|
from app.scodoc.gen_tables import GenTable
|
|
from app.scodoc.codes_cursus import code_semestre_validant, code_semestre_attente
|
|
import sco_version
|
|
|
|
|
|
def etud_get_poursuite_info(sem: dict, etud: dict) -> dict:
|
|
"""{ 'nom' : ..., 'semlist' : [ { 'semestre_id': , 'moy' : ... }, {}, ...] }"""
|
|
infos = {}
|
|
infos.update(etud) # copie nom, prenom, civilite, ...
|
|
|
|
# Now add each semester, starting from the first one
|
|
semlist = []
|
|
current_id = sem["semestre_id"]
|
|
for sem_id in range(1, current_id + 1):
|
|
sem_descr = None
|
|
for s in etud["sems"]:
|
|
if s["semestre_id"] == sem_id:
|
|
etudid = etud["etudid"]
|
|
formsemestre = FormSemestre.query.get_or_404(s["formsemestre_id"])
|
|
nt: NotesTableCompat = res_sem.load_formsemestre_results(formsemestre)
|
|
dec = nt.get_etud_decision_sem(etudid)
|
|
# Moyennes et rangs des UE
|
|
ues = nt.get_ues_stat_dict(filter_sport=True)
|
|
moy_ues = []
|
|
for ue in ues:
|
|
ue_status = nt.get_etud_ue_status(etudid, ue["ue_id"])
|
|
if ue_status:
|
|
moy_ues.append(
|
|
(
|
|
ue["acronyme"],
|
|
scu.fmt_note(ue_status["moy"]),
|
|
)
|
|
)
|
|
else:
|
|
moy_ues.append((ue["acronyme"], ""))
|
|
rg_ues = [
|
|
("rang_" + ue["acronyme"], nt.ue_rangs[ue["ue_id"]][0][etudid])
|
|
for ue in ues
|
|
]
|
|
|
|
# Moyennes et rang des modules
|
|
modimpls = nt.get_modimpls_dict() # recupération des modules
|
|
modules = []
|
|
rangs = []
|
|
for ue in ues: # on parcourt chaque UE
|
|
for modimpl in modimpls: # dans chaque UE les modules
|
|
if modimpl["module"]["ue_id"] == ue["ue_id"]:
|
|
code_module = modimpl["module"]["code"] or ""
|
|
note_module = scu.fmt_note(
|
|
nt.get_etud_mod_moy(modimpl["moduleimpl_id"], etudid)
|
|
)
|
|
# si étudiant inscrit au module, sauf BUT
|
|
if (note_module != "NI") and not nt.is_apc:
|
|
if nt.mod_rangs is not None:
|
|
rang_module = nt.mod_rangs[
|
|
modimpl["moduleimpl_id"]
|
|
][0][etudid]
|
|
else:
|
|
rang_module = ""
|
|
modules.append([code_module, note_module])
|
|
rangs.append(["rang_" + code_module, rang_module])
|
|
|
|
# Absences
|
|
nbabsnj, nbabsjust, _ = sco_assiduites.get_assiduites_count(
|
|
etudid, nt.sem
|
|
)
|
|
# En BUT, prend tout, sinon ne prend que les semestre validés par le jury
|
|
if nt.is_apc or (
|
|
dec
|
|
# not sem_descr pour ne prendre que le semestre validé le plus récent:
|
|
and not sem_descr
|
|
and (
|
|
code_semestre_validant(dec["code"])
|
|
or code_semestre_attente(dec["code"])
|
|
)
|
|
and nt.get_etud_etat(etudid) == scu.INSCRIT
|
|
):
|
|
d = [
|
|
("moy", scu.fmt_note(nt.get_etud_moy_gen(etudid))),
|
|
("moy_promo", scu.fmt_note(nt.moy_moy)),
|
|
("rang", nt.get_etud_rang(etudid)),
|
|
("effectif", len(nt.T)),
|
|
("date_debut", s["date_debut"]),
|
|
("date_fin", s["date_fin"]),
|
|
("periode", "%s - %s" % (s["mois_debut"], s["mois_fin"])),
|
|
("AbsNonJust", nbabsnj),
|
|
("AbsJust", nbabsjust),
|
|
]
|
|
# ajout des 2 champs notes des modules et classement dans chaque module
|
|
d += moy_ues + rg_ues + modules + rangs
|
|
sem_descr = collections.OrderedDict(d)
|
|
if not sem_descr:
|
|
sem_descr = collections.OrderedDict(
|
|
[
|
|
("moy", ""),
|
|
("moy_promo", ""),
|
|
("rang", ""),
|
|
("effectif", ""),
|
|
("date_debut", ""),
|
|
("date_fin", ""),
|
|
("periode", ""),
|
|
]
|
|
)
|
|
sem_descr["semestre_id"] = sem_id
|
|
semlist.append(sem_descr)
|
|
|
|
infos["semlist"] = semlist
|
|
return infos
|
|
|
|
|
|
def _flatten_info(info):
|
|
"""met la liste des infos semestres "a plat"
|
|
S1_moy, S1_rang, ..., S2_moy, ...
|
|
"""
|
|
ids = []
|
|
for s in info["semlist"]:
|
|
for k, v in s.items():
|
|
if k != "semestre_id":
|
|
label = "S%s_%s" % (s["semestre_id"], k)
|
|
info[label] = v
|
|
ids.append(label)
|
|
return ids
|
|
|
|
|
|
def _get_etud_info_groupes(group_ids, etat=None):
|
|
"""liste triée d'infos (dict) sur les etudiants du groupe indiqué.
|
|
Attention: lent, car plusieurs requetes SQL par etudiant !
|
|
"""
|
|
etuds = []
|
|
for group_id in group_ids:
|
|
members = sco_groups.get_group_members(group_id, etat=etat)
|
|
for m in members:
|
|
etud = sco_etud.get_etud_info(etudid=m["etudid"], filled=True)[0]
|
|
etuds.append(etud)
|
|
|
|
return etuds
|
|
|
|
|
|
def formsemestre_poursuite_report(formsemestre_id, fmt="html"):
|
|
"""Table avec informations "poursuite" """
|
|
sem = sco_formsemestre.get_formsemestre(formsemestre_id)
|
|
etuds = _get_etud_info_groupes([sco_groups.get_default_group(formsemestre_id)])
|
|
|
|
infos = []
|
|
ids = []
|
|
for etud in etuds:
|
|
fiche_url = url_for(
|
|
"scolar.fiche_etud", scodoc_dept=g.scodoc_dept, etudid=etud["etudid"]
|
|
)
|
|
etud["_nom_target"] = fiche_url
|
|
etud["_prenom_target"] = fiche_url
|
|
etud["_nom_td_attrs"] = f"""id="{etud['etudid']}" class="etudinfo" """
|
|
info = etud_get_poursuite_info(sem, etud)
|
|
idd = _flatten_info(info)
|
|
# On recupere la totalite des UEs dans ids
|
|
for key in idd:
|
|
if key not in ids:
|
|
ids += [key]
|
|
info["etudid"] = etud["etudid"]
|
|
infos.append(info)
|
|
#
|
|
column_ids = (
|
|
(("etudid",) if fmt.startswith("xls") else ())
|
|
+ ("civilite_str", "nom", "prenom", "annee", "date_naissance")
|
|
+ tuple(ids)
|
|
+ ("debouche",)
|
|
)
|
|
titles = {}
|
|
for c in column_ids:
|
|
titles[c] = c
|
|
tab = GenTable(
|
|
titles=titles,
|
|
columns_ids=column_ids,
|
|
rows=infos,
|
|
# html_col_width='4em',
|
|
html_sortable=True,
|
|
html_class="table_leftalign table_listegroupe",
|
|
pdf_link=False, # pas d'export pdf
|
|
preferences=sco_preferences.SemPreferences(formsemestre_id),
|
|
table_id="formsemestre_poursuite_report",
|
|
)
|
|
tab.filename = scu.make_filename("poursuite " + sem["titreannee"])
|
|
|
|
tab.origin = (
|
|
"Généré par %s le " % sco_version.SCONAME + scu.timedate_human_repr() + ""
|
|
)
|
|
tab.caption = "Récapitulatif %s." % sem["titreannee"]
|
|
tab.html_caption = "Récapitulatif %s." % sem["titreannee"]
|
|
tab.base_url = "%s?formsemestre_id=%s" % (request.base_url, formsemestre_id)
|
|
return tab.make_page(
|
|
title="""<h2 class="formsemestre">Poursuite d'études</h2>""",
|
|
init_qtip=True,
|
|
javascripts=["js/etud_info.js"],
|
|
fmt=fmt,
|
|
with_html_headers=True,
|
|
)
|