ng-easycommerce-v18 0.3.23-beta.1 → 0.3.23-beta.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/README.md +5 -1
- package/assets/ec-i18n/en.json +1 -0
- package/assets/ec-i18n/es.json +1 -0
- package/esm2022/lib/ec-components/cart-ec/cart-item-ec/cart-item-ec.component.mjs +37 -9
- package/esm2022/lib/ec-components/product-ec/product-ec.component.mjs +54 -6
- package/esm2022/lib/ec-services/cart.service.mjs +22 -4
- package/esm2022/lib/ec-services/product-detail.service.mjs +11 -1
- package/esm2022/lib/interfaces/product.mjs +1 -1
- package/fesm2022/ng-easycommerce-v18.mjs +120 -16
- package/fesm2022/ng-easycommerce-v18.mjs.map +1 -1
- package/lib/ec-components/cart-ec/cart-item-ec/cart-item-ec.component.d.ts +1 -1
- package/lib/ec-components/product-ec/product-ec.component.d.ts +4 -0
- package/lib/ec-services/cart.service.d.ts +1 -1
- package/lib/interfaces/product.d.ts +6 -0
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -1,4 +1,6 @@
|
|
|
1
|
-
# Version 0.3.23-beta.
|
|
1
|
+
# Version 0.3.23-beta.2
|
|
2
|
+
- se hace merge entre la version 0.3.23-beta.1 y 0.3.19-beta.1
|
|
3
|
+
# Version 0.3.23-beta.1
|
|
2
4
|
- Primera direccion en checkout predeterminada por defecto
|
|
3
5
|
# Version 0.3.22
|
|
4
6
|
- Limpieza de importaciones innecesarias en `header-ec.component` para reducir peso y evitar dependencias sin uso.
|
|
@@ -10,6 +12,8 @@
|
|
|
10
12
|
- Unificación del flujo de filtrado: ahora toda la lógica se basa en URL (params + queryParams) y se hidrata desde `FiltersService.hydrateFromRoute()`.
|
|
11
13
|
- Se optimiza el runtime inicial para evitar ciclos innecesarios y cargar filtros antes del primer render.
|
|
12
14
|
- Se centraliza la sincronización de rangos de precio (filters ↔ pagination ↔ products).
|
|
15
|
+
# version 0.3.19-beta.1
|
|
16
|
+
- Se añade en carrito sidebar product detail y producto las validaciones necesarias para que compruebe si hay un maximo , un minimo o multiplo.
|
|
13
17
|
# Version 0.3.17
|
|
14
18
|
- En el `header-ec.component.ts` se hacen public algunos injectores para poder usarlo en el `header.component.ts` de los frontends.
|
|
15
19
|
# Version 0.3.16
|
package/assets/ec-i18n/en.json
CHANGED
|
@@ -176,6 +176,7 @@
|
|
|
176
176
|
"mens-fashion": "mens's fashion",
|
|
177
177
|
"mercado-pago": "Mercado Pago",
|
|
178
178
|
"minimum-items-quantity": "The minimum quantity allowed for this product is {{quantity}}",
|
|
179
|
+
"multiple-quantity-required": "The quantity must be a multiple of {{multiple}}.",
|
|
179
180
|
"mismatched-password": "Password do not match",
|
|
180
181
|
"most-expensive-first": "Most expensive first",
|
|
181
182
|
"my-account": "My Account",
|
package/assets/ec-i18n/es.json
CHANGED
|
@@ -182,6 +182,7 @@
|
|
|
182
182
|
"men-fashion": "Hombres",
|
|
183
183
|
"mercado-pago": "Mercado Pago",
|
|
184
184
|
"minimum-items-quantity": "La cantidad minima de compra del producto es {{quantity}}",
|
|
185
|
+
"multiple-quantity-required": "La cantidad debe ser múltiplo de {{multiple}}",
|
|
185
186
|
"mismatched-password": "Las contraseña no coinciden",
|
|
186
187
|
"most-expensive-first": "Mayor precio",
|
|
187
188
|
"my-account": "Mi cuenta",
|
|
@@ -24,28 +24,56 @@ export class CartItemEcComponent {
|
|
|
24
24
|
// console.log(this.item, this.mediaUrl);
|
|
25
25
|
}
|
|
26
26
|
updateQuantity(stock) {
|
|
27
|
+
const originalQuantity = this.item.quantity; // Guardar cantidad original
|
|
27
28
|
if (this.quantity > 0 && this.quantity <= stock) {
|
|
28
|
-
this._cartService.updateItemQuantity(this.item, this.quantity);
|
|
29
|
+
const success = this._cartService.updateItemQuantity(this.item, this.quantity);
|
|
30
|
+
// Si la validación falló, restaurar la cantidad original
|
|
31
|
+
if (!success) {
|
|
32
|
+
this.quantity = originalQuantity;
|
|
33
|
+
}
|
|
29
34
|
}
|
|
30
35
|
else {
|
|
31
|
-
|
|
36
|
+
// Restaurar cantidad original si está fuera de rango
|
|
37
|
+
this.quantity = originalQuantity;
|
|
32
38
|
this._toastService.show('out-of-stock-actually');
|
|
33
39
|
}
|
|
34
40
|
}
|
|
35
|
-
|
|
41
|
+
plus(stock, value = 1) {
|
|
36
42
|
if (this.isQuantityUpdating)
|
|
37
43
|
return;
|
|
38
44
|
this.isQuantityUpdating = true;
|
|
39
|
-
let
|
|
40
|
-
|
|
45
|
+
let newQuantity = Number(this.quantity) + value; // Forzar a número
|
|
46
|
+
// Verificar stock ANTES de llamar al servicio
|
|
47
|
+
if (newQuantity > stock) {
|
|
48
|
+
this._toastService.show('out-of-stock-actually');
|
|
49
|
+
setTimeout(() => { this.isQuantityUpdating = false; }, 1000);
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
const success = this._cartService.updateItemQuantity(this.item, newQuantity);
|
|
53
|
+
// Solo actualizar el input si la operación fue exitosa
|
|
54
|
+
if (success) {
|
|
55
|
+
this.quantity = newQuantity;
|
|
56
|
+
}
|
|
57
|
+
// Si success es false, NO mostrar mensaje aquí porque ya lo mostró validateQuantity()
|
|
41
58
|
setTimeout(() => { this.isQuantityUpdating = false; }, 1000);
|
|
42
59
|
}
|
|
43
|
-
|
|
60
|
+
less(stock, value = 1) {
|
|
44
61
|
if (this.isQuantityUpdating)
|
|
45
62
|
return;
|
|
46
63
|
this.isQuantityUpdating = true;
|
|
47
|
-
let
|
|
48
|
-
|
|
64
|
+
let newQuantity = Number(this.quantity) - value; // Forzar a número
|
|
65
|
+
// Verificar cantidad mínima ANTES de llamar al servicio
|
|
66
|
+
if (newQuantity <= 0) {
|
|
67
|
+
// No permitir cantidades menores o iguales a 0
|
|
68
|
+
setTimeout(() => { this.isQuantityUpdating = false; }, 1000);
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
const success = this._cartService.updateItemQuantity(this.item, newQuantity);
|
|
72
|
+
// Solo actualizar el input si la operación fue exitosa
|
|
73
|
+
if (success) {
|
|
74
|
+
this.quantity = newQuantity;
|
|
75
|
+
}
|
|
76
|
+
// Si success es false, NO mostrar mensaje aquí porque ya lo mostró validateQuantity()
|
|
49
77
|
setTimeout(() => { this.isQuantityUpdating = false; }, 1000);
|
|
50
78
|
}
|
|
51
79
|
deleteCartItem() {
|
|
@@ -117,4 +145,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
117
145
|
}], inSidebar: [{
|
|
118
146
|
type: Input
|
|
119
147
|
}] } });
|
|
120
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cart-item-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/cart-ec/cart-item-ec/cart-item-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/cart-ec/cart-item-ec/cart-item-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnInit } from '@angular/core';\r\nimport { CoreConstantsService } from '../../../constants';\r\nimport { CartService, ToastService, ParametersService } from '../../../ec-services';\r\nimport { TitleCasePipe } from '@angular/common';\r\nimport { EcCurrencySymbolPipe } from '../../../ec-pipe';\r\nimport { RouterLink } from '@angular/router';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { Observable } from 'rxjs';\r\nimport { Parameter } from '../../../interfaces';\r\n\r\n\r\n@Component({\r\n\tselector: 'app-cart-item-ec',\r\n\tstandalone: true,\r\n\timports: [TitleCasePipe, EcCurrencySymbolPipe, RouterLink, FormsModule],\r\n\ttemplateUrl: './cart-item-ec.component.html',\r\n\tstyleUrl: './cart-item-ec.component.scss'\r\n})\r\nexport class CartItemEcComponent implements OnInit {\r\n\t@Input({\r\n\t\trequired: true\r\n\t}) item: any;\r\n\r\n\t@Input() inSidebar: boolean = false;\r\n\r\n\tprivate _cartService: CartService = inject(CartService);\r\n\tprivate _toastService: ToastService = inject(ToastService);\r\n\tprivate _constants: CoreConstantsService = inject(CoreConstantsService);\r\n\tprivate parametersService = inject(ParametersService)\r\n\r\n\tpublic mediaUrl = this._constants.mediaUrl();\r\n\tpublic quantity = 0;\r\n\tpublic variantsToShow: string[] = ['TALLA', 'COLOR'];\r\n\tpublic updateStock: boolean = false;\r\n\tpublic isQuantityUpdating = false;\r\n\r\n\tngOnInit() {\r\n\t\tthis.quantity = this.item.quantity\r\n\t\t// console.log(this.item, this.mediaUrl);\r\n\t}\r\n\r\n\tupdateQuantity(stock: number) {\r\n\t\tif (this.quantity > 0 && this.quantity <= stock) {\r\n\t\t\tthis._cartService.updateItemQuantity(this.item, this.quantity)\r\n\t\t} else {\r\n\t\t\tthis.quantity = this.item.quantity\r\n\t\t\tthis._toastService.show('out-of-stock-actually');\r\n\t\t}\r\n\t}\r\n\r\n\tless(stock: number, value: number = 1) {\r\n\t\tif (this.isQuantityUpdating) return;\r\n\t\tthis.isQuantityUpdating = true;\r\n\t\tlet quantity = this.item.quantity - value;\r\n\t\tquantity > 0 && quantity <= stock ? this._cartService.updateItemQuantity(this.item, quantity) : this._toastService.show('out-of-stock-actually');\r\n\t\tsetTimeout(() => { this.isQuantityUpdating = false; }, 1000);\r\n\t}\r\n\r\n\tplus(stock: number, value: number = 1) {\r\n\t\tif (this.isQuantityUpdating) return;\r\n\t\tthis.isQuantityUpdating = true;\r\n\t\tlet quantity = this.item.quantity + value;\r\n\t\tquantity > 0 && quantity <= stock ? this._cartService.updateItemQuantity(this.item, quantity) : this._toastService.show('out-of-stock-actually');\r\n\t\tsetTimeout(() => { this.isQuantityUpdating = false; }, 1000);\r\n\t}\r\n\r\n\tdeleteCartItem() {\r\n\t\tthis._cartService.deleteCartItem(this.item)\r\n\t}\r\n\r\n\tgetVariants(product: any): any {\r\n\t\tlet item = product.product\r\n\r\n\t\tif (item.variants && item.variants.length && item.variants[0].options && item.variants[0].options.length) {\r\n\t\t\tlet options = item.variants[0].options\r\n\t\t\toptions = options.filter((option: any) => this.variantsToShow.find((name: any) => option.hasOwnProperty(name)))\r\n\t\t\tlet optionsMap: any = []\r\n\t\t\toptions.map((option: any) => {\r\n\t\t\t\tObject.keys(option).forEach((entry) => {\r\n\t\t\t\t\tlet optionAdded = false;  // Variable para evitar múltiples inserciones\r\n\r\n\t\t\t\t\titem.options.forEach((option2: any) => {\r\n\t\t\t\t\t\toption2.values.forEach((color: any) => {\r\n\t\t\t\t\t\t\tif (color.code == option[entry] && !optionAdded) {\r\n\t\t\t\t\t\t\t\toptionsMap.push({ name: entry, value: option[entry], image: color.image, mobileImage: color.mobileImage });\r\n\t\t\t\t\t\t\t\toptionAdded = true; // Marcar opción agregada\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t});\r\n\t\t\t\t\t});\r\n\t\t\t\t});\r\n\t\t\t\treturn optionsMap;\r\n\t\t\t});\r\n\r\n\t\t\treturn optionsMap;\r\n\r\n\t\t} else {\r\n\t\t\treturn null\r\n\t\t}\r\n\t}\r\n\r\n\tcreateDiscountMessage(saleprice: number, price: number): string {\r\n\t\tif (isNaN(saleprice) || isNaN(price) || saleprice >= price || saleprice <= 0 || price <= 0) {\r\n\t\t\treturn '';\r\n\t\t}\r\n\r\n\t\tlet discountPercentage = Math.round(((price - saleprice) / price) * 100);\r\n\r\n\t\tlet discountMessage = `${discountPercentage}% OFF`;\r\n\t\treturn discountMessage;\r\n\t}\r\n\r\n\tcheckStock(product: any): any {\r\n\t\tfor (const variant of product.product.variants) {\r\n\t\t\tif (product.variant_id === variant.code) {\r\n\t\t\t\tif (variant.stock === 0) {\r\n\t\t\t\t\tthis.updateStock = true;\r\n\t\t\t\t\treturn '0';\r\n\t\t\t\t} else if (product.quantity > variant.stock) {\r\n\t\t\t\t\tthis.updateStock = true;\r\n\t\t\t\t\treturn variant.stock;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn false; // Solo se ejecuta si no se cumple la condición\r\n\t}\r\n\r\n\t// PARAMETROS\r\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters()\r\n\tpublic hasParams = this.parametersService.hasParams\r\n}\r\n","@if(!inSidebar){\r\n<p>cart-item-ec works!</p>\r\n}@else{\r\n\r\n<div class=\"row\">\r\n    <div class=\"col-3\">\r\n        @let product= item.product;\r\n        @if(item.variant_id && product.variants.length>0){\r\n        <img [src]=\"mediaUrl +  product.variants[0].images[0]\" alt=\"\" class=\"img-fluid\">\r\n        }@else{\r\n        <img [src]=\"mediaUrl +  product.picturesdefault[0]\" alt=\"\" class=\"img-fluid\">\r\n        }\r\n    </div>\r\n    <div class=\"col-7\">\r\n        <div class=\"info d-flex flex-column align-items-start\">\r\n            @if (item.product.special_mark?.length > 0 || item.product.saleprice) {\r\n                    <div class=\"marcas\">\r\n                        <img [src]=\"mediaUrl + (item.product.special_mark?.[0]?.images[0] || '')\" alt=\"\">\r\n\r\n                        @if (item.product.saleprice) {\r\n                        <div class=\"tag-dsc\">\r\n                            {{\r\n                            createDiscountMessage(item.product.saleprice,\r\n                            item.product.price)\r\n                            }}\r\n                        </div>\r\n                        }\r\n                    </div>\r\n                    }\r\n            <a class=\"title text-dark text-decoration-none m-0 p-0 h6 mb-0\"\r\n                [routerLink]=\"['/product', item.variant_id]\">{{\r\n                item.product.name | titlecase\r\n                }}</a>\r\n            <div class=\"qty1\">\r\n                <span>{{ item.product.id}}</span>\r\n            </div>\r\n            <div class=\"price h6 fw-bold mb-0 pb-0\">{{ item.product.price | ecCurrencySymbol\r\n                }}</div>\r\n            @if(getVariants(item); as options){\r\n            <div class=\"d-flex align-items-center p-0\">\r\n                @for(option of options; track $index){\r\n                <span class=\"me-1\"> {{option.name | titlecase}}:</span>\r\n                @if(option.name == 'COLOR'){\r\n                <div class=\"p-2 rounded\" [style.background]=\"'#' + option.value\"></div>\r\n                }@else{\r\n                <b>{{option.value}}</b>\r\n                }\r\n                }\r\n            </div>\r\n            }\r\n            <div class=\"campoCantidad mt-2\">\r\n                <div class=\"numero\">\r\n                    <button (click)=\"less(item.product.variants[0]?.stock)\" class=\"btn btn-outline-secondary\"\r\n                        type=\"button\" id=\"button-addon1\">\r\n                        <i class=\"fa fa-minus\" aria-hidden=\"true\"></i>\r\n                    </button>\r\n                    <input type=\"text\" class=\"form-control text-center\" placeholder=\"\"\r\n                        aria-label=\"Example text with button addon\" aria-describedby=\"button-addon1\"\r\n                        [value]=\"item.quantity\" min=\"1\" step=\"1\" [(ngModel)]=\"quantity\"\r\n                        (change)=\"updateQuantity(item.product.variants[0]?.stock)\">\r\n                    <button (click)=\"plus(item.product.variants[0]?.stock)\" class=\"btn btn-outline-secondary\"\r\n                        type=\"button\" id=\"button-addon1\">\r\n                        <i class=\"fa fa-plus\" aria-hidden=\"true\"></i>\r\n                    </button>\r\n                </div>\r\n\r\n            </div>\r\n        </div>\r\n    </div>\r\n    <div class=\"col-2\">\r\n        <a (click)=\"deleteCartItem()\" class=\"btn botBorrar\"><i class=\"fa fa-trash\" aria-hidden=\"true\"></i></a>\r\n    </div>\r\n</div>\r\n\r\n\r\n\r\n\r\n\r\n}"]}
|
|
148
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cart-item-ec.component.js","sourceRoot":"","sources":["../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/cart-ec/cart-item-ec/cart-item-ec.component.ts","../../../../../../../projects/ng-easycommerce-v18/src/lib/ec-components/cart-ec/cart-item-ec/cart-item-ec.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnInit } from '@angular/core';\r\nimport { CoreConstantsService } from '../../../constants';\r\nimport { CartService, ToastService, ParametersService } from '../../../ec-services';\r\nimport { TitleCasePipe } from '@angular/common';\r\nimport { EcCurrencySymbolPipe } from '../../../ec-pipe';\r\nimport { RouterLink } from '@angular/router';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { Observable } from 'rxjs';\r\nimport { Parameter } from '../../../interfaces';\r\n\r\n\r\n@Component({\r\n\tselector: 'app-cart-item-ec',\r\n\tstandalone: true,\r\n\timports: [TitleCasePipe, EcCurrencySymbolPipe, RouterLink, FormsModule],\r\n\ttemplateUrl: './cart-item-ec.component.html',\r\n\tstyleUrl: './cart-item-ec.component.scss'\r\n})\r\nexport class CartItemEcComponent implements OnInit {\r\n\t@Input({\r\n\t\trequired: true\r\n\t}) item: any;\r\n\r\n\t@Input() inSidebar: boolean = false;\r\n\r\n\tprivate _cartService: CartService = inject(CartService);\r\n\tprivate _toastService: ToastService = inject(ToastService);\r\n\tprivate _constants: CoreConstantsService = inject(CoreConstantsService);\r\n\tprivate parametersService = inject(ParametersService)\r\n\r\n\tpublic mediaUrl = this._constants.mediaUrl();\r\n\tpublic quantity = 0;\r\n\tpublic variantsToShow: string[] = ['TALLA', 'COLOR'];\r\n\tpublic updateStock: boolean = false;\r\n\tpublic isQuantityUpdating = false;\r\n\r\n\tngOnInit() {\r\n\t\tthis.quantity = this.item.quantity\r\n\t\t// console.log(this.item, this.mediaUrl);\r\n\t}\r\n\r\n\tupdateQuantity(stock: number) {\r\n\t\tconst originalQuantity = this.item.quantity; // Guardar cantidad original\r\n\r\n\t\tif (this.quantity > 0 && this.quantity <= stock) {\r\n\t\t\tconst success = this._cartService.updateItemQuantity(this.item, this.quantity);\r\n\r\n\t\t\t// Si la validación falló, restaurar la cantidad original\r\n\t\t\tif (!success) {\r\n\t\t\t\tthis.quantity = originalQuantity;\r\n\t\t\t}\r\n\t\t} else {\r\n\t\t\t// Restaurar cantidad original si está fuera de rango\r\n\t\t\tthis.quantity = originalQuantity;\r\n\t\t\tthis._toastService.show('out-of-stock-actually');\r\n\t\t}\r\n\t}\r\n\r\n\tplus(stock: number, value: number = 1) {\r\n\t\tif (this.isQuantityUpdating) return;\r\n\t\tthis.isQuantityUpdating = true;\r\n\r\n\t\tlet newQuantity = Number(this.quantity) + value; // Forzar a número\r\n\r\n\t\t// Verificar stock ANTES de llamar al servicio\r\n\t\tif (newQuantity > stock) {\r\n\t\t\tthis._toastService.show('out-of-stock-actually');\r\n\t\t\tsetTimeout(() => { this.isQuantityUpdating = false; }, 1000);\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tconst success = this._cartService.updateItemQuantity(this.item, newQuantity);\r\n\r\n\t\t// Solo actualizar el input si la operación fue exitosa\r\n\t\tif (success) {\r\n\t\t\tthis.quantity = newQuantity;\r\n\t\t}\r\n\t\t// Si success es false, NO mostrar mensaje aquí porque ya lo mostró validateQuantity()\r\n\r\n\t\tsetTimeout(() => { this.isQuantityUpdating = false; }, 1000);\r\n\t}\r\n\r\n\tless(stock: number, value: number = 1) {\r\n\t\tif (this.isQuantityUpdating) return;\r\n\t\tthis.isQuantityUpdating = true;\r\n\r\n\t\tlet newQuantity = Number(this.quantity) - value; // Forzar a número\r\n\r\n\t\t// Verificar cantidad mínima ANTES de llamar al servicio\r\n\t\tif (newQuantity <= 0) {\r\n\t\t\t// No permitir cantidades menores o iguales a 0\r\n\t\t\tsetTimeout(() => { this.isQuantityUpdating = false; }, 1000);\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tconst success = this._cartService.updateItemQuantity(this.item, newQuantity);\r\n\r\n\t\t// Solo actualizar el input si la operación fue exitosa\r\n\t\tif (success) {\r\n\t\t\tthis.quantity = newQuantity;\r\n\t\t}\r\n\t\t// Si success es false, NO mostrar mensaje aquí porque ya lo mostró validateQuantity()\r\n\r\n\t\tsetTimeout(() => { this.isQuantityUpdating = false; }, 1000);\r\n\t}\r\n\r\n\tdeleteCartItem() {\r\n\t\tthis._cartService.deleteCartItem(this.item)\r\n\t}\r\n\r\n\tgetVariants(product: any): any {\r\n\t\tlet item = product.product\r\n\r\n\t\tif (item.variants && item.variants.length && item.variants[0].options && item.variants[0].options.length) {\r\n\t\t\tlet options = item.variants[0].options\r\n\t\t\toptions = options.filter((option: any) => this.variantsToShow.find((name: any) => option.hasOwnProperty(name)))\r\n\t\t\tlet optionsMap: any = []\r\n\t\t\toptions.map((option: any) => {\r\n\t\t\t\tObject.keys(option).forEach((entry) => {\r\n\t\t\t\t\tlet optionAdded = false;  // Variable para evitar múltiples inserciones\r\n\r\n\t\t\t\t\titem.options.forEach((option2: any) => {\r\n\t\t\t\t\t\toption2.values.forEach((color: any) => {\r\n\t\t\t\t\t\t\tif (color.code == option[entry] && !optionAdded) {\r\n\t\t\t\t\t\t\t\toptionsMap.push({ name: entry, value: option[entry], image: color.image, mobileImage: color.mobileImage });\r\n\t\t\t\t\t\t\t\toptionAdded = true; // Marcar opción agregada\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t});\r\n\t\t\t\t\t});\r\n\t\t\t\t});\r\n\t\t\t\treturn optionsMap;\r\n\t\t\t});\r\n\r\n\t\t\treturn optionsMap;\r\n\r\n\t\t} else {\r\n\t\t\treturn null\r\n\t\t}\r\n\t}\r\n\r\n\tcreateDiscountMessage(saleprice: number, price: number): string {\r\n\t\tif (isNaN(saleprice) || isNaN(price) || saleprice >= price || saleprice <= 0 || price <= 0) {\r\n\t\t\treturn '';\r\n\t\t}\r\n\r\n\t\tlet discountPercentage = Math.round(((price - saleprice) / price) * 100);\r\n\r\n\t\tlet discountMessage = `${discountPercentage}% OFF`;\r\n\t\treturn discountMessage;\r\n\t}\r\n\r\n\tcheckStock(product: any): any {\r\n\t\tfor (const variant of product.product.variants) {\r\n\t\t\tif (product.variant_id === variant.code) {\r\n\t\t\t\tif (variant.stock === 0) {\r\n\t\t\t\t\tthis.updateStock = true;\r\n\t\t\t\t\treturn '0';\r\n\t\t\t\t} else if (product.quantity > variant.stock) {\r\n\t\t\t\t\tthis.updateStock = true;\r\n\t\t\t\t\treturn variant.stock;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn false; // Solo se ejecuta si no se cumple la condición\r\n\t}\r\n\r\n\t// PARAMETROS\r\n\tpublic parameters$: Observable<Parameter[] | null> = this.parametersService.getParameters()\r\n\tpublic hasParams = this.parametersService.hasParams\r\n}\r\n","@if(!inSidebar){\r\n<p>cart-item-ec works!</p>\r\n}@else{\r\n\r\n<div class=\"row\">\r\n    <div class=\"col-3\">\r\n        @let product= item.product;\r\n        @if(item.variant_id && product.variants.length>0){\r\n        <img [src]=\"mediaUrl +  product.variants[0].images[0]\" alt=\"\" class=\"img-fluid\">\r\n        }@else{\r\n        <img [src]=\"mediaUrl +  product.picturesdefault[0]\" alt=\"\" class=\"img-fluid\">\r\n        }\r\n    </div>\r\n    <div class=\"col-7\">\r\n        <div class=\"info d-flex flex-column align-items-start\">\r\n            @if (item.product.special_mark?.length > 0 || item.product.saleprice) {\r\n                    <div class=\"marcas\">\r\n                        <img [src]=\"mediaUrl + (item.product.special_mark?.[0]?.images[0] || '')\" alt=\"\">\r\n\r\n                        @if (item.product.saleprice) {\r\n                        <div class=\"tag-dsc\">\r\n                            {{\r\n                            createDiscountMessage(item.product.saleprice,\r\n                            item.product.price)\r\n                            }}\r\n                        </div>\r\n                        }\r\n                    </div>\r\n                    }\r\n            <a class=\"title text-dark text-decoration-none m-0 p-0 h6 mb-0\"\r\n                [routerLink]=\"['/product', item.variant_id]\">{{\r\n                item.product.name | titlecase\r\n                }}</a>\r\n            <div class=\"qty1\">\r\n                <span>{{ item.product.id}}</span>\r\n            </div>\r\n            <div class=\"price h6 fw-bold mb-0 pb-0\">{{ item.product.price | ecCurrencySymbol\r\n                }}</div>\r\n            @if(getVariants(item); as options){\r\n            <div class=\"d-flex align-items-center p-0\">\r\n                @for(option of options; track $index){\r\n                <span class=\"me-1\"> {{option.name | titlecase}}:</span>\r\n                @if(option.name == 'COLOR'){\r\n                <div class=\"p-2 rounded\" [style.background]=\"'#' + option.value\"></div>\r\n                }@else{\r\n                <b>{{option.value}}</b>\r\n                }\r\n                }\r\n            </div>\r\n            }\r\n            <div class=\"campoCantidad mt-2\">\r\n                <div class=\"numero\">\r\n                    <button (click)=\"less(item.product.variants[0]?.stock)\" class=\"btn btn-outline-secondary\"\r\n                        type=\"button\" id=\"button-addon1\">\r\n                        <i class=\"fa fa-minus\" aria-hidden=\"true\"></i>\r\n                    </button>\r\n                    <input type=\"text\" class=\"form-control text-center\" placeholder=\"\"\r\n                        aria-label=\"Example text with button addon\" aria-describedby=\"button-addon1\"\r\n                        [value]=\"item.quantity\" min=\"1\" step=\"1\" [(ngModel)]=\"quantity\"\r\n                        (change)=\"updateQuantity(item.product.variants[0]?.stock)\">\r\n                    <button (click)=\"plus(item.product.variants[0]?.stock)\" class=\"btn btn-outline-secondary\"\r\n                        type=\"button\" id=\"button-addon1\">\r\n                        <i class=\"fa fa-plus\" aria-hidden=\"true\"></i>\r\n                    </button>\r\n                </div>\r\n\r\n            </div>\r\n        </div>\r\n    </div>\r\n    <div class=\"col-2\">\r\n        <a (click)=\"deleteCartItem()\" class=\"btn botBorrar\"><i class=\"fa fa-trash\" aria-hidden=\"true\"></i></a>\r\n    </div>\r\n</div>\r\n\r\n\r\n\r\n\r\n\r\n}"]}
|
|
@@ -93,7 +93,6 @@ export class ProductEcComponent {
|
|
|
93
93
|
return Math.round(discount);
|
|
94
94
|
}
|
|
95
95
|
plus(stock, multipleQuantity) {
|
|
96
|
-
console.log('Aumentando cantidad');
|
|
97
96
|
const current = Number(this.quantity()); // <-- fuerza a número
|
|
98
97
|
if (multipleQuantity && multipleQuantity > 0) {
|
|
99
98
|
stock
|
|
@@ -111,7 +110,6 @@ export class ProductEcComponent {
|
|
|
111
110
|
}
|
|
112
111
|
}
|
|
113
112
|
less(multipleQuantity) {
|
|
114
|
-
console.log('Disminuyendo cantidad');
|
|
115
113
|
const current = Number(this.quantity()); // <-- fuerza a número
|
|
116
114
|
if (multipleQuantity && multipleQuantity > 0) {
|
|
117
115
|
current > multipleQuantity
|
|
@@ -123,7 +121,6 @@ export class ProductEcComponent {
|
|
|
123
121
|
}
|
|
124
122
|
}
|
|
125
123
|
addToCart() {
|
|
126
|
-
console.log('Añadiendo al carrito');
|
|
127
124
|
if (this.isAddingToCart() || this.quantity() <= 0)
|
|
128
125
|
return;
|
|
129
126
|
// Verificar que tenemos el producto y sus variantes
|
|
@@ -142,22 +139,73 @@ export class ProductEcComponent {
|
|
|
142
139
|
this._toastService.show('out-of-stock-actually');
|
|
143
140
|
return;
|
|
144
141
|
}
|
|
142
|
+
// NUEVAS VALIDACIONES: Verificar restricciones del producto
|
|
143
|
+
if (!this.validateQuantity(this.quantity())) {
|
|
144
|
+
return; // La validación ya mostró el toast correspondiente
|
|
145
|
+
}
|
|
145
146
|
this.isAddingToCart.set(true);
|
|
146
147
|
try {
|
|
147
148
|
// Agregar al carrito usando CartService directamente
|
|
148
149
|
this._cartService.addToCart(this.product, this.quantity(), firstVariant.code);
|
|
149
|
-
console.log('Producto agregado al carrito exitosamente');
|
|
150
150
|
// Resetear cantidad a 1 después de agregar al carrito
|
|
151
151
|
this.quantity.set(1);
|
|
152
152
|
}
|
|
153
153
|
catch (error) {
|
|
154
|
-
console.error('Error al agregar al carrito:', error);
|
|
155
154
|
this._toastService.show('cant-buy');
|
|
156
155
|
}
|
|
157
156
|
setTimeout(() => {
|
|
158
157
|
this.isAddingToCart.set(false);
|
|
159
158
|
}, 2000);
|
|
160
159
|
}
|
|
160
|
+
/**
|
|
161
|
+
* Valida las restricciones de cantidad del producto
|
|
162
|
+
*/
|
|
163
|
+
validateQuantity(quantity) {
|
|
164
|
+
if (!this.product.variants || this.product.variants.length === 0) {
|
|
165
|
+
this._toastService.show('cant-buy');
|
|
166
|
+
return false;
|
|
167
|
+
}
|
|
168
|
+
const variant = this.product.variants[0];
|
|
169
|
+
// Obtener cantidad actual en el carrito
|
|
170
|
+
const actualQuantity = this._cartService.getCountFromItemInCart(variant.code);
|
|
171
|
+
const totalQuantity = Number(actualQuantity) + Number(quantity);
|
|
172
|
+
// Obtener restricciones del producto/variante (con valores por defecto)
|
|
173
|
+
const maximumItemsQuantity = this.product.maximumItemsQuantity ||
|
|
174
|
+
variant.maximumItemsQuantity ||
|
|
175
|
+
999999;
|
|
176
|
+
const minimumItemsQuantity = this.product.minimumItemsQuantity ||
|
|
177
|
+
variant.minimumItemsQuantity ||
|
|
178
|
+
1;
|
|
179
|
+
const multipleQuantity = this.product.multipleQuantity ||
|
|
180
|
+
variant.multipleQuantity ||
|
|
181
|
+
1;
|
|
182
|
+
// Validar cantidad máxima
|
|
183
|
+
if (totalQuantity > maximumItemsQuantity) {
|
|
184
|
+
this._toastService.show('maximum-items-quantity', { quantity: maximumItemsQuantity });
|
|
185
|
+
return false;
|
|
186
|
+
}
|
|
187
|
+
// Validar cantidad mínima
|
|
188
|
+
if (totalQuantity < minimumItemsQuantity) {
|
|
189
|
+
this._toastService.show('minimum-items-quantity', { quantity: minimumItemsQuantity });
|
|
190
|
+
return false;
|
|
191
|
+
}
|
|
192
|
+
// Validar múltiplo de cantidad
|
|
193
|
+
if (multipleQuantity && multipleQuantity > 1) {
|
|
194
|
+
if (totalQuantity % multipleQuantity !== 0) {
|
|
195
|
+
this._toastService.show('multiple-quantity-required', {
|
|
196
|
+
multiple: multipleQuantity,
|
|
197
|
+
current: totalQuantity
|
|
198
|
+
});
|
|
199
|
+
return false;
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
// Validar que no exceda el stock total
|
|
203
|
+
if (totalQuantity > variant.stock) {
|
|
204
|
+
this._toastService.show('out-of-stock-actually');
|
|
205
|
+
return false;
|
|
206
|
+
}
|
|
207
|
+
return true; // Todas las validaciones pasaron
|
|
208
|
+
}
|
|
161
209
|
checkStock(stock) {
|
|
162
210
|
if (this.quantity() >= stock)
|
|
163
211
|
this.quantity.set(stock);
|
|
@@ -195,4 +243,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
195
243
|
}], isCollection: [{
|
|
196
244
|
type: Input
|
|
197
245
|
}] } });
|
|
198
|
-
//# 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, Injector, WritableSignal, signal, } from '@angular/core';\r\nimport { Product } from '../../interfaces';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { AnalyticsService, ChannelService, AuthService, CartService, ToastService } 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\nimport { FormsModule } from '@angular/forms';\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, FormsModule],\r\n\ttemplateUrl: './product-ec.component.html',\r\n\tstyleUrl: './product-ec.component.scss'\r\n})\r\nexport class ProductEcComponent implements OnInit {\r\n\tpublic injector: Injector = inject(Injector);\r\n\tprivate routerService = inject(Router);\r\n\tprivate _cartService: CartService = inject(CartService);\r\n\tprivate _toastService: ToastService = inject(ToastService);\r\n\r\n\tpublic isAddingToCart: WritableSignal<boolean> = signal(false);\r\n\tpublic quantity: WritableSignal<number> = signal(1);\r\n\r\n\t/**\r\n\t * Navega al detalle del producto y sube al inicio de la página\r\n\t * @param productId ID del producto\r\n\t */\r\n\tnavigateToProduct(productId: string): void {\r\n\t\tthis.routerService.navigate(['/product', productId]).then(() => {\r\n\t\t\t// Scroll suave al inicio de la página\r\n\t\t\twindow.scrollTo({\r\n\t\t\t\ttop: 0,\r\n\t\t\t\tleft: 0,\r\n\t\t\t\tbehavior: 'smooth'\r\n\t\t\t});\r\n\t\t});\r\n\t}\r\n\r\n\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\tthis.loaded.emit(this.product.id);\r\n\r\n\t}\r\n\tpublic showPricesOnlyToLoggedUsers: boolean = false;\r\n\tpublic hidePrices: boolean = false;\r\n\r\n\tprivate _authService: AuthService = inject(AuthService)\r\n\tpublic isAuthenticated$ = this._authService.isAuthenticated();\r\n\tconstructor() {\r\n\t\tthis.injector.get(ChannelService).channel$.subscribe(channel => {\r\n\t\t\tthis.showPricesOnlyToLoggedUsers = !!channel.showPricesOnlyToLoggedUsers;\r\n\t\t\tthis.hidePrices = !!channel.hidePrices;\r\n\t\t});\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\t\tif (url) {\r\n\t\t\twindow.open(url, '_blank');\r\n\t\t}\r\n\t}\r\n\r\n\tget shouldShowPrice(): boolean {\r\n\t\treturn !this.product?.special_mark || this.product.special_mark.length === 0 || this.product.special_mark[0]?.showPrice;\r\n\t}\r\n\r\n\tget hasDiscount(): boolean {\r\n\t\treturn this.product.saleprice && this.product.saleprice !== this.product.price;\r\n\t}\r\n\r\n\tget discountPercentage(): number | null {\r\n\t\tif (!this.hasDiscount) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\r\n\t\tconst originalPrice = parseFloat(this.product.price?.toString() || '0');\r\n\t\tconst salePrice = parseFloat(this.product.saleprice?.toString() || '0');\r\n\r\n\t\t// Validar que los precios sean válidos\r\n\t\tif (originalPrice <= 0 || salePrice <= 0 || salePrice >= originalPrice) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\r\n\t\t// Calcular el porcentaje: ((precio_original - precio_oferta) / precio_original) * 100\r\n\t\tconst discount = ((originalPrice - salePrice) / originalPrice) * 100;\r\n\r\n\t\treturn Math.round(discount);\r\n\t}\r\n\tplus(stock?: number, multipleQuantity?: number) {\r\n\t\tconsole.log('Aumentando cantidad');\r\n\t\tconst current = Number(this.quantity()); // <-- fuerza a número\r\n\t\tif (multipleQuantity && multipleQuantity > 0) {\r\n\t\t\tstock\r\n\t\t\t\t? (current < stock\r\n\t\t\t\t\t? this.quantity.set(current + multipleQuantity)\r\n\t\t\t\t\t: this._toastService.show('out-of-stock-actually'))\r\n\t\t\t\t: this.quantity.set(current + multipleQuantity);\r\n\t\t} else {\r\n\t\t\tstock\r\n\t\t\t\t? (current < stock\r\n\t\t\t\t\t? this.quantity.set(current + 1)\r\n\t\t\t\t\t: this._toastService.show('out-of-stock-actually'))\r\n\t\t\t\t: this.quantity.set(current + 1);\r\n\t\t}\r\n\t}\r\n\r\n\tless(multipleQuantity?: number) {\r\n\t\tconsole.log('Disminuyendo cantidad');\r\n\t\tconst current = Number(this.quantity()); // <-- fuerza a número\r\n\t\tif (multipleQuantity && multipleQuantity > 0) {\r\n\t\t\tcurrent > multipleQuantity\r\n\t\t\t\t? this.quantity.set(current - multipleQuantity)\r\n\t\t\t\t: null;\r\n\t\t} else {\r\n\t\t\tcurrent > 1 ? this.quantity.set(current - 1) : null;\r\n\t\t}\r\n\t}\r\n\r\n\taddToCart() {\r\n\t\tconsole.log('Añadiendo al carrito');\r\n\t\tif (this.isAddingToCart() || this.quantity() <= 0) return;\r\n\r\n\t\t// Verificar que tenemos el producto y sus variantes\r\n\t\tif (!this.product || !this.product.variants || this.product.variants.length === 0) {\r\n\t\t\tthis._toastService.show('cant-buy');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// Verificar stock\r\n\t\tconst firstVariant = this.product.variants[0];\r\n\t\tif (!firstVariant.stock || firstVariant.stock <= 0) {\r\n\t\t\tthis._toastService.show('out-of-stock');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// Verificar que no se supere el stock disponible\r\n\t\tif (this.quantity() > firstVariant.stock) {\r\n\t\t\tthis._toastService.show('out-of-stock-actually');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tthis.isAddingToCart.set(true);\r\n\r\n\t\ttry {\r\n\t\t\t// Agregar al carrito usando CartService directamente\r\n\t\t\tthis._cartService.addToCart(this.product, this.quantity(), firstVariant.code);\r\n\t\t\tconsole.log('Producto agregado al carrito exitosamente');\r\n\r\n\t\t\t// Resetear cantidad a 1 después de agregar al carrito\r\n\t\t\tthis.quantity.set(1);\r\n\t\t} catch (error) {\r\n\t\t\tconsole.error('Error al agregar al carrito:', error);\r\n\t\t\tthis._toastService.show('cant-buy');\r\n\t\t}\r\n\r\n\t\tsetTimeout(() => {\r\n\t\t\tthis.isAddingToCart.set(false);\r\n\t\t}, 2000);\r\n\t}\r\n\tcheckStock(stock: number) {\r\n\t\tif (this.quantity() >= stock)\r\n\t\t\tthis.quantity.set(stock);\r\n\t}\r\n\r\n\tonQuantityChange(event: Event) {\r\n\t\tconst target = event.target as HTMLInputElement;\r\n\t\tconst value = +target.value;\r\n\r\n\t\t// Validar que el valor sea mayor a 0\r\n\t\tif (value > 0) {\r\n\t\t\tthis.quantity.set(value);\r\n\t\t\t// Validar contra el stock disponible\r\n\t\t\tconst maxStock = this.product?.variants?.[0]?.stock || this.product?.stock || 0;\r\n\t\t\tthis.checkStock(maxStock);\r\n\t\t} else {\r\n\t\t\t// Si el valor es 0 o negativo, resetear a 1\r\n\t\t\tthis.quantity.set(1);\r\n\t\t}\r\n\t}\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    @if(!hidePrices){\r\n    @if(!showPricesOnlyToLoggedUsers || isAuthenticated$){\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    }}\r\n</a>"]}
|
|
246
|
+
//# 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, Injector, WritableSignal, signal, } from '@angular/core';\r\nimport { Product } from '../../interfaces';\r\nimport { CoreConstantsService } from '../../constants';\r\nimport { AnalyticsService, ChannelService, AuthService, CartService, ToastService } 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\nimport { FormsModule } from '@angular/forms';\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, FormsModule],\r\n\ttemplateUrl: './product-ec.component.html',\r\n\tstyleUrl: './product-ec.component.scss'\r\n})\r\nexport class ProductEcComponent implements OnInit {\r\n\tpublic injector: Injector = inject(Injector);\r\n\tprivate routerService = inject(Router);\r\n\tprivate _cartService: CartService = inject(CartService);\r\n\tprivate _toastService: ToastService = inject(ToastService);\r\n\r\n\tpublic isAddingToCart: WritableSignal<boolean> = signal(false);\r\n\tpublic quantity: WritableSignal<number> = signal(1);\r\n\r\n\t/**\r\n\t * Navega al detalle del producto y sube al inicio de la página\r\n\t * @param productId ID del producto\r\n\t */\r\n\tnavigateToProduct(productId: string): void {\r\n\t\tthis.routerService.navigate(['/product', productId]).then(() => {\r\n\t\t\t// Scroll suave al inicio de la página\r\n\t\t\twindow.scrollTo({\r\n\t\t\t\ttop: 0,\r\n\t\t\t\tleft: 0,\r\n\t\t\t\tbehavior: 'smooth'\r\n\t\t\t});\r\n\t\t});\r\n\t}\r\n\r\n\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\tthis.loaded.emit(this.product.id);\r\n\r\n\t}\r\n\tpublic showPricesOnlyToLoggedUsers: boolean = false;\r\n\tpublic hidePrices: boolean = false;\r\n\r\n\tprivate _authService: AuthService = inject(AuthService)\r\n\tpublic isAuthenticated$ = this._authService.isAuthenticated();\r\n\tconstructor() {\r\n\t\tthis.injector.get(ChannelService).channel$.subscribe(channel => {\r\n\t\t\tthis.showPricesOnlyToLoggedUsers = !!channel.showPricesOnlyToLoggedUsers;\r\n\t\t\tthis.hidePrices = !!channel.hidePrices;\r\n\t\t});\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\t\tif (url) {\r\n\t\t\twindow.open(url, '_blank');\r\n\t\t}\r\n\t}\r\n\r\n\tget shouldShowPrice(): boolean {\r\n\t\treturn !this.product?.special_mark || this.product.special_mark.length === 0 || this.product.special_mark[0]?.showPrice;\r\n\t}\r\n\r\n\tget hasDiscount(): boolean {\r\n\t\treturn this.product.saleprice && this.product.saleprice !== this.product.price;\r\n\t}\r\n\r\n\tget discountPercentage(): number | null {\r\n\t\tif (!this.hasDiscount) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\r\n\t\tconst originalPrice = parseFloat(this.product.price?.toString() || '0');\r\n\t\tconst salePrice = parseFloat(this.product.saleprice?.toString() || '0');\r\n\r\n\t\t// Validar que los precios sean válidos\r\n\t\tif (originalPrice <= 0 || salePrice <= 0 || salePrice >= originalPrice) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\r\n\t\t// Calcular el porcentaje: ((precio_original - precio_oferta) / precio_original) * 100\r\n\t\tconst discount = ((originalPrice - salePrice) / originalPrice) * 100;\r\n\r\n\t\treturn Math.round(discount);\r\n\t}\r\n\tplus(stock?: number, multipleQuantity?: number) {\r\n\t\tconst current = Number(this.quantity()); // <-- fuerza a número\r\n\t\tif (multipleQuantity && multipleQuantity > 0) {\r\n\t\t\tstock\r\n\t\t\t\t? (current < stock\r\n\t\t\t\t\t? this.quantity.set(current + multipleQuantity)\r\n\t\t\t\t\t: this._toastService.show('out-of-stock-actually'))\r\n\t\t\t\t: this.quantity.set(current + multipleQuantity);\r\n\t\t} else {\r\n\t\t\tstock\r\n\t\t\t\t? (current < stock\r\n\t\t\t\t\t? this.quantity.set(current + 1)\r\n\t\t\t\t\t: this._toastService.show('out-of-stock-actually'))\r\n\t\t\t\t: this.quantity.set(current + 1);\r\n\t\t}\r\n\t}\r\n\r\n\tless(multipleQuantity?: number) {\r\n\t\tconst current = Number(this.quantity()); // <-- fuerza a número\r\n\t\tif (multipleQuantity && multipleQuantity > 0) {\r\n\t\t\tcurrent > multipleQuantity\r\n\t\t\t\t? this.quantity.set(current - multipleQuantity)\r\n\t\t\t\t: null;\r\n\t\t} else {\r\n\t\t\tcurrent > 1 ? this.quantity.set(current - 1) : null;\r\n\t\t}\r\n\t}\r\n\r\n\taddToCart() {\r\n\t\tif (this.isAddingToCart() || this.quantity() <= 0) return;\r\n\r\n\t\t// Verificar que tenemos el producto y sus variantes\r\n\t\tif (!this.product || !this.product.variants || this.product.variants.length === 0) {\r\n\t\t\tthis._toastService.show('cant-buy');\r\n\t\t\treturn;\r\n\t\t}\r\n\t\t// Verificar stock\r\n\t\tconst firstVariant = this.product.variants[0];\r\n\t\tif (!firstVariant.stock || firstVariant.stock <= 0) {\r\n\t\t\tthis._toastService.show('out-of-stock');\r\n\t\t\treturn;\r\n\t\t}\r\n\t\t// Verificar que no se supere el stock disponible\r\n\t\tif (this.quantity() > firstVariant.stock) {\r\n\t\t\tthis._toastService.show('out-of-stock-actually');\r\n\t\t\treturn;\r\n\t\t}\r\n\t\t// NUEVAS VALIDACIONES: Verificar restricciones del producto\r\n\t\tif (!this.validateQuantity(this.quantity())) {\r\n\t\t\treturn; // La validación ya mostró el toast correspondiente\r\n\t\t}\r\n\t\tthis.isAddingToCart.set(true);\r\n\t\ttry {\r\n\t\t\t// Agregar al carrito usando CartService directamente\r\n\t\t\tthis._cartService.addToCart(this.product, this.quantity(), firstVariant.code);\r\n\t\t\t// Resetear cantidad a 1 después de agregar al carrito\r\n\t\t\tthis.quantity.set(1);\r\n\t\t} catch (error) {\r\n\t\t\tthis._toastService.show('cant-buy');\r\n\t\t}\r\n\r\n\t\tsetTimeout(() => {\r\n\t\t\tthis.isAddingToCart.set(false);\r\n\t\t}, 2000);\r\n\t}\r\n\r\n\t/**\r\n\t * Valida las restricciones de cantidad del producto\r\n\t */\r\n\tprivate validateQuantity(quantity: number): boolean {\r\n\t\tif (!this.product.variants || this.product.variants.length === 0) {\r\n\t\t\tthis._toastService.show('cant-buy');\r\n\t\t\treturn false;\r\n\t\t}\r\n\r\n\t\tconst variant = this.product.variants[0];\r\n\r\n\t\t// Obtener cantidad actual en el carrito\r\n\t\tconst actualQuantity = this._cartService.getCountFromItemInCart(variant.code);\r\n\t\tconst totalQuantity = Number(actualQuantity) + Number(quantity);\r\n\t\t// Obtener restricciones del producto/variante (con valores por defecto)\r\n\t\tconst maximumItemsQuantity = this.product.maximumItemsQuantity ||\r\n\t\t\tvariant.maximumItemsQuantity ||\r\n\t\t\t999999;\r\n\t\tconst minimumItemsQuantity = this.product.minimumItemsQuantity ||\r\n\t\t\tvariant.minimumItemsQuantity ||\r\n\t\t\t1;\r\n\t\tconst multipleQuantity = this.product.multipleQuantity ||\r\n\t\t\tvariant.multipleQuantity ||\r\n\t\t\t1;\r\n\r\n\t\t// Validar cantidad máxima\r\n\t\tif (totalQuantity > maximumItemsQuantity) {\r\n\t\t\tthis._toastService.show('maximum-items-quantity', { quantity: maximumItemsQuantity });\r\n\t\t\treturn false;\r\n\t\t}\r\n\r\n\t\t// Validar cantidad mínima\r\n\t\tif (totalQuantity < minimumItemsQuantity) {\r\n\t\t\tthis._toastService.show('minimum-items-quantity', { quantity: minimumItemsQuantity });\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\t// Validar múltiplo de cantidad\r\n\t\tif (multipleQuantity && multipleQuantity > 1) {\r\n\t\t\tif (totalQuantity % multipleQuantity !== 0) {\r\n\t\t\t\tthis._toastService.show('multiple-quantity-required', {\r\n\t\t\t\t\tmultiple: multipleQuantity,\r\n\t\t\t\t\tcurrent: totalQuantity\r\n\t\t\t\t});\r\n\t\t\t\treturn false;\r\n\t\t\t}\r\n\t\t}\r\n\t\t// Validar que no exceda el stock total\r\n\t\tif (totalQuantity > variant.stock) {\r\n\t\t\tthis._toastService.show('out-of-stock-actually');\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\treturn true; // Todas las validaciones pasaron\r\n\t}\r\n\tcheckStock(stock: number) {\r\n\t\tif (this.quantity() >= stock)\r\n\t\t\tthis.quantity.set(stock);\r\n\t}\r\n\r\n\tonQuantityChange(event: Event) {\r\n\t\tconst target = event.target as HTMLInputElement;\r\n\t\tconst value = +target.value;\r\n\t\t// Validar que el valor sea mayor a 0\r\n\t\tif (value > 0) {\r\n\t\t\tthis.quantity.set(value);\r\n\t\t\t// Validar contra el stock disponible\r\n\t\t\tconst maxStock = this.product?.variants?.[0]?.stock || this.product?.stock || 0;\r\n\t\t\tthis.checkStock(maxStock);\r\n\t\t} else {\r\n\t\t\t// Si el valor es 0 o negativo, resetear a 1\r\n\t\t\tthis.quantity.set(1);\r\n\t\t}\r\n\t}\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    @if(!hidePrices){\r\n    @if(!showPricesOnlyToLoggedUsers || isAuthenticated$){\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    }}\r\n</a>"]}
|