Newer
Older
import { Component, ApplicationRef, OnInit, OnDestroy, HostListener, ViewChild, ComponentRef } from "@angular/core";
import { Router, Event, NavigationEnd } from "@angular/router";
import { environment } from "../environments/environment";
import { InternationalisationService } from "./services/internationalisation/internationalisation.service";
import { ErrorService } from "./services/error/error.service";
import { FormulaireService } from "./services/formulaire/formulaire.service";
import { FormulaireDefinition } from "./formulaire/definition/form-definition";
import { ServiceFactory } from "./services/service-factory";
import { ParamService } from "./services/param/param.service";
import { ApplicationSetupService } from "./services/app-setup/app-setup.service";
import { HttpService } from "./services/http/http.service";
import { LoadCalcDialogAnchorDirective } from "./components/load-calculator/load-calculator-anchor.directive";
import { LoadCalculatorComponent } from "./components/load-calculator/load-calculator.component";
import { SaveCalcDialogAnchorDirective } from "./components/save-calculator/save-calculator-anchor.directive";
import { SaveCalculatorComponent } from "./components/save-calculator/save-calculator.component";
import { MatSidenav, MatToolbar } from "@angular/material";
selector: "nghyd-app",
templateUrl: "./app.component.html",
styleUrls: ["./app.component.scss"],
providers: [ErrorService]
export class AppComponent implements OnInit, OnDestroy, Observer {
@ViewChild("sidenav")
public sidenav: MatSidenav;
@ViewChild("navbar")
public navbar: MatToolbar;
francois.grand
committed
/**
* liste des calculettes. Forme des objets :
* "title": nom de la calculette
* "uid": id unique du formulaire
*/
/**
* id du formulaire courant
* on utilise pas directement FormulaireService.currentFormId pour éviter l'erreur ExpressionChangedAfterItHasBeenCheckedError
*/
private _currentFormId: number;
francois.grand
committed
@ViewChild(LoadCalcDialogAnchorDirective)
private appLoadCalcDialogAnchor: LoadCalcDialogAnchorDirective;
francois.grand
committed
francois.grand
committed
@ViewChild(SaveCalcDialogAnchorDirective)
private appSaveCalcDialogAnchor: SaveCalcDialogAnchorDirective;
francois.grand
committed
/**
* composant actuellement affiché par l'élément <router-outlet>
*/
private _routerCurrentComponent: Component;
constructor(
private intlService: InternationalisationService,
francois.grand
committed
private paramService: ParamService,
private appSetupService: ApplicationSetupService,
private appRef: ApplicationRef,
private errorService: ErrorService,
private router: Router,
francois.grand
committed
private formulaireService: FormulaireService,
francois.grand
committed
ServiceFactory.instance.applicationSetupService = appSetupService;
ServiceFactory.instance.paramService = paramService;
ServiceFactory.instance.internationalisationService = intlService;
francois.grand
committed
ServiceFactory.instance.formulaireService = formulaireService;
francois.grand
committed
ServiceFactory.instance.httpService = httpService;
francois.grand
committed
this.router.events.subscribe((event: Event) => {
// close side navigation when clicking a calculator tab
if (event instanceof NavigationEnd) {
this.sidenav.close();
}
});
}
francois.grand
committed
ngOnInit() {
francois.grand
committed
this.intlService.addObserver(this);
this.intlService.setLocale("fr");
this.formulaireService.addObserver(this);
francois.grand
committed
this.subscribeErrorService();
this._innerWidth = window.innerWidth;
francois.grand
committed
}
ngOnDestroy() {
this.unsubscribeErrorService();
this.formulaireService.removeObserver(this);
}
@HostListener("window:resize", ["$event"])
onResize(event) {
this._innerWidth = window.innerWidth;
}
public get uitextSidenavNewCalc() {
return this.intlService.localizeText("INFO_MENU_NOUVELLE_CALC");
}
public get uitextSidenavParams() {
return this.intlService.localizeText("INFO_SETUP_TITLE");
}
public get uitextSidenavLoadSession() {
return this.intlService.localizeText("INFO_MENU_LOAD_SESSION_TITLE");
}
public get uitextSidenavHelp() {
return this.intlService.localizeText("INFO_MENU_HELP_TITLE");
public get calculators() {
return this._calculators;
}
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
151
152
153
154
155
156
157
158
159
public setActiveCalc(uid: string) {
this._calculators.forEach((calc) => {
calc.active = (calc.uid === uid);
});
}
/**
* Returns true if sum of open calculator tabs witdh is lower than navbar
* available space (ie. if navbar is not overflowing), false otherwise
*/
public get tabsFitInNavbar() {
// manual breakpoints
// @WARNING keep in sync with .calculator-buttons sizes in app.component.scss
let tabsLimit = 2;
if (this._innerWidth > 320) {
tabsLimit = 3;
}
if (this._innerWidth > 480) {
tabsLimit = 3;
}
if (this._innerWidth > 640) {
tabsLimit = 4;
}
if (this._innerWidth > 800) {
tabsLimit = 6;
}
if (this._innerWidth > 1200) {
tabsLimit = 8;
}
const fits = this._calculators.length <= tabsLimit;
return fits;
}
francois.grand
committed
/**
* abonnement au service d'erreurs
*/
private subscribeErrorService() {
this.errorService.addObserver(this);
francois.grand
committed
}
francois.grand
committed
private unsubscribeErrorService() {
this.errorService.removeObserver(this);
}
francois.grand
committed
private updateLocale() {
const tag = this.intlService.currentLanguage.tag;
document["locale"] = tag;
francois.grand
committed
// location.reload(true);
// this.cdRef.markForCheck();
// this.cdRef.detectChanges();
this.appRef.tick();
}
francois.grand
committed
// interface Observer
update(sender: any, data: any): void {
if (sender instanceof ErrorService) {
// on ouvre un dialogue avec le message d'erreur reçu
// if (this._displayErrorDialog) {
// let dialogRef = this.dialog.open(AlertDialog);
// let ad: AlertDialog = dialogRef.componentInstance;
// ad.text = String(data);
// }
// else
console.log(data);
switch (data["action"]) {
const f: FormulaireDefinition = data["form"];
this._calculators.push(
{
"title": f.calculatorName,
"type": this.formulaireService.getLocalisedTitleFromCalculatorType(f.calculatorType),
"uid": f.uid
}
);
this.setActiveCalc(f.uid);
this._tabsChanged = true;
// abonnement en tant qu'observateur du nouveau formulaire
f.addObserver(this);
break;
francois.grand
committed
francois.grand
committed
this.toList();
break;
case "currentFormChanged":
/*
utilisation de setTimeout() pour éviter le message console ExpressionChangedAfterItHasBeenCheckedError
relatif au getter getHighlightClass() (qui change de valeur quand le formulaire courant change)
*/
setTimeout(() => {
this._currentFormId = data["formId"];
}, 1);
break;
francois.grand
committed
case "saveForm":
this.saveForm(data["form"]);
break;
const form: FormulaireDefinition = data["form"];
this.closeCalculator(form);
} else if (sender instanceof InternationalisationService) {
francois.grand
committed
this.updateLocale();
} else if (sender instanceof FormulaireDefinition) {
switch (data["action"]) {
case "nameChanged":
this.updateCalculatorTitle(sender, data["name"]);
break;
}
}
francois.grand
committed
}
private getCalculatorIndexFromId(formId: string) {
const index = this._calculators.reduce((resultIndex, calc, currIndex) => {
if (resultIndex === -1 && calc["uid"] === formId) {
return resultIndex;
}, -1);
return index;
}
private updateCalculatorTitle(f: FormulaireDefinition, title: string) {
const formIndex = this.getCalculatorIndexFromId(f.uid);
this._calculators[formIndex]["title"] = title;
this._tabsChanged = true; // recompute size !
}
francois.grand
committed
/**
* sauvegarde du/des formulaires
* @param form formulaire à sélectionner par défaut dans la liste
*/
francois.grand
committed
private saveForm(form: FormulaireDefinition) {
francois.grand
committed
// création du dialogue de sélection des formulaires à sauver
const compRef: ComponentRef<SaveCalculatorComponent> = this.appSaveCalcDialogAnchor.createDialog();
francois.grand
committed
francois.grand
committed
// création de la liste des formulaires
francois.grand
committed
for (const c of this._calculators) {
const uid = c["uid"];
francois.grand
committed
list.push({
"selected": uid === form.uid,
"title": c["title"],
"uid": uid
francois.grand
committed
}
francois.grand
committed
// passage de la liste, récupération d'une Promise pour traiter le résultat
const prom: Promise<any[]> = compRef.instance.run(list);
let name = compRef.instance.filename;
// ajout extension ".json"
const re = /.+\.json/;
const match = re.exec(name.toLowerCase());
name = name + ".json";
francois.grand
committed
}
private saveSession(calcList: any[], filename) {
if (c.selected) {
francois.grand
committed
console.log(c.title);
const form: FormulaireDefinition = this.formulaireService.getFormulaireFromId(c.uid);
elems.push(form.JSONserialise());
}
const session = { "session": { "elements": elems } };
this.formulaireService.saveSession(session, filename);
francois.grand
committed
}
private closeCalculator(form: FormulaireDefinition) {
const formId: string = form.uid;
// désabonnement en tant qu'observateur
form.removeObserver(this);
// recherche de la calculette correspondante à formId
const closedIndex = this.getCalculatorIndexFromId(formId);
* détermination de la nouvelle calculette à afficher :
* - celle après celle supprimée
* - ou celle avant celle supprimée si on supprime la dernière
*/
const l = this._calculators.length;
if (l > 1) {
newId = this._calculators[closedIndex - 1]["uid"];
newId = this._calculators[closedIndex + 1]["uid"];
}
// suppression
this._calculators = this._calculators.filter(calc => {
return formId !== calc["uid"];
});
// MAJ affichage
if (newId === null) {
this._currentFormId = null;
}
private toList() {
/**
* récupération du composant affiché par le routeur
*/
public onRouterOutletActivated(a) {
this._routerCurrentComponent = a;
}
private getHighlightClass(uid: number) {
return uid === this._currentFormId ? "blue darken-2" : "";
francois.grand
committed
// flag d'affichage des repères des colonnes Bootstrap : uniquement en mode dev
// cf. src/environments/*, ng build --env=<mode> (par ex : ng build --env=prod)
public get ruler(): boolean {
francois.grand
committed
// return !environment.production;
return false;
francois.grand
committed
}
// création du dialogue de sélection des formulaires à sauver
const compRef: ComponentRef<LoadCalculatorComponent> = this.appLoadCalcDialogAnchor.createDialog();
const prom: Promise<any[]> = compRef.instance.run();
prom.then(list => {
francois.grand
committed
this.formulaireService.loadSession(compRef.instance.selectedFile, compRef.instance.calculators);
public getDateRevision(): string[] {
const dr: string[] = [jalhydDateRev, nghydDateRev];
return dr;
}
francois.grand
committed
/**
* détection de la fermeture de la page/navigateur et demande de confirmation
*/
francois.grand
committed
confirmExit($event) {
francois.grand
committed
if (environment.production) {
// affecter une valeur différente de null provoque l'affichage d'un dialogue de confirmation, mais le texte n'est pas affiché
francois.grand
committed
}
francois.grand
committed
}