ng-easycommerce-v18 0.3.17-beta.1 → 0.3.17-beta.3

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.
package/README.md CHANGED
@@ -1,3 +1,5 @@
1
+ # Version 0.3.17-beta.2
2
+ - Ya estaba contemplado isVisible en el componente `menu-ec.component`, pero se adapto para que devuelva solo las categorias, atributos y secciones visibles y no tener que filtrarlos en el frontend.
1
3
  # Version 0.3.17-beta.1
2
4
  - Se agrego el isVisible de las categorias a su interfaz
3
5
  # Version 0.3.15
@@ -5,6 +7,10 @@
5
7
  - Compatibilidad completa con mobile y desktop: cierre automático de pestaña o redirección según navegador.
6
8
  - Nuevo control de estado visual (idle / pagando / finalizando) para evitar repeticiones o confusión del usuario.
7
9
  - Simplificación de `catch` (RedsysCatchEcComponent) y limpieza de código legacy.
10
+ # Version 0.3.14-beta.2
11
+ - Pruebas para errores en console al desplegar
12
+ # Version 0.3.14-beta.1
13
+ - Pruebas para errores en console al desplegar
8
14
  # Version 0.3.13
9
15
  - Se hace estable 0.3.12-beta.1.
10
16
  - Se hace estable feature/pop-up-inicial.
@@ -1,3 +1,28 @@
1
+ /**
2
+ * Función auxiliar para remover acentos de forma segura para SSR
3
+ */
4
+ function safeRemoveAccents(str) {
5
+ if (typeof window !== 'undefined' && typeof String.prototype.normalize === 'function') {
6
+ return str.normalize("NFD").replace(/[\u0300-\u036f]/g, "");
7
+ }
8
+ // Fallback para SSR - remover acentos manualmente
9
+ return str.replace(/[àáâãäå]/g, 'a')
10
+ .replace(/[èéêë]/g, 'e')
11
+ .replace(/[ìíîï]/g, 'i')
12
+ .replace(/[òóôõö]/g, 'o')
13
+ .replace(/[ùúûü]/g, 'u')
14
+ .replace(/[ýÿ]/g, 'y')
15
+ .replace(/[ñ]/g, 'n')
16
+ .replace(/[ç]/g, 'c')
17
+ .replace(/[ÀÁÂÃÄÅ]/g, 'A')
18
+ .replace(/[ÈÉÊË]/g, 'E')
19
+ .replace(/[ÌÍÎÏ]/g, 'I')
20
+ .replace(/[ÒÓÔÕÖ]/g, 'O')
21
+ .replace(/[ÙÚÛÜ]/g, 'U')
22
+ .replace(/[ÝŸ]/g, 'Y')
23
+ .replace(/[Ñ]/g, 'N')
24
+ .replace(/[Ç]/g, 'C');
25
+ }
1
26
  export class Filter {
2
27
  data = [];
3
28
  multi = false;
@@ -17,7 +42,7 @@ export class Filter {
17
42
  throw new Error("Method not implemented.");
18
43
  }
19
44
  removeAccents = (str) => {
20
- return str.normalize("NFD").replace(/[\u0300-\u036f]/g, "");
45
+ return safeRemoveAccents(str);
21
46
  };
22
47
  setSelected(element, value) {
23
48
  //console.log(element, value);
@@ -80,4 +105,4 @@ export class Filter {
80
105
  return result;
81
106
  };
82
107
  }
83
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/classes/filters/filter.ts"],"names":[],"mappings":"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","sourcesContent":["import { FilterElement, FilterType, IFilter } from \"../../interfaces\";\n\nexport class Filter implements IFilter  {\n    \n    data: FilterElement[] = [];\n    multi: boolean = false;\n    \n    type(): FilterType {\n        throw new Error(\"Method not implemented.\");\n    }\n    setContent(options: any): void {\n        throw new Error(\"Method not implemented.\");\n    }\n    getContent(): FilterElement[] {\n        throw new Error(\"Method not implemented.\");\n    }\n    toUrlParams(): string {\n        throw new Error(\"Method not implemented.\");\n    }\n    createElement(filter: any, original: any, initialValue?: any) {\n        throw new Error(\"Method not implemented.\");\n    }\n    removeAccents = (str:string) => {\n        return str.normalize(\"NFD\").replace(/[\\u0300-\\u036f]/g, \"\");\n    }\n\n    setSelected(element:any, value?:any){\n        //console.log(element, value);\n        ((element.multi != undefined) && !element.multi) ? this.cleanFilter(this.data, element) : null;\n        this.setSelectedValue(element, value);\n    }\n\n    cleanFilter = (data:any, element_to_change:any) => {\n        //console.log(\"cleanFilter\");\n        element_to_change && data.forEach((element:any) => {\n            element_to_change != element ? element.selected = false : null;\n            (element.children && element.children.length > 0) && this.cleanFilter(element.children, element_to_change);\n        })\n    }\n\n    setSelectedValue(element:any, value?:any, sublist?:any, multi?:any){\n        let elements = sublist || this.data;\n        let index = elements.findIndex((subelement:any) => {\n            return (element.code || element) == subelement.code;\n        });\n        if (index >= 0) {\n            elements[index].selected = elements[index].selected ? false : true;\n            !multi && elements.forEach((child:any, i:number) => {\n                if (i != index) child.selected = false;\n            });\n            return {\n                elements,\n                founded: true\n            };\n        } else {\n            elements.some((elem:any) => elem.type == 'sub') && elements.some(\n                (filter:any) => {\n                    let founded = false;\n                    if (filter.type == 'sub') {\n                        const result = this.setSelectedValue(element, value, filter.children, filter.multi);\n                        if (result && result.founded) {\n                            filter.children = result.elements;\n                            founded = true;\n                        }\n                    }\n                    return founded;\n                }\n            )\n        }\n        return null;\n    };\n\n    getSelectedList = (): any => this.findSelected(this.data, []);\n\n    findSelected = (elements: any[], result:any) => {\n        let element = elements.find(elem => elem.selected);\n        if (element) {\n            result.push(element.code);\n            return result;\n        }\n        elements.forEach(elem => {\n            if (elem.children && elem.children.length) {\n                let new_result = this.findSelected(elem.children, result);\n                if (new_result) result = new_result;\n            }\n        })\n        return result;\n    }\n}\n"]}
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/classes/filters/filter.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,SAAS,iBAAiB,CAAC,GAAW;IAClC,IAAI,OAAO,MAAM,KAAK,WAAW,IAAI,OAAO,MAAM,CAAC,SAAS,CAAC,SAAS,KAAK,UAAU,EAAE,CAAC;QACpF,OAAO,GAAG,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,kBAAkB,EAAE,EAAE,CAAC,CAAC;IAChE,CAAC;IACD,kDAAkD;IAClD,OAAO,GAAG,CAAC,OAAO,CAAC,WAAW,EAAE,GAAG,CAAC;SAC/B,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;SACvB,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;SACvB,OAAO,CAAC,UAAU,EAAE,GAAG,CAAC;SACxB,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;SACvB,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC;SACrB,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;SACpB,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;SACpB,OAAO,CAAC,WAAW,EAAE,GAAG,CAAC;SACzB,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;SACvB,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;SACvB,OAAO,CAAC,UAAU,EAAE,GAAG,CAAC;SACxB,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;SACvB,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC;SACrB,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;SACpB,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;AAC9B,CAAC;AAED,MAAM,OAAO,MAAM;IAEf,IAAI,GAAoB,EAAE,CAAC;IAC3B,KAAK,GAAY,KAAK,CAAC;IAEvB,IAAI;QACA,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;IAC/C,CAAC;IACD,UAAU,CAAC,OAAY;QACnB,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;IAC/C,CAAC;IACD,UAAU;QACN,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;IAC/C,CAAC;IACD,WAAW;QACP,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;IAC/C,CAAC;IACD,aAAa,CAAC,MAAW,EAAE,QAAa,EAAE,YAAkB;QACxD,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;IAC/C,CAAC;IACD,aAAa,GAAG,CAAC,GAAU,EAAE,EAAE;QAC3B,OAAO,iBAAiB,CAAC,GAAG,CAAC,CAAC;IAClC,CAAC,CAAA;IAED,WAAW,CAAC,OAAW,EAAE,KAAU;QAC/B,8BAA8B;QAC9B,CAAC,CAAC,OAAO,CAAC,KAAK,IAAI,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAC/F,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;IAC1C,CAAC;IAED,WAAW,GAAG,CAAC,IAAQ,EAAE,iBAAqB,EAAE,EAAE;QAC9C,6BAA6B;QAC7B,iBAAiB,IAAI,IAAI,CAAC,OAAO,CAAC,CAAC,OAAW,EAAE,EAAE;YAC9C,iBAAiB,IAAI,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,GAAG,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;YAC/D,CAAC,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,QAAQ,EAAE,iBAAiB,CAAC,CAAC;QAC/G,CAAC,CAAC,CAAA;IACN,CAAC,CAAA;IAED,gBAAgB,CAAC,OAAW,EAAE,KAAU,EAAE,OAAY,EAAE,KAAU;QAC9D,IAAI,QAAQ,GAAG,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC;QACpC,IAAI,KAAK,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC,UAAc,EAAE,EAAE;YAC9C,OAAO,CAAC,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,IAAI,UAAU,CAAC,IAAI,CAAC;QACxD,CAAC,CAAC,CAAC;QACH,IAAI,KAAK,IAAI,CAAC,EAAE,CAAC;YACb,QAAQ,CAAC,KAAK,CAAC,CAAC,QAAQ,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;YACnE,CAAC,KAAK,IAAI,QAAQ,CAAC,OAAO,CAAC,CAAC,KAAS,EAAE,CAAQ,EAAE,EAAE;gBAC/C,IAAI,CAAC,IAAI,KAAK;oBAAE,KAAK,CAAC,QAAQ,GAAG,KAAK,CAAC;YAC3C,CAAC,CAAC,CAAC;YACH,OAAO;gBACH,QAAQ;gBACR,OAAO,EAAE,IAAI;aAChB,CAAC;QACN,CAAC;aAAM,CAAC;YACJ,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAQ,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC,IAAI,QAAQ,CAAC,IAAI,CAC5D,CAAC,MAAU,EAAE,EAAE;gBACX,IAAI,OAAO,GAAG,KAAK,CAAC;gBACpB,IAAI,MAAM,CAAC,IAAI,IAAI,KAAK,EAAE,CAAC;oBACvB,MAAM,MAAM,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,KAAK,EAAE,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;oBACpF,IAAI,MAAM,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;wBAC3B,MAAM,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;wBAClC,OAAO,GAAG,IAAI,CAAC;oBACnB,CAAC;gBACL,CAAC;gBACD,OAAO,OAAO,CAAC;YACnB,CAAC,CACJ,CAAA;QACL,CAAC;QACD,OAAO,IAAI,CAAC;IAChB,CAAC;IAAA,CAAC;IAEF,eAAe,GAAG,GAAQ,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;IAE9D,YAAY,GAAG,CAAC,QAAe,EAAE,MAAU,EAAE,EAAE;QAC3C,IAAI,OAAO,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACnD,IAAI,OAAO,EAAE,CAAC;YACV,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;YAC1B,OAAO,MAAM,CAAC;QAClB,CAAC;QACD,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACpB,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACxC,IAAI,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;gBAC1D,IAAI,UAAU;oBAAE,MAAM,GAAG,UAAU,CAAC;YACxC,CAAC;QACL,CAAC,CAAC,CAAA;QACF,OAAO,MAAM,CAAC;IAClB,CAAC,CAAA;CACJ","sourcesContent":["import { FilterElement, FilterType, IFilter } from \"../../interfaces\";\n\n/**\n * Función auxiliar para remover acentos de forma segura para SSR\n */\nfunction safeRemoveAccents(str: string): string {\n    if (typeof window !== 'undefined' && typeof String.prototype.normalize === 'function') {\n        return str.normalize(\"NFD\").replace(/[\\u0300-\\u036f]/g, \"\");\n    }\n    // Fallback para SSR - remover acentos manualmente\n    return str.replace(/[àáâãäå]/g, 'a')\n        .replace(/[èéêë]/g, 'e')\n        .replace(/[ìíîï]/g, 'i')\n        .replace(/[òóôõö]/g, 'o')\n        .replace(/[ùúûü]/g, 'u')\n        .replace(/[ýÿ]/g, 'y')\n        .replace(/[ñ]/g, 'n')\n        .replace(/[ç]/g, 'c')\n        .replace(/[ÀÁÂÃÄÅ]/g, 'A')\n        .replace(/[ÈÉÊË]/g, 'E')\n        .replace(/[ÌÍÎÏ]/g, 'I')\n        .replace(/[ÒÓÔÕÖ]/g, 'O')\n        .replace(/[ÙÚÛÜ]/g, 'U')\n        .replace(/[ÝŸ]/g, 'Y')\n        .replace(/[Ñ]/g, 'N')\n        .replace(/[Ç]/g, 'C');\n}\n\nexport class Filter implements IFilter  {\n    \n    data: FilterElement[] = [];\n    multi: boolean = false;\n    \n    type(): FilterType {\n        throw new Error(\"Method not implemented.\");\n    }\n    setContent(options: any): void {\n        throw new Error(\"Method not implemented.\");\n    }\n    getContent(): FilterElement[] {\n        throw new Error(\"Method not implemented.\");\n    }\n    toUrlParams(): string {\n        throw new Error(\"Method not implemented.\");\n    }\n    createElement(filter: any, original: any, initialValue?: any) {\n        throw new Error(\"Method not implemented.\");\n    }\n    removeAccents = (str:string) => {\n        return safeRemoveAccents(str);\n    }\n\n    setSelected(element:any, value?:any){\n        //console.log(element, value);\n        ((element.multi != undefined) && !element.multi) ? this.cleanFilter(this.data, element) : null;\n        this.setSelectedValue(element, value);\n    }\n\n    cleanFilter = (data:any, element_to_change:any) => {\n        //console.log(\"cleanFilter\");\n        element_to_change && data.forEach((element:any) => {\n            element_to_change != element ? element.selected = false : null;\n            (element.children && element.children.length > 0) && this.cleanFilter(element.children, element_to_change);\n        })\n    }\n\n    setSelectedValue(element:any, value?:any, sublist?:any, multi?:any){\n        let elements = sublist || this.data;\n        let index = elements.findIndex((subelement:any) => {\n            return (element.code || element) == subelement.code;\n        });\n        if (index >= 0) {\n            elements[index].selected = elements[index].selected ? false : true;\n            !multi && elements.forEach((child:any, i:number) => {\n                if (i != index) child.selected = false;\n            });\n            return {\n                elements,\n                founded: true\n            };\n        } else {\n            elements.some((elem:any) => elem.type == 'sub') && elements.some(\n                (filter:any) => {\n                    let founded = false;\n                    if (filter.type == 'sub') {\n                        const result = this.setSelectedValue(element, value, filter.children, filter.multi);\n                        if (result && result.founded) {\n                            filter.children = result.elements;\n                            founded = true;\n                        }\n                    }\n                    return founded;\n                }\n            )\n        }\n        return null;\n    };\n\n    getSelectedList = (): any => this.findSelected(this.data, []);\n\n    findSelected = (elements: any[], result:any) => {\n        let element = elements.find(elem => elem.selected);\n        if (element) {\n            result.push(element.code);\n            return result;\n        }\n        elements.forEach(elem => {\n            if (elem.children && elem.children.length) {\n                let new_result = this.findSelected(elem.children, result);\n                if (new_result) result = new_result;\n            }\n        })\n        return result;\n    }\n}\n"]}
@@ -1,92 +1,86 @@
1
- import { inject, Injectable } from '@angular/core';
1
+ import { inject, Injectable, PLATFORM_ID, Inject } from '@angular/core';
2
+ import { isPlatformServer } from '@angular/common';
2
3
  import { ENVIRONMENT_TOKEN } from '../providers';
3
4
  import { LocalStorageService } from '../ec-services';
4
5
  import { TranslateService } from '@ngx-translate/core';
5
6
  import * as i0 from "@angular/core";
6
7
  /**
7
- * Servicio que provee de datos que estan relacionado con las peticiones a la API
8
- * @export
9
- * @class ApiConstantsService
8
+ * Servicio que provee de datos relacionados con las peticiones a la API
10
9
  */
11
10
  export class ApiConstantsService {
11
+ ssrApiUrl;
12
12
  _localStorage = inject(LocalStorageService);
13
13
  _translate = inject(TranslateService);
14
- /**
15
- * Contiene los datos provisto por el frontend en el archivo environment.ts
16
- */
17
14
  environment = inject(ENVIRONMENT_TOKEN);
15
+ platformId = inject(PLATFORM_ID);
16
+ _channel;
17
+ LOCALE;
18
+ SHOP_API_URL = 'shop-api/';
19
+ CMS_URL = 'cms/';
20
+ constructor(ssrApiUrl) {
21
+ this.ssrApiUrl = ssrApiUrl;
22
+ this._channel = this.environment.channel;
23
+ this.LOCALE = this.environment.locale;
24
+ }
18
25
  /**
19
26
  * Canal actual del frontend
20
27
  */
21
28
  get CHANNEL() {
22
- // Verificar si estamos en el navegador (no en SSR)
23
29
  if (typeof window !== 'undefined') {
24
- // Primero intenta leer de window.__env (configurado por Docker)
25
30
  const windowEnv = window.__env;
26
- if (windowEnv?.channel) {
31
+ if (windowEnv?.channel)
27
32
  return windowEnv.channel;
28
- }
29
33
  }
30
- // Fallback al environment
31
34
  return this._channel;
32
35
  }
33
36
  set CHANNEL(value) {
34
37
  this._channel = value;
35
38
  }
36
- _channel;
37
- /**
38
- * Locale actual del frontend
39
- */
40
- LOCALE;
41
- /**
42
- * URL para las peticiones a shop-api
43
- */
44
- SHOP_API_URL = 'shop-api/';
45
- /**
46
- * URL para las peticiones a cms
47
- */
48
- CMS_URL = 'cms/';
49
- constructor() {
50
- this._channel = this.environment.channel;
51
- this.LOCALE = this.environment.locale;
52
- }
53
39
  /**
54
40
  * URL del backend para realizar las peticiones
55
41
  */
56
42
  get API_URL() {
57
- // Verificar si estamos en el navegador (no en SSR)
43
+ // Browser runtime-config
58
44
  if (typeof window !== 'undefined') {
59
- // Primero intenta leer de window.__env (configurado por Docker)
60
45
  const windowEnv = window.__env;
61
- if (windowEnv?.apiUrl) {
62
- return windowEnv.apiUrl;
63
- }
46
+ if (windowEnv?.apiBaseUrl)
47
+ return windowEnv.apiBaseUrl;
48
+ }
49
+ // SSR
50
+ if (isPlatformServer(this.platformId)) {
51
+ return this.ssrApiUrl;
64
52
  }
65
- // Fallback al environment (para SSR y como backup)
53
+ // Fallback environment
66
54
  return this.environment.apiUrl ?? '';
67
55
  }
68
56
  /**
69
- * Retorna la url base
70
- * @returns {string}
57
+ * URL base completa para shop-api con channel y locale
58
+ */
59
+ getShopApiBase() {
60
+ const channel = this.CHANNEL || 'undefined';
61
+ const locale = this.LOCALE || 'undefined';
62
+ return `${this.API_URL}/${this.SHOP_API_URL}${channel}/?locale=${locale}`;
63
+ }
64
+ /**
65
+ * Retorna la url base general (sin shop-api)
71
66
  */
72
67
  getUrlBase() {
73
68
  return this.API_URL;
74
69
  }
75
70
  /**
76
71
  * Cambia el canal actual
77
- * @param code
78
- * @returns
79
72
  */
80
- setChannel(code) { this.CHANNEL = code; }
73
+ setChannel(code) {
74
+ this.CHANNEL = code;
75
+ }
81
76
  setLocale(locale) {
82
77
  this._localStorage.setItem(this.LOCALE_KEY, locale);
83
78
  this._translate.use(locale.split('_')[0]);
84
79
  this.LOCALE = locale;
85
80
  }
86
- //Storage key
87
81
  LOCALE_KEY = 'LOCALE';
88
82
  CHANNEL_KEY = 'CHANNEL';
89
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ApiConstantsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
83
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ApiConstantsService, deps: [{ token: 'API_URL' }], target: i0.ɵɵFactoryTarget.Injectable });
90
84
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ApiConstantsService, providedIn: 'root' });
91
85
  }
92
86
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ApiConstantsService, decorators: [{
@@ -94,5 +88,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
94
88
  args: [{
95
89
  providedIn: 'root'
96
90
  }]
97
- }], ctorParameters: () => [] });
98
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmNvbnN0YW50cy5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctZWFzeWNvbW1lcmNlLXYxOC9zcmMvbGliL2NvbnN0YW50cy9hcGkuY29uc3RhbnRzLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkQsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sY0FBYyxDQUFDO0FBRWpELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3JELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDOztBQUd2RDs7OztHQUlHO0FBSUgsTUFBTSxPQUFPLG1CQUFtQjtJQUN2QixhQUFhLEdBQXdCLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBQ2pFLFVBQVUsR0FBcUIsTUFBTSxDQUFDLGdCQUFnQixDQUFDLENBQUM7SUFFaEU7O09BRUc7SUFDSyxXQUFXLEdBQWdCLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO0lBRTdEOztPQUVHO0lBQ0gsSUFBVyxPQUFPO1FBQ2pCLG1EQUFtRDtRQUNuRCxJQUFJLE9BQU8sTUFBTSxLQUFLLFdBQVcsRUFBRSxDQUFDO1lBQ25DLGdFQUFnRTtZQUNoRSxNQUFNLFNBQVMsR0FBSSxNQUFjLENBQUMsS0FBSyxDQUFDO1lBQ3hDLElBQUksU0FBUyxFQUFFLE9BQU8sRUFBRSxDQUFDO2dCQUN4QixPQUFPLFNBQVMsQ0FBQyxPQUFPLENBQUM7WUFDMUIsQ0FBQztRQUNGLENBQUM7UUFFRCwwQkFBMEI7UUFDMUIsT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDO0lBQ3RCLENBQUM7SUFFRCxJQUFXLE9BQU8sQ0FBQyxLQUFhO1FBQy9CLElBQUksQ0FBQyxRQUFRLEdBQUcsS0FBSyxDQUFDO0lBQ3ZCLENBQUM7SUFFTyxRQUFRLENBQVM7SUFDekI7O09BRUc7SUFDSSxNQUFNLENBQVM7SUFFdEI7O09BRUc7SUFDYSxZQUFZLEdBQVcsV0FBVyxDQUFDO0lBQ25EOztPQUVHO0lBQ2EsT0FBTyxHQUFXLE1BQU0sQ0FBQztJQUV6QztRQUNDLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxPQUFPLENBQUM7UUFDekMsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQztJQUN2QyxDQUFDO0lBRUQ7O09BRUc7SUFDSCxJQUFXLE9BQU87UUFDakIsbURBQW1EO1FBQ25ELElBQUksT0FBTyxNQUFNLEtBQUssV0FBVyxFQUFFLENBQUM7WUFDbkMsZ0VBQWdFO1lBQ2hFLE1BQU0sU0FBUyxHQUFJLE1BQWMsQ0FBQyxLQUFLLENBQUM7WUFDeEMsSUFBSSxTQUFTLEVBQUUsTUFBTSxFQUFFLENBQUM7Z0JBQ3ZCLE9BQU8sU0FBUyxDQUFDLE1BQU0sQ0FBQztZQUN6QixDQUFDO1FBQ0YsQ0FBQztRQUVELG1EQUFtRDtRQUNuRCxPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxJQUFJLEVBQUUsQ0FBQztJQUN0QyxDQUFDO0lBQ0Q7OztPQUdHO0lBQ0ksVUFBVTtRQUNoQixPQUFPLElBQUksQ0FBQyxPQUFPLENBQUM7SUFDckIsQ0FBQztJQUNEOzs7O09BSUc7SUFDSSxVQUFVLENBQUMsSUFBWSxJQUFRLElBQUksQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDLENBQUEsQ0FBQztJQUVwRCxTQUFTLENBQUMsTUFBYztRQUM5QixJQUFJLENBQUMsYUFBYSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsVUFBVSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBQ3BELElBQUksQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQTtRQUN6QyxJQUFJLENBQUMsTUFBTSxHQUFHLE1BQU0sQ0FBQztJQUN0QixDQUFDO0lBRUQsYUFBYTtJQUNJLFVBQVUsR0FBVSxRQUFRLENBQUM7SUFDN0IsV0FBVyxHQUFVLFNBQVMsQ0FBQzt3R0F4RnBDLG1CQUFtQjs0R0FBbkIsbUJBQW1CLGNBRmxCLE1BQU07OzRGQUVQLG1CQUFtQjtrQkFIL0IsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpbmplY3QsIEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEVOVklST05NRU5UX1RPS0VOIH0gZnJvbSAnLi4vcHJvdmlkZXJzJztcbmltcG9ydCB7IEVudmlyb25tZW50IH0gZnJvbSAnLi4vaW50ZXJmYWNlcyc7XG5pbXBvcnQgeyBMb2NhbFN0b3JhZ2VTZXJ2aWNlIH0gZnJvbSAnLi4vZWMtc2VydmljZXMnO1xuaW1wb3J0IHsgVHJhbnNsYXRlU2VydmljZSB9IGZyb20gJ0BuZ3gtdHJhbnNsYXRlL2NvcmUnO1xuaW1wb3J0IHsgQ29yZUNvbnN0YW50c1NlcnZpY2UgfSBmcm9tICcuL2NvcmUuY29uc3RhbnRzLnNlcnZpY2UnO1xuXG4vKipcbiAqIFNlcnZpY2lvIHF1ZSBwcm92ZWUgZGUgZGF0b3MgcXVlIGVzdGFuIHJlbGFjaW9uYWRvIGNvbiBsYXMgcGV0aWNpb25lcyBhIGxhIEFQSVxuICogQGV4cG9ydFxuICogQGNsYXNzIEFwaUNvbnN0YW50c1NlcnZpY2VcbiAqL1xuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgQXBpQ29uc3RhbnRzU2VydmljZSB7XG5cdHByaXZhdGUgX2xvY2FsU3RvcmFnZTogTG9jYWxTdG9yYWdlU2VydmljZSA9IGluamVjdChMb2NhbFN0b3JhZ2VTZXJ2aWNlKTtcblx0cHJpdmF0ZSBfdHJhbnNsYXRlOiBUcmFuc2xhdGVTZXJ2aWNlID0gaW5qZWN0KFRyYW5zbGF0ZVNlcnZpY2UpO1xuXG5cdC8qKlxuXHQgKiBDb250aWVuZSBsb3MgZGF0b3MgcHJvdmlzdG8gcG9yIGVsIGZyb250ZW5kIGVuIGVsIGFyY2hpdm8gZW52aXJvbm1lbnQudHNcblx0ICovXG5cdHByaXZhdGUgZW52aXJvbm1lbnQ6IEVudmlyb25tZW50ID0gaW5qZWN0KEVOVklST05NRU5UX1RPS0VOKTtcblx0XG5cdC8qKlxuXHQgKiBDYW5hbCBhY3R1YWwgZGVsIGZyb250ZW5kXG5cdCAqL1xuXHRwdWJsaWMgZ2V0IENIQU5ORUwoKTogc3RyaW5nIHtcblx0XHQvLyBWZXJpZmljYXIgc2kgZXN0YW1vcyBlbiBlbCBuYXZlZ2Fkb3IgKG5vIGVuIFNTUilcblx0XHRpZiAodHlwZW9mIHdpbmRvdyAhPT0gJ3VuZGVmaW5lZCcpIHtcblx0XHRcdC8vIFByaW1lcm8gaW50ZW50YSBsZWVyIGRlIHdpbmRvdy5fX2VudiAoY29uZmlndXJhZG8gcG9yIERvY2tlcilcblx0XHRcdGNvbnN0IHdpbmRvd0VudiA9ICh3aW5kb3cgYXMgYW55KS5fX2Vudjtcblx0XHRcdGlmICh3aW5kb3dFbnY/LmNoYW5uZWwpIHtcblx0XHRcdFx0cmV0dXJuIHdpbmRvd0Vudi5jaGFubmVsO1xuXHRcdFx0fVxuXHRcdH1cblx0XHRcblx0XHQvLyBGYWxsYmFjayBhbCBlbnZpcm9ubWVudFxuXHRcdHJldHVybiB0aGlzLl9jaGFubmVsO1xuXHR9XG5cdFxuXHRwdWJsaWMgc2V0IENIQU5ORUwodmFsdWU6IHN0cmluZykge1xuXHRcdHRoaXMuX2NoYW5uZWwgPSB2YWx1ZTtcblx0fVxuXHRcblx0cHJpdmF0ZSBfY2hhbm5lbDogc3RyaW5nO1xuXHQvKipcblx0ICogTG9jYWxlIGFjdHVhbCBkZWwgZnJvbnRlbmRcblx0ICovXG5cdHB1YmxpYyBMT0NBTEU6IHN0cmluZztcblx0XG5cdC8qKlxuXHQgKiBVUkwgcGFyYSBsYXMgcGV0aWNpb25lcyBhIHNob3AtYXBpXG5cdCAqL1xuXHRwdWJsaWMgcmVhZG9ubHkgU0hPUF9BUElfVVJMOiBzdHJpbmcgPSAnc2hvcC1hcGkvJztcblx0LyoqXG5cdCAqIFVSTCBwYXJhIGxhcyBwZXRpY2lvbmVzIGEgY21zXG5cdCAqL1xuXHRwdWJsaWMgcmVhZG9ubHkgQ01TX1VSTDogc3RyaW5nID0gJ2Ntcy8nO1xuXG5cdGNvbnN0cnVjdG9yKCkge1xuXHRcdHRoaXMuX2NoYW5uZWwgPSB0aGlzLmVudmlyb25tZW50LmNoYW5uZWw7XG5cdFx0dGhpcy5MT0NBTEUgPSB0aGlzLmVudmlyb25tZW50LmxvY2FsZTtcblx0fVxuXG5cdC8qKlxuXHQgKiBVUkwgZGVsIGJhY2tlbmQgcGFyYSByZWFsaXphciBsYXMgcGV0aWNpb25lc1xuXHQgKi9cblx0cHVibGljIGdldCBBUElfVVJMKCk6IHN0cmluZyB7XG5cdFx0Ly8gVmVyaWZpY2FyIHNpIGVzdGFtb3MgZW4gZWwgbmF2ZWdhZG9yIChubyBlbiBTU1IpXG5cdFx0aWYgKHR5cGVvZiB3aW5kb3cgIT09ICd1bmRlZmluZWQnKSB7XG5cdFx0XHQvLyBQcmltZXJvIGludGVudGEgbGVlciBkZSB3aW5kb3cuX19lbnYgKGNvbmZpZ3VyYWRvIHBvciBEb2NrZXIpXG5cdFx0XHRjb25zdCB3aW5kb3dFbnYgPSAod2luZG93IGFzIGFueSkuX19lbnY7XG5cdFx0XHRpZiAod2luZG93RW52Py5hcGlVcmwpIHtcblx0XHRcdFx0cmV0dXJuIHdpbmRvd0Vudi5hcGlVcmw7XG5cdFx0XHR9XG5cdFx0fVxuXHRcdFxuXHRcdC8vIEZhbGxiYWNrIGFsIGVudmlyb25tZW50IChwYXJhIFNTUiB5IGNvbW8gYmFja3VwKVxuXHRcdHJldHVybiB0aGlzLmVudmlyb25tZW50LmFwaVVybCA/PyAnJztcblx0fVxuXHQvKipcblx0ICogUmV0b3JuYSBsYSB1cmwgYmFzZVxuXHQgKiBAcmV0dXJucyB7c3RyaW5nfVxuXHQgKi9cblx0cHVibGljIGdldFVybEJhc2UoKTogc3RyaW5nIHsgXG5cdFx0cmV0dXJuIHRoaXMuQVBJX1VSTDtcblx0fSBcblx0LyoqXG5cdCAqIENhbWJpYSBlbCBjYW5hbCBhY3R1YWxcblx0ICogQHBhcmFtIGNvZGVcblx0ICogQHJldHVybnMgXG5cdCAqL1xuXHRwdWJsaWMgc2V0Q2hhbm5lbChjb2RlOiBzdHJpbmcpOnZvaWR7IHRoaXMuQ0hBTk5FTCA9IGNvZGU7fSBcblxuXHRwdWJsaWMgc2V0TG9jYWxlKGxvY2FsZTogc3RyaW5nKTp2b2lke1xuXHRcdHRoaXMuX2xvY2FsU3RvcmFnZS5zZXRJdGVtKHRoaXMuTE9DQUxFX0tFWSwgbG9jYWxlKTtcblx0XHR0aGlzLl90cmFuc2xhdGUudXNlKGxvY2FsZS5zcGxpdCgnXycpWzBdKVxuXHRcdHRoaXMuTE9DQUxFID0gbG9jYWxlO1xuXHR9XG5cblx0Ly9TdG9yYWdlIGtleVxuXHRwcml2YXRlIHJlYWRvbmx5IExPQ0FMRV9LRVk6c3RyaW5nID0gJ0xPQ0FMRSc7XG5cdHByaXZhdGUgcmVhZG9ubHkgQ0hBTk5FTF9LRVk6c3RyaW5nID0gJ0NIQU5ORUwnO1xufVxuIl19
91
+ }], ctorParameters: () => [{ type: undefined, decorators: [{
92
+ type: Inject,
93
+ args: ['API_URL']
94
+ }] }] });
95
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmNvbnN0YW50cy5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctZWFzeWNvbW1lcmNlLXYxOC9zcmMvbGliL2NvbnN0YW50cy9hcGkuY29uc3RhbnRzLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxVQUFVLEVBQUUsV0FBVyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN4RSxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUNuRCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFFakQsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDckQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUM7O0FBRXZEOztHQUVHO0FBSUgsTUFBTSxPQUFPLG1CQUFtQjtJQWFTO0lBWi9CLGFBQWEsR0FBd0IsTUFBTSxDQUFDLG1CQUFtQixDQUFDLENBQUM7SUFDakUsVUFBVSxHQUFxQixNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztJQUV4RCxXQUFXLEdBQWdCLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO0lBQ3JELFVBQVUsR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7SUFFakMsUUFBUSxDQUFTO0lBQ2xCLE1BQU0sQ0FBUztJQUVOLFlBQVksR0FBVyxXQUFXLENBQUM7SUFDbkMsT0FBTyxHQUFXLE1BQU0sQ0FBQztJQUV6QyxZQUF1QyxTQUFpQjtRQUFqQixjQUFTLEdBQVQsU0FBUyxDQUFRO1FBQ3RELElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxPQUFPLENBQUM7UUFDekMsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQztJQUN4QyxDQUFDO0lBRUQ7O09BRUc7SUFDSCxJQUFXLE9BQU87UUFDaEIsSUFBSSxPQUFPLE1BQU0sS0FBSyxXQUFXLEVBQUUsQ0FBQztZQUNsQyxNQUFNLFNBQVMsR0FBSSxNQUFjLENBQUMsS0FBSyxDQUFDO1lBQ3hDLElBQUksU0FBUyxFQUFFLE9BQU87Z0JBQUUsT0FBTyxTQUFTLENBQUMsT0FBTyxDQUFDO1FBQ25ELENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxRQUFRLENBQUM7SUFDdkIsQ0FBQztJQUVELElBQVcsT0FBTyxDQUFDLEtBQWE7UUFDOUIsSUFBSSxDQUFDLFFBQVEsR0FBRyxLQUFLLENBQUM7SUFDeEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsSUFBVyxPQUFPO1FBQ2hCLHlCQUF5QjtRQUN6QixJQUFJLE9BQU8sTUFBTSxLQUFLLFdBQVcsRUFBRSxDQUFDO1lBQ2xDLE1BQU0sU0FBUyxHQUFJLE1BQWMsQ0FBQyxLQUFLLENBQUM7WUFDeEMsSUFBSSxTQUFTLEVBQUUsVUFBVTtnQkFBRSxPQUFPLFNBQVMsQ0FBQyxVQUFVLENBQUM7UUFDekQsQ0FBQztRQUVELE1BQU07UUFDTixJQUFJLGdCQUFnQixDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsRUFBRSxDQUFDO1lBQ3RDLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQztRQUN4QixDQUFDO1FBRUQsdUJBQXVCO1FBQ3ZCLE9BQU8sSUFBSSxDQUFDLFdBQVcsQ0FBQyxNQUFNLElBQUksRUFBRSxDQUFDO0lBQ3ZDLENBQUM7SUFFRDs7T0FFRztJQUNJLGNBQWM7UUFDbkIsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLE9BQU8sSUFBSSxXQUFXLENBQUM7UUFDNUMsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sSUFBSSxXQUFXLENBQUM7UUFDMUMsT0FBTyxHQUFHLElBQUksQ0FBQyxPQUFPLElBQUksSUFBSSxDQUFDLFlBQVksR0FBRyxPQUFPLFlBQVksTUFBTSxFQUFFLENBQUM7SUFDNUUsQ0FBQztJQUVEOztPQUVHO0lBQ0ksVUFBVTtRQUNmLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQztJQUN0QixDQUFDO0lBRUQ7O09BRUc7SUFDSSxVQUFVLENBQUMsSUFBWTtRQUM1QixJQUFJLENBQUMsT0FBTyxHQUFHLElBQUksQ0FBQztJQUN0QixDQUFDO0lBRU0sU0FBUyxDQUFDLE1BQWM7UUFDN0IsSUFBSSxDQUFDLGFBQWEsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLFVBQVUsRUFBRSxNQUFNLENBQUMsQ0FBQztRQUNwRCxJQUFJLENBQUMsVUFBVSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDMUMsSUFBSSxDQUFDLE1BQU0sR0FBRyxNQUFNLENBQUM7SUFDdkIsQ0FBQztJQUVnQixVQUFVLEdBQVcsUUFBUSxDQUFDO0lBQzlCLFdBQVcsR0FBVyxTQUFTLENBQUM7d0dBbEZ0QyxtQkFBbUIsa0JBYVYsU0FBUzs0R0FibEIsbUJBQW1CLGNBRmxCLE1BQU07OzRGQUVQLG1CQUFtQjtrQkFIL0IsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkI7OzBCQWNjLE1BQU07MkJBQUMsU0FBUyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGluamVjdCwgSW5qZWN0YWJsZSwgUExBVEZPUk1fSUQsIEluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgaXNQbGF0Zm9ybVNlcnZlciB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBFTlZJUk9OTUVOVF9UT0tFTiB9IGZyb20gJy4uL3Byb3ZpZGVycyc7XG5pbXBvcnQgeyBFbnZpcm9ubWVudCB9IGZyb20gJy4uL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgTG9jYWxTdG9yYWdlU2VydmljZSB9IGZyb20gJy4uL2VjLXNlcnZpY2VzJztcbmltcG9ydCB7IFRyYW5zbGF0ZVNlcnZpY2UgfSBmcm9tICdAbmd4LXRyYW5zbGF0ZS9jb3JlJztcblxuLyoqXG4gKiBTZXJ2aWNpbyBxdWUgcHJvdmVlIGRlIGRhdG9zIHJlbGFjaW9uYWRvcyBjb24gbGFzIHBldGljaW9uZXMgYSBsYSBBUElcbiAqL1xuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgQXBpQ29uc3RhbnRzU2VydmljZSB7XG4gIHByaXZhdGUgX2xvY2FsU3RvcmFnZTogTG9jYWxTdG9yYWdlU2VydmljZSA9IGluamVjdChMb2NhbFN0b3JhZ2VTZXJ2aWNlKTtcbiAgcHJpdmF0ZSBfdHJhbnNsYXRlOiBUcmFuc2xhdGVTZXJ2aWNlID0gaW5qZWN0KFRyYW5zbGF0ZVNlcnZpY2UpO1xuXG4gIHByaXZhdGUgZW52aXJvbm1lbnQ6IEVudmlyb25tZW50ID0gaW5qZWN0KEVOVklST05NRU5UX1RPS0VOKTtcbiAgcHJpdmF0ZSBwbGF0Zm9ybUlkID0gaW5qZWN0KFBMQVRGT1JNX0lEKTtcblxuICBwcml2YXRlIF9jaGFubmVsOiBzdHJpbmc7XG4gIHB1YmxpYyBMT0NBTEU6IHN0cmluZztcblxuICBwdWJsaWMgcmVhZG9ubHkgU0hPUF9BUElfVVJMOiBzdHJpbmcgPSAnc2hvcC1hcGkvJztcbiAgcHVibGljIHJlYWRvbmx5IENNU19VUkw6IHN0cmluZyA9ICdjbXMvJztcblxuICBjb25zdHJ1Y3RvcihASW5qZWN0KCdBUElfVVJMJykgcHJpdmF0ZSBzc3JBcGlVcmw6IHN0cmluZykge1xuICAgIHRoaXMuX2NoYW5uZWwgPSB0aGlzLmVudmlyb25tZW50LmNoYW5uZWw7XG4gICAgdGhpcy5MT0NBTEUgPSB0aGlzLmVudmlyb25tZW50LmxvY2FsZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYW5hbCBhY3R1YWwgZGVsIGZyb250ZW5kXG4gICAqL1xuICBwdWJsaWMgZ2V0IENIQU5ORUwoKTogc3RyaW5nIHtcbiAgICBpZiAodHlwZW9mIHdpbmRvdyAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgIGNvbnN0IHdpbmRvd0VudiA9ICh3aW5kb3cgYXMgYW55KS5fX2VudjtcbiAgICAgIGlmICh3aW5kb3dFbnY/LmNoYW5uZWwpIHJldHVybiB3aW5kb3dFbnYuY2hhbm5lbDtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuX2NoYW5uZWw7XG4gIH1cblxuICBwdWJsaWMgc2V0IENIQU5ORUwodmFsdWU6IHN0cmluZykge1xuICAgIHRoaXMuX2NoYW5uZWwgPSB2YWx1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBVUkwgZGVsIGJhY2tlbmQgcGFyYSByZWFsaXphciBsYXMgcGV0aWNpb25lc1xuICAgKi9cbiAgcHVibGljIGdldCBBUElfVVJMKCk6IHN0cmluZyB7XG4gICAgLy8gQnJvd3NlciBydW50aW1lLWNvbmZpZ1xuICAgIGlmICh0eXBlb2Ygd2luZG93ICE9PSAndW5kZWZpbmVkJykge1xuICAgICAgY29uc3Qgd2luZG93RW52ID0gKHdpbmRvdyBhcyBhbnkpLl9fZW52O1xuICAgICAgaWYgKHdpbmRvd0Vudj8uYXBpQmFzZVVybCkgcmV0dXJuIHdpbmRvd0Vudi5hcGlCYXNlVXJsO1xuICAgIH1cblxuICAgIC8vIFNTUlxuICAgIGlmIChpc1BsYXRmb3JtU2VydmVyKHRoaXMucGxhdGZvcm1JZCkpIHtcbiAgICAgIHJldHVybiB0aGlzLnNzckFwaVVybDtcbiAgICB9XG5cbiAgICAvLyBGYWxsYmFjayBlbnZpcm9ubWVudFxuICAgIHJldHVybiB0aGlzLmVudmlyb25tZW50LmFwaVVybCA/PyAnJztcbiAgfVxuXG4gIC8qKlxuICAgKiBVUkwgYmFzZSBjb21wbGV0YSBwYXJhIHNob3AtYXBpIGNvbiBjaGFubmVsIHkgbG9jYWxlXG4gICAqL1xuICBwdWJsaWMgZ2V0U2hvcEFwaUJhc2UoKTogc3RyaW5nIHtcbiAgICBjb25zdCBjaGFubmVsID0gdGhpcy5DSEFOTkVMIHx8ICd1bmRlZmluZWQnO1xuICAgIGNvbnN0IGxvY2FsZSA9IHRoaXMuTE9DQUxFIHx8ICd1bmRlZmluZWQnO1xuICAgIHJldHVybiBgJHt0aGlzLkFQSV9VUkx9LyR7dGhpcy5TSE9QX0FQSV9VUkx9JHtjaGFubmVsfS8/bG9jYWxlPSR7bG9jYWxlfWA7XG4gIH1cblxuICAvKipcbiAgICogUmV0b3JuYSBsYSB1cmwgYmFzZSBnZW5lcmFsIChzaW4gc2hvcC1hcGkpXG4gICAqL1xuICBwdWJsaWMgZ2V0VXJsQmFzZSgpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLkFQSV9VUkw7XG4gIH1cblxuICAvKipcbiAgICogQ2FtYmlhIGVsIGNhbmFsIGFjdHVhbFxuICAgKi9cbiAgcHVibGljIHNldENoYW5uZWwoY29kZTogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5DSEFOTkVMID0gY29kZTtcbiAgfVxuXG4gIHB1YmxpYyBzZXRMb2NhbGUobG9jYWxlOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLl9sb2NhbFN0b3JhZ2Uuc2V0SXRlbSh0aGlzLkxPQ0FMRV9LRVksIGxvY2FsZSk7XG4gICAgdGhpcy5fdHJhbnNsYXRlLnVzZShsb2NhbGUuc3BsaXQoJ18nKVswXSk7XG4gICAgdGhpcy5MT0NBTEUgPSBsb2NhbGU7XG4gIH1cblxuICBwcml2YXRlIHJlYWRvbmx5IExPQ0FMRV9LRVk6IHN0cmluZyA9ICdMT0NBTEUnO1xuICBwcml2YXRlIHJlYWRvbmx5IENIQU5ORUxfS0VZOiBzdHJpbmcgPSAnQ0hBTk5FTCc7XG59XG4iXX0=
@@ -117,6 +117,22 @@ export class MenuEcComponent {
117
117
  hasVisibleProperty(category) {
118
118
  return category.isVisible === true;
119
119
  }
120
+ filterVisibleTree(list) {
121
+ return (list ?? [])
122
+ .filter((n) => n?.isVisible === true)
123
+ .map(n => ({ ...n, children: this.filterVisibleTree(n.children) }));
124
+ }
125
+ // Exponé streams ya filtrados
126
+ categoriesVisible$ = this.categories$.pipe(map((list) => this.filterVisibleTree(list)));
127
+ sectionsVisible$ = this.sections$.pipe(map((list) => this.filterVisibleTree(list)));
128
+ attributesVisible$ = this.attributes$.pipe(map((list) => this.filterVisibleTree(list)));
129
+ // Helpers de conveniencia opcionales
130
+ getVisibleChildren(node) {
131
+ return this.filterVisibleTree(node?.children);
132
+ }
133
+ hasVisibleChildren(node) {
134
+ return this.getVisibleChildren(node).length > 0;
135
+ }
120
136
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MenuEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
121
137
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MenuEcComponent, isStandalone: true, selector: "lib-footer-ec", ngImport: i0, template: '<p>Menu and Footer Helper Component</p>', isInline: true });
122
138
  }
@@ -129,4 +145,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
129
145
  template: '<p>Menu and Footer Helper Component</p>'
130
146
  }]
131
147
  }], ctorParameters: () => [] });
132
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/abstractions-components/menu-ec.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, afterRender, Component, inject } from '@angular/core';\nimport { AuthService, OptionsService, ParametersService } from '../../ec-services';\nimport { map, Observable } from 'rxjs';\nimport { CoreConstantsService } from '../../constants';\nimport { Attribute, Category, Parameter, Section } from '../../interfaces';\nimport { ToastService } from '../../ec-services/toast.service';\n/**\n * Para diferenciar los tipos de Menu\n */\ntype MenuType = \"category\" | \"attribute\" | \"section\"\n/**\n * Componente que sirve para abstraer la funcionalidad que comparten \n * el menú y el footer.\n * @class MenuEcComponent\n */\n@Component({\n\tselector: 'lib-footer-ec',\n\tstandalone: true,\n\timports: [],\n\ttemplate: '<p>Menu and Footer Helper Component</p>'\n})\nexport class MenuEcComponent {\n\tprivate optionsService = inject(OptionsService)\n\tprivate parametersService = inject(ParametersService)\n\tprivate _authService: AuthService = inject(AuthService)\n\tprivate _toastService: ToastService = inject(ToastService)\n\t/**\n\t * Constantes del core.\n\t */\n\tprivate consts = inject(CoreConstantsService)\n\n\t/**\n\t * Observable de categorias\n\t */\n\tpublic categories$: Observable<Category[]> = this.optionsService.getCategories()\n\t/**\n\t * Observable de secciones\n\t */\n\tpublic sections$: Observable<Section[]> = this.optionsService.getSections()\n\t/**\n\t * Observable de atributos\n\t */\n\tpublic attributes$: Observable<Attribute[]> = this.optionsService.getAttributes()\n\t/**\n\t * Observable de parametros\n\t */\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters()\n\t/**\n\t * Url para las imagenes guardades en el backend.\n\t */\n\tpublic mediaUrl = this.consts.mediaUrl()\n\t/**\n\t * Funcion para saber si existen un parametro. \n\t */\n\tpublic hasParams = this.parametersService.hasParams\n\n\tconstructor() { }\n\n\tprotected addChildren = (menuType: MenuType, code: string, item: any, stackPlace = 'end'): void => {\n\n\t}\n\n\t/* protected addChildren = (menuItems: any[], code: string, item: any, stackPlace = 'end') => {\n\t  (stackPlace === 'start')\n\t\t? menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.unshift(item)\n\t\t: menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.push(item)\n    \n\t} */\n\t/**\n\t * Actualiza las categorias agregando nuevas.\n\t * @param categories \n\t */\n\tprotected updateCategories(categories: Category[]) {\n\t\tthis.categories$ = this.categories$.pipe(map((data: Category[]) => {\n\t\t\treturn [...data, ...categories]\n\t\t}))\n\t}\n\t/**\n\t * Actualiza las secciones agregando nuevas.\n\t * @param sections \n\t */\n\tprotected updateSections(sections: Section[]) {\n\t\tthis.sections$ = this.sections$.pipe(map((data: Section[]) => {\n\t\t\treturn [...data, ...sections]\n\t\t}))\n\t}\n\t/**\n\t * Actualiza el menú de atributos agredando nuevos.\n\t * @param attributes \n\t */\n\tprotected updateAttributes(attributes: Attribute[]) {\n\t\tthis.attributes$ = this.attributes$.pipe(map((data: Attribute[]) => {\n\t\t\treturn [...data, ...attributes]\n\t\t}))\n\t}\n\n\tpublic isAuthenticated() {\n\t\treturn this._authService.isAuthenticated();\n\t}\n\n\tpublic logout() {\n\t\tthis._authService.logout();\n\t\tthis._toastService.show('logout-success');\n\t\twindow.location.href = '/home'; // Redirige a la página de inicio después del logout\n\t}\n\n\t/**\n\t * Añade varios hijos a una sección por code.\n\t * @param sectionCode Código de la sección\n\t * @param children Array de hijos a agregar\n\t * @param stackPlace 'start' | 'end' (por defecto 'end')\n\t */\n\tpublic addSectionChildren(sectionCode: string, children: any[], stackPlace: 'start' | 'end' = 'end') {\n\t\tthis.sections$ = this.sections$.pipe(\n\t\t\tmap((sections: Section[]) => {\n\t\t\t\treturn sections.map(section => {\n\t\t\t\t\tif (section.code?.toLowerCase() === sectionCode.toLowerCase()) {\n\t\t\t\t\t\tif (!section.children) section.children = [];\n\t\t\t\t\t\tif (stackPlace === 'start') {\n\t\t\t\t\t\t\tsection.children = [...children, ...section.children];\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tsection.children = [...section.children, ...children];\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\treturn section;\n\t\t\t\t});\n\t\t\t})\n\t\t);\n\t}\n\t/**\n   * Verifica si una categoría tiene la propiedad isVisible y está marcada como visible\n   * @param category - La categoría a verificar\n   * @returns true si la categoría es visible, false en caso contrario\n   */\n\thasVisibleProperty(category: Category): boolean {\n\t\treturn (category as any).isVisible === true;\n\t}\n}"]}
148
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/abstractions-components/menu-ec.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, afterRender, Component, inject } from '@angular/core';\nimport { AuthService, OptionsService, ParametersService } from '../../ec-services';\nimport { map, Observable } from 'rxjs';\nimport { CoreConstantsService } from '../../constants';\nimport { Attribute, Category, Parameter, Section } from '../../interfaces';\nimport { ToastService } from '../../ec-services/toast.service';\n/**\n * Para diferenciar los tipos de Menu\n */\ntype MenuType = \"category\" | \"attribute\" | \"section\"\ntype TreeNode<T> = T & { isVisible?: boolean; children?: T[] };\n/**\n * Componente que sirve para abstraer la funcionalidad que comparten \n * el menú y el footer.\n * @class MenuEcComponent\n */\n@Component({\n\tselector: 'lib-footer-ec',\n\tstandalone: true,\n\timports: [],\n\ttemplate: '<p>Menu and Footer Helper Component</p>'\n})\nexport class MenuEcComponent {\n\tprivate optionsService = inject(OptionsService)\n\tprivate parametersService = inject(ParametersService)\n\tprivate _authService: AuthService = inject(AuthService)\n\tprivate _toastService: ToastService = inject(ToastService)\n\t/**\n\t * Constantes del core.\n\t */\n\tprivate consts = inject(CoreConstantsService)\n\n\t/**\n\t * Observable de categorias\n\t */\n\tpublic categories$: Observable<Category[]> = this.optionsService.getCategories()\n\t/**\n\t * Observable de secciones\n\t */\n\tpublic sections$: Observable<Section[]> = this.optionsService.getSections()\n\t/**\n\t * Observable de atributos\n\t */\n\tpublic attributes$: Observable<Attribute[]> = this.optionsService.getAttributes()\n\t/**\n\t * Observable de parametros\n\t */\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters()\n\t/**\n\t * Url para las imagenes guardades en el backend.\n\t */\n\tpublic mediaUrl = this.consts.mediaUrl()\n\t/**\n\t * Funcion para saber si existen un parametro. \n\t */\n\tpublic hasParams = this.parametersService.hasParams\n\n\tconstructor() { }\n\n\tprotected addChildren = (menuType: MenuType, code: string, item: any, stackPlace = 'end'): void => {\n\n\t}\n\n\t/* protected addChildren = (menuItems: any[], code: string, item: any, stackPlace = 'end') => {\n\t  (stackPlace === 'start')\n\t\t? menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.unshift(item)\n\t\t: menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.push(item)\n    \n\t} */\n\t/**\n\t * Actualiza las categorias agregando nuevas.\n\t * @param categories \n\t */\n\tprotected updateCategories(categories: Category[]) {\n\t\tthis.categories$ = this.categories$.pipe(map((data: Category[]) => {\n\t\t\treturn [...data, ...categories]\n\t\t}))\n\t}\n\t/**\n\t * Actualiza las secciones agregando nuevas.\n\t * @param sections \n\t */\n\tprotected updateSections(sections: Section[]) {\n\t\tthis.sections$ = this.sections$.pipe(map((data: Section[]) => {\n\t\t\treturn [...data, ...sections]\n\t\t}))\n\t}\n\t/**\n\t * Actualiza el menú de atributos agredando nuevos.\n\t * @param attributes \n\t */\n\tprotected updateAttributes(attributes: Attribute[]) {\n\t\tthis.attributes$ = this.attributes$.pipe(map((data: Attribute[]) => {\n\t\t\treturn [...data, ...attributes]\n\t\t}))\n\t}\n\n\tpublic isAuthenticated() {\n\t\treturn this._authService.isAuthenticated();\n\t}\n\n\tpublic logout() {\n\t\tthis._authService.logout();\n\t\tthis._toastService.show('logout-success');\n\t\twindow.location.href = '/home'; // Redirige a la página de inicio después del logout\n\t}\n\n\t/**\n\t * Añade varios hijos a una sección por code.\n\t * @param sectionCode Código de la sección\n\t * @param children Array de hijos a agregar\n\t * @param stackPlace 'start' | 'end' (por defecto 'end')\n\t */\n\tpublic addSectionChildren(sectionCode: string, children: any[], stackPlace: 'start' | 'end' = 'end') {\n\t\tthis.sections$ = this.sections$.pipe(\n\t\t\tmap((sections: Section[]) => {\n\t\t\t\treturn sections.map(section => {\n\t\t\t\t\tif (section.code?.toLowerCase() === sectionCode.toLowerCase()) {\n\t\t\t\t\t\tif (!section.children) section.children = [];\n\t\t\t\t\t\tif (stackPlace === 'start') {\n\t\t\t\t\t\t\tsection.children = [...children, ...section.children];\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tsection.children = [...section.children, ...children];\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\treturn section;\n\t\t\t\t});\n\t\t\t})\n\t\t);\n\t}\n\t/**\n   * Verifica si una categoría tiene la propiedad isVisible y está marcada como visible\n   * @param category - La categoría a verificar\n   * @returns true si la categoría es visible, false en caso contrario\n   */\n\thasVisibleProperty(category: Category): boolean {\n\t\treturn (category as any).isVisible === true;\n\t}\n\n\tprivate filterVisibleTree<T extends { isVisible?: boolean; children?: T[] }>(list?: T[]): T[] {\n\t\treturn (list ?? [])\n\t\t\t.filter((n): n is T => n?.isVisible === true)\n\t\t\t.map(n => ({ ...n, children: this.filterVisibleTree(n.children) } as T));\n\t}\n\n\t// Exponé streams ya filtrados\n\tpublic categoriesVisible$: Observable<Category[]> = this.categories$.pipe(\n\t\tmap((list: Category[]) => this.filterVisibleTree<Category>(list))\n\t);\n\n\tpublic sectionsVisible$: Observable<Section[]> = this.sections$.pipe(\n\t\tmap((list: Section[]) => this.filterVisibleTree<Section>(list))\n\t);\n\n\tpublic attributesVisible$: Observable<Attribute[]> = this.attributes$.pipe(\n\t\tmap((list: Attribute[]) => this.filterVisibleTree<Attribute>(list))\n\t);\n\n\t// Helpers de conveniencia opcionales\n\tpublic getVisibleChildren<T extends { isVisible?: boolean; children?: T[] }>(node?: T): T[] {\n\t\treturn this.filterVisibleTree(node?.children);\n\t}\n\n\tpublic hasVisibleChildren<T extends { isVisible?: boolean; children?: T[] }>(node?: T): boolean {\n\t\treturn this.getVisibleChildren(node).length > 0;\n\t}\n}"]}
@@ -1,7 +1,7 @@
1
1
  import { Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal } from '@angular/core';
2
2
  import { BlockEcComponent } from '../../abstractions-components';
3
3
  import { AnalyticsService } from '../../../ec-services';
4
- import { CommonModule } from '@angular/common';
4
+ import { CommonModule, isPlatformBrowser } from '@angular/common';
5
5
  // import function to register Swiper custom elements
6
6
  import { register } from 'swiper/element/bundle';
7
7
  import { ProductEcComponent } from "../../product-ec/product-ec.component";
@@ -66,7 +66,7 @@ export class BlockProductsEcComponent extends BlockEcComponent {
66
66
  * Permite personalización de las imágenes de las flechas mediante @Input.
67
67
  */
68
68
  setupSwiperNavigation() {
69
- if (this.meta?.styles?.carrousel !== false) {
69
+ if (this.meta?.styles?.carrousel !== false && isPlatformBrowser(this.platformId)) {
70
70
  // Usar setTimeout para asegurar que el swiper esté inicializado
71
71
  setTimeout(() => {
72
72
  this.initializeSwiperWithCustomNavigation();
@@ -78,6 +78,8 @@ export class BlockProductsEcComponent extends BlockEcComponent {
78
78
  * Esta función puede ser movida al componente base para reutilización.
79
79
  */
80
80
  initializeSwiperWithCustomNavigation() {
81
+ if (!isPlatformBrowser(this.platformId))
82
+ return;
81
83
  const prevButton = document.getElementById(`${this.meta?.code}-prev`);
82
84
  const nextButton = document.getElementById(`${this.meta?.code}-next`);
83
85
  const swiperElement = document.getElementById(this.meta?.code);
@@ -216,4 +218,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
216
218
  required: true
217
219
  }]
218
220
  }] } });
219
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block-products-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, AfterViewInit, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal, TemplateRef } from '@angular/core';\nimport { BlockEcComponent } from '../../abstractions-components';\nimport { SwiperOptions } from 'swiper/types';\nimport { AnalyticsService } from '../../../ec-services';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\n// import function to register Swiper custom elements\nimport { register, SwiperContainer } from 'swiper/element/bundle';\nimport { ProductEcComponent } from \"../../product-ec/product-ec.component\";\n\nregister()\n/**\n * Componen para manejar los bloques de productos.\n * @extends {BlockEcComponent}\n * @class BlockProductsEcComponent\n */\n@Component({\n\tselector: 'app-block-products-ec',\n\tstandalone: true,\n\timports: [CommonModule, ProductEcComponent],\n\ttemplateUrl: './block-products-ec.component.html',\n\tstyleUrl: './block-products-ec.component.scss',\n\tschemas: [CUSTOM_ELEMENTS_SCHEMA]\n})\nexport class BlockProductsEcComponent extends BlockEcComponent implements AfterViewInit {\n\t\t// Personalización de las flechas de navegación\n\t// Por defecto usa símbolos de texto, pero se pueden especificar imágenes\n\t@Input() prevArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() nextArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() prevArrowText: string = '<';\n\t@Input() nextArrowText: string = '>';\n\t/**\n\t * Servicio de Analytics\n\t */\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\n\t/**\n\t * Signal utlizado para guarda el contenedor del carrusel\n\t */\n\tswiperElement = signal<SwiperContainer | null>(null)\n\t/**\n\t * Input que recibe un template para el producto.\n\t */\n\t@Input() appProduct: TemplateRef<any> | undefined;\n\t/**\n\t * Colección de productos.\n\t */\n\t@Input({\n\t\trequired: true\n\t}) products: any; // Product[];\n\t/**\n\t * Bloque principal que contiene los productos\n\t */\n\t@Input({\n\t\trequired: true\n\t}) meta: any;\n\n\tngAfterViewInit() {\n\t\tthis.setupSwiperNavigation();\n\t}\n\t\n\tprivate document?: Document;\n\tprivate platformId: any = inject(PLATFORM_ID)\n\n\t/**\n\t * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias \n\t * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente\n\t * en el Dom el element `<swiper-container>` para poder configurarlo.\n\t */\n\tconstructor() {\n\t\tsuper()\n\t}\n\t/**\n\t * Aplica el evento `select_promotion` junto con el banner que interactua.\n\t * @param banner \n\t */\n\tselectPromotion(item: any): void {\n\t\tthis.analyticsService.callEvent('select_promotion', item)\n\t}\n\n\t/**\n\t * Configura la navegación personalizada del Swiper.\n\t * Esta función está diseñada para ser movida al componente base BlockProductsEcComponent.\n\t * Permite personalización de las imágenes de las flechas mediante @Input.\n\t */\n\tprivate setupSwiperNavigation() {\n\t\tif (this.meta?.styles?.carrousel !== false) {\n\t\t\t// Usar setTimeout para asegurar que el swiper esté inicializado\n\t\t\tsetTimeout(() => {\n\t\t\t\tthis.initializeSwiperWithCustomNavigation();\n\t\t\t}, 200);\n\t\t}\n\t}\n\n\t/**\n\t * Inicializa el Swiper con navegación personalizada.\n\t * Esta función puede ser movida al componente base para reutilización.\n\t */\n\tprivate initializeSwiperWithCustomNavigation() {\n\t\tconst prevButton = document.getElementById(`${this.meta?.code}-prev`);\n\t\tconst nextButton = document.getElementById(`${this.meta?.code}-next`);\n\t\tconst swiperElement = document.getElementById(this.meta?.code) as any;\n\n\t\tif (prevButton && nextButton && swiperElement) {\n\t\t\t// console.log('Configurando navegación personalizada para:', this.meta?.code);\n\t\t\t\n\t\t\tconst swiperConfig = this.getSwiperConfiguration();\n\n\t\t\t// Verificar si el Swiper ya está inicializado\n\t\t\tif (!swiperElement.swiper) {\n\t\t\t\tthis.initializeNewSwiper(swiperElement, swiperConfig);\n\t\t\t} else {\n\t\t\t\tthis.updateExistingSwiper(swiperElement);\n\t\t\t}\n\n\t\t\t// Configurar los event listeners para los botones\n\t\t\tthis.setupNavigationEventListeners(prevButton, nextButton, swiperElement);\n\t\t\t\n\t\t\t// console.log('Event listeners configurados para los botones de navegación');\n\t\t} else {\n\t\t\t// console.log('No se pudieron encontrar los elementos:', {\n\t\t\t// \tprevButton: !!prevButton,\n\t\t\t// \tnextButton: !!nextButton,\n\t\t\t// \tswiperElement: !!swiperElement\n\t\t\t// });\n\t\t}\n\t}\n\n\t/**\n\t * Obtiene la configuración base del Swiper.\n\t * Esta configuración puede ser personalizada en el futuro mediante @Input.\n\t */\n\tprivate getSwiperConfiguration() {\n\t\treturn {\n\t\t\tslidesPerView: 'auto',\n\t\t\tspaceBetween: 16,\n\t\t\tslidesPerGroup: 1, // Importante: moverse de uno en uno\n\t\t\tnavigation: false, // Deshabilitamos la navegación por defecto\n\t\t\tpagination: false,\n\t\t\tloop: true,\n\t\t\tgrabCursor: true,\n\t\t\tautoplay: true,\n\t\t\tbreakpoints: {\n\t\t\t\t320: {\n\t\t\t\t\tslidesPerView: 1,\n\t\t\t\t\tspaceBetween: 8,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t576: {\n\t\t\t\t\tslidesPerView: 2,\n\t\t\t\t\tspaceBetween: 12,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t768: {\n\t\t\t\t\tslidesPerView: 3,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t1024: {\n\t\t\t\t\tslidesPerView: 4,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * Inicializa un nuevo Swiper con la configuración proporcionada.\n\t */\n\tprivate initializeNewSwiper(swiperElement: any, config: any) {\n\t\t// Asignar configuración al elemento\n\t\tObject.assign(swiperElement, config);\n\t\t\n\t\t// Inicializar el Swiper\n\t\tswiperElement.initialize();\n\t\t\n\t\t// console.log('Swiper inicializado con configuración personalizada');\n\t}\n\n\t/**\n\t * Actualiza un Swiper existente para asegurar la configuración correcta.\n\t */\n\tprivate updateExistingSwiper(swiperElement: any) {\n\t\t// console.log('Actualizando configuración de Swiper existente');\n\t\t\n\t\t// Forzar slidesPerGroup a 1\n\t\tswiperElement.swiper.params.slidesPerGroup = 1;\n\t\tswiperElement.swiper.allowSlideNext = true;\n\t\tswiperElement.swiper.allowSlidePrev = true;\n\t\t\n\t\t// Actualizar también los breakpoints si existen\n\t\tif (swiperElement.swiper.params.breakpoints) {\n\t\t\tObject.keys(swiperElement.swiper.params.breakpoints).forEach(key => {\n\t\t\t\tswiperElement.swiper.params.breakpoints[key].slidesPerGroup = 1;\n\t\t\t});\n\t\t}\n\t\t\n\t\tswiperElement.swiper.update();\n\t}\n\n\t/**\n\t * Configura los event listeners para los botones de navegación.\n\t */\n\tprivate setupNavigationEventListeners(prevButton: Element, nextButton: Element, swiperElement: any) {\n\t\tprevButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón anterior');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slidePrev();\n\t\t\t}\n\t\t});\n\n\t\tnextButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón siguiente');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slideNext();\n\t\t\t}\n\t\t});\n\t}\n}\n","<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\n\n    <div class=\"blockProduct block-product\">\n        @if(meta.name){\n        <div class=\"row\">\n            <div class=\"col-12 mt-4\">\n                <h2  class=\"font-weight-normal font-gd\">\n                    <span>{{meta.name}}</span>\n                </h2>\n            </div>\n        </div>\n        }\n\n\n        @if(meta.styles && meta.styles.carrousel == false){\n        <div class=\"row \">\n            @for (product of products; track $index) {\n            <div [class]=\"'item '+ ' col-'+ (meta.styles.items?.sm) + ' col-md-' + (meta.styles.items?.md)  + ' col-lg-' + (meta.styles.items?.lg)  + ' px-2'\" [id]=\"$index\">\n                <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\n                <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n            </div>\n            }\n        </div>\n        } @else {\n        <div class=\"container position-relative\">\n            <swiper-container \n                init=\"false\" \n                [id]=\"meta?.code\"\n                slides-per-view=\"auto\"\n                space-between=\"16\"\n                slides-per-group=\"1\"\n                navigation=\"false\"\n                pagination=\"false\"\n                loop=\"false\">\n                @for (product of products; track $index) {\n                <swiper-slide id=\"swiper-slide\">\n                    <ng-container\n                        *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n                </swiper-slide>\n                }\n            </swiper-container>\n            \n            <!-- Botones de navegación personalizados -->\n            <div class=\"swiper-navigation\">\n                <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\n                    @if(prevArrowImage) {\n                        <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{prevArrowText}}</span>\n                    }\n                </div>\n                <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\n                    @if(nextArrowImage) {\n                        <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{nextArrowText}}</span>\n                    }\n                </div>\n            </div>\n        </div>\n        }\n    </div>\n</section>\n\n\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\n<ng-template #defaultAppProduct let-product=\"product\">\n    <app-product-ec [product]=\"product\"></app-product-ec>\n</ng-template>"]}
221
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block-products-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, AfterViewInit, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal, TemplateRef } from '@angular/core';\nimport { BlockEcComponent } from '../../abstractions-components';\nimport { SwiperOptions } from 'swiper/types';\nimport { AnalyticsService } from '../../../ec-services';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\n// import function to register Swiper custom elements\nimport { register, SwiperContainer } from 'swiper/element/bundle';\nimport { ProductEcComponent } from \"../../product-ec/product-ec.component\";\n\nregister()\n/**\n * Componen para manejar los bloques de productos.\n * @extends {BlockEcComponent}\n * @class BlockProductsEcComponent\n */\n@Component({\n\tselector: 'app-block-products-ec',\n\tstandalone: true,\n\timports: [CommonModule, ProductEcComponent],\n\ttemplateUrl: './block-products-ec.component.html',\n\tstyleUrl: './block-products-ec.component.scss',\n\tschemas: [CUSTOM_ELEMENTS_SCHEMA]\n})\nexport class BlockProductsEcComponent extends BlockEcComponent implements AfterViewInit {\n\t\t// Personalización de las flechas de navegación\n\t// Por defecto usa símbolos de texto, pero se pueden especificar imágenes\n\t@Input() prevArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() nextArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() prevArrowText: string = '<';\n\t@Input() nextArrowText: string = '>';\n\t/**\n\t * Servicio de Analytics\n\t */\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\n\t/**\n\t * Signal utlizado para guarda el contenedor del carrusel\n\t */\n\tswiperElement = signal<SwiperContainer | null>(null)\n\t/**\n\t * Input que recibe un template para el producto.\n\t */\n\t@Input() appProduct: TemplateRef<any> | undefined;\n\t/**\n\t * Colección de productos.\n\t */\n\t@Input({\n\t\trequired: true\n\t}) products: any; // Product[];\n\t/**\n\t * Bloque principal que contiene los productos\n\t */\n\t@Input({\n\t\trequired: true\n\t}) meta: any;\n\n\tngAfterViewInit() {\n\t\tthis.setupSwiperNavigation();\n\t}\n\t\n\tprivate document?: Document;\n\tprivate platformId: any = inject(PLATFORM_ID)\n\n\t/**\n\t * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias \n\t * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente\n\t * en el Dom el element `<swiper-container>` para poder configurarlo.\n\t */\n\tconstructor() {\n\t\tsuper()\n\t}\n\t/**\n\t * Aplica el evento `select_promotion` junto con el banner que interactua.\n\t * @param banner \n\t */\n\tselectPromotion(item: any): void {\n\t\tthis.analyticsService.callEvent('select_promotion', item)\n\t}\n\n\t/**\n\t * Configura la navegación personalizada del Swiper.\n\t * Esta función está diseñada para ser movida al componente base BlockProductsEcComponent.\n\t * Permite personalización de las imágenes de las flechas mediante @Input.\n\t */\n\tprivate setupSwiperNavigation() {\n\t\tif (this.meta?.styles?.carrousel !== false && isPlatformBrowser(this.platformId)) {\n\t\t\t// Usar setTimeout para asegurar que el swiper esté inicializado\n\t\t\tsetTimeout(() => {\n\t\t\t\tthis.initializeSwiperWithCustomNavigation();\n\t\t\t}, 200);\n\t\t}\n\t}\n\n\t/**\n\t * Inicializa el Swiper con navegación personalizada.\n\t * Esta función puede ser movida al componente base para reutilización.\n\t */\n\tprivate initializeSwiperWithCustomNavigation() {\n\t\tif (!isPlatformBrowser(this.platformId)) return;\n\t\t\n\t\tconst prevButton = document.getElementById(`${this.meta?.code}-prev`);\n\t\tconst nextButton = document.getElementById(`${this.meta?.code}-next`);\n\t\tconst swiperElement = document.getElementById(this.meta?.code) as any;\n\n\t\tif (prevButton && nextButton && swiperElement) {\n\t\t\t// console.log('Configurando navegación personalizada para:', this.meta?.code);\n\t\t\t\n\t\t\tconst swiperConfig = this.getSwiperConfiguration();\n\n\t\t\t// Verificar si el Swiper ya está inicializado\n\t\t\tif (!swiperElement.swiper) {\n\t\t\t\tthis.initializeNewSwiper(swiperElement, swiperConfig);\n\t\t\t} else {\n\t\t\t\tthis.updateExistingSwiper(swiperElement);\n\t\t\t}\n\n\t\t\t// Configurar los event listeners para los botones\n\t\t\tthis.setupNavigationEventListeners(prevButton, nextButton, swiperElement);\n\t\t\t\n\t\t\t// console.log('Event listeners configurados para los botones de navegación');\n\t\t} else {\n\t\t\t// console.log('No se pudieron encontrar los elementos:', {\n\t\t\t// \tprevButton: !!prevButton,\n\t\t\t// \tnextButton: !!nextButton,\n\t\t\t// \tswiperElement: !!swiperElement\n\t\t\t// });\n\t\t}\n\t}\n\n\t/**\n\t * Obtiene la configuración base del Swiper.\n\t * Esta configuración puede ser personalizada en el futuro mediante @Input.\n\t */\n\tprivate getSwiperConfiguration() {\n\t\treturn {\n\t\t\tslidesPerView: 'auto',\n\t\t\tspaceBetween: 16,\n\t\t\tslidesPerGroup: 1, // Importante: moverse de uno en uno\n\t\t\tnavigation: false, // Deshabilitamos la navegación por defecto\n\t\t\tpagination: false,\n\t\t\tloop: true,\n\t\t\tgrabCursor: true,\n\t\t\tautoplay: true,\n\t\t\tbreakpoints: {\n\t\t\t\t320: {\n\t\t\t\t\tslidesPerView: 1,\n\t\t\t\t\tspaceBetween: 8,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t576: {\n\t\t\t\t\tslidesPerView: 2,\n\t\t\t\t\tspaceBetween: 12,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t768: {\n\t\t\t\t\tslidesPerView: 3,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t1024: {\n\t\t\t\t\tslidesPerView: 4,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * Inicializa un nuevo Swiper con la configuración proporcionada.\n\t */\n\tprivate initializeNewSwiper(swiperElement: any, config: any) {\n\t\t// Asignar configuración al elemento\n\t\tObject.assign(swiperElement, config);\n\t\t\n\t\t// Inicializar el Swiper\n\t\tswiperElement.initialize();\n\t\t\n\t\t// console.log('Swiper inicializado con configuración personalizada');\n\t}\n\n\t/**\n\t * Actualiza un Swiper existente para asegurar la configuración correcta.\n\t */\n\tprivate updateExistingSwiper(swiperElement: any) {\n\t\t// console.log('Actualizando configuración de Swiper existente');\n\t\t\n\t\t// Forzar slidesPerGroup a 1\n\t\tswiperElement.swiper.params.slidesPerGroup = 1;\n\t\tswiperElement.swiper.allowSlideNext = true;\n\t\tswiperElement.swiper.allowSlidePrev = true;\n\t\t\n\t\t// Actualizar también los breakpoints si existen\n\t\tif (swiperElement.swiper.params.breakpoints) {\n\t\t\tObject.keys(swiperElement.swiper.params.breakpoints).forEach(key => {\n\t\t\t\tswiperElement.swiper.params.breakpoints[key].slidesPerGroup = 1;\n\t\t\t});\n\t\t}\n\t\t\n\t\tswiperElement.swiper.update();\n\t}\n\n\t/**\n\t * Configura los event listeners para los botones de navegación.\n\t */\n\tprivate setupNavigationEventListeners(prevButton: Element, nextButton: Element, swiperElement: any) {\n\t\tprevButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón anterior');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slidePrev();\n\t\t\t}\n\t\t});\n\n\t\tnextButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón siguiente');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slideNext();\n\t\t\t}\n\t\t});\n\t}\n}\n","<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\n\n    <div class=\"blockProduct block-product\">\n        @if(meta.name){\n        <div class=\"row\">\n            <div class=\"col-12 mt-4\">\n                <h2  class=\"font-weight-normal font-gd\">\n                    <span>{{meta.name}}</span>\n                </h2>\n            </div>\n        </div>\n        }\n\n\n        @if(meta.styles && meta.styles.carrousel == false){\n        <div class=\"row \">\n            @for (product of products; track $index) {\n            <div [class]=\"'item '+ ' col-'+ (meta.styles.items?.sm) + ' col-md-' + (meta.styles.items?.md)  + ' col-lg-' + (meta.styles.items?.lg)  + ' px-2'\" [id]=\"$index\">\n                <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\n                <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n            </div>\n            }\n        </div>\n        } @else {\n        <div class=\"container position-relative\">\n            <swiper-container \n                init=\"false\" \n                [id]=\"meta?.code\"\n                slides-per-view=\"auto\"\n                space-between=\"16\"\n                slides-per-group=\"1\"\n                navigation=\"false\"\n                pagination=\"false\"\n                loop=\"false\">\n                @for (product of products; track $index) {\n                <swiper-slide id=\"swiper-slide\">\n                    <ng-container\n                        *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n                </swiper-slide>\n                }\n            </swiper-container>\n            \n            <!-- Botones de navegación personalizados -->\n            <div class=\"swiper-navigation\">\n                <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\n                    @if(prevArrowImage) {\n                        <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{prevArrowText}}</span>\n                    }\n                </div>\n                <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\n                    @if(nextArrowImage) {\n                        <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{nextArrowText}}</span>\n                    }\n                </div>\n            </div>\n        </div>\n        }\n    </div>\n</section>\n\n\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\n<ng-template #defaultAppProduct let-product=\"product\">\n    <app-product-ec [product]=\"product\"></app-product-ec>\n</ng-template>"]}