ng-easycommerce-v18 0.3.20-beta.2 → 0.3.21-beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -16
- package/esm2022/lib/classes/filters/attributes-filter.mjs +4 -74
- package/esm2022/lib/classes/filters/category-filter.mjs +26 -105
- package/esm2022/lib/classes/filters/filter-factory.mjs +3 -7
- package/esm2022/lib/classes/filters/price_range-filter.mjs +3 -3
- package/esm2022/lib/constants/core.constants.service.mjs +1 -12
- package/esm2022/lib/ec-components/collection-ec/collection-ec.component.mjs +17 -41
- package/esm2022/lib/ec-components/filters-ec/filters-ec.component.mjs +9 -42
- package/esm2022/lib/ec-components/header-ec/header-ec.component.mjs +5 -13
- package/esm2022/lib/ec-components/price-range-filter/price-range-filter.component.mjs +2 -13
- package/esm2022/lib/ec-components/widgets-ec/redsys-catch-ec/redsys-catch-ec.component.mjs +17 -7
- package/esm2022/lib/ec-services/filters.service.mjs +18 -124
- package/esm2022/lib/ec-services/pagination.service.mjs +22 -70
- package/esm2022/lib/ec-services/products.service.mjs +3 -5
- package/fesm2022/ng-easycommerce-v18.mjs +113 -497
- package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
- package/lib/classes/filters/attributes-filter.d.ts +0 -24
- package/lib/classes/filters/category-filter.d.ts +3 -30
- package/lib/constants/core.constants.service.d.ts +0 -7
- package/lib/ec-components/collection-ec/collection-ec.component.d.ts +4 -5
- package/lib/ec-components/filters-ec/filters-ec.component.d.ts +0 -13
- package/lib/ec-components/price-range-filter/price-range-filter.component.d.ts +0 -2
- package/lib/ec-services/filters.service.d.ts +1 -18
- package/lib/ec-services/pagination.service.d.ts +5 -21
- package/lib/ec-services/products.service.d.ts +1 -1
- package/package.json +1 -1
|
@@ -3,13 +3,10 @@ import { PriceRangeFilter } from '../../classes/filters/price_range-filter';
|
|
|
3
3
|
import { CommonModule } from '@angular/common';
|
|
4
4
|
import { EcCurrencySymbolPipe } from '../../ec-pipe';
|
|
5
5
|
import { FiltersService } from '../../ec-services';
|
|
6
|
-
import { Router, ActivatedRoute } from '@angular/router';
|
|
7
6
|
import * as i0 from "@angular/core";
|
|
8
7
|
import * as i1 from "@angular/common";
|
|
9
8
|
export class PriceRangeFilterComponent {
|
|
10
9
|
_filtersService = inject(FiltersService);
|
|
11
|
-
router = inject(Router);
|
|
12
|
-
route = inject(ActivatedRoute);
|
|
13
10
|
priceGap = 1;
|
|
14
11
|
roundStep = 5;
|
|
15
12
|
filter;
|
|
@@ -46,15 +43,7 @@ export class PriceRangeFilterComponent {
|
|
|
46
43
|
return;
|
|
47
44
|
const min = filter.currentMinPrice ?? filter.minPrice;
|
|
48
45
|
const max = filter.currentMaxPrice ?? filter.maxPrice;
|
|
49
|
-
this.
|
|
50
|
-
relativeTo: this.route,
|
|
51
|
-
queryParams: {
|
|
52
|
-
price_min: min !== filter.minPrice ? min : null,
|
|
53
|
-
price_max: max !== filter.maxPrice ? max : null,
|
|
54
|
-
page: null,
|
|
55
|
-
},
|
|
56
|
-
queryParamsHandling: 'merge'
|
|
57
|
-
});
|
|
46
|
+
this._filtersService.updatePriceRangeFilter(min, max);
|
|
58
47
|
}
|
|
59
48
|
getMinValue(filter) {
|
|
60
49
|
return (filter instanceof PriceRangeFilter && filter.currentMinPrice != null)
|
|
@@ -151,4 +140,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
151
140
|
}], rangeChanged: [{
|
|
152
141
|
type: Output
|
|
153
142
|
}] } });
|
|
154
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"price-range-filter.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/price-range-filter/price-range-filter.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/price-range-filter/price-range-filter.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { PriceRangeFilter } from '../../classes/filters/price_range-filter';\nimport { CommonModule } from '@angular/common';\nimport { EcCurrencySymbolPipe } from '../../ec-pipe';\nimport { FiltersService } from '../../ec-services';\nimport { Router, ActivatedRoute } from '@angular/router';\n\n@Component({\n  selector: 'lib-price-range-filter',\n  standalone: true,\n  imports: [EcCurrencySymbolPipe, CommonModule],\n  templateUrl: './price-range-filter.component.html',\n  styleUrl: './price-range-filter.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PriceRangeFilterComponent {\n\n  private _filtersService: FiltersService = inject(FiltersService);\n  private router: Router = inject(Router);\n  private route: ActivatedRoute = inject(ActivatedRoute);\n\n  priceGap = 1;\n  roundStep = 5;\n\n  @Input() filter!: PriceRangeFilter;\n  @Output() rangeChanged = new EventEmitter<{ min: number; max: number }>();\n\n  getSliderStyle(filter: any): { [klass: string]: string } {\n    if (!(filter instanceof PriceRangeFilter)) return { left: '0%', right: '0%' };\n\n    const left = this.calculateLeftPercentage(filter);\n    const right = this.calculateRightPercentage(filter);\n    return { left: `${left}%`, right: `${right}%` };\n  }\n\n  // Métodos para actualizar la barra del filtro por precio\n  calculateLeftPercentage(filter: any): number {\n    if (!(filter instanceof PriceRangeFilter)) return 0;\n\n    const min = filter.minPrice;\n    const max = filter.maxPrice;\n    const currentMin = filter.currentMinPrice ?? min;\n\n    const clampedMin = Math.max(min, Math.min(currentMin, max));\n    return Math.max(0, Math.min(((clampedMin - min) / (max - min)) * 100, 100));\n  }\n\n  calculateRightPercentage(filter: any): number {\n    if (!(filter instanceof PriceRangeFilter)) return 0;\n\n    const min = filter.minPrice;\n    const max = filter.maxPrice;\n    const currentMax = filter.currentMaxPrice ?? max;\n\n    const clampedMax = Math.max(min, Math.min(currentMax, max));\n    return Math.max(0, Math.min(100 - ((clampedMax - min) / (max - min)) * 100, 100));\n  }\n\n  // Método para actualizar los valores de precio seleccionados\n  updatePriceRange(filter: any) {\n    if (!(filter instanceof PriceRangeFilter)) return;\n\n    const min = filter.currentMinPrice ?? filter.minPrice;\n    const max = filter.currentMaxPrice ?? filter.maxPrice;\n\n    this.router.navigate([], {\n      relativeTo: this.route,\n      queryParams: {\n        price_min: min !== filter.minPrice ? min : null,\n        price_max: max !== filter.maxPrice ? max : null,\n        page: null,\n      },\n      queryParamsHandling: 'merge'\n    });\n  }\n\n  getMinValue(filter: any): number {\n    return (filter instanceof PriceRangeFilter && filter.currentMinPrice != null)\n      ? filter.currentMinPrice\n      : filter.minPrice;\n  }\n\n  getMaxValue(filter: any): number {\n    return (filter instanceof PriceRangeFilter && filter.currentMaxPrice != null)\n      ? filter.currentMaxPrice\n      : filter.maxPrice;\n  }\n\n  /**\n  * Redondea al múltiplo de 'step' respetando el redondeo entero:\n  * - Mínimo: redondea hacia abajo.\n  * - Máximo: redondea hacia arriba.\n  */\n  onRangeChange(event: Event, type: 'min' | 'max', filter: any) {\n    const input = event.target as HTMLInputElement;\n    let value = parseFloat(input.value);\n    value = this.roundToNearest(value, this.roundStep, type, filter);\n    this.updateFilterPrices(type, value, filter, this.priceGap);\n  }\n\n  protected roundToNearest(value: number, step: number, type: 'min' | 'max', filter: any): number {\n    let rounded = Math.round(value / step) * step;\n    if (type === 'max' && rounded < filter.maxPrice) {\n      rounded = Math.ceil(value / step) * step;\n    }\n    if (type === 'min' && rounded > filter.minPrice) {\n      rounded = Math.floor(value / step) * step;\n    }\n    return rounded;\n  }\n\n  /**\n  * Actualiza los valores del filtro permitiendo que un input empuje al otro.\n  */\n  protected updateFilterPrices(type: 'min' | 'max', value: number, filter: any, priceGap: number) {\n    const globalMin = filter.minPrice ?? 0;\n    const globalMax = filter.maxPrice;\n\n    let currentMin = filter.currentMinPrice ?? globalMin;\n    let currentMax = filter.currentMaxPrice ?? globalMax;\n\n    if (type === 'min') {\n      let newMin = Math.min(Math.max(value, globalMin), globalMax);\n      if (newMin + priceGap > currentMax) {\n        let newMaxCandidate = newMin + priceGap;\n        if (newMaxCandidate <= globalMax) {\n          currentMax = newMaxCandidate;\n        } else {\n          newMin = globalMax - priceGap;\n        }\n      }\n      filter.currentMinPrice = newMin;\n      filter.currentMaxPrice = currentMax;\n    } else if (type === 'max') {\n      let newMax = Math.max(Math.min(value, globalMax), globalMin);\n      if (newMax - priceGap < currentMin) {\n        let newMinCandidate = newMax - priceGap;\n        if (newMinCandidate >= globalMin) {\n          currentMin = newMinCandidate;\n        } else {\n          newMax = globalMin + priceGap;\n        }\n      }\n      filter.currentMaxPrice = newMax;\n      filter.currentMinPrice = currentMin;\n    }\n  }\n\n  scrollUp = () => {\n    window.scroll(0, 0);\n    return true\n  }\n\n  getFormattedMinPrice(filter: any): string {\n    const value = (filter instanceof PriceRangeFilter && filter.currentMinPrice != null)\n      ? filter.currentMinPrice\n      : filter.minPrice;\n    return value;\n  }\n\n  getFormattedMaxPrice(filter: any): string {\n    const value = (filter instanceof PriceRangeFilter && filter.currentMaxPrice != null)\n      ? filter.currentMaxPrice\n      : filter.maxPrice;\n    return value;\n  }\n}\n","<div class=\"slider\">\n    <div class=\"slider\">\n        <div class=\"progress\" [ngStyle]=\"getSliderStyle(filter)\">\n        </div>\n    </div>\n</div>\n<div class=\"range-input d-flex gap-2\">\n    <input #rangeMin type=\"range\" class=\"range-min\" [min]=\"filter.minPrice\" [max]=\"filter.maxPrice\"\n        [value]=\"getMinValue(filter)\" (input)=\"onRangeChange($event, 'min', filter)\"\n        (change)=\"updatePriceRange(filter); scrollUp()\">\n    <input #rangeMax type=\"range\" class=\"range-max\" [min]=\"filter.minPrice\" [max]=\"filter.maxPrice\"\n        [value]=\"getMaxValue(filter)\" (input)=\"onRangeChange($event, 'max', filter)\"\n        (change)=\"updatePriceRange(filter); scrollUp()\">\n</div>\n<!-- precios Mínimo / Máximo -->\n<div class=\"price-container mt-2\">\n    <div class=\"price-label-wrapper\">\n        <span class=\"price-label\">Mínimo:</span>\n        <span class=\"price-value\">\n            {{ (getFormattedMinPrice(filter) | ecCurrencySymbol) || '0' }}\n        </span>\n    </div>\n    <div class=\"price-label-wrapper\">\n        <span class=\"price-label\">Máximo:</span>\n        <span class=\"price-value price-value-right\">\n            {{ getFormattedMaxPrice(filter) | ecCurrencySymbol }}\n        </span>\n    </div>\n</div>"]}
|
|
143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"price-range-filter.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/price-range-filter/price-range-filter.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/price-range-filter/price-range-filter.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACxG,OAAO,EAAE,gBAAgB,EAAE,MAAM,0CAA0C,CAAC;AAC5E,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,oBAAoB,EAAE,MAAM,eAAe,CAAC;AACrD,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;;;AAUnD,MAAM,OAAO,yBAAyB;IAE5B,eAAe,GAAmB,MAAM,CAAC,cAAc,CAAC,CAAC;IAEjE,QAAQ,GAAG,CAAC,CAAC;IACb,SAAS,GAAG,CAAC,CAAC;IAEL,MAAM,CAAoB;IACzB,YAAY,GAAG,IAAI,YAAY,EAAgC,CAAC;IAE1E,cAAc,CAAC,MAAW;QACxB,IAAI,CAAC,CAAC,MAAM,YAAY,gBAAgB,CAAC;YAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;QAE9E,MAAM,IAAI,GAAG,IAAI,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;QAClD,MAAM,KAAK,GAAG,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC;QACpD,OAAO,EAAE,IAAI,EAAE,GAAG,IAAI,GAAG,EAAE,KAAK,EAAE,GAAG,KAAK,GAAG,EAAE,CAAC;IAClD,CAAC;IAED,yDAAyD;IACzD,uBAAuB,CAAC,MAAW;QACjC,IAAI,CAAC,CAAC,MAAM,YAAY,gBAAgB,CAAC;YAAE,OAAO,CAAC,CAAC;QAEpD,MAAM,GAAG,GAAG,MAAM,CAAC,QAAQ,CAAC;QAC5B,MAAM,GAAG,GAAG,MAAM,CAAC,QAAQ,CAAC;QAC5B,MAAM,UAAU,GAAG,MAAM,CAAC,eAAe,IAAI,GAAG,CAAC;QAEjD,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,CAAC,CAAC;QAC5D,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,UAAU,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,GAAG,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC;IAC9E,CAAC;IAED,wBAAwB,CAAC,MAAW;QAClC,IAAI,CAAC,CAAC,MAAM,YAAY,gBAAgB,CAAC;YAAE,OAAO,CAAC,CAAC;QAEpD,MAAM,GAAG,GAAG,MAAM,CAAC,QAAQ,CAAC;QAC5B,MAAM,GAAG,GAAG,MAAM,CAAC,QAAQ,CAAC;QAC5B,MAAM,UAAU,GAAG,MAAM,CAAC,eAAe,IAAI,GAAG,CAAC;QAEjD,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,GAAG,CAAC,CAAC,CAAC;QAC5D,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC,CAAC,UAAU,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,GAAG,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC;IACpF,CAAC;IAED,6DAA6D;IAC7D,gBAAgB,CAAC,MAAW;QAC1B,IAAI,CAAC,CAAC,MAAM,YAAY,gBAAgB,CAAC;YAAE,OAAO;QAElD,MAAM,GAAG,GAAG,MAAM,CAAC,eAAe,IAAI,MAAM,CAAC,QAAQ,CAAC;QACtD,MAAM,GAAG,GAAG,MAAM,CAAC,eAAe,IAAI,MAAM,CAAC,QAAQ,CAAC;QACtD,IAAI,CAAC,eAAe,CAAC,sBAAsB,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IACxD,CAAC;IAED,WAAW,CAAC,MAAW;QACrB,OAAO,CAAC,MAAM,YAAY,gBAAgB,IAAI,MAAM,CAAC,eAAe,IAAI,IAAI,CAAC;YAC3E,CAAC,CAAC,MAAM,CAAC,eAAe;YACxB,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC;IACtB,CAAC;IAED,WAAW,CAAC,MAAW;QACrB,OAAO,CAAC,MAAM,YAAY,gBAAgB,IAAI,MAAM,CAAC,eAAe,IAAI,IAAI,CAAC;YAC3E,CAAC,CAAC,MAAM,CAAC,eAAe;YACxB,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC;IACtB,CAAC;IAED;;;;MAIE;IACF,aAAa,CAAC,KAAY,EAAE,IAAmB,EAAE,MAAW;QAC1D,MAAM,KAAK,GAAG,KAAK,CAAC,MAA0B,CAAC;QAC/C,IAAI,KAAK,GAAG,UAAU,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACpC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,MAAM,CAAC,CAAC;QACjE,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC9D,CAAC;IAES,cAAc,CAAC,KAAa,EAAE,IAAY,EAAE,IAAmB,EAAE,MAAW;QACpF,IAAI,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC;QAC9C,IAAI,IAAI,KAAK,KAAK,IAAI,OAAO,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAC;YAChD,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC;QAC3C,CAAC;QACD,IAAI,IAAI,KAAK,KAAK,IAAI,OAAO,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAC;YAChD,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC;QAC5C,CAAC;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;MAEE;IACQ,kBAAkB,CAAC,IAAmB,EAAE,KAAa,EAAE,MAAW,EAAE,QAAgB;QAC5F,MAAM,SAAS,GAAG,MAAM,CAAC,QAAQ,IAAI,CAAC,CAAC;QACvC,MAAM,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAC;QAElC,IAAI,UAAU,GAAG,MAAM,CAAC,eAAe,IAAI,SAAS,CAAC;QACrD,IAAI,UAAU,GAAG,MAAM,CAAC,eAAe,IAAI,SAAS,CAAC;QAErD,IAAI,IAAI,KAAK,KAAK,EAAE,CAAC;YACnB,IAAI,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,SAAS,CAAC,EAAE,SAAS,CAAC,CAAC;YAC7D,IAAI,MAAM,GAAG,QAAQ,GAAG,UAAU,EAAE,CAAC;gBACnC,IAAI,eAAe,GAAG,MAAM,GAAG,QAAQ,CAAC;gBACxC,IAAI,eAAe,IAAI,SAAS,EAAE,CAAC;oBACjC,UAAU,GAAG,eAAe,CAAC;gBAC/B,CAAC;qBAAM,CAAC;oBACN,MAAM,GAAG,SAAS,GAAG,QAAQ,CAAC;gBAChC,CAAC;YACH,CAAC;YACD,MAAM,CAAC,eAAe,GAAG,MAAM,CAAC;YAChC,MAAM,CAAC,eAAe,GAAG,UAAU,CAAC;QACtC,CAAC;aAAM,IAAI,IAAI,KAAK,KAAK,EAAE,CAAC;YAC1B,IAAI,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,SAAS,CAAC,EAAE,SAAS,CAAC,CAAC;YAC7D,IAAI,MAAM,GAAG,QAAQ,GAAG,UAAU,EAAE,CAAC;gBACnC,IAAI,eAAe,GAAG,MAAM,GAAG,QAAQ,CAAC;gBACxC,IAAI,eAAe,IAAI,SAAS,EAAE,CAAC;oBACjC,UAAU,GAAG,eAAe,CAAC;gBAC/B,CAAC;qBAAM,CAAC;oBACN,MAAM,GAAG,SAAS,GAAG,QAAQ,CAAC;gBAChC,CAAC;YACH,CAAC;YACD,MAAM,CAAC,eAAe,GAAG,MAAM,CAAC;YAChC,MAAM,CAAC,eAAe,GAAG,UAAU,CAAC;QACtC,CAAC;IACH,CAAC;IAED,QAAQ,GAAG,GAAG,EAAE;QACd,MAAM,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACpB,OAAO,IAAI,CAAA;IACb,CAAC,CAAA;IAED,oBAAoB,CAAC,MAAW;QAC9B,MAAM,KAAK,GAAG,CAAC,MAAM,YAAY,gBAAgB,IAAI,MAAM,CAAC,eAAe,IAAI,IAAI,CAAC;YAClF,CAAC,CAAC,MAAM,CAAC,eAAe;YACxB,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC;QACpB,OAAO,KAAK,CAAC;IACf,CAAC;IAED,oBAAoB,CAAC,MAAW;QAC9B,MAAM,KAAK,GAAG,CAAC,MAAM,YAAY,gBAAgB,IAAI,MAAM,CAAC,eAAe,IAAI,IAAI,CAAC;YAClF,CAAC,CAAC,MAAM,CAAC,eAAe;YACxB,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC;QACpB,OAAO,KAAK,CAAC;IACf,CAAC;wGA3IU,yBAAyB;4FAAzB,yBAAyB,2JCdtC,4xCA4BM,8nDDnBM,oBAAoB,wDAAE,YAAY;;4FAKjC,yBAAyB;kBARrC,SAAS;+BACE,wBAAwB,cACtB,IAAI,WACP,CAAC,oBAAoB,EAAE,YAAY,CAAC,mBAG5B,uBAAuB,CAAC,MAAM;8BAStC,MAAM;sBAAd,KAAK;gBACI,YAAY;sBAArB,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { PriceRangeFilter } from '../../classes/filters/price_range-filter';\nimport { CommonModule } from '@angular/common';\nimport { EcCurrencySymbolPipe } from '../../ec-pipe';\nimport { FiltersService } from '../../ec-services';\n\n@Component({\n  selector: 'lib-price-range-filter',\n  standalone: true,\n  imports: [EcCurrencySymbolPipe, CommonModule],\n  templateUrl: './price-range-filter.component.html',\n  styleUrl: './price-range-filter.component.scss',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PriceRangeFilterComponent {\n\n  private _filtersService: FiltersService = inject(FiltersService);\n\n  priceGap = 1;\n  roundStep = 5;\n\n  @Input() filter!: PriceRangeFilter;\n  @Output() rangeChanged = new EventEmitter<{ min: number; max: number }>();\n\n  getSliderStyle(filter: any): { [klass: string]: string } {\n    if (!(filter instanceof PriceRangeFilter)) return { left: '0%', right: '0%' };\n\n    const left = this.calculateLeftPercentage(filter);\n    const right = this.calculateRightPercentage(filter);\n    return { left: `${left}%`, right: `${right}%` };\n  }\n\n  // Métodos para actualizar la barra del filtro por precio\n  calculateLeftPercentage(filter: any): number {\n    if (!(filter instanceof PriceRangeFilter)) return 0;\n\n    const min = filter.minPrice;\n    const max = filter.maxPrice;\n    const currentMin = filter.currentMinPrice ?? min;\n\n    const clampedMin = Math.max(min, Math.min(currentMin, max));\n    return Math.max(0, Math.min(((clampedMin - min) / (max - min)) * 100, 100));\n  }\n\n  calculateRightPercentage(filter: any): number {\n    if (!(filter instanceof PriceRangeFilter)) return 0;\n\n    const min = filter.minPrice;\n    const max = filter.maxPrice;\n    const currentMax = filter.currentMaxPrice ?? max;\n\n    const clampedMax = Math.max(min, Math.min(currentMax, max));\n    return Math.max(0, Math.min(100 - ((clampedMax - min) / (max - min)) * 100, 100));\n  }\n\n  // Método para actualizar los valores de precio seleccionados\n  updatePriceRange(filter: any) {\n    if (!(filter instanceof PriceRangeFilter)) return;\n\n    const min = filter.currentMinPrice ?? filter.minPrice;\n    const max = filter.currentMaxPrice ?? filter.maxPrice;\n    this._filtersService.updatePriceRangeFilter(min, max);\n  }\n\n  getMinValue(filter: any): number {\n    return (filter instanceof PriceRangeFilter && filter.currentMinPrice != null)\n      ? filter.currentMinPrice\n      : filter.minPrice;\n  }\n\n  getMaxValue(filter: any): number {\n    return (filter instanceof PriceRangeFilter && filter.currentMaxPrice != null)\n      ? filter.currentMaxPrice\n      : filter.maxPrice;\n  }\n\n  /**\n  * Redondea al múltiplo de 'step' respetando el redondeo entero:\n  * - Mínimo: redondea hacia abajo.\n  * - Máximo: redondea hacia arriba.\n  */\n  onRangeChange(event: Event, type: 'min' | 'max', filter: any) {\n    const input = event.target as HTMLInputElement;\n    let value = parseFloat(input.value);\n    value = this.roundToNearest(value, this.roundStep, type, filter);\n    this.updateFilterPrices(type, value, filter, this.priceGap);\n  }\n\n  protected roundToNearest(value: number, step: number, type: 'min' | 'max', filter: any): number {\n    let rounded = Math.round(value / step) * step;\n    if (type === 'max' && rounded < filter.maxPrice) {\n      rounded = Math.ceil(value / step) * step;\n    }\n    if (type === 'min' && rounded > filter.minPrice) {\n      rounded = Math.floor(value / step) * step;\n    }\n    return rounded;\n  }\n\n  /**\n  * Actualiza los valores del filtro permitiendo que un input empuje al otro.\n  */\n  protected updateFilterPrices(type: 'min' | 'max', value: number, filter: any, priceGap: number) {\n    const globalMin = filter.minPrice ?? 0;\n    const globalMax = filter.maxPrice;\n\n    let currentMin = filter.currentMinPrice ?? globalMin;\n    let currentMax = filter.currentMaxPrice ?? globalMax;\n\n    if (type === 'min') {\n      let newMin = Math.min(Math.max(value, globalMin), globalMax);\n      if (newMin + priceGap > currentMax) {\n        let newMaxCandidate = newMin + priceGap;\n        if (newMaxCandidate <= globalMax) {\n          currentMax = newMaxCandidate;\n        } else {\n          newMin = globalMax - priceGap;\n        }\n      }\n      filter.currentMinPrice = newMin;\n      filter.currentMaxPrice = currentMax;\n    } else if (type === 'max') {\n      let newMax = Math.max(Math.min(value, globalMax), globalMin);\n      if (newMax - priceGap < currentMin) {\n        let newMinCandidate = newMax - priceGap;\n        if (newMinCandidate >= globalMin) {\n          currentMin = newMinCandidate;\n        } else {\n          newMax = globalMin + priceGap;\n        }\n      }\n      filter.currentMaxPrice = newMax;\n      filter.currentMinPrice = currentMin;\n    }\n  }\n\n  scrollUp = () => {\n    window.scroll(0, 0);\n    return true\n  }\n\n  getFormattedMinPrice(filter: any): string {\n    const value = (filter instanceof PriceRangeFilter && filter.currentMinPrice != null)\n      ? filter.currentMinPrice\n      : filter.minPrice;\n    return value;\n  }\n\n  getFormattedMaxPrice(filter: any): string {\n    const value = (filter instanceof PriceRangeFilter && filter.currentMaxPrice != null)\n      ? filter.currentMaxPrice\n      : filter.maxPrice;\n    return value;\n  }\n}\n","<div class=\"slider\">\n    <div class=\"slider\">\n        <div class=\"progress\" [ngStyle]=\"getSliderStyle(filter)\">\n        </div>\n    </div>\n</div>\n<div class=\"range-input d-flex gap-2\">\n    <input #rangeMin type=\"range\" class=\"range-min\" [min]=\"filter.minPrice\" [max]=\"filter.maxPrice\"\n        [value]=\"getMinValue(filter)\" (input)=\"onRangeChange($event, 'min', filter)\"\n        (change)=\"updatePriceRange(filter); scrollUp()\">\n    <input #rangeMax type=\"range\" class=\"range-max\" [min]=\"filter.minPrice\" [max]=\"filter.maxPrice\"\n        [value]=\"getMaxValue(filter)\" (input)=\"onRangeChange($event, 'max', filter)\"\n        (change)=\"updatePriceRange(filter); scrollUp()\">\n</div>\n<!-- precios Mínimo / Máximo -->\n<div class=\"price-container mt-2\">\n    <div class=\"price-label-wrapper\">\n        <span class=\"price-label\">Mínimo:</span>\n        <span class=\"price-value\">\n            {{ (getFormattedMinPrice(filter) | ecCurrencySymbol) || '0' }}\n        </span>\n    </div>\n    <div class=\"price-label-wrapper\">\n        <span class=\"price-label\">Máximo:</span>\n        <span class=\"price-value price-value-right\">\n            {{ getFormattedMaxPrice(filter) | ecCurrencySymbol }}\n        </span>\n    </div>\n</div>"]}
|
|
@@ -127,16 +127,26 @@ export class RedsysCatchEcComponent extends ComponentHelper {
|
|
|
127
127
|
}
|
|
128
128
|
/** Intenta cerrar la pestaña actual; si falla, ejecuta el callback de fallback. */
|
|
129
129
|
tryCloseSelf(onFail) {
|
|
130
|
-
if (!isPlatformBrowser(this.platformId))
|
|
131
|
-
|
|
132
|
-
|
|
130
|
+
if (!isPlatformBrowser(this.platformId)) {
|
|
131
|
+
onFail();
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
// Fallback de seguridad: si después de ~800 ms la página sigue visible,
|
|
135
|
+
// asumimos que window.close() fue ignorado.
|
|
136
|
+
const fallback = setTimeout(() => {
|
|
137
|
+
// Si la pestaña siguiera abierta, document.hidden suele ser false.
|
|
138
|
+
if (!document.hidden) {
|
|
139
|
+
onFail();
|
|
140
|
+
}
|
|
141
|
+
}, 800);
|
|
133
142
|
try {
|
|
134
143
|
window.close();
|
|
135
|
-
|
|
144
|
+
// Si window.close() lanza excepción, limpiamos el timeout y hacemos fallback inmediato.
|
|
136
145
|
}
|
|
137
|
-
catch {
|
|
138
|
-
|
|
146
|
+
catch {
|
|
147
|
+
clearTimeout(fallback);
|
|
139
148
|
onFail();
|
|
149
|
+
}
|
|
140
150
|
}
|
|
141
151
|
/** Oculta header/footer para esta pantalla mínima. */
|
|
142
152
|
hideHeaderFooter() {
|
|
@@ -190,4 +200,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
190
200
|
type: Inject,
|
|
191
201
|
args: [PLATFORM_ID]
|
|
192
202
|
}] }] });
|
|
193
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redsys-catch-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/widgets-ec/redsys-catch-ec/redsys-catch-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/widgets-ec/redsys-catch-ec/redsys-catch-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit, OnDestroy, Renderer2, ElementRef, Inject, PLATFORM_ID } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { combineLatest, Subscription } from 'rxjs';\nimport { ComponentHelper } from '../../../classes/component-helper';\nimport { CartService } from '../../../ec-services';\nimport { CheckoutService } from '../../../ec-services';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\nimport { DOCUMENT } from '@angular/common';\nimport { LoadingFullEcComponent } from '../loading/loading-full-ec/loading-full-ec.component';\n\ntype MpState = 'success' | 'pending' | 'failure' | 'cancel';\n\n/**\n * Catch genérico para redirecciones de pagos.\n * - Normaliza el estado recibido por params/query.\n * - Informa el resultado al opener (postMessage), BroadcastChannel y localStorage.\n * - Intenta cerrarse; si el cierre falla, redirige según el estado.\n */\n@Component({\n    selector: 'app-redsys-catch-ec',\n    standalone: true,\n    imports: [CommonModule, LoadingFullEcComponent],\n    templateUrl: './redsys-catch-ec.component.html',\n    styleUrls: ['./redsys-catch-ec.component.scss']\n})\nexport class RedsysCatchEcComponent extends ComponentHelper implements OnInit, OnDestroy {\n\n    public message = '';\n    private subscription: Subscription | null = null;\n    private sid = '';\n    private bc?: BroadcastChannel;\n\n    constructor(\n        public activedRoute: ActivatedRoute,\n        public router: Router,\n        public checkoutService: CheckoutService,\n        private renderer: Renderer2,\n        private elementRef: ElementRef,\n        @Inject(DOCUMENT) private document: Document,\n        @Inject(PLATFORM_ID) private platformId: any\n    ) {\n        super();\n\n        this.hideHeaderFooter();\n        this.ecOnConstruct();\n    }\n\n    ngOnInit() {\n        if (isPlatformBrowser(this.platformId) && 'BroadcastChannel' in window) {\n            this.bc = new BroadcastChannel('mp_payment');\n        }\n\n        this.subscription = combineLatest([this.activedRoute.params, this.activedRoute.queryParams])\n            .subscribe(([routeParams, q]) => {\n                let stateStr: string | undefined = routeParams['state'];\n                if (stateStr === 'statuspayment') stateStr = q['status'];\n\n                const statusParam = (stateStr || q['status'] || q['state'] || '').toString();\n                const state = this.normalizeState(statusParam);\n\n                this.sid = (q['sid'] || (isPlatformBrowser(this.platformId) ? localStorage.getItem('mp:sid') : '') || '').toString();\n                this.storeTotalAmount(q);\n\n                this.setStateInLocal('Su pago fue procesado con éxito.', state);\n                this.signalState(state);\n\n                this.tryCloseSelf(() => {\n                    const target = (state === 'success' || state === 'pending')\n                        ? ['/checkout/order_success']\n                        : ['/checkout'];\n                    setTimeout(() => this.router.navigate(target), 4500);\n                });\n            });\n        this.ecOnInit();\n    }\n\n    ngOnDestroy() {\n        this.subscription?.unsubscribe();\n        this.bc?.close();\n        this.showHeaderFooter();\n    }\n\n    /** Guarda total_amount si viene desde el PSP. */\n    private storeTotalAmount(queryParams: any): void {\n        const totalAmount = queryParams['total_amount'];\n        if (totalAmount && isPlatformBrowser(this.platformId)) {\n            localStorage.setItem('total_amount', totalAmount);\n        }\n    }\n\n    /** Setea mensaje y estado en storages. */\n    private setStateInLocal = (mensaje: string, state: string): void => {\n        this.message = mensaje;\n        if (!isPlatformBrowser(this.platformId)) return;\n        try { localStorage.setItem('state', state); } catch { }\n        try { sessionStorage.setItem('modalnews', 'false'); } catch { }\n    }\n\n    /** Normaliza estados heterogéneos de distintos gateways. */\n    private normalizeState(raw: string): MpState {\n        const v = (raw || '').toLowerCase();\n        if (v === '200' || v === 'ok' || v === 'success') return 'success';\n        if (v === 'pending') return 'pending';\n        if (v === 'cancel') return 'cancel';\n        return 'failure'; // failure, 0, error, rejected, chargeback, desconocidos\n    }\n\n    /**\n     * Informa el resultado: postMessage al opener, BroadcastChannel y localStorage\n     * (este último permite polling en la pestaña madre).\n     */\n    private signalState(state: MpState) {\n        if (!isPlatformBrowser(this.platformId)) return;\n        const sid = this.sid || localStorage.getItem('mp:sid') || '';\n        const payload = { type: 'mp:state', sid, state };\n\n        try { window.opener && window.opener.postMessage(payload, '*'); } catch { }\n        try { this.bc?.postMessage(payload); } catch { }\n        try { localStorage.setItem(`mp:state:${sid}`, state); } catch { }\n        try { localStorage.setItem('state', state); } catch { }\n    }\n\n    /** Intenta cerrar la pestaña actual; si falla, ejecuta el callback de fallback. */\n    private tryCloseSelf(onFail: () => void) {\n        if (!isPlatformBrowser(this.platformId)) return onFail();\n        let attempted = false;\n        try { window.close(); attempted = true; } catch { }\n        if (!attempted) onFail();\n    }\n\n    /** Oculta header/footer para esta pantalla mínima. */\n    private hideHeaderFooter(): void {\n        if (!isPlatformBrowser(this.platformId)) return;\n        const header = this.document.querySelector('header');\n        const footer = this.document.querySelector('footer');\n        if (header) this.renderer.setStyle(header, 'display', 'none');\n        if (footer) this.renderer.setStyle(footer, 'display', 'none');\n    }\n\n    /** Restaura header/footer al salir. */\n    private showHeaderFooter(): void {\n        if (!isPlatformBrowser(this.platformId)) return;\n        const header = this.document.querySelector('header');\n        const footer = this.document.querySelector('footer');\n        if (header) this.renderer.removeStyle(header, 'display');\n        if (footer) this.renderer.removeStyle(footer, 'display');\n    }\n\n    private setStateInSession(mensaje: string, state: MpState) {\n        this.message = mensaje;\n        if (!isPlatformBrowser(this.platformId)) return;\n        try { sessionStorage.setItem('state', state); } catch { }\n        try { localStorage.setItem('state', state); } catch { }\n        try { sessionStorage.setItem('modalnews', 'false'); } catch { }\n    }\n\n\n}\n","<div id=\"container\">\n    <div class=\"row\">\n        <div class=\"col align-self-center\">\n            <h4 class=\"titpage center-block text-center font-nexa font-lg my-3\">{{ message | uppercase }}</h4>\n        </div>\n    </div>\n    <div class=\"row\">\n        <div class=\"col align-self-center\">\n            <h5 class=\"center-block text-center font-nexa my-3\">Redirigiendo en segundos...</h5>\n            <br>\n            <div class=\"d-flex flex-column jusitfy-content-center align-items-center\">\n                <app-loading-full-ec></app-loading-full-ec>\n            </div>\n        </div>\n    </div>\n</div>"]}
|
|
203
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redsys-catch-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/widgets-ec/redsys-catch-ec/redsys-catch-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/widgets-ec/redsys-catch-ec/redsys-catch-ec.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAA4C,MAAM,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAEzG,OAAO,EAAE,aAAa,EAAgB,MAAM,MAAM,CAAC;AACnD,OAAO,EAAE,eAAe,EAAE,MAAM,mCAAmC,CAAC;AAGpE,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AAClE,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,sBAAsB,EAAE,MAAM,sDAAsD,CAAC;;;;;AAI9F;;;;;GAKG;AAQH,MAAM,OAAO,sBAAuB,SAAQ,eAAe;IAQ5C;IACA;IACA;IACC;IACA;IACkB;IACG;IAZ1B,OAAO,GAAG,EAAE,CAAC;IACZ,YAAY,GAAwB,IAAI,CAAC;IACzC,GAAG,GAAG,EAAE,CAAC;IACT,EAAE,CAAoB;IAE9B,YACW,YAA4B,EAC5B,MAAc,EACd,eAAgC,EAC/B,QAAmB,EACnB,UAAsB,EACJ,QAAkB,EACf,UAAe;QAE5C,KAAK,EAAE,CAAC;QARD,iBAAY,GAAZ,YAAY,CAAgB;QAC5B,WAAM,GAAN,MAAM,CAAQ;QACd,oBAAe,GAAf,eAAe,CAAiB;QAC/B,aAAQ,GAAR,QAAQ,CAAW;QACnB,eAAU,GAAV,UAAU,CAAY;QACJ,aAAQ,GAAR,QAAQ,CAAU;QACf,eAAU,GAAV,UAAU,CAAK;QAI5C,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,aAAa,EAAE,CAAC;IACzB,CAAC;IAED,QAAQ;QACJ,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,kBAAkB,IAAI,MAAM,EAAE,CAAC;YACrE,IAAI,CAAC,EAAE,GAAG,IAAI,gBAAgB,CAAC,YAAY,CAAC,CAAC;QACjD,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,aAAa,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;aACvF,SAAS,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;YAC5B,IAAI,QAAQ,GAAuB,WAAW,CAAC,OAAO,CAAC,CAAC;YACxD,IAAI,QAAQ,KAAK,eAAe;gBAAE,QAAQ,GAAG,CAAC,CAAC,QAAQ,CAAC,CAAC;YAEzD,MAAM,WAAW,GAAG,CAAC,QAAQ,IAAI,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;YAC7E,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;YAE/C,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;YACrH,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC;YAEzB,IAAI,CAAC,eAAe,CAAC,kCAAkC,EAAE,KAAK,CAAC,CAAC;YAChE,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAExB,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE;gBACnB,MAAM,MAAM,GAAG,CAAC,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,SAAS,CAAC;oBACvD,CAAC,CAAC,CAAC,yBAAyB,CAAC;oBAC7B,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;gBACpB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,IAAI,CAAC,CAAC;YACzD,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QACP,IAAI,CAAC,QAAQ,EAAE,CAAC;IACpB,CAAC;IAED,WAAW;QACP,IAAI,CAAC,YAAY,EAAE,WAAW,EAAE,CAAC;QACjC,IAAI,CAAC,EAAE,EAAE,KAAK,EAAE,CAAC;QACjB,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC5B,CAAC;IAED,iDAAiD;IACzC,gBAAgB,CAAC,WAAgB;QACrC,MAAM,WAAW,GAAG,WAAW,CAAC,cAAc,CAAC,CAAC;QAChD,IAAI,WAAW,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACpD,YAAY,CAAC,OAAO,CAAC,cAAc,EAAE,WAAW,CAAC,CAAC;QACtD,CAAC;IACL,CAAC;IAED,0CAA0C;IAClC,eAAe,GAAG,CAAC,OAAe,EAAE,KAAa,EAAQ,EAAE;QAC/D,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC;YAAC,YAAY,CAAC,OAAO,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;QACvD,IAAI,CAAC;YAAC,cAAc,CAAC,OAAO,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;IACnE,CAAC,CAAA;IAED,4DAA4D;IACpD,cAAc,CAAC,GAAW;QAC9B,MAAM,CAAC,GAAG,CAAC,GAAG,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;QACpC,IAAI,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,SAAS;YAAE,OAAO,SAAS,CAAC;QACnE,IAAI,CAAC,KAAK,SAAS;YAAE,OAAO,SAAS,CAAC;QACtC,IAAI,CAAC,KAAK,QAAQ;YAAE,OAAO,QAAQ,CAAC;QACpC,OAAO,SAAS,CAAC,CAAC,wDAAwD;IAC9E,CAAC;IAED;;;OAGG;IACK,WAAW,CAAC,KAAc;QAC9B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC;YAAE,OAAO;QAChD,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,IAAI,YAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QAC7D,MAAM,OAAO,GAAG,EAAE,IAAI,EAAE,UAAU,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC;QAEjD,IAAI,CAAC;YAAC,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;QAC3E,IAAI,CAAC;YAAC,IAAI,CAAC,EAAE,EAAE,WAAW,CAAC,OAAO,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;QAChD,IAAI,CAAC;YAAC,YAAY,CAAC,OAAO,CAAC,YAAY,GAAG,EAAE,EAAE,KAAK,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;QACjE,IAAI,CAAC;YAAC,YAAY,CAAC,OAAO,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;IAC3D,CAAC;IAED,mFAAmF;IAC3E,YAAY,CAAC,MAAkB;QACnC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACtC,MAAM,EAAE,CAAC;YACT,OAAO;QACX,CAAC;QACD,wEAAwE;QACxE,4CAA4C;QAC5C,MAAM,QAAQ,GAAG,UAAU,CAAC,GAAG,EAAE;YAC7B,mEAAmE;YACnE,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACnB,MAAM,EAAE,CAAC;YACb,CAAC;QACL,CAAC,EAAE,GAAG,CAAC,CAAC;QACR,IAAI,CAAC;YACD,MAAM,CAAC,KAAK,EAAE,CAAC;YACf,wFAAwF;QAC5F,CAAC;QAAC,MAAM,CAAC;YACL,YAAY,CAAC,QAAQ,CAAC,CAAC;YACvB,MAAM,EAAE,CAAC;QACb,CAAC;IACL,CAAC;IAED,sDAAsD;IAC9C,gBAAgB;QACpB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC;YAAE,OAAO;QAChD,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QACrD,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QACrD,IAAI,MAAM;YAAE,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC;QAC9D,IAAI,MAAM;YAAE,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC;IAClE,CAAC;IAED,uCAAuC;IAC/B,gBAAgB;QACpB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC;YAAE,OAAO;QAChD,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QACrD,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QACrD,IAAI,MAAM;YAAE,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;QACzD,IAAI,MAAM;YAAE,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;IAC7D,CAAC;IAEO,iBAAiB,CAAC,OAAe,EAAE,KAAc;QACrD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC;YAAE,OAAO;QAChD,IAAI,CAAC;YAAC,cAAc,CAAC,OAAO,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;QACzD,IAAI,CAAC;YAAC,YAAY,CAAC,OAAO,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;QACvD,IAAI,CAAC;YAAC,cAAc,CAAC,OAAO,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,CAAC;IACnE,CAAC;wGAhJQ,sBAAsB,wJAanB,QAAQ,aACR,WAAW;4FAdd,sBAAsB,sGCzBnC,2nBAeM,mZDMQ,YAAY,4FAAE,sBAAsB;;4FAIrC,sBAAsB;kBAPlC,SAAS;+BACI,qBAAqB,cACnB,IAAI,WACP,CAAC,YAAY,EAAE,sBAAsB,CAAC;;0BAiB1C,MAAM;2BAAC,QAAQ;;0BACf,MAAM;2BAAC,WAAW","sourcesContent":["import { Component, OnInit, OnDestroy, Renderer2, ElementRef, Inject, PLATFORM_ID } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { combineLatest, Subscription } from 'rxjs';\nimport { ComponentHelper } from '../../../classes/component-helper';\nimport { CartService } from '../../../ec-services';\nimport { CheckoutService } from '../../../ec-services';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\nimport { DOCUMENT } from '@angular/common';\nimport { LoadingFullEcComponent } from '../loading/loading-full-ec/loading-full-ec.component';\n\ntype MpState = 'success' | 'pending' | 'failure' | 'cancel';\n\n/**\n * Catch genérico para redirecciones de pagos.\n * - Normaliza el estado recibido por params/query.\n * - Informa el resultado al opener (postMessage), BroadcastChannel y localStorage.\n * - Intenta cerrarse; si el cierre falla, redirige según el estado.\n */\n@Component({\n    selector: 'app-redsys-catch-ec',\n    standalone: true,\n    imports: [CommonModule, LoadingFullEcComponent],\n    templateUrl: './redsys-catch-ec.component.html',\n    styleUrls: ['./redsys-catch-ec.component.scss']\n})\nexport class RedsysCatchEcComponent extends ComponentHelper implements OnInit, OnDestroy {\n\n    public message = '';\n    private subscription: Subscription | null = null;\n    private sid = '';\n    private bc?: BroadcastChannel;\n\n    constructor(\n        public activedRoute: ActivatedRoute,\n        public router: Router,\n        public checkoutService: CheckoutService,\n        private renderer: Renderer2,\n        private elementRef: ElementRef,\n        @Inject(DOCUMENT) private document: Document,\n        @Inject(PLATFORM_ID) private platformId: any\n    ) {\n        super();\n\n        this.hideHeaderFooter();\n        this.ecOnConstruct();\n    }\n\n    ngOnInit() {\n        if (isPlatformBrowser(this.platformId) && 'BroadcastChannel' in window) {\n            this.bc = new BroadcastChannel('mp_payment');\n        }\n\n        this.subscription = combineLatest([this.activedRoute.params, this.activedRoute.queryParams])\n            .subscribe(([routeParams, q]) => {\n                let stateStr: string | undefined = routeParams['state'];\n                if (stateStr === 'statuspayment') stateStr = q['status'];\n\n                const statusParam = (stateStr || q['status'] || q['state'] || '').toString();\n                const state = this.normalizeState(statusParam);\n\n                this.sid = (q['sid'] || (isPlatformBrowser(this.platformId) ? localStorage.getItem('mp:sid') : '') || '').toString();\n                this.storeTotalAmount(q);\n\n                this.setStateInLocal('Su pago fue procesado con éxito.', state);\n                this.signalState(state);\n\n                this.tryCloseSelf(() => {\n                    const target = (state === 'success' || state === 'pending')\n                        ? ['/checkout/order_success']\n                        : ['/checkout'];\n                    setTimeout(() => this.router.navigate(target), 4500);\n                });\n            });\n        this.ecOnInit();\n    }\n\n    ngOnDestroy() {\n        this.subscription?.unsubscribe();\n        this.bc?.close();\n        this.showHeaderFooter();\n    }\n\n    /** Guarda total_amount si viene desde el PSP. */\n    private storeTotalAmount(queryParams: any): void {\n        const totalAmount = queryParams['total_amount'];\n        if (totalAmount && isPlatformBrowser(this.platformId)) {\n            localStorage.setItem('total_amount', totalAmount);\n        }\n    }\n\n    /** Setea mensaje y estado en storages. */\n    private setStateInLocal = (mensaje: string, state: string): void => {\n        this.message = mensaje;\n        if (!isPlatformBrowser(this.platformId)) return;\n        try { localStorage.setItem('state', state); } catch { }\n        try { sessionStorage.setItem('modalnews', 'false'); } catch { }\n    }\n\n    /** Normaliza estados heterogéneos de distintos gateways. */\n    private normalizeState(raw: string): MpState {\n        const v = (raw || '').toLowerCase();\n        if (v === '200' || v === 'ok' || v === 'success') return 'success';\n        if (v === 'pending') return 'pending';\n        if (v === 'cancel') return 'cancel';\n        return 'failure'; // failure, 0, error, rejected, chargeback, desconocidos\n    }\n\n    /**\n     * Informa el resultado: postMessage al opener, BroadcastChannel y localStorage\n     * (este último permite polling en la pestaña madre).\n     */\n    private signalState(state: MpState) {\n        if (!isPlatformBrowser(this.platformId)) return;\n        const sid = this.sid || localStorage.getItem('mp:sid') || '';\n        const payload = { type: 'mp:state', sid, state };\n\n        try { window.opener && window.opener.postMessage(payload, '*'); } catch { }\n        try { this.bc?.postMessage(payload); } catch { }\n        try { localStorage.setItem(`mp:state:${sid}`, state); } catch { }\n        try { localStorage.setItem('state', state); } catch { }\n    }\n\n    /** Intenta cerrar la pestaña actual; si falla, ejecuta el callback de fallback. */\n    private tryCloseSelf(onFail: () => void) {\n        if (!isPlatformBrowser(this.platformId)) {\n            onFail();\n            return;\n        }\n        // Fallback de seguridad: si después de ~800 ms la página sigue visible,\n        // asumimos que window.close() fue ignorado.\n        const fallback = setTimeout(() => {\n            // Si la pestaña siguiera abierta, document.hidden suele ser false.\n            if (!document.hidden) {\n                onFail();\n            }\n        }, 800);\n        try {\n            window.close();\n            // Si window.close() lanza excepción, limpiamos el timeout y hacemos fallback inmediato.\n        } catch {\n            clearTimeout(fallback);\n            onFail();\n        }\n    }\n\n    /** Oculta header/footer para esta pantalla mínima. */\n    private hideHeaderFooter(): void {\n        if (!isPlatformBrowser(this.platformId)) return;\n        const header = this.document.querySelector('header');\n        const footer = this.document.querySelector('footer');\n        if (header) this.renderer.setStyle(header, 'display', 'none');\n        if (footer) this.renderer.setStyle(footer, 'display', 'none');\n    }\n\n    /** Restaura header/footer al salir. */\n    private showHeaderFooter(): void {\n        if (!isPlatformBrowser(this.platformId)) return;\n        const header = this.document.querySelector('header');\n        const footer = this.document.querySelector('footer');\n        if (header) this.renderer.removeStyle(header, 'display');\n        if (footer) this.renderer.removeStyle(footer, 'display');\n    }\n\n    private setStateInSession(mensaje: string, state: MpState) {\n        this.message = mensaje;\n        if (!isPlatformBrowser(this.platformId)) return;\n        try { sessionStorage.setItem('state', state); } catch { }\n        try { localStorage.setItem('state', state); } catch { }\n        try { sessionStorage.setItem('modalnews', 'false'); } catch { }\n    }\n\n\n}\n","<div id=\"container\">\n    <div class=\"row\">\n        <div class=\"col align-self-center\">\n            <h4 class=\"titpage center-block text-center font-nexa font-lg my-3\">{{ message | uppercase }}</h4>\n        </div>\n    </div>\n    <div class=\"row\">\n        <div class=\"col align-self-center\">\n            <h5 class=\"center-block text-center font-nexa my-3\">Redirigiendo en segundos...</h5>\n            <br>\n            <div class=\"d-flex flex-column jusitfy-content-center align-items-center\">\n                <app-loading-full-ec></app-loading-full-ec>\n            </div>\n        </div>\n    </div>\n</div>"]}
|
|
@@ -2,7 +2,6 @@ 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';
|
|
6
5
|
import * as i0 from "@angular/core";
|
|
7
6
|
export class FiltersService {
|
|
8
7
|
_apiConsts = inject(ApiConstantsService);
|
|
@@ -22,7 +21,6 @@ export class FiltersService {
|
|
|
22
21
|
limit: 0
|
|
23
22
|
};
|
|
24
23
|
_filtersInitialized = false;
|
|
25
|
-
_lastRouteQuery = null;
|
|
26
24
|
constructor() {
|
|
27
25
|
//this._defaultFilters = this._consts.getDefaultFilters()
|
|
28
26
|
}
|
|
@@ -43,20 +41,10 @@ export class FiltersService {
|
|
|
43
41
|
let extra_params = '';
|
|
44
42
|
this._filtersSubject.value.forEach(filter => {
|
|
45
43
|
const extra = filter.toUrlParams();
|
|
46
|
-
|
|
47
|
-
extra_params += extra;
|
|
48
|
-
}
|
|
44
|
+
extra.split('=')[1] != '' ? extra_params += extra : null;
|
|
49
45
|
});
|
|
50
|
-
|
|
51
|
-
|
|
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
|
-
}
|
|
46
|
+
if (search_value)
|
|
47
|
+
extra_params += ('&criteria[search][type]=contains&criteria[search][value]=' + search_value);
|
|
60
48
|
return this.productsFilterApi(extra_params);
|
|
61
49
|
}
|
|
62
50
|
isUpdated(paginationSettings) {
|
|
@@ -72,7 +60,7 @@ export class FiltersService {
|
|
|
72
60
|
this._paginationSettings = paginationSettings;
|
|
73
61
|
return change;
|
|
74
62
|
}
|
|
75
|
-
setFilters(paginationSettings, search_value
|
|
63
|
+
setFilters(paginationSettings, search_value) {
|
|
76
64
|
this._paginationSettings = paginationSettings;
|
|
77
65
|
let final_filters = [];
|
|
78
66
|
let filtersToProcess = this._optionsFilters?.includes('all')
|
|
@@ -85,30 +73,15 @@ export class FiltersService {
|
|
|
85
73
|
if (filter) {
|
|
86
74
|
final_filters.push(filter);
|
|
87
75
|
}
|
|
76
|
+
else {
|
|
77
|
+
// console.warn(`❌ Failed to create filter for type: ${filterType}`);
|
|
78
|
+
}
|
|
88
79
|
});
|
|
89
80
|
});
|
|
90
81
|
this._defaultFilters?.forEach(filterDefault => {
|
|
91
82
|
let filter = final_filters.find(filter => filter.type() == filterDefault.filter_type);
|
|
92
83
|
filter && filterDefault.codes.forEach(value => filter.setSelected(value));
|
|
93
84
|
});
|
|
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
|
-
}
|
|
112
85
|
this._filtersSubject.next(final_filters);
|
|
113
86
|
}
|
|
114
87
|
getFilters(paginationSettings) {
|
|
@@ -123,7 +96,9 @@ export class FiltersService {
|
|
|
123
96
|
runInInjectionContext(this.environmentInjector, () => {
|
|
124
97
|
const filterFactory = new FilterFactory();
|
|
125
98
|
filtersToProcess?.forEach(filterType => {
|
|
99
|
+
// console.log('Creating filter for type:', filterType);
|
|
126
100
|
const filter = filterFactory.create(filterType, settings);
|
|
101
|
+
// console.log('Created filter:', filter);
|
|
127
102
|
if (filter) {
|
|
128
103
|
final_filters.push(filter);
|
|
129
104
|
}
|
|
@@ -148,6 +123,7 @@ export class FiltersService {
|
|
|
148
123
|
if (filterObj.type() !== 'price_range') {
|
|
149
124
|
let index = final_filters.findIndex(filter => filter.type() == filterObj.type());
|
|
150
125
|
final_filters[index].setSelected(filterElem, filterElem.value || filterElem.code);
|
|
126
|
+
console.log(index, final_filters);
|
|
151
127
|
}
|
|
152
128
|
this._filtersSubject.next(final_filters);
|
|
153
129
|
// }
|
|
@@ -157,6 +133,13 @@ export class FiltersService {
|
|
|
157
133
|
runInInjectionContext(this.environmentInjector, () => {
|
|
158
134
|
const filterFactory = new FilterFactory();
|
|
159
135
|
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
|
+
}) */
|
|
160
143
|
this._filtersSubject.value.forEach((value) => {
|
|
161
144
|
if (value.type() == "attributes") {
|
|
162
145
|
//value.data = [];
|
|
@@ -176,95 +159,6 @@ export class FiltersService {
|
|
|
176
159
|
pr.setSelected(min, max);
|
|
177
160
|
this._filtersSubject.next([...filters]);
|
|
178
161
|
}
|
|
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
|
-
}
|
|
268
162
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FiltersService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
269
163
|
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FiltersService, providedIn: 'root' });
|
|
270
164
|
}
|
|
@@ -274,4 +168,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
274
168
|
providedIn: 'root'
|
|
275
169
|
}]
|
|
276
170
|
}], ctorParameters: () => [] });
|
|
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"]}
|
|
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"]}
|