ng-easycommerce-v18 0.3.22-beta.2 → 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 (25) hide show
  1. package/README.md +6 -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/price_range-filter.mjs +3 -3
  6. package/esm2022/lib/constants/core.constants.service.mjs +12 -1
  7. package/esm2022/lib/ec-components/collection-ec/collection-ec.component.mjs +41 -17
  8. package/esm2022/lib/ec-components/filters-ec/filters-ec.component.mjs +42 -9
  9. package/esm2022/lib/ec-components/header-ec/header-ec.component.mjs +12 -6
  10. package/esm2022/lib/ec-components/price-range-filter/price-range-filter.component.mjs +13 -2
  11. package/esm2022/lib/ec-services/filters.service.mjs +124 -18
  12. package/esm2022/lib/ec-services/pagination.service.mjs +70 -22
  13. package/esm2022/lib/ec-services/products.service.mjs +5 -3
  14. package/fesm2022/ng-easycommerce-v18.mjs +490 -98
  15. package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
  16. package/lib/classes/filters/attributes-filter.d.ts +24 -0
  17. package/lib/classes/filters/category-filter.d.ts +30 -3
  18. package/lib/constants/core.constants.service.d.ts +7 -0
  19. package/lib/ec-components/collection-ec/collection-ec.component.d.ts +5 -4
  20. package/lib/ec-components/filters-ec/filters-ec.component.d.ts +13 -0
  21. package/lib/ec-components/price-range-filter/price-range-filter.component.d.ts +2 -0
  22. package/lib/ec-services/filters.service.d.ts +18 -1
  23. package/lib/ec-services/pagination.service.d.ts +21 -5
  24. package/lib/ec-services/products.service.d.ts +1 -1
  25. package/package.json +1 -1
@@ -2,6 +2,7 @@ import { EnvironmentInjector, inject, Injectable, runInInjectionContext } from '
2
2
  import { BehaviorSubject } from 'rxjs';
3
3
  import { FilterFactory } from '../classes';
4
4
  import { ApiConstantsService, CoreConstantsService } from '../constants';
5
+ import { PriceRangeFilter } from '../classes/filters/price_range-filter';
5
6
  import * as i0 from "@angular/core";
6
7
  export class FiltersService {
7
8
  _apiConsts = inject(ApiConstantsService);
@@ -21,6 +22,7 @@ export class FiltersService {
21
22
  limit: 0
22
23
  };
23
24
  _filtersInitialized = false;
25
+ _lastRouteQuery = null;
24
26
  constructor() {
25
27
  //this._defaultFilters = this._consts.getDefaultFilters()
26
28
  }
@@ -41,10 +43,20 @@ export class FiltersService {
41
43
  let extra_params = '';
42
44
  this._filtersSubject.value.forEach(filter => {
43
45
  const extra = filter.toUrlParams();
44
- extra.split('=')[1] != '' ? extra_params += extra : null;
46
+ if (extra && extra.split('=')[1] !== '') {
47
+ extra_params += extra;
48
+ }
45
49
  });
46
- if (search_value)
47
- extra_params += ('&criteria[search][type]=contains&criteria[search][value]=' + search_value);
50
+ // Si en la URL ya venía `attributeCodes` pero aún ningún filtro lo agregó,
51
+ // lo volvemos a sumar a mano. Esto evita que al refrescar (SSR) se pierda el filtro
52
+ // aunque el AttributesFilter todavía no haya terminado de hidratarse.
53
+ const attributeCodesFromRoute = this._lastRouteQuery?.['attributeCodes'];
54
+ if (attributeCodesFromRoute && !extra_params.includes('attributeCodes=')) {
55
+ extra_params += `&attributeCodes=${attributeCodesFromRoute}`;
56
+ }
57
+ if (search_value) {
58
+ extra_params += '&criteria[search][type]=contains&criteria[search][value]=' + search_value;
59
+ }
48
60
  return this.productsFilterApi(extra_params);
49
61
  }
50
62
  isUpdated(paginationSettings) {
@@ -60,7 +72,7 @@ export class FiltersService {
60
72
  this._paginationSettings = paginationSettings;
61
73
  return change;
62
74
  }
63
- setFilters(paginationSettings, search_value) {
75
+ setFilters(paginationSettings, search_value, routeQueryParams) {
64
76
  this._paginationSettings = paginationSettings;
65
77
  let final_filters = [];
66
78
  let filtersToProcess = this._optionsFilters?.includes('all')
@@ -73,15 +85,30 @@ export class FiltersService {
73
85
  if (filter) {
74
86
  final_filters.push(filter);
75
87
  }
76
- else {
77
- // console.warn(`❌ Failed to create filter for type: ${filterType}`);
78
- }
79
88
  });
80
89
  });
81
90
  this._defaultFilters?.forEach(filterDefault => {
82
91
  let filter = final_filters.find(filter => filter.type() == filterDefault.filter_type);
83
92
  filter && filterDefault.codes.forEach(value => filter.setSelected(value));
84
93
  });
94
+ // hidratar price_range desde la URL
95
+ if (routeQueryParams) {
96
+ const pr = final_filters.find(f => f.type() === 'price_range');
97
+ if (pr) {
98
+ const rawMin = routeQueryParams['price_min'];
99
+ const rawMax = routeQueryParams['price_max'];
100
+ const min = rawMin != null ? Number(rawMin) : null;
101
+ const max = rawMax != null ? Number(rawMax) : null;
102
+ if ((min != null && !Number.isNaN(min)) || (max != null && !Number.isNaN(max))) {
103
+ if (min != null && !Number.isNaN(min)) {
104
+ pr.currentMinPrice = min;
105
+ }
106
+ if (max != null && !Number.isNaN(max)) {
107
+ pr.currentMaxPrice = max;
108
+ }
109
+ }
110
+ }
111
+ }
85
112
  this._filtersSubject.next(final_filters);
86
113
  }
87
114
  getFilters(paginationSettings) {
@@ -96,9 +123,7 @@ export class FiltersService {
96
123
  runInInjectionContext(this.environmentInjector, () => {
97
124
  const filterFactory = new FilterFactory();
98
125
  filtersToProcess?.forEach(filterType => {
99
- // console.log('Creating filter for type:', filterType);
100
126
  const filter = filterFactory.create(filterType, settings);
101
- // console.log('Created filter:', filter);
102
127
  if (filter) {
103
128
  final_filters.push(filter);
104
129
  }
@@ -123,7 +148,6 @@ export class FiltersService {
123
148
  if (filterObj.type() !== 'price_range') {
124
149
  let index = final_filters.findIndex(filter => filter.type() == filterObj.type());
125
150
  final_filters[index].setSelected(filterElem, filterElem.value || filterElem.code);
126
- console.log(index, final_filters);
127
151
  }
128
152
  this._filtersSubject.next(final_filters);
129
153
  // }
@@ -133,13 +157,6 @@ export class FiltersService {
133
157
  runInInjectionContext(this.environmentInjector, () => {
134
158
  const filterFactory = new FilterFactory();
135
159
  const filter = [];
136
- /* claves.forEach((key:any) => {
137
- filter.push(filterFactory.create(key))
138
- paginationFilters[key].forEach((value:any) => {
139
- console.log(filter[0].createElement(value,paginationFilters[key]));
140
- })
141
-
142
- }) */
143
160
  this._filtersSubject.value.forEach((value) => {
144
161
  if (value.type() == "attributes") {
145
162
  //value.data = [];
@@ -159,6 +176,95 @@ export class FiltersService {
159
176
  pr.setSelected(min, max);
160
177
  this._filtersSubject.next([...filters]);
161
178
  }
179
+ /**
180
+ * Fuerza a emitir de nuevo los filtros actuales.
181
+ * Útil cuando un filtro async (categorías, atributos) termina de cargarse
182
+ * y necesitamos que PaginationService vuelva a armar la URL.
183
+ */
184
+ refreshFilters() {
185
+ const current = this._filtersSubject.value;
186
+ // Emitimos una copia para que los subscribers detecten el cambio
187
+ this._filtersSubject.next([...current]);
188
+ }
189
+ /**
190
+ * Punto central donde se hidratan los filtros a partir de la URL:
191
+ * - Lee type/value de la ruta (ej. /collection/categories/:value).
192
+ * - Lee query params (category, price_min, price_max, attributeCodes, search, etc.).
193
+ * - Crea instancias de filtros y les delega la hidratación específica.
194
+ *
195
+ * Este método se llama tanto en SSR como en navegador.
196
+ */
197
+ hydrateFromRoute(paginationSettings, routeParams, routeQuery) {
198
+ // hidratar search desde la URL ===
199
+ const searchFromUrl = routeQuery?.['search'];
200
+ if (typeof searchFromUrl === 'string' && searchFromUrl.trim() !== '') {
201
+ // Guardamos el valor para que otros componentes (Header, Collection) lo reutilicen
202
+ this._consts.searchValue = searchFromUrl.trim();
203
+ }
204
+ else {
205
+ // Si no viene `search` en la URL, limpiamos el estado global para no arrastrar búsquedas viejas
206
+ this._consts.searchValue = '';
207
+ }
208
+ // Guardamos la última query completa para usarla en generateFinalApi (parche attributeCodes)
209
+ this._lastRouteQuery = routeQuery;
210
+ // Guardamos las settings actuales (type, value, limit, etc.)
211
+ this._paginationSettings = paginationSettings;
212
+ // Creamos los filtros base (attributes, categories, dynamics, sort, price_range, etc.)
213
+ let final_filters = [];
214
+ let filtersToProcess = this._optionsFilters?.includes('all')
215
+ ? ['attributes', 'categories', 'dynamics', 'sort', 'price_range']
216
+ : this._optionsFilters;
217
+ runInInjectionContext(this.environmentInjector, () => {
218
+ const filterFactory = new FilterFactory();
219
+ filtersToProcess?.forEach(filterType => {
220
+ const filter = filterFactory.create(filterType, paginationSettings);
221
+ if (filter) {
222
+ final_filters.push(filter);
223
+ }
224
+ });
225
+ });
226
+ // Aplicamos defaultFilters si corresponde
227
+ this._defaultFilters?.forEach(filterDefault => {
228
+ const filter = final_filters.find(f => f.type() === filterDefault.filter_type);
229
+ filter && filterDefault.codes.forEach(value => filter.setSelected(value));
230
+ });
231
+ // Hidratar filtros con slug y query params genéricos
232
+ final_filters.forEach(f => {
233
+ const anyFilter = f;
234
+ // type/value en la URL: /collection/categories/:value
235
+ const urlType = routeParams['type'];
236
+ const urlValue = routeParams['value'];
237
+ // Por ejemplo, el CategoryFilter implementa setFromSlug para marcar la categoría correcta
238
+ if (urlType === 'categories' && urlValue && typeof anyFilter.setFromSlug === 'function') {
239
+ anyFilter.setFromSlug(urlValue);
240
+ }
241
+ // query params genéricos: ?category=0101&price_min=...&price_max=...&attributeCodes=...
242
+ // Cada filtro que lo soporte implementa hydrateFromQuery y decide qué leer.
243
+ if (typeof anyFilter.hydrateFromQuery === 'function') {
244
+ anyFilter.hydrateFromQuery(routeQuery);
245
+ }
246
+ });
247
+ // Hidratar específicamente el PriceRangeFilter desde la URL
248
+ const priceFilter = final_filters.find((f) => f instanceof PriceRangeFilter);
249
+ if (priceFilter && routeQuery) {
250
+ const minFromUrl = routeQuery['price_min'];
251
+ const maxFromUrl = routeQuery['price_max'];
252
+ if (minFromUrl != null || maxFromUrl != null) {
253
+ const min = minFromUrl != null ? Number(minFromUrl) : priceFilter.minPrice;
254
+ const max = maxFromUrl != null ? Number(maxFromUrl) : priceFilter.maxPrice;
255
+ // usamos la API del filtro para setear el rango
256
+ priceFilter.setSelected(min, max);
257
+ }
258
+ else {
259
+ // si no hay nada en la URL, lo reseteamos
260
+ priceFilter.reset();
261
+ }
262
+ }
263
+ // Emitimos filtros ya hidratados para que PaginationService y Collection reaccionen
264
+ this._filtersSubject.next(final_filters);
265
+ // Marcamos que el sistema de filtros está listo (útil para mostrar spinners, etc.)
266
+ this._readySubject.next(true);
267
+ }
162
268
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FiltersService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
163
269
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FiltersService, providedIn: 'root' });
164
270
  }
@@ -168,4 +274,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
168
274
  providedIn: 'root'
169
275
  }]
170
276
  }], ctorParameters: () => [] });
171
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filters.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/filters.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { EnvironmentInjector, inject, Injectable, runInInjectionContext } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { Filter, FilterFactory } from '../classes';\nimport { DefaultFilter, FilterElement, PaginationSettings, FilterType } from '../interfaces';\nimport { ApiConstantsService, CoreConstantsService } from '../constants';\nimport { PriceRangeFilter } from '../classes/filters/price_range-filter';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class FiltersService {\n    private _apiConsts: ApiConstantsService = inject(ApiConstantsService);\n    private _consts: CoreConstantsService = inject(CoreConstantsService);\n    private _filtersSubject: BehaviorSubject<Filter[]> = new BehaviorSubject<Filter[]>([]);\n    public filters$: Observable<Filter[]> = this._filtersSubject.asObservable();\n\n    private _apiUrlSubject: BehaviorSubject<string> = new BehaviorSubject<string>('');\n    public apiUrl$: Observable<string> = this._apiUrlSubject.asObservable();\n\n    private _readySubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n    public ready$: Observable<boolean> = this._readySubject.asObservable();\n\n    private environmentInjector = inject(EnvironmentInjector)\n\n    private _optionsFilters: (FilterType | 'all')[] = ['all'];\n    private _defaultFilters: DefaultFilter[] = [];\n\n    private _paginationSettings: PaginationSettings = {\n        value: '',\n        type: '',\n        limit: 0\n    };\n\n    private _filtersInitialized = false;\n\n    constructor() {\n        //this._defaultFilters = this._consts.getDefaultFilters()\n    }\n\n    //API URL\n    /**\n     * Retorna la url para el filtrado\n     * @param filters \n     * @returns \n     */\n    productsFilterApi(filters = '') { return 'shop-api/' + this._apiConsts.CHANNEL + '/product-search/' + '?locale=' + this._apiConsts.LOCALE + filters };\n    /**\n     * Arma los filtros y retorna la url completa con los filtros aplicados.\n     * @param search_value valor a buscar\n     * @returns \n     */\n    generateFinalApi(search_value?: string) {\n        let extra_params = '';\n        this._filtersSubject.value.forEach(filter => {\n            const extra = filter.toUrlParams();\n            extra.split('=')[1] != '' ? extra_params += extra : null;\n        })\n        if (search_value) extra_params += ('&criteria[search][type]=contains&criteria[search][value]=' + search_value);\n\n        return this.productsFilterApi(extra_params);\n    }\n\n    private isUpdated(paginationSettings: PaginationSettings) {\n        const keys: string[] = Object.keys(paginationSettings);\n        let change: boolean = false;\n        keys.forEach((key: string) => {\n            const value = (paginationSettings as Record<string, any>)[key];\n            const value2 = (this._paginationSettings as Record<string, any>)[key];\n            if (value != value2) change = true;\n        })\n        if (change) this._paginationSettings = paginationSettings;\n        return change;\n    }\n\n    setFilters(paginationSettings: PaginationSettings, search_value?: string) {\n        this._paginationSettings = paginationSettings;\n        let final_filters: Filter[] = [];\n        let filtersToProcess = this._optionsFilters?.includes('all')\n            ? (['attributes', 'categories', 'dynamics', 'sort', 'price_range'] as FilterType[])\n            : this._optionsFilters;\n\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            filtersToProcess?.forEach(filterType => {\n                const filter = filterFactory.create(filterType, paginationSettings);\n                if (filter) {\n                    final_filters.push(filter);\n                } else {\n                    // console.warn(`❌ Failed to create filter for type: ${filterType}`);\n                }\n            });\n        })\n\n        this._defaultFilters?.forEach(filterDefault => {\n            let filter = final_filters.find(filter => filter.type() == filterDefault.filter_type);\n            filter && filterDefault.codes.forEach(value => filter.setSelected(value));\n        })\n        this._filtersSubject.next(final_filters);\n    }\n\n    getFilters(paginationSettings?: PaginationSettings) {\n        if (this._filtersInitialized) return;\n        this._filtersInitialized = true;\n        const settings = paginationSettings || this._paginationSettings;\n        let final_filters: Filter[] = [];\n        let filtersToProcess = this._optionsFilters?.includes('all')\n            ? (['attributes', 'categories', 'dynamics', 'sort', 'price_range'] as FilterType[])\n            : this._optionsFilters;\n\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            filtersToProcess?.forEach(filterType => {\n                // console.log('Creating filter for type:', filterType);\n                const filter = filterFactory.create(filterType, settings);\n                // console.log('Created filter:', filter);\n                if (filter) {\n                    final_filters.push(filter);\n                } else {\n                    // console.warn(`Failed to create filter for type: ${filterType}`);\n                }\n            });\n        });\n\n        this._defaultFilters?.forEach(filterDefault => {\n            let filter = final_filters.find(filter => filter.type() == filterDefault.filter_type);\n            filter && filterDefault.codes.forEach(value => filter.setSelected(value));\n        });\n        this._filtersSubject.next(final_filters);\n        return final_filters;\n    }\n\n    getSpecificFilters(types: FilterType[]): Filter[] {\n        const filters: Filter[] = this._filtersSubject.value.filter(filter => types.includes(filter.type()))\n        return filters;\n    }\n\n    setFilterSelected(filterObj: Filter, filterElem: FilterElement) {\n        let final_filters: Filter[] = this._filtersSubject.value;\n        if (filterObj.type() !== 'price_range') {\n            let index = final_filters.findIndex(filter => filter.type() == filterObj.type());\n            final_filters[index].setSelected(filterElem, filterElem.value || filterElem.code);\n            console.log(index, final_filters)\n        }\n        this._filtersSubject.next(final_filters);\n        // }\n    }\n\n    updateFilters(paginationFilters: any): void {\n        const claves = Object.keys(paginationFilters)\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            const filter: Filter[] = []\n            /* claves.forEach((key:any) => {\n                filter.push(filterFactory.create(key))\n                paginationFilters[key].forEach((value:any) => {\n                     console.log(filter[0].createElement(value,paginationFilters[key]));\n                })\n               \n            }) */\n            this._filtersSubject.value.forEach((value: any) => {\n                if (value.type() == \"attributes\") {\n                    //value.data = [];\n                }\n            })\n        })\n    }\n\n    /** \n     * Actualiza internamente el rango de precio y re-emite \n     * el array de filtros para que PaginationService vuelva a llamar \n     */\n    public updatePriceRangeFilter(min: number, max: number): void {\n        const filters = this._filtersSubject.value;\n        const pr = filters.find(f => f.type() === 'price_range') as PriceRangeFilter | undefined;\n        if (!pr) return;\n        pr.setSelected(min, max);\n        this._filtersSubject.next([...filters]);\n    }\n}\n\n// export type FilterType = 'attributes' | 'categories' | 'dynamics' | 'sort' | 'price_range'; // Use imported FilterType instead\n"]}
277
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filters.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/filters.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,MAAM,EAAE,UAAU,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAC/F,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;AACnD,OAAO,EAAU,aAAa,EAAE,MAAM,YAAY,CAAC;AAEnD,OAAO,EAAE,mBAAmB,EAAE,oBAAoB,EAAE,MAAM,cAAc,CAAC;AACzE,OAAO,EAAE,gBAAgB,EAAE,MAAM,uCAAuC,CAAC;;AAMzE,MAAM,OAAO,cAAc;IACf,UAAU,GAAwB,MAAM,CAAC,mBAAmB,CAAC,CAAC;IAC9D,OAAO,GAAyB,MAAM,CAAC,oBAAoB,CAAC,CAAC;IAC7D,eAAe,GAA8B,IAAI,eAAe,CAAW,EAAE,CAAC,CAAC;IAChF,QAAQ,GAAyB,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;IAEpE,cAAc,GAA4B,IAAI,eAAe,CAAS,EAAE,CAAC,CAAC;IAC3E,OAAO,GAAuB,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAAC;IAEhE,aAAa,GAA6B,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;IAC/E,MAAM,GAAwB,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;IAE/D,mBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAA;IAEjD,eAAe,GAA2B,CAAC,KAAK,CAAC,CAAC;IAClD,eAAe,GAAoB,EAAE,CAAC;IAEtC,mBAAmB,GAAuB;QAC9C,KAAK,EAAE,EAAE;QACT,IAAI,EAAE,EAAE;QACR,KAAK,EAAE,CAAC;KACX,CAAC;IAEM,mBAAmB,GAAG,KAAK,CAAC;IAC5B,eAAe,GAAkB,IAAI,CAAC;IAE9C;QACI,yDAAyD;IAC7D,CAAC;IAED,SAAS;IACT;;;;OAIG;IACH,iBAAiB,CAAC,OAAO,GAAG,EAAE,IAAI,OAAO,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,GAAG,kBAAkB,GAAG,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,OAAO,CAAA,CAAC,CAAC;IAAA,CAAC;IACtJ;;;;OAIG;IACH,gBAAgB,CAAC,YAAqB;QAClC,IAAI,YAAY,GAAG,EAAE,CAAC;QAEtB,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACxC,MAAM,KAAK,GAAG,MAAM,CAAC,WAAW,EAAE,CAAC;YACnC,IAAI,KAAK,IAAI,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC;gBACtC,YAAY,IAAI,KAAK,CAAC;YAC1B,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,2EAA2E;QAC3E,oFAAoF;QACpF,sEAAsE;QACtE,MAAM,uBAAuB,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,gBAAgB,CAAC,CAAC;QACzE,IAAI,uBAAuB,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;YACvE,YAAY,IAAI,mBAAmB,uBAAuB,EAAE,CAAC;QACjE,CAAC;QAED,IAAI,YAAY,EAAE,CAAC;YACf,YAAY,IAAI,2DAA2D,GAAG,YAAY,CAAC;QAC/F,CAAC;QAED,OAAO,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;IAChD,CAAC;IAEO,SAAS,CAAC,kBAAsC;QACpD,MAAM,IAAI,GAAa,MAAM,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QACvD,IAAI,MAAM,GAAY,KAAK,CAAC;QAC5B,IAAI,CAAC,OAAO,CAAC,CAAC,GAAW,EAAE,EAAE;YACzB,MAAM,KAAK,GAAI,kBAA0C,CAAC,GAAG,CAAC,CAAC;YAC/D,MAAM,MAAM,GAAI,IAAI,CAAC,mBAA2C,CAAC,GAAG,CAAC,CAAC;YACtE,IAAI,KAAK,IAAI,MAAM;gBAAE,MAAM,GAAG,IAAI,CAAC;QACvC,CAAC,CAAC,CAAA;QACF,IAAI,MAAM;YAAE,IAAI,CAAC,mBAAmB,GAAG,kBAAkB,CAAC;QAC1D,OAAO,MAAM,CAAC;IAClB,CAAC;IAED,UAAU,CACN,kBAAsC,EACtC,YAAqB,EACrB,gBAAsB;QAEtB,IAAI,CAAC,mBAAmB,GAAG,kBAAkB,CAAC;QAC9C,IAAI,aAAa,GAAa,EAAE,CAAC;QACjC,IAAI,gBAAgB,GAAG,IAAI,CAAC,eAAe,EAAE,QAAQ,CAAC,KAAK,CAAC;YACxD,CAAC,CAAE,CAAC,YAAY,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,EAAE,aAAa,CAAkB;YACnF,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC;QAE3B,qBAAqB,CAAC,IAAI,CAAC,mBAAmB,EAAE,GAAG,EAAE;YACjD,MAAM,aAAa,GAAG,IAAI,aAAa,EAAE,CAAC;YAC1C,gBAAgB,EAAE,OAAO,CAAC,UAAU,CAAC,EAAE;gBACnC,MAAM,MAAM,GAAG,aAAa,CAAC,MAAM,CAAC,UAAU,EAAE,kBAAkB,CAAC,CAAC;gBACpE,IAAI,MAAM,EAAE,CAAC;oBACT,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/B,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,eAAe,EAAE,OAAO,CAAC,aAAa,CAAC,EAAE;YAC1C,IAAI,MAAM,GAAG,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,aAAa,CAAC,WAAW,CAAC,CAAC;YACtF,MAAM,IAAI,aAAa,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;QAEH,oCAAoC;QACpC,IAAI,gBAAgB,EAAE,CAAC;YACnB,MAAM,EAAE,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,aAAa,CAAiC,CAAC;YAC/F,IAAI,EAAE,EAAE,CAAC;gBACL,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;gBAC7C,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;gBAE7C,MAAM,GAAG,GAAG,MAAM,IAAI,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;gBACnD,MAAM,GAAG,GAAG,MAAM,IAAI,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;gBAEnD,IAAI,CAAC,GAAG,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC;oBAC7E,IAAI,GAAG,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC;wBACpC,EAAE,CAAC,eAAe,GAAG,GAAG,CAAC;oBAC7B,CAAC;oBACD,IAAI,GAAG,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC;wBACpC,EAAE,CAAC,eAAe,GAAG,GAAG,CAAC;oBAC7B,CAAC;gBACL,CAAC;YACL,CAAC;QACL,CAAC;QAED,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IAC7C,CAAC;IAED,UAAU,CAAC,kBAAuC;QAC9C,IAAI,IAAI,CAAC,mBAAmB;YAAE,OAAO;QACrC,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;QAChC,MAAM,QAAQ,GAAG,kBAAkB,IAAI,IAAI,CAAC,mBAAmB,CAAC;QAChE,IAAI,aAAa,GAAa,EAAE,CAAC;QACjC,IAAI,gBAAgB,GAAG,IAAI,CAAC,eAAe,EAAE,QAAQ,CAAC,KAAK,CAAC;YACxD,CAAC,CAAE,CAAC,YAAY,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,EAAE,aAAa,CAAkB;YACnF,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC;QAE3B,qBAAqB,CAAC,IAAI,CAAC,mBAAmB,EAAE,GAAG,EAAE;YACjD,MAAM,aAAa,GAAG,IAAI,aAAa,EAAE,CAAC;YAC1C,gBAAgB,EAAE,OAAO,CAAC,UAAU,CAAC,EAAE;gBACnC,MAAM,MAAM,GAAG,aAAa,CAAC,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAC1D,IAAI,MAAM,EAAE,CAAC;oBACT,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/B,CAAC;qBAAM,CAAC;oBACJ,mEAAmE;gBACvE,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,eAAe,EAAE,OAAO,CAAC,aAAa,CAAC,EAAE;YAC1C,IAAI,MAAM,GAAG,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,aAAa,CAAC,WAAW,CAAC,CAAC;YACtF,MAAM,IAAI,aAAa,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACzC,OAAO,aAAa,CAAC;IACzB,CAAC;IAED,kBAAkB,CAAC,KAAmB;QAClC,MAAM,OAAO,GAAa,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC,CAAA;QACpG,OAAO,OAAO,CAAC;IACnB,CAAC;IAED,iBAAiB,CAAC,SAAiB,EAAE,UAAyB;QAC1D,IAAI,aAAa,GAAa,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC;QACzD,IAAI,SAAS,CAAC,IAAI,EAAE,KAAK,aAAa,EAAE,CAAC;YACrC,IAAI,KAAK,GAAG,aAAa,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;YACjF,aAAa,CAAC,KAAK,CAAC,CAAC,WAAW,CAAC,UAAU,EAAE,UAAU,CAAC,KAAK,IAAI,UAAU,CAAC,IAAI,CAAC,CAAC;QACtF,CAAC;QACD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACzC,IAAI;IACR,CAAC;IAED,aAAa,CAAC,iBAAsB;QAChC,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAA;QAC7C,qBAAqB,CAAC,IAAI,CAAC,mBAAmB,EAAE,GAAG,EAAE;YACjD,MAAM,aAAa,GAAG,IAAI,aAAa,EAAE,CAAC;YAC1C,MAAM,MAAM,GAAa,EAAE,CAAA;YAC3B,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,KAAU,EAAE,EAAE;gBAC9C,IAAI,KAAK,CAAC,IAAI,EAAE,IAAI,YAAY,EAAE,CAAC;oBAC/B,kBAAkB;gBACtB,CAAC;YACL,CAAC,CAAC,CAAA;QACN,CAAC,CAAC,CAAA;IACN,CAAC;IAED;;;OAGG;IACI,sBAAsB,CAAC,GAAW,EAAE,GAAW;QAClD,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC;QAC3C,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,aAAa,CAAiC,CAAC;QACzF,IAAI,CAAC,EAAE;YAAE,OAAO;QAChB,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;QACzB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAC5C,CAAC;IAED;;;;OAIG;IACI,cAAc;QACjB,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC;QAC3C,iEAAiE;QACjE,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAC5C,CAAC;IAED;;;;;;;OAOG;IACH,gBAAgB,CACZ,kBAAsC,EACtC,WAAmB,EACnB,UAAkB;QAElB,mCAAmC;QACnC,MAAM,aAAa,GAAG,UAAU,EAAE,CAAC,QAAQ,CAAC,CAAC;QAC7C,IAAI,OAAO,aAAa,KAAK,QAAQ,IAAI,aAAa,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;YACnE,mFAAmF;YACnF,IAAI,CAAC,OAAO,CAAC,WAAW,GAAG,aAAa,CAAC,IAAI,EAAE,CAAC;QACpD,CAAC;aAAM,CAAC;YACJ,gGAAgG;YAChG,IAAI,CAAC,OAAO,CAAC,WAAW,GAAG,EAAE,CAAC;QAClC,CAAC;QAED,6FAA6F;QAC7F,IAAI,CAAC,eAAe,GAAG,UAAU,CAAC;QAElC,6DAA6D;QAC7D,IAAI,CAAC,mBAAmB,GAAG,kBAAkB,CAAC;QAE9C,uFAAuF;QACvF,IAAI,aAAa,GAAa,EAAE,CAAC;QACjC,IAAI,gBAAgB,GAAG,IAAI,CAAC,eAAe,EAAE,QAAQ,CAAC,KAAK,CAAC;YACxD,CAAC,CAAE,CAAC,YAAY,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,EAAE,aAAa,CAAkB;YACnF,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC;QAE3B,qBAAqB,CAAC,IAAI,CAAC,mBAAmB,EAAE,GAAG,EAAE;YACjD,MAAM,aAAa,GAAG,IAAI,aAAa,EAAE,CAAC;YAC1C,gBAAgB,EAAE,OAAO,CAAC,UAAU,CAAC,EAAE;gBACnC,MAAM,MAAM,GAAG,aAAa,CAAC,MAAM,CAAC,UAAU,EAAE,kBAAkB,CAAC,CAAC;gBACpE,IAAI,MAAM,EAAE,CAAC;oBACT,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/B,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,0CAA0C;QAC1C,IAAI,CAAC,eAAe,EAAE,OAAO,CAAC,aAAa,CAAC,EAAE;YAC1C,MAAM,MAAM,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,aAAa,CAAC,WAAW,CAAC,CAAC;YAC/E,MAAM,IAAI,aAAa,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;QAEH,qDAAqD;QACrD,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;YACtB,MAAM,SAAS,GAAG,CAAQ,CAAC;YAE3B,sDAAsD;YACtD,MAAM,OAAO,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC;YACpC,MAAM,QAAQ,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;YAEtC,0FAA0F;YAC1F,IAAI,OAAO,KAAK,YAAY,IAAI,QAAQ,IAAI,OAAO,SAAS,CAAC,WAAW,KAAK,UAAU,EAAE,CAAC;gBACtF,SAAS,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;YACpC,CAAC;YAED,wFAAwF;YACxF,4EAA4E;YAC5E,IAAI,OAAO,SAAS,CAAC,gBAAgB,KAAK,UAAU,EAAE,CAAC;gBACnD,SAAS,CAAC,gBAAgB,CAAC,UAAU,CAAC,CAAC;YAC3C,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,4DAA4D;QAC5D,MAAM,WAAW,GAAG,aAAa,CAAC,IAAI,CAClC,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,YAAY,gBAAgB,CACf,CAAC;QAElC,IAAI,WAAW,IAAI,UAAU,EAAE,CAAC;YAC5B,MAAM,UAAU,GAAG,UAAU,CAAC,WAAW,CAAC,CAAC;YAC3C,MAAM,UAAU,GAAG,UAAU,CAAC,WAAW,CAAC,CAAC;YAE3C,IAAI,UAAU,IAAI,IAAI,IAAI,UAAU,IAAI,IAAI,EAAE,CAAC;gBAC3C,MAAM,GAAG,GAAG,UAAU,IAAI,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,QAAQ,CAAC;gBAC3E,MAAM,GAAG,GAAG,UAAU,IAAI,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,QAAQ,CAAC;gBAE3E,gDAAgD;gBAChD,WAAW,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;YACtC,CAAC;iBAAM,CAAC;gBACJ,0CAA0C;gBAC1C,WAAW,CAAC,KAAK,EAAE,CAAC;YACxB,CAAC;QACL,CAAC;QAED,oFAAoF;QACpF,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAEzC,mFAAmF;QACnF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAClC,CAAC;wGAlTQ,cAAc;4GAAd,cAAc,cAFX,MAAM;;4FAET,cAAc;kBAH1B,UAAU;mBAAC;oBACR,UAAU,EAAE,MAAM;iBACrB","sourcesContent":["import { EnvironmentInjector, inject, Injectable, runInInjectionContext } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { Filter, FilterFactory } from '../classes';\nimport { DefaultFilter, FilterElement, PaginationSettings, FilterType } from '../interfaces';\nimport { ApiConstantsService, CoreConstantsService } from '../constants';\nimport { PriceRangeFilter } from '../classes/filters/price_range-filter';\nimport { Params } from '@angular/router';\n\n@Injectable({\n    providedIn: 'root'\n})\nexport class FiltersService {\n    private _apiConsts: ApiConstantsService = inject(ApiConstantsService);\n    private _consts: CoreConstantsService = inject(CoreConstantsService);\n    private _filtersSubject: BehaviorSubject<Filter[]> = new BehaviorSubject<Filter[]>([]);\n    public filters$: Observable<Filter[]> = this._filtersSubject.asObservable();\n\n    private _apiUrlSubject: BehaviorSubject<string> = new BehaviorSubject<string>('');\n    public apiUrl$: Observable<string> = this._apiUrlSubject.asObservable();\n\n    private _readySubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n    public ready$: Observable<boolean> = this._readySubject.asObservable();\n\n    private environmentInjector = inject(EnvironmentInjector)\n\n    private _optionsFilters: (FilterType | 'all')[] = ['all'];\n    private _defaultFilters: DefaultFilter[] = [];\n\n    private _paginationSettings: PaginationSettings = {\n        value: '',\n        type: '',\n        limit: 0\n    };\n\n    private _filtersInitialized = false;\n    private _lastRouteQuery: Params | null = null;\n\n    constructor() {\n        //this._defaultFilters = this._consts.getDefaultFilters()\n    }\n\n    //API URL\n    /**\n     * Retorna la url para el filtrado\n     * @param filters \n     * @returns \n     */\n    productsFilterApi(filters = '') { return 'shop-api/' + this._apiConsts.CHANNEL + '/product-search/' + '?locale=' + this._apiConsts.LOCALE + filters };\n    /**\n     * Arma los filtros y retorna la url completa con los filtros aplicados.\n     * @param search_value valor a buscar\n     * @returns \n     */\n    generateFinalApi(search_value?: string): string {\n        let extra_params = '';\n\n        this._filtersSubject.value.forEach(filter => {\n            const extra = filter.toUrlParams();\n            if (extra && extra.split('=')[1] !== '') {\n                extra_params += extra;\n            }\n        });\n\n        // Si en la URL ya venía `attributeCodes` pero aún ningún filtro lo agregó,\n        // lo volvemos a sumar a mano. Esto evita que al refrescar (SSR) se pierda el filtro\n        // aunque el AttributesFilter todavía no haya terminado de hidratarse.\n        const attributeCodesFromRoute = this._lastRouteQuery?.['attributeCodes'];\n        if (attributeCodesFromRoute && !extra_params.includes('attributeCodes=')) {\n            extra_params += `&attributeCodes=${attributeCodesFromRoute}`;\n        }\n\n        if (search_value) {\n            extra_params += '&criteria[search][type]=contains&criteria[search][value]=' + search_value;\n        }\n\n        return this.productsFilterApi(extra_params);\n    }\n\n    private isUpdated(paginationSettings: PaginationSettings) {\n        const keys: string[] = Object.keys(paginationSettings);\n        let change: boolean = false;\n        keys.forEach((key: string) => {\n            const value = (paginationSettings as Record<string, any>)[key];\n            const value2 = (this._paginationSettings as Record<string, any>)[key];\n            if (value != value2) change = true;\n        })\n        if (change) this._paginationSettings = paginationSettings;\n        return change;\n    }\n\n    setFilters(\n        paginationSettings: PaginationSettings,\n        search_value?: string,\n        routeQueryParams?: any\n    ) {\n        this._paginationSettings = paginationSettings;\n        let final_filters: Filter[] = [];\n        let filtersToProcess = this._optionsFilters?.includes('all')\n            ? (['attributes', 'categories', 'dynamics', 'sort', 'price_range'] as FilterType[])\n            : this._optionsFilters;\n\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            filtersToProcess?.forEach(filterType => {\n                const filter = filterFactory.create(filterType, paginationSettings);\n                if (filter) {\n                    final_filters.push(filter);\n                }\n            });\n        });\n\n        this._defaultFilters?.forEach(filterDefault => {\n            let filter = final_filters.find(filter => filter.type() == filterDefault.filter_type);\n            filter && filterDefault.codes.forEach(value => filter.setSelected(value));\n        });\n\n        // hidratar price_range desde la URL\n        if (routeQueryParams) {\n            const pr = final_filters.find(f => f.type() === 'price_range') as PriceRangeFilter | undefined;\n            if (pr) {\n                const rawMin = routeQueryParams['price_min'];\n                const rawMax = routeQueryParams['price_max'];\n\n                const min = rawMin != null ? Number(rawMin) : null;\n                const max = rawMax != null ? Number(rawMax) : null;\n\n                if ((min != null && !Number.isNaN(min)) || (max != null && !Number.isNaN(max))) {\n                    if (min != null && !Number.isNaN(min)) {\n                        pr.currentMinPrice = min;\n                    }\n                    if (max != null && !Number.isNaN(max)) {\n                        pr.currentMaxPrice = max;\n                    }\n                }\n            }\n        }\n\n        this._filtersSubject.next(final_filters);\n    }\n\n    getFilters(paginationSettings?: PaginationSettings) {\n        if (this._filtersInitialized) return;\n        this._filtersInitialized = true;\n        const settings = paginationSettings || this._paginationSettings;\n        let final_filters: Filter[] = [];\n        let filtersToProcess = this._optionsFilters?.includes('all')\n            ? (['attributes', 'categories', 'dynamics', 'sort', 'price_range'] as FilterType[])\n            : this._optionsFilters;\n\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            filtersToProcess?.forEach(filterType => {\n                const filter = filterFactory.create(filterType, settings);\n                if (filter) {\n                    final_filters.push(filter);\n                } else {\n                    // console.warn(`Failed to create filter for type: ${filterType}`);\n                }\n            });\n        });\n\n        this._defaultFilters?.forEach(filterDefault => {\n            let filter = final_filters.find(filter => filter.type() == filterDefault.filter_type);\n            filter && filterDefault.codes.forEach(value => filter.setSelected(value));\n        });\n        this._filtersSubject.next(final_filters);\n        return final_filters;\n    }\n\n    getSpecificFilters(types: FilterType[]): Filter[] {\n        const filters: Filter[] = this._filtersSubject.value.filter(filter => types.includes(filter.type()))\n        return filters;\n    }\n\n    setFilterSelected(filterObj: Filter, filterElem: FilterElement) {\n        let final_filters: Filter[] = this._filtersSubject.value;\n        if (filterObj.type() !== 'price_range') {\n            let index = final_filters.findIndex(filter => filter.type() == filterObj.type());\n            final_filters[index].setSelected(filterElem, filterElem.value || filterElem.code);\n        }\n        this._filtersSubject.next(final_filters);\n        // }\n    }\n\n    updateFilters(paginationFilters: any): void {\n        const claves = Object.keys(paginationFilters)\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            const filter: Filter[] = []\n            this._filtersSubject.value.forEach((value: any) => {\n                if (value.type() == \"attributes\") {\n                    //value.data = [];\n                }\n            })\n        })\n    }\n\n    /** \n     * Actualiza internamente el rango de precio y re-emite \n     * el array de filtros para que PaginationService vuelva a llamar \n     */\n    public updatePriceRangeFilter(min: number, max: number): void {\n        const filters = this._filtersSubject.value;\n        const pr = filters.find(f => f.type() === 'price_range') as PriceRangeFilter | undefined;\n        if (!pr) return;\n        pr.setSelected(min, max);\n        this._filtersSubject.next([...filters]);\n    }\n\n    /**\n     * Fuerza a emitir de nuevo los filtros actuales.\n     * Útil cuando un filtro async (categorías, atributos) termina de cargarse\n     * y necesitamos que PaginationService vuelva a armar la URL.\n     */\n    public refreshFilters(): void {\n        const current = this._filtersSubject.value;\n        // Emitimos una copia para que los subscribers detecten el cambio\n        this._filtersSubject.next([...current]);\n    }\n\n    /**\n     * Punto central donde se hidratan los filtros a partir de la URL:\n     * - Lee type/value de la ruta (ej. /collection/categories/:value).\n     * - Lee query params (category, price_min, price_max, attributeCodes, search, etc.).\n     * - Crea instancias de filtros y les delega la hidratación específica.\n     *\n     * Este método se llama tanto en SSR como en navegador.\n     */\n    hydrateFromRoute(\n        paginationSettings: PaginationSettings,\n        routeParams: Params,\n        routeQuery: Params\n    ): void {\n        // hidratar search desde la URL ===\n        const searchFromUrl = routeQuery?.['search'];\n        if (typeof searchFromUrl === 'string' && searchFromUrl.trim() !== '') {\n            // Guardamos el valor para que otros componentes (Header, Collection) lo reutilicen\n            this._consts.searchValue = searchFromUrl.trim();\n        } else {\n            // Si no viene `search` en la URL, limpiamos el estado global para no arrastrar búsquedas viejas\n            this._consts.searchValue = '';\n        }\n\n        // Guardamos la última query completa para usarla en generateFinalApi (parche attributeCodes)\n        this._lastRouteQuery = routeQuery;\n\n        // Guardamos las settings actuales (type, value, limit, etc.)\n        this._paginationSettings = paginationSettings;\n\n        // Creamos los filtros base (attributes, categories, dynamics, sort, price_range, etc.)\n        let final_filters: Filter[] = [];\n        let filtersToProcess = this._optionsFilters?.includes('all')\n            ? (['attributes', 'categories', 'dynamics', 'sort', 'price_range'] as FilterType[])\n            : this._optionsFilters;\n\n        runInInjectionContext(this.environmentInjector, () => {\n            const filterFactory = new FilterFactory();\n            filtersToProcess?.forEach(filterType => {\n                const filter = filterFactory.create(filterType, paginationSettings);\n                if (filter) {\n                    final_filters.push(filter);\n                }\n            });\n        });\n\n        // Aplicamos defaultFilters si corresponde\n        this._defaultFilters?.forEach(filterDefault => {\n            const filter = final_filters.find(f => f.type() === filterDefault.filter_type);\n            filter && filterDefault.codes.forEach(value => filter.setSelected(value));\n        });\n\n        // Hidratar filtros con slug y query params genéricos\n        final_filters.forEach(f => {\n            const anyFilter = f as any;\n\n            // type/value en la URL: /collection/categories/:value\n            const urlType = routeParams['type'];\n            const urlValue = routeParams['value'];\n\n            // Por ejemplo, el CategoryFilter implementa setFromSlug para marcar la categoría correcta\n            if (urlType === 'categories' && urlValue && typeof anyFilter.setFromSlug === 'function') {\n                anyFilter.setFromSlug(urlValue);\n            }\n\n            // query params genéricos: ?category=0101&price_min=...&price_max=...&attributeCodes=...\n            // Cada filtro que lo soporte implementa hydrateFromQuery y decide qué leer.\n            if (typeof anyFilter.hydrateFromQuery === 'function') {\n                anyFilter.hydrateFromQuery(routeQuery);\n            }\n        });\n\n        // Hidratar específicamente el PriceRangeFilter desde la URL\n        const priceFilter = final_filters.find(\n            (f: Filter) => f instanceof PriceRangeFilter\n        ) as PriceRangeFilter | undefined;\n\n        if (priceFilter && routeQuery) {\n            const minFromUrl = routeQuery['price_min'];\n            const maxFromUrl = routeQuery['price_max'];\n\n            if (minFromUrl != null || maxFromUrl != null) {\n                const min = minFromUrl != null ? Number(minFromUrl) : priceFilter.minPrice;\n                const max = maxFromUrl != null ? Number(maxFromUrl) : priceFilter.maxPrice;\n\n                // usamos la API del filtro para setear el rango\n                priceFilter.setSelected(min, max);\n            } else {\n                // si no hay nada en la URL, lo reseteamos\n                priceFilter.reset();\n            }\n        }\n\n        // Emitimos filtros ya hidratados para que PaginationService y Collection reaccionen\n        this._filtersSubject.next(final_filters);\n\n        // Marcamos que el sistema de filtros está listo (útil para mostrar spinners, etc.)\n        this._readySubject.next(true);\n    }\n}\n\n// export type FilterType = 'attributes' | 'categories' | 'dynamics' | 'sort' | 'price_range'; // Use imported FilterType instead\n"]}
@@ -1,21 +1,59 @@
1
1
  import { inject, Injectable, signal } from '@angular/core';
2
2
  import { ConnectionService } from '../api';
3
- import { BehaviorSubject, filter, map, shareReplay, switchMap } from 'rxjs';
3
+ import { BehaviorSubject, distinctUntilChanged, filter, map, shareReplay, switchMap, tap } from 'rxjs';
4
4
  import { FiltersService } from './filters.service';
5
5
  import { CoreConstantsService } from '../constants';
6
6
  import * as i0 from "@angular/core";
7
7
  /**
8
- * Servicio para manejar la paginación de los productos.
9
- * @class PaginationService
8
+ * Servicio para manejar la paginación y la carga de productos.
9
+ *
10
+ * Se encarga de:
11
+ * - Escuchar los cambios en los filtros (categorías, atributos, precio, búsqueda, etc.).
12
+ * - Construir la URL al backend con esos filtros.
13
+ * - Pedir los productos al endpoint de product-search.
14
+ * - Exponer un observable con la última página de productos (`paginationData$`)
15
+ * y otros helpers (precio mínimo/máximo, siguiente página, etc.).
10
16
  */
11
17
  export class PaginationService {
12
18
  _connectionService = inject(ConnectionService);
13
19
  _filtersService = inject(FiltersService);
14
20
  _constants = inject(CoreConstantsService);
15
- paginationData$ = this._filtersService.filters$.pipe(shareReplay(1), filter(filters => filters && filters.length > 0), switchMap((filters) => {
21
+ /**
22
+ * Flujo principal: a partir de los filtros → arma URL → consulta backend → devuelve productos.
23
+ *
24
+ * Pasos:
25
+ * 1) Espera a que `FiltersService` emita filtros válidos.
26
+ * 2) Construye la URL final con `buildUrl()`.
27
+ * 3) Evita repetir llamadas si la URL no cambió (`distinctUntilChanged`).
28
+ * 4) Llama al backend con `getData(url)`.
29
+ * 5) Comparte el último resultado con todos los suscriptores (`shareReplay(1)`).
30
+ */
31
+ paginationData$ = this._filtersService.filters$.pipe(tap(filters => {
32
+ }), filter(filters => {
33
+ if (!filters || !filters.length)
34
+ return false;
35
+ const categoryFilter = filters.find(f => f.type() === 'categories');
36
+ const hasCategorySelected = !!categoryFilter?.getSelectedList()?.length;
37
+ const isCategoryRoute = this._constants.currentRouteType === 'categories';
38
+ // Si estamos en ruta de categorías y todavía no se marcó ninguna,
39
+ // esperamos a que el CategoryFilter se hidrate antes de disparar la llamada.
40
+ if (isCategoryRoute && !hasCategorySelected) {
41
+ return false;
42
+ }
43
+ return true;
44
+ }),
45
+ // 2) Convertimos filtros -> URL
46
+ map(filters => {
16
47
  const url = this.buildUrl(filters);
17
- return this.getData(url);
18
- }));
48
+ return url;
49
+ }),
50
+ // 3) Solo seguimos si la URL cambió respecto de la emisión anterior
51
+ distinctUntilChanged(),
52
+ // 4) Llamamos al backend con la URL construida
53
+ tap(url => {
54
+ }), switchMap(url => this.getData(url)),
55
+ // 5) Reutilizar resultado si alguien más se suscribe (evita repetir la petición)
56
+ shareReplay(1));
19
57
  _dataPagination = signal({
20
58
  attributes: [],
21
59
  category: [],
@@ -84,7 +122,8 @@ export class PaginationService {
84
122
  this._dataPagination.set({ ...response, called: true });
85
123
  this._finished = (response.page == response.pages);
86
124
  this._waiting = false;
87
- // Emitir los valores de price_min y price_max a través de priceRangeSubject
125
+ // Emitir los valores de price_min y price_max a través de priceRangeSubject,
126
+ // para que otros componentes (ej. filtro de precio) puedan mostrarlos.
88
127
  this.priceRangeSubject.next({
89
128
  price_min: response.price_min,
90
129
  price_max: response.price_max
@@ -144,27 +183,36 @@ export class PaginationService {
144
183
  * @returns
145
184
  */
146
185
  buildUrl(filters) {
147
- if (this._constants.searchValue) {
148
- return this._filtersService.generateFinalApi(this._constants.searchValue);
149
- }
150
- else {
151
- return this._filtersService.generateFinalApi();
152
- }
186
+ const url = this._constants.searchValue
187
+ ? this._filtersService.generateFinalApi(this._constants.searchValue)
188
+ : this._filtersService.generateFinalApi();
189
+ return url;
153
190
  }
154
191
  /**
155
- * Devuelve un observable de los productos que devolvio la última página obtenida.
192
+ * Devuelve un observable de los productos que devolvió la última página obtenida.
156
193
  * @param url
157
194
  * @returns
158
195
  */
159
196
  getData(url) {
160
- this._resetSubject.next(true);
161
- this._nextProductsSubject.next([]);
162
- this._connectionService.get(url, { limit: 10, page: 1 }).pipe(map(res => {
163
- //console.log(res)
164
- this._nextProductsSubject.next(res.items);
197
+ // Detectar la página desde la URL (?page=n). Si no viene, asumimos page=1.
198
+ const pageMatch = url.match(/[?&]page=(\d+)/);
199
+ const page = pageMatch ? Number(pageMatch[1]) : 1;
200
+ // Si es la primera página, reseteamos estado interno
201
+ if (page === 1) {
202
+ this._resetSubject.next(true);
203
+ this._nextProductsSubject.next([]);
204
+ }
205
+ // Llamamos al backend con el page correcto
206
+ return this._connectionService.get(url, { limit: 10, page }).pipe(tap((res) => {
207
+ // Actualizar datos de paginación + price_min/price_max
165
208
  res.links ? this.updatePageData(res) : this.finish(res);
166
- })).subscribe();
167
- return this.nextProducts$;
209
+ }), map((res) => {
210
+ const items = res.items ?? [];
211
+ // Mantener nextProducts$ sincronizado (por compatibilidad)
212
+ this._nextProductsSubject.next(items);
213
+ // Lo que ve paginationData$ (switchMap) son estos items
214
+ return items;
215
+ }));
168
216
  }
169
217
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PaginationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
170
218
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PaginationService, providedIn: 'root' });
@@ -175,4 +223,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
175
223
  providedIn: 'root'
176
224
  }]
177
225
  }], ctorParameters: () => [] });
178
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/pagination.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable, signal } from '@angular/core';\nimport { ConnectionService } from '../api';\nimport { BehaviorSubject, filter, map, Observable, shareReplay, switchMap, tap } from 'rxjs';\nimport { Pagination, Product } from '../interfaces';\nimport { FiltersService } from './filters.service';\nimport { Filter } from '../classes';\nimport { CoreConstantsService } from '../constants';\n/**\n * Servicio para manejar la paginación de los productos.\n * @class PaginationService\n */\n@Injectable({\n\tprovidedIn: 'root'\n})\nexport class PaginationService {\n\n\tprivate _connectionService: ConnectionService = inject(ConnectionService);\n\tprivate _filtersService: FiltersService = inject(FiltersService)\n\tprivate _constants: CoreConstantsService = inject(CoreConstantsService);\n\t\n\tpublic paginationData$ = this._filtersService.filters$.pipe(\n\t\tshareReplay(1),\n\t\tfilter(filters => filters && filters.length > 0),\n\t\tswitchMap((filters) => {\t\t\t\n\t\t\tconst url = this.buildUrl(filters);\t\t\t\n\t\t\treturn this.getData(url);\n\t\t})\n\t)\n\n\tprivate _dataPagination: any = signal<Pagination>({\n\t\tattributes: [],\n\t\tcategory: [],\n\t\toptions: [],\n\t\tlimit: 0,\n\t\tpage: 0,\n\t\tpages: 0,\n\t\ttotal: 0,\n\t\titems: [],\n\t\tcalled: false,\n\t\tprice_max: 0,\n\t\tprice_min: 0\n\t});\n\n\tprivate _waiting: boolean = false;\n\tprivate _finished: boolean = false;\n\tprivate _apiUrl: string = '';\n\tprivate _start: boolean = false;\n\n\tprivate _nextProductsSubject: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([]);\n\tpublic nextProducts$ = this._nextProductsSubject.asObservable();\n\n\tprivate _resetSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n\tpublic reset$ = this._resetSubject.asObservable();\n\n\tprivate priceRangeSubject = new BehaviorSubject<{ price_min: number, price_max: number }>({ price_min: 0, price_max: 0 });\n\tpublic priceRange$ = this.priceRangeSubject.asObservable();\n\t\n\tconstructor() { }\n\n\t/**\n\t * Inicializa la configuración para el Servicio de Paginación.\n\t * @param apiUrl Url con los filtros aplicados. \n\t * @param limit limite de productos en cada petición.\n\t */\n\tinitialize(apiUrl: string, limit?: number) {\n\t\t//console.log(apiUrl)\n\t\tthis._apiUrl = apiUrl;\n\t\tthis._dataPagination().limit = limit || this._dataPagination().limit || 10;\n\t\tthis._finished = false;\n\t\tthis._start = true;\n\t\tthis._nextProductsSubject.next([]);\n\t}\n\t/**\n\t * Obtengo los datos de la siguiente página.\n\t * @param next \n\t * @returns \n\t */\n\tgetNext(next?: any): boolean {\n\t\tthis._waiting = true;\n\t\tthis._nextProductsSubject.next([]);\n\t\tthis._apiUrl = this._apiUrl.replace('%20', ' ');// mas para cuando es un search value.\n\t\tlet nextProducts;\n\t\tif (this._start) {\n\t\t\tnextProducts = this._connectionService.get(this._apiUrl, { limit: this._dataPagination().limit, page: 1 })\n\t\t\tthis._start = false\n\t\t} else {\n\t\t\tnextProducts = this._connectionService.get(this._dataPagination().links.next.slice(1));\n\t\t}\n\n\t\t!this._finished && nextProducts.subscribe(\n\t\t\t(response: Pagination) => {\n\t\t\t\tresponse.links ? this.updatePageData(response) : this.finish(response)\n\t\t\t\tthis._nextProductsSubject.next(response.items);\n\t\t\t}\n\t\t)\n\t\treturn true;\n\t}\n\t/**\n\t * Actualiza los datos de la pagina.\n\t * @param response respuesta del backend o objecto del tipo `Pagination`.\n\t */\n\tupdatePageData(response: Pagination): void {\n\t\tthis._dataPagination.set({ ...response, called: true })\n\t\tthis._finished = (response.page == response.pages)\n\t\tthis._waiting = false;\n\t\t // Emitir los valores de price_min y price_max a través de priceRangeSubject\n        this.priceRangeSubject.next({\n            price_min: response.price_min,\n            price_max: response.price_max\n        });\n\n\t}\n\t/**\n\t * Marca el final de las paginas, es decir cuando ya no queda mas elementos que mostrar.\n\t * @param response \n\t */\n\tfinish(response: Pagination): void {\n\t\tthis._finished = true;\n\t\tthis._waiting = false;\n\t}\n\t/**\n\t * Obtengo los productos de la pagina cargada actual. \n\t * @returns \n\t */\n\tpublic getNextProducts(): Product[] {\n\t\treturn this._nextProductsSubject.value;\n\t}\n\t/**\n\t * Devuelve si esta esperando o no la petición al backend.\n\t * @returns \n\t */\n\tpublic getWaiting(): boolean { return this._waiting; }\n\t/**\n\t * Setea la variable waiting.\n\t * @param value \n\t */\n\tpublic setWaiting(value: boolean): void { this._waiting = value }\n\t/**\n\t * Getter para la variable `_finished`.\n\t * @returns \n\t */\n\tpublic getFinished(): boolean { return this._finished; }\n\t/**\n\t * Devuelve el total de Items.\n\t * @returns \n\t */\n\tpublic getTotalItems(): number {\n\t\treturn this._dataPagination().limit * this._dataPagination().page;\n\t}\n\t/**\n\t * Obtengo un objecto con los datos de los filtros resultantes.\n\t * @returns \n\t */\n\tpublic getFiltersPagination(): any {\n\t\treturn {\n\t\t\t\"categories\": this._dataPagination().category,\n\t\t\t\"attributes\": this._dataPagination().attributes,\n\t\t\t\"options\": this._dataPagination().options\n\t\t}\n\t}\n\t/**\n\t * Para contruiste la url en base a los filtros aplicados.\n\t * @param filters \n\t * @returns \n\t */\n\tpublic buildUrl(filters: Filter[]) {\t\t\n\t\tif (this._constants.searchValue){\n\t\t\treturn this._filtersService.generateFinalApi(this._constants.searchValue);\n\t\t} else{\n\t\t\treturn this._filtersService.generateFinalApi();\n\t\t}\n\t\t\n\t}\n\t/**\n\t * Devuelve un observable de los productos que devolvio la última página obtenida.\n\t * @param url \n\t * @returns \n\t */\n\tpublic getData(url: string): Observable<any> {\n\t\tthis._resetSubject.next(true);\n\t\tthis._nextProductsSubject.next([]);\n\t\tthis._connectionService.get(url, { limit: 10, page: 1 }).pipe(\n\t\t\tmap(res =>{ \n\t\t\t\t//console.log(res)\n\t\t\t\tthis._nextProductsSubject.next(res.items);\n\t\t\t\tres.links ? this.updatePageData(res) : this.finish(res)\n\t\t\t})).subscribe();\n\n\t\treturn this.nextProducts$;\n\t}\n}\n\n"]}
226
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/pagination.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,QAAQ,CAAC;AAC3C,OAAO,EAAE,eAAe,EAAE,oBAAoB,EAAE,MAAM,EAAE,GAAG,EAAc,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AAEnH,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAEnD,OAAO,EAAE,oBAAoB,EAAE,MAAM,cAAc,CAAC;;AACpD;;;;;;;;;GASG;AAIH,MAAM,OAAO,iBAAiB;IAErB,kBAAkB,GAAsB,MAAM,CAAC,iBAAiB,CAAC,CAAC;IAClE,eAAe,GAAmB,MAAM,CAAC,cAAc,CAAC,CAAA;IACxD,UAAU,GAAyB,MAAM,CAAC,oBAAoB,CAAC,CAAC;IAExE;;;;;;;;;OASG;IACI,eAAe,GAAG,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,IAAI,CAC1D,GAAG,CAAC,OAAO,CAAC,EAAE;IACd,CAAC,CAAC,EACF,MAAM,CAAC,OAAO,CAAC,EAAE;QAChB,IAAI,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM;YAAE,OAAO,KAAK,CAAC;QAE9C,MAAM,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,YAAY,CAAC,CAAC;QACpE,MAAM,mBAAmB,GAAG,CAAC,CAAC,cAAc,EAAE,eAAe,EAAE,EAAE,MAAM,CAAC;QAExE,MAAM,eAAe,GACpB,IAAI,CAAC,UAAU,CAAC,gBAAgB,KAAK,YAAY,CAAC;QAEnD,kEAAkE;QAClE,6EAA6E;QAC7E,IAAI,eAAe,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC7C,OAAO,KAAK,CAAC;QACd,CAAC;QAED,OAAO,IAAI,CAAC;IACb,CAAC,CAAC;IAEF,gCAAgC;IAChC,GAAG,CAAC,OAAO,CAAC,EAAE;QACb,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QACnC,OAAO,GAAG,CAAC;IACZ,CAAC,CAAC;IAEF,oEAAoE;IACpE,oBAAoB,EAAE;IAEtB,+CAA+C;IAC/C,GAAG,CAAC,GAAG,CAAC,EAAE;IACV,CAAC,CAAC,EACF,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAEnC,iFAAiF;IACjF,WAAW,CAAC,CAAC,CAAC,CACd,CAAC;IAEM,eAAe,GAAQ,MAAM,CAAa;QACjD,UAAU,EAAE,EAAE;QACd,QAAQ,EAAE,EAAE;QACZ,OAAO,EAAE,EAAE;QACX,KAAK,EAAE,CAAC;QACR,IAAI,EAAE,CAAC;QACP,KAAK,EAAE,CAAC;QACR,KAAK,EAAE,CAAC;QACR,KAAK,EAAE,EAAE;QACT,MAAM,EAAE,KAAK;QACb,SAAS,EAAE,CAAC;QACZ,SAAS,EAAE,CAAC;KACZ,CAAC,CAAC;IAEK,QAAQ,GAAY,KAAK,CAAC;IAC1B,SAAS,GAAY,KAAK,CAAC;IAC3B,OAAO,GAAW,EAAE,CAAC;IACrB,MAAM,GAAY,KAAK,CAAC;IAExB,oBAAoB,GAA+B,IAAI,eAAe,CAAY,EAAE,CAAC,CAAC;IACvF,aAAa,GAAG,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;IAExD,aAAa,GAA6B,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;IAC/E,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;IAE1C,iBAAiB,GAAG,IAAI,eAAe,CAA2C,EAAE,SAAS,EAAE,CAAC,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,CAAC;IACnH,WAAW,GAAG,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;IAE3D,gBAAgB,CAAC;IAEjB;;;;OAIG;IACH,UAAU,CAAC,MAAc,EAAE,KAAc;QACxC,qBAAqB;QACrB,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC;QACtB,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,GAAG,KAAK,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,IAAI,EAAE,CAAC;QAC3E,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACnB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACpC,CAAC;IACD;;;;OAIG;IACH,OAAO,CAAC,IAAU;QACjB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QACrB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACnC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,CAAA,sCAAsC;QACtF,IAAI,YAAY,CAAC;QACjB,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,YAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC,CAAA;YAC1G,IAAI,CAAC,MAAM,GAAG,KAAK,CAAA;QACpB,CAAC;aAAM,CAAC;YACP,YAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACxF,CAAC;QAED,CAAC,IAAI,CAAC,SAAS,IAAI,YAAY,CAAC,SAAS,CACxC,CAAC,QAAoB,EAAE,EAAE;YACxB,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAA;YACtE,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAChD,CAAC,CACD,CAAA;QACD,OAAO,IAAI,CAAC;IACb,CAAC;IACD;;;OAGG;IACH,cAAc,CAAC,QAAoB;QAClC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,EAAE,GAAG,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAA;QACvD,IAAI,CAAC,SAAS,GAAG,CAAC,QAAQ,CAAC,IAAI,IAAI,QAAQ,CAAC,KAAK,CAAC,CAAA;QAClD,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACtB,6EAA6E;QAC7E,uEAAuE;QACvE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC;YAC3B,SAAS,EAAE,QAAQ,CAAC,SAAS;YAC7B,SAAS,EAAE,QAAQ,CAAC,SAAS;SAC7B,CAAC,CAAC;IAEJ,CAAC;IACD;;;OAGG;IACH,MAAM,CAAC,QAAoB;QAC1B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;IACvB,CAAC;IACD;;;OAGG;IACI,eAAe;QACrB,OAAO,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC;IACxC,CAAC;IACD;;;OAGG;IACI,UAAU,KAAc,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IACtD;;;OAGG;IACI,UAAU,CAAC,KAAc,IAAU,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAA,CAAC,CAAC;IACjE;;;OAGG;IACI,WAAW,KAAc,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;IACxD;;;OAGG;IACI,aAAa;QACnB,OAAO,IAAI,CAAC,eAAe,EAAE,CAAC,KAAK,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,IAAI,CAAC;IACnE,CAAC;IACD;;;OAGG;IACI,oBAAoB;QAC1B,OAAO;YACN,YAAY,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC,QAAQ;YAC7C,YAAY,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC,UAAU;YAC/C,SAAS,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC,OAAO;SACzC,CAAA;IACF,CAAC;IACD;;;;OAIG;IACI,QAAQ,CAAC,OAAiB;QAChC,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,WAAW;YACtC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC;YACpE,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,gBAAgB,EAAE,CAAC;QAE3C,OAAO,GAAG,CAAC;IACZ,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAC,GAAW;QAEzB,2EAA2E;QAC3E,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;QAC9C,MAAM,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAElD,qDAAqD;QACrD,IAAI,IAAI,KAAK,CAAC,EAAE,CAAC;YAChB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC9B,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACpC,CAAC;QAED,2CAA2C;QAC3C,OAAO,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAChE,GAAG,CAAC,CAAC,GAAe,EAAE,EAAE;YACvB,uDAAuD;YACvD,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACzD,CAAC,CAAC,EACF,GAAG,CAAC,CAAC,GAAe,EAAE,EAAE;YACvB,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC;YAE9B,2DAA2D;YAC3D,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAEtC,wDAAwD;YACxD,OAAO,KAAK,CAAC;QACd,CAAC,CAAC,CACF,CAAC;IACH,CAAC;wGAzOW,iBAAiB;4GAAjB,iBAAiB,cAFjB,MAAM;;4FAEN,iBAAiB;kBAH7B,UAAU;mBAAC;oBACX,UAAU,EAAE,MAAM;iBAClB","sourcesContent":["import { inject, Injectable, signal } from '@angular/core';\nimport { ConnectionService } from '../api';\nimport { BehaviorSubject, distinctUntilChanged, filter, map, Observable, shareReplay, switchMap, tap } from 'rxjs';\nimport { Pagination, Product } from '../interfaces';\nimport { FiltersService } from './filters.service';\nimport { Filter } from '../classes';\nimport { CoreConstantsService } from '../constants';\n/**\n * Servicio para manejar la paginación y la carga de productos.\n *\n * Se encarga de:\n * - Escuchar los cambios en los filtros (categorías, atributos, precio, búsqueda, etc.).\n * - Construir la URL al backend con esos filtros.\n * - Pedir los productos al endpoint de product-search.\n * - Exponer un observable con la última página de productos (`paginationData$`)\n *   y otros helpers (precio mínimo/máximo, siguiente página, etc.).\n */\n@Injectable({\n\tprovidedIn: 'root'\n})\nexport class PaginationService {\n\n\tprivate _connectionService: ConnectionService = inject(ConnectionService);\n\tprivate _filtersService: FiltersService = inject(FiltersService)\n\tprivate _constants: CoreConstantsService = inject(CoreConstantsService);\n\n\t/**\n\t * Flujo principal: a partir de los filtros → arma URL → consulta backend → devuelve productos.\n\t *\n\t * Pasos:\n\t * 1) Espera a que `FiltersService` emita filtros válidos.\n\t * 2) Construye la URL final con `buildUrl()`.\n\t * 3) Evita repetir llamadas si la URL no cambió (`distinctUntilChanged`).\n\t * 4) Llama al backend con `getData(url)`.\n\t * 5) Comparte el último resultado con todos los suscriptores (`shareReplay(1)`).\n\t */\n\tpublic paginationData$ = this._filtersService.filters$.pipe(\n\t\ttap(filters => {\n\t\t}),\n\t\tfilter(filters => {\n\t\t\tif (!filters || !filters.length) return false;\n\n\t\t\tconst categoryFilter = filters.find(f => f.type() === 'categories');\n\t\t\tconst hasCategorySelected = !!categoryFilter?.getSelectedList()?.length;\n\n\t\t\tconst isCategoryRoute =\n\t\t\t\tthis._constants.currentRouteType === 'categories';\n\n\t\t\t// Si estamos en ruta de categorías y todavía no se marcó ninguna,\n\t\t\t// esperamos a que el CategoryFilter se hidrate antes de disparar la llamada.\n\t\t\tif (isCategoryRoute && !hasCategorySelected) {\n\t\t\t\treturn false;\n\t\t\t}\n\n\t\t\treturn true;\n\t\t}),\n\n\t\t// 2) Convertimos filtros -> URL\n\t\tmap(filters => {\n\t\t\tconst url = this.buildUrl(filters);\n\t\t\treturn url;\n\t\t}),\n\n\t\t// 3) Solo seguimos si la URL cambió respecto de la emisión anterior\n\t\tdistinctUntilChanged(),\n\n\t\t// 4) Llamamos al backend con la URL construida\n\t\ttap(url => {\n\t\t}),\n\t\tswitchMap(url => this.getData(url)),\n\n\t\t// 5) Reutilizar resultado si alguien más se suscribe (evita repetir la petición)\n\t\tshareReplay(1)\n\t);\n\n\tprivate _dataPagination: any = signal<Pagination>({\n\t\tattributes: [],\n\t\tcategory: [],\n\t\toptions: [],\n\t\tlimit: 0,\n\t\tpage: 0,\n\t\tpages: 0,\n\t\ttotal: 0,\n\t\titems: [],\n\t\tcalled: false,\n\t\tprice_max: 0,\n\t\tprice_min: 0\n\t});\n\n\tprivate _waiting: boolean = false;\n\tprivate _finished: boolean = false;\n\tprivate _apiUrl: string = '';\n\tprivate _start: boolean = false;\n\n\tprivate _nextProductsSubject: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([]);\n\tpublic nextProducts$ = this._nextProductsSubject.asObservable();\n\n\tprivate _resetSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n\tpublic reset$ = this._resetSubject.asObservable();\n\n\tprivate priceRangeSubject = new BehaviorSubject<{ price_min: number, price_max: number }>({ price_min: 0, price_max: 0 });\n\tpublic priceRange$ = this.priceRangeSubject.asObservable();\n\n\tconstructor() { }\n\n\t/**\n\t * Inicializa la configuración para el Servicio de Paginación.\n\t * @param apiUrl Url con los filtros aplicados. \n\t * @param limit limite de productos en cada petición.\n\t */\n\tinitialize(apiUrl: string, limit?: number) {\n\t\t//console.log(apiUrl)\n\t\tthis._apiUrl = apiUrl;\n\t\tthis._dataPagination().limit = limit || this._dataPagination().limit || 10;\n\t\tthis._finished = false;\n\t\tthis._start = true;\n\t\tthis._nextProductsSubject.next([]);\n\t}\n\t/**\n\t * Obtengo los datos de la siguiente página.\n\t * @param next \n\t * @returns \n\t */\n\tgetNext(next?: any): boolean {\n\t\tthis._waiting = true;\n\t\tthis._nextProductsSubject.next([]);\n\t\tthis._apiUrl = this._apiUrl.replace('%20', ' ');// mas para cuando es un search value.\n\t\tlet nextProducts;\n\t\tif (this._start) {\n\t\t\tnextProducts = this._connectionService.get(this._apiUrl, { limit: this._dataPagination().limit, page: 1 })\n\t\t\tthis._start = false\n\t\t} else {\n\t\t\tnextProducts = this._connectionService.get(this._dataPagination().links.next.slice(1));\n\t\t}\n\n\t\t!this._finished && nextProducts.subscribe(\n\t\t\t(response: Pagination) => {\n\t\t\t\tresponse.links ? this.updatePageData(response) : this.finish(response)\n\t\t\t\tthis._nextProductsSubject.next(response.items);\n\t\t\t}\n\t\t)\n\t\treturn true;\n\t}\n\t/**\n\t * Actualiza los datos de la pagina.\n\t * @param response respuesta del backend o objecto del tipo `Pagination`.\n\t */\n\tupdatePageData(response: Pagination): void {\n\t\tthis._dataPagination.set({ ...response, called: true })\n\t\tthis._finished = (response.page == response.pages)\n\t\tthis._waiting = false;\n\t\t// Emitir los valores de price_min y price_max a través de priceRangeSubject,\n\t\t// para que otros componentes (ej. filtro de precio) puedan mostrarlos.\n\t\tthis.priceRangeSubject.next({\n\t\t\tprice_min: response.price_min,\n\t\t\tprice_max: response.price_max\n\t\t});\n\n\t}\n\t/**\n\t * Marca el final de las paginas, es decir cuando ya no queda mas elementos que mostrar.\n\t * @param response \n\t */\n\tfinish(response: Pagination): void {\n\t\tthis._finished = true;\n\t\tthis._waiting = false;\n\t}\n\t/**\n\t * Obtengo los productos de la pagina cargada actual. \n\t * @returns \n\t */\n\tpublic getNextProducts(): Product[] {\n\t\treturn this._nextProductsSubject.value;\n\t}\n\t/**\n\t * Devuelve si esta esperando o no la petición al backend.\n\t * @returns \n\t */\n\tpublic getWaiting(): boolean { return this._waiting; }\n\t/**\n\t * Setea la variable waiting.\n\t * @param value \n\t */\n\tpublic setWaiting(value: boolean): void { this._waiting = value }\n\t/**\n\t * Getter para la variable `_finished`.\n\t * @returns \n\t */\n\tpublic getFinished(): boolean { return this._finished; }\n\t/**\n\t * Devuelve el total de Items.\n\t * @returns \n\t */\n\tpublic getTotalItems(): number {\n\t\treturn this._dataPagination().limit * this._dataPagination().page;\n\t}\n\t/**\n\t * Obtengo un objecto con los datos de los filtros resultantes.\n\t * @returns \n\t */\n\tpublic getFiltersPagination(): any {\n\t\treturn {\n\t\t\t\"categories\": this._dataPagination().category,\n\t\t\t\"attributes\": this._dataPagination().attributes,\n\t\t\t\"options\": this._dataPagination().options\n\t\t}\n\t}\n\t/**\n\t * Para contruiste la url en base a los filtros aplicados.\n\t * @param filters \n\t * @returns \n\t */\n\tpublic buildUrl(filters: Filter[]) {\n\t\tconst url = this._constants.searchValue\n\t\t\t? this._filtersService.generateFinalApi(this._constants.searchValue)\n\t\t\t: this._filtersService.generateFinalApi();\n\n\t\treturn url;\n\t}\n\n\t/**\n\t * Devuelve un observable de los productos que devolvió la última página obtenida.\n\t * @param url \n\t * @returns \n\t */\n\tpublic getData(url: string): Observable<Product[]> {\n\n\t\t// Detectar la página desde la URL (?page=n). Si no viene, asumimos page=1.\n\t\tconst pageMatch = url.match(/[?&]page=(\\d+)/);\n\t\tconst page = pageMatch ? Number(pageMatch[1]) : 1;\n\n\t\t// Si es la primera página, reseteamos estado interno\n\t\tif (page === 1) {\n\t\t\tthis._resetSubject.next(true);\n\t\t\tthis._nextProductsSubject.next([]);\n\t\t}\n\n\t\t// Llamamos al backend con el page correcto\n\t\treturn this._connectionService.get(url, { limit: 10, page }).pipe(\n\t\t\ttap((res: Pagination) => {\n\t\t\t\t// Actualizar datos de paginación + price_min/price_max\n\t\t\t\tres.links ? this.updatePageData(res) : this.finish(res);\n\t\t\t}),\n\t\t\tmap((res: Pagination) => {\n\t\t\t\tconst items = res.items ?? [];\n\n\t\t\t\t// Mantener nextProducts$ sincronizado (por compatibilidad)\n\t\t\t\tthis._nextProductsSubject.next(items);\n\n\t\t\t\t// Lo que ve paginationData$ (switchMap) son estos items\n\t\t\t\treturn items;\n\t\t\t})\n\t\t);\n\t}\n}\n\n"]}