From 17fea2272f5c765b26d19fe5c793238240bfb744 Mon Sep 17 00:00:00 2001 From: Thibaut Valentin Date: Fri, 20 Jun 2025 16:31:35 +0200 Subject: [PATCH] feat: club membre export --- .../ffsaf/domain/service/MembreService.java | 12 +- .../ffsaf/rest/MembreClubEndpoints.java | 16 ++ .../rest/data/SimpleMembreInOutData.java | 40 ++++ src/main/webapp/package-lock.json | 185 +++++++++++++++- src/main/webapp/package.json | 4 +- src/main/webapp/src/pages/MemberList.jsx | 199 +++++++++++++++++- 6 files changed, 448 insertions(+), 8 deletions(-) create mode 100644 src/main/java/fr/titionfire/ffsaf/rest/data/SimpleMembreInOutData.java diff --git a/src/main/java/fr/titionfire/ffsaf/domain/service/MembreService.java b/src/main/java/fr/titionfire/ffsaf/domain/service/MembreService.java index 69694df..87614d4 100644 --- a/src/main/java/fr/titionfire/ffsaf/domain/service/MembreService.java +++ b/src/main/java/fr/titionfire/ffsaf/domain/service/MembreService.java @@ -10,6 +10,7 @@ import fr.titionfire.ffsaf.net2.request.SReqComb; import fr.titionfire.ffsaf.rest.data.MeData; import fr.titionfire.ffsaf.rest.data.SimpleLicence; import fr.titionfire.ffsaf.rest.data.SimpleMembre; +import fr.titionfire.ffsaf.rest.data.SimpleMembreInOutData; import fr.titionfire.ffsaf.rest.exception.DBadRequestException; import fr.titionfire.ffsaf.rest.exception.DForbiddenException; import fr.titionfire.ffsaf.rest.exception.DNotFoundException; @@ -108,8 +109,8 @@ public class MembreService { return repository.find("userId = ?1", subject).firstResult() .chain(membreModel -> { PanacheQuery query = repository.find( - "club = ?1 AND (" + FIND_NAME_REQUEST + ")", - Sort.ascending("fname", "lname"), membreModel.getClub(), finalSearch) + "club = ?2 AND (" + FIND_NAME_REQUEST + ")", + Sort.ascending("fname", "lname"), finalSearch, membreModel.getClub()) .page(Page.ofSize(limit)); return getPageResult(query, limit, page); }); @@ -130,6 +131,13 @@ public class MembreService { .invoke(result::setResult)); } + public Uni> getAllExport(String subject) { + return repository.find("userId = ?1", subject).firstResult() + .chain(membreModel -> repository.list("club = ?1", membreModel.getClub())) + .chain(membres -> licenceRepository.list("saison = ?1 AND membre IN ?2", Utils.getSaison(), membres) + .map(l -> membres.stream().map(m -> SimpleMembreInOutData.fromModel(m, l)).toList())); + } + public Uni getById(long id) { return repository.findById(id); } diff --git a/src/main/java/fr/titionfire/ffsaf/rest/MembreClubEndpoints.java b/src/main/java/fr/titionfire/ffsaf/rest/MembreClubEndpoints.java index aa9b49e..81ef479 100644 --- a/src/main/java/fr/titionfire/ffsaf/rest/MembreClubEndpoints.java +++ b/src/main/java/fr/titionfire/ffsaf/rest/MembreClubEndpoints.java @@ -2,6 +2,7 @@ package fr.titionfire.ffsaf.rest; import fr.titionfire.ffsaf.domain.service.MembreService; import fr.titionfire.ffsaf.rest.data.SimpleMembre; +import fr.titionfire.ffsaf.rest.data.SimpleMembreInOutData; import fr.titionfire.ffsaf.rest.exception.DInternalError; import fr.titionfire.ffsaf.rest.from.ClubMemberForm; import fr.titionfire.ffsaf.rest.from.FullMemberForm; @@ -21,6 +22,8 @@ import org.eclipse.microprofile.openapi.annotations.responses.APIResponse; import org.eclipse.microprofile.openapi.annotations.responses.APIResponses; import org.eclipse.microprofile.openapi.annotations.tags.Tag; +import java.util.List; + @Tag(name = "Membre club", description = "Gestion des membres (pour les clubs)") @RolesAllowed({"club_president", "club_secretaire", "club_respo_intra"}) @Path("api/member") @@ -55,6 +58,19 @@ public class MembreClubEndpoints { return membreService.search(limit, page - 1, search, securityCtx.getSubject()); } + @GET + @Path("club/export") + @Produces(MediaType.APPLICATION_JSON) + @Operation(summary = "Exporte les membres du club", description = "Exporte les membres du club") + @APIResponses(value = { + @APIResponse(responseCode = "200", description = "Les membres du club ont été exportés avec succès"), + @APIResponse(responseCode = "403", description = "Accès refusé"), + @APIResponse(responseCode = "500", description = "Erreur interne du serveur") + }) + public Uni> exportMembre() { + return membreService.getAllExport(securityCtx.getSubject()); + } + @PUT @Path("club/{id}") @Produces(MediaType.TEXT_PLAIN) diff --git a/src/main/java/fr/titionfire/ffsaf/rest/data/SimpleMembreInOutData.java b/src/main/java/fr/titionfire/ffsaf/rest/data/SimpleMembreInOutData.java new file mode 100644 index 0000000..c31a174 --- /dev/null +++ b/src/main/java/fr/titionfire/ffsaf/rest/data/SimpleMembreInOutData.java @@ -0,0 +1,40 @@ +package fr.titionfire.ffsaf.rest.data; + +import fr.titionfire.ffsaf.data.model.LicenceModel; +import fr.titionfire.ffsaf.data.model.MembreModel; +import io.quarkus.runtime.annotations.RegisterForReflection; +import lombok.AllArgsConstructor; +import lombok.Data; + +import java.util.Date; +import java.util.List; + +@Data +@RegisterForReflection +@AllArgsConstructor +public class SimpleMembreInOutData { + Integer licence; + String nom; + String prenom; + String email; + String genre; + Date birthdate; + boolean licenceCurrent; + String certif; + + public static SimpleMembreInOutData fromModel(MembreModel membreModel, List lc) { + LicenceModel currentLicence = lc.stream().filter(l -> l.getMembre().getId().equals(membreModel.getId())) + .findFirst().orElse(null); + + return new SimpleMembreInOutData( + membreModel.getLicence(), + membreModel.getLname(), + membreModel.getFname(), + membreModel.getEmail(), + membreModel.getGenre().str, + membreModel.getBirth_date(), + currentLicence != null, + currentLicence == null ? null : currentLicence.getCertificate() + ); + } +} diff --git a/src/main/webapp/package-lock.json b/src/main/webapp/package-lock.json index b83100b..9cf6137 100644 --- a/src/main/webapp/package-lock.json +++ b/src/main/webapp/package-lock.json @@ -24,7 +24,9 @@ "react-loader-spinner": "^6.1.6", "react-router-dom": "^6.21.2", "react-toastify": "^10.0.4", - "recharts": "^2.15.1" + "recharts": "^2.15.1", + "xlsx": "^0.18.5", + "xlsx-js-style": "^1.2.0" }, "devDependencies": { "@types/react": "^18.2.43", @@ -1410,6 +1412,14 @@ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, + "node_modules/adler-32": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/adler-32/-/adler-32-1.3.1.tgz", + "integrity": "sha512-ynZ4w/nUUv5rrsR8UUGoe1VC9hZj6V5hU9Qw1HlMDJGEJw5S7TfTErWTjMys6M7vr0YWcPqs3qAr4ss0nDfP+A==", + "engines": { + "node": ">=0.8" + } + }, "node_modules/ajv": { "version": "6.12.6", "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", @@ -1698,6 +1708,18 @@ } ] }, + "node_modules/cfb": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/cfb/-/cfb-1.2.2.tgz", + "integrity": "sha512-KfdUZsSOw19/ObEWasvBP/Ac4reZvAGauZhs6S/gqNhXhI7cKwvlH7ulj+dOEYnca4bm4SGo8C1bTAQvnTjgQA==", + "dependencies": { + "adler-32": "~1.3.0", + "crc-32": "~1.2.0" + }, + "engines": { + "node": ">=0.8" + } + }, "node_modules/chalk": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", @@ -1720,6 +1742,14 @@ "node": ">=6" } }, + "node_modules/codepage": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/codepage/-/codepage-1.15.0.tgz", + "integrity": "sha512-3g6NUTPd/YtuuGrhMnOMRjFc+LJw/bnMp3+0r/Wcz3IXUuCosKRJvMphm5+Q+bvTVGcJJuRvVLuYba+WojaFaA==", + "engines": { + "node": ">=0.8" + } + }, "node_modules/color-convert": { "version": "1.9.3", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", @@ -1746,6 +1776,11 @@ "node": ">= 0.8" } }, + "node_modules/commander": { + "version": "2.17.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.17.1.tgz", + "integrity": "sha512-wPMUt6FnH2yzG95SA6mzjQOEKUU3aLaDEmzs1ti+1E9h+CsrZghRlqEM/EJ4KscsQVG8uNN4uVreUeT8+drlgg==" + }, "node_modules/concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", @@ -1758,6 +1793,17 @@ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", "dev": true }, + "node_modules/crc-32": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz", + "integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==", + "bin": { + "crc32": "bin/crc32.njs" + }, + "engines": { + "node": ">=0.8" + } + }, "node_modules/cross-spawn": { "version": "7.0.3", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", @@ -2477,6 +2523,14 @@ "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==" }, + "node_modules/exit-on-epipe": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/exit-on-epipe/-/exit-on-epipe-1.0.1.tgz", + "integrity": "sha512-h2z5mrROTxce56S+pnvAV890uu7ls7f1kEvVGJbw1OlFH3/mlJ5bkXu0KRyW94v37zzHPiUd55iLn3DA7TjWpw==", + "engines": { + "node": ">=0.8" + } + }, "node_modules/fast-deep-equal": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", @@ -2512,6 +2566,11 @@ "reusify": "^1.0.4" } }, + "node_modules/fflate": { + "version": "0.3.11", + "resolved": "https://registry.npmjs.org/fflate/-/fflate-0.3.11.tgz", + "integrity": "sha512-Rr5QlUeGN1mbOHlaqcSYMKVpPbgLy0AWT/W0EHxA6NGI12yO1jpoui2zBBvU2G824ltM6Ut8BFgfHSBGfkmS0A==" + }, "node_modules/file-entry-cache": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", @@ -2601,6 +2660,14 @@ "node": ">= 6" } }, + "node_modules/frac": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/frac/-/frac-1.1.2.tgz", + "integrity": "sha512-w/XBfkibaTl3YDqASwfDUqkna4Z2p9cFSr1aHDt0WoMTECnRfBOv2WArlZILlqgWlmdIlALXGpM2AOhEk5W3IA==", + "engines": { + "node": ">=0.8" + } + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -3725,6 +3792,17 @@ "node": ">= 0.8.0" } }, + "node_modules/printj": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/printj/-/printj-1.1.2.tgz", + "integrity": "sha512-zA2SmoLaxZyArQTOPj5LXecR+RagfPSU5Kw1qP+jkWeNlrq+eJZyY2oS68SU1Z/7/myXM4lo9716laOFAVStCQ==", + "bin": { + "printj": "bin/printj.njs" + }, + "engines": { + "node": ">=0.8" + } + }, "node_modules/proj4": { "version": "2.11.0", "resolved": "https://registry.npmjs.org/proj4/-/proj4-2.11.0.tgz", @@ -4237,6 +4315,17 @@ "node": ">=0.10.0" } }, + "node_modules/ssf": { + "version": "0.11.2", + "resolved": "https://registry.npmjs.org/ssf/-/ssf-0.11.2.tgz", + "integrity": "sha512-+idbmIXoYET47hH+d7dfm2epdOMUDjqcB4648sTZ+t2JwoyBFL/insLfB/racrDmsKB3diwsDA696pZMieAC5g==", + "dependencies": { + "frac": "~1.1.2" + }, + "engines": { + "node": ">=0.8" + } + }, "node_modules/string.prototype.matchall": { "version": "4.0.10", "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.10.tgz", @@ -4759,12 +4848,106 @@ "resolved": "https://registry.npmjs.org/wkt-parser/-/wkt-parser-1.3.3.tgz", "integrity": "sha512-ZnV3yH8/k58ZPACOXeiHaMuXIiaTk1t0hSUVisbO0t4RjA5wPpUytcxeyiN2h+LZRrmuHIh/1UlrR9e7DHDvTw==" }, + "node_modules/wmf": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wmf/-/wmf-1.0.2.tgz", + "integrity": "sha512-/p9K7bEh0Dj6WbXg4JG0xvLQmIadrner1bi45VMJTfnbVHsc7yIajZyoSoK60/dtVBs12Fm6WkUI5/3WAVsNMw==", + "engines": { + "node": ">=0.8" + } + }, + "node_modules/word": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/word/-/word-0.3.0.tgz", + "integrity": "sha512-OELeY0Q61OXpdUfTp+oweA/vtLVg5VDOXh+3he3PNzLGG/y0oylSOC1xRVj0+l4vQ3tj/bB1HVHv1ocXkQceFA==", + "engines": { + "node": ">=0.8" + } + }, "node_modules/wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", "dev": true }, + "node_modules/xlsx": { + "version": "0.18.5", + "resolved": "https://registry.npmjs.org/xlsx/-/xlsx-0.18.5.tgz", + "integrity": "sha512-dmg3LCjBPHZnQp5/F/+nnTa+miPJxUXB6vtk42YjBBKayDNagxGEeIdWApkYPOf3Z3pm3k62Knjzp7lMeTEtFQ==", + "dependencies": { + "adler-32": "~1.3.0", + "cfb": "~1.2.1", + "codepage": "~1.15.0", + "crc-32": "~1.2.1", + "ssf": "~0.11.2", + "wmf": "~1.0.1", + "word": "~0.3.0" + }, + "bin": { + "xlsx": "bin/xlsx.njs" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/xlsx-js-style": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/xlsx-js-style/-/xlsx-js-style-1.2.0.tgz", + "integrity": "sha512-DDT4FXFSWfT4DXMSok/m3TvmP1gvO3dn0Eu/c+eXHW5Kzmp7IczNkxg/iEPnImbG9X0Vb8QhROda5eatSR/97Q==", + "dependencies": { + "adler-32": "~1.2.0", + "cfb": "^1.1.4", + "codepage": "~1.14.0", + "commander": "~2.17.1", + "crc-32": "~1.2.0", + "exit-on-epipe": "~1.0.1", + "fflate": "^0.3.8", + "ssf": "~0.11.2", + "wmf": "~1.0.1", + "word": "~0.3.0" + }, + "bin": { + "xlsx": "bin/xlsx.njs" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/xlsx-js-style/node_modules/adler-32": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/adler-32/-/adler-32-1.2.0.tgz", + "integrity": "sha512-/vUqU/UY4MVeFsg+SsK6c+/05RZXIHZMGJA+PX5JyWI0ZRcBpupnRuPLU/NXXoFwMYCPCoxIfElM2eS+DUXCqQ==", + "dependencies": { + "exit-on-epipe": "~1.0.1", + "printj": "~1.1.0" + }, + "bin": { + "adler32": "bin/adler32.njs" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/xlsx-js-style/node_modules/codepage": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/codepage/-/codepage-1.14.0.tgz", + "integrity": "sha512-iz3zJLhlrg37/gYRWgEPkaFTtzmnEv1h+r7NgZum2lFElYQPi0/5bnmuDfODHxfp0INEfnRqyfyeIJDbb7ahRw==", + "dependencies": { + "commander": "~2.14.1", + "exit-on-epipe": "~1.0.1" + }, + "bin": { + "codepage": "bin/codepage.njs" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/xlsx-js-style/node_modules/codepage/node_modules/commander": { + "version": "2.14.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.14.1.tgz", + "integrity": "sha512-+YR16o3rK53SmWHU3rEM3tPAh2rwb1yPcQX5irVn7mb0gXbwuCCrnkbV5+PBfETdfg1vui07nM6PCG1zndcjQw==" + }, "node_modules/yallist": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", diff --git a/src/main/webapp/package.json b/src/main/webapp/package.json index 3384fa4..0cc49fc 100644 --- a/src/main/webapp/package.json +++ b/src/main/webapp/package.json @@ -26,7 +26,9 @@ "react-loader-spinner": "^6.1.6", "react-router-dom": "^6.21.2", "react-toastify": "^10.0.4", - "recharts": "^2.15.1" + "recharts": "^2.15.1", + "xlsx": "^0.18.5", + "xlsx-js-style": "^1.2.0" }, "devDependencies": { "@types/react": "^18.2.43", diff --git a/src/main/webapp/src/pages/MemberList.jsx b/src/main/webapp/src/pages/MemberList.jsx index bb91ef0..79bd089 100644 --- a/src/main/webapp/src/pages/MemberList.jsx +++ b/src/main/webapp/src/pages/MemberList.jsx @@ -3,13 +3,13 @@ import {useFetch} from "../hooks/useFetch.js"; import {AxiosError} from "../components/AxiosError.jsx"; import {ThreeDots} from "react-loader-spinner"; import {useEffect, useState} from "react"; -import {Input} from "../components/Input.jsx"; import {useLocation, useNavigate} from "react-router-dom"; import {Checkbox} from "../components/MemberCustomFiels.jsx"; -import axios from "axios"; +import * as Tools from "../utils/Tools.js"; import {apiAxios, errFormater} from "../utils/Tools.js"; import {toast} from "react-toastify"; import {SearchBar} from "../components/SearchBar.jsx"; +import * as XLSX from "xlsx-js-style"; export function MemberList({source}) { const {hash} = useLocation(); @@ -100,12 +100,202 @@ export function MemberList({source}) { clubFilter={clubFilter} setClubFilter={setClubFilter} source={source}/> + +
+
Gestion groupée
+
+ +
+ +
+
} +function FileOutput() { + function formatColumnDate(worksheet, col) { + const range = XLSX.utils.decode_range(worksheet['!ref']) + // note: range.s.r + 1 skips the header row + for (let row = range.s.r + 1; row <= range.e.r; ++row) { + const ref = XLSX.utils.encode_cell({r: row, c: col}) + if (worksheet[ref] && worksheet[ref].t === "n") { + worksheet[ref].v = Math.trunc(worksheet[ref].v) + } else { + worksheet[ref].t = "n" + } + worksheet[ref].z = "dd/mm/yyyy" + } + } + + const handleFileDownload = () => { + toast.promise( + apiAxios.get(`/member/club/export`), + { + pending: "Exportation des licences...", + success: "Licences exportées", + error: { + render({data}) { + return errFormater(data, "Impossible d'exporté les licences") + } + } + }) + .then(data => { + const dataOut = [] + for (const e of data.data) { + const tmp = { + licence: e.licence, + nom: e.nom, + prenom: e.prenom, + email: e.email, + genre: e.genre, + birthdate: new Date(e.birthdate), + licenceCurrent: e.licenceCurrent ? 'X' : '', + certif: e.certif ? e.certif.split("¤")[0] : '', + certifDate: e.certif ? new Date(e.certif.split("¤")[1]) : '', + } + + //tmp.birthdate.setMilliseconds(0); + //tmp.birthdate.setSeconds(0); + //tmp.birthdate.setMinutes(0); + //tmp.birthdate.setHours(0); +// + //console.log(tmp.birthdate); + dataOut.push(tmp) + } + + const wb = XLSX.utils.book_new(); + const ws = XLSX.utils.json_to_sheet(dataOut); + XLSX.utils.sheet_add_aoa(ws, [["Licence", "Nom", "Prénom", "Email", "Genre", "Date de naissance", "Licence en cours", "Nom médecin certificat", "Date certificat"]], {origin: 'A1'}); + // XLSX.utils.sheet_add_json(ws, dataOut, {skipHeader: true, origin: 'A2'}); + + formatColumnDate(ws, 5) + formatColumnDate(ws, 8) + console.log(ws) + //ws["!data"][0][0].z = "yyyy-mm-dd hh:mm:ss" + ws["!cols"] = [{wch: 7}, {wch: 16}, {wch: 16}, {wch: 30}, {wch: 9}, {wch: 12}, {wch: 6}, {wch: 13}, {wch: 12}] + + XLSX.utils.book_append_sheet(wb, ws, `Saison ${Tools.getSaison()}-${Tools.getSaison() + 1}`); + XLSX.writeFile(wb, "output.xlsx"); + }); + }; + + return ( +
+ + À utiliser comme template pour mettre à jour les informations +
+ ); + +} + + +function FileInput() { + const [data, setData] = useState(null); + + const re = /^(([^<>()[\].,;:\s@"]+(\.[^<>()[\].,;:\s@"]+)*)|(".+"))@(([^<>()[\].,;:\s@"]+\.)+[^<>()[\].,;:\s@"]{2,})$/i; + + function excelDateToJSDate(serial) { + const utcDays = Math.floor(serial - 25569); + const utcValue = utcDays * 86400; + return new Date(utcValue * 1000); + } + + const handleFileUpload = (e) => { + const file = e.target.files[0]; + const reader = new FileReader(); + + reader.onload = (event) => { + const workbook = XLSX.read(event.target.result, {type: 'binary'}); + const sheet = workbook.Sheets[`Saison ${Tools.getSaison()}-${Tools.getSaison() + 1}`]; + const sheetData = XLSX.utils.sheet_to_json(sheet); + + const dataOut = [] + let error = 0; + let cetifNotFill = 0; + for (let i = 0; i < sheetData.length; i++) { + const line = sheetData[i]; + // noinspection NonAsciiCharacters,JSNonASCIINames + const tmp = { + licence: line["Licence"], + nom: line["Nom"], + prenom: line["Prénom"], + email: line["Email"], + genre: line["Genre"], + birthdate: line["Date de naissance"], + licenceCurrent: line["Licence en cours"] === undefined ? false : line["Licence en cours"].toLowerCase() === "x", + certif: "", + } + + if (tmp.nom === undefined || tmp.nom === "") { + toast.error("Nom vide à la ligne " + (i + 2)) + error++; + } + if (tmp.prenom === undefined || tmp.prenom === "") { + toast.error("Prénom vide à la ligne " + (i + 2)) + error++; + } + + if (tmp.licenceCurrent) { // need check full data + if (tmp.email === undefined || tmp.email === "") { + toast.error("Email vide à la ligne " + (i + 2)) + error++; + } + if (!re.test(tmp.email)) { + toast.error("Email invalide à la ligne " + (i + 2)) + error++; + } + // noinspection NonAsciiCharacters,JSNonASCIINames + if (line["Nom médecin certificat"] === undefined || line["Nom médecin certificat"] === "" || + line["Date certificat"] === undefined || line["Date certificat"] === "") { + cetifNotFill++; + } + try { + // noinspection JSNonASCIINames + tmp.certif = line["Nom médecin certificat"] + "¤" + excelDateToJSDate(line["Date certificat"]).toISOString() + } catch (e) { + toast.error("Format de la date de certificat invalide à la ligne " + (i + 2)) + error++; + } + + if (tmp.birthdate === undefined || tmp.birthdate === "") { + toast.error("Date de naissance vide à la ligne " + (i + 2)) + error++; + } + try { + tmp.birthdate = excelDateToJSDate(tmp.birthdate).toISOString(); + } catch (e) { + toast.error("Format de la date de naissance invalide à la ligne " + (i + 2)) + error++; + } + } + dataOut.push(tmp) + } + + if (error > 0) { + toast.error(`${error} erreur(s) dans le fichier, opération annulée`) + } else { + console.log(dataOut); + setData(sheetData); + } + }; + + reader.readAsBinaryString(file); + }; + + return ( +
+ Charger l'Excel +
+ +
+ Merci d'utiliser le fichier ci-dessus comme base, ne pas renommer les colonnes ni modifier les n° de licences. +
+ ); +} + function MakeCentralPanel({data, visibleMember, navigate, showLicenceState, page, source}) { const pages = [] for (let i = 1; i <= data.page_count; i++) { @@ -119,7 +309,8 @@ function MakeCentralPanel({data, visibleMember, navigate, showLicenceState, page Ligne {((page - 1) * data.page_size) + 1} à { (page * data.page_size > data.result_count) ? data.result_count : (page * data.page_size)} (page {page} sur {data.page_count})
- {visibleMember.map(member => ())} + {visibleMember.map(member => ( + ))}
@@ -198,4 +389,4 @@ function Def() {
  • -} \ No newline at end of file +}