ng-easycommerce-v18 0.3.20-beta.1 → 0.3.21-beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -16
- package/esm2022/lib/classes/filters/attributes-filter.mjs +4 -74
- package/esm2022/lib/classes/filters/category-filter.mjs +26 -105
- package/esm2022/lib/classes/filters/filter-factory.mjs +3 -7
- package/esm2022/lib/classes/filters/price_range-filter.mjs +3 -3
- package/esm2022/lib/constants/api.constants.service.mjs +15 -19
- package/esm2022/lib/constants/core.constants.service.mjs +5 -16
- package/esm2022/lib/ec-components/abstractions-components/menu-ec.component.mjs +1 -17
- package/esm2022/lib/ec-components/auth-ec/login-form-ec/login-form-ec.component.mjs +16 -26
- package/esm2022/lib/ec-components/auth-ec/password-reset-ec/password-reset-ec.component.mjs +21 -25
- package/esm2022/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.mjs +16 -4
- package/esm2022/lib/ec-components/collection-ec/collection-ec.component.mjs +17 -41
- package/esm2022/lib/ec-components/filters-ec/filters-ec.component.mjs +14 -72
- package/esm2022/lib/ec-components/header-ec/header-ec.component.mjs +25 -41
- package/esm2022/lib/ec-components/price-range-filter/price-range-filter.component.mjs +2 -13
- package/esm2022/lib/ec-components/product-detail-ec/product-detail-ec.component.mjs +7 -12
- package/esm2022/lib/ec-components/stores-ec/stores-ec.component.mjs +9 -18
- package/esm2022/lib/ec-components/widgets-ec/decidir-ec/decidir-ec.component.mjs +6 -12
- package/esm2022/lib/ec-components/widgets-ec/magnizoom-ec/magnizoom-ec.component.mjs +4 -6
- package/esm2022/lib/ec-components/widgets-ec/redsys-catch-ec/redsys-catch-ec.component.mjs +17 -7
- package/esm2022/lib/ec-services/analytics/google-analytics.service.mjs +4 -4
- package/esm2022/lib/ec-services/analytics/gtm.service.mjs +6 -10
- package/esm2022/lib/ec-services/analytics/metricool-pixel.service.mjs +18 -17
- package/esm2022/lib/ec-services/checkout.service.mjs +2 -4
- package/esm2022/lib/ec-services/filters.service.mjs +18 -124
- package/esm2022/lib/ec-services/index.mjs +1 -3
- package/esm2022/lib/ec-services/pagination.service.mjs +22 -70
- package/esm2022/lib/ec-services/products.service.mjs +3 -5
- package/esm2022/lib/interceptors/index.mjs +1 -2
- package/esm2022/lib/interfaces/environment.mjs +1 -1
- package/esm2022/lib/interfaces/filter.mjs +1 -1
- package/esm2022/lib/interfaces/index.mjs +1 -2
- package/esm2022/lib/interfaces/options.mjs +1 -1
- package/esm2022/lib/providers/index.mjs +1 -2
- package/fesm2022/ng-easycommerce-v18.mjs +316 -1186
- package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
- package/lib/classes/filters/attributes-filter.d.ts +0 -24
- package/lib/classes/filters/category-filter.d.ts +3 -30
- package/lib/constants/api.constants.service.d.ts +0 -1
- package/lib/constants/core.constants.service.d.ts +0 -7
- package/lib/ec-components/abstractions-components/menu-ec.component.d.ts +0 -12
- package/lib/ec-components/auth-ec/login-form-ec/login-form-ec.component.d.ts +0 -2
- package/lib/ec-components/collection-ec/collection-ec.component.d.ts +4 -5
- package/lib/ec-components/filters-ec/filters-ec.component.d.ts +4 -25
- package/lib/ec-components/price-range-filter/price-range-filter.component.d.ts +0 -2
- package/lib/ec-services/analytics/gtm.service.d.ts +1 -1
- package/lib/ec-services/filters.service.d.ts +1 -18
- package/lib/ec-services/index.d.ts +0 -2
- package/lib/ec-services/pagination.service.d.ts +5 -21
- package/lib/ec-services/products.service.d.ts +1 -1
- package/lib/interceptors/index.d.ts +0 -1
- package/lib/interfaces/environment.d.ts +0 -1
- package/lib/interfaces/filter.d.ts +0 -1
- package/lib/interfaces/index.d.ts +0 -1
- package/lib/interfaces/options.d.ts +0 -2
- package/lib/providers/index.d.ts +0 -1
- package/package.json +1 -1
- package/esm2022/lib/ec-services/base-api.service.mjs +0 -148
- package/esm2022/lib/ec-services/runtime-config.service.mjs +0 -190
- package/esm2022/lib/interceptors/runtime-config.interceptor.mjs +0 -41
- package/esm2022/lib/interfaces/runtime-config.mjs +0 -2
- package/esm2022/lib/providers/provideRuntimeConfig.mjs +0 -42
- package/lib/ec-services/base-api.service.d.ts +0 -64
- package/lib/ec-services/runtime-config.service.d.ts +0 -63
- package/lib/interceptors/runtime-config.interceptor.d.ts +0 -7
- package/lib/interfaces/runtime-config.d.ts +0 -22
- package/lib/providers/provideRuntimeConfig.d.ts +0 -13
package/esm2022/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.mjs
CHANGED
|
@@ -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
|
|
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";
|
|
@@ -78,12 +78,24 @@ 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) || typeof document === 'undefined')
|
|
82
|
-
return;
|
|
83
81
|
const prevButton = document.getElementById(`${this.meta?.code}-prev`);
|
|
84
82
|
const nextButton = document.getElementById(`${this.meta?.code}-next`);
|
|
85
83
|
const swiperElement = document.getElementById(this.meta?.code);
|
|
86
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 {
|
|
87
99
|
// console.log('No se pudieron encontrar los elementos:', {
|
|
88
100
|
// prevButton: !!prevButton,
|
|
89
101
|
// nextButton: !!nextButton,
|
|
@@ -204,4 +216,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
204
216
|
required: true
|
|
205
217
|
}]
|
|
206
218
|
}] } });
|
|
207
|
-
//# 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,EAAkC,SAAS,EAAE,sBAAsB,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,EAAE,MAAM,EAAe,MAAM,eAAe,CAAC;AACnJ,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;IAC5D,+CAA+C;IAChD,yEAAyE;IAChE,cAAc,CAAU,CAAC,mCAAmC;IAC5D,cAAc,CAAU,CAAC,mCAAmC;IAC5D,aAAa,GAAW,GAAG,CAAC;IAC5B,aAAa,GAAW,GAAG,CAAC;IACrC;;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;IAEb,eAAe;QACd,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC9B,CAAC;IAEO,QAAQ,CAAY;IACpB,UAAU,GAAQ,MAAM,CAAC,WAAW,CAAC,CAAA;IAE7C;;;;OAIG;IACH;QACC,KAAK,EAAE,CAAA;IACR,CAAC;IACD;;;OAGG;IACH,eAAe,CAAC,IAAS;QACxB,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAA;IAC1D,CAAC;IAED;;;;OAIG;IACK,qBAAqB;QAC5B,IAAI,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,KAAK,KAAK,EAAE,CAAC;YAC5C,gEAAgE;YAChE,UAAU,CAAC,GAAG,EAAE;gBACf,IAAI,CAAC,oCAAoC,EAAE,CAAC;YAC7C,CAAC,EAAE,GAAG,CAAC,CAAC;QACT,CAAC;IACF,CAAC;IAED;;;OAGG;IACK,oCAAoC;QAC3C,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,OAAO,QAAQ,KAAK,WAAW;YAAE,OAAO;QAEnF,MAAM,UAAU,GAAG,QAAQ,CAAC,cAAc,CAAC,GAAG,IAAI,CAAC,IAAI,EAAE,IAAI,OAAO,CAAC,CAAC;QACtE,MAAM,UAAU,GAAG,QAAQ,CAAC,cAAc,CAAC,GAAG,IAAI,CAAC,IAAI,EAAE,IAAI,OAAO,CAAC,CAAC;QACtE,MAAM,aAAa,GAAG,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAQ,CAAC;QAEtE,IAAI,UAAU,IAAI,UAAU,IAAI,aAAa,EAAE,CAAC;YAC/C,2DAA2D;YAC3D,6BAA6B;YAC7B,6BAA6B;YAC7B,kCAAkC;YAClC,MAAM;QACP,CAAC;IACF,CAAC;IAED;;;OAGG;IACK,sBAAsB;QAC7B,OAAO;YACN,aAAa,EAAE,MAAM;YACrB,YAAY,EAAE,EAAE;YAChB,cAAc,EAAE,CAAC,EAAE,oCAAoC;YACvD,UAAU,EAAE,KAAK,EAAE,2CAA2C;YAC9D,UAAU,EAAE,KAAK;YACjB,IAAI,EAAE,IAAI;YACV,UAAU,EAAE,IAAI;YAChB,QAAQ,EAAE,IAAI;YACd,WAAW,EAAE;gBACZ,GAAG,EAAE;oBACJ,aAAa,EAAE,CAAC;oBAChB,YAAY,EAAE,CAAC;oBACf,cAAc,EAAE,CAAC;iBACjB;gBACD,GAAG,EAAE;oBACJ,aAAa,EAAE,CAAC;oBAChB,YAAY,EAAE,EAAE;oBAChB,cAAc,EAAE,CAAC;iBACjB;gBACD,GAAG,EAAE;oBACJ,aAAa,EAAE,CAAC;oBAChB,YAAY,EAAE,EAAE;oBAChB,cAAc,EAAE,CAAC;iBACjB;gBACD,IAAI,EAAE;oBACL,aAAa,EAAE,CAAC;oBAChB,YAAY,EAAE,EAAE;oBAChB,cAAc,EAAE,CAAC;iBACjB;aACD;SACD,CAAC;IACH,CAAC;IAED;;OAEG;IACK,mBAAmB,CAAC,aAAkB,EAAE,MAAW;QAC1D,oCAAoC;QACpC,MAAM,CAAC,MAAM,CAAC,aAAa,EAAE,MAAM,CAAC,CAAC;QAErC,wBAAwB;QACxB,aAAa,CAAC,UAAU,EAAE,CAAC;QAE3B,sEAAsE;IACvE,CAAC;IAED;;OAEG;IACK,oBAAoB,CAAC,aAAkB;QAC9C,iEAAiE;QAEjE,4BAA4B;QAC5B,aAAa,CAAC,MAAM,CAAC,MAAM,CAAC,cAAc,GAAG,CAAC,CAAC;QAC/C,aAAa,CAAC,MAAM,CAAC,cAAc,GAAG,IAAI,CAAC;QAC3C,aAAa,CAAC,MAAM,CAAC,cAAc,GAAG,IAAI,CAAC;QAE3C,gDAAgD;QAChD,IAAI,aAAa,CAAC,MAAM,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;YAC7C,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;gBAClE,aAAa,CAAC,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,cAAc,GAAG,CAAC,CAAC;YACjE,CAAC,CAAC,CAAC;QACJ,CAAC;QAED,aAAa,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;IAC/B,CAAC;IAED;;OAEG;IACK,6BAA6B,CAAC,UAAmB,EAAE,UAAmB,EAAE,aAAkB;QACjG,UAAU,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE;YAC1C,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,CAAC,CAAC,eAAe,EAAE,CAAC;YACpB,0CAA0C;YAC1C,IAAI,aAAa,CAAC,MAAM,EAAE,CAAC;gBAC1B,aAAa,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC;YAClC,CAAC;QACF,CAAC,CAAC,CAAC;QAEH,UAAU,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE;YAC1C,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,CAAC,CAAC,eAAe,EAAE,CAAC;YACpB,2CAA2C;YAC3C,IAAI,aAAa,CAAC,MAAM,EAAE,CAAC;gBAC1B,aAAa,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC;YAClC,CAAC;QACF,CAAC,CAAC,CAAC;IACJ,CAAC;wGAvLW,wBAAwB;4FAAxB,wBAAwB,sTCvBrC,m7FAoEc,yDDlDH,YAAY,oSAAE,kBAAkB;;4FAK9B,wBAAwB;kBARpC,SAAS;+BACC,uBAAuB,cACrB,IAAI,WACP,CAAC,YAAY,EAAE,kBAAkB,CAAC,WAGlC,CAAC,sBAAsB,CAAC;wDAKxB,cAAc;sBAAtB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBAYG,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, 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\tif (!isPlatformBrowser(this.platformId) || typeof document === 'undefined') return;\n\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('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>"]}
|
|
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>"]}
|
|
@@ -2,7 +2,7 @@ import { Component, inject, Input, PLATFORM_ID, signal } from '@angular/core';
|
|
|
2
2
|
import { AuthService, FiltersService, OptionsService, ProductsService } from '../../ec-services';
|
|
3
3
|
import { ActivatedRoute } from '@angular/router';
|
|
4
4
|
import { CoreConstantsService } from '../../constants';
|
|
5
|
-
import { combineLatest
|
|
5
|
+
import { combineLatest } from 'rxjs';
|
|
6
6
|
import { InfiniteScrollDirective } from "ngx-infinite-scroll";
|
|
7
7
|
import { isPlatformBrowser } from '@angular/common';
|
|
8
8
|
import * as i0 from "@angular/core";
|
|
@@ -10,7 +10,6 @@ export class CollectionEcComponent {
|
|
|
10
10
|
_productsService = inject(ProductsService);
|
|
11
11
|
_activeRoute = inject(ActivatedRoute);
|
|
12
12
|
_optionsService = inject(OptionsService);
|
|
13
|
-
_filtersService = inject(FiltersService);
|
|
14
13
|
params$ = this._activeRoute.params;
|
|
15
14
|
//public ready = this._optionsService.ready
|
|
16
15
|
queryParams$ = this._activeRoute.queryParams;
|
|
@@ -22,63 +21,40 @@ export class CollectionEcComponent {
|
|
|
22
21
|
defaultFilters = [];
|
|
23
22
|
loading = false;
|
|
24
23
|
countProducts = signal(0);
|
|
25
|
-
loaded =
|
|
24
|
+
loaded = false;
|
|
26
25
|
optionsFilters = ['all'];
|
|
27
26
|
filters_sort = [];
|
|
27
|
+
_filtersService = inject(FiltersService);
|
|
28
28
|
filters$ = this._filtersService.filters$;
|
|
29
29
|
ready$ = this._filtersService.ready$;
|
|
30
30
|
window;
|
|
31
31
|
isList = false;
|
|
32
|
-
destroy$ = new Subject();
|
|
33
32
|
setAsList = (value) => this.isList = value;
|
|
34
33
|
ngOnInit() {
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
}
|
|
38
|
-
combineLatest([this.params$, this.queryParams$])
|
|
39
|
-
.pipe(map(([params, query]) => ({ params, query })), distinctUntilChanged((a, b) => JSON.stringify(a) === JSON.stringify(b)), takeUntil(this.destroy$))
|
|
40
|
-
.subscribe(({ params, query }) => {
|
|
41
|
-
// Guardamos el tipo de ruta actual en las constantes (ej: 'categories', 'sections', etc.)
|
|
42
|
-
const routeType = params['type'] || null;
|
|
43
|
-
this.constanst.setCurrentRouteType(routeType);
|
|
44
|
-
const paginationSettings = {
|
|
45
|
-
latest: true,
|
|
46
|
-
limit: 10,
|
|
47
|
-
type: params['type'] || null,
|
|
48
|
-
value: params['value'] || null
|
|
49
|
-
};
|
|
50
|
-
// Punto clave:
|
|
51
|
-
// A partir de la URL (params + query) reconstruimos los filtros
|
|
52
|
-
// (categorías, atributos, rango de precios, etc.)
|
|
53
|
-
// Esto permite:
|
|
54
|
-
// - Soportar F5 / recarga sin perder filtros
|
|
55
|
-
// - Navegar con URL compartibles (deep linking)
|
|
56
|
-
this._filtersService.hydrateFromRoute(paginationSettings, params, query);
|
|
57
|
-
});
|
|
34
|
+
this.getProducts();
|
|
35
|
+
this.window?.scroll(0, 0);
|
|
58
36
|
}
|
|
59
37
|
//protected readonly questions = signal<Question[]>([]);
|
|
60
38
|
total = 0;
|
|
61
39
|
platformId = inject(PLATFORM_ID);
|
|
62
40
|
constructor() {
|
|
63
|
-
// Guardamos window sólo en Browser para evitar errores en SSR
|
|
64
41
|
if (isPlatformBrowser(this.platformId)) {
|
|
65
42
|
this.window = window;
|
|
66
43
|
}
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
44
|
+
}
|
|
45
|
+
getProducts() {
|
|
46
|
+
combineLatest([this.params$, this.queryParams$]).subscribe({
|
|
47
|
+
next: ([params, queryParams]) => {
|
|
48
|
+
const paginationSettings = {
|
|
49
|
+
latest: true,
|
|
50
|
+
limit: 10,
|
|
51
|
+
type: params['type'] || null,
|
|
52
|
+
value: params['value'] || null
|
|
53
|
+
};
|
|
54
|
+
this._productsService.getProductsForFilter(paginationSettings, queryParams["search"]);
|
|
75
55
|
}
|
|
76
56
|
});
|
|
77
57
|
}
|
|
78
|
-
ngOnDestroy() {
|
|
79
|
-
this.destroy$.next();
|
|
80
|
-
this.destroy$.complete();
|
|
81
|
-
}
|
|
82
58
|
onScroll() {
|
|
83
59
|
this.loading = true;
|
|
84
60
|
this._productsService.updateProducts();
|
|
@@ -96,4 +72,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
96
72
|
}], ctorParameters: () => [], propDecorators: { optionsFilters: [{
|
|
97
73
|
type: Input
|
|
98
74
|
}] } });
|
|
99
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collection-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/collection-ec/collection-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/collection-ec/collection-ec.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAmB,SAAS,EAAgB,MAAM,EAAE,KAAK,EAAqB,WAAW,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAEhI,OAAO,EAAE,WAAW,EAAE,cAAc,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACjG,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,oBAAoB,EAAY,GAAG,EAAc,OAAO,EAAQ,SAAS,EAAE,MAAM,MAAM,CAAC;AAEhH,OAAO,EAAE,uBAAuB,EAAE,MAAM,qBAAqB,CAAC;AAC9D,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;;AAQpD,MAAM,OAAO,qBAAqB;IAEvB,gBAAgB,GAAoB,MAAM,CAAC,eAAe,CAAC,CAAA;IAC3D,YAAY,GAAmB,MAAM,CAAC,cAAc,CAAC,CAAA;IACrD,eAAe,GAAmB,MAAM,CAAC,cAAc,CAAC,CAAA;IAC1D,eAAe,GAAmB,MAAM,CAAC,cAAc,CAAC,CAAC;IAE1D,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAA;IACzC,2CAA2C;IACpC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,CAAA;IAC5C,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAA;IAClD,uBAAuB;IAEhB,WAAW,GAAgB,MAAM,CAAC,WAAW,CAAC,CAAA;IAC9C,SAAS,GAAyB,MAAM,CAAC,oBAAoB,CAAC,CAAA;IAE9D,QAAQ,GAAc,EAAE,CAAC;IACzB,cAAc,GAAoB,EAAE,CAAC;IACrC,OAAO,GAAY,KAAK,CAAC;IACzB,aAAa,GAAG,MAAM,CAAS,CAAC,CAAC,CAAA;IAChC,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IAEtB,cAAc,GAAiB,CAAC,KAAK,CAAC,CAAC;IAEhD,YAAY,GAAa,EAAE,CAAC;IAErB,QAAQ,GAAyB,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC;IAC/D,MAAM,GAAwB,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC;IAEzD,MAAM,CAAU;IAEjB,MAAM,GAAY,KAAK,CAAA;IACtB,QAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;IACvC,SAAS,GAAG,CAAC,KAAc,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,KAAK,CAAA;IAEnD,QAAQ;QACP,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACxC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3B,CAAC;QAED,aAAa,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;aAC9C,IAAI,CACJ,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC,EAC7C,oBAAoB,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,EACvE,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CACxB;aACA,SAAS,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,EAAE,EAAE;YAChC,0FAA0F;YAC1F,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC;YACzC,IAAI,CAAC,SAAS,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAE9C,MAAM,kBAAkB,GAAuB;gBAC9C,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,EAAE;gBACT,IAAI,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,IAAI;gBAC5B,KAAK,EAAE,MAAM,CAAC,OAAO,CAAC,IAAI,IAAI;aAC9B,CAAC;YAEF,gBAAgB;YAChB,mEAAmE;YACnE,qDAAqD;YACrD,mBAAmB;YACnB,gDAAgD;YAChD,mDAAmD;YACnD,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,kBAAkB,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC;QAC1E,CAAC,CAAC,CAAC;IACL,CAAC;IAED,wDAAwD;IAChD,KAAK,GAAG,CAAC,CAAC;IACV,UAAU,GAAQ,MAAM,CAAC,WAAW,CAAC,CAAA;IAE7C;QACC,8DAA8D;QAC9D,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACxC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAA;QAErB,CAAC;QACD,2CAA2C;QAC3C,oFAAoF;QACpF,uEAAuE;QACvE,IAAI,CAAC,SAAS;aACZ,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,QAAQ,CAAC,EAAE;YACrB,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;gBACxC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACvB,CAAC;QACF,CAAC,CAAC,CAAC;IACL,CAAC;IAED,WAAW;QACV,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC1B,CAAC;IAED,QAAQ;QACP,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,gBAAgB,CAAC,cAAc,EAAE,CAAA;IACvC,CAAC;IAED,kEAAkE;IAC3D,eAAe,CAAC,OAAgB;QACtC,OAAO,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IAChD,CAAC;wGAvGW,qBAAqB;4FAArB,qBAAqB,2HChBlC,sZAgBA;;4FDAa,qBAAqB;kBAPjC,SAAS;+BACC,mBAAmB,cACjB,IAAI,WACP,CAAC,uBAAuB,CAAC;wDA0BzB,cAAc;sBAAtB,KAAK","sourcesContent":["import { afterNextRender, Component, HostListener, inject, Input, OnDestroy, OnInit, PLATFORM_ID, signal } from '@angular/core';\nimport { DefaultFilter, FilterType, PaginationSettings, Product } from '../../interfaces';\nimport { AuthService, FiltersService, OptionsService, ProductsService } from '../../ec-services';\nimport { ActivatedRoute } from '@angular/router';\nimport { CoreConstantsService } from '../../constants';\nimport { combineLatest, distinctUntilChanged, forkJoin, map, Observable, Subject, take, takeUntil } from 'rxjs';\nimport { Filter } from '../../classes';\nimport { InfiniteScrollDirective } from \"ngx-infinite-scroll\";\nimport { isPlatformBrowser } from '@angular/common';\n@Component({\n\tselector: 'lib-collection-ec',\n\tstandalone: true,\n\timports: [InfiniteScrollDirective],\n\ttemplateUrl: './collection-ec.component.html',\n\tstyleUrl: './collection-ec.component.scss'\n})\nexport class CollectionEcComponent implements OnInit, OnDestroy {\n\n\tprotected _productsService: ProductsService = inject(ProductsService)\n\tprotected _activeRoute: ActivatedRoute = inject(ActivatedRoute)\n\tprotected _optionsService: OptionsService = inject(OptionsService)\n\tprivate _filtersService: FiltersService = inject(FiltersService);\n\n\tpublic params$ = this._activeRoute.params\n\t//public ready = this._optionsService.ready\n\tpublic queryParams$ = this._activeRoute.queryParams\n\tpublic products$ = this._productsService.products$\n\t//public products$ = []\n\n\tpublic authService: AuthService = inject(AuthService)\n\tpublic constanst: CoreConstantsService = inject(CoreConstantsService)\n\n\tpublic products: Product[] = [];\n\tpublic defaultFilters: DefaultFilter[] = [];\n\tpublic loading: Boolean = false;\n\tpublic countProducts = signal<number>(0)\n\tprivate loaded = signal(false);\n\n\t@Input() optionsFilters: FilterType[] = ['all'];\n\n\tfilters_sort: Filter[] = [];\n\n\tpublic filters$: Observable<Filter[]> = this._filtersService.filters$;\n\tpublic ready$: Observable<boolean> = this._filtersService.ready$;\n\n\tprivate window?: Window;\n\n\tpublic isList: boolean = false\n\tprivate destroy$ = new Subject<void>();\n\tsetAsList = (value: boolean) => this.isList = value\n\n\tngOnInit(): void {\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\tthis.window?.scroll(0, 0);\n\t\t}\n\n\t\tcombineLatest([this.params$, this.queryParams$])\n\t\t\t.pipe(\n\t\t\t\tmap(([params, query]) => ({ params, query })),\n\t\t\t\tdistinctUntilChanged((a, b) => JSON.stringify(a) === JSON.stringify(b)),\n\t\t\t\ttakeUntil(this.destroy$)\n\t\t\t)\n\t\t\t.subscribe(({ params, query }) => {\n\t\t\t\t// Guardamos el tipo de ruta actual en las constantes (ej: 'categories', 'sections', etc.)\n\t\t\t\tconst routeType = params['type'] || null;\n\t\t\t\tthis.constanst.setCurrentRouteType(routeType);\n\n\t\t\t\tconst paginationSettings: PaginationSettings = {\n\t\t\t\t\tlatest: true,\n\t\t\t\t\tlimit: 10,\n\t\t\t\t\ttype: params['type'] || null,\n\t\t\t\t\tvalue: params['value'] || null\n\t\t\t\t};\n\n\t\t\t\t//  Punto clave:\n\t\t\t\t//    A partir de la URL (params + query) reconstruimos los filtros\n\t\t\t\t//    (categorías, atributos, rango de precios, etc.)\n\t\t\t\t//    Esto permite:\n\t\t\t\t//    - Soportar F5 / recarga sin perder filtros\n\t\t\t\t//    - Navegar con URL compartibles (deep linking)\n\t\t\t\tthis._filtersService.hydrateFromRoute(paginationSettings, params, query);\n\t\t\t});\n\t}\n\n\t//protected readonly questions = signal<Question[]>([]);\n\tprivate total = 0;\n\tprivate platformId: any = inject(PLATFORM_ID)\n\n\tconstructor() {\n\t\t// Guardamos window sólo en Browser para evitar errores en SSR\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\tthis.window = window\n\n\t\t}\n\t\t//  Nos suscribimos al stream de productos.\n\t\t//    Cuando ProductsService emite, marcamos `loaded` en true, pero solo en Browser.\n\t\t//    En SSR no lo marcamos para evitar cambios de estado innecesarios.\n\t\tthis.products$\n\t\t\t.pipe(takeUntil(this.destroy$))\n\t\t\t.subscribe(products => {\n\t\t\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\t\t\tthis.loaded.set(true);\n\t\t\t\t}\n\t\t\t});\n\t}\n\n\tngOnDestroy(): void {\n\t\tthis.destroy$.next();\n\t\tthis.destroy$.complete();\n\t}\n\n\tonScroll() {\n\t\tthis.loading = true;\n\t\tthis._productsService.updateProducts()\n\t}\n\n\t// Método para saber si un producto tiene stock usando el servicio\n\tpublic productHasStock(product: Product): boolean {\n\t\treturn this._productsService.hasStock(product);\n\t}\n}\n","<section class=\"container-xl mb-5\">\n    <div class=\"row m-0 p-0\">\n        <div class=\"d-none d-lg-block col-3\">\n            <!--Filtros-->\n        </div>\n\n        <div class=\"col-12 col-lg-9 mt-2 p-0\">\n            <div class=\"container\">\n                <!-- @for(['a', 'b', 'c', 'd'] as product){\n\n                } -->\n            </div>\n        </div>\n\n    </div>\n</section>\n"]}
|
|
75
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collection-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/collection-ec/collection-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/collection-ec/collection-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, Component, HostListener, inject, Input, OnInit, PLATFORM_ID, signal } from '@angular/core';\nimport { DefaultFilter, FilterType, PaginationSettings, Product } from '../../interfaces';\nimport { AuthService, FiltersService, OptionsService, ProductsService } from '../../ec-services';\nimport { ActivatedRoute } from '@angular/router';\nimport { CoreConstantsService } from '../../constants';\nimport { combineLatest, forkJoin, map, Observable, take } from 'rxjs';\nimport { Filter } from '../../classes';\nimport { InfiniteScrollDirective } from \"ngx-infinite-scroll\";\nimport { isPlatformBrowser } from '@angular/common';\n@Component({\n\tselector: 'lib-collection-ec',\n\tstandalone: true,\n\timports: [InfiniteScrollDirective],\n\ttemplateUrl: './collection-ec.component.html',\n\tstyleUrl: './collection-ec.component.scss'\n})\nexport class CollectionEcComponent implements OnInit {\n\n\tprotected _productsService: ProductsService = inject(ProductsService)\n\tprotected _activeRoute: ActivatedRoute = inject(ActivatedRoute)\n\tprotected _optionsService: OptionsService = inject(OptionsService)\n\n\tpublic params$ = this._activeRoute.params\n\t//public ready = this._optionsService.ready\n\tpublic queryParams$ = this._activeRoute.queryParams\n\tpublic products$ = this._productsService.products$\n\t//public products$ = []\n\n\tpublic authService: AuthService = inject(AuthService)\n\tpublic constanst: CoreConstantsService = inject(CoreConstantsService)\n\n\tpublic products: Product[] = [];\n\tpublic defaultFilters: DefaultFilter[] = [];\n\tpublic loading: Boolean = false;\n\tpublic countProducts = signal<number>(0)\n\tprivate loaded = false;\n\n\t@Input() optionsFilters: FilterType[] = ['all'];\n\n\tfilters_sort: Filter[] = [];\n\n\tprivate _filtersService: FiltersService = inject(FiltersService)\n\tpublic filters$: Observable<Filter[]> = this._filtersService.filters$;\n\tpublic ready$: Observable<boolean> = this._filtersService.ready$;\n\n\tprivate window?: Window;\n\n\tpublic isList: boolean = false\n\tsetAsList = (value: boolean) => this.isList = value\n\n\tngOnInit(): void {\n\t\tthis.getProducts();\n\t\tthis.window?.scroll(0, 0)\n\t}\n\n\t//protected readonly questions = signal<Question[]>([]);\n\tprivate total = 0;\n\tprivate platformId: any = inject(PLATFORM_ID)\n\n\tconstructor() {\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\tthis.window = window\n\n\t\t}\n\t}\n\n\tgetProducts(): void {\n\t\tcombineLatest([this.params$, this.queryParams$]).subscribe({\n\t\t\tnext: ([params, queryParams]) => {\n\t\t\t\tconst paginationSettings: PaginationSettings = {\n\t\t\t\t\tlatest: true,\n\t\t\t\t\tlimit: 10,\n\t\t\t\t\ttype: params['type'] || null,\n\t\t\t\t\tvalue: params['value'] || null\n\t\t\t\t}\n\t\t\t\tthis._productsService.getProductsForFilter(paginationSettings, queryParams[\"search\"]);\n\t\t\t}\n\t\t})\n\t}\n\n\tonScroll() {\n\t\tthis.loading = true;\n\t\tthis._productsService.updateProducts()\n\t}\n\n\t// Método para saber si un producto tiene stock usando el servicio\n\tpublic productHasStock(product: Product): boolean {\n\t\treturn this._productsService.hasStock(product);\n\t}\n}\n","<section class=\"container-xl mb-5\">\n    <div class=\"row m-0 p-0\">\n        <div class=\"d-none d-lg-block col-3\">\n            <!--Filtros-->\n        </div>\n\n        <div class=\"col-12 col-lg-9 mt-2 p-0\">\n            <div class=\"container\">\n                <!-- @for(['a', 'b', 'c', 'd'] as product){\n\n                } -->\n            </div>\n        </div>\n\n    </div>\n</section>\n"]}
|
|
@@ -3,7 +3,7 @@ import { FiltersService, AuthService } from '../../ec-services';
|
|
|
3
3
|
import { Filter } from '../../classes';
|
|
4
4
|
import { ProductsService } from '../../ec-services/products.service';
|
|
5
5
|
import { CoreConstantsService } from '../../constants';
|
|
6
|
-
import {
|
|
6
|
+
import { Router } from '@angular/router';
|
|
7
7
|
import { PriceRangeFilter } from '../../classes/filters/price_range-filter';
|
|
8
8
|
import { ChannelService } from '../../ec-services/channel.service';
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
@@ -19,7 +19,6 @@ export class FiltersEcComponent {
|
|
|
19
19
|
injector = inject(Injector);
|
|
20
20
|
isAuthenticated$ = this._authService.isAuthenticated();
|
|
21
21
|
hidePrices = false;
|
|
22
|
-
route = inject(ActivatedRoute);
|
|
23
22
|
setSelect;
|
|
24
23
|
ngOnInit() {
|
|
25
24
|
}
|
|
@@ -49,56 +48,26 @@ export class FiltersEcComponent {
|
|
|
49
48
|
.pop()
|
|
50
49
|
selectedOption && this._filtersService.setFilterSelected(this.filters[0], selectedOption) */
|
|
51
50
|
}
|
|
52
|
-
/**
|
|
53
|
-
* Maneja el click sobre un elemento de filtro (categoría, atributo, etc.).
|
|
54
|
-
*
|
|
55
|
-
* - Para categorías: navega a la URL de la categoría.
|
|
56
|
-
* - Para atributos: actualiza la query string de la URL con `attributeCodes`.
|
|
57
|
-
* - Para otros filtros: delega en FiltersService para marcar seleccionado.
|
|
58
|
-
*
|
|
59
|
-
* La idea es que **la URL siempre represente los filtros aplicados**,
|
|
60
|
-
* de modo que:
|
|
61
|
-
* - al hacer F5 no se pierdan los filtros,
|
|
62
|
-
* - los links sean compartibles (deep linking).
|
|
63
|
-
*/
|
|
64
51
|
setSelected(filter, selected) {
|
|
65
52
|
if (!filter || !selected) {
|
|
53
|
+
console.error('Filter or selected element is undefined:', { filter, selected });
|
|
66
54
|
return;
|
|
67
55
|
}
|
|
68
|
-
// Si el elemento está marcado como no visible, no hacemos nada
|
|
69
|
-
if (selected.isVisible === false)
|
|
70
|
-
return;
|
|
71
56
|
if (typeof filter.setSelected !== 'function') {
|
|
57
|
+
console.error('filter.setSelected is not a function. Filter might not be an instance of the expected class:', filter);
|
|
72
58
|
return;
|
|
73
59
|
}
|
|
74
60
|
try {
|
|
61
|
+
// this._filtersService.setFilterSelected(filter, selected);
|
|
75
62
|
if (filter.type() === 'categories') {
|
|
76
|
-
// Para categorías usamos navegación por path (ej: /collection/categories/camas-elasticas)
|
|
77
63
|
if (selected.path) {
|
|
78
64
|
this.router.navigate([selected.path]);
|
|
79
65
|
}
|
|
80
|
-
return;
|
|
81
66
|
}
|
|
82
|
-
if (filter.type() === 'attributes') {
|
|
83
|
-
//
|
|
84
|
-
|
|
85
|
-
// Actualizamos la URL manteniendo la ruta, pero agregando `attributeCodes`
|
|
86
|
-
// Ejemplo de resultado:
|
|
87
|
-
// /collection/categories/camas-elasticas?attributeCodes=ABC123&...
|
|
88
|
-
//
|
|
89
|
-
// También reseteamos la página a null para que vuelva a la primera página
|
|
90
|
-
// cuando se aplica un nuevo filtro.
|
|
91
|
-
this.router.navigate([], {
|
|
92
|
-
relativeTo: this.route,
|
|
93
|
-
queryParams: {
|
|
94
|
-
attributeCodes: code,
|
|
95
|
-
page: null
|
|
96
|
-
},
|
|
97
|
-
queryParamsHandling: 'merge'
|
|
98
|
-
});
|
|
99
|
-
return;
|
|
67
|
+
else if (filter.type() === 'attributes') {
|
|
68
|
+
// Manejar la navegación para atributos
|
|
69
|
+
this._filtersService.setFilterSelected(filter, selected);
|
|
100
70
|
}
|
|
101
|
-
this._filtersService.setFilterSelected(filter, selected);
|
|
102
71
|
}
|
|
103
72
|
catch (error) {
|
|
104
73
|
console.error("Error while setting selected filter:", error);
|
|
@@ -122,9 +91,7 @@ export class FiltersEcComponent {
|
|
|
122
91
|
});
|
|
123
92
|
}
|
|
124
93
|
// close = () => {
|
|
125
|
-
//
|
|
126
|
-
// this.document.getElementById("filtros").classList.remove('in');
|
|
127
|
-
// }
|
|
94
|
+
// this.document.getElementById("filtros").classList.remove('in');
|
|
128
95
|
// return true;
|
|
129
96
|
// };
|
|
130
97
|
close = () => {
|
|
@@ -134,9 +101,7 @@ export class FiltersEcComponent {
|
|
|
134
101
|
return true;
|
|
135
102
|
};
|
|
136
103
|
scrollUp = () => {
|
|
137
|
-
|
|
138
|
-
window.scroll(0, 0);
|
|
139
|
-
}
|
|
104
|
+
window.scroll(0, 0);
|
|
140
105
|
return true;
|
|
141
106
|
};
|
|
142
107
|
hasAppliedFilters() {
|
|
@@ -157,36 +122,13 @@ export class FiltersEcComponent {
|
|
|
157
122
|
}) ?? false;
|
|
158
123
|
}
|
|
159
124
|
/**
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
125
|
+
* Verifica si una categoría tiene la propiedad isVisible y está marcada como visible
|
|
126
|
+
* @param category - La categoría a verificar
|
|
127
|
+
* @returns true si la categoría es visible, false en caso contrario
|
|
128
|
+
*/
|
|
164
129
|
hasVisibleProperty(category) {
|
|
165
130
|
return category.isVisible === true;
|
|
166
131
|
}
|
|
167
|
-
/** Lista visible (filtra recursivo por isVisible) */
|
|
168
|
-
getVisibleData(filter) {
|
|
169
|
-
if (!filter)
|
|
170
|
-
return [];
|
|
171
|
-
return this.filterVisibleTree(filter.data);
|
|
172
|
-
}
|
|
173
|
-
/** Children visibles de un nodo */
|
|
174
|
-
getVisibleChildren(node) {
|
|
175
|
-
return this.filterVisibleTree(node?.children ?? []);
|
|
176
|
-
}
|
|
177
|
-
/** Tiene hijos visibles? */
|
|
178
|
-
hasVisibleChildren(node) {
|
|
179
|
-
return this.getVisibleChildren(node).length > 0;
|
|
180
|
-
}
|
|
181
|
-
/** Utilidad recursiva */
|
|
182
|
-
filterVisibleTree(list = []) {
|
|
183
|
-
return (list ?? [])
|
|
184
|
-
.filter(n => n.isVisible === true)
|
|
185
|
-
.map(n => ({
|
|
186
|
-
...n,
|
|
187
|
-
children: this.filterVisibleTree(n.children ?? [])
|
|
188
|
-
}));
|
|
189
|
-
}
|
|
190
132
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FiltersEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
191
133
|
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FiltersEcComponent, isStandalone: true, selector: "lib-filters-ec", inputs: { setSelect: "setSelect" }, ngImport: i0, template: "<p>filters-ec works!</p>\n", styles: [""] });
|
|
192
134
|
}
|
|
@@ -196,4 +138,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
196
138
|
}], ctorParameters: () => [], propDecorators: { setSelect: [{
|
|
197
139
|
type: Input
|
|
198
140
|
}] } });
|
|
199
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filters-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/filters-ec/filters-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/filters-ec/filters-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, ElementRef, ViewChild, Injector } from '@angular/core';\nimport { FiltersService, AuthService } from '../../ec-services';\nimport { Filter } from '../../classes';\nimport { FilterElement, FilterType, PaginationSettings } from '../../interfaces';\nimport { ProductsService } from '../../ec-services/products.service';\nimport { CoreConstantsService } from '../../constants';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { PriceRangeFilter } from '../../classes/filters/price_range-filter';\nimport { ChannelService } from '../../ec-services/channel.service';\n\n\n@Component({\n    selector: 'lib-filters-ec',\n    standalone: true,\n    imports: [],\n    templateUrl: './filters-ec.component.html',\n    styleUrl: './filters-ec.component.scss'\n})\nexport class FiltersEcComponent {\n\n    private _authService: AuthService = inject(AuthService);\n    private _filtersService: FiltersService = inject(FiltersService);\n    private _productsService: ProductsService = inject(ProductsService);\n    private _constants: CoreConstantsService = inject(CoreConstantsService);\n    private router: Router = inject(Router);\n    public filters: Filter[] = [];\n    public filter$ = this._filtersService.filters$;\n    public showPricesOnlyToLoggedUsers: boolean = false;\n    public injector: Injector = inject(Injector);\n    public isAuthenticated$ = this._authService.isAuthenticated();\n    public hidePrices: boolean = false;\n    private route: ActivatedRoute = inject(ActivatedRoute);\n\n    @Input() setSelect: any;\n\n    ngOnInit() {\n\n    }\n\n    constructor() {\n        this._filtersService.filters$.subscribe((filters: Filter[]) => {\n            this.filters = filters;\n        });\n        this.injector.get(ChannelService).channel$.subscribe(channel => {\n            this.showPricesOnlyToLoggedUsers = !!channel.showPricesOnlyToLoggedUsers;\n            this.hidePrices = !!channel.hidePrices;\n        });\n    }\n    /** 1) Para cuando me piden 'price_range', devuelvo PriceRangeFilter */\n    public getSpecificFilter(type: 'price_range'): PriceRangeFilter | null;\n\n    /** 2) Para cualquier otro FilterType, devuelvo Filter */\n    public getSpecificFilter(type: FilterType): Filter | null;\n    public getSpecificFilter(type: FilterType): Filter | null {\n        const filter = this.filters.find(f => f.type() === type);\n        if (type === 'price_range' && filter && !(filter instanceof Filter)) {\n            console.error(`Expected a Filter instance but got`, filter);\n        }\n        return filter || null;\n    }\n\n    /*Obtiene el filtro elegido en el select, lo pasa al setSelect\n    para que se marque seleccionado*/\n    onSelect(event: any, filter: FilterElement[] | null) {\n        // console.log(filter)\n        /*  const selectedOption = filter\n             .flatMap(category => category.children)\n             .filter(option => option?.code === event.target.value)\n             .pop()\n         selectedOption && this._filtersService.setFilterSelected(this.filters[0], selectedOption) */\n    }\n\n    /**\n     * Maneja el click sobre un elemento de filtro (categoría, atributo, etc.).\n     *\n     * - Para categorías: navega a la URL de la categoría.\n     * - Para atributos: actualiza la query string de la URL con `attributeCodes`.\n     * - Para otros filtros: delega en FiltersService para marcar seleccionado.\n     *\n     * La idea es que **la URL siempre represente los filtros aplicados**,\n     * de modo que:\n     *   - al hacer F5 no se pierdan los filtros,\n     *   - los links sean compartibles (deep linking).\n     */\n    setSelected(filter: Filter | null, selected: FilterElement | null) {\n        if (!filter || !selected) {\n            return;\n        }\n\n        // Si el elemento está marcado como no visible, no hacemos nada\n        if ((selected as any).isVisible === false) return;\n\n        if (typeof (filter as any).setSelected !== 'function') {\n            return;\n        }\n\n        try {\n            if (filter.type() === 'categories') {\n                // Para categorías usamos navegación por path (ej: /collection/categories/camas-elasticas)\n                if (selected.path) {\n                    this.router.navigate([selected.path]);\n                }\n                return;\n            }\n            if (filter.type() === 'attributes') {\n                // El backend espera que le mandemos el \"code\" del atributo en el query param\n                const code = (selected as any).code || (selected as any).value || null;\n\n                // Actualizamos la URL manteniendo la ruta, pero agregando `attributeCodes`\n                // Ejemplo de resultado:\n                //   /collection/categories/camas-elasticas?attributeCodes=ABC123&...\n                //\n                // También reseteamos la página a null para que vuelva a la primera página\n                // cuando se aplica un nuevo filtro.\n                this.router.navigate([], {\n                    relativeTo: this.route,\n                    queryParams: {\n                        attributeCodes: code,\n                        page: null\n                    },\n                    queryParamsHandling: 'merge'\n                });\n\n                return;\n            }\n            this._filtersService.setFilterSelected(filter, selected);\n\n        } catch (error) {\n            console.error(\"Error while setting selected filter:\", error);\n        }\n    }\n\n    uniqueSelection(filter: { data: any[]; }, filterChild: any) {\n        // Buscar si hay un filtro de tipo PriceRangeFilter en la lista de filtros\n        const priceRangeFilter = this.filters.find(f => f instanceof PriceRangeFilter);\n\n        if (priceRangeFilter) {\n            priceRangeFilter.reset();\n        }\n        filter.data.forEach(filterElement => {\n            if (filterElement.selected) {\n                filterElement.selected = false;\n            }\n            filterElement.children.forEach((filterChildren: FilterElement) => {\n                if (filterChildren != filterChild && filterChildren.selected) {\n                    filterChildren.selected = false;\n                }\n            });\n        });\n    }\n\n    //  close = () => {\n    //       if (typeof document !== 'undefined') {\n    //         this.document.getElementById(\"filtros\").classList.remove('in');\n    //       }\n    //       return true;\n    //  };\n\n\n    close = () => {\n        // this.consts.mobile()\n        //      ? this.document.getElementById(\"accordionExample\").classList.remove('show')\n        //      : this.document.getElementById(accordion_id).classList.remove('show');\n        return true;\n    };\n\n\n    scrollUp = () => {\n        if (typeof window !== 'undefined') {\n            window.scroll(0, 0);\n        }\n        return true\n    }\n\n\n\n\n\n    hasAppliedFilters(): boolean {\n        return this.filters?.some((filter: any) => {\n            if (filter.type?.() === 'price_range') {\n                const minSet = filter.currentMinPrice !== null && filter.currentMinPrice !== 0;\n                const maxSet = filter.currentMaxPrice !== null && filter.currentMaxPrice !== filter.maxPrice;\n                if (minSet || maxSet) {\n                    return true;\n                }\n            }\n\n            return filter.data?.some((filterElement: any) => {\n                if (Array.isArray(filterElement.children)) {\n                    return filterElement.children.some((child: any) => child.selected);\n                }\n                return filterElement.selected;\n            });\n        }) ?? false;\n    }\n    /**\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    hasVisibleProperty(category: FilterElement): boolean {\n        return (category as any).isVisible === true;\n    }\n\n    /** Lista visible (filtra recursivo por isVisible) */\n    public getVisibleData(filter: Filter | null): FilterElement[] {\n        if (!filter) return [];\n        return this.filterVisibleTree(filter.data);\n    }\n\n    /** Children visibles de un nodo */\n    public getVisibleChildren(node?: FilterElement): FilterElement[] {\n        return this.filterVisibleTree(node?.children ?? []);\n    }\n\n    /** Tiene hijos visibles? */\n    public hasVisibleChildren(node?: FilterElement): boolean {\n        return this.getVisibleChildren(node).length > 0;\n    }\n\n    /** Utilidad recursiva */\n    private filterVisibleTree(list: FilterElement[] = []): FilterElement[] {\n        return (list ?? [])\n            .filter(n => (n as any).isVisible === true)\n            .map(n => ({\n                ...n,\n                children: this.filterVisibleTree(n.children ?? [])\n            }));\n    }\n}\n","<p>filters-ec works!</p>\n"]}
|
|
141
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filters-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/filters-ec/filters-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/filters-ec/filters-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, ElementRef, ViewChild, Injector } from '@angular/core';\nimport { FiltersService , AuthService} from '../../ec-services';\nimport { Filter } from '../../classes';\nimport { FilterElement, FilterType, PaginationSettings } from '../../interfaces';\nimport { ProductsService } from '../../ec-services/products.service';\nimport { CoreConstantsService } from '../../constants';\nimport { Router } from '@angular/router';\nimport { PriceRangeFilter } from '../../classes/filters/price_range-filter';\nimport { ChannelService } from '../../ec-services/channel.service';\n\n\n@Component({\n    selector: 'lib-filters-ec',\n    standalone: true,\n    imports: [],\n    templateUrl: './filters-ec.component.html',\n    styleUrl: './filters-ec.component.scss'\n})\nexport class FiltersEcComponent {\n\n    private _authService: AuthService = inject(AuthService);\n    private _filtersService: FiltersService = inject(FiltersService);\n    private _productsService: ProductsService = inject(ProductsService);\n    private _constants: CoreConstantsService = inject(CoreConstantsService);\n    private router: Router = inject(Router);\n    public filters: Filter[] = [];\n    public filter$ = this._filtersService.filters$;\n    public showPricesOnlyToLoggedUsers: boolean = false;\n    public injector: Injector = inject(Injector);\n\tpublic isAuthenticated$ = this._authService.isAuthenticated();\n    public hidePrices: boolean = false;\n    \n    @Input() setSelect: any;\n\n    ngOnInit() {\n\n    }\n\n    constructor() {\n        this._filtersService.filters$.subscribe((filters: Filter[]) => {\n            this.filters = filters;\n        });\n        this.injector.get(ChannelService).channel$.subscribe(channel => {\n            this.showPricesOnlyToLoggedUsers = !!channel.showPricesOnlyToLoggedUsers;\n            this.hidePrices = !!channel.hidePrices;\n        });\n    }\n    /** 1) Para cuando me piden 'price_range', devuelvo PriceRangeFilter */\n    public getSpecificFilter(type: 'price_range'): PriceRangeFilter | null;\n\n    /** 2) Para cualquier otro FilterType, devuelvo Filter */\n    public getSpecificFilter(type: FilterType): Filter | null;\n    public getSpecificFilter(type: FilterType): Filter | null {\n        const filter = this.filters.find(f => f.type() === type);\n        if (type === 'price_range' && filter && !(filter instanceof Filter)) {\n            console.error(`Expected a Filter instance but got`, filter);\n        }\n        return filter || null;\n    }\n\n    /*Obtiene el filtro elegido en el select, lo pasa al setSelect\n    para que se marque seleccionado*/\n    onSelect(event: any, filter: FilterElement[] | null) {\n        // console.log(filter)\n        /*  const selectedOption = filter\n             .flatMap(category => category.children)\n             .filter(option => option?.code === event.target.value)\n             .pop()\n         selectedOption && this._filtersService.setFilterSelected(this.filters[0], selectedOption) */\n    }\n\n    setSelected(filter: Filter | null, selected: FilterElement | null) {\n        if (!filter || !selected) {\n            console.error('Filter or selected element is undefined:', { filter, selected });\n            return;\n        }\n\n        if (typeof (filter as any).setSelected !== 'function') {\n            console.error('filter.setSelected is not a function. Filter might not be an instance of the expected class:', filter);\n            return;\n        }\n\n        try {\n            // this._filtersService.setFilterSelected(filter, selected);\n            if (filter.type() === 'categories') {\n                if (selected.path) {\n                    this.router.navigate([selected.path]);\n                }\n            } else if (filter.type() === 'attributes') {\n                // Manejar la navegación para atributos\n                this._filtersService.setFilterSelected(filter, selected);\n            }\n\n\n        } catch (error) {\n            console.error(\"Error while setting selected filter:\", error);\n        }\n    }\n\n    uniqueSelection(filter: { data: any[]; }, filterChild: any) {\n        // Buscar si hay un filtro de tipo PriceRangeFilter en la lista de filtros\n        const priceRangeFilter = this.filters.find(f => f instanceof PriceRangeFilter);\n\n        if (priceRangeFilter) {\n            priceRangeFilter.reset();\n        }\n        filter.data.forEach(filterElement => {\n            if (filterElement.selected) {\n                filterElement.selected = false;\n            }\n            filterElement.children.forEach((filterChildren: FilterElement) => {\n                if (filterChildren != filterChild && filterChildren.selected) {\n                    filterChildren.selected = false;\n                }\n            });\n        });\n    }\n\n    //  close = () => {\n    //       this.document.getElementById(\"filtros\").classList.remove('in');\n    //       return true;\n    //  };\n\n\n    close = () => {\n        // this.consts.mobile()\n        //      ? this.document.getElementById(\"accordionExample\").classList.remove('show')\n        //      : this.document.getElementById(accordion_id).classList.remove('show');\n        return true;\n    };\n\n\n    scrollUp = () => {\n        window.scroll(0, 0);\n        return true\n    }\n\n\n\n\n\n    hasAppliedFilters(): boolean {\n        return this.filters?.some((filter: any) => {\n            if (filter.type?.() === 'price_range') {\n                const minSet = filter.currentMinPrice !== null && filter.currentMinPrice !== 0;\n                const maxSet = filter.currentMaxPrice !== null && filter.currentMaxPrice !== filter.maxPrice;\n                if (minSet || maxSet) {\n                    return true;\n                }\n            }\n\n            return filter.data?.some((filterElement: any) => {\n                if (Array.isArray(filterElement.children)) {\n                    return filterElement.children.some((child: any) => child.selected);\n                }\n                return filterElement.selected;\n            });\n        }) ?? false;\n    }\n/**\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    hasVisibleProperty(category: FilterElement): boolean {\n        return (category as any).isVisible === true;\n    }\n}\n","<p>filters-ec works!</p>\n"]}
|