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,
Pab,
Props,
Cloisons,
CloisonAval
} 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";
import { FormulairePab } from "../../formulaire/definition/concrete/form-pab";
import { FormulaireMacrorugoCompound } from "../../formulaire/definition/concrete/form-macrorugo-compound";
export class FormulaireService extends Observable {
private calculatorPaths = {};
private _formulaires: FormulaireDefinition[];
private _currentFormId: string = null;
/** to avoid loading language files multiple times */
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";
this.calculatorPaths[CalculatorType.Pab] = "pab";
this.calculatorPaths[CalculatorType.MacroRugoCompound] = "macrorugo-compound";
private get _intlService(): I18nService {
return this.i18nService;
francois.grand
committed
}
private get _httpService(): HttpService {
francois.grand
committed
}
public get formulaires(): FormulaireDefinition[] {
return this._formulaires;
}
public get languageCache() {
return this._languageCache;
}
/**
* Loads the localisation file dedicated to calculator type ct; tries the current
* language then the fallback language; uses cache if available
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)) {
});
} 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`);
}
/**
* Retourne le titre cour du type de module de calcul, dans la langue en cours
* (pour les titres d'onglets par défaut)
*/
mathias.chouet
committed
public getLocalisedShortTitleFromCalculatorType(type: any) {
if (typeof type !== "string") { // retrocompatibility for old file format
type = CalculatorType[type];
}
const sCalculator: string = type.toUpperCase();
return this._intlService.localizeText(`INFO_${sCalculator}_TITRE_COURT`);
}
/**
* Returns variable name from symbol
* @param calcType
* @param symbol
*/
public expandVariableName(calcType: CalculatorType, symbol: string): string {
let s = "";
// language cache…
let langCache = this.languageCache;
if (langCache && langCache[calcType]) {
langCache = langCache[calcType]; // …for target Nub type
}
if (langCache && langCache[this.intlService.currentLanguage]) {
langCache = langCache[this.intlService.currentLanguage]; // … for current language
}
if (langCache && langCache[symbol] !== undefined) {
s = this.intlService.localizeText(symbol, langCache);
} else {
// is symbol of the form ouvrages[i]… ?
const re = /([A-Z,a-z]+)\[(\d+)\]\.(.+)/;
const match = re.exec(symbol);
if (match) {
// Les libellés correspondants sont INFO OUVRAGE et INFO_LIB_OUVRAGE_XXX
s = this.intlService.localizeText(`INFO_${match[1].toUpperCase()}`)
+ " n°" + (+match[2] + 1) + ": "
+ this.expandVariableName(calcType, `${match[1].toUpperCase()}_${match[3].toUpperCase()}`);
} else {
s = this.intlService.localizeText("INFO_LIB_" + symbol.toLocaleUpperCase());
}
}
return s;
}
/**
* Returns variable name and unit from symbol
* @param calcType
* @param symbol
*/
public expandVariableNameAndUnit(calcType: CalculatorType, symbol: string): string {
let s = this.expandVariableName(calcType, symbol);
let langCache = this.languageCache; // language cache…
if (langCache && langCache[calcType]) {
langCache = langCache[calcType]; // …for target Nub type
}
if (langCache && langCache[this.intlService.currentLanguage]) {
langCache = langCache[this.intlService.currentLanguage]; // … for current language
}
mathias.chouet
committed
// remove device number before looking for unit (hacky)
let symbolBase = symbol.toLocaleUpperCase();
const idx = symbolBase.indexOf("].");
if (idx !== -1) {
symbolBase = symbolBase.substring(idx + 2);
}
const unitKey = "UNIT_" + symbolBase;
if (langCache && langCache[unitKey] !== undefined) {
s = s + " (" + this.intlService.localizeText(unitKey, langCache) + ")";
}
return s;
}
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
/**
* 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();
case CalculatorType.Pab:
f = new FormulairePab();
break;
case CalculatorType.MacroRugoCompound:
f = new FormulaireMacrorugoCompound();
break;
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
// (when loading session only)
if (f.currentNub instanceof ParallelStructure) {
for (const struct of f.currentNub.structures) {
for (const e of f.allFormElements) {
if (e instanceof FieldsetContainer) { // @TODO manage many containers one day ?
e.addFromTemplate(0, undefined, struct);
}
}
}
}
// when creating a new Pab, add one wall with one device, plus the downwall
// (when loading session, those items are already present)
if (
f instanceof FormulairePab
&& f.currentNub instanceof Pab
&& f.currentNub.children.length === 0
&& f.currentNub.downWall === undefined
) {
// 1. one wall
const newWall = Session.getInstance().createNub(
new Props({
calcType: CalculatorType.Cloisons
})
) as Cloisons;
// add new default device for new wall
const newDevice = Session.getInstance().createNub(
new Props({
calcType: CalculatorType.Structure,
loiDebit: newWall.getDefaultLoiDebit()
})
);
newWall.addChild(newDevice);
f.pabNub.addChild(newWall);
// 2. downwall
const newDownWall = Session.getInstance().createNub(
new Props({
// add new default device for new downwall
const newDownwallDevice = Session.getInstance().createNub(
new Props({
calcType: CalculatorType.Structure,
loiDebit: newDownWall.getDefaultLoiDebit()
})
);
newDownWall.addChild(newDownwallDevice);
f.pabNub.downWall = newDownWall;
}
// @TODO add aprons for existing Aprons if needed
// (when loading session only)
/* if (f.currentNub instanceof MacrorugoCompound) {
for (const struct of f.currentNub.structures) {
for (const e of f.allFormElements) {
if (e instanceof FieldsetContainer) { // @TODO manage many containers one day ?
e.addFromTemplate(0, undefined, struct);
}
}
}
} */
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
mathias.chouet
committed
this.resetAllDependingFormsResults(form, [], undefined, true, 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és
francois.grand
committed
* @param f fichier session
* @param formInfos infos sur les modules de calcul @see DialogLoadSessionComponent.calculators
francois.grand
committed
*/
public async loadSession(f: File, formInfos: any[]): Promise<{ hasErrors: boolean, loaded: string[] }> {
const uids: string[] = [];
formInfos.forEach((fi) => {
if (fi.selected) {
uids.push(fi.uid);
const res = Session.getInstance().unserialise(s, uids);
const newNubs = res.nubs;
// for each new Nub, create the related form, set its title
for (let i = 0; i < newNubs.length; i++) {
const nn = newNubs[i];
let title;
if (nn.meta && nn.meta.title) {
title = nn.meta.title;
}
await this.loadLocalisation(nn.nub.calcType);
await this.createFormulaire(nn.nub.calcType, nn.nub, title); // await guarantees loading order
}
// forward errors
return {
hasErrors: res.hasErrors,
loaded: newNubs.map(n => n.nub.uid)
// forward errors to caller to avoid "Uncaught (in promise)"
/**
* 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épendances) d'un fichier session
francois.grand
committed
* @param f fichier session
*/
public calculatorInfosFromSessionFile(f: File): Promise<{ nubs: any[], formatVersion: string }> {
francois.grand
committed
return this.readSingleFile(f).then(s => {
// return value
const res: { nubs: any[], formatVersion: string } = {
nubs: [],
formatVersion: ""
};
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: [],
type: e.props.calcType
};
// 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) => {
// version du format de fichier
if (data.header && data.header.format_version) {
res.formatVersion = data.header.format_version;
}
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
mathias.chouet
committed
* @param symbol symbol of the parameter whose value change triggered this method
* @param forceResetAllDependencies if true, even non-calculated non-modified parameters
* links will be considered as dependencies @see jalhyd#98
mathias.chouet
committed
public resetAllDependingFormsResults(
f: FormulaireDefinition,
visited: string[] = [],
symbol?: string,
forceResetAllDependencies: boolean = false,
notify: boolean = true
) {
const dependingNubs = Session.getInstance().getDependingNubs(f.currentNub.uid, symbol, forceResetAllDependencies);
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
}