2022-02-21 19:25:38 +01:00
|
|
|
##############################################################################
|
|
|
|
# ScoDoc
|
|
|
|
# Copyright (c) 1999 - 2022 Emmanuel Viennet. All rights reserved.
|
|
|
|
# See LICENSE
|
|
|
|
##############################################################################
|
|
|
|
|
|
|
|
"""Génération bulletin BUT au format PDF standard
|
|
|
|
"""
|
2022-03-10 20:44:01 +01:00
|
|
|
from reportlab.platypus import Paragraph, Spacer
|
2022-02-21 19:25:38 +01:00
|
|
|
|
2022-03-10 00:50:36 +01:00
|
|
|
from app.scodoc.sco_pdf import blue, cm, mm
|
2022-02-21 19:25:38 +01:00
|
|
|
|
|
|
|
from app.scodoc import gen_tables
|
2022-03-16 18:51:22 +01:00
|
|
|
from app.scodoc.sco_codes_parcours import UE_SPORT
|
2022-02-21 19:25:38 +01:00
|
|
|
from app.scodoc.sco_utils import fmt_note
|
|
|
|
|
|
|
|
from app.scodoc.sco_bulletins_standard import BulletinGeneratorStandard
|
|
|
|
|
|
|
|
|
|
|
|
class BulletinGeneratorStandardBUT(BulletinGeneratorStandard):
|
|
|
|
"""Génération du bulletin de BUT au format PDF.
|
|
|
|
|
|
|
|
self.infos est le dict issu de BulletinBUT.bulletin_etud_complet()
|
|
|
|
"""
|
|
|
|
|
|
|
|
list_in_menu = False # spécialisation du BulletinGeneratorStandard, ne pas présenter à l'utilisateur
|
2022-03-13 22:22:54 +01:00
|
|
|
scale_table_in_page = False # pas de mise à l'échelle pleine page auto
|
|
|
|
multi_pages = True # plusieurs pages par bulletins
|
2022-03-12 09:40:48 +01:00
|
|
|
small_fontsize = "8"
|
2022-02-21 19:25:38 +01:00
|
|
|
|
|
|
|
def bul_table(self, format="html"):
|
|
|
|
"""Génère la table centrale du bulletin de notes
|
|
|
|
Renvoie:
|
|
|
|
- en HTML: une chaine
|
|
|
|
- en PDF: une liste d'objets PLATYPUS (eg instance de Table).
|
|
|
|
"""
|
2022-03-10 00:50:36 +01:00
|
|
|
tables_infos = [
|
|
|
|
# ---- TABLE SYNTHESE UES
|
|
|
|
self.but_table_synthese_ues(),
|
|
|
|
]
|
2022-03-10 09:28:59 +01:00
|
|
|
if self.version != "short":
|
|
|
|
tables_infos += [
|
|
|
|
# ---- TABLE RESSOURCES
|
|
|
|
self.but_table_ressources(),
|
|
|
|
# ---- TABLE SAE
|
|
|
|
self.but_table_saes(),
|
|
|
|
]
|
2022-03-10 00:50:36 +01:00
|
|
|
objects = []
|
|
|
|
for i, (col_keys, rows, pdf_style, col_widths) in enumerate(tables_infos):
|
|
|
|
table = gen_tables.GenTable(
|
|
|
|
rows=rows,
|
|
|
|
columns_ids=col_keys,
|
|
|
|
pdf_table_style=pdf_style,
|
|
|
|
pdf_col_widths=[col_widths[k] for k in col_keys],
|
|
|
|
preferences=self.preferences,
|
|
|
|
html_class="notes_bulletin",
|
|
|
|
html_class_ignore_default=True,
|
|
|
|
html_with_td_classes=True,
|
|
|
|
)
|
|
|
|
table_objects = table.gen(format=format)
|
|
|
|
objects += table_objects
|
|
|
|
# objects += [KeepInFrame(0, 0, table_objects, mode="shrink")]
|
|
|
|
if i != 2:
|
|
|
|
objects.append(Spacer(1, 6 * mm))
|
2022-02-21 19:25:38 +01:00
|
|
|
|
2022-03-10 00:50:36 +01:00
|
|
|
return objects
|
2022-02-21 19:25:38 +01:00
|
|
|
|
2022-03-10 00:50:36 +01:00
|
|
|
def but_table_synthese_ues(self, title_bg=(182, 235, 255)):
|
2022-02-21 19:25:38 +01:00
|
|
|
"""La table de synthèse; pour chaque UE, liste des ressources et SAÉs avec leurs notes
|
|
|
|
et leurs coefs.
|
|
|
|
Renvoie: colkeys, P, pdf_style, colWidths
|
|
|
|
- colkeys: nom des colonnes de la table (clés)
|
|
|
|
- P : table (liste de dicts de chaines de caracteres)
|
|
|
|
- pdf_style : commandes table Platypus
|
|
|
|
- largeurs de colonnes pour PDF
|
|
|
|
"""
|
|
|
|
col_widths = {
|
|
|
|
"titre": None,
|
|
|
|
"moyenne": 2 * cm,
|
|
|
|
"coef": 2 * cm,
|
|
|
|
}
|
2022-03-10 00:50:36 +01:00
|
|
|
title_bg = tuple(x / 255.0 for x in title_bg)
|
2022-03-12 09:40:48 +01:00
|
|
|
nota_bene = "La moyenne des ressources et SAÉs dans une UE dépend des poids donnés aux évaluations."
|
2022-03-10 00:50:36 +01:00
|
|
|
# elems pour générer table avec gen_table (liste de dicts)
|
|
|
|
rows = [
|
|
|
|
# Ligne de titres
|
|
|
|
{
|
|
|
|
"titre": "Unités d'enseignement",
|
2022-03-12 09:40:48 +01:00
|
|
|
"moyenne": Paragraph("<para align=right><b>Note/20</b></para>"),
|
2022-03-10 00:50:36 +01:00
|
|
|
"coef": "Coef.",
|
2022-03-10 20:44:01 +01:00
|
|
|
"_coef_pdf": Paragraph("<para align=right><b><i>Coef.</i></b></para>"),
|
2022-03-10 00:50:36 +01:00
|
|
|
"_css_row_class": "note_bold",
|
|
|
|
"_pdf_row_markup": ["b"],
|
2022-03-12 09:40:48 +01:00
|
|
|
"_pdf_style": [
|
|
|
|
("BACKGROUND", (0, 0), (-1, 0), title_bg),
|
|
|
|
# ("BOTTOMPADDING", (0, 0), (-1, 0), 7),
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"titre": nota_bene,
|
|
|
|
"_titre_pdf": Paragraph(
|
|
|
|
f"<para fontSize={self.small_fontsize}><i>{nota_bene}</i></para>"
|
|
|
|
),
|
|
|
|
"_titre_colspan": 3,
|
2022-03-10 00:50:36 +01:00
|
|
|
"_pdf_style": [
|
|
|
|
("BACKGROUND", (0, 0), (-1, 0), title_bg),
|
|
|
|
("BOTTOMPADDING", (0, 0), (-1, 0), 7),
|
|
|
|
(
|
|
|
|
"LINEBELOW",
|
|
|
|
(0, 0),
|
|
|
|
(-1, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
|
|
|
blue,
|
|
|
|
),
|
|
|
|
],
|
2022-03-12 09:40:48 +01:00
|
|
|
},
|
2022-03-10 00:50:36 +01:00
|
|
|
]
|
2022-03-10 20:44:01 +01:00
|
|
|
col_keys = ["titre", "coef", "moyenne"] # noms des colonnes à afficher
|
2022-02-21 19:25:38 +01:00
|
|
|
for ue_acronym, ue in self.infos["ues"].items():
|
2022-03-16 18:51:22 +01:00
|
|
|
self.ue_rows(rows, ue_acronym, ue, title_bg)
|
2022-03-10 00:50:36 +01:00
|
|
|
# Global pdf style commands:
|
|
|
|
pdf_style = [
|
|
|
|
("VALIGN", (0, 0), (-1, -1), "TOP"),
|
|
|
|
("BOX", (0, 0), (-1, -1), 0.4, blue), # ajoute cadre extérieur bleu:
|
|
|
|
]
|
|
|
|
return col_keys, rows, pdf_style, col_widths
|
|
|
|
|
2022-03-16 18:51:22 +01:00
|
|
|
def ue_rows(self, rows: list, ue_acronym: str, ue: dict, title_bg: tuple):
|
|
|
|
"Décrit une UE dans la table synthèse: titre, sous-titre et liste modules"
|
|
|
|
# 1er ligne titre UE
|
|
|
|
moy_ue = ue.get("moyenne")
|
|
|
|
t = {
|
|
|
|
"titre": f"{ue_acronym} - {ue['titre']}",
|
|
|
|
"moyenne": Paragraph(
|
|
|
|
f"""<para align=right><b>{moy_ue.get("value", "-") if moy_ue is not None else "-"}</b></para>"""
|
|
|
|
),
|
|
|
|
"_css_row_class": "note_bold",
|
|
|
|
"_pdf_row_markup": ["b"],
|
|
|
|
"_pdf_style": [
|
|
|
|
(
|
|
|
|
"LINEABOVE",
|
|
|
|
(0, 0),
|
|
|
|
(-1, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
|
|
|
self.PDF_LINECOLOR,
|
|
|
|
),
|
|
|
|
("BACKGROUND", (0, 0), (-1, 0), title_bg),
|
|
|
|
("BOTTOMPADDING", (0, 0), (-1, 0), 7),
|
|
|
|
],
|
|
|
|
}
|
|
|
|
rows.append(t)
|
|
|
|
if ue["type"] == UE_SPORT:
|
|
|
|
self.ue_sport_rows(rows, ue, title_bg)
|
|
|
|
else:
|
|
|
|
self.ue_std_rows(rows, ue, title_bg)
|
|
|
|
|
|
|
|
def ue_std_rows(self, rows: list, ue: dict, title_bg: tuple):
|
|
|
|
"Lignes décrivant une UE standard dans la table de synthèse"
|
|
|
|
# 2eme ligne titre UE (bonus/malus/ects)
|
|
|
|
if "ECTS" in ue:
|
|
|
|
ects_txt = f'ECTS: {ue["ECTS"]["acquis"]:.3g} / {ue["ECTS"]["total"]:.3g}'
|
|
|
|
else:
|
|
|
|
ects_txt = ""
|
|
|
|
t = {
|
|
|
|
"titre": f"""Bonus: {ue['bonus']} - Malus: {
|
|
|
|
ue["malus"]}""",
|
|
|
|
"coef": ects_txt,
|
|
|
|
"_coef_pdf": Paragraph(f"""<para align=left>{ects_txt}</para>"""),
|
|
|
|
"_coef_colspan": 2,
|
|
|
|
"_pdf_style": [
|
|
|
|
("BACKGROUND", (0, 0), (-1, 0), title_bg),
|
|
|
|
("LINEBELOW", (0, 0), (-1, 0), self.PDF_LINEWIDTH, self.PDF_LINECOLOR),
|
|
|
|
# cadre autour du bonus/malus, gris clair
|
|
|
|
("BOX", (0, 0), (0, 0), self.PDF_LINEWIDTH, (0.7, 0.7, 0.7)),
|
|
|
|
],
|
|
|
|
}
|
|
|
|
rows.append(t)
|
|
|
|
|
|
|
|
# Liste chaque ressource puis chaque SAE
|
|
|
|
for mod_type in ("ressources", "saes"):
|
|
|
|
for mod_code, mod in ue[mod_type].items():
|
|
|
|
t = {
|
|
|
|
"titre": f"{mod_code} {self.infos[mod_type][mod_code]['titre']}",
|
|
|
|
"moyenne": Paragraph(f'<para align=right>{mod["moyenne"]}</para>'),
|
|
|
|
"coef": mod["coef"],
|
|
|
|
"_coef_pdf": Paragraph(
|
|
|
|
f"<para align=right><i>{mod['coef']}</i></para>"
|
|
|
|
),
|
|
|
|
"_pdf_style": [
|
|
|
|
(
|
|
|
|
"LINEBELOW",
|
|
|
|
(0, 0),
|
|
|
|
(-1, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
|
|
|
(0.7, 0.7, 0.7), # gris clair
|
|
|
|
)
|
|
|
|
],
|
|
|
|
}
|
|
|
|
rows.append(t)
|
|
|
|
|
|
|
|
def ue_sport_rows(self, rows: list, ue: dict, title_bg: tuple):
|
|
|
|
"Lignes décrivant l'UE bonus dans la table de synthèse"
|
|
|
|
# UE BONUS
|
|
|
|
for mod_code, mod in ue["modules"].items():
|
|
|
|
rows.append(
|
|
|
|
{
|
|
|
|
"titre": f"{mod_code} {mod['titre']}",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
self.evaluations_rows(rows, mod["evaluations"])
|
|
|
|
|
2022-03-10 00:50:36 +01:00
|
|
|
def but_table_ressources(self):
|
|
|
|
"""La table de synthèse; pour chaque ressources, note et liste d'évaluations
|
|
|
|
Renvoie: colkeys, P, pdf_style, colWidths
|
|
|
|
"""
|
|
|
|
return self.bul_table_modules(
|
|
|
|
mod_type="ressources", title="Ressources", title_bg=(248, 200, 68)
|
|
|
|
)
|
|
|
|
|
|
|
|
def but_table_saes(self):
|
|
|
|
"table des SAEs"
|
|
|
|
return self.bul_table_modules(
|
|
|
|
mod_type="saes",
|
|
|
|
title="Situations d'apprentissage et d'évaluation",
|
|
|
|
title_bg=(198, 255, 171),
|
|
|
|
)
|
|
|
|
|
|
|
|
def bul_table_modules(self, mod_type=None, title="", title_bg=(248, 200, 68)):
|
|
|
|
"""Table ressources ou SAEs
|
|
|
|
- colkeys: nom des colonnes de la table (clés)
|
|
|
|
- P : table (liste de dicts de chaines de caracteres)
|
|
|
|
- pdf_style : commandes table Platypus
|
|
|
|
- largeurs de colonnes pour PDF
|
|
|
|
"""
|
2022-03-10 20:44:01 +01:00
|
|
|
# UE à utiliser pour les poids (# colonne/UE)
|
2022-03-16 18:51:22 +01:00
|
|
|
ue_infos = self.infos["ues"]
|
|
|
|
ue_acros = list(
|
|
|
|
[k for k in ue_infos if ue_infos[k]["type"] != UE_SPORT]
|
|
|
|
) # ['RT1.1', 'RT2.1', 'RT3.1']
|
2022-03-10 20:44:01 +01:00
|
|
|
# Colonnes à afficher:
|
|
|
|
col_keys = ["titre"] + ue_acros + ["coef", "moyenne"]
|
|
|
|
# Largeurs des colonnes:
|
2022-03-10 00:50:36 +01:00
|
|
|
col_widths = {
|
|
|
|
"titre": None,
|
2022-03-10 20:44:01 +01:00
|
|
|
# "poids": None,
|
2022-03-10 00:50:36 +01:00
|
|
|
"moyenne": 2 * cm,
|
|
|
|
"coef": 2 * cm,
|
|
|
|
}
|
2022-03-10 20:44:01 +01:00
|
|
|
for ue_acro in ue_acros:
|
|
|
|
col_widths[ue_acro] = 12 * mm # largeur col. poids
|
|
|
|
|
2022-03-10 00:50:36 +01:00
|
|
|
title_bg = tuple(x / 255.0 for x in title_bg)
|
|
|
|
# elems pour générer table avec gen_table (liste de dicts)
|
2022-03-10 20:44:01 +01:00
|
|
|
# Ligne de titres
|
|
|
|
t = {
|
|
|
|
"titre": title,
|
|
|
|
# "_titre_colspan": 1 + len(ue_acros),
|
|
|
|
"moyenne": "Note/20",
|
|
|
|
"coef": "Coef.",
|
|
|
|
"_coef_pdf": Paragraph("<para align=right><i>Coef.</i></para>"),
|
|
|
|
"_css_row_class": "note_bold",
|
|
|
|
"_pdf_row_markup": ["b"],
|
|
|
|
"_pdf_style": [
|
|
|
|
("BACKGROUND", (0, 0), (-1, 0), title_bg),
|
|
|
|
("BOTTOMPADDING", (0, 0), (-1, 0), 7),
|
|
|
|
(
|
|
|
|
"LINEBELOW",
|
|
|
|
(0, 0),
|
|
|
|
(-1, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
|
|
|
blue,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
}
|
|
|
|
for ue_acro in ue_acros:
|
|
|
|
t[ue_acro] = Paragraph(
|
2022-03-12 09:40:48 +01:00
|
|
|
f"<para align=right fontSize={self.small_fontsize}><i>{ue_acro}</i></para>"
|
2022-03-10 20:44:01 +01:00
|
|
|
)
|
|
|
|
rows = [t]
|
2022-03-10 00:50:36 +01:00
|
|
|
for mod_code, mod in self.infos[mod_type].items():
|
|
|
|
# 1er ligne titre module
|
|
|
|
t = {
|
|
|
|
"titre": f"{mod_code} - {mod['titre']}",
|
2022-03-10 20:44:01 +01:00
|
|
|
"_titre_colspan": 2 + len(ue_acros),
|
2022-03-10 00:50:36 +01:00
|
|
|
"_css_row_class": "note_bold",
|
|
|
|
"_pdf_row_markup": ["b"],
|
|
|
|
"_pdf_style": [
|
|
|
|
(
|
|
|
|
"LINEABOVE",
|
|
|
|
(0, 0),
|
|
|
|
(-1, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
2022-02-21 19:25:38 +01:00
|
|
|
self.PDF_LINECOLOR,
|
2022-03-10 00:50:36 +01:00
|
|
|
),
|
|
|
|
("BACKGROUND", (0, 0), (-1, 0), title_bg),
|
|
|
|
("BOTTOMPADDING", (0, 0), (-1, 0), 7),
|
2022-02-21 19:25:38 +01:00
|
|
|
],
|
|
|
|
}
|
2022-03-10 00:50:36 +01:00
|
|
|
rows.append(t)
|
|
|
|
# Evaluations:
|
2022-03-16 18:51:22 +01:00
|
|
|
self.evaluations_rows(rows, mod["evaluations"], ue_acros)
|
|
|
|
|
2022-02-21 19:25:38 +01:00
|
|
|
# Global pdf style commands:
|
|
|
|
pdf_style = [
|
|
|
|
("VALIGN", (0, 0), (-1, -1), "TOP"),
|
|
|
|
("BOX", (0, 0), (-1, -1), 0.4, blue), # ajoute cadre extérieur bleu:
|
|
|
|
]
|
2022-03-10 00:50:36 +01:00
|
|
|
return col_keys, rows, pdf_style, col_widths
|
2022-03-16 18:51:22 +01:00
|
|
|
|
|
|
|
def evaluations_rows(self, rows, evaluations, ue_acros=()):
|
|
|
|
"lignes des évaluations"
|
|
|
|
for e in evaluations:
|
|
|
|
t = {
|
|
|
|
"titre": f"{e['description']}",
|
|
|
|
"moyenne": e["note"]["value"],
|
|
|
|
"coef": e["coef"],
|
|
|
|
"_coef_pdf": Paragraph(
|
|
|
|
f"<para align=right fontSize={self.small_fontsize}><i>{e['coef']}</i></para>"
|
|
|
|
),
|
|
|
|
"_pdf_style": [
|
|
|
|
(
|
|
|
|
"LINEBELOW",
|
|
|
|
(0, 0),
|
|
|
|
(-1, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
|
|
|
(0.7, 0.7, 0.7), # gris clair
|
|
|
|
)
|
|
|
|
],
|
|
|
|
}
|
|
|
|
col_idx = 1 # 1ere col. poids
|
|
|
|
for ue_acro in ue_acros:
|
|
|
|
t[ue_acro] = Paragraph(
|
|
|
|
f"""<para align=right fontSize={self.small_fontsize}><i>{e["poids"].get(ue_acro, "") or ""}</i></para>"""
|
|
|
|
)
|
|
|
|
t["_pdf_style"].append(
|
|
|
|
(
|
|
|
|
"BOX",
|
|
|
|
(col_idx, 0),
|
|
|
|
(col_idx, 0),
|
|
|
|
self.PDF_LINEWIDTH,
|
|
|
|
(0.7, 0.7, 0.7), # gris clair
|
|
|
|
),
|
|
|
|
)
|
|
|
|
col_idx += 1
|
|
|
|
rows.append(t)
|