ng-easycommerce-v18 0.2.0 → 0.2.2
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/assets/ec-i18n/ct.json +2 -1
- package/assets/ec-i18n/en.json +2 -1
- package/assets/ec-i18n/es.json +7 -1
- package/assets/ec-i18n/fr.json +2 -1
- package/assets/ec-i18n/gl.json +2 -1
- package/assets/ec-i18n/pr.json +2 -1
- package/esm2022/lib/classes/filters/filter-factory.mjs +4 -1
- package/esm2022/lib/classes/filters/price_range-filter.mjs +59 -0
- package/esm2022/lib/ec-components/auth-ec/login-form-ec/login-form-ec.component.mjs +3 -3
- package/esm2022/lib/ec-components/blocks-ec/block-products-ec/block-products-ec.component.mjs +3 -2
- package/esm2022/lib/ec-components/cart-ec/cart-ec.component.mjs +9 -1
- package/esm2022/lib/ec-components/checkout-ec/dataform-ec/dataform-ec.component.mjs +24 -3
- package/esm2022/lib/ec-components/collection-ec/collection-ec.component.mjs +1 -5
- package/esm2022/lib/ec-components/coupon-ec/coupon-ec.component.mjs +3 -3
- package/esm2022/lib/ec-components/filters-ec/filters-ec.component.mjs +38 -91
- package/esm2022/lib/ec-components/index.mjs +2 -1
- package/esm2022/lib/ec-components/price-range-filter/price-range-filter.component.mjs +143 -0
- package/esm2022/lib/ec-components/product-detail-ec/product-detail-ec.component.mjs +39 -3
- package/esm2022/lib/ec-components/product-ec/product-ec.component.mjs +16 -2
- package/esm2022/lib/ec-components/related-products-ec/related-products-ec.component.mjs +27 -16
- package/esm2022/lib/ec-components/widgets-ec/price-ec/price-ec.component.mjs +73 -10
- package/esm2022/lib/ec-services/filters.service.mjs +45 -12
- package/esm2022/lib/ec-services/pagination.service.mjs +8 -7
- package/esm2022/lib/ec-services/product-detail.service.mjs +50 -35
- package/esm2022/lib/ec-services/products.service.mjs +18 -4
- package/esm2022/lib/interfaces/filter.mjs +1 -1
- package/esm2022/lib/interfaces/pagination.mjs +1 -1
- package/fesm2022/ng-easycommerce-v18.mjs +528 -180
- package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
- package/lib/classes/filters/price_range-filter.d.ts +26 -0
- package/lib/ec-components/cart-ec/cart-ec.component.d.ts +1 -0
- package/lib/ec-components/checkout-ec/dataform-ec/dataform-ec.component.d.ts +1 -0
- package/lib/ec-components/filters-ec/filters-ec.component.d.ts +7 -22
- package/lib/ec-components/index.d.ts +1 -0
- package/lib/ec-components/price-range-filter/price-range-filter.component.d.ts +37 -0
- package/lib/ec-components/product-detail-ec/product-detail-ec.component.d.ts +4 -0
- package/lib/ec-components/product-ec/product-ec.component.d.ts +1 -0
- package/lib/ec-components/related-products-ec/related-products-ec.component.d.ts +4 -2
- package/lib/ec-components/widgets-ec/price-ec/price-ec.component.d.ts +30 -10
- package/lib/ec-services/filters.service.d.ts +6 -1
- package/lib/ec-services/product-detail.service.d.ts +4 -4
- package/lib/ec-services/products.service.d.ts +1 -0
- package/lib/interfaces/filter.d.ts +1 -1
- package/lib/interfaces/pagination.d.ts +2 -0
- package/package.json +1 -1
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Injector } from '@angular/core';
|
|
1
|
+
import { computed, Injector } from '@angular/core';
|
|
2
2
|
import { ParametersService, ChannelService, CartService } from '../../ec-services';
|
|
3
3
|
import { Component, inject, signal } from '@angular/core';
|
|
4
4
|
import { Location } from '@angular/common';
|
|
@@ -35,9 +35,12 @@ export class ProductDetailEcComponent {
|
|
|
35
35
|
success = false;
|
|
36
36
|
categories = [];
|
|
37
37
|
quantity = signal(1);
|
|
38
|
+
isAddingToCart = signal(false);
|
|
38
39
|
parameters$ = this.parametersService.getParameters();
|
|
39
40
|
hasParams = this.parametersService.hasParams;
|
|
40
41
|
creditAccountShowPrices = null;
|
|
42
|
+
// Agregar esta propiedad al inicio de la clase
|
|
43
|
+
productSignal = signal(null);
|
|
41
44
|
constructor() {
|
|
42
45
|
this._activedRoute.params.subscribe(queryParams => {
|
|
43
46
|
this.code = queryParams["id"];
|
|
@@ -56,6 +59,9 @@ export class ProductDetailEcComponent {
|
|
|
56
59
|
this._cartService.showPrice$.subscribe((showPrice) => {
|
|
57
60
|
this.creditAccountShowPrices = showPrice;
|
|
58
61
|
});
|
|
62
|
+
this.product$.subscribe(product => {
|
|
63
|
+
this.productSignal.set(product);
|
|
64
|
+
});
|
|
59
65
|
}
|
|
60
66
|
ngOnInit() {
|
|
61
67
|
this.product$.subscribe(product => {
|
|
@@ -83,7 +89,13 @@ export class ProductDetailEcComponent {
|
|
|
83
89
|
return this._domSanitizer.bypassSecurityTrustHtml(decodedHtml);
|
|
84
90
|
}
|
|
85
91
|
addToCart() {
|
|
86
|
-
this.
|
|
92
|
+
if (this.isAddingToCart() || this.quantity() <= 0)
|
|
93
|
+
return;
|
|
94
|
+
this.isAddingToCart.set(true);
|
|
95
|
+
this._productDetailService.addToCart(this.quantity());
|
|
96
|
+
setTimeout(() => {
|
|
97
|
+
this.isAddingToCart.set(false);
|
|
98
|
+
}, 2000);
|
|
87
99
|
}
|
|
88
100
|
plus(stock, multipleQuantity) {
|
|
89
101
|
if (multipleQuantity && multipleQuantity > 0) {
|
|
@@ -124,6 +136,30 @@ export class ProductDetailEcComponent {
|
|
|
124
136
|
this.reset = !this.reset;
|
|
125
137
|
this.success = true;
|
|
126
138
|
};
|
|
139
|
+
// Computed signals más eficientes
|
|
140
|
+
hasDiscount = computed(() => {
|
|
141
|
+
const product = this.productSignal();
|
|
142
|
+
if (!product?.saleprice || !product?.price) {
|
|
143
|
+
return false;
|
|
144
|
+
}
|
|
145
|
+
const originalPrice = parseFloat(product.price.toString());
|
|
146
|
+
const salePrice = parseFloat(product.saleprice.toString());
|
|
147
|
+
return salePrice > 0 && originalPrice > salePrice;
|
|
148
|
+
});
|
|
149
|
+
discountPercentage = computed(() => {
|
|
150
|
+
if (!this.hasDiscount())
|
|
151
|
+
return null;
|
|
152
|
+
const product = this.productSignal();
|
|
153
|
+
if (!product)
|
|
154
|
+
return null;
|
|
155
|
+
const originalPrice = parseFloat(product.price?.toString() || '0');
|
|
156
|
+
const salePrice = parseFloat(product.saleprice?.toString() || '0');
|
|
157
|
+
if (originalPrice <= 0 || salePrice <= 0 || salePrice >= originalPrice) {
|
|
158
|
+
return null;
|
|
159
|
+
}
|
|
160
|
+
const discount = ((originalPrice - salePrice) / originalPrice) * 100;
|
|
161
|
+
return Math.round(discount);
|
|
162
|
+
});
|
|
127
163
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProductDetailEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
128
164
|
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ProductDetailEcComponent, isStandalone: true, selector: "app-product-detail-ec", ngImport: i0, template: "", styles: [""] });
|
|
129
165
|
}
|
|
@@ -131,4 +167,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
131
167
|
type: Component,
|
|
132
168
|
args: [{ selector: 'app-product-detail-ec', standalone: true, imports: [], template: "" }]
|
|
133
169
|
}], ctorParameters: () => [] });
|
|
134
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-detail-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-detail-ec/product-detail-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-detail-ec/product-detail-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { computed , Injector} from '@angular/core';\r\nimport { ParametersService, ChannelService, CartService } from '../../ec-services';\r\nimport {  Parameter } from '../../interfaces';\r\nimport { Component, inject, WritableSignal, signal, PLATFORM_ID, Inject } from '@angular/core';\r\nimport { Location } from '@angular/common';\r\nimport { Meta, Title } from '@angular/platform-browser';\r\nimport { ActivatedRoute } from '@angular/router';\r\nimport { ProductDetailService, ProductsService, ToastService } from '../../ec-services';\r\nimport { DomSanitizer , SafeHtml  } from '@angular/platform-browser';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { Observable } from 'rxjs';\r\nimport { Product } from '../../interfaces';\r\nimport * as he from 'he';\r\nimport { OptionsService } from '../../ec-services';\r\n\r\n\r\n\r\n@Component({\r\n  selector: 'app-product-detail-ec',\r\n  standalone: true,\r\n  imports: [],\r\n  templateUrl: './product-detail-ec.component.html',\r\n  styleUrls: ['./product-detail-ec.component.scss']\r\n})\r\nexport class ProductDetailEcComponent {\r\n  private _cartService: CartService = inject(CartService)\r\n  private _optionsService: OptionsService = inject(OptionsService);\r\n\tprivate _productDetailService: ProductDetailService = inject(ProductDetailService);\r\n\tprivate _productService: ProductsService = inject(ProductsService);\r\n\tprivate _activedRoute: ActivatedRoute = inject(ActivatedRoute);\r\n\tprivate _domSanitizer: DomSanitizer = inject(DomSanitizer);\r\n\tprivate _consts: CoreConstantsService = inject(CoreConstantsService);\r\n\tprivate _toastService: ToastService = inject(ToastService);\r\n\tprivate parametersService = inject(ParametersService);\r\n\tpublic injector: Injector = inject(Injector);\r\n\tprivate _meta: Meta = inject(Meta);\r\n\tprivate _title: Title = inject(Title);\r\n\tprivate _location: Location = inject(Location);\r\n\tpublic product$: Observable<Product | null> = this._productDetailService.product$;\r\n\tpublic options$: Observable<any> = this._productDetailService.options$;\r\n\tpublic data$: Observable<any> = this._productDetailService.associatedData$;\r\n\r\n\tpublic mediaUrl = this._consts.mediaUrl();\r\n\tpublic code: string = '';\r\n\tpublic showFormContact = false;\r\n\tpublic showReviews: boolean = false;\r\n  public reset: boolean = false;\r\n\t public success: boolean = false;\r\n   public categories: any[] = [];\r\n\r\n\tpublic quantity: WritableSignal<number> = signal(1);\r\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters();\r\n\tpublic hasParams = this.parametersService.hasParams\r\n  public creditAccountShowPrices: boolean | null = null;\r\n\r\n\tconstructor() {\r\n\t\tthis._activedRoute.params.subscribe(queryParams => {\r\n\t\t\tthis.code = queryParams[\"id\"];\r\n\t\t\tthis._productDetailService.loadProduct(this.code);\r\n\t\t  });\r\n\t\tthis._activedRoute.params.subscribe(queryParams => {\r\n\t\t\tthis.code = queryParams[\"id\"];\r\n\t\t\tthis._productDetailService.loadProduct(this.code);\r\n\t\t})\r\n\t\t this.injector.get(ChannelService).channel$.subscribe(channel => {\r\n           this.showReviews = channel.showReviews;\r\n     });\r\n     this._optionsService.getCategories().subscribe(res => {\r\n        this.categories = res;\r\n      }\r\n     );\r\n     this._cartService.showPrice$.subscribe((showPrice) => {\r\n            this.creditAccountShowPrices = showPrice;\r\n        });\r\n\t}\r\n\r\n  ngOnInit() {\r\n    this.product$.subscribe(product => {\r\n      if (product) {\r\n        this.updateMetaTags(product);\r\n      }\r\n    });\r\n  }\r\n\r\n  updateMetaTags(product: Product) {\r\n\tconst descripcionLimpia = he.decode(product.description || '').replace(/<[^>]*>/g, ' ').replace(/\\s+/g, ' ').trim();\r\n    const currentUrl =this._consts.url() +  this._location.path(true);  // `true` incluye el query string\r\n\r\n\r\n\r\n    this._meta.updateTag({ property: 'og:title', content: product.name || '' });\r\n    this._meta.updateTag({ property: 'og:description', content: descripcionLimpia || '' });\r\n    this._meta.updateTag({ property: 'og:image', content: this._consts.mediaImageUrl(Array.isArray(product.picturesdefault) ? product.picturesdefault[0] : product.picturesdefault) || '' });\r\n    this._meta.updateTag({ property: 'og:url', content: currentUrl });\r\n    this._meta.updateTag({ property: 'og:type', content: 'product' });\r\n  }\r\n  decodeHtml(html: string): string {\r\n    const txt = document.createElement('textarea');\r\n    txt.innerHTML = html;\r\n    return txt.value;\r\n    \r\n  }\r\n  sanitizedHtml(html: string): SafeHtml  {\r\n    const decodedHtml = this.decodeHtml(html);\r\n    return this._domSanitizer.bypassSecurityTrustHtml(decodedHtml);\r\n  }\r\n\r\n  addToCart() {\r\n    this.quantity() > 0 && this._productDetailService.addToCart(this.quantity());\r\n  }\r\n\r\n  plus(stock?: number, multipleQuantity?: number) {\r\n    if (multipleQuantity && multipleQuantity > 0) {\r\n      stock ? (this.quantity() < stock\r\n        ? this.quantity.update(value => value + multipleQuantity)\r\n        : this._toastService.show('out-of-stock-actually'))\r\n        : this.quantity.update(value => value + multipleQuantity)\r\n    } else {\r\n      stock ? (this.quantity() < stock\r\n        ? this.quantity.update(value => value + 1)\r\n        : this._toastService.show('out-of-stock-actually'))\r\n        : this.quantity.update(value => value + 1)\r\n    }\r\n  }\r\n\r\n  less(multipleQuantity?: any) {\r\n    if (multipleQuantity && multipleQuantity > 0) {\r\n      this.quantity() > multipleQuantity ? this.quantity.update(value => value - multipleQuantity) : null;\r\n    } else {\r\n      this.quantity() > 1 ? this.quantity.update(value => value - 1) : null;\r\n    }\r\n  }\r\n\r\n  checkStock(stock: number) {\r\n    if (this.quantity() >= stock)\r\n      this.quantity.set(stock);\r\n  }\r\n\r\n  public goToSection(section: string): void {\r\n    this.showFormContact = true;\r\n    setTimeout(() => {\r\n      const element = document.getElementById(section);\r\n      if (element) {\r\n        element.scrollIntoView({ behavior: 'smooth' });\r\n      }\r\n    }, 500);\r\n  }\r\n\r\n  resetForm = () => {\r\n    this.reset = !this.reset\r\n    this.success = true\r\n}\r\n}\r\n",""]}
|
|
170
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-detail-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-detail-ec/product-detail-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-detail-ec/product-detail-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { computed, Injector } from '@angular/core';\r\nimport { ParametersService, ChannelService, CartService } from '../../ec-services';\r\nimport { Parameter } from '../../interfaces';\r\nimport { Component, inject, WritableSignal, signal, PLATFORM_ID, Inject } from '@angular/core';\r\nimport { Location } from '@angular/common';\r\nimport { Meta, Title } from '@angular/platform-browser';\r\nimport { ActivatedRoute } from '@angular/router';\r\nimport { ProductDetailService, ProductsService, ToastService } from '../../ec-services';\r\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { Observable } from 'rxjs';\r\nimport { Product } from '../../interfaces';\r\nimport * as he from 'he';\r\nimport { OptionsService } from '../../ec-services';\r\n\r\n\r\n\r\n@Component({\r\n  selector: 'app-product-detail-ec',\r\n  standalone: true,\r\n  imports: [],\r\n  templateUrl: './product-detail-ec.component.html',\r\n  styleUrls: ['./product-detail-ec.component.scss']\r\n})\r\nexport class ProductDetailEcComponent {\r\n  private _cartService: CartService = inject(CartService)\r\n  private _optionsService: OptionsService = inject(OptionsService);\r\n  private _productDetailService: ProductDetailService = inject(ProductDetailService);\r\n  private _productService: ProductsService = inject(ProductsService);\r\n  private _activedRoute: ActivatedRoute = inject(ActivatedRoute);\r\n  private _domSanitizer: DomSanitizer = inject(DomSanitizer);\r\n  private _consts: CoreConstantsService = inject(CoreConstantsService);\r\n  private _toastService: ToastService = inject(ToastService);\r\n  private parametersService = inject(ParametersService);\r\n  public injector: Injector = inject(Injector);\r\n  private _meta: Meta = inject(Meta);\r\n  private _title: Title = inject(Title);\r\n  private _location: Location = inject(Location);\r\n  public product$: Observable<Product | null> = this._productDetailService.product$;\r\n  public options$: Observable<any> = this._productDetailService.options$;\r\n  public data$: Observable<any> = this._productDetailService.associatedData$;\r\n\r\n  public mediaUrl = this._consts.mediaUrl();\r\n  public code: string = '';\r\n  public showFormContact = false;\r\n  public showReviews: boolean = false;\r\n  public reset: boolean = false;\r\n  public success: boolean = false;\r\n  public categories: any[] = [];\r\n\r\n  public quantity: WritableSignal<number> = signal(1);\r\n  public isAddingToCart: WritableSignal<boolean> = signal(false);\r\n  public parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters();\r\n  public hasParams = this.parametersService.hasParams\r\n  public creditAccountShowPrices: boolean | null = null;\r\n  // Agregar esta propiedad al inicio de la clase\r\n  private productSignal = signal<Product | null>(null);\r\n\r\n\r\n  constructor() {\r\n    this._activedRoute.params.subscribe(queryParams => {\r\n      this.code = queryParams[\"id\"];\r\n      this._productDetailService.loadProduct(this.code);\r\n    });\r\n    this._activedRoute.params.subscribe(queryParams => {\r\n      this.code = queryParams[\"id\"];\r\n      this._productDetailService.loadProduct(this.code);\r\n    })\r\n    this.injector.get(ChannelService).channel$.subscribe(channel => {\r\n      this.showReviews = channel.showReviews;\r\n    });\r\n    this._optionsService.getCategories().subscribe(res => {\r\n      this.categories = res;\r\n    }\r\n    );\r\n    this._cartService.showPrice$.subscribe((showPrice) => {\r\n      this.creditAccountShowPrices = showPrice;\r\n    });\r\n\r\n    this.product$.subscribe(product => {\r\n      this.productSignal.set(product);\r\n    });\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.product$.subscribe(product => {\r\n      if (product) {\r\n        this.updateMetaTags(product);\r\n      }\r\n    });\r\n  }\r\n\r\n  updateMetaTags(product: Product) {\r\n    const descripcionLimpia = he.decode(product.description || '').replace(/<[^>]*>/g, ' ').replace(/\\s+/g, ' ').trim();\r\n    const currentUrl = this._consts.url() + this._location.path(true);  // `true` incluye el query string\r\n\r\n\r\n\r\n    this._meta.updateTag({ property: 'og:title', content: product.name || '' });\r\n    this._meta.updateTag({ property: 'og:description', content: descripcionLimpia || '' });\r\n    this._meta.updateTag({ property: 'og:image', content: this._consts.mediaImageUrl(Array.isArray(product.picturesdefault) ? product.picturesdefault[0] : product.picturesdefault) || '' });\r\n    this._meta.updateTag({ property: 'og:url', content: currentUrl });\r\n    this._meta.updateTag({ property: 'og:type', content: 'product' });\r\n  }\r\n  decodeHtml(html: string): string {\r\n    const txt = document.createElement('textarea');\r\n    txt.innerHTML = html;\r\n    return txt.value;\r\n\r\n  }\r\n  sanitizedHtml(html: string): SafeHtml {\r\n    const decodedHtml = this.decodeHtml(html);\r\n    return this._domSanitizer.bypassSecurityTrustHtml(decodedHtml);\r\n  }\r\n\r\n  addToCart() {\r\n    if (this.isAddingToCart() || this.quantity() <= 0) return;\r\n\r\n    this.isAddingToCart.set(true);\r\n    this._productDetailService.addToCart(this.quantity());\r\n\r\n    setTimeout(() => {\r\n      this.isAddingToCart.set(false);\r\n    }, 2000);\r\n  }\r\n\r\n  plus(stock?: number, multipleQuantity?: number) {\r\n    if (multipleQuantity && multipleQuantity > 0) {\r\n      stock ? (this.quantity() < stock\r\n        ? this.quantity.update(value => value + multipleQuantity)\r\n        : this._toastService.show('out-of-stock-actually'))\r\n        : this.quantity.update(value => value + multipleQuantity)\r\n    } else {\r\n      stock ? (this.quantity() < stock\r\n        ? this.quantity.update(value => value + 1)\r\n        : this._toastService.show('out-of-stock-actually'))\r\n        : this.quantity.update(value => value + 1)\r\n    }\r\n  }\r\n\r\n  less(multipleQuantity?: any) {\r\n    if (multipleQuantity && multipleQuantity > 0) {\r\n      this.quantity() > multipleQuantity ? this.quantity.update(value => value - multipleQuantity) : null;\r\n    } else {\r\n      this.quantity() > 1 ? this.quantity.update(value => value - 1) : null;\r\n    }\r\n  }\r\n\r\n  checkStock(stock: number) {\r\n    if (this.quantity() >= stock)\r\n      this.quantity.set(stock);\r\n  }\r\n\r\n  public goToSection(section: string): void {\r\n    this.showFormContact = true;\r\n    setTimeout(() => {\r\n      const element = document.getElementById(section);\r\n      if (element) {\r\n        element.scrollIntoView({ behavior: 'smooth' });\r\n      }\r\n    }, 500);\r\n  }\r\n\r\n  resetForm = () => {\r\n    this.reset = !this.reset\r\n    this.success = true\r\n  }\r\n\r\n  // Computed signals más eficientes\r\n  public hasDiscount = computed(() => {\r\n    const product = this.productSignal();\r\n    if (!product?.saleprice || !product?.price) {\r\n      return false;\r\n    }\r\n\r\n    const originalPrice = parseFloat(product.price.toString());\r\n    const salePrice = parseFloat(product.saleprice.toString());\r\n\r\n    return salePrice > 0 && originalPrice > salePrice;\r\n  });\r\n\r\n  public discountPercentage = computed(() => {\r\n    if (!this.hasDiscount()) return null;\r\n\r\n    const product = this.productSignal();\r\n    if (!product) return null;\r\n\r\n    const originalPrice = parseFloat(product.price?.toString() || '0');\r\n    const salePrice = parseFloat(product.saleprice?.toString() || '0');\r\n\r\n    if (originalPrice <= 0 || salePrice <= 0 || salePrice >= originalPrice) {\r\n      return null;\r\n    }\r\n\r\n    const discount = ((originalPrice - salePrice) / originalPrice) * 100;\r\n    return Math.round(discount);\r\n  });\r\n\r\n\r\n}\r\n",""]}
|
|
@@ -47,8 +47,22 @@ export class ProductEcComponent {
|
|
|
47
47
|
get hasDiscount() {
|
|
48
48
|
return this.product.saleprice && this.product.saleprice !== this.product.price;
|
|
49
49
|
}
|
|
50
|
+
get discountPercentage() {
|
|
51
|
+
if (!this.hasDiscount) {
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
const originalPrice = parseFloat(this.product.price?.toString() || '0');
|
|
55
|
+
const salePrice = parseFloat(this.product.saleprice?.toString() || '0');
|
|
56
|
+
// Validar que los precios sean válidos
|
|
57
|
+
if (originalPrice <= 0 || salePrice <= 0 || salePrice >= originalPrice) {
|
|
58
|
+
return null;
|
|
59
|
+
}
|
|
60
|
+
// Calcular el porcentaje: ((precio_original - precio_oferta) / precio_original) * 100
|
|
61
|
+
const discount = ((originalPrice - salePrice) / originalPrice) * 100;
|
|
62
|
+
return Math.round(discount);
|
|
63
|
+
}
|
|
50
64
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProductEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
51
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ProductEcComponent, isStandalone: true, selector: "app-product-ec", inputs: { product: "product", isProductBox: "isProductBox", isCollection: "isCollection" }, outputs: { loaded: "loaded" }, ngImport: i0, template: "<a [routerLink]=\"['/product', product.id]\" class=\"text-decoration-none producto\">\r\n <!-- Marca especial y descuento -->\r\n <!-- <div *ngIf=\"product.saleprice || (product.special_mark && product.special_mark !== null && product.special_mark !== undefined && product.special_mark.length >0)\"\r\n class=\"marcas\">\r\n <div *ecProductStock=\"product\" [ecProductMini]=\"product.special_mark\"></div>\r\n <ng-container *ngIf=\"shouldShowPrice\">\r\n <div *ecProductStock=\"product\" [ngClass]=\"{'tag-dsc float-right': product.saleprice}\"\r\n [ecProductOff]=\"product\">\r\n </div>\r\n </ng-container>\r\n </div> -->\r\n\r\n <!-- Imagen del producto -->\r\n <div class=\"foto\">\r\n @if(product.picturesdefault){\r\n @if (product.picturesdefault && product.picturesdefault.length > 1 ) {\r\n <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n <img [src]=\"mediaUrl + product.picturesdefault[1]\" alt=\"Imagen secundaria\" class=\"w-100 pic02\" />\r\n } @else {\r\n <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n }\r\n }\r\n </div>\r\n <!-- Precio -->\r\n\r\n\r\n <!-- Nombre del producto -->\r\n <h6 class=\"title\">{{ product.name | titlecase }}</h6>\r\n\r\n <div class=\"sku\" [innerHTML]=\"product.shortdetails\"></div>\r\n\r\n @if (shouldShowPrice) {\r\n <app-price-ec [price]=\"product.price\" [saleprice]=\"product.saleprice\" class=\"\" />\r\n }\r\n\r\n <div class=\"fixBottom\">\r\n\r\n <!-- Bot\u00F3n de acciones -->\r\n <!-- <ng-container *ecProductStock=\"product; else noStock\"> -->\r\n <!-- Cuando no tiene marca especial o es de tipo 'standard' -->\r\n @if (!product.special_mark || product.special_mark.length === 0 || product.special_mark[0]?.type ===\r\n 'standard') {\r\n <button class=\"btn standard\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n <ng-container *ngIf=\"isCollection; else normalText\">\r\n <span> {{(\"buy\" | translate) | uppercase}} </span>\r\n </ng-container>\r\n <ng-template #normalText>\r\n {{(\"buy\" | translate) | uppercase}}\r\n </ng-template>\r\n </button>\r\n }@else {\r\n <!-- Caso 1: Agotado o Disponible muy pronto -->\r\n @if (product.special_mark[0]?.type === 'out_of_stock' || product.special_mark[0]?.type === 'coming_soon') {\r\n <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n @if(isCollection){\r\n <span>{{ product.special_mark[0].name | uppercase }} </span>\r\n }@else {\r\n {{ product.special_mark[0]?.name | uppercase }}\r\n }\r\n </button>}\r\n <!-- Caso 2: Contacto por WhatsApp -->\r\n @if (product.special_mark[0].type === 'whatsapp_contact') {\r\n <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\"\r\n (click)=\"openWhatsApp(product.special_mark[0]?.whatsappContact)\">\r\n @if(isCollection){\r\n <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n }@else {\r\n {{ product.special_mark[0]?.name | uppercase }}\r\n }\r\n\r\n </button>\r\n }\r\n <!-- Caso 3: Solicitar m\u00E1s informaci\u00F3n -->\r\n @if (product.special_mark[0]?.type === 'more_info') {\r\n <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n @if(isCollection){\r\n <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n }@else {\r\n {{ product.special_mark[0]?.name | uppercase }}\r\n }\r\n </button>\r\n }\r\n }\r\n </div>\r\n</a>", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.UpperCasePipe, name: "uppercase" }, { kind: "pipe", type: i1.TitleCasePipe, name: "titlecase" }, { kind: "component", type: PriceEcComponent, selector: "app-price-ec", inputs: ["price", "saleprice"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
|
|
65
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ProductEcComponent, isStandalone: true, selector: "app-product-ec", inputs: { product: "product", isProductBox: "isProductBox", isCollection: "isCollection" }, outputs: { loaded: "loaded" }, ngImport: i0, template: "<a [routerLink]=\"['/product', product.id]\" class=\"text-decoration-none producto\">\r\n <!-- Marca especial y descuento -->\r\n <!-- <div *ngIf=\"product.saleprice || (product.special_mark && product.special_mark !== null && product.special_mark !== undefined && product.special_mark.length >0)\"\r\n class=\"marcas\">\r\n <div *ecProductStock=\"product\" [ecProductMini]=\"product.special_mark\"></div>\r\n <ng-container *ngIf=\"shouldShowPrice\">\r\n <div *ecProductStock=\"product\" [ngClass]=\"{'tag-dsc float-right': product.saleprice}\"\r\n [ecProductOff]=\"product\">\r\n </div>\r\n </ng-container>\r\n </div> -->\r\n\r\n <!-- Imagen del producto -->\r\n <div class=\"foto\">\r\n @if(product.picturesdefault){\r\n @if (product.picturesdefault && product.picturesdefault.length > 1 ) {\r\n <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n <img [src]=\"mediaUrl + product.picturesdefault[1]\" alt=\"Imagen secundaria\" class=\"w-100 pic02\" />\r\n } @else {\r\n <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n }\r\n }\r\n </div>\r\n <!-- Precio -->\r\n\r\n\r\n <!-- Nombre del producto -->\r\n <h6 class=\"title\">{{ product.name | titlecase }}</h6>\r\n\r\n <div class=\"sku\" [innerHTML]=\"product.shortdetails\"></div>\r\n\r\n @if (shouldShowPrice) {\r\n <app-price-ec [price]=\"product.price\" [saleprice]=\"product.saleprice\" class=\"\" />\r\n }\r\n\r\n <div class=\"fixBottom\">\r\n\r\n <!-- Bot\u00F3n de acciones -->\r\n <!-- <ng-container *ecProductStock=\"product; else noStock\"> -->\r\n <!-- Cuando no tiene marca especial o es de tipo 'standard' -->\r\n @if (!product.special_mark || product.special_mark.length === 0 || product.special_mark[0]?.type ===\r\n 'standard') {\r\n <button class=\"btn standard\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n <ng-container *ngIf=\"isCollection; else normalText\">\r\n <span> {{(\"buy\" | translate) | uppercase}} </span>\r\n </ng-container>\r\n <ng-template #normalText>\r\n {{(\"buy\" | translate) | uppercase}}\r\n </ng-template>\r\n </button>\r\n }@else {\r\n <!-- Caso 1: Agotado o Disponible muy pronto -->\r\n @if (product.special_mark[0]?.type === 'out_of_stock' || product.special_mark[0]?.type === 'coming_soon') {\r\n <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n @if(isCollection){\r\n <span>{{ product.special_mark[0].name | uppercase }} </span>\r\n }@else {\r\n {{ product.special_mark[0]?.name | uppercase }}\r\n }\r\n </button>}\r\n <!-- Caso 2: Contacto por WhatsApp -->\r\n @if (product.special_mark[0].type === 'whatsapp_contact') {\r\n <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\"\r\n (click)=\"openWhatsApp(product.special_mark[0]?.whatsappContact)\">\r\n @if(isCollection){\r\n <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n }@else {\r\n {{ product.special_mark[0]?.name | uppercase }}\r\n }\r\n\r\n </button>\r\n }\r\n <!-- Caso 3: Solicitar m\u00E1s informaci\u00F3n -->\r\n @if (product.special_mark[0]?.type === 'more_info') {\r\n <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n @if(isCollection){\r\n <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n }@else {\r\n {{ product.special_mark[0]?.name | uppercase }}\r\n }\r\n </button>\r\n }\r\n }\r\n </div>\r\n</a>", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.UpperCasePipe, name: "uppercase" }, { kind: "pipe", type: i1.TitleCasePipe, name: "titlecase" }, { kind: "component", type: PriceEcComponent, selector: "app-price-ec", inputs: ["price", "saleprice", "basePrice", "taxeAmount", "taxes", "priceSize", "showTaxLegendOnly", "disableTaxInfo", "customPriceTemplate", "customSalePriceTemplate", "customSimplePriceTemplate", "customSimpleSalePriceTemplate", "customTaxTemplate", "customOnlyTaxLabelTemplate"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
|
|
52
66
|
}
|
|
53
67
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProductEcComponent, decorators: [{
|
|
54
68
|
type: Component,
|
|
@@ -65,4 +79,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
65
79
|
}], isCollection: [{
|
|
66
80
|
type: Input
|
|
67
81
|
}] } });
|
|
68
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-ec/product-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-ec/product-ec.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAEvF,OAAO,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AACvD,OAAO,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAC;AACrD,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,gBAAgB,EAAE,MAAM,2CAA2C,CAAC;AAC7E,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;;;;AACtD;;;GAGG;AAQH,MAAM,OAAO,kBAAkB;IAC9B;;OAEG;IAIH,OAAO,CAAW;IAClB;;OAEG;IACM,YAAY,GAAW,IAAI,CAAC;IACrC;;OAEG;IACO,MAAM,GAAG,IAAI,YAAY,EAAU,CAAC;IAErC,YAAY,GAAY,KAAK,CAAC;IAEvC,QAAQ;QACN,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IACpC,CAAC;IACO,MAAM,GAAwB,MAAM,CAAC,oBAAoB,CAAC,CAAA;IAC1D,gBAAgB,GAAqB,MAAM,CAAC,gBAAgB,CAAC,CAAA;IAC7D,MAAM,GAAU,MAAM,CAAC,MAAM,CAAC,CAAA;IACtC;;OAEG;IACI,QAAQ,GAAU,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAA;IAC/C,YAAY,CAAC,GAAW;QACjB,IAAI,GAAG,EAAE,CAAC;YACN,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;QAC/B,CAAC;IACL,CAAC;IAED,IAAI,eAAe;QACf,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC;IAC5H,CAAC;IAED,IAAI,WAAW;QACX,OAAO,IAAI,CAAC,OAAO,CAAC,SAAS,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,KAAK,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC;IACnF,CAAC;wGAzCQ,kBAAkB;4FAAlB,kBAAkB,qMCnB/B,08HAoFI,yDDrEO,YAAY,2VAAE,gBAAgB,yFAAE,UAAU,mOAAG,eAAe;;4FAI1D,kBAAkB;kBAP9B,SAAS;+BACC,gBAAgB,cACd,IAAI,WACP,CAAC,YAAY,EAAE,gBAAgB,EAAE,UAAU,EAAG,eAAe,CAAC;8BAWvE,OAAO;sBAHN,KAAK;uBAAC;wBACN,QAAQ,EAAE,IAAI;qBACd;gBAKQ,YAAY;sBAApB,KAAK;gBAII,MAAM;sBAAf,MAAM;gBAEE,YAAY;sBAApB,KAAK","sourcesContent":["import { Component, EventEmitter, inject, Input, OnInit, Output } from '@angular/core';\r\nimport { Product } from '../../interfaces';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { AnalyticsService } from '../../ec-services';\r\nimport { Router, RouterLink } from '@angular/router';\r\nimport { CommonModule } from '@angular/common';\r\nimport { PriceEcComponent } from \"../widgets-ec/price-ec/price-ec.component\";\r\nimport { TranslateModule } from '@ngx-translate/core';\r\n/**\r\n * Componente que se encarga de manejar un producto.\r\n * @class ProductEcComponent\r\n */\r\n@Component({\r\n\tselector: 'app-product-ec',\r\n\tstandalone: true,\r\n\timports: [CommonModule, PriceEcComponent, RouterLink , TranslateModule],\r\n\ttemplateUrl: './product-ec.component.html',\r\n\tstyleUrl: './product-ec.component.scss'\r\n})\r\nexport class ProductEcComponent implements OnInit {\r\n\t/**\r\n\t * Datos del producto.\r\n\t */\r\n\t@Input({\r\n\t\trequired: true\r\n\t})\r\n\tproduct!: Product;\r\n\t/**\r\n\t * Se usa para saber si el producto es de tipo Box o no y definir la vista adecuada.\r\n\t */\r\n\t@Input() isProductBox:boolean = true;\r\n\t/**\r\n\t * Constantes del core\r\n\t */\r\n\t@Output() loaded = new EventEmitter<number>();\r\n\r\n\t@Input() isCollection: boolean = false;\r\n\r\n\tngOnInit() {\r\n\t  this.loaded.emit(this.product.id);\r\n\t}\r\n\tprivate consts:CoreConstantsService = inject(CoreConstantsService)\r\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\r\n\tprivate router:Router = inject(Router)\r\n\t/**\r\n\t * URL para las imagenes del producto.\r\n\t */\r\n\tpublic mediaUrl:string = this.consts.mediaUrl()\r\n\topenWhatsApp(url: string): void {\r\n        if (url) {\r\n            window.open(url, '_blank');\r\n        }\r\n    }\r\n\r\n    get shouldShowPrice(): boolean {\r\n        return !this.product?.special_mark || this.product.special_mark.length === 0 || this.product.special_mark[0]?.showPrice;\r\n    }\r\n\r\n    get hasDiscount(): boolean {\r\n        return this.product.saleprice && this.product.saleprice !== this.product.price;\r\n    }\r\n}\r\n","<a [routerLink]=\"['/product', product.id]\" class=\"text-decoration-none producto\">\r\n    <!-- Marca especial y descuento -->\r\n    <!-- <div *ngIf=\"product.saleprice || (product.special_mark && product.special_mark !== null && product.special_mark !== undefined && product.special_mark.length >0)\"\r\n        class=\"marcas\">\r\n        <div *ecProductStock=\"product\" [ecProductMini]=\"product.special_mark\"></div>\r\n        <ng-container *ngIf=\"shouldShowPrice\">\r\n            <div *ecProductStock=\"product\" [ngClass]=\"{'tag-dsc float-right': product.saleprice}\"\r\n                [ecProductOff]=\"product\">\r\n            </div>\r\n        </ng-container>\r\n    </div> -->\r\n\r\n    <!-- Imagen del producto -->\r\n    <div class=\"foto\">\r\n        @if(product.picturesdefault){\r\n        @if (product.picturesdefault && product.picturesdefault.length > 1 ) {\r\n        <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n        <img [src]=\"mediaUrl + product.picturesdefault[1]\" alt=\"Imagen secundaria\" class=\"w-100 pic02\" />\r\n        } @else {\r\n        <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n        }\r\n        }\r\n    </div>\r\n    <!-- Precio -->\r\n\r\n\r\n    <!-- Nombre del producto -->\r\n    <h6 class=\"title\">{{ product.name | titlecase }}</h6>\r\n\r\n    <div class=\"sku\" [innerHTML]=\"product.shortdetails\"></div>\r\n\r\n    @if (shouldShowPrice) {\r\n    <app-price-ec [price]=\"product.price\" [saleprice]=\"product.saleprice\" class=\"\" />\r\n    }\r\n\r\n    <div class=\"fixBottom\">\r\n\r\n        <!-- Botón de acciones -->\r\n        <!-- <ng-container *ecProductStock=\"product; else noStock\"> -->\r\n        <!-- Cuando no tiene marca especial o es de tipo 'standard' -->\r\n        @if (!product.special_mark || product.special_mark.length === 0 || product.special_mark[0]?.type ===\r\n        'standard') {\r\n        <button class=\"btn standard\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n            <ng-container *ngIf=\"isCollection; else normalText\">\r\n                <span> {{(\"buy\" | translate) | uppercase}} </span>\r\n            </ng-container>\r\n            <ng-template #normalText>\r\n                {{(\"buy\" | translate) | uppercase}}\r\n            </ng-template>\r\n        </button>\r\n        }@else {\r\n        <!-- Caso 1: Agotado o Disponible muy pronto -->\r\n        @if (product.special_mark[0]?.type === 'out_of_stock' || product.special_mark[0]?.type === 'coming_soon') {\r\n        <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n            @if(isCollection){\r\n            <span>{{ product.special_mark[0].name | uppercase }} </span>\r\n            }@else {\r\n            {{ product.special_mark[0]?.name | uppercase }}\r\n            }\r\n        </button>}\r\n        <!-- Caso 2: Contacto por WhatsApp -->\r\n        @if (product.special_mark[0].type === 'whatsapp_contact') {\r\n        <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\"\r\n            (click)=\"openWhatsApp(product.special_mark[0]?.whatsappContact)\">\r\n            @if(isCollection){\r\n            <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n            }@else {\r\n            {{ product.special_mark[0]?.name | uppercase }}\r\n            }\r\n\r\n        </button>\r\n        }\r\n        <!-- Caso 3: Solicitar más información -->\r\n        @if (product.special_mark[0]?.type === 'more_info') {\r\n        <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n            @if(isCollection){\r\n            <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n            }@else {\r\n            {{ product.special_mark[0]?.name | uppercase }}\r\n            }\r\n        </button>\r\n    }\r\n        }\r\n    </div>\r\n</a>"]}
|
|
82
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"product-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-ec/product-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/product-ec/product-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, inject, Input, OnInit, Output } from '@angular/core';\r\nimport { Product } from '../../interfaces';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { AnalyticsService } from '../../ec-services';\r\nimport { Router, RouterLink } from '@angular/router';\r\nimport { CommonModule } from '@angular/common';\r\nimport { PriceEcComponent } from \"../widgets-ec/price-ec/price-ec.component\";\r\nimport { TranslateModule } from '@ngx-translate/core';\r\n/**\r\n * Componente que se encarga de manejar un producto.\r\n * @class ProductEcComponent\r\n */\r\n@Component({\r\n\tselector: 'app-product-ec',\r\n\tstandalone: true,\r\n\timports: [CommonModule, PriceEcComponent, RouterLink , TranslateModule],\r\n\ttemplateUrl: './product-ec.component.html',\r\n\tstyleUrl: './product-ec.component.scss'\r\n})\r\nexport class ProductEcComponent implements OnInit {\r\n\t/**\r\n\t * Datos del producto.\r\n\t */\r\n\t@Input({\r\n\t\trequired: true\r\n\t})\r\n\tproduct!: Product;\r\n\t/**\r\n\t * Se usa para saber si el producto es de tipo Box o no y definir la vista adecuada.\r\n\t */\r\n\t@Input() isProductBox:boolean = true;\r\n\t/**\r\n\t * Constantes del core\r\n\t */\r\n\t@Output() loaded = new EventEmitter<number>();\r\n\r\n\t@Input() isCollection: boolean = false;\r\n\r\n\tngOnInit() {\r\n\t  this.loaded.emit(this.product.id);\r\n\t}\r\n\tprivate consts:CoreConstantsService = inject(CoreConstantsService)\r\n\tprivate analyticsService: AnalyticsService = inject(AnalyticsService)\r\n\tprivate router:Router = inject(Router)\r\n\t/**\r\n\t * URL para las imagenes del producto.\r\n\t */\r\n\tpublic mediaUrl:string = this.consts.mediaUrl()\r\n\topenWhatsApp(url: string): void {\r\n        if (url) {\r\n            window.open(url, '_blank');\r\n        }\r\n    }\r\n\r\n    get shouldShowPrice(): boolean {\r\n        return !this.product?.special_mark || this.product.special_mark.length === 0 || this.product.special_mark[0]?.showPrice;\r\n    }\r\n\r\n    get hasDiscount(): boolean {\r\n        return this.product.saleprice && this.product.saleprice !== this.product.price;\r\n    }\r\n\r\n\tget discountPercentage(): number | null {\r\n        if (!this.hasDiscount) {\r\n            return null;\r\n        }\r\n\r\n        const originalPrice = parseFloat(this.product.price?.toString() || '0');\r\n        const salePrice = parseFloat(this.product.saleprice?.toString() || '0');\r\n\r\n        // Validar que los precios sean válidos\r\n        if (originalPrice <= 0 || salePrice <= 0 || salePrice >= originalPrice) {\r\n            return null;\r\n        }\r\n\r\n        // Calcular el porcentaje: ((precio_original - precio_oferta) / precio_original) * 100\r\n        const discount = ((originalPrice - salePrice) / originalPrice) * 100;\r\n        \r\n        return Math.round(discount);\r\n    }\r\n}\r\n","<a [routerLink]=\"['/product', product.id]\" class=\"text-decoration-none producto\">\r\n    <!-- Marca especial y descuento -->\r\n    <!-- <div *ngIf=\"product.saleprice || (product.special_mark && product.special_mark !== null && product.special_mark !== undefined && product.special_mark.length >0)\"\r\n        class=\"marcas\">\r\n        <div *ecProductStock=\"product\" [ecProductMini]=\"product.special_mark\"></div>\r\n        <ng-container *ngIf=\"shouldShowPrice\">\r\n            <div *ecProductStock=\"product\" [ngClass]=\"{'tag-dsc float-right': product.saleprice}\"\r\n                [ecProductOff]=\"product\">\r\n            </div>\r\n        </ng-container>\r\n    </div> -->\r\n\r\n    <!-- Imagen del producto -->\r\n    <div class=\"foto\">\r\n        @if(product.picturesdefault){\r\n        @if (product.picturesdefault && product.picturesdefault.length > 1 ) {\r\n        <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n        <img [src]=\"mediaUrl + product.picturesdefault[1]\" alt=\"Imagen secundaria\" class=\"w-100 pic02\" />\r\n        } @else {\r\n        <img [src]=\"mediaUrl + product.picturesdefault[0]\" alt=\"Imagen principal\" class=\"w-100 pic01\" />\r\n        }\r\n        }\r\n    </div>\r\n    <!-- Precio -->\r\n\r\n\r\n    <!-- Nombre del producto -->\r\n    <h6 class=\"title\">{{ product.name | titlecase }}</h6>\r\n\r\n    <div class=\"sku\" [innerHTML]=\"product.shortdetails\"></div>\r\n\r\n    @if (shouldShowPrice) {\r\n    <app-price-ec [price]=\"product.price\" [saleprice]=\"product.saleprice\" class=\"\" />\r\n    }\r\n\r\n    <div class=\"fixBottom\">\r\n\r\n        <!-- Botón de acciones -->\r\n        <!-- <ng-container *ecProductStock=\"product; else noStock\"> -->\r\n        <!-- Cuando no tiene marca especial o es de tipo 'standard' -->\r\n        @if (!product.special_mark || product.special_mark.length === 0 || product.special_mark[0]?.type ===\r\n        'standard') {\r\n        <button class=\"btn standard\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n            <ng-container *ngIf=\"isCollection; else normalText\">\r\n                <span> {{(\"buy\" | translate) | uppercase}} </span>\r\n            </ng-container>\r\n            <ng-template #normalText>\r\n                {{(\"buy\" | translate) | uppercase}}\r\n            </ng-template>\r\n        </button>\r\n        }@else {\r\n        <!-- Caso 1: Agotado o Disponible muy pronto -->\r\n        @if (product.special_mark[0]?.type === 'out_of_stock' || product.special_mark[0]?.type === 'coming_soon') {\r\n        <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n            @if(isCollection){\r\n            <span>{{ product.special_mark[0].name | uppercase }} </span>\r\n            }@else {\r\n            {{ product.special_mark[0]?.name | uppercase }}\r\n            }\r\n        </button>}\r\n        <!-- Caso 2: Contacto por WhatsApp -->\r\n        @if (product.special_mark[0].type === 'whatsapp_contact') {\r\n        <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\"\r\n            (click)=\"openWhatsApp(product.special_mark[0]?.whatsappContact)\">\r\n            @if(isCollection){\r\n            <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n            }@else {\r\n            {{ product.special_mark[0]?.name | uppercase }}\r\n            }\r\n\r\n        </button>\r\n        }\r\n        <!-- Caso 3: Solicitar más información -->\r\n        @if (product.special_mark[0]?.type === 'more_info') {\r\n        <button class=\"btn\" [ngClass]=\"isCollection ? 'px-2 w-100 d-sm-block' : 'py-2 px-4'\">\r\n            @if(isCollection){\r\n            <span>{{ product.special_mark[0]?.name | uppercase }}</span>\r\n            }@else {\r\n            {{ product.special_mark[0]?.name | uppercase }}\r\n            }\r\n        </button>\r\n    }\r\n        }\r\n    </div>\r\n</a>"]}
|
|
@@ -1,11 +1,12 @@
|
|
|
1
|
-
import { Component, inject, Input, CUSTOM_ELEMENTS_SCHEMA, signal, PLATFORM_ID
|
|
1
|
+
import { Component, inject, Input, CUSTOM_ELEMENTS_SCHEMA, signal, PLATFORM_ID } from '@angular/core';
|
|
2
2
|
import { AnalyticsService, ProductsService } from '../../ec-services';
|
|
3
3
|
import { Router } from '@angular/router';
|
|
4
4
|
import { BehaviorSubject, take } from 'rxjs';
|
|
5
5
|
import { AsyncPipe, CommonModule, isPlatformBrowser } from '@angular/common';
|
|
6
6
|
import { ProductEcComponent } from '../product-ec/product-ec.component';
|
|
7
|
+
import { BlockEcComponent } from '../abstractions-components';
|
|
7
8
|
import * as i0 from "@angular/core";
|
|
8
|
-
export class RelatedProductsEcComponent {
|
|
9
|
+
export class RelatedProductsEcComponent extends BlockEcComponent {
|
|
9
10
|
_productsService = inject(ProductsService);
|
|
10
11
|
_analyticsService = inject(AnalyticsService);
|
|
11
12
|
_router = inject(Router);
|
|
@@ -23,10 +24,12 @@ export class RelatedProductsEcComponent {
|
|
|
23
24
|
_relatedProductsSubject = new BehaviorSubject([]);
|
|
24
25
|
relatedProducts$ = this._relatedProductsSubject.asObservable();
|
|
25
26
|
keywordsToCompare = ['compare', 'comparar', 'comparacion', 'comparación', 'compared'];
|
|
26
|
-
|
|
27
|
+
swiperOptions = () => {
|
|
27
28
|
return {
|
|
28
29
|
navigation: true,
|
|
29
30
|
loop: true,
|
|
31
|
+
autoHeight: true,
|
|
32
|
+
autoplay: true,
|
|
30
33
|
breakpoints: {
|
|
31
34
|
0: {
|
|
32
35
|
slidesPerView: 1
|
|
@@ -35,10 +38,10 @@ export class RelatedProductsEcComponent {
|
|
|
35
38
|
slidesPerView: 2
|
|
36
39
|
},
|
|
37
40
|
768: {
|
|
38
|
-
slidesPerView:
|
|
41
|
+
slidesPerView: 2,
|
|
39
42
|
},
|
|
40
43
|
992: {
|
|
41
|
-
slidesPerView:
|
|
44
|
+
slidesPerView: 3,
|
|
42
45
|
},
|
|
43
46
|
1200: {
|
|
44
47
|
slidesPerView: 4,
|
|
@@ -47,18 +50,11 @@ export class RelatedProductsEcComponent {
|
|
|
47
50
|
};
|
|
48
51
|
};
|
|
49
52
|
constructor() {
|
|
53
|
+
super();
|
|
50
54
|
if (isPlatformBrowser(this.platformId)) {
|
|
51
55
|
this.document = document;
|
|
52
56
|
}
|
|
53
|
-
|
|
54
|
-
const swiperElemConstructor = this.document?.querySelector('#relatedProductsSwiper');
|
|
55
|
-
console.log('swiperElemConstructor', swiperElemConstructor);
|
|
56
|
-
if (swiperElemConstructor) {
|
|
57
|
-
Object.assign(swiperElemConstructor, this.swiperRelatedOptions());
|
|
58
|
-
this.swiperElement.set(swiperElemConstructor);
|
|
59
|
-
this.swiperElement()?.initialize();
|
|
60
|
-
}
|
|
61
|
-
});
|
|
57
|
+
// Swiper initialization moved to initSwiper() method called after data loads
|
|
62
58
|
}
|
|
63
59
|
ngOnInit() {
|
|
64
60
|
console.log(this.productID);
|
|
@@ -70,8 +66,23 @@ export class RelatedProductsEcComponent {
|
|
|
70
66
|
const relatedProducts = res.filter((elem) => !this.includeKeyword(elem.title.toLowerCase()));
|
|
71
67
|
this._relatedProductsSubject.next(relatedProducts);
|
|
72
68
|
res.map((products) => this._analyticsService.callEvent('view_item_list', { products: products.items, item_list_name: products.title || 'Related Products', item_list_id: products.id || 'related-products' }));
|
|
69
|
+
// Inicializar swiper después de que los datos estén disponibles
|
|
70
|
+
setTimeout(() => {
|
|
71
|
+
this.initSwiper();
|
|
72
|
+
}, 100);
|
|
73
73
|
});
|
|
74
74
|
}
|
|
75
|
+
initSwiper() {
|
|
76
|
+
if (isPlatformBrowser(this.platformId)) {
|
|
77
|
+
const swiperElemConstructor = this.document?.querySelector('#relatedProductsSwiper');
|
|
78
|
+
console.log('swiperElemConstructor', swiperElemConstructor);
|
|
79
|
+
if (swiperElemConstructor) {
|
|
80
|
+
Object.assign(swiperElemConstructor, this.swiperOptions());
|
|
81
|
+
this.swiperElement.set(swiperElemConstructor);
|
|
82
|
+
this.swiperElement()?.initialize();
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
75
86
|
includeKeyword = (word) => {
|
|
76
87
|
let result = false;
|
|
77
88
|
this.keywordsToCompare.forEach(keyword => {
|
|
@@ -80,7 +91,7 @@ export class RelatedProductsEcComponent {
|
|
|
80
91
|
return result;
|
|
81
92
|
};
|
|
82
93
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RelatedProductsEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
83
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RelatedProductsEcComponent, isStandalone: true, selector: "app-related-products-ec", inputs: { appProduct: "appProduct", productID: "productID" }, ngImport: i0, template: "\r\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }] });
|
|
94
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RelatedProductsEcComponent, isStandalone: true, selector: "app-related-products-ec", inputs: { appProduct: "appProduct", productID: "productID" }, usesInheritance: true, ngImport: i0, template: "\r\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }] });
|
|
84
95
|
}
|
|
85
96
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RelatedProductsEcComponent, decorators: [{
|
|
86
97
|
type: Component,
|
|
@@ -93,4 +104,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
93
104
|
required: true
|
|
94
105
|
}]
|
|
95
106
|
}] } });
|
|
96
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"related-products-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/related-products-ec/related-products-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/related-products-ec/related-products-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, TemplateRef, CUSTOM_ELEMENTS_SCHEMA, signal, PLATFORM_ID, afterNextRender } from '@angular/core';\r\nimport { AnalyticsService, ProductsService } from '../../ec-services';\r\nimport { Router } from '@angular/router';\r\nimport { BehaviorSubject, Observable, take } from 'rxjs';\r\nimport { AsyncPipe, CommonModule, isPlatformBrowser } from '@angular/common';\r\nimport { Product } from '../../interfaces';\r\nimport { ProductEcComponent } from '../product-ec/product-ec.component';\r\nimport { SwiperContainer } from 'swiper/element';\r\nimport { SwiperOptions } from 'swiper/types';\r\n\r\n@Component({\r\n    selector: 'app-related-products-ec',\r\n    standalone: true,\r\n    imports: [AsyncPipe, ProductEcComponent, CommonModule],\r\n    templateUrl: './related-products-ec.component.html',\r\n    styleUrl: './related-products-ec.component.scss',\r\n    schemas: [CUSTOM_ELEMENTS_SCHEMA]\r\n})\r\nexport class RelatedProductsEcComponent {\r\n    private _productsService: ProductsService = inject(ProductsService)\r\n    private _analyticsService: AnalyticsService = inject(AnalyticsService)\r\n    private _router: Router = inject(Router)\r\n    \t/**\r\n\t * Signal utlizado para guarda el contenedor del carrusel\r\n\t */\r\n\tswiperElement = signal<SwiperContainer | null>(null)\r\n    /**\r\n     * Input que recibe un template para el producto.\r\n     */\r\n    @Input() appProduct: TemplateRef<any> | undefined;\r\n    @Input({\r\n        required: true\r\n    }) productID: any;\r\n\r\n        private document?: Document;\r\n        private platformId: any = inject(PLATFORM_ID)\r\n    \r\n    private _relatedProductsSubject:BehaviorSubject<Product[]> =  new BehaviorSubject<Product[]>([])\r\n    public relatedProducts$:Observable<any[]> = this._relatedProductsSubject.asObservable();\r\n    private keywordsToCompare:string[] = ['compare', 'comparar', 'comparacion', 'comparación', 'compared'];\r\n    swiperRelatedOptions = (): SwiperOptions => {\r\n        return {\r\n            navigation: true,\r\n            loop: true,\r\n            breakpoints: {\r\n                0: {\r\n                    slidesPerView: 1\r\n                },\r\n                576: {\r\n                    slidesPerView: 2\r\n                },\r\n                768: {\r\n                    slidesPerView: 3,\r\n                },\r\n                992: {\r\n                    slidesPerView: 4,\r\n                },\r\n                1200: {\r\n                    slidesPerView: 4,\r\n                }\r\n            }\r\n        }\r\n    }\r\n    constructor() {\r\n        if (isPlatformBrowser(this.platformId)) {\r\n            this.document = document\r\n        }\r\n        afterNextRender(() => {\r\n            const swiperElemConstructor = this.document?.querySelector('#relatedProductsSwiper')\r\n            console.log('swiperElemConstructor', swiperElemConstructor)\r\n            if (swiperElemConstructor) {\r\n                Object.assign(swiperElemConstructor!, this.swiperRelatedOptions())\r\n                this.swiperElement.set(swiperElemConstructor as SwiperContainer)\r\n                this.swiperElement()?.initialize();\r\n            }\r\n        })\r\n    }\r\n\r\n    ngOnInit() {\r\n        console.log(this.productID)\r\n        this.load(this.productID)\r\n    }\r\n\r\n    private load(product_id:any) {\r\n        this._productsService.getRelatedProducts(product_id).pipe(take(1)).subscribe(\r\n            res => {\r\n                console.log('response',res);\r\n                const relatedProducts:any = res.filter((elem:any) => !this.includeKeyword(elem.title.toLowerCase()))\r\n                this._relatedProductsSubject.next(relatedProducts)\r\n                res.map((products:any) => this._analyticsService.callEvent('view_item_list', { products: products.items, item_list_name: products.title || 'Related Products', item_list_id: products.id || 'related-products' }))\r\n            }\r\n        )\r\n    }\r\n\r\n    includeKeyword = (word:string) => {\r\n        let result = false\r\n        this.keywordsToCompare.forEach( keyword => {\r\n            word.includes(keyword) ? result = true : null\r\n        })\r\n        return result\r\n    }\r\n}\r\n\r\n","\r\n"]}
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"related-products-ec.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/related-products-ec/related-products-ec.component.ts","../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/related-products-ec/related-products-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, TemplateRef, CUSTOM_ELEMENTS_SCHEMA, signal, PLATFORM_ID, afterNextRender } from '@angular/core';\r\nimport { AnalyticsService, ProductsService } from '../../ec-services';\r\nimport { Router } from '@angular/router';\r\nimport { BehaviorSubject, Observable, take } from 'rxjs';\r\nimport { AsyncPipe, CommonModule, isPlatformBrowser } from '@angular/common';\r\nimport { Product } from '../../interfaces';\r\nimport { ProductEcComponent } from '../product-ec/product-ec.component';\r\nimport { SwiperContainer } from 'swiper/element';\r\nimport { SwiperOptions } from 'swiper/types';\r\nimport { BlockEcComponent } from '../abstractions-components';\r\n\r\n@Component({\r\n    selector: 'app-related-products-ec',\r\n    standalone: true,\r\n    imports: [AsyncPipe, ProductEcComponent, CommonModule],\r\n    templateUrl: './related-products-ec.component.html',\r\n    styleUrl: './related-products-ec.component.scss',\r\n    schemas: [CUSTOM_ELEMENTS_SCHEMA]\r\n})\r\nexport class RelatedProductsEcComponent extends BlockEcComponent{\r\n    private _productsService: ProductsService = inject(ProductsService)\r\n    private _analyticsService: AnalyticsService = inject(AnalyticsService)\r\n    private _router: Router = inject(Router)\r\n    \t/**\r\n\t * Signal utlizado para guarda el contenedor del carrusel\r\n\t */\r\n\tswiperElement = signal<SwiperContainer | null>(null)\r\n    /**\r\n     * Input que recibe un template para el producto.\r\n     */\r\n    @Input() appProduct: TemplateRef<any> | undefined;\r\n    @Input({\r\n        required: true\r\n    }) productID: any;\r\n\r\n        private document?: Document;\r\n        private platformId: any = inject(PLATFORM_ID)\r\n    \r\n    private _relatedProductsSubject:BehaviorSubject<Product[]> =  new BehaviorSubject<Product[]>([])\r\n    public relatedProducts$:Observable<any[]> = this._relatedProductsSubject.asObservable();\r\n    private keywordsToCompare:string[] = ['compare', 'comparar', 'comparacion', 'comparación', 'compared'];\r\n    override swiperOptions = (): SwiperOptions => {\r\n        return {\r\n            navigation: true,\r\n            loop: true,\r\n            autoHeight: true,\r\n            autoplay: true,\r\n            breakpoints: {\r\n                0: {\r\n                    slidesPerView: 1\r\n                },\r\n                576: {\r\n                    slidesPerView: 2\r\n                },\r\n                768: {\r\n                    slidesPerView: 2,\r\n                },\r\n                992: {\r\n                    slidesPerView: 3,\r\n                },\r\n                1200: {\r\n                    slidesPerView: 4,\r\n                }\r\n            }\r\n        }\r\n    }\r\n    constructor() {\r\n        super()\r\n        if (isPlatformBrowser(this.platformId)) {\r\n            this.document = document\r\n        }\r\n        // Swiper initialization moved to initSwiper() method called after data loads\r\n    }\r\n\r\n    ngOnInit() {\r\n        console.log(this.productID)\r\n        this.load(this.productID)\r\n    }\r\n\r\n    private load(product_id:any) {\r\n        this._productsService.getRelatedProducts(product_id).pipe(take(1)).subscribe(\r\n            res => {\r\n                console.log('response',res);\r\n                const relatedProducts:any = res.filter((elem:any) => !this.includeKeyword(elem.title.toLowerCase()))\r\n                this._relatedProductsSubject.next(relatedProducts)\r\n                res.map((products:any) => this._analyticsService.callEvent('view_item_list', { products: products.items, item_list_name: products.title || 'Related Products', item_list_id: products.id || 'related-products' }))\r\n                \r\n                // Inicializar swiper después de que los datos estén disponibles\r\n                setTimeout(() => {\r\n                    this.initSwiper();\r\n                }, 100);\r\n            }\r\n        )\r\n    }\r\n\r\n    private initSwiper() {\r\n        if (isPlatformBrowser(this.platformId)) {\r\n            const swiperElemConstructor = this.document?.querySelector('#relatedProductsSwiper')\r\n            console.log('swiperElemConstructor', swiperElemConstructor)\r\n            if (swiperElemConstructor) {\r\n                Object.assign(swiperElemConstructor!, this.swiperOptions())\r\n                this.swiperElement.set(swiperElemConstructor as SwiperContainer)\r\n                this.swiperElement()?.initialize();\r\n            }\r\n        }\r\n    }\r\n\r\n    includeKeyword = (word:string) => {\r\n        let result = false\r\n        this.keywordsToCompare.forEach( keyword => {\r\n            word.includes(keyword) ? result = true : null\r\n        })\r\n        return result\r\n    }\r\n}\r\n\r\n","\r\n"]}
|
|
@@ -1,27 +1,90 @@
|
|
|
1
|
-
import { Component, Input } from '@angular/core';
|
|
1
|
+
import { Component, inject, Input } from '@angular/core';
|
|
2
2
|
import { EcCurrencySymbolPipe } from "../../../ec-pipe/ec-currency-symbol.pipe";
|
|
3
|
+
import { ChannelService } from '../../../ec-services';
|
|
4
|
+
import { TranslateModule } from '@ngx-translate/core';
|
|
5
|
+
import { CommonModule } from '@angular/common';
|
|
3
6
|
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/common";
|
|
8
|
+
import * as i2 from "@ngx-translate/core";
|
|
4
9
|
/**
|
|
5
10
|
* Componente para manejar la vista del precio de un producto.
|
|
6
11
|
*/
|
|
7
12
|
export class PriceEcComponent {
|
|
8
|
-
|
|
9
|
-
* Precio de un producto.
|
|
10
|
-
*/
|
|
13
|
+
// Precios
|
|
11
14
|
price;
|
|
12
|
-
/**
|
|
13
|
-
* Precio de venta de un producto.
|
|
14
|
-
*/
|
|
15
15
|
saleprice;
|
|
16
|
+
basePrice;
|
|
17
|
+
taxeAmount;
|
|
18
|
+
taxes = { Name: '', taxeAmount: 0 };
|
|
19
|
+
// Opciones
|
|
20
|
+
priceSize = false;
|
|
21
|
+
showTaxLegendOnly = false;
|
|
22
|
+
disableTaxInfo = false;
|
|
23
|
+
// Templates personalizados (proyección)
|
|
24
|
+
customPriceTemplate = null;
|
|
25
|
+
customSalePriceTemplate = null;
|
|
26
|
+
customSimplePriceTemplate = null;
|
|
27
|
+
customSimpleSalePriceTemplate = null;
|
|
28
|
+
customTaxTemplate = null;
|
|
29
|
+
customOnlyTaxLabelTemplate = null;
|
|
30
|
+
_channelService = inject(ChannelService);
|
|
31
|
+
showTaxLegend = false;
|
|
32
|
+
hideTaxes = false;
|
|
33
|
+
ngOnInit() {
|
|
34
|
+
this._channelService.channel$.subscribe(cfg => {
|
|
35
|
+
this.showTaxLegend = !!cfg.showTaxLegend;
|
|
36
|
+
this.hideTaxes = !!cfg.hideTaxes;
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
/** Determina si mostrar la sección de impuestos */
|
|
40
|
+
get shouldShowTaxes() {
|
|
41
|
+
return !this.disableTaxInfo
|
|
42
|
+
&& this.showTaxLegend
|
|
43
|
+
&& !this.hideTaxes
|
|
44
|
+
&& !!this.basePrice
|
|
45
|
+
&& !!this.taxeAmount;
|
|
46
|
+
}
|
|
47
|
+
/** Clase CSS base según tamaño */
|
|
48
|
+
get priceClass() {
|
|
49
|
+
return this.priceSize ? 'price-sm' : 'price';
|
|
50
|
+
}
|
|
51
|
+
/** Helper para detectar rangos */
|
|
52
|
+
hasRange(value) {
|
|
53
|
+
return !!value && value.split(' - ').length === 2;
|
|
54
|
+
}
|
|
16
55
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PriceEcComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
17
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
56
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: PriceEcComponent, isStandalone: true, selector: "app-price-ec", inputs: { price: "price", saleprice: "saleprice", basePrice: "basePrice", taxeAmount: "taxeAmount", taxes: "taxes", priceSize: "priceSize", showTaxLegendOnly: "showTaxLegendOnly", disableTaxInfo: "disableTaxInfo", customPriceTemplate: "customPriceTemplate", customSalePriceTemplate: "customSalePriceTemplate", customSimplePriceTemplate: "customSimplePriceTemplate", customSimpleSalePriceTemplate: "customSimpleSalePriceTemplate", customTaxTemplate: "customTaxTemplate", customOnlyTaxLabelTemplate: "customOnlyTaxLabelTemplate" }, ngImport: i0, template: "<ng-container *ngIf=\"customSalePriceTemplate || customPriceTemplate; else defaultPriceBlock\">\r\n <ng-container *ngTemplateOutlet=\"customSalePriceTemplate || customPriceTemplate\">\r\n </ng-container>\r\n</ng-container>\r\n\r\n<ng-template #defaultPriceBlock>\r\n <div class=\"price\">\r\n <!-- Con precio de oferta -->\r\n <div *ngIf=\"saleprice; else onlyPriceBlock\" class=\"line-height-custom\">\r\n\r\n <!-- Precio original como rango o tachado simple -->\r\n <div *ngIf=\"hasRange(price); else simplePriceDel\" class=\"price-whithSaleprice\">\r\n <del class=\"\">\r\n {{\r\n price!.split(' - ')[0] | ecCurrencySymbol\r\n }}\r\n {{\r\n price!.split(' - ')[1] | ecCurrencySymbol\r\n }}\r\n </del>\r\n </div>\r\n\r\n <!-- Oferta como rango o simple -->\r\n <div *ngIf=\"hasRange(saleprice); else simpleSalePrice\" class=\"\">\r\n {{\r\n saleprice!.split(' - ')[0] | ecCurrencySymbol\r\n }}\r\n {{\r\n saleprice!.split(' - ')[1] | ecCurrencySymbol\r\n }}\r\n </div>\r\n </div>\r\n\r\n <!-- S\u00F3lo precio sin oferta -->\r\n <ng-template #onlyPriceBlock>\r\n <div *ngIf=\"hasRange(price); else simplePrice\" class=\"price-onlyPrice\">\r\n {{\r\n price!.split(' - ')[0] | ecCurrencySymbol\r\n }} -\r\n {{\r\n price!.split(' - ')[1] | ecCurrencySymbol\r\n }}\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Fallback simple price -->\r\n <ng-template #simplePrice>\r\n <ng-container *ngIf=\"customSimplePriceTemplate; else fallbackSimplePrice\">\r\n <ng-container *ngTemplateOutlet=\"customSimplePriceTemplate\"></ng-container>\r\n </ng-container>\r\n </ng-template>\r\n <ng-template #fallbackSimplePrice>\r\n <del> </del>\r\n <div class=\"price-simplePrice\">\r\n {{ price | ecCurrencySymbol }}\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Fallback simple price-del -->\r\n <ng-template #simplePriceDel>\r\n <div class=\"price-simpleDel\">\r\n <del class=\"\">{{ price | ecCurrencySymbol }}</del>\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Fallback simple saleprice -->\r\n <ng-template #simpleSalePrice>\r\n <ng-container *ngIf=\"customSimpleSalePriceTemplate; else fallbackSimpleSalePrice\">\r\n <ng-container *ngTemplateOutlet=\"customSimpleSalePriceTemplate\"></ng-container>\r\n </ng-container>\r\n </ng-template>\r\n <ng-template #fallbackSimpleSalePrice>\r\n <div class=\"price-simpleSaleprice\">\r\n {{ saleprice | ecCurrencySymbol }}\r\n </div>\r\n </ng-template>\r\n </div>\r\n</ng-template>\r\n\r\n<!-- Secci\u00F3n de impuestos -->\r\n<ng-container *ngIf=\"shouldShowTaxes\">\r\n <!-- S\u00F3lo leyenda -->\r\n <ng-container *ngIf=\"showTaxLegendOnly; else detailedTaxBlock\">\r\n <ng-container *ngIf=\"customOnlyTaxLabelTemplate; else defaultOnlyTaxLabel\">\r\n <ng-container *ngTemplateOutlet=\"customOnlyTaxLabelTemplate\"></ng-container>\r\n </ng-container>\r\n <ng-template #defaultOnlyTaxLabel>\r\n <p class=\"taxes-title\">\r\n {{ 'price-without-national-taxes' | translate }}:\r\n {{ basePrice! | ecCurrencySymbol }}\r\n </p>\r\n </ng-template>\r\n </ng-container>\r\n\r\n <!-- Detalle impuestos -->\r\n <ng-template #detailedTaxBlock>\r\n <ng-container *ngIf=\"customTaxTemplate; else defaultTaxBlock\">\r\n <ng-container *ngTemplateOutlet=\"customTaxTemplate\"></ng-container>\r\n </ng-container>\r\n </ng-template>\r\n</ng-container>\r\n\r\n<ng-template #defaultTaxBlock>\r\n <div class=\"taxes-section\">\r\n <p class=\"taxes-title\">\r\n {{ 'price-without-national-taxes' | translate }}:\r\n {{ basePrice! | ecCurrencySymbol }}\r\n </p>\r\n <ul class=\"taxes-list\">\r\n <li>\r\n {{ taxes.Name }}: {{ taxeAmount | ecCurrencySymbol }}\r\n </li>\r\n </ul>\r\n </div>\r\n</ng-template>", styles: [".price-sm{font-size:13px}.price{font-size:18px}.line-height-custom{line-height:1.2}.lnth{text-decoration:line-through;color:gray}.taxes-section{margin-top:.5rem;border-radius:.5rem;font-size:.95rem;color:#333;line-height:1.4;max-width:400px}.taxes-title{font-weight:500;margin-bottom:.2rem;font-size:.7rem;color:#222}.taxes-list{list-style:none;padding:0;margin:0}.taxes-list li{display:flex;justify-content:space-between;margin-bottom:.1rem;font-size:.65rem}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: EcCurrencySymbolPipe, name: "ecCurrencySymbol" }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }] });
|
|
18
57
|
}
|
|
19
58
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: PriceEcComponent, decorators: [{
|
|
20
59
|
type: Component,
|
|
21
|
-
args: [{ selector: 'app-price-ec', standalone: true, imports: [EcCurrencySymbolPipe], template: " <
|
|
60
|
+
args: [{ selector: 'app-price-ec', standalone: true, imports: [CommonModule, EcCurrencySymbolPipe, TranslateModule], template: "<ng-container *ngIf=\"customSalePriceTemplate || customPriceTemplate; else defaultPriceBlock\">\r\n <ng-container *ngTemplateOutlet=\"customSalePriceTemplate || customPriceTemplate\">\r\n </ng-container>\r\n</ng-container>\r\n\r\n<ng-template #defaultPriceBlock>\r\n <div class=\"price\">\r\n <!-- Con precio de oferta -->\r\n <div *ngIf=\"saleprice; else onlyPriceBlock\" class=\"line-height-custom\">\r\n\r\n <!-- Precio original como rango o tachado simple -->\r\n <div *ngIf=\"hasRange(price); else simplePriceDel\" class=\"price-whithSaleprice\">\r\n <del class=\"\">\r\n {{\r\n price!.split(' - ')[0] | ecCurrencySymbol\r\n }}\r\n {{\r\n price!.split(' - ')[1] | ecCurrencySymbol\r\n }}\r\n </del>\r\n </div>\r\n\r\n <!-- Oferta como rango o simple -->\r\n <div *ngIf=\"hasRange(saleprice); else simpleSalePrice\" class=\"\">\r\n {{\r\n saleprice!.split(' - ')[0] | ecCurrencySymbol\r\n }}\r\n {{\r\n saleprice!.split(' - ')[1] | ecCurrencySymbol\r\n }}\r\n </div>\r\n </div>\r\n\r\n <!-- S\u00F3lo precio sin oferta -->\r\n <ng-template #onlyPriceBlock>\r\n <div *ngIf=\"hasRange(price); else simplePrice\" class=\"price-onlyPrice\">\r\n {{\r\n price!.split(' - ')[0] | ecCurrencySymbol\r\n }} -\r\n {{\r\n price!.split(' - ')[1] | ecCurrencySymbol\r\n }}\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Fallback simple price -->\r\n <ng-template #simplePrice>\r\n <ng-container *ngIf=\"customSimplePriceTemplate; else fallbackSimplePrice\">\r\n <ng-container *ngTemplateOutlet=\"customSimplePriceTemplate\"></ng-container>\r\n </ng-container>\r\n </ng-template>\r\n <ng-template #fallbackSimplePrice>\r\n <del> </del>\r\n <div class=\"price-simplePrice\">\r\n {{ price | ecCurrencySymbol }}\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Fallback simple price-del -->\r\n <ng-template #simplePriceDel>\r\n <div class=\"price-simpleDel\">\r\n <del class=\"\">{{ price | ecCurrencySymbol }}</del>\r\n </div>\r\n </ng-template>\r\n\r\n <!-- Fallback simple saleprice -->\r\n <ng-template #simpleSalePrice>\r\n <ng-container *ngIf=\"customSimpleSalePriceTemplate; else fallbackSimpleSalePrice\">\r\n <ng-container *ngTemplateOutlet=\"customSimpleSalePriceTemplate\"></ng-container>\r\n </ng-container>\r\n </ng-template>\r\n <ng-template #fallbackSimpleSalePrice>\r\n <div class=\"price-simpleSaleprice\">\r\n {{ saleprice | ecCurrencySymbol }}\r\n </div>\r\n </ng-template>\r\n </div>\r\n</ng-template>\r\n\r\n<!-- Secci\u00F3n de impuestos -->\r\n<ng-container *ngIf=\"shouldShowTaxes\">\r\n <!-- S\u00F3lo leyenda -->\r\n <ng-container *ngIf=\"showTaxLegendOnly; else detailedTaxBlock\">\r\n <ng-container *ngIf=\"customOnlyTaxLabelTemplate; else defaultOnlyTaxLabel\">\r\n <ng-container *ngTemplateOutlet=\"customOnlyTaxLabelTemplate\"></ng-container>\r\n </ng-container>\r\n <ng-template #defaultOnlyTaxLabel>\r\n <p class=\"taxes-title\">\r\n {{ 'price-without-national-taxes' | translate }}:\r\n {{ basePrice! | ecCurrencySymbol }}\r\n </p>\r\n </ng-template>\r\n </ng-container>\r\n\r\n <!-- Detalle impuestos -->\r\n <ng-template #detailedTaxBlock>\r\n <ng-container *ngIf=\"customTaxTemplate; else defaultTaxBlock\">\r\n <ng-container *ngTemplateOutlet=\"customTaxTemplate\"></ng-container>\r\n </ng-container>\r\n </ng-template>\r\n</ng-container>\r\n\r\n<ng-template #defaultTaxBlock>\r\n <div class=\"taxes-section\">\r\n <p class=\"taxes-title\">\r\n {{ 'price-without-national-taxes' | translate }}:\r\n {{ basePrice! | ecCurrencySymbol }}\r\n </p>\r\n <ul class=\"taxes-list\">\r\n <li>\r\n {{ taxes.Name }}: {{ taxeAmount | ecCurrencySymbol }}\r\n </li>\r\n </ul>\r\n </div>\r\n</ng-template>", styles: [".price-sm{font-size:13px}.price{font-size:18px}.line-height-custom{line-height:1.2}.lnth{text-decoration:line-through;color:gray}.taxes-section{margin-top:.5rem;border-radius:.5rem;font-size:.95rem;color:#333;line-height:1.4;max-width:400px}.taxes-title{font-weight:500;margin-bottom:.2rem;font-size:.7rem;color:#222}.taxes-list{list-style:none;padding:0;margin:0}.taxes-list li{display:flex;justify-content:space-between;margin-bottom:.1rem;font-size:.65rem}\n"] }]
|
|
22
61
|
}], propDecorators: { price: [{
|
|
23
62
|
type: Input
|
|
24
63
|
}], saleprice: [{
|
|
25
64
|
type: Input
|
|
65
|
+
}], basePrice: [{
|
|
66
|
+
type: Input
|
|
67
|
+
}], taxeAmount: [{
|
|
68
|
+
type: Input
|
|
69
|
+
}], taxes: [{
|
|
70
|
+
type: Input
|
|
71
|
+
}], priceSize: [{
|
|
72
|
+
type: Input
|
|
73
|
+
}], showTaxLegendOnly: [{
|
|
74
|
+
type: Input
|
|
75
|
+
}], disableTaxInfo: [{
|
|
76
|
+
type: Input
|
|
77
|
+
}], customPriceTemplate: [{
|
|
78
|
+
type: Input
|
|
79
|
+
}], customSalePriceTemplate: [{
|
|
80
|
+
type: Input
|
|
81
|
+
}], customSimplePriceTemplate: [{
|
|
82
|
+
type: Input
|
|
83
|
+
}], customSimpleSalePriceTemplate: [{
|
|
84
|
+
type: Input
|
|
85
|
+
}], customTaxTemplate: [{
|
|
86
|
+
type: Input
|
|
87
|
+
}], customOnlyTaxLabelTemplate: [{
|
|
88
|
+
type: Input
|
|
26
89
|
}] } });
|
|
27
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpY2UtZWMuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctZWFzeWNvbW1lcmNlLXYxOC9zcmMvbGliL2VjLWNvbXBvbmVudHMvd2lkZ2V0cy1lYy9wcmljZS1lYy9wcmljZS1lYy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvZWMtY29tcG9uZW50cy93aWRnZXRzLWVjL3ByaWNlLWVjL3ByaWNlLWVjLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2pELE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDBDQUEwQyxDQUFDOztBQUNoRjs7R0FFRztBQVFILE1BQU0sT0FBTyxnQkFBZ0I7SUFDM0I7O09BRUc7SUFDSyxLQUFLLENBQU07SUFDbkI7O09BRUc7SUFDSyxTQUFTLENBQU07d0dBUlosZ0JBQWdCOzRGQUFoQixnQkFBZ0IsNEhDWjdCLDhoREFrRFEscUREMUNJLG9CQUFvQjs7NEZBSW5CLGdCQUFnQjtrQkFQNUIsU0FBUzsrQkFDRSxjQUFjLGNBQ1osSUFBSSxXQUNQLENBQUMsb0JBQW9CLENBQUM7OEJBUXZCLEtBQUs7c0JBQWIsS0FBSztnQkFJRyxTQUFTO3NCQUFqQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBFY0N1cnJlbmN5U3ltYm9sUGlwZSB9IGZyb20gXCIuLi8uLi8uLi9lYy1waXBlL2VjLWN1cnJlbmN5LXN5bWJvbC5waXBlXCI7XHJcbi8qKlxyXG4gKiBDb21wb25lbnRlIHBhcmEgbWFuZWphciBsYSB2aXN0YSBkZWwgcHJlY2lvIGRlIHVuIHByb2R1Y3RvLlxyXG4gKi9cclxuQENvbXBvbmVudCh7XHJcbiAgc2VsZWN0b3I6ICdhcHAtcHJpY2UtZWMnLFxyXG4gIHN0YW5kYWxvbmU6IHRydWUsXHJcbiAgaW1wb3J0czogW0VjQ3VycmVuY3lTeW1ib2xQaXBlXSxcclxuICB0ZW1wbGF0ZVVybDogJy4vcHJpY2UtZWMuY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsOiAnLi9wcmljZS1lYy5jb21wb25lbnQuc2NzcydcclxufSlcclxuZXhwb3J0IGNsYXNzIFByaWNlRWNDb21wb25lbnQge1xyXG4gIC8qKlxyXG4gICAqIFByZWNpbyBkZSB1biBwcm9kdWN0by5cclxuICAgKi9cclxuXHRASW5wdXQoKSBwcmljZTogYW55O1xyXG4gIC8qKlxyXG4gICAqIFByZWNpbyBkZSB2ZW50YSBkZSB1biBwcm9kdWN0by5cclxuICAgKi9cclxuXHRASW5wdXQoKSBzYWxlcHJpY2U6IGFueTtcclxufVxyXG4iLCIgPGRpdiBjbGFzcz1cInByaWNlXCI+XHJcbiAgICBAaWYoc2FsZXByaWNlKXtcclxuICAgIDwhLS1XaXRoIHNhbGVwcmljZS0tPlxyXG4gXHJcbiAgICBAaWYocHJpY2UgJiYgcHJpY2Uuc3BsaXQoJyAtICcpLmxlbmd0aCA+IDEpe1xyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJwcmljZS13aGl0aFNhbGVwcmljZVwiPlxyXG4gICAgICAgICAgICA8ZGVsIGNsYXNzPVwiXCI+XHJcbiAgICAgICAgICAgICAgICB7eyBwcmljZS5zcGxpdCgnIC0gJylbMF0gfCBlY0N1cnJlbmN5U3ltYm9sIH19XHJcbiAgICAgICAgICAgICAgICB7eyBwcmljZS5zcGxpdCgnIC0gJylbMV0gfCBlY0N1cnJlbmN5U3ltYm9sIH19XHJcbiAgICAgICAgICAgIDwvZGVsPlxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgfUBlbHNlIHtcclxuICAgICAgICA8IS0tU2ltcGxlIFByaWNlIERlbC0tPlxyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJwcmljZS1zaW1wbGVEZWxcIj5cclxuICAgICAgICAgICAgPGRlbCBjbGFzcz1cIlwiPlxyXG4gICAgICAgICAgICAgICAge3sgcHJpY2UgfCBlY0N1cnJlbmN5U3ltYm9sIH19PC9kZWw+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICB9XHJcbiAgICBAaWYoc2FsZXByaWNlICYmIHNhbGVwcmljZS5zcGxpdCgnIC0gJykubGVuZ3RoID4gMSl7XHJcbiAgICAgICAgPGRpdiBjbGFzcz1cIlwiPlxyXG4gICAgICAgICAgICB7e1xyXG4gICAgICAgICAgICBzYWxlcHJpY2Uuc3BsaXQoJyAtICcpWzBdIHwgZWNDdXJyZW5jeVN5bWJvbFxyXG4gICAgICAgICAgICB9fVxyXG4gICAgICAgICAgICB7e1xyXG4gICAgICAgICAgICBzYWxlcHJpY2Uuc3BsaXQoJyAtICcpWzFdIHwgZWNDdXJyZW5jeVN5bWJvbFxyXG4gICAgICAgICAgICB9fVxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgfSBAZWxzZSB7XHJcbiAgICAgICAgPCEtLVNpbXBsZSBzYWxlcHJpY2UtLT5cclxuICAgICAgICA8ZGl2IGNsYXNzPVwicHJpY2Utc2ltcGxlU2FsZXByaWNlXCI+XHJcbiAgICAgICAgICAgIHt7IHNhbGVwcmljZSB8IGVjQ3VycmVuY3lTeW1ib2wgfX1cclxuICAgICAgIDwvZGl2PlxyXG4gICAgfVxyXG5cclxufUBlbHNlIHtcclxuICAgIDwhLS1Pbmx5IFByaWNlLS0+XHJcblxyXG4gICAgQGlmKHByaWNlICYmIHByaWNlLnNwbGl0KCcgLSAnKS5sZW5ndGggPiAxKXtcclxuICAgICAgICA8ZGl2IGNsYXNzPVwicHJpY2Utb25seVByaWNlXCI+XHJcbiAgICAgICAgICAgIHt7IHByaWNlLnNwbGl0KCcgLSAnKVswXSB8IGVjQ3VycmVuY3lTeW1ib2wgfX0gLVxyXG4gICAgICAgICAgICB7eyBwcmljZS5zcGxpdCgnIC0gJylbMV0gfCBlY0N1cnJlbmN5U3ltYm9sIH19XHJcbiAgICAgICAgPC9kaXY+IFxyXG4gICAgfSBAZWxzZSB7XHJcbiAgICAgICAgPCEtLVNpbXBsZSBwcmljZS0tPlxyXG4gICAgICAgIDxkZWw+Jm5ic3A7PC9kZWw+XHJcbiAgICAgICAgPGRpdiBjbGFzcz1cInByaWNlLXNpbXBsZVByaWNlXCI+XHJcbiAgICAgICAgICAgIHt7IHByaWNlIHwgZWNDdXJyZW5jeVN5bWJvbCB9fVxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgfVxyXG59XHJcbjwvZGl2PiAgIl19
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"price-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/widgets-ec/price-ec/price-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/widgets-ec/price-ec/price-ec.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAuB,MAAM,eAAe,CAAC;AAC9E,OAAO,EAAE,oBAAoB,EAAE,MAAM,0CAA0C,CAAC;AAChF,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;;;;AAC/C;;GAEG;AAQH,MAAM,OAAO,gBAAgB;IAC3B,UAAU;IACD,KAAK,CAAU;IACf,SAAS,CAAU;IACnB,SAAS,CAAU;IACnB,UAAU,CAAU;IACpB,KAAK,GAAyC,EAAE,IAAI,EAAE,EAAE,EAAE,UAAU,EAAE,CAAC,EAAE,CAAC;IAEnF,WAAW;IACF,SAAS,GAAY,KAAK,CAAC;IAC3B,iBAAiB,GAAY,KAAK,CAAC;IACnC,cAAc,GAAY,KAAK,CAAC;IAEzC,wCAAwC;IAC/B,mBAAmB,GAAsC,IAAI,CAAC;IAC9D,uBAAuB,GAAkC,IAAI,CAAC;IAC9D,yBAAyB,GAAgC,IAAI,CAAC;IAC9D,6BAA6B,GAA4B,IAAI,CAAC;IAC9D,iBAAiB,GAAwC,IAAI,CAAC;IAC9D,0BAA0B,GAA+B,IAAI,CAAC;IAE/D,eAAe,GAAmB,MAAM,CAAC,cAAc,CAAC,CAAC;IAC1D,aAAa,GAAG,KAAK,CAAC;IACtB,SAAS,GAAG,KAAK,CAAC;IAEzB,QAAQ;QACN,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE;YAC5C,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,GAAG,CAAC,aAAa,CAAC;YACzC,IAAI,CAAC,SAAS,GAAO,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC;QACvC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,mDAAmD;IACnD,IAAI,eAAe;QACjB,OAAO,CAAC,IAAI,CAAC,cAAc;eACpB,IAAI,CAAC,aAAa;eAClB,CAAC,IAAI,CAAC,SAAS;eACf,CAAC,CAAC,IAAI,CAAC,SAAS;eAChB,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC;IAC3B,CAAC;IAED,kCAAkC;IAClC,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC;IAC/C,CAAC;IAED,kCAAkC;IAClC,QAAQ,CAAC,KAAc;QACrB,OAAO,CAAC,CAAC,KAAK,IAAI,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC;IACpD,CAAC;wGAjDU,gBAAgB;4FAAhB,gBAAgB,0lBCf7B,++IAkHc,sgBDvGF,YAAY,qSAAE,oBAAoB,wDAAE,eAAe;;4FAIlD,gBAAgB;kBAP5B,SAAS;+BACE,cAAc,cACZ,IAAI,WACP,CAAC,YAAY,EAAE,oBAAoB,EAAE,eAAe,CAAC;8BAMrD,KAAK;sBAAb,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,UAAU;sBAAlB,KAAK;gBACG,KAAK;sBAAb,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBACG,iBAAiB;sBAAzB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBAGG,mBAAmB;sBAA3B,KAAK;gBACG,uBAAuB;sBAA/B,KAAK;gBACG,yBAAyB;sBAAjC,KAAK;gBACG,6BAA6B;sBAArC,KAAK;gBACG,iBAAiB;sBAAzB,KAAK;gBACG,0BAA0B;sBAAlC,KAAK","sourcesContent":["import { Component, inject, Input, OnInit, TemplateRef } from '@angular/core';\r\nimport { EcCurrencySymbolPipe } from \"../../../ec-pipe/ec-currency-symbol.pipe\";\r\nimport { ChannelService } from '../../../ec-services';\r\nimport { TranslateModule } from '@ngx-translate/core';\r\nimport { CommonModule } from '@angular/common';\r\n/**\r\n * Componente para manejar la vista del precio de un producto.\r\n */\r\n@Component({\r\n  selector: 'app-price-ec',\r\n  standalone: true,\r\n  imports: [CommonModule, EcCurrencySymbolPipe, TranslateModule],\r\n  templateUrl: './price-ec.component.html',\r\n  styleUrl: './price-ec.component.scss'\r\n})\r\nexport class PriceEcComponent implements OnInit {\r\n  // Precios\r\n  @Input() price!: string;\r\n  @Input() saleprice?: string;\r\n  @Input() basePrice?: string;\r\n  @Input() taxeAmount?: number;\r\n  @Input() taxes: { Name: string; taxeAmount: number } = { Name: '', taxeAmount: 0 };\r\n\r\n  // Opciones\r\n  @Input() priceSize: boolean = false;\r\n  @Input() showTaxLegendOnly: boolean = false;\r\n  @Input() disableTaxInfo: boolean = false;\r\n\r\n  // Templates personalizados (proyección)\r\n  @Input() customPriceTemplate: TemplateRef<any> | null           = null;\r\n  @Input() customSalePriceTemplate: TemplateRef<any> | null       = null;\r\n  @Input() customSimplePriceTemplate: TemplateRef<any> | null     = null;\r\n  @Input() customSimpleSalePriceTemplate: TemplateRef<any> | null = null;\r\n  @Input() customTaxTemplate: TemplateRef<any> | null             = null;\r\n  @Input() customOnlyTaxLabelTemplate: TemplateRef<any> | null    = null;\r\n\r\n  private _channelService: ChannelService = inject(ChannelService);\r\n  public showTaxLegend = false;\r\n  public hideTaxes = false;\r\n\r\n  ngOnInit() {\r\n    this._channelService.channel$.subscribe(cfg => {\r\n      this.showTaxLegend = !!cfg.showTaxLegend;\r\n      this.hideTaxes     = !!cfg.hideTaxes;\r\n    });\r\n  }\r\n\r\n  /** Determina si mostrar la sección de impuestos */\r\n  get shouldShowTaxes(): boolean {    \r\n    return !this.disableTaxInfo\r\n        && this.showTaxLegend\r\n        && !this.hideTaxes\r\n        && !!this.basePrice\r\n        && !!this.taxeAmount;\r\n  }\r\n\r\n  /** Clase CSS base según tamaño */\r\n  get priceClass(): string {\r\n    return this.priceSize ? 'price-sm' : 'price';\r\n  }\r\n\r\n  /** Helper para detectar rangos */\r\n  hasRange(value?: string): boolean {\r\n    return !!value && value.split(' - ').length === 2;\r\n  }\r\n}\r\n","<ng-container *ngIf=\"customSalePriceTemplate || customPriceTemplate; else defaultPriceBlock\">\r\n    <ng-container *ngTemplateOutlet=\"customSalePriceTemplate || customPriceTemplate\">\r\n    </ng-container>\r\n</ng-container>\r\n\r\n<ng-template #defaultPriceBlock>\r\n    <div class=\"price\">\r\n        <!-- Con precio de oferta -->\r\n        <div *ngIf=\"saleprice; else onlyPriceBlock\" class=\"line-height-custom\">\r\n\r\n            <!-- Precio original como rango o tachado simple -->\r\n            <div *ngIf=\"hasRange(price); else simplePriceDel\" class=\"price-whithSaleprice\">\r\n                <del class=\"\">\r\n                    {{\r\n                    price!.split(' - ')[0] | ecCurrencySymbol\r\n                    }}\r\n                    {{\r\n                    price!.split(' - ')[1] | ecCurrencySymbol\r\n                    }}\r\n                </del>\r\n            </div>\r\n\r\n            <!-- Oferta como rango o simple -->\r\n            <div *ngIf=\"hasRange(saleprice); else simpleSalePrice\" class=\"\">\r\n                {{\r\n                saleprice!.split(' - ')[0] | ecCurrencySymbol\r\n                }}\r\n                {{\r\n                saleprice!.split(' - ')[1] | ecCurrencySymbol\r\n                }}\r\n            </div>\r\n        </div>\r\n\r\n        <!-- Sólo precio sin oferta -->\r\n        <ng-template #onlyPriceBlock>\r\n            <div *ngIf=\"hasRange(price); else simplePrice\" class=\"price-onlyPrice\">\r\n                {{\r\n                price!.split(' - ')[0] | ecCurrencySymbol\r\n                }} -\r\n                {{\r\n                price!.split(' - ')[1] | ecCurrencySymbol\r\n                }}\r\n            </div>\r\n        </ng-template>\r\n\r\n        <!-- Fallback simple price -->\r\n        <ng-template #simplePrice>\r\n            <ng-container *ngIf=\"customSimplePriceTemplate; else fallbackSimplePrice\">\r\n                <ng-container *ngTemplateOutlet=\"customSimplePriceTemplate\"></ng-container>\r\n            </ng-container>\r\n        </ng-template>\r\n        <ng-template #fallbackSimplePrice>\r\n            <del>&nbsp;</del>\r\n            <div class=\"price-simplePrice\">\r\n                {{ price | ecCurrencySymbol }}\r\n            </div>\r\n        </ng-template>\r\n\r\n        <!-- Fallback simple price-del -->\r\n        <ng-template #simplePriceDel>\r\n            <div class=\"price-simpleDel\">\r\n                <del class=\"\">{{ price | ecCurrencySymbol }}</del>\r\n            </div>\r\n        </ng-template>\r\n\r\n        <!-- Fallback simple saleprice -->\r\n        <ng-template #simpleSalePrice>\r\n            <ng-container *ngIf=\"customSimpleSalePriceTemplate; else fallbackSimpleSalePrice\">\r\n                <ng-container *ngTemplateOutlet=\"customSimpleSalePriceTemplate\"></ng-container>\r\n            </ng-container>\r\n        </ng-template>\r\n        <ng-template #fallbackSimpleSalePrice>\r\n            <div class=\"price-simpleSaleprice\">\r\n                {{ saleprice | ecCurrencySymbol }}\r\n            </div>\r\n        </ng-template>\r\n    </div>\r\n</ng-template>\r\n\r\n<!-- Sección de impuestos -->\r\n<ng-container *ngIf=\"shouldShowTaxes\">\r\n    <!-- Sólo leyenda -->\r\n    <ng-container *ngIf=\"showTaxLegendOnly; else detailedTaxBlock\">\r\n        <ng-container *ngIf=\"customOnlyTaxLabelTemplate; else defaultOnlyTaxLabel\">\r\n            <ng-container *ngTemplateOutlet=\"customOnlyTaxLabelTemplate\"></ng-container>\r\n        </ng-container>\r\n        <ng-template #defaultOnlyTaxLabel>\r\n            <p class=\"taxes-title\">\r\n                {{ 'price-without-national-taxes' | translate }}:\r\n                {{ basePrice! | ecCurrencySymbol }}\r\n            </p>\r\n        </ng-template>\r\n    </ng-container>\r\n\r\n    <!-- Detalle impuestos -->\r\n    <ng-template #detailedTaxBlock>\r\n        <ng-container *ngIf=\"customTaxTemplate; else defaultTaxBlock\">\r\n            <ng-container *ngTemplateOutlet=\"customTaxTemplate\"></ng-container>\r\n        </ng-container>\r\n    </ng-template>\r\n</ng-container>\r\n\r\n<ng-template #defaultTaxBlock>\r\n    <div class=\"taxes-section\">\r\n        <p class=\"taxes-title\">\r\n            {{ 'price-without-national-taxes' | translate }}:\r\n            {{ basePrice! | ecCurrencySymbol }}\r\n        </p>\r\n        <ul class=\"taxes-list\">\r\n            <li>\r\n                {{ taxes.Name }}: {{ taxeAmount | ecCurrencySymbol }}\r\n            </li>\r\n        </ul>\r\n    </div>\r\n</ng-template>"]}
|