2020-09-26 16:19:37 +02:00
|
|
|
# -*- mode: python -*-
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
##############################################################################
|
|
|
|
#
|
|
|
|
# Gestion scolarite IUT
|
|
|
|
#
|
2023-12-31 23:04:06 +01:00
|
|
|
# Copyright (c) 1999 - 2024 Emmanuel Viennet. All rights reserved.
|
2020-09-26 16:19:37 +02:00
|
|
|
#
|
|
|
|
# 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@viennet.net
|
|
|
|
#
|
|
|
|
##############################################################################
|
|
|
|
|
|
|
|
"""ScoDoc: génération feuille émargement et placement
|
|
|
|
|
2021-09-18 14:03:36 +02:00
|
|
|
Contribution J.-M. Place 2021
|
|
|
|
basée sur une idée de M. Salomon, UFC / IUT DE BELFORT-MONTBÉLIARD, 2016
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
"""
|
2021-02-03 22:00:41 +01:00
|
|
|
import random
|
2021-08-21 00:24:51 +02:00
|
|
|
import time
|
2021-09-05 14:50:35 +02:00
|
|
|
from copy import copy
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-09-05 14:50:35 +02:00
|
|
|
import wtforms.validators
|
2021-09-17 09:44:47 +02:00
|
|
|
from flask import request, render_template
|
2021-09-18 13:42:19 +02:00
|
|
|
from flask_login import current_user
|
2021-09-05 14:50:35 +02:00
|
|
|
from flask_wtf import FlaskForm
|
2021-09-17 09:44:47 +02:00
|
|
|
from openpyxl.styles import PatternFill, Alignment, Border, Side, Font
|
2021-09-05 14:50:35 +02:00
|
|
|
from wtforms import (
|
|
|
|
StringField,
|
|
|
|
SubmitField,
|
|
|
|
SelectField,
|
|
|
|
RadioField,
|
|
|
|
HiddenField,
|
|
|
|
SelectMultipleField,
|
|
|
|
)
|
2024-10-20 11:02:08 +02:00
|
|
|
from app.models import Evaluation, Identite, ModuleImpl
|
2021-06-19 23:21:37 +02:00
|
|
|
import app.scodoc.sco_utils as scu
|
2024-08-24 14:39:19 +02:00
|
|
|
from app.scodoc import sco_preferences
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc import sco_evaluations
|
|
|
|
from app.scodoc import sco_excel
|
2021-09-17 09:44:47 +02:00
|
|
|
from app.scodoc.sco_excel import ScoExcelBook, COLORS
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc import sco_groups
|
2021-09-17 09:44:47 +02:00
|
|
|
from app.scodoc.gen_tables import GenTable
|
2021-08-21 17:07:44 +02:00
|
|
|
import sco_version
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
COORD = "Coordonnées"
|
|
|
|
SEQ = "Continue"
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
TOUS = "Tous"
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
def _get_group_info(evaluation_id):
|
|
|
|
# groupes
|
2021-09-17 15:59:47 +02:00
|
|
|
groups = sco_groups.do_evaluation_listegroupes(evaluation_id, include_default=True)
|
2021-09-17 12:18:55 +02:00
|
|
|
has_groups = False
|
|
|
|
groups_tree = {}
|
|
|
|
for group in groups:
|
|
|
|
partition = group["partition_name"] or TOUS
|
|
|
|
group_id = group["group_id"]
|
|
|
|
group_name = group["group_name"] or TOUS
|
|
|
|
if partition not in groups_tree:
|
|
|
|
groups_tree[partition] = {}
|
|
|
|
groups_tree[partition][group_name] = group_id
|
2024-10-20 11:02:08 +02:00
|
|
|
has_groups = partition != TOUS
|
|
|
|
|
|
|
|
nb_groups = sum(len(pg) for pg in groups_tree.values())
|
2021-09-17 12:18:55 +02:00
|
|
|
return groups_tree, has_groups, nb_groups
|
2021-09-05 14:50:35 +02:00
|
|
|
|
2021-09-17 15:59:47 +02:00
|
|
|
|
2021-09-05 14:50:35 +02:00
|
|
|
class PlacementForm(FlaskForm):
|
2021-09-17 09:44:47 +02:00
|
|
|
"""Formulaire pour placement des étudiants en Salle"""
|
|
|
|
|
2021-09-05 14:50:35 +02:00
|
|
|
evaluation_id = HiddenField("evaluation_id")
|
|
|
|
file_format = RadioField(
|
|
|
|
"Format de fichier",
|
|
|
|
choices=["pdf", "xls"],
|
|
|
|
validators=[
|
|
|
|
wtforms.validators.DataRequired("indiquez le format du fichier attendu"),
|
|
|
|
],
|
|
|
|
)
|
2024-10-20 11:02:08 +02:00
|
|
|
surveillants = StringField("Surveillants", validators=[], description="texte libre")
|
|
|
|
batiment = StringField("Bâtiment", description="texte libre")
|
|
|
|
salle = StringField("Salle", description="texte libre")
|
2021-09-05 14:50:35 +02:00
|
|
|
nb_rangs = SelectField(
|
2021-09-18 14:21:15 +02:00
|
|
|
"nb de places en largeur",
|
|
|
|
coerce=int,
|
|
|
|
choices=[3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14],
|
|
|
|
description="largeur de la salle, en nombre de places",
|
2021-09-05 14:50:35 +02:00
|
|
|
)
|
|
|
|
etiquetage = RadioField(
|
|
|
|
"Numérotation",
|
2021-09-12 09:31:07 +02:00
|
|
|
choices=[SEQ, COORD],
|
2021-09-05 14:50:35 +02:00
|
|
|
validators=[
|
|
|
|
wtforms.validators.DataRequired("indiquez le style de numérotation"),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
groups = SelectMultipleField(
|
|
|
|
"Groupe(s)",
|
2021-09-19 08:21:08 +02:00
|
|
|
validators=[],
|
2021-09-05 14:50:35 +02:00
|
|
|
)
|
|
|
|
submit = SubmitField("OK")
|
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
def __init__(self, formdata=None, data=None):
|
|
|
|
super().__init__(formdata=formdata, data=data)
|
|
|
|
self.groups_tree = {}
|
|
|
|
self.has_groups = None
|
|
|
|
self.nb_groups = None
|
2021-09-19 08:21:08 +02:00
|
|
|
self.tous_id = None
|
2021-09-17 12:18:55 +02:00
|
|
|
self.set_evaluation_infos(data["evaluation_id"])
|
|
|
|
|
2021-09-11 18:33:55 +02:00
|
|
|
def set_evaluation_infos(self, evaluation_id):
|
2021-09-17 09:44:47 +02:00
|
|
|
"""Initialise les données du formulaire avec les données de l'évaluation."""
|
2024-02-26 17:20:36 +01:00
|
|
|
_ = Evaluation.get_evaluation(evaluation_id) # check exist ?
|
2021-09-17 15:59:47 +02:00
|
|
|
self.groups_tree, self.has_groups, self.nb_groups = _get_group_info(
|
|
|
|
evaluation_id
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
2021-09-19 08:21:08 +02:00
|
|
|
choices = []
|
2024-10-20 11:02:08 +02:00
|
|
|
for partition, groupes in self.groups_tree.items():
|
|
|
|
for groupe in groupes:
|
2021-09-19 08:21:08 +02:00
|
|
|
if (
|
|
|
|
groupe == TOUS
|
|
|
|
): # Affichage et valeur spécifique pour le groupe TOUS
|
2024-10-20 11:02:08 +02:00
|
|
|
self.tous_id = str(groupes[groupe])
|
2021-09-19 08:21:08 +02:00
|
|
|
choices.append((TOUS, TOUS))
|
|
|
|
else:
|
2021-09-17 09:44:47 +02:00
|
|
|
groupe_id = str(self.groups_tree[partition][groupe])
|
2024-10-20 11:02:08 +02:00
|
|
|
choices.append((groupe_id, f"{str(groupe)} ({partition})"))
|
2021-09-19 08:21:08 +02:00
|
|
|
self.groups.choices = choices
|
2024-10-20 11:02:08 +02:00
|
|
|
# self.groups.default = [TOUS] # Ne fonctionnne pas...
|
|
|
|
# (ni dans la déclaration de PlaceForm.groups)
|
|
|
|
# la réponse [] est de toute façon transposée en [ self.tous_id ]
|
|
|
|
# lors du traitement (cas du groupe unique)
|
2021-09-05 14:50:35 +02:00
|
|
|
|
|
|
|
|
2021-09-12 07:04:05 +02:00
|
|
|
class _DistributeurContinu:
|
|
|
|
"""Distribue les places selon un ordre numérique."""
|
|
|
|
|
2021-09-17 05:54:11 +02:00
|
|
|
def __init__(self):
|
2021-09-12 07:04:05 +02:00
|
|
|
self.position = 1
|
|
|
|
|
|
|
|
def suivant(self):
|
2021-09-17 12:18:55 +02:00
|
|
|
"""Retounre la désignation de la place suivante"""
|
2021-09-12 07:04:05 +02:00
|
|
|
retour = self.position
|
|
|
|
self.position += 1
|
|
|
|
return retour
|
|
|
|
|
|
|
|
|
|
|
|
class _Distributeur2D:
|
|
|
|
"""Distribue les places selon des coordonnées sur nb_rangs."""
|
|
|
|
|
|
|
|
def __init__(self, nb_rangs):
|
|
|
|
self.nb_rangs = nb_rangs
|
|
|
|
self.rang = 1
|
|
|
|
self.index = 1
|
|
|
|
|
|
|
|
def suivant(self):
|
2021-09-17 12:18:55 +02:00
|
|
|
"""Retounre la désignation de la place suivante"""
|
2021-09-12 07:04:05 +02:00
|
|
|
retour = (self.index, self.rang)
|
|
|
|
self.rang += 1
|
|
|
|
if self.rang > self.nb_rangs:
|
|
|
|
self.rang = 1
|
|
|
|
self.index += 1
|
|
|
|
return retour
|
|
|
|
|
|
|
|
|
2021-09-11 19:35:30 +02:00
|
|
|
def placement_eval_selectetuds(evaluation_id):
|
|
|
|
"""Creation de l'écran de placement"""
|
2021-09-05 14:50:35 +02:00
|
|
|
form = PlacementForm(
|
|
|
|
request.form,
|
2021-09-17 12:18:55 +02:00
|
|
|
data={"evaluation_id": int(evaluation_id), "groups": TOUS},
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
2021-09-05 14:50:35 +02:00
|
|
|
if form.validate_on_submit():
|
2021-09-12 09:31:07 +02:00
|
|
|
runner = PlacementRunner(form)
|
|
|
|
if not runner.check_placement():
|
|
|
|
return (
|
|
|
|
"""<h2>Génération du placement impossible pour %s</h2>
|
|
|
|
<p>(vérifiez que le semestre n'est pas verrouillé et que vous
|
|
|
|
avez l'autorisation d'effectuer cette opération)</p>
|
|
|
|
<p><a href="moduleimpl_status?moduleimpl_id=%s">Continuer</a></p>
|
|
|
|
"""
|
|
|
|
% runner.__dict__
|
|
|
|
)
|
2021-09-17 09:44:47 +02:00
|
|
|
return runner.exec_placement() # calcul et generation du fichier
|
2024-08-24 14:39:19 +02:00
|
|
|
|
|
|
|
return render_template(
|
|
|
|
"scodoc/forms/placement.j2",
|
|
|
|
evaluations_description=sco_evaluations.evaluation_describe(
|
|
|
|
evaluation_id=evaluation_id
|
|
|
|
),
|
|
|
|
form=form,
|
|
|
|
)
|
2021-09-05 14:50:35 +02:00
|
|
|
|
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
class PlacementRunner:
|
2021-09-17 10:59:26 +02:00
|
|
|
"""Execution de l'action définie par le formulaire"""
|
2021-09-17 12:18:55 +02:00
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
def __init__(self, form):
|
|
|
|
"""Calcul et génération du fichier sur la base des données du formulaire"""
|
|
|
|
self.evaluation_id = form["evaluation_id"].data
|
|
|
|
self.etiquetage = form["etiquetage"].data
|
|
|
|
self.surveillants = form["surveillants"].data
|
|
|
|
self.batiment = form["batiment"].data
|
|
|
|
self.salle = form["salle"].data
|
|
|
|
self.nb_rangs = form["nb_rangs"].data
|
|
|
|
self.file_format = form["file_format"].data
|
2021-09-19 08:21:08 +02:00
|
|
|
if len(form["groups"].data) == 0:
|
|
|
|
self.groups_ids = [form.tous_id]
|
2024-10-20 11:02:08 +02:00
|
|
|
else: # On remplace le mot-clé TOUS par l'identifiant de ce groupe
|
2021-09-19 08:21:08 +02:00
|
|
|
self.groups_ids = [
|
|
|
|
gid if gid != TOUS else form.tous_id for gid in form["groups"].data
|
|
|
|
]
|
2024-10-20 11:02:08 +02:00
|
|
|
evl = Evaluation.get_evaluation(self.evaluation_id)
|
|
|
|
self.evaluation = evl
|
2021-09-12 09:31:07 +02:00
|
|
|
self.groups = sco_groups.listgroups(self.groups_ids)
|
|
|
|
self.gr_title_filename = sco_groups.listgroups_filename(self.groups)
|
|
|
|
self.current_user = current_user
|
2024-10-20 11:02:08 +02:00
|
|
|
self.moduleimpl_id = evl.moduleimpl_id
|
|
|
|
self.moduleimpl: ModuleImpl = evl.moduleimpl
|
|
|
|
self.moduleimpl_data = self.moduleimpl.to_dict()
|
|
|
|
mod = evl.moduleimpl.module
|
|
|
|
self.module_data = mod.to_dict()
|
|
|
|
self.evalname = f"""{mod.code or "?"}-{
|
|
|
|
evl.date_debut.strftime("%Y-%m-%d_%Hh%M") if evl.date_debut else ""}"""
|
|
|
|
if evl.description:
|
2024-02-26 17:20:36 +01:00
|
|
|
self.evaltitre = self.evaluation.description
|
2021-09-12 09:31:07 +02:00
|
|
|
else:
|
2024-02-26 17:20:36 +01:00
|
|
|
self.evaltitre = f"""évaluation{
|
2024-10-20 11:02:08 +02:00
|
|
|
evl.date_debut.strftime(' du ' + scu.DATEATIME_FMT)
|
|
|
|
if evl.date_debut else ''}"""
|
2021-09-17 09:44:47 +02:00
|
|
|
self.desceval = [ # une liste de chaines: description de l'evaluation
|
2024-10-20 11:02:08 +02:00
|
|
|
evl.moduleimpl.formsemestre.titre_annee(),
|
|
|
|
f"""Module : {mod.code or "?"} - {evl.moduleimpl.module.abbrev or ""}""",
|
|
|
|
f"Surveillants : {self.surveillants}",
|
|
|
|
f"Bâtiment : {self.batiment} - Salle : {self.salle}",
|
|
|
|
f"Évaluation : {self.evaltitre} (coef. {evl.coefficient:g})",
|
2021-09-17 09:44:47 +02:00
|
|
|
]
|
2021-09-17 12:18:55 +02:00
|
|
|
self.styles = None
|
|
|
|
self.plan = None
|
|
|
|
self.listetud = None
|
2021-09-12 09:31:07 +02:00
|
|
|
|
|
|
|
def check_placement(self):
|
2021-09-17 10:59:26 +02:00
|
|
|
"""Vérifie que l'utilisateur courant a le droit d'édition sur les notes"""
|
2021-09-12 09:31:07 +02:00
|
|
|
# Check access (admin, respformation, and responsable_id)
|
2024-02-04 23:08:08 +01:00
|
|
|
return self.moduleimpl.can_edit_notes(self.current_user)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-09-17 09:44:47 +02:00
|
|
|
def exec_placement(self):
|
2021-09-17 12:18:55 +02:00
|
|
|
"""Excéute l'action liée au formulaire"""
|
2021-09-12 09:31:07 +02:00
|
|
|
self._repartition()
|
|
|
|
if self.file_format == "xls":
|
|
|
|
return self._production_xls()
|
2021-09-17 10:59:26 +02:00
|
|
|
return self._production_pdf()
|
2021-09-12 09:31:07 +02:00
|
|
|
|
|
|
|
def _repartition(self):
|
|
|
|
"""
|
|
|
|
Calcule le placement. retourne une liste de couples ((nom, prenom), position)
|
|
|
|
"""
|
|
|
|
# Construit liste des etudiants et les réparti
|
|
|
|
self.groups = sco_groups.listgroups(self.groups_ids)
|
|
|
|
self.listetud = self._build_listetud()
|
|
|
|
self.plan = self._affectation_places()
|
|
|
|
|
2024-10-20 11:02:08 +02:00
|
|
|
def _build_listetud(self) -> list[tuple[str, str, int]]:
|
|
|
|
# tous les etudiants ?
|
|
|
|
get_all_students = None in [g["group_name"] for g in self.groups]
|
2021-11-20 17:21:51 +01:00
|
|
|
etudid_etats = sco_groups.do_evaluation_listeetuds_groups(
|
2021-09-12 09:31:07 +02:00
|
|
|
self.evaluation_id,
|
|
|
|
self.groups,
|
2021-09-17 12:18:55 +02:00
|
|
|
getallstudents=get_all_students,
|
2022-01-16 23:47:52 +01:00
|
|
|
include_demdef=True,
|
2021-09-12 09:31:07 +02:00
|
|
|
)
|
2024-10-20 11:02:08 +02:00
|
|
|
listetud = [] # liste de tuples (nom, prenom, etudid)
|
2021-11-20 17:21:51 +01:00
|
|
|
for etudid, etat in etudid_etats:
|
2024-10-20 11:02:08 +02:00
|
|
|
if etat != scu.DEMISSION:
|
|
|
|
# infos identite etudiant
|
|
|
|
etud = Identite.get_etud(etudid)
|
|
|
|
nom = etud.nom.upper()
|
|
|
|
prenom = etud.prenom.lower().capitalize()
|
|
|
|
listetud.append((nom, prenom, etud.id))
|
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
random.shuffle(listetud)
|
2024-10-20 11:02:08 +02:00
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
return listetud
|
|
|
|
|
|
|
|
def _affectation_places(self):
|
|
|
|
plan = []
|
|
|
|
if self.etiquetage == SEQ:
|
|
|
|
distributeur = _DistributeurContinu()
|
2021-09-11 19:35:30 +02:00
|
|
|
else:
|
2021-09-12 09:31:07 +02:00
|
|
|
distributeur = _Distributeur2D(self.nb_rangs)
|
|
|
|
for etud in self.listetud:
|
|
|
|
plan.append((etud, distributeur.suivant()))
|
|
|
|
return plan
|
|
|
|
|
|
|
|
def _production_xls(self):
|
2024-10-20 11:02:08 +02:00
|
|
|
filename = f"placement_{self.evalname}_{self.gr_title_filename}"
|
2021-09-12 09:31:07 +02:00
|
|
|
xls = self._excel_feuille_placement()
|
2021-09-21 22:19:08 +02:00
|
|
|
return scu.send_file(xls, filename, scu.XLSX_SUFFIX, mime=scu.XLSX_MIMETYPE)
|
2021-09-12 09:31:07 +02:00
|
|
|
|
|
|
|
def _production_pdf(self):
|
2022-10-02 23:43:29 +02:00
|
|
|
pdf_title = "<br>".join(self.desceval)
|
2024-04-02 23:37:23 +02:00
|
|
|
pdf_title += f"""\nDate : {self.evaluation.date_debut.strftime(scu.DATE_FMT)
|
2024-02-26 17:20:36 +01:00
|
|
|
if self.evaluation.date_debut else '-'
|
|
|
|
} - Horaire : {self.evaluation.heure_debut()} à {self.evaluation.heure_fin()
|
|
|
|
}"""
|
2024-10-20 11:02:08 +02:00
|
|
|
filename = f"placement_{self.evalname}_{self.gr_title_filename}"
|
2020-09-26 16:19:37 +02:00
|
|
|
titles = {
|
|
|
|
"nom": "Nom",
|
2024-10-20 11:02:08 +02:00
|
|
|
"prenom": "Prénom",
|
2020-09-26 16:19:37 +02:00
|
|
|
"colonne": "Colonne",
|
|
|
|
"ligne": "Ligne",
|
|
|
|
"place": "Place",
|
|
|
|
}
|
2021-09-12 09:31:07 +02:00
|
|
|
if self.etiquetage == COORD:
|
2020-09-26 16:19:37 +02:00
|
|
|
columns_ids = ["nom", "prenom", "colonne", "ligne"]
|
|
|
|
else:
|
|
|
|
columns_ids = ["nom", "prenom", "place"]
|
|
|
|
|
|
|
|
rows = []
|
2021-09-13 07:16:37 +02:00
|
|
|
for etud in sorted(self.plan, key=lambda item: item[0][0]): # sort by name
|
2021-09-12 09:31:07 +02:00
|
|
|
if self.etiquetage == COORD:
|
2020-09-26 16:19:37 +02:00
|
|
|
rows.append(
|
|
|
|
{
|
2021-09-12 09:31:07 +02:00
|
|
|
"nom": etud[0][0],
|
|
|
|
"prenom": etud[0][1],
|
|
|
|
"colonne": etud[1][0],
|
|
|
|
"ligne": etud[1][1],
|
2020-09-26 16:19:37 +02:00
|
|
|
}
|
|
|
|
)
|
|
|
|
else:
|
2021-09-12 09:31:07 +02:00
|
|
|
rows.append({"nom": etud[0][0], "prenom": etud[0][1], "place": etud[1]})
|
2020-09-26 16:19:37 +02:00
|
|
|
tab = GenTable(
|
|
|
|
titles=titles,
|
|
|
|
columns_ids=columns_ids,
|
|
|
|
rows=rows,
|
|
|
|
filename=filename,
|
2024-10-20 11:02:08 +02:00
|
|
|
origin=f"Généré par {sco_version.SCONAME} le {scu.timedate_human_repr()}",
|
2020-09-26 16:19:37 +02:00
|
|
|
pdf_title=pdf_title,
|
|
|
|
# pdf_shorttitle = '',
|
2021-09-17 09:44:47 +02:00
|
|
|
preferences=sco_preferences.SemPreferences(
|
|
|
|
self.moduleimpl_data["formsemestre_id"]
|
|
|
|
),
|
2024-05-22 00:06:30 +02:00
|
|
|
table_id="placement_pdf",
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
2023-09-21 10:20:19 +02:00
|
|
|
return tab.make_page(fmt="pdf", with_html_headers=False)
|
2021-09-12 09:31:07 +02:00
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
def _one_header(self, worksheet):
|
2021-09-12 09:31:07 +02:00
|
|
|
cells = [
|
2021-09-17 12:18:55 +02:00
|
|
|
worksheet.make_cell("Nom", self.styles["2bi"]),
|
|
|
|
worksheet.make_cell("Prénom", self.styles["2bi"]),
|
2021-09-12 09:31:07 +02:00
|
|
|
]
|
|
|
|
if self.etiquetage == COORD:
|
2021-09-17 12:18:55 +02:00
|
|
|
cells.append(worksheet.make_cell("Colonne", self.styles["2bi"]))
|
|
|
|
cells.append(worksheet.make_cell("Ligne", self.styles["2bi"]))
|
2020-09-26 16:19:37 +02:00
|
|
|
else:
|
2021-09-17 12:18:55 +02:00
|
|
|
cells.append(worksheet.make_cell("Place", self.styles["2bi"]))
|
2021-09-12 09:31:07 +02:00
|
|
|
return cells
|
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
def _headers(self, worksheet, nb_listes):
|
2021-09-12 09:31:07 +02:00
|
|
|
cells = []
|
|
|
|
for _ in range(nb_listes):
|
2021-09-17 12:18:55 +02:00
|
|
|
cells += self._one_header(worksheet)
|
|
|
|
cells.append(worksheet.make_cell(""))
|
|
|
|
worksheet.append_row(cells)
|
2021-09-12 09:31:07 +02:00
|
|
|
|
|
|
|
def _make_styles(self, ws0, ws1):
|
|
|
|
# polices
|
|
|
|
font0 = Font(name="Calibri", bold=True, size=12)
|
|
|
|
font1b = copy(font0)
|
|
|
|
font1b.size = 9
|
|
|
|
font1i = Font(name="Arial", italic=True, size=10)
|
|
|
|
font1o = Font(name="Arial", outline=True, size=10)
|
|
|
|
font2bi = Font(name="Arial", bold=True, italic=True, size=8)
|
|
|
|
font2 = Font(name="Arial", size=10)
|
|
|
|
|
|
|
|
# bordures
|
|
|
|
side_double = Side(border_style="double", color=COLORS.BLACK.value)
|
|
|
|
side_thin = Side(border_style="thin", color=COLORS.BLACK.value)
|
|
|
|
|
|
|
|
# bordures
|
|
|
|
border1t = Border(left=side_double, top=side_double, right=side_double)
|
|
|
|
border1bb = Border(left=side_double, bottom=side_double, right=side_double)
|
|
|
|
border1bm = Border(left=side_double, right=side_double)
|
|
|
|
border1m = Border(left=side_double, bottom=side_thin, right=side_double)
|
|
|
|
border2m = Border(top=side_thin, bottom=side_thin)
|
|
|
|
border2r = Border(top=side_thin, bottom=side_thin, right=side_thin)
|
|
|
|
border2l = Border(left=side_thin, top=side_thin, bottom=side_thin)
|
|
|
|
border2b = Border(
|
|
|
|
left=side_thin, top=side_thin, bottom=side_thin, right=side_thin
|
|
|
|
)
|
|
|
|
|
|
|
|
# alignements
|
|
|
|
align_center_center = Alignment(horizontal="center", vertical="center")
|
|
|
|
align_right_bottom = Alignment(horizontal="right", vertical="bottom")
|
|
|
|
align_left_center = Alignment(horizontal="left", vertical="center")
|
|
|
|
align_right_center = Alignment(horizontal="right", vertical="center")
|
|
|
|
|
|
|
|
# patterns
|
|
|
|
pattern = PatternFill(
|
|
|
|
fill_type="solid", fgColor=sco_excel.COLORS.LIGHT_YELLOW.value
|
|
|
|
)
|
|
|
|
|
|
|
|
# styles
|
|
|
|
self.styles = {
|
|
|
|
"titres": sco_excel.excel_make_style(font_name="Arial", bold=True, size=12),
|
|
|
|
"1t": ws0.excel_make_composite_style(
|
|
|
|
font=font0, alignment=align_center_center, border=border1t
|
|
|
|
),
|
|
|
|
"1m": ws0.excel_make_composite_style(
|
|
|
|
font=font1b, alignment=align_center_center, border=border1m
|
|
|
|
),
|
|
|
|
"1bm": ws0.excel_make_composite_style(
|
|
|
|
font=font1b, alignment=align_center_center, border=border1bm
|
|
|
|
),
|
|
|
|
"1bb": ws0.excel_make_composite_style(
|
|
|
|
font=font1o, alignment=align_right_bottom, border=border1bb
|
|
|
|
),
|
|
|
|
"2b": ws1.excel_make_composite_style(
|
|
|
|
font=font1i, alignment=align_center_center, border=border2b
|
|
|
|
),
|
|
|
|
"2bi": ws1.excel_make_composite_style(
|
|
|
|
font=font2bi,
|
|
|
|
alignment=align_center_center,
|
|
|
|
border=border2b,
|
|
|
|
fill=pattern,
|
|
|
|
),
|
|
|
|
"2l": ws1.excel_make_composite_style(
|
|
|
|
font=font2, alignment=align_left_center, border=border2l
|
|
|
|
),
|
|
|
|
"2m1": ws1.excel_make_composite_style(
|
|
|
|
font=font2, alignment=align_left_center, border=border2m
|
|
|
|
),
|
|
|
|
"2m2": ws1.excel_make_composite_style(
|
|
|
|
font=font2, alignment=align_right_center, border=border2m
|
|
|
|
),
|
|
|
|
"2r": ws1.excel_make_composite_style(
|
|
|
|
font=font2, alignment=align_right_center, border=border2r
|
|
|
|
),
|
|
|
|
}
|
|
|
|
|
2021-09-17 10:59:26 +02:00
|
|
|
def _titres(self, worksheet):
|
2024-10-20 11:02:08 +02:00
|
|
|
date_time = time.strftime(scu.DATEATIME_FMT)
|
2021-09-17 10:59:26 +02:00
|
|
|
worksheet.append_single_cell_row(
|
2024-10-20 11:02:08 +02:00
|
|
|
f"Feuille placement etudiants éditée le {date_time}", self.styles["titres"]
|
2021-09-12 09:31:07 +02:00
|
|
|
)
|
|
|
|
for line, desceval in enumerate(self.desceval):
|
|
|
|
if line in [1, 4, 7]:
|
2021-09-17 10:59:26 +02:00
|
|
|
worksheet.append_blank_row()
|
|
|
|
worksheet.append_single_cell_row(desceval, self.styles["titres"])
|
|
|
|
worksheet.append_single_cell_row(
|
2024-04-02 23:37:23 +02:00
|
|
|
f"""Date : {self.evaluation.date_debut.strftime(scu.DATE_FMT)
|
2024-02-26 17:20:36 +01:00
|
|
|
if self.evaluation.date_debut else '-'
|
|
|
|
} - Horaire : {self.evaluation.heure_debut()} à {self.evaluation.heure_fin()
|
|
|
|
}""",
|
2021-09-12 09:31:07 +02:00
|
|
|
self.styles["titres"],
|
2021-09-05 14:50:35 +02:00
|
|
|
)
|
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
def _feuille0(self, ws0, space):
|
|
|
|
self._titres(ws0)
|
|
|
|
# entetes colonnes - feuille0
|
|
|
|
cells = [ws0.make_cell()]
|
|
|
|
for col in range(self.nb_rangs):
|
2024-10-20 11:02:08 +02:00
|
|
|
cells.append(ws0.make_cell(f"colonne {col + 1}", self.styles["2b"]))
|
2021-09-12 09:31:07 +02:00
|
|
|
ws0.append_row(cells)
|
|
|
|
|
|
|
|
# etudiants - feuille0
|
|
|
|
place = 1
|
2021-09-17 05:54:11 +02:00
|
|
|
col = 0
|
|
|
|
rang = 1
|
|
|
|
# Chaque rang est affiché sur 3 lignes xlsx (notées A, B, C)
|
|
|
|
# ligne A: le nom, ligne B: le prénom, ligne C: un espace ou la place
|
|
|
|
cells_a = [ws0.make_cell(rang, self.styles["2b"])]
|
|
|
|
cells_b = [ws0.make_cell("", self.styles["2b"])]
|
|
|
|
cells_c = [ws0.make_cell("", self.styles["2b"])]
|
|
|
|
row = 13 # première ligne de signature
|
2021-09-19 09:21:37 +02:00
|
|
|
rang += 1
|
2021-09-17 05:54:11 +02:00
|
|
|
for linetud in self.plan:
|
|
|
|
cells_a.append(ws0.make_cell(linetud[0][0], self.styles["1t"])) # nom
|
|
|
|
cells_b.append(ws0.make_cell(linetud[0][1], self.styles["1m"])) # prenom
|
|
|
|
if self.etiquetage == COORD:
|
|
|
|
cell_c = ws0.make_cell("", self.styles["1bb"])
|
2020-09-26 16:19:37 +02:00
|
|
|
else:
|
2024-10-20 11:02:08 +02:00
|
|
|
cell_c = ws0.make_cell(f"place {place}", self.styles["1bb"])
|
2021-09-12 09:31:07 +02:00
|
|
|
place = place + 1
|
2021-09-17 05:54:11 +02:00
|
|
|
cells_c.append(cell_c)
|
2021-09-12 09:31:07 +02:00
|
|
|
ws0.set_row_dimension_height(row, space / 25)
|
2021-09-17 05:54:11 +02:00
|
|
|
row += 3
|
|
|
|
col += 1
|
|
|
|
if col == self.nb_rangs: # On a fini la rangée courante
|
2021-09-17 09:44:47 +02:00
|
|
|
ws0.append_row(cells_a) # on affiche les 3 lignes construites
|
2021-09-17 05:54:11 +02:00
|
|
|
ws0.append_row(cells_b)
|
|
|
|
ws0.append_row(cells_c)
|
2021-09-17 09:44:47 +02:00
|
|
|
cells_a = [
|
|
|
|
ws0.make_cell(rang, self.styles["2b"])
|
|
|
|
] # on réinitialise les 3 lignes
|
2021-09-17 05:54:11 +02:00
|
|
|
cells_b = [ws0.make_cell("", self.styles["2b"])]
|
|
|
|
cells_c = [ws0.make_cell("", self.styles["2b"])]
|
|
|
|
col = 0
|
|
|
|
rang += 1
|
|
|
|
# publication du rang final incomplet
|
|
|
|
ws0.append_row(cells_a) # Affiche des 3 lignes (dernières lignes incomplètes)
|
|
|
|
ws0.append_row(cells_b)
|
|
|
|
ws0.append_row(cells_c)
|
|
|
|
ws0.set_row_dimension_height(row, space / 25)
|
2021-09-12 09:31:07 +02:00
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
def _feuille1(self, worksheet, maxlines):
|
2021-09-12 09:31:07 +02:00
|
|
|
# etudiants - feuille1
|
|
|
|
# structuration:
|
|
|
|
# 1 page = maxlistes listes
|
|
|
|
# 1 liste = 3 ou 4 colonnes(excel) (selon numbering) et (maximum maxlines) lignes
|
|
|
|
maxlistes = 2 # nombre de listes par page
|
|
|
|
# computes excel columns widths
|
|
|
|
if self.etiquetage == COORD:
|
|
|
|
gabarit = [16, 18, 6, 6, 2]
|
|
|
|
else:
|
|
|
|
gabarit = [16, 18, 12, 2]
|
|
|
|
widths = []
|
|
|
|
for _ in range(maxlistes):
|
|
|
|
widths += gabarit
|
2021-09-17 12:18:55 +02:00
|
|
|
worksheet.set_column_dimension_width(value=widths)
|
2021-09-12 09:31:07 +02:00
|
|
|
nb_etu_restant = len(self.listetud)
|
2021-09-17 12:18:55 +02:00
|
|
|
self._titres(worksheet)
|
2021-09-12 09:31:07 +02:00
|
|
|
nb_listes = min(
|
|
|
|
maxlistes, nb_etu_restant // maxlines + 1
|
|
|
|
) # nombre de colonnes dans la page
|
2021-09-17 12:18:55 +02:00
|
|
|
self._headers(worksheet, nb_listes)
|
2021-09-12 09:31:07 +02:00
|
|
|
# construction liste alphabétique
|
|
|
|
# Affichage
|
2021-09-17 09:44:47 +02:00
|
|
|
lines = [[] for _ in range(maxlines)]
|
|
|
|
lineno = 0
|
2021-09-12 09:31:07 +02:00
|
|
|
col = 0
|
2021-09-17 05:54:11 +02:00
|
|
|
for etud in sorted(self.plan, key=lambda e: e[0][0]): # tri alphabétique
|
2021-09-12 09:31:07 +02:00
|
|
|
# check for skip of list or page
|
|
|
|
if col > 0: # add a empty cell between lists
|
2021-09-17 12:18:55 +02:00
|
|
|
lines[lineno].append(worksheet.make_cell())
|
|
|
|
lines[lineno].append(worksheet.make_cell(etud[0][0], self.styles["2l"]))
|
|
|
|
lines[lineno].append(worksheet.make_cell(etud[0][1], self.styles["2m1"]))
|
2021-09-12 09:31:07 +02:00
|
|
|
if self.etiquetage == COORD:
|
2021-09-17 12:18:55 +02:00
|
|
|
lines[lineno].append(
|
|
|
|
worksheet.make_cell(etud[1][1], self.styles["2m2"])
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
2021-09-17 12:18:55 +02:00
|
|
|
lines[lineno].append(worksheet.make_cell(etud[1][0], self.styles["2r"]))
|
2021-09-12 09:31:07 +02:00
|
|
|
else:
|
2021-09-17 12:18:55 +02:00
|
|
|
lines[lineno].append(worksheet.make_cell(etud[1], self.styles["2r"]))
|
2021-09-17 09:44:47 +02:00
|
|
|
lineno = lineno + 1
|
|
|
|
if lineno >= maxlines: # fin de liste
|
2021-09-12 09:31:07 +02:00
|
|
|
col = col + 1
|
2021-09-17 09:44:47 +02:00
|
|
|
lineno = 0
|
2021-09-12 09:31:07 +02:00
|
|
|
if col >= maxlistes: # fin de page
|
2021-09-17 09:44:47 +02:00
|
|
|
for line_cells in lines:
|
2021-09-17 12:18:55 +02:00
|
|
|
worksheet.append_row(line_cells)
|
2021-09-17 09:44:47 +02:00
|
|
|
lines = [[] for _ in range(maxlines)]
|
2021-09-12 09:31:07 +02:00
|
|
|
col = 0
|
2021-09-17 12:18:55 +02:00
|
|
|
worksheet.append_blank_row()
|
2021-09-12 09:31:07 +02:00
|
|
|
nb_etu_restant -= maxlistes * maxlines
|
|
|
|
nb_listes = min(
|
|
|
|
maxlistes, nb_etu_restant // maxlines + 1
|
|
|
|
) # nombre de colonnes dans la page
|
2021-09-17 12:18:55 +02:00
|
|
|
self._headers(worksheet, nb_listes)
|
2021-09-17 09:44:47 +02:00
|
|
|
for line_cells in lines:
|
2021-09-17 12:18:55 +02:00
|
|
|
worksheet.append_row(line_cells)
|
2021-09-12 09:31:07 +02:00
|
|
|
|
|
|
|
def _excel_feuille_placement(self):
|
|
|
|
"""Genere feuille excel pour placement des etudiants.
|
|
|
|
E: evaluation (dict)
|
|
|
|
lines: liste de tuples
|
|
|
|
(etudid, nom, prenom, etat, groupe, val, explanation)
|
|
|
|
"""
|
|
|
|
sem_preferences = sco_preferences.SemPreferences()
|
|
|
|
space = sem_preferences.get("feuille_placement_emargement")
|
|
|
|
maxlines = sem_preferences.get("feuille_placement_positions")
|
|
|
|
nb_rangs = int(self.nb_rangs)
|
|
|
|
column_width_ratio = (
|
|
|
|
1 / 250
|
|
|
|
) # changement d unités entre pyExcelerator et openpyxl
|
|
|
|
|
2021-09-17 10:59:26 +02:00
|
|
|
workbook = ScoExcelBook()
|
2021-09-12 09:31:07 +02:00
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
sheet_name_0 = "Emargement"
|
|
|
|
ws0 = workbook.create_sheet(sheet_name_0)
|
2021-09-12 09:31:07 +02:00
|
|
|
# ajuste largeurs colonnes (unite inconnue, empirique)
|
|
|
|
width = 4500 * column_width_ratio
|
|
|
|
if nb_rangs > 5:
|
|
|
|
width = 22500 * column_width_ratio // nb_rangs
|
|
|
|
|
|
|
|
ws0.set_column_dimension_width("A", 750 * column_width_ratio)
|
|
|
|
for col in range(nb_rangs):
|
|
|
|
ws0.set_column_dimension_width(
|
2021-09-17 10:26:20 +02:00
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"[col + 1 : col + 2], width
|
2021-09-12 09:31:07 +02:00
|
|
|
)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-09-17 12:18:55 +02:00
|
|
|
sheet_name_1 = "Positions"
|
|
|
|
ws1 = workbook.create_sheet(sheet_name_1)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-09-12 09:31:07 +02:00
|
|
|
self._make_styles(ws0, ws1)
|
|
|
|
self._feuille0(ws0, space)
|
|
|
|
self._feuille1(ws1, maxlines)
|
2021-09-17 10:59:26 +02:00
|
|
|
return workbook.generate()
|