Newer
Older
francois.grand
committed
import { Injectable } from "@angular/core";
francois.grand
committed
import { decode } from "he";
import { saveAs } from "file-saver";
import { CalculatorType, EnumEx, Observable, ParamDefinition, Session, Nub, ParallelStructure } from "jalhyd";
francois.grand
committed
import { HttpService } from "../../services/http/http.service";
import { I18nService } from "../../services/internationalisation/internationalisation.service";
francois.grand
committed
import { FormulaireDefinition } from "../../formulaire/definition/form-definition";
francois.grand
committed
import { FormulaireElement } from "../../formulaire/formulaire-element";
import { InputField } from "../../formulaire/input-field";
francois.grand
committed
import { SelectField } from "../../formulaire/select-field";
import { CheckField } from "../../formulaire/check-field";
import { StringMap } from "../../stringmap";
import { FormulaireBase } from "../../formulaire/definition/concrete/form-base";
francois.grand
committed
import { FormulaireLechaptCalmon } from "../../formulaire/definition/concrete/form-lechapt-calmon";
import { FormulaireSectionParametree } from "../../formulaire/definition/concrete/form-section-parametree";
import { FormulaireCourbeRemous } from "../../formulaire/definition/concrete/form-courbe-remous";
import { FormulaireRegimeUniforme } from "../../formulaire/definition/concrete/form-regime-uniforme";
import { FormulaireParallelStructure } from "../../formulaire/definition/concrete/form-parallel-structures";
francois.grand
committed
import { NgParameter } from "../../formulaire/ngparam";
import { FieldsetContainer } from "../..//formulaire/fieldset-container";
export class FormulaireService extends Observable {
private _formulaires: FormulaireDefinition[];
private _currentFormId: string = null;
private i18nService: I18nService,
private get _intlService(): I18nService {
return this.i18nService;
francois.grand
committed
}
private get _httpService(): HttpService {
francois.grand
committed
}
public get formulaires(): FormulaireDefinition[] {
return this._formulaires;
}
private loadLocalisation(calc: CalculatorType): Promise<any> {
const f: string = this.getConfigPathPrefix(calc) + this._intlService.currentLanguage.tag + ".json";
const prom = this._httpService.httpGetRequestPromise(f);
}
/**
* met à jour la langue du formulaire
* @param formId id unique du formulaire
* @param localisation ensemble id-message traduit
*/
private updateFormulaireLocalisation(formId: string, localisation: StringMap) {
f.updateLocalisation(localisation);
break;
}
}
/**
* charge la localisation et met à jour la langue du formulaire
*/
private loadUpdateFormulaireLocalisation(f: FormulaireDefinition): Promise<FormulaireDefinition> {
return this.loadLocalisation(f.calculatorType)
.then(localisation => {
this.updateFormulaireLocalisation(f.uid, localisation);
return f;
});
}
public updateLocalisation() {
for (const c of EnumEx.getValues(CalculatorType)) {
const prom: Promise<StringMap> = this.loadLocalisation(c);
prom.then(loc => {
for (const f of this._formulaires) {
if (f.calculatorType === c) {
this.updateFormulaireLocalisation(f.uid, loc);
/**
* Retourne le titre complet du type de module de calcul, dans la langue en cours
*/
public getLocalisedTitleFromCalculatorType(type: CalculatorType) {
const sCalculator: string = CalculatorType[type].toUpperCase();
return this._intlService.localizeText(`INFO_${sCalculator}_TITRE`);
}
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
/**
* Retourne le titre cour du type de module de calcul, dans la langue en cours
* (pour les titres d'onglets par défaut)
*/
public getLocalisedShortTitleFromCalculatorType(type: CalculatorType) {
const sCalculator: string = CalculatorType[type].toUpperCase();
return this._intlService.localizeText(`INFO_${sCalculator}_TITRE_COURT`);
}
/**
* Checks if the given calculator name (tab title) is already used by any existing
* form; if so, adds a number after it
*/
private suffixNameIfNeeded(name: string) {
let found = false;
let maxNumber = 0;
// extract base name
let baseName = name;
const re1 = new RegExp("^.+( \\d+)$");
const matches1 = re1.exec(name);
if (matches1) {
baseName = baseName.replace(matches1[1], "");
}
// browse session calculators
const re2 = new RegExp("^" + baseName + "( (\\d+))?$");
for (const f of this.formulaires) {
const matches2 = re2.exec(f.calculatorName);
if (matches2) {
found = true;
if (matches2[2] !== undefined) {
const nb = Number(matches2[2]);
maxNumber = Math.max(maxNumber, nb);
}
}
}
// suffix if needed
if (found) {
name = baseName + " " + (maxNumber + 1);
}
return name;
}
public loadConfig(ct: CalculatorType): Promise<any> {
const f: string = this.getConfigPathPrefix(ct) + "config.json";
return this._httpService.httpGetRequestPromise(f);
private newFormulaire(ct: CalculatorType, jsonState?: {}): FormulaireDefinition {
francois.grand
committed
let f: FormulaireDefinition;
switch (ct) {
case CalculatorType.ConduiteDistributrice:
case CalculatorType.PabDimensions:
case CalculatorType.PabPuissance:
case CalculatorType.MacroRugo:
f = new FormulaireBase();
francois.grand
committed
break;
case CalculatorType.LechaptCalmon:
francois.grand
committed
f = new FormulaireLechaptCalmon();
francois.grand
committed
break;
case CalculatorType.SectionParametree:
francois.grand
committed
f = new FormulaireSectionParametree();
francois.grand
committed
break;
case CalculatorType.RegimeUniforme:
francois.grand
committed
f = new FormulaireRegimeUniforme();
francois.grand
committed
break;
case CalculatorType.CourbeRemous:
francois.grand
committed
f = new FormulaireCourbeRemous();
francois.grand
committed
break;
case CalculatorType.ParallelStructure:
case CalculatorType.Cloisons:
f = new FormulaireParallelStructure();
francois.grand
committed
default:
throw new Error(`FormulaireService.newFormulaire() : type de module de calcul ${CalculatorType[ct]} non pris en charge`);
francois.grand
committed
}
f.defaultProperties["calcType"] = ct;
return f;
}
/**
* crée un formulaire d'un type donné
* @param ct type de formulaire
* @param nub nub existant à associer au formulaire (chargement de session / duplication de module)
* @param calculatorName nom du module, à afficher dans l'interface
public createFormulaire(ct: CalculatorType, nub?: Nub, calculatorName?: string): Promise<FormulaireDefinition> {
// Crée un formulaire du bon type
const f: FormulaireDefinition = this.newFormulaire(ct);
// Charge la configuration dépendamment du type
return prom.then(s => {
// Associe le Nub fourni (chargement de session / duplication de module), sinon en crée un nouveau
if (nub) {
f.currentNub = nub;
} else {
f.initNub();
}
// Restaure le nom du module, sinon affecte le nom par défaut
let tempName: string;
if (calculatorName) {
tempName = decode(this.getLocalisedShortTitleFromCalculatorType(ct));
// Suffixe le nom du module si nécessaire
f.calculatorName = this.suffixNameIfNeeded(tempName);
// add fieldsets for existing structures if needed
if (f.currentNub instanceof ParallelStructure) {
for (s of f.currentNub.structures) {
for (const e of f.allFormElements) {
if (e instanceof FieldsetContainer) { // @TODO manage many containers one day ?
e.addFromTemplate(0, undefined, s);
}
}
}
}
return this.loadUpdateFormulaireLocalisation(f);
mathias.chouet
committed
this.notifyObservers({
"action": "createForm",
"form": fi
});
mathias.chouet
committed
/**
* Trick to notify param-link components that parent form name changed
* @TODO find a way to make param-link components directly observe FormDefinition
*/
public propagateFormNameChange(f: FormulaireDefinition, name: string) {
this.notifyObservers({
"action": "formNameChanged",
"form": f,
"value": name
});
}
public getFormulaireFromId(uid: string): FormulaireDefinition {
for (const f of this._formulaires) {
if (f.uid === uid) {
public getInputField(formId: string, elemId: string): InputField {
const s = this.getFormulaireElementById(formId, elemId);
if (!(s instanceof InputField)) {
throw new Error("Form element with id '" + elemId + "' is not an input");
}
public getCheckField(formId: string, elemId: string): CheckField {
const s = this.getFormulaireElementById(formId, elemId);
throw new Error("Form element with id '" + elemId + "' is not a checkbox");
public getSelectField(formId: string, elemId: string): SelectField {
const s = this.getFormulaireElementById(formId, elemId);
throw new Error("Form element with id '" + elemId + "' is not a select");
private getFormulaireElementById(formId: string, elemId: string): FormulaireElement {
const s = f.getFormulaireNodeById(elemId);
francois.grand
committed
return s as FormulaireElement;
francois.grand
committed
public getParamdefParentForm(prm: ParamDefinition): FormulaireDefinition {
for (const f of this._formulaires) {
for (const p of f.allFormElements) {
if (p instanceof NgParameter) {
if (p.paramDefinition.uid === prm.uid) {
francois.grand
committed
return f;
francois.grand
committed
}
/**
* retrouve un formulaire à partir d'un uid de Nub
*/
public getFormulaireFromNubId(uid: string) {
for (const f of this._formulaires) {
if (f.hasNubId(uid)) {
return f;
}
}
}
public getConfigPathPrefix(ct: CalculatorType): string {
if (ct === undefined) {
throw new Error("FormulaireService.getConfigPathPrefix() : invalid undefined CalculatorType");
}
switch (ct) {
case CalculatorType.ConduiteDistributrice:
return "app/calculators/cond_distri/cond_distri.";
case CalculatorType.LechaptCalmon:
return "app/calculators/lechapt-calmon/lechapt-calmon.";
case CalculatorType.SectionParametree:
return "app/calculators/section-param/section-param.";
case CalculatorType.RegimeUniforme:
return "app/calculators/regime-uniforme/regime-uniforme.";
francois.grand
committed
case CalculatorType.CourbeRemous:
return "app/calculators/remous/remous.";
case CalculatorType.PabDimensions:
return "app/calculators/pab-dimensions/pab-dimensions.";
case CalculatorType.PabPuissance:
return "app/calculators/pab-puissance/pab-puissance.";
case CalculatorType.Structure:
return "app/calculators/ouvrages/ouvrages.";
case CalculatorType.ParallelStructure:
return "app/calculators/parallel-structures/parallel-structures.";
case CalculatorType.Dever:
return "app/calculators/dever/dever.";
case CalculatorType.Cloisons:
return "app/calculators/cloisons/cloisons.";
case CalculatorType.MacroRugo:
return "app/calculators/macrorugo/macrorugo.";
throw new Error("FormulaireService.getConfigPathPrefix() : valeur de CalculatorType " + ct + " non implémentée");
/**
* Supprime le formulaire ciblé, et demande à JaLHyd d'effacer son Nub de la Session
* @param uid formulaire à supprimer
*/
public requestCloseForm(uid: string) {
const form = this.getFormulaireFromId(uid);
const nub = form.currentNub;
if (form) {
this._formulaires = this._formulaires.filter(f => f.uid !== uid);
this.notifyObservers({
"form": form
if (nub) {
Session.getInstance().deleteNub(nub);
}
}
public get currentFormId() {
return this._currentFormId;
}
public setCurrentForm(formId: string) {
const form = this.getFormulaireFromId(formId);
this._currentFormId = null;
this.notifyObservers({
"action": "invalidFormId",
"formId": formId
});
this._currentFormId = formId;
this.notifyObservers({
"action": "currentFormChanged",
"formId": formId
francois.grand
committed
private get currentForm(): FormulaireDefinition {
return this.getFormulaireFromId(this._currentFormId);
}
public get currentFormHasResults(): boolean {
const form = this.currentForm;
francois.grand
committed
return form.hasResults;
francois.grand
committed
return false;
}
francois.grand
committed
private readSingleFile(file: File): Promise<any> {
return new Promise<any>((resolve, reject) => {
fr.onload = () => {
resolve(fr.result);
};
fr.onerror = () => {
fr.abort();
reject(new Error(`Erreur de lecture du fichier ${file.name}`));
};
fr.readAsText(file);
});
}
francois.grand
committed
/**
* charge une session en tenant compte des modules de calcul sélectionnées
francois.grand
committed
* @param f fichier session
* @param formInfos infos sur les modules de calcul @see DialogLoadSessionComponent.calculators
francois.grand
committed
*/
public loadSession(f: File, formInfos: any[]) {
this.readSingleFile(f).then(s => {
const uids: string[] = [];
formInfos.forEach((fi) => {
if (fi.selected) {
uids.push(fi.uid);
});
const newNubs = Session.getInstance().unserialise(s, uids);
// for each new Nub, create the related form, set its title
newNubs.forEach((nn) => {
this.createFormulaire(nn.nub.calcType, nn.nub, nn.meta.title);
});
}).catch(err => {
throw err;
});
/**
* Sends an UTF-8 text file for download
*/
public downloadTextFile(session: string, filename: string = "file_1") {
const blob = new Blob([session], { type: "text/plain;charset=utf-8" });
saveAs(blob, filename);
francois.grand
committed
}
francois.grand
committed
/**
* obtient des infos (nom, uid des modules de calcul) d'un fichier session
francois.grand
committed
* @param f fichier session
*/
public calculatorInfosFromSessionFile(f: File): Promise<any[]> {
return this.readSingleFile(f).then(s => {
const res: any[] = [];
const data = JSON.parse(s);
// liste des noms de modules de calcul
if (data.session && Array.isArray(data.session)) {
data.session.forEach((e: any) => {
res.push({
uid: e.uid,
title: e.meta && e.meta.title ? e.meta.title : undefined
});
});
francois.grand
committed
return res;
});
}
francois.grand
committed
public saveForm(f: FormulaireDefinition) {
this.notifyObservers({
"action": "saveForm",
"form": f
});
}
/**
* met à jour les liens d'un formulaire
* @param json conf du formulaire
* @param formInfos métadonnées sur les formulaires chargés
* @param form formulaire dont on met à jour les liens
* @param uidMap table de correspondance uid dans le fichier de conf <-> uid en mémoire
*/
private updateFormLinks(json: {}, formInfos: any[], form: FormulaireDefinition, uidMap: {}[]) {
for (const i of formInfos) {
if (i["uid"] === json["uid"] && i["selected"]) {
form.updateParamsLinks(json, uidMap);
}
}
}
/**
* MAJ des liens entre paramètres lors de la désérialisation
/* private updateParamsLinks(json: {}, formInfos: any[], oldFormCount: number) {
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
// table de correspondance des uid fichier <-> objets mémoire
// forme : tableau d'objets de la forme :
// { "type" : <type de l'objet. "form" pour formulaire>,
// "old": <uid dans le fichier>,
// "new": <uid de l'objet mémoire>}
const uidMap = [];
for (const ks in json) {
switch (ks) {
case "elements":
let n = oldFormCount;
for (const e of json[ks]) {
if (Object.keys(e)[0] === "form") {
uidMap.push({
"type": "form",
"old": e["form"]["uid"],
"new": this._formulaires[n].uid
});
n++;
}
}
}
}
// MAJ liens
for (const ks in json) {
switch (ks) {
case "elements":
let n = 0;
for (const e of json[ks]) {
if (Object.keys(e)[0] === "form") {
this.updateFormLinks(e["form"], formInfos, this._formulaires[n + oldFormCount], uidMap);
n++;
}
}
break;
default:
throw new Error(`session file : invalid key '${ks}' in session object`);
}
}
francois.grand
committed
/**
francois.grand
committed
* @returns liste des valeurs liables à un paramètre sous la forme d'un tableau d'objets
* {"param":<paramètre lié>, "nub":<Nub d'origine du paramètre lié>, "formTitle":<nom du module de calcul liée au nub>}
francois.grand
committed
* @param p paramètre qui sert de clé de recherche des paramètres liables
francois.grand
committed
*/
francois.grand
committed
public getLinkableValues(p: NgParameter): any[] {
francois.grand
committed
for (const f of this._formulaires) {
francois.grand
committed
// nub associé au formulaire
francois.grand
committed
try {
// on vérifie que le paramètre en entrée appartient au nub
const np = sn.getParameter(p.symbol);
// si oui, on demande à exclure des valeurs retournées le résultat du même nom que le paramètre
François
committed
const exclude = np !== undefined ? p.paramDefinition.uid === np.uid : false;
// valeurs liables
const ps = sn.getLinkableValues(p.paramDefinition, undefined, exclude);
for (const npp of ps) {
npp["formTitle"] = f.calculatorName;
res.push(npp);
francois.grand
committed
// p.symbol n'existe pas dans le nub testé
}
francois.grand
committed
}
francois.grand
committed
return res;
}
francois.grand
committed
/**
* filtre les valeurs liables à un paramètre :
* - supprime les valeurs non affichées dans leur parent (ce n'est pas le cas par ex
* pour Q, Z1, Z2 dans les ouvrages enfants des ouvrages //)
francois.grand
committed
* @param values valeurs liables (modifié par la méthode)
*/
public filterLinkableValues(values: any[]): any[] {
// suppression des paramètres non affichés
for (let i = values.length - 1; i >= 0; i--) {
const v = values[i].value;
if (v instanceof ParamDefinition) {
// pour chaque paramètre...
const prm: ParamDefinition = v;
const parentForm: FormulaireDefinition = this.getParamdefParentForm(prm);
// ... on cherche s'il est affiché dans son parent
if (parentForm !== undefined) {
for (const fe of parentForm.allFormElements) {
if (fe instanceof NgParameter) {
francois.grand
committed
if (fe.paramDefinition.uid === prm.uid) {
found = true;
break;
}
francois.grand
committed
francois.grand
committed
values.splice(i, 1);
francois.grand
committed
}
}
return values;
}