# -*- mode: python -*-
# -*- coding: utf-8 -*-
##############################################################################
#
# Gestion scolarite IUT
#
# Copyright (c) 1999 - 2021 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@viennet.net
#
##############################################################################
"""Ajout/Modification/Supression UE
"""
import notesdb as ndb
import sco_utils as scu
from notes_log import log
from TrivialFormulator import TrivialFormulator, TF
from gen_tables import GenTable
import sco_groups
import sco_formsemestre
import sco_formsemestre_validation
import sco_codes_parcours
import sco_tag_module
from sco_permissions import ScoChangeFormation, ScoEditFormationTags, ScoImplement
from sco_exceptions import ScoValueError, ScoLockedFormError
def ue_create(context, formation_id=None, REQUEST=None):
"""Creation d'une UE"""
return ue_edit(context, create=True, formation_id=formation_id, REQUEST=REQUEST)
def ue_edit(context, ue_id=None, create=False, formation_id=None, REQUEST=None):
"""Modification ou creation d'une UE"""
create = int(create)
if not create:
U = context.do_ue_list(args={"ue_id": ue_id})
if not U:
raise ScoValueError("UE inexistante !")
U = U[0]
formation_id = U["formation_id"]
title = "Modification de l'UE %(titre)s" % U
initvalues = U
submitlabel = "Modifier les valeurs"
else:
title = "Création d'une UE"
initvalues = {}
submitlabel = "Créer cette UE"
Fol = context.formation_list(args={"formation_id": formation_id})
if not Fol:
raise ScoValueError(
"Formation %s inexistante ! (si vous avez suivi un lien valide, merci de signaler le problème)"
% formation_id
)
Fo = Fol[0]
parcours = sco_codes_parcours.get_parcours_from_code(Fo["type_parcours"])
H = [
context.sco_header(REQUEST, page_title=title, javascripts=["js/edit_ue.js"]),
"
" + title,
" (formation %(acronyme)s, version %(version)s)
" % Fo,
"""
Les UE sont des groupes de modules dans une formation donnée, utilisés pour l'évaluation (on calcule des moyennes par UE et applique des seuils ("barres")).
Note: L'UE n'a pas de coefficient associé. Seuls les modules ont des coefficients.
""",
]
ue_types = parcours.ALLOWED_UE_TYPES
ue_types.sort()
ue_types_names = [sco_codes_parcours.UE_TYPE_NAME[k] for k in ue_types]
ue_types = [str(x) for x in ue_types]
fw = [
("ue_id", {"input_type": "hidden"}),
("create", {"input_type": "hidden", "default": create}),
("formation_id", {"input_type": "hidden", "default": formation_id}),
("titre", {"size": 30, "explanation": "nom de l'UE"}),
("acronyme", {"size": 8, "explanation": "abbréviation", "allow_null": False}),
(
"numero",
{
"size": 2,
"explanation": "numéro (1,2,3,4) de l'UE pour l'ordre d'affichage",
"type": "int",
},
),
(
"type",
{
"explanation": "type d'UE",
"input_type": "menu",
"allowed_values": ue_types,
"labels": ue_types_names,
},
),
(
"ects",
{
"size": 4,
"type": "float",
"title": "ECTS",
"explanation": "nombre de crédits ECTS",
},
),
(
"coefficient",
{
"size": 4,
"type": "float",
"title": "Coefficient",
"explanation": """les coefficients d'UE ne sont utilisés que lorsque
l'option Utiliser les coefficients d'UE pour calculer la moyenne générale
est activée. Par défaut, le coefficient d'une UE est simplement la somme des
coefficients des modules dans lesquels l'étudiant a des notes.
""",
},
),
(
"ue_code",
{
"size": 12,
"title": "Code UE",
"explanation": "code interne (optionnel). Toutes les UE partageant le même code (et le même code de formation) sont compatibles (compensation de semestres, capitalisation d'UE). Voir liste ci-dessous.",
},
),
(
"code_apogee",
{
"title": "Code Apogée",
"size": 15,
"explanation": "code élément pédagogique Apogée (optionnel) ou liste de codes ELP séparés par une virgule",
},
),
]
if parcours.UE_IS_MODULE:
# demande le semestre pour creer le module immediatement:
semestres_indices = range(1, parcours.NB_SEM + 1)
fw.append(
(
"semestre_id",
{
"input_type": "menu",
"type": "int",
"title": scu.strcapitalize(parcours.SESSION_NAME),
"explanation": "%s de début du module dans la formation"
% parcours.SESSION_NAME,
"labels": [str(x) for x in semestres_indices],
"allowed_values": semestres_indices,
},
)
)
if create and not parcours.UE_IS_MODULE:
fw.append(
(
"create_matiere",
{
"input_type": "boolcheckbox",
"default": False,
"title": "Créer matière identique",
"explanation": "créer immédiatement une matière dans cette UE (utile si on n'utilise pas de matières)",
},
)
)
tf = TrivialFormulator(
REQUEST.URL0, REQUEST.form, fw, initvalues=initvalues, submitlabel=submitlabel
)
if tf[0] == 0:
X = """
"""
return "\n".join(H) + tf[1] + X + context.sco_footer(REQUEST)
else:
if create:
if not tf[2]["ue_code"]:
del tf[2]["ue_code"]
if not tf[2]["numero"]:
if not "semestre_id" in tf[2]:
tf[2]["semestre_id"] = 0
# numero regroupant par semestre ou année:
tf[2]["numero"] = next_ue_numero(
context, formation_id, int(tf[2]["semestre_id"] or 0)
)
ue_id = context.do_ue_create(tf[2], REQUEST)
if parcours.UE_IS_MODULE or tf[2]["create_matiere"]:
matiere_id = context.do_matiere_create(
{"ue_id": ue_id, "titre": tf[2]["titre"], "numero": 1}, REQUEST
)
if parcours.UE_IS_MODULE:
# dans ce mode, crée un (unique) module dans l'UE:
_ = context.do_module_create(
{
"titre": tf[2]["titre"],
"code": tf[2]["acronyme"],
"coefficient": 1.0, # tous les modules auront coef 1, et on utilisera les ECTS
"ue_id": ue_id,
"matiere_id": matiere_id,
"formation_id": formation_id,
"semestre_id": tf[2]["semestre_id"],
},
REQUEST,
)
else:
do_ue_edit(context, tf[2])
return REQUEST.RESPONSE.redirect(
context.NotesURL() + "/ue_list?formation_id=" + formation_id
)
def _add_ue_semestre_id(context, ue_list):
"""ajoute semestre_id dans les ue, en regardant le premier module de chacune.
Les UE sans modules se voient attribuer le numero UE_SEM_DEFAULT (1000000),
qui les place à la fin de la liste.
"""
for ue in ue_list:
Modlist = context.do_module_list(args={"ue_id": ue["ue_id"]})
if Modlist:
ue["semestre_id"] = Modlist[0]["semestre_id"]
else:
ue["semestre_id"] = 1000000
def next_ue_numero(context, formation_id, semestre_id=None):
"""Numero d'une nouvelle UE dans cette formation.
Si le semestre est specifie, cherche les UE ayant des modules de ce semestre
"""
ue_list = context.do_ue_list(args={"formation_id": formation_id})
if not ue_list:
return 0
if semestre_id is None:
return ue_list[-1]["numero"] + 1000
else:
# Avec semestre: (prend le semestre du 1er module de l'UE)
_add_ue_semestre_id(context, ue_list)
ue_list_semestre = [ue for ue in ue_list if ue["semestre_id"] == semestre_id]
if ue_list_semestre:
return ue_list_semestre[-1]["numero"] + 10
else:
return ue_list[-1]["numero"] + 1000
def ue_delete(
context, ue_id=None, delete_validations=False, dialog_confirmed=False, REQUEST=None
):
"""Delete an UE"""
ue = context.do_ue_list(args={"ue_id": ue_id})
if not ue:
raise ScoValueError("UE inexistante !")
ue = ue[0]
if not dialog_confirmed:
return context.confirmDialog(
"Suppression de l'UE %(titre)s (%(acronyme)s))
" % ue,
dest_url="",
REQUEST=REQUEST,
parameters={"ue_id": ue_id},
cancel_url="ue_list?formation_id=%s" % ue["formation_id"],
)
return context._do_ue_delete(
ue_id, delete_validations=delete_validations, REQUEST=REQUEST
)
def ue_list(context, formation_id=None, msg="", REQUEST=None):
"""Liste des matières et modules d'une formation, avec liens pour
editer (si non verrouillée).
"""
authuser = REQUEST.AUTHENTICATED_USER
F = context.formation_list(args={"formation_id": formation_id})
if not F:
raise ScoValueError("invalid formation_id")
F = F[0]
parcours = sco_codes_parcours.get_parcours_from_code(F["type_parcours"])
locked = context.formation_has_locked_sems(formation_id)
ue_list = context.do_ue_list(args={"formation_id": formation_id})
# tri par semestre et numero:
_add_ue_semestre_id(context, ue_list)
ue_list.sort(key=lambda u: (u["semestre_id"], u["numero"]))
has_duplicate_ue_codes = len(set([ue["ue_code"] for ue in ue_list])) != len(ue_list)
perm_change = authuser.has_permission(ScoChangeFormation, context)
# editable = (not locked) and perm_change
# On autorise maintanant la modification des formations qui ont des semestres verrouillés,
# sauf si cela affect les notes passées (verrouillées):
# - pas de modif des modules utilisés dans des semestres verrouillés
# - pas de changement des codes d'UE utilisés dans des semestres verrouillés
editable = perm_change
tag_editable = authuser.has_permission(ScoEditFormationTags, context) or perm_change
if locked:
lockicon = scu.icontag("lock32_img", title="verrouillé")
else:
lockicon = ""
arrow_up, arrow_down, arrow_none = sco_groups.getArrowIconsTags(context, REQUEST)
delete_icon = scu.icontag(
"delete_small_img", title="Supprimer (module inutilisé)", alt="supprimer"
)
delete_disabled_icon = scu.icontag(
"delete_small_dis_img", title="Suppression impossible (module utilisé)"
)
H = [
context.sco_header(
REQUEST,
cssstyles=["libjs/jQuery-tagEditor/jquery.tag-editor.css"],
javascripts=[
"libjs/jinplace-1.2.1.min.js",
"js/ue_list.js",
"libjs/jQuery-tagEditor/jquery.tag-editor.min.js",
"libjs/jQuery-tagEditor/jquery.caret.min.js",
"js/module_tag_editor.js",
],
page_title="Programme %s" % F["acronyme"],
),
"""Formation %(titre)s (%(acronyme)s) [version %(version)s] code %(formation_code)s"""
% F,
lockicon,
"
",
]
if locked:
H.append(
"""Cette formation est verrouillée car %d semestres verrouillés s'y réferent.
Si vous souhaitez modifier cette formation (par exemple pour y ajouter un module), vous devez:
- soit créer une nouvelle version de cette formation pour pouvoir l'éditer librement (vous pouvez passer par la fonction "Associer à une nouvelle version du programme" (menu "Semestre") si vous avez un semestre en cours);
- soit déverrouiller le ou les semestres qui s'y réfèrent (attention, en principe ces semestres sont archivés
et ne devraient pas être modifiés).
"""
% len(locked)
)
if msg:
H.append('' + msg + "
")
if has_duplicate_ue_codes:
H.append(
"""Attention: plusieurs UE de cette formation ont le même code. Il faut corriger cela ci-dessous, sinon les calculs d'ECTS seront erronés !
"""
)
# Description de la formation
H.append('")
# Description des UE/matières/modules
H.append('") # formation_ue_list
H.append("
"""
% F
)
if perm_change:
H.append(
"""
"""
)
for sem in sco_formsemestre.do_formsemestre_list(
context, args={"formation_id": formation_id}
):
H.append(
'- %(titremois)s'
% sem
)
if sem["etat"] != "1":
H.append(" [verrouillé]")
else:
H.append(
' Modifier'
% sem
)
H.append("
")
H.append("
")
if authuser.has_permission(ScoImplement, context):
H.append(
""""""
% F
)
# (debug) Vérifier cohérence
warn, _ = sco_formsemestre_validation.check_formation_ues(context, formation_id)
H.append(warn)
H.append(context.sco_footer(REQUEST))
return "".join(H)
def ue_sharing_code(context, ue_code=None, ue_id=None, hide_ue_id=None):
"""HTML list of UE sharing this code
Either ue_code or ue_id may be specified.
"""
if ue_id:
ue = context.do_ue_list(args={"ue_id": ue_id})[0]
if not ue_code:
ue_code = ue["ue_code"]
F = context.formation_list(args={"formation_id": ue["formation_id"]})[0]
formation_code = F["formation_code"]
ue_list_all = context.do_ue_list(args={"ue_code": ue_code})
if ue_id:
# retire les UE d'autres formations:
# log('checking ucode %s formation %s' % (ue_code, formation_code))
ue_list = []
for ue in ue_list_all:
F = context.formation_list(args={"formation_id": ue["formation_id"]})[0]
if formation_code == F["formation_code"]:
ue_list.append(ue)
else:
ue_list = ue_list_all
if hide_ue_id: # enlève l'ue de depart
ue_list = [ue for ue in ue_list if ue["ue_id"] != hide_ue_id]
if not ue_list:
if ue_id:
return """Seule UE avec code %s""" % ue_code
else:
return """Aucune UE avec code %s""" % ue_code
H = []
if ue_id:
H.append('Autres UE avec le code %s:' % ue_code)
else:
H.append('UE avec le code %s:' % ue_code)
H.append("")
for ue in ue_list:
F = context.formation_list(args={"formation_id": ue["formation_id"]})[0]
H.append(
'- %s (%s) dans %s (%s), version %s
'
% (
ue["acronyme"],
ue["titre"],
F["formation_id"],
F["acronyme"],
F["titre"],
F["version"],
)
)
H.append("
")
return "\n".join(H)
def do_ue_edit(context, args, bypass_lock=False, dont_invalidate_cache=False):
"edit an UE"
# check
ue_id = args["ue_id"]
ue = context.do_ue_list({"ue_id": ue_id})[0]
if (not bypass_lock) and context.ue_is_locked(ue["ue_id"]):
raise ScoLockedFormError()
# check: acronyme unique dans cette formation
if args.has_key("acronyme"):
new_acro = args["acronyme"]
ues = context.do_ue_list(
{"formation_id": ue["formation_id"], "acronyme": new_acro}
)
if ues and ues[0]["ue_id"] != ue_id:
raise ScoValueError('Acronyme d\'UE "%s" déjà utilisé !' % args["acronyme"])
# On ne peut pas supprimer le code UE:
if args.has_key("ue_code") and not args["ue_code"]:
del args["ue_code"]
cnx = context.GetDBConnexion()
context._ueEditor.edit(cnx, args)
if not dont_invalidate_cache:
# Invalide les semestres utilisant cette formation:
for sem in sco_formsemestre.do_formsemestre_list(
context, args={"formation_id": ue["formation_id"]}
):
context._inval_cache(
formsemestre_id=sem["formsemestre_id"]
) # > formation (ue) modif.
# essai edition en ligne:
def edit_ue_set_code_apogee(context, id=None, value=None, REQUEST=None):
"set UE code apogee"
ue_id = id
value = value.strip("-_ \t")
log("edit_ue_set_code_apogee: ue_id=%s code_apogee=%s" % (ue_id, value))
ues = context.do_ue_list(args={"ue_id": ue_id})
if not ues:
return "ue invalide"
do_ue_edit(
context,
{"ue_id": ue_id, "code_apogee": value},
bypass_lock=True,
dont_invalidate_cache=False,
)
if not value:
value = scu.APO_MISSING_CODE_STR
return value
# ---- Table recap formation
def formation_table_recap(context, formation_id, format="html", REQUEST=None):
"""Table recapitulant formation."""
F = context.formation_list(args={"formation_id": formation_id})
if not F:
raise ScoValueError("invalid formation_id")
F = F[0]
T = []
ue_list = context.do_ue_list(args={"formation_id": formation_id})
for UE in ue_list:
Matlist = context.do_matiere_list(args={"ue_id": UE["ue_id"]})
for Mat in Matlist:
Modlist = context.do_module_list(args={"matiere_id": Mat["matiere_id"]})
for Mod in Modlist:
Mod["nb_moduleimpls"] = context.module_count_moduleimpls(
Mod["module_id"]
)
#
T.append(
{
"UE_acro": UE["acronyme"],
"Mat_tit": Mat["titre"],
"Mod_tit": Mod["abbrev"] or Mod["titre"],
"Mod_code": Mod["code"],
"Mod_coef": Mod["coefficient"],
"Mod_sem": Mod["semestre_id"],
"nb_moduleimpls": Mod["nb_moduleimpls"],
"heures_cours": Mod["heures_cours"],
"heures_td": Mod["heures_td"],
"heures_tp": Mod["heures_tp"],
"ects": Mod["ects"],
}
)
columns_ids = [
"UE_acro",
"Mat_tit",
"Mod_tit",
"Mod_code",
"Mod_coef",
"Mod_sem",
"nb_moduleimpls",
"heures_cours",
"heures_td",
"heures_tp",
"ects",
]
titles = {
"UE_acro": "UE",
"Mat_tit": "Matière",
"Mod_tit": "Module",
"Mod_code": "Code",
"Mod_coef": "Coef.",
"Mod_sem": "Sem.",
"nb_moduleimpls": "Nb utilisé",
"heures_cours": "Cours (h)",
"heures_td": "TD (h)",
"heures_tp": "TP (h)",
"ects": "ECTS",
}
title = (
"""Formation %(titre)s (%(acronyme)s) [version %(version)s] code %(formation_code)s"""
% F
)
tab = GenTable(
columns_ids=columns_ids,
rows=T,
titles=titles,
origin="Généré par %s le " % scu.VERSION.SCONAME
+ scu.timedate_human_repr()
+ "",
caption=title,
html_caption=title,
html_class="table_leftalign",
base_url="%s?formation_id=%s" % (REQUEST.URL0, formation_id),
page_title=title,
html_title="" + title + "
",
pdf_title=title,
preferences=context.get_preferences(),
)
return tab.make_page(context, format=format, REQUEST=REQUEST)
def ue_list_semestre_ids(context, ue):
"""Liste triée des numeros de semestres des modules dans cette UE
Il est recommandable que tous les modules d'une UE aient le même indice de semestre.
Mais cela n'a pas toujours été le cas dans les programmes pédagogiques officiels,
aussi ScoDoc laisse le choix.
"""
Modlist = context.do_module_list(args={"ue_id": ue["ue_id"]})
return sorted(list(set([mod["semestre_id"] for mod in Modlist])))