turbogui-angular 15.2.0 → 15.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -239,8 +239,8 @@ export class DialogService extends SingletoneStrictClass {
239
239
  * - viewContainerRef: This is important if we want to propagate providers from a parent component to this dialog. We must specify
240
240
  * this reference to make sure the same services injected on the parent are available too at the child dialog
241
241
  *
242
- * @param callback A function that will be called after the dialog is closed and will receive a selection object with the numeric index and value for
243
- * the option that's been selected by the user. if no option was selected, index will be -1 and value null
242
+ * @param callback A function that will be called after the dialog is closed. It will receive a selection object with two properties: index and value. Those
243
+ * will contain the index and value from the options array that's selected by the user. if no option selected, index will be -1 and value null
244
244
  */
245
245
  addDialog(dialogComponentClass, properties, callback = null) {
246
246
  if (!this._isEnabled) {
@@ -403,4 +403,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
403
403
  providedIn: 'root',
404
404
  }]
405
405
  }], ctorParameters: () => [{ type: i0.RendererFactory2 }, { type: i1.MatSnackBar }, { type: i2.MatDialog }, { type: i0.Injector }, { type: i0.ApplicationRef }, { type: i0.ComponentFactoryResolver }] });
406
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/dialog.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { ArrayUtils, NumericUtils } from 'turbocommons-ts';\r\nimport { Type, Injectable, ComponentFactoryResolver, Injector, ApplicationRef, Renderer2, RendererFactory2, ViewContainerRef } from '@angular/core';\r\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\r\nimport { MatSnackBar, MatSnackBarConfig } from '@angular/material/snack-bar';\r\nimport { BusyStateBaseComponent } from '../view/components/busy-state-base/busy-state-base.component';\r\nimport { ComponentPortal, DomPortalOutlet } from '@angular/cdk/portal';\r\nimport { DialogBaseComponent } from '../view/components/dialog-base/dialog-base.component';\r\nimport { DialogDateSelectionComponent } from '../view/components/dialog-date-selection/dialog-date-selection.component';\r\nimport { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\n\r\n\r\n/**\r\n * Manages the application modal and non modal floating elements\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class DialogService extends SingletoneStrictClass {\r\n\r\n\r\n    /**\r\n     * Used to modify the busy state component that is shown by default by the addModalBusyState() method.\r\n     *\r\n     * @see this.addModalBusyState()\r\n     */\r\n    customBusyStateComponentClass: Type<BusyStateBaseComponent> = BusyStateBaseComponent;\r\n\r\n\r\n    /**\r\n     * Check public getter for docs\r\n     */\r\n    private _isEnabled = true;\r\n\r\n\r\n    /**\r\n     * Tells if the main application is currently showing a busy state that blocks all user interaction\r\n     */\r\n    private _isShowingBusyState = false;\r\n\r\n\r\n    /**\r\n     * A reference to the modal busy state component that is initialized only the first time it is called.\r\n     *\r\n     * (To append the busy state dynamically, we use the Portal concept from the angular material library)\r\n     */\r\n    private _componentPortal: ComponentPortal<BusyStateBaseComponent> | null = null;\r\n\r\n\r\n    /**\r\n     * A reference to the modal busy state container where the component will be added\r\n     */\r\n    private _modalBusyStateHost: DomPortalOutlet | null = null;\r\n\r\n\r\n    /**\r\n     * Tells if the manager is currently showing a snackbar element or not\r\n     */\r\n    private _isShowingSnackBar = false;\r\n\r\n\r\n    /**\r\n     * Contains a list of the dialogs that are currently visible to the user.\r\n     * Each item in this list is a hash that is computed when dialog is created to uniquely identify it.\r\n     *\r\n     * Empty list means no dialogs are currently visible\r\n     */\r\n    private _activeDialogs: string[] = [];\r\n\r\n\r\n    /**\r\n     * Contains a list with all the MatDialog instances that are currently visible to the user.\r\n     * The list uses the same order as the list of _activeDialogs hash values\r\n     */\r\n    private _activeDialogInstances: MatDialogRef<DialogBaseComponent>[] = [];\r\n\r\n\r\n    /**\r\n     * Used to store the initialized Renderer 2 instance that is used by this class\r\n     */\r\n    private readonly _renderer: Renderer2;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the window beforeunload event listener once not used anymore\r\n     */\r\n    private _windowBeforeUnloadUnListen: (() => void) | null = null;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the document keydown event listener once not used anymore\r\n     */\r\n    private _documentKeydownUnlisten: (() => void) | null = null;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the document mousedown event listener once not used anymore\r\n     */\r\n    private _documentMousedownUnlisten: (() => void) | null = null;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the document pointerdown event listener once not used anymore\r\n     */\r\n    private _documentPointerdownUnlisten: (() => void) | null = null;\r\n\r\n\r\n    constructor(rendererFactory: RendererFactory2,\r\n                private readonly matSnackBar: MatSnackBar,\r\n                private readonly matDialog: MatDialog,\r\n                private readonly injector: Injector,\r\n                private readonly applicationRef: ApplicationRef,\r\n                private readonly componentFactoryResolver: ComponentFactoryResolver) {\r\n\r\n\t\tsuper(DialogService);\r\n\r\n        this._renderer = rendererFactory.createRenderer(null, null);\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if this dialog service is enabled or not. If dialog service is disabled, none of it's features will work\r\n     * This is used to block all dialog features normally when shutting down the application\r\n     *\r\n     * Use with caution. When this is set to false, dialog service stops working.\r\n     */\r\n    set isEnabled(v: boolean) {\r\n\r\n        if (v === this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n\r\n        this._isEnabled = v;\r\n    }\r\n\r\n\r\n    /**\r\n     * Enables a warning that will be shown to the user when he/she tries to close the application.\r\n     * This warning will prompt the user to continue with the exit process or cancel it.\r\n     * The specific texts of this message are a generic ones and cannot be changed.\r\n     *\r\n     * IMPORTANT: This method must be called after the main application has been initialized in order to work,\r\n     * otherwise it will do nothing.\r\n     */\r\n    addCloseApplicationWarning() {\r\n\r\n        if (this._windowBeforeUnloadUnListen === null) {\r\n\r\n            this._windowBeforeUnloadUnListen = this._renderer.listen('window', 'beforeunload',\r\n                (event) => event.returnValue = true);\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Remove the close application warning message if previously assigned\r\n     */\r\n    removeCloseApplicationWarning() {\r\n\r\n        if (this._windowBeforeUnloadUnListen !== null) {\r\n\r\n            this._windowBeforeUnloadUnListen();\r\n            this._windowBeforeUnloadUnListen = null;\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Change the application visual appearance so the user perceives that the application is\r\n     * currently busy. While modal busy state is enabled, no user input is possible neither via\r\n     * keyboard, mouse or touch. Use this state when performing server requests or operations that\r\n     * must block the user interaction with the application. To allow user interaction again, you must\r\n     * call removeModalBusyState()\r\n     *\r\n     * Notice: We can modify the busy state visual component that is shown by this method. To do it, we must\r\n     * set this.customBusyStateComponentClass property with our own custom busy state component class. (We can do it at\r\n     * our main application component constructor for example). Our custom component must extend the\r\n     * BusyStateBaseComponent one to add its own visual appearance.\r\n     *\r\n     * @see this.customBusyStateComponentClass\r\n     */\r\n    addModalBusyState() {\r\n\r\n        if (!this._isEnabled || this._isShowingBusyState) {\r\n\r\n            return;\r\n        }\r\n\r\n        this._disableUserInteraction();\r\n\r\n        // Dynamically create the busy state component reference if this is the first time\r\n        if (this._componentPortal === null) {\r\n\r\n            this._componentPortal = new ComponentPortal(this.customBusyStateComponentClass);\r\n\r\n            // Create a PortalHost with document.body as its anchor element\r\n            this._modalBusyStateHost = new DomPortalOutlet(\r\n                    document.body,\r\n                    this.componentFactoryResolver,\r\n                    this.applicationRef,\r\n                    this.injector);\r\n        }\r\n\r\n        (this._modalBusyStateHost as DomPortalOutlet).attach(this._componentPortal);\r\n\r\n        this._isShowingBusyState = true;\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if the application is currently locked in a modal busy state (caused by an addModalBusyState() call)\r\n     */\r\n    get isShowingBusyState() {\r\n\r\n        return this._isShowingBusyState;\r\n    }\r\n\r\n\r\n    /**\r\n     * Cancel the application busy state and restore it back to normal so user interaction is allowed again\r\n     */\r\n    removeModalBusyState() {\r\n\r\n        if (!this._isEnabled || !this._isShowingBusyState) {\r\n\r\n            return;\r\n        }\r\n\r\n        if (this._componentPortal !== null) {\r\n\r\n            (this._modalBusyStateHost as DomPortalOutlet).detach();\r\n        }\r\n\r\n        this._enableUserInteraction();\r\n\r\n        this._isShowingBusyState = false;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * TODO - adapt from TS version\r\n     */\r\n    addToolTip() {\r\n\r\n        // TODO - adapt from TS version\r\n    }\r\n\r\n\r\n    /**\r\n     * Show a non modal snackbar notification to the user (Only one snack-bar can ever be opened at the same time).\r\n     *\r\n     * Snackbars inform users of a process that an app has performed or will perform. They appear temporarily, towards the bottom or top of the screen.\r\n     * They shouldn’t interrupt the user experience, and they don’t require user input to disappear.\r\n     *\r\n     * @param config A MatSnackBarConfig instance with the customizations we want for this snackbar\r\n     * @param message The message to show on the snackbar\r\n     * @param action If not empty, the text to place on the snackbar confirmation button\r\n     * @param actionCallback A method to execute once the user clicks into the action button.\r\n     *\r\n     * @return void\r\n     */\r\n    addSnackBar(config: MatSnackBarConfig, message: string, action = '', actionCallback: (() => void) | null = null) {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n\r\n        if (this._isShowingSnackBar) {\r\n\r\n            throw new Error('Trying to show a snackbar while another one is still visible');\r\n        }\r\n\r\n        this._isShowingSnackBar = true;\r\n\r\n        const snackBarRef = this.matSnackBar.open(message, action === '' ? undefined : action, config);\r\n\r\n        if (actionCallback !== null) {\r\n\r\n            snackBarRef.onAction().subscribe(() => {\r\n\r\n                actionCallback();\r\n            });\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if the application is currently showing a snackbar or not\r\n     */\r\n    get isShowingSnackBar() {\r\n\r\n        return this._isShowingSnackBar;\r\n    }\r\n\r\n\r\n    /**\r\n     * Force the removal of the snack bar dialog if it exists.\r\n     *\r\n     * If no snackbar is currently visible, this method will do nothing\r\n     */\r\n    removeSnackBar() {\r\n\r\n        if (!this._isEnabled || !this._isShowingSnackBar) {\r\n\r\n            return;\r\n        }\r\n\r\n        this.matSnackBar.dismiss();\r\n\r\n        this._isShowingSnackBar = false;\r\n    }\r\n\r\n\r\n    /**\r\n     * Show a dialog with one or more options that can be used to close it. We can use any of the predefined dialog types that are bundled with\r\n     * this library or extend DialogBaseComponent to create our own custom ones.\r\n     *\r\n     * @param dialogComponentClass A class for a component that extends DialogBaseComponent, which will be the dialog that is shown to the user.\r\n     * @param properties An object containing the different visual and textual options that this dialog allows:\r\n     *            - id: The html unique identifier that the dialog will have once created. If not specified, no id will be explicitly set\r\n     *            - width: 50% by default. Specify the css value for the default dialog width. As the dialog is responsive, the value will be automatically\r\n     *              reduced if the available screen is not enough, and will reach the desired value otherwise. We can set any css unit like pixels, \r\n     *              %, vh, vw, or any other. For example: '400px', '50%', etc.\r\n     *            - maxWidth: Defines the maximum width that the dialog will have regarding the viewport. We can specify it in % or vw, just like is done in\r\n     *              css. By default it is defined as 96vw, which will fit 96% of the viewport on small devices\r\n     *            - height: TODO docs\r\n     *            - maxHeight: TODO docs\r\n     *            - modal: True (default) if selecting an option is mandatory to close the dialog, false if the dialog can be closed\r\n     *              by the user clicking outside it \r\n     *            - texts: A list with strings containing the dialog texts, sorted by importance. When dialog has a title, this should\r\n     *              be placed first, subtitle second and so (Each dialog may accept different number of texts).\r\n     *            - options: A list of strings that will be used as button captions for each one of the accepted dialog options\r\n     *            - viewContainerRef: This is important if we want to propagate providers from a parent component to this dialog. We must specify \r\n\t *              this reference to make sure the same services injected on the parent are available too at the child dialog \r\n     * \r\n     * @param callback A function that will be called after the dialog is closed and will receive a selection object with the numeric index and value for\r\n     *        the option that's been selected by the user. if no option was selected, index will be -1 and value null\r\n     */\r\n    addDialog(dialogComponentClass: Type<DialogBaseComponent>,\r\n              properties: {id?: string,\r\n\t\t\t\t\t\t   width?: string,\r\n                           maxWidth?: string,\r\n                           height?: string,\r\n                           maxHeight?: string,\r\n                           modal?: boolean,\r\n                           texts?: string[],\r\n                           options?: string[],\r\n                           viewContainerRef?: ViewContainerRef}, \r\n              callback: null | ((selection: {index:number, value?: any}) => void) = null) {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n        \r\n        // Set the default values for non specified properties\r\n        properties.modal = properties.modal ?? true;\r\n        properties.texts = properties.texts ?? [];\r\n        properties.options = properties.options ?? [];\r\n\r\n        // Generate a string to uniquely identify this dialog on the list of active dialogs\r\n        // A dialog is considered as unique if the dialog id and texts are exactly the same. We do not take options into consideration\r\n        // as there may be dialogs with a big amount of options available.\r\n        let className = (dialogComponentClass as any).DIALOG_CLASS_NAME;\r\n        \r\n        if(className === ''){\r\n        \r\n            throw new Error(`The static property DIALOG_CLASS_NAME is not defined or is empty for this dialog component (${dialogComponentClass})`);     \r\n        }\r\n        \r\n        const dialogHash = className + properties.texts.join('');\r\n\r\n        // identical dialogs won't be allowed at the same time\r\n        if (this._activeDialogs.includes(dialogHash)) {\r\n\r\n            return;\r\n        }\r\n\r\n        const dialogRef = this.matDialog.open(dialogComponentClass, {\r\n            width: properties.width ?? \"50%\",\r\n            maxWidth: properties.maxWidth ?? \"96vw\",\r\n            disableClose: properties.modal,\r\n            autoFocus: false,\r\n            closeOnNavigation: !properties.modal,\r\n            viewContainerRef: properties.viewContainerRef,\r\n            data: { texts: properties.texts, options: properties.options }\r\n          });      \r\n\t\t\r\n\t\t// Assign the dialog ID only if specifically set on properties\r\n\t\tif(properties.id && properties.id !== undefined){\r\n\t\t\t\r\n\t\t\tdialogRef.id = properties.id;\r\n\t\t}\r\n\t\t\r\n        this._activeDialogs.push(dialogHash);\r\n        this._activeDialogInstances.push(dialogRef);\r\n\r\n        dialogRef.beforeClosed().subscribe((selection:{index:number, value?:any}) => {\r\n\r\n            this._activeDialogs = ArrayUtils.removeElement(this._activeDialogs, dialogHash);\r\n            this._activeDialogInstances = ArrayUtils.removeElement(this._activeDialogInstances, dialogRef);\r\n\r\n            if(!properties.modal && selection === undefined){\r\n            \r\n                selection = { index: -1 };\r\n            \r\n            }else if (!NumericUtils.isInteger(selection.index)) {\r\n\r\n                throw new Error(`closeDialog() expects index to be an integer`);               \r\n            }\r\n\r\n            if (callback !== null) {\r\n\r\n                if(selection.index >= 0 && selection.value === null){\r\n                    \r\n                    selection.value = properties.options![selection.index];\r\n                }\r\n\r\n                (callback as ((selection:{index:number, value?:any}) => void))(selection);\r\n            }\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Show a dialog with a calendar to let the user pick a date.\r\n     *\r\n     * @param properties An object containing the different visual and textual options that this dialog allows:\r\n     *            - id: The html unique identifier that the dialog will have once created. If not specified, no id will be explicitly set\r\n     *            - width: Specify the css value for the default dialog width. As the dialog is responsive, the value will be automatically\r\n     *              reduced if the available screen is not enough, and will reach the desired value otherwise. We can set any css unit like pixels, \r\n     *              %, vh, vw, or any other. For example: '400px', '50%', etc.\r\n     *            - maxWidth: Defines the maximum width that the dialog will have regarding the viewport. We can specify it in % or vw, just like is done in\r\n     *              css. By default it is defined as 96vw, which will fit 96% of the viewport on small devices\r\n     *            - height: TODO docs\r\n     *            - maxHeight: TODO docs\r\n     *            - modal: True (default) if selecting an option is mandatory to close the dialog, false if the dialog can be closed\r\n     *              by the user clicking outside it \r\n     *            - title: An optional dialog title\r\n     *            - viewContainerRef: This is important to propagate providers from a parent component to this dialog. We must specify \r\n\t *              this reference to make sure the same services injected on the parent are available too at the child dialog \r\n     * @param callback A function to be called after the dialog is closed. It will receive a Date() object selected by the user or null if no selection happened\r\n     */\r\n    addDateSelectionDialog(properties: {id?: string,\r\n                                        width?: string,\r\n                                        maxWidth?: string,\r\n                                        height?: string,\r\n                                        maxHeight?: string,\r\n                                        modal?: boolean,\r\n                                        title?: string,\r\n                           \t\t\t\tviewContainerRef: ViewContainerRef},\r\n                           callback: ((selectedDate: null | Date) => void)) {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n        \r\n        this.addDialog(DialogDateSelectionComponent,\r\n            {\r\n                id: properties.id ?? undefined,\r\n                width: properties.width ?? \"50%\",\r\n                maxWidth: properties.maxWidth ?? \"96vw\",\r\n                height: properties.height ?? \"50%\",\r\n                maxHeight: properties.maxHeight ?? \"92vw\",\r\n                modal: properties.modal ?? false,\r\n                texts: [properties.title ?? ''],\r\n                viewContainerRef: properties.viewContainerRef\r\n            },(selection) => {\r\n                \r\n                callback(selection.index === -1 ? null : (selection.value as Date));  \r\n            });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Force the removal of all the dialogs that are currently visible.\r\n     *\r\n     * If no dialogs are currently visible, this method will do nothing\r\n     */\r\n    removeAllDialogs() {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n\r\n        for (const dialogRef of this._activeDialogInstances) {\r\n\r\n            dialogRef.close({index:-1});\r\n        }\r\n        \r\n        this._activeDialogs = [];\r\n        this._activeDialogInstances = [];\r\n    }\r\n\r\n\r\n    /**\r\n     * TODO - translate from TS version\r\n     */\r\n//    addSideNav(){\r\n//\r\n//    }\r\n\r\n\r\n    /**\r\n     * TODO - translate from TS version\r\n     */\r\n//    get isShowingSideNav(){\r\n//\r\n//    }\r\n\r\n\r\n    /**\r\n     * TODO - translate from TS version\r\n     */\r\n//    removeSideNav(){\r\n//\r\n//    }\r\n\r\n    /**\r\n     * Block all the user interactions with the application (keyboard, touch, mouse, ...)\r\n     */\r\n    private _disableUserInteraction() {\r\n\r\n        if (this._documentKeydownUnlisten === null) {\r\n\r\n            this._documentKeydownUnlisten = this._renderer.listen('document', 'keydown', (event) => event.preventDefault());\r\n        }\r\n\r\n        if (this._documentMousedownUnlisten === null) {\r\n\r\n            this._documentMousedownUnlisten = this._renderer.listen('document', 'mousedown', (event) => event.preventDefault());\r\n        }\r\n\r\n        if (this._documentPointerdownUnlisten === null) {\r\n\r\n            this._documentPointerdownUnlisten = this._renderer.listen('document', 'pointerdown', (event) => event.preventDefault());\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Restore the user interactions that were previously disabled with _disableUserInteraction method\r\n     */\r\n    private _enableUserInteraction() {\r\n\r\n        if (this._documentKeydownUnlisten !== null) {\r\n\r\n            this._documentKeydownUnlisten();\r\n            this._documentKeydownUnlisten = null;\r\n        }\r\n\r\n        if (this._documentMousedownUnlisten !== null) {\r\n\r\n            this._documentMousedownUnlisten();\r\n            this._documentMousedownUnlisten = null;\r\n        }\r\n\r\n        if (this._documentPointerdownUnlisten !== null) {\r\n\r\n            this._documentPointerdownUnlisten();\r\n            this._documentMousedownUnlisten = null;\r\n        }\r\n    }\r\n}\r\n"]}
406
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/dialog.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { ArrayUtils, NumericUtils } from 'turbocommons-ts';\r\nimport { Type, Injectable, ComponentFactoryResolver, Injector, ApplicationRef, Renderer2, RendererFactory2, ViewContainerRef } from '@angular/core';\r\nimport { MatDialog, MatDialogRef } from '@angular/material/dialog';\r\nimport { MatSnackBar, MatSnackBarConfig } from '@angular/material/snack-bar';\r\nimport { BusyStateBaseComponent } from '../view/components/busy-state-base/busy-state-base.component';\r\nimport { ComponentPortal, DomPortalOutlet } from '@angular/cdk/portal';\r\nimport { DialogBaseComponent } from '../view/components/dialog-base/dialog-base.component';\r\nimport { DialogDateSelectionComponent } from '../view/components/dialog-date-selection/dialog-date-selection.component';\r\nimport { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\n\r\n\r\n/**\r\n * Manages the application modal and non modal floating elements\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class DialogService extends SingletoneStrictClass {\r\n\r\n\r\n    /**\r\n     * Used to modify the busy state component that is shown by default by the addModalBusyState() method.\r\n     *\r\n     * @see this.addModalBusyState()\r\n     */\r\n    customBusyStateComponentClass: Type<BusyStateBaseComponent> = BusyStateBaseComponent;\r\n\r\n\r\n    /**\r\n     * Check public getter for docs\r\n     */\r\n    private _isEnabled = true;\r\n\r\n\r\n    /**\r\n     * Tells if the main application is currently showing a busy state that blocks all user interaction\r\n     */\r\n    private _isShowingBusyState = false;\r\n\r\n\r\n    /**\r\n     * A reference to the modal busy state component that is initialized only the first time it is called.\r\n     *\r\n     * (To append the busy state dynamically, we use the Portal concept from the angular material library)\r\n     */\r\n    private _componentPortal: ComponentPortal<BusyStateBaseComponent> | null = null;\r\n\r\n\r\n    /**\r\n     * A reference to the modal busy state container where the component will be added\r\n     */\r\n    private _modalBusyStateHost: DomPortalOutlet | null = null;\r\n\r\n\r\n    /**\r\n     * Tells if the manager is currently showing a snackbar element or not\r\n     */\r\n    private _isShowingSnackBar = false;\r\n\r\n\r\n    /**\r\n     * Contains a list of the dialogs that are currently visible to the user.\r\n     * Each item in this list is a hash that is computed when dialog is created to uniquely identify it.\r\n     *\r\n     * Empty list means no dialogs are currently visible\r\n     */\r\n    private _activeDialogs: string[] = [];\r\n\r\n\r\n    /**\r\n     * Contains a list with all the MatDialog instances that are currently visible to the user.\r\n     * The list uses the same order as the list of _activeDialogs hash values\r\n     */\r\n    private _activeDialogInstances: MatDialogRef<DialogBaseComponent>[] = [];\r\n\r\n\r\n    /**\r\n     * Used to store the initialized Renderer 2 instance that is used by this class\r\n     */\r\n    private readonly _renderer: Renderer2;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the window beforeunload event listener once not used anymore\r\n     */\r\n    private _windowBeforeUnloadUnListen: (() => void) | null = null;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the document keydown event listener once not used anymore\r\n     */\r\n    private _documentKeydownUnlisten: (() => void) | null = null;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the document mousedown event listener once not used anymore\r\n     */\r\n    private _documentMousedownUnlisten: (() => void) | null = null;\r\n\r\n\r\n    /**\r\n     * Method that is used to delete the document pointerdown event listener once not used anymore\r\n     */\r\n    private _documentPointerdownUnlisten: (() => void) | null = null;\r\n\r\n\r\n    constructor(rendererFactory: RendererFactory2,\r\n                private readonly matSnackBar: MatSnackBar,\r\n                private readonly matDialog: MatDialog,\r\n                private readonly injector: Injector,\r\n                private readonly applicationRef: ApplicationRef,\r\n                private readonly componentFactoryResolver: ComponentFactoryResolver) {\r\n\r\n\t\tsuper(DialogService);\r\n\r\n        this._renderer = rendererFactory.createRenderer(null, null);\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if this dialog service is enabled or not. If dialog service is disabled, none of it's features will work\r\n     * This is used to block all dialog features normally when shutting down the application\r\n     *\r\n     * Use with caution. When this is set to false, dialog service stops working.\r\n     */\r\n    set isEnabled(v: boolean) {\r\n\r\n        if (v === this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n\r\n        this._isEnabled = v;\r\n    }\r\n\r\n\r\n    /**\r\n     * Enables a warning that will be shown to the user when he/she tries to close the application.\r\n     * This warning will prompt the user to continue with the exit process or cancel it.\r\n     * The specific texts of this message are a generic ones and cannot be changed.\r\n     *\r\n     * IMPORTANT: This method must be called after the main application has been initialized in order to work,\r\n     * otherwise it will do nothing.\r\n     */\r\n    addCloseApplicationWarning() {\r\n\r\n        if (this._windowBeforeUnloadUnListen === null) {\r\n\r\n            this._windowBeforeUnloadUnListen = this._renderer.listen('window', 'beforeunload',\r\n                (event) => event.returnValue = true);\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Remove the close application warning message if previously assigned\r\n     */\r\n    removeCloseApplicationWarning() {\r\n\r\n        if (this._windowBeforeUnloadUnListen !== null) {\r\n\r\n            this._windowBeforeUnloadUnListen();\r\n            this._windowBeforeUnloadUnListen = null;\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Change the application visual appearance so the user perceives that the application is\r\n     * currently busy. While modal busy state is enabled, no user input is possible neither via\r\n     * keyboard, mouse or touch. Use this state when performing server requests or operations that\r\n     * must block the user interaction with the application. To allow user interaction again, you must\r\n     * call removeModalBusyState()\r\n     *\r\n     * Notice: We can modify the busy state visual component that is shown by this method. To do it, we must\r\n     * set this.customBusyStateComponentClass property with our own custom busy state component class. (We can do it at\r\n     * our main application component constructor for example). Our custom component must extend the\r\n     * BusyStateBaseComponent one to add its own visual appearance.\r\n     *\r\n     * @see this.customBusyStateComponentClass\r\n     */\r\n    addModalBusyState() {\r\n\r\n        if (!this._isEnabled || this._isShowingBusyState) {\r\n\r\n            return;\r\n        }\r\n\r\n        this._disableUserInteraction();\r\n\r\n        // Dynamically create the busy state component reference if this is the first time\r\n        if (this._componentPortal === null) {\r\n\r\n            this._componentPortal = new ComponentPortal(this.customBusyStateComponentClass);\r\n\r\n            // Create a PortalHost with document.body as its anchor element\r\n            this._modalBusyStateHost = new DomPortalOutlet(\r\n                    document.body,\r\n                    this.componentFactoryResolver,\r\n                    this.applicationRef,\r\n                    this.injector);\r\n        }\r\n\r\n        (this._modalBusyStateHost as DomPortalOutlet).attach(this._componentPortal);\r\n\r\n        this._isShowingBusyState = true;\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if the application is currently locked in a modal busy state (caused by an addModalBusyState() call)\r\n     */\r\n    get isShowingBusyState() {\r\n\r\n        return this._isShowingBusyState;\r\n    }\r\n\r\n\r\n    /**\r\n     * Cancel the application busy state and restore it back to normal so user interaction is allowed again\r\n     */\r\n    removeModalBusyState() {\r\n\r\n        if (!this._isEnabled || !this._isShowingBusyState) {\r\n\r\n            return;\r\n        }\r\n\r\n        if (this._componentPortal !== null) {\r\n\r\n            (this._modalBusyStateHost as DomPortalOutlet).detach();\r\n        }\r\n\r\n        this._enableUserInteraction();\r\n\r\n        this._isShowingBusyState = false;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * TODO - adapt from TS version\r\n     */\r\n    addToolTip() {\r\n\r\n        // TODO - adapt from TS version\r\n    }\r\n\r\n\r\n    /**\r\n     * Show a non modal snackbar notification to the user (Only one snack-bar can ever be opened at the same time).\r\n     *\r\n     * Snackbars inform users of a process that an app has performed or will perform. They appear temporarily, towards the bottom or top of the screen.\r\n     * They shouldn’t interrupt the user experience, and they don’t require user input to disappear.\r\n     *\r\n     * @param config A MatSnackBarConfig instance with the customizations we want for this snackbar\r\n     * @param message The message to show on the snackbar\r\n     * @param action If not empty, the text to place on the snackbar confirmation button\r\n     * @param actionCallback A method to execute once the user clicks into the action button.\r\n     *\r\n     * @return void\r\n     */\r\n    addSnackBar(config: MatSnackBarConfig, message: string, action = '', actionCallback: (() => void) | null = null) {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n\r\n        if (this._isShowingSnackBar) {\r\n\r\n            throw new Error('Trying to show a snackbar while another one is still visible');\r\n        }\r\n\r\n        this._isShowingSnackBar = true;\r\n\r\n        const snackBarRef = this.matSnackBar.open(message, action === '' ? undefined : action, config);\r\n\r\n        if (actionCallback !== null) {\r\n\r\n            snackBarRef.onAction().subscribe(() => {\r\n\r\n                actionCallback();\r\n            });\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if the application is currently showing a snackbar or not\r\n     */\r\n    get isShowingSnackBar() {\r\n\r\n        return this._isShowingSnackBar;\r\n    }\r\n\r\n\r\n    /**\r\n     * Force the removal of the snack bar dialog if it exists.\r\n     *\r\n     * If no snackbar is currently visible, this method will do nothing\r\n     */\r\n    removeSnackBar() {\r\n\r\n        if (!this._isEnabled || !this._isShowingSnackBar) {\r\n\r\n            return;\r\n        }\r\n\r\n        this.matSnackBar.dismiss();\r\n\r\n        this._isShowingSnackBar = false;\r\n    }\r\n\r\n\r\n    /**\r\n     * Show a dialog with one or more options that can be used to close it. We can use any of the predefined dialog types that are bundled with\r\n     * this library or extend DialogBaseComponent to create our own custom ones.\r\n     *\r\n     * @param dialogComponentClass A class for a component that extends DialogBaseComponent, which will be the dialog that is shown to the user.\r\n     * @param properties An object containing the different visual and textual options that this dialog allows:\r\n     *            - id: The html unique identifier that the dialog will have once created. If not specified, no id will be explicitly set\r\n     *            - width: 50% by default. Specify the css value for the default dialog width. As the dialog is responsive, the value will be automatically\r\n     *              reduced if the available screen is not enough, and will reach the desired value otherwise. We can set any css unit like pixels, \r\n     *              %, vh, vw, or any other. For example: '400px', '50%', etc.\r\n     *            - maxWidth: Defines the maximum width that the dialog will have regarding the viewport. We can specify it in % or vw, just like is done in\r\n     *              css. By default it is defined as 96vw, which will fit 96% of the viewport on small devices\r\n     *            - height: TODO docs\r\n     *            - maxHeight: TODO docs\r\n     *            - modal: True (default) if selecting an option is mandatory to close the dialog, false if the dialog can be closed\r\n     *              by the user clicking outside it \r\n     *            - texts: A list with strings containing the dialog texts, sorted by importance. When dialog has a title, this should\r\n     *              be placed first, subtitle second and so (Each dialog may accept different number of texts).\r\n     *            - options: A list of strings that will be used as button captions for each one of the accepted dialog options\r\n     *            - viewContainerRef: This is important if we want to propagate providers from a parent component to this dialog. We must specify \r\n\t *              this reference to make sure the same services injected on the parent are available too at the child dialog \r\n     * \r\n     * @param callback A function that will be called after the dialog is closed. It will receive a selection object with two properties: index and value. Those \r\n     *        will contain the index and value from the options array that's selected by the user. if no option selected, index will be -1 and value null\r\n     */\r\n    addDialog(dialogComponentClass: Type<DialogBaseComponent>,\r\n              properties: {id?: string,\r\n\t\t\t\t\t\t   width?: string,\r\n                           maxWidth?: string,\r\n                           height?: string,\r\n                           maxHeight?: string,\r\n                           modal?: boolean,\r\n                           texts?: string[],\r\n                           options?: string[],\r\n                           viewContainerRef?: ViewContainerRef}, \r\n              callback: null | ((selection: {index:number, value?: any}) => void) = null) {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n        \r\n        // Set the default values for non specified properties\r\n        properties.modal = properties.modal ?? true;\r\n        properties.texts = properties.texts ?? [];\r\n        properties.options = properties.options ?? [];\r\n\r\n        // Generate a string to uniquely identify this dialog on the list of active dialogs\r\n        // A dialog is considered as unique if the dialog id and texts are exactly the same. We do not take options into consideration\r\n        // as there may be dialogs with a big amount of options available.\r\n        let className = (dialogComponentClass as any).DIALOG_CLASS_NAME;\r\n        \r\n        if(className === ''){\r\n        \r\n            throw new Error(`The static property DIALOG_CLASS_NAME is not defined or is empty for this dialog component (${dialogComponentClass})`);     \r\n        }\r\n        \r\n        const dialogHash = className + properties.texts.join('');\r\n\r\n        // identical dialogs won't be allowed at the same time\r\n        if (this._activeDialogs.includes(dialogHash)) {\r\n\r\n            return;\r\n        }\r\n\r\n        const dialogRef = this.matDialog.open(dialogComponentClass, {\r\n            width: properties.width ?? \"50%\",\r\n            maxWidth: properties.maxWidth ?? \"96vw\",\r\n            disableClose: properties.modal,\r\n            autoFocus: false,\r\n            closeOnNavigation: !properties.modal,\r\n            viewContainerRef: properties.viewContainerRef,\r\n            data: { texts: properties.texts, options: properties.options }\r\n          });      \r\n\t\t\r\n\t\t// Assign the dialog ID only if specifically set on properties\r\n\t\tif(properties.id && properties.id !== undefined){\r\n\t\t\t\r\n\t\t\tdialogRef.id = properties.id;\r\n\t\t}\r\n\t\t\r\n        this._activeDialogs.push(dialogHash);\r\n        this._activeDialogInstances.push(dialogRef);\r\n\r\n        dialogRef.beforeClosed().subscribe((selection:{index:number, value?:any}) => {\r\n\r\n            this._activeDialogs = ArrayUtils.removeElement(this._activeDialogs, dialogHash);\r\n            this._activeDialogInstances = ArrayUtils.removeElement(this._activeDialogInstances, dialogRef);\r\n\r\n            if(!properties.modal && selection === undefined){\r\n            \r\n                selection = { index: -1 };\r\n            \r\n            }else if (!NumericUtils.isInteger(selection.index)) {\r\n\r\n                throw new Error(`closeDialog() expects index to be an integer`);               \r\n            }\r\n\r\n            if (callback !== null) {\r\n\r\n                if(selection.index >= 0 && selection.value === null){\r\n                    \r\n                    selection.value = properties.options![selection.index];\r\n                }\r\n\r\n                (callback as ((selection:{index:number, value?:any}) => void))(selection);\r\n            }\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Show a dialog with a calendar to let the user pick a date.\r\n     *\r\n     * @param properties An object containing the different visual and textual options that this dialog allows:\r\n     *            - id: The html unique identifier that the dialog will have once created. If not specified, no id will be explicitly set\r\n     *            - width: Specify the css value for the default dialog width. As the dialog is responsive, the value will be automatically\r\n     *              reduced if the available screen is not enough, and will reach the desired value otherwise. We can set any css unit like pixels, \r\n     *              %, vh, vw, or any other. For example: '400px', '50%', etc.\r\n     *            - maxWidth: Defines the maximum width that the dialog will have regarding the viewport. We can specify it in % or vw, just like is done in\r\n     *              css. By default it is defined as 96vw, which will fit 96% of the viewport on small devices\r\n     *            - height: TODO docs\r\n     *            - maxHeight: TODO docs\r\n     *            - modal: True (default) if selecting an option is mandatory to close the dialog, false if the dialog can be closed\r\n     *              by the user clicking outside it \r\n     *            - title: An optional dialog title\r\n     *            - viewContainerRef: This is important to propagate providers from a parent component to this dialog. We must specify \r\n\t *              this reference to make sure the same services injected on the parent are available too at the child dialog \r\n     * @param callback A function to be called after the dialog is closed. It will receive a Date() object selected by the user or null if no selection happened\r\n     */\r\n    addDateSelectionDialog(properties: {id?: string,\r\n                                        width?: string,\r\n                                        maxWidth?: string,\r\n                                        height?: string,\r\n                                        maxHeight?: string,\r\n                                        modal?: boolean,\r\n                                        title?: string,\r\n                           \t\t\t\tviewContainerRef: ViewContainerRef},\r\n                           callback: ((selectedDate: null | Date) => void)) {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n        \r\n        this.addDialog(DialogDateSelectionComponent,\r\n            {\r\n                id: properties.id ?? undefined,\r\n                width: properties.width ?? \"50%\",\r\n                maxWidth: properties.maxWidth ?? \"96vw\",\r\n                height: properties.height ?? \"50%\",\r\n                maxHeight: properties.maxHeight ?? \"92vw\",\r\n                modal: properties.modal ?? false,\r\n                texts: [properties.title ?? ''],\r\n                viewContainerRef: properties.viewContainerRef\r\n            },(selection) => {\r\n                \r\n                callback(selection.index === -1 ? null : (selection.value as Date));  \r\n            });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Force the removal of all the dialogs that are currently visible.\r\n     *\r\n     * If no dialogs are currently visible, this method will do nothing\r\n     */\r\n    removeAllDialogs() {\r\n\r\n        if (!this._isEnabled) {\r\n\r\n            return;\r\n        }\r\n\r\n        for (const dialogRef of this._activeDialogInstances) {\r\n\r\n            dialogRef.close({index:-1});\r\n        }\r\n        \r\n        this._activeDialogs = [];\r\n        this._activeDialogInstances = [];\r\n    }\r\n\r\n\r\n    /**\r\n     * TODO - translate from TS version\r\n     */\r\n//    addSideNav(){\r\n//\r\n//    }\r\n\r\n\r\n    /**\r\n     * TODO - translate from TS version\r\n     */\r\n//    get isShowingSideNav(){\r\n//\r\n//    }\r\n\r\n\r\n    /**\r\n     * TODO - translate from TS version\r\n     */\r\n//    removeSideNav(){\r\n//\r\n//    }\r\n\r\n    /**\r\n     * Block all the user interactions with the application (keyboard, touch, mouse, ...)\r\n     */\r\n    private _disableUserInteraction() {\r\n\r\n        if (this._documentKeydownUnlisten === null) {\r\n\r\n            this._documentKeydownUnlisten = this._renderer.listen('document', 'keydown', (event) => event.preventDefault());\r\n        }\r\n\r\n        if (this._documentMousedownUnlisten === null) {\r\n\r\n            this._documentMousedownUnlisten = this._renderer.listen('document', 'mousedown', (event) => event.preventDefault());\r\n        }\r\n\r\n        if (this._documentPointerdownUnlisten === null) {\r\n\r\n            this._documentPointerdownUnlisten = this._renderer.listen('document', 'pointerdown', (event) => event.preventDefault());\r\n        }\r\n    }\r\n\r\n\r\n    /**\r\n     * Restore the user interactions that were previously disabled with _disableUserInteraction method\r\n     */\r\n    private _enableUserInteraction() {\r\n\r\n        if (this._documentKeydownUnlisten !== null) {\r\n\r\n            this._documentKeydownUnlisten();\r\n            this._documentKeydownUnlisten = null;\r\n        }\r\n\r\n        if (this._documentMousedownUnlisten !== null) {\r\n\r\n            this._documentMousedownUnlisten();\r\n            this._documentMousedownUnlisten = null;\r\n        }\r\n\r\n        if (this._documentPointerdownUnlisten !== null) {\r\n\r\n            this._documentPointerdownUnlisten();\r\n            this._documentMousedownUnlisten = null;\r\n        }\r\n    }\r\n}\r\n"]}
@@ -23,6 +23,8 @@ export class TurboApiCallerService extends SingletoneStrictClass {
23
23
  this.browserService = browserService;
24
24
  /**
25
25
  * URI Path to the web service that performs the user login
26
+ * TODO - crear una clase apipaths amb les constants de totes les rutes i moure aquestes 3 alla
27
+ * aquesta classe haura de ser extesa a la nostra app amb les altres que es fagin servir
26
28
  */
27
29
  this.loginServiceURI = 'users/login';
28
30
  /**
@@ -35,6 +37,8 @@ export class TurboApiCallerService extends SingletoneStrictClass {
35
37
  this.mailVerifyServiceURI = 'users/user-mail-verify';
36
38
  /**
37
39
  * The username that is currently defined and will be used by the login methods
40
+ * TODO - We must use a setter and a getter of user and psw. If any of the values change
41
+ * we must clear the logged state and the token.
38
42
  */
39
43
  this.userName = '';
40
44
  /**
@@ -140,10 +144,13 @@ export class TurboApiCallerService extends SingletoneStrictClass {
140
144
  ;
141
145
  }
142
146
  /**
143
- * Authenticates the user by sending an encoded credentials request to the login web service.
147
+ * Authenticates the userName and password that are currently defined at the respective properties of this service.
144
148
  * Returns a promise that resolves with the server's response or rejects with an error if the login fails.
145
149
  * Path to the login service must be correctly defined at this.loginWebService
146
150
  *
151
+ * The authentication process is performed by sending an encoded credentials request to the login web service using the
152
+ * currently defined user name and psw.
153
+ *
147
154
  * Here's an example of a call:
148
155
  *
149
156
  * login().then(response => {
@@ -194,7 +201,8 @@ export class TurboApiCallerService extends SingletoneStrictClass {
194
201
  return !StringUtils.isEmpty(this.userName) && !StringUtils.isEmpty(this.password);
195
202
  }
196
203
  /**
197
- * Tells if exists a user that is currently logged or not
204
+ * Tells if the user name and psw that are specified on this service are currently logged or not. This means
205
+ * also a token is active.
198
206
  */
199
207
  get isLogged() {
200
208
  return this._isLogged;
@@ -389,4 +397,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
389
397
  providedIn: 'root',
390
398
  }]
391
399
  }], ctorParameters: () => [{ type: i1.DialogService }, { type: i2.BrowserService }] });
392
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"turbo-api-caller.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/turbo-api-caller.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { Injectable } from '@angular/core';\r\nimport { DialogService } from './dialog.service';\r\nimport { ConversionUtils, HTTPManager, HTTPManagerPostRequest, StringUtils } from 'turbocommons-ts';\r\nimport { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\r\nimport { DialogErrorComponent } from '../view/components/dialog-error/dialog-error.component';\r\nimport { BrowserService } from './browser.service';\r\n\r\n\r\n/**\r\n * Allows us to easily perform requests to a remote API that is developed with Turbo framework.\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class TurboApiCallerService extends SingletoneStrictClass {\r\n    \r\n\r\n    /**\r\n     * URI Path to the web service that performs the user login\r\n     */\r\n    loginServiceURI = 'users/login';\r\n    \r\n    \r\n    /**\r\n     * URI Path to the web service that performs the user log out\r\n     */\r\n    logOutServiceURI = 'users/logout';\r\n    \r\n    \r\n    /**\r\n     * URI Path to the web service that performs the verification for a user email\r\n     */\r\n    mailVerifyServiceURI = 'users/user-mail-verify';\r\n\r\n\r\n    /**\r\n     * The username that is currently defined and will be used by the login methods\r\n     */\r\n    userName = '';\r\n\r\n\r\n    /**\r\n     * The password for the user that is currently defined and will be used by the login methods\r\n     */\r\n    password = '';\r\n    \r\n    \r\n    /**\r\n     * Contains the last email account that has been verified (or tried to verify) by this service, if any\r\n     */\r\n    private _lastVerifiedMail = '';\r\n\r\n\r\n    /**\r\n     * Check public getter for docs\r\n     */\r\n    private _isLogged = false;\r\n\r\n\r\n    /**\r\n     * @see token() getter for more info\r\n     */\r\n    private _token = '';\r\n    \r\n    \r\n    /**\r\n     * List of operations that are allowed to the currently loged user. It gets filled just after login is performed\r\n     */\r\n    private _operations:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * Private http service instance that will be exclusive to this turbo api caller service. It is a duplicate.\r\n     */\r\n    private httpManager: HTTPManager;\r\n    \r\n    \r\n    constructor(private dialogService: DialogService, \r\n                private browserService: BrowserService) {\r\n        \r\n        super(TurboApiCallerService);\r\n        \r\n        // Create a fresh instance of the http service so we can use it independently\r\n        this.httpManager = new HTTPManager();\r\n        this._clearUserAndToken();\r\n    }\r\n    \r\n\r\n    /** \r\n     * Define the root url that will be used for all the API calls\r\n     */\r\n    set baseUrl(url:string){\r\n        \r\n        this.httpManager.baseUrl = url;\r\n    }\r\n    \r\n    \r\n    /** \r\n     * Define the root url that will be used for all the API calls\r\n     */\r\n    get baseUrl(){\r\n        \r\n        return this.httpManager.baseUrl;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * If the current browser URL contains a hash fragment (The part after the # character) that contains encoded user or password\r\n     * values, this method will parse and automatically set them as the credentials to use.\r\n     * \r\n     * The URL hash is not altered in any way by this method.\r\n     *\r\n     * @returns True if any credentials were found and loaded, false if nothing happened\r\n     */\r\n    loadCredentialsFromURLHashFragment() {\r\n    \r\n        // If the hash fragment is empty, nothing to do\r\n        if(!this.browserService.isCurrentUrlWithHashFragment()){\r\n           \r\n           return false;\r\n        }\r\n       \r\n        let valuesFound = false;\r\n       \r\n        // Split the hash fragment to obtain the different values that contains\r\n        let hashData = this.browserService.getCurrentUrlHashFragment().split('/');\r\n       \r\n        // User name is the first element of the hash fragment. If we found a value, \r\n        // we will fill it automatically\r\n        if(hashData.length > 0){\r\n           \r\n            valuesFound = true;\r\n            this.userName = ConversionUtils.base64ToString(hashData[0]);\r\n        }\r\n       \r\n        // Auto fill the password if it is received\r\n        if(hashData.length > 3){\r\n        \r\n            valuesFound = true;\r\n            this.password = ConversionUtils.base64ToString(hashData[3]);\r\n        }\r\n       \r\n        return valuesFound;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * If the current browser URL contains a hash fragment (The part after the # character) that contains encoded user, mail and\r\n     * has verification code values, this method will perform the request to server to verify that email account for the user.\r\n     * \r\n     * The URI path to the mail verification service must be correctly defined at this.mailVerifyServiceURI property.\r\n     *\r\n     * @returns A promise that resolves with the server response if the verification is successful,\r\n     *          or rejects with an error if the verification fails. Response will have 4 possible values:\r\n     *               undefined - meaning that the URL hash fragment didn't contain valid verification values\r\n     *              -1 - meaning verification failed\r\n     *               0 - meaning verification succeeded\r\n     *               1 - meaning the email was already previouly verified\r\n     */\r\n    verifyUserMailFromURLHashFragment() {\r\n\r\n        // If the hash fragment is empty, nothing to do\r\n        if(this.browserService.isCurrentUrlWithHashFragment()){\r\n                   \r\n            let hashData = this.browserService.getCurrentUrlHashFragment().split('/');\r\n                    \r\n            if(hashData.length >= 3){\r\n                        \r\n                // Call for the user mail verification if user, mail and hash are received\r\n                let user = ConversionUtils.base64ToString(hashData[0]);\r\n                let mail = ConversionUtils.base64ToString(hashData[1]);\r\n                let hash = ConversionUtils.base64ToString(hashData[2]);\r\n                        \r\n                if(!StringUtils.isEmpty(user) && !StringUtils.isEmpty(mail) && !StringUtils.isEmpty(hash)){\r\n                    \r\n                    this._lastVerifiedMail = mail;\r\n                    \r\n                    return this.call(this.mailVerifyServiceURI, {userName: user, mail: mail, hash: hash});\r\n                }\r\n            }\r\n        }\r\n        \r\n        return Promise.resolve(undefined);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Obtain the lat user mail account that has been verified (or attempted to verify) by this service.\r\n     */\r\n    getLastVerifiedMail(){\r\n\r\n        return this._lastVerifiedMail;;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Authenticates the user by sending an encoded credentials request to the login web service.\r\n     * Returns a promise that resolves with the server's response or rejects with an error if the login fails.\r\n     * Path to the login service must be correctly defined at this.loginWebService\r\n     *\r\n     * Here's an example of a call:\r\n     * \r\n     *  login().then(response => {\r\n     *     console.log('Login successful:', response);\r\n     *  }).catch(() => {\r\n     *     console.error('Login failed:');\r\n     *  });\r\n     * \r\n     * @returns A promise that resolves with the server response if the login is successful,\r\n     *          or rejects with an error if the login fails.\r\n     */\r\n    login() {\r\n        \r\n        this.dialogService.addModalBusyState(); \r\n        \r\n        return new Promise((resolve:(r:any) => any, reject) => {\r\n        \r\n            const request = new HTTPManagerPostRequest(this.loginServiceURI);\r\n                    \r\n            request.ignoreGlobalPostParams = true;\r\n               \r\n            const encodedCredentials = ConversionUtils.stringToBase64(\r\n                ConversionUtils.stringToBase64(this.userName) + ',' + ConversionUtils.stringToBase64(this.password));\r\n    \r\n            request.parameters = { data: encodedCredentials };\r\n            \r\n            request.successCallback = (response) => {\r\n            \r\n                if (response !== '') {\r\n        \r\n                    response = JSON.parse(response);            \r\n        \r\n                    this._isLogged = true;\r\n                    this._token = response.token;\r\n                    this._operations = response.operations;\r\n                    this.httpManager.setGlobalPostParam('token', response.token);\r\n        \r\n                    resolve(response);\r\n                    \r\n                } else {\r\n        \r\n                    this._clearUserAndToken();\r\n        \r\n                    reject(new Error());\r\n                }\r\n            };\r\n            \r\n            request.errorCallback = () => {\r\n\r\n                this._clearUserAndToken();\r\n                \r\n                reject(new Error());               \r\n            };\r\n            \r\n            request.finallyCallback = () => {\r\n               \r\n               this.dialogService.removeModalBusyState();\r\n            };\r\n        \r\n            this.httpManager.execute(request);\r\n        }); \r\n    }\r\n    \r\n\r\n    /**\r\n     * Checks if the user and password credentials are filled and not empty.\r\n     * If any of the two values is empty, false will be returned\r\n     */\r\n    get isUserAndPswDefined(): boolean  {\r\n\r\n        return !StringUtils.isEmpty(this.userName) && !StringUtils.isEmpty(this.password);\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if exists a user that is currently logged or not\r\n     */\r\n    get isLogged(): boolean {\r\n\r\n        return this._isLogged;\r\n    }\r\n\r\n\r\n    /**\r\n     * Gives the value for the currently active user authentication token or an empty string if no user logged\r\n     */\r\n    get token() {\r\n\r\n        return this._token;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Checks if the currently logged user is allowed to perform the specified operation.\r\n     * This will check the user permisions as defined on server side and return true if the operation is allowed\r\n     * or false if not. We can then decide what to do (hide an app section, block a button, etc)\r\n     * \r\n     * @param operation The name of the operation to check \r\n     * \r\n     * @returns True if the operation is allowed, false otherwise \r\n     */\r\n    isUserAllowedTo(operation:string): boolean {\r\n    \r\n        return this._operations.includes(operation);\r\n    }\r\n        \r\n    \r\n    /**\r\n     * Performs a standard request to an API service and returns a promise that resolves with the response data.\r\n     * \r\n     * Following is an example of a call:\r\n     * \r\n     *     this.apiService.call('users/user-create', this.userData, {handleErrors: false}).then(response => {\r\n     *          \r\n     *          console.log('Success:', response);\r\n     *          \r\n     *      }).catch(error => {\r\n     *          \r\n     *          console.error('Error:', error.message);\r\n     *      });\r\n     *\r\n     * @param apiPath - A relative URL (based on the defined base root url) that defines the path to the service to call.\r\n     *        For example: 'users/login'\r\n     * @param parameters - An object containing key-value pairs that are sent as the request body.\r\n     *        token parameter is not necessary, it is automatically appended\r\n     * @param options An object defining several options for the request:\r\n     *        resultFormat: 'JSON' by default. The expected format of the response. \r\n     *        busyState: Enabled by default. Enables or disables the busy state to lock user interaction while performing the http calls.\r\n     *        handleErrors: Enabled by default. If set to true, an error dialog will be automatically shown when the call fails.\r\n     *        If set to false, the promise will generate a reject error that must be handled by our code.\r\n     *       \r\n     * @returns A promise that resolves with the response data correctly parsed if the request is successful, or rejects \r\n     *          with an error (containing all the error response details) if the request fails.\r\n     */\r\n    call(apiPath:string, parameters = {}, options: {resultFormat?:'STRING'|'JSON', busyState?:boolean, handleErrors?:boolean} = {}){\r\n    \r\n        // Set the default values for non specified properties\r\n        options.resultFormat = options.resultFormat ?? 'JSON';\r\n        options.busyState = options.busyState ?? true;\r\n        options.handleErrors = options.handleErrors ?? true;\r\n        \r\n        if(options.busyState){\r\n           \r\n           this.dialogService.addModalBusyState(); \r\n        }\r\n        \r\n        return new Promise((resolve:(r:any) => any, reject) => {\r\n            \r\n            // Create the request object\r\n            const request = new HTTPManagerPostRequest(\r\n                apiPath, \r\n                options.resultFormat === 'STRING' ? HTTPManagerPostRequest.STRING : HTTPManagerPostRequest.JSON\r\n            );\r\n\r\n            request.parameters = parameters;\r\n\r\n            request.successCallback = (response: any) => {\r\n                \r\n                resolve(response);\r\n            };\r\n\r\n            request.errorCallback = (errorMsg, errorCode, response) => {\r\n\r\n                if(options.handleErrors){\r\n                \r\n                    this.showErrorDialog(errorMsg, errorCode, response);\r\n                \r\n                }else{\r\n                    \r\n                    reject(new Error(errorMsg + ' ' + errorCode + ' ' + response));\r\n                }                \r\n            };\r\n            \r\n            request.finallyCallback = () => {\r\n                \r\n                if(options.busyState){\r\n                                \r\n                    this.dialogService.removeModalBusyState();\r\n                }\r\n            };\r\n            \r\n            this.httpManager.execute(request);\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Performs a request to chain several api calls into a single http request, via the chain services mechanism of the turboframework API.\r\n     * Returns a promise that resolves with an array of response objects. One for each of the request calls.\r\n     *\r\n     * @param apiPath - A relative URL (based on the defined base root url) that defines the path to the root of the chain services call.\r\n     *        For example: 'turbosite/chain/chain-services'\r\n     * @param services - An array of objects, were each object contains a valid structure for the chain services call. \r\n     *        Token parameter is not necessary, it is automatically appended\r\n     * @param options An object defining several options for the request:\r\n     *        busyState: Enables or disables the busy state to lock user interaction while performing the http calls. Enabled by default\r\n     *        handleErrors: Enabled by default. If set to true, an error dialog will be automatically shown when the call fails.\r\n     *        If set to false, the promise will generate a reject error that must be handled by our code.\r\n     * \r\n     * @returns A promise that resolves with the response data correctly parsed if the request is successful, or rejects \r\n     *          with an error (containing all the error response details) if the request fails.\r\n     */\r\n    callChain(apiPath:string, services:{}[], options: {busyState?:boolean, handleErrors?:boolean} = {}){\r\n    \r\n        // Set the default values for non specified properties\r\n        options.busyState = options.busyState ?? true;\r\n        options.handleErrors = options.handleErrors ?? true;\r\n        \r\n        if(options.busyState){\r\n           \r\n           this.dialogService.addModalBusyState(); \r\n        }\r\n                \r\n        return new Promise((resolve:(r:any) => any, reject) => {\r\n            \r\n            const request = new HTTPManagerPostRequest(apiPath, HTTPManagerPostRequest.JSON);\r\n                        \r\n            request.ignoreGlobalPostParams = true;      \r\n\r\n            request.parameters = { services: services };\r\n            \r\n            request.successCallback = (response:any[]) => {\r\n            \r\n                resolve(response);\r\n            };      \r\n\r\n            request.errorCallback = (errorMsg, errorCode, response) => {\r\n\r\n                if(options.handleErrors){\r\n                \r\n                    this.showErrorDialog(errorMsg, errorCode, response);    \r\n                \r\n                }else{\r\n                    \r\n                    reject(new Error(errorMsg + ' ' + errorCode + ' ' + response));\r\n                }                \r\n            };\r\n                        \r\n            request.finallyCallback = () => {\r\n                \r\n                if(options.busyState){\r\n                                \r\n                    this.dialogService.removeModalBusyState();\r\n                }\r\n            };\r\n            \r\n            this.httpManager.execute(request);\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to show an error dialog when a request fails and error handling is enabled\r\n     */\r\n    private showErrorDialog(errorMsg:string, errorCode:number, response:any){\r\n        \r\n        errorMsg = errorMsg + '\\n\\n' + response;\r\n                                                    \r\n        if(StringUtils.isEmpty(errorMsg)){\r\n        \r\n            errorMsg = 'Unknown error. Make sure Internet connection is available';    \r\n        }\r\n\r\n        this.dialogService.addDialog(DialogErrorComponent,\r\n            {\r\n                width:'50vw',\r\n                texts: ['Error: ' + errorCode, errorMsg],\r\n                options: ['Ok']\r\n            }); \r\n    }\r\n    \r\n    \r\n    /**\r\n     * Perform the logout for the currently logged user\r\n     *\r\n     * @param options An object defining several options for the request:\r\n     *        handleErrors: Enabled by default. If set to true, an error dialog will be automatically shown when the call fails.\r\n     *        If set to false, the promise will generate a reject error that must be handled by our code.\r\n     * \r\n     * @returns A promise that resolves correctly if the logout is correct, or rejects with an error (containing all the error \r\n     *          response details) if the request fails.\r\n     */\r\n    logout(options: {handleErrors?:boolean} = {}) {\r\n        \r\n        // Set the default values for non specified properties\r\n        options.handleErrors = options.handleErrors ?? true;\r\n                \r\n        this.dialogService.addModalBusyState(); \r\n        \r\n        return new Promise((resolve, reject) => {\r\n                \r\n            const request = new HTTPManagerPostRequest(this.logOutServiceURI);\r\n                    \r\n            request.parameters = { token: this._token };\r\n            \r\n            request.successCallback = () => {\r\n            \r\n                this._clearUserAndToken();\r\n                \r\n                resolve(undefined);\r\n            };\r\n            \r\n            request.errorCallback = (errorMsg, errorCode, response) => {\r\n                \r\n                if(options.handleErrors){\r\n                                \r\n                    this.showErrorDialog(errorMsg, errorCode, response);    \r\n                \r\n                }else{\r\n                    \r\n                    reject(new Error(errorMsg + ' ' + errorCode + ' ' + response));\r\n                }               \r\n            };\r\n            \r\n            request.finallyCallback = () => {\r\n               \r\n               this.dialogService.removeModalBusyState();\r\n            };\r\n        \r\n            this.httpManager.execute(request);\r\n        }); \r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux methot to clear all the currently logged user data\r\n     */\r\n    private _clearUserAndToken(){\r\n        \r\n        this.userName = '';\r\n        this.password = '';\r\n        this._isLogged = false;\r\n        this._token = '';\r\n        this.httpManager.setGlobalPostParam('token', '-'); \r\n    }\r\n}\r\n"]}
400
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"turbo-api-caller.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/turbo-api-caller.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { Injectable } from '@angular/core';\r\nimport { DialogService } from './dialog.service';\r\nimport { ConversionUtils, HTTPManager, HTTPManagerPostRequest, StringUtils } from 'turbocommons-ts';\r\nimport { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\r\nimport { DialogErrorComponent } from '../view/components/dialog-error/dialog-error.component';\r\nimport { BrowserService } from './browser.service';\r\n\r\n\r\n/**\r\n * Allows us to easily perform requests to a remote API that is developed with Turbo framework.\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class TurboApiCallerService extends SingletoneStrictClass {\r\n    \r\n\r\n    /**\r\n     * URI Path to the web service that performs the user login\r\n     * TODO - crear una clase apipaths amb les constants de totes les rutes i moure aquestes 3 alla\r\n     * aquesta classe haura de ser extesa a la nostra app amb les altres que es fagin servir\r\n     */\r\n    loginServiceURI = 'users/login';\r\n    \r\n    \r\n    /**\r\n     * URI Path to the web service that performs the user log out\r\n     */\r\n    logOutServiceURI = 'users/logout';\r\n    \r\n    \r\n    /**\r\n     * URI Path to the web service that performs the verification for a user email\r\n     */\r\n    mailVerifyServiceURI = 'users/user-mail-verify';\r\n\r\n\r\n    /**\r\n     * The username that is currently defined and will be used by the login methods\r\n     * TODO - We must use a setter and a getter of user and psw. If any of the values change\r\n     * we must clear the logged state and the token. \r\n     */\r\n    userName = '';\r\n\r\n\r\n    /**\r\n     * The password for the user that is currently defined and will be used by the login methods\r\n     */\r\n    password = '';\r\n    \r\n    \r\n    /**\r\n     * Contains the last email account that has been verified (or tried to verify) by this service, if any\r\n     */\r\n    private _lastVerifiedMail = '';\r\n\r\n\r\n    /**\r\n     * Check public getter for docs\r\n     */\r\n    private _isLogged = false;\r\n\r\n\r\n    /**\r\n     * @see token() getter for more info\r\n     */\r\n    private _token = '';\r\n    \r\n    \r\n    /**\r\n     * List of operations that are allowed to the currently loged user. It gets filled just after login is performed\r\n     */\r\n    private _operations:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * Private http service instance that will be exclusive to this turbo api caller service. It is a duplicate.\r\n     */\r\n    private httpManager: HTTPManager;\r\n    \r\n    \r\n    constructor(private dialogService: DialogService, \r\n                private browserService: BrowserService) {\r\n        \r\n        super(TurboApiCallerService);\r\n        \r\n        // Create a fresh instance of the http service so we can use it independently\r\n        this.httpManager = new HTTPManager();\r\n        this._clearUserAndToken();\r\n    }\r\n    \r\n\r\n    /** \r\n     * Define the root url that will be used for all the API calls\r\n     */\r\n    set baseUrl(url:string){\r\n        \r\n        this.httpManager.baseUrl = url;\r\n    }\r\n    \r\n    \r\n    /** \r\n     * Define the root url that will be used for all the API calls\r\n     */\r\n    get baseUrl(){\r\n        \r\n        return this.httpManager.baseUrl;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * If the current browser URL contains a hash fragment (The part after the # character) that contains encoded user or password\r\n     * values, this method will parse and automatically set them as the credentials to use.\r\n     * \r\n     * The URL hash is not altered in any way by this method.\r\n     *\r\n     * @returns True if any credentials were found and loaded, false if nothing happened\r\n     */\r\n    loadCredentialsFromURLHashFragment() {\r\n    \r\n        // If the hash fragment is empty, nothing to do\r\n        if(!this.browserService.isCurrentUrlWithHashFragment()){\r\n           \r\n           return false;\r\n        }\r\n       \r\n        let valuesFound = false;\r\n       \r\n        // Split the hash fragment to obtain the different values that contains\r\n        let hashData = this.browserService.getCurrentUrlHashFragment().split('/');\r\n       \r\n        // User name is the first element of the hash fragment. If we found a value, \r\n        // we will fill it automatically\r\n        if(hashData.length > 0){\r\n           \r\n            valuesFound = true;\r\n            this.userName = ConversionUtils.base64ToString(hashData[0]);\r\n        }\r\n       \r\n        // Auto fill the password if it is received\r\n        if(hashData.length > 3){\r\n        \r\n            valuesFound = true;\r\n            this.password = ConversionUtils.base64ToString(hashData[3]);\r\n        }\r\n       \r\n        return valuesFound;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * If the current browser URL contains a hash fragment (The part after the # character) that contains encoded user, mail and\r\n     * has verification code values, this method will perform the request to server to verify that email account for the user.\r\n     * \r\n     * The URI path to the mail verification service must be correctly defined at this.mailVerifyServiceURI property.\r\n     *\r\n     * @returns A promise that resolves with the server response if the verification is successful,\r\n     *          or rejects with an error if the verification fails. Response will have 4 possible values:\r\n     *               undefined - meaning that the URL hash fragment didn't contain valid verification values\r\n     *              -1 - meaning verification failed\r\n     *               0 - meaning verification succeeded\r\n     *               1 - meaning the email was already previouly verified\r\n     */\r\n    verifyUserMailFromURLHashFragment() {\r\n\r\n        // If the hash fragment is empty, nothing to do\r\n        if(this.browserService.isCurrentUrlWithHashFragment()){\r\n                   \r\n            let hashData = this.browserService.getCurrentUrlHashFragment().split('/');\r\n                    \r\n            if(hashData.length >= 3){\r\n                        \r\n                // Call for the user mail verification if user, mail and hash are received\r\n                let user = ConversionUtils.base64ToString(hashData[0]);\r\n                let mail = ConversionUtils.base64ToString(hashData[1]);\r\n                let hash = ConversionUtils.base64ToString(hashData[2]);\r\n                        \r\n                if(!StringUtils.isEmpty(user) && !StringUtils.isEmpty(mail) && !StringUtils.isEmpty(hash)){\r\n                    \r\n                    this._lastVerifiedMail = mail;\r\n                    \r\n                    return this.call(this.mailVerifyServiceURI, {userName: user, mail: mail, hash: hash});\r\n                }\r\n            }\r\n        }\r\n        \r\n        return Promise.resolve(undefined);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Obtain the lat user mail account that has been verified (or attempted to verify) by this service.\r\n     */\r\n    getLastVerifiedMail(){\r\n\r\n        return this._lastVerifiedMail;;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Authenticates the userName and password that are currently defined at the respective properties of this service.\r\n     * Returns a promise that resolves with the server's response or rejects with an error if the login fails.\r\n     * Path to the login service must be correctly defined at this.loginWebService\r\n     * \r\n     * The authentication process is performed by sending an encoded credentials request to the login web service using the \r\n     * currently defined user name and psw.\r\n     *\r\n     * Here's an example of a call:\r\n     * \r\n     *  login().then(response => {\r\n     *     console.log('Login successful:', response);\r\n     *  }).catch(() => {\r\n     *     console.error('Login failed:');\r\n     *  });\r\n     * \r\n     * @returns A promise that resolves with the server response if the login is successful,\r\n     *          or rejects with an error if the login fails.\r\n     */\r\n    login() {\r\n        \r\n        this.dialogService.addModalBusyState(); \r\n        \r\n        return new Promise((resolve:(r:any) => any, reject) => {\r\n        \r\n            const request = new HTTPManagerPostRequest(this.loginServiceURI);\r\n                    \r\n            request.ignoreGlobalPostParams = true;\r\n               \r\n            const encodedCredentials = ConversionUtils.stringToBase64(\r\n                ConversionUtils.stringToBase64(this.userName) + ',' + ConversionUtils.stringToBase64(this.password));\r\n    \r\n            request.parameters = { data: encodedCredentials };\r\n            \r\n            request.successCallback = (response) => {\r\n            \r\n                if (response !== '') {\r\n        \r\n                    response = JSON.parse(response);            \r\n        \r\n                    this._isLogged = true;\r\n                    this._token = response.token;\r\n                    this._operations = response.operations;\r\n                    this.httpManager.setGlobalPostParam('token', response.token);\r\n        \r\n                    resolve(response);\r\n                    \r\n                } else {\r\n        \r\n                    this._clearUserAndToken();\r\n        \r\n                    reject(new Error());\r\n                }\r\n            };\r\n            \r\n            request.errorCallback = () => {\r\n\r\n                this._clearUserAndToken();\r\n                \r\n                reject(new Error());               \r\n            };\r\n            \r\n            request.finallyCallback = () => {\r\n               \r\n               this.dialogService.removeModalBusyState();\r\n            };\r\n        \r\n            this.httpManager.execute(request);\r\n        }); \r\n    }\r\n    \r\n\r\n    /**\r\n     * Checks if the user and password credentials are filled and not empty.\r\n     * If any of the two values is empty, false will be returned\r\n     */\r\n    get isUserAndPswDefined(): boolean  {\r\n\r\n        return !StringUtils.isEmpty(this.userName) && !StringUtils.isEmpty(this.password);\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if the user name and psw that are specified on this service are currently logged or not. This means\r\n     * also a token is active.\r\n     */\r\n    get isLogged(): boolean {\r\n\r\n        return this._isLogged;\r\n    }\r\n\r\n\r\n    /**\r\n     * Gives the value for the currently active user authentication token or an empty string if no user logged\r\n     */\r\n    get token() {\r\n\r\n        return this._token;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Checks if the currently logged user is allowed to perform the specified operation.\r\n     * This will check the user permisions as defined on server side and return true if the operation is allowed\r\n     * or false if not. We can then decide what to do (hide an app section, block a button, etc)\r\n     * \r\n     * @param operation The name of the operation to check \r\n     * \r\n     * @returns True if the operation is allowed, false otherwise \r\n     */\r\n    isUserAllowedTo(operation:string): boolean {\r\n    \r\n        return this._operations.includes(operation);\r\n    }\r\n        \r\n    \r\n    /**\r\n     * Performs a standard request to an API service and returns a promise that resolves with the response data.\r\n     * \r\n     * Following is an example of a call:\r\n     * \r\n     *     this.apiService.call('users/user-create', this.userData, {handleErrors: false}).then(response => {\r\n     *          \r\n     *          console.log('Success:', response);\r\n     *          \r\n     *      }).catch(error => {\r\n     *          \r\n     *          console.error('Error:', error.message);\r\n     *      });\r\n     *\r\n     * @param apiPath - A relative URL (based on the defined base root url) that defines the path to the service to call.\r\n     *        For example: 'users/login'\r\n     * @param parameters - An object containing key-value pairs that are sent as the request body.\r\n     *        token parameter is not necessary, it is automatically appended\r\n     * @param options An object defining several options for the request:\r\n     *        resultFormat: 'JSON' by default. The expected format of the response. \r\n     *        busyState: Enabled by default. Enables or disables the busy state to lock user interaction while performing the http calls.\r\n     *        handleErrors: Enabled by default. If set to true, an error dialog will be automatically shown when the call fails.\r\n     *        If set to false, the promise will generate a reject error that must be handled by our code.\r\n     *       \r\n     * @returns A promise that resolves with the response data correctly parsed if the request is successful, or rejects \r\n     *          with an error (containing all the error response details) if the request fails.\r\n     */\r\n    call(apiPath:string, parameters = {}, options: {resultFormat?:'STRING'|'JSON', busyState?:boolean, handleErrors?:boolean} = {}){\r\n    \r\n        // Set the default values for non specified properties\r\n        options.resultFormat = options.resultFormat ?? 'JSON';\r\n        options.busyState = options.busyState ?? true;\r\n        options.handleErrors = options.handleErrors ?? true;\r\n        \r\n        if(options.busyState){\r\n           \r\n           this.dialogService.addModalBusyState(); \r\n        }\r\n        \r\n        return new Promise((resolve:(r:any) => any, reject) => {\r\n            \r\n            // Create the request object\r\n            const request = new HTTPManagerPostRequest(\r\n                apiPath, \r\n                options.resultFormat === 'STRING' ? HTTPManagerPostRequest.STRING : HTTPManagerPostRequest.JSON\r\n            );\r\n\r\n            request.parameters = parameters;\r\n\r\n            request.successCallback = (response: any) => {\r\n                \r\n                resolve(response);\r\n            };\r\n\r\n            request.errorCallback = (errorMsg, errorCode, response) => {\r\n\r\n                if(options.handleErrors){\r\n                \r\n                    this.showErrorDialog(errorMsg, errorCode, response);\r\n                \r\n                }else{\r\n                    \r\n                    reject(new Error(errorMsg + ' ' + errorCode + ' ' + response));\r\n                }                \r\n            };\r\n            \r\n            request.finallyCallback = () => {\r\n                \r\n                if(options.busyState){\r\n                                \r\n                    this.dialogService.removeModalBusyState();\r\n                }\r\n            };\r\n            \r\n            this.httpManager.execute(request);\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Performs a request to chain several api calls into a single http request, via the chain services mechanism of the turboframework API.\r\n     * Returns a promise that resolves with an array of response objects. One for each of the request calls.\r\n     *\r\n     * @param apiPath - A relative URL (based on the defined base root url) that defines the path to the root of the chain services call.\r\n     *        For example: 'turbosite/chain/chain-services'\r\n     * @param services - An array of objects, were each object contains a valid structure for the chain services call. \r\n     *        Token parameter is not necessary, it is automatically appended\r\n     * @param options An object defining several options for the request:\r\n     *        busyState: Enables or disables the busy state to lock user interaction while performing the http calls. Enabled by default\r\n     *        handleErrors: Enabled by default. If set to true, an error dialog will be automatically shown when the call fails.\r\n     *        If set to false, the promise will generate a reject error that must be handled by our code.\r\n     * \r\n     * @returns A promise that resolves with the response data correctly parsed if the request is successful, or rejects \r\n     *          with an error (containing all the error response details) if the request fails.\r\n     */\r\n    callChain(apiPath:string, services:{}[], options: {busyState?:boolean, handleErrors?:boolean} = {}){\r\n    \r\n        // Set the default values for non specified properties\r\n        options.busyState = options.busyState ?? true;\r\n        options.handleErrors = options.handleErrors ?? true;\r\n        \r\n        if(options.busyState){\r\n           \r\n           this.dialogService.addModalBusyState(); \r\n        }\r\n                \r\n        return new Promise((resolve:(r:any) => any, reject) => {\r\n            \r\n            const request = new HTTPManagerPostRequest(apiPath, HTTPManagerPostRequest.JSON);\r\n                        \r\n            request.ignoreGlobalPostParams = true;      \r\n\r\n            request.parameters = { services: services };\r\n            \r\n            request.successCallback = (response:any[]) => {\r\n            \r\n                resolve(response);\r\n            };      \r\n\r\n            request.errorCallback = (errorMsg, errorCode, response) => {\r\n\r\n                if(options.handleErrors){\r\n                \r\n                    this.showErrorDialog(errorMsg, errorCode, response);    \r\n                \r\n                }else{\r\n                    \r\n                    reject(new Error(errorMsg + ' ' + errorCode + ' ' + response));\r\n                }                \r\n            };\r\n                        \r\n            request.finallyCallback = () => {\r\n                \r\n                if(options.busyState){\r\n                                \r\n                    this.dialogService.removeModalBusyState();\r\n                }\r\n            };\r\n            \r\n            this.httpManager.execute(request);\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to show an error dialog when a request fails and error handling is enabled\r\n     */\r\n    private showErrorDialog(errorMsg:string, errorCode:number, response:any){\r\n        \r\n        errorMsg = errorMsg + '\\n\\n' + response;\r\n                                                    \r\n        if(StringUtils.isEmpty(errorMsg)){\r\n        \r\n            errorMsg = 'Unknown error. Make sure Internet connection is available';    \r\n        }\r\n\r\n        this.dialogService.addDialog(DialogErrorComponent,\r\n            {\r\n                width:'50vw',\r\n                texts: ['Error: ' + errorCode, errorMsg],\r\n                options: ['Ok']\r\n            }); \r\n    }\r\n    \r\n    \r\n    /**\r\n     * Perform the logout for the currently logged user\r\n     *\r\n     * @param options An object defining several options for the request:\r\n     *        handleErrors: Enabled by default. If set to true, an error dialog will be automatically shown when the call fails.\r\n     *        If set to false, the promise will generate a reject error that must be handled by our code.\r\n     * \r\n     * @returns A promise that resolves correctly if the logout is correct, or rejects with an error (containing all the error \r\n     *          response details) if the request fails.\r\n     */\r\n    logout(options: {handleErrors?:boolean} = {}) {\r\n        \r\n        // Set the default values for non specified properties\r\n        options.handleErrors = options.handleErrors ?? true;\r\n                \r\n        this.dialogService.addModalBusyState(); \r\n        \r\n        return new Promise((resolve, reject) => {\r\n                \r\n            const request = new HTTPManagerPostRequest(this.logOutServiceURI);\r\n                    \r\n            request.parameters = { token: this._token };\r\n            \r\n            request.successCallback = () => {\r\n            \r\n                this._clearUserAndToken();\r\n                \r\n                resolve(undefined);\r\n            };\r\n            \r\n            request.errorCallback = (errorMsg, errorCode, response) => {\r\n                \r\n                if(options.handleErrors){\r\n                                \r\n                    this.showErrorDialog(errorMsg, errorCode, response);    \r\n                \r\n                }else{\r\n                    \r\n                    reject(new Error(errorMsg + ' ' + errorCode + ' ' + response));\r\n                }               \r\n            };\r\n            \r\n            request.finallyCallback = () => {\r\n               \r\n               this.dialogService.removeModalBusyState();\r\n            };\r\n        \r\n            this.httpManager.execute(request);\r\n        }); \r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux methot to clear all the currently logged user data\r\n     */\r\n    private _clearUserAndToken(){\r\n        \r\n        this.userName = '';\r\n        this.password = '';\r\n        this._isLogged = false;\r\n        this._token = '';\r\n        this.httpManager.setGlobalPostParam('token', '-'); \r\n    }\r\n}\r\n"]}
@@ -30,6 +30,10 @@ export class ViewsService extends SingletoneStrictClass {
30
30
  * Contains the reference to the currently loaded view component
31
31
  */
32
32
  this._currentComponentRef = null;
33
+ /**
34
+ * Flag that stores if any view is in the process of being loaded
35
+ */
36
+ this._isLoadingView = false;
33
37
  }
34
38
  /**
35
39
  * Tells if there's any view currently loaded on the application views container
@@ -51,20 +55,25 @@ export class ViewsService extends SingletoneStrictClass {
51
55
  }
52
56
  /**
53
57
  * Create a new view instance with the specified type and add it to the current application views container. Any currently loaded
54
- * view will be removed
58
+ * view will be removed.
55
59
  *
56
- * Make sure this method is called when all the visual components of the application have been created (ngAfterViewInit)
60
+ * If we push a view while another one is in the process of being loaded, the new push will be ignored.
57
61
  *
58
- * @param view The classname for the view that we want to create and add to the views container (must extend View base class).
62
+ * Make sure this method is called when all the visual components of the application have been created (ngAfterViewInit)
59
63
  *
60
- * @return The instance of the newly added and created view.
64
+ * @param view The classname for the view that we want to create (must extend View base class). A new angular object
65
+ * will be instantiated and loaded into the views container.
61
66
  */
62
67
  async pushView(view) {
63
- this.verifyViewsContainerExist();
64
68
  // If the loaded view is the same as the specified one, we will do nothing
65
69
  if (this._loadedViewClass === view) {
66
- return this._currentComponentRef;
70
+ return;
67
71
  }
72
+ // If a view is already being loaded, nothing to do
73
+ if (this._isLoadingView) {
74
+ return;
75
+ }
76
+ this._isLoadingView = true;
68
77
  // If a view is already loaded, we will unload it first
69
78
  if (this._loadedViewClass !== null) {
70
79
  await this.removeCurrentView();
@@ -83,18 +92,17 @@ export class ViewsService extends SingletoneStrictClass {
83
92
  fadeInPlayer.onDone(() => {
84
93
  this._currentComponentRef = newComponentRef;
85
94
  this._loadedViewClass = view;
95
+ this._isLoadingView = false;
86
96
  resolve();
87
97
  });
88
98
  fadeInPlayer.play();
89
99
  });
90
- return this._currentComponentRef;
91
100
  }
92
101
  /**
93
102
  * Delete the currently loaded view from the views container, and leave it empty.
94
103
  * If no view is currently loaded, this method will do nothing
95
104
  */
96
105
  async popView() {
97
- this.verifyViewsContainerExist();
98
106
  if (this._loadedViewClass !== null && this._currentComponentRef) {
99
107
  await this.removeCurrentView();
100
108
  }
@@ -103,6 +111,9 @@ export class ViewsService extends SingletoneStrictClass {
103
111
  * Aux method to remove the currently loaded view from the views container using a promise
104
112
  */
105
113
  removeCurrentView() {
114
+ if (this._viewContainerRef === null) {
115
+ throw new Error('Views container not defined. Please declare a <views-container> element in your application');
116
+ }
106
117
  return new Promise((resolve) => {
107
118
  if (this._currentComponentRef) {
108
119
  const element = this._currentComponentRef.location.nativeElement;
@@ -125,14 +136,6 @@ export class ViewsService extends SingletoneStrictClass {
125
136
  }
126
137
  });
127
138
  }
128
- /**
129
- * Auxiliary method to test if the views container instance is available on the application
130
- */
131
- verifyViewsContainerExist() {
132
- if (this._viewContainerRef === null) {
133
- throw new Error('Views container not defined. Please declare a <views-container> element in your application');
134
- }
135
- }
136
139
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: ViewsService, deps: [{ token: i1.AnimationBuilder }], target: i0.ɵɵFactoryTarget.Injectable }); }
137
140
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: ViewsService, providedIn: 'root' }); }
138
141
  }
@@ -142,4 +145,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
142
145
  providedIn: 'root',
143
146
  }]
144
147
  }], ctorParameters: () => [{ type: i1.AnimationBuilder }] });
145
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"views.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/views.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { Type, ViewContainerRef, Injectable, ComponentRef } from '@angular/core';\r\nimport { View } from '../model/classes/View';\r\nimport { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\nimport { AnimationBuilder, animate, style } from '@angular/animations';\n\r\n\r\n/**\r\n * Manages adding, removing and manipulating the application views\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class ViewsService extends SingletoneStrictClass {\r\n\r\n\r\n    /**\r\n     * See getter method for docs\r\n     */\r\n    private _loadedViewClass: Type<View> | null = null;\r\n\r\n\r\n    /**\r\n     * See setter method for docs\r\n     */\r\n    private _viewContainerRef: ViewContainerRef | null = null;\r\n    \r\n\r\n    /**\r\n     * Contains the reference to the currently loaded view component\r\n     */\r\n    private _currentComponentRef: ComponentRef<View> | null = null;\r\n    \r\n\r\n    constructor(private animationBuilder: AnimationBuilder) {\r\n\r\n\t\tsuper(ViewsService);\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if there's any view currently loaded on the application views container\r\n     */\r\n    get isViewLoaded() {\r\n\r\n        return this._loadedViewClass !== null;\r\n    }\r\n\r\n\r\n    /**\r\n     * Stores a reference to the visual element that will be used as the container for all the views.\r\n     * This is automatically assigned by the tg-views-container component once it is created.\r\n     *\r\n     * @see ViewsContainerComponent for more information\r\n     */\r\n    set viewContainerRef(ref: ViewContainerRef | null) {\r\n\r\n        if (ref !== null && this._viewContainerRef !== null) {\r\n\r\n            throw new Error('Views container already defined. Only one views container element can exist on an application');\r\n        }\r\n\r\n        this._viewContainerRef = ref;\r\n    }\r\n\r\n\r\n    /**\r\n     * Create a new view instance with the specified type and add it to the current application views container. Any currently loaded \r\n     * view will be removed\r\n     *\r\n     * Make sure this method is called when all the visual components of the application have been created (ngAfterViewInit)\r\n     *\r\n     * @param view The classname for the view that we want to create and add to the views container (must extend View base class).\r\n     *\r\n     * @return The instance of the newly added and created view. \r\n     */\r\n    async pushView(view: Type<View>) {\r\n        \r\n        this.verifyViewsContainerExist();\r\n                \r\n        // If the loaded view is the same as the specified one, we will do nothing\r\n        if (this._loadedViewClass === view) {\r\n\r\n            return this._currentComponentRef;\r\n        }\r\n                    \r\n        // If a view is already loaded, we will unload it first\r\n        if (this._loadedViewClass !== null) {\r\n\r\n            await this.removeCurrentView();\r\n        }\r\n\r\n        // Create the new view\r\n        const newComponentRef = (this._viewContainerRef as ViewContainerRef).createComponent(view);\r\n        newComponentRef.changeDetectorRef.detectChanges();\r\n\r\n        // Set initial opacity to 0\r\n        newComponentRef.location.nativeElement.style.opacity = '0';\r\n\r\n        // Fade in the new view\r\n        const fadeInPlayer = this.animationBuilder.build([\r\n            style({ opacity: 0 }),\r\n            animate('300ms ease', style({ opacity: 1 }))\r\n        ]).create(newComponentRef.location.nativeElement);\r\n\r\n        await new Promise<void>((resolve) => {\r\n            fadeInPlayer.onDone(() => {\r\n                this._currentComponentRef = newComponentRef;\r\n                this._loadedViewClass = view;\r\n                resolve();\r\n            });\r\n            fadeInPlayer.play();\r\n        });\r\n\r\n        return this._currentComponentRef;\r\n    }\r\n\r\n\r\n    /**\r\n     * Delete the currently loaded view from the views container, and leave it empty.\r\n     * If no view is currently loaded, this method will do nothing\r\n     */\r\n    async popView() {\r\n\r\n        this.verifyViewsContainerExist();\r\n\r\n        if (this._loadedViewClass !== null && this._currentComponentRef) {\r\n            \r\n            await this.removeCurrentView();\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to remove the currently loaded view from the views container using a promise\r\n     */\r\n    private removeCurrentView(): Promise<void> {\r\n        \r\n        return new Promise<void>((resolve) => {\r\n            \r\n            if (this._currentComponentRef) {\r\n                \r\n                const element = this._currentComponentRef.location.nativeElement;\r\n                \r\n                const fadeOutPlayer = this.animationBuilder.build([\r\n                    style({ opacity: 1 }),\r\n                    animate('300ms ease', style({ opacity: 0 }))\r\n                ]).create(element);\r\n\r\n                fadeOutPlayer.onDone(() => {\r\n                    \r\n                    if(this._viewContainerRef !== null){\r\n                        \r\n                        this._viewContainerRef.clear();\r\n                    }\r\n                    \r\n                    this._currentComponentRef = null;\r\n                    this._loadedViewClass = null;\r\n                    resolve();\r\n                });\r\n\r\n                fadeOutPlayer.play();\r\n                \r\n            } else {\r\n                \r\n                resolve();\r\n            }\r\n        });\r\n    }\r\n\r\n\r\n    /**\r\n     * Auxiliary method to test if the views container instance is available on the application\r\n     */\r\n    private verifyViewsContainerExist() {\r\n\r\n        if (this._viewContainerRef === null) {\r\n\r\n            throw new Error('Views container not defined. Please declare a <views-container> element in your application');\r\n        }\r\n    }\r\n}\r\n"]}
148
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"views.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/views.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * TurboGUI is A library that helps with the most common and generic UI elements and functionalities\r\n *\r\n * Website : -> http://www.turbogui.org\r\n * License : -> Licensed under the Apache License, Version 2.0. You may not use this file except in compliance with the License.\r\n * License Url : -> http://www.apache.org/licenses/LICENSE-2.0\r\n * CopyRight : -> Copyright 2018 Edertone Advanded Solutions. https://www.edertone.com\r\n */\r\n\r\nimport { Type, ViewContainerRef, Injectable, ComponentRef } from '@angular/core';\r\nimport { View } from '../model/classes/View';\r\nimport { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\nimport { AnimationBuilder, animate, style } from '@angular/animations';\n\r\n\r\n/**\r\n * Manages adding, removing and manipulating the application views\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class ViewsService extends SingletoneStrictClass {\r\n\r\n\r\n    /**\r\n     * See getter method for docs\r\n     */\r\n    private _loadedViewClass: Type<View> | null = null;\r\n\r\n\r\n    /**\r\n     * See setter method for docs\r\n     */\r\n    private _viewContainerRef: ViewContainerRef | null = null;\r\n    \r\n\r\n    /**\r\n     * Contains the reference to the currently loaded view component\r\n     */\r\n    private _currentComponentRef: ComponentRef<View> | null = null;\r\n    \r\n    \r\n    /**\r\n     * Flag that stores if any view is in the process of being loaded\r\n     */\r\n    private _isLoadingView = false;\r\n    \r\n\r\n    constructor(private animationBuilder: AnimationBuilder) {\r\n\r\n\t\tsuper(ViewsService);\r\n    }\r\n\r\n\r\n    /**\r\n     * Tells if there's any view currently loaded on the application views container\r\n     */\r\n    get isViewLoaded() {\r\n\r\n        return this._loadedViewClass !== null;\r\n    }\r\n\r\n\r\n    /**\r\n     * Stores a reference to the visual element that will be used as the container for all the views.\r\n     * This is automatically assigned by the tg-views-container component once it is created.\r\n     *\r\n     * @see ViewsContainerComponent for more information\r\n     */\r\n    set viewContainerRef(ref: ViewContainerRef | null) {\r\n\r\n        if (ref !== null && this._viewContainerRef !== null) {\r\n\r\n            throw new Error('Views container already defined. Only one views container element can exist on an application');\r\n        }\r\n\r\n        this._viewContainerRef = ref;\r\n    }\r\n\r\n\r\n    /**\r\n     * Create a new view instance with the specified type and add it to the current application views container. Any currently loaded \r\n     * view will be removed.\r\n     * \r\n     * If we push a view while another one is in the process of being loaded, the new push will be ignored.\r\n     *\r\n     * Make sure this method is called when all the visual components of the application have been created (ngAfterViewInit)\r\n     *\r\n     * @param view The classname for the view that we want to create (must extend View base class). A new angular object \r\n     *        will be instantiated and loaded into the views container.\r\n     */\r\n    async pushView(view: Type<View>) {\r\n        \r\n        // If the loaded view is the same as the specified one, we will do nothing\r\n        if (this._loadedViewClass === view) {\r\n\r\n            return;\r\n        }\r\n                    \r\n        // If a view is already being loaded, nothing to do\r\n        if(this._isLoadingView){\r\n            \r\n            return;\r\n        }\r\n        \r\n        this._isLoadingView = true;\r\n                \r\n        // If a view is already loaded, we will unload it first\r\n        if (this._loadedViewClass !== null) {\r\n\r\n            await this.removeCurrentView();\r\n        }\r\n\r\n        // Create the new view\r\n        const newComponentRef = (this._viewContainerRef as ViewContainerRef).createComponent(view);\r\n        newComponentRef.changeDetectorRef.detectChanges();\r\n\r\n        // Set initial opacity to 0\r\n        newComponentRef.location.nativeElement.style.opacity = '0';\r\n\r\n        // Fade in the new view\r\n        const fadeInPlayer = this.animationBuilder.build([\r\n            style({ opacity: 0 }),\r\n            animate('300ms ease', style({ opacity: 1 }))\r\n        ]).create(newComponentRef.location.nativeElement);\r\n\r\n        await new Promise<void>((resolve) => {\r\n            fadeInPlayer.onDone(() => {\r\n                this._currentComponentRef = newComponentRef;\r\n                this._loadedViewClass = view;\r\n                this._isLoadingView = false;\r\n                resolve();\r\n            });\r\n            fadeInPlayer.play();\r\n        });\r\n    }\r\n\r\n\r\n    /**\r\n     * Delete the currently loaded view from the views container, and leave it empty.\r\n     * If no view is currently loaded, this method will do nothing\r\n     */\r\n    async popView() {\r\n\r\n        if (this._loadedViewClass !== null && this._currentComponentRef) {\r\n            \r\n            await this.removeCurrentView();\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to remove the currently loaded view from the views container using a promise\r\n     */\r\n    private removeCurrentView(): Promise<void> {\r\n        \r\n        if (this._viewContainerRef === null) {\r\n\r\n            throw new Error('Views container not defined. Please declare a <views-container> element in your application');\r\n        }\r\n\r\n        return new Promise<void>((resolve) => {\r\n            \r\n            if (this._currentComponentRef) {\r\n                \r\n                const element = this._currentComponentRef.location.nativeElement;\r\n                \r\n                const fadeOutPlayer = this.animationBuilder.build([\r\n                    style({ opacity: 1 }),\r\n                    animate('300ms ease', style({ opacity: 0 }))\r\n                ]).create(element);\r\n\r\n                fadeOutPlayer.onDone(() => {\r\n                    \r\n                    if(this._viewContainerRef !== null){\r\n                        \r\n                        this._viewContainerRef.clear();\r\n                    }\r\n                    \r\n                    this._currentComponentRef = null;\r\n                    this._loadedViewClass = null;\r\n                    resolve();\r\n                });\r\n\r\n                fadeOutPlayer.play();\r\n                \r\n            } else {\r\n                \r\n                resolve();\r\n            }\r\n        });\r\n    }\r\n}\r\n"]}