@snabcentr/client-ui 3.46.0 → 3.47.6

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.
Files changed (30) hide show
  1. package/configurators/index.d.ts +2 -0
  2. package/configurators/models/index.d.ts +2 -0
  3. package/configurators/models/sandwich/sc-i-configurator-search-product-sandwich.d.ts +22 -0
  4. package/configurators/models/sandwich/sc-i-sandwich-settings.d.ts +18 -0
  5. package/configurators/sandwich/index.d.ts +3 -0
  6. package/configurators/sandwich/sandwich-skeleton/sc-sandwich-skeleton.component.d.ts +8 -0
  7. package/configurators/sandwich/sc-i-new-cart-item-sandwich.d.ts +14 -0
  8. package/configurators/sandwich/sc-sandwich.component.d.ts +138 -0
  9. package/directives/select-on-focusin/sc-select-on-focusin.directive.d.ts +1 -1
  10. package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +2 -2
  11. package/esm2022/cart/cart-item/sc-cart-item.component.mjs +3 -3
  12. package/esm2022/configurators/index.mjs +3 -0
  13. package/esm2022/configurators/models/index.mjs +3 -0
  14. package/esm2022/configurators/models/sandwich/sc-i-configurator-search-product-sandwich.mjs +2 -0
  15. package/esm2022/configurators/models/sandwich/sc-i-sandwich-settings.mjs +2 -0
  16. package/esm2022/configurators/sandwich/index.mjs +4 -0
  17. package/esm2022/configurators/sandwich/sandwich-skeleton/sc-sandwich-skeleton.component.mjs +14 -0
  18. package/esm2022/configurators/sandwich/sc-i-new-cart-item-sandwich.mjs +2 -0
  19. package/esm2022/configurators/sandwich/sc-sandwich.component.mjs +311 -0
  20. package/esm2022/directives/select-on-focusin/sc-select-on-focusin.directive.mjs +3 -3
  21. package/esm2022/providers/index.mjs +2 -1
  22. package/esm2022/providers/sc-debounce-time-default.mjs +9 -0
  23. package/esm2022/public-api.mjs +2 -1
  24. package/fesm2022/snabcentr-client-ui.mjs +322 -13
  25. package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
  26. package/package.json +1 -1
  27. package/providers/index.d.ts +1 -0
  28. package/providers/sc-debounce-time-default.d.ts +5 -0
  29. package/public-api.d.ts +1 -0
  30. package/styles/tailwind/tailwind.scss +61 -1
@@ -0,0 +1,311 @@
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 } 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 } from '@taiga-ui/cdk';
9
+ import { TuiButton, TuiDataList, TuiError, tuiItemsHandlersProvider, TuiLabel, 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, 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
+ * Результат расчета оптимизации раскладки.
100
+ */
101
+ this.calculateResult$ = 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
+ }), catchError(() => of(null)));
111
+ }));
112
+ /**
113
+ * Валидатор для ширины изделия.
114
+ */
115
+ this.validatorWidth$ = tuiControlValue(this.product).pipe(map((product) => Validators.min(product?.properties?.width ?? 0)));
116
+ /**
117
+ * Валидатор для длины изделия.
118
+ */
119
+ this.validatorLength$ = tuiControlValue(this.product).pipe(map((product) => Validators.min(product?.properties?.length ?? 0)));
120
+ /**
121
+ * Признак того, что выполняется запрос на отправку данных выполняется.
122
+ */
123
+ this.isSubmitLoading = signal(false);
124
+ /**
125
+ * Ссылка для автоматического управления уничтожением зависимостей.
126
+ */
127
+ this.destroyRef = inject(DestroyRef);
128
+ }
129
+ /**
130
+ * Массив изделий.
131
+ */
132
+ get items() {
133
+ return this.form.controls.items;
134
+ }
135
+ /** @inheritDoc */
136
+ ngOnInit() {
137
+ this.setDefaultFormValue();
138
+ this.product.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((product) => {
139
+ if (product) {
140
+ this.items.controls.forEach((item) => {
141
+ const value = item.getRawValue();
142
+ if ((product.properties?.width && value.width < product.properties.width) || (product.properties?.length && value.length < product.properties.length)) {
143
+ item.patchValue({
144
+ width: Math.min(value.width, product.properties.width ?? value.width),
145
+ length: Math.min(value.length, product.properties.length ?? value.length),
146
+ });
147
+ }
148
+ });
149
+ }
150
+ });
151
+ }
152
+ /**
153
+ * Создает группу для изделия.
154
+ *
155
+ * @param item Параметры изделия.
156
+ */
157
+ createItemGroup(item) {
158
+ return new FormGroup({
159
+ width: new FormControl(item?.width ?? 0, {
160
+ validators: [Validators.required, Validators.min(this.settings()?.minWidth ?? 0.01)],
161
+ nonNullable: true,
162
+ }),
163
+ length: new FormControl(item?.length ?? 0, {
164
+ validators: [Validators.required, Validators.min(this.settings()?.minLength ?? 0.01)],
165
+ nonNullable: true,
166
+ }),
167
+ quantity: new FormControl(item?.quantity ?? 1, {
168
+ validators: [Validators.required, Validators.min(1)],
169
+ nonNullable: true,
170
+ }),
171
+ });
172
+ }
173
+ /**
174
+ * Добавляет новое изделие в форму.
175
+ */
176
+ addItem() {
177
+ this.items.push(this.createItemGroup());
178
+ }
179
+ /**
180
+ * Удаляет изделие из формы.
181
+ *
182
+ * @param index Индекс изделия.
183
+ */
184
+ removeItem(index) {
185
+ this.items.removeAt(index);
186
+ }
187
+ /**
188
+ * Возвращает итоговую стоимость.
189
+ *
190
+ * @param calculateResult Результат расчета оптимизации раскладки.
191
+ */
192
+ getTotalCost(calculateResult) {
193
+ if (!this.product.value?.costRub) {
194
+ return 0;
195
+ }
196
+ const quantity = this.form.get('quantity')?.value ?? 1;
197
+ const totalCost = calculateResult.quantity * this.product.value.costRub * quantity;
198
+ return Math.round(totalCost * 100) / 100;
199
+ }
200
+ /**
201
+ * Обработчик добавления/редактирования продукта в корзине.
202
+ *
203
+ * @param calculateResult Результат расчета оптимизации раскладки.
204
+ */
205
+ onSubmit(calculateResult) {
206
+ if (!calculateResult || this.form.invalid) {
207
+ return;
208
+ }
209
+ const formValue = { ...this.form.value, productId: this.product.value?.id ?? null, calculationId: calculateResult.id };
210
+ // Удаляем items, так как они не нужны в {@link ScINewCartItemSandwich}.
211
+ delete formValue.items;
212
+ const cartItem = this.cartItem();
213
+ const newCartItem$ = cartItem
214
+ ? this.cartService.patchCartItem$(cartItem.id, this.convertersService.removeNull(formValue))
215
+ : this.cartService.addToCart$(this.convertersService.removeNull(formValue));
216
+ this.isSubmitLoading.set(true);
217
+ newCartItem$
218
+ .pipe(tap(() => {
219
+ if (this.context) {
220
+ this.context.$implicit.complete();
221
+ }
222
+ else {
223
+ this.setDefaultFormValue();
224
+ }
225
+ }), catchError((error) => {
226
+ if (error instanceof HttpErrorResponse) {
227
+ const { errors, message } = error.error;
228
+ if (errors) {
229
+ Object.keys(errors).forEach((key) => {
230
+ // eslint-disable-next-line security/detect-object-injection
231
+ this.form.get(key)?.setErrors({ serverResponse: errors[key] });
232
+ });
233
+ }
234
+ else if (message) {
235
+ this.form.setErrors({ serverResponse: [message] });
236
+ }
237
+ }
238
+ return of();
239
+ }), finalize(() => {
240
+ this.isSubmitLoading.set(false);
241
+ }))
242
+ .subscribe();
243
+ }
244
+ /**
245
+ * Устанавливает начальные значения для полей формы.
246
+ */
247
+ setDefaultFormValue() {
248
+ this.form.reset({
249
+ productCategoryId: Number(this.categoryId()),
250
+ configurator: this.editor(),
251
+ quantity: this.cartItem()?.quantity ?? null,
252
+ marker: this.cartItem()?.marker ?? '',
253
+ });
254
+ this.items.clear();
255
+ this.product.reset(this.cartItem()?.product ?? null);
256
+ const configuratorParameters = this.cartItem()?.configuratorParams;
257
+ configuratorParameters?.items.forEach((item) => {
258
+ this.items.push(this.createItemGroup({ width: item.width, length: item.length, quantity: item.quantity }));
259
+ });
260
+ if (!configuratorParameters?.items.length) {
261
+ this.addItem();
262
+ }
263
+ }
264
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSandwichComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
265
+ 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: [
266
+ tuiNumberFormatProvider({ precision: 0 }),
267
+ tuiInputNumberOptionsProvider({
268
+ min: 0,
269
+ }),
270
+ tuiItemsHandlersProvider({
271
+ stringify: signal((x) => x.name),
272
+ identityMatcher: signal((a, b) => a.id === b.id),
273
+ }),
274
+ ], 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 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 tuiChevron [tuiTextfieldCleaner]=\"false\">\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 <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 index; let index = $index) {\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 @if (calculateResult && product.value) {\n <label tuiLabel>\n \u0418\u0442\u043E\u0433\u043E\u0432\u043E\u0435 \u043A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u043B\u0438\u0441\u0442\u043E\u0432 \u0434\u043B\u044F \u043F\u0440\u043E\u0438\u0437\u0432\u043E\u0434\u0441\u0442\u0432\u0430, \u0448\u0442\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"quantity\"\n [readOnly]=\"true\"\n [focused]=\"false\"\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n </label>\n }\n\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-textfield>\n <input\n tuiTextfield\n formControlName=\"marker\"\n placeholder=\"\u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\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 <p class=\"mt-2\">\n \u0418\u0442\u043E\u0433\u043E: <span class=\"text-xl font-bold\">{{ getTotalCost(calculateResult).toLocaleString() }} {{ product.value.currency.symbol }}</span>\n </p>\n }\n\n <tui-error [error]=\"[] | tuiFieldError | async\" />\n <button\n [disabled]=\"form.invalid || !calculateResult || 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]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
275
+ }
276
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSandwichComponent, decorators: [{
277
+ type: Component,
278
+ args: [{ standalone: true, selector: 'sc-sandwich', imports: [
279
+ TuiAppearance,
280
+ TuiWithAppearance,
281
+ TuiIcons,
282
+ TuiWithIcons,
283
+ TuiButton,
284
+ FormsModule,
285
+ ReactiveFormsModule,
286
+ ScNextInputFocusModule,
287
+ TuiLabel,
288
+ TuiDataListWrapperComponent,
289
+ TuiError,
290
+ TuiTextfield,
291
+ TuiChevron,
292
+ TuiSelect,
293
+ TuiInputNumber,
294
+ TuiNumberFormat,
295
+ AsyncPipe,
296
+ TuiFieldErrorPipe,
297
+ TuiInputSlider,
298
+ TuiDataList,
299
+ ScSelectOnFocusinDirective,
300
+ ], providers: [
301
+ tuiNumberFormatProvider({ precision: 0 }),
302
+ tuiInputNumberOptionsProvider({
303
+ min: 0,
304
+ }),
305
+ tuiItemsHandlersProvider({
306
+ stringify: signal((x) => x.name),
307
+ identityMatcher: signal((a, b) => a.id === b.id),
308
+ }),
309
+ ], 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 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 tuiChevron [tuiTextfieldCleaner]=\"false\">\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 <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 index; let index = $index) {\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 @if (calculateResult && product.value) {\n <label tuiLabel>\n \u0418\u0442\u043E\u0433\u043E\u0432\u043E\u0435 \u043A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u043B\u0438\u0441\u0442\u043E\u0432 \u0434\u043B\u044F \u043F\u0440\u043E\u0438\u0437\u0432\u043E\u0434\u0441\u0442\u0432\u0430, \u0448\u0442\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"quantity\"\n [readOnly]=\"true\"\n [focused]=\"false\"\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n </label>\n }\n\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-textfield>\n <input\n tuiTextfield\n formControlName=\"marker\"\n placeholder=\"\u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\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 <p class=\"mt-2\">\n \u0418\u0442\u043E\u0433\u043E: <span class=\"text-xl font-bold\">{{ getTotalCost(calculateResult).toLocaleString() }} {{ product.value.currency.symbol }}</span>\n </p>\n }\n\n <tui-error [error]=\"[] | tuiFieldError | async\" />\n <button\n [disabled]=\"form.invalid || !calculateResult || 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"] }]
310
+ }] });
311
+ //# 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 } 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 } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDataList, TuiDialogContext, TuiError, tuiItemsHandlersProvider, TuiLabel, TuiNumberFormat, tuiNumberFormatProvider, TuiTextfield } 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, 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    ],\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     * Результат расчета оптимизации раскладки.\n     */\n    protected readonly calculateResult$: 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                catchError(() => of(null))\n            );\n        })\n    );\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) {\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 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 tuiChevron [tuiTextfieldCleaner]=\"false\">\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            <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 index; let index = $index) {\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        @if (calculateResult && product.value) {\n            <label tuiLabel>\n                Итоговое количество листов для производства, шт\n                <tui-textfield>\n                    <input\n                        tuiInputNumber\n                        formControlName=\"quantity\"\n                        [readOnly]=\"true\"\n                        [focused]=\"false\"\n                        placeholder=\"Количество\"\n                        autocomplete=\"quantity\"\n                    />\n                </tui-textfield>\n            </label>\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            <p class=\"mt-2\">\n                Итого: <span class=\"text-xl font-bold\">{{ getTotalCost(calculateResult).toLocaleString() }} {{ product.value.currency.symbol }}</span>\n            </p>\n        }\n\n        <tui-error [error]=\"[] | tuiFieldError | async\" />\n        <button\n            [disabled]=\"form.invalid || !calculateResult || isSubmitLoading()\"\n            tuiButton\n            class=\"self-center\"\n            iconStart=\"@tui.check\"\n        >\n            {{ !cartItem() ? 'Добавить в корзину' : 'Изменить' }}\n        </button>\n    </form>\n</div>\n"]}
@@ -16,16 +16,16 @@ export class ScSelectOnFocusinDirective {
16
16
  }
17
17
  }
18
18
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSelectOnFocusinDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
19
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.14", type: ScSelectOnFocusinDirective, isStandalone: true, selector: "tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber]", host: { listeners: { "focusin": "onFocusIn($event.target)" } }, ngImport: i0 }); }
19
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.14", type: ScSelectOnFocusinDirective, isStandalone: true, selector: "tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber], input[tuiTextfield], input[tuiInputSlider]", host: { listeners: { "focusin": "onFocusIn($event.target)" } }, ngImport: i0 }); }
20
20
  }
21
21
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSelectOnFocusinDirective, decorators: [{
22
22
  type: Directive,
23
23
  args: [{
24
24
  standalone: true,
25
- selector: 'tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber]',
25
+ selector: 'tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber], input[tuiTextfield], input[tuiInputSlider]',
26
26
  host: {
27
27
  '(focusin)': 'onFocusIn($event.target)',
28
28
  },
29
29
  }]
30
30
  }] });
31
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Mtc2VsZWN0LW9uLWZvY3VzaW4uZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2RpcmVjdGl2ZXMvc2VsZWN0LW9uLWZvY3VzaW4vc2Mtc2VsZWN0LW9uLWZvY3VzaW4uZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRTFDOztHQUVHO0FBUUgsTUFBTSxPQUFPLDBCQUEwQjtJQUNuQzs7OztPQUlHO0lBQ0gsa0RBQWtEO0lBQ3hDLFNBQVMsQ0FBQyxNQUFtQjtRQUNuQyxJQUFJLE1BQU0sWUFBWSxnQkFBZ0IsRUFBRSxDQUFDO1lBQ3JDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUNwQixDQUFDO0lBQ0wsQ0FBQzsrR0FYUSwwQkFBMEI7bUdBQTFCLDBCQUEwQjs7NEZBQTFCLDBCQUEwQjtrQkFQdEMsU0FBUzttQkFBQztvQkFDUCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFLHlHQUF5RztvQkFDbkgsSUFBSSxFQUFFO3dCQUNGLFdBQVcsRUFBRSwwQkFBMEI7cUJBQzFDO2lCQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICog0JTQuNGA0LXQutGC0LjQstCwINC00LvRjyDQvtCx0YDQsNCx0L7RgtC60Lgg0YHQvtCx0YvRgtC40Y8g0YTQvtC60YPRgdCwINC/0L7Qu9GPINCy0LLQvtC00LAsINC00LvRjyDQv9C+0YHQu9C10LTRg9GO0YnQtdCz0L4g0LLRi9C00LXQu9C10L3QuNGPINGB0L7QtNC10YDQttC40LzQvtCz0L4g0L/QvtC70Y8g0LLQstC+0LTQsC5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBzZWxlY3RvcjogJ3R1aS1pbnB1dC1udW1iZXIsIHR1aS1pbnB1dCwgdHVpLWlucHV0LXBob25lLCB0dWktaW5wdXQtZGF0ZSwgdHVpLWlucHV0LXBhc3N3b3JkLCBpbnB1dFt0dWlJbnB1dE51bWJlcl0nLFxuICAgIGhvc3Q6IHtcbiAgICAgICAgJyhmb2N1c2luKSc6ICdvbkZvY3VzSW4oJGV2ZW50LnRhcmdldCknLFxuICAgIH0sXG59KVxuZXhwb3J0IGNsYXNzIFNjU2VsZWN0T25Gb2N1c2luRGlyZWN0aXZlIHtcbiAgICAvKipcbiAgICAgKiDQodC70YPRiNCw0YLQtdC70Ywg0Lgg0L7QsdGA0LDQsdC+0YLRh9C40Log0YHQvtCx0YvRgtC40Y8gYGZvY3VzaW5gLlxuICAgICAqXG4gICAgICogQHBhcmFtIHRhcmdldCDQptC10LvQtdCy0L7QuSDQvtCx0YrQtdC60YIg0YHQvtCx0YvRgtC40Y8gYGZvY3VzaW5gLlxuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBjbGFzcy1tZXRob2RzLXVzZS10aGlzXG4gICAgcHJvdGVjdGVkIG9uRm9jdXNJbih0YXJnZXQ6IEhUTUxFbGVtZW50KTogdm9pZCB7XG4gICAgICAgIGlmICh0YXJnZXQgaW5zdGFuY2VvZiBIVE1MSW5wdXRFbGVtZW50KSB7XG4gICAgICAgICAgICB0YXJnZXQuc2VsZWN0KCk7XG4gICAgICAgIH1cbiAgICB9XG59XG4iXX0=
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Mtc2VsZWN0LW9uLWZvY3VzaW4uZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2RpcmVjdGl2ZXMvc2VsZWN0LW9uLWZvY3VzaW4vc2Mtc2VsZWN0LW9uLWZvY3VzaW4uZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRTFDOztHQUVHO0FBUUgsTUFBTSxPQUFPLDBCQUEwQjtJQUNuQzs7OztPQUlHO0lBQ0gsa0RBQWtEO0lBQ3hDLFNBQVMsQ0FBQyxNQUFtQjtRQUNuQyxJQUFJLE1BQU0sWUFBWSxnQkFBZ0IsRUFBRSxDQUFDO1lBQ3JDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUNwQixDQUFDO0lBQ0wsQ0FBQzsrR0FYUSwwQkFBMEI7bUdBQTFCLDBCQUEwQjs7NEZBQTFCLDBCQUEwQjtrQkFQdEMsU0FBUzttQkFBQztvQkFDUCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFLHFKQUFxSjtvQkFDL0osSUFBSSxFQUFFO3dCQUNGLFdBQVcsRUFBRSwwQkFBMEI7cUJBQzFDO2lCQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICog0JTQuNGA0LXQutGC0LjQstCwINC00LvRjyDQvtCx0YDQsNCx0L7RgtC60Lgg0YHQvtCx0YvRgtC40Y8g0YTQvtC60YPRgdCwINC/0L7Qu9GPINCy0LLQvtC00LAsINC00LvRjyDQv9C+0YHQu9C10LTRg9GO0YnQtdCz0L4g0LLRi9C00LXQu9C10L3QuNGPINGB0L7QtNC10YDQttC40LzQvtCz0L4g0L/QvtC70Y8g0LLQstC+0LTQsC5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBzZWxlY3RvcjogJ3R1aS1pbnB1dC1udW1iZXIsIHR1aS1pbnB1dCwgdHVpLWlucHV0LXBob25lLCB0dWktaW5wdXQtZGF0ZSwgdHVpLWlucHV0LXBhc3N3b3JkLCBpbnB1dFt0dWlJbnB1dE51bWJlcl0sIGlucHV0W3R1aVRleHRmaWVsZF0sIGlucHV0W3R1aUlucHV0U2xpZGVyXScsXG4gICAgaG9zdDoge1xuICAgICAgICAnKGZvY3VzaW4pJzogJ29uRm9jdXNJbigkZXZlbnQudGFyZ2V0KScsXG4gICAgfSxcbn0pXG5leHBvcnQgY2xhc3MgU2NTZWxlY3RPbkZvY3VzaW5EaXJlY3RpdmUge1xuICAgIC8qKlxuICAgICAqINCh0LvRg9GI0LDRgtC10LvRjCDQuCDQvtCx0YDQsNCx0L7RgtGH0LjQuiDRgdC+0LHRi9GC0LjRjyBgZm9jdXNpbmAuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gdGFyZ2V0INCm0LXQu9C10LLQvtC5INC+0LHRitC10LrRgiDRgdC+0LHRi9GC0LjRjyBgZm9jdXNpbmAuXG4gICAgICovXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGNsYXNzLW1ldGhvZHMtdXNlLXRoaXNcbiAgICBwcm90ZWN0ZWQgb25Gb2N1c0luKHRhcmdldDogSFRNTEVsZW1lbnQpOiB2b2lkIHtcbiAgICAgICAgaWYgKHRhcmdldCBpbnN0YW5jZW9mIEhUTUxJbnB1dEVsZW1lbnQpIHtcbiAgICAgICAgICAgIHRhcmdldC5zZWxlY3QoKTtcbiAgICAgICAgfVxuICAgIH1cbn1cbiJdfQ==
@@ -10,4 +10,5 @@ export * from './sc-date-formatter';
10
10
  export * from './sc-date-value-transformer';
11
11
  export * from './sc-notify-when-in-stock-required-fields';
12
12
  export * from './sc-banner-duration';
13
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvcHJvdmlkZXJzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsZ0NBQWdDLENBQUM7QUFDL0MsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsOEJBQThCLENBQUM7QUFDN0MsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLHdCQUF3QixDQUFDO0FBQ3ZDLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxxQkFBcUIsQ0FBQztBQUNwQyxjQUFjLHFCQUFxQixDQUFDO0FBQ3BDLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLHNCQUFzQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9zYy12ZXJpZmljYXRpb24tY29kZS10aW1lb3V0JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtYWxsb3ctc2VsZWN0LXRlcm1pbmF0ZWQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1oZWxwLW5vdGlmaWNhdGlvbi1saW1pdCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWhlbHAtbm90aWZpY2F0aW9uLWNsb3NlJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtbWF4LWZpbGVzLWluLWZvcm0taW5wdXQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1wYWdlLXNpemUtb3B0aW9ucyc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWNvdW50cnktaWRzJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtdXNlci1wcm92aWRlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1kYXRlLWZvcm1hdHRlcic7XG5leHBvcnQgKiBmcm9tICcuL3NjLWRhdGUtdmFsdWUtdHJhbnNmb3JtZXInO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1ub3RpZnktd2hlbi1pbi1zdG9jay1yZXF1aXJlZC1maWVsZHMnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1iYW5uZXItZHVyYXRpb24nO1xuIl19
13
+ export * from './sc-debounce-time-default';
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvcHJvdmlkZXJzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsZ0NBQWdDLENBQUM7QUFDL0MsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsOEJBQThCLENBQUM7QUFDN0MsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLHdCQUF3QixDQUFDO0FBQ3ZDLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxxQkFBcUIsQ0FBQztBQUNwQyxjQUFjLHFCQUFxQixDQUFDO0FBQ3BDLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLHNCQUFzQixDQUFDO0FBQ3JDLGNBQWMsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3NjLXZlcmlmaWNhdGlvbi1jb2RlLXRpbWVvdXQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1hbGxvdy1zZWxlY3QtdGVybWluYXRlZCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWhlbHAtbm90aWZpY2F0aW9uLWxpbWl0JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtaGVscC1ub3RpZmljYXRpb24tY2xvc2UnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1tYXgtZmlsZXMtaW4tZm9ybS1pbnB1dCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLXBhZ2Utc2l6ZS1vcHRpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtY291bnRyeS1pZHMnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy11c2VyLXByb3ZpZGVycyc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWRhdGUtZm9ybWF0dGVyJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtZGF0ZS12YWx1ZS10cmFuc2Zvcm1lcic7XG5leHBvcnQgKiBmcm9tICcuL3NjLW5vdGlmeS13aGVuLWluLXN0b2NrLXJlcXVpcmVkLWZpZWxkcyc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWJhbm5lci1kdXJhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL3NjLWRlYm91bmNlLXRpbWUtZGVmYXVsdCc7XG4iXX0=
@@ -0,0 +1,9 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ /**
3
+ * Длительность {@link debounceTime} в миллисекундах по умолчанию.
4
+ */
5
+ export const SC_DEBOUNCE_TIME_DEFAULT = new InjectionToken('scDebounceTimeDefault', {
6
+ providedIn: 'root',
7
+ factory: () => 300,
8
+ });
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtZGVib3VuY2UtdGltZS1kZWZhdWx0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL3Byb3ZpZGVycy9zYy1kZWJvdW5jZS10aW1lLWRlZmF1bHQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUUvQzs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLHdCQUF3QixHQUEyQixJQUFJLGNBQWMsQ0FBUyx1QkFBdUIsRUFBRTtJQUNoSCxVQUFVLEVBQUUsTUFBTTtJQUNsQixPQUFPLEVBQUUsR0FBRyxFQUFFLENBQUMsR0FBRztDQUNyQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG4vKipcbiAqINCU0LvQuNGC0LXQu9GM0L3QvtGB0YLRjCB7QGxpbmsgZGVib3VuY2VUaW1lfSDQsiDQvNC40LvQu9C40YHQtdC60YPQvdC00LDRhSDQv9C+INGD0LzQvtC70YfQsNC90LjRji5cbiAqL1xuZXhwb3J0IGNvbnN0IFNDX0RFQk9VTkNFX1RJTUVfREVGQVVMVDogSW5qZWN0aW9uVG9rZW48bnVtYmVyPiA9IG5ldyBJbmplY3Rpb25Ub2tlbjxudW1iZXI+KCdzY0RlYm91bmNlVGltZURlZmF1bHQnLCB7XG4gICAgcHJvdmlkZWRJbjogJ3Jvb3QnLFxuICAgIGZhY3Rvcnk6ICgpID0+IDMwMCxcbn0pO1xuIl19
@@ -8,6 +8,7 @@ export * from './accordion';
8
8
  export * from './banner';
9
9
  export * from './cart';
10
10
  export * from './catalog';
11
+ export * from './configurators';
11
12
  export * from './contragents';
12
13
  export * from './contacts';
13
14
  export * from './delivery-address';
@@ -34,4 +35,4 @@ export * from './validators';
34
35
  export * from './verification';
35
36
  export * from './profile';
36
37
  export * from './schemas';
37
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxvQkFBb0IsQ0FBQztBQUNuQyxjQUFjLFdBQVcsQ0FBQztBQUMxQixjQUFjLGNBQWMsQ0FBQztBQUM3QixjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsVUFBVSxDQUFDO0FBQ3pCLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsUUFBUSxDQUFDO0FBQ3ZCLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLFFBQVEsQ0FBQztBQUN2QixjQUFjLGNBQWMsQ0FBQztBQUM3QixjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsV0FBVyxDQUFDO0FBQzFCLGNBQWMsV0FBVyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiB1aVxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vbW9kZWxzJztcbmV4cG9ydCAqIGZyb20gJy4vc2VydmljZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9hdXRoJztcbmV4cG9ydCAqIGZyb20gJy4vYWNjb3JkaW9uJztcbmV4cG9ydCAqIGZyb20gJy4vYmFubmVyJztcbmV4cG9ydCAqIGZyb20gJy4vY2FydCc7XG5leHBvcnQgKiBmcm9tICcuL2NhdGFsb2cnO1xuZXhwb3J0ICogZnJvbSAnLi9jb250cmFnZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnRhY3RzJztcbmV4cG9ydCAqIGZyb20gJy4vZGVsaXZlcnktYWRkcmVzcyc7XG5leHBvcnQgKiBmcm9tICcuL2RpYWxvZ3MnO1xuZXhwb3J0ICogZnJvbSAnLi9kaXJlY3RpdmVzJztcbmV4cG9ydCAqIGZyb20gJy4vZXJyb3ItaGFuZGxlcic7XG5leHBvcnQgKiBmcm9tICcuL2ZpbGVzJztcbmV4cG9ydCAqIGZyb20gJy4vZm9ybS1maWVsZHMnO1xuZXhwb3J0ICogZnJvbSAnLi9pY29ucyc7XG5leHBvcnQgKiBmcm9tICcuL2xvYWRlcic7XG5leHBvcnQgKiBmcm9tICcuL2JyYW5kcy1saXN0JztcbmV4cG9ydCAqIGZyb20gJy4vbmV3cyc7XG5leHBvcnQgKiBmcm9tICcuL25vaW5kZXgtd3JhcHBlcic7XG5leHBvcnQgKiBmcm9tICcuL29yZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vcGFnZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL3NhbXBsZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9wcm92aWRlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9mZWVkYmFjayc7XG5leHBvcnQgKiBmcm9tICcuL3FyY29kZSc7XG5leHBvcnQgKiBmcm9tICcuL3NoYXJlLWJ1dHRvbic7XG5leHBvcnQgKiBmcm9tICcuL3VzZXInO1xuZXhwb3J0ICogZnJvbSAnLi92YWxpZGF0b3JzJztcbmV4cG9ydCAqIGZyb20gJy4vdmVyaWZpY2F0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vcHJvZmlsZSc7XG5leHBvcnQgKiBmcm9tICcuL3NjaGVtYXMnO1xuIl19
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxpQkFBaUIsQ0FBQztBQUNoQyxjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLFlBQVksQ0FBQztBQUMzQixjQUFjLG9CQUFvQixDQUFDO0FBQ25DLGNBQWMsV0FBVyxDQUFDO0FBQzFCLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxtQkFBbUIsQ0FBQztBQUNsQyxjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLFdBQVcsQ0FBQztBQUMxQixjQUFjLGFBQWEsQ0FBQztBQUM1QixjQUFjLFlBQVksQ0FBQztBQUMzQixjQUFjLFVBQVUsQ0FBQztBQUN6QixjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsUUFBUSxDQUFDO0FBQ3ZCLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxXQUFXLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIHVpXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9tb2RlbHMnO1xuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcyc7XG5leHBvcnQgKiBmcm9tICcuL2F1dGgnO1xuZXhwb3J0ICogZnJvbSAnLi9hY2NvcmRpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9iYW5uZXInO1xuZXhwb3J0ICogZnJvbSAnLi9jYXJ0JztcbmV4cG9ydCAqIGZyb20gJy4vY2F0YWxvZyc7XG5leHBvcnQgKiBmcm9tICcuL2NvbmZpZ3VyYXRvcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9jb250cmFnZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnRhY3RzJztcbmV4cG9ydCAqIGZyb20gJy4vZGVsaXZlcnktYWRkcmVzcyc7XG5leHBvcnQgKiBmcm9tICcuL2RpYWxvZ3MnO1xuZXhwb3J0ICogZnJvbSAnLi9kaXJlY3RpdmVzJztcbmV4cG9ydCAqIGZyb20gJy4vZXJyb3ItaGFuZGxlcic7XG5leHBvcnQgKiBmcm9tICcuL2ZpbGVzJztcbmV4cG9ydCAqIGZyb20gJy4vZm9ybS1maWVsZHMnO1xuZXhwb3J0ICogZnJvbSAnLi9pY29ucyc7XG5leHBvcnQgKiBmcm9tICcuL2xvYWRlcic7XG5leHBvcnQgKiBmcm9tICcuL2JyYW5kcy1saXN0JztcbmV4cG9ydCAqIGZyb20gJy4vbmV3cyc7XG5leHBvcnQgKiBmcm9tICcuL25vaW5kZXgtd3JhcHBlcic7XG5leHBvcnQgKiBmcm9tICcuL29yZGVyJztcbmV4cG9ydCAqIGZyb20gJy4vcGFnZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL3NhbXBsZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9wcm92aWRlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9mZWVkYmFjayc7XG5leHBvcnQgKiBmcm9tICcuL3FyY29kZSc7XG5leHBvcnQgKiBmcm9tICcuL3NoYXJlLWJ1dHRvbic7XG5leHBvcnQgKiBmcm9tICcuL3VzZXInO1xuZXhwb3J0ICogZnJvbSAnLi92YWxpZGF0b3JzJztcbmV4cG9ydCAqIGZyb20gJy4vdmVyaWZpY2F0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vcHJvZmlsZSc7XG5leHBvcnQgKiBmcm9tICcuL3NjaGVtYXMnO1xuIl19