ng-easycommerce-v18 0.1.4 → 0.1.5

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.
@@ -406,5 +406,6 @@
406
406
  "login-error": "Error al iniciar sesión. Por favor, intente nuevamente más tarde.",
407
407
  "login-success": "Inicio de sesión exitoso",
408
408
  "logout-success": "Sesión cerrada exitosamente",
409
+ "quantity-not-exceeded": "La cantidad mínima de compra es {{amount}}",
409
410
  "login-form-incomplete": "El formulario de inicio de sesión está incompleto o contiene datos inválidos."
410
411
  }
@@ -84,6 +84,7 @@ export class MenuEcComponent {
84
84
  logout() {
85
85
  this._authService.logout();
86
86
  this._toastService.show('logout-success');
87
+ window.location.href = '/home'; // Redirige a la página de inicio después del logout
87
88
  }
88
89
  /**
89
90
  * Añade varios hijos a una sección por code.
@@ -120,4 +121,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
120
121
  template: '<p>Menu and Footer Helper Component</p>'
121
122
  }]
122
123
  }], ctorParameters: () => [] });
123
- //# 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';\r\nimport { AuthService, OptionsService, ParametersService } from '../../ec-services';\r\nimport { map, Observable } from 'rxjs';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { Attribute, Category, Parameter, Section } from '../../interfaces';\r\nimport { ToastService } from '../../ec-services/toast.service';\r\n/**\r\n * Para diferenciar los tipos de Menu\r\n */\r\ntype MenuType = \"category\" | \"attribute\" | \"section\"\r\n/**\r\n * Componente que sirve para abstraer la funcionalidad que comparten \r\n * el menú y el footer.\r\n * @class MenuEcComponent\r\n */\r\n@Component({\r\n\tselector: 'lib-footer-ec',\r\n\tstandalone: true,\r\n\timports: [],\r\n\ttemplate: '<p>Menu and Footer Helper Component</p>'\r\n})\r\nexport class MenuEcComponent {\r\n\tprivate optionsService = inject(OptionsService)\r\n\tprivate parametersService = inject(ParametersService)\r\n\tprivate _authService: AuthService = inject(AuthService)\r\n\tprivate _toastService: ToastService = inject(ToastService)\r\n\t/**\r\n\t * Constantes del core.\r\n\t */\r\n\tprivate consts = inject(CoreConstantsService)\r\n\r\n\t/**\r\n\t * Observable de categorias\r\n\t */\r\n\tpublic categories$: Observable<Category[]> = this.optionsService.getCategories()\r\n\t/**\r\n\t * Observable de secciones\r\n\t */\r\n\tpublic sections$: Observable<Section[]> = this.optionsService.getSections()\r\n\t/**\r\n\t * Observable de atributos\r\n\t */\r\n\tpublic attributes$: Observable<Attribute[]> = this.optionsService.getAttributes()\r\n\t/**\r\n\t * Observable de parametros\r\n\t */\r\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters()\r\n\t/**\r\n\t * Url para las imagenes guardades en el backend.\r\n\t */\r\n\tpublic mediaUrl = this.consts.mediaUrl()\r\n\t/**\r\n\t * Funcion para saber si existen un parametro. \r\n\t */\r\n\tpublic hasParams = this.parametersService.hasParams\r\n\r\n\tconstructor() {}\r\n\r\n\tprotected addChildren = (menuType: MenuType, code: string, item: any, stackPlace = 'end'): void => {\r\n\r\n\t}\r\n\r\n\t/* protected addChildren = (menuItems: any[], code: string, item: any, stackPlace = 'end') => {\r\n\t  (stackPlace === 'start')\r\n\t\t? menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.unshift(item)\r\n\t\t: menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.push(item)\r\n    \r\n\t} */\r\n\t/**\r\n\t * Actualiza las categorias agregando nuevas.\r\n\t * @param categories \r\n\t */\r\n\tprotected updateCategories(categories: Category[]) {\r\n\t\tthis.categories$ = this.categories$.pipe(map((data: Category[]) => {\r\n\t\t\treturn [...data, ...categories]\r\n\t\t}))\r\n\t}\r\n\t/**\r\n\t * Actualiza las secciones agregando nuevas.\r\n\t * @param sections \r\n\t */\r\n\tprotected updateSections(sections: Section[]) {\r\n\t\tthis.sections$ = this.sections$.pipe(map((data: Section[]) => {\r\n\t\t\treturn [...data, ...sections]\r\n\t\t}))\r\n\t}\r\n\t/**\r\n\t * Actualiza el menú de atributos agredando nuevos.\r\n\t * @param attributes \r\n\t */\r\n\tprotected updateAttributes(attributes: Attribute[]) {\r\n\t\tthis.attributes$ = this.attributes$.pipe(map((data: Attribute[]) => {\r\n\t\t\treturn [...data, ...attributes]\r\n\t\t}))\r\n\t}\r\n\r\n\tpublic isAuthenticated() {\r\n\t\treturn this._authService.isAuthenticated();\r\n\t}\r\n\r\n\tpublic logout() {\r\n\t\tthis._authService.logout();\r\n\t\tthis._toastService.show('logout-success');\r\n\t}\r\n\r\n\t/**\r\n\t * Añade varios hijos a una sección por code.\r\n\t * @param sectionCode Código de la sección\r\n\t * @param children Array de hijos a agregar\r\n\t * @param stackPlace 'start' | 'end' (por defecto 'end')\r\n\t */\r\n\tpublic addSectionChildren(sectionCode: string, children: any[], stackPlace: 'start' | 'end' = 'end') {\r\n\t\tthis.sections$ = this.sections$.pipe(\r\n\t\t\tmap((sections: Section[]) => {\r\n\t\t\t\treturn sections.map(section => {\r\n\t\t\t\t\tif (section.code?.toLowerCase() === sectionCode.toLowerCase()) {\r\n\t\t\t\t\t\tif (!section.children) section.children = [];\r\n\t\t\t\t\t\tif (stackPlace === 'start') {\r\n\t\t\t\t\t\t\tsection.children = [...children, ...section.children];\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\tsection.children = [...section.children, ...children];\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t\treturn section;\r\n\t\t\t\t});\r\n\t\t\t})\r\n\t\t);\r\n\t}\r\n}"]}
124
+ //# 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';\r\nimport { AuthService, OptionsService, ParametersService } from '../../ec-services';\r\nimport { map, Observable } from 'rxjs';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { Attribute, Category, Parameter, Section } from '../../interfaces';\r\nimport { ToastService } from '../../ec-services/toast.service';\r\n/**\r\n * Para diferenciar los tipos de Menu\r\n */\r\ntype MenuType = \"category\" | \"attribute\" | \"section\"\r\n/**\r\n * Componente que sirve para abstraer la funcionalidad que comparten \r\n * el menú y el footer.\r\n * @class MenuEcComponent\r\n */\r\n@Component({\r\n\tselector: 'lib-footer-ec',\r\n\tstandalone: true,\r\n\timports: [],\r\n\ttemplate: '<p>Menu and Footer Helper Component</p>'\r\n})\r\nexport class MenuEcComponent {\r\n\tprivate optionsService = inject(OptionsService)\r\n\tprivate parametersService = inject(ParametersService)\r\n\tprivate _authService: AuthService = inject(AuthService)\r\n\tprivate _toastService: ToastService = inject(ToastService)\r\n\t/**\r\n\t * Constantes del core.\r\n\t */\r\n\tprivate consts = inject(CoreConstantsService)\r\n\r\n\t/**\r\n\t * Observable de categorias\r\n\t */\r\n\tpublic categories$: Observable<Category[]> = this.optionsService.getCategories()\r\n\t/**\r\n\t * Observable de secciones\r\n\t */\r\n\tpublic sections$: Observable<Section[]> = this.optionsService.getSections()\r\n\t/**\r\n\t * Observable de atributos\r\n\t */\r\n\tpublic attributes$: Observable<Attribute[]> = this.optionsService.getAttributes()\r\n\t/**\r\n\t * Observable de parametros\r\n\t */\r\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters()\r\n\t/**\r\n\t * Url para las imagenes guardades en el backend.\r\n\t */\r\n\tpublic mediaUrl = this.consts.mediaUrl()\r\n\t/**\r\n\t * Funcion para saber si existen un parametro. \r\n\t */\r\n\tpublic hasParams = this.parametersService.hasParams\r\n\r\n\tconstructor() {}\r\n\r\n\tprotected addChildren = (menuType: MenuType, code: string, item: any, stackPlace = 'end'): void => {\r\n\r\n\t}\r\n\r\n\t/* protected addChildren = (menuItems: any[], code: string, item: any, stackPlace = 'end') => {\r\n\t  (stackPlace === 'start')\r\n\t\t? menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.unshift(item)\r\n\t\t: menuItems.find(item => item.code?.toLowerCase().includes(code))?.children.push(item)\r\n    \r\n\t} */\r\n\t/**\r\n\t * Actualiza las categorias agregando nuevas.\r\n\t * @param categories \r\n\t */\r\n\tprotected updateCategories(categories: Category[]) {\r\n\t\tthis.categories$ = this.categories$.pipe(map((data: Category[]) => {\r\n\t\t\treturn [...data, ...categories]\r\n\t\t}))\r\n\t}\r\n\t/**\r\n\t * Actualiza las secciones agregando nuevas.\r\n\t * @param sections \r\n\t */\r\n\tprotected updateSections(sections: Section[]) {\r\n\t\tthis.sections$ = this.sections$.pipe(map((data: Section[]) => {\r\n\t\t\treturn [...data, ...sections]\r\n\t\t}))\r\n\t}\r\n\t/**\r\n\t * Actualiza el menú de atributos agredando nuevos.\r\n\t * @param attributes \r\n\t */\r\n\tprotected updateAttributes(attributes: Attribute[]) {\r\n\t\tthis.attributes$ = this.attributes$.pipe(map((data: Attribute[]) => {\r\n\t\t\treturn [...data, ...attributes]\r\n\t\t}))\r\n\t}\r\n\r\n\tpublic isAuthenticated() {\r\n\t\treturn this._authService.isAuthenticated();\r\n\t}\r\n\r\n\tpublic logout() {\r\n\t\tthis._authService.logout();\r\n\t\tthis._toastService.show('logout-success');\r\n\t\t window.location.href = '/home'; // Redirige a la página de inicio después del logout\r\n\t}\r\n\r\n\t/**\r\n\t * Añade varios hijos a una sección por code.\r\n\t * @param sectionCode Código de la sección\r\n\t * @param children Array de hijos a agregar\r\n\t * @param stackPlace 'start' | 'end' (por defecto 'end')\r\n\t */\r\n\tpublic addSectionChildren(sectionCode: string, children: any[], stackPlace: 'start' | 'end' = 'end') {\r\n\t\tthis.sections$ = this.sections$.pipe(\r\n\t\t\tmap((sections: Section[]) => {\r\n\t\t\t\treturn sections.map(section => {\r\n\t\t\t\t\tif (section.code?.toLowerCase() === sectionCode.toLowerCase()) {\r\n\t\t\t\t\t\tif (!section.children) section.children = [];\r\n\t\t\t\t\t\tif (stackPlace === 'start') {\r\n\t\t\t\t\t\t\tsection.children = [...children, ...section.children];\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\tsection.children = [...section.children, ...children];\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t\treturn section;\r\n\t\t\t\t});\r\n\t\t\t})\r\n\t\t);\r\n\t}\r\n}"]}
@@ -1,7 +1,7 @@
1
- import { afterNextRender, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal } from '@angular/core';
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, isPlatformBrowser } from '@angular/common';
4
+ import { CommonModule } 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";
@@ -14,6 +14,12 @@ register();
14
14
  * @class BlockProductsEcComponent
15
15
  */
16
16
  export class BlockProductsEcComponent extends BlockEcComponent {
17
+ // Personalización de las flechas de navegación
18
+ // Por defecto usa símbolos de texto, pero se pueden especificar imágenes
19
+ prevArrowImage; // undefined = usa símbolo de texto
20
+ nextArrowImage; // undefined = usa símbolo de texto
21
+ prevArrowText = '<';
22
+ nextArrowText = '>';
17
23
  /**
18
24
  * Servicio de Analytics
19
25
  */
@@ -34,66 +40,169 @@ export class BlockProductsEcComponent extends BlockEcComponent {
34
40
  * Bloque principal que contiene los productos
35
41
  */
36
42
  meta;
43
+ ngAfterViewInit() {
44
+ this.setupSwiperNavigation();
45
+ }
46
+ document;
47
+ platformId = inject(PLATFORM_ID);
48
+ /**
49
+ * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias
50
+ * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente
51
+ * en el Dom el element `<swiper-container>` para poder configurarlo.
52
+ */
53
+ constructor() {
54
+ super();
55
+ }
56
+ /**
57
+ * Aplica el evento `select_promotion` junto con el banner que interactua.
58
+ * @param banner
59
+ */
60
+ selectPromotion(item) {
61
+ this.analyticsService.callEvent('select_promotion', item);
62
+ }
63
+ /**
64
+ * Configura la navegación personalizada del Swiper.
65
+ * Esta función está diseñada para ser movida al componente base BlockProductsEcComponent.
66
+ * Permite personalización de las imágenes de las flechas mediante @Input.
67
+ */
68
+ setupSwiperNavigation() {
69
+ if (this.meta?.styles?.carrousel !== false) {
70
+ // Usar setTimeout para asegurar que el swiper esté inicializado
71
+ setTimeout(() => {
72
+ this.initializeSwiperWithCustomNavigation();
73
+ }, 200);
74
+ }
75
+ }
76
+ /**
77
+ * Inicializa el Swiper con navegación personalizada.
78
+ * Esta función puede ser movida al componente base para reutilización.
79
+ */
80
+ initializeSwiperWithCustomNavigation() {
81
+ const prevButton = document.getElementById(`${this.meta?.code}-prev`);
82
+ const nextButton = document.getElementById(`${this.meta?.code}-next`);
83
+ const swiperElement = document.getElementById(this.meta?.code);
84
+ if (prevButton && nextButton && swiperElement) {
85
+ console.log('Configurando navegación personalizada para:', this.meta?.code);
86
+ const swiperConfig = this.getSwiperConfiguration();
87
+ // Verificar si el Swiper ya está inicializado
88
+ if (!swiperElement.swiper) {
89
+ this.initializeNewSwiper(swiperElement, swiperConfig);
90
+ }
91
+ else {
92
+ this.updateExistingSwiper(swiperElement);
93
+ }
94
+ // Configurar los event listeners para los botones
95
+ this.setupNavigationEventListeners(prevButton, nextButton, swiperElement);
96
+ console.log('Event listeners configurados para los botones de navegación');
97
+ }
98
+ else {
99
+ console.log('No se pudieron encontrar los elementos:', {
100
+ prevButton: !!prevButton,
101
+ nextButton: !!nextButton,
102
+ swiperElement: !!swiperElement
103
+ });
104
+ }
105
+ }
37
106
  /**
38
- * Método original en: {@link BlockEcComponent}
107
+ * Obtiene la configuración base del Swiper.
108
+ * Esta configuración puede ser personalizada en el futuro mediante @Input.
39
109
  */
40
- swiperOptions = () => {
110
+ getSwiperConfiguration() {
41
111
  return {
42
- navigation: true,
112
+ slidesPerView: 'auto',
113
+ spaceBetween: 16,
114
+ slidesPerGroup: 1, // Importante: moverse de uno en uno
115
+ navigation: false, // Deshabilitamos la navegación por defecto
116
+ pagination: false,
117
+ loop: false,
118
+ grabCursor: true,
43
119
  breakpoints: {
44
- 0: {
45
- slidesPerView: 1
120
+ 320: {
121
+ slidesPerView: 1,
122
+ spaceBetween: 8,
123
+ slidesPerGroup: 1
46
124
  },
47
125
  576: {
48
- slidesPerView: 2
126
+ slidesPerView: 2,
127
+ spaceBetween: 12,
128
+ slidesPerGroup: 1
49
129
  },
50
130
  768: {
51
131
  slidesPerView: 3,
132
+ spaceBetween: 16,
133
+ slidesPerGroup: 1
52
134
  },
53
- 992: {
54
- slidesPerView: 4,
55
- },
56
- 1200: {
135
+ 1024: {
57
136
  slidesPerView: 4,
137
+ spaceBetween: 16,
138
+ slidesPerGroup: 1
58
139
  }
59
140
  }
60
141
  };
61
- };
62
- document;
63
- platformId = inject(PLATFORM_ID);
142
+ }
64
143
  /**
65
- * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias
66
- * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente
67
- * en el Dom el element `<swiper-container>` para poder configurarlo.
144
+ * Inicializa un nuevo Swiper con la configuración proporcionada.
68
145
  */
69
- constructor() {
70
- super();
71
- if (isPlatformBrowser(this.platformId)) {
72
- this.document = document;
146
+ initializeNewSwiper(swiperElement, config) {
147
+ // Asignar configuración al elemento
148
+ Object.assign(swiperElement, config);
149
+ // Inicializar el Swiper
150
+ swiperElement.initialize();
151
+ console.log('Swiper inicializado con configuración personalizada');
152
+ }
153
+ /**
154
+ * Actualiza un Swiper existente para asegurar la configuración correcta.
155
+ */
156
+ updateExistingSwiper(swiperElement) {
157
+ console.log('Actualizando configuración de Swiper existente');
158
+ // Forzar slidesPerGroup a 1
159
+ swiperElement.swiper.params.slidesPerGroup = 1;
160
+ swiperElement.swiper.allowSlideNext = true;
161
+ swiperElement.swiper.allowSlidePrev = true;
162
+ // Actualizar también los breakpoints si existen
163
+ if (swiperElement.swiper.params.breakpoints) {
164
+ Object.keys(swiperElement.swiper.params.breakpoints).forEach(key => {
165
+ swiperElement.swiper.params.breakpoints[key].slidesPerGroup = 1;
166
+ });
73
167
  }
74
- afterNextRender(() => {
75
- const swiperElemConstructor = this.document?.querySelector('#' + this.meta?.code);
76
- if (swiperElemConstructor) {
77
- Object.assign(swiperElemConstructor, this.swiperOptions());
78
- this.swiperElement.set(swiperElemConstructor);
79
- this.swiperElement()?.initialize();
80
- }
81
- });
168
+ swiperElement.swiper.update();
82
169
  }
83
170
  /**
84
- * Aplica el evento `select_promotion` junto con el banner que interactua.
85
- * @param banner
171
+ * Configura los event listeners para los botones de navegación.
86
172
  */
87
- selectPromotion(item) {
88
- this.analyticsService.callEvent('select_promotion', item);
173
+ setupNavigationEventListeners(prevButton, nextButton, swiperElement) {
174
+ prevButton.addEventListener('click', (e) => {
175
+ e.preventDefault();
176
+ e.stopPropagation();
177
+ console.log('Click en botón anterior');
178
+ if (swiperElement.swiper) {
179
+ swiperElement.swiper.slidePrev();
180
+ }
181
+ });
182
+ nextButton.addEventListener('click', (e) => {
183
+ e.preventDefault();
184
+ e.stopPropagation();
185
+ console.log('Click en botón siguiente');
186
+ if (swiperElement.swiper) {
187
+ swiperElement.swiper.slideNext();
188
+ }
189
+ });
89
190
  }
90
191
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BlockProductsEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
91
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: BlockProductsEcComponent, isStandalone: true, selector: "app-block-products-ec", inputs: { appProduct: "appProduct", products: "products", meta: "meta" }, usesInheritance: true, ngImport: i0, template: "<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\r\n\r\n <div class=\"blockProduct block-product\">\r\n @if(meta.name){\r\n <div class=\"row\">\r\n <div class=\"col-12 mt-4\">\r\n <h2 class=\"font-weight-normal font-gd\">\r\n <span>{{meta.name}}</span>\r\n </h2>\r\n </div>\r\n </div>\r\n }\r\n\r\n\r\n @if(meta.styles && meta.styles.carrousel == false){\r\n <div class=\"row \">\r\n @for (product of products; track $index) {\r\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\">\r\n <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\r\n <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </div>\r\n }\r\n </div>\r\n } @else {\r\n <div class=\"container\">\r\n <swiper-container init=\"false\" [id]=\"meta?.code\">\r\n @for (product of products; track $index) {\r\n <swiper-slide id=\"swiper-slide\">\r\n <ng-container\r\n *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </swiper-slide>\r\n }\r\n </swiper-container>\r\n </div>\r\n }\r\n </div>\r\n</section>\r\n\r\n\r\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\r\n<ng-template #defaultAppProduct let-product=\"product\">\r\n <app-product-ec [product]=\"product\"></app-product-ec>\r\n</ng-template>", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: ProductEcComponent, selector: "app-product-ec", inputs: ["product", "isProductBox", "isCollection"], outputs: ["loaded"] }] });
192
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: BlockProductsEcComponent, isStandalone: true, selector: "app-block-products-ec", inputs: { prevArrowImage: "prevArrowImage", nextArrowImage: "nextArrowImage", prevArrowText: "prevArrowText", nextArrowText: "nextArrowText", appProduct: "appProduct", products: "products", meta: "meta" }, usesInheritance: true, ngImport: i0, template: "<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\r\n\r\n <div class=\"blockProduct block-product\">\r\n @if(meta.name){\r\n <div class=\"row\">\r\n <div class=\"col-12 mt-4\">\r\n <h2 class=\"font-weight-normal font-gd\">\r\n <span>{{meta.name}}</span>\r\n </h2>\r\n </div>\r\n </div>\r\n }\r\n\r\n\r\n @if(meta.styles && meta.styles.carrousel == false){\r\n <div class=\"row \">\r\n @for (product of products; track $index) {\r\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\">\r\n <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\r\n <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </div>\r\n }\r\n </div>\r\n } @else {\r\n <div class=\"container position-relative\">\r\n <swiper-container \r\n init=\"false\" \r\n [id]=\"meta?.code\"\r\n slides-per-view=\"auto\"\r\n space-between=\"16\"\r\n slides-per-group=\"1\"\r\n navigation=\"false\"\r\n pagination=\"false\"\r\n loop=\"false\">\r\n @for (product of products; track $index) {\r\n <swiper-slide id=\"swiper-slide\">\r\n <ng-container\r\n *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </swiper-slide>\r\n }\r\n </swiper-container>\r\n \r\n <!-- Botones de navegaci\u00F3n personalizados -->\r\n <div class=\"swiper-navigation\">\r\n <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\r\n @if(prevArrowImage) {\r\n <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\r\n } @else {\r\n <span class=\"arrow-text\">{{prevArrowText}}</span>\r\n }\r\n </div>\r\n <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\r\n @if(nextArrowImage) {\r\n <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\r\n } @else {\r\n <span class=\"arrow-text\">{{nextArrowText}}</span>\r\n }\r\n </div>\r\n </div>\r\n </div>\r\n }\r\n </div>\r\n</section>\r\n\r\n\r\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\r\n<ng-template #defaultAppProduct let-product=\"product\">\r\n <app-product-ec [product]=\"product\"></app-product-ec>\r\n</ng-template>", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: ProductEcComponent, selector: "app-product-ec", inputs: ["product", "isProductBox", "isCollection"], outputs: ["loaded"] }] });
92
193
  }
93
194
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BlockProductsEcComponent, decorators: [{
94
195
  type: Component,
95
- args: [{ selector: 'app-block-products-ec', standalone: true, imports: [CommonModule, ProductEcComponent], schemas: [CUSTOM_ELEMENTS_SCHEMA], template: "<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\r\n\r\n <div class=\"blockProduct block-product\">\r\n @if(meta.name){\r\n <div class=\"row\">\r\n <div class=\"col-12 mt-4\">\r\n <h2 class=\"font-weight-normal font-gd\">\r\n <span>{{meta.name}}</span>\r\n </h2>\r\n </div>\r\n </div>\r\n }\r\n\r\n\r\n @if(meta.styles && meta.styles.carrousel == false){\r\n <div class=\"row \">\r\n @for (product of products; track $index) {\r\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\">\r\n <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\r\n <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </div>\r\n }\r\n </div>\r\n } @else {\r\n <div class=\"container\">\r\n <swiper-container init=\"false\" [id]=\"meta?.code\">\r\n @for (product of products; track $index) {\r\n <swiper-slide id=\"swiper-slide\">\r\n <ng-container\r\n *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </swiper-slide>\r\n }\r\n </swiper-container>\r\n </div>\r\n }\r\n </div>\r\n</section>\r\n\r\n\r\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\r\n<ng-template #defaultAppProduct let-product=\"product\">\r\n <app-product-ec [product]=\"product\"></app-product-ec>\r\n</ng-template>" }]
96
- }], ctorParameters: () => [], propDecorators: { appProduct: [{
196
+ args: [{ selector: 'app-block-products-ec', standalone: true, imports: [CommonModule, ProductEcComponent], schemas: [CUSTOM_ELEMENTS_SCHEMA], template: "<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\r\n\r\n <div class=\"blockProduct block-product\">\r\n @if(meta.name){\r\n <div class=\"row\">\r\n <div class=\"col-12 mt-4\">\r\n <h2 class=\"font-weight-normal font-gd\">\r\n <span>{{meta.name}}</span>\r\n </h2>\r\n </div>\r\n </div>\r\n }\r\n\r\n\r\n @if(meta.styles && meta.styles.carrousel == false){\r\n <div class=\"row \">\r\n @for (product of products; track $index) {\r\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\">\r\n <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\r\n <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </div>\r\n }\r\n </div>\r\n } @else {\r\n <div class=\"container position-relative\">\r\n <swiper-container \r\n init=\"false\" \r\n [id]=\"meta?.code\"\r\n slides-per-view=\"auto\"\r\n space-between=\"16\"\r\n slides-per-group=\"1\"\r\n navigation=\"false\"\r\n pagination=\"false\"\r\n loop=\"false\">\r\n @for (product of products; track $index) {\r\n <swiper-slide id=\"swiper-slide\">\r\n <ng-container\r\n *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n </swiper-slide>\r\n }\r\n </swiper-container>\r\n \r\n <!-- Botones de navegaci\u00F3n personalizados -->\r\n <div class=\"swiper-navigation\">\r\n <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\r\n @if(prevArrowImage) {\r\n <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\r\n } @else {\r\n <span class=\"arrow-text\">{{prevArrowText}}</span>\r\n }\r\n </div>\r\n <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\r\n @if(nextArrowImage) {\r\n <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\r\n } @else {\r\n <span class=\"arrow-text\">{{nextArrowText}}</span>\r\n }\r\n </div>\r\n </div>\r\n </div>\r\n }\r\n </div>\r\n</section>\r\n\r\n\r\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\r\n<ng-template #defaultAppProduct let-product=\"product\">\r\n <app-product-ec [product]=\"product\"></app-product-ec>\r\n</ng-template>" }]
197
+ }], ctorParameters: () => [], propDecorators: { prevArrowImage: [{
198
+ type: Input
199
+ }], nextArrowImage: [{
200
+ type: Input
201
+ }], prevArrowText: [{
202
+ type: Input
203
+ }], nextArrowText: [{
204
+ type: Input
205
+ }], appProduct: [{
97
206
  type: Input
98
207
  }], products: [{
99
208
  type: Input,
@@ -106,4 +215,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
106
215
  required: true
107
216
  }]
108
217
  }] } });
109
- //# 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":"AAAA,OAAO,EAAE,eAAe,EAAE,SAAS,EAAE,sBAAsB,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,EAAE,MAAM,EAAe,MAAM,eAAe,CAAC;AACpI,OAAO,EAAE,gBAAgB,EAAE,MAAM,+BAA+B,CAAC;AAEjE,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AAClE,qDAAqD;AACrD,OAAO,EAAE,QAAQ,EAAmB,MAAM,uBAAuB,CAAC;AAClE,OAAO,EAAE,kBAAkB,EAAE,MAAM,uCAAuC,CAAC;;;AAE3E,QAAQ,EAAE,CAAA;AACV;;;;GAIG;AASH,MAAM,OAAO,wBAAyB,SAAQ,gBAAgB;IAC7D;;OAEG;IACK,gBAAgB,GAAqB,MAAM,CAAC,gBAAgB,CAAC,CAAA;IACrE;;OAEG;IACH,aAAa,GAAG,MAAM,CAAyB,IAAI,CAAC,CAAA;IACpD;;OAEG;IACM,UAAU,CAA+B;IAClD;;OAEG;IAGA,QAAQ,CAAM,CAAC,aAAa;IAC/B;;OAEG;IAGA,IAAI,CAAM;IACb;;OAEG;IACM,aAAa,GAAG,GAAkB,EAAE;QAC5C,OAAO;YACN,UAAU,EAAE,IAAI;YAChB,WAAW,EAAE;gBACZ,CAAC,EAAE;oBACF,aAAa,EAAE,CAAC;iBAChB;gBACD,GAAG,EAAE;oBACJ,aAAa,EAAE,CAAC;iBAChB;gBACD,GAAG,EAAE;oBACJ,aAAa,EAAE,CAAC;iBAChB;gBACD,GAAG,EAAE;oBACJ,aAAa,EAAE,CAAC;iBAChB;gBACD,IAAI,EAAE;oBACL,aAAa,EAAE,CAAC;iBAChB;aACD;SACD,CAAA;IACF,CAAC,CAAA;IACO,QAAQ,CAAY;IACpB,UAAU,GAAQ,MAAM,CAAC,WAAW,CAAC,CAAA;IAE7C;;;;OAIG;IACH;QACC,KAAK,EAAE,CAAA;QACP,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACxC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAA;QACzB,CAAC;QACD,eAAe,CAAC,GAAG,EAAE;YACpB,MAAM,qBAAqB,GAAG,IAAI,CAAC,QAAQ,EAAE,aAAa,CAAC,GAAG,GAAG,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAA;YACjF,IAAI,qBAAqB,EAAE,CAAC;gBAC3B,MAAM,CAAC,MAAM,CAAC,qBAAsB,EAAE,IAAI,CAAC,aAAa,EAAE,CAAC,CAAA;gBAC3D,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,qBAAwC,CAAC,CAAA;gBAChE,IAAI,CAAC,aAAa,EAAE,EAAE,UAAU,EAAE,CAAC;YACpC,CAAC;QACF,CAAC,CAAC,CAAA;IACH,CAAC;IACD;;;OAGG;IACH,eAAe,CAAC,IAAS;QACxB,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAA;IAC1D,CAAC;wGA9EW,wBAAwB;4FAAxB,wBAAwB,kLCvBrC,q5DA0Cc,yDDxBH,YAAY,oSAAE,kBAAkB;;4FAK9B,wBAAwB;kBARpC,SAAS;+BACC,uBAAuB,cACrB,IAAI,WACP,CAAC,YAAY,EAAE,kBAAkB,CAAC,WAGlC,CAAC,sBAAsB,CAAC;wDAcxB,UAAU;sBAAlB,KAAK;gBAMH,QAAQ;sBAFV,KAAK;uBAAC;wBACN,QAAQ,EAAE,IAAI;qBACd;gBAME,IAAI;sBAFN,KAAK;uBAAC;wBACN,QAAQ,EAAE,IAAI;qBACd","sourcesContent":["import { afterNextRender, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal, TemplateRef } from '@angular/core';\r\nimport { BlockEcComponent } from '../../abstractions-components';\r\nimport { SwiperOptions } from 'swiper/types';\r\nimport { AnalyticsService } from '../../../ec-services';\r\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\r\n// import function to register Swiper custom elements\r\nimport { register, SwiperContainer } from 'swiper/element/bundle';\r\nimport { ProductEcComponent } from \"../../product-ec/product-ec.component\";\r\n\r\nregister()\r\n/**\r\n * Componen para manejar los bloques de productos.\r\n * @extends {BlockEcComponent}\r\n * @class BlockProductsEcComponent\r\n */\r\n@Component({\r\n\tselector: 'app-block-products-ec',\r\n\tstandalone: true,\r\n\timports: [CommonModule, ProductEcComponent],\r\n\ttemplateUrl: './block-products-ec.component.html',\r\n\tstyleUrl: './block-products-ec.component.scss',\r\n\tschemas: [CUSTOM_ELEMENTS_SCHEMA]\r\n})\r\nexport class BlockProductsEcComponent extends BlockEcComponent {\r\n\t/**\r\n\t * Servicio de Analytics\r\n\t */\r\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\r\n\t/**\r\n\t * Signal utlizado para guarda el contenedor del carrusel\r\n\t */\r\n\tswiperElement = signal<SwiperContainer | null>(null)\r\n\t/**\r\n\t * Input que recibe un template para el producto.\r\n\t */\r\n\t@Input() appProduct: TemplateRef<any> | undefined;\r\n\t/**\r\n\t * Colección de productos.\r\n\t */\r\n\t@Input({\r\n\t\trequired: true\r\n\t}) products: any; // Product[];\r\n\t/**\r\n\t * Bloque principal que contiene los productos\r\n\t */\r\n\t@Input({\r\n\t\trequired: true\r\n\t}) meta: any;\r\n\t/**\r\n\t * Método original en: {@link BlockEcComponent}\r\n\t */\r\n\toverride swiperOptions = (): SwiperOptions => {\r\n\t\treturn {\r\n\t\t\tnavigation: true,\r\n\t\t\tbreakpoints: {\r\n\t\t\t\t0: {\r\n\t\t\t\t\tslidesPerView: 1\r\n\t\t\t\t},\r\n\t\t\t\t576: {\r\n\t\t\t\t\tslidesPerView: 2\r\n\t\t\t\t},\r\n\t\t\t\t768: {\r\n\t\t\t\t\tslidesPerView: 3,\r\n\t\t\t\t},\r\n\t\t\t\t992: {\r\n\t\t\t\t\tslidesPerView: 4,\r\n\t\t\t\t},\r\n\t\t\t\t1200: {\r\n\t\t\t\t\tslidesPerView: 4,\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\tprivate document?: Document;\r\n\tprivate platformId: any = inject(PLATFORM_ID)\r\n\r\n\t/**\r\n\t * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias \r\n\t * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente\r\n\t * en el Dom el element `<swiper-container>` para poder configurarlo.\r\n\t */\r\n\tconstructor() {\r\n\t\tsuper()\r\n\t\tif (isPlatformBrowser(this.platformId)) {\r\n\t\t\tthis.document = document\r\n\t\t}\r\n\t\tafterNextRender(() => {\r\n\t\t\tconst swiperElemConstructor = this.document?.querySelector('#' + this.meta?.code)\r\n\t\t\tif (swiperElemConstructor) {\r\n\t\t\t\tObject.assign(swiperElemConstructor!, this.swiperOptions())\r\n\t\t\t\tthis.swiperElement.set(swiperElemConstructor as SwiperContainer)\r\n\t\t\t\tthis.swiperElement()?.initialize();\r\n\t\t\t}\r\n\t\t})\r\n\t}\r\n\t/**\r\n\t * Aplica el evento `select_promotion` junto con el banner que interactua.\r\n\t * @param banner \r\n\t */\r\n\tselectPromotion(item: any): void {\r\n\t\tthis.analyticsService.callEvent('select_promotion', item)\r\n\t}\r\n}\r\n","<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\r\n\r\n    <div class=\"blockProduct block-product\">\r\n        @if(meta.name){\r\n        <div class=\"row\">\r\n            <div class=\"col-12 mt-4\">\r\n                <h2  class=\"font-weight-normal font-gd\">\r\n                    <span>{{meta.name}}</span>\r\n                </h2>\r\n            </div>\r\n        </div>\r\n        }\r\n\r\n\r\n        @if(meta.styles && meta.styles.carrousel == false){\r\n        <div class=\"row \">\r\n            @for (product of products; track $index) {\r\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\">\r\n                <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\r\n                <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n            </div>\r\n            }\r\n        </div>\r\n        } @else {\r\n        <div class=\"container\">\r\n            <swiper-container init=\"false\" [id]=\"meta?.code\">\r\n                @for (product of products; track $index) {\r\n                <swiper-slide id=\"swiper-slide\">\r\n                    <ng-container\r\n                        *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n                </swiper-slide>\r\n                }\r\n            </swiper-container>\r\n        </div>\r\n        }\r\n    </div>\r\n</section>\r\n\r\n\r\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\r\n<ng-template #defaultAppProduct let-product=\"product\">\r\n    <app-product-ec [product]=\"product\"></app-product-ec>\r\n</ng-template>"]}
218
+ //# 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';\r\nimport { BlockEcComponent } from '../../abstractions-components';\r\nimport { SwiperOptions } from 'swiper/types';\r\nimport { AnalyticsService } from '../../../ec-services';\r\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\r\n// import function to register Swiper custom elements\r\nimport { register, SwiperContainer } from 'swiper/element/bundle';\r\nimport { ProductEcComponent } from \"../../product-ec/product-ec.component\";\r\n\r\nregister()\r\n/**\r\n * Componen para manejar los bloques de productos.\r\n * @extends {BlockEcComponent}\r\n * @class BlockProductsEcComponent\r\n */\r\n@Component({\r\n\tselector: 'app-block-products-ec',\r\n\tstandalone: true,\r\n\timports: [CommonModule, ProductEcComponent],\r\n\ttemplateUrl: './block-products-ec.component.html',\r\n\tstyleUrl: './block-products-ec.component.scss',\r\n\tschemas: [CUSTOM_ELEMENTS_SCHEMA]\r\n})\r\nexport class BlockProductsEcComponent extends BlockEcComponent implements AfterViewInit {\r\n\t\t// Personalización de las flechas de navegación\r\n\t// Por defecto usa símbolos de texto, pero se pueden especificar imágenes\r\n\t@Input() prevArrowImage?: string; // undefined = usa símbolo de texto\r\n\t@Input() nextArrowImage?: string; // undefined = usa símbolo de texto\r\n\t@Input() prevArrowText: string = '<';\r\n\t@Input() nextArrowText: string = '>';\r\n\t/**\r\n\t * Servicio de Analytics\r\n\t */\r\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\r\n\t/**\r\n\t * Signal utlizado para guarda el contenedor del carrusel\r\n\t */\r\n\tswiperElement = signal<SwiperContainer | null>(null)\r\n\t/**\r\n\t * Input que recibe un template para el producto.\r\n\t */\r\n\t@Input() appProduct: TemplateRef<any> | undefined;\r\n\t/**\r\n\t * Colección de productos.\r\n\t */\r\n\t@Input({\r\n\t\trequired: true\r\n\t}) products: any; // Product[];\r\n\t/**\r\n\t * Bloque principal que contiene los productos\r\n\t */\r\n\t@Input({\r\n\t\trequired: true\r\n\t}) meta: any;\r\n\r\n\tngAfterViewInit() {\r\n\t\tthis.setupSwiperNavigation();\r\n\t}\r\n\t\r\n\tprivate document?: Document;\r\n\tprivate platformId: any = inject(PLATFORM_ID)\r\n\r\n\t/**\r\n\t * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias \r\n\t * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente\r\n\t * en el Dom el element `<swiper-container>` para poder configurarlo.\r\n\t */\r\n\tconstructor() {\r\n\t\tsuper()\r\n\t}\r\n\t/**\r\n\t * Aplica el evento `select_promotion` junto con el banner que interactua.\r\n\t * @param banner \r\n\t */\r\n\tselectPromotion(item: any): void {\r\n\t\tthis.analyticsService.callEvent('select_promotion', item)\r\n\t}\r\n\r\n\t/**\r\n\t * Configura la navegación personalizada del Swiper.\r\n\t * Esta función está diseñada para ser movida al componente base BlockProductsEcComponent.\r\n\t * Permite personalización de las imágenes de las flechas mediante @Input.\r\n\t */\r\n\tprivate setupSwiperNavigation() {\r\n\t\tif (this.meta?.styles?.carrousel !== false) {\r\n\t\t\t// Usar setTimeout para asegurar que el swiper esté inicializado\r\n\t\t\tsetTimeout(() => {\r\n\t\t\t\tthis.initializeSwiperWithCustomNavigation();\r\n\t\t\t}, 200);\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Inicializa el Swiper con navegación personalizada.\r\n\t * Esta función puede ser movida al componente base para reutilización.\r\n\t */\r\n\tprivate initializeSwiperWithCustomNavigation() {\r\n\t\tconst prevButton = document.getElementById(`${this.meta?.code}-prev`);\r\n\t\tconst nextButton = document.getElementById(`${this.meta?.code}-next`);\r\n\t\tconst swiperElement = document.getElementById(this.meta?.code) as any;\r\n\r\n\t\tif (prevButton && nextButton && swiperElement) {\r\n\t\t\tconsole.log('Configurando navegación personalizada para:', this.meta?.code);\r\n\t\t\t\r\n\t\t\tconst swiperConfig = this.getSwiperConfiguration();\r\n\r\n\t\t\t// Verificar si el Swiper ya está inicializado\r\n\t\t\tif (!swiperElement.swiper) {\r\n\t\t\t\tthis.initializeNewSwiper(swiperElement, swiperConfig);\r\n\t\t\t} else {\r\n\t\t\t\tthis.updateExistingSwiper(swiperElement);\r\n\t\t\t}\r\n\r\n\t\t\t// Configurar los event listeners para los botones\r\n\t\t\tthis.setupNavigationEventListeners(prevButton, nextButton, swiperElement);\r\n\t\t\t\r\n\t\t\tconsole.log('Event listeners configurados para los botones de navegación');\r\n\t\t} else {\r\n\t\t\tconsole.log('No se pudieron encontrar los elementos:', {\r\n\t\t\t\tprevButton: !!prevButton,\r\n\t\t\t\tnextButton: !!nextButton,\r\n\t\t\t\tswiperElement: !!swiperElement\r\n\t\t\t});\r\n\t\t}\r\n\t}\r\n\r\n\t/**\r\n\t * Obtiene la configuración base del Swiper.\r\n\t * Esta configuración puede ser personalizada en el futuro mediante @Input.\r\n\t */\r\n\tprivate getSwiperConfiguration() {\r\n\t\treturn {\r\n\t\t\tslidesPerView: 'auto',\r\n\t\t\tspaceBetween: 16,\r\n\t\t\tslidesPerGroup: 1, // Importante: moverse de uno en uno\r\n\t\t\tnavigation: false, // Deshabilitamos la navegación por defecto\r\n\t\t\tpagination: false,\r\n\t\t\tloop: false,\r\n\t\t\tgrabCursor: true,\r\n\t\t\tbreakpoints: {\r\n\t\t\t\t320: {\r\n\t\t\t\t\tslidesPerView: 1,\r\n\t\t\t\t\tspaceBetween: 8,\r\n\t\t\t\t\tslidesPerGroup: 1\r\n\t\t\t\t},\r\n\t\t\t\t576: {\r\n\t\t\t\t\tslidesPerView: 2,\r\n\t\t\t\t\tspaceBetween: 12,\r\n\t\t\t\t\tslidesPerGroup: 1\r\n\t\t\t\t},\r\n\t\t\t\t768: {\r\n\t\t\t\t\tslidesPerView: 3,\r\n\t\t\t\t\tspaceBetween: 16,\r\n\t\t\t\t\tslidesPerGroup: 1\r\n\t\t\t\t},\r\n\t\t\t\t1024: {\r\n\t\t\t\t\tslidesPerView: 4,\r\n\t\t\t\t\tspaceBetween: 16,\r\n\t\t\t\t\tslidesPerGroup: 1\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t};\r\n\t}\r\n\r\n\t/**\r\n\t * Inicializa un nuevo Swiper con la configuración proporcionada.\r\n\t */\r\n\tprivate initializeNewSwiper(swiperElement: any, config: any) {\r\n\t\t// Asignar configuración al elemento\r\n\t\tObject.assign(swiperElement, config);\r\n\t\t\r\n\t\t// Inicializar el Swiper\r\n\t\tswiperElement.initialize();\r\n\t\t\r\n\t\tconsole.log('Swiper inicializado con configuración personalizada');\r\n\t}\r\n\r\n\t/**\r\n\t * Actualiza un Swiper existente para asegurar la configuración correcta.\r\n\t */\r\n\tprivate updateExistingSwiper(swiperElement: any) {\r\n\t\tconsole.log('Actualizando configuración de Swiper existente');\r\n\t\t\r\n\t\t// Forzar slidesPerGroup a 1\r\n\t\tswiperElement.swiper.params.slidesPerGroup = 1;\r\n\t\tswiperElement.swiper.allowSlideNext = true;\r\n\t\tswiperElement.swiper.allowSlidePrev = true;\r\n\t\t\r\n\t\t// Actualizar también los breakpoints si existen\r\n\t\tif (swiperElement.swiper.params.breakpoints) {\r\n\t\t\tObject.keys(swiperElement.swiper.params.breakpoints).forEach(key => {\r\n\t\t\t\tswiperElement.swiper.params.breakpoints[key].slidesPerGroup = 1;\r\n\t\t\t});\r\n\t\t}\r\n\t\t\r\n\t\tswiperElement.swiper.update();\r\n\t}\r\n\r\n\t/**\r\n\t * Configura los event listeners para los botones de navegación.\r\n\t */\r\n\tprivate setupNavigationEventListeners(prevButton: Element, nextButton: Element, swiperElement: any) {\r\n\t\tprevButton.addEventListener('click', (e) => {\r\n\t\t\te.preventDefault();\r\n\t\t\te.stopPropagation();\r\n\t\t\tconsole.log('Click en botón anterior');\r\n\t\t\tif (swiperElement.swiper) {\r\n\t\t\t\tswiperElement.swiper.slidePrev();\r\n\t\t\t}\r\n\t\t});\r\n\r\n\t\tnextButton.addEventListener('click', (e) => {\r\n\t\t\te.preventDefault();\r\n\t\t\te.stopPropagation();\r\n\t\t\tconsole.log('Click en botón siguiente');\r\n\t\t\tif (swiperElement.swiper) {\r\n\t\t\t\tswiperElement.swiper.slideNext();\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n}\r\n","<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\r\n\r\n    <div class=\"blockProduct block-product\">\r\n        @if(meta.name){\r\n        <div class=\"row\">\r\n            <div class=\"col-12 mt-4\">\r\n                <h2  class=\"font-weight-normal font-gd\">\r\n                    <span>{{meta.name}}</span>\r\n                </h2>\r\n            </div>\r\n        </div>\r\n        }\r\n\r\n\r\n        @if(meta.styles && meta.styles.carrousel == false){\r\n        <div class=\"row \">\r\n            @for (product of products; track $index) {\r\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\">\r\n                <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\r\n                <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n            </div>\r\n            }\r\n        </div>\r\n        } @else {\r\n        <div class=\"container position-relative\">\r\n            <swiper-container \r\n                init=\"false\" \r\n                [id]=\"meta?.code\"\r\n                slides-per-view=\"auto\"\r\n                space-between=\"16\"\r\n                slides-per-group=\"1\"\r\n                navigation=\"false\"\r\n                pagination=\"false\"\r\n                loop=\"false\">\r\n                @for (product of products; track $index) {\r\n                <swiper-slide id=\"swiper-slide\">\r\n                    <ng-container\r\n                        *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\r\n                </swiper-slide>\r\n                }\r\n            </swiper-container>\r\n            \r\n            <!-- Botones de navegación personalizados -->\r\n            <div class=\"swiper-navigation\">\r\n                <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\r\n                    @if(prevArrowImage) {\r\n                        <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\r\n                    } @else {\r\n                        <span class=\"arrow-text\">{{prevArrowText}}</span>\r\n                    }\r\n                </div>\r\n                <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\r\n                    @if(nextArrowImage) {\r\n                        <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\r\n                    } @else {\r\n                        <span class=\"arrow-text\">{{nextArrowText}}</span>\r\n                    }\r\n                </div>\r\n            </div>\r\n        </div>\r\n        }\r\n    </div>\r\n</section>\r\n\r\n\r\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\r\n<ng-template #defaultAppProduct let-product=\"product\">\r\n    <app-product-ec [product]=\"product\"></app-product-ec>\r\n</ng-template>"]}
@@ -70,7 +70,7 @@ export class BlocksEcComponent {
70
70
  }
71
71
  }
72
72
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BlocksEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
73
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: BlocksEcComponent, isStandalone: true, selector: "app-blocks-ec", inputs: { templates: "templates", show_loading: "show_loading", section: "section", blockFilters: "blockFilters" }, ngImport: i0, template: "@if(blocks$ | async; as blocks){\r\n @if(blocks.length > 0){\r\n <div class=\"container-fluid px-0\">\r\n <div class=\"row\">\r\n @for (block of blocks; track $index) {\r\n @switch (block.contentType) {\r\n @case ('banner') {\r\n @switch(block.design){\r\n @case ('full') {\r\n <app-block-banner-full-ec [banners]=\"block.banners\" [meta]=\"block\" />\r\n }\r\n @case ('boxes') {\r\n <app-block-banner-box-ec [banners]=\"block.banners\" [meta]=\"block\"/>\r\n }\r\n }\r\n }\r\n @case ('html') {\r\n @if(showBlock(block)){\r\n <app-block-html-ec [html_content]=\"getHTMLContent(block)\" />\r\n }\r\n }\r\n @case ('products') {\r\n <app-block-products-ec [products]=\"block.products?.items\" [meta]=\"block\" />\r\n }\r\n @case ('contact_form') {\r\n @if(isNewsletter(block)){\r\n <app-block-newsletter-ec [block]=\"block.contactForm\" />\r\n } @else {\r\n <app-block-form-contact-ec [block]=\"block.contactForm\" />\r\n }\r\n }\r\n }\r\n }\r\n </div>\r\n </div>\r\n }\r\n}", styles: [""], dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: BlockBannerFullEcComponent, selector: "app-block-banner-full-ec", inputs: ["banners", "meta"] }, { kind: "component", type: BlockBannerBoxEcComponent, selector: "app-block-banner-box-ec", inputs: ["banners", "meta"] }, { kind: "component", type: BlockHtmlEcComponent, selector: "app-block-html-ec", inputs: ["html_content"] }, { kind: "component", type: BlockProductsEcComponent, selector: "app-block-products-ec", inputs: ["appProduct", "products", "meta"] }, { kind: "component", type: BlockNewsletterEcComponent, selector: "app-block-newsletter-ec", inputs: ["block", "success_message", "subject"] }, { kind: "component", type: BlockFormContactEcComponent, selector: "app-block-form-contact-ec", inputs: ["block", "success_message", "redirect", "subject"] }] });
73
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: BlocksEcComponent, isStandalone: true, selector: "app-blocks-ec", inputs: { templates: "templates", show_loading: "show_loading", section: "section", blockFilters: "blockFilters" }, ngImport: i0, template: "@if(blocks$ | async; as blocks){\r\n @if(blocks.length > 0){\r\n <div class=\"container-fluid px-0\">\r\n <div class=\"row\">\r\n @for (block of blocks; track $index) {\r\n @switch (block.contentType) {\r\n @case ('banner') {\r\n @switch(block.design){\r\n @case ('full') {\r\n <app-block-banner-full-ec [banners]=\"block.banners\" [meta]=\"block\" />\r\n }\r\n @case ('boxes') {\r\n <app-block-banner-box-ec [banners]=\"block.banners\" [meta]=\"block\"/>\r\n }\r\n }\r\n }\r\n @case ('html') {\r\n @if(showBlock(block)){\r\n <app-block-html-ec [html_content]=\"getHTMLContent(block)\" />\r\n }\r\n }\r\n @case ('products') {\r\n <app-block-products-ec [products]=\"block.products?.items\" [meta]=\"block\" />\r\n }\r\n @case ('contact_form') {\r\n @if(isNewsletter(block)){\r\n <app-block-newsletter-ec [block]=\"block.contactForm\" />\r\n } @else {\r\n <app-block-form-contact-ec [block]=\"block.contactForm\" />\r\n }\r\n }\r\n }\r\n }\r\n </div>\r\n </div>\r\n }\r\n}", styles: [""], dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "component", type: BlockBannerFullEcComponent, selector: "app-block-banner-full-ec", inputs: ["banners", "meta"] }, { kind: "component", type: BlockBannerBoxEcComponent, selector: "app-block-banner-box-ec", inputs: ["banners", "meta"] }, { kind: "component", type: BlockHtmlEcComponent, selector: "app-block-html-ec", inputs: ["html_content"] }, { kind: "component", type: BlockProductsEcComponent, selector: "app-block-products-ec", inputs: ["prevArrowImage", "nextArrowImage", "prevArrowText", "nextArrowText", "appProduct", "products", "meta"] }, { kind: "component", type: BlockNewsletterEcComponent, selector: "app-block-newsletter-ec", inputs: ["block", "success_message", "subject"] }, { kind: "component", type: BlockFormContactEcComponent, selector: "app-block-form-contact-ec", inputs: ["block", "success_message", "redirect", "subject"] }] });
74
74
  }
75
75
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BlocksEcComponent, decorators: [{
76
76
  type: Component,
@@ -96,4 +96,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
96
96
  }], blockFilters: [{
97
97
  type: Input
98
98
  }] } });
99
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"blocks-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/blocks-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/blocks-ec.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAuB,MAAM,eAAe,CAAC;AAE9E,OAAO,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAEtD,OAAO,EAAE,0BAA0B,EAAE,MAAM,uDAAuD,CAAC;AACnG,OAAO,EAAE,yBAAyB,EAAE,MAAM,qDAAqD,CAAC;AAChG,OAAO,EAAE,oBAAoB,EAAE,MAAM,yCAAyC,CAAC;AAC/E,OAAO,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAC;AACtD,OAAO,EAAE,wBAAwB,EAAE,MAAM,iDAAiD,CAAC;AAC3F,OAAO,EAAE,0BAA0B,EAAE,MAAM,qDAAqD,CAAC;AACjG,OAAO,EAAE,2BAA2B,EAAE,MAAM,yDAAyD,CAAC;;AACtG;;;;;GAKG;AAiBH,MAAM,OAAO,iBAAiB;IAC7B;;OAEG;IACK,cAAc,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;IAC/C;;;OAGG;IACK,aAAa,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACpD;;OAEG;IACI,OAAO,CAAmC;IACjD;;OAEG;IACK,WAAW,GAAkB,IAAI,CAAC;IAC1C;;;OAGG;IACM,SAAS,CAAoB;IACtC;;OAEG;IACM,YAAY,GAAY,KAAK,CAAC;IACvC;;OAEG;IACH,IAEO,OAAO,CAAC,KAAoB;QAClC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,UAAU,EAAE,CAAC;IACnB,CAAC;IACQ,YAAY,GAAoB,IAAI,CAAC;IAC9C,gBAAgB,CAAC;IAEjB,QAAQ,KAAW,CAAC;IACpB;;;;OAIG;IACH,cAAc,GAAG,CAAC,KAAa,EAAE,EAAE,CAAC,KAAK,CAAC,YAAY,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;IAEhH,YAAY,CAAC,KAAU,IAAa,OAAO,KAAK,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA,CAAC,CAAC;IAAA,CAAC;IAErF,SAAS,GAAG,CAAC,KAAa,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAA;IAEhG;;KAEI;IACI,UAAU;QACjB,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAChE,CAAC;IACF,CAAC;wGA1DW,iBAAiB;4FAAjB,iBAAiB,6LClC9B,0uDAoCC,qDDdC,SAAS,8CAET,0BAA0B,kGAC1B,yBAAyB,iGACzB,oBAAoB,wFACpB,wBAAwB,8GACxB,0BAA0B,qHAC1B,2BAA2B;;4FAKhB,iBAAiB;kBAhB7B,SAAS;+BACC,eAAe,cACb,IAAI,WACP;wBACR,SAAS;wBACT,QAAQ;wBACR,0BAA0B;wBAC1B,yBAAyB;wBACzB,oBAAoB;wBACpB,wBAAwB;wBACxB,0BAA0B;wBAC1B,2BAA2B;qBAC3B;wDA0BQ,SAAS;sBAAjB,KAAK;gBAIG,YAAY;sBAApB,KAAK;gBAMC,OAAO;sBAFb,KAAK;uBAAC;wBACN,QAAQ,EAAE,IAAI;qBACd;gBAIQ,YAAY;sBAApB,KAAK","sourcesContent":["import { Component, inject, Input, OnInit, TemplateRef } from '@angular/core';\r\nimport { Observable } from 'rxjs';\r\nimport { BlocksService } from '../../ec-services';\r\nimport { AsyncPipe, JsonPipe } from '@angular/common';\r\nimport { IBlock } from '../../interfaces';\r\nimport { BlockBannerFullEcComponent } from './block-banner-full-ec/block-banner-full-ec.component';\r\nimport { BlockBannerBoxEcComponent } from \"./block-banner-box-ec/block-banner-box-ec.component\";\r\nimport { BlockHtmlEcComponent } from \"./block-html-ec/block-html-ec.component\";\r\nimport { ApiConstantsService } from '../../constants';\r\nimport { BlockProductsEcComponent } from \"./block-products-ec/block-products-ec.component\";\r\nimport { BlockNewsletterEcComponent } from \"./block-newsletter-ec/block-newsletter-ec.component\";\r\nimport { BlockFormContactEcComponent } from \"./block-form-contact-ec/block-form-contact-ec.component\";\r\n/**\r\n * Componente que es utilizado para manejar los distintos tipos\r\n * de bloques y distribuirlos por la aplicación.\r\n * @implements {OnInit}\r\n * @class BlocksEcComponent\r\n */\r\n@Component({\r\n\tselector: 'app-blocks-ec',\r\n\tstandalone: true,\r\n\timports: [\r\n\t\tAsyncPipe,\r\n\t\tJsonPipe,\r\n\t\tBlockBannerFullEcComponent,\r\n\t\tBlockBannerBoxEcComponent,\r\n\t\tBlockHtmlEcComponent,\r\n\t\tBlockProductsEcComponent,\r\n\t\tBlockNewsletterEcComponent,\r\n\t\tBlockFormContactEcComponent\r\n\t],\r\n\ttemplateUrl: './blocks-ec.component.html',\r\n\tstyleUrl: './blocks-ec.component.scss'\r\n})\r\nexport class BlocksEcComponent implements OnInit {\r\n\t/**\r\n\t * Servicio de Bloques\r\n\t */\r\n\tprivate _blocksService = inject(BlocksService);\r\n\t/**\tprivate _blocksService = inject(BlocksService);\r\n\r\n\t * Constantes de la API\r\n\t */\r\n\tprivate _apiConstants = inject(ApiConstantsService);\r\n\t/**\r\n\t * Observable de bloques\r\n\t */\r\n\tpublic blocks$: Observable<IBlock[]> | undefined;\r\n\t/**\r\n\t * Nombre de la sección que alberga a los bloques.\r\n\t */\r\n\tprivate _strsection: string | null = null;\r\n\t/**\r\n\t * Colección de Templates para reemplazar en los distintos tipos\r\n\t * de bloque por bloques locales generados en el frontend.\r\n\t */\r\n\t@Input() templates!: TemplateRef<any>;\r\n\t/**\r\n\t * Indicador para cuando se esten cargando los bloques.\r\n\t */\r\n\t@Input() show_loading: boolean = false;\r\n\t/**\r\n\t * Nombre de la sección del bloque. \r\n\t */\r\n\t@Input({\r\n\t\trequired: true,\r\n\t}) set section(value: string | null) {\r\n\t\tthis._strsection = value;\r\n\t\tthis.loadBlocks();\r\n\t}\r\n\t@Input() blockFilters: string[] | null = null;\r\n\tconstructor() { }\r\n\r\n\tngOnInit(): void { }\r\n\t/**\r\n\t * Obtiene el contenido html de un bloque, este método es usado cuando el bloque es de tipo **HTML**\r\n\t * @param block \r\n\t * @returns \r\n\t */\r\n\tgetHTMLContent = (block: IBlock) => block.translations && block.translations[this._apiConstants.LOCALE].content;\r\n\r\n\tisNewsletter(block: any): boolean { return block.contactForm.code.includes('news') };\r\n\r\n\tshowBlock = (block: IBlock) => this.blockFilters ? this.blockFilters.includes(block.code) : true\r\n\r\n\t/**\r\n   * Carga los bloques de la sección actual.\r\n   */\r\n\tprivate loadBlocks(): void {\r\n\t\tif (this._strsection) {\r\n\t\t\tthis.blocks$ = this._blocksService.getBlocks(this._strsection);\r\n\t\t}\r\n\t}\r\n}\t\r\n","@if(blocks$ | async; as blocks){\r\n    @if(blocks.length > 0){\r\n        <div class=\"container-fluid px-0\">\r\n            <div class=\"row\">\r\n                @for (block of blocks; track $index) {\r\n                    @switch (block.contentType) {\r\n                        @case ('banner') {\r\n                            @switch(block.design){\r\n                                @case ('full') {\r\n                                    <app-block-banner-full-ec [banners]=\"block.banners\" [meta]=\"block\" />\r\n                                }\r\n                                @case ('boxes') {\r\n                                    <app-block-banner-box-ec [banners]=\"block.banners\" [meta]=\"block\"/>\r\n                                }\r\n                            }\r\n                        }\r\n                        @case ('html') {\r\n                            @if(showBlock(block)){\r\n                                <app-block-html-ec [html_content]=\"getHTMLContent(block)\" />\r\n                            }\r\n                        }\r\n                        @case ('products') {\r\n                            <app-block-products-ec [products]=\"block.products?.items\" [meta]=\"block\" />\r\n                        }\r\n                        @case ('contact_form') {\r\n                            @if(isNewsletter(block)){\r\n                                <app-block-newsletter-ec [block]=\"block.contactForm\" />\r\n                            } @else {\r\n                                <app-block-form-contact-ec [block]=\"block.contactForm\" />\r\n                            }\r\n                        }\r\n                    }\r\n                }\r\n            </div>\r\n        </div>\r\n    }\r\n}"]}
99
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"blocks-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/blocks-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/blocks-ec.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAuB,MAAM,eAAe,CAAC;AAE9E,OAAO,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAEtD,OAAO,EAAE,0BAA0B,EAAE,MAAM,uDAAuD,CAAC;AACnG,OAAO,EAAE,yBAAyB,EAAE,MAAM,qDAAqD,CAAC;AAChG,OAAO,EAAE,oBAAoB,EAAE,MAAM,yCAAyC,CAAC;AAC/E,OAAO,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAC;AACtD,OAAO,EAAE,wBAAwB,EAAE,MAAM,iDAAiD,CAAC;AAC3F,OAAO,EAAE,0BAA0B,EAAE,MAAM,qDAAqD,CAAC;AACjG,OAAO,EAAE,2BAA2B,EAAE,MAAM,yDAAyD,CAAC;;AACtG;;;;;GAKG;AAiBH,MAAM,OAAO,iBAAiB;IAC7B;;OAEG;IACK,cAAc,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;IAC/C;;;OAGG;IACK,aAAa,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACpD;;OAEG;IACI,OAAO,CAAmC;IACjD;;OAEG;IACK,WAAW,GAAkB,IAAI,CAAC;IAC1C;;;OAGG;IACM,SAAS,CAAoB;IACtC;;OAEG;IACM,YAAY,GAAY,KAAK,CAAC;IACvC;;OAEG;IACH,IAEO,OAAO,CAAC,KAAoB;QAClC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,UAAU,EAAE,CAAC;IACnB,CAAC;IACQ,YAAY,GAAoB,IAAI,CAAC;IAC9C,gBAAgB,CAAC;IAEjB,QAAQ,KAAW,CAAC;IACpB;;;;OAIG;IACH,cAAc,GAAG,CAAC,KAAa,EAAE,EAAE,CAAC,KAAK,CAAC,YAAY,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;IAEhH,YAAY,CAAC,KAAU,IAAa,OAAO,KAAK,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA,CAAC,CAAC;IAAA,CAAC;IAErF,SAAS,GAAG,CAAC,KAAa,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAA;IAEhG;;KAEI;IACI,UAAU;QACjB,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAChE,CAAC;IACF,CAAC;wGA1DW,iBAAiB;4FAAjB,iBAAiB,6LClC9B,0uDAoCC,qDDdC,SAAS,8CAET,0BAA0B,kGAC1B,yBAAyB,iGACzB,oBAAoB,wFACpB,wBAAwB,oLACxB,0BAA0B,qHAC1B,2BAA2B;;4FAKhB,iBAAiB;kBAhB7B,SAAS;+BACC,eAAe,cACb,IAAI,WACP;wBACR,SAAS;wBACT,QAAQ;wBACR,0BAA0B;wBAC1B,yBAAyB;wBACzB,oBAAoB;wBACpB,wBAAwB;wBACxB,0BAA0B;wBAC1B,2BAA2B;qBAC3B;wDA0BQ,SAAS;sBAAjB,KAAK;gBAIG,YAAY;sBAApB,KAAK;gBAMC,OAAO;sBAFb,KAAK;uBAAC;wBACN,QAAQ,EAAE,IAAI;qBACd;gBAIQ,YAAY;sBAApB,KAAK","sourcesContent":["import { Component, inject, Input, OnInit, TemplateRef } from '@angular/core';\r\nimport { Observable } from 'rxjs';\r\nimport { BlocksService } from '../../ec-services';\r\nimport { AsyncPipe, JsonPipe } from '@angular/common';\r\nimport { IBlock } from '../../interfaces';\r\nimport { BlockBannerFullEcComponent } from './block-banner-full-ec/block-banner-full-ec.component';\r\nimport { BlockBannerBoxEcComponent } from \"./block-banner-box-ec/block-banner-box-ec.component\";\r\nimport { BlockHtmlEcComponent } from \"./block-html-ec/block-html-ec.component\";\r\nimport { ApiConstantsService } from '../../constants';\r\nimport { BlockProductsEcComponent } from \"./block-products-ec/block-products-ec.component\";\r\nimport { BlockNewsletterEcComponent } from \"./block-newsletter-ec/block-newsletter-ec.component\";\r\nimport { BlockFormContactEcComponent } from \"./block-form-contact-ec/block-form-contact-ec.component\";\r\n/**\r\n * Componente que es utilizado para manejar los distintos tipos\r\n * de bloques y distribuirlos por la aplicación.\r\n * @implements {OnInit}\r\n * @class BlocksEcComponent\r\n */\r\n@Component({\r\n\tselector: 'app-blocks-ec',\r\n\tstandalone: true,\r\n\timports: [\r\n\t\tAsyncPipe,\r\n\t\tJsonPipe,\r\n\t\tBlockBannerFullEcComponent,\r\n\t\tBlockBannerBoxEcComponent,\r\n\t\tBlockHtmlEcComponent,\r\n\t\tBlockProductsEcComponent,\r\n\t\tBlockNewsletterEcComponent,\r\n\t\tBlockFormContactEcComponent\r\n\t],\r\n\ttemplateUrl: './blocks-ec.component.html',\r\n\tstyleUrl: './blocks-ec.component.scss'\r\n})\r\nexport class BlocksEcComponent implements OnInit {\r\n\t/**\r\n\t * Servicio de Bloques\r\n\t */\r\n\tprivate _blocksService = inject(BlocksService);\r\n\t/**\tprivate _blocksService = inject(BlocksService);\r\n\r\n\t * Constantes de la API\r\n\t */\r\n\tprivate _apiConstants = inject(ApiConstantsService);\r\n\t/**\r\n\t * Observable de bloques\r\n\t */\r\n\tpublic blocks$: Observable<IBlock[]> | undefined;\r\n\t/**\r\n\t * Nombre de la sección que alberga a los bloques.\r\n\t */\r\n\tprivate _strsection: string | null = null;\r\n\t/**\r\n\t * Colección de Templates para reemplazar en los distintos tipos\r\n\t * de bloque por bloques locales generados en el frontend.\r\n\t */\r\n\t@Input() templates!: TemplateRef<any>;\r\n\t/**\r\n\t * Indicador para cuando se esten cargando los bloques.\r\n\t */\r\n\t@Input() show_loading: boolean = false;\r\n\t/**\r\n\t * Nombre de la sección del bloque. \r\n\t */\r\n\t@Input({\r\n\t\trequired: true,\r\n\t}) set section(value: string | null) {\r\n\t\tthis._strsection = value;\r\n\t\tthis.loadBlocks();\r\n\t}\r\n\t@Input() blockFilters: string[] | null = null;\r\n\tconstructor() { }\r\n\r\n\tngOnInit(): void { }\r\n\t/**\r\n\t * Obtiene el contenido html de un bloque, este método es usado cuando el bloque es de tipo **HTML**\r\n\t * @param block \r\n\t * @returns \r\n\t */\r\n\tgetHTMLContent = (block: IBlock) => block.translations && block.translations[this._apiConstants.LOCALE].content;\r\n\r\n\tisNewsletter(block: any): boolean { return block.contactForm.code.includes('news') };\r\n\r\n\tshowBlock = (block: IBlock) => this.blockFilters ? this.blockFilters.includes(block.code) : true\r\n\r\n\t/**\r\n   * Carga los bloques de la sección actual.\r\n   */\r\n\tprivate loadBlocks(): void {\r\n\t\tif (this._strsection) {\r\n\t\t\tthis.blocks$ = this._blocksService.getBlocks(this._strsection);\r\n\t\t}\r\n\t}\r\n}\t\r\n","@if(blocks$ | async; as blocks){\r\n    @if(blocks.length > 0){\r\n        <div class=\"container-fluid px-0\">\r\n            <div class=\"row\">\r\n                @for (block of blocks; track $index) {\r\n                    @switch (block.contentType) {\r\n                        @case ('banner') {\r\n                            @switch(block.design){\r\n                                @case ('full') {\r\n                                    <app-block-banner-full-ec [banners]=\"block.banners\" [meta]=\"block\" />\r\n                                }\r\n                                @case ('boxes') {\r\n                                    <app-block-banner-box-ec [banners]=\"block.banners\" [meta]=\"block\"/>\r\n                                }\r\n                            }\r\n                        }\r\n                        @case ('html') {\r\n                            @if(showBlock(block)){\r\n                                <app-block-html-ec [html_content]=\"getHTMLContent(block)\" />\r\n                            }\r\n                        }\r\n                        @case ('products') {\r\n                            <app-block-products-ec [products]=\"block.products?.items\" [meta]=\"block\" />\r\n                        }\r\n                        @case ('contact_form') {\r\n                            @if(isNewsletter(block)){\r\n                                <app-block-newsletter-ec [block]=\"block.contactForm\" />\r\n                            } @else {\r\n                                <app-block-form-contact-ec [block]=\"block.contactForm\" />\r\n                            }\r\n                        }\r\n                    }\r\n                }\r\n            </div>\r\n        </div>\r\n    }\r\n}"]}
@@ -1,8 +1,11 @@
1
1
  import { Component, inject } from '@angular/core';
2
2
  import { Router } from '@angular/router';
3
- import { CartService, AuthService } from '../../ec-services';
3
+ import { CartService, ToastService, AuthService, ChannelService } from '../../ec-services';
4
4
  import * as i0 from "@angular/core";
5
5
  export class CartEcComponent {
6
+ _channelService = inject(ChannelService);
7
+ channel;
8
+ _toastrService = inject(ToastService);
6
9
  _cartService = inject(CartService);
7
10
  router = inject(Router);
8
11
  cartItems$ = this._cartService.cartItems$;
@@ -14,6 +17,14 @@ export class CartEcComponent {
14
17
  isAuthenticated$ = this._authService.isAuthenticated();
15
18
  getTotalAmount = this._cartService.getTotalAmount();
16
19
  couponCode$ = this._cartService.getCouponCode();
20
+ constructor() {
21
+ //console.log("constructo.....");
22
+ this._channelService.channel$.subscribe((res) => {
23
+ //console.log("construct")
24
+ this.channel = res;
25
+ //this.initializeSteps();
26
+ });
27
+ }
17
28
  removeCoupon() {
18
29
  console.log(this.couponCode$);
19
30
  this._cartService.removeCoupon();
@@ -28,11 +39,24 @@ export class CartEcComponent {
28
39
  return false;
29
40
  }
30
41
  }
42
+ getMinimumPurchaseAmount = () => {
43
+ return this.channel.type == 'b2b' ? this.channel.wholesalerMinimumPurchaseAmount : this.channel.retailerMinimumPurchaseAmount;
44
+ };
45
+ exceedsMinimumAmount = (value) => {
46
+ if (value >= this.getMinimumPurchaseAmount()) {
47
+ this.router.navigate(['/checkout']);
48
+ return true;
49
+ }
50
+ else {
51
+ this._toastrService.show('quantity-not-exceeded', { amount: this.getMinimumPurchaseAmount() });
52
+ return false;
53
+ }
54
+ };
31
55
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CartEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
32
56
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CartEcComponent, isStandalone: true, selector: "lib-cart-ec", ngImport: i0, template: "<p>cart-ec works!</p>\r\n", styles: [""] });
33
57
  }
34
58
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CartEcComponent, decorators: [{
35
59
  type: Component,
36
60
  args: [{ selector: 'lib-cart-ec', standalone: true, imports: [], template: "<p>cart-ec works!</p>\r\n" }]
37
- }] });
38
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FydC1lYy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvZWMtY29tcG9uZW50cy9jYXJ0LWVjL2NhcnQtZWMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctZWFzeWNvbW1lcmNlLXYxOC9zcmMvbGliL2VjLWNvbXBvbmVudHMvY2FydC1lYy9jYXJ0LWVjLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2xELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFpQixXQUFXLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQzs7QUFVNUUsTUFBTSxPQUFPLGVBQWU7SUFFbkIsWUFBWSxHQUFnQixNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDakQsTUFBTSxHQUFXLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUVoQyxVQUFVLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxVQUFVLENBQUM7SUFDMUMsZUFBZSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztJQUN4RCxxQkFBcUIsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLHVCQUF1QixFQUFFLENBQUM7SUFDcEUsWUFBWSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsY0FBYyxFQUFFLENBQUM7SUFDbEQsWUFBWSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsY0FBYyxFQUFFLENBQUM7SUFFakQsWUFBWSxHQUFnQixNQUFNLENBQUMsV0FBVyxDQUFDLENBQUE7SUFFaEQsZ0JBQWdCLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxlQUFlLEVBQUUsQ0FBQztJQUN2RCxjQUFjLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxjQUFjLEVBQUUsQ0FBQztJQUVwRCxXQUFXLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxhQUFhLEVBQUUsQ0FBQztJQUV2RCxZQUFZO1FBQ1gsT0FBTyxDQUFDLEdBQUcsQ0FBRSxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUE7UUFDOUIsSUFBSSxDQUFDLFlBQVksQ0FBQyxZQUFZLEVBQUUsQ0FBQTtJQUNqQyxDQUFDO0lBRUQsZ0JBQWdCO1FBQ1QsSUFBSSxJQUFJLENBQUMsWUFBWSxDQUFDLGdDQUFnQyxFQUFFLEVBQUUsQ0FBQztZQUN2RCxJQUFJLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxXQUFXLENBQUMsQ0FBQztZQUN2QyxPQUFPLElBQUksQ0FBQztRQUNoQixDQUFDO2FBQU0sQ0FBQztZQUNKLE9BQU8sQ0FBQyxJQUFJLENBQUMsb0VBQW9FLENBQUMsQ0FBQztZQUNuRixPQUFPLEtBQUssQ0FBQztRQUNqQixDQUFDO0lBQ0wsQ0FBQzt3R0EvQlEsZUFBZTs0RkFBZixlQUFlLHVFQ1o1QiwyQkFDQTs7NEZEV2EsZUFBZTtrQkFQM0IsU0FBUzsrQkFDQyxhQUFhLGNBQ1gsSUFBSSxXQUNQLEVBQUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBSb3V0ZXIgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xyXG5pbXBvcnQgeyBDYXJ0U2VydmljZSwgVG9hc3RTZXJ2aWNlICwgQXV0aFNlcnZpY2UgfSBmcm9tICcuLi8uLi9lYy1zZXJ2aWNlcyc7XHJcbmltcG9ydCB7IENvcmVDb25zdGFudHNTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vY29uc3RhbnRzJztcclxuXHJcbkBDb21wb25lbnQoe1xyXG5cdHNlbGVjdG9yOiAnbGliLWNhcnQtZWMnLFxyXG5cdHN0YW5kYWxvbmU6IHRydWUsXHJcblx0aW1wb3J0czogW10sXHJcblx0dGVtcGxhdGVVcmw6ICcuL2NhcnQtZWMuY29tcG9uZW50Lmh0bWwnLFxyXG5cdHN0eWxlVXJsOiAnLi9jYXJ0LWVjLmNvbXBvbmVudC5zY3NzJ1xyXG59KVxyXG5leHBvcnQgY2xhc3MgQ2FydEVjQ29tcG9uZW50IHtcclxuXHJcblx0cHJpdmF0ZSBfY2FydFNlcnZpY2U6IENhcnRTZXJ2aWNlID0gaW5qZWN0KENhcnRTZXJ2aWNlKTtcclxuXHRwdWJsaWMgcm91dGVyOiBSb3V0ZXIgPSBpbmplY3QoUm91dGVyKTtcclxuXHJcblx0cHVibGljIGNhcnRJdGVtcyQgPSB0aGlzLl9jYXJ0U2VydmljZS5jYXJ0SXRlbXMkO1xyXG5cdHB1YmxpYyBzdWJUb3RhbEFtb3VudCQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRTdWJUb3RhbEFtb3VudCgpO1xyXG5cdHB1YmxpYyB0b3RhbFByb21vdGlvbkFtb3VudCQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRUb3RhbFByb21vdGlvbkFtb3VudCgpO1xyXG5cdHB1YmxpYyB0YXhlc0Ftb3VudCQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRUYXhlc0Ftb3VudCgpO1xyXG5cdHB1YmxpYyB0b3RhbEFtb3VudCQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRUb3RhbEFtb3VudCgpO1xyXG5cdFxyXG5cdHByaXZhdGUgX2F1dGhTZXJ2aWNlOiBBdXRoU2VydmljZSA9IGluamVjdChBdXRoU2VydmljZSlcclxuXHJcblx0cHVibGljIGlzQXV0aGVudGljYXRlZCQgPSB0aGlzLl9hdXRoU2VydmljZS5pc0F1dGhlbnRpY2F0ZWQoKTtcclxuXHRwdWJsaWMgZ2V0VG90YWxBbW91bnQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRUb3RhbEFtb3VudCgpO1xyXG5cclxuXHRwdWJsaWMgY291cG9uQ29kZSQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRDb3Vwb25Db2RlKCk7XHJcblx0XHJcblx0cmVtb3ZlQ291cG9uKCl7XHJcblx0XHRjb25zb2xlLmxvZyAodGhpcy5jb3Vwb25Db2RlJClcclxuXHRcdHRoaXMuX2NhcnRTZXJ2aWNlLnJlbW92ZUNvdXBvbigpXHJcblx0fVxyXG5cclxuXHRyZWRpcmVjdENoZWNrb3V0KCk6IGJvb2xlYW4ge1xyXG4gICAgICAgIGlmICh0aGlzLl9jYXJ0U2VydmljZS5oYXNTdWZmaWNpZW50Q3JlZGl0c0ZvckNhcnRUb3RhbCgpKSB7XHJcbiAgICAgICAgICAgIHRoaXMucm91dGVyLm5hdmlnYXRlQnlVcmwoYC9jaGVja291dGApO1xyXG4gICAgICAgICAgICByZXR1cm4gdHJ1ZTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICBjb25zb2xlLndhcm4oXCJObyBzZSBwdWVkZSByZWRpcmlnaXIgYWwgY2hlY2tvdXQgZGViaWRvIGEgdW5hIHZhbGlkYWNpw7NuIGZhbGxpZGEuXCIpO1xyXG4gICAgICAgICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxufVxyXG4iLCI8cD5jYXJ0LWVjIHdvcmtzITwvcD5cclxuIl19
61
+ }], ctorParameters: () => [] });
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FydC1lYy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvZWMtY29tcG9uZW50cy9jYXJ0LWVjL2NhcnQtZWMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctZWFzeWNvbW1lcmNlLXYxOC9zcmMvbGliL2VjLWNvbXBvbmVudHMvY2FydC1lYy9jYXJ0LWVjLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2xELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFFLFlBQVksRUFBRSxXQUFXLEVBQUUsY0FBYyxFQUFFLE1BQU0sbUJBQW1CLENBQUM7O0FBVTNGLE1BQU0sT0FBTyxlQUFlO0lBQ25CLGVBQWUsR0FBbUIsTUFBTSxDQUFDLGNBQWMsQ0FBQyxDQUFDO0lBQzFELE9BQU8sQ0FBTTtJQUVaLGNBQWMsR0FBaUIsTUFBTSxDQUFDLFlBQVksQ0FBQyxDQUFDO0lBRXBELFlBQVksR0FBZ0IsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBQ2pELE1BQU0sR0FBVyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUM7SUFFaEMsVUFBVSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsVUFBVSxDQUFDO0lBQzFDLGVBQWUsR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLGlCQUFpQixFQUFFLENBQUM7SUFDeEQscUJBQXFCLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyx1QkFBdUIsRUFBRSxDQUFDO0lBQ3BFLFlBQVksR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLGNBQWMsRUFBRSxDQUFDO0lBQ2xELFlBQVksR0FBRyxJQUFJLENBQUMsWUFBWSxDQUFDLGNBQWMsRUFBRSxDQUFDO0lBRWpELFlBQVksR0FBZ0IsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFBO0lBRWhELGdCQUFnQixHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsZUFBZSxFQUFFLENBQUM7SUFDdkQsY0FBYyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsY0FBYyxFQUFFLENBQUM7SUFFcEQsV0FBVyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsYUFBYSxFQUFFLENBQUM7SUFDdkQ7UUFDQyxpQ0FBaUM7UUFDakMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxRQUFRLENBQUMsU0FBUyxDQUN0QyxDQUFDLEdBQVEsRUFBRSxFQUFFO1lBQ1osMEJBQTBCO1lBQzFCLElBQUksQ0FBQyxPQUFPLEdBQUcsR0FBRyxDQUFDO1lBQ25CLHlCQUF5QjtRQUMxQixDQUFDLENBQ0QsQ0FBQTtJQUNGLENBQUM7SUFDRCxZQUFZO1FBQ1gsT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUE7UUFDN0IsSUFBSSxDQUFDLFlBQVksQ0FBQyxZQUFZLEVBQUUsQ0FBQTtJQUNqQyxDQUFDO0lBRUQsZ0JBQWdCO1FBQ2YsSUFBSSxJQUFJLENBQUMsWUFBWSxDQUFDLGdDQUFnQyxFQUFFLEVBQUUsQ0FBQztZQUMxRCxJQUFJLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxXQUFXLENBQUMsQ0FBQztZQUN2QyxPQUFPLElBQUksQ0FBQztRQUNiLENBQUM7YUFBTSxDQUFDO1lBQ1AsT0FBTyxDQUFDLElBQUksQ0FBQyxvRUFBb0UsQ0FBQyxDQUFDO1lBQ25GLE9BQU8sS0FBSyxDQUFDO1FBQ2QsQ0FBQztJQUNGLENBQUM7SUFFRCx3QkFBd0IsR0FBRyxHQUFHLEVBQUU7UUFDL0IsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksSUFBSSxLQUFLLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsK0JBQStCLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsNkJBQTZCLENBQUE7SUFFOUgsQ0FBQyxDQUFBO0lBQ0Qsb0JBQW9CLEdBQUcsQ0FBQyxLQUFhLEVBQUUsRUFBRTtRQUN4QyxJQUFJLEtBQUssSUFBSSxJQUFJLENBQUMsd0JBQXdCLEVBQUUsRUFBRSxDQUFDO1lBQzlDLElBQUksQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQztZQUNwQyxPQUFPLElBQUksQ0FBQztRQUNiLENBQUM7YUFBTSxDQUFDO1lBQ1AsSUFBSSxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsdUJBQXVCLEVBQUUsRUFBRSxNQUFNLEVBQUUsSUFBSSxDQUFDLHdCQUF3QixFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBQy9GLE9BQU8sS0FBSyxDQUFDO1FBQ2QsQ0FBQztJQUNGLENBQUMsQ0FBQTt3R0ExRFcsZUFBZTs0RkFBZixlQUFlLHVFQ1o1QiwyQkFDQTs7NEZEV2EsZUFBZTtrQkFQM0IsU0FBUzsrQkFDQyxhQUFhLGNBQ1gsSUFBSSxXQUNQLEVBQUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBSb3V0ZXIgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xyXG5pbXBvcnQgeyBDYXJ0U2VydmljZSwgVG9hc3RTZXJ2aWNlLCBBdXRoU2VydmljZSwgQ2hhbm5lbFNlcnZpY2UgfSBmcm9tICcuLi8uLi9lYy1zZXJ2aWNlcyc7XHJcbmltcG9ydCB7IENvcmVDb25zdGFudHNTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vY29uc3RhbnRzJztcclxuXHJcbkBDb21wb25lbnQoe1xyXG5cdHNlbGVjdG9yOiAnbGliLWNhcnQtZWMnLFxyXG5cdHN0YW5kYWxvbmU6IHRydWUsXHJcblx0aW1wb3J0czogW10sXHJcblx0dGVtcGxhdGVVcmw6ICcuL2NhcnQtZWMuY29tcG9uZW50Lmh0bWwnLFxyXG5cdHN0eWxlVXJsOiAnLi9jYXJ0LWVjLmNvbXBvbmVudC5zY3NzJ1xyXG59KVxyXG5leHBvcnQgY2xhc3MgQ2FydEVjQ29tcG9uZW50IHtcclxuXHRwcml2YXRlIF9jaGFubmVsU2VydmljZTogQ2hhbm5lbFNlcnZpY2UgPSBpbmplY3QoQ2hhbm5lbFNlcnZpY2UpO1xyXG5cdHB1YmxpYyBjaGFubmVsOiBhbnk7XHJcblxyXG5cdHByaXZhdGUgX3RvYXN0clNlcnZpY2U6IFRvYXN0U2VydmljZSA9IGluamVjdChUb2FzdFNlcnZpY2UpO1xyXG5cclxuXHRwcml2YXRlIF9jYXJ0U2VydmljZTogQ2FydFNlcnZpY2UgPSBpbmplY3QoQ2FydFNlcnZpY2UpO1xyXG5cdHB1YmxpYyByb3V0ZXI6IFJvdXRlciA9IGluamVjdChSb3V0ZXIpO1xyXG5cclxuXHRwdWJsaWMgY2FydEl0ZW1zJCA9IHRoaXMuX2NhcnRTZXJ2aWNlLmNhcnRJdGVtcyQ7XHJcblx0cHVibGljIHN1YlRvdGFsQW1vdW50JCA9IHRoaXMuX2NhcnRTZXJ2aWNlLmdldFN1YlRvdGFsQW1vdW50KCk7XHJcblx0cHVibGljIHRvdGFsUHJvbW90aW9uQW1vdW50JCA9IHRoaXMuX2NhcnRTZXJ2aWNlLmdldFRvdGFsUHJvbW90aW9uQW1vdW50KCk7XHJcblx0cHVibGljIHRheGVzQW1vdW50JCA9IHRoaXMuX2NhcnRTZXJ2aWNlLmdldFRheGVzQW1vdW50KCk7XHJcblx0cHVibGljIHRvdGFsQW1vdW50JCA9IHRoaXMuX2NhcnRTZXJ2aWNlLmdldFRvdGFsQW1vdW50KCk7XHJcblxyXG5cdHByaXZhdGUgX2F1dGhTZXJ2aWNlOiBBdXRoU2VydmljZSA9IGluamVjdChBdXRoU2VydmljZSlcclxuXHJcblx0cHVibGljIGlzQXV0aGVudGljYXRlZCQgPSB0aGlzLl9hdXRoU2VydmljZS5pc0F1dGhlbnRpY2F0ZWQoKTtcclxuXHRwdWJsaWMgZ2V0VG90YWxBbW91bnQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRUb3RhbEFtb3VudCgpO1xyXG5cclxuXHRwdWJsaWMgY291cG9uQ29kZSQgPSB0aGlzLl9jYXJ0U2VydmljZS5nZXRDb3Vwb25Db2RlKCk7XHJcblx0Y29uc3RydWN0b3IoKSB7XHJcblx0XHQvL2NvbnNvbGUubG9nKFwiY29uc3RydWN0by4uLi4uXCIpO1xyXG5cdFx0dGhpcy5fY2hhbm5lbFNlcnZpY2UuY2hhbm5lbCQuc3Vic2NyaWJlKFxyXG5cdFx0XHQocmVzOiBhbnkpID0+IHtcclxuXHRcdFx0XHQvL2NvbnNvbGUubG9nKFwiY29uc3RydWN0XCIpXHJcblx0XHRcdFx0dGhpcy5jaGFubmVsID0gcmVzO1xyXG5cdFx0XHRcdC8vdGhpcy5pbml0aWFsaXplU3RlcHMoKTtcclxuXHRcdFx0fVxyXG5cdFx0KVxyXG5cdH1cclxuXHRyZW1vdmVDb3Vwb24oKSB7XHJcblx0XHRjb25zb2xlLmxvZyh0aGlzLmNvdXBvbkNvZGUkKVxyXG5cdFx0dGhpcy5fY2FydFNlcnZpY2UucmVtb3ZlQ291cG9uKClcclxuXHR9XHJcblxyXG5cdHJlZGlyZWN0Q2hlY2tvdXQoKTogYm9vbGVhbiB7XHJcblx0XHRpZiAodGhpcy5fY2FydFNlcnZpY2UuaGFzU3VmZmljaWVudENyZWRpdHNGb3JDYXJ0VG90YWwoKSkge1xyXG5cdFx0XHR0aGlzLnJvdXRlci5uYXZpZ2F0ZUJ5VXJsKGAvY2hlY2tvdXRgKTtcclxuXHRcdFx0cmV0dXJuIHRydWU7XHJcblx0XHR9IGVsc2Uge1xyXG5cdFx0XHRjb25zb2xlLndhcm4oXCJObyBzZSBwdWVkZSByZWRpcmlnaXIgYWwgY2hlY2tvdXQgZGViaWRvIGEgdW5hIHZhbGlkYWNpw7NuIGZhbGxpZGEuXCIpO1xyXG5cdFx0XHRyZXR1cm4gZmFsc2U7XHJcblx0XHR9XHJcblx0fVxyXG5cclxuXHRnZXRNaW5pbXVtUHVyY2hhc2VBbW91bnQgPSAoKSA9PiB7XHJcblx0XHRyZXR1cm4gdGhpcy5jaGFubmVsLnR5cGUgPT0gJ2IyYicgPyB0aGlzLmNoYW5uZWwud2hvbGVzYWxlck1pbmltdW1QdXJjaGFzZUFtb3VudCA6IHRoaXMuY2hhbm5lbC5yZXRhaWxlck1pbmltdW1QdXJjaGFzZUFtb3VudFxyXG5cclxuXHR9XHJcblx0ZXhjZWVkc01pbmltdW1BbW91bnQgPSAodmFsdWU6IG51bWJlcikgPT4ge1xyXG5cdFx0aWYgKHZhbHVlID49IHRoaXMuZ2V0TWluaW11bVB1cmNoYXNlQW1vdW50KCkpIHtcclxuXHRcdFx0dGhpcy5yb3V0ZXIubmF2aWdhdGUoWycvY2hlY2tvdXQnXSk7XHJcblx0XHRcdHJldHVybiB0cnVlO1xyXG5cdFx0fSBlbHNlIHtcclxuXHRcdFx0dGhpcy5fdG9hc3RyU2VydmljZS5zaG93KCdxdWFudGl0eS1ub3QtZXhjZWVkZWQnLCB7IGFtb3VudDogdGhpcy5nZXRNaW5pbXVtUHVyY2hhc2VBbW91bnQoKSB9KTtcclxuXHRcdFx0cmV0dXJuIGZhbHNlO1xyXG5cdFx0fVxyXG5cdH1cclxuXHJcbn1cclxuIiwiPHA+Y2FydC1lYyB3b3JrcyE8L3A+XHJcbiJdfQ==