2020-09-26 16:19:37 +02:00
# -*- mode: python -*-
# -*- coding: utf-8 -*-
##############################################################################
#
# Gestion scolarite IUT
#
2021-01-01 17:51:08 +01:00
# Copyright (c) 1999 - 2021 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
#
##############################################################################
2021-06-19 23:21:37 +02:00
""" Ajout/Modification/Suppression modules
2020-09-26 16:19:37 +02:00
( portage from DTML )
"""
2021-07-31 19:01:10 +03:00
import flask
from flask import url_for , g
2021-06-19 23:21:37 +02:00
import app . scodoc . notesdb as ndb
import app . scodoc . sco_utils as scu
from app . scodoc . notes_log import log
from app . scodoc . TrivialFormulator import TrivialFormulator
from app . scodoc . sco_permissions import Permission
from app . scodoc . sco_exceptions import ScoValueError , ScoLockedFormError , ScoGenError
from app . scodoc import html_sco_header
from app . scodoc import sco_codes_parcours
from app . scodoc import sco_edit_matiere
from app . scodoc import sco_moduleimpl
from app . scodoc import sco_news
2020-09-26 16:19:37 +02:00
_MODULE_HELP = """ <p class= " help " >
Les modules sont décrits dans le programme pédagogique . Un module est pour ce
logiciel l ' unité pédagogique élémentaire. On va lui associer une note
à travers des < em > évaluations < / em > . < br / >
Cette note ( moyenne de module ) sera utilisée pour calculer la moyenne
générale ( et la moyenne de l ' UE à laquelle appartient le module). Pour
cela , on utilisera le < em > coefficient < / em > associé au module .
< / p >
< p class = " help " > Un module possède un enseignant responsable
( typiquement celui qui dispense le cours magistral ) . On peut associer
au module une liste d ' enseignants (typiquement les chargés de TD).
Tous ces enseignants , plus le responsable du semestre , pourront
saisir et modifier les notes de ce module .
< / p > """
2021-06-16 18:18:32 +02:00
_moduleEditor = ndb . EditableTable (
" notes_modules " ,
" module_id " ,
(
" module_id " ,
" titre " ,
" code " ,
" abbrev " ,
" heures_cours " ,
" heures_td " ,
" heures_tp " ,
" coefficient " ,
" ue_id " ,
" matiere_id " ,
" formation_id " ,
" semestre_id " ,
" numero " ,
" code_apogee " ,
" module_type "
#'ects'
) ,
sortkey = " numero, code, titre " ,
output_formators = {
" heures_cours " : ndb . float_null_is_zero ,
" heures_td " : ndb . float_null_is_zero ,
" heures_tp " : ndb . float_null_is_zero ,
" numero " : ndb . int_null_is_zero ,
" coefficient " : ndb . float_null_is_zero ,
" module_type " : ndb . int_null_is_zero
#'ects' : ndb.float_null_is_null
} ,
)
def do_module_list ( context , * args , * * kw ) :
" list modules "
cnx = ndb . GetDBConnexion ( )
return _moduleEditor . list ( cnx , * args , * * kw )
2021-08-15 23:10:15 +02:00
def do_module_create ( context , args ) - > int :
2021-06-16 18:18:32 +02:00
" create a module "
# create
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_formations
2021-06-16 18:18:32 +02:00
cnx = ndb . GetDBConnexion ( )
r = _moduleEditor . create ( cnx , args )
# news
2021-08-19 10:28:35 +02:00
F = sco_formations . formation_list ( args = { " formation_id " : args [ " formation_id " ] } ) [ 0 ]
2021-06-16 18:18:32 +02:00
sco_news . add (
2021-06-19 23:21:37 +02:00
typ = sco_news . NEWS_FORM ,
2021-06-16 18:18:32 +02:00
object = args [ " formation_id " ] ,
text = " Modification de la formation %(acronyme)s " % F ,
2021-08-01 11:16:16 +03:00
max_frequency = 3 ,
2021-06-16 18:18:32 +02:00
)
return r
2020-09-26 16:19:37 +02:00
def module_create ( context , matiere_id = None , REQUEST = None ) :
2021-01-01 18:40:47 +01:00
""" Creation d ' un module """
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_formations
from app . scodoc import sco_edit_ue
2021-08-09 10:09:04 +02:00
if matiere_id is None :
2020-09-26 16:19:37 +02:00
raise ScoValueError ( " invalid matiere ! " )
2021-06-16 18:18:32 +02:00
M = sco_edit_matiere . do_matiere_list ( context , args = { " matiere_id " : matiere_id } ) [ 0 ]
UE = sco_edit_ue . do_ue_list ( context , args = { " ue_id " : M [ " ue_id " ] } ) [ 0 ]
2021-08-19 10:28:35 +02:00
Fo = sco_formations . formation_list ( args = { " formation_id " : UE [ " formation_id " ] } ) [ 0 ]
2020-09-26 16:19:37 +02:00
parcours = sco_codes_parcours . get_parcours_from_code ( Fo [ " type_parcours " ] )
2021-07-09 17:47:06 +02:00
semestres_indices = list ( range ( 1 , parcours . NB_SEM + 1 ) )
2020-09-26 16:19:37 +02:00
H = [
2021-07-29 17:31:15 +03:00
html_sco_header . sco_header ( page_title = " Création d ' un module " ) ,
2020-09-26 16:19:37 +02:00
""" <h2>Création d ' un module dans la matière %(titre)s """ % M ,
""" (UE %(acronyme)s )</h2> """ % UE ,
_MODULE_HELP ,
]
# cherche le numero adequat (pour placer le module en fin de liste)
2021-06-16 18:18:32 +02:00
Mods = do_module_list ( context , args = { " matiere_id " : matiere_id } )
2020-09-26 16:19:37 +02:00
if Mods :
default_num = max ( [ m [ " numero " ] for m in Mods ] ) + 10
else :
default_num = 10
tf = TrivialFormulator (
REQUEST . URL0 ,
REQUEST . form ,
(
(
" code " ,
{
" size " : 10 ,
" explanation " : " code du module (doit être unique dans la formation) " ,
" allow_null " : False ,
" validator " : lambda val , field , formation_id = Fo [
" formation_id "
] : check_module_code_unicity ( val , field , formation_id , context ) ,
} ,
) ,
( " titre " , { " size " : 30 , " explanation " : " nom du module " } ) ,
( " abbrev " , { " size " : 20 , " explanation " : " nom abrégé (pour bulletins) " } ) ,
(
" module_type " ,
{
" input_type " : " menu " ,
" title " : " Type " ,
" explanation " : " " ,
" labels " : ( " Standard " , " Malus " ) ,
2021-02-04 20:02:44 +01:00
" allowed_values " : ( str ( scu . MODULE_STANDARD ) , str ( scu . MODULE_MALUS ) ) ,
2020-09-26 16:19:37 +02:00
} ,
) ,
(
" heures_cours " ,
{ " size " : 4 , " type " : " float " , " explanation " : " nombre d ' heures de cours " } ,
) ,
(
" heures_td " ,
{
" size " : 4 ,
" type " : " float " ,
" explanation " : " nombre d ' heures de Travaux Dirigés " ,
} ,
) ,
(
" heures_tp " ,
{
" size " : 4 ,
" type " : " float " ,
" explanation " : " nombre d ' heures de Travaux Pratiques " ,
} ,
) ,
(
" coefficient " ,
{
" size " : 4 ,
" type " : " float " ,
" explanation " : " coefficient dans la formation (PPN) " ,
" allow_null " : False ,
} ,
) ,
# ('ects', { 'size' : 4, 'type' : 'float', 'title' : 'ECTS', 'explanation' : 'nombre de crédits ECTS (inutilisés: les crédits sont associés aux UE)' }),
( " formation_id " , { " default " : UE [ " formation_id " ] , " input_type " : " hidden " } ) ,
( " ue_id " , { " default " : M [ " ue_id " ] , " input_type " : " hidden " } ) ,
( " matiere_id " , { " default " : M [ " matiere_id " ] , " input_type " : " hidden " } ) ,
(
" semestre_id " ,
{
" input_type " : " menu " ,
" type " : " int " ,
2021-02-04 20:02:44 +01:00
" title " : scu . strcapitalize ( parcours . SESSION_NAME ) ,
2020-09-26 16:19:37 +02:00
" explanation " : " %s de début du module dans la formation standard "
% parcours . SESSION_NAME ,
" labels " : [ str ( x ) for x in semestres_indices ] ,
" allowed_values " : semestres_indices ,
} ,
) ,
(
" code_apogee " ,
{
" title " : " Code Apogée " ,
2021-06-24 23:09:06 +02:00
" size " : 25 ,
" explanation " : " (optionnel) code élément pédagogique Apogée ou liste de codes ELP séparés par des virgules " ,
2020-09-26 16:19:37 +02:00
} ,
) ,
(
" numero " ,
{
" size " : 2 ,
" explanation " : " numéro (1,2,3,4...) pour ordre d ' affichage " ,
" type " : " int " ,
" default " : default_num ,
} ,
) ,
) ,
submitlabel = " Créer ce module " ,
)
if tf [ 0 ] == 0 :
2021-07-29 11:19:00 +03:00
return " \n " . join ( H ) + tf [ 1 ] + html_sco_header . sco_footer ( )
2020-09-26 16:19:37 +02:00
else :
2021-07-31 19:01:10 +03:00
do_module_create ( context , tf [ 2 ] )
return flask . redirect (
url_for (
" notes.ue_list " ,
scodoc_dept = g . scodoc_dept ,
formation_id = UE [ " formation_id " ] ,
)
2020-09-26 16:19:37 +02:00
)
2021-07-31 19:01:10 +03:00
def do_module_delete ( context , oid ) :
2021-06-16 18:18:32 +02:00
" delete module "
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_formations
2021-06-16 18:18:32 +02:00
mod = do_module_list ( context , { " module_id " : oid } ) [ 0 ]
2021-06-19 23:21:37 +02:00
if module_is_locked ( context , mod [ " module_id " ] ) :
2021-06-16 18:18:32 +02:00
raise ScoLockedFormError ( )
# S'il y a des moduleimpls, on ne peut pas detruire le module !
mods = sco_moduleimpl . do_moduleimpl_list ( context , module_id = oid )
if mods :
err_page = scu . confirm_dialog (
message = """ <h3>Destruction du module impossible car il est utilisé dans des semestres existants !</h3> """ ,
helpmsg = """ Il faut d ' abord supprimer le semestre. Mais il est peut être préférable de laisser ce programme intact et d ' en créer une nouvelle version pour la modifier. """ ,
dest_url = " ue_list " ,
parameters = { " formation_id " : mod [ " formation_id " ] } ,
)
raise ScoGenError ( err_page )
# delete
cnx = ndb . GetDBConnexion ( )
_moduleEditor . delete ( cnx , oid )
# news
2021-08-19 10:28:35 +02:00
F = sco_formations . formation_list ( args = { " formation_id " : mod [ " formation_id " ] } ) [ 0 ]
2021-06-16 18:18:32 +02:00
sco_news . add (
2021-06-19 23:21:37 +02:00
typ = sco_news . NEWS_FORM ,
2021-06-16 18:18:32 +02:00
object = mod [ " formation_id " ] ,
text = " Modification de la formation %(acronyme)s " % F ,
2021-08-01 11:16:16 +03:00
max_frequency = 3 ,
2021-06-16 18:18:32 +02:00
)
2020-09-26 16:19:37 +02:00
def module_delete ( context , module_id = None , REQUEST = None ) :
""" Delete a module """
if not module_id :
raise ScoValueError ( " invalid module ! " )
2021-06-16 18:18:32 +02:00
Mods = do_module_list ( context , args = { " module_id " : module_id } )
2020-09-26 16:19:37 +02:00
if not Mods :
raise ScoValueError ( " Module inexistant ! " )
Mod = Mods [ 0 ]
H = [
2021-07-29 17:31:15 +03:00
html_sco_header . sco_header ( page_title = " Suppression d ' un module " ) ,
2020-09-26 16:19:37 +02:00
""" <h2>Suppression du module %(titre)s ( %(code)s )</h2> """ % Mod ,
]
2021-08-09 10:09:04 +02:00
dest_url = scu . NotesURL ( ) + " /ue_list?formation_id= " + str ( Mod [ " formation_id " ] )
2020-09-26 16:19:37 +02:00
tf = TrivialFormulator (
REQUEST . URL0 ,
REQUEST . form ,
( ( " module_id " , { " input_type " : " hidden " } ) , ) ,
initvalues = Mod ,
submitlabel = " Confirmer la suppression " ,
cancelbutton = " Annuler " ,
)
if tf [ 0 ] == 0 :
2021-07-29 11:19:00 +03:00
return " \n " . join ( H ) + tf [ 1 ] + html_sco_header . sco_footer ( )
2020-09-26 16:19:37 +02:00
elif tf [ 0 ] == - 1 :
2021-07-31 19:01:10 +03:00
return flask . redirect ( dest_url )
2020-09-26 16:19:37 +02:00
else :
2021-07-31 19:01:10 +03:00
do_module_delete ( context , module_id )
return flask . redirect ( dest_url )
2020-09-26 16:19:37 +02:00
2021-06-16 18:18:32 +02:00
def do_module_edit ( context , val ) :
" edit a module "
2021-07-19 20:53:01 +03:00
from app . scodoc import sco_edit_formation
2021-06-16 18:18:32 +02:00
# check
mod = do_module_list ( context , { " module_id " : val [ " module_id " ] } ) [ 0 ]
if module_is_locked ( context , mod [ " module_id " ] ) :
# formation verrouillée: empeche de modifier certains champs:
protected_fields = ( " coefficient " , " ue_id " , " matiere_id " , " semestre_id " )
for f in protected_fields :
if f in val :
del val [ f ]
# edit
cnx = ndb . GetDBConnexion ( )
_moduleEditor . edit ( cnx , val )
2021-07-19 20:53:01 +03:00
sco_edit_formation . invalidate_sems_in_formation ( mod [ " formation_id " ] )
2021-06-16 18:18:32 +02:00
2020-09-26 16:19:37 +02:00
def check_module_code_unicity ( code , field , formation_id , context , module_id = None ) :
" true si code module unique dans la formation "
2021-06-16 18:18:32 +02:00
Mods = do_module_list ( context , args = { " code " : code , " formation_id " : formation_id } )
2020-09-26 16:19:37 +02:00
if module_id : # edition: supprime le module en cours
Mods = [ m for m in Mods if m [ " module_id " ] != module_id ]
return len ( Mods ) == 0
def module_edit ( context , module_id = None , REQUEST = None ) :
""" Edit a module """
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_formations
from app . scodoc import sco_tag_module
2020-09-26 16:19:37 +02:00
if not module_id :
raise ScoValueError ( " invalid module ! " )
2021-06-16 18:18:32 +02:00
Mod = do_module_list ( context , args = { " module_id " : module_id } )
2020-09-26 16:19:37 +02:00
if not Mod :
raise ScoValueError ( " invalid module ! " )
Mod = Mod [ 0 ]
2021-06-13 18:29:53 +02:00
unlocked = not module_is_locked ( context , module_id )
2021-08-19 10:28:35 +02:00
Fo = sco_formations . formation_list ( args = { " formation_id " : Mod [ " formation_id " ] } ) [ 0 ]
2020-09-26 16:19:37 +02:00
parcours = sco_codes_parcours . get_parcours_from_code ( Fo [ " type_parcours " ] )
2021-02-03 22:00:41 +01:00
M = ndb . SimpleDictFetch (
2021-08-08 17:38:46 +02:00
""" SELECT ue.acronyme, mat.*, mat.id AS matiere_id
FROM notes_matieres mat , notes_ue ue
WHERE mat . ue_id = ue . id
2021-08-09 10:09:04 +02:00
AND ue . formation_id = % ( formation_id ) d
2021-08-08 17:38:46 +02:00
ORDER BY ue . numero , mat . numero
""" ,
2020-09-26 16:19:37 +02:00
{ " formation_id " : Mod [ " formation_id " ] } ,
)
Mnames = [ " %s / %s " % ( x [ " acronyme " ] , x [ " titre " ] ) for x in M ]
Mids = [ " %s ! %s " % ( x [ " ue_id " ] , x [ " matiere_id " ] ) for x in M ]
Mod [ " ue_matiere_id " ] = " %s ! %s " % ( Mod [ " ue_id " ] , Mod [ " matiere_id " ] )
2021-07-09 17:47:06 +02:00
semestres_indices = list ( range ( 1 , parcours . NB_SEM + 1 ) )
2020-09-26 16:19:37 +02:00
2021-08-09 10:09:04 +02:00
dest_url = scu . NotesURL ( ) + " /ue_list?formation_id= " + str ( Mod [ " formation_id " ] )
2020-09-26 16:19:37 +02:00
H = [
2021-06-13 18:29:53 +02:00
html_sco_header . sco_header (
2020-09-26 16:19:37 +02:00
page_title = " Modification du module %(titre)s " % Mod ,
cssstyles = [ " libjs/jQuery-tagEditor/jquery.tag-editor.css " ] ,
javascripts = [
" libjs/jQuery-tagEditor/jquery.tag-editor.min.js " ,
" libjs/jQuery-tagEditor/jquery.caret.min.js " ,
" js/module_tag_editor.js " ,
] ,
) ,
""" <h2>Modification du module %(titre)s """ % Mod ,
""" (formation %(acronyme)s , version %(version)s )</h2> """ % Fo ,
_MODULE_HELP ,
]
if not unlocked :
H . append (
""" <div class= " ue_warning " ><span>Formation verrouillée, seuls certains éléments peuvent être modifiés</span></div> """
)
tf = TrivialFormulator (
REQUEST . URL0 ,
REQUEST . form ,
(
(
" code " ,
{
" size " : 10 ,
" explanation " : " code du module (doit être unique dans la formation) " ,
" allow_null " : False ,
" validator " : lambda val , field , formation_id = Mod [
" formation_id "
] : check_module_code_unicity (
val , field , formation_id , context , module_id = module_id
) ,
} ,
) ,
( " titre " , { " size " : 30 , " explanation " : " nom du module " } ) ,
( " abbrev " , { " size " : 20 , " explanation " : " nom abrégé (pour bulletins) " } ) ,
(
" module_type " ,
{
" input_type " : " menu " ,
" title " : " Type " ,
" explanation " : " " ,
" labels " : ( " Standard " , " Malus " ) ,
2021-02-04 20:02:44 +01:00
" allowed_values " : ( str ( scu . MODULE_STANDARD ) , str ( scu . MODULE_MALUS ) ) ,
2020-09-26 16:19:37 +02:00
" enabled " : unlocked ,
} ,
) ,
(
" heures_cours " ,
{ " size " : 4 , " type " : " float " , " explanation " : " nombre d ' heures de cours " } ,
) ,
(
" heures_td " ,
{
" size " : 4 ,
" type " : " float " ,
" explanation " : " nombre d ' heures de Travaux Dirigés " ,
} ,
) ,
(
" heures_tp " ,
{
" size " : 4 ,
" type " : " float " ,
" explanation " : " nombre d ' heures de Travaux Pratiques " ,
} ,
) ,
(
" coefficient " ,
{
" size " : 4 ,
" type " : " float " ,
" explanation " : " coefficient dans la formation (PPN) " ,
" allow_null " : False ,
" enabled " : unlocked ,
} ,
) ,
# ('ects', { 'size' : 4, 'type' : 'float', 'title' : 'ECTS', 'explanation' : 'nombre de crédits ECTS', 'enabled' : unlocked }),
( " formation_id " , { " input_type " : " hidden " } ) ,
( " ue_id " , { " input_type " : " hidden " } ) ,
( " module_id " , { " input_type " : " hidden " } ) ,
(
" ue_matiere_id " ,
{
" input_type " : " menu " ,
" title " : " Matière " ,
" explanation " : " un module appartient à une seule matière. " ,
" labels " : Mnames ,
" allowed_values " : Mids ,
" enabled " : unlocked ,
} ,
) ,
(
" semestre_id " ,
{
" input_type " : " menu " ,
" type " : " int " ,
" title " : parcours . SESSION_NAME . capitalize ( ) ,
" explanation " : " %s de début du module dans la formation standard "
% parcours . SESSION_NAME ,
" labels " : [ str ( x ) for x in semestres_indices ] ,
" allowed_values " : semestres_indices ,
" enabled " : unlocked ,
} ,
) ,
(
" code_apogee " ,
{
" title " : " Code Apogée " ,
2021-06-24 23:09:06 +02:00
" size " : 25 ,
" explanation " : " (optionnel) code élément pédagogique Apogée ou liste de codes ELP séparés par des virgules " ,
2020-09-26 16:19:37 +02:00
} ,
) ,
(
" numero " ,
{
" size " : 2 ,
" explanation " : " numéro (1,2,3,4...) pour ordre d ' affichage " ,
" type " : " int " ,
} ,
) ,
) ,
html_foot_markup = """ <div style= " width: 90 % ; " ><span class= " sco_tag_edit " ><textarea data-module_id= " {} " class= " module_tag_editor " > {} </textarea></span></div> """ . format (
module_id , " , " . join ( sco_tag_module . module_tag_list ( context , module_id ) )
) ,
initvalues = Mod ,
submitlabel = " Modifier ce module " ,
)
if tf [ 0 ] == 0 :
2021-07-29 11:19:00 +03:00
return " \n " . join ( H ) + tf [ 1 ] + html_sco_header . sco_footer ( )
2020-09-26 16:19:37 +02:00
elif tf [ 0 ] == - 1 :
2021-07-31 19:01:10 +03:00
return flask . redirect ( dest_url )
2020-09-26 16:19:37 +02:00
else :
# l'UE peut changer
tf [ 2 ] [ " ue_id " ] , tf [ 2 ] [ " matiere_id " ] = tf [ 2 ] [ " ue_matiere_id " ] . split ( " ! " )
# Check unicité code module dans la formation
2021-06-16 18:18:32 +02:00
do_module_edit ( context , tf [ 2 ] )
2021-07-31 19:01:10 +03:00
return flask . redirect ( dest_url )
2020-09-26 16:19:37 +02:00
# Edition en ligne du code Apogee
def edit_module_set_code_apogee ( context , id = None , value = None , REQUEST = None ) :
2021-06-24 23:09:06 +02:00
" Set UE code apogee "
2020-09-26 16:19:37 +02:00
module_id = id
value = value . strip ( " -_ \t " )
log ( " edit_module_set_code_apogee: module_id= %s code_apogee= %s " % ( module_id , value ) )
2021-06-16 18:18:32 +02:00
modules = do_module_list ( context , args = { " module_id " : module_id } )
2020-09-26 16:19:37 +02:00
if not modules :
2021-06-24 23:09:06 +02:00
return " module invalide " # should not occur
2020-09-26 16:19:37 +02:00
2021-06-16 18:18:32 +02:00
do_module_edit ( context , { " module_id " : module_id , " code_apogee " : value } )
2020-09-26 16:19:37 +02:00
if not value :
2021-02-04 20:02:44 +01:00
value = scu . APO_MISSING_CODE_STR
2020-09-26 16:19:37 +02:00
return value
def module_list ( context , formation_id , REQUEST = None ) :
""" Liste des modules de la formation
( XXX inutile ou a revoir )
"""
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_formations
2020-09-26 16:19:37 +02:00
if not formation_id :
raise ScoValueError ( " invalid formation ! " )
2021-08-19 10:28:35 +02:00
F = sco_formations . formation_list ( args = { " formation_id " : formation_id } ) [ 0 ]
2020-09-26 16:19:37 +02:00
H = [
2021-07-29 17:31:15 +03:00
html_sco_header . sco_header ( page_title = " Liste des modules de %(titre)s " % F ) ,
2020-09-26 16:19:37 +02:00
""" <h2>Listes des modules dans la formation %(titre)s ( %(acronyme)s )</h2> """
% F ,
' <ul class= " notes_module_list " > ' ,
]
2021-06-16 10:15:46 +02:00
editable = REQUEST . AUTHENTICATED_USER . has_permission ( Permission . ScoChangeFormation )
2020-09-26 16:19:37 +02:00
2021-06-16 18:18:32 +02:00
for Mod in do_module_list ( context , args = { " formation_id " : formation_id } ) :
2020-09-26 16:19:37 +02:00
H . append ( ' <li class= " notes_module_list " > %s ' % Mod )
if editable :
H . append ( ' <a href= " module_edit?module_id= %(module_id)s " >modifier</a> ' % Mod )
H . append (
' <a href= " module_delete?module_id= %(module_id)s " >supprimer</a> ' % Mod
)
H . append ( " </li> " )
H . append ( " </ul> " )
2021-07-29 11:19:00 +03:00
H . append ( html_sco_header . sco_footer ( ) )
2020-09-26 16:19:37 +02:00
return " \n " . join ( H )
2021-06-13 18:29:53 +02:00
def module_is_locked ( context , module_id ) :
""" True if module should not be modified
( used in a locked formsemestre )
"""
r = ndb . SimpleDictFetch (
2021-08-08 17:38:46 +02:00
""" SELECT mi.id
FROM notes_modules mod , notes_formsemestre sem , notes_moduleimpl mi
WHERE mi . module_id = mod . id
AND mi . formsemestre_id = sem . id
AND mi . module_id = % ( module_id ) s
AND sem . etat = false
2021-06-13 18:29:53 +02:00
""" ,
{ " module_id " : module_id } ,
)
return len ( r ) > 0
def module_count_moduleimpls ( context , module_id ) :
" Number of moduleimpls using this module "
mods = sco_moduleimpl . do_moduleimpl_list ( context , module_id = module_id )
return len ( mods )
2020-09-26 16:19:37 +02:00
def formation_add_malus_modules ( context , formation_id , titre = None , REQUEST = None ) :
2021-01-01 18:40:47 +01:00
""" Création d ' un module de " malus " dans chaque UE d ' une formation """
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_edit_ue
2021-06-16 18:18:32 +02:00
ue_list = sco_edit_ue . do_ue_list ( context , args = { " formation_id " : formation_id } )
2020-09-26 16:19:37 +02:00
for ue in ue_list :
# Un seul module de malus par UE:
nb_mod_malus = len (
[
mod
2021-06-16 18:18:32 +02:00
for mod in do_module_list ( context , args = { " ue_id " : ue [ " ue_id " ] } )
2021-02-04 20:02:44 +01:00
if mod [ " module_type " ] == scu . MODULE_MALUS
2020-09-26 16:19:37 +02:00
]
)
if nb_mod_malus == 0 :
ue_add_malus_module ( context , ue [ " ue_id " ] , titre = titre , REQUEST = REQUEST )
if REQUEST :
2021-08-09 10:09:04 +02:00
return flask . redirect ( " ue_list?formation_id= " + str ( formation_id ) )
2020-09-26 16:19:37 +02:00
def ue_add_malus_module ( context , ue_id , titre = None , code = None , REQUEST = None ) :
2021-01-01 18:40:47 +01:00
""" Add a malus module in this ue """
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_edit_ue
2021-06-16 18:18:32 +02:00
ue = sco_edit_ue . do_ue_list ( context , args = { " ue_id " : ue_id } ) [ 0 ]
2020-09-26 16:19:37 +02:00
if titre is None :
titre = " "
if code is None :
code = " MALUS %d " % ue [ " numero " ]
# Tout module doit avoir un semestre_id (indice 1, 2, ...)
semestre_ids = sco_edit_ue . ue_list_semestre_ids ( context , ue )
if semestre_ids :
semestre_id = semestre_ids [ 0 ]
else :
# c'est ennuyeux: dans ce cas, on pourrait demander à indiquer explicitement
# le semestre ? ou affecter le malus au semestre 1 ???
raise ScoValueError (
" Impossible d ' ajouter un malus s ' il n ' y a pas d ' autres modules "
)
# Matiere pour placer le module malus
2021-06-16 18:18:32 +02:00
Matlist = sco_edit_matiere . do_matiere_list ( context , args = { " ue_id " : ue_id } )
2020-09-26 16:19:37 +02:00
numero = max ( [ mat [ " numero " ] for mat in Matlist ] ) + 10
2021-06-16 18:18:32 +02:00
matiere_id = sco_edit_matiere . do_matiere_create (
2021-07-31 19:01:10 +03:00
context , { " ue_id " : ue_id , " titre " : " Malus " , " numero " : numero }
2020-09-26 16:19:37 +02:00
)
2021-06-16 18:18:32 +02:00
module_id = do_module_create (
context ,
2020-09-26 16:19:37 +02:00
{
" titre " : titre ,
" code " : code ,
" coefficient " : 0.0 , # unused
" ue_id " : ue_id ,
" matiere_id " : matiere_id ,
" formation_id " : ue [ " formation_id " ] ,
" semestre_id " : semestre_id ,
2021-02-04 20:02:44 +01:00
" module_type " : scu . MODULE_MALUS ,
2020-09-26 16:19:37 +02:00
} ,
)
return module_id