Skip to content
Snippets Groups Projects
formulaire.service.ts 22.1 KiB
Newer Older
import { saveAs } from "file-saver";
import { CalculatorType, EnumEx, Observable, ParamDefinition, Session, Nub, ParallelStructure } from "jalhyd";
import { HttpService } from "../../services/http/http.service";
import { I18nService } from "../../services/internationalisation/internationalisation.service";
import { FormulaireDefinition } from "../../formulaire/definition/form-definition";
import { FormulaireElement } from "../../formulaire/formulaire-element";
import { InputField } from "../../formulaire/input-field";
import { SelectField } from "../../formulaire/select-field";
import { CheckField } from "../../formulaire/check-field";
import { StringMap } from "../../stringmap";
import { FormulaireBase } from "../../formulaire/definition/concrete/form-base";
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";
import { NgParameter } from "../../formulaire/ngparam";
import { FieldsetContainer } from "../..//formulaire/fieldset-container";

@Injectable()
export class FormulaireService extends Observable {
    private _formulaires: FormulaireDefinition[];

    private _currentFormId: string = null;
    constructor(
        private i18nService: I18nService,
        private httpService: HttpService) {

        super();
        this._formulaires = [];
    }

    private get _intlService(): I18nService {
        return this.i18nService;
        return this.httpService;
    public get formulaires(): FormulaireDefinition[] {
        return this._formulaires;
    }

    private loadLocalisation(calc: CalculatorType): Promise<any> {
mathias.chouet's avatar
mathias.chouet committed
        const f: string = this.getConfigPathPrefix(calc) + this._intlService.currentLanguage.tag + ".json";
        const prom = this._httpService.httpGetRequestPromise(f);
        return prom.then((j) => {
            return j as StringMap;
    }

    /**
     * met à jour la langue du formulaire
     * @param formId id unique du formulaire
     * @param localisation ensemble id-message traduit
     */
    private updateFormulaireLocalisation(formId: string, localisation: StringMap) {
mathias.chouet's avatar
mathias.chouet committed
        for (const f of this._formulaires) {
David Dorchies's avatar
David Dorchies committed
            if (f.uid === formId) {
                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);
            });
    }

    public updateLocalisation() {
        for (const c of EnumEx.getValues(CalculatorType)) {
            const prom: Promise<StringMap> = this.loadLocalisation(c);
            prom.then(loc => {
David Dorchies's avatar
David Dorchies committed
                for (const f of this._formulaires) {
                    if (f.calculatorType === c) {
                        this.updateFormulaireLocalisation(f.uid, loc);
mathias.chouet's avatar
mathias.chouet committed
    /**
     * 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`);
mathias.chouet's avatar
mathias.chouet committed
    /**
     * 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 {
        let f: FormulaireDefinition;
        switch (ct) {
            case CalculatorType.ConduiteDistributrice:
            case CalculatorType.PabDimensions:
            case CalculatorType.PabPuissance:
            case CalculatorType.MacroRugo:
                f = new FormulaireBase();
            case CalculatorType.Dever:
            case CalculatorType.Cloisons:
                f = new FormulaireParallelStructure();
                throw new Error(`FormulaireService.newFormulaire() : type de module de calcul ${CalculatorType[ct]} non pris en charge`);
        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);
        this._formulaires.push(f);
        // Charge la configuration dépendamment du type
mathias.chouet's avatar
mathias.chouet committed
        const prom: Promise<any> = this.loadConfig(ct);
mathias.chouet's avatar
mathias.chouet committed
            f.preparseConfig(s);
mathias.chouet's avatar
mathias.chouet committed

            // Associe le Nub fourni (chargement de session / duplication de module), sinon en crée un nouveau
            if (nub) {
                f.currentNub = nub;
            } else {
                f.initNub();
            }
mathias.chouet's avatar
mathias.chouet committed

            // Restaure le nom du module, sinon affecte le nom par défaut
            let tempName: string;
mathias.chouet's avatar
mathias.chouet committed
                tempName = calculatorName;
mathias.chouet's avatar
mathias.chouet committed
                tempName = decode(this.getLocalisedShortTitleFromCalculatorType(ct));
mathias.chouet's avatar
mathias.chouet committed
            // Suffixe le nom du module si nécessaire
            f.calculatorName = this.suffixNameIfNeeded(tempName);

            f.parseConfig();
mathias.chouet's avatar
mathias.chouet committed

            // add fieldsets for existing structures if needed
            if (f.currentNub instanceof ParallelStructure) {
                for (s of f.currentNub.structures) {
                    for (const e of f.allFormElements) {
mathias.chouet's avatar
mathias.chouet committed
                        if (e instanceof FieldsetContainer) { // @TODO manage many containers one day ?
            return this.loadUpdateFormulaireLocalisation(f);
mathias.chouet's avatar
mathias.chouet committed
        }).then(fi => {
            fi.applyDependencies();
            this.notifyObservers({
                "action": "createForm",
                "form": fi
            });
mathias.chouet's avatar
mathias.chouet committed
            return fi;
    /**
     * 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 {
mathias.chouet's avatar
mathias.chouet committed
        for (const f of this._formulaires) {
            if (f.uid === uid) {
                return f;
mathias.chouet's avatar
mathias.chouet committed
            }
        }
    public getInputField(formId: string, elemId: string): InputField {
        const s = this.getFormulaireElementById(formId, elemId);
mathias.chouet's avatar
mathias.chouet committed
        if (!(s instanceof InputField)) {
            throw new Error("Form element with id '" + elemId + "' is not an input");
        }
        return <InputField>s;
    }

    public getCheckField(formId: string, elemId: string): CheckField {
        const s = this.getFormulaireElementById(formId, elemId);
mathias.chouet's avatar
mathias.chouet committed
        if (!(s instanceof CheckField)) {
            throw new Error("Form element with id '" + elemId + "' is not a checkbox");
mathias.chouet's avatar
mathias.chouet committed
        }
        return <CheckField>s;
    public getSelectField(formId: string, elemId: string): SelectField {
        const s = this.getFormulaireElementById(formId, elemId);
mathias.chouet's avatar
mathias.chouet committed
        if (!(s instanceof SelectField)) {
            throw new Error("Form element with id '" + elemId + "' is not a select");
mathias.chouet's avatar
mathias.chouet committed
        }
        return <SelectField>s;
    private getFormulaireElementById(formId: string, elemId: string): FormulaireElement {
mathias.chouet's avatar
mathias.chouet committed
        for (const f of this._formulaires) {
David Dorchies's avatar
David Dorchies committed
            if (f.uid === formId) {
                const s = f.getFormulaireNodeById(elemId);
mathias.chouet's avatar
mathias.chouet committed
                if (s !== undefined) {
mathias.chouet's avatar
mathias.chouet committed
                }
mathias.chouet's avatar
mathias.chouet committed
        }
    public getParamdefParentForm(prm: ParamDefinition): FormulaireDefinition {
mathias.chouet's avatar
mathias.chouet committed
        for (const f of this._formulaires) {
            for (const p of f.allFormElements) {
                if (p instanceof NgParameter) {
                    if (p.paramDefinition.uid === prm.uid) {
mathias.chouet's avatar
mathias.chouet 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 {
mathias.chouet's avatar
mathias.chouet committed
        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.";

            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.";

francois.grand's avatar
francois.grand committed
            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.";

mathias.chouet's avatar
mathias.chouet committed
                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({
                "action": "closeForm",
        if (nub) {
            Session.getInstance().deleteNub(nub);
        }
    }

    public get currentFormId() {
        return this._currentFormId;
    }

    public setCurrentForm(formId: string) {
        const form = this.getFormulaireFromId(formId);
David Dorchies's avatar
David Dorchies committed
        if (form === undefined) {
            this._currentFormId = null;
            this.notifyObservers({
                "action": "invalidFormId",
                "formId": formId
            });
David Dorchies's avatar
David Dorchies committed
        } else {
            this._currentFormId = formId;
            this.notifyObservers({
                "action": "currentFormChanged",
                "formId": formId

    private get currentForm(): FormulaireDefinition {
        return this.getFormulaireFromId(this._currentFormId);
    }

    public get currentFormHasResults(): boolean {
        const form = this.currentForm;
mathias.chouet's avatar
mathias.chouet committed
        if (form !== undefined) {
mathias.chouet's avatar
mathias.chouet committed
        }
    private readSingleFile(file: File): Promise<any> {
        return new Promise<any>((resolve, reject) => {
            const fr = new FileReader();

            fr.onload = () => {
                resolve(fr.result);
            };

            fr.onerror = () => {
                fr.abort();
                reject(new Error(`Erreur de lecture du fichier ${file.name}`));
            };

            fr.readAsText(file);
        });
    }

     * charge une session en tenant compte des modules de calcul sélectionnées
     * @param formInfos infos sur les modules de calcul @see DialogLoadSessionComponent.calculators
     */
    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" });
     * obtient des infos (nom, uid des modules de calcul) d'un fichier session
     * @param f fichier session
     */
    public calculatorInfosFromSessionFile(f: File): Promise<any[]> {
        return this.readSingleFile(f).then(s => {
            const res: any[] = [];
            // 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
                    });
                });
mathias.chouet's avatar
mathias.chouet 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) {
        // 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`);
            }
        }
     * @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>}
     * @param p paramètre qui sert de clé de recherche des paramètres liables
    public getLinkableValues(p: NgParameter): any[] {
        const res: any[] = [];
mathias.chouet's avatar
mathias.chouet committed
        if (p !== undefined) {
                const sn = f.currentNub;
                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
                    const exclude = np !== undefined ? p.paramDefinition.uid === np.uid : false;
                    // valeurs liables
                    const ps = sn.getLinkableValues(p.paramDefinition, undefined, exclude);
mathias.chouet's avatar
mathias.chouet committed
                    for (const npp of ps) {
                        npp["formTitle"] = f.calculatorName;
                        res.push(npp);
mathias.chouet's avatar
mathias.chouet committed
                } catch (e) {
mathias.chouet's avatar
mathias.chouet committed
        }

    /**
     * filtre les valeurs liables à un paramètre :
mathias.chouet's avatar
mathias.chouet committed
     * - 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 //)
     * @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
                let found = false;
mathias.chouet's avatar
mathias.chouet committed
                if (parentForm !== undefined) {
                    for (const fe of parentForm.allFormElements) {
                        if (fe instanceof NgParameter) {
                            if (fe.paramDefinition.uid === prm.uid) {
                                found = true;
                                break;
                            }
mathias.chouet's avatar
mathias.chouet committed
                if (!found) {
mathias.chouet's avatar
mathias.chouet committed
                }