const Papa = require("papaparse"); const errors = require("./errors.js"); import { papaParseDatas, papaParseErrors, papaParseMeta } from "./papaParseInterfaces"; import { domElement, selectors } from "./freeDatas2HTMLInterfaces"; export class freeDatas2HTML { private _datasViewElt: domElement = { id:"", eltDOM:undefined }; private _datasSourceUrl: string = ""; private _datasSelectors: selectors[] = []; public parseMeta: papaParseMeta|undefined = undefined; public parseDatas: papaParseDatas[] = []; public parseErrors: papaParseErrors[] = []; public datasHTML: string = ""; public stopIfParseErrors: boolean = false; set datasViewElt(elt: domElement) { let checkContainerExist=document.getElementById(elt.id); if(checkContainerExist === null) throw new Error(errors.elementNotFound+elt.id); else { this._datasViewElt.id=elt.id; this._datasViewElt.eltDOM=checkContainerExist; } } set datasSourceUrl(url: string) { if(url.trim().length === 0) throw new Error(errors.needUrl); else this._datasSourceUrl=url.trim(); } set datasSelectors(selectionElts: selectors[]) { this._datasSelectors=[]; let checkContainerExist: HTMLElement|null; for(let i = 0; i < selectionElts.length; i++) { checkContainerExist=document.getElementById(selectionElts[i].id); if(checkContainerExist === null) console.error(errors.elementNotFound+selectionElts[i].id); else if(Number.isInteger( selectionElts[i].datasFielNb) === false || selectionElts[i].datasFielNb < 0) console.error(errors.needNaturalNumber); else { selectionElts[i].eltDOM=checkContainerExist; this._datasSelectors.push(selectionElts[i]); } } } get datasSelectors() : selectors[] { return this._datasSelectors; } public async parse(): Promise { const converter=this; return new Promise((resolve,reject) => { if(converter._datasSourceUrl !== "" ) { Papa.parse(converter._datasSourceUrl, { quoteChar: '"', header: true, complete: function(results :any) { converter.parseErrors=results.errors; converter.parseDatas=results.data; // Attention, papaParse peut accepter un nom de colonne vide let realFields: string[]=[]; for(let i in results.meta.fields) { if(results.meta.fields[i].trim() !== "") realFields.push(results.meta.fields[i]); } results.meta.fields=realFields; converter.parseMeta=results.meta; resolve(true); }, error:function(error :any) { reject(new Error(errors.parserFail)); }, download: true, skipEmptyLines: true, }); } else reject(new Error(errors.needUrl)); }); } public async run(): Promise { if (this._datasViewElt.eltDOM === undefined) throw new Error(errors.needDatasElt); if(this._datasSourceUrl === "" ) throw new Error(errors.needUrl); await this.parse(); if(this.parseDatas.length === 0 || this.parseMeta!.fields === undefined) // je force avec "!", car l'existence de parseMeta est certaine après parse(). throw new Error(errors.datasNotFound); else if(this.stopIfParseErrors && this.parseErrors.length!==0) console.error(this.parseErrors); else { let converter=this; // Affichage initial des données du fichier this.datasHTML=this.createDatasHTML(this.parseMeta!.fields, this.parseDatas); this._datasViewElt.eltDOM.innerHTML=this.datasHTML; // Si demandé, création des listes permettant de filter les données if(this._datasSelectors.length > 0) { // Les colonnes devant servir de filtre existent-elles dans le fichier ? let selectorsHTML : string [] = []; for(let i in this._datasSelectors) { if(this._datasSelectors[i].datasFielNb > (this.parseMeta!.fields.length-1)) throw new Error(errors.selectorFieldNotFound); else { let values=[], colName=this.parseMeta!.fields[this._datasSelectors[i].datasFielNb]; for (let row in this.parseDatas) { if(values.indexOf(this.parseDatas[row][colName].trim()) === -1) values.push(this.parseDatas[row][colName].trim()); // Des espaces gauche pourraient fausser le classement alphabétique. // Donc réutiliser le trim() lorsque l'on filtre l'affichage des données. } if(values.length > 0) { values.sort(); // à revoir, car gère mal la casse, les nombres, etc. this._datasSelectors[i].name=colName; this._datasSelectors[i].values=values; selectorsHTML[i]=""; this._datasSelectors[i].eltDOM!.innerHTML=selectorsHTML[i]; let selectElement = document.getElementById("freeDatas2HTMLSelector"+i) as HTMLInputElement; selectElement.addEventListener('change', function(e) { converter.datasHTML=converter.createDatasHTML(converter.parseMeta!.fields as string[], converter.parseDatas); converter._datasViewElt.eltDOM!.innerHTML=converter.datasHTML; }); } } } } return true; } } private createDatasHTML(fields: string[], datas: any[]) : string { // Je vérifie si des valeurs ont été sélectionnées pour filter les données. let checkSelectorExist: HTMLSelectElement|null, filters: any[] = []; for(let i in this._datasSelectors) { // Attention : je peux avoir des _datasSelectors fournis, mais pas de liste dans le DOM si aucune donnée ou autre problème. checkSelectorExist=document.querySelector("#"+ this._datasSelectors[i].id+" select"); if(checkSelectorExist != null && checkSelectorExist.value != "0") filters.push({ field: this._datasSelectors[i].name, value: this._datasSelectors[i].values![checkSelectorExist.selectedIndex-1] }); // Attention : si on récupère innerHTML du select, certains caractères peuvent être modifiés ! } // Création du tableau de données : let datasHTML=""; for (let i in fields) datasHTML+=""; datasHTML+=""; for (let row in datas) { let visible=true; if(filters.length !== 0) { for(let i in filters) { if(datas[row][filters[i].field].trim() != filters[i].value) visible=false; } } if(visible) { datasHTML+=""; for(let field in datas[row]) { // Attention : si les erreurs papaParse ne sont pas bloquantes, il peut y avoir des données en trop, avec comme nom de colonne : "__parsed_extra" if(fields.indexOf(field) !== -1) datasHTML+=""; } datasHTML+=""; } } datasHTML+="
"+fields[i]+"
"+datas[row][field]+"
"; return datasHTML; } }