# -*- 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 # ############################################################################## """ 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(etudid, dates) CountAbs(etudid, debut, fin, matin=None, moduleimpl_id=None) CountAbsJust(etudid, debut, fin, matin=None, moduleimpl_id=None) ListeAbsJust(etudid, datedebut) [pas de fin ?] ListeAbsNonJust(etudid, datedebut) [pas de fin ?] ListeJustifs(etudid, datedebut, datefin=None, only_no_abs=True) ListeAbsJour(date, am=True, pm=True, is_abs=None, is_just=None) ListeAbsNonJustJour(date, am=True, pm=True) """ import string import re import time import datetime import dateutil import dateutil.parser import calendar import urllib import cgi import jaxml from flask import g from flask import current_app from app.decorators import ( scodoc7func, ScoDoc7Context, permission_required, admin_required, login_required, ) 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.notes_log import log from app.scodoc.scolog import logdb from app.scodoc.sco_permissions import Permission from app.scodoc.sco_exceptions import ScoValueError, ScoInvalidDateError 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_notification from app.scodoc import sco_abs_views from app.scodoc import sco_compute_moy from app.scodoc import sco_core from app.scodoc import sco_etud from app.scodoc import sco_excel from app.scodoc import sco_formsemestre from app.scodoc import sco_groups from app.scodoc import sco_groups_view from app.scodoc import sco_moduleimpl from app.scodoc import sco_preferences CSSSTYLES = html_sco_header.BOOTSTRAP_MULTISELECT_CSS context = ScoDoc7Context("absences") def sco_publish(route, function, permission): """Declare a route for a python function, protected by permission and called following ScoDoc 7 Zope standards. """ bp.route(route)(permission_required(permission)(scodoc7func(context)(function))) def _toboolean(x): "convert a value to boolean" return x # not necessary anymore ! # -------------------------------------------------------------------- # # ABSENCES (/ScoDoc//Scolarite/Absences/...) # # -------------------------------------------------------------------- sco_publish("/index_html", sco_abs_views.index_html, Permission.ScoView) 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 ) sco_publish( "/doSignaleAbsence", sco_abs_views.doSignaleAbsence, Permission.ScoAbsChange ) sco_publish( "/JustifAbsenceEtud", sco_abs_views.JustifAbsenceEtud, Permission.ScoAbsChange ) sco_publish("/doJustifAbsence", sco_abs_views.doJustifAbsence, Permission.ScoAbsChange) sco_publish( "/AnnuleAbsenceEtud", sco_abs_views.AnnuleAbsenceEtud, Permission.ScoAbsChange ) sco_publish("/doAnnuleAbsence", sco_abs_views.doAnnuleAbsence, Permission.ScoAbsChange) sco_publish("/doAnnuleJustif", sco_abs_views.doAnnuleJustif, Permission.ScoAbsChange) sco_publish( "/AnnuleAbsencesDatesNoJust", sco_abs_views.AnnuleAbsencesDatesNoJust, Permission.ScoAbsChange, ) sco_publish("/ListeAbsEtud", sco_abs_views.ListeAbsEtud, Permission.ScoView) # -------------------------------------------------------------------- # # SQL METHODS (xxx #sco8 not views => à déplacer) # # -------------------------------------------------------------------- sco_publish("/CountAbs", sco_abs.CountAbs, Permission.ScoView) sco_publish("/CountAbsJust", sco_abs.CountAbsJust, Permission.ScoView) @bp.route("/doSignaleAbsenceGrSemestre") @permission_required(Permission.ScoAbsChange) @scodoc7func(context) def doSignaleAbsenceGrSemestre( context, moduleimpl_id=None, abslist=[], dates="", etudids="", destination=None, REQUEST=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. """ if etudids: etudids = 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( context, etudid, dates, moduleimpl_id, REQUEST ) return "Absences effacées" # 2- Ajoute les absences if abslist: sco_abs._add_abslist(context, abslist, REQUEST, moduleimpl_id) return "Absences ajoutées" return "" # ------------ HTML Interfaces @bp.route("/SignaleAbsenceGrHebdo") @permission_required(Permission.ScoAbsChange) @scodoc7func(context) def SignaleAbsenceGrHebdo( context, datelundi, group_ids=[], destination="", moduleimpl_id=None, REQUEST=None ): "Saisie hebdomadaire des absences" if not moduleimpl_id: moduleimpl_id = None groups_infos = sco_groups_view.DisplayedGroupsInfos( context, group_ids, moduleimpl_id=moduleimpl_id, REQUEST=REQUEST ) if not groups_infos.members: return ( html_sco_header.sco_header( context, page_title="Saisie des absences", REQUEST=REQUEST ) + "

Aucun étudiant !

" + html_sco_header.sco_footer(REQUEST) ) base_url = "SignaleAbsenceGrHebdo?datelundi=%s&%s&destination=%s" % ( datelundi, groups_infos.groups_query_args, urllib.quote(destination), ) formsemestre_id = groups_infos.formsemestre_id require_module = sco_preferences.get_preference( context, "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( context.Notes, 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 = sco_core.get_notes_cache(context).get_NotesTable( context.Notes, formsemestre_id ) sem = sco_formsemestre.do_formsemestre_list( context, {"formsemestre_id": formsemestre_id} )[0] # calcule dates jours de cette semaine # liste de dates iso "yyyy-mm-dd" datessem = [ndb.DateDMYtoISO(datelundi)] for _ in sco_abs.day_names(context)[1:]: datessem.append(sco_abs.next_iso_day(context, 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 + ' ' + groups_infos.groups_titles + "" H = [ html_sco_header.sco_header( context, 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, REQUEST=REQUEST, ), """

Saisie des absences %s %s, semaine du lundi %s

Groupes: %s
""" % ( gr_tit, sem["titre_num"], datelundi, groups_infos.formsemestre_id, datelundi, destination, moduleimpl_id or "", sco_groups_view.menu_groups_choice( context, groups_infos, submit_on_change=True ), ), ] # modimpls_list = [] # Initialize with first student ues = nt.get_ues(etudid=etuds[0]["etudid"]) for ue in ues: modimpls_list += nt.get_modimpls(ue_id=ue["ue_id"]) # Add modules other students are subscribed to for etud in etuds[1:]: modimpls_etud = [] ues = nt.get_ues(etudid=etud["etudid"]) for ue in ues: modimpls_etud += nt.get_modimpls(ue_id=ue["ue_id"]) modimpls_list += [m for m in modimpls_etud if m not in modimpls_list] menu_module = "" for modimpl in modimpls_list: if modimpl["moduleimpl_id"] == moduleimpl_id: sel = "selected" else: sel = "" menu_module += ( """\n""" % { "modimpl_id": modimpl["moduleimpl_id"], "modname": modimpl["module"]["code"] + " " + (modimpl["module"]["abbrev"] or modimpl["module"]["titre"]), "sel": sel, } ) if moduleimpl_id: sel = "" else: sel = "selected" # aucun module specifie H.append( """Module concerné:
""" % {"menu_module": menu_module, "url": base_url, "sel": sel} ) H += _gen_form_saisie_groupe( context, etuds, datessem, destination, moduleimpl_id, require_module ) H.append(html_sco_header.sco_footer(REQUEST)) return "\n".join(H) @bp.route("/SignaleAbsenceGrSemestre") @permission_required(Permission.ScoAbsChange) @scodoc7func(context) def SignaleAbsenceGrSemestre( context, datedebut, datefin, destination="", group_ids=[], # list of groups to display nbweeks=4, # ne montre que les nbweeks dernieres semaines moduleimpl_id=None, REQUEST=None, ): """Saisie des absences sur une journée sur un semestre (ou intervalle de dates) entier""" groups_infos = sco_groups_view.DisplayedGroupsInfos( context, group_ids, REQUEST=REQUEST ) if not groups_infos.members: return ( html_sco_header.sco_header( context, page_title="Saisie des absences", REQUEST=REQUEST ) + "

Aucun étudiant !

" + html_sco_header.sco_footer(REQUEST) ) formsemestre_id = groups_infos.formsemestre_id require_module = sco_preferences.get_preference( context, "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( context.Notes, 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.quote(destination), ) ) base_url = base_url_noweeks + "&nbweeks=%s" % nbweeks # sans le moduleimpl_id if etuds: nt = sco_core.get_notes_cache(context).get_NotesTable( context.Notes, formsemestre_id ) sem = sco_formsemestre.do_formsemestre_list( context, {"formsemestre_id": formsemestre_id} )[0] work_saturday = sco_abs.is_work_saturday(context) 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() 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(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=" + moduleimpl_id # dates = [x.ISO() for x in dates] dayname = sco_abs.day_names(context)[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 + '' + groups_infos.groups_titles + "" H = [ html_sco_header.sco_header( context, page_title="Saisie des absences", init_qtip=True, javascripts=["js/etud_info.js", "js/abs_ajax.js"], no_side_bar=1, REQUEST=REQUEST, ), """

Saisie des absences %s %s, les %s

%s """ % (gr_tit, sem["titre_num"], dayname, url_link_semaines, msg), ] # if etuds: modimpls_list = [] # Initialize with first student ues = nt.get_ues(etudid=etuds[0]["etudid"]) for ue in ues: modimpls_list += nt.get_modimpls(ue_id=ue["ue_id"]) # Add modules other students are subscribed to for etud in etuds[1:]: modimpls_etud = [] ues = nt.get_ues(etudid=etud["etudid"]) for ue in ues: modimpls_etud += nt.get_modimpls(ue_id=ue["ue_id"]) modimpls_list += [m for m in modimpls_etud if m not in modimpls_list] menu_module = "" for modimpl in modimpls_list: if modimpl["moduleimpl_id"] == moduleimpl_id: sel = "selected" else: sel = "" menu_module += ( """\n""" % { "modimpl_id": modimpl["moduleimpl_id"], "modname": modimpl["module"]["code"] + " " + (modimpl["module"]["abbrev"] or modimpl["module"]["titre"]), "sel": sel, } ) if moduleimpl_id: sel = "" else: sel = "selected" # aucun module specifie H.append( """

Module concerné par ces absences (%(optionel_txt)s):

""" % { "menu_module": menu_module, "url": base_url, "sel": sel, "optionel_txt": 'requis' if require_module else "optionnel", } ) H += _gen_form_saisie_groupe( context, etuds, dates, destination, moduleimpl_id, require_module ) H.append(html_sco_header.sco_footer(REQUEST)) return "\n".join(H) def _gen_form_saisie_groupe( context, 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 = [ """

""" % ( "true" if (require_module and not moduleimpl_id) else "false", len(etuds), ) ] # 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(context) for d in odates: idx_jour = d.weekday() noms_jours.append(jn[idx_jour]) for jour in noms_jours: H.append( '" ) H.append("") for d in odates: H.append( '" ) H.append("") H.append("" * len(dates)) H.append("") # if not etuds: H.append( '' ) i = 1 cnx = ndb.GetDBConnexion() cursor = cnx.cursor(cursor_factory=ndb.ScoDocCursor) for etud in etuds: i += 1 etudid = etud["etudid"] # UE capitalisee dans semestre courant ? cap = [] if etud["cursem"]: nt = sco_core.get_notes_cache(context).get_NotesTable( context.Notes, etud["cursem"]["formsemestre_id"] ) # > get_ues, get_etud_ue_status for ue in nt.get_ues(): status = nt.get_etud_ue_status(etudid, ue["ue_id"]) if status["is_capitalized"]: cap.append(ue["acronyme"]) if cap: capstr = ' (%s cap.)' % ", ".join(cap) else: capstr = "" 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, etudid, etudid, etud["nomprenom"], capstr) ) etud_abs = sco_abs.list_abs_in_range( context, 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_matin_class, bulle_am, 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( '' % ( bulle_pm, etudid + ":" + date + ":" + "pm", checked, etudid, date + ":pm", ) ) H.append("") H.append("
%d étudiants' + jour + "
 ' + d.strftime("%d/%m/%Y") + "
 AMPM
Aucun étudiant inscrit !
%s%s
") # place la liste des etudiants et les dates pour pouvoir effacer les absences H.append( '' % ",".join([etud["etudid"] for etud in etuds]) ) H.append('' % dates[0]) H.append('' % dates[-1]) H.append('' % ",".join(dates)) H.append( '' % urllib.quote(destination) ) # # version pour formulaire avec AJAX (Yann LB) H.append( """

Les cases cochées correspondent à des absences. Les absences saisies ne sont pas justifiées (sauf si un justificatif a été entré par ailleurs).

Si vous "décochez" une case, l'absence correspondante sera supprimée. Attention, les modifications sont automatiquement entregistrées au fur et à mesure.

""" % destination ) return H @bp.route("/EtatAbsencesGr") @permission_required(Permission.ScoView) @scodoc7func(context) # ported from dtml def EtatAbsencesGr( context, group_ids=[], # list of groups to display debut="", fin="", with_boursier=True, # colonne boursier format="html", REQUEST=None, ): """Liste les absences de groupes""" datedebut = ndb.DateDMYtoISO(debut) datefin = ndb.DateDMYtoISO(fin) # Informations sur les groupes à afficher: groups_infos = sco_groups_view.DisplayedGroupsInfos( context, group_ids, REQUEST=REQUEST ) 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.CountAbs( context, etudid=etud["etudid"], debut=datedebut, fin=datefin ) nbabsjust = context.CountAbsJust( etudid=etud["etudid"], debut=datedebut, fin=datefin ) nbjustifs_noabs = len( sco_abs.ListeJustifs( context, etudid=etud["etudid"], datedebut=datedebut, 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": etud["boursier"], } ) 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 = ' ' + groups_infos.groups_titles + "" else: h = groups_infos.groups_titles gr_tit = p + h title = "Etat des absences %s" % gr_tit if format == "xls" or format == "xml" or format == "json": columns_ids = ["etudid"] + columns_ids tab = GenTable( columns_ids=columns_ids, rows=T, preferences=sco_preferences.SemPreferences(context, 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( context, REQUEST, page_title=title, init_qtip=True, javascripts=["js/etud_info.js"], ), html_title=context.Notes.html_sem_header( REQUEST, "%s" % title, sem, with_page_header=False ) + "

Période du %s au %s (nombre de demi-journées)
" % (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="""

Justifs non utilisés: nombre de demi-journées avec justificatif mais sans absences relevées.

Cliquez sur un nom pour afficher le calendrier des absences
ou entrez une date pour visualiser les absents un jour donné :

%s
""" % (REQUEST.URL0, formsemestre_id, groups_infos.get_form_elem()), ) return tab.make_page(context, format=format, REQUEST=REQUEST) @bp.route("/EtatAbsencesDate") @permission_required(Permission.ScoView) @scodoc7func(context) def EtatAbsencesDate( context, group_ids=[], date=None, REQUEST=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( context, group_ids, REQUEST=REQUEST ) H = [ html_sco_header.sco_header( context, page_title="Etat des absences", REQUEST=REQUEST ) ] if date: dateiso = ndb.DateDMYtoISO(date) nbetud = 0 t_nbabsjustam = 0 t_nbabsam = 0 t_nbabsjustpm = 0 t_nbabspm = 0 H.append("

État des absences le %s

" % date) H.append( """ """ ) for etud in groups_infos.members: nbabsam = sco_abs.CountAbs( context, etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=1 ) nbabspm = sco_abs.CountAbs( context, etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=0 ) if (nbabsam != 0) or (nbabspm != 0): nbetud += 1 nbabsjustam = context.CountAbsJust( etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=1 ) nbabsjustpm = context.CountAbsJust( etudid=etud["etudid"], debut=dateiso, fin=dateiso, matin=0 ) H.append( """") H.append( """""" % (t_nbabsam, t_nbabsjustam, t_nbabspm, t_nbabsjustpm) ) H.append("
  MatinAprès-midi
%(nomprenom)s""" % etud ) # """ if nbabsam != 0: if nbabsjustam: H.append("Just.") t_nbabsjustam += 1 else: H.append("Abs.") t_nbabsam += 1 else: H.append("") H.append('') if nbabspm != 0: if nbabsjustpm: H.append("Just.") t_nbabsjustam += 1 else: H.append("Abs.") t_nbabspm += 1 else: H.append("") H.append("
%d abs, %d just.%d abs, %d just.
") if nbetud == 0: H.append("

Aucune absence !

") else: H.append( """

Erreur: vous n'avez pas choisi de date !

Continuer""" % REQUEST.HTTP_REFERER ) return "\n".join(H) + html_sco_header.sco_footer(REQUEST) # ----- Gestion des "billets d'absence": signalement par les etudiants eux mêmes (à travers le portail) @bp.route("/AddBilletAbsence") @permission_required(Permission.ScoAbsAddBillet) @scodoc7func(context) def AddBilletAbsence( context, begin, end, description, etudid=False, code_nip=None, code_ine=None, justified=True, REQUEST=None, xml_reply=True, ): """Memorise un "billet" begin et end sont au format ISO (eg "1999-01-08 04:05:06") """ t0 = time.time() # check etudid etuds = sco_etud.get_etud_info( etudid=etudid, code_nip=code_nip, REQUEST=REQUEST, filled=True ) if not etuds: return scu.log_unknown_etud(context, REQUEST=REQUEST) etud = etuds[0] # 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 = int(justified) # cnx = ndb.GetDBConnexion() billet_id = sco_abs.billet_absence_create( cnx, { "etudid": etud["etudid"], "abs_begin": begin, "abs_end": end, "description": description, "etat": 0, "justified": justified, }, ) if xml_reply: # Renvoie le nouveau billet en XML if REQUEST: REQUEST.RESPONSE.setHeader("content-type", scu.XML_MIMETYPE) billets = sco_abs.billet_absence_list(cnx, {"billet_id": billet_id}) tab = _tableBillets(context, billets, etud=etud) log("AddBilletAbsence: new billet_id=%s (%gs)" % (billet_id, time.time() - t0)) return tab.make_page(context, REQUEST=REQUEST, format="xml") else: return billet_id @bp.route("/AddBilletAbsenceForm") @permission_required(Permission.ScoAbsAddBillet) @scodoc7func(context) def AddBilletAbsenceForm(context, etudid, REQUEST=None): """Formulaire ajout billet (pour tests seulement, le vrai formulaire accessible aux etudiants étant sur le portail étudiant). """ etud = sco_etud.get_etud_info(etudid=etudid, filled=1, REQUEST=REQUEST)[0] H = [ html_sco_header.sco_header( context, REQUEST, page_title="Billet d'absence de %s" % etud["nomprenom"] ) ] tf = TrivialFormulator( REQUEST.URL0, REQUEST.form, ( ("etudid", {"input_type": "hidden"}), ("begin", {"input_type": "date"}), ("end", {"input_type": "date"}), ( "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(REQUEST) elif tf[0] == -1: return REQUEST.RESPONSE.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( context, begin, end, tf[2]["description"], etudid=etudid, xml_reply=True, justified=tf[2]["justified"], ) ) return REQUEST.RESPONSE.redirect("listeBilletsEtud?etudid=" + etudid) def _tableBillets(context, billets, etud=None, title=""): for b in billets: if b["abs_begin"].hour < 12: m = " matin" else: m = " après-midi" b["abs_begin_str"] = b["abs_begin"].strftime("%d/%m/%Y") + m if b["abs_end"].hour < 12: m = " matin" else: m = " après-midi" b["abs_end_str"] = b["abs_end"].strftime("%d/%m/%Y") + m if b["etat"] == 0: if b["justified"] == 0: b["etat_str"] = "à traiter" else: b["etat_str"] = "à justifier" b["_etat_str_target"] = ( "ProcessBilletAbsenceForm?billet_id=%s" % b["billet_id"] ) if etud: b["_etat_str_target"] += "&etudid=%s" % etud["etudid"] b["_billet_id_target"] = b["_etat_str_target"] else: b["etat_str"] = "ok" if not etud: # ajoute info etudiant e = sco_etud.get_etud_info(etudid=b["etudid"], filled=1) if not e: b["nomprenom"] = "???" # should not occur else: b["nomprenom"] = e[0]["nomprenom"] b["_nomprenom_target"] = "ficheEtud?etudid=%s" % b["etudid"] if etud and not title: title = "Billets d'absence déclarés par %(nomprenom)s" % etud else: title = title columns_ids = ["billet_id"] if not etud: columns_ids += ["nomprenom"] columns_ids += ["abs_begin_str", "abs_end_str", "description", "etat_str"] tab = GenTable( titles={ "billet_id": "Numéro", "abs_begin_str": "Début", "abs_end_str": "Fin", "description": "Raison de l'absence", "etat_str": "Etat", }, columns_ids=columns_ids, page_title=title, html_title="

%s

" % title, preferences=sco_preferences.SemPreferences( context, ), rows=billets, html_sortable=True, ) return tab @bp.route("/listeBilletsEtud") @permission_required(Permission.ScoView) @scodoc7func(context) def listeBilletsEtud(context, etudid=False, REQUEST=None, format="html"): """Liste billets pour un etudiant""" etuds = sco_etud.get_etud_info(etudid=etudid, filled=1, REQUEST=REQUEST) if not etuds: return scu.log_unknown_etud(context, format=format, REQUEST=REQUEST) etud = etuds[0] cnx = ndb.GetDBConnexion() billets = sco_abs.billet_absence_list(cnx, {"etudid": etud["etudid"]}) tab = _tableBillets(context, billets, etud=etud) return tab.make_page(context, REQUEST=REQUEST, format=format) @bp.route("/XMLgetBilletsEtud") @permission_required(Permission.ScoView) @scodoc7func(context) def XMLgetBilletsEtud(context, etudid=False, REQUEST=None): """Liste billets pour un etudiant""" if not sco_preferences.get_preference(context, "handle_billets_abs"): return "" t0 = time.time() r = listeBilletsEtud(context, etudid, REQUEST=REQUEST, format="xml") log("XMLgetBilletsEtud (%gs)" % (time.time() - t0)) return r @bp.route("/listeBillets") @permission_required(Permission.ScoView) @scodoc7func(context) def listeBillets(context, REQUEST=None): """Page liste des billets non traités et formulaire recherche d'un billet""" cnx = ndb.GetDBConnexion() billets = sco_abs.billet_absence_list(cnx, {"etat": 0}) tab = _tableBillets(context, billets) T = tab.html() H = [ html_sco_header.sco_header( context, REQUEST, page_title="Billet d'absence non traités" ), "

Billets d'absence en attente de traitement (%d)

" % len(billets), ] tf = TrivialFormulator( REQUEST.URL0, REQUEST.form, (("billet_id", {"input_type": "text", "title": "Numéro du billet"}),), submitbutton=False, ) if tf[0] == 0: return "\n".join(H) + tf[1] + T + html_sco_header.sco_footer(REQUEST) else: return REQUEST.RESPONSE.redirect( "ProcessBilletAbsenceForm?billet_id=" + tf[2]["billet_id"] ) @bp.route("/deleteBilletAbsence") @permission_required(Permission.ScoAbsChange) @scodoc7func(context) def deleteBilletAbsence(context, billet_id, REQUEST=None, dialog_confirmed=False): """Supprime un billet.""" cnx = ndb.GetDBConnexion() billets = sco_abs.billet_absence_list(cnx, {"billet_id": billet_id}) if not billets: return REQUEST.RESPONSE.redirect( "listeBillets?head_message=Billet%%20%s%%20inexistant !" % billet_id ) if not dialog_confirmed: tab = _tableBillets(context, billets) return context.confirmDialog( """

Supprimer ce billet ?

""" + tab.html(), dest_url="", REQUEST=REQUEST, cancel_url="listeBillets", parameters={"billet_id": billet_id}, ) sco_abs.billet_absence_delete(cnx, billet_id) return REQUEST.RESPONSE.redirect("listeBillets?head_message=Billet%20supprimé") def _ProcessBilletAbsence(context, billet, estjust, description, REQUEST): """Traite un billet: ajoute absence(s) et éventuellement justificatifs, et change l'état du billet à 1. NB: actuellement, les heures ne sont utilisées que pour déterminer si matin et/ou après-midi. """ cnx = ndb.GetDBConnexion() if billet["etat"] != 0: log("billet=%s" % billet) log("billet deja traité !") 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(context, datedebut, datefin) # commence après-midi ? if dates and billet["abs_begin"].hour > 11: sco_abs.add_absence( context, billet["etudid"], dates[0], 0, estjust, REQUEST, description=description, ) n += 1 dates = dates[1:] # termine matin ? if dates and billet["abs_end"].hour < 12: sco_abs.add_absence( context, billet["etudid"], dates[-1], 1, estjust, REQUEST, description=description, ) n += 1 dates = dates[:-1] for jour in dates: sco_abs.add_absence( context, billet["etudid"], jour, 0, estjust, REQUEST, description=description, ) sco_abs.add_absence( context, billet["etudid"], jour, 1, estjust, REQUEST, description=description, ) n += 2 # 2- change etat du billet sco_abs.billet_absence_edit(cnx, {"billet_id": billet["billet_id"], "etat": 1}) return n @bp.route("/ProcessBilletAbsenceForm") @permission_required(Permission.ScoAbsChange) @scodoc7func(context) def ProcessBilletAbsenceForm(context, billet_id, REQUEST=None): """Formulaire traitement d'un billet""" cnx = ndb.GetDBConnexion() billets = sco_abs.billet_absence_list(cnx, {"billet_id": billet_id}) if not billets: return REQUEST.RESPONSE.redirect( "listeBillets?head_message=Billet%%20%s%%20inexistant !" % billet_id ) billet = billets[0] etudid = billet["etudid"] etud = sco_etud.get_etud_info(etudid=etudid, filled=1, REQUEST=REQUEST)[0] H = [ html_sco_header.sco_header( context, REQUEST, page_title="Traitement billet d'absence de %s" % etud["nomprenom"], ), '

Traitement du billet %s : %s

' % (billet_id, etudid, etud["nomprenom"]), ] tf = TrivialFormulator( REQUEST.URL0, REQUEST.form, ( ("billet_id", {"input_type": "hidden"}), ( "etudid", {"input_type": "hidden"}, ), # pour centrer l'UI sur l'étudiant ( "estjust", {"input_type": "boolcheckbox", "title": "Absences justifiées"}, ), ("description", {"input_type": "text", "size": 42, "title": "Raison"}), ), initvalues={ "description": billet["description"], "estjust": billet["justified"], "etudid": etudid, }, submitlabel="Enregistrer ces absences", ) if tf[0] == 0: tab = _tableBillets(context, [billet], etud=etud) H.append(tab.html()) if billet["justified"] == 1: H.append( """

L'étudiant pense pouvoir justifier cette absence.
Vérifiez le justificatif avant d'enregistrer.

""" ) F = ( """

Supprimer ce billet (utiliser en cas d'erreur, par ex. billet en double)

""" % billet_id ) F += '

Liste de tous les billets en attente

' return "\n".join(H) + "
" + tf[1] + F + html_sco_header.sco_footer(REQUEST) elif tf[0] == -1: return REQUEST.RESPONSE.redirect(scu.ScoURL()) else: n = _ProcessBilletAbsence( context, billet, tf[2]["estjust"], tf[2]["description"], REQUEST ) if tf[2]["estjust"]: j = "justifiées" else: j = "non justifiées" H.append('
') 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( '

Autre billets en attente

Billets déclarés par %s

' % (etud["nomprenom"]) ) billets = sco_abs.billet_absence_list(cnx, {"etudid": etud["etudid"]}) tab = _tableBillets(context, billets, etud=etud) H.append(tab.html()) return "\n".join(H) + html_sco_header.sco_footer(REQUEST) @bp.route("/XMLgetAbsEtud") @permission_required(Permission.ScoView) @scodoc7func(context) def XMLgetAbsEtud(context, beg_date="", end_date="", REQUEST=None): """returns list of absences in date interval""" t0 = time.time() etud = sco_etud.get_etud_info(REQUEST=REQUEST)[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 = sco_abs.ListeAbsDate(context, etud["etudid"], beg_date, end_date) REQUEST.RESPONSE.setHeader("content-type", scu.XML_MIMETYPE) doc = jaxml.XML_document(encoding=scu.SCO_ENCODING) doc.absences(etudid=etud["etudid"], beg_date=beg_date, end_date=end_date) doc._push() for a in Abs: if a["estabs"]: # ne donne pas les justifications si pas d'absence doc._push() doc.abs( begin=a["begin"], end=a["end"], description=a["description"], justified=a["estjust"], ) doc._pop() doc._pop() log("XMLgetAbsEtud (%gs)" % (time.time() - t0)) return repr(doc) context.populate(globals())