FreeDatas2HTML/src/freeDatas2HTML.ts

208 lines
8.8 KiB
TypeScript
Raw Normal View History

const Papa = require("papaparse");
const errors = require("./errors.js");
import { papaParseDatas, papaParseErrors, papaParseMeta } from "./papaParseInterfaces";
import { domElement, selectors } from "./freeDatas2HTMLInterfaces";
2021-08-05 18:24:37 +02:00
export class freeDatas2HTML
2021-08-05 18:24:37 +02:00
{
private _datasViewElt: domElement = { id:"", eltDOM:undefined };
private _datasSourceUrl: string = "";
2021-08-05 18:24:37 +02:00
private _datasSelectors: selectors[] = [];
public parseMeta: papaParseMeta|undefined = undefined;
public parseDatas: papaParseDatas[] = [];
public parseErrors: papaParseErrors[] = [];
2021-08-12 16:05:12 +02:00
public datasHTML: string = "";
public stopIfParseErrors: boolean = false;
2021-08-05 18:24:37 +02:00
set datasViewElt(elt: domElement)
2021-08-05 18:24:37 +02:00
{
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;
}
2021-08-05 18:24:37 +02:00
}
set datasSourceUrl(url: string)
2021-08-05 18:24:37 +02:00
{
if(url.trim().length === 0)
throw new Error(errors.needUrl);
else
this._datasSourceUrl=url.trim();
2021-08-05 18:24:37 +02:00
}
2021-08-12 16:05:12 +02:00
set datasSelectors(selectionElts: selectors[])
2021-08-05 18:24:37 +02:00
{
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]);
}
}
2021-08-05 18:24:37 +02:00
}
get datasSelectors() : selectors[]
{
return this._datasSelectors;
}
public async parse(): Promise<any>
{
const converter=this;
return new Promise((resolve,reject) =>
{
if(converter._datasSourceUrl !== "" )
{
Papa.parse(converter._datasSourceUrl,
{
quoteChar: '"',
header: true,
complete: function(results :any)
{
2021-08-12 16:05:12 +02:00
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<any>
{
if (this._datasViewElt.eltDOM === undefined)
throw new Error(errors.needDatasElt);
2021-08-12 16:05:12 +02:00
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
{
2021-08-12 16:05:12 +02:00
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)
{
2021-08-12 16:05:12 +02:00
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]="<label for='freeDatas2HTMLSelector"+i+"'>"+colName+" : </label><select name='freeDatas2HTMLSelector"+i+"' id='freeDatas2HTMLSelector"+i+"'><option value='0'>----</option>";
for(let j in values)
selectorsHTML[i]+="<option value='"+(Number(j)+1)+"'>"+values[j]+"</option>";
selectorsHTML[i]+="</select>";
2021-08-12 16:05:12 +02:00
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
{
2021-08-12 16:05:12 +02:00
// 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)
{
2021-08-12 16:05:12 +02:00
// 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 !
2021-08-12 16:05:12 +02:00
}
// Création du tableau de données :
let datasHTML="<table><thead>";
for (let i in fields)
datasHTML+="<th>"+fields[i]+"</th>";
datasHTML+="</thead><tbody>";
for (let row in datas)
{
2021-08-12 16:05:12 +02:00
let visible=true;
if(filters.length !== 0)
{
for(let i in filters)
{
2021-08-12 16:05:12 +02:00
if(datas[row][filters[i].field].trim() != filters[i].value)
visible=false;
}
}
2021-08-12 16:05:12 +02:00
if(visible)
{
datasHTML+="<tr>";
for(let field in datas[row])
{
2021-08-12 16:05:12 +02:00
// 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+="<td>"+datas[row][field]+"</td>";
}
datasHTML+="</tr>";
}
}
datasHTML+="</tbody></table>";
return datasHTML;
}
2021-08-12 16:05:12 +02:00
}