Newer
Older
import { Component, ApplicationRef, OnInit, OnDestroy, HostListener, ViewChild, ComponentRef } from '@angular/core';
//import { MdDialog } from '@angular/material';
import { Router, ActivatedRoute } from '@angular/router';
import { Observer } from "jalhyd";
francois.grand
committed
import { environment } from '../environments/environment';
francois.grand
committed
import { InternationalisationService, Language, LanguageCode } from './services/internationalisation/internationalisation.service';
francois.grand
committed
import { ErrorService } from './services/error/error.service';
// import { AlertDialog } from './components/alert-dialog/alert-dialog.component';
import { FormulaireService } from './services/formulaire/formulaire.service';
francois.grand
committed
import { FormulaireDefinition } from './formulaire/definition/form-definition';
francois.grand
committed
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 { HelpService } from './services/help/help.service';
import { HelpComponent } from './components/help/help.component';
francois.grand
committed
import { LoadCalcDialogAnchorDirective } from './components/load-calculator/load-calculator-anchor.directive';
import { LoadCalculatorComponent } from './components/load-calculator/load-calculator.component';
francois.grand
committed
import { SaveCalcDialogAnchorDirective } from './components/save-calculator/save-calculator-anchor.directive';
import { SaveCalculatorComponent } from './components/save-calculator/save-calculator.component';
francois.grand
committed
selector: 'nghyd-app',
export class AppComponent implements OnInit, OnDestroy, Observer {
private _displayErrorDialog: boolean = false;
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 loadCalcDialogAnchor: LoadCalcDialogAnchorDirective;
francois.grand
committed
@ViewChild(SaveCalcDialogAnchorDirective)
private saveCalcDialogAnchor: SaveCalcDialogAnchorDirective;
/**
* 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,
private route: ActivatedRoute,
francois.grand
committed
private formulaireService: FormulaireService,
private httpService: HttpService,
private helpService: HelpService
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;
ServiceFactory.instance.helpService = helpService;
francois.grand
committed
francois.grand
committed
// process.on('unhandledRejection', (reason, p) => {
// console.log('Unhandled Rejection at: Promise', p, 'reason:', reason);
// // Stack Trace
// console.log(reason.stack);
// });
francois.grand
committed
ngOnInit() {
francois.grand
committed
this.intlService.addObserver(this);
this.intlService.setLocale("fr");
this.formulaireService.addObserver(this);
francois.grand
committed
this.subscribeErrorService();
}
ngOnDestroy() {
this.unsubscribeErrorService();
this.formulaireService.removeObserver(this);
}
private get uitextSidenavNewCalc() {
return this.intlService.localizeText("INFO_MENU_NOUVELLE_CALC");
}
private get uitextSidenavParams() {
return this.intlService.localizeText("INFO_SETUP_TITLE");
}
private get uitextSidenavLoadSession() {
return "Charger une session";
}
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() {
francois.grand
committed
let tag = this.intlService.currentLanguage.tag;
document['locale'] = tag;
// 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);
}
else if (sender instanceof FormulaireService) {
switch (data["action"]) {
const f: FormulaireDefinition = data["form"];
this._calculators.push(
{
"title": f.calculatorName,
"uid": String(f.uid)
}
);
// 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);
francois.grand
committed
else if (sender instanceof InternationalisationService) {
this.updateLocale();
}
else if (sender instanceof FormulaireDefinition) {
switch (data["action"]) {
case "nameChanged":
this.updateCalculatorTitle(sender, data["name"]);
break;
case "requestHelp":
this.toHelp(sender);
}
}
francois.grand
committed
}
private getCalculatorIndexFromId(formId: number) {
const index = this._calculators.reduce((resultIndex, calc, currIndex) => {
if (resultIndex == -1 && calc["uid"] == formId)
resultIndex = currIndex;
return resultIndex;
}, -1);
return index;
}
private updateCalculatorTitle(f: FormulaireDefinition, title: string) {
const formIndex = this.getCalculatorIndexFromId(f.uid);
this._calculators[formIndex]["title"] = title;
}
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
francois.grand
committed
const compRef: ComponentRef<SaveCalculatorComponent> = this.saveCalcDialogAnchor.createDialog();
francois.grand
committed
// création de la liste des formulaires
francois.grand
committed
let list = [];
for (const c of this._calculators) {
const uid = +c["uid"];
list.push({
"selected": uid === form.uid,
"title": c["title"],
"uid": uid
})
}
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);
prom.then(list => {
let name = compRef.instance.filename;
// ajout extension ".json"
const re = /.+\.json/;
const match = re.exec(name.toLowerCase());
if (match === null)
name = name + ".json";
this.saveSession(list, name)
francois.grand
committed
});
}
private saveSession(calcList: any[], filename) {
let elems = [];
francois.grand
committed
for (const c of calcList)
if (c.selected) {
francois.grand
committed
console.log(c.title);
const form: FormulaireDefinition = this.formulaireService.getFormulaireFromId(c.uid);
elems.push(form.JSONserialise());
let session = { "session": { "elements": elems } };
this.formulaireService.saveSession(session, filename);
francois.grand
committed
}
private closeCalculator(form: FormulaireDefinition) {
const formId: number = 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
*/
let newId: number = -1;
const l = this._calculators.length;
if (l > 1) {
if (closedIndex == l - 1)
newId = +this._calculators[closedIndex - 1]["uid"];
else
newId = +this._calculators[closedIndex + 1]["uid"];
}
// suppression
this._calculators = this._calculators.filter(calc => {
return formId != +calc["uid"]
});
// MAJ affichage
this._currentFormId = -1;
}
else
this.toCalc(newId);
}
private toList() {
this.router.navigate(['/list']);
}
private toCalc(id: number) {
this.router.navigate(['/calculator', id]);
}
private toHelp(form: FormulaireDefinition) {
// Activated.firstChild.component : type du composant actuellement affiché par le router outlet
this.router.navigate(['/help']).then(_ => {
const helpComp = this._routerCurrentComponent as HelpComponent;
helpComp.formHelp = form;
})
}
/**
* récupération du composant affiché par le routeur
*/
private 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)
private get ruler(): boolean {
francois.grand
committed
// return !environment.production;
return false;
francois.grand
committed
}
// sidenav
private openNav() {
document.getElementById("mySidenav").style.width = "300px";
}
private closeNav() {
document.getElementById("mySidenav").style.width = "0";
}
private newCalc() {
this.closeNav();
francois.grand
committed
this.toList();
}
private loadSession() {
this.closeNav();
// création du dialogue de sélection des formulaires à sauver
const compRef: ComponentRef<LoadCalculatorComponent> = this.loadCalcDialogAnchor.createDialog();
const prom: Promise<any[]> = compRef.instance.run();
prom.then(list => {
francois.grand
committed
this.formulaireService.loadSession(compRef.instance.selectedFile, compRef.instance.calculators);
compRef.destroy();
});
}
private params() {
this.closeNav();
this.router.navigate(['/setup']);
}
francois.grand
committed
// /sidenav
/**
* détection de la fermeture de la page/navigateur et demande de confirmation
*/
@HostListener('window:beforeunload', ['$event'])
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é
$event.returnValue = 'Your data will be lost !';
}
francois.grand
committed
}