turbogui-angular 15.3.0 → 16.0.0

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.
@@ -234,8 +234,9 @@ export class DialogService extends SingletoneStrictClass {
234
234
  * - modal: True (default) if selecting an option is mandatory to close the dialog, false if the dialog can be closed
235
235
  * by the user clicking outside it
236
236
  * - texts: A list with strings containing the dialog texts, sorted by importance. When dialog has a title, this should
237
- * be placed first, subtitle second and so (Each dialog may accept different number of texts).
237
+ * be placed first, subtitle second and so (Each dialog may accept a different custom number of texts).
238
238
  * - options: A list of strings that will be used as button captions for each one of the accepted dialog options
239
+ * - data: An object that we can use to pass any extra data that we want to the dialog
239
240
  * - viewContainerRef: This is important if we want to propagate providers from a parent component to this dialog. We must specify
240
241
  * this reference to make sure the same services injected on the parent are available too at the child dialog
241
242
  *
@@ -250,6 +251,7 @@ export class DialogService extends SingletoneStrictClass {
250
251
  properties.modal = properties.modal ?? true;
251
252
  properties.texts = properties.texts ?? [];
252
253
  properties.options = properties.options ?? [];
254
+ properties.data = properties.data ?? {};
253
255
  // Generate a string to uniquely identify this dialog on the list of active dialogs
254
256
  // A dialog is considered as unique if the dialog id and texts are exactly the same. We do not take options into consideration
255
257
  // as there may be dialogs with a big amount of options available.
@@ -269,7 +271,7 @@ export class DialogService extends SingletoneStrictClass {
269
271
  autoFocus: false,
270
272
  closeOnNavigation: !properties.modal,
271
273
  viewContainerRef: properties.viewContainerRef,
272
- data: { texts: properties.texts, options: properties.options }
274
+ data: { texts: properties.texts, options: properties.options, data: properties.data }
273
275
  });
274
276
  // Assign the dialog ID only if specifically set on properties
275
277
  if (properties.id && properties.id !== undefined) {
@@ -403,4 +405,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
403
405
  providedIn: 'root',
404
406
  }]
405
407
  }], 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":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC3D,OAAO,EAAQ,UAAU,EAAqG,MAAM,eAAe,CAAC;AAGpJ,OAAO,EAAE,sBAAsB,EAAE,MAAM,8DAA8D,CAAC;AACtG,OAAO,EAAE,eAAe,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAEvE,OAAO,EAAE,4BAA4B,EAAE,MAAM,0EAA0E,CAAC;AACxH,OAAO,EAAE,qBAAqB,EAAE,MAAM,wCAAwC,CAAC;;;;AAG/E;;GAEG;AAIH,MAAM,OAAO,aAAc,SAAQ,qBAAqB;IAyFpD,YAAY,eAAiC,EAChB,WAAwB,EACxB,SAAoB,EACpB,QAAkB,EAClB,cAA8B,EAC9B,wBAAkD;QAEjF,KAAK,CAAC,aAAa,CAAC,CAAC;QANU,gBAAW,GAAX,WAAW,CAAa;QACxB,cAAS,GAAT,SAAS,CAAW;QACpB,aAAQ,GAAR,QAAQ,CAAU;QAClB,mBAAc,GAAd,cAAc,CAAgB;QAC9B,6BAAwB,GAAxB,wBAAwB,CAA0B;QA3F/E;;;;WAIG;QACH,kCAA6B,GAAiC,sBAAsB,CAAC;QAGrF;;WAEG;QACK,eAAU,GAAG,IAAI,CAAC;QAG1B;;WAEG;QACK,wBAAmB,GAAG,KAAK,CAAC;QAGpC;;;;WAIG;QACK,qBAAgB,GAAmD,IAAI,CAAC;QAGhF;;WAEG;QACK,wBAAmB,GAA2B,IAAI,CAAC;QAG3D;;WAEG;QACK,uBAAkB,GAAG,KAAK,CAAC;QAGnC;;;;;WAKG;QACK,mBAAc,GAAa,EAAE,CAAC;QAGtC;;;WAGG;QACK,2BAAsB,GAAwC,EAAE,CAAC;QASzE;;WAEG;QACK,gCAA2B,GAAwB,IAAI,CAAC;QAGhE;;WAEG;QACK,6BAAwB,GAAwB,IAAI,CAAC;QAG7D;;WAEG;QACK,+BAA0B,GAAwB,IAAI,CAAC;QAG/D;;WAEG;QACK,iCAA4B,GAAwB,IAAI,CAAC;QAY7D,IAAI,CAAC,SAAS,GAAG,eAAe,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IAChE,CAAC;IAGD;;;;;OAKG;IACH,IAAI,SAAS,CAAC,CAAU;QAEpB,IAAI,CAAC,KAAK,IAAI,CAAC,UAAU,EAAE;YAEvB,OAAO;SACV;QAED,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;IACxB,CAAC;IAGD;;;;;;;OAOG;IACH,0BAA0B;QAEtB,IAAI,IAAI,CAAC,2BAA2B,KAAK,IAAI,EAAE;YAE3C,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,EAAE,cAAc,EAC7E,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC,CAAC;SAC5C;IACL,CAAC;IAGD;;OAEG;IACH,6BAA6B;QAEzB,IAAI,IAAI,CAAC,2BAA2B,KAAK,IAAI,EAAE;YAE3C,IAAI,CAAC,2BAA2B,EAAE,CAAC;YACnC,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC;SAC3C;IACL,CAAC;IAGD;;;;;;;;;;;;;OAaG;IACH,iBAAiB;QAEb,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAE9C,OAAO;SACV;QAED,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAE/B,kFAAkF;QAClF,IAAI,IAAI,CAAC,gBAAgB,KAAK,IAAI,EAAE;YAEhC,IAAI,CAAC,gBAAgB,GAAG,IAAI,eAAe,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;YAEhF,+DAA+D;YAC/D,IAAI,CAAC,mBAAmB,GAAG,IAAI,eAAe,CACtC,QAAQ,CAAC,IAAI,EACb,IAAI,CAAC,wBAAwB,EAC7B,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,QAAQ,CAAC,CAAC;SAC1B;QAEA,IAAI,CAAC,mBAAuC,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAE5E,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;IACpC,CAAC;IAGD;;OAEG;IACH,IAAI,kBAAkB;QAElB,OAAO,IAAI,CAAC,mBAAmB,CAAC;IACpC,CAAC;IAGD;;OAEG;IACH,oBAAoB;QAEhB,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE;YAE/C,OAAO;SACV;QAED,IAAI,IAAI,CAAC,gBAAgB,KAAK,IAAI,EAAE;YAE/B,IAAI,CAAC,mBAAuC,CAAC,MAAM,EAAE,CAAC;SAC1D;QAED,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAE9B,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;IACrC,CAAC;IAGD;;OAEG;IACH,UAAU;QAEN,+BAA+B;IACnC,CAAC;IAGD;;;;;;;;;;;;OAYG;IACH,WAAW,CAAC,MAAyB,EAAE,OAAe,EAAE,MAAM,GAAG,EAAE,EAAE,iBAAsC,IAAI;QAE3G,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YAElB,OAAO;SACV;QAED,IAAI,IAAI,CAAC,kBAAkB,EAAE;YAEzB,MAAM,IAAI,KAAK,CAAC,8DAA8D,CAAC,CAAC;SACnF;QAED,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;QAE/B,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,KAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;QAE/F,IAAI,cAAc,KAAK,IAAI,EAAE;YAEzB,WAAW,CAAC,QAAQ,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE;gBAElC,cAAc,EAAE,CAAC;YACrB,CAAC,CAAC,CAAC;SACN;IACL,CAAC;IAGD;;OAEG;IACH,IAAI,iBAAiB;QAEjB,OAAO,IAAI,CAAC,kBAAkB,CAAC;IACnC,CAAC;IAGD;;;;OAIG;IACH,cAAc;QAEV,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,kBAAkB,EAAE;YAE9C,OAAO;SACV;QAED,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;QAE3B,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;IACpC,CAAC;IAGD;;;;;;;;;;;;;;;;;;;;;;;;OAwBG;IACH,SAAS,CAAC,oBAA+C,EAC/C,UAQiD,EACjD,WAAsE,IAAI;QAEhF,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YAElB,OAAO;SACV;QAED,sDAAsD;QACtD,UAAU,CAAC,KAAK,GAAG,UAAU,CAAC,KAAK,IAAI,IAAI,CAAC;QAC5C,UAAU,CAAC,KAAK,GAAG,UAAU,CAAC,KAAK,IAAI,EAAE,CAAC;QAC1C,UAAU,CAAC,OAAO,GAAG,UAAU,CAAC,OAAO,IAAI,EAAE,CAAC;QAE9C,mFAAmF;QACnF,8HAA8H;QAC9H,kEAAkE;QAClE,IAAI,SAAS,GAAI,oBAA4B,CAAC,iBAAiB,CAAC;QAEhE,IAAG,SAAS,KAAK,EAAE,EAAC;YAEhB,MAAM,IAAI,KAAK,CAAC,+FAA+F,oBAAoB,GAAG,CAAC,CAAC;SAC3I;QAED,MAAM,UAAU,GAAG,SAAS,GAAG,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEzD,sDAAsD;QACtD,IAAI,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;YAE1C,OAAO;SACV;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,oBAAoB,EAAE;YACxD,KAAK,EAAE,UAAU,CAAC,KAAK,IAAI,KAAK;YAChC,QAAQ,EAAE,UAAU,CAAC,QAAQ,IAAI,MAAM;YACvC,YAAY,EAAE,UAAU,CAAC,KAAK;YAC9B,SAAS,EAAE,KAAK;YAChB,iBAAiB,EAAE,CAAC,UAAU,CAAC,KAAK;YACpC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB;YAC7C,IAAI,EAAE,EAAE,KAAK,EAAE,UAAU,CAAC,KAAK,EAAE,OAAO,EAAE,UAAU,CAAC,OAAO,EAAE;SAC/D,CAAC,CAAC;QAEX,8DAA8D;QAC9D,IAAG,UAAU,CAAC,EAAE,IAAI,UAAU,CAAC,EAAE,KAAK,SAAS,EAAC;YAE/C,SAAS,CAAC,EAAE,GAAG,UAAU,CAAC,EAAE,CAAC;SAC7B;QAEK,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACrC,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAE5C,SAAS,CAAC,YAAY,EAAE,CAAC,SAAS,CAAC,CAAC,SAAoC,EAAE,EAAE;YAExE,IAAI,CAAC,cAAc,GAAG,UAAU,CAAC,aAAa,CAAC,IAAI,CAAC,cAAc,EAAE,UAAU,CAAC,CAAC;YAChF,IAAI,CAAC,sBAAsB,GAAG,UAAU,CAAC,aAAa,CAAC,IAAI,CAAC,sBAAsB,EAAE,SAAS,CAAC,CAAC;YAE/F,IAAG,CAAC,UAAU,CAAC,KAAK,IAAI,SAAS,KAAK,SAAS,EAAC;gBAE5C,SAAS,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC,EAAE,CAAC;aAE7B;iBAAK,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;gBAEhD,MAAM,IAAI,KAAK,CAAC,8CAA8C,CAAC,CAAC;aACnE;YAED,IAAI,QAAQ,KAAK,IAAI,EAAE;gBAEnB,IAAG,SAAS,CAAC,KAAK,IAAI,CAAC,IAAI,SAAS,CAAC,KAAK,KAAK,IAAI,EAAC;oBAEhD,SAAS,CAAC,KAAK,GAAG,UAAU,CAAC,OAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;iBAC1D;gBAEA,QAA6D,CAAC,SAAS,CAAC,CAAC;aAC7E;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAGD;;;;;;;;;;;;;;;;;;OAkBG;IACH,sBAAsB,CAAC,UAOuC,EACvC,QAA+C;QAElE,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YAElB,OAAO;SACV;QAED,IAAI,CAAC,SAAS,CAAC,4BAA4B,EACvC;YACI,EAAE,EAAE,UAAU,CAAC,EAAE,IAAI,SAAS;YAC9B,KAAK,EAAE,UAAU,CAAC,KAAK,IAAI,KAAK;YAChC,QAAQ,EAAE,UAAU,CAAC,QAAQ,IAAI,MAAM;YACvC,MAAM,EAAE,UAAU,CAAC,MAAM,IAAI,KAAK;YAClC,SAAS,EAAE,UAAU,CAAC,SAAS,IAAI,MAAM;YACzC,KAAK,EAAE,UAAU,CAAC,KAAK,IAAI,KAAK;YAChC,KAAK,EAAE,CAAC,UAAU,CAAC,KAAK,IAAI,EAAE,CAAC;YAC/B,gBAAgB,EAAE,UAAU,CAAC,gBAAgB;SAChD,EAAC,CAAC,SAAS,EAAE,EAAE;YAEZ,QAAQ,CAAC,SAAS,CAAC,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAE,SAAS,CAAC,KAAc,CAAC,CAAC;QACxE,CAAC,CAAC,CAAC;IACX,CAAC;IAGD;;;;OAIG;IACH,gBAAgB;QAEZ,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YAElB,OAAO;SACV;QAED,KAAK,MAAM,SAAS,IAAI,IAAI,CAAC,sBAAsB,EAAE;YAEjD,SAAS,CAAC,KAAK,CAAC,EAAC,KAAK,EAAC,CAAC,CAAC,EAAC,CAAC,CAAC;SAC/B;QAED,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;QACzB,IAAI,CAAC,sBAAsB,GAAG,EAAE,CAAC;IACrC,CAAC;IAGD;;OAEG;IACP,mBAAmB;IACnB,EAAE;IACF,OAAO;IAGH;;OAEG;IACP,6BAA6B;IAC7B,EAAE;IACF,OAAO;IAGH;;OAEG;IACP,sBAAsB;IACtB,EAAE;IACF,OAAO;IAEH;;OAEG;IACK,uBAAuB;QAE3B,IAAI,IAAI,CAAC,wBAAwB,KAAK,IAAI,EAAE;YAExC,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,EAAE,SAAS,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC,CAAC;SACnH;QAED,IAAI,IAAI,CAAC,0BAA0B,KAAK,IAAI,EAAE;YAE1C,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,EAAE,WAAW,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC,CAAC;SACvH;QAED,IAAI,IAAI,CAAC,4BAA4B,KAAK,IAAI,EAAE;YAE5C,IAAI,CAAC,4BAA4B,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,EAAE,aAAa,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,cAAc,EAAE,CAAC,CAAC;SAC3H;IACL,CAAC;IAGD;;OAEG;IACK,sBAAsB;QAE1B,IAAI,IAAI,CAAC,wBAAwB,KAAK,IAAI,EAAE;YAExC,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAChC,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC;SACxC;QAED,IAAI,IAAI,CAAC,0BAA0B,KAAK,IAAI,EAAE;YAE1C,IAAI,CAAC,0BAA0B,EAAE,CAAC;YAClC,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC;SAC1C;QAED,IAAI,IAAI,CAAC,4BAA4B,KAAK,IAAI,EAAE;YAE5C,IAAI,CAAC,4BAA4B,EAAE,CAAC;YACpC,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC;SAC1C;IACL,CAAC;8GApiBQ,aAAa;kHAAb,aAAa,cAFZ,MAAM;;2FAEP,aAAa;kBAHzB,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","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"]}
408
+ //# 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 a different custom 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     *            - data: An object that we can use to pass any extra data that we want to the dialog\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                           data?: any,\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        properties.data = properties.data ?? {};\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, data: properties.data }\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"]}
@@ -109,6 +109,8 @@ export class LocalesService extends SingletoneStrictClass {
109
109
  }
110
110
  /**
111
111
  * Initializes the translation system by loading and parsing bundle files from the specified translations path.
112
+ * After the promise finishes, the class will contain all the translation data and will be ready to translate any
113
+ * provided key.
112
114
  *
113
115
  * @param translationsPath - Url where the translations Json structure of libraries/bundles/locales/keys is available.
114
116
  * @param locales An array of locale codes (e.g., ['en_US', 'es_ES', 'fr_FR']) to load. These will be added to the translation
@@ -206,8 +208,8 @@ export class LocalesService extends SingletoneStrictClass {
206
208
  * @param string key The key we want to read from the specified resource bundle
207
209
  * @param string bundlePath A string with the format 'library_name/bundle_name' that is used to locate the bundle were the key to translate is found
208
210
  * @param array replaceWildcards A list of values that will replace wildcards that may be found on the translated text. Each wildcard
209
- * will be replaced with the element whose index on toReplace matches it. Check the documentation for this.wildCardsFormat
210
- * property to know more about how to setup wildcards.
211
+ * will be replaced with the element whose index on replaceWildcards matches it. Check the documentation for this.wildCardsFormat
212
+ * property to know more about how to setup wildcards on your translations.
211
213
  *
212
214
  * @see setWildCardsFormat()
213
215
  *
@@ -489,4 +491,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImpor
489
491
  providedIn: 'root',
490
492
  }]
491
493
  }], ctorParameters: () => [] });
492
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"locales.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/locales.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 { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\r\n\r\n\r\n/**\r\n * Manages application text translations and languages\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class LocalesService extends SingletoneStrictClass {\r\n\r\n\r\n    /**\r\n     * if the class has been correctly initialized and translations have been correctly loaded\r\n     */\r\n    private _isInitialized = false;\r\n        \r\n\r\n    /**\r\n     * @see getLocales()\r\n     */\r\n    private _locales:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * @see getLanguages()\r\n     */\r\n    private _languages:string[] = [];\r\n        \r\n    \r\n    /**\r\n     * Stores all the loaded localization data by library name, bundle name, key and locales\r\n     */\r\n    private _loadedTranslations:any = {};\r\n    \r\n    \r\n    /**\r\n     * Stores a memory cache to improve performance when outputing translations\r\n     */\r\n    private _keyValuesCache:any = {};\r\n\r\n\r\n    /**\r\n     * @see setWildCardsFormat()\r\n     */\r\n    private _wildCardsFormat = '{N}';\r\n\r\n\r\n    /**\r\n     * @see setMissingKeyFormat()\r\n     */\r\n    private _missingKeyFormat = '$exception';\r\n\r\n\r\n    /**\r\n     * Stores a hash value that is used to improve the performance for translation t() methods.\r\n     * This is computed based on _wildCardsFormat plus _missingKeyFormat plus the current primary locale\r\n     * Methods that change these values will recalculate the hash string, so when calling translation methods, the\r\n     * performance will be as fast as possible.\r\n     */\r\n    private _cacheHashBaseString = '';\r\n    \r\n    \r\n    /**\r\n     * Fully featured translation manager to be used with any application that requires text internationalization.\r\n     */\r\n    constructor() {\r\n\r\n        super(LocalesService);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Wildcards are string fragments that are placed inside the translated texts. Their main purpose is to be replaced at\r\n     * runtime by custom values like for example a user name, a date, a numeric value, etc..\r\n     *\r\n     * This class helps with this process by including a parameter called 'toReplace' on all ->t methods which allows us\r\n     * to specify a string or list of strings that will replace the respective wildcards on the translated text. Each wildcard\r\n     * must follow the format specified here, and contain a numeric digit that will be used to find the replacement text at the\r\n     * 'toReplace' list. For example, if we define $N as the wildcard format, and we have a translation that contains $0, $1, $2,\r\n     * $0 will be replaced with the first element on toReplace, $1 with the second and so.\r\n     *\r\n     * We usually set this before initializing the class translation data\r\n     *\r\n     * Notice that N is mandayory on the wildcards format and the first index value is 0.\r\n     *\r\n     * @param value The wildcards format we want to set\r\n     * \r\n     * @returns The value that's been set\r\n     */\r\n    setWildCardsFormat(value:string) {\r\n\r\n        if(!value.includes('N')){\r\n        \r\n            throw new Error(\"N is mandatory to replace wildcards\");\r\n        }\r\n\r\n        this._cacheHashBaseString = value + this._missingKeyFormat + ((this._locales.length > 0) ? this._locales[0] : '');\r\n\r\n        this._wildCardsFormat = value;\r\n        \r\n        return value;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Defines the behaviour for get(), getStartCase(), etc... methods when a key is not found on\r\n     * a bundle or the bundle does not exist\r\n     *\r\n     * If missingKeyFormat is an empty string, all missing keys will return an empty value (not recommended)\r\n     *\r\n     * If missingKeyFormat contains a string, that string will be always returned for missing keys\r\n     *\r\n     * If missingKeyFormat contains a string with one of the following predefined wildcards:<br>\r\n     *    - $key will be replaced with key name. Example: get(\"NAME\") will output [NAME] if key is not found and missingKeyFormat = '[$key]'<br>\r\n     *    - $exception (default value) will throw an exception with the problem cause description.\r\n     *\r\n     * @param value The missing key format we want to set\r\n     * \r\n     * @returns The value that's been set\r\n     */\r\n    setMissingKeyFormat(value:string) {\r\n\r\n        this._cacheHashBaseString = this._wildCardsFormat + value + ((this._locales.length > 0) ? this._locales[0] : '');\r\n\r\n        this._missingKeyFormat = value;\r\n        \r\n        return value;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * @see setMissingKeyFormat()\r\n     */\r\n    getMissingKeyFormat() {\r\n\r\n        return this._missingKeyFormat;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Initializes the translation system by loading and parsing bundle files from the specified translations path.\r\n     * \r\n     * @param translationsPath - Url where the translations Json structure of libraries/bundles/locales/keys is available.\r\n     * @param locales An array of locale codes (e.g., ['en_US', 'es_ES', 'fr_FR']) to load. These will be added to the translation\r\n     *        path using the following format: translationsPath/en_US-es_ES-fr_FR. The order of this array will determine the\r\n     *        translation priority\r\n     * @param parameters Any extra parameters to be attached to the translationsPath after the locales like /param1/param2/ etc\r\n     * \r\n     * @return A promise that will resolve if the translations get correctly loaded, or reject with an error if load fails \r\n     */\r\n    initialize(translationsPath:string, locales:string[], parameters:string[]){\r\n    \r\n        this._isInitialized = false;\r\n        this._loadedTranslations = {};\r\n        \r\n        // Validate received locales are correct\r\n        for(const locale of locales) {\r\n\r\n            this._validateLocaleString(locale);\r\n        }\r\n        \r\n        let translationsFullPath = translationsPath + '/' + locales.join('-') + '/' + parameters.join('/');\r\n        \r\n        return new Promise((resolve, reject) => {\r\n        \r\n            fetch(translationsFullPath).then(response => {\r\n                \r\n                if (!response.ok) {\r\n                    \r\n                  throw new Error(`HTTP error! status: ${response.status}`);\r\n                }\r\n                \r\n                return response.json();\r\n          \r\n            }).then(data => {\r\n                \r\n                this._loadedTranslations = data;\r\n                this._isInitialized = true;\r\n                this._locales = locales;\r\n                this._languages = locales.map((l: string) => l.substring(0, 2));\r\n                this._cacheHashBaseString = this._wildCardsFormat + this._missingKeyFormat + this._locales[0];\r\n                \r\n                resolve(undefined);\r\n          \r\n            }).catch(error => {\r\n            \r\n                reject(new Error(`ERROR LOADING LOCALES FROM: ${translationsFullPath}\\n` + error));\r\n            });\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Check if the class has been correctly initialized and translations have been correctly loaded\r\n     */\r\n    isInitialized(){\r\n\r\n        return this._isInitialized;\r\n    }\r\n\r\n\r\n    /**\r\n     * Aux method to verify that this class is correctly initialized with translation data\r\n     */\r\n    private _validateInitialized(){\r\n\r\n        if(!this._isInitialized){\r\n\r\n            throw new Error('LocalesManager not initialized');\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Checks if the specified locale is currently loaded for the currently defined bundles and paths.\r\n     *\r\n     * @param locale A locale to check. For example 'en_US'\r\n     *\r\n     * @return True if the locale is currently loaded on the class, false if not.\r\n     */\r\n    isLocaleLoaded(locale:string){\r\n\r\n        this._validateLocaleString(locale);\r\n\r\n        return this._locales.includes(locale);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to validate that a locale string is correctly formatted\r\n     *\r\n     * @param string $locale A locale string\r\n     */\r\n    private _validateLocaleString(locale:string){\r\n        \r\n        if(!/^[a-z]{2}_[A-Z]{2}$/.test(locale)) {\r\n\r\n            throw new Error('locale must be a valid xx_XX value');\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Checks if the specified 2 digit language is currently loaded for the currently defined bundles and paths.\r\n     *\r\n     * @param language A language to check. For example 'en'\r\n     *\r\n     * @return True if the language is currently loaded on the class, false if not.\r\n     */\r\n    isLanguageLoaded(language:string){\r\n\r\n        this._validateLanguageString(language);\r\n\r\n        return this._languages.includes(language);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to validate that a language string is correctly formatted\r\n     *\r\n     * @param language A 2 digit language string\r\n     */\r\n    _validateLanguageString(language:string){\r\n\r\n        if (!/^[a-z]{2}$/.test(language)) {\r\n\r\n            throw new Error('language must be a valid 2 digit value');\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation to the current primary locale for the given key, library and bundle\r\n     *\r\n     * @param string key The key we want to read from the specified resource bundle\r\n     * @param string bundlePath A string with the format 'library_name/bundle_name' that is used to locate the bundle were the key to translate is found\r\n     * @param array replaceWildcards A list of values that will replace wildcards that may be found on the translated text. Each wildcard\r\n     *        will be replaced with the element whose index on toReplace matches it. Check the documentation for this.wildCardsFormat\r\n     *        property to know more about how to setup wildcards.\r\n     *\r\n     * @see setWildCardsFormat()\r\n     *\r\n     * @return The translated text\r\n     */\r\n    t(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        this._validateInitialized();\r\n\r\n        // Create a cache key to improve performance when requesting the same key translation several times\r\n        const cacheKey = `${this._cacheHashBaseString}${key}${bundlePath}${replaceWildcards.join('')}`;\r\n\r\n        if (!this._keyValuesCache[cacheKey]) {\r\n\r\n            this._forceNonEmptyString(key, '', 'key must be non empty string');\r\n            this._forceNonEmptyString(bundlePath, '', 'bundlePath must be non empty string');\r\n            \r\n            const [library, bundle] = bundlePath.split('/');\r\n\r\n            this._forceNonEmptyString(library, '', 'no library specified on bundlePath');\r\n            this._forceNonEmptyString(bundle, '', 'no bundle specified on bundlePath');\r\n            \r\n            const replacementsCount = replaceWildcards.length;\r\n\r\n            // Loop all the locales to find the first one with a value for the specified key\r\n            for (const locale of this._locales) {\r\n\r\n                if (this._loadedTranslations[library]?.[bundle]?.[locale]?.[key]) {\r\n\r\n                    let result = this._loadedTranslations[library][bundle][locale][key];\r\n\r\n                    // Replace all wildcards on the text with the specified replacements if any\r\n                    for (let i = 0; i < replacementsCount; i++) {\r\n\r\n                        result = this._replace(result, this._replace(this._wildCardsFormat, 'N', i.toString()), replaceWildcards[i]);\r\n                    }\r\n\r\n                    this._keyValuesCache[cacheKey] = result;\r\n                    \r\n                    return result;\r\n                }\r\n            }\r\n\r\n            // Check if an exception needs to be thrown if the specified key is not found on this bundle\r\n            if (this._missingKeyFormat.includes('$exception')) {\r\n                \r\n                throw new Error(`key <${key}> not found on ${bundlePath}`);\r\n            }\r\n\r\n            this._keyValuesCache[cacheKey] = this._replace(this._missingKeyFormat, '$key', key);\r\n        }\r\n\r\n        return this._keyValuesCache[cacheKey];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as a string with all words first character capitalized\r\n     * and all the rest of the word with lower case\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tStartCase(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        return this.t(key, bundlePath, replaceWildcards).split(' ')\r\n                .map((word:any) => word ? word[0].toUpperCase() + word.slice(1).toLowerCase() : '').join(' ');\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as an all upper case string\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tAllUpperCase(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        return this.t(key, bundlePath, replaceWildcards).toUpperCase();\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as an all lower case string\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tAllLowerCase(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        return this.t(key, bundlePath, replaceWildcards).toLowerCase();\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as a string with the first character as Upper case\r\n     * and all the rest as lower case\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tFirstUpperRestLower(key:string, bundlePath:string, replaceWildcards:string[] = []){\r\n\r\n        const string = this.t(key, bundlePath, replaceWildcards);\r\n        \r\n        return string.charAt(0).toUpperCase() + string.slice(1).toLowerCase();\r\n    }\r\n    \r\n    \r\n    /**\r\n     * A list of strings containing the locales that are used by this class to translate the given keys, sorted by preference.\r\n     * Each string is formatted as a standard locale code with language and country joined by an underscore, like: en_US, fr_FR\r\n     *\r\n     * When a key and bundle are requested for translation, the class will check on the first language of this\r\n     * list for a translated text. If missing, the next one will be used, and so. This list is constructed after initialize\r\n     * methods is called.\r\n     *\r\n     * @example: After loading the following list of locales ['en_US', 'es_ES', 'fr_FR'] if we call LocalesManager.t('HELLO', 'lib1/greetings')\r\n     * the localization manager will try to locate the en_US value for the HELLO tag on the greetings bundle for the library lib1.\r\n     * If the tag is not found for the specified locale and bundle, the same search will be performed for the es_ES locale, and so, till a\r\n     * value is found or no more locales are defined.\r\n     */\r\n    getLocales(){\r\n\r\n        return this._locales;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * A list of strings containing the languages that are used by this class to translate the given keys, sorted by preference.\r\n     * Each string is formatted as a 2 digit language code, like: en, fr\r\n     *\r\n     * This list is the same as the locales() one, but containing only the language part of each locale (the first two digits)\r\n     *\r\n     * @see getLocales()\r\n     */\r\n    getLanguages(){\r\n\r\n        return this._languages;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the first locale from the list of loaded locales, which is the currently used to search for translated texts.\r\n     *\r\n     * @returns The locale that is defined as the primary one. For example: en_US, es_ES, ..\r\n     */\r\n    getPrimaryLocale(){\r\n\r\n        this._validateInitialized();\r\n\r\n        return this._locales[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the first language from the list of loaded locales, which is the currently used to search for translated texts.\r\n     *\r\n     * @returns The 2 digit language code that is defined as the primary one. For example: en, es, ..\r\n     */\r\n    getPrimaryLanguage(){\r\n\r\n        this._validateInitialized();\r\n\r\n        return this._languages[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Define the locale that will be placed at the front of the currently loaded locales list (moving all the others one position to the right).\r\n     *\r\n     * This will be the first locale to use when trying to get a translation.\r\n     *\r\n     * @param locale A currently loaded locale that will be moved to the first position of the loaded locales list. If the specified locale\r\n     *        is not currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLocale(locale:string){\r\n\r\n        this._validateInitialized();\r\n\r\n        if(!this.isLocaleLoaded(locale)){\r\n\r\n            throw new Error(locale + ' not loaded');\r\n        }\r\n\r\n        let result = [locale];\r\n\r\n        for (let l of this._locales){\r\n        \r\n            if(l !== locale){\r\n\r\n                result.push(l);\r\n            }\r\n        }\r\n\r\n        this._locales = result;\r\n        this._languages = this._locales.map((l: string) => l.substring(0, 2));\r\n        this._cacheHashBaseString = this._wildCardsFormat + this._missingKeyFormat + this._locales[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Moves the specified locales to the beginning of the locales list. This also alters the translation priority by setting the first\r\n     * provided locale as the most prioritary, the second as the next one and so.\r\n     *\r\n     * This method basically works exactly the same way as setPrimaryLocale but letting us add many locales at once.\r\n     *\r\n     * @see setPrimaryLocale()\r\n     *\r\n     * @param locales A list of locales to be moved to the beginning of the translation priority. First locales item will be the prefered\r\n     *        locale for translation, second will be the next one in case some key is not translated for the first one and so. If any of the\r\n     *        specified locales is not currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLocales(locales:string[]){\r\n\r\n        if(!Array.isArray(locales) ||\r\n            (new Set(locales).size !== locales.length) ||\r\n            locales.length === 0){\r\n\r\n                throw new Error('locales must be non empty string array with no duplicate elements');\r\n        }\r\n\r\n        for (let i = locales.length - 1; i >= 0; i--) {\r\n\r\n            this.setPrimaryLocale(locales[i]);\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Define the 2 digit language that will be placed at the front of the currently loaded locales list (moving all the others one position to the right).\r\n     *\r\n     * This will be the first language to use when trying to get a translation.\r\n     *\r\n     * @param language A 2 digit language code that matches with any of the currently loaded locales, which will\r\n     *        be moved to the first position of the loaded locales list. If the specified language does not match with\r\n     *        a locale that is currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLanguage(language:string){\r\n\r\n        for(let locale of this._locales){\r\n        \r\n            if(locale.substring(0, 2) === language){\r\n\r\n                this.setPrimaryLocale(locale);\r\n\r\n                return;\r\n            }\r\n        }\r\n\r\n        throw new Error(language + ' not loaded');\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Moves the locales that match the specified languages to the beginning of the locales list.\r\n     * Works the same as setPrimaryLocales() but with a list of the 2 digit language codes that match the respective locales.\r\n     *\r\n     * @see setPrimaryLocale()\r\n     * @see setPrimaryLanguage()\r\n     *\r\n     * @param languages A list of 2 digit language codes to be moved to the beginning of the translation priority. If any of the\r\n     *        specified languages does not match with a locale that is currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLanguages(languages:string[]){\r\n\r\n        if(!Array.isArray(languages) ||\r\n           (new Set(languages).size !== languages.length) ||\r\n            languages.length === 0){\r\n\r\n            throw new Error('languages must be non empty string array with no duplicate elements');\r\n        }\r\n\r\n        for (let i = languages.length - 1; i >= 0; i--) {\r\n\r\n            this.setPrimaryLanguage(languages[i]);\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Change the loaded locales translation preference order. The same locales that are currently loaded must be passed\r\n     * but with a different order to change the translation priority.\r\n     *\r\n     * @param locales A list with the new locales translation priority\r\n     *\r\n     * @returns void\r\n     */\r\n    setLocalesOrder(locales:string[]){\r\n\r\n        if(locales.length !== this._locales.length){\r\n\r\n            throw new Error('locales must contain all the currently loaded locales');\r\n        }\r\n\r\n        this._validateInitialized();\r\n\r\n        for(let locale of locales){\r\n        \r\n            if(!this.isLocaleLoaded(locale)){\r\n\r\n                throw new Error(locale + ' not loaded');\r\n            }\r\n        }\r\n\r\n        this._locales = locales;\r\n        this._languages = this._locales.map((l: string) => l.substring(0, 2));\r\n        this._cacheHashBaseString = this._wildCardsFormat + this._missingKeyFormat + this._locales[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * This is an aux method to implement the TurboCommons StringUtils replace method.\r\n     * It is exactly the same as the one on the library, but we implement it here to avoid having a dependency with TurboCommons\r\n     */\r\n    private _replace(string: string, search:string, replacement: string) {\r\n        \r\n        const escapedSearch = search.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\r\n           \r\n        return string.replace(new RegExp(escapedSearch, 'g'), replacement);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * This is an aux method to implement the TurboCommons StringUtils isEmpty method.\r\n     * It is exactly the same as the one on the library, but we implement it here to avoid having a dependency with TurboCommons\r\n     */\r\n    private _isEmpty(string:string) {\r\n        \r\n        let isString = (typeof string === 'string' || (string as any) instanceof String);\r\n                \r\n        // Throw exception if non string value was received\r\n        if(!isString){\r\n\r\n            // Empty or null value is considered empty\r\n            if(string == null || string == ''){\r\n    \r\n                return true;\r\n            }\r\n            \r\n            throw new Error(\"value is not a string\");\r\n        }\r\n\r\n        return string.replace(/[ \\n\\r\\t]/g, '') === '';\r\n    }\r\n    \r\n    \r\n    /**\r\n     * This is an aux method to implement the TurboCommons StringUtils forceNonEmptyString method.\r\n     * It is exactly the same as the one on the library, but we implement it here to avoid having a dependency with TurboCommons\r\n     */\r\n    private _forceNonEmptyString(value:any, valueName = '', errorMessage = 'must be a non empty string'){\r\n\r\n        let isString = (typeof value === 'string' || value instanceof String);\r\n        \r\n        if(!isString || this._isEmpty(value)){\r\n\r\n            throw new Error(valueName + ' ' + errorMessage);\r\n        }\r\n    }\r\n}\r\n"]}
494
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"locales.service.js","sourceRoot":"","sources":["../../../../../projects/turbogui-angular/src/main/controller/locales.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 { SingletoneStrictClass } from '../model/classes/SingletoneStrictClass';\r\n\r\n\r\n/**\r\n * Manages application text translations and languages\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class LocalesService extends SingletoneStrictClass {\r\n\r\n\r\n    /**\r\n     * if the class has been correctly initialized and translations have been correctly loaded\r\n     */\r\n    private _isInitialized = false;\r\n        \r\n\r\n    /**\r\n     * @see getLocales()\r\n     */\r\n    private _locales:string[] = [];\r\n    \r\n    \r\n    /**\r\n     * @see getLanguages()\r\n     */\r\n    private _languages:string[] = [];\r\n        \r\n    \r\n    /**\r\n     * Stores all the loaded localization data by library name, bundle name, key and locales\r\n     */\r\n    private _loadedTranslations:any = {};\r\n    \r\n    \r\n    /**\r\n     * Stores a memory cache to improve performance when outputing translations\r\n     */\r\n    private _keyValuesCache:any = {};\r\n\r\n\r\n    /**\r\n     * @see setWildCardsFormat()\r\n     */\r\n    private _wildCardsFormat = '{N}';\r\n\r\n\r\n    /**\r\n     * @see setMissingKeyFormat()\r\n     */\r\n    private _missingKeyFormat = '$exception';\r\n\r\n\r\n    /**\r\n     * Stores a hash value that is used to improve the performance for translation t() methods.\r\n     * This is computed based on _wildCardsFormat plus _missingKeyFormat plus the current primary locale\r\n     * Methods that change these values will recalculate the hash string, so when calling translation methods, the\r\n     * performance will be as fast as possible.\r\n     */\r\n    private _cacheHashBaseString = '';\r\n    \r\n    \r\n    /**\r\n     * Fully featured translation manager to be used with any application that requires text internationalization.\r\n     */\r\n    constructor() {\r\n\r\n        super(LocalesService);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Wildcards are string fragments that are placed inside the translated texts. Their main purpose is to be replaced at\r\n     * runtime by custom values like for example a user name, a date, a numeric value, etc..\r\n     *\r\n     * This class helps with this process by including a parameter called 'toReplace' on all ->t methods which allows us\r\n     * to specify a string or list of strings that will replace the respective wildcards on the translated text. Each wildcard\r\n     * must follow the format specified here, and contain a numeric digit that will be used to find the replacement text at the\r\n     * 'toReplace' list. For example, if we define $N as the wildcard format, and we have a translation that contains $0, $1, $2,\r\n     * $0 will be replaced with the first element on toReplace, $1 with the second and so.\r\n     *\r\n     * We usually set this before initializing the class translation data\r\n     *\r\n     * Notice that N is mandayory on the wildcards format and the first index value is 0.\r\n     *\r\n     * @param value The wildcards format we want to set\r\n     * \r\n     * @returns The value that's been set\r\n     */\r\n    setWildCardsFormat(value:string) {\r\n\r\n        if(!value.includes('N')){\r\n        \r\n            throw new Error(\"N is mandatory to replace wildcards\");\r\n        }\r\n\r\n        this._cacheHashBaseString = value + this._missingKeyFormat + ((this._locales.length > 0) ? this._locales[0] : '');\r\n\r\n        this._wildCardsFormat = value;\r\n        \r\n        return value;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Defines the behaviour for get(), getStartCase(), etc... methods when a key is not found on\r\n     * a bundle or the bundle does not exist\r\n     *\r\n     * If missingKeyFormat is an empty string, all missing keys will return an empty value (not recommended)\r\n     *\r\n     * If missingKeyFormat contains a string, that string will be always returned for missing keys\r\n     *\r\n     * If missingKeyFormat contains a string with one of the following predefined wildcards:<br>\r\n     *    - $key will be replaced with key name. Example: get(\"NAME\") will output [NAME] if key is not found and missingKeyFormat = '[$key]'<br>\r\n     *    - $exception (default value) will throw an exception with the problem cause description.\r\n     *\r\n     * @param value The missing key format we want to set\r\n     * \r\n     * @returns The value that's been set\r\n     */\r\n    setMissingKeyFormat(value:string) {\r\n\r\n        this._cacheHashBaseString = this._wildCardsFormat + value + ((this._locales.length > 0) ? this._locales[0] : '');\r\n\r\n        this._missingKeyFormat = value;\r\n        \r\n        return value;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * @see setMissingKeyFormat()\r\n     */\r\n    getMissingKeyFormat() {\r\n\r\n        return this._missingKeyFormat;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Initializes the translation system by loading and parsing bundle files from the specified translations path.\r\n     * After the promise finishes, the class will contain all the translation data and will be ready to translate any \r\n     * provided key.\r\n     * \r\n     * @param translationsPath - Url where the translations Json structure of libraries/bundles/locales/keys is available.\r\n     * @param locales An array of locale codes (e.g., ['en_US', 'es_ES', 'fr_FR']) to load. These will be added to the translation\r\n     *        path using the following format: translationsPath/en_US-es_ES-fr_FR. The order of this array will determine the\r\n     *        translation priority\r\n     * @param parameters Any extra parameters to be attached to the translationsPath after the locales like /param1/param2/ etc\r\n     * \r\n     * @return A promise that will resolve if the translations get correctly loaded, or reject with an error if load fails \r\n     */\r\n    initialize(translationsPath:string, locales:string[], parameters:string[]){\r\n    \r\n        this._isInitialized = false;\r\n        this._loadedTranslations = {};\r\n        \r\n        // Validate received locales are correct\r\n        for(const locale of locales) {\r\n\r\n            this._validateLocaleString(locale);\r\n        }\r\n        \r\n        let translationsFullPath = translationsPath + '/' + locales.join('-') + '/' + parameters.join('/');\r\n        \r\n        return new Promise((resolve, reject) => {\r\n        \r\n            fetch(translationsFullPath).then(response => {\r\n                \r\n                if (!response.ok) {\r\n                    \r\n                  throw new Error(`HTTP error! status: ${response.status}`);\r\n                }\r\n                \r\n                return response.json();\r\n          \r\n            }).then(data => {\r\n                \r\n                this._loadedTranslations = data;\r\n                this._isInitialized = true;\r\n                this._locales = locales;\r\n                this._languages = locales.map((l: string) => l.substring(0, 2));\r\n                this._cacheHashBaseString = this._wildCardsFormat + this._missingKeyFormat + this._locales[0];\r\n                \r\n                resolve(undefined);\r\n          \r\n            }).catch(error => {\r\n            \r\n                reject(new Error(`ERROR LOADING LOCALES FROM: ${translationsFullPath}\\n` + error));\r\n            });\r\n        });\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Check if the class has been correctly initialized and translations have been correctly loaded\r\n     */\r\n    isInitialized(){\r\n\r\n        return this._isInitialized;\r\n    }\r\n\r\n\r\n    /**\r\n     * Aux method to verify that this class is correctly initialized with translation data\r\n     */\r\n    private _validateInitialized(){\r\n\r\n        if(!this._isInitialized){\r\n\r\n            throw new Error('LocalesManager not initialized');\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Checks if the specified locale is currently loaded for the currently defined bundles and paths.\r\n     *\r\n     * @param locale A locale to check. For example 'en_US'\r\n     *\r\n     * @return True if the locale is currently loaded on the class, false if not.\r\n     */\r\n    isLocaleLoaded(locale:string){\r\n\r\n        this._validateLocaleString(locale);\r\n\r\n        return this._locales.includes(locale);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to validate that a locale string is correctly formatted\r\n     *\r\n     * @param string $locale A locale string\r\n     */\r\n    private _validateLocaleString(locale:string){\r\n        \r\n        if(!/^[a-z]{2}_[A-Z]{2}$/.test(locale)) {\r\n\r\n            throw new Error('locale must be a valid xx_XX value');\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Checks if the specified 2 digit language is currently loaded for the currently defined bundles and paths.\r\n     *\r\n     * @param language A language to check. For example 'en'\r\n     *\r\n     * @return True if the language is currently loaded on the class, false if not.\r\n     */\r\n    isLanguageLoaded(language:string){\r\n\r\n        this._validateLanguageString(language);\r\n\r\n        return this._languages.includes(language);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Aux method to validate that a language string is correctly formatted\r\n     *\r\n     * @param language A 2 digit language string\r\n     */\r\n    _validateLanguageString(language:string){\r\n\r\n        if (!/^[a-z]{2}$/.test(language)) {\r\n\r\n            throw new Error('language must be a valid 2 digit value');\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation to the current primary locale for the given key, library and bundle\r\n     *\r\n     * @param string key The key we want to read from the specified resource bundle\r\n     * @param string bundlePath A string with the format 'library_name/bundle_name' that is used to locate the bundle were the key to translate is found\r\n     * @param array replaceWildcards A list of values that will replace wildcards that may be found on the translated text. Each wildcard\r\n     *        will be replaced with the element whose index on replaceWildcards matches it. Check the documentation for this.wildCardsFormat\r\n     *        property to know more about how to setup wildcards on your translations.\r\n     *\r\n     * @see setWildCardsFormat()\r\n     *\r\n     * @return The translated text\r\n     */\r\n    t(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        this._validateInitialized();\r\n\r\n        // Create a cache key to improve performance when requesting the same key translation several times\r\n        const cacheKey = `${this._cacheHashBaseString}${key}${bundlePath}${replaceWildcards.join('')}`;\r\n\r\n        if (!this._keyValuesCache[cacheKey]) {\r\n\r\n            this._forceNonEmptyString(key, '', 'key must be non empty string');\r\n            this._forceNonEmptyString(bundlePath, '', 'bundlePath must be non empty string');\r\n            \r\n            const [library, bundle] = bundlePath.split('/');\r\n\r\n            this._forceNonEmptyString(library, '', 'no library specified on bundlePath');\r\n            this._forceNonEmptyString(bundle, '', 'no bundle specified on bundlePath');\r\n            \r\n            const replacementsCount = replaceWildcards.length;\r\n\r\n            // Loop all the locales to find the first one with a value for the specified key\r\n            for (const locale of this._locales) {\r\n\r\n                if (this._loadedTranslations[library]?.[bundle]?.[locale]?.[key]) {\r\n\r\n                    let result = this._loadedTranslations[library][bundle][locale][key];\r\n\r\n                    // Replace all wildcards on the text with the specified replacements if any\r\n                    for (let i = 0; i < replacementsCount; i++) {\r\n\r\n                        result = this._replace(result, this._replace(this._wildCardsFormat, 'N', i.toString()), replaceWildcards[i]);\r\n                    }\r\n\r\n                    this._keyValuesCache[cacheKey] = result;\r\n                    \r\n                    return result;\r\n                }\r\n            }\r\n\r\n            // Check if an exception needs to be thrown if the specified key is not found on this bundle\r\n            if (this._missingKeyFormat.includes('$exception')) {\r\n                \r\n                throw new Error(`key <${key}> not found on ${bundlePath}`);\r\n            }\r\n\r\n            this._keyValuesCache[cacheKey] = this._replace(this._missingKeyFormat, '$key', key);\r\n        }\r\n\r\n        return this._keyValuesCache[cacheKey];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as a string with all words first character capitalized\r\n     * and all the rest of the word with lower case\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tStartCase(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        return this.t(key, bundlePath, replaceWildcards).split(' ')\r\n                .map((word:any) => word ? word[0].toUpperCase() + word.slice(1).toLowerCase() : '').join(' ');\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as an all upper case string\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tAllUpperCase(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        return this.t(key, bundlePath, replaceWildcards).toUpperCase();\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as an all lower case string\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tAllLowerCase(key:string, bundlePath:string, replaceWildcards:string[] = []) {\r\n\r\n        return this.t(key, bundlePath, replaceWildcards).toLowerCase();\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the translation for the given key and bundle as a string with the first character as Upper case\r\n     * and all the rest as lower case\r\n     *\r\n     * @see t()\r\n     *\r\n     * @returns The localized and case formatted text\r\n     */\r\n    tFirstUpperRestLower(key:string, bundlePath:string, replaceWildcards:string[] = []){\r\n\r\n        const string = this.t(key, bundlePath, replaceWildcards);\r\n        \r\n        return string.charAt(0).toUpperCase() + string.slice(1).toLowerCase();\r\n    }\r\n    \r\n    \r\n    /**\r\n     * A list of strings containing the locales that are used by this class to translate the given keys, sorted by preference.\r\n     * Each string is formatted as a standard locale code with language and country joined by an underscore, like: en_US, fr_FR\r\n     *\r\n     * When a key and bundle are requested for translation, the class will check on the first language of this\r\n     * list for a translated text. If missing, the next one will be used, and so. This list is constructed after initialize\r\n     * methods is called.\r\n     *\r\n     * @example: After loading the following list of locales ['en_US', 'es_ES', 'fr_FR'] if we call LocalesManager.t('HELLO', 'lib1/greetings')\r\n     * the localization manager will try to locate the en_US value for the HELLO tag on the greetings bundle for the library lib1.\r\n     * If the tag is not found for the specified locale and bundle, the same search will be performed for the es_ES locale, and so, till a\r\n     * value is found or no more locales are defined.\r\n     */\r\n    getLocales(){\r\n\r\n        return this._locales;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * A list of strings containing the languages that are used by this class to translate the given keys, sorted by preference.\r\n     * Each string is formatted as a 2 digit language code, like: en, fr\r\n     *\r\n     * This list is the same as the locales() one, but containing only the language part of each locale (the first two digits)\r\n     *\r\n     * @see getLocales()\r\n     */\r\n    getLanguages(){\r\n\r\n        return this._languages;\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the first locale from the list of loaded locales, which is the currently used to search for translated texts.\r\n     *\r\n     * @returns The locale that is defined as the primary one. For example: en_US, es_ES, ..\r\n     */\r\n    getPrimaryLocale(){\r\n\r\n        this._validateInitialized();\r\n\r\n        return this._locales[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Get the first language from the list of loaded locales, which is the currently used to search for translated texts.\r\n     *\r\n     * @returns The 2 digit language code that is defined as the primary one. For example: en, es, ..\r\n     */\r\n    getPrimaryLanguage(){\r\n\r\n        this._validateInitialized();\r\n\r\n        return this._languages[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Define the locale that will be placed at the front of the currently loaded locales list (moving all the others one position to the right).\r\n     *\r\n     * This will be the first locale to use when trying to get a translation.\r\n     *\r\n     * @param locale A currently loaded locale that will be moved to the first position of the loaded locales list. If the specified locale\r\n     *        is not currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLocale(locale:string){\r\n\r\n        this._validateInitialized();\r\n\r\n        if(!this.isLocaleLoaded(locale)){\r\n\r\n            throw new Error(locale + ' not loaded');\r\n        }\r\n\r\n        let result = [locale];\r\n\r\n        for (let l of this._locales){\r\n        \r\n            if(l !== locale){\r\n\r\n                result.push(l);\r\n            }\r\n        }\r\n\r\n        this._locales = result;\r\n        this._languages = this._locales.map((l: string) => l.substring(0, 2));\r\n        this._cacheHashBaseString = this._wildCardsFormat + this._missingKeyFormat + this._locales[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Moves the specified locales to the beginning of the locales list. This also alters the translation priority by setting the first\r\n     * provided locale as the most prioritary, the second as the next one and so.\r\n     *\r\n     * This method basically works exactly the same way as setPrimaryLocale but letting us add many locales at once.\r\n     *\r\n     * @see setPrimaryLocale()\r\n     *\r\n     * @param locales A list of locales to be moved to the beginning of the translation priority. First locales item will be the prefered\r\n     *        locale for translation, second will be the next one in case some key is not translated for the first one and so. If any of the\r\n     *        specified locales is not currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLocales(locales:string[]){\r\n\r\n        if(!Array.isArray(locales) ||\r\n            (new Set(locales).size !== locales.length) ||\r\n            locales.length === 0){\r\n\r\n                throw new Error('locales must be non empty string array with no duplicate elements');\r\n        }\r\n\r\n        for (let i = locales.length - 1; i >= 0; i--) {\r\n\r\n            this.setPrimaryLocale(locales[i]);\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Define the 2 digit language that will be placed at the front of the currently loaded locales list (moving all the others one position to the right).\r\n     *\r\n     * This will be the first language to use when trying to get a translation.\r\n     *\r\n     * @param language A 2 digit language code that matches with any of the currently loaded locales, which will\r\n     *        be moved to the first position of the loaded locales list. If the specified language does not match with\r\n     *        a locale that is currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLanguage(language:string){\r\n\r\n        for(let locale of this._locales){\r\n        \r\n            if(locale.substring(0, 2) === language){\r\n\r\n                this.setPrimaryLocale(locale);\r\n\r\n                return;\r\n            }\r\n        }\r\n\r\n        throw new Error(language + ' not loaded');\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Moves the locales that match the specified languages to the beginning of the locales list.\r\n     * Works the same as setPrimaryLocales() but with a list of the 2 digit language codes that match the respective locales.\r\n     *\r\n     * @see setPrimaryLocale()\r\n     * @see setPrimaryLanguage()\r\n     *\r\n     * @param languages A list of 2 digit language codes to be moved to the beginning of the translation priority. If any of the\r\n     *        specified languages does not match with a locale that is currently loaded, an exception will happen.\r\n     *\r\n     * @returns void\r\n     */\r\n    setPrimaryLanguages(languages:string[]){\r\n\r\n        if(!Array.isArray(languages) ||\r\n           (new Set(languages).size !== languages.length) ||\r\n            languages.length === 0){\r\n\r\n            throw new Error('languages must be non empty string array with no duplicate elements');\r\n        }\r\n\r\n        for (let i = languages.length - 1; i >= 0; i--) {\r\n\r\n            this.setPrimaryLanguage(languages[i]);\r\n        }\r\n    }\r\n    \r\n    \r\n    /**\r\n     * Change the loaded locales translation preference order. The same locales that are currently loaded must be passed\r\n     * but with a different order to change the translation priority.\r\n     *\r\n     * @param locales A list with the new locales translation priority\r\n     *\r\n     * @returns void\r\n     */\r\n    setLocalesOrder(locales:string[]){\r\n\r\n        if(locales.length !== this._locales.length){\r\n\r\n            throw new Error('locales must contain all the currently loaded locales');\r\n        }\r\n\r\n        this._validateInitialized();\r\n\r\n        for(let locale of locales){\r\n        \r\n            if(!this.isLocaleLoaded(locale)){\r\n\r\n                throw new Error(locale + ' not loaded');\r\n            }\r\n        }\r\n\r\n        this._locales = locales;\r\n        this._languages = this._locales.map((l: string) => l.substring(0, 2));\r\n        this._cacheHashBaseString = this._wildCardsFormat + this._missingKeyFormat + this._locales[0];\r\n    }\r\n    \r\n    \r\n    /**\r\n     * This is an aux method to implement the TurboCommons StringUtils replace method.\r\n     * It is exactly the same as the one on the library, but we implement it here to avoid having a dependency with TurboCommons\r\n     */\r\n    private _replace(string: string, search:string, replacement: string) {\r\n        \r\n        const escapedSearch = search.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\r\n           \r\n        return string.replace(new RegExp(escapedSearch, 'g'), replacement);\r\n    }\r\n    \r\n    \r\n    /**\r\n     * This is an aux method to implement the TurboCommons StringUtils isEmpty method.\r\n     * It is exactly the same as the one on the library, but we implement it here to avoid having a dependency with TurboCommons\r\n     */\r\n    private _isEmpty(string:string) {\r\n        \r\n        let isString = (typeof string === 'string' || (string as any) instanceof String);\r\n                \r\n        // Throw exception if non string value was received\r\n        if(!isString){\r\n\r\n            // Empty or null value is considered empty\r\n            if(string == null || string == ''){\r\n    \r\n                return true;\r\n            }\r\n            \r\n            throw new Error(\"value is not a string\");\r\n        }\r\n\r\n        return string.replace(/[ \\n\\r\\t]/g, '') === '';\r\n    }\r\n    \r\n    \r\n    /**\r\n     * This is an aux method to implement the TurboCommons StringUtils forceNonEmptyString method.\r\n     * It is exactly the same as the one on the library, but we implement it here to avoid having a dependency with TurboCommons\r\n     */\r\n    private _forceNonEmptyString(value:any, valueName = '', errorMessage = 'must be a non empty string'){\r\n\r\n        let isString = (typeof value === 'string' || value instanceof String);\r\n        \r\n        if(!isString || this._isEmpty(value)){\r\n\r\n            throw new Error(valueName + ' ' + errorMessage);\r\n        }\r\n    }\r\n}\r\n"]}