@snabcentr/client-ui 4.11.8 → 4.15.0
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/auth/sc-auth.module.d.ts +2 -1
- package/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.d.ts +19 -8
- package/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.d.ts +6 -7
- package/auth/sc-simple-sign-up-form/sc-simple-sign-up-form.component.d.ts +9 -2
- package/auth/sign-up-form/sc-sign-up-form.component.d.ts +14 -9
- package/catalog/catalog-filters/index.d.ts +3 -0
- package/catalog/catalog-filters/sc-catalog-filters.component.d.ts +99 -0
- package/catalog/catalog-filters/tokens/sc-catalog-products-filters.d.ts +7 -0
- package/catalog/catalog-filters/tokens/sc-catalog-show-products-recursively.d.ts +5 -0
- package/catalog/index.d.ts +1 -0
- package/catalog/input-quantity/sc-input-quantity.component.d.ts +2 -2
- package/configurators/index.d.ts +2 -0
- package/configurators/models/index.d.ts +2 -0
- package/configurators/models/sandwich/sc-i-configurator-search-product-sandwich.d.ts +22 -0
- package/configurators/models/sandwich/sc-i-sandwich-settings.d.ts +18 -0
- package/configurators/sandwich/index.d.ts +3 -0
- package/configurators/sandwich/sandwich-skeleton/sc-sandwich-skeleton.component.d.ts +8 -0
- package/configurators/sandwich/sc-i-new-cart-item-sandwich.d.ts +14 -0
- package/configurators/sandwich/sc-sandwich.component.d.ts +146 -0
- package/contragents/add-contragent-dialog/sc-add-contragent-dialog.component.d.ts +21 -24
- package/contragents/new-contragent-form/sc-new-contragent-form.component.d.ts +12 -18
- package/directives/select-on-focusin/sc-select-on-focusin.directive.d.ts +1 -1
- package/esm2022/auth/sc-auth.module.mjs +11 -5
- package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +37 -18
- package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +20 -18
- package/esm2022/auth/sc-simple-sign-up-form/sc-simple-sign-up-form.component.mjs +20 -5
- package/esm2022/auth/sign-up-form/sc-sign-up-form.component.mjs +37 -36
- package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +4 -4
- package/esm2022/cart/cart-item/sc-cart-item.component.mjs +3 -3
- package/esm2022/catalog/catalog-filters/index.mjs +4 -0
- package/esm2022/catalog/catalog-filters/sc-catalog-filters.component.mjs +202 -0
- package/esm2022/catalog/catalog-filters/tokens/sc-catalog-products-filters.mjs +10 -0
- package/esm2022/catalog/catalog-filters/tokens/sc-catalog-show-products-recursively.mjs +6 -0
- package/esm2022/catalog/category-card/sc-category-card.component.mjs +3 -3
- package/esm2022/catalog/index.mjs +2 -1
- package/esm2022/catalog/input-quantity/sc-input-quantity.component.mjs +19 -7
- package/esm2022/catalog/price-warehouse-stock/sc-price-warehouse-stock.component.mjs +3 -3
- package/esm2022/configurators/index.mjs +3 -0
- package/esm2022/configurators/models/index.mjs +3 -0
- package/esm2022/configurators/models/sandwich/sc-i-configurator-search-product-sandwich.mjs +2 -0
- package/esm2022/configurators/models/sandwich/sc-i-sandwich-settings.mjs +2 -0
- package/esm2022/configurators/sandwich/index.mjs +4 -0
- package/esm2022/configurators/sandwich/sandwich-skeleton/sc-sandwich-skeleton.component.mjs +14 -0
- package/esm2022/configurators/sandwich/sc-i-new-cart-item-sandwich.mjs +2 -0
- package/esm2022/configurators/sandwich/sc-sandwich.component.mjs +322 -0
- package/esm2022/contragents/add-contragent-dialog/sc-add-contragent-dialog.component.mjs +38 -50
- package/esm2022/contragents/new-contragent-form/sc-new-contragent-form.component.mjs +28 -46
- package/esm2022/directives/select-on-focusin/sc-select-on-focusin.directive.mjs +3 -3
- package/esm2022/methods/index.mjs +2 -0
- package/esm2022/methods/sc-get-current-route.mjs +14 -0
- package/esm2022/noindex-wrapper/sc-noindex-wrapper.component.mjs +5 -3
- package/esm2022/pages/frequently-asked-questions-with-groups/sc-frequently-asked-questions-with-groups.component.mjs +3 -2
- package/esm2022/providers/index.mjs +3 -1
- package/esm2022/providers/sc-category.providers.mjs +43 -0
- package/esm2022/providers/sc-debounce-time-default.mjs +9 -0
- package/esm2022/public-api.mjs +3 -1
- package/esm2022/samples/preview-sample/sc-preview-sample.component.mjs +3 -3
- package/esm2022/user/user-managers/sc-user-managers.component.mjs +22 -24
- package/esm2022/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +14 -14
- package/fesm2022/snabcentr-client-ui.mjs +976 -386
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/methods/index.d.ts +1 -0
- package/methods/sc-get-current-route.d.ts +8 -0
- package/package.json +19 -19
- package/providers/index.d.ts +2 -0
- package/providers/sc-category.providers.d.ts +11 -0
- package/providers/sc-debounce-time-default.d.ts +5 -0
- package/public-api.d.ts +2 -0
- package/styles/tailwind/tailwind.scss +76 -4
- package/user/user-managers/sc-user-managers.component.d.ts +7 -10
|
@@ -0,0 +1,322 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/unbound-method */
|
|
2
|
+
import { AsyncPipe } from '@angular/common';
|
|
3
|
+
import { HttpErrorResponse } from '@angular/common/http';
|
|
4
|
+
import { ChangeDetectionStrategy, Component, DestroyRef, inject, input, output, signal } from '@angular/core';
|
|
5
|
+
import { takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
6
|
+
import { FormArray, FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';
|
|
7
|
+
import { ScCartService, ScCatalogService, ScConfiguratorService, ScConvertersService, } from '@snabcentr/client-core';
|
|
8
|
+
import { tuiControlValue, tuiIsFalsy, tuiIsPresent } from '@taiga-ui/cdk';
|
|
9
|
+
import { TuiButton, TuiDataList, TuiError, tuiItemsHandlersProvider, TuiLabel, TuiLoader, TuiNumberFormat, tuiNumberFormatProvider, TuiTextfield, } from '@taiga-ui/core';
|
|
10
|
+
import { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';
|
|
11
|
+
import { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';
|
|
12
|
+
import { TuiChevron, TuiDataListWrapperComponent, TuiFieldErrorPipe, TuiInputNumber, tuiInputNumberOptionsProvider, TuiInputSlider, TuiSelect } from '@taiga-ui/kit';
|
|
13
|
+
import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
|
|
14
|
+
import { catchError, combineLatest, debounceTime, filter, finalize, map, of, share, startWith, switchMap, tap } from 'rxjs';
|
|
15
|
+
import { ScNextInputFocusModule } from '../../directives/next-input-focus/sc-next-input-focus.module';
|
|
16
|
+
import { ScSelectOnFocusinDirective } from '../../directives/select-on-focusin/sc-select-on-focusin.directive';
|
|
17
|
+
import { SC_DEBOUNCE_TIME_DEFAULT } from '../../providers';
|
|
18
|
+
import { stepValidator } from '../../validators/step-validator';
|
|
19
|
+
import * as i0 from "@angular/core";
|
|
20
|
+
import * as i1 from "@angular/forms";
|
|
21
|
+
import * as i2 from "../../directives/next-input-focus/sc-next-input-focus.directive";
|
|
22
|
+
import * as i3 from "@taiga-ui/core";
|
|
23
|
+
import * as i4 from "@taiga-ui/kit";
|
|
24
|
+
import * as i5 from "@taiga-ui/kit/components/slider";
|
|
25
|
+
/**
|
|
26
|
+
* Конфигуратор распила сэндвич-панелей.
|
|
27
|
+
*/
|
|
28
|
+
export class ScSandwichComponent {
|
|
29
|
+
constructor() {
|
|
30
|
+
/**
|
|
31
|
+
* Данные настроек окружения.
|
|
32
|
+
*/
|
|
33
|
+
this.debounceTimeDefault = inject(SC_DEBOUNCE_TIME_DEFAULT);
|
|
34
|
+
/**
|
|
35
|
+
* Настройки для конфигуратора сэндвич-панелей.
|
|
36
|
+
*/
|
|
37
|
+
this.settings = input();
|
|
38
|
+
/**
|
|
39
|
+
* Идентификатор категории.
|
|
40
|
+
*/
|
|
41
|
+
this.categoryId = input.required();
|
|
42
|
+
/**
|
|
43
|
+
* Название конфигуратора.
|
|
44
|
+
*/
|
|
45
|
+
this.editor = input.required();
|
|
46
|
+
/**
|
|
47
|
+
* Позиция товара/услуги в корзине.
|
|
48
|
+
*/
|
|
49
|
+
this.cartItem = input.required();
|
|
50
|
+
/**
|
|
51
|
+
* Событие переключения отображения товаров.
|
|
52
|
+
*/
|
|
53
|
+
this.toggleShowEvent = output();
|
|
54
|
+
/**
|
|
55
|
+
* Форма добавления/редактирования продукта в корзине.
|
|
56
|
+
*/
|
|
57
|
+
this.form = new FormGroup({
|
|
58
|
+
productCategoryId: new FormControl(null, { validators: Validators.required, nonNullable: false }),
|
|
59
|
+
configurator: new FormControl(null, { validators: Validators.required, nonNullable: false }),
|
|
60
|
+
quantity: new FormControl(null, { validators: [Validators.required, stepValidator(1)], nonNullable: false }),
|
|
61
|
+
marker: new FormControl('', { nonNullable: true }),
|
|
62
|
+
items: new FormArray([]),
|
|
63
|
+
calculationId: new FormControl(null, { nonNullable: false }),
|
|
64
|
+
});
|
|
65
|
+
/**
|
|
66
|
+
* Сервис для работы с корзиной.
|
|
67
|
+
*/
|
|
68
|
+
this.cartService = inject(ScCartService);
|
|
69
|
+
/**
|
|
70
|
+
* Сервис конвертации данных.
|
|
71
|
+
*/
|
|
72
|
+
this.convertersService = inject(ScConvertersService);
|
|
73
|
+
/**
|
|
74
|
+
* Контекст диалогового окна, в котором открыт компонент.
|
|
75
|
+
*/
|
|
76
|
+
this.context = inject(POLYMORPHEUS_CONTEXT, { optional: true });
|
|
77
|
+
/**
|
|
78
|
+
* Сервис для работы с каталогом.
|
|
79
|
+
*/
|
|
80
|
+
this.catalogService = inject(ScCatalogService);
|
|
81
|
+
/**
|
|
82
|
+
* Сервис для работы с конфигураторами.
|
|
83
|
+
*/
|
|
84
|
+
this.configuratorService = inject(ScConfiguratorService);
|
|
85
|
+
/**
|
|
86
|
+
* {@link Observable} списка товаров категории.
|
|
87
|
+
*/
|
|
88
|
+
this.products$ = toObservable(this.categoryId).pipe(switchMap((categoryId) => {
|
|
89
|
+
if (!categoryId) {
|
|
90
|
+
return of([]);
|
|
91
|
+
}
|
|
92
|
+
return this.catalogService.getProductsCategoryCached$({ categoryId: Number(categoryId) }).pipe(map((products) => products ?? []));
|
|
93
|
+
}));
|
|
94
|
+
/**
|
|
95
|
+
* Выбранный продукт (материал для раскроя).
|
|
96
|
+
*/
|
|
97
|
+
this.product = new FormControl(null, { validators: Validators.required, nonNullable: false });
|
|
98
|
+
/**
|
|
99
|
+
* {@link Observable} расчета оптимизации раскладки.
|
|
100
|
+
*/
|
|
101
|
+
this.calculate$ = combineLatest({
|
|
102
|
+
productId: tuiControlValue(this.product).pipe(map((product) => product?.id ?? null)),
|
|
103
|
+
items: tuiControlValue(this.items),
|
|
104
|
+
}).pipe(debounceTime(this.debounceTimeDefault), filter(() => this.product.valid && this.form.controls.items.valid), switchMap(({ productId, items }) => {
|
|
105
|
+
if (productId === null || items.length === 0) {
|
|
106
|
+
return of(null);
|
|
107
|
+
}
|
|
108
|
+
return this.configuratorService.calculate$(this.categoryId(), this.editor(), { productId, items }).pipe(tap((calculateResult) => {
|
|
109
|
+
this.form.get('quantity')?.setValue(calculateResult.quantity);
|
|
110
|
+
}), startWith(null), catchError(() => of(null)));
|
|
111
|
+
}), share(), tap((calculateResult) => {
|
|
112
|
+
console.log('calculateResult', calculateResult);
|
|
113
|
+
}));
|
|
114
|
+
/**
|
|
115
|
+
* Результат расчета оптимизации раскладки.
|
|
116
|
+
*/
|
|
117
|
+
this.calculateResult$ = this.calculate$.pipe(filter(tuiIsPresent));
|
|
118
|
+
/**
|
|
119
|
+
* Сигнал состояния загрузки расчета.
|
|
120
|
+
*/
|
|
121
|
+
this.isCalculateLoading = toSignal(this.calculate$.pipe(map(tuiIsFalsy)), { initialValue: false });
|
|
122
|
+
/**
|
|
123
|
+
* Валидатор для ширины изделия.
|
|
124
|
+
*/
|
|
125
|
+
this.validatorWidth$ = tuiControlValue(this.product).pipe(map((product) => Validators.min(product?.properties?.width ?? 0)));
|
|
126
|
+
/**
|
|
127
|
+
* Валидатор для длины изделия.
|
|
128
|
+
*/
|
|
129
|
+
this.validatorLength$ = tuiControlValue(this.product).pipe(map((product) => Validators.min(product?.properties?.length ?? 0)));
|
|
130
|
+
/**
|
|
131
|
+
* Признак того, что выполняется запрос на отправку данных выполняется.
|
|
132
|
+
*/
|
|
133
|
+
this.isSubmitLoading = signal(false);
|
|
134
|
+
/**
|
|
135
|
+
* Ссылка для автоматического управления уничтожением зависимостей.
|
|
136
|
+
*/
|
|
137
|
+
this.destroyRef = inject(DestroyRef);
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Массив изделий.
|
|
141
|
+
*/
|
|
142
|
+
get items() {
|
|
143
|
+
return this.form.controls.items;
|
|
144
|
+
}
|
|
145
|
+
/** @inheritDoc */
|
|
146
|
+
ngOnInit() {
|
|
147
|
+
this.setDefaultFormValue();
|
|
148
|
+
this.product.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((product) => {
|
|
149
|
+
if (product) {
|
|
150
|
+
this.items.controls.forEach((item) => {
|
|
151
|
+
const value = item.getRawValue();
|
|
152
|
+
if ((product.properties?.width && value.width < product.properties.width) || (product.properties?.length && value.length < product.properties.length)) {
|
|
153
|
+
item.patchValue({
|
|
154
|
+
width: Math.min(value.width, product.properties.width ?? value.width),
|
|
155
|
+
length: Math.min(value.length, product.properties.length ?? value.length),
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Создает группу для изделия.
|
|
164
|
+
*
|
|
165
|
+
* @param item Параметры изделия.
|
|
166
|
+
*/
|
|
167
|
+
createItemGroup(item) {
|
|
168
|
+
return new FormGroup({
|
|
169
|
+
width: new FormControl(item?.width ?? 0, {
|
|
170
|
+
validators: [Validators.required, Validators.min(this.settings()?.minWidth ?? 0.01)],
|
|
171
|
+
nonNullable: true,
|
|
172
|
+
}),
|
|
173
|
+
length: new FormControl(item?.length ?? 0, {
|
|
174
|
+
validators: [Validators.required, Validators.min(this.settings()?.minLength ?? 0.01)],
|
|
175
|
+
nonNullable: true,
|
|
176
|
+
}),
|
|
177
|
+
quantity: new FormControl(item?.quantity ?? 1, {
|
|
178
|
+
validators: [Validators.required, Validators.min(1)],
|
|
179
|
+
nonNullable: true,
|
|
180
|
+
}),
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Добавляет новое изделие в форму.
|
|
185
|
+
*/
|
|
186
|
+
addItem() {
|
|
187
|
+
this.items.push(this.createItemGroup());
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* Удаляет изделие из формы.
|
|
191
|
+
*
|
|
192
|
+
* @param index Индекс изделия.
|
|
193
|
+
*/
|
|
194
|
+
removeItem(index) {
|
|
195
|
+
this.items.removeAt(index);
|
|
196
|
+
}
|
|
197
|
+
/**
|
|
198
|
+
* Возвращает итоговую стоимость.
|
|
199
|
+
*
|
|
200
|
+
* @param calculateResult Результат расчета оптимизации раскладки.
|
|
201
|
+
*/
|
|
202
|
+
getTotalCost(calculateResult) {
|
|
203
|
+
if (!this.product.value?.costRub) {
|
|
204
|
+
return 0;
|
|
205
|
+
}
|
|
206
|
+
const quantity = this.form.get('quantity')?.value ?? 1;
|
|
207
|
+
const totalCost = calculateResult.quantity * this.product.value.costRub * quantity;
|
|
208
|
+
return Math.round(totalCost * 100) / 100;
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* Обработчик добавления/редактирования продукта в корзине.
|
|
212
|
+
*
|
|
213
|
+
* @param calculateResult Результат расчета оптимизации раскладки.
|
|
214
|
+
*/
|
|
215
|
+
onSubmit(calculateResult) {
|
|
216
|
+
if (!calculateResult || this.form.invalid || !this.isCalculateLoading()) {
|
|
217
|
+
return;
|
|
218
|
+
}
|
|
219
|
+
const formValue = { ...this.form.value, productId: this.product.value?.id ?? null, calculationId: calculateResult.id };
|
|
220
|
+
// Удаляем items, так как они не нужны в {@link ScINewCartItemSandwich}.
|
|
221
|
+
delete formValue.items;
|
|
222
|
+
const cartItem = this.cartItem();
|
|
223
|
+
const newCartItem$ = cartItem
|
|
224
|
+
? this.cartService.patchCartItem$(cartItem.id, this.convertersService.removeNull(formValue))
|
|
225
|
+
: this.cartService.addToCart$(this.convertersService.removeNull(formValue));
|
|
226
|
+
this.isSubmitLoading.set(true);
|
|
227
|
+
newCartItem$
|
|
228
|
+
.pipe(tap(() => {
|
|
229
|
+
if (this.context) {
|
|
230
|
+
this.context.$implicit.complete();
|
|
231
|
+
}
|
|
232
|
+
else {
|
|
233
|
+
this.setDefaultFormValue();
|
|
234
|
+
}
|
|
235
|
+
}), catchError((error) => {
|
|
236
|
+
if (error instanceof HttpErrorResponse) {
|
|
237
|
+
const { errors, message } = error.error;
|
|
238
|
+
if (errors) {
|
|
239
|
+
Object.keys(errors).forEach((key) => {
|
|
240
|
+
// eslint-disable-next-line security/detect-object-injection
|
|
241
|
+
this.form.get(key)?.setErrors({ serverResponse: errors[key] });
|
|
242
|
+
});
|
|
243
|
+
}
|
|
244
|
+
else if (message) {
|
|
245
|
+
this.form.setErrors({ serverResponse: [message] });
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
return of();
|
|
249
|
+
}), finalize(() => {
|
|
250
|
+
this.isSubmitLoading.set(false);
|
|
251
|
+
}))
|
|
252
|
+
.subscribe();
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Устанавливает начальные значения для полей формы.
|
|
256
|
+
*/
|
|
257
|
+
setDefaultFormValue() {
|
|
258
|
+
this.form.reset({
|
|
259
|
+
productCategoryId: Number(this.categoryId()),
|
|
260
|
+
configurator: this.editor(),
|
|
261
|
+
quantity: this.cartItem()?.quantity ?? null,
|
|
262
|
+
marker: this.cartItem()?.marker ?? '',
|
|
263
|
+
});
|
|
264
|
+
this.items.clear();
|
|
265
|
+
this.product.reset(this.cartItem()?.product ?? null);
|
|
266
|
+
const configuratorParameters = this.cartItem()?.configuratorParams;
|
|
267
|
+
configuratorParameters?.items.forEach((item) => {
|
|
268
|
+
this.items.push(this.createItemGroup({ width: item.width, length: item.length, quantity: item.quantity }));
|
|
269
|
+
});
|
|
270
|
+
if (!configuratorParameters?.items.length) {
|
|
271
|
+
this.addItem();
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSandwichComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
275
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScSandwichComponent, isStandalone: true, selector: "sc-sandwich", inputs: { settings: { classPropertyName: "settings", publicName: "settings", isSignal: true, isRequired: false, transformFunction: null }, categoryId: { classPropertyName: "categoryId", publicName: "categoryId", isSignal: true, isRequired: true, transformFunction: null }, editor: { classPropertyName: "editor", publicName: "editor", isSignal: true, isRequired: true, transformFunction: null }, cartItem: { classPropertyName: "cartItem", publicName: "cartItem", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { toggleShowEvent: "toggleShowEvent" }, providers: [
|
|
276
|
+
tuiNumberFormatProvider({ precision: 0 }),
|
|
277
|
+
tuiInputNumberOptionsProvider({
|
|
278
|
+
min: 0,
|
|
279
|
+
}),
|
|
280
|
+
tuiItemsHandlersProvider({
|
|
281
|
+
stringify: signal((x) => x.name),
|
|
282
|
+
identityMatcher: signal((a, b) => a.id === b.id),
|
|
283
|
+
}),
|
|
284
|
+
], ngImport: i0, template: "@if (!cartItem() && settings()?.allowShowTable) {\n <div class=\"mb-5 flex justify-center\">\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"toggleShowEvent.emit()\"\n type=\"button\"\n iconStart=\"@tui.layout-list\"\n >\n \u041F\u043E\u043A\u0430\u0437\u0430\u0442\u044C \u0432 \u0432\u0438\u0434\u0435 \u0441\u043F\u0438\u0441\u043A\u0430\n </button>\n </div>\n}\n<div class=\"\">\n @let calculateResult = calculateResult$ | async;\n @let products = products$ | async;\n @let productValue = product.value;\n @let validatorWidth = validatorWidth$ | async;\n @let validatorLength = validatorLength$ | async;\n\n <form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit(calculateResult)\"\n ScNextInputFocus\n class=\"flex flex-col gap-3\"\n >\n <label\n tuiLabel\n class=\"grow\"\n >\n \u0422\u043E\u0432\u0430\u0440\n <tui-textfield\n tuiChevron\n [tuiTextfieldCleaner]=\"false\"\n >\n <input\n tuiChevron\n tuiSelect\n [formControl]=\"product\"\n placeholder=\"\u0412\u044B\u0431\u0435\u0440\u0438\u0442\u0435 \u043C\u0430\u0442\u0435\u0440\u0438\u0430\u043B\"\n />\n\n <tui-data-list-wrapper\n *tuiTextfieldDropdown\n new\n [items]=\"products\"\n />\n </tui-textfield>\n @if (productValue) {\n <div class=\"ml-2 text-tui-text-secondary\">\n \u0421\u0442\u043E\u0438\u043C\u043E\u0441\u0442\u044C:\n <span class=\"font-bold\">{{ productValue.costRub?.toLocaleString() }} {{ productValue.currency.symbol }}/\u0448\u0442.</span>\n </div>\n }\n <tui-error\n [formControl]=\"product\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n @if (product.value && validatorWidth && validatorLength) {\n <p class=\"w-full font-bold\">\u0420\u0430\u0437\u043C\u0435\u0440\u044B \u0438\u0437\u0434\u0435\u043B\u0438\u0439</p>\n <p class=\"w-full\">\n \u041C\u0438\u043D\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0448\u0438\u0440\u0438\u043D\u0430: {{ settings()?.minWidth }} \u043C\u043C.\n <br />\n \u041C\u0430\u043A\u0441\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0448\u0438\u0440\u0438\u043D\u0430: {{ product.value.properties?.width }} \u043C\u043C.\n <br />\n \u041C\u0438\u043D\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0434\u043B\u0438\u043D\u0430: {{ settings()?.minLength }} \u043C\u043C.\n <br />\n \u041C\u0430\u043A\u0441\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0434\u043B\u0438\u043D\u0430: {{ product.value.properties?.length }} \u043C\u043C.\n <br />\n </p>\n <div\n formArrayName=\"items\"\n class=\"flex flex-col gap-3\"\n >\n @for (item of items.controls; track item) {\n <div\n class=\"flex grow gap-2\"\n [formGroupName]=\"$index\"\n >\n <div class=\"grid grow grid-cols-3 gap-2\">\n <label tuiLabel>\n \u0428\u0438\u0440\u0438\u043D\u0430, \u043C\u043C\n <tui-textfield>\n <input\n tuiInputSlider\n formControlName=\"width\"\n [min]=\"settings()?.minWidth ?? 0\"\n [max]=\"product.value.properties?.width ?? 0\"\n [tuiNumberFormat]=\"{ precision: 0 }\"\n placeholder=\"\u0428\u0438\u0440\u0438\u043D\u0430\"\n autocomplete=\"off\"\n />\n @if (product.value) {\n <input\n tuiSlider\n type=\"range\"\n />\n }\n </tui-textfield>\n @if (product.value) {\n <div class=\"slider-ticks-labels\">\n <span> 0 </span>\n <span>{{ product.value.properties?.width }}</span>\n </div>\n }\n <tui-error\n formControlName=\"width\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <label tuiLabel>\n \u0414\u043B\u0438\u043D\u0430, \u043C\u043C\n <tui-textfield>\n <input\n tuiInputSlider\n formControlName=\"length\"\n [min]=\"settings()?.minLength ?? 0\"\n [max]=\"product.value.properties?.length ?? 0\"\n [tuiNumberFormat]=\"{ precision: 0 }\"\n placeholder=\"\u0414\u043B\u0438\u043D\u0430\"\n autocomplete=\"off\"\n />\n @if (product.value) {\n <input\n tuiSlider\n type=\"range\"\n />\n }\n </tui-textfield>\n @if (product.value) {\n <div class=\"slider-ticks-labels\">\n <span> 0 </span>\n <span>{{ product.value.properties?.length }}</span>\n </div>\n }\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n <label tuiLabel>\n <span class=\"whitespace-nowrap\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442</span>\n <tui-textfield [tuiTextfieldCleaner]=\"false\">\n <input\n tuiInputNumber\n formControlName=\"quantity\"\n [min]=\"1\"\n [tuiNumberFormat]=\"{ precision: 0 }\"\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n autocomplete=\"off\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n </div>\n <button\n tuiIconButton\n (click)=\"removeItem($index)\"\n [disabled]=\"items.length <= 1\"\n size=\"m\"\n type=\"button\"\n iconStart=\"@tui.trash-2\"\n appearance=\"secondary\"\n class=\"mt-6\"\n [style.flex]=\"'0 0 auto'\"\n ></button>\n </div>\n }\n </div>\n <button\n tuiButton\n appearance=\"secondary\"\n [disabled]=\"items.invalid\"\n (click)=\"addItem()\"\n type=\"button\"\n class=\"self-center\"\n iconStart=\"@tui.plus\"\n >\n \u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0438\u0437\u0434\u0435\u043B\u0438\u0435\n </button>\n } @else if (product.value) {\n <tui-error error=\"\u0420\u0430\u0441\u0447\u0435\u0442 \u043F\u043E \u043F\u0440\u043E\u0434\u0443\u043A\u0442\u0443 \u043D\u0435\u0432\u043E\u0437\u043C\u043E\u0436\u0435\u043D. \u041E\u0431\u0440\u0430\u0442\u0438\u0442\u0435\u0441\u044C \u043A \u043C\u0435\u043D\u0435\u0434\u0436\u0435\u0440\u0443\" />\n }\n\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 \u0440\u0430\u0441\u043F\u0438\u043B\u0430\n <tui-textfield>\n <input\n tuiTextfield\n formControlName=\"marker\"\n placeholder=\"\u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 \u0440\u0430\u0441\u043F\u0438\u043B\u0430\"\n autocomplete=\"marker\"\n />\n </tui-textfield>\n\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n @if (calculateResult && product.value) {\n <tui-loader [showLoader]=\"isCalculateLoading()\" >\n <div class=\"mt-2 flex flex-wrap items-end gap-2\">\n <div>\n \u0418\u0442\u043E\u0433\u043E: <span class=\"text-xl font-bold whitespace-nowrap\">{{ getTotalCost(calculateResult).toLocaleString() }} {{ product.value.currency.symbol }}</span>\n </div>\n @if (calculateResult?.quantity && productValue) {\n <div class=\"text-lg text-sc-dark-grey whitespace-nowrap\">\n <span class=\"font-bold\">({{ calculateResult.quantity }} \u0448\u0442.</span> x\n <span class=\"font-bold\">{{ productValue.costRub?.toLocaleString() }} {{ productValue.currency.symbol }}/\u0448\u0442.)</span>\n </div>\n }\n </div>\n </tui-loader>\n }\n\n <tui-error [error]=\"[] | tuiFieldError | async\" />\n <button\n [disabled]=\"form.invalid || !calculateResult || isCalculateLoading() || isSubmitLoading()\"\n tuiButton\n class=\"self-center\"\n iconStart=\"@tui.check\"\n >\n {{ !cartItem() ? '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' : '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' }}\n </button>\n </form>\n</div>\n", styles: [":host{display:block;width:100%;max-width:30rem}.slider-ticks-labels{--t-offset: calc($thumb / 2);display:flex;font:var(--tui-font-text-s);margin-inline-start:var(--t-offset);margin-inline-end:var(--t-offset);color:var(--tui-text-secondary)}.slider-ticks-labels>*{position:relative;flex:2;text-align:center}.slider-ticks-labels>*:first-child{left:calc(-1 * var(--t-offset));inset-inline-start:calc(-1 * var(--t-offset));flex:1;text-align:start}.slider-ticks-labels>*:last-child{right:calc(-1 * var(--t-offset));flex:1;text-align:end}@supports (inset-inline-end: 0){.slider-ticks-labels>*:last-child{right:unset;inset-inline-end:calc(-1 * var(--t-offset))}}tui-input-slider+.slider-ticks-labels{margin-inline-start:calc(var(--tui-radius-m) / 2 + var(--t-offset))}tui-textfield+.slider-ticks-labels{--t-offset: calc(var(--tui-radius-m) / 2 + $thumb / 2)}tui-textfield[data-size=l]+.slider-ticks-labels{--t-offset: calc(var(--tui-radius-l) / 2 + $thumb / 2)}tui-input-range:not([new])+.slider-ticks-labels,tui-range+.slider-ticks-labels{--t-offset: $thumb}tui-input-range[new]+.slider-ticks-labels{--t-offset: calc(map-get($track-inset, $input-size) + $thumb)}\n"], dependencies: [{ kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.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: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "directive", type: i1.FormArrayName, selector: "[formArrayName]", inputs: ["formArrayName"] }, { kind: "ngmodule", type: ScNextInputFocusModule }, { kind: "directive", type: i2.ScNextInputFocusDirective, selector: "form[ScNextInputFocus]" }, { kind: "directive", type: TuiLabel, selector: "label[tuiLabel]" }, { kind: "component", type: TuiDataListWrapperComponent, selector: "tui-data-list-wrapper:not([labels]), tui-data-list-wrapper:not([labels])[new]", inputs: ["items", "disabledItemHandler", "emptyContent", "size", "itemContent"], outputs: ["itemClick"] }, { kind: "component", type: TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "component", type: i3.TuiTextfieldComponent, selector: "tui-textfield:not([multi])" }, { kind: "directive", type: i3.TuiTextfieldDirective, selector: "input[tuiTextfield]:not([tuiInputCard]):not([tuiInputExpire]):not([tuiInputCVC])" }, { kind: "directive", type: i3.TuiTextfieldOptionsDirective, selector: "[tuiTextfieldAppearance],[tuiTextfieldSize],[tuiTextfieldCleaner]", inputs: ["tuiTextfieldAppearance", "tuiTextfieldSize", "tuiTextfieldCleaner"] }, { kind: "directive", type: i3.TuiTextfieldDropdownDirective, selector: "ng-template[tuiTextfieldDropdown]" }, { kind: "directive", type: TuiChevron, selector: "[tuiChevron]", inputs: ["tuiChevron"] }, { kind: "directive", type: i4.TuiSelectDirective, selector: "input[tuiSelect]" }, { kind: "directive", type: i4.TuiInputNumberDirective, selector: "input[tuiInputNumber]", inputs: ["min", "max", "prefix", "postfix"] }, { kind: "directive", type: TuiNumberFormat, selector: "[tuiNumberFormat]", inputs: ["tuiNumberFormat"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "pipe", type: TuiFieldErrorPipe, name: "tuiFieldError" }, { kind: "component", type: i5.TuiSliderComponent, selector: "input[type=range][tuiSlider]", inputs: ["size", "segments"] }, { kind: "directive", type: i4.TuiInputSliderDirective, selector: "input[tuiInputSlider]" }, { kind: "directive", type: ScSelectOnFocusinDirective, selector: "tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber], input[tuiTextfield], input[tuiInputSlider]" }, { kind: "component", type: TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
285
|
+
}
|
|
286
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSandwichComponent, decorators: [{
|
|
287
|
+
type: Component,
|
|
288
|
+
args: [{ standalone: true, selector: 'sc-sandwich', imports: [
|
|
289
|
+
TuiAppearance,
|
|
290
|
+
TuiWithAppearance,
|
|
291
|
+
TuiIcons,
|
|
292
|
+
TuiWithIcons,
|
|
293
|
+
TuiButton,
|
|
294
|
+
FormsModule,
|
|
295
|
+
ReactiveFormsModule,
|
|
296
|
+
ScNextInputFocusModule,
|
|
297
|
+
TuiLabel,
|
|
298
|
+
TuiDataListWrapperComponent,
|
|
299
|
+
TuiError,
|
|
300
|
+
TuiTextfield,
|
|
301
|
+
TuiChevron,
|
|
302
|
+
TuiSelect,
|
|
303
|
+
TuiInputNumber,
|
|
304
|
+
TuiNumberFormat,
|
|
305
|
+
AsyncPipe,
|
|
306
|
+
TuiFieldErrorPipe,
|
|
307
|
+
TuiInputSlider,
|
|
308
|
+
TuiDataList,
|
|
309
|
+
ScSelectOnFocusinDirective,
|
|
310
|
+
TuiLoader,
|
|
311
|
+
], providers: [
|
|
312
|
+
tuiNumberFormatProvider({ precision: 0 }),
|
|
313
|
+
tuiInputNumberOptionsProvider({
|
|
314
|
+
min: 0,
|
|
315
|
+
}),
|
|
316
|
+
tuiItemsHandlersProvider({
|
|
317
|
+
stringify: signal((x) => x.name),
|
|
318
|
+
identityMatcher: signal((a, b) => a.id === b.id),
|
|
319
|
+
}),
|
|
320
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (!cartItem() && settings()?.allowShowTable) {\n <div class=\"mb-5 flex justify-center\">\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"toggleShowEvent.emit()\"\n type=\"button\"\n iconStart=\"@tui.layout-list\"\n >\n \u041F\u043E\u043A\u0430\u0437\u0430\u0442\u044C \u0432 \u0432\u0438\u0434\u0435 \u0441\u043F\u0438\u0441\u043A\u0430\n </button>\n </div>\n}\n<div class=\"\">\n @let calculateResult = calculateResult$ | async;\n @let products = products$ | async;\n @let productValue = product.value;\n @let validatorWidth = validatorWidth$ | async;\n @let validatorLength = validatorLength$ | async;\n\n <form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit(calculateResult)\"\n ScNextInputFocus\n class=\"flex flex-col gap-3\"\n >\n <label\n tuiLabel\n class=\"grow\"\n >\n \u0422\u043E\u0432\u0430\u0440\n <tui-textfield\n tuiChevron\n [tuiTextfieldCleaner]=\"false\"\n >\n <input\n tuiChevron\n tuiSelect\n [formControl]=\"product\"\n placeholder=\"\u0412\u044B\u0431\u0435\u0440\u0438\u0442\u0435 \u043C\u0430\u0442\u0435\u0440\u0438\u0430\u043B\"\n />\n\n <tui-data-list-wrapper\n *tuiTextfieldDropdown\n new\n [items]=\"products\"\n />\n </tui-textfield>\n @if (productValue) {\n <div class=\"ml-2 text-tui-text-secondary\">\n \u0421\u0442\u043E\u0438\u043C\u043E\u0441\u0442\u044C:\n <span class=\"font-bold\">{{ productValue.costRub?.toLocaleString() }} {{ productValue.currency.symbol }}/\u0448\u0442.</span>\n </div>\n }\n <tui-error\n [formControl]=\"product\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n @if (product.value && validatorWidth && validatorLength) {\n <p class=\"w-full font-bold\">\u0420\u0430\u0437\u043C\u0435\u0440\u044B \u0438\u0437\u0434\u0435\u043B\u0438\u0439</p>\n <p class=\"w-full\">\n \u041C\u0438\u043D\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0448\u0438\u0440\u0438\u043D\u0430: {{ settings()?.minWidth }} \u043C\u043C.\n <br />\n \u041C\u0430\u043A\u0441\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0448\u0438\u0440\u0438\u043D\u0430: {{ product.value.properties?.width }} \u043C\u043C.\n <br />\n \u041C\u0438\u043D\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0434\u043B\u0438\u043D\u0430: {{ settings()?.minLength }} \u043C\u043C.\n <br />\n \u041C\u0430\u043A\u0441\u0438\u043C\u0430\u043B\u044C\u043D\u0430\u044F \u0434\u043B\u0438\u043D\u0430: {{ product.value.properties?.length }} \u043C\u043C.\n <br />\n </p>\n <div\n formArrayName=\"items\"\n class=\"flex flex-col gap-3\"\n >\n @for (item of items.controls; track item) {\n <div\n class=\"flex grow gap-2\"\n [formGroupName]=\"$index\"\n >\n <div class=\"grid grow grid-cols-3 gap-2\">\n <label tuiLabel>\n \u0428\u0438\u0440\u0438\u043D\u0430, \u043C\u043C\n <tui-textfield>\n <input\n tuiInputSlider\n formControlName=\"width\"\n [min]=\"settings()?.minWidth ?? 0\"\n [max]=\"product.value.properties?.width ?? 0\"\n [tuiNumberFormat]=\"{ precision: 0 }\"\n placeholder=\"\u0428\u0438\u0440\u0438\u043D\u0430\"\n autocomplete=\"off\"\n />\n @if (product.value) {\n <input\n tuiSlider\n type=\"range\"\n />\n }\n </tui-textfield>\n @if (product.value) {\n <div class=\"slider-ticks-labels\">\n <span> 0 </span>\n <span>{{ product.value.properties?.width }}</span>\n </div>\n }\n <tui-error\n formControlName=\"width\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <label tuiLabel>\n \u0414\u043B\u0438\u043D\u0430, \u043C\u043C\n <tui-textfield>\n <input\n tuiInputSlider\n formControlName=\"length\"\n [min]=\"settings()?.minLength ?? 0\"\n [max]=\"product.value.properties?.length ?? 0\"\n [tuiNumberFormat]=\"{ precision: 0 }\"\n placeholder=\"\u0414\u043B\u0438\u043D\u0430\"\n autocomplete=\"off\"\n />\n @if (product.value) {\n <input\n tuiSlider\n type=\"range\"\n />\n }\n </tui-textfield>\n @if (product.value) {\n <div class=\"slider-ticks-labels\">\n <span> 0 </span>\n <span>{{ product.value.properties?.length }}</span>\n </div>\n }\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n <label tuiLabel>\n <span class=\"whitespace-nowrap\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442</span>\n <tui-textfield [tuiTextfieldCleaner]=\"false\">\n <input\n tuiInputNumber\n formControlName=\"quantity\"\n [min]=\"1\"\n [tuiNumberFormat]=\"{ precision: 0 }\"\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n autocomplete=\"off\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n </div>\n <button\n tuiIconButton\n (click)=\"removeItem($index)\"\n [disabled]=\"items.length <= 1\"\n size=\"m\"\n type=\"button\"\n iconStart=\"@tui.trash-2\"\n appearance=\"secondary\"\n class=\"mt-6\"\n [style.flex]=\"'0 0 auto'\"\n ></button>\n </div>\n }\n </div>\n <button\n tuiButton\n appearance=\"secondary\"\n [disabled]=\"items.invalid\"\n (click)=\"addItem()\"\n type=\"button\"\n class=\"self-center\"\n iconStart=\"@tui.plus\"\n >\n \u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0438\u0437\u0434\u0435\u043B\u0438\u0435\n </button>\n } @else if (product.value) {\n <tui-error error=\"\u0420\u0430\u0441\u0447\u0435\u0442 \u043F\u043E \u043F\u0440\u043E\u0434\u0443\u043A\u0442\u0443 \u043D\u0435\u0432\u043E\u0437\u043C\u043E\u0436\u0435\u043D. \u041E\u0431\u0440\u0430\u0442\u0438\u0442\u0435\u0441\u044C \u043A \u043C\u0435\u043D\u0435\u0434\u0436\u0435\u0440\u0443\" />\n }\n\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 \u0440\u0430\u0441\u043F\u0438\u043B\u0430\n <tui-textfield>\n <input\n tuiTextfield\n formControlName=\"marker\"\n placeholder=\"\u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 \u0440\u0430\u0441\u043F\u0438\u043B\u0430\"\n autocomplete=\"marker\"\n />\n </tui-textfield>\n\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n @if (calculateResult && product.value) {\n <tui-loader [showLoader]=\"isCalculateLoading()\" >\n <div class=\"mt-2 flex flex-wrap items-end gap-2\">\n <div>\n \u0418\u0442\u043E\u0433\u043E: <span class=\"text-xl font-bold whitespace-nowrap\">{{ getTotalCost(calculateResult).toLocaleString() }} {{ product.value.currency.symbol }}</span>\n </div>\n @if (calculateResult?.quantity && productValue) {\n <div class=\"text-lg text-sc-dark-grey whitespace-nowrap\">\n <span class=\"font-bold\">({{ calculateResult.quantity }} \u0448\u0442.</span> x\n <span class=\"font-bold\">{{ productValue.costRub?.toLocaleString() }} {{ productValue.currency.symbol }}/\u0448\u0442.)</span>\n </div>\n }\n </div>\n </tui-loader>\n }\n\n <tui-error [error]=\"[] | tuiFieldError | async\" />\n <button\n [disabled]=\"form.invalid || !calculateResult || isCalculateLoading() || isSubmitLoading()\"\n tuiButton\n class=\"self-center\"\n iconStart=\"@tui.check\"\n >\n {{ !cartItem() ? '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' : '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' }}\n </button>\n </form>\n</div>\n", styles: [":host{display:block;width:100%;max-width:30rem}.slider-ticks-labels{--t-offset: calc($thumb / 2);display:flex;font:var(--tui-font-text-s);margin-inline-start:var(--t-offset);margin-inline-end:var(--t-offset);color:var(--tui-text-secondary)}.slider-ticks-labels>*{position:relative;flex:2;text-align:center}.slider-ticks-labels>*:first-child{left:calc(-1 * var(--t-offset));inset-inline-start:calc(-1 * var(--t-offset));flex:1;text-align:start}.slider-ticks-labels>*:last-child{right:calc(-1 * var(--t-offset));flex:1;text-align:end}@supports (inset-inline-end: 0){.slider-ticks-labels>*:last-child{right:unset;inset-inline-end:calc(-1 * var(--t-offset))}}tui-input-slider+.slider-ticks-labels{margin-inline-start:calc(var(--tui-radius-m) / 2 + var(--t-offset))}tui-textfield+.slider-ticks-labels{--t-offset: calc(var(--tui-radius-m) / 2 + $thumb / 2)}tui-textfield[data-size=l]+.slider-ticks-labels{--t-offset: calc(var(--tui-radius-l) / 2 + $thumb / 2)}tui-input-range:not([new])+.slider-ticks-labels,tui-range+.slider-ticks-labels{--t-offset: $thumb}tui-input-range[new]+.slider-ticks-labels{--t-offset: calc(map-get($track-inset, $input-size) + $thumb)}\n"] }]
|
|
321
|
+
}] });
|
|
322
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sandwich.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/configurators/sandwich/sc-sandwich.component.ts","../../../../../projects/client-ui/configurators/sandwich/sc-sandwich.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/unbound-method */\n\nimport { AsyncPipe } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, DestroyRef, inject, input, OnInit, output, signal } from '@angular/core';\nimport { takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { FormArray, FormControl, FormGroup, FormsModule, ReactiveFormsModule, ValidatorFn, Validators } from '@angular/forms';\nimport {\n    ISandwichCalculate,\n    ISandwichCalculateItem,\n    ScCart,\n    ScCartItem,\n    ScCartService,\n    ScCatalogService,\n    ScConfiguratorService,\n    ScConvertersService,\n    ScIConfiguratorCalculateResult,\n    ScProduct,\n} from '@snabcentr/client-core';\nimport { tuiControlValue, tuiIsFalsy, tuiIsPresent } from '@taiga-ui/cdk';\nimport {\n    TuiButton,\n    TuiDataList,\n    TuiDialogContext,\n    TuiError,\n    tuiItemsHandlersProvider,\n    TuiLabel,\n    TuiLoader,\n    TuiNumberFormat,\n    tuiNumberFormatProvider,\n    TuiTextfield,\n} from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiChevron, TuiDataListWrapperComponent, TuiFieldErrorPipe, TuiInputNumber, tuiInputNumberOptionsProvider, TuiInputSlider, TuiSelect } from '@taiga-ui/kit';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, combineLatest, debounceTime, filter, finalize, map, Observable, of, share, startWith, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\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 { SC_DEBOUNCE_TIME_DEFAULT } from '../../providers';\nimport { stepValidator } from '../../validators/step-validator';\nimport { ScISandwichSettings } from '../models/sandwich/sc-i-sandwich-settings';\nimport { ScINewCartItemSandwich } from './sc-i-new-cart-item-sandwich';\n\n/**\n * Конфигуратор распила сэндвич-панелей.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-sandwich',\n    templateUrl: './sc-sandwich.component.html',\n    styleUrl: './sc-sandwich.component.scss',\n    imports: [\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        FormsModule,\n        ReactiveFormsModule,\n        ScNextInputFocusModule,\n        TuiLabel,\n        TuiDataListWrapperComponent,\n        TuiError,\n        TuiTextfield,\n        TuiChevron,\n        TuiSelect,\n        TuiInputNumber,\n        TuiNumberFormat,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputSlider,\n        TuiDataList,\n        ScSelectOnFocusinDirective,\n        TuiLoader,\n    ],\n    providers: [\n        tuiNumberFormatProvider({ precision: 0 }),\n        tuiInputNumberOptionsProvider({\n            min: 0,\n        }),\n        tuiItemsHandlersProvider({\n            stringify: signal((x: ScProduct) => x.name),\n            identityMatcher: signal((a: ScProduct, b: ScProduct) => a.id === b.id),\n        }),\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSandwichComponent implements OnInit {\n    /**\n     * Данные настроек окружения.\n     */\n    private readonly debounceTimeDefault: number = inject<number>(SC_DEBOUNCE_TIME_DEFAULT);\n\n    /**\n     * Настройки для конфигуратора сэндвич-панелей.\n     */\n    public readonly settings = input<ScISandwichSettings>();\n\n    /**\n     * Идентификатор категории.\n     */\n    public readonly categoryId = input.required<number | string>();\n\n    /**\n     * Название конфигуратора.\n     */\n    public readonly editor = input.required<string>();\n\n    /**\n     * Позиция товара/услуги в корзине.\n     */\n    public readonly cartItem = input.required<ScCartItem | undefined>();\n\n    /**\n     * Событие переключения отображения товаров.\n     */\n    public readonly toggleShowEvent = output();\n\n    /**\n     * Форма добавления/редактирования продукта в корзине.\n     */\n    protected readonly form = new FormGroup({\n        productCategoryId: new FormControl<number | null>(null, { validators: Validators.required, nonNullable: false }),\n        configurator: new FormControl<string | null>(null, { validators: Validators.required, nonNullable: false }),\n        quantity: new FormControl<number | null>(null, { validators: [Validators.required, stepValidator(1)], nonNullable: false }),\n        marker: new FormControl<string>('', { nonNullable: true }),\n        items: new FormArray<ReturnType<ScSandwichComponent['createItemGroup']>>([]),\n        calculationId: new FormControl<string | null>(null, { nonNullable: false }),\n    });\n\n    /**\n     * Сервис для работы с корзиной.\n     */\n    private readonly cartService: ScCartService = inject(ScCartService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    private readonly context: TuiDialogContext | undefined = inject(POLYMORPHEUS_CONTEXT, { optional: true }) as TuiDialogContext | undefined;\n\n    /**\n     * Сервис для работы с каталогом.\n     */\n    private readonly catalogService: ScCatalogService = inject(ScCatalogService);\n\n    /**\n     * Сервис для работы с конфигураторами.\n     */\n    private readonly configuratorService: ScConfiguratorService = inject(ScConfiguratorService);\n\n    /**\n     * {@link Observable} списка товаров категории.\n     */\n    protected readonly products$: Observable<ScProduct[]> = toObservable(this.categoryId).pipe(\n        switchMap((categoryId) => {\n            if (!categoryId) {\n                return of([]);\n            }\n\n            return this.catalogService.getProductsCategoryCached$({ categoryId: Number(categoryId) }).pipe(map((products) => products ?? []));\n        })\n    );\n\n    /**\n     * Выбранный продукт (материал для раскроя).\n     */\n    protected readonly product = new FormControl<ScProduct | null>(null, { validators: Validators.required, nonNullable: false });\n\n    /**\n     * {@link Observable} расчета оптимизации раскладки.\n     */\n    private readonly calculate$: Observable<ScIConfiguratorCalculateResult | null> = combineLatest({\n        productId: tuiControlValue<ScProduct | null>(this.product).pipe(map((product) => product?.id ?? null)),\n        items: tuiControlValue<ISandwichCalculateItem[]>(this.items),\n    }).pipe(\n        debounceTime(this.debounceTimeDefault),\n        filter(() => this.product.valid && this.form.controls.items.valid),\n        switchMap(({ productId, items }) => {\n            if (productId === null || items.length === 0) {\n                return of(null);\n            }\n\n            return this.configuratorService.calculate$(this.categoryId(), this.editor(), { productId, items }).pipe(\n                tap((calculateResult) => {\n                    this.form.get('quantity')?.setValue(calculateResult.quantity);\n                }),\n                startWith(null),\n                catchError(() => of(null))\n            );\n        }),\n        share(),\n        tap((calculateResult) => {\n            console.log('calculateResult', calculateResult);\n        })\n    );\n\n    /**\n     * Результат расчета оптимизации раскладки.\n     */\n    protected readonly calculateResult$: Observable<ScIConfiguratorCalculateResult> = this.calculate$.pipe(filter(tuiIsPresent));\n\n    /**\n     * Сигнал состояния загрузки расчета.\n     */\n    protected readonly isCalculateLoading = toSignal(this.calculate$.pipe(map(tuiIsFalsy)), { initialValue: false });\n\n    /**\n     * Валидатор для ширины изделия.\n     */\n    protected readonly validatorWidth$: Observable<ValidatorFn | null> = tuiControlValue<ScProduct | null>(this.product).pipe(\n        map((product) => Validators.min(product?.properties?.width ?? 0))\n    );\n\n    /**\n     * Валидатор для длины изделия.\n     */\n    protected readonly validatorLength$: Observable<ValidatorFn | null> = tuiControlValue<ScProduct | null>(this.product).pipe(\n        map((product) => Validators.min(product?.properties?.length ?? 0))\n    );\n\n    /**\n     * Признак того, что выполняется запрос на отправку данных выполняется.\n     */\n    protected readonly isSubmitLoading = signal(false);\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private readonly destroyRef: DestroyRef = inject(DestroyRef);\n\n    /**\n     * Массив изделий.\n     */\n    get items(): FormArray<ReturnType<ScSandwichComponent['createItemGroup']>> {\n        return this.form.controls.items;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.setDefaultFormValue();\n\n        this.product.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((product) => {\n            if (product) {\n                this.items.controls.forEach((item) => {\n                    const value = item.getRawValue();\n\n                    if ((product.properties?.width && value.width < product.properties.width) || (product.properties?.length && value.length < product.properties.length)) {\n                        item.patchValue({\n                            width: Math.min(value.width, product.properties.width ?? value.width),\n                            length: Math.min(value.length, product.properties.length ?? value.length),\n                        });\n                    }\n                });\n            }\n        });\n    }\n\n    /**\n     * Создает группу для изделия.\n     *\n     * @param item Параметры изделия.\n     */\n    private createItemGroup(item?: ISandwichCalculateItem): FormGroup<{\n        width: FormControl<number>;\n        length: FormControl<number>;\n        quantity: FormControl<number>;\n    }> {\n        return new FormGroup({\n            width: new FormControl<number>(item?.width ?? 0, {\n                validators: [Validators.required, Validators.min(this.settings()?.minWidth ?? 0.01)],\n                nonNullable: true,\n            }),\n            length: new FormControl<number>(item?.length ?? 0, {\n                validators: [Validators.required, Validators.min(this.settings()?.minLength ?? 0.01)],\n                nonNullable: true,\n            }),\n            quantity: new FormControl<number>(item?.quantity ?? 1, {\n                validators: [Validators.required, Validators.min(1)],\n                nonNullable: true,\n            }),\n        });\n    }\n\n    /**\n     * Добавляет новое изделие в форму.\n     */\n    protected addItem(): void {\n        this.items.push(this.createItemGroup());\n    }\n\n    /**\n     * Удаляет изделие из формы.\n     *\n     * @param index Индекс изделия.\n     */\n    protected removeItem(index: number): void {\n        this.items.removeAt(index);\n    }\n\n    /**\n     * Возвращает итоговую стоимость.\n     *\n     * @param calculateResult Результат расчета оптимизации раскладки.\n     */\n    protected getTotalCost(calculateResult: ScIConfiguratorCalculateResult): number {\n        if (!this.product.value?.costRub) {\n            return 0;\n        }\n\n        const quantity = this.form.get('quantity')?.value ?? 1;\n        const totalCost = calculateResult.quantity * this.product.value.costRub * quantity;\n\n        return Math.round(totalCost * 100) / 100;\n    }\n\n    /**\n     * Обработчик добавления/редактирования продукта в корзине.\n     *\n     * @param calculateResult Результат расчета оптимизации раскладки.\n     */\n    protected onSubmit(calculateResult: ScIConfiguratorCalculateResult | null): void {\n        if (!calculateResult || this.form.invalid || !this.isCalculateLoading()) {\n            return;\n        }\n\n        const formValue = { ...this.form.value, productId: this.product.value?.id ?? null, calculationId: calculateResult.id };\n        // Удаляем items, так как они не нужны в {@link ScINewCartItemSandwich}.\n        delete formValue.items;\n        const cartItem = this.cartItem();\n\n        const newCartItem$: Observable<ScCart> = cartItem\n            ? this.cartService.patchCartItem$(cartItem.id, this.convertersService.removeNull(formValue) as ScINewCartItemSandwich)\n            : this.cartService.addToCart$(this.convertersService.removeNull(formValue) as ScINewCartItemSandwich);\n\n        this.isSubmitLoading.set(true);\n\n        newCartItem$\n            .pipe(\n                tap(() => {\n                    if (this.context) {\n                        this.context.$implicit.complete();\n                    } else {\n                        this.setDefaultFormValue();\n                    }\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        if (errors) {\n                            Object.keys(errors).forEach((key) => {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            });\n                        } else if (message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n                    return of();\n                }),\n                finalize(() => {\n                    this.isSubmitLoading.set(false);\n                })\n            )\n            .subscribe();\n    }\n\n    /**\n     * Устанавливает начальные значения для полей формы.\n     */\n    private setDefaultFormValue(): void {\n        this.form.reset({\n            productCategoryId: Number(this.categoryId()),\n            configurator: this.editor(),\n            quantity: this.cartItem()?.quantity ?? null,\n            marker: this.cartItem()?.marker ?? '',\n        });\n\n        this.items.clear();\n\n        this.product.reset(this.cartItem()?.product ?? null);\n\n        const configuratorParameters = this.cartItem()?.configuratorParams as ISandwichCalculate | undefined;\n\n        configuratorParameters?.items.forEach((item) => {\n            this.items.push(this.createItemGroup({ width: item.width, length: item.length, quantity: item.quantity }));\n        });\n\n        if (!configuratorParameters?.items.length) {\n            this.addItem();\n        }\n    }\n}\n","@if (!cartItem() && settings()?.allowShowTable) {\n    <div class=\"mb-5 flex justify-center\">\n        <button\n            tuiButton\n            appearance=\"secondary\"\n            (click)=\"toggleShowEvent.emit()\"\n            type=\"button\"\n            iconStart=\"@tui.layout-list\"\n        >\n            Показать в виде списка\n        </button>\n    </div>\n}\n<div class=\"\">\n    @let calculateResult = calculateResult$ | async;\n    @let products = products$ | async;\n    @let productValue = product.value;\n    @let validatorWidth = validatorWidth$ | async;\n    @let validatorLength = validatorLength$ | async;\n\n    <form\n        [formGroup]=\"form\"\n        (ngSubmit)=\"onSubmit(calculateResult)\"\n        ScNextInputFocus\n        class=\"flex flex-col gap-3\"\n    >\n        <label\n            tuiLabel\n            class=\"grow\"\n        >\n            Товар\n            <tui-textfield\n                tuiChevron\n                [tuiTextfieldCleaner]=\"false\"\n            >\n                <input\n                    tuiChevron\n                    tuiSelect\n                    [formControl]=\"product\"\n                    placeholder=\"Выберите материал\"\n                />\n\n                <tui-data-list-wrapper\n                    *tuiTextfieldDropdown\n                    new\n                    [items]=\"products\"\n                />\n            </tui-textfield>\n            @if (productValue) {\n                <div class=\"ml-2 text-tui-text-secondary\">\n                    Стоимость:\n                    <span class=\"font-bold\">{{ productValue.costRub?.toLocaleString() }} {{ productValue.currency.symbol }}/шт.</span>\n                </div>\n            }\n            <tui-error\n                [formControl]=\"product\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        @if (product.value && validatorWidth && validatorLength) {\n            <p class=\"w-full font-bold\">Размеры изделий</p>\n            <p class=\"w-full\">\n                Минимальная ширина: {{ settings()?.minWidth }} мм.\n                <br />\n                Максимальная ширина: {{ product.value.properties?.width }} мм.\n                <br />\n                Минимальная длина: {{ settings()?.minLength }} мм.\n                <br />\n                Максимальная длина: {{ product.value.properties?.length }} мм.\n                <br />\n            </p>\n            <div\n                formArrayName=\"items\"\n                class=\"flex flex-col gap-3\"\n            >\n                @for (item of items.controls; track item) {\n                    <div\n                        class=\"flex grow gap-2\"\n                        [formGroupName]=\"$index\"\n                    >\n                        <div class=\"grid grow grid-cols-3 gap-2\">\n                            <label tuiLabel>\n                                Ширина, мм\n                                <tui-textfield>\n                                    <input\n                                        tuiInputSlider\n                                        formControlName=\"width\"\n                                        [min]=\"settings()?.minWidth ?? 0\"\n                                        [max]=\"product.value.properties?.width ?? 0\"\n                                        [tuiNumberFormat]=\"{ precision: 0 }\"\n                                        placeholder=\"Ширина\"\n                                        autocomplete=\"off\"\n                                    />\n                                    @if (product.value) {\n                                        <input\n                                            tuiSlider\n                                            type=\"range\"\n                                        />\n                                    }\n                                </tui-textfield>\n                                @if (product.value) {\n                                    <div class=\"slider-ticks-labels\">\n                                        <span> 0 </span>\n                                        <span>{{ product.value.properties?.width }}</span>\n                                    </div>\n                                }\n                                <tui-error\n                                    formControlName=\"width\"\n                                    [error]=\"[] | tuiFieldError | async\"\n                                />\n                            </label>\n\n                            <label tuiLabel>\n                                Длина, мм\n                                <tui-textfield>\n                                    <input\n                                        tuiInputSlider\n                                        formControlName=\"length\"\n                                        [min]=\"settings()?.minLength ?? 0\"\n                                        [max]=\"product.value.properties?.length ?? 0\"\n                                        [tuiNumberFormat]=\"{ precision: 0 }\"\n                                        placeholder=\"Длина\"\n                                        autocomplete=\"off\"\n                                    />\n                                    @if (product.value) {\n                                        <input\n                                            tuiSlider\n                                            type=\"range\"\n                                        />\n                                    }\n                                </tui-textfield>\n                                @if (product.value) {\n                                    <div class=\"slider-ticks-labels\">\n                                        <span> 0 </span>\n                                        <span>{{ product.value.properties?.length }}</span>\n                                    </div>\n                                }\n                                <tui-error\n                                    formControlName=\"length\"\n                                    [error]=\"[] | tuiFieldError | async\"\n                                />\n                            </label>\n                            <label tuiLabel>\n                                <span class=\"whitespace-nowrap\">Количество, шт</span>\n                                <tui-textfield [tuiTextfieldCleaner]=\"false\">\n                                    <input\n                                        tuiInputNumber\n                                        formControlName=\"quantity\"\n                                        [min]=\"1\"\n                                        [tuiNumberFormat]=\"{ precision: 0 }\"\n                                        placeholder=\"Количество\"\n                                        autocomplete=\"off\"\n                                    />\n                                </tui-textfield>\n                                <tui-error\n                                    formControlName=\"quantity\"\n                                    [error]=\"[] | tuiFieldError | async\"\n                                />\n                            </label>\n                        </div>\n                        <button\n                            tuiIconButton\n                            (click)=\"removeItem($index)\"\n                            [disabled]=\"items.length <= 1\"\n                            size=\"m\"\n                            type=\"button\"\n                            iconStart=\"@tui.trash-2\"\n                            appearance=\"secondary\"\n                            class=\"mt-6\"\n                            [style.flex]=\"'0 0 auto'\"\n                        ></button>\n                    </div>\n                }\n            </div>\n            <button\n                tuiButton\n                appearance=\"secondary\"\n                [disabled]=\"items.invalid\"\n                (click)=\"addItem()\"\n                type=\"button\"\n                class=\"self-center\"\n                iconStart=\"@tui.plus\"\n            >\n                Добавить изделие\n            </button>\n        } @else if (product.value) {\n            <tui-error error=\"Расчет по продукту невозможен. Обратитесь к менеджеру\" />\n        }\n\n        <label tuiLabel>\n            Маркировка распила\n            <tui-textfield>\n                <input\n                    tuiTextfield\n                    formControlName=\"marker\"\n                    placeholder=\"Маркировка распила\"\n                    autocomplete=\"marker\"\n                />\n            </tui-textfield>\n\n            <tui-error\n                formControlName=\"marker\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        @if (calculateResult && product.value) {\n            <tui-loader [showLoader]=\"isCalculateLoading()\" >\n                <div class=\"mt-2 flex flex-wrap items-end gap-2\">\n                    <div>\n                        Итого: <span class=\"text-xl font-bold whitespace-nowrap\">{{ getTotalCost(calculateResult).toLocaleString() }} {{ product.value.currency.symbol }}</span>\n                    </div>\n                    @if (calculateResult?.quantity && productValue) {\n                        <div class=\"text-lg text-sc-dark-grey whitespace-nowrap\">\n                            <span class=\"font-bold\">({{ calculateResult.quantity }} шт.</span> x\n                            <span class=\"font-bold\">{{ productValue.costRub?.toLocaleString() }} {{ productValue.currency.symbol }}/шт.)</span>\n                        </div>\n                    }\n                </div>\n            </tui-loader>\n        }\n\n        <tui-error [error]=\"[] | tuiFieldError | async\" />\n        <button\n            [disabled]=\"form.invalid || !calculateResult || isCalculateLoading() || isSubmitLoading()\"\n            tuiButton\n            class=\"self-center\"\n            iconStart=\"@tui.check\"\n        >\n            {{ !cartItem() ? 'Добавить в корзину' : 'Изменить' }}\n        </button>\n    </form>\n</div>\n"]}
|