1522 lines
50 KiB
Python
1522 lines
50 KiB
Python
# -*- coding: utf-8 -*-
|
|
##############################################################################
|
|
#
|
|
# Gestion scolarite IUT
|
|
#
|
|
# Copyright (c) 1999 - 2022 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
|
|
#
|
|
##############################################################################
|
|
|
|
"""
|
|
Module absences: issu de ScoDoc7 / ZAbsences.py
|
|
|
|
Emmanuel Viennet, 2021
|
|
|
|
Gestion des absences (v4)
|
|
|
|
Code dérivé de la partie la plus ancienne de ScoDoc, et à revoir.
|
|
|
|
L'API de plus bas niveau est en gros:
|
|
|
|
AnnuleAbsencesDatesNoJust( dates)
|
|
count_abs(etudid, debut, fin, matin=None, moduleimpl_id=None)
|
|
count_abs_just(etudid, debut, fin, matin=None, moduleimpl_id=None)
|
|
list_abs_just(etudid, datedebut) [pas de fin ?]
|
|
list_abs_non_just(etudid, datedebut) [pas de fin ?]
|
|
list_abs_justifs(etudid, datedebut, datefin=None, only_no_abs=True)
|
|
|
|
list_abs_jour(date, am=True, pm=True, is_abs=None, is_just=None)
|
|
list_abs_non_just_jour(date, am=True, pm=True)
|
|
|
|
"""
|
|
|
|
import calendar
|
|
import datetime
|
|
import dateutil
|
|
import dateutil.parser
|
|
import re
|
|
import time
|
|
import urllib
|
|
from xml.etree import ElementTree
|
|
|
|
import flask
|
|
from flask import g, request
|
|
from flask import abort, flash, url_for
|
|
from flask_login import current_user
|
|
|
|
from app import db, log
|
|
from app import api
|
|
from app.comp import res_sem
|
|
from app.comp.res_compat import NotesTableCompat
|
|
from app.decorators import (
|
|
scodoc,
|
|
scodoc7func,
|
|
permission_required,
|
|
permission_required_compat_scodoc7,
|
|
)
|
|
from app.models import FormSemestre, GroupDescr, Partition
|
|
from app.models.absences import BilletAbsence
|
|
from app.models.etudiants import Identite
|
|
from app.views import absences_bp as bp
|
|
|
|
# ---------------
|
|
from app.scodoc import sco_utils as scu
|
|
from app.scodoc import notesdb as ndb
|
|
from app.scodoc.scolog import logdb
|
|
from app.scodoc.sco_permissions import Permission
|
|
from app.scodoc.sco_exceptions import ScoValueError, APIInvalidParams
|
|
from app.scodoc.TrivialFormulator import TrivialFormulator
|
|
from app.scodoc.gen_tables import GenTable
|
|
from app.scodoc import html_sco_header
|
|
from app.scodoc import sco_abs
|
|
from app.scodoc import sco_abs_billets
|
|
from app.scodoc import sco_abs_views
|
|
from app.scodoc import sco_etud
|
|
from app.scodoc import sco_find_etud
|
|
from app.scodoc import sco_formsemestre
|
|
from app.scodoc import sco_groups_view
|
|
from app.scodoc import sco_moduleimpl
|
|
from app.scodoc import sco_preferences
|
|
from app.scodoc import sco_xml
|
|
|
|
|
|
CSSSTYLES = html_sco_header.BOOTSTRAP_MULTISELECT_CSS
|
|
|
|
|
|
def sco_publish(route, function, permission, methods=["GET"]):
|
|
"""Declare a route for a python function,
|
|
protected by permission and called following ScoDoc 7 Zope standards.
|
|
"""
|
|
return bp.route(route, methods=methods)(
|
|
scodoc(permission_required(permission)(scodoc7func(function)))
|
|
)
|
|
|
|
|
|
# --------------------------------------------------------------------
|
|
#
|
|
# ABSENCES (/ScoDoc/<dept>/Scolarite/Absences/...)
|
|
#
|
|
# --------------------------------------------------------------------
|
|
|
|
|
|
@bp.route("/")
|
|
@bp.route("/index_html")
|
|
@scodoc
|
|
@permission_required(Permission.ScoView)
|
|
@scodoc7func
|
|
def index_html():
|
|
"""Gestionnaire absences, page principale"""
|
|
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title="Saisie des absences",
|
|
cssstyles=["css/calabs.css"],
|
|
javascripts=["js/calabs.js"],
|
|
),
|
|
"""<h2>Traitement des absences</h2>
|
|
<p class="help">
|
|
Pour saisir des absences ou consulter les états, il est recommandé par passer par
|
|
le semestre concerné (saisie par jours nommés ou par semaines).
|
|
</p>
|
|
""",
|
|
]
|
|
H.append(
|
|
"""<p class="help">Pour signaler, annuler ou justifier une absence pour un seul étudiant,
|
|
choisissez d'abord concerné:</p>"""
|
|
)
|
|
H.append(sco_find_etud.form_search_etud())
|
|
if current_user.has_permission(
|
|
Permission.ScoAbsChange
|
|
) and sco_preferences.get_preference("handle_billets_abs"):
|
|
H.append(
|
|
f"""
|
|
<h2 style="margin-top: 30px;">Billets d'absence</h2>
|
|
<ul><li><a href="{url_for("absences.list_billets", scodoc_dept=g.scodoc_dept)
|
|
}">Traitement des billets d'absence en attente</a>
|
|
</li></ul>
|
|
"""
|
|
)
|
|
H.append(html_sco_header.sco_footer())
|
|
return "\n".join(H)
|
|
|
|
|
|
@bp.route("/choix_semaine")
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsChange)
|
|
@scodoc7func
|
|
def choix_semaine(group_id):
|
|
"""Page choix semaine sur calendrier pour saisie absences d'un groupe"""
|
|
group = (
|
|
GroupDescr.query.filter_by(id=group_id)
|
|
.join(Partition)
|
|
.join(FormSemestre)
|
|
.filter_by(dept_id=g.scodoc_dept_id)
|
|
.first_or_404()
|
|
)
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title="Saisie des absences",
|
|
cssstyles=["css/calabs.css"],
|
|
javascripts=["js/calabs.js"],
|
|
),
|
|
f"""
|
|
<h2>Saisie des Absences</h2>
|
|
<form action="SignaleAbsenceGrHebdo" id="formw">
|
|
<p>
|
|
<span style="font-weight: bold; font-size:120%;">
|
|
Saisie par semaine </span> - Groupe: {group.get_nom_with_part()}
|
|
<input name="datelundi" type="hidden" value="x"/>
|
|
<input name="group_ids" type="hidden" value="{group_id}"/>
|
|
</p>
|
|
""",
|
|
cal_select_week(),
|
|
"""<p class="help">Sélectionner le groupe d'étudiants, puis cliquez sur une semaine pour
|
|
saisir les absences de toute cette semaine.</p>
|
|
</form>
|
|
""",
|
|
html_sco_header.sco_footer(),
|
|
]
|
|
return "\n".join(H)
|
|
|
|
|
|
def cal_select_week(year=None):
|
|
"display calendar allowing week selection"
|
|
if not year:
|
|
year = scu.annee_scolaire()
|
|
sems = sco_formsemestre.do_formsemestre_list()
|
|
if not sems:
|
|
js = ""
|
|
else:
|
|
js = 'onmouseover="highlightweek(this);" onmouseout="deselectweeks();" onclick="wclick(this);"'
|
|
C = sco_abs.YearTable(int(year), dayattributes=js)
|
|
return C
|
|
|
|
|
|
sco_publish("/EtatAbsences", sco_abs_views.EtatAbsences, Permission.ScoView)
|
|
sco_publish("/CalAbs", sco_abs_views.CalAbs, Permission.ScoView)
|
|
sco_publish(
|
|
"/SignaleAbsenceEtud",
|
|
sco_abs_views.SignaleAbsenceEtud,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/doSignaleAbsence",
|
|
sco_abs_views.doSignaleAbsence,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/JustifAbsenceEtud",
|
|
sco_abs_views.JustifAbsenceEtud,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/doJustifAbsence",
|
|
sco_abs_views.doJustifAbsence,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/AnnuleAbsenceEtud",
|
|
sco_abs_views.AnnuleAbsenceEtud,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/doAnnuleAbsence",
|
|
sco_abs_views.doAnnuleAbsence,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/doAnnuleJustif",
|
|
sco_abs_views.doAnnuleJustif,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
sco_publish(
|
|
"/AnnuleAbsencesDatesNoJust",
|
|
sco_abs_views.AnnuleAbsencesDatesNoJust,
|
|
Permission.ScoAbsChange,
|
|
methods=["GET", "POST"],
|
|
)
|
|
|
|
# Antédiluvienne fonction: #deprecated
|
|
@bp.route("/ListeAbsEtud", methods=["GET", "POST"]) # pour compat anciens clients PHP
|
|
@scodoc
|
|
@permission_required_compat_scodoc7(Permission.ScoView)
|
|
@scodoc7func
|
|
def ListeAbsEtud(
|
|
etudid=None,
|
|
code_nip=None,
|
|
with_evals=True,
|
|
format="html",
|
|
absjust_only=0,
|
|
sco_year=None,
|
|
):
|
|
return sco_abs_views.ListeAbsEtud(
|
|
etudid=etudid,
|
|
code_nip=str(code_nip),
|
|
with_evals=with_evals,
|
|
format=format,
|
|
absjust_only=absjust_only,
|
|
sco_year=sco_year,
|
|
)
|
|
|
|
|
|
# --------------------------------------------------------------------
|
|
#
|
|
# SQL METHODS (xxx #sco8 not views => à déplacer)
|
|
#
|
|
# --------------------------------------------------------------------
|
|
|
|
# API backward compatibility
|
|
sco_publish("/CountAbs", sco_abs.count_abs, Permission.ScoView)
|
|
sco_publish("/CountAbsJust", sco_abs.count_abs_just, Permission.ScoView)
|
|
# TODO nouvel appel rendnat les deux valeurs et utilisant le cache
|
|
|
|
|
|
@bp.route("/doSignaleAbsenceGrSemestre", methods=["GET", "POST"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsChange)
|
|
@scodoc7func
|
|
def doSignaleAbsenceGrSemestre(
|
|
moduleimpl_id=None,
|
|
abslist=[],
|
|
dates="",
|
|
etudids="",
|
|
destination=None,
|
|
):
|
|
"""Enregistre absences aux dates indiquees (abslist et dates).
|
|
dates est une liste de dates ISO (séparées par des ',').
|
|
Efface les absences aux dates indiquées par dates,
|
|
ou bien ajoute celles de abslist.
|
|
"""
|
|
moduleimpl_id = moduleimpl_id or None
|
|
if etudids:
|
|
etudids = [int(x) for x in str(etudids).split(",")]
|
|
else:
|
|
etudids = []
|
|
if dates:
|
|
dates = dates.split(",")
|
|
else:
|
|
dates = []
|
|
|
|
# 1- Efface les absences
|
|
if dates:
|
|
for etudid in etudids:
|
|
sco_abs_views.AnnuleAbsencesDatesNoJust(etudid, dates, moduleimpl_id)
|
|
return "Absences effacées"
|
|
|
|
# 2- Ajoute les absences
|
|
if abslist:
|
|
sco_abs.add_abslist(abslist, moduleimpl_id)
|
|
return "Absences ajoutées"
|
|
|
|
return ("", 204)
|
|
|
|
|
|
# ------------ HTML Interfaces
|
|
@bp.route("/SignaleAbsenceGrHebdo", methods=["GET", "POST"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsChange)
|
|
@scodoc7func
|
|
def SignaleAbsenceGrHebdo(
|
|
datelundi, group_ids=[], destination="", moduleimpl_id=None, formsemestre_id=None
|
|
):
|
|
"Saisie hebdomadaire des absences"
|
|
if not moduleimpl_id:
|
|
moduleimpl_id = None
|
|
|
|
groups_infos = sco_groups_view.DisplayedGroupsInfos(
|
|
group_ids, moduleimpl_id=moduleimpl_id, formsemestre_id=formsemestre_id
|
|
)
|
|
if not groups_infos.members:
|
|
return (
|
|
html_sco_header.sco_header(page_title="Saisie des absences")
|
|
+ "<h3>Aucun étudiant !</h3>"
|
|
+ html_sco_header.sco_footer()
|
|
)
|
|
|
|
base_url = "SignaleAbsenceGrHebdo?datelundi=%s&%s&destination=%s" % (
|
|
datelundi,
|
|
groups_infos.groups_query_args,
|
|
urllib.parse.quote(destination),
|
|
)
|
|
|
|
formsemestre_id = groups_infos.formsemestre_id
|
|
formsemestre = FormSemestre.query.get_or_404(formsemestre_id)
|
|
if formsemestre.dept_id != g.scodoc_dept_id:
|
|
abort(404, "groupes inexistants dans ce département")
|
|
require_module = sco_preferences.get_preference(
|
|
"abs_require_module", formsemestre_id
|
|
)
|
|
etuds = [
|
|
sco_etud.get_etud_info(etudid=m["etudid"], filled=True)[0]
|
|
for m in groups_infos.members
|
|
]
|
|
# Restreint aux inscrits au module sélectionné
|
|
if moduleimpl_id:
|
|
mod_inscrits = set(
|
|
[
|
|
x["etudid"]
|
|
for x in sco_moduleimpl.do_moduleimpl_inscription_list(
|
|
moduleimpl_id=moduleimpl_id
|
|
)
|
|
]
|
|
)
|
|
etuds_inscrits_module = [e for e in etuds if e["etudid"] in mod_inscrits]
|
|
if etuds_inscrits_module:
|
|
etuds = etuds_inscrits_module
|
|
else:
|
|
# Si aucun etudiant n'est inscrit au module choisi...
|
|
moduleimpl_id = None
|
|
|
|
nt: NotesTableCompat = res_sem.load_formsemestre_results(formsemestre)
|
|
sem = formsemestre.to_dict()
|
|
|
|
# calcule dates jours de cette semaine
|
|
# liste de dates iso "yyyy-mm-dd"
|
|
datessem = [ndb.DateDMYtoISO(datelundi)]
|
|
for _ in sco_abs.day_names()[1:]:
|
|
datessem.append(sco_abs.next_iso_day(datessem[-1]))
|
|
#
|
|
if groups_infos.tous_les_etuds_du_sem:
|
|
gr_tit = "en"
|
|
else:
|
|
if len(groups_infos.group_ids) > 1:
|
|
p = "des groupes"
|
|
else:
|
|
p = "du groupe"
|
|
gr_tit = p + ' <span class="fontred">' + groups_infos.groups_titles + "</span>"
|
|
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title="Saisie hebdomadaire des absences",
|
|
init_qtip=True,
|
|
javascripts=html_sco_header.BOOTSTRAP_MULTISELECT_JS
|
|
+ [
|
|
"js/etud_info.js",
|
|
"js/abs_ajax.js",
|
|
"js/groups_view.js",
|
|
],
|
|
cssstyles=CSSSTYLES,
|
|
no_side_bar=1,
|
|
),
|
|
"""<table border="0" cellspacing="16"><tr><td>
|
|
<h2>Saisie des absences %s %s,
|
|
<span class="fontred">semaine du lundi %s</span></h2>
|
|
<div>
|
|
<form id="group_selector" method="get">
|
|
<input type="hidden" name="formsemestre_id" id="formsemestre_id" value="%s"/>
|
|
<input type="hidden" name="datelundi" id="datelundi" value="%s"/>
|
|
<input type="hidden" name="destination" id="destination" value="%s"/>
|
|
<input type="hidden" name="moduleimpl_id" id="moduleimpl_id_o" value="%s"/>
|
|
Groupes: %s
|
|
</form>
|
|
<form id="abs_form">
|
|
"""
|
|
% (
|
|
gr_tit,
|
|
sem["titre_num"],
|
|
datelundi,
|
|
groups_infos.formsemestre_id,
|
|
datelundi,
|
|
destination,
|
|
moduleimpl_id or "",
|
|
sco_groups_view.menu_groups_choice(groups_infos, submit_on_change=True),
|
|
),
|
|
]
|
|
#
|
|
modimpls_list = []
|
|
ues = nt.get_ues_stat_dict()
|
|
for ue in ues:
|
|
modimpls_list += nt.get_modimpls_dict(ue_id=ue["ue_id"])
|
|
|
|
menu_module = ""
|
|
for modimpl in modimpls_list:
|
|
if modimpl["moduleimpl_id"] == moduleimpl_id:
|
|
sel = "selected"
|
|
else:
|
|
sel = ""
|
|
menu_module += (
|
|
"""<option value="%(modimpl_id)s" %(sel)s>%(modname)s</option>\n"""
|
|
% {
|
|
"modimpl_id": modimpl["moduleimpl_id"],
|
|
"modname": (modimpl["module"]["code"] or "")
|
|
+ " "
|
|
+ (modimpl["module"]["abbrev"] or modimpl["module"]["titre"] or ""),
|
|
"sel": sel,
|
|
}
|
|
)
|
|
if moduleimpl_id:
|
|
sel = ""
|
|
else:
|
|
sel = "selected" # aucun module specifie
|
|
|
|
H.append(
|
|
"""Module concerné:
|
|
<select id="moduleimpl_id" name="moduleimpl_id" onchange="change_moduleimpl('%(url)s')">
|
|
<option value="" %(sel)s>non spécifié</option>
|
|
%(menu_module)s
|
|
</select>
|
|
</div>"""
|
|
% {"menu_module": menu_module, "url": base_url, "sel": sel}
|
|
)
|
|
|
|
H += _gen_form_saisie_groupe(
|
|
etuds, datessem, destination, moduleimpl_id, require_module
|
|
)
|
|
|
|
H.append(html_sco_header.sco_footer())
|
|
return "\n".join(H)
|
|
|
|
|
|
@bp.route("/SignaleAbsenceGrSemestre", methods=["GET", "POST"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsChange)
|
|
@scodoc7func
|
|
def SignaleAbsenceGrSemestre(
|
|
datedebut,
|
|
datefin,
|
|
destination="",
|
|
group_ids=(), # list of groups to display
|
|
nbweeks=4, # ne montre que les nbweeks dernieres semaines
|
|
moduleimpl_id=None,
|
|
):
|
|
"""Saisie des absences sur une journée sur un semestre (ou intervalle de dates) entier"""
|
|
groups_infos = sco_groups_view.DisplayedGroupsInfos(group_ids)
|
|
if not groups_infos.members:
|
|
return (
|
|
html_sco_header.sco_header(page_title="Saisie des absences")
|
|
+ "<h3>Aucun étudiant !</h3>"
|
|
+ html_sco_header.sco_footer()
|
|
)
|
|
formsemestre_id = groups_infos.formsemestre_id
|
|
formsemestre: FormSemestre = FormSemestre.query.get_or_404(formsemestre_id)
|
|
if formsemestre.dept_id != g.scodoc_dept_id:
|
|
return abort(404, "groupes inexistants dans ce département")
|
|
sem = formsemestre.to_dict()
|
|
require_module = sco_preferences.get_preference(
|
|
"abs_require_module", formsemestre_id
|
|
)
|
|
etuds = [
|
|
sco_etud.get_etud_info(etudid=m["etudid"], filled=True)[0]
|
|
for m in groups_infos.members
|
|
]
|
|
# Restreint aux inscrits au module sélectionné
|
|
if moduleimpl_id:
|
|
mod_inscrits = set(
|
|
[
|
|
x["etudid"]
|
|
for x in sco_moduleimpl.do_moduleimpl_inscription_list(
|
|
moduleimpl_id=moduleimpl_id
|
|
)
|
|
]
|
|
)
|
|
etuds = [e for e in etuds if e["etudid"] in mod_inscrits]
|
|
if not moduleimpl_id:
|
|
moduleimpl_id = None
|
|
base_url_noweeks = (
|
|
"SignaleAbsenceGrSemestre?datedebut=%s&datefin=%s&%s&destination=%s"
|
|
% (
|
|
datedebut,
|
|
datefin,
|
|
groups_infos.groups_query_args,
|
|
urllib.parse.quote(destination),
|
|
)
|
|
)
|
|
base_url = base_url_noweeks + "&nbweeks=%s" % nbweeks # sans le moduleimpl_id
|
|
|
|
if etuds:
|
|
nt: NotesTableCompat = res_sem.load_formsemestre_results(formsemestre)
|
|
|
|
work_saturday = sco_abs.is_work_saturday()
|
|
jourdebut = sco_abs.ddmmyyyy(datedebut, work_saturday=work_saturday)
|
|
jourfin = sco_abs.ddmmyyyy(datefin, work_saturday=work_saturday)
|
|
today = sco_abs.ddmmyyyy(
|
|
time.strftime("%d/%m/%Y", time.localtime()),
|
|
work_saturday=work_saturday,
|
|
)
|
|
today.next_day()
|
|
if jourfin > today: # ne propose jamais les semaines dans le futur
|
|
jourfin = today
|
|
if jourdebut > today:
|
|
raise ScoValueError("date de début dans le futur (%s) !" % jourdebut)
|
|
#
|
|
if not jourdebut.iswork() or jourdebut > jourfin:
|
|
raise ValueError(
|
|
"date debut invalide (%s, ouvrable=%d)"
|
|
% (str(jourdebut), jourdebut.iswork())
|
|
)
|
|
# calcule dates
|
|
dates = [] # sco_abs.ddmmyyyy instances
|
|
d = sco_abs.ddmmyyyy(datedebut, work_saturday=work_saturday)
|
|
while d <= jourfin:
|
|
dates.append(d)
|
|
d = d.next_day(7) # avance d'une semaine
|
|
#
|
|
msg = "Montrer seulement les 4 dernières semaines"
|
|
nwl = 4
|
|
if nbweeks:
|
|
nbweeks = int(nbweeks)
|
|
if nbweeks > 0:
|
|
dates = dates[-nbweeks:]
|
|
msg = "Montrer toutes les semaines"
|
|
nwl = 0
|
|
url_link_semaines = base_url_noweeks + "&nbweeks=%s" % nwl
|
|
if moduleimpl_id:
|
|
url_link_semaines += "&moduleimpl_id=" + str(moduleimpl_id)
|
|
#
|
|
dates = [x.ISO() for x in dates]
|
|
day_name = sco_abs.day_names()[jourdebut.weekday]
|
|
|
|
if groups_infos.tous_les_etuds_du_sem:
|
|
gr_tit = "en"
|
|
else:
|
|
if len(groups_infos.group_ids) > 1:
|
|
p = "des groupes "
|
|
else:
|
|
p = "du groupe "
|
|
gr_tit = p + '<span class="fontred">' + groups_infos.groups_titles + "</span>"
|
|
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title=f"Saisie des absences du {day_name}",
|
|
init_qtip=True,
|
|
javascripts=["js/etud_info.js", "js/abs_ajax.js"],
|
|
no_side_bar=1,
|
|
),
|
|
f"""<table border="0" cellspacing="16"><tr><td>
|
|
<h2>Saisie des absences {gr_tit} {sem["titre_num"]},
|
|
les <span class="fontred">{day_name}s</span></h2>
|
|
<p>
|
|
<a href="{url_link_semaines}">{msg}</a>
|
|
<form id="abs_form" action="doSignaleAbsenceGrSemestre" method="post">
|
|
""",
|
|
]
|
|
#
|
|
if etuds:
|
|
modimpls_list = []
|
|
ues = nt.get_ues_stat_dict()
|
|
for ue in ues:
|
|
modimpls_list += nt.get_modimpls_dict(ue_id=ue["ue_id"])
|
|
|
|
menu_module = ""
|
|
for modimpl in modimpls_list:
|
|
if modimpl["moduleimpl_id"] == moduleimpl_id:
|
|
sel = "selected"
|
|
else:
|
|
sel = ""
|
|
menu_module += (
|
|
"""<option value="%(modimpl_id)s" %(sel)s>%(modname)s</option>\n"""
|
|
% {
|
|
"modimpl_id": modimpl["moduleimpl_id"],
|
|
"modname": (modimpl["module"]["code"] or "")
|
|
+ " "
|
|
+ (modimpl["module"]["abbrev"] or modimpl["module"]["titre"] or ""),
|
|
"sel": sel,
|
|
}
|
|
)
|
|
if moduleimpl_id:
|
|
sel = ""
|
|
else:
|
|
sel = "selected" # aucun module specifie
|
|
H.append(
|
|
"""<p>
|
|
Module concerné par ces absences (%(optionel_txt)s):
|
|
<select id="moduleimpl_id" name="moduleimpl_id"
|
|
onchange="document.location='%(url)s&moduleimpl_id='+document.getElementById('moduleimpl_id').value">
|
|
<option value="" %(sel)s>non spécifié</option>
|
|
%(menu_module)s
|
|
</select>
|
|
</p>"""
|
|
% {
|
|
"menu_module": menu_module,
|
|
"url": base_url,
|
|
"sel": sel,
|
|
"optionel_txt": '<span class="redboldtext">requis</span>'
|
|
if require_module
|
|
else "optionnel",
|
|
}
|
|
)
|
|
|
|
H += _gen_form_saisie_groupe(
|
|
etuds, dates, destination, moduleimpl_id, require_module
|
|
)
|
|
H.append(html_sco_header.sco_footer())
|
|
return "\n".join(H)
|
|
|
|
|
|
def _gen_form_saisie_groupe(
|
|
etuds, dates, destination="", moduleimpl_id=None, require_module=False
|
|
):
|
|
"""Formulaire saisie absences
|
|
|
|
Args:
|
|
etuds: liste des étudiants
|
|
dates: liste ordonnée de dates iso, par exemple: [ '2020-12-24', ... ]
|
|
moduleimpl_id: optionnel, module concerné.
|
|
"""
|
|
H = [
|
|
f"""
|
|
<script type="text/javascript">
|
|
$(function() {{
|
|
$(".abs_form_table input").prop( "disabled", {
|
|
"true" if (require_module and not moduleimpl_id) else "false"
|
|
} );
|
|
}});
|
|
function colorize(obj) {{
|
|
if (obj.checked) {{
|
|
obj.parentNode.className = 'absent';
|
|
}} else {{
|
|
obj.parentNode.className = 'present';
|
|
}}
|
|
}}
|
|
function on_toggled(obj, etudid, dat) {{
|
|
colorize(obj);
|
|
if (obj.checked) {{
|
|
ajaxFunction('add', etudid, dat);
|
|
}} else {{
|
|
ajaxFunction('remove', etudid, dat);
|
|
}}
|
|
}}
|
|
</script>
|
|
<div id="AjaxDiv"></div>
|
|
<br>
|
|
<table rules="cols" frame="box" class="abs_form_table">
|
|
<tr><th class="formabs_contetud">{len(etuds)} étudiants</th>
|
|
"""
|
|
]
|
|
# Dates
|
|
odates = [datetime.date(*[int(x) for x in d.split("-")]) for d in dates]
|
|
begin = dates[0]
|
|
end = dates[-1]
|
|
# Titres colonnes
|
|
noms_jours = [] # eg [ "Lundi", "mardi", "Samedi", ... ]
|
|
jn = sco_abs.day_names()
|
|
for d in odates:
|
|
idx_jour = d.weekday()
|
|
noms_jours.append(jn[idx_jour])
|
|
for jour in noms_jours:
|
|
H.append(
|
|
f"""<th colspan="2" width="100px" style="padding-left: 5px; padding-right: 5px;">
|
|
{ jour }
|
|
</th>"""
|
|
)
|
|
H.append("</tr><tr><td> </td>")
|
|
for d in odates:
|
|
H.append(
|
|
f"""<th colspan="2" width="100px" style="padding-left: 5px; padding-right: 5px;">
|
|
{ d.strftime("%d/%m/%Y") }
|
|
</th>"""
|
|
)
|
|
H.append("</tr><tr><td> </td>")
|
|
H.append("<th>AM</th><th>PM</th>" * len(dates))
|
|
H.append("</tr>")
|
|
#
|
|
if not etuds:
|
|
H.append(
|
|
'<tr><td><span class="redboldtext">Aucun étudiant inscrit !</span></td></tr>'
|
|
)
|
|
i = 1
|
|
cnx = ndb.GetDBConnexion()
|
|
cursor = cnx.cursor(cursor_factory=ndb.ScoDocCursor)
|
|
for etud in etuds:
|
|
i += 1
|
|
etudid = etud["etudid"]
|
|
etud_class = "etudinfo" # css
|
|
# UE capitalisee dans semestre courant ?
|
|
cap = []
|
|
if etud["cursem"]:
|
|
formsemestre = FormSemestre.query.get_or_404(
|
|
etud["cursem"]["formsemestre_id"]
|
|
)
|
|
nt: NotesTableCompat = res_sem.load_formsemestre_results(formsemestre)
|
|
for ue in nt.get_ues_stat_dict():
|
|
ue_status = nt.get_etud_ue_status(etudid, ue["ue_id"])
|
|
if ue_status and ue_status["is_capitalized"]:
|
|
cap.append(ue["acronyme"])
|
|
if cap:
|
|
capstr = ' <span class="capstr">(%s cap.)</span>' % ", ".join(cap)
|
|
else:
|
|
capstr = ""
|
|
if etud["etatincursem"] == "D":
|
|
capstr += ' <span class="capstr">(dém.)</span>'
|
|
etud_class += " etuddem"
|
|
tr_class = ("row_1", "row_2", "row_3")[i % 3]
|
|
td_matin_class = ("matin_1", "matin_2", "matin_3")[i % 3]
|
|
|
|
H.append(
|
|
'<tr class="%s"><td><b class="%s" id="%s"><a class="discretelink" href="%s" target="new">%s</a></b>%s</td>'
|
|
% (
|
|
tr_class,
|
|
etud_class,
|
|
etudid,
|
|
url_for("scolar.ficheEtud", scodoc_dept=g.scodoc_dept, etudid=etudid),
|
|
etud["nomprenom"],
|
|
capstr,
|
|
)
|
|
)
|
|
etud_abs = sco_abs.list_abs_in_range(
|
|
etudid, begin, end, moduleimpl_id=moduleimpl_id, cursor=cursor
|
|
)
|
|
for d in odates:
|
|
date = d.strftime("%Y-%m-%d")
|
|
# matin
|
|
is_abs = {"jour": d, "matin": True} in etud_abs
|
|
if is_abs:
|
|
checked = "checked"
|
|
else:
|
|
checked = ""
|
|
# bulle lors du passage souris
|
|
coljour = sco_abs.DAYNAMES[(calendar.weekday(d.year, d.month, d.day))]
|
|
datecol = coljour + " " + d.strftime("%d/%m/%Y")
|
|
bulle_am = '"' + etud["nomprenom"] + " - " + datecol + ' (matin)"'
|
|
bulle_pm = '"' + etud["nomprenom"] + " - " + datecol + ' (ap.midi)"'
|
|
|
|
H.append(
|
|
'<td class="%s"><a title=%s><input type="checkbox" name="abslist:list" value="%s" %s onclick="on_toggled(this, \'%s\', \'%s\')"/></a></td>'
|
|
% (
|
|
td_matin_class,
|
|
bulle_am,
|
|
str(etudid) + ":" + date + ":" + "am",
|
|
checked,
|
|
etudid,
|
|
date + ":am",
|
|
)
|
|
)
|
|
# après-midi
|
|
is_abs = {"jour": d, "matin": False} in etud_abs
|
|
if is_abs:
|
|
checked = "checked"
|
|
else:
|
|
checked = ""
|
|
H.append(
|
|
'<td><a title=%s><input type="checkbox" name="abslist:list" value="%s" %s onclick="on_toggled(this, \'%s\', \'%s\')"/></a></td>'
|
|
% (
|
|
bulle_pm,
|
|
str(etudid) + ":" + date + ":" + "pm",
|
|
checked,
|
|
etudid,
|
|
date + ":pm",
|
|
)
|
|
)
|
|
H.append("</tr>")
|
|
H.append("</table>")
|
|
# place la liste des etudiants et les dates pour pouvoir effacer les absences
|
|
H.append(
|
|
'<input type="hidden" name="etudids" value="%s"/>'
|
|
% ",".join([str(etud["etudid"]) for etud in etuds])
|
|
)
|
|
H.append('<input type="hidden" name="datedebut" value="%s"/>' % dates[0])
|
|
H.append('<input type="hidden" name="datefin" value="%s"/>' % dates[-1])
|
|
H.append('<input type="hidden" name="dates" value="%s"/>' % ",".join(dates))
|
|
H.append(
|
|
'<input type="hidden" name="destination" value="%s"/>'
|
|
% urllib.parse.quote(destination)
|
|
)
|
|
#
|
|
# version pour formulaire avec AJAX (Yann LB)
|
|
H.append(
|
|
"""
|
|
</p>
|
|
</form>
|
|
</p>
|
|
</td></tr></table>
|
|
<p class="help">Les cases cochées correspondent à des absences.
|
|
Les absences saisies ne sont pas justifiées (sauf si un justificatif a été entré
|
|
par ailleurs).
|
|
</p><p class="help">Si vous "décochez" une case, l'absence correspondante sera supprimée.
|
|
Attention, les modifications sont automatiquement entregistrées au fur et à mesure.
|
|
</p>
|
|
"""
|
|
)
|
|
return H
|
|
|
|
|
|
@bp.route("/EtatAbsencesGr")
|
|
@scodoc
|
|
@permission_required(Permission.ScoView)
|
|
@scodoc7func # ported from dtml
|
|
def EtatAbsencesGr(
|
|
group_ids=[], # list of groups to display
|
|
debut="",
|
|
fin="",
|
|
with_boursier=True, # colonne boursier
|
|
format="html",
|
|
):
|
|
"""Liste les absences de groupes"""
|
|
datedebut = ndb.DateDMYtoISO(debut)
|
|
datefin = ndb.DateDMYtoISO(fin)
|
|
# Informations sur les groupes à afficher:
|
|
groups_infos = sco_groups_view.DisplayedGroupsInfos(group_ids)
|
|
formsemestre_id = groups_infos.formsemestre_id
|
|
sem = groups_infos.formsemestre
|
|
|
|
# Construit tableau (etudid, statut, nomprenom, nbJust, nbNonJust, NbTotal)
|
|
T = []
|
|
for m in groups_infos.members:
|
|
etud = sco_etud.get_etud_info(etudid=m["etudid"], filled=True)[0]
|
|
nbabs = sco_abs.count_abs(etudid=etud["etudid"], debut=datedebut, fin=datefin)
|
|
nbabsjust = sco_abs.count_abs_just(
|
|
etudid=etud["etudid"], debut=datedebut, fin=datefin
|
|
)
|
|
nbjustifs_noabs = len(
|
|
sco_abs.list_abs_justifs(
|
|
etudid=etud["etudid"],
|
|
datedebut=datedebut,
|
|
datefin=datefin,
|
|
only_no_abs=True,
|
|
)
|
|
)
|
|
# retrouve sem dans etud['sems']
|
|
s = None
|
|
for s in etud["sems"]:
|
|
if s["formsemestre_id"] == formsemestre_id:
|
|
break
|
|
if not s or s["formsemestre_id"] != formsemestre_id:
|
|
raise ValueError(
|
|
"EtatAbsencesGr: can't retreive sem"
|
|
) # bug or malicious arg
|
|
T.append(
|
|
{
|
|
"etudid": etud["etudid"],
|
|
"etatincursem": s["ins"]["etat"],
|
|
"nomprenom": etud["nomprenom"],
|
|
"nbabsjust": nbabsjust,
|
|
"nbabsnonjust": nbabs - nbabsjust,
|
|
"nbabs": nbabs,
|
|
"nbjustifs_noabs": nbjustifs_noabs,
|
|
"_nomprenom_target": "CalAbs?etudid=%s" % etud["etudid"],
|
|
"_nomprenom_td_attrs": 'id="%s" class="etudinfo"' % etud["etudid"],
|
|
"boursier": "oui" if etud["boursier"] else "non",
|
|
}
|
|
)
|
|
if s["ins"]["etat"] == "D":
|
|
T[-1]["_css_row_class"] = "etuddem"
|
|
T[-1]["nomprenom"] += " (dem)"
|
|
columns_ids = [
|
|
"nomprenom",
|
|
"nbjustifs_noabs",
|
|
"nbabsjust",
|
|
"nbabsnonjust",
|
|
"nbabs",
|
|
]
|
|
if with_boursier:
|
|
columns_ids[1:1] = ["boursier"]
|
|
if groups_infos.tous_les_etuds_du_sem:
|
|
gr_tit = ""
|
|
else:
|
|
if len(groups_infos.group_ids) > 1:
|
|
p = "des groupes"
|
|
else:
|
|
p = "du groupe"
|
|
if format == "html":
|
|
h = ' <span class="fontred">' + groups_infos.groups_titles + "</span>"
|
|
else:
|
|
h = groups_infos.groups_titles
|
|
gr_tit = p + h
|
|
|
|
title = f"État des absences {gr_tit}"
|
|
if format == "xls" or format == "xml" or format == "json":
|
|
columns_ids = ["etudid"] + columns_ids
|
|
# --- Formulaire choix dates début / fin
|
|
form_date = (
|
|
f"""
|
|
<form action="{groups_infos.base_url}" method="get">
|
|
<input type="hidden" name="group_ids" value="{group_ids}">
|
|
<span style="font-size: 120%"><b>Période du
|
|
<input type="text" name="debut" size="10" value="{debut}" class="datepicker"
|
|
onchange="validate_date(this);">
|
|
au
|
|
<input type="text" name="fin" size="10" value="{fin}" class="datepicker"
|
|
onchange="validate_date(this);">
|
|
</b></span>
|
|
|
|
(nombre de <em>demi-journées</em>)
|
|
</form>"""
|
|
+ """
|
|
<script>
|
|
function validate_date(el) {
|
|
const regex = /^[0-3]?[0-9]\/[0-9]{1,2}\/[0-9]{1,4}$/;
|
|
if (regex.test(el.value)) {
|
|
return el.form.submit();
|
|
}
|
|
return false;
|
|
}
|
|
</script>
|
|
"""
|
|
)
|
|
tab = GenTable(
|
|
columns_ids=columns_ids,
|
|
rows=T,
|
|
preferences=sco_preferences.SemPreferences(formsemestre_id),
|
|
titles={
|
|
"etatincursem": "Etat",
|
|
"nomprenom": "Nom",
|
|
"nbabsjust": "Justifiées",
|
|
"nbabsnonjust": "Non justifiées",
|
|
"nbabs": "Total",
|
|
"nbjustifs_noabs": "Justifs non utilisés",
|
|
"boursier": "Bourse",
|
|
},
|
|
html_sortable=True,
|
|
html_class="table_leftalign",
|
|
html_header=html_sco_header.sco_header(
|
|
page_title=title,
|
|
init_qtip=True,
|
|
javascripts=["js/etud_info.js"],
|
|
),
|
|
html_title=html_sco_header.html_sem_header(title, with_page_header=False)
|
|
+ form_date,
|
|
# "<p>Période du %s au %s (nombre de <b>demi-journées</b>)<br>" % (debut, fin),
|
|
base_url="%s&formsemestre_id=%s&debut=%s&fin=%s"
|
|
% (groups_infos.base_url, formsemestre_id, debut, fin),
|
|
filename="etat_abs_"
|
|
+ scu.make_filename(
|
|
"%s de %s" % (groups_infos.groups_filename, sem["titreannee"])
|
|
),
|
|
caption=title,
|
|
html_next_section="""</table>
|
|
<p class="help">
|
|
Justifs non utilisés: nombre de demi-journées avec justificatif mais sans absences relevées.
|
|
</p>
|
|
<p class="help">
|
|
Cliquez sur un nom pour afficher le calendrier des absences<br>
|
|
ou entrez une date pour visualiser les absents un jour donné :
|
|
</p>
|
|
<div style="margin-bottom: 10px;">
|
|
<form action="EtatAbsencesDate" method="get" action="%s">
|
|
<input type="hidden" name="formsemestre_id" value="%s">
|
|
%s
|
|
<input type="text" name="date" size="10" class="datepicker"/>
|
|
<input type="submit" name="" value="visualiser les absences">
|
|
</form></div>
|
|
"""
|
|
% (request.base_url, formsemestre_id, groups_infos.get_form_elem()),
|
|
)
|
|
return tab.make_page(format=format)
|
|
|
|
|
|
@bp.route("/EtatAbsencesDate")
|
|
@scodoc
|
|
@permission_required(Permission.ScoView)
|
|
@scodoc7func
|
|
def EtatAbsencesDate(group_ids=[], date=None): # list of groups to display
|
|
# ported from dtml
|
|
"""Etat des absences pour un groupe à une date donnée"""
|
|
# Informations sur les groupes à afficher:
|
|
groups_infos = sco_groups_view.DisplayedGroupsInfos(group_ids)
|
|
H = [html_sco_header.sco_header(page_title="Etat des absences")]
|
|
if date:
|
|
dateiso = ndb.DateDMYtoISO(date)
|
|
nbetud = 0
|
|
t_nbabsjustam = 0
|
|
t_nbabsam = 0
|
|
t_nbabsjustpm = 0
|
|
t_nbabspm = 0
|
|
H.append(f"<h2>État des absences le {date}</h2>")
|
|
H.append(
|
|
"""<table border="0" cellspacing="4" cellpadding="0">
|
|
<tr>
|
|
<th> </th>
|
|
<th style="width: 10em;">Matin</th>
|
|
<th style="width: 10em;">Après-midi</th>
|
|
</tr>
|
|
"""
|
|
)
|
|
for etud in groups_infos.members:
|
|
nbabsam = sco_abs.count_abs(
|
|
etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=1
|
|
)
|
|
nbabspm = sco_abs.count_abs(
|
|
etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=0
|
|
)
|
|
if (nbabsam != 0) or (nbabspm != 0):
|
|
nbetud += 1
|
|
nbabsjustam = sco_abs.count_abs_just(
|
|
etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=1
|
|
)
|
|
nbabsjustpm = sco_abs.count_abs_just(
|
|
etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=0
|
|
)
|
|
H.append(
|
|
f"""<tr bgcolor="#FFFFFF">
|
|
<td>
|
|
<a href="{url_for("absences.CalAbs",
|
|
scodoc_dept=g.scodoc_dept, etudid=etud["etudid"])
|
|
}"><font color="#A00000">{etud["nomprenom"]}</font></a>
|
|
</td>
|
|
<td align="center">
|
|
"""
|
|
)
|
|
if nbabsam != 0:
|
|
if nbabsjustam:
|
|
H.append("Just.")
|
|
t_nbabsjustam += 1
|
|
else:
|
|
H.append("Abs.")
|
|
t_nbabsam += 1
|
|
else:
|
|
H.append("")
|
|
H.append('</td><td align="center">')
|
|
if nbabspm != 0:
|
|
if nbabsjustpm:
|
|
H.append("Just.")
|
|
t_nbabsjustam += 1
|
|
else:
|
|
H.append("Abs.")
|
|
t_nbabspm += 1
|
|
else:
|
|
H.append("")
|
|
H.append("</td></tr>")
|
|
H.append(
|
|
f"""<tr bgcolor="#FFFFFF">
|
|
<td></td>
|
|
<td>{t_nbabsam} abs, {t_nbabsjustam} just.</td>
|
|
<td>{t_nbabspm} abs, {t_nbabsjustpm} just.</td>
|
|
</tr>
|
|
</table>
|
|
"""
|
|
)
|
|
if nbetud == 0:
|
|
H.append("<p>Aucune absence !</p>")
|
|
else:
|
|
H.append(
|
|
"""<h2>Erreur: vous n'avez pas choisi de date !</h2>
|
|
"""
|
|
)
|
|
|
|
return "\n".join(H) + html_sco_header.sco_footer()
|
|
|
|
|
|
# ----- Gestion des "billets d'absence": signalement par les etudiants eux mêmes (à travers le portail)
|
|
@bp.route("/AddBilletAbsence", methods=["GET", "POST"]) # API ScoDoc 7 compat
|
|
@scodoc
|
|
@permission_required_compat_scodoc7(Permission.ScoAbsAddBillet)
|
|
@scodoc7func
|
|
def AddBilletAbsence(
|
|
begin,
|
|
end,
|
|
description,
|
|
etudid=None,
|
|
code_nip=None,
|
|
code_ine=None,
|
|
justified=True,
|
|
format="json",
|
|
xml_reply=True, # deprecated
|
|
):
|
|
"""Mémorise un "billet"
|
|
begin et end sont au format ISO (eg "1999-01-08 04:05:06")
|
|
"""
|
|
log("Warning: calling deprecated AddBilletAbsence")
|
|
begin = str(begin)
|
|
end = str(end)
|
|
code_nip = str(code_nip) if code_nip else None
|
|
|
|
etud = api.tools.get_etud(etudid=etudid, nip=code_nip, ine=code_ine)
|
|
# check dates
|
|
begin_date = dateutil.parser.isoparse(begin) # may raises ValueError
|
|
end_date = dateutil.parser.isoparse(end)
|
|
if begin_date > end_date:
|
|
raise ValueError("invalid dates")
|
|
#
|
|
justified = bool(justified)
|
|
xml_reply = bool(xml_reply)
|
|
if xml_reply: # backward compat
|
|
format = "xml"
|
|
#
|
|
billet = BilletAbsence(
|
|
etudid=etud.id,
|
|
abs_begin=begin,
|
|
abs_end=end,
|
|
description=description,
|
|
etat=False,
|
|
justified=justified,
|
|
)
|
|
db.session.add(billet)
|
|
db.session.commit()
|
|
|
|
# Renvoie le nouveau billet au format demandé
|
|
table = sco_abs_billets.table_billets([billet], etud=etud)
|
|
log(f"AddBilletAbsence: new billet_id={billet.id}")
|
|
return table.make_page(format=format)
|
|
|
|
|
|
@bp.route("/add_billets_absence_form", methods=["GET", "POST"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsAddBillet)
|
|
@scodoc7func
|
|
def add_billets_absence_form(etudid):
|
|
"""Formulaire ajout billet (pour tests seulement, le vrai
|
|
formulaire accessible aux etudiants étant sur le portail étudiant).
|
|
"""
|
|
etud = sco_etud.get_etud_info(filled=True, etudid=etudid)[0]
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title="Billet d'absence de %s" % etud["nomprenom"]
|
|
)
|
|
]
|
|
tf = TrivialFormulator(
|
|
request.base_url,
|
|
scu.get_request_args(),
|
|
(
|
|
("etudid", {"input_type": "hidden"}),
|
|
("begin", {"input_type": "datedmy"}),
|
|
("end", {"input_type": "datedmy"}),
|
|
(
|
|
"justified",
|
|
{"input_type": "boolcheckbox", "default": 0, "title": "Justifiée"},
|
|
),
|
|
("description", {"input_type": "textarea"}),
|
|
),
|
|
)
|
|
if tf[0] == 0:
|
|
return "\n".join(H) + tf[1] + html_sco_header.sco_footer()
|
|
elif tf[0] == -1:
|
|
return flask.redirect(scu.ScoURL())
|
|
else:
|
|
e = tf[2]["begin"].split("/")
|
|
begin = e[2] + "-" + e[1] + "-" + e[0] + " 00:00:00"
|
|
e = tf[2]["end"].split("/")
|
|
end = e[2] + "-" + e[1] + "-" + e[0] + " 00:00:00"
|
|
log(
|
|
AddBilletAbsence(
|
|
begin,
|
|
end,
|
|
tf[2]["description"],
|
|
etudid=etudid,
|
|
xml_reply=True,
|
|
justified=tf[2]["justified"],
|
|
)
|
|
)
|
|
return flask.redirect("billets_etud?etudid=" + str(etudid))
|
|
|
|
|
|
@bp.route("/billets_etud/<int:etudid>")
|
|
@scodoc
|
|
@permission_required(Permission.ScoView)
|
|
def billets_etud(etudid=False, format=False):
|
|
"""Liste billets pour un étudiant"""
|
|
fmt = format or request.args.get("format", "html")
|
|
if not fmt in {"html", "json", "xml", "xls", "xlsx"}:
|
|
return ScoValueError("Format invalide")
|
|
table = sco_abs_billets.table_billets_etud(etudid)
|
|
if table:
|
|
return table.make_page(format=fmt)
|
|
return ""
|
|
|
|
|
|
# DEEPRECATED: pour compat anciens clients PHP
|
|
@bp.route("/XMLgetBilletsEtud", methods=["GET", "POST"])
|
|
@scodoc
|
|
@permission_required_compat_scodoc7(Permission.ScoView)
|
|
@scodoc7func
|
|
def XMLgetBilletsEtud(etudid=False, code_nip=False):
|
|
"""Liste billets pour un etudiant"""
|
|
log("Warning: called deprecated XMLgetBilletsEtud")
|
|
if etudid is False:
|
|
etud = Identite.query.filter_by(
|
|
code_nip=str(code_nip), dept_id=g.scodoc_dept_id
|
|
).first_or_404()
|
|
etudid = etud.id
|
|
table = sco_abs_billets.table_billets_etud(etudid)
|
|
if table:
|
|
return table.make_page(format="xml")
|
|
return ""
|
|
|
|
|
|
@bp.route("/list_billets", methods=["GET"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoView)
|
|
@scodoc7func
|
|
def list_billets():
|
|
"""Page liste des billets non traités pour tous les étudiants du département
|
|
et formulaire recherche d'un billet.
|
|
"""
|
|
table = sco_abs_billets.table_billets_etud(etat=False)
|
|
T = table.html()
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title="Billet d'absence non traités",
|
|
javascripts=["js/etud_info.js"],
|
|
init_qtip=True,
|
|
),
|
|
f"<h2>Billets d'absence en attente de traitement ({table.get_nb_rows()})</h2>",
|
|
]
|
|
|
|
tf = TrivialFormulator(
|
|
request.base_url,
|
|
scu.get_request_args(),
|
|
(("billet_id", {"input_type": "text", "title": "Numéro du billet :"}),),
|
|
method="get",
|
|
submitbutton=False,
|
|
)
|
|
if tf[0] == 0:
|
|
return "\n".join(H) + tf[1] + T + html_sco_header.sco_footer()
|
|
else:
|
|
return flask.redirect(
|
|
url_for(
|
|
"absences.process_billet_absence_form",
|
|
billet_id=tf[2]["billet_id"],
|
|
scodoc_dept=g.scodoc_dept,
|
|
)
|
|
)
|
|
|
|
|
|
@bp.route("/delete_billets_absence", methods=["POST", "GET"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsChange)
|
|
@scodoc7func
|
|
def delete_billets_absence(billet_id, dialog_confirmed=False):
|
|
"""Supprime un billet."""
|
|
billet: BilletAbsence = (
|
|
BilletAbsence.query.filter_by(id=billet_id)
|
|
.join(Identite)
|
|
.filter_by(dept_id=g.scodoc_dept_id)
|
|
.first_or_404()
|
|
)
|
|
if not dialog_confirmed:
|
|
tab = sco_abs_billets.table_billets([billet])
|
|
return scu.confirm_dialog(
|
|
"""<h2>Supprimer ce billet ?</h2>""" + tab.html(),
|
|
dest_url="",
|
|
cancel_url="list_billets",
|
|
parameters={"billet_id": billet_id},
|
|
)
|
|
|
|
db.session.delete(billet)
|
|
db.session.commit()
|
|
|
|
flash("Billet supprimé")
|
|
return flask.redirect(url_for("absences.list_billets", scodoc_dept=g.scodoc_dept))
|
|
|
|
|
|
def _ProcessBilletAbsence(
|
|
billet: BilletAbsence, estjust: bool, description: str
|
|
) -> int:
|
|
"""Traite un billet: ajoute absence(s) et éventuellement justificatifs,
|
|
et change l'état du billet à True.
|
|
return: nombre de demi-journées d'absence ajoutées, -1 si billet déjà traité.
|
|
NB: actuellement, les heures ne sont utilisées que pour déterminer
|
|
si matin et/ou après-midi.
|
|
"""
|
|
if billet.etat:
|
|
log(f"billet deja traite: {billet} !")
|
|
return -1
|
|
n = 0 # nombre de demi-journées d'absence ajoutées
|
|
|
|
# 1-- Ajout des absences (et justifs)
|
|
datedebut = billet.abs_begin.strftime("%d/%m/%Y")
|
|
datefin = billet.abs_end.strftime("%d/%m/%Y")
|
|
dates = sco_abs.DateRangeISO(datedebut, datefin)
|
|
# commence après-midi ?
|
|
if dates and billet.abs_begin.hour > 11:
|
|
sco_abs.add_absence(
|
|
billet.etudid,
|
|
dates[0],
|
|
0,
|
|
estjust,
|
|
description=description,
|
|
)
|
|
n += 1
|
|
dates = dates[1:]
|
|
# termine matin ?
|
|
if dates and billet.abs_end.hour < 12:
|
|
sco_abs.add_absence(
|
|
billet.etudid,
|
|
dates[-1],
|
|
1,
|
|
estjust,
|
|
description=description,
|
|
)
|
|
n += 1
|
|
dates = dates[:-1]
|
|
|
|
for jour in dates:
|
|
sco_abs.add_absence(
|
|
billet.etudid,
|
|
jour,
|
|
0,
|
|
estjust,
|
|
description=description,
|
|
)
|
|
sco_abs.add_absence(
|
|
billet.etudid,
|
|
jour,
|
|
1,
|
|
estjust,
|
|
description=description,
|
|
)
|
|
n += 2
|
|
|
|
# 2- Change état du billet
|
|
billet.etat = True
|
|
db.session.add(billet)
|
|
db.session.commit()
|
|
return n
|
|
|
|
|
|
@bp.route("/process_billet_absence_form", methods=["POST", "GET"])
|
|
@scodoc
|
|
@permission_required(Permission.ScoAbsChange)
|
|
@scodoc7func
|
|
def process_billet_absence_form(billet_id: int):
|
|
"""Formulaire traitement d'un billet"""
|
|
if not isinstance(billet_id, int):
|
|
raise abort(404, "billet_id invalide")
|
|
billet: BilletAbsence = (
|
|
BilletAbsence.query.filter_by(id=billet_id)
|
|
.join(Identite)
|
|
.filter_by(dept_id=g.scodoc_dept_id)
|
|
.first()
|
|
)
|
|
if billet is None:
|
|
raise ScoValueError(
|
|
f"Aucun billet avec le numéro <tt>{billet_id}</tt> dans ce département.",
|
|
dest_url=url_for("absences.list_billets", scodoc_dept=g.scodoc_dept),
|
|
)
|
|
etud = billet.etudiant
|
|
|
|
H = [
|
|
html_sco_header.sco_header(
|
|
page_title=f"Traitement billet d'absence de {etud.nomprenom}",
|
|
),
|
|
f"""<h2>Traitement du billet {billet.id} : <a class="discretelink" href="{
|
|
url_for("scolar.ficheEtud", scodoc_dept=g.scodoc_dept, etudid=etud.id)
|
|
}">{etud.nomprenom}</a></h2>
|
|
""",
|
|
]
|
|
|
|
tf = TrivialFormulator(
|
|
request.base_url,
|
|
scu.get_request_args(),
|
|
(
|
|
("billet_id", {"input_type": "hidden"}),
|
|
(
|
|
"etudid",
|
|
{"input_type": "hidden"},
|
|
),
|
|
(
|
|
"estjust",
|
|
{"input_type": "boolcheckbox", "title": "Absences justifiées"},
|
|
),
|
|
("description", {"input_type": "text", "size": 42, "title": "Raison"}),
|
|
),
|
|
initvalues={
|
|
"description": billet.description or "",
|
|
"estjust": billet.justified,
|
|
"etudid": etud.id,
|
|
},
|
|
submitlabel="Enregistrer ces absences",
|
|
)
|
|
if tf[0] == 0:
|
|
tab = sco_abs_billets.table_billets([billet], etud=etud)
|
|
H.append(tab.html())
|
|
if billet.justified:
|
|
H.append(
|
|
"""<p>L'étudiant pense pouvoir justifier cette absence.<br>
|
|
<em>Vérifiez le justificatif avant d'enregistrer.</em></p>"""
|
|
)
|
|
F = f"""<p><a class="stdlink" href="{
|
|
url_for("absences.delete_billets_absence",
|
|
scodoc_dept=g.scodoc_dept, billet_id=billet_id)
|
|
}">Supprimer ce billet</a>
|
|
(utiliser en cas d'erreur, par ex. billet en double)
|
|
</p>
|
|
<p><a class="stdlink" href="{
|
|
url_for("absences.list_billets", scodoc_dept=g.scodoc_dept)
|
|
}">Liste de tous les billets en attente</a>
|
|
</p>
|
|
"""
|
|
|
|
return "\n".join(H) + "<br>" + tf[1] + F + html_sco_header.sco_footer()
|
|
elif tf[0] == -1:
|
|
return flask.redirect(scu.ScoURL())
|
|
else:
|
|
n = _ProcessBilletAbsence(billet, tf[2]["estjust"], tf[2]["description"])
|
|
if tf[2]["estjust"]:
|
|
j = "justifiées"
|
|
else:
|
|
j = "non justifiées"
|
|
H.append('<div class="head_message">')
|
|
if n > 0:
|
|
H.append("%d absences (1/2 journées) %s ajoutées" % (n, j))
|
|
elif n == 0:
|
|
H.append("Aucun jour d'absence dans les dates indiquées !")
|
|
elif n < 0:
|
|
H.append("Ce billet avait déjà été traité !")
|
|
H.append(
|
|
f"""</div><p><a class="stdlink" href="{
|
|
url_for("absences.list_billets", scodoc_dept=g.scodoc_dept)
|
|
}">Autre billets en attente</a>
|
|
</p>
|
|
<h4>Billets déclarés par {etud.nomprenom}</h4>
|
|
"""
|
|
)
|
|
billets = (
|
|
BilletAbsence.query.filter_by(etudid=etud.id)
|
|
.join(Identite)
|
|
.filter_by(dept_id=g.scodoc_dept_id)
|
|
)
|
|
tab = sco_abs_billets.table_billets(billets, etud=etud)
|
|
H.append(tab.html())
|
|
return "\n".join(H) + html_sco_header.sco_footer()
|
|
|
|
|
|
@bp.route("/XMLgetAbsEtud", methods=["GET", "POST"]) # pour compat anciens clients PHP
|
|
@scodoc
|
|
@permission_required_compat_scodoc7(Permission.ScoView)
|
|
@scodoc7func
|
|
def XMLgetAbsEtud(beg_date="", end_date=""):
|
|
"""returns list of absences in date interval"""
|
|
t0 = time.time()
|
|
etuds = sco_etud.get_etud_info(filled=False)
|
|
if not etuds:
|
|
raise APIInvalidParams("étudiant inconnu")
|
|
etud = etuds[0]
|
|
exp = re.compile(r"^(\d{4})\D?(0[1-9]|1[0-2])\D?([12]\d|0[1-9]|3[01])$")
|
|
if not exp.match(beg_date):
|
|
raise ScoValueError("invalid date: %s" % beg_date)
|
|
if not exp.match(end_date):
|
|
raise ScoValueError("invalid date: %s" % end_date)
|
|
|
|
abs_list = sco_abs.list_abs_date(etud["etudid"], beg_date, end_date)
|
|
|
|
doc = ElementTree.Element(
|
|
"absences", etudid=str(etud["etudid"]), beg_date=beg_date, end_date=end_date
|
|
)
|
|
for a in abs_list:
|
|
if a["estabs"]: # ne donne pas les justifications si pas d'absence
|
|
doc.append(
|
|
ElementTree.Element(
|
|
"abs",
|
|
begin=a["begin"],
|
|
end=a["end"],
|
|
description=a["description"],
|
|
justified=str(int(a["estjust"])),
|
|
)
|
|
)
|
|
log("XMLgetAbsEtud (%gs)" % (time.time() - t0))
|
|
data = sco_xml.XML_HEADER + ElementTree.tostring(doc).decode(scu.SCO_ENCODING)
|
|
return scu.send_file(data, mime=scu.XML_MIMETYPE, attached=False)
|