/* Module par Seb. L. */ class releveBUT extends HTMLElement { constructor() { super(); this.shadow = this.attachShadow({ mode: "open" }); /* Config par defaut */ this.config = { showURL: true, }; /* Template du module */ this.shadow.innerHTML = this.template(); /* Style du module */ const styles = document.createElement("link"); styles.setAttribute("rel", "stylesheet"); if (location.href.includes("ScoDoc")) { styles.setAttribute( "href", removeLastTwoComponents(getCurrentScriptPath()) + "/css/releve-but.css" ); // Scodoc } else { styles.setAttribute("href", "/assets/styles/releve-but.css"); // Passerelle } this.shadow.appendChild(styles); } listeOnOff() { this.parentElement.parentElement.classList.toggle("listeOff"); this.parentElement.parentElement .querySelectorAll(".moduleOnOff") .forEach((e) => { e.classList.remove("moduleOnOff"); }); } moduleOnOff() { this.parentElement.classList.toggle("moduleOnOff"); } goTo() { let module = this.dataset.module; this.parentElement.parentElement.parentElement.parentElement .querySelector("#Module_" + module) .scrollIntoView(); } set setConfig(config) { this.config.showURL = config.showURL ?? this.config.showURL; } set showData(data) { // this.showInformations(data); this.showSemestre(data); this.showSynthese(data); this.showEvaluations(data); this.showCustom(data); this.setOptions(data.options); this.shadow.querySelectorAll(".CTA_Liste").forEach((e) => { e.addEventListener("click", this.listeOnOff); }); this.shadow.querySelectorAll(".ue, .module").forEach((e) => { e.addEventListener("click", this.moduleOnOff); }); this.shadow .querySelectorAll(":not(.ueBonus)+.syntheseModule") .forEach((e) => { e.addEventListener("click", this.goTo); }); this.shadow.children[0].classList.add("ready"); } template() { return ` <div> <div class="wait"></div> <main class="releve"> <!---------------------------------------------------------------------------------------> <!-- Zone spéciale pour que les IUT puisse ajouter des infos locales sur la passerelle --> <!---------------------------------------------------------------------------------------> <section class=custom></section> <!---------------------------> <!-- Semestre --> <!---------------------------> <section> <h2 id="identite_etudiant"></h2> <div> <div class=infoSemestre></div> <div> <div class=decision_annee></div> <div class=decision></div> <div class="ects" id="ects_tot"></div> <div class=dateInscription>Inscrit le </div> <em>Les moyennes ci-dessus servent à situer l'étudiant dans la promotion et ne correspondent pas à des validations de compétences ou d'UE.</em> </div> </div> </section> <!---------------------------> <!-- Synthèse --> <!---------------------------> <section> <div> <div> <h2>Unités d'enseignement</h2> <em>La moyenne des ressources dans une UE dépend des poids donnés aux évaluations.</em> </div> <div class=CTA_Liste> Liste <svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" fill="none" stroke="#ffffff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <path d="M18 15l-6-6-6 6" /> </svg> </div> </div> <div class=synthese></div> </section> <!---------------------------> <!-- Evaluations --> <!---------------------------> <section> <div> <h2>Ressources</h2> <div class=CTA_Liste> Liste <svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" fill="none" stroke="#ffffff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <path d="M18 15l-6-6-6 6" /> </svg> </div> </div> <div class=evaluations></div> </section> <section> <div> <h2>Situations d'apprentissage et d'évaluation (SAÉ)</h2> <div class=CTA_Liste> Liste <svg xmlns="http://www.w3.org/2000/svg" width="26" height="26" viewBox="0 0 24 24" fill="none" stroke="#ffffff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <path d="M18 15l-6-6-6 6" /> </svg> </div> </div> <div class=sae></div> </section> </main> </div>`; } /********************************/ /* Informations sur l'étudiant */ /********************************/ showInformations(data) { this.shadow.querySelector(".studentPic").src = data.etudiant.photo_url || "default_Student.svg"; let output = ""; if (this.config.showURL) { output += `<a href="${data.etudiant.fiche_url}" class=info_etudiant>`; } else { output += `<div class=info_etudiant>`; } output += ` <div class=civilite> ${this.civilite(data.etudiant.civilite)} ${data.etudiant.nom} ${data.etudiant.prenom}`; if (data.etudiant.date_naissance) { output += ` <div class=dateNaissance>né${ data.etudiant.civilite == "F" ? "e" : "" } le ${this.ISOToDate(data.etudiant.date_naissance)}</div>`; } output += ` </div> <div class=numerosEtudiant> Numéro étudiant : ${data.etudiant.code_nip || "~"} - Code INE : ${data.etudiant.code_ine || "~"} </div> <div>${data.formation.titre}</div> `; if (this.config.showURL) { output += `</a>`; } else { output += `</div>`; } this.shadow.querySelector(".infoEtudiant").innerHTML = output; } /*******************************/ /* Affichage local */ /*******************************/ showCustom(data) { this.shadow.querySelector(".custom").innerHTML = data.custom || ""; } /*******************************/ /* Information sur le semestre */ /*******************************/ showSemestre(data) { let correspondanceCodes = { ADM: "Admis", AJD: "Admis par décision de jury", PASD: "Passage de droit : tout n'est pas validé, mais d'après les règles du BUT, vous passez", PAS1NCI: "Vous passez par décision de jury mais attention, vous n'avez pas partout le niveau suffisant", RED: "Ajourné mais autorisé à redoubler", NAR: "Non admis et non autorisé à redoubler : réorientation", DEM: "Démission", ABAN: "Abandon constaté sans lettre de démission", RAT: "En attente d'un rattrapage", EXCLU: "Exclusion dans le cadre d'une décision disciplinaire", DEF: "Défaillance : non évalué par manque d'assiduité", ABL: "Année blanche", }; this.shadow.querySelector( "#identite_etudiant" ).innerHTML = ` <a href="${data.etudiant.fiche_url}">${data.etudiant.nomprenom}</a> `; this.shadow.querySelector(".dateInscription").innerHTML += this.ISOToDate( data.semestre.inscription ); let output = ""; if (!data.options.block_moyenne_generale) { output += ` <div> <div class=enteteSemestre>Moyenne</div><div class=enteteSemestre>${data.semestre.notes.value}</div> <div class=rang>Rang :</div><div class=rang>${data.semestre.rang.value} / ${data.semestre.rang.total}</div> <div>Max. promo. :</div><div>${data.semestre.notes.max}</div> <div>Moy. promo. :</div><div>${data.semestre.notes.moy}</div> <div>Min. promo. :</div><div>${data.semestre.notes.min}</div> </div> `; } output += ` ${(() => { if ( !data.semestre.rang.groupes || Object.keys(data.semestre.rang.groupes).length == 0 ) { return ""; } let output = ""; let [idGroupe, dataGroupe] = Object.entries( data.semestre.rang.groupes )[0]; output += `<div> <div class=enteteSemestre>${data.semestre.groupes[0]?.group_name}</div><div></div> <div class=rang>Rang :</div><div class=rang>${dataGroupe.value} / ${dataGroupe.total}</div> </div>`; // <div>Max. promo. :</div><div>${dataGroupe.max || "-"}</div> // <div>Moy. promo. :</div><div>${dataGroupe.moy || "-"}</div> // <div>Min. promo. :</div><div>${dataGroupe.min || "-"}</div> return output; })()} <div class=absencesRecap> <div class=enteteSemestre>Absences</div><div class=enteteSemestre>${ data.semestre.absences?.metrique ?? "1/2 jour." }</div> <div class=abs>Non justifiées</div> <div>${data.semestre.absences?.injustifie ?? "-"}</div> <div class=abs>Total</div><div>${data.semestre.absences?.total ?? "-"}</div> </div>`; if (data.semestre.decision_rcue?.length) { output += ` <div> <div class=enteteSemestre>RCUE</div><div></div> ${(() => { let output = ""; data.semestre.decision_rcue.forEach((competence) => { output += `<div class=competence>${competence.niveau.competence.titre}</div><div>${competence.code}</div>`; }); return output; })()} </div> </div>`; } if (data.semestre.decision_ue?.length) { output += ` <div> <div class=enteteSemestre>UE</div><div></div> ${(() => { let output = ""; data.semestre.decision_ue.forEach((ue) => { output += `<div class=competence>${ue.acronyme}</div><div>${ue.code}</div>`; }); return output; })()} </div> </div>`; } output += ` <a class=photo href="${data.etudiant.fiche_url}"> <img src="${ data.etudiant.photo_url || "default_Student.svg" }" alt="photo de l'étudiant" title="fiche de l'étudiant" height="120" border="0"> </a>`; /*${data.semestre.groupes.map(groupe => { return ` <div> <div class=enteteSemestre>Groupe</div><div class=enteteSemestre>${groupe.nom}</div> <div class=rang>Rang :</div><div class=rang>${groupe.rang.value} / ${groupe.rang.total}</div> <div>Max. groupe :</div><div>${groupe.notes.max}</div> <div>Moy. groupe :</div><div>${groupe.notes.min}</div> <div>Min. groupe :</div><div>${groupe.notes.min}</div> </div> `; }).join("") }*/ this.shadow.querySelector(".infoSemestre").innerHTML = output; /*if(data.semestre.decision_annee?.code){ this.shadow.querySelector(".decision_annee").innerHTML = "Décision année : " + data.semestre.decision_annee.code + " - " + correspondanceCodes[data.semestre.decision_annee.code]; }*/ this.shadow.querySelector(".decision").innerHTML = data.semestre.situation || ""; /*if (data.semestre.decision?.code) { this.shadow.querySelector(".decision").innerHTML = "Décision jury: " + (data.semestre.decision?.code || ""); }*/ this.shadow.querySelector("#ects_tot").innerHTML = "ECTS : " + (data.semestre.ECTS?.acquis ?? "-") + " / " + (data.semestre.ECTS?.total ?? "-"); } /*******************************/ /* Synthèse */ /*******************************/ showSynthese(data) { let output = ``; /* Fusion et tri des UE et UE capitalisées */ let fusionUE = [ ...Object.entries(data.ues), ...Object.entries(data.ues_capitalisees), ].sort((a, b) => { return a[1].numero - b[1].numero; }); /* Affichage */ fusionUE.forEach(([ue, dataUE]) => { if (dataUE.type == 1) { // UE Sport / Bonus output += ` <div> <div class="ue ueBonus"> <h3>Bonus</h3> <div>${dataUE.bonus_description}</div> </div> ${this.ueSport(dataUE.modules)} </div> `; } else { output += ` <div> <div class="ue ${dataUE.date_capitalisation ? "capitalisee" : ""}"> <h3> ${ue}${dataUE.titre ? " - " + dataUE.titre : ""} </h3> <div> <div class=moyenne>Moyenne : ${ dataUE.moyenne?.value || dataUE.moyenne || "-" }</div> <div class=ue_rang>Rang : ${dataUE.moyenne?.rang} / ${ dataUE.moyenne?.total }</div> <div class=info>`; if (!dataUE.date_capitalisation) { output += ` Bonus : ${dataUE.bonus || 0} - `; if (dataUE.malus >= 0) { output += `Malus : ${dataUE.malus || 0}`; } else { output += `Bonus complémentaire : ${ -dataUE.malus || 0 }`; } } else { output += ` le ${this.ISOToDate( dataUE.date_capitalisation.split("T")[0] )} <a href="${dataUE.bul_orig_url}">dans ce semestre</a>`; } output += ` <span class=ects> - ECTS : ${dataUE.ECTS?.acquis ?? "-"} / ${ dataUE.ECTS?.total ?? "-" } </span> </div> </div>`; /*<div class=absences> <div>Abs N.J.</div><div>${dataUE.absences?.injustifie || 0}</div> <div>Total</div><div>${dataUE.absences?.total || 0}</div> </div>*/ output += "</div>"; if (!dataUE.date_capitalisation) { output += this.synthese(data, dataUE.ressources) + this.synthese(data, dataUE.saes); } output += "</div>"; } }); this.shadow.querySelector(".synthese").innerHTML = output; } synthese(data, modules) { let output = ""; Object.entries(modules).forEach(([module, dataModule]) => { let titre = data.ressources[module]?.titre || data.saes[module]?.titre; //let url = data.ressources[module]?.url || data.saes[module]?.url; output += ` <div class=syntheseModule data-module="${module.replace(/[^a-zA-Z0-9]/g, "")}"> <div>${module} - ${titre}</div> <div> ${dataModule.moyenne} <em>Coef. ${dataModule.coef}</em> </div> </div> `; }); return output; } ueSport(modules) { let output = ""; Object.values(modules).forEach((module) => { Object.values(module.evaluations).forEach((evaluation) => { output += ` <div class=syntheseModule> <div>${module.titre} - ${evaluation.description || "Note"}</div> <div> ${evaluation.note.value ?? "-"} <em>Coef. ${evaluation.coef ?? "*"}</em> </div> </div> `; }); }); return output; } /*******************************/ /* Evaluations */ /*******************************/ showEvaluations(data) { this.shadow.querySelector(".evaluations").innerHTML = this.module( data.ressources ); this.shadow.querySelector(".sae").innerHTML += this.module(data.saes); } module(module) { let output = ""; Object.entries(module).forEach(([numero, content]) => { output += ` <div id="Module_${numero.replace(/[^a-zA-Z0-9]/g, "")}"> <div class=module> <h3>${this.URL(content.url, `${numero} - ${content.titre}`)}</h3> <div> <div class=moyenne>Moyenne indicative : ${ content.moyenne.value }</div> <div class=info> Classe : ${content.moyenne.moy} - Max : ${content.moyenne.max} - Min : ${content.moyenne.min} </div> </div> <div class=absences> <div>Abs inj.</div><div>${content.absences?.injustifie || 0}</div> <div>Total</div><div>${content.absences?.total || 0}</div> </div> </div> ${this.evaluation(content.evaluations)} </div> `; }); return output; } evaluation(evaluations) { let output = ""; evaluations.forEach((evaluation) => { output += ` <div class="eval ${evaluation.evaluation_type == 3 ? "eval-bonus" : ""}"> <div>${this.URL(evaluation.url, evaluation.description || "Évaluation")}</div> <div> ${evaluation.note.value} <em>${evaluation.evaluation_type == 0 ? "Coef." : evaluation.evaluation_type == 3 ? "Bonus" : "" } ${evaluation.coef ?? ""}</em> </div> <div class=complement> <div>${evaluation.evaluation_type == 0 ? "Coef." : ""}</div><div>${evaluation.coef ?? ""}</div> <div>Max. promo.</div><div>${evaluation.note.max}</div> <div>Moy. promo.</div><div>${evaluation.note.moy}</div> <div>Min. promo.</div><div>${evaluation.note.min}</div> ${Object.entries(evaluation.poids) .map(([UE, poids]) => { return ` <div>Poids ${UE}</div> <div>${poids}</div> `; }) .join("")} </div> </div> `; }); return output; } /********************/ /* Options */ /********************/ setOptions(options) { Object.entries(options).forEach(([option, value]) => { if (value === false) { this.shadow.children[0].classList.add(option.replace("show", "hide")); } }); } /********************/ /* Fonctions d'aide */ /********************/ URL(href, content) { if (this.config.showURL) { return `<a href=${href}>${content}</a>`; } else { return content; } } civilite(txt) { switch (txt) { case "M": return "M."; case "F": return "Mme"; default: return ""; } } ISOToDate(ISO) { return ISO.split("-").reverse().join("/"); } } customElements.define("releve-but", releveBUT);