<style> .wait { position: fixed; width: 50px; height: 10px; background: #424242; top: calc(50% - 50px); left: 50%; margin-left: -25px; animation: wait 0.6s ease-out alternate infinite; } @keyframes wait { 100% { transform: translateY(-30px) rotate(360deg) } } main h2 { border-bottom: 4px solid #0069d9; grid-column: 1 / -1; margin-bottom: 2px; } main { font-family: Verdana, Geneva, Tahoma, sans-serif; margin-right: 16px; padding: 8px; border-radius: 12px; background: #717171; display: flex; flex-wrap: wrap; gap: 8px; } section { background: #dbccb8; padding: 8px; border-radius: 8px; } .moitemoite { display: grid; grid-template-columns: 1fr 1fr; gap: 8px; } @media screen and (max-width: 1000px) { .moitemoite { grid-template-columns: 1fr; } } .notes { opacity: 0.4; pointer-events: none; } .parcours, .notes { display: inline-grid; grid-template-columns: auto auto; align-items: center; grid-row-gap: 4px; grid-column-gap: 16px; margin-bottom: 16px; background: #FFF; border: 1px solid #0069d9; border-radius: 8px; padding: 8px 16px; } h3, .resultats small { grid-column: 1 / -1; margin: 4px 0; } .parcours form { display: inline; } .parcours input { padding: 4px 8px; border: 1px solid #aaa; border-radius: 4px; width: 40px; } .dropZone { border: 2px dashed #0069d9; padding: 8px 16px; text-align: center; } .notes select { padding: 4px 8px; } .donnees, .resultats { display: inline-grid; } .donnees { background: #aaa; padding: 4px; border-radius: 4px; } .donnees>* { border-radius: 4px; border: 1px solid #aaa; } .donnees>*, .resultats>*:not(h2, button) { background: #fff; outline: 1px solid #aaa; padding: 4px 8px; } .entete { background: #0069d9 !important; color: #FFF; } .criteres { text-align: right; } .criteres:not(.criteres+.criteres), .voeux:not(.voeux+.voeux), .entete+.entete { margin-left: 8px; } small { display: block; margin-bottom: 4px; padding: 2px 4px; background: #fff; border-radius: 4px; font-style: italic; } button { background: rgb(134, 179, 0); color: #FFF; font-size: 16px; padding: 16px 32px; border: none; border-radius: 4px; box-shadow: 0 2px 2px rgba(0, 0, 0, 0.26); cursor: pointer; } button:hover { color: #ddd; } button:active { transform: translateY(2px); box-shadow: initial; } .grid { display: grid; grid-template-columns: 1fr 1fr; gap: 4px; } button.autre { background: #0c9; } .resultats button { background: #90c; margin-bottom: 4px; } b { color: #000; } .moitemoite input:invalid, [contenteditable]:empty { animation: focus .5s alternate infinite; } @keyframes focus { 0% { outline: 1px solid rgba(204, 0, 153, 0) } 100% { outline: 1px solid rgba(204, 0, 153, 1) } } [data-actif] { cursor: pointer; } [data-actif=false] { background: #ccc; } [data-highlight=true] { outline: 2px solid #90c !important; border-radius: 4px; z-index: 1; } </style> <main class="moitemoite"> <div class="wait"></div> <section> <h2>Données</h2> <div class="parcours"></div> <div class="notes"></div> <div style="display:flex; gap: 8px;"> <button onclick="exportData()">Exporter les données</button> <!--<form class="dropZone"> <div><b>Importer les données</b></div> <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="#0099cc" stroke-width="2" stroke-linecap="round"><path pathlength="100" d="M21.44 11.05l-9.19 9.19a6 6 0 0 1-8.49-8.49l9.19-9.19a4 4 0 0 1 5.66 5.66l-9.2 9.19a2 2 0 0 1-2.83-2.83l8.49-8.48"></path></svg> Déposez une fichier ou <br> <label><input type="file" accept="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"></label> </form>--> </div> <small style="margin-top: 4px">💡 Vous pouvez ensuite copier / coller, dans les cases blanches ci-dessous, les "notes" et "voeux", directement depuis le tableur.</small> <small>💡 Les voeux doivent être une série d'entiers en partant de 1 - exemple : 3, 1 et 2.</small> <small>💡 Cliquez sur un étudiant pour qu'il ne fasse pas parti du tri.</small> <div class="donnees"></div> </section> <section> <h2>Résultats</h2> <div class=grid> <button class="actionTri" data-tri="notes">Trier uniquement par <b>notes</b></button> <button class="actionTri" data-tri="voeux">Trier par <b>voeux</b> puis par notes</button> </div> <small> 💡 Vous pouvez exporter les résultats pour les utiliser dans l'éditeur de groupes/partitions. </small> <div class="resultats"></div> </section> </main> <script src="{{scu.STATIC_DIR}}/libjs/xlsx-populate-1.21.0.min.js"></script> <script> /************************/ /* A générer par Scodoc */ /************************/ let choixNotes = [ "RCUE1", "RCUE2", "RCUE3", "RCUE4", "RCUE5" ] let formsemestre; let parcours = {}; let nbParcours; /*let parcours = { 1448: { nom: "Communication", places: 28, // Modifiable par l'utilisateur etudiants: [] // Résultat du tri }, 1449: { nom: "Création numérique", places: 28, // Modifiable par l'utilisateur etudiants: [] // Résultat du tri }, 1450: { nom: "Développement Web", places: 28, // Modifiable par l'utilisateur etudiants: [] // Résultat du tri } }*/ let etudiants = {}; /*let etudiants = { 123: { actif: true, nom: "Jean Bono", criteres: { 1448: 10, // Modifiable par l'utilisateur 1449: 12, // Modifiable par l'utilisateur 1450: 8 // Modifiable par l'utilisateur }, voeux: { 1448: 1, // Modifiable par l'utilisateur 1449: 3, // Modifiable par l'utilisateur 1450: 2 // Modifiable par l'utilisateur } }, 124: { actif: true, nom: "Etudiant 2", criteres: { 1448: 11, // Modifiable par l'utilisateur 1449: 12.5, // Modifiable par l'utilisateur 1450: 8 // Modifiable par l'utilisateur }, voeux: { 1448: 2, // Modifiable par l'utilisateur 1449: 1, // Modifiable par l'utilisateur 1450: 3 // Modifiable par l'utilisateur } }, 125: { actif: true, nom: "Etudiant 3", criteres: { 1448: 10, // Modifiable par l'utilisateur 1449: 12.5, // Modifiable par l'utilisateur 1450: 15 // Modifiable par l'utilisateur }, voeux: { 1448: 3, // Modifiable par l'utilisateur 1449: 2, // Modifiable par l'utilisateur 1450: 1 // Modifiable par l'utilisateur } } }*/ /************************/ /* Chargement des datas */ /************************/ go(); async function go() { document.querySelector('.wait').style.display = ""; let params = (new URL(document.location)).searchParams; formsemestre = params.get('formsemestre_id'); let parcoursRaw = await fetchData("/ScoDoc/{{formsemestre.departement.acronym}}/api/formsemestre/" + formsemestre + "/partitions"); let etudiantsRaw = await fetchData("/ScoDoc/{{formsemestre.departement.acronym}}/api/formsemestre/" + formsemestre + "/etudiants"); //let decisionsRaw = await fetchData("/ScoDoc/{{formsemestre.departement.acronym}}/api/formsemestre/" + formsemestre + "/decisions_jury"); let savedData = await getFromScodoc(); etudiantsRaw.sort((a, b) => { return (a.nom + a.prenom).localeCompare(b.nom + b.prenom) }); processParcours(parcoursRaw, savedData); processNotes(); processEtudiants(etudiantsRaw, savedData); processEvents(); document.querySelector("body").classList.add("loaded"); document.querySelector('.wait').style.display = "none"; } function fetchData(request) { return fetch(request) .then(r => { return r.json() }) .then(data => { return data; }).catch(error => { document.querySelector("main").innerHTML = "<h2>Une erreur s'est produite lors du transfert des données.</h2>"; throw 'Fin du script - données invalides'; }) } /************************/ /* Génération du front */ /************************/ function processParcours(parcoursRaw, savedData) { Object.values(parcoursRaw).forEach(partition => { if (partition.partition_name == "Parcours") { Object.values(partition.groups).forEach(group => { parcours[group.id] = { nom: group.group_name, places: savedData.parcours?.[group.id].places || 28, etudiants: [] // Résultat du tri } }) } }) nbParcours = Object.keys(parcours).length; /* Zones parcours */ let output = "<h3>Places</h3>"; Object.entries(parcours).forEach(([idParcours, dataParcours]) => { output += ` <div data-idparcours=${idParcours}>${dataParcours.nom}</div> <div><form onsubmit="return false;"><input type=number required min=0 data-idparcours="${idParcours}" value="${dataParcours.places}"></form> places</div> `; }) document.querySelector(".parcours").innerHTML = output; } function processNotes() { /* Zones critère de notes */ let output = "<h3>Appliquer les notes</h3>"; Object.entries(parcours).forEach(([idParcours, dataParcours]) => { output += ` <div data-idparcours=${idParcours}>${dataParcours.nom}</div> <div> <select data-idparcours=${idParcours}> ${( () => { let output = "<option>-</option>"; choixNotes.forEach(notes => { output += `<option value="${notes}">${notes}</option>`; }) return output; } )()} </select> </div> `; }) document.querySelector(".notes").innerHTML = output; } function processEtudiants(etudiantsRaw, savedData) { /*id123: { actif: true, nom: "Jean Bono", criteres: { 1448: 10, // Modifiable par l'utilisateur 1449: 12, // Modifiable par l'utilisateur 1450: 8 // Modifiable par l'utilisateur }, voeux: { 1448: 1, // Modifiable par l'utilisateur 1449: 3, // Modifiable par l'utilisateur 1450: 2 // Modifiable par l'utilisateur } },*/ etudiantsRaw.forEach(etudiant => { saved = savedData.etudiants?.['id' + etudiant.id] || {}; etudiants['id' + etudiant.id] = { actif: saved.actif == false ? false : true, nom: etudiant.nom + " " + etudiant.prenom, criteres: saved.criteres || {}, voeux: saved.voeux || {} }; if (Object.keys(saved).length == 0) Object.keys(parcours).forEach((idParcours, index) => { etudiants['id' + etudiant.id].criteres[idParcours] = 10; etudiants['id' + etudiant.id].voeux[idParcours] = index + 1; }) }) /* Zone étudiants */ let output = ` <div class=entete>Etudiants</div> <div class=entete style="grid-column: span ${nbParcours}">Notes</div> <div class=entete style="grid-column: span ${nbParcours}">Voeux</div>`; Object.entries(etudiants).forEach(([id, etudiant]) => { output += ` <div data-actif=${etudiant.actif}>${etudiant.nom}</div> ${display(id, etudiant, "criteres", true)} ${display(id, etudiant, "voeux", true)} `; }) document.querySelector(".donnees").style.gridTemplateColumns = `repeat(${1 + nbParcours * 2}, auto)`; document.querySelector(".donnees").innerHTML = output; } /**********************************/ /* Affichage des critères / voeux */ /**********************************/ function display(id, datas, key, editable) { let output = ""; Object.entries(datas[key]).forEach(([idParcours, data]) => { output += `<div class=${key} data-id=${id} data-idParcours=${idParcours} contenteditable=${editable}>${data}</div>`; }) return output; } /**********************************/ /* Gestion des événements */ /**********************************/ function processEvents() { document.querySelectorAll(".parcours input").forEach(e => e.addEventListener("change", changeNbParcours)); document.querySelectorAll(".notes select").forEach(e => e.addEventListener("change", changeNotes)); document.querySelectorAll(".actionTri").forEach(e => e.addEventListener("click", processData)); document.querySelectorAll(".donnees>.criteres, .donnees>.voeux").forEach(e => e.addEventListener("paste", pasteData)); document.querySelectorAll("[data-actif]").forEach(e => e.addEventListener("mousedown", event => { event.preventDefault() })); document.querySelectorAll("[data-actif]").forEach(e => e.addEventListener("click", changeActif)); document.querySelectorAll("[contenteditable]").forEach(e => e.addEventListener("keypress", changeListe)); document.querySelectorAll("[contenteditable]").forEach(e => e.addEventListener("keyup", saveLocal)); document.querySelectorAll("[contenteditable]").forEach(e => e.addEventListener("keyup", saveToScodoc)); document.querySelectorAll("[data-idparcours]").forEach(e => e.addEventListener("mouseover", setHighlight)); document.querySelectorAll("[data-idparcours]").forEach(e => e.addEventListener("mouseleave", resetHighlight)); } /***************************************/ /* Traitement des entrées utilisateurs */ /***************************************/ function changeNbParcours() { let idParcours = this.dataset.idparcours; let places = this.value; parcours[idParcours].places = places; saveToScodoc(); } function changeNotes() { let idParcours = this.dataset.idparcours; let notes = this.value; /******* SCODOC - récupérer et modifier les valeurs des notes ******/ } function changeActif() { let statut = this.dataset.actif; if (statut == "true") { statut = false; } else { statut = true; } this.dataset.actif = statut; etudiants[this.nextElementSibling.dataset.id].actif = statut; saveToScodoc(); } function changeListe(event) { let type = this.className; /* Vérification des touches */ if (type == "criteres") { if (!/[0-9.]/.test(event.key)) { event.preventDefault(); } } else { if (!/[0-9]/.test(event.key)) { event.preventDefault(); } } if (event.key == "." && /[.]/.test(this.innerText)) { event.preventDefault(); } } function saveLocal(event) { /* Enregistement des données */ let type = this.className; let parcours = this.dataset.idparcours; let id = this.dataset.id; etudiants[id][type][parcours] = parseFloat(this.innerText); } /*function importList(semestre) { console.log(parcours); /******* SCODOC - enregistrer la répartition des étudiants dans le parcours ******/ /*document.querySelector(".resultats>h2").innerHTML += `<div>✔️ Groupes importés dans le ${semestre}</div>`; }*/ function setHighlight() { document.querySelectorAll(`[data-idparcours="${this.dataset.idparcours}"]`).forEach(e => { e.dataset.highlight = true; }) } function resetHighlight() { document.querySelectorAll(`[data-highlight=true]`).forEach(e => { e.dataset.highlight = false; }) } function saveToScodoc() { let data = { parcours: parcours, etudiants: etudiants }; fetch( "/ScoDoc/{{formsemestre.departement.acronym}}/api/formsemestre/" + formsemestre + "/groups_save_auto_assignment", { method: "POST", headers: { "Content-Type": "application/json", }, body: JSON.stringify(data) } ); } async function getFromScodoc() { let dataRaw = await fetch("/ScoDoc/{{formsemestre.departement.acronym}}/api/formsemestre/" + formsemestre + "/groups_get_auto_assignment"); let data = await dataRaw.text(); if (data == "") { return {}; } return JSON.parse(data); } /***************/ /* Algo de tri */ /***************/ function processData() { let typeTri = this.dataset.tri; // Vérifications nombre de place let placesTotales = 0; Object.values(parcours).forEach(p => { placesTotales += p.places }) if (placesTotales < Object.keys(etudiants).length) { document.querySelector(".resultats").innerHTML = "<h2>⚠️ Le nombre de places est inférieur au nombre d'étudiants ⚠️</h2>"; return; } // Vérification champs non remplis if (typeTri == "notes") { var recherche = ".moitemoite input:invalid"; } else { var recherche = ".moitemoite input:invalid, [contenteditable]:empty"; } if (document.querySelector(recherche)) { document.querySelector(".resultats").innerHTML = "<h2>⚠️ Veuillez remplir tous les champs ⚠️</h2>"; return; } // Remise à 0 des résultats Object.values(parcours).map(p => p.etudiants = []); // Liste des étudiants à traiter let etudiantsID = Object.keys(etudiants); /* Pour chaque niveau de voeux */ for (let numVoeux = 1; numVoeux < nbParcours + 1; numVoeux++) { /* Ce qu'il restera à traiter au round suivant */ let aTraiter = []; etudiantsID.forEach(id => { if (etudiants[id].actif == false) return; if (typeTri == "notes") { // Récupère le parcours en fonction des notes et du round var groupe = Object.entries(etudiants[id].criteres).sort(([, a], [, b]) => b - a)[numVoeux - 1][0]; } else { // Trouve le parcours qui correspond au numéro du voeux var groupe = Object.keys(etudiants[id].voeux).find(key => etudiants[id].voeux[key] == numVoeux); } if (!groupe) { sco_message("Il semblerait que les voeux de " + etudiants[id].nom + " ne soit pas consécutifs en partant de 1."); } if (parcours[groupe].places == 0) { aTraiter.push(id); } else if (parcours[groupe].etudiants.length < parcours[groupe].places) { // Le groupe n'est pas plein, on y ajoute l'étudiant parcours[groupe].etudiants.push(id); } else { aTraiter.push(groupePlein(groupe, id)); } }) etudiantsID = aTraiter; } showResults(); } function groupePlein(groupe, id) { // Trie des étudiants dans le parcours par rapport au critère parcours[groupe].etudiants = parcours[groupe].etudiants.sort((a, b) => { return etudiants[a].criteres[groupe] - etudiants[b].criteres[groupe]; }) // Si les deux étudiants ont le même niveau : aléatoire let diff = etudiants[parcours[groupe].etudiants[0]].criteres[groupe] - etudiants[id].criteres[groupe]; if (diff == 0) { if (Math.random() < 0.5) { diff = 1; } else { diff = -1; } } // On ajoute dans le parcours le meilleur et on met à traiter l'autre if (diff < 0) { parcours[groupe].etudiants.push(id); return parcours[groupe].etudiants.shift(); } else { return id; } } /***************************/ /* Affichage des résultats */ /***************************/ function showResults() { let output = ` <button class="autre" onclick="exportResults()">Exporter les résultats</button>`; /*` <h2> Importer les résultats dans Scodoc<br> <button onclick='importList("S1")'>S1</button> <button onclick='importList("S2")'>S2</button> <button onclick='importList("S3")'>S3</button> <button onclick='importList("S4")'>S4</button> <button onclick='importList("S5")'>S5</button> <button onclick='importList("S6")'>S6</button> </h2> `*/; Object.values(parcours).forEach(p => { output += `<h2>${p.nom} - ${p.etudiants.length}</h2> <div class=entete>Etudiants</div> <div class=entete style="grid-column: span ${nbParcours}">Notes</div> <div class=entete style="grid-column: span ${nbParcours}">Voeux</div>`; p.etudiants.forEach(id => { output += `<div>${etudiants[id].nom}</div> ${display(id, etudiants[id], "criteres", false)} ${display(id, etudiants[id], "voeux", false)} `; }) }) document.querySelector(".resultats").style.gridTemplateColumns = `repeat(${1 + nbParcours * 2}, auto)`; document.querySelector(".resultats").innerHTML = output; } /*****************************/ /* Copier / coller des datas */ /*****************************/ function pasteData(event) { event.preventDefault(); let data = event.clipboardData.getData('Text'); let lignes = data.split(/\r\n|\r|\n/g); currentTarget = this; // Calcul du décalage dans la grille let offset = -1; do { offset++; currentTarget = currentTarget.previousElementSibling; } while (currentTarget.className != "") // Collage des données currentTarget = this; lignes.forEach(ligne => { let zone = ligne.split(/\t/g); for (let i = 0; i < zone.length; i++) { if (!currentTarget) return; currentTarget.innerText = zone[i]; currentTarget = currentTarget.nextElementSibling; if (currentTarget?.className == "") { for (let j = 0; j <= offset; j++) { currentTarget = currentTarget.nextElementSibling; } return; } } }) updateStudentsDatas(); } function updateStudentsDatas() { document.querySelectorAll(".criteres, .voeux").forEach(e => { etudiants[e.dataset.id][e.className][e.dataset.idparcours] = parseFloat(e.innerText); }) saveStudentsData(); } function saveStudentsData() { /******* SCODOC - enregistrer les données étudiantes ******/ console.log(etudiants); } /***************/ /* Export xlsx */ /***************/ function saveFile(name, workbook) { workbook.outputAsync() .then(function (blob) { var url = window.URL.createObjectURL(blob); var a = document.createElement("a"); document.body.appendChild(a); a.href = url; a.download = name + ".xlsx"; a.click(); window.URL.revokeObjectURL(url); document.body.removeChild(a); }); } function changeChar(char, nb) { return String.fromCharCode(char.charCodeAt(0) + nb); } async function exportData() { XlsxPopulate.fromBlankAsync() .then(workbook => { const sheet = workbook.sheet(0); sheet.name("Données"); sheet.cell("A1").value("id"); sheet.cell("B1").value("Actif"); sheet.cell("C1").value("Nom"); Object.values(parcours).forEach((p, index) => { sheet.cell(changeChar("D", index) + "1").value("Note " + p.nom); sheet.cell(changeChar("D", index + nbParcours) + "1").value("Voeu " + p.nom); sheet.cell(changeChar("D", index + nbParcours) + "2").value(p.places); }) Object.entries(etudiants).forEach(([id, etudiant], index) => { sheet.cell("A" + (index + 3)).value(id.slice(2)); sheet.cell("B" + (index + 3)).value(etudiant.actif); sheet.cell("C" + (index + 3)).value(etudiant.nom); let offset = 0; Object.values(etudiant.criteres).forEach(critere => { sheet.cell(changeChar("D", offset) + (index + 3)).value(critere); offset++; }) Object.values(etudiant.voeux).forEach(voeux => { sheet.cell(changeChar("D", offset) + (index + 3)).value(voeux); offset++; }) }) sheet.column("A").width(5); sheet.column("B").width(5); sheet.column("C").width(20); sheet.column("D").width(20); sheet.column("E").width(20); sheet.column("F").width(20); sheet.column("G").width(20); sheet.column("H").width(20); sheet.column("I").width(20); sheet.column("J").width(20); sheet.column("K").width(20); sheet.column("L").width(20); saveFile("Donnees groupes - " + formsemestre, workbook); }); } async function exportResults() { XlsxPopulate.fromBlankAsync() .then(workbook => { const sheet = workbook.sheet(0); sheet.name("Résultats"); let colonne = 1; Object.values(parcours).forEach((dataParcours, index) => { let ligne = 1; sheet.row(ligne++).cell(colonne).value(dataParcours.nom).style("bold", true); sheet.row(ligne).cell(colonne).value("id").style("bold", true); sheet.row(ligne++).cell(colonne + 1).value("Nom").style("bold", true).column().width(30); dataParcours.etudiants.forEach(etudiant => { sheet.row(ligne).cell(colonne).value(etudiant.slice(2)); sheet.row(ligne++).cell(colonne + 1).value(etudiants[etudiant].nom); }) colonne += 3; }) saveFile("Resultats groupes - " + formsemestre, workbook); }); } </script>