ng-easycommerce-v18 0.3.22-beta.1 → 0.3.22

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.
Files changed (33) hide show
  1. package/README.md +8 -2
  2. package/esm2022/lib/classes/filters/attributes-filter.mjs +74 -4
  3. package/esm2022/lib/classes/filters/category-filter.mjs +105 -26
  4. package/esm2022/lib/classes/filters/filter-factory.mjs +7 -3
  5. package/esm2022/lib/classes/filters/filter.mjs +27 -2
  6. package/esm2022/lib/classes/filters/price_range-filter.mjs +3 -3
  7. package/esm2022/lib/constants/core.constants.service.mjs +12 -1
  8. package/esm2022/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.mjs +5 -3
  9. package/esm2022/lib/ec-components/collection-ec/collection-ec.component.mjs +41 -17
  10. package/esm2022/lib/ec-components/filters-ec/filters-ec.component.mjs +42 -9
  11. package/esm2022/lib/ec-components/header-ec/header-ec.component.mjs +42 -27
  12. package/esm2022/lib/ec-components/price-range-filter/price-range-filter.component.mjs +13 -2
  13. package/esm2022/lib/ec-components/related-products-ec/related-products-ec.component.mjs +6 -4
  14. package/esm2022/lib/ec-components/widgets-ec/magnizoom-ec/magnizoom-ec.component.mjs +4 -2
  15. package/esm2022/lib/ec-services/analytics/facebook-pixel.service.mjs +4 -2
  16. package/esm2022/lib/ec-services/analytics/google-analytics.service.mjs +4 -2
  17. package/esm2022/lib/ec-services/filters.service.mjs +124 -18
  18. package/esm2022/lib/ec-services/options.service.mjs +27 -3
  19. package/esm2022/lib/ec-services/pagination.service.mjs +70 -22
  20. package/esm2022/lib/ec-services/products.service.mjs +5 -3
  21. package/fesm2022/ng-easycommerce-v18.mjs +586 -126
  22. package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
  23. package/lib/classes/filters/attributes-filter.d.ts +24 -0
  24. package/lib/classes/filters/category-filter.d.ts +30 -3
  25. package/lib/constants/core.constants.service.d.ts +7 -0
  26. package/lib/ec-components/collection-ec/collection-ec.component.d.ts +5 -4
  27. package/lib/ec-components/filters-ec/filters-ec.component.d.ts +13 -0
  28. package/lib/ec-components/price-range-filter/price-range-filter.component.d.ts +2 -0
  29. package/lib/ec-services/filters.service.d.ts +18 -1
  30. package/lib/ec-services/options.service.d.ts +4 -0
  31. package/lib/ec-services/pagination.service.d.ts +21 -5
  32. package/lib/ec-services/products.service.d.ts +1 -1
  33. package/package.json +1 -1
@@ -30,6 +30,11 @@ export class CoreConstantsService {
30
30
  * Guarda la variable window del web browser.
31
31
  */
32
32
  window;
33
+ /**
34
+ * Tipo de ruta actual (ej: 'categories', 'attributes', etc).
35
+ * Lo usamos para que PaginationService sepa qué filtro esperar.
36
+ */
37
+ currentRouteType = null;
33
38
  constructor() {
34
39
  if (isPlatformBrowser(this.platformId)) {
35
40
  this.window = window;
@@ -307,6 +312,12 @@ export class CoreConstantsService {
307
312
  return this.apiConstants.CHANNEL;
308
313
  }
309
314
  form_sender = false;
315
+ setCurrentRouteType(type) {
316
+ this.currentRouteType = type;
317
+ }
318
+ getCurrentRouteType() {
319
+ return this.currentRouteType;
320
+ }
310
321
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CoreConstantsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
311
322
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CoreConstantsService, providedIn: 'root' });
312
323
  }
@@ -316,4 +327,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
316
327
  providedIn: 'root'
317
328
  }]
318
329
  }], ctorParameters: () => [] });
319
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"core.constants.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/constants/core.constants.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable, PLATFORM_ID } from '@angular/core';\nimport { ApiConstantsService } from './api.constants.service';\nimport { isPlatformBrowser, DOCUMENT } from '@angular/common';\nimport { CurrencyConfig, currencyType, ParamsProductsWithUniqueVariant, Environment } from '../interfaces';\nimport { ENVIRONMENT_TOKEN } from '../providers';\n/**\n * Servicio que provee las constantes necesarias que se utilizan por toda la aplicación.\n * @export\n * @class CoreConstantsService\n */\n@Injectable({\n\tprovidedIn: 'root'\n})\nexport class CoreConstantsService {\n\n\tpublic searchValue: string = '';\n\t/**\n\t * Servicio que contiene las constantes usadas para la API\n\t */\n\tprivate apiConstants = inject(ApiConstantsService)\n\t/**\n\t * Provee el ID de la plataforma en donde se esta ejecutando el codigo, esto sirve para poder diferenciar si es de un servidor o un navegador web.\n\t */\n\tprivate platformId: any = inject(PLATFORM_ID)\n\t/**\n\t * Document token para acceso SSR-compatible al documento\n\t */\n\tprivate document: Document = inject(DOCUMENT)\n\t/**\n\t * Contiene los datos provisto por el frontend en el archivo environment.ts\n\t */\n\tprivate environment: Environment = inject(ENVIRONMENT_TOKEN)\n\t/**\n\t * Guarda la variable window del web browser.\n\t */\n\tprivate window?:Window;\n\n\tconstructor() {\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\tthis.window = window\n\t\t}\n\t}\n\t/**\n\t * Path usado en las colecciones de productos.\n\t */\n\tpublic PRODUCT_LIST_ROUTE: string = 'collection/';\n\t/**\n\t * Path usado para acceder a las secciones.\n\t */\n\tpublic SECTIONS_ROUTE: string = 'sections/';\n\t/**\n\t * Path que aloja la carpeta de los assets.\n\t */\n\tpublic readonly ASSETS_URL: string = this.apiConstants.API_URL;\n\t/**\n\t * Retorna la url donde se guarda las imagenes.\n\t * @param postMedia \n\t * @returns \n\t */\n\tpublic mediaUrl = (postMedia?: string) => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/' : '') + (postMedia ? postMedia : \"image/\");\n\t/**\n\t * Retorna la url donde se guarda las imagenes.\n\t * @param postMedia \n\t * @returns \n\t */\n\tpublic mediaImageUrl = (postMedia?: string) => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/image/' : '') + (postMedia ? postMedia : \"image/\");\n\t/**\n\t * Retorna la url del sitio.\n\t * @param postMedia \n\t * @returns \n\t */\n\tpublic url = () => this.apiConstants.API_URL.replace(/\\/$/, ''); // remove last slash\t\n\n\t/**\n\t * URL del sitio frontend - Compatible con Angular SSR\n\t */\n\tpublic get FRONTEND_URL(): string {\n\t\t// Verificar si estamos en el navegador\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\t// Primero intenta leer de window.__env (configurado por Docker/CI)\n\t\t\tconst windowEnv = (this.window as any)?.__env;\n\t\t\tif (windowEnv?.frontendUrl) {\n\t\t\t\t// Log para debugging en desarrollo\n\t\t\t\tif (!windowEnv.frontendUrl.startsWith('http')) {\n\t\t\t\t\tconsole.warn('[FRONTEND_URL] Invalid frontendUrl format:', windowEnv.frontendUrl);\n\t\t\t\t}\n\t\t\t\treturn windowEnv.frontendUrl;\n\t\t\t}\n\t\t\t\n\t\t\t// Si no hay configuración específica, construir desde window.location\n\t\t\tif (this.window?.location) {\n\t\t\t\tconst url = `${this.window.location.protocol}//${this.window.location.host}`;\n\t\t\t\t// Verificar que no sea localhost o 127.0.0.1 en producción\n\t\t\t\tif (url.includes('127.0.0.1') || url.includes('localhost')) {\n\t\t\t\t\tconsole.warn('[FRONTEND_URL] Using localhost URL in production, consider setting window.__env.frontendUrl');\n\t\t\t\t}\n\t\t\t\treturn url;\n\t\t\t}\n\t\t}\n\t\t\n\t\t// Para SSR, intentar construir desde el document si está disponible\n\t\tif (this.document?.location) {\n\t\t\treturn `${this.document.location.protocol}//${this.document.location.host}`;\n\t\t}\n\t\t\n\t\t// Intentar obtener desde environment\n\t\tif (this.environment.frontendUrl) {\n\t\t\treturn this.environment.frontendUrl;\n\t\t}\n\t\t\n\t\t// Último fallback - usar la URL del API si está configurada y es absoluta\n\t\tconst apiUrl = this.apiConstants.API_URL;\n\t\tif (apiUrl && (apiUrl.startsWith('http://') || apiUrl.startsWith('https://'))) {\n\t\t\ttry {\n\t\t\t\tconst url = new URL(apiUrl);\n\t\t\t\treturn `${url.protocol}//${url.host}`;\n\t\t\t} catch (e) {\n\t\t\t\tconsole.warn('[FRONTEND_URL] Could not parse API URL as fallback:', apiUrl);\n\t\t\t}\n\t\t}\n\t\t\n\t\t// Fallback final - retornar URL vacía\n\t\tconsole.warn('[FRONTEND_URL] No valid frontend URL found, this may cause issues with meta tags');\n\t\treturn '';\n\t}\n\n\t/**\n\t * Retorna la URL completa del frontend con una ruta opcional\n\t * @param path - Ruta opcional para agregar a la URL base\n\t * @returns {string} URL completa del frontend\n\t */\n\tpublic getFrontendUrl = (path?: string): string => {\n\t\tconst baseUrl = this.FRONTEND_URL;\n\t\tif (!path) {\n\t\t\treturn baseUrl;\n\t\t}\n\t\t\n\t\t// Asegurar que la ruta comience con '/' y no tenga doble slash\n\t\tconst cleanPath = path.startsWith('/') ? path : `/${path}`;\n\t\tconst cleanBaseUrl = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl;\n\t\t\n\t\treturn `${cleanBaseUrl}${cleanPath}`;\n\t}\n\n\t/**\n\t * Retorna la URL absoluta de una imagen para meta tags (compatible con SSR)\n\t * @param postMedia - Ruta de la imagen\n\t * @returns {string} URL absoluta de la imagen\n\t */\n\tpublic getAbsoluteImageUrl = (postMedia?: string): string => {\n\t\t// Si la URL ya es absoluta, devolverla tal como está\n\t\tif (postMedia && (postMedia.startsWith('http://') || postMedia.startsWith('https://'))) {\n\t\t\treturn postMedia;\n\t\t}\n\n\t\t// Construir la URL de la imagen usando la URL del API\n\t\tconst imageUrl = this.mediaImageUrl(postMedia);\n\t\t\n\t\t// Si la URL de la imagen ya es absoluta, devolverla\n\t\tif (imageUrl.startsWith('http://') || imageUrl.startsWith('https://')) {\n\t\t\treturn imageUrl;\n\t\t}\n\t\t\n\t\t// Si es una URL relativa, convertirla a absoluta usando la URL base del API\n\t\tconst apiBaseUrl = this.apiConstants.API_URL;\n\t\tif (apiBaseUrl.startsWith('http://') || apiBaseUrl.startsWith('https://')) {\n\t\t\t// API_URL ya es absoluta\n\t\t\treturn imageUrl;\n\t\t}\n\t\t\n\t\t// Si API_URL es relativa, usar la URL del frontend como base\n\t\tconst frontendUrl = this.FRONTEND_URL;\n\t\tif (!frontendUrl) {\n\t\t\treturn imageUrl; // Fallback si no podemos determinar la URL del frontend\n\t\t}\n\t\t\n\t\t// Construir URL absoluta\n\t\tconst cleanApiUrl = apiBaseUrl.startsWith('/') ? apiBaseUrl.substring(1) : apiBaseUrl;\n\t\tconst cleanFrontendUrl = frontendUrl.endsWith('/') ? frontendUrl.slice(0, -1) : frontendUrl;\n\t\tconst cleanImageUrl = imageUrl.startsWith('/') ? imageUrl.substring(1) : imageUrl;\n\t\t\n\t\treturn `${cleanFrontendUrl}/${cleanApiUrl}${cleanImageUrl}`;\n\t}\n\n\t/**\n\t * Retorna la url de las imagenes de los banners\n\t * @returns \n\t */\n\tpublic mediaBannerUrl = () => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/' : '') + \"banner-image/\";\n\t\n\tmediaStoreUrl = () => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/' : '') + \"store-image/\";\n\t\n\t/**\n\t * Retorna `true` si la vista es mobile, `false` caso contrario.\n\t * @returns {boolean}\n\t */\n\tpublic mobileScreen = (): boolean => this.window && this.window?.innerWidth < 750 || false;\n\t/**\n\t * Contiene los valores para la moneda actual\n\t */\n\tpublic currency: CurrencyConfig = {\n\t\tcode: 'ARG',\n\t\tsymbol: '$',\n\t\tposition: 'Left',\n\t\twithoutDecimal: false\n\t};\n\n\t/**\n\t * Arreglo de valores usados para el pipe currencyPrice\n\t */\n\tpublic currencyTypeConfig: CurrencyConfig[] = [\n\t\t{\n\t\t\tcode: 'EUR',\n\t\t\tsymbol: '€',\n\t\t\tposition: 'Right',\n\t\t\twithoutDecimal: false\n\n\t\t},\n\t\t{\n\t\t\tcode: 'ARG',\n\t\t\tsymbol: '$',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'ARS',\n\t\t\tsymbol: '$',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'USD',\n\t\t\tsymbol: '$',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'PYG',\n\t\t\tsymbol: 'Gs.',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'PEN',\n\t\t\tsymbol: 'S/',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'SEK',\n\t\t\tsymbol: 'SEK',\n\t\t\tposition: 'Right',\n\t\t\twithoutDecimal: true\n\t\t},\n\t]\n\t/**\n\t * Cambia la configuración de la moneda.\n\t * @param currencyTypeConfig Arreglo con las distintas monedas.\n\t */\n\tpublic setCurrencyTypeConfig(currencyTypeConfig: CurrencyConfig[]) {\n\t\tthis.currencyTypeConfig = currencyTypeConfig;\n\t}\n\n\t/**\n\t * @param currencyTypeFind Definir el code del tipo de moneda que se quiera utilizar.\n\t * Si el tipo es CUSTOM se toman los valores del segundo parametro donde se puede definir la simbología y posesión que se quiera.\n\t * @param customType Formato del parámetro ejemplo { code: 'ARG', symbol: '$', position: 'L' , withoutDecimals: false } ,\n\t * todas las key son opcionales y por defecto se toman los valores del ejemplo.\n\t * @returns se retorna la configuración que se seteo a la propiedad currency del servicio.\n\t */\n\tpublic setCurrency(currencyTypeFind: currencyType, customType?: CurrencyConfig): CurrencyConfig {\n\t\tthis.currency = this.currencyTypeConfig.find((ct: CurrencyConfig) => ct.code == currencyTypeFind.toUpperCase()) || this.currency\n\t\tcustomType && (this.currency = { ...this.currency, ...customType })\n\t\treturn this.currency;\n\t};\n\n\n\tpublic modifyChannelConfig(channelConfig: any) {\n\t\tif (channelConfig.channel || channelConfig.code)\n\t\t\tthis.apiConstants.setChannel(channelConfig.channel || channelConfig.code)\n\n\t\tif (channelConfig.locale || channelConfig.locales.code)\n\t\t\tthis.apiConstants.setLocale(channelConfig.locale || channelConfig.locales.code)\n\n\t\tif (channelConfig.currency || channelConfig.baseCurrency)\n\t\t\tthis.setCurrency(channelConfig.currency || channelConfig.baseCurrency)\n\n\t\treturn channelConfig;\n\t}\n\n\t/**\n\t* @description se utiliza para configurar que filtros del tipo `SortFilter` se quiere mostrar en la vista.\n\t* Se usan los codigo de los filtros para indicar cuales se deben mostrar.\n\t*/\n\tprivate sortFilters: string[] = ['price', 'order-alph', 'recent']\n\n\t/**\n\t *\n\t * @param filtersCode arreglo de strings con los codigos de los filtros.\n\t * Si se le pasa el parametro **all** este mostrara todos.\n\t * @example price | order-alph | recent | update | all\n\t * @returns\n\t */\n\tsetSortFilters = (filtersCode: string[] = ['all']) => this.sortFilters = filtersCode;\n\n\tgetSortFilters = () => this.sortFilters;\n\n\t// ABOUT PRODUCTS \n\tprivate productsWithUniqueVariant: boolean = false;\n\tsetProductWithUniqueVariant(value: boolean) { this.productsWithUniqueVariant = value; }\n\tgetProductWithUniqueVariant() { return this.productsWithUniqueVariant }\n\n\t/**\n\t* @description esta variable se utiliza para configurar los parametros cuando se desea mostrar\n\t* variantes el listado de productos.\n\t* @param {string}type tipo de options\n\t* @param {string}code codigo de options\n\t* @param {boolean}defaultFirstOption `true` toma el primer option del producto en general\n\t* @param {boolean}checkIfStock `true` se realiza un chequeo de si la variante tiene stock para mostrarla\n\t* @example  ```javascript\n\t* {\n\t   type: 'color',\n\t   code: 'color',\n\t   defaultFirstOption: false ,\n\t   checkIfStock:true\n\t   } ```\n\t*/\n\tprivate paramsProductsWithUniqueVariant: ParamsProductsWithUniqueVariant = {\n\t\ttype: 'color',\n\t\tcode: 'color',\n\t\tdefaultFirstOption: false,\n\t\tcheckIfStock: true\n\t}\n\n\tsetParamsProductsWithUniqueVariant = (value: ParamsProductsWithUniqueVariant) => {\n\t\tthis.paramsProductsWithUniqueVariant = {\n\t\t\t...this.paramsProductsWithUniqueVariant,\n\t\t\t...value\n\t\t}\n\t}\n\n\tgetParamsProductsWithUniqueVariant = () => this.paramsProductsWithUniqueVariant\n\t\n\tpublic getChannel(){\n\t\treturn this.apiConstants.CHANNEL;\n\t}\n\n\tpublic form_sender: boolean = false;\n}\n"]}
330
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"core.constants.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/constants/core.constants.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable, PLATFORM_ID } from '@angular/core';\nimport { ApiConstantsService } from './api.constants.service';\nimport { isPlatformBrowser, DOCUMENT } from '@angular/common';\nimport { CurrencyConfig, currencyType, ParamsProductsWithUniqueVariant, Environment } from '../interfaces';\nimport { ENVIRONMENT_TOKEN } from '../providers';\n/**\n * Servicio que provee las constantes necesarias que se utilizan por toda la aplicación.\n * @export\n * @class CoreConstantsService\n */\n@Injectable({\n\tprovidedIn: 'root'\n})\nexport class CoreConstantsService {\n\n\tpublic searchValue: string = '';\n\t/**\n\t * Servicio que contiene las constantes usadas para la API\n\t */\n\tprivate apiConstants = inject(ApiConstantsService)\n\t/**\n\t * Provee el ID de la plataforma en donde se esta ejecutando el codigo, esto sirve para poder diferenciar si es de un servidor o un navegador web.\n\t */\n\tprivate platformId: any = inject(PLATFORM_ID)\n\t/**\n\t * Document token para acceso SSR-compatible al documento\n\t */\n\tprivate document: Document = inject(DOCUMENT)\n\t/**\n\t * Contiene los datos provisto por el frontend en el archivo environment.ts\n\t */\n\tprivate environment: Environment = inject(ENVIRONMENT_TOKEN)\n\t/**\n\t * Guarda la variable window del web browser.\n\t */\n\tprivate window?:Window;\n\n\t/**\n\t * Tipo de ruta actual (ej: 'categories', 'attributes', etc).\n\t * Lo usamos para que PaginationService sepa qué filtro esperar.\n\t */\n\tpublic currentRouteType: string | null = null;\n\n\tconstructor() {\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\tthis.window = window\n\t\t}\n\t}\n\t/**\n\t * Path usado en las colecciones de productos.\n\t */\n\tpublic PRODUCT_LIST_ROUTE: string = 'collection/';\n\t/**\n\t * Path usado para acceder a las secciones.\n\t */\n\tpublic SECTIONS_ROUTE: string = 'sections/';\n\t/**\n\t * Path que aloja la carpeta de los assets.\n\t */\n\tpublic readonly ASSETS_URL: string = this.apiConstants.API_URL;\n\t/**\n\t * Retorna la url donde se guarda las imagenes.\n\t * @param postMedia \n\t * @returns \n\t */\n\tpublic mediaUrl = (postMedia?: string) => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/' : '') + (postMedia ? postMedia : \"image/\");\n\t/**\n\t * Retorna la url donde se guarda las imagenes.\n\t * @param postMedia \n\t * @returns \n\t */\n\tpublic mediaImageUrl = (postMedia?: string) => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/image/' : '') + (postMedia ? postMedia : \"image/\");\n\t/**\n\t * Retorna la url del sitio.\n\t * @param postMedia \n\t * @returns \n\t */\n\tpublic url = () => this.apiConstants.API_URL.replace(/\\/$/, ''); // remove last slash\t\n\n\t/**\n\t * URL del sitio frontend - Compatible con Angular SSR\n\t */\n\tpublic get FRONTEND_URL(): string {\n\t\t// Verificar si estamos en el navegador\n\t\tif (isPlatformBrowser(this.platformId)) {\n\t\t\t// Primero intenta leer de window.__env (configurado por Docker/CI)\n\t\t\tconst windowEnv = (this.window as any)?.__env;\n\t\t\tif (windowEnv?.frontendUrl) {\n\t\t\t\t// Log para debugging en desarrollo\n\t\t\t\tif (!windowEnv.frontendUrl.startsWith('http')) {\n\t\t\t\t\tconsole.warn('[FRONTEND_URL] Invalid frontendUrl format:', windowEnv.frontendUrl);\n\t\t\t\t}\n\t\t\t\treturn windowEnv.frontendUrl;\n\t\t\t}\n\t\t\t\n\t\t\t// Si no hay configuración específica, construir desde window.location\n\t\t\tif (this.window?.location) {\n\t\t\t\tconst url = `${this.window.location.protocol}//${this.window.location.host}`;\n\t\t\t\t// Verificar que no sea localhost o 127.0.0.1 en producción\n\t\t\t\tif (url.includes('127.0.0.1') || url.includes('localhost')) {\n\t\t\t\t\tconsole.warn('[FRONTEND_URL] Using localhost URL in production, consider setting window.__env.frontendUrl');\n\t\t\t\t}\n\t\t\t\treturn url;\n\t\t\t}\n\t\t}\n\t\t\n\t\t// Para SSR, intentar construir desde el document si está disponible\n\t\tif (this.document?.location) {\n\t\t\treturn `${this.document.location.protocol}//${this.document.location.host}`;\n\t\t}\n\t\t\n\t\t// Intentar obtener desde environment\n\t\tif (this.environment.frontendUrl) {\n\t\t\treturn this.environment.frontendUrl;\n\t\t}\n\t\t\n\t\t// Último fallback - usar la URL del API si está configurada y es absoluta\n\t\tconst apiUrl = this.apiConstants.API_URL;\n\t\tif (apiUrl && (apiUrl.startsWith('http://') || apiUrl.startsWith('https://'))) {\n\t\t\ttry {\n\t\t\t\tconst url = new URL(apiUrl);\n\t\t\t\treturn `${url.protocol}//${url.host}`;\n\t\t\t} catch (e) {\n\t\t\t\tconsole.warn('[FRONTEND_URL] Could not parse API URL as fallback:', apiUrl);\n\t\t\t}\n\t\t}\n\t\t\n\t\t// Fallback final - retornar URL vacía\n\t\tconsole.warn('[FRONTEND_URL] No valid frontend URL found, this may cause issues with meta tags');\n\t\treturn '';\n\t}\n\n\t/**\n\t * Retorna la URL completa del frontend con una ruta opcional\n\t * @param path - Ruta opcional para agregar a la URL base\n\t * @returns {string} URL completa del frontend\n\t */\n\tpublic getFrontendUrl = (path?: string): string => {\n\t\tconst baseUrl = this.FRONTEND_URL;\n\t\tif (!path) {\n\t\t\treturn baseUrl;\n\t\t}\n\t\t\n\t\t// Asegurar que la ruta comience con '/' y no tenga doble slash\n\t\tconst cleanPath = path.startsWith('/') ? path : `/${path}`;\n\t\tconst cleanBaseUrl = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl;\n\t\t\n\t\treturn `${cleanBaseUrl}${cleanPath}`;\n\t}\n\n\t/**\n\t * Retorna la URL absoluta de una imagen para meta tags (compatible con SSR)\n\t * @param postMedia - Ruta de la imagen\n\t * @returns {string} URL absoluta de la imagen\n\t */\n\tpublic getAbsoluteImageUrl = (postMedia?: string): string => {\n\t\t// Si la URL ya es absoluta, devolverla tal como está\n\t\tif (postMedia && (postMedia.startsWith('http://') || postMedia.startsWith('https://'))) {\n\t\t\treturn postMedia;\n\t\t}\n\n\t\t// Construir la URL de la imagen usando la URL del API\n\t\tconst imageUrl = this.mediaImageUrl(postMedia);\n\t\t\n\t\t// Si la URL de la imagen ya es absoluta, devolverla\n\t\tif (imageUrl.startsWith('http://') || imageUrl.startsWith('https://')) {\n\t\t\treturn imageUrl;\n\t\t}\n\t\t\n\t\t// Si es una URL relativa, convertirla a absoluta usando la URL base del API\n\t\tconst apiBaseUrl = this.apiConstants.API_URL;\n\t\tif (apiBaseUrl.startsWith('http://') || apiBaseUrl.startsWith('https://')) {\n\t\t\t// API_URL ya es absoluta\n\t\t\treturn imageUrl;\n\t\t}\n\t\t\n\t\t// Si API_URL es relativa, usar la URL del frontend como base\n\t\tconst frontendUrl = this.FRONTEND_URL;\n\t\tif (!frontendUrl) {\n\t\t\treturn imageUrl; // Fallback si no podemos determinar la URL del frontend\n\t\t}\n\t\t\n\t\t// Construir URL absoluta\n\t\tconst cleanApiUrl = apiBaseUrl.startsWith('/') ? apiBaseUrl.substring(1) : apiBaseUrl;\n\t\tconst cleanFrontendUrl = frontendUrl.endsWith('/') ? frontendUrl.slice(0, -1) : frontendUrl;\n\t\tconst cleanImageUrl = imageUrl.startsWith('/') ? imageUrl.substring(1) : imageUrl;\n\t\t\n\t\treturn `${cleanFrontendUrl}/${cleanApiUrl}${cleanImageUrl}`;\n\t}\n\n\t/**\n\t * Retorna la url de las imagenes de los banners\n\t * @returns \n\t */\n\tpublic mediaBannerUrl = () => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/' : '') + \"banner-image/\";\n\t\n\tmediaStoreUrl = () => this.ASSETS_URL + (this.ASSETS_URL == this.apiConstants.API_URL ? 'media/' : '') + \"store-image/\";\n\t\n\t/**\n\t * Retorna `true` si la vista es mobile, `false` caso contrario.\n\t * @returns {boolean}\n\t */\n\tpublic mobileScreen = (): boolean => this.window && this.window?.innerWidth < 750 || false;\n\t/**\n\t * Contiene los valores para la moneda actual\n\t */\n\tpublic currency: CurrencyConfig = {\n\t\tcode: 'ARG',\n\t\tsymbol: '$',\n\t\tposition: 'Left',\n\t\twithoutDecimal: false\n\t};\n\n\t/**\n\t * Arreglo de valores usados para el pipe currencyPrice\n\t */\n\tpublic currencyTypeConfig: CurrencyConfig[] = [\n\t\t{\n\t\t\tcode: 'EUR',\n\t\t\tsymbol: '€',\n\t\t\tposition: 'Right',\n\t\t\twithoutDecimal: false\n\n\t\t},\n\t\t{\n\t\t\tcode: 'ARG',\n\t\t\tsymbol: '$',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'ARS',\n\t\t\tsymbol: '$',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'USD',\n\t\t\tsymbol: '$',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'PYG',\n\t\t\tsymbol: 'Gs.',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'PEN',\n\t\t\tsymbol: 'S/',\n\t\t\tposition: 'Left',\n\t\t\twithoutDecimal: false\n\t\t},\n\t\t{\n\t\t\tcode: 'SEK',\n\t\t\tsymbol: 'SEK',\n\t\t\tposition: 'Right',\n\t\t\twithoutDecimal: true\n\t\t},\n\t]\n\t/**\n\t * Cambia la configuración de la moneda.\n\t * @param currencyTypeConfig Arreglo con las distintas monedas.\n\t */\n\tpublic setCurrencyTypeConfig(currencyTypeConfig: CurrencyConfig[]) {\n\t\tthis.currencyTypeConfig = currencyTypeConfig;\n\t}\n\n\t/**\n\t * @param currencyTypeFind Definir el code del tipo de moneda que se quiera utilizar.\n\t * Si el tipo es CUSTOM se toman los valores del segundo parametro donde se puede definir la simbología y posesión que se quiera.\n\t * @param customType Formato del parámetro ejemplo { code: 'ARG', symbol: '$', position: 'L' , withoutDecimals: false } ,\n\t * todas las key son opcionales y por defecto se toman los valores del ejemplo.\n\t * @returns se retorna la configuración que se seteo a la propiedad currency del servicio.\n\t */\n\tpublic setCurrency(currencyTypeFind: currencyType, customType?: CurrencyConfig): CurrencyConfig {\n\t\tthis.currency = this.currencyTypeConfig.find((ct: CurrencyConfig) => ct.code == currencyTypeFind.toUpperCase()) || this.currency\n\t\tcustomType && (this.currency = { ...this.currency, ...customType })\n\t\treturn this.currency;\n\t};\n\n\n\tpublic modifyChannelConfig(channelConfig: any) {\n\t\tif (channelConfig.channel || channelConfig.code)\n\t\t\tthis.apiConstants.setChannel(channelConfig.channel || channelConfig.code)\n\n\t\tif (channelConfig.locale || channelConfig.locales.code)\n\t\t\tthis.apiConstants.setLocale(channelConfig.locale || channelConfig.locales.code)\n\n\t\tif (channelConfig.currency || channelConfig.baseCurrency)\n\t\t\tthis.setCurrency(channelConfig.currency || channelConfig.baseCurrency)\n\n\t\treturn channelConfig;\n\t}\n\n\t/**\n\t* @description se utiliza para configurar que filtros del tipo `SortFilter` se quiere mostrar en la vista.\n\t* Se usan los codigo de los filtros para indicar cuales se deben mostrar.\n\t*/\n\tprivate sortFilters: string[] = ['price', 'order-alph', 'recent']\n\n\t/**\n\t *\n\t * @param filtersCode arreglo de strings con los codigos de los filtros.\n\t * Si se le pasa el parametro **all** este mostrara todos.\n\t * @example price | order-alph | recent | update | all\n\t * @returns\n\t */\n\tsetSortFilters = (filtersCode: string[] = ['all']) => this.sortFilters = filtersCode;\n\n\tgetSortFilters = () => this.sortFilters;\n\n\t// ABOUT PRODUCTS \n\tprivate productsWithUniqueVariant: boolean = false;\n\tsetProductWithUniqueVariant(value: boolean) { this.productsWithUniqueVariant = value; }\n\tgetProductWithUniqueVariant() { return this.productsWithUniqueVariant }\n\n\t/**\n\t* @description esta variable se utiliza para configurar los parametros cuando se desea mostrar\n\t* variantes el listado de productos.\n\t* @param {string}type tipo de options\n\t* @param {string}code codigo de options\n\t* @param {boolean}defaultFirstOption `true` toma el primer option del producto en general\n\t* @param {boolean}checkIfStock `true` se realiza un chequeo de si la variante tiene stock para mostrarla\n\t* @example  ```javascript\n\t* {\n\t   type: 'color',\n\t   code: 'color',\n\t   defaultFirstOption: false ,\n\t   checkIfStock:true\n\t   } ```\n\t*/\n\tprivate paramsProductsWithUniqueVariant: ParamsProductsWithUniqueVariant = {\n\t\ttype: 'color',\n\t\tcode: 'color',\n\t\tdefaultFirstOption: false,\n\t\tcheckIfStock: true\n\t}\n\n\tsetParamsProductsWithUniqueVariant = (value: ParamsProductsWithUniqueVariant) => {\n\t\tthis.paramsProductsWithUniqueVariant = {\n\t\t\t...this.paramsProductsWithUniqueVariant,\n\t\t\t...value\n\t\t}\n\t}\n\n\tgetParamsProductsWithUniqueVariant = () => this.paramsProductsWithUniqueVariant\n\t\n\tpublic getChannel(){\n\t\treturn this.apiConstants.CHANNEL;\n\t}\n\n\tpublic form_sender: boolean = false;\n\n\tsetCurrentRouteType(type: string | null) {\n\t\tthis.currentRouteType = type;\n\t}\n\n\tgetCurrentRouteType(): string | null {\n\t\treturn this.currentRouteType;\n\t}\n}\n"]}
@@ -1,7 +1,7 @@
1
1
  import { Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal } from '@angular/core';
2
2
  import { BlockEcComponent } from '../../abstractions-components';
3
3
  import { AnalyticsService } from '../../../ec-services';
4
- import { CommonModule } from '@angular/common';
4
+ import { CommonModule, isPlatformBrowser } from '@angular/common';
5
5
  // import function to register Swiper custom elements
6
6
  import { register } from 'swiper/element/bundle';
7
7
  import { ProductEcComponent } from "../../product-ec/product-ec.component";
@@ -66,7 +66,7 @@ export class BlockProductsEcComponent extends BlockEcComponent {
66
66
  * Permite personalización de las imágenes de las flechas mediante @Input.
67
67
  */
68
68
  setupSwiperNavigation() {
69
- if (this.meta?.styles?.carrousel !== false) {
69
+ if (this.meta?.styles?.carrousel !== false && isPlatformBrowser(this.platformId)) {
70
70
  // Usar setTimeout para asegurar que el swiper esté inicializado
71
71
  setTimeout(() => {
72
72
  this.initializeSwiperWithCustomNavigation();
@@ -78,6 +78,8 @@ export class BlockProductsEcComponent extends BlockEcComponent {
78
78
  * Esta función puede ser movida al componente base para reutilización.
79
79
  */
80
80
  initializeSwiperWithCustomNavigation() {
81
+ if (!isPlatformBrowser(this.platformId))
82
+ return;
81
83
  const prevButton = document.getElementById(`${this.meta?.code}-prev`);
82
84
  const nextButton = document.getElementById(`${this.meta?.code}-next`);
83
85
  const swiperElement = document.getElementById(this.meta?.code);
@@ -216,4 +218,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
216
218
  required: true
217
219
  }]
218
220
  }] } });
219
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block-products-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, AfterViewInit, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal, TemplateRef } from '@angular/core';\nimport { BlockEcComponent } from '../../abstractions-components';\nimport { SwiperOptions } from 'swiper/types';\nimport { AnalyticsService } from '../../../ec-services';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\n// import function to register Swiper custom elements\nimport { register, SwiperContainer } from 'swiper/element/bundle';\nimport { ProductEcComponent } from \"../../product-ec/product-ec.component\";\n\nregister()\n/**\n * Componen para manejar los bloques de productos.\n * @extends {BlockEcComponent}\n * @class BlockProductsEcComponent\n */\n@Component({\n\tselector: 'app-block-products-ec',\n\tstandalone: true,\n\timports: [CommonModule, ProductEcComponent],\n\ttemplateUrl: './block-products-ec.component.html',\n\tstyleUrl: './block-products-ec.component.scss',\n\tschemas: [CUSTOM_ELEMENTS_SCHEMA]\n})\nexport class BlockProductsEcComponent extends BlockEcComponent implements AfterViewInit {\n\t\t// Personalización de las flechas de navegación\n\t// Por defecto usa símbolos de texto, pero se pueden especificar imágenes\n\t@Input() prevArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() nextArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() prevArrowText: string = '<';\n\t@Input() nextArrowText: string = '>';\n\t/**\n\t * Servicio de Analytics\n\t */\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\n\t/**\n\t * Signal utlizado para guarda el contenedor del carrusel\n\t */\n\tswiperElement = signal<SwiperContainer | null>(null)\n\t/**\n\t * Input que recibe un template para el producto.\n\t */\n\t@Input() appProduct: TemplateRef<any> | undefined;\n\t/**\n\t * Colección de productos.\n\t */\n\t@Input({\n\t\trequired: true\n\t}) products: any; // Product[];\n\t/**\n\t * Bloque principal que contiene los productos\n\t */\n\t@Input({\n\t\trequired: true\n\t}) meta: any;\n\n\tngAfterViewInit() {\n\t\tthis.setupSwiperNavigation();\n\t}\n\t\n\tprivate document?: Document;\n\tprivate platformId: any = inject(PLATFORM_ID)\n\n\t/**\n\t * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias \n\t * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente\n\t * en el Dom el element `<swiper-container>` para poder configurarlo.\n\t */\n\tconstructor() {\n\t\tsuper()\n\t}\n\t/**\n\t * Aplica el evento `select_promotion` junto con el banner que interactua.\n\t * @param banner \n\t */\n\tselectPromotion(item: any): void {\n\t\tthis.analyticsService.callEvent('select_promotion', item)\n\t}\n\n\t/**\n\t * Configura la navegación personalizada del Swiper.\n\t * Esta función está diseñada para ser movida al componente base BlockProductsEcComponent.\n\t * Permite personalización de las imágenes de las flechas mediante @Input.\n\t */\n\tprivate setupSwiperNavigation() {\n\t\tif (this.meta?.styles?.carrousel !== false) {\n\t\t\t// Usar setTimeout para asegurar que el swiper esté inicializado\n\t\t\tsetTimeout(() => {\n\t\t\t\tthis.initializeSwiperWithCustomNavigation();\n\t\t\t}, 200);\n\t\t}\n\t}\n\n\t/**\n\t * Inicializa el Swiper con navegación personalizada.\n\t * Esta función puede ser movida al componente base para reutilización.\n\t */\n\tprivate initializeSwiperWithCustomNavigation() {\n\t\tconst prevButton = document.getElementById(`${this.meta?.code}-prev`);\n\t\tconst nextButton = document.getElementById(`${this.meta?.code}-next`);\n\t\tconst swiperElement = document.getElementById(this.meta?.code) as any;\n\n\t\tif (prevButton && nextButton && swiperElement) {\n\t\t\t// console.log('Configurando navegación personalizada para:', this.meta?.code);\n\t\t\t\n\t\t\tconst swiperConfig = this.getSwiperConfiguration();\n\n\t\t\t// Verificar si el Swiper ya está inicializado\n\t\t\tif (!swiperElement.swiper) {\n\t\t\t\tthis.initializeNewSwiper(swiperElement, swiperConfig);\n\t\t\t} else {\n\t\t\t\tthis.updateExistingSwiper(swiperElement);\n\t\t\t}\n\n\t\t\t// Configurar los event listeners para los botones\n\t\t\tthis.setupNavigationEventListeners(prevButton, nextButton, swiperElement);\n\t\t\t\n\t\t\t// console.log('Event listeners configurados para los botones de navegación');\n\t\t} else {\n\t\t\t// console.log('No se pudieron encontrar los elementos:', {\n\t\t\t// \tprevButton: !!prevButton,\n\t\t\t// \tnextButton: !!nextButton,\n\t\t\t// \tswiperElement: !!swiperElement\n\t\t\t// });\n\t\t}\n\t}\n\n\t/**\n\t * Obtiene la configuración base del Swiper.\n\t * Esta configuración puede ser personalizada en el futuro mediante @Input.\n\t */\n\tprivate getSwiperConfiguration() {\n\t\treturn {\n\t\t\tslidesPerView: 'auto',\n\t\t\tspaceBetween: 16,\n\t\t\tslidesPerGroup: 1, // Importante: moverse de uno en uno\n\t\t\tnavigation: false, // Deshabilitamos la navegación por defecto\n\t\t\tpagination: false,\n\t\t\tloop: true,\n\t\t\tgrabCursor: true,\n\t\t\tautoplay: true,\n\t\t\tbreakpoints: {\n\t\t\t\t320: {\n\t\t\t\t\tslidesPerView: 1,\n\t\t\t\t\tspaceBetween: 8,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t576: {\n\t\t\t\t\tslidesPerView: 2,\n\t\t\t\t\tspaceBetween: 12,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t768: {\n\t\t\t\t\tslidesPerView: 3,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t1024: {\n\t\t\t\t\tslidesPerView: 4,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * Inicializa un nuevo Swiper con la configuración proporcionada.\n\t */\n\tprivate initializeNewSwiper(swiperElement: any, config: any) {\n\t\t// Asignar configuración al elemento\n\t\tObject.assign(swiperElement, config);\n\t\t\n\t\t// Inicializar el Swiper\n\t\tswiperElement.initialize();\n\t\t\n\t\t// console.log('Swiper inicializado con configuración personalizada');\n\t}\n\n\t/**\n\t * Actualiza un Swiper existente para asegurar la configuración correcta.\n\t */\n\tprivate updateExistingSwiper(swiperElement: any) {\n\t\t// console.log('Actualizando configuración de Swiper existente');\n\t\t\n\t\t// Forzar slidesPerGroup a 1\n\t\tswiperElement.swiper.params.slidesPerGroup = 1;\n\t\tswiperElement.swiper.allowSlideNext = true;\n\t\tswiperElement.swiper.allowSlidePrev = true;\n\t\t\n\t\t// Actualizar también los breakpoints si existen\n\t\tif (swiperElement.swiper.params.breakpoints) {\n\t\t\tObject.keys(swiperElement.swiper.params.breakpoints).forEach(key => {\n\t\t\t\tswiperElement.swiper.params.breakpoints[key].slidesPerGroup = 1;\n\t\t\t});\n\t\t}\n\t\t\n\t\tswiperElement.swiper.update();\n\t}\n\n\t/**\n\t * Configura los event listeners para los botones de navegación.\n\t */\n\tprivate setupNavigationEventListeners(prevButton: Element, nextButton: Element, swiperElement: any) {\n\t\tprevButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón anterior');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slidePrev();\n\t\t\t}\n\t\t});\n\n\t\tnextButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón siguiente');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slideNext();\n\t\t\t}\n\t\t});\n\t}\n}\n","<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\n\n    <div class=\"blockProduct block-product\">\n        @if(meta.name){\n        <div class=\"row\">\n            <div class=\"col-12 mt-4\">\n                <h2  class=\"font-weight-normal font-gd\">\n                    <span>{{meta.name}}</span>\n                </h2>\n            </div>\n        </div>\n        }\n\n\n        @if(meta.styles && meta.styles.carrousel == false){\n        <div class=\"row \">\n            @for (product of products; track $index) {\n            <div [class]=\"'item '+ ' col-'+ (meta.styles.items?.sm) + ' col-md-' + (meta.styles.items?.md)  + ' col-lg-' + (meta.styles.items?.lg)  + ' px-2'\" [id]=\"$index\">\n                <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\n                <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n            </div>\n            }\n        </div>\n        } @else {\n        <div class=\"container position-relative\">\n            <swiper-container \n                init=\"false\" \n                [id]=\"meta?.code\"\n                slides-per-view=\"auto\"\n                space-between=\"16\"\n                slides-per-group=\"1\"\n                navigation=\"false\"\n                pagination=\"false\"\n                loop=\"false\">\n                @for (product of products; track $index) {\n                <swiper-slide id=\"swiper-slide\">\n                    <ng-container\n                        *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n                </swiper-slide>\n                }\n            </swiper-container>\n            \n            <!-- Botones de navegación personalizados -->\n            <div class=\"swiper-navigation\">\n                <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\n                    @if(prevArrowImage) {\n                        <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{prevArrowText}}</span>\n                    }\n                </div>\n                <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\n                    @if(nextArrowImage) {\n                        <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{nextArrowText}}</span>\n                    }\n                </div>\n            </div>\n        </div>\n        }\n    </div>\n</section>\n\n\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\n<ng-template #defaultAppProduct let-product=\"product\">\n    <app-product-ec [product]=\"product\"></app-product-ec>\n</ng-template>"]}
221
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block-products-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { afterNextRender, AfterViewInit, Component, CUSTOM_ELEMENTS_SCHEMA, inject, Input, PLATFORM_ID, signal, TemplateRef } from '@angular/core';\nimport { BlockEcComponent } from '../../abstractions-components';\nimport { SwiperOptions } from 'swiper/types';\nimport { AnalyticsService } from '../../../ec-services';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\n// import function to register Swiper custom elements\nimport { register, SwiperContainer } from 'swiper/element/bundle';\nimport { ProductEcComponent } from \"../../product-ec/product-ec.component\";\n\nregister()\n/**\n * Componen para manejar los bloques de productos.\n * @extends {BlockEcComponent}\n * @class BlockProductsEcComponent\n */\n@Component({\n\tselector: 'app-block-products-ec',\n\tstandalone: true,\n\timports: [CommonModule, ProductEcComponent],\n\ttemplateUrl: './block-products-ec.component.html',\n\tstyleUrl: './block-products-ec.component.scss',\n\tschemas: [CUSTOM_ELEMENTS_SCHEMA]\n})\nexport class BlockProductsEcComponent extends BlockEcComponent implements AfterViewInit {\n\t\t// Personalización de las flechas de navegación\n\t// Por defecto usa símbolos de texto, pero se pueden especificar imágenes\n\t@Input() prevArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() nextArrowImage?: string; // undefined = usa símbolo de texto\n\t@Input() prevArrowText: string = '<';\n\t@Input() nextArrowText: string = '>';\n\t/**\n\t * Servicio de Analytics\n\t */\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\n\t/**\n\t * Signal utlizado para guarda el contenedor del carrusel\n\t */\n\tswiperElement = signal<SwiperContainer | null>(null)\n\t/**\n\t * Input que recibe un template para el producto.\n\t */\n\t@Input() appProduct: TemplateRef<any> | undefined;\n\t/**\n\t * Colección de productos.\n\t */\n\t@Input({\n\t\trequired: true\n\t}) products: any; // Product[];\n\t/**\n\t * Bloque principal que contiene los productos\n\t */\n\t@Input({\n\t\trequired: true\n\t}) meta: any;\n\n\tngAfterViewInit() {\n\t\tthis.setupSwiperNavigation();\n\t}\n\t\n\tprivate document?: Document;\n\tprivate platformId: any = inject(PLATFORM_ID)\n\n\t/**\n\t * Ejecuta el método `afterNextRender`para cargar las configuraciones necesarias \n\t * para el carrusel del banners. Esto debe ser asi debido a que ya debe estar presente\n\t * en el Dom el element `<swiper-container>` para poder configurarlo.\n\t */\n\tconstructor() {\n\t\tsuper()\n\t}\n\t/**\n\t * Aplica el evento `select_promotion` junto con el banner que interactua.\n\t * @param banner \n\t */\n\tselectPromotion(item: any): void {\n\t\tthis.analyticsService.callEvent('select_promotion', item)\n\t}\n\n\t/**\n\t * Configura la navegación personalizada del Swiper.\n\t * Esta función está diseñada para ser movida al componente base BlockProductsEcComponent.\n\t * Permite personalización de las imágenes de las flechas mediante @Input.\n\t */\n\tprivate setupSwiperNavigation() {\n\t\tif (this.meta?.styles?.carrousel !== false && isPlatformBrowser(this.platformId)) {\n\t\t\t// Usar setTimeout para asegurar que el swiper esté inicializado\n\t\t\tsetTimeout(() => {\n\t\t\t\tthis.initializeSwiperWithCustomNavigation();\n\t\t\t}, 200);\n\t\t}\n\t}\n\n\t/**\n\t * Inicializa el Swiper con navegación personalizada.\n\t * Esta función puede ser movida al componente base para reutilización.\n\t */\n\tprivate initializeSwiperWithCustomNavigation() {\n\t\tif (!isPlatformBrowser(this.platformId)) return;\n\t\t\n\t\tconst prevButton = document.getElementById(`${this.meta?.code}-prev`);\n\t\tconst nextButton = document.getElementById(`${this.meta?.code}-next`);\n\t\tconst swiperElement = document.getElementById(this.meta?.code) as any;\n\n\t\tif (prevButton && nextButton && swiperElement) {\n\t\t\t// console.log('Configurando navegación personalizada para:', this.meta?.code);\n\t\t\t\n\t\t\tconst swiperConfig = this.getSwiperConfiguration();\n\n\t\t\t// Verificar si el Swiper ya está inicializado\n\t\t\tif (!swiperElement.swiper) {\n\t\t\t\tthis.initializeNewSwiper(swiperElement, swiperConfig);\n\t\t\t} else {\n\t\t\t\tthis.updateExistingSwiper(swiperElement);\n\t\t\t}\n\n\t\t\t// Configurar los event listeners para los botones\n\t\t\tthis.setupNavigationEventListeners(prevButton, nextButton, swiperElement);\n\t\t\t\n\t\t\t// console.log('Event listeners configurados para los botones de navegación');\n\t\t} else {\n\t\t\t// console.log('No se pudieron encontrar los elementos:', {\n\t\t\t// \tprevButton: !!prevButton,\n\t\t\t// \tnextButton: !!nextButton,\n\t\t\t// \tswiperElement: !!swiperElement\n\t\t\t// });\n\t\t}\n\t}\n\n\t/**\n\t * Obtiene la configuración base del Swiper.\n\t * Esta configuración puede ser personalizada en el futuro mediante @Input.\n\t */\n\tprivate getSwiperConfiguration() {\n\t\treturn {\n\t\t\tslidesPerView: 'auto',\n\t\t\tspaceBetween: 16,\n\t\t\tslidesPerGroup: 1, // Importante: moverse de uno en uno\n\t\t\tnavigation: false, // Deshabilitamos la navegación por defecto\n\t\t\tpagination: false,\n\t\t\tloop: true,\n\t\t\tgrabCursor: true,\n\t\t\tautoplay: true,\n\t\t\tbreakpoints: {\n\t\t\t\t320: {\n\t\t\t\t\tslidesPerView: 1,\n\t\t\t\t\tspaceBetween: 8,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t576: {\n\t\t\t\t\tslidesPerView: 2,\n\t\t\t\t\tspaceBetween: 12,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t768: {\n\t\t\t\t\tslidesPerView: 3,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t},\n\t\t\t\t1024: {\n\t\t\t\t\tslidesPerView: 4,\n\t\t\t\t\tspaceBetween: 16,\n\t\t\t\t\tslidesPerGroup: 1\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * Inicializa un nuevo Swiper con la configuración proporcionada.\n\t */\n\tprivate initializeNewSwiper(swiperElement: any, config: any) {\n\t\t// Asignar configuración al elemento\n\t\tObject.assign(swiperElement, config);\n\t\t\n\t\t// Inicializar el Swiper\n\t\tswiperElement.initialize();\n\t\t\n\t\t// console.log('Swiper inicializado con configuración personalizada');\n\t}\n\n\t/**\n\t * Actualiza un Swiper existente para asegurar la configuración correcta.\n\t */\n\tprivate updateExistingSwiper(swiperElement: any) {\n\t\t// console.log('Actualizando configuración de Swiper existente');\n\t\t\n\t\t// Forzar slidesPerGroup a 1\n\t\tswiperElement.swiper.params.slidesPerGroup = 1;\n\t\tswiperElement.swiper.allowSlideNext = true;\n\t\tswiperElement.swiper.allowSlidePrev = true;\n\t\t\n\t\t// Actualizar también los breakpoints si existen\n\t\tif (swiperElement.swiper.params.breakpoints) {\n\t\t\tObject.keys(swiperElement.swiper.params.breakpoints).forEach(key => {\n\t\t\t\tswiperElement.swiper.params.breakpoints[key].slidesPerGroup = 1;\n\t\t\t});\n\t\t}\n\t\t\n\t\tswiperElement.swiper.update();\n\t}\n\n\t/**\n\t * Configura los event listeners para los botones de navegación.\n\t */\n\tprivate setupNavigationEventListeners(prevButton: Element, nextButton: Element, swiperElement: any) {\n\t\tprevButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón anterior');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slidePrev();\n\t\t\t}\n\t\t});\n\n\t\tnextButton.addEventListener('click', (e) => {\n\t\t\te.preventDefault();\n\t\t\te.stopPropagation();\n\t\t\t// console.log('Click en botón siguiente');\n\t\t\tif (swiperElement.swiper) {\n\t\t\t\tswiperElement.swiper.slideNext();\n\t\t\t}\n\t\t});\n\t}\n}\n","<section [ngClass]=\"trimClassBlock(meta.code) + ' container-fluid'\">\n\n    <div class=\"blockProduct block-product\">\n        @if(meta.name){\n        <div class=\"row\">\n            <div class=\"col-12 mt-4\">\n                <h2  class=\"font-weight-normal font-gd\">\n                    <span>{{meta.name}}</span>\n                </h2>\n            </div>\n        </div>\n        }\n\n\n        @if(meta.styles && meta.styles.carrousel == false){\n        <div class=\"row \">\n            @for (product of products; track $index) {\n            <div [class]=\"'item '+ ' col-'+ (meta.styles.items?.sm) + ' col-md-' + (meta.styles.items?.md)  + ' col-lg-' + (meta.styles.items?.lg)  + ' px-2'\" [id]=\"$index\">\n                <!-- verifica que si vienen un template para un custom appProduct llamado \"appProduct\" con un objeto \"product\"- sino usa por defecto el del core -->\n                <ng-container *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n            </div>\n            }\n        </div>\n        } @else {\n        <div class=\"container position-relative\">\n            <swiper-container \n                init=\"false\" \n                [id]=\"meta?.code\"\n                slides-per-view=\"auto\"\n                space-between=\"16\"\n                slides-per-group=\"1\"\n                navigation=\"false\"\n                pagination=\"false\"\n                loop=\"false\">\n                @for (product of products; track $index) {\n                <swiper-slide id=\"swiper-slide\">\n                    <ng-container\n                        *ngTemplateOutlet=\"appProduct ? appProduct : defaultAppProduct; context: {product:product}\"></ng-container>\n                </swiper-slide>\n                }\n            </swiper-container>\n            \n            <!-- Botones de navegación personalizados -->\n            <div class=\"swiper-navigation\">\n                <div class=\"swiper-button-prev\" [id]=\"meta?.code + '-prev'\">\n                    @if(prevArrowImage) {\n                        <img [src]=\"prevArrowImage\" alt=\"Anterior\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{prevArrowText}}</span>\n                    }\n                </div>\n                <div class=\"swiper-button-next\" [id]=\"meta?.code + '-next'\">\n                    @if(nextArrowImage) {\n                        <img [src]=\"nextArrowImage\" alt=\"Siguiente\" />\n                    } @else {\n                        <span class=\"arrow-text\">{{nextArrowText}}</span>\n                    }\n                </div>\n            </div>\n        </div>\n        }\n    </div>\n</section>\n\n\n<!-- componente por defecto (tomara como producto el contexto pasado como \"product\") -->\n<ng-template #defaultAppProduct let-product=\"product\">\n    <app-product-ec [product]=\"product\"></app-product-ec>\n</ng-template>"]}
@@ -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 } from 'rxjs';
5
+ import { combineLatest, distinctUntilChanged, map, Subject, takeUntil } 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,6 +10,7 @@ export class CollectionEcComponent {
10
10
  _productsService = inject(ProductsService);
11
11
  _activeRoute = inject(ActivatedRoute);
12
12
  _optionsService = inject(OptionsService);
13
+ _filtersService = inject(FiltersService);
13
14
  params$ = this._activeRoute.params;
14
15
  //public ready = this._optionsService.ready
15
16
  queryParams$ = this._activeRoute.queryParams;
@@ -21,40 +22,63 @@ export class CollectionEcComponent {
21
22
  defaultFilters = [];
22
23
  loading = false;
23
24
  countProducts = signal(0);
24
- loaded = false;
25
+ loaded = signal(false);
25
26
  optionsFilters = ['all'];
26
27
  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();
32
33
  setAsList = (value) => this.isList = value;
33
34
  ngOnInit() {
34
- this.getProducts();
35
- this.window?.scroll(0, 0);
35
+ if (isPlatformBrowser(this.platformId)) {
36
+ this.window?.scroll(0, 0);
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
+ });
36
58
  }
37
59
  //protected readonly questions = signal<Question[]>([]);
38
60
  total = 0;
39
61
  platformId = inject(PLATFORM_ID);
40
62
  constructor() {
63
+ // Guardamos window sólo en Browser para evitar errores en SSR
41
64
  if (isPlatformBrowser(this.platformId)) {
42
65
  this.window = window;
43
66
  }
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"]);
67
+ // Nos suscribimos al stream de productos.
68
+ // Cuando ProductsService emite, marcamos `loaded` en true, pero solo en Browser.
69
+ // En SSR no lo marcamos para evitar cambios de estado innecesarios.
70
+ this.products$
71
+ .pipe(takeUntil(this.destroy$))
72
+ .subscribe(products => {
73
+ if (isPlatformBrowser(this.platformId)) {
74
+ this.loaded.set(true);
55
75
  }
56
76
  });
57
77
  }
78
+ ngOnDestroy() {
79
+ this.destroy$.next();
80
+ this.destroy$.complete();
81
+ }
58
82
  onScroll() {
59
83
  this.loading = true;
60
84
  this._productsService.updateProducts();
@@ -72,4 +96,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
72
96
  }], ctorParameters: () => [], propDecorators: { optionsFilters: [{
73
97
  type: Input
74
98
  }] } });
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"]}
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"]}
@@ -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 { Router } from '@angular/router';
6
+ import { ActivatedRoute, 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,6 +19,7 @@ export class FiltersEcComponent {
19
19
  injector = inject(Injector);
20
20
  isAuthenticated$ = this._authService.isAuthenticated();
21
21
  hidePrices = false;
22
+ route = inject(ActivatedRoute);
22
23
  setSelect;
23
24
  ngOnInit() {
24
25
  }
@@ -48,26 +49,56 @@ export class FiltersEcComponent {
48
49
  .pop()
49
50
  selectedOption && this._filtersService.setFilterSelected(this.filters[0], selectedOption) */
50
51
  }
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
+ */
51
64
  setSelected(filter, selected) {
52
65
  if (!filter || !selected) {
53
- console.error('Filter or selected element is undefined:', { filter, selected });
54
66
  return;
55
67
  }
68
+ // Si el elemento está marcado como no visible, no hacemos nada
69
+ if (selected.isVisible === false)
70
+ return;
56
71
  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);
58
72
  return;
59
73
  }
60
74
  try {
61
- // this._filtersService.setFilterSelected(filter, selected);
62
75
  if (filter.type() === 'categories') {
76
+ // Para categorías usamos navegación por path (ej: /collection/categories/camas-elasticas)
63
77
  if (selected.path) {
64
78
  this.router.navigate([selected.path]);
65
79
  }
80
+ return;
66
81
  }
67
- else if (filter.type() === 'attributes') {
68
- // Manejar la navegación para atributos
69
- this._filtersService.setFilterSelected(filter, selected);
82
+ if (filter.type() === 'attributes') {
83
+ // El backend espera que le mandemos el "code" del atributo en el query param
84
+ const code = selected.code || selected.value || null;
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;
70
100
  }
101
+ this._filtersService.setFilterSelected(filter, selected);
71
102
  }
72
103
  catch (error) {
73
104
  console.error("Error while setting selected filter:", error);
@@ -101,7 +132,9 @@ export class FiltersEcComponent {
101
132
  return true;
102
133
  };
103
134
  scrollUp = () => {
104
- window.scroll(0, 0);
135
+ if (typeof window !== 'undefined') {
136
+ window.scroll(0, 0);
137
+ }
105
138
  return true;
106
139
  };
107
140
  hasAppliedFilters() {
@@ -138,4 +171,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
138
171
  }], ctorParameters: () => [], propDecorators: { setSelect: [{
139
172
  type: Input
140
173
  }] } });
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"]}
174
+ //# 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\tpublic 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    //       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        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","<p>filters-ec works!</p>\n"]}