@snabcentr/client-ui 3.32.3 → 3.32.4
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/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.d.ts +1 -1
- package/cart/index.d.ts +1 -1
- package/cart/{sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component.d.ts → sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component.d.ts} +3 -3
- package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +6 -6
- package/esm2022/cart/index.mjs +2 -2
- package/esm2022/cart/sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component.mjs +110 -0
- package/fesm2022/snabcentr-client-ui.mjs +11 -11
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +1 -1
- package/release_notes.tmp +2 -10
- package/esm2022/cart/sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component.mjs +0 -110
@@ -42,7 +42,7 @@ export declare class ScAddOrEditingCartItemFormComponent implements OnInit {
|
|
42
42
|
/**
|
43
43
|
* Минимальный метраж для товара.
|
44
44
|
*/
|
45
|
-
protected readonly minLength: import("@angular/core").Signal<number>;
|
45
|
+
protected readonly minLength: import("@angular/core").Signal<number | undefined>;
|
46
46
|
/**
|
47
47
|
* Максимальный метраж для товара.
|
48
48
|
*/
|
package/cart/index.d.ts
CHANGED
@@ -1,3 +1,3 @@
|
|
1
1
|
export * from './cart-item/sc-cart-item.component';
|
2
|
-
export * from './sc-
|
2
|
+
export * from './sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component';
|
3
3
|
export * from './add-or-editing-cart-item-dialog';
|
@@ -6,7 +6,7 @@ import * as i0 from "@angular/core";
|
|
6
6
|
/**
|
7
7
|
* Компонент диалога для добавления товаров в корзину из CSV файла.
|
8
8
|
*/
|
9
|
-
export declare class
|
9
|
+
export declare class ScCartAddProductsFromCsvDialogComponent {
|
10
10
|
/**
|
11
11
|
* Контекст диалогового окна, в котором открыт компонент.
|
12
12
|
*/
|
@@ -57,6 +57,6 @@ export declare class ScCarAddProductsFromCsvDialogComponent {
|
|
57
57
|
* @param blob Бинарный объект.
|
58
58
|
*/
|
59
59
|
private downloadExampleFile;
|
60
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<
|
61
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<
|
60
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScCartAddProductsFromCsvDialogComponent, never>;
|
61
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScCartAddProductsFromCsvDialogComponent, "sc-cart-add-products-from-csv-dialog", never, {}, {}, never, never, true, never>;
|
62
62
|
}
|
@@ -61,7 +61,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
61
61
|
/**
|
62
62
|
* Минимальный метраж для товара.
|
63
63
|
*/
|
64
|
-
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount)
|
64
|
+
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));
|
65
65
|
/**
|
66
66
|
* Максимальный метраж для товара.
|
67
67
|
*/
|
@@ -76,8 +76,8 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
76
76
|
this.lengthHint = computed(() => {
|
77
77
|
const min = this.minLength();
|
78
78
|
const max = this.maxLength();
|
79
|
-
// eslint-disable-next-line sonarjs/no-nested-conditional,
|
80
|
-
return
|
79
|
+
// eslint-disable-next-line sonarjs/no-nested-conditional, sonarjs/prefer-nullish-coalescing
|
80
|
+
return min || max ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';
|
81
81
|
});
|
82
82
|
/**
|
83
83
|
* {@link Output} события добавления товара в корзину.
|
@@ -108,7 +108,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
108
108
|
});
|
109
109
|
if (this.productIsMeasurable()) {
|
110
110
|
// Добавляем поле ввода длины, если товар измеряемый.
|
111
|
-
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));
|
111
|
+
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength() ?? 0.01, [Validators.required, Validators.min(0.01)]));
|
112
112
|
}
|
113
113
|
// Устанавливаем количество из корзины или кратности товара.
|
114
114
|
this.form.patchValue({
|
@@ -161,7 +161,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
161
161
|
return;
|
162
162
|
}
|
163
163
|
const length = lengthControl.value ?? 0;
|
164
|
-
let newLength = Math.max(this.minLength(), length + step);
|
164
|
+
let newLength = Math.max(this.minLength() ?? step, length + step);
|
165
165
|
// Округляем до ближайшего кратного числа.
|
166
166
|
newLength = Math.round(newLength / step) * step;
|
167
167
|
lengthControl.setValue(Number(newLength.toFixed(2)));
|
@@ -230,4 +230,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
230
230
|
TuiChip,
|
231
231
|
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- \u0424\u043E\u0440\u043C\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F \u0438\u0437\u043C\u0435\u0440\u044F\u0435\u043C\u043E\u0433\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n<form\n *ngIf=\"product\"\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit()\"\n ScNextInputFocus\n class=\"flex flex-col gap-2\"\n>\n @let step = lengthStep();\n\n <!-- \u0414\u043B\u0438\u043D\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 (\u043C\u0435\u0442\u0440\u0430\u0436) -->\n <label\n *ngIf=\"productIsMeasurable()\"\n tuiLabel\n >\n \u041C\u0435\u0442\u0440\u0430\u0436, {{ product().unit }}\n @if (!step || maxLength()) {\n {{ lengthHint() }}\n }\n\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"length\"\n [tuiNumberFormat]=\"{ precision: 2 }\"\n [max]=\"maxLength() || null\"\n [min]=\"minLength() || null\"\n (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n autocomplete=\"length\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n <p\n *ngIf=\"lengthStep()\"\n class=\"tui-form__field-note\"\n >\n \u041C\u0435\u0442\u0440\u0430\u0436 \u0434\u043E\u043B\u0436\u0435\u043D \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u0435\u043D {{ lengthStep() }}\n </p>\n </label>\n\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430 -->\n <label tuiLabel>\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442.\n <tui-textfield>\n <tui-chip\n *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n size=\"s\"\n appearance=\"negative\"\n class=\"font-bold\"\n >\n x {{ step }} {{ product().unit }}\n </tui-chip>\n\n <input\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n tuiInputNumber\n [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n [min]=\"productMultiplicity()\"\n (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n formControlName=\"quantity\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n <p class=\"tui-form__field-note\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0434\u043E\u043B\u0436\u043D\u043E \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u043D\u043E {{ productMultiplicity() }}</p>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 -->\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-input formControlName=\"marker\">\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <input\n tuiTextfieldLegacy\n autocomplete=\"marker\"\n />\n </tui-input>\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <div class=\"flex flex-col items-center\">\n @let cost = totalCost$ | async;\n\n <div *ngIf=\"cost\">\n \u0418\u0442\u043E\u0433\u043E:<span class=\"text-2xl font-bold\">\n {{ cost | tuiAmount | async }}\n {{ product().currency }}\n </span>\n </div>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <button\n tuiButton\n iconStart=\"@tui.check\"\n [disabled]=\"form.invalid\"\n [loading]=\"isLoading()\"\n type=\"submit\"\n class=\"mt-2\"\n >\n {{ cartItem() ? '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' : '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' }}\n </button>\n </div>\n</form>\n" }]
|
232
232
|
}] });
|
233
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"AAAA,0FAA0F;AAE1F,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAC5G,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,WAAW,EAAE,mBAAmB,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AACtG,OAAO,EAEH,mBAAmB,EAInB,aAAa,EACb,sBAAsB,EACtB,oBAAoB,GACvB,MAAM,wBAAwB,CAAC;AAChC,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,kCAAkC,EAAE,MAAM,eAAe,CAAC;AACpF,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,QAAQ,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAC9F,OAAO,EAAE,aAAa,EAAE,iBAAiB,EAAE,MAAM,sCAAsC,CAAC;AACxF,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,MAAM,iCAAiC,CAAC;AACzE,OAAO,EAAE,gBAAgB,EAAE,OAAO,EAAE,iBAAiB,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC7F,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AACxC,OAAO,EAAE,YAAY,EAAE,oBAAoB,EAAE,MAAM,EAAE,GAAG,EAAc,QAAQ,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AAExG,OAAO,EAAE,sBAAsB,EAAE,MAAM,iEAAiE,CAAC;AACzG,OAAO,EAAE,0BAA0B,EAAE,MAAM,sEAAsE,CAAC;AAClH,OAAO,EAAE,aAAa,EAAE,MAAM,oCAAoC,CAAC;;;;;;;;AAEnE;;GAEG;AA+BH,MAAM,OAAO,mCAAmC;IA9BhD;QA+BI;;WAEG;QACa,SAAI,GAAG,IAAI,SAAS,CAIjC;YACC,QAAQ,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC;YACnE,MAAM,EAAE,IAAI,WAAW,CAAgB,EAAE,CAAC;SAC7C,CAAC,CAAC;QAEH;;WAEG;QACa,YAAO,GAAG,KAAK,CAAC,QAAQ,EAAa,CAAC;QAEtD;;WAEG;QACgB,yBAAoB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,CAAC,sBAAsB,CAAC,CAAC,CAAC,CAAC;QAEvH;;WAEG;QACa,aAAQ,GAAG,KAAK,EAA0B,CAAC;QAE3D;;WAEG;QACa,cAAS,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC;QAEzC;;WAEG;QACgB,wBAAmB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAE9G;;WAEG;QACgB,wBAAmB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAE9G;;WAEG;QACgB,cAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,SAAS,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;QAE/J;;WAEG;QACgB,cAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;QAEpF;;WAEG;QACgB,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,UAAU,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC;QAE5J;;WAEG;QACgB,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC1C,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;YAC7B,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;YAE7B,sGAAsG;YACtG,OAAO,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,GAAG,IAAI,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,GAAG,CAAC,CAAC,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC,EAAE,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QACrI,CAAC,CAAC,CAAC;QAOH;;WAEG;QACa,cAAS,GAAG,MAAM,EAAsB,CAAC;QAEzD;;WAEG;QACa,iBAAY,GAAG,MAAM,EAAyC,CAAC;QAE/E;;WAEG;QACc,gBAAW,GAAkB,MAAM,CAAC,aAAa,CAAC,CAAC;QAEpE;;WAEG;QACc,uBAAkB,GAAyB,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAEzF;;WAEG;QACc,sBAAiB,GAAwB,MAAM,CAAC,mBAAmB,CAAC,CAAC;KAoIzF;IAlIG,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,CAAC;YACzC,MAAM,EAAE,sBAAsB,CAAC,eAAe;YAC9C,MAAM,EAAE,EAAE,UAAU,EAAE,IAAI,CAAC,OAAO,EAAE,CAAC,EAAE,EAAE;SAC5C,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,mBAAmB,EAAE,EAAE,CAAC;YAC7B,qDAAqD;YACrD,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,IAAI,WAAW,CAAgB,IAAI,CAAC,QAAQ,EAAE,EAAE,MAAM,IAAI,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7J,CAAC;QAED,4DAA4D;QAC5D,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;YACjB,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,EAAE,QAAQ,IAAI,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YAC3F,MAAM,EAAE,IAAI,CAAC,QAAQ,EAAE,EAAE,MAAM,IAAI,EAAE;SACxC,CAAC,CAAC;QAEH,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAE/B,IAAI,IAAI,CAAC,oBAAoB,EAAE,IAAI,IAAI,CAAC,UAAU,EAAE,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,IAAI,IAAI,EAAE,CAAC;YAChG,sEAAsE;YACtE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC;QACpF,CAAC;QAED,2CAA2C;QAC3C,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,aAAa,CAAC,aAAa,CAAC,IAAI,CAAC,WAAW,CAAC,uBAAuB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,CAAC;QACnH,sCAAsC;QACtC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,aAAa,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,UAAU,EAAE,UAAU,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,QAAQ,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAE1K,qCAAqC;QACrC,IAAI,CAAC,UAAU,GAAG,eAAe,CAG9B,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CACd,YAAY,CAAC,CAAC,CAAC,EAAE,4BAA4B;QAC7C,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAC7B,oBAAoB,CAAC,CAAC,QAAQ,EAAE,OAAO,EAAE,EAAE,CAAC,QAAQ,CAAC,MAAM,KAAK,OAAO,CAAC,MAAM,IAAI,QAAQ,CAAC,QAAQ,KAAK,OAAO,CAAC,QAAQ,CAAC,EAAE,gCAAgC;QAC3J,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,KAA8C,CAAC,EACnE,QAAQ,EAAE,EACV,GAAG,CAAC,CAAC,CAAC,QAAQ,EAAE,OAAO,CAAC,EAAE,EAAE;YACxB,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YAE9C,mFAAmF;YACnF,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE,IAAI,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,IAAI,EAAE,CAAC;gBAC9G,OAAO,OAAO,CAAC;YACnB,CAAC;YAED,oDAAoD;YACpD,IAAI,QAAQ,CAAC,QAAQ,KAAK,OAAO,CAAC,QAAQ,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;gBAC9E,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,GAAG,IAAI,CAAC;gBAC1D,aAAa,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;YACvC,CAAC;YAED,gFAAgF;YAChF,IAAI,QAAQ,CAAC,MAAM,KAAK,OAAO,CAAC,MAAM,IAAI,aAAa,CAAC,KAAK,EAAE,CAAC;gBAC5D,MAAM,QAAQ,GAAG,aAAa,CAAC,KAAK,GAAG,IAAI,CAAC;gBAC5C,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;gBAEjD,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC;YAChD,CAAC;YAED,sDAAsD;YACtD,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC;QACxD,CAAC,CAAC;QACF,gDAAgD;QAChD,GAAG,CAAC,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,OAAO,IAAI,CAAC,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,CAAC,GAAG,QAAQ,CAAC,EACvF,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,CAAC,CAC5C,CAAC;IACN,CAAC;IAED;;;;OAIG;IACO,YAAY,CAAC,IAAY;QAC/B,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAE9C,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,aAAa,EAAE,CAAC;YAChD,OAAO;QACX,CAAC;QAED,MAAM,MAAM,GAAG,aAAa,CAAC,KAAK,IAAI,CAAC,CAAC;QACxC,IAAI,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,EAAE,MAAM,GAAG,IAAI,CAAC,CAAC;QAC1D,0CAA0C;QAC1C,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC;QAChD,aAAa,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACzD,CAAC;IAED;;;;OAIG;IACO,cAAc,CAAC,IAAY;QACjC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,IAAI,CAAC,CAAC;QAExD,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,mBAAmB,EAAE,EAAE,QAAQ,GAAG,CAAC,QAAQ,GAAG,IAAI,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC;IACpH,CAAC;IAED;;OAEG;IACO,QAAQ;QACd,uCAAuC;QACvC,MAAM,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAyB,IAAI,CAAC,IAAI,CAAC,KAA+B,CAAC,CAAC;QAEnH,2DAA2D;QAC3D,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YACnB,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC;QACxC,CAAC;QAED,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAE/B,6IAA6I;QAC7I,IAAI,IAAI,CAAC,mBAAmB,EAAE,IAAI,IAAI,CAAC,oBAAoB,EAAE,IAAI,IAAI,EAAE,CAAC;YACpE,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC;QACxB,CAAC;QAED,8FAA8F;QAC9F,kCAAkC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE9C,6DAA6D;QAC7D,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YAClB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAClC,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC/B,CAAC;IACL,CAAC;+GApOQ,mCAAmC;mGAAnC,mCAAmC,6iBC9DhD,68JAmHA,4CD9EQ,IAAI,4FACJ,WAAW,2pBACX,sBAAsB,6HACtB,mBAAmB,gVACnB,QAAQ,4DACR,0BAA0B,oJAC1B,eAAe,2FACf,QAAQ,wEACR,cAAc,+SAKd,SAAS,oIACT,gBAAgB,kHAChB,SAAS,yCACT,iBAAiB,2SAIjB,aAAa,kDACb,OAAO;;4FAIF,mCAAmC;kBA9B/C,SAAS;iCACM,IAAI,YACN,kCAAkC,WAEnC;wBACL,IAAI;wBACJ,WAAW;wBACX,sBAAsB;wBACtB,mBAAmB;wBACnB,QAAQ;wBACR,0BAA0B;wBAC1B,eAAe;wBACf,QAAQ;wBACR,cAAc;wBACd,aAAa;wBACb,iBAAiB;wBACjB,QAAQ;wBACR,YAAY;wBACZ,SAAS;wBACT,gBAAgB;wBAChB,SAAS;wBACT,iBAAiB;wBACjB,cAAc;wBACd,eAAe;wBACf,YAAY;wBACZ,aAAa;wBACb,OAAO;qBACV,mBACgB,uBAAuB,CAAC,MAAM","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { isUndefined } from 'lodash-es';\nimport { debounceTime, distinctUntilChanged, filter, map, Observable, pairwise, startWith } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Признак того, что для указанной категории или продукта действует конфигуратор длины.\n     */\n    protected readonly isLengthConfigurator = computed(() => Boolean(this.product().properties?.['isLengthConfigurator']));\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount) ?? 1);\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, @typescript-eslint/no-unnecessary-condition\n        return (min ?? max) ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected totalCost$: Observable<number>;\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n\n        if (this.productIsMeasurable()) {\n            // Добавляем поле ввода длины, если товар измеряемый.\n            this.form.addControl('length', new FormControl<number | null>(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));\n        }\n\n        // Устанавливаем количество из корзины или кратности товара.\n        this.form.patchValue({\n            quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),\n            marker: this.cartItem()?.marker ?? '',\n        });\n\n        const step = this.lengthStep();\n\n        if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {\n            // Если есть конфигуратор длины, рассчитываем начально значение длины.\n            this.form.controls.length?.patchValue(this.form.controls.quantity.value * step);\n        }\n\n        // Добавляем валидацию шага для количества.\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        // Добавляем валидацию шага для длины.\n        this.form.get('length')?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        // Считаем итоговую стоимость заказа.\n        this.totalCost$ = tuiControlValue<{\n            length?: number;\n            quantity: number;\n        }>(this.form).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            filter(() => this.form.valid),\n            distinctUntilChanged((previous, current) => previous.length === current.length && previous.quantity === current.quantity), // Только при изменении значений\n            startWith(this.form.value as { length?: number; quantity: number }),\n            pairwise(),\n            map(([previous, current]) => {\n                const lengthControl = this.form.get('length');\n\n                // Если нет конфигуратора длины или некорректные данные — возвращаем без изменений.\n                if (!this.isLengthConfigurator() || isUndefined(current.length) || !lengthControl || this.form.invalid || !step) {\n                    return current;\n                }\n\n                // Если изменилось количество — пересчитываем длину.\n                if (previous.quantity !== current.quantity && this.form.controls.quantity.value) {\n                    const newValue = this.form.controls.quantity.value * step;\n                    lengthControl.patchValue(newValue);\n                }\n\n                // Если изменилась длина — пересчитываем количество и возвращаем значение длины.\n                if (previous.length !== current.length && lengthControl.value) {\n                    const newValue = lengthControl.value / step;\n                    this.form.controls.quantity.patchValue(newValue);\n\n                    return { length: step, quantity: newValue };\n                }\n\n                // Возвращаем итоговые значения для расчёта стоимости.\n                return { length: step, quantity: current.quantity };\n            }),\n            // Считаем стоимость: цена * длина * количество.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * (length ?? 1) * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        );\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        const lengthControl = this.form.get('length');\n\n        if (!this.productIsMeasurable() || !lengthControl) {\n            return;\n        }\n\n        const length = lengthControl.value ?? 0;\n        let newLength = Math.max(this.minLength(), length + step);\n        // Округляем до ближайшего кратного числа.\n        newLength = Math.round(newLength / step) * step;\n        lengthControl.setValue(Number(newLength.toFixed(2)));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        // Удаляем null-поля из значения формы.\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        // Если это добавление нового товара — добавляем productId.\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        // Если товар измеряемый и есть конфигуратор длины — устанавливаем длину по шагу (количество уже будет вычислено по формуле Метраж/min-count)\n        if (this.productIsMeasurable() && this.isLengthConfigurator() && step) {\n            value.length = step;\n        }\n\n        // Помечаем все поля формы как \"touch\" и запускаем валидацию для отображения ошибок с сервера.\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        // В зависимости от режима — редактируем или добавляем товар.\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    @let step = lengthStep();\n\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }}\n        @if (!step || maxLength()) {\n            {{ lengthHint() }}\n        }\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            <tui-chip\n                *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost$ | async;\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount | async }}\n                {{ product().currency }}\n            </span>\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|
233
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { isUndefined } from 'lodash-es';\nimport { debounceTime, distinctUntilChanged, filter, map, Observable, pairwise, startWith } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Признак того, что для указанной категории или продукта действует конфигуратор длины.\n     */\n    protected readonly isLengthConfigurator = computed(() => Boolean(this.product().properties?.['isLengthConfigurator']));\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, sonarjs/prefer-nullish-coalescing\n        return min || max ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected totalCost$: Observable<number>;\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n\n        if (this.productIsMeasurable()) {\n            // Добавляем поле ввода длины, если товар измеряемый.\n            this.form.addControl('length', new FormControl<number | null>(this.cartItem()?.length ?? this.minLength() ?? 0.01, [Validators.required, Validators.min(0.01)]));\n        }\n\n        // Устанавливаем количество из корзины или кратности товара.\n        this.form.patchValue({\n            quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),\n            marker: this.cartItem()?.marker ?? '',\n        });\n\n        const step = this.lengthStep();\n\n        if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {\n            // Если есть конфигуратор длины, рассчитываем начально значение длины.\n            this.form.controls.length?.patchValue(this.form.controls.quantity.value * step);\n        }\n\n        // Добавляем валидацию шага для количества.\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        // Добавляем валидацию шага для длины.\n        this.form.get('length')?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        // Считаем итоговую стоимость заказа.\n        this.totalCost$ = tuiControlValue<{\n            length?: number;\n            quantity: number;\n        }>(this.form).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            filter(() => this.form.valid),\n            distinctUntilChanged((previous, current) => previous.length === current.length && previous.quantity === current.quantity), // Только при изменении значений\n            startWith(this.form.value as { length?: number; quantity: number }),\n            pairwise(),\n            map(([previous, current]) => {\n                const lengthControl = this.form.get('length');\n\n                // Если нет конфигуратора длины или некорректные данные — возвращаем без изменений.\n                if (!this.isLengthConfigurator() || isUndefined(current.length) || !lengthControl || this.form.invalid || !step) {\n                    return current;\n                }\n\n                // Если изменилось количество — пересчитываем длину.\n                if (previous.quantity !== current.quantity && this.form.controls.quantity.value) {\n                    const newValue = this.form.controls.quantity.value * step;\n                    lengthControl.patchValue(newValue);\n                }\n\n                // Если изменилась длина — пересчитываем количество и возвращаем значение длины.\n                if (previous.length !== current.length && lengthControl.value) {\n                    const newValue = lengthControl.value / step;\n                    this.form.controls.quantity.patchValue(newValue);\n\n                    return { length: step, quantity: newValue };\n                }\n\n                // Возвращаем итоговые значения для расчёта стоимости.\n                return { length: step, quantity: current.quantity };\n            }),\n            // Считаем стоимость: цена * длина * количество.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * (length ?? 1) * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        );\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        const lengthControl = this.form.get('length');\n\n        if (!this.productIsMeasurable() || !lengthControl) {\n            return;\n        }\n\n        const length = lengthControl.value ?? 0;\n        let newLength = Math.max(this.minLength() ?? step, length + step);\n        // Округляем до ближайшего кратного числа.\n        newLength = Math.round(newLength / step) * step;\n        lengthControl.setValue(Number(newLength.toFixed(2)));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        // Удаляем null-поля из значения формы.\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        // Если это добавление нового товара — добавляем productId.\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        // Если товар измеряемый и есть конфигуратор длины — устанавливаем длину по шагу (количество уже будет вычислено по формуле Метраж/min-count)\n        if (this.productIsMeasurable() && this.isLengthConfigurator() && step) {\n            value.length = step;\n        }\n\n        // Помечаем все поля формы как \"touch\" и запускаем валидацию для отображения ошибок с сервера.\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        // В зависимости от режима — редактируем или добавляем товар.\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    @let step = lengthStep();\n\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }}\n        @if (!step || maxLength()) {\n            {{ lengthHint() }}\n        }\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            <tui-chip\n                *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost$ | async;\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount | async }}\n                {{ product().currency }}\n            </span>\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|
package/esm2022/cart/index.mjs
CHANGED
@@ -1,4 +1,4 @@
|
|
1
1
|
export * from './cart-item/sc-cart-item.component';
|
2
|
-
export * from './sc-
|
2
|
+
export * from './sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component';
|
3
3
|
export * from './add-or-editing-cart-item-dialog';
|
4
|
-
//# sourceMappingURL=data:application/json;base64,
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9DQUFvQyxDQUFDO0FBQ25ELGNBQWMsdUZBQXVGLENBQUM7QUFDdEcsY0FBYyxtQ0FBbUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vY2FydC1pdGVtL3NjLWNhcnQtaXRlbS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1jYXJ0LWFkZC1wcm9kdWN0cy1mcm9tLWNzdi1kaWFsb2cvc2MtY2FydC1hZGQtcHJvZHVjdHMtZnJvbS1jc3YtZGlhbG9nLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2FkZC1vci1lZGl0aW5nLWNhcnQtaXRlbS1kaWFsb2cnO1xuIl19
|
@@ -0,0 +1,110 @@
|
|
1
|
+
import { NgIf } from '@angular/common';
|
2
|
+
import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
|
3
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
4
|
+
import { FormControl, ReactiveFormsModule } from '@angular/forms';
|
5
|
+
import { WA_WINDOW } from '@ng-web-apis/common';
|
6
|
+
import { ScCartService, ScUploadedFile } from '@snabcentr/client-core';
|
7
|
+
import { tuiIsPresent } from '@taiga-ui/cdk';
|
8
|
+
import { TuiButton, TuiLink, TuiLoader } from '@taiga-ui/core';
|
9
|
+
import { TuiButtonLoading, TuiFiles } from '@taiga-ui/kit';
|
10
|
+
import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
|
11
|
+
import { filter, map, Observable, share, startWith, Subject, switchMap, tap } from 'rxjs';
|
12
|
+
import * as i0 from "@angular/core";
|
13
|
+
import * as i1 from "@taiga-ui/kit";
|
14
|
+
import * as i2 from "@angular/forms";
|
15
|
+
/**
|
16
|
+
* Компонент диалога для добавления товаров в корзину из CSV файла.
|
17
|
+
*/
|
18
|
+
export class ScCartAddProductsFromCsvDialogComponent {
|
19
|
+
constructor() {
|
20
|
+
/**
|
21
|
+
* Контекст диалогового окна, в котором открыт компонент.
|
22
|
+
*/
|
23
|
+
this.context = inject(POLYMORPHEUS_CONTEXT);
|
24
|
+
/**
|
25
|
+
* Сервис для работы с корзиной.
|
26
|
+
*/
|
27
|
+
this.cartService = inject(ScCartService);
|
28
|
+
/**
|
29
|
+
* Объект окна.
|
30
|
+
*/
|
31
|
+
this.window = inject(WA_WINDOW);
|
32
|
+
/**
|
33
|
+
* {@link Subject} события скачивания цен каталога.
|
34
|
+
*/
|
35
|
+
this.onDownloadClick = new Subject();
|
36
|
+
/**
|
37
|
+
* {@link Observable} запроса на скачивание цен каталога.
|
38
|
+
*/
|
39
|
+
this.downloadRequest$ = this.onDownloadClick.pipe(switchMap(() => this.cartService.getCartCsvExample$().pipe(tap((blob) => {
|
40
|
+
this.downloadExampleFile(blob);
|
41
|
+
}), startWith(null))), share(),
|
42
|
+
// eslint-disable-next-line unicorn/no-useless-undefined
|
43
|
+
startWith(undefined));
|
44
|
+
/**
|
45
|
+
* Признак того, что запрос выполняется.
|
46
|
+
*/
|
47
|
+
this.isDownloadLoading = toSignal(this.downloadRequest$.pipe(map((value) => value === null)), {
|
48
|
+
initialValue: false,
|
49
|
+
});
|
50
|
+
/**
|
51
|
+
* {@link Subject} события отправки формы.
|
52
|
+
*/
|
53
|
+
this.onSubmit$ = new Subject();
|
54
|
+
/**
|
55
|
+
* {@link Observable} запроса на добавление товаров из CSV файла.
|
56
|
+
*/
|
57
|
+
this.submitRequest$ = this.onSubmit$.pipe(map(() => this.control.value), filter(tuiIsPresent), switchMap((file) => new Observable((observer) => {
|
58
|
+
const reader = new FileReader();
|
59
|
+
reader.addEventListener('load', () => {
|
60
|
+
observer.next(new ScUploadedFile(file.name, reader.result));
|
61
|
+
});
|
62
|
+
reader.addEventListener('error', (e) => {
|
63
|
+
observer.error(e);
|
64
|
+
});
|
65
|
+
reader.readAsDataURL(file);
|
66
|
+
return () => {
|
67
|
+
reader.abort();
|
68
|
+
};
|
69
|
+
})), switchMap((file) => this.cartService.addProductsFromCsv$(file).pipe(tap(() => {
|
70
|
+
this.context.$implicit.complete();
|
71
|
+
}), startWith(null))));
|
72
|
+
/**
|
73
|
+
* Признак того, что запрос на добавление товаров выполняется.
|
74
|
+
*/
|
75
|
+
this.isSubmitLoading = toSignal(this.submitRequest$.pipe(map((value) => value === null)), {
|
76
|
+
initialValue: false,
|
77
|
+
});
|
78
|
+
/**
|
79
|
+
* Поле для загрузки файла.
|
80
|
+
*/
|
81
|
+
this.control = new FormControl(null);
|
82
|
+
}
|
83
|
+
/**
|
84
|
+
* Метод для удаления загруженного файла.
|
85
|
+
*/
|
86
|
+
removeFile() {
|
87
|
+
this.control.setValue(null);
|
88
|
+
}
|
89
|
+
/**
|
90
|
+
* Метод для скачивания примера файла.
|
91
|
+
*
|
92
|
+
* @param blob Бинарный объект.
|
93
|
+
*/
|
94
|
+
downloadExampleFile(blob) {
|
95
|
+
const url = this.window.URL.createObjectURL(blob);
|
96
|
+
// используем ссылку и download, чтобы указать название файла.
|
97
|
+
const a = this.window.document.createElement('a');
|
98
|
+
a.href = url;
|
99
|
+
a.download = 'Пример файла добавления товаров в корзину.csv';
|
100
|
+
a.click();
|
101
|
+
this.window.URL.revokeObjectURL(url);
|
102
|
+
}
|
103
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScCartAddProductsFromCsvDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
104
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScCartAddProductsFromCsvDialogComponent, isStandalone: true, selector: "sc-cart-add-products-from-csv-dialog", ngImport: i0, template: "<div class=\"flex flex-col items-center gap-8\">\n <tui-loader\n [overlay]=\"true\"\n [showLoader]=\"isDownloadLoading()\"\n size=\"s\"\n class=\"mt-8\"\n >\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"onDownloadClick.next()\"\n >\n \u041F\u0440\u0438\u043C\u0435\u0440 .csv \u0444\u0430\u0439\u043B\u0430\n </button>\n </tui-loader>\n\n <div class=\"flex w-full flex-col gap-1\">\n <label\n tuiInputFiles\n class=\"w-full\"\n >\n <input\n accept=\"text/csv\"\n tuiInputFiles\n [formControl]=\"control\"\n />\n </label>\n\n <tui-files class=\"tui-space_top-1\">\n <tui-file\n *ngIf=\"control.value as file\"\n [file]=\"file\"\n (remove)=\"removeFile()\"\n />\n </tui-files>\n </div>\n <div class=\"flex gap-2\">\n <button\n tuiButton\n [disabled]=\"!control.value\"\n [loading]=\"isSubmitLoading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n (click)=\"onSubmit$.next()\"\n >\n \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044C\n </button>\n <button\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "component", type: TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "component", type: i1.TuiFile, selector: "tui-file,a[tuiFile],button[tuiFile]", inputs: ["file", "state", "size", "showDelete", "showSize", "leftContent"], outputs: ["remove"] }, { kind: "component", type: i1.TuiInputFiles, selector: "label[tuiInputFiles]" }, { kind: "component", type: i1.TuiFilesComponent, selector: "tui-files", inputs: ["max", "expanded"], outputs: ["expandedChange"] }, { kind: "directive", type: i1.TuiInputFilesDirective, selector: "input[tuiInputFiles]", outputs: ["reject"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
105
|
+
}
|
106
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScCartAddProductsFromCsvDialogComponent, decorators: [{
|
107
|
+
type: Component,
|
108
|
+
args: [{ standalone: true, selector: 'sc-cart-add-products-from-csv-dialog', imports: [TuiLink, TuiLoader, TuiFiles, ReactiveFormsModule, NgIf, TuiButton, TuiButtonLoading], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center gap-8\">\n <tui-loader\n [overlay]=\"true\"\n [showLoader]=\"isDownloadLoading()\"\n size=\"s\"\n class=\"mt-8\"\n >\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"onDownloadClick.next()\"\n >\n \u041F\u0440\u0438\u043C\u0435\u0440 .csv \u0444\u0430\u0439\u043B\u0430\n </button>\n </tui-loader>\n\n <div class=\"flex w-full flex-col gap-1\">\n <label\n tuiInputFiles\n class=\"w-full\"\n >\n <input\n accept=\"text/csv\"\n tuiInputFiles\n [formControl]=\"control\"\n />\n </label>\n\n <tui-files class=\"tui-space_top-1\">\n <tui-file\n *ngIf=\"control.value as file\"\n [file]=\"file\"\n (remove)=\"removeFile()\"\n />\n </tui-files>\n </div>\n <div class=\"flex gap-2\">\n <button\n tuiButton\n [disabled]=\"!control.value\"\n [loading]=\"isSubmitLoading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n (click)=\"onSubmit$.next()\"\n >\n \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044C\n </button>\n <button\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n </div>\n</div>\n" }]
|
109
|
+
}] });
|
110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-add-products-from-csv-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component.ts","../../../../../projects/client-ui/cart/sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, inject, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { WA_WINDOW } from '@ng-web-apis/common';\nimport { ScCartService, ScUploadedFile } from '@snabcentr/client-core';\nimport { tuiIsPresent } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiLink, TuiLoader } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFiles } from '@taiga-ui/kit';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { filter, map, Observable, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\n/**\n * Компонент диалога для добавления товаров в корзину из CSV файла.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-cart-add-products-from-csv-dialog',\n    templateUrl: './sc-cart-add-products-from-csv-dialog.component.html',\n    imports: [TuiLink, TuiLoader, TuiFiles, ReactiveFormsModule, NgIf, TuiButton, TuiButtonLoading],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCartAddProductsFromCsvDialogComponent {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext = inject<TuiDialogContext>(POLYMORPHEUS_CONTEXT);\n\n    /**\n     * Сервис для работы с корзиной.\n     */\n    private readonly cartService: ScCartService = inject(ScCartService);\n\n    /**\n     * Объект окна.\n     */\n    private readonly window: Window & typeof globalThis = inject<Window & typeof globalThis>(WA_WINDOW);\n\n    /**\n     * {@link Subject} события скачивания цен каталога.\n     */\n    protected readonly onDownloadClick: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на скачивание цен каталога.\n     */\n    protected readonly downloadRequest$ = this.onDownloadClick.pipe(\n        switchMap(() =>\n            this.cartService.getCartCsvExample$().pipe(\n                tap((blob: Blob) => {\n                    this.downloadExampleFile(blob);\n                }),\n                startWith(null)\n            )\n        ),\n        share(),\n        // eslint-disable-next-line unicorn/no-useless-undefined\n        startWith(undefined)\n    );\n\n    /**\n     * Признак того, что запрос выполняется.\n     */\n    protected readonly isDownloadLoading: Signal<boolean> = toSignal(this.downloadRequest$.pipe(map((value) => value === null)), {\n        initialValue: false,\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit$: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на добавление товаров из CSV файла.\n     */\n    protected readonly submitRequest$ = this.onSubmit$.pipe(\n        map(() => this.control.value),\n        filter(tuiIsPresent),\n        switchMap(\n            (file: File) =>\n                new Observable<ScUploadedFile>((observer) => {\n                    const reader = new FileReader();\n\n                    reader.addEventListener('load', () => {\n                        observer.next(new ScUploadedFile(file.name, reader.result as string));\n                    });\n\n                    reader.addEventListener('error', (e) => {\n                        observer.error(e);\n                    });\n\n                    reader.readAsDataURL(file);\n\n                    return () => {\n                        reader.abort();\n                    };\n                })\n        ),\n        switchMap((file) =>\n            this.cartService.addProductsFromCsv$(file).pipe(\n                tap(() => {\n                    this.context.$implicit.complete();\n                }),\n                startWith(null)\n            )\n        )\n    );\n\n    /**\n     * Признак того, что запрос на добавление товаров выполняется.\n     */\n    protected readonly isSubmitLoading: Signal<boolean> = toSignal(this.submitRequest$.pipe(map((value) => value === null)), {\n        initialValue: false,\n    });\n\n    /**\n     * Поле для загрузки файла.\n     */\n    protected readonly control: FormControl<File | null> = new FormControl<File | null>(null);\n\n    /**\n     * Метод для удаления загруженного файла.\n     */\n    protected removeFile(): void {\n        this.control.setValue(null);\n    }\n\n    /**\n     * Метод для скачивания примера файла.\n     *\n     * @param blob Бинарный объект.\n     */\n    private downloadExampleFile(blob: Blob): void {\n        const url = this.window.URL.createObjectURL(blob);\n        // используем ссылку и download, чтобы указать название файла.\n        const a = this.window.document.createElement('a');\n\n        a.href = url;\n        a.download = 'Пример файла добавления товаров в корзину.csv';\n        a.click();\n        this.window.URL.revokeObjectURL(url);\n    }\n}\n","<div class=\"flex flex-col items-center gap-8\">\n    <tui-loader\n        [overlay]=\"true\"\n        [showLoader]=\"isDownloadLoading()\"\n        size=\"s\"\n        class=\"mt-8\"\n    >\n        <button\n            tuiLink\n            [pseudo]=\"true\"\n            type=\"button\"\n            (click)=\"onDownloadClick.next()\"\n        >\n            Пример .csv файла\n        </button>\n    </tui-loader>\n\n    <div class=\"flex w-full flex-col gap-1\">\n        <label\n            tuiInputFiles\n            class=\"w-full\"\n        >\n            <input\n                accept=\"text/csv\"\n                tuiInputFiles\n                [formControl]=\"control\"\n            />\n        </label>\n\n        <tui-files class=\"tui-space_top-1\">\n            <tui-file\n                *ngIf=\"control.value as file\"\n                [file]=\"file\"\n                (remove)=\"removeFile()\"\n            />\n        </tui-files>\n    </div>\n    <div class=\"flex gap-2\">\n        <button\n            tuiButton\n            [disabled]=\"!control.value\"\n            [loading]=\"isSubmitLoading()\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n            (click)=\"onSubmit$.next()\"\n        >\n            Загрузить\n        </button>\n        <button\n            tuiButton\n            (click)=\"context.$implicit.complete()\"\n            type=\"button\"\n            appearance=\"secondary\"\n        >\n            Отмена\n        </button>\n    </div>\n</div>\n"]}
|
@@ -4190,7 +4190,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
4190
4190
|
/**
|
4191
4191
|
* Компонент диалога для добавления товаров в корзину из CSV файла.
|
4192
4192
|
*/
|
4193
|
-
class
|
4193
|
+
class ScCartAddProductsFromCsvDialogComponent {
|
4194
4194
|
constructor() {
|
4195
4195
|
/**
|
4196
4196
|
* Контекст диалогового окна, в котором открыт компонент.
|
@@ -4275,12 +4275,12 @@ class ScCarAddProductsFromCsvDialogComponent {
|
|
4275
4275
|
a.click();
|
4276
4276
|
this.window.URL.revokeObjectURL(url);
|
4277
4277
|
}
|
4278
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type:
|
4279
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type:
|
4278
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScCartAddProductsFromCsvDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
4279
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScCartAddProductsFromCsvDialogComponent, isStandalone: true, selector: "sc-cart-add-products-from-csv-dialog", ngImport: i0, template: "<div class=\"flex flex-col items-center gap-8\">\n <tui-loader\n [overlay]=\"true\"\n [showLoader]=\"isDownloadLoading()\"\n size=\"s\"\n class=\"mt-8\"\n >\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"onDownloadClick.next()\"\n >\n \u041F\u0440\u0438\u043C\u0435\u0440 .csv \u0444\u0430\u0439\u043B\u0430\n </button>\n </tui-loader>\n\n <div class=\"flex w-full flex-col gap-1\">\n <label\n tuiInputFiles\n class=\"w-full\"\n >\n <input\n accept=\"text/csv\"\n tuiInputFiles\n [formControl]=\"control\"\n />\n </label>\n\n <tui-files class=\"tui-space_top-1\">\n <tui-file\n *ngIf=\"control.value as file\"\n [file]=\"file\"\n (remove)=\"removeFile()\"\n />\n </tui-files>\n </div>\n <div class=\"flex gap-2\">\n <button\n tuiButton\n [disabled]=\"!control.value\"\n [loading]=\"isSubmitLoading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n (click)=\"onSubmit$.next()\"\n >\n \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044C\n </button>\n <button\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "component", type: TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "component", type: i2$1.TuiFile, selector: "tui-file,a[tuiFile],button[tuiFile]", inputs: ["file", "state", "size", "showDelete", "showSize", "leftContent"], outputs: ["remove"] }, { kind: "component", type: i2$1.TuiInputFiles, selector: "label[tuiInputFiles]" }, { kind: "component", type: i2$1.TuiFilesComponent, selector: "tui-files", inputs: ["max", "expanded"], outputs: ["expandedChange"] }, { kind: "directive", type: i2$1.TuiInputFilesDirective, selector: "input[tuiInputFiles]", outputs: ["reject"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2$3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2$3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2$3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
4280
4280
|
}
|
4281
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type:
|
4281
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScCartAddProductsFromCsvDialogComponent, decorators: [{
|
4282
4282
|
type: Component,
|
4283
|
-
args: [{ standalone: true, selector: 'sc-
|
4283
|
+
args: [{ standalone: true, selector: 'sc-cart-add-products-from-csv-dialog', imports: [TuiLink, TuiLoader, TuiFiles, ReactiveFormsModule, NgIf, TuiButton, TuiButtonLoading], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center gap-8\">\n <tui-loader\n [overlay]=\"true\"\n [showLoader]=\"isDownloadLoading()\"\n size=\"s\"\n class=\"mt-8\"\n >\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"onDownloadClick.next()\"\n >\n \u041F\u0440\u0438\u043C\u0435\u0440 .csv \u0444\u0430\u0439\u043B\u0430\n </button>\n </tui-loader>\n\n <div class=\"flex w-full flex-col gap-1\">\n <label\n tuiInputFiles\n class=\"w-full\"\n >\n <input\n accept=\"text/csv\"\n tuiInputFiles\n [formControl]=\"control\"\n />\n </label>\n\n <tui-files class=\"tui-space_top-1\">\n <tui-file\n *ngIf=\"control.value as file\"\n [file]=\"file\"\n (remove)=\"removeFile()\"\n />\n </tui-files>\n </div>\n <div class=\"flex gap-2\">\n <button\n tuiButton\n [disabled]=\"!control.value\"\n [loading]=\"isSubmitLoading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n (click)=\"onSubmit$.next()\"\n >\n \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044C\n </button>\n <button\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n </div>\n</div>\n" }]
|
4284
4284
|
}] });
|
4285
4285
|
|
4286
4286
|
/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */
|
@@ -4323,7 +4323,7 @@ class ScAddOrEditingCartItemFormComponent {
|
|
4323
4323
|
/**
|
4324
4324
|
* Минимальный метраж для товара.
|
4325
4325
|
*/
|
4326
|
-
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount)
|
4326
|
+
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));
|
4327
4327
|
/**
|
4328
4328
|
* Максимальный метраж для товара.
|
4329
4329
|
*/
|
@@ -4338,8 +4338,8 @@ class ScAddOrEditingCartItemFormComponent {
|
|
4338
4338
|
this.lengthHint = computed(() => {
|
4339
4339
|
const min = this.minLength();
|
4340
4340
|
const max = this.maxLength();
|
4341
|
-
// eslint-disable-next-line sonarjs/no-nested-conditional,
|
4342
|
-
return
|
4341
|
+
// eslint-disable-next-line sonarjs/no-nested-conditional, sonarjs/prefer-nullish-coalescing
|
4342
|
+
return min || max ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';
|
4343
4343
|
});
|
4344
4344
|
/**
|
4345
4345
|
* {@link Output} события добавления товара в корзину.
|
@@ -4370,7 +4370,7 @@ class ScAddOrEditingCartItemFormComponent {
|
|
4370
4370
|
});
|
4371
4371
|
if (this.productIsMeasurable()) {
|
4372
4372
|
// Добавляем поле ввода длины, если товар измеряемый.
|
4373
|
-
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));
|
4373
|
+
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength() ?? 0.01, [Validators.required, Validators.min(0.01)]));
|
4374
4374
|
}
|
4375
4375
|
// Устанавливаем количество из корзины или кратности товара.
|
4376
4376
|
this.form.patchValue({
|
@@ -4423,7 +4423,7 @@ class ScAddOrEditingCartItemFormComponent {
|
|
4423
4423
|
return;
|
4424
4424
|
}
|
4425
4425
|
const length = lengthControl.value ?? 0;
|
4426
|
-
let newLength = Math.max(this.minLength(), length + step);
|
4426
|
+
let newLength = Math.max(this.minLength() ?? step, length + step);
|
4427
4427
|
// Округляем до ближайшего кратного числа.
|
4428
4428
|
newLength = Math.round(newLength / step) * step;
|
4429
4429
|
lengthControl.setValue(Number(newLength.toFixed(2)));
|
@@ -7765,5 +7765,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
7765
7765
|
* Generated bundle index. Do not edit.
|
7766
7766
|
*/
|
7767
7767
|
|
7768
|
-
export { AbstractScPriceCard, AuthMethod, CURRENT_COUNTRY_ID, FilesAndDocumentsComponent, FilesAndDocumentsModule, FinishDateTimeTransformerDirective, IS_DEFAULT_COUNTRY, MAX_FILES_IN_FORM_INPUT, SC_ALLOW_SELECT_TERMINATED, SC_DATE_FORMATTER, SC_ERROR_CHANGE_HANDLER, SC_HELP_NOTIFICATION_CLOSE, SC_HELP_NOTIFICATION_LIMIT, SC_MANAGER_QR_HANDLER, SC_PAGE_SIZE_OPTIONS$1 as SC_PAGE_SIZE_OPTIONS, SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT, SC_USER_CITY_INFO, SC_USER_INFO, SC_USER_PROVIDERS, SC_VERIFICATION_CODE_TIMEOUT, ScAccordionComponent, ScAccordionContentDirective, ScAccordionModule, ScAddContactDialogComponent, ScAddContragentBankAccountsDialogComponent, ScAddContragentDialogComponent, ScAddDeliveryAddressDialogComponent, ScAddOrEditingCartItemDialogComponent, ScAddOrEditingCartItemFormComponent, ScAddressesSelectionFieldComponent, ScAuthModule, ScBannerComponent, ScBannerModule, ScBrandsListComponent, ScBrandsListModule,
|
7768
|
+
export { AbstractScPriceCard, AuthMethod, CURRENT_COUNTRY_ID, FilesAndDocumentsComponent, FilesAndDocumentsModule, FinishDateTimeTransformerDirective, IS_DEFAULT_COUNTRY, MAX_FILES_IN_FORM_INPUT, SC_ALLOW_SELECT_TERMINATED, SC_DATE_FORMATTER, SC_ERROR_CHANGE_HANDLER, SC_HELP_NOTIFICATION_CLOSE, SC_HELP_NOTIFICATION_LIMIT, SC_MANAGER_QR_HANDLER, SC_PAGE_SIZE_OPTIONS$1 as SC_PAGE_SIZE_OPTIONS, SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT, SC_USER_CITY_INFO, SC_USER_INFO, SC_USER_PROVIDERS, SC_VERIFICATION_CODE_TIMEOUT, ScAccordionComponent, ScAccordionContentDirective, ScAccordionModule, ScAddContactDialogComponent, ScAddContragentBankAccountsDialogComponent, ScAddContragentDialogComponent, ScAddDeliveryAddressDialogComponent, ScAddOrEditingCartItemDialogComponent, ScAddOrEditingCartItemFormComponent, ScAddressesSelectionFieldComponent, ScAuthModule, ScBannerComponent, ScBannerModule, ScBrandsListComponent, ScBrandsListModule, ScCartAddProductsFromCsvDialogComponent, ScCartItemComponent, ScCatalogModule, ScCategoryCardComponent, ScContactsAccordionComponent, ScContactsModule, ScContragentsAccordionComponent, ScContragentsAccordionItemComponent, ScContragentsModule, ScDeliveryAddressAccordionComponent, ScDeliveryAddressAccordionItemComponent, ScDeliveryAddressModule, ScDownloadPriceListComponent, ScEmailLinkDirective, ScErrorBlockStatusComponent, ScErrorHandlerComponent, ScFavoriteButtonComponent, ScFeedbackFormComponent, ScFormFieldsModule, ScFormatDatePipe, ScFrequentlyAskedQuestionsComponent, ScFrequentlyAskedQuestionsGroupSelectorComponent, ScFrequentlyAskedQuestionsWithGroupsComponent, ScGratitudeComponent, ScHelpNotificationService, ScInputQuantityComponent, ScLinks, ScManagerCardComponent, ScManagerCardPushComponent, ScNewContactFormComponent, ScNewContragentBankAccountsFormComponent, ScNewContragentFormComponent, ScNewsCardComponent, ScNewsCardSkeletonComponent, ScNewsModule, ScNextInputFocusDirective, ScNextInputFocusModule, ScNotifyWhenInStockDialogComponent, ScOrderItemMobileComponent, ScOrderModule, ScPaymentStatusComponent, ScPhoneFormatPipe, ScPreviewSampleComponent, ScPreviewSampleModule, ScPreviewSamplesMosquitoComponent, ScPriceCardComponent, ScPriceCardInlineComponent, ScPriceHistoryComponent, ScPriceListPaginationComponent, ScPriceWarehouseStockComponent, ScProductInAllWarehousesPipe, ScProfileAccordionsContentComponent, ScProfileModule, ScQRCodeDialogComponent, ScQRCodeModule, ScResetUserPasswordComponent, ScResourcePreviewComponent, ScSelectOnFocusinDirective, ScShareButtonComponent, ScShareButtonModule, ScSignInFormByEmailComponent, ScSignInFormByPhoneComponent, ScSignInFormComponent, ScSignUpFormComponent, ScSimpleSignUpFormComponent, ScSuggestionFieldComponent, ScTelLinkDirective, ScTerminalLinkDirective, ScUpdateUserInfoDialogComponent, ScUserManagersComponent, ScUserModule, ScUserPhoneApproveDialogComponent, ScVerificationModule, ScVerificationPhoneCheckFormComponent, TreeDirective, TreeIconService, TreeLoaderService, TreeTopDirective, phoneValidator, scAtLeastOneRequiredValidator, scBicValidator, scClientUiIconsName, scCorrespondentAccountValidator, scPasswordConfirmMatchingValidator, stepValidator, tuiDateValueTransformerDefaultProvider };
|
7769
7769
|
//# sourceMappingURL=snabcentr-client-ui.mjs.map
|