ng-easycommerce-v18 0.5.2 → 0.5.3

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 CHANGED
@@ -1,5 +1,6 @@
1
+ # Version 0.5.3
2
+ - Se modifica servicio de producto y paginacion porque el collection lanza endpoint de pagina 2 pero no la muestra y el resto si.
1
3
  # Version 0.5.2
2
- - Se modifica la funcion para limpiar la descripcion sin html ni css.
3
4
  - Se agrega descrption2 al observable Product para recibirlo del endpoint.
4
5
  - Se modifica la funcion para limpiar la descripcion sin html ni css.
5
6
  # Version 0.5.1
@@ -84,12 +84,12 @@ export class PaginationService {
84
84
  * @param limit limite de productos en cada petición.
85
85
  */
86
86
  initialize(apiUrl, limit) {
87
- //console.log(apiUrl)
88
87
  this._apiUrl = apiUrl;
89
88
  this._dataPagination().limit = limit || this._dataPagination().limit || 10;
90
89
  this._finished = false;
90
+ this._waiting = false; // Asegurar que no está bloqueado
91
91
  this._start = true;
92
- this._nextProductsSubject.next([]);
92
+ // NO emitir array vacío aquí para evitar triggers innecesarios
93
93
  }
94
94
  /**
95
95
  * Obtengo los datos de la siguiente página.
@@ -97,20 +97,31 @@ export class PaginationService {
97
97
  * @returns
98
98
  */
99
99
  getNext(next) {
100
- this._waiting = true;
101
- this._nextProductsSubject.next([]);
102
- this._apiUrl = this._apiUrl.replace('%20', ' '); // mas para cuando es un search value.
103
- let nextProducts;
104
- if (this._start) {
105
- nextProducts = this._connectionService.get(this._apiUrl, { limit: this._dataPagination().limit, page: 1 });
106
- this._start = false;
100
+ // Si ya estamos esperando una respuesta o ya terminamos, no hacer nada
101
+ if (this._waiting || this._finished) {
102
+ return false;
107
103
  }
108
- else {
109
- nextProducts = this._connectionService.get(this._dataPagination().links.next.slice(1));
104
+ // Verificar que hay un link next disponible
105
+ if (!this._dataPagination().links?.next) {
106
+ return false;
110
107
  }
111
- !this._finished && nextProducts.subscribe((response) => {
108
+ this._waiting = true;
109
+ this._apiUrl = this._apiUrl.replace('%20', ' '); // mas para cuando es un search value.
110
+ // Siempre usar el link "next" de la respuesta anterior
111
+ const nextProducts = this._connectionService.get(this._dataPagination().links.next.slice(1));
112
+ nextProducts.subscribe((response) => {
112
113
  response.links ? this.updatePageData(response) : this.finish(response);
113
- this._nextProductsSubject.next(response.items);
114
+ // Solo emitir si hay productos reales
115
+ if (response.items && response.items.length > 0) {
116
+ this._nextProductsSubject.next(response.items);
117
+ }
118
+ else {
119
+ // Si no hay productos, igual liberamos el waiting
120
+ this._waiting = false;
121
+ }
122
+ }, (error) => {
123
+ // En caso de error, liberar el bloqueo
124
+ this._waiting = false;
114
125
  });
115
126
  return true;
116
127
  }
@@ -200,7 +211,10 @@ export class PaginationService {
200
211
  // Si es la primera página, reseteamos estado interno
201
212
  if (page === 1) {
202
213
  this._resetSubject.next(true);
203
- this._nextProductsSubject.next([]);
214
+ // Resetear estado de paginación para permitir scroll
215
+ this._finished = false;
216
+ this._waiting = false;
217
+ this._start = false; // Ya no es el inicio después de getData
204
218
  }
205
219
  // Llamamos al backend con el page correcto
206
220
  return this._connectionService.get(url, { limit: 10, page }).pipe(tap((res) => {
@@ -221,4 +235,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
221
235
  providedIn: 'root'
222
236
  }]
223
237
  }], ctorParameters: () => [] });
224
- //# 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';\r\nimport { ConnectionService } from '../api';\r\nimport { BehaviorSubject, distinctUntilChanged, filter, map, Observable, shareReplay, switchMap, tap } from 'rxjs';\r\nimport { Pagination, Product } from '../interfaces';\r\nimport { FiltersService } from './filters.service';\r\nimport { Filter } from '../classes';\r\nimport { CoreConstantsService } from '../constants';\r\n/**\r\n * Servicio para manejar la paginación y la carga de productos.\r\n *\r\n * Se encarga de:\r\n * - Escuchar los cambios en los filtros (categorías, atributos, precio, búsqueda, etc.).\r\n * - Construir la URL al backend con esos filtros.\r\n * - Pedir los productos al endpoint de product-search.\r\n * - Exponer un observable con la última página de productos (`paginationData$`)\r\n *   y otros helpers (precio mínimo/máximo, siguiente página, etc.).\r\n */\r\n@Injectable({\r\n\tprovidedIn: 'root'\r\n})\r\nexport class PaginationService {\r\n\r\n\tprivate _connectionService: ConnectionService = inject(ConnectionService);\r\n\tprivate _filtersService: FiltersService = inject(FiltersService)\r\n\tprivate _constants: CoreConstantsService = inject(CoreConstantsService);\r\n\r\n\t/**\r\n\t * Flujo principal: a partir de los filtros → arma URL → consulta backend → devuelve productos.\r\n\t *\r\n\t * Pasos:\r\n\t * 1) Espera a que `FiltersService` emita filtros válidos.\r\n\t * 2) Construye la URL final con `buildUrl()`.\r\n\t * 3) Evita repetir llamadas si la URL no cambió (`distinctUntilChanged`).\r\n\t * 4) Llama al backend con `getData(url)`.\r\n\t * 5) Comparte el último resultado con todos los suscriptores (`shareReplay(1)`).\r\n\t */\r\n\tpublic paginationData$ = this._filtersService.filters$.pipe(\r\n\t\ttap(filters => {\r\n\t\t}),\r\n\t\tfilter(filters => {\r\n\t\t\tif (!filters || !filters.length) return false;\r\n\r\n\t\t\tconst categoryFilter = filters.find(f => f.type() === 'categories');\r\n\t\t\tconst hasCategorySelected = !!categoryFilter?.getSelectedList()?.length;\r\n\r\n\t\t\tconst isCategoryRoute =\r\n\t\t\t\tthis._constants.currentRouteType === 'categories';\r\n\r\n\t\t\t// Si estamos en ruta de categorías y todavía no se marcó ninguna,\r\n\t\t\t// esperamos a que el CategoryFilter se hidrate antes de disparar la llamada.\r\n\t\t\tif (isCategoryRoute && !hasCategorySelected) {\r\n\t\t\t\treturn false;\r\n\t\t\t}\r\n\r\n\t\t\treturn true;\r\n\t\t}),\r\n\r\n\t\t// 2) Convertimos filtros -> URL\r\n\t\tmap(filters => {\r\n\t\t\tconst url = this.buildUrl(filters);\r\n\t\t\treturn url;\r\n\t\t}),\r\n\r\n\t\t// 3) Solo seguimos si la URL cambió respecto de la emisión anterior\r\n\t\tdistinctUntilChanged(),\r\n\r\n\t\t// 4) Llamamos al backend con la URL construida\r\n\t\ttap(url => {\r\n\t\t}),\r\n\t\tswitchMap(url => this.getData(url)),\r\n\r\n\t\t// 5) Reutilizar resultado si alguien más se suscribe (evita repetir la petición)\r\n\t\tshareReplay(1)\r\n\t);\r\n\r\n\tprivate _dataPagination: any = signal<Pagination>({\r\n\t\tattributes: [],\r\n\t\tcategory: [],\r\n\t\toptions: [],\r\n\t\tlimit: 0,\r\n\t\tpage: 0,\r\n\t\tpages: 0,\r\n\t\ttotal: 0,\r\n\t\titems: [],\r\n\t\tcalled: false,\r\n\t\tprice_max: 0,\r\n\t\tprice_min: 0\r\n\t});\r\n\r\n\tprivate _waiting: boolean = false;\r\n\tprivate _finished: boolean = false;\r\n\tprivate _apiUrl: string = '';\r\n\tprivate _start: boolean = false;\r\n\r\n\tprivate _nextProductsSubject: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([]);\r\n\tpublic nextProducts$ = this._nextProductsSubject.asObservable();\r\n\r\n\tprivate _resetSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\r\n\tpublic reset$ = this._resetSubject.asObservable();\r\n\r\n\tprivate priceRangeSubject = new BehaviorSubject<{ price_min: number, price_max: number }>({ price_min: 0, price_max: 0 });\r\n\tpublic priceRange$ = this.priceRangeSubject.asObservable();\r\n\r\n\tconstructor() { }\r\n\r\n\t/**\r\n\t * Inicializa la configuración para el Servicio de Paginación.\r\n\t * @param apiUrl Url con los filtros aplicados. \r\n\t * @param limit limite de productos en cada petición.\r\n\t */\r\n\tinitialize(apiUrl: string, limit?: number) {\r\n\t\t//console.log(apiUrl)\r\n\t\tthis._apiUrl = apiUrl;\r\n\t\tthis._dataPagination().limit = limit || this._dataPagination().limit || 10;\r\n\t\tthis._finished = false;\r\n\t\tthis._start = true;\r\n\t\tthis._nextProductsSubject.next([]);\r\n\t}\r\n\t/**\r\n\t * Obtengo los datos de la siguiente página.\r\n\t * @param next \r\n\t * @returns \r\n\t */\r\n\tgetNext(next?: any): boolean {\r\n\t\tthis._waiting = true;\r\n\t\tthis._nextProductsSubject.next([]);\r\n\t\tthis._apiUrl = this._apiUrl.replace('%20', ' ');// mas para cuando es un search value.\r\n\t\tlet nextProducts;\r\n\t\tif (this._start) {\r\n\t\t\tnextProducts = this._connectionService.get(this._apiUrl, { limit: this._dataPagination().limit, page: 1 })\r\n\t\t\tthis._start = false\r\n\t\t} else {\r\n\t\t\tnextProducts = this._connectionService.get(this._dataPagination().links.next.slice(1));\r\n\t\t}\r\n\r\n\t\t!this._finished && nextProducts.subscribe(\r\n\t\t\t(response: Pagination) => {\r\n\t\t\t\tresponse.links ? this.updatePageData(response) : this.finish(response)\r\n\t\t\t\tthis._nextProductsSubject.next(response.items);\r\n\t\t\t}\r\n\t\t)\r\n\t\treturn true;\r\n\t}\r\n\t/**\r\n\t * Actualiza los datos de la pagina.\r\n\t * @param response respuesta del backend o objecto del tipo `Pagination`.\r\n\t */\r\n\tupdatePageData(response: Pagination): void {\r\n\t\tthis._dataPagination.set({ ...response, called: true })\r\n\t\tthis._finished = (response.page == response.pages)\r\n\t\tthis._waiting = false;\r\n\t\t// Emitir los valores de price_min y price_max a través de priceRangeSubject,\r\n\t\t// para que otros componentes (ej. filtro de precio) puedan mostrarlos.\r\n\t\tthis.priceRangeSubject.next({\r\n\t\t\tprice_min: response.price_min,\r\n\t\t\tprice_max: response.price_max\r\n\t\t});\r\n\r\n\t}\r\n\t/**\r\n\t * Marca el final de las paginas, es decir cuando ya no queda mas elementos que mostrar.\r\n\t * @param response \r\n\t */\r\n\tfinish(response: Pagination): void {\r\n\t\tthis._finished = true;\r\n\t\tthis._waiting = false;\r\n\t}\r\n\t/**\r\n\t * Obtengo los productos de la pagina cargada actual. \r\n\t * @returns \r\n\t */\r\n\tpublic getNextProducts(): Product[] {\r\n\t\treturn this._nextProductsSubject.value;\r\n\t}\r\n\t/**\r\n\t * Devuelve si esta esperando o no la petición al backend.\r\n\t * @returns \r\n\t */\r\n\tpublic getWaiting(): boolean { return this._waiting; }\r\n\t/**\r\n\t * Setea la variable waiting.\r\n\t * @param value \r\n\t */\r\n\tpublic setWaiting(value: boolean): void { this._waiting = value }\r\n\t/**\r\n\t * Getter para la variable `_finished`.\r\n\t * @returns \r\n\t */\r\n\tpublic getFinished(): boolean { return this._finished; }\r\n\t/**\r\n\t * Devuelve el total de Items.\r\n\t * @returns \r\n\t */\r\n\tpublic getTotalItems(): number {\r\n\t\treturn this._dataPagination().limit * this._dataPagination().page;\r\n\t}\r\n\t/**\r\n\t * Obtengo un objecto con los datos de los filtros resultantes.\r\n\t * @returns \r\n\t */\r\n\tpublic getFiltersPagination(): any {\r\n\t\treturn {\r\n\t\t\t\"categories\": this._dataPagination().category,\r\n\t\t\t\"attributes\": this._dataPagination().attributes,\r\n\t\t\t\"options\": this._dataPagination().options\r\n\t\t}\r\n\t}\r\n\t/**\r\n\t * Para contruiste la url en base a los filtros aplicados.\r\n\t * @param filters \r\n\t * @returns \r\n\t */\r\n\tpublic buildUrl(filters: Filter[]) {\r\n\t\tconst url = this._constants.searchValue\r\n\t\t\t? this._filtersService.generateFinalApi(this._constants.searchValue)\r\n\t\t\t: this._filtersService.generateFinalApi();\r\n\r\n\t\treturn url;\r\n\t}\r\n\r\n\t/**\r\n\t * Devuelve un observable de los productos que devolvió la última página obtenida.\r\n\t * @param url \r\n\t * @returns \r\n\t */\r\n\tpublic getData(url: string): Observable<Product[]> {\r\n\r\n\t\t// Detectar la página desde la URL (?page=n). Si no viene, asumimos page=1.\r\n\t\tconst pageMatch = url.match(/[?&]page=(\\d+)/);\r\n\t\tconst page = pageMatch ? Number(pageMatch[1]) : 1;\r\n\r\n\t\t// Si es la primera página, reseteamos estado interno\r\n\t\tif (page === 1) {\r\n\t\t\tthis._resetSubject.next(true);\r\n\t\t\tthis._nextProductsSubject.next([]);\r\n\t\t}\r\n\r\n\t\t// Llamamos al backend con el page correcto\r\n\t\treturn this._connectionService.get(url, { limit: 10, page }).pipe(\r\n\t\t\ttap((res: Pagination) => {\r\n\t\t\t\t// Actualizar datos de paginación + price_min/price_max\r\n\t\t\t\tres.links ? this.updatePageData(res) : this.finish(res);\r\n\t\t\t}),\r\n\t\t\tmap((res: Pagination) => {\r\n\t\t\t\tconst items = res.items ?? [];\r\n\r\n\t\t\t\t// Lo que ve paginationData$ (switchMap) son estos items\r\n\t\t\t\treturn items;\r\n\t\t\t})\r\n\t\t);\r\n\t}\r\n}"]}
238
+ //# 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';\r\nimport { ConnectionService } from '../api';\r\nimport { BehaviorSubject, distinctUntilChanged, filter, map, Observable, shareReplay, switchMap, tap } from 'rxjs';\r\nimport { Pagination, Product } from '../interfaces';\r\nimport { FiltersService } from './filters.service';\r\nimport { Filter } from '../classes';\r\nimport { CoreConstantsService } from '../constants';\r\n/**\r\n * Servicio para manejar la paginación y la carga de productos.\r\n *\r\n * Se encarga de:\r\n * - Escuchar los cambios en los filtros (categorías, atributos, precio, búsqueda, etc.).\r\n * - Construir la URL al backend con esos filtros.\r\n * - Pedir los productos al endpoint de product-search.\r\n * - Exponer un observable con la última página de productos (`paginationData$`)\r\n *   y otros helpers (precio mínimo/máximo, siguiente página, etc.).\r\n */\r\n@Injectable({\r\n\tprovidedIn: 'root'\r\n})\r\nexport class PaginationService {\r\n\r\n\tprivate _connectionService: ConnectionService = inject(ConnectionService);\r\n\tprivate _filtersService: FiltersService = inject(FiltersService)\r\n\tprivate _constants: CoreConstantsService = inject(CoreConstantsService);\r\n\r\n\t/**\r\n\t * Flujo principal: a partir de los filtros → arma URL → consulta backend → devuelve productos.\r\n\t *\r\n\t * Pasos:\r\n\t * 1) Espera a que `FiltersService` emita filtros válidos.\r\n\t * 2) Construye la URL final con `buildUrl()`.\r\n\t * 3) Evita repetir llamadas si la URL no cambió (`distinctUntilChanged`).\r\n\t * 4) Llama al backend con `getData(url)`.\r\n\t * 5) Comparte el último resultado con todos los suscriptores (`shareReplay(1)`).\r\n\t */\r\n\tpublic paginationData$ = this._filtersService.filters$.pipe(\r\n\t\ttap(filters => {\r\n\t\t}),\r\n\t\tfilter(filters => {\r\n\t\t\tif (!filters || !filters.length) return false;\r\n\r\n\t\t\tconst categoryFilter = filters.find(f => f.type() === 'categories');\r\n\t\t\tconst hasCategorySelected = !!categoryFilter?.getSelectedList()?.length;\r\n\r\n\t\t\tconst isCategoryRoute =\r\n\t\t\t\tthis._constants.currentRouteType === 'categories';\r\n\r\n\t\t\t// Si estamos en ruta de categorías y todavía no se marcó ninguna,\r\n\t\t\t// esperamos a que el CategoryFilter se hidrate antes de disparar la llamada.\r\n\t\t\tif (isCategoryRoute && !hasCategorySelected) {\r\n\t\t\t\treturn false;\r\n\t\t\t}\r\n\r\n\t\t\treturn true;\r\n\t\t}),\r\n\r\n\t\t// 2) Convertimos filtros -> URL\r\n\t\tmap(filters => {\r\n\t\t\tconst url = this.buildUrl(filters);\r\n\t\t\treturn url;\r\n\t\t}),\r\n\r\n\t\t// 3) Solo seguimos si la URL cambió respecto de la emisión anterior\r\n\t\tdistinctUntilChanged(),\r\n\r\n\t\t// 4) Llamamos al backend con la URL construida\r\n\t\ttap(url => {\r\n\t\t}),\r\n\t\tswitchMap(url => this.getData(url)),\r\n\r\n\t\t// 5) Reutilizar resultado si alguien más se suscribe (evita repetir la petición)\r\n\t\tshareReplay(1)\r\n\t);\r\n\r\n\tprivate _dataPagination: any = signal<Pagination>({\r\n\t\tattributes: [],\r\n\t\tcategory: [],\r\n\t\toptions: [],\r\n\t\tlimit: 0,\r\n\t\tpage: 0,\r\n\t\tpages: 0,\r\n\t\ttotal: 0,\r\n\t\titems: [],\r\n\t\tcalled: false,\r\n\t\tprice_max: 0,\r\n\t\tprice_min: 0\r\n\t});\r\n\r\n\tprivate _waiting: boolean = false;\r\n\tprivate _finished: boolean = false;\r\n\tprivate _apiUrl: string = '';\r\n\tprivate _start: boolean = false;\r\n\r\n\tprivate _nextProductsSubject: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([]);\r\n\tpublic nextProducts$ = this._nextProductsSubject.asObservable();\r\n\r\n\tprivate _resetSubject: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\r\n\tpublic reset$ = this._resetSubject.asObservable();\r\n\r\n\tprivate priceRangeSubject = new BehaviorSubject<{ price_min: number, price_max: number }>({ price_min: 0, price_max: 0 });\r\n\tpublic priceRange$ = this.priceRangeSubject.asObservable();\r\n\r\n\tconstructor() { }\r\n\r\n\t/**\r\n\t * Inicializa la configuración para el Servicio de Paginación.\r\n\t * @param apiUrl Url con los filtros aplicados. \r\n\t * @param limit limite de productos en cada petición.\r\n\t */\r\n\tinitialize(apiUrl: string, limit?: number) {\r\n\t\tthis._apiUrl = apiUrl;\r\n\t\tthis._dataPagination().limit = limit || this._dataPagination().limit || 10;\r\n\t\tthis._finished = false;\r\n\t\tthis._waiting = false; // Asegurar que no está bloqueado\r\n\t\tthis._start = true;\r\n\t\t// NO emitir array vacío aquí para evitar triggers innecesarios\r\n\t}\r\n\t/**\r\n\t * Obtengo los datos de la siguiente página.\r\n\t * @param next \r\n\t * @returns \r\n\t */\r\n\tgetNext(next?: any): boolean {\r\n\t\t// Si ya estamos esperando una respuesta o ya terminamos, no hacer nada\r\n\t\tif (this._waiting || this._finished) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\r\n\t\t// Verificar que hay un link next disponible\r\n\t\tif (!this._dataPagination().links?.next) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\r\n\t\tthis._waiting = true;\r\n\t\tthis._apiUrl = this._apiUrl.replace('%20', ' ');// mas para cuando es un search value.\r\n\t\t\r\n\t\t// Siempre usar el link \"next\" de la respuesta anterior\r\n\t\tconst nextProducts = this._connectionService.get(this._dataPagination().links.next.slice(1));\r\n\r\n\t\tnextProducts.subscribe(\r\n\t\t\t(response: Pagination) => {\r\n\t\t\t\tresponse.links ? this.updatePageData(response) : this.finish(response)\r\n\t\t\t\t// Solo emitir si hay productos reales\r\n\t\t\t\tif (response.items && response.items.length > 0) {\r\n\t\t\t\t\tthis._nextProductsSubject.next(response.items);\r\n\t\t\t\t} else {\r\n\t\t\t\t\t// Si no hay productos, igual liberamos el waiting\r\n\t\t\t\t\tthis._waiting = false;\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\t(error) => {\r\n\t\t\t\t// En caso de error, liberar el bloqueo\r\n\t\t\t\tthis._waiting = false;\r\n\t\t\t}\r\n\t\t)\r\n\t\treturn true;\r\n\t}\r\n\t/**\r\n\t * Actualiza los datos de la pagina.\r\n\t * @param response respuesta del backend o objecto del tipo `Pagination`.\r\n\t */\r\n\tupdatePageData(response: Pagination): void {\r\n\t\tthis._dataPagination.set({ ...response, called: true })\r\n\t\tthis._finished = (response.page == response.pages)\r\n\t\tthis._waiting = false;\r\n\t\t// Emitir los valores de price_min y price_max a través de priceRangeSubject,\r\n\t\t// para que otros componentes (ej. filtro de precio) puedan mostrarlos.\r\n\t\tthis.priceRangeSubject.next({\r\n\t\t\tprice_min: response.price_min,\r\n\t\t\tprice_max: response.price_max\r\n\t\t});\r\n\r\n\t}\r\n\t/**\r\n\t * Marca el final de las paginas, es decir cuando ya no queda mas elementos que mostrar.\r\n\t * @param response \r\n\t */\r\n\tfinish(response: Pagination): void {\r\n\t\tthis._finished = true;\r\n\t\tthis._waiting = false;\r\n\t}\r\n\t/**\r\n\t * Obtengo los productos de la pagina cargada actual. \r\n\t * @returns \r\n\t */\r\n\tpublic getNextProducts(): Product[] {\r\n\t\treturn this._nextProductsSubject.value;\r\n\t}\r\n\t/**\r\n\t * Devuelve si esta esperando o no la petición al backend.\r\n\t * @returns \r\n\t */\r\n\tpublic getWaiting(): boolean { return this._waiting; }\r\n\t/**\r\n\t * Setea la variable waiting.\r\n\t * @param value \r\n\t */\r\n\tpublic setWaiting(value: boolean): void { this._waiting = value }\r\n\t/**\r\n\t * Getter para la variable `_finished`.\r\n\t * @returns \r\n\t */\r\n\tpublic getFinished(): boolean { return this._finished; }\r\n\t/**\r\n\t * Devuelve el total de Items.\r\n\t * @returns \r\n\t */\r\n\tpublic getTotalItems(): number {\r\n\t\treturn this._dataPagination().limit * this._dataPagination().page;\r\n\t}\r\n\t/**\r\n\t * Obtengo un objecto con los datos de los filtros resultantes.\r\n\t * @returns \r\n\t */\r\n\tpublic getFiltersPagination(): any {\r\n\t\treturn {\r\n\t\t\t\"categories\": this._dataPagination().category,\r\n\t\t\t\"attributes\": this._dataPagination().attributes,\r\n\t\t\t\"options\": this._dataPagination().options\r\n\t\t}\r\n\t}\r\n\t/**\r\n\t * Para contruiste la url en base a los filtros aplicados.\r\n\t * @param filters \r\n\t * @returns \r\n\t */\r\n\tpublic buildUrl(filters: Filter[]) {\r\n\t\tconst url = this._constants.searchValue\r\n\t\t\t? this._filtersService.generateFinalApi(this._constants.searchValue)\r\n\t\t\t: this._filtersService.generateFinalApi();\r\n\r\n\t\treturn url;\r\n\t}\r\n\r\n\t/**\r\n\t * Devuelve un observable de los productos que devolvió la última página obtenida.\r\n\t * @param url \r\n\t * @returns \r\n\t */\r\n\tpublic getData(url: string): Observable<Product[]> {\r\n\r\n\t\t// Detectar la página desde la URL (?page=n). Si no viene, asumimos page=1.\r\n\t\tconst pageMatch = url.match(/[?&]page=(\\d+)/);\r\n\t\tconst page = pageMatch ? Number(pageMatch[1]) : 1;\r\n\r\n\t\t// Si es la primera página, reseteamos estado interno\r\n\t\tif (page === 1) {\r\n\t\t\tthis._resetSubject.next(true);\r\n\t\t\t// Resetear estado de paginación para permitir scroll\r\n\t\t\tthis._finished = false;\r\n\t\t\tthis._waiting = false;\r\n\t\t\tthis._start = false; // Ya no es el inicio después de getData\r\n\t\t}\r\n\r\n\t\t// Llamamos al backend con el page correcto\r\n\t\treturn this._connectionService.get(url, { limit: 10, page }).pipe(\r\n\t\t\ttap((res: Pagination) => {\r\n\t\t\t\t// Actualizar datos de paginación + price_min/price_max\r\n\t\t\t\tres.links ? this.updatePageData(res) : this.finish(res);\r\n\t\t\t}),\r\n\t\t\tmap((res: Pagination) => {\r\n\t\t\t\tconst items = res.items ?? [];\r\n\r\n\t\t\t\t// Lo que ve paginationData$ (switchMap) son estos items\r\n\t\t\t\treturn items;\r\n\t\t\t})\r\n\t\t);\r\n\t}\r\n}"]}
@@ -19,57 +19,65 @@ export class ProductsService {
19
19
  filtersSubject = new BehaviorSubject([]);
20
20
  filters$ = this.filtersSubject.asObservable();
21
21
  searchValue = signal('');
22
+ // Semáforo para evitar peticiones simultáneas (ej. pedir pág 3 mientras carga pág 2)
23
+ isLoadingNextPage = false;
22
24
  constructor() {
25
+ // 1. Resetear todo al cambiar filtros o limpiar
23
26
  this._paginationService.reset$.subscribe(res => {
27
+ this.isLoadingNextPage = false;
24
28
  this._productsSubject.next([]);
25
29
  });
30
+ // 2. Datos principales / Carga inicial (SOLO página 1)
31
+ // Este observable se dispara cuando cambian los filtros o se recarga la página
26
32
  this._paginationService.paginationData$.subscribe(products => {
27
- const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();
28
- if (productsWithUniqueVariant)
29
- products = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());
30
- const currentProducts = this._productsSubject.value;
31
- this._productsSubject.next([...currentProducts, ...products]);
33
+ // Para la carga inicial, REEMPLAZAMOS los productos (no agregamos)
34
+ if (products && products.length > 0) {
35
+ this._productsSubject.next(products);
36
+ }
37
+ // Liberamos el semáforo después de procesar
38
+ this.isLoadingNextPage = false;
32
39
  });
33
- // Suscribirse a nextProducts$ SOLO para páginas posteriores (scroll infinito)
34
- // Este observable se emite cuando se llama a getNext() desde updateProducts()
35
- let isFirstLoad = true;
40
+ // 3. Paginación (Scroll infinito - páginas 2, 3, 4...)
41
+ // Este observable se dispara SOLO cuando llamamos explícitamente a getNext()
36
42
  this._paginationService.nextProducts$.subscribe(products => {
37
- // Ignorar la primera emisión porque ya la maneja paginationData$
38
- if (isFirstLoad && products.length > 0) {
39
- isFirstLoad = false;
40
- return;
41
- }
42
- if (products.length > 0 && !isFirstLoad) {
43
- const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();
44
- let processedProducts = products;
45
- if (productsWithUniqueVariant)
46
- processedProducts = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());
47
- const currentProducts = this._productsSubject.value;
48
- this._productsSubject.next([...currentProducts, ...processedProducts]);
43
+ // SIEMPRE liberamos el semáforo al recibir respuesta, aunque sea vacía
44
+ this.isLoadingNextPage = false;
45
+ if (products && products.length > 0) {
46
+ this.appendProductsSafe(products);
49
47
  }
50
48
  });
51
- // Resetear flag cuando cambian los filtros
52
- this._paginationService.reset$.subscribe(() => {
53
- isFirstLoad = true;
54
- });
55
- // Suscribirse a los cambios en el rango de precios
49
+ // 4. Filtros de precio
56
50
  this._paginationService.priceRange$.subscribe(range => {
57
51
  let min = Math.floor(range.price_min);
58
52
  let max = Math.ceil(range.price_max);
59
53
  this.updatePriceRangeFilter(min, max);
60
54
  });
55
+ // 5. Gestión de filtros generales
61
56
  this._filtersService.filters$.subscribe(filters => {
62
57
  this.filtersSubject.next(filters);
63
58
  });
64
59
  }
65
- appendProducts(products) {
60
+ /**
61
+ * Método seguro para añadir productos.
62
+ * Verifica por ID que el producto no exista ya en la lista para evitar duplicados visuales
63
+ * y resuelve el problema de la "Página 2 perdida".
64
+ */
65
+ appendProductsSafe(products) {
66
66
  const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();
67
- let output = products;
67
+ let candidates = products;
68
+ // 1. Procesar variantes si es necesario
68
69
  if (productsWithUniqueVariant) {
69
- output = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());
70
+ candidates = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());
70
71
  }
71
72
  const currentProducts = this._productsSubject.value;
72
- this._productsSubject.next([...currentProducts, ...output]);
73
+ // 2. FILTRADO DE DUPLICADOS: La clave del arreglo
74
+ // Solo añadimos productos cuyo ID no esté ya en la lista actual.
75
+ // Esto permite que si 'nextProducts' emite la P1 de nuevo, la ignoremos,
76
+ // pero si emite la P2, la agreguemos.
77
+ const newProducts = candidates.filter(candidate => !currentProducts.some(existing => existing.id === candidate.id));
78
+ if (newProducts.length > 0) {
79
+ this._productsSubject.next([...currentProducts, ...newProducts]);
80
+ }
73
81
  }
74
82
  updatePriceRangeFilter(min, max) {
75
83
  const final_filters = this.filtersSubject.value;
@@ -88,8 +96,6 @@ export class ProductsService {
88
96
  relatedProductsApi = (key) => 'shop-api/' + this._apiConsts.CHANNEL + '/products/related/by-code/' + key + '?locale=' + this._apiConsts.LOCALE;
89
97
  /**
90
98
  * Setea los filtros para obtener los productos.
91
- * @param paginationSettings
92
- * @param searchValue
93
99
  */
94
100
  getProductsForFilter(paginationSettings, searchValue, routeQueryParams) {
95
101
  searchValue ? this.searchValue.set(searchValue) : this.searchValue.set('');
@@ -99,37 +105,19 @@ export class ProductsService {
99
105
  }
100
106
  /**
101
107
  * Actualiza los productos con los de la siguiente pagina.
108
+ * Implementa bloqueo para evitar saltos de página por scroll rápido.
102
109
  */
103
110
  updateProducts() {
104
- if (!this._paginationService.getWaiting()) {
111
+ // Solo pedimos si el servicio no está esperando Y nosotros no tenemos el semáforo en rojo
112
+ if (!this._paginationService.getWaiting() && !this.isLoadingNextPage) {
113
+ this.isLoadingNextPage = true;
105
114
  this._paginationService.getNext();
106
115
  }
107
116
  }
108
- /**
109
- * Devuelve el producto por codigo.
110
- * @param code Codigo del producto
111
- * @param variant Para indicar si es variante o no.
112
- * @returns
113
- */
114
117
  getProductByCode(code, variant) {
115
118
  return this._connection.get(this.productByCodeApi(code, variant));
116
119
  }
117
- /**
118
- * @description Guarda la reseña de un producto.
119
- * @param code
120
- * @param form_data
121
- * @returns
122
- */
123
120
  saveReviews = (code, form_data) => this._connection.post(this.saveProductReviewsApi(code), form_data);
124
- /**
125
- * Recibe una lista de productos y retorna los mismos productos tantas veces como variantes tenga.
126
- * cada producto retornado tendrá una única variante.
127
- * @param products Array. Lista de productos.
128
- * @param withOption 'type' tipo de opción, code: código de la opción. 'defaultFirstOption': en caso de que no se pueda desdoblar por la opción indicada si el valor es verdadero se intentara desdoblar por la primera opción asociada al producto.
129
- * 'checkIfStock': en caso de que esta valor sea verdadero se realiza un control de que la variantes cuenten con stock para ser un producto desdoblado por la variante, si el valor es falso no se realiza dicho control y el producto se desdoblara con stock en 0.
130
- * @returns arreglo con los productos desdoblados por la opción solicitada.
131
- * (opción por defecto que se intenta hacer el desdoblamiento 'color', o bien si en le parámetro withOption el atributo 'defaultFirstOption' es verdadero se desdobla por la primera opción asociada al producto)
132
- */
133
121
  getProductsWithUniqueVariant = (products, withOption = { type: 'color', code: 'color', defaultFirstOption: false, checkIfStock: true }) => {
134
122
  !withOption['code'] && withOption['type'] && (withOption['code'] = withOption['type']) || !withOption['code'] && (withOption['code'] = 'color');
135
123
  !withOption['type'] && withOption['code'] && (withOption['type'] = withOption['code']) || !withOption['type'] && (withOption['type'] = 'color');
@@ -143,20 +131,14 @@ export class ProductsService {
143
131
  option.values.forEach((optionValue) => {
144
132
  let modified_product = JSON.parse(JSON.stringify(product));
145
133
  modified_product.options.find((option) => option.type.toLowerCase() == withOption.type?.toLowerCase()).values = [{ code: optionValue.code, name: optionValue.name }];
146
- //busco una variante con ese code para asignarle la lista de imagenes
147
134
  let variantAux = variants.find(variant => ((withOption.checkIfStock && variant.stock > 0) || withOption.checkIfStock == false)
148
135
  && variant.options.length
149
136
  && variant.options.some((op) => (op.hasOwnProperty(withOption.code?.toLowerCase()) || op.hasOwnProperty(withOption.code?.toUpperCase()))
150
137
  && (op[`${withOption.code?.toLowerCase()}`] || op[`${withOption.code?.toUpperCase()}`]) === optionValue.code));
151
- //console.log("---",variantAux);
152
138
  variantAux && productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue));
153
139
  });
154
140
  }
155
141
  else {
156
- /**
157
- * se modifica el caso para cuando se toma el primer `option` del producto para
158
- * armar la lista de productos con variantes.
159
- */
160
142
  if (withOption.defaultFirstOption && product.options?.length) {
161
143
  let firstOptions = product.options[0];
162
144
  product.options[0].values.forEach((optionValue) => {
@@ -185,8 +167,8 @@ export class ProductsService {
185
167
  };
186
168
  productWithVariantValues = (product, variant, optionValue) => {
187
169
  product.picturesdefault = [...variant.images];
188
- product.variant_id = variant.code; //lo dejo aca hasta asegurarnos de que no se estaba usando en otra parte.
189
- product.stock = variant.stock || 0; //lo dejo aca hasta asegurarnos de que no se estaba usando en otra parte.
170
+ product.variant_id = variant.code;
171
+ product.stock = variant.stock || 0;
190
172
  product.price = variant.price.toString() || '0';
191
173
  product.currentOption = {
192
174
  ...optionValue,
@@ -204,15 +186,12 @@ export class ProductsService {
204
186
  return this._connection.get(this.relatedProductsApi(product_id));
205
187
  }
206
188
  hasStock(product) {
207
- // Si el producto tiene la propiedad 'stock' y es mayor a 0
208
189
  if (typeof product.stock === 'number') {
209
190
  return product.stock > 0;
210
191
  }
211
- // Si el producto tiene variantes, verifica si alguna tiene stock
212
192
  if (Array.isArray(product.variants)) {
213
193
  return product.variants.some(variant => variant.stock > 0);
214
194
  }
215
- // Si no tiene información de stock, asumimos que no hay stock
216
195
  return false;
217
196
  }
218
197
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProductsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
@@ -224,4 +203,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
224
203
  providedIn: 'root'
225
204
  }]
226
205
  }], ctorParameters: () => [] });
227
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"products.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/products.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { EnvironmentInjector, inject, Injectable, runInInjectionContext, signal } from '@angular/core';\r\nimport { BehaviorSubject, map, Observable, ReplaySubject, shareReplay, switchMap } from 'rxjs';\r\nimport { DefaultFilter, FilterElement, FilterType, IFilter, Options, PaginationSettings, ParamsProductsWithUniqueVariant, Product } from '../interfaces';\r\nimport { ConnectionService } from '../api';\r\nimport { ApiConstantsService, CoreConstantsService } from '../constants';\r\nimport { OptionsService } from './options.service';\r\nimport { PaginationService } from './pagination.service';\r\nimport { Filter, FilterFactory, } from '../classes';\r\nimport { PriceRangeFilter } from '../classes/filters/price_range-filter';\r\nimport { FiltersService } from './filters.service'\r\n\r\n@Injectable({\r\n    providedIn: 'root'\r\n})\r\nexport class ProductsService {\r\n\r\n    private _connection: ConnectionService = inject(ConnectionService);\r\n    private _apiConsts: ApiConstantsService = inject(ApiConstantsService);\r\n    private _optionsService: OptionsService = inject(OptionsService);\r\n    private _paginationService: PaginationService = inject(PaginationService);\r\n    private _consts: CoreConstantsService = inject(CoreConstantsService);\r\n\r\n    private _productsSubject: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([])\r\n    public products$: Observable<Product[]> = this._productsSubject.asObservable();\r\n    private _filtersService: FiltersService = inject(FiltersService)\r\n    private filtersSubject: BehaviorSubject<Filter[]> = new BehaviorSubject<Filter[]>([]);\r\n    public filters$ = this.filtersSubject.asObservable();\r\n\r\n    private searchValue = signal<string>('');\r\n\r\n    constructor() {\r\n        this._paginationService.reset$.subscribe(res => {\r\n            this._productsSubject.next([]);\r\n        });\r\n\r\n        this._paginationService.paginationData$.subscribe(\r\n            products => {\r\n                const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();\r\n                if (productsWithUniqueVariant)\r\n                    products = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());\r\n\r\n                const currentProducts = this._productsSubject.value;\r\n                this._productsSubject.next([...currentProducts, ...products]);\r\n            }\r\n        );\r\n        \r\n        // Suscribirse a nextProducts$ SOLO para páginas posteriores (scroll infinito)\r\n        // Este observable se emite cuando se llama a getNext() desde updateProducts()\r\n        let isFirstLoad = true;\r\n        this._paginationService.nextProducts$.subscribe(\r\n            products => {\r\n                // Ignorar la primera emisión porque ya la maneja paginationData$\r\n                if (isFirstLoad && products.length > 0) {\r\n                    isFirstLoad = false;\r\n                    return;\r\n                }\r\n                \r\n                if (products.length > 0 && !isFirstLoad) {\r\n                    const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();\r\n                    let processedProducts = products;\r\n                    if (productsWithUniqueVariant)\r\n                        processedProducts = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());\r\n\r\n                    const currentProducts = this._productsSubject.value;\r\n                    this._productsSubject.next([...currentProducts, ...processedProducts]);\r\n                }\r\n            }\r\n        );\r\n        \r\n        // Resetear flag cuando cambian los filtros\r\n        this._paginationService.reset$.subscribe(() => {\r\n            isFirstLoad = true;\r\n        });\r\n        \r\n        // Suscribirse a los cambios en el rango de precios\r\n        this._paginationService.priceRange$.subscribe(range => {\r\n            let min = Math.floor(range.price_min);\r\n            let max = Math.ceil(range.price_max);\r\n            this.updatePriceRangeFilter(min, max);\r\n        });\r\n        this._filtersService.filters$.subscribe(filters => {\r\n            this.filtersSubject.next(filters);\r\n        });\r\n    }\r\n\r\n    private appendProducts(products: Product[]) {\r\n        const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();\r\n        let output = products;\r\n        if (productsWithUniqueVariant) {\r\n            output = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());\r\n        }\r\n\r\n        const currentProducts = this._productsSubject.value;\r\n        this._productsSubject.next([...currentProducts, ...output]);\r\n    }\r\n\r\n    private updatePriceRangeFilter(min: number, max: number) {\r\n        const final_filters: Filter[] = this.filtersSubject.value;\r\n        const priceRangeFilter = final_filters.find(filter => filter instanceof PriceRangeFilter);\r\n\r\n        if (priceRangeFilter) {\r\n            (priceRangeFilter as PriceRangeFilter).updatePrices(min, max);\r\n            this.filtersSubject.next(final_filters);\r\n        }\r\n    }\r\n\r\n\r\n\r\n    //API URL\r\n    private _shopApiUrl = this._apiConsts.SHOP_API_URL;\r\n    productByCodeApi(code: string, variant: boolean) { return this._shopApiUrl + this._apiConsts.CHANNEL + '/products/by-code/' + (variant ? 'variant/' : '') + code + '?locale=' + this._apiConsts.LOCALE; }\r\n    // review\r\n    saveProductReviewsApi(code: string) { return this._shopApiUrl + this._apiConsts.CHANNEL + '/products/by-code/' + code + '/reviews'; }\r\n    //related products\r\n    relatedProductsApi = (key: string): string => 'shop-api/' + this._apiConsts.CHANNEL + '/products/related/by-code/' + key + '?locale=' + this._apiConsts.LOCALE;\r\n\r\n    /**\r\n     * Setea los filtros para obtener los productos.\r\n     * @param paginationSettings \r\n     * @param searchValue \r\n     */\r\n    getProductsForFilter(\r\n        paginationSettings?: PaginationSettings,\r\n        searchValue?: any,\r\n        routeQueryParams?: any\r\n    ): any {\r\n        searchValue ? this.searchValue.set(searchValue) : this.searchValue.set('');\r\n        if (paginationSettings) {\r\n            this._filtersService.setFilters(paginationSettings, searchValue, routeQueryParams);\r\n        }\r\n    }\r\n    /**\r\n     * Actualiza los productos con los de la siguiente pagina.\r\n     */\r\n    updateProducts() {\r\n        if (!this._paginationService.getWaiting()) {\r\n            this._paginationService.getNext()\r\n        }\r\n    }\r\n    /**\r\n     * Devuelve el producto por codigo.\r\n     * @param code Codigo del producto\r\n     * @param variant Para indicar si es variante o no.\r\n     * @returns \r\n     */\r\n    getProductByCode(code: string, variant: boolean): Observable<Product> {\r\n        return this._connection.get(this.productByCodeApi(code, variant));\r\n    }\r\n\r\n    /**\r\n    * @description Guarda la reseña de un producto.\r\n    * @param code\r\n    * @param form_data\r\n    * @returns\r\n    */\r\n    saveReviews = (code: string, form_data: any) => this._connection.post(this.saveProductReviewsApi(code), form_data)\r\n\r\n\r\n    /**\r\n     * Recibe una lista de productos y retorna los mismos productos tantas veces como variantes tenga.\r\n     * cada producto retornado tendrá una única variante.\r\n     * @param products  Array. Lista de productos.\r\n     * @param withOption 'type' tipo de opción, code: código de la opción. 'defaultFirstOption': en caso de que no se pueda desdoblar por la opción indicada si el valor es verdadero se intentara desdoblar por la primera opción asociada al producto.\r\n     * 'checkIfStock': en caso de que esta valor sea verdadero se realiza un control de que la variantes cuenten con stock para ser un producto desdoblado por la variante, si el valor es falso no se realiza dicho control y el producto se desdoblara con stock en 0.\r\n     * @returns arreglo con los productos desdoblados por la opción solicitada.\r\n     * (opción por defecto que se intenta hacer el desdoblamiento 'color', o bien si en le parámetro withOption el atributo 'defaultFirstOption' es verdadero se desdobla por la primera opción asociada al producto)\r\n     */\r\n    getProductsWithUniqueVariant = (products: Product[], withOption: ParamsProductsWithUniqueVariant = { type: 'color', code: 'color', defaultFirstOption: false, checkIfStock: true }) => {\r\n\r\n        !withOption['code'] && withOption['type'] && (withOption['code'] = withOption['type']) || !withOption['code'] && (withOption['code'] = 'color')\r\n        !withOption['type'] && withOption['code'] && (withOption['type'] = withOption['code']) || !withOption['type'] && (withOption['type'] = 'color')\r\n        !withOption.hasOwnProperty('defaultFirstOption') && (withOption['defaultFirstOption'] = false)\r\n        !withOption.hasOwnProperty('checkIfStock') && (withOption['checkIfStock'] = true)\r\n\r\n        let productsUniqueVariant: any[] = [];\r\n        products.forEach((product: any) => {\r\n            let variants = [...product?.variants];\r\n            let option = product.options.find((opt: any) => opt.type.toLowerCase() === withOption.type?.toLowerCase());\r\n            if (option) {\r\n                option.values.forEach((optionValue: any) => {\r\n                    let modified_product = JSON.parse(JSON.stringify(product));\r\n                    modified_product.options.find((option: any) => option.type.toLowerCase() == withOption.type?.toLowerCase()).values = [{ code: optionValue.code, name: optionValue.name }];\r\n                    //busco una variante con ese code para asignarle la lista de imagenes\r\n                    let variantAux = variants.find(\r\n                        variant => ((withOption.checkIfStock && variant.stock > 0) || withOption.checkIfStock == false)\r\n                            && variant.options.length\r\n                            && variant.options.some((op: any) => (op.hasOwnProperty(withOption.code?.toLowerCase()) || op.hasOwnProperty(withOption.code?.toUpperCase()))\r\n                                && (op[`${withOption.code?.toLowerCase()}`] || op[`${withOption.code?.toUpperCase()}`]) === optionValue.code)\r\n\r\n                    );\r\n                    //console.log(\"---\",variantAux);\r\n                    variantAux && productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue));\r\n                })\r\n            } else {\r\n                /**\r\n                 * se modifica el caso para cuando se toma el primer `option` del producto para\r\n                 * armar la lista de productos con variantes.\r\n                 */\r\n                if (withOption.defaultFirstOption && product.options?.length) {\r\n                    let firstOptions = product.options[0]\r\n                    product.options[0].values.forEach((optionValue: any) => {\r\n\r\n                        let modified_product = JSON.parse(JSON.stringify(product));\r\n                        modified_product.options.find((option: Options) => option.type == firstOptions.type).values = [{ code: optionValue.code, name: optionValue.name }];\r\n\r\n                        let variantAux = variants.find(variant => {\r\n                            return variant.options.find((elem: any) => elem[firstOptions.code] == optionValue.code)\r\n                        })\r\n\r\n                        if (variantAux && variantAux.images.length == 0) {\r\n                            variantAux.images = product.picturesdefault\r\n                        }\r\n\r\n                        if (withOption.checkIfStock) {\r\n                            (variantAux && variantAux.stock > 0) ? productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue)) : null;\r\n                        } else {\r\n                            variantAux && productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue))\r\n                        }\r\n\r\n                    })\r\n                } else {\r\n                    productsUniqueVariant.push(product)\r\n                }\r\n\r\n            }\r\n        })\r\n        return productsUniqueVariant.length > 0 ? productsUniqueVariant : products;\r\n    }\r\n\r\n    productWithVariantValues = (product: Product, variant: any, optionValue: any) => {\r\n        product.picturesdefault = [...variant.images]\r\n        product.variant_id = variant.code; //lo dejo aca hasta asegurarnos de que no se estaba usando en otra parte.\r\n        product.stock = variant.stock || 0; //lo dejo aca hasta asegurarnos de que no se estaba usando en otra parte.\r\n        product.price = variant.price.toString() || '0';\r\n        product.currentOption = {\r\n            ...optionValue,\r\n            productCode: product.id,\r\n            variantCode: variant.code,\r\n            stock: variant.stock || 0,\r\n            finalPrice: variant.saleprice || variant.price || 0,\r\n        }\r\n        return product\r\n    }\r\n\r\n    private getObjectWithVariant = (): ParamsProductsWithUniqueVariant => {\r\n        return this._consts.getParamsProductsWithUniqueVariant()\r\n    }\r\n\r\n    getRelatedProducts(product_id: any) {\r\n        return this._connection.get(this.relatedProductsApi(product_id));\r\n    }\r\n\r\n    hasStock(product: Product): boolean {\r\n        // Si el producto tiene la propiedad 'stock' y es mayor a 0\r\n        if (typeof product.stock === 'number') {\r\n            return product.stock > 0;\r\n        }\r\n        // Si el producto tiene variantes, verifica si alguna tiene stock\r\n        if (Array.isArray(product.variants)) {\r\n            return product.variants.some(variant => variant.stock > 0);\r\n        }\r\n        // Si no tiene información de stock, asumimos que no hay stock\r\n        return false;\r\n    }\r\n}\r\n"]}
206
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"products.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/products.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { EnvironmentInjector, inject, Injectable, runInInjectionContext, signal } from '@angular/core';\r\nimport { BehaviorSubject, map, Observable, ReplaySubject, shareReplay, switchMap } from 'rxjs';\r\nimport { DefaultFilter, FilterElement, FilterType, IFilter, Options, PaginationSettings, ParamsProductsWithUniqueVariant, Product } from '../interfaces';\r\nimport { ConnectionService } from '../api';\r\nimport { ApiConstantsService, CoreConstantsService } from '../constants';\r\nimport { OptionsService } from './options.service';\r\nimport { PaginationService } from './pagination.service';\r\nimport { Filter, FilterFactory, } from '../classes';\r\nimport { PriceRangeFilter } from '../classes/filters/price_range-filter';\r\nimport { FiltersService } from './filters.service'\r\n\r\n@Injectable({\r\n    providedIn: 'root'\r\n})\r\nexport class ProductsService {\r\n\r\n    private _connection: ConnectionService = inject(ConnectionService);\r\n    private _apiConsts: ApiConstantsService = inject(ApiConstantsService);\r\n    private _optionsService: OptionsService = inject(OptionsService);\r\n    private _paginationService: PaginationService = inject(PaginationService);\r\n    private _consts: CoreConstantsService = inject(CoreConstantsService);\r\n\r\n    private _productsSubject: BehaviorSubject<Product[]> = new BehaviorSubject<Product[]>([])\r\n    public products$: Observable<Product[]> = this._productsSubject.asObservable();\r\n    private _filtersService: FiltersService = inject(FiltersService)\r\n    private filtersSubject: BehaviorSubject<Filter[]> = new BehaviorSubject<Filter[]>([]);\r\n    public filters$ = this.filtersSubject.asObservable();\r\n\r\n    private searchValue = signal<string>('');\r\n\r\n    // Semáforo para evitar peticiones simultáneas (ej. pedir pág 3 mientras carga pág 2)\r\n    private isLoadingNextPage = false;\r\n\r\n    constructor() {\r\n        // 1. Resetear todo al cambiar filtros o limpiar\r\n        this._paginationService.reset$.subscribe(res => {\r\n            this.isLoadingNextPage = false;\r\n            this._productsSubject.next([]);\r\n        });\r\n\r\n        // 2. Datos principales / Carga inicial (SOLO página 1)\r\n        // Este observable se dispara cuando cambian los filtros o se recarga la página\r\n        this._paginationService.paginationData$.subscribe(\r\n            products => {\r\n                // Para la carga inicial, REEMPLAZAMOS los productos (no agregamos)\r\n                if (products && products.length > 0) {\r\n                    this._productsSubject.next(products);\r\n                }\r\n                // Liberamos el semáforo después de procesar\r\n                this.isLoadingNextPage = false;\r\n            }\r\n        );\r\n\r\n        // 3. Paginación (Scroll infinito - páginas 2, 3, 4...)\r\n        // Este observable se dispara SOLO cuando llamamos explícitamente a getNext()\r\n        this._paginationService.nextProducts$.subscribe(\r\n            products => {\r\n                // SIEMPRE liberamos el semáforo al recibir respuesta, aunque sea vacía\r\n                this.isLoadingNextPage = false;\r\n\r\n                if (products && products.length > 0) {\r\n                    this.appendProductsSafe(products);\r\n                }\r\n            }\r\n        );\r\n\r\n        // 4. Filtros de precio\r\n        this._paginationService.priceRange$.subscribe(range => {\r\n            let min = Math.floor(range.price_min);\r\n            let max = Math.ceil(range.price_max);\r\n            this.updatePriceRangeFilter(min, max);\r\n        });\r\n        \r\n        // 5. Gestión de filtros generales\r\n        this._filtersService.filters$.subscribe(filters => {\r\n            this.filtersSubject.next(filters);\r\n        });\r\n    }\r\n\r\n    /**\r\n     * Método seguro para añadir productos.\r\n     * Verifica por ID que el producto no exista ya en la lista para evitar duplicados visuales\r\n     * y resuelve el problema de la \"Página 2 perdida\".\r\n     */\r\n    private appendProductsSafe(products: Product[]) {\r\n        const productsWithUniqueVariant = this._consts.getProductWithUniqueVariant();\r\n        let candidates = products;\r\n\r\n        // 1. Procesar variantes si es necesario\r\n        if (productsWithUniqueVariant) {\r\n            candidates = this.getProductsWithUniqueVariant(products, this.getObjectWithVariant());\r\n        }\r\n\r\n        const currentProducts = this._productsSubject.value;\r\n\r\n        // 2. FILTRADO DE DUPLICADOS: La clave del arreglo\r\n        // Solo añadimos productos cuyo ID no esté ya en la lista actual.\r\n        // Esto permite que si 'nextProducts' emite la P1 de nuevo, la ignoremos,\r\n        // pero si emite la P2, la agreguemos.\r\n        const newProducts = candidates.filter(candidate => \r\n            !currentProducts.some(existing => existing.id === candidate.id)\r\n        );\r\n\r\n        if (newProducts.length > 0) {\r\n            this._productsSubject.next([...currentProducts, ...newProducts]);\r\n        }\r\n    }\r\n\r\n    private updatePriceRangeFilter(min: number, max: number) {\r\n        const final_filters: Filter[] = this.filtersSubject.value;\r\n        const priceRangeFilter = final_filters.find(filter => filter instanceof PriceRangeFilter);\r\n\r\n        if (priceRangeFilter) {\r\n            (priceRangeFilter as PriceRangeFilter).updatePrices(min, max);\r\n            this.filtersSubject.next(final_filters);\r\n        }\r\n    }\r\n\r\n    //API URL\r\n    private _shopApiUrl = this._apiConsts.SHOP_API_URL;\r\n    productByCodeApi(code: string, variant: boolean) { return this._shopApiUrl + this._apiConsts.CHANNEL + '/products/by-code/' + (variant ? 'variant/' : '') + code + '?locale=' + this._apiConsts.LOCALE; }\r\n    // review\r\n    saveProductReviewsApi(code: string) { return this._shopApiUrl + this._apiConsts.CHANNEL + '/products/by-code/' + code + '/reviews'; }\r\n    //related products\r\n    relatedProductsApi = (key: string): string => 'shop-api/' + this._apiConsts.CHANNEL + '/products/related/by-code/' + key + '?locale=' + this._apiConsts.LOCALE;\r\n\r\n    /**\r\n     * Setea los filtros para obtener los productos.\r\n     */\r\n    getProductsForFilter(\r\n        paginationSettings?: PaginationSettings,\r\n        searchValue?: any,\r\n        routeQueryParams?: any\r\n    ): any {\r\n        searchValue ? this.searchValue.set(searchValue) : this.searchValue.set('');\r\n        if (paginationSettings) {\r\n            this._filtersService.setFilters(paginationSettings, searchValue, routeQueryParams);\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Actualiza los productos con los de la siguiente pagina.\r\n     * Implementa bloqueo para evitar saltos de página por scroll rápido.\r\n     */\r\n    updateProducts() {\r\n        // Solo pedimos si el servicio no está esperando Y nosotros no tenemos el semáforo en rojo\r\n        if (!this._paginationService.getWaiting() && !this.isLoadingNextPage) {\r\n            this.isLoadingNextPage = true;\r\n            this._paginationService.getNext();\r\n        }\r\n    }\r\n\r\n    getProductByCode(code: string, variant: boolean): Observable<Product> {\r\n        return this._connection.get(this.productByCodeApi(code, variant));\r\n    }\r\n\r\n    saveReviews = (code: string, form_data: any) => this._connection.post(this.saveProductReviewsApi(code), form_data)\r\n\r\n    getProductsWithUniqueVariant = (products: Product[], withOption: ParamsProductsWithUniqueVariant = { type: 'color', code: 'color', defaultFirstOption: false, checkIfStock: true }) => {\r\n\r\n        !withOption['code'] && withOption['type'] && (withOption['code'] = withOption['type']) || !withOption['code'] && (withOption['code'] = 'color')\r\n        !withOption['type'] && withOption['code'] && (withOption['type'] = withOption['code']) || !withOption['type'] && (withOption['type'] = 'color')\r\n        !withOption.hasOwnProperty('defaultFirstOption') && (withOption['defaultFirstOption'] = false)\r\n        !withOption.hasOwnProperty('checkIfStock') && (withOption['checkIfStock'] = true)\r\n\r\n        let productsUniqueVariant: any[] = [];\r\n        products.forEach((product: any) => {\r\n            let variants = [...product?.variants];\r\n            let option = product.options.find((opt: any) => opt.type.toLowerCase() === withOption.type?.toLowerCase());\r\n            if (option) {\r\n                option.values.forEach((optionValue: any) => {\r\n                    let modified_product = JSON.parse(JSON.stringify(product));\r\n                    modified_product.options.find((option: any) => option.type.toLowerCase() == withOption.type?.toLowerCase()).values = [{ code: optionValue.code, name: optionValue.name }];\r\n                    \r\n                    let variantAux = variants.find(\r\n                        variant => ((withOption.checkIfStock && variant.stock > 0) || withOption.checkIfStock == false)\r\n                            && variant.options.length\r\n                            && variant.options.some((op: any) => (op.hasOwnProperty(withOption.code?.toLowerCase()) || op.hasOwnProperty(withOption.code?.toUpperCase()))\r\n                                && (op[`${withOption.code?.toLowerCase()}`] || op[`${withOption.code?.toUpperCase()}`]) === optionValue.code)\r\n\r\n                    );\r\n                    variantAux && productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue));\r\n                })\r\n            } else {\r\n                if (withOption.defaultFirstOption && product.options?.length) {\r\n                    let firstOptions = product.options[0]\r\n                    product.options[0].values.forEach((optionValue: any) => {\r\n\r\n                        let modified_product = JSON.parse(JSON.stringify(product));\r\n                        modified_product.options.find((option: Options) => option.type == firstOptions.type).values = [{ code: optionValue.code, name: optionValue.name }];\r\n\r\n                        let variantAux = variants.find(variant => {\r\n                            return variant.options.find((elem: any) => elem[firstOptions.code] == optionValue.code)\r\n                        })\r\n\r\n                        if (variantAux && variantAux.images.length == 0) {\r\n                            variantAux.images = product.picturesdefault\r\n                        }\r\n\r\n                        if (withOption.checkIfStock) {\r\n                            (variantAux && variantAux.stock > 0) ? productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue)) : null;\r\n                        } else {\r\n                            variantAux && productsUniqueVariant.push(this.productWithVariantValues(modified_product, variantAux, optionValue))\r\n                        }\r\n\r\n                    })\r\n                } else {\r\n                    productsUniqueVariant.push(product)\r\n                }\r\n\r\n            }\r\n        })\r\n        return productsUniqueVariant.length > 0 ? productsUniqueVariant : products;\r\n    }\r\n\r\n    productWithVariantValues = (product: Product, variant: any, optionValue: any) => {\r\n        product.picturesdefault = [...variant.images]\r\n        product.variant_id = variant.code; \r\n        product.stock = variant.stock || 0; \r\n        product.price = variant.price.toString() || '0';\r\n        product.currentOption = {\r\n            ...optionValue,\r\n            productCode: product.id,\r\n            variantCode: variant.code,\r\n            stock: variant.stock || 0,\r\n            finalPrice: variant.saleprice || variant.price || 0,\r\n        }\r\n        return product\r\n    }\r\n\r\n    private getObjectWithVariant = (): ParamsProductsWithUniqueVariant => {\r\n        return this._consts.getParamsProductsWithUniqueVariant()\r\n    }\r\n\r\n    getRelatedProducts(product_id: any) {\r\n        return this._connection.get(this.relatedProductsApi(product_id));\r\n    }\r\n\r\n    hasStock(product: Product): boolean {\r\n        if (typeof product.stock === 'number') {\r\n            return product.stock > 0;\r\n        }\r\n        if (Array.isArray(product.variants)) {\r\n            return product.variants.some(variant => variant.stock > 0);\r\n        }\r\n        return false;\r\n    }\r\n}"]}