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.
- package/README.md +6 -2
- package/esm2022/lib/classes/filters/attributes-filter.mjs +74 -4
- package/esm2022/lib/classes/filters/category-filter.mjs +105 -26
- package/esm2022/lib/classes/filters/filter-factory.mjs +7 -3
- package/esm2022/lib/classes/filters/price_range-filter.mjs +3 -3
- package/esm2022/lib/constants/core.constants.service.mjs +12 -1
- package/esm2022/lib/ec-components/collection-ec/collection-ec.component.mjs +41 -17
- package/esm2022/lib/ec-components/filters-ec/filters-ec.component.mjs +42 -9
- package/esm2022/lib/ec-components/header-ec/header-ec.component.mjs +12 -6
- package/esm2022/lib/ec-components/price-range-filter/price-range-filter.component.mjs +13 -2
- package/esm2022/lib/ec-services/filters.service.mjs +124 -18
- package/esm2022/lib/ec-services/pagination.service.mjs +70 -22
- package/esm2022/lib/ec-services/products.service.mjs +5 -3
- package/fesm2022/ng-easycommerce-v18.mjs +490 -98
- package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
- package/lib/classes/filters/attributes-filter.d.ts +24 -0
- package/lib/classes/filters/category-filter.d.ts +30 -3
- package/lib/constants/core.constants.service.d.ts +7 -0
- package/lib/ec-components/collection-ec/collection-ec.component.d.ts +5 -4
- package/lib/ec-components/filters-ec/filters-ec.component.d.ts +13 -0
- package/lib/ec-components/price-range-filter/price-range-filter.component.d.ts +2 -0
- package/lib/ec-services/filters.service.d.ts +18 -1
- package/lib/ec-services/pagination.service.d.ts +21 -5
- package/lib/ec-services/products.service.d.ts +1 -1
- 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]
|
|
46
|
+
if (extra && extra.split('=')[1] !== '') {
|
|
47
|
+
extra_params += extra;
|
|
48
|
+
}
|
|
45
49
|
});
|
|
46
|
-
|
|
47
|
-
|
|
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":"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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';\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
|
|
9
|
-
*
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
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
|
|
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
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
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
|
-
})
|
|
167
|
-
|
|
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":"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","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"]}
|