Newer
Older
francois.grand
committed
import { Injectable } from "@angular/core";
francois.grand
committed
import { decode } from "he";
import { saveAs } from "file-saver";
import { CalculatorType, LinkedValue, 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 { 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";
import { ApplicationSetupService } from "../app-setup/app-setup.service";
mathias.chouet
committed
import { NotificationsService } from "../notifications/notifications.service";
export class FormulaireService extends Observable {
private calculatorPaths = {};
private _formulaires: FormulaireDefinition[];
private _currentFormId: string = null;
/** to avoid loading language files multiple times */
private languageCache = {};
private i18nService: I18nService,
private appSetupService: ApplicationSetupService,
mathias.chouet
committed
private httpService: HttpService,
private intlService: I18nService,
private notificationsService: NotificationsService
// register calculator config paths here
this.calculatorPaths[CalculatorType.ConduiteDistributrice] = "cond_distri";
this.calculatorPaths[CalculatorType.LechaptCalmon] = "lechapt-calmon";
this.calculatorPaths[CalculatorType.SectionParametree] = "section-param";
this.calculatorPaths[CalculatorType.RegimeUniforme] = "regime-uniforme";
this.calculatorPaths[CalculatorType.CourbeRemous] = "remous";
this.calculatorPaths[CalculatorType.PabChute] = "pab-chute";
this.calculatorPaths[CalculatorType.PabDimensions] = "pab-dimensions";
this.calculatorPaths[CalculatorType.PabNombre] = "pab-nombre";
this.calculatorPaths[CalculatorType.PabPuissance] = "pab-puissance";
this.calculatorPaths[CalculatorType.Structure] = "ouvrages";
this.calculatorPaths[CalculatorType.ParallelStructure] = "parallel-structures";
this.calculatorPaths[CalculatorType.Dever] = "dever";
this.calculatorPaths[CalculatorType.Cloisons] = "cloisons";
this.calculatorPaths[CalculatorType.MacroRugo] = "macrorugo";
private get _intlService(): I18nService {
return this.i18nService;
francois.grand
committed
}
private get _httpService(): HttpService {
francois.grand
committed
}
public get formulaires(): FormulaireDefinition[] {
return this._formulaires;
}
/**
* Loads the localisation file dedicated to calculator type ct; tries the current
* language then the fallback language
*/
private loadLocalisation(calc: CalculatorType): Promise<any> {
const lang = this._intlService.currentLanguage;
return this.loadLocalisationForLang(calc, lang).then((localisation) => {
return localisation as StringMap;
}).catch((e) => {
console.error(e);
// try default lang (the one in the config file) ?
const fallbackLang = this.appSetupService.fallbackLanguage;
if (lang !== fallbackLang) {
console.error(`trying fallback language: ${fallbackLang}`);
return this.loadLocalisationForLang(calc, fallbackLang);
}
* Loads the localisation file dedicated to calculator type ct for language lang;
* keeps it in cache for subsequent calls ()
*/
private loadLocalisationForLang(calc: CalculatorType, lang: string): Promise<any> {
const ct = String(calc);
// already in cache ?
if (Object.keys(this.languageCache).includes(ct) && Object.keys(this.languageCache[calc]).includes(lang)) {
return new Promise((resolve) => {
resolve(this.languageCache[ct][lang]);
});
} else {
const f: string = this.getConfigPathPrefix(calc) + lang + ".json";
return this._httpService.httpGetRequestPromise(f).then((localisation) => {
this.languageCache[ct] = this.languageCache[ct] || {};
this.languageCache[ct][lang] = localisation;
return localisation as StringMap;
}).catch((e) => {
throw new Error(`LOCALISATION_FILE_NOT_FOUND "${f}"`);
});
}
}
/**
* Loads localisation file corresponding to current language then updates all form strings,
* only if form language was not already set to current language
public loadUpdateFormulaireLocalisation(f: FormulaireDefinition): Promise<FormulaireDefinition> {
const requiredLang = this._intlService.currentLanguage;
if (requiredLang !== f.currentLanguage) {
return this.loadLocalisation(f.calculatorType).then(localisation => {
f.updateLocalisation(localisation, requiredLang);
return f;
});
}
/**
* 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`);
}
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
/**
* 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.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:
f = new FormulaireBase();
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);
}
}
}
}
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 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 (! this.calculatorPaths.hasOwnProperty(ct)) {
throw new Error("FormulaireService.getConfigPathPrefix() : valeur de CalculatorType " + ct + " non implémentée");
return "app/calculators/" + this.calculatorPaths[ct] + "/" + this.calculatorPaths[ct] + ".";
/**
* 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
// reset the result panels of all forms depending on this one
// @TODO UI should detect model change instead of doing this manually
this.resetAllDependingFormsResults(form, [], false);
if (nub) {
// reset model results (important, also resets dependent Nubs results in chain)
form.currentNub.resetResult();
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) => {
const nubInfo = {
uid: e.uid,
title: e.meta && e.meta.title ? e.meta.title : undefined,
requires: [],
children: []
};
// list linked params dependencies for each Nub
if (e.parameters) {
e.parameters.forEach((p) => {
if (p.targetNub && ! nubInfo.requires.includes(p.targetNub)) {
nubInfo.requires.push(p.targetNub);
}
});
}
// list children nubs for each Nub
if (e.children) {
e.children.forEach((p) => {
nubInfo.children.push(p.uid);
});
}
res.push(nubInfo);
francois.grand
committed
return res;
});
}
francois.grand
committed
public saveForm(f: FormulaireDefinition) {
this.notifyObservers({
"action": "saveForm",
"form": f
});
}
* Demande à la Session JalHYd la liste des paramètres/résultats pouvant être liés au
* paramètre fourni
public getLinkableValues(p: NgParameter): LinkedValue[] {
let linkableValues: LinkedValue[] = [];
if (p) {
linkableValues = Session.getInstance().getLinkableValues(p.paramDefinition);
// join form names to ease usage
for (let i = 0; i < linkableValues.length; i++) {
const lv = linkableValues[i];
for (const f of this._formulaires) {
if (f.currentNub) {
if (f.currentNub.uid === lv.nub.uid) {
lv.meta["formTitle"] = f.calculatorName;
} else {
// child structures ?
for (const s of f.currentNub.getChildren()) {
if (s.uid === lv.nub.uid) {
lv.meta["formTitle"] = f.calculatorName;
francois.grand
committed
}
francois.grand
committed
}
linkableValues = this.filterLinkableValues(linkableValues);
return linkableValues;
francois.grand
committed
}
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[] {
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
values.splice(i, 1);
francois.grand
committed
}
}
return values;
}
/**
* Resets the results of all forms depending on the given form "f"
* @param f
*/
public resetAllDependingFormsResults(f: FormulaireDefinition, visited: string[] = [], notify: boolean = true) {
const dependingNubs = Session.getInstance().getDependingNubs(f.currentNub.uid);
for (const dn of dependingNubs) {
if (! visited.includes(dn.uid)) {
const form = this.getFormulaireFromNubId(dn.uid);
mathias.chouet
committed
if (form) {
const hadResults = form.hasResults;
// form might not have a result, but still have another form depending on it !
form.resetResults(visited);
if (hadResults) {
if (notify) {
this.notificationsService.notify(
this.intlService.localizeText("INFO_SNACKBAR_RESULTS_INVALIDATED") + " " + form.calculatorName,
1500
mathias.chouet
committed
);
}
mathias.chouet
committed
}