@snabcentr/client-ui 3.30.0 → 3.31.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/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.d.ts +97 -0
- package/cart/add-or-editing-cart-item-dialog/index.d.ts +2 -0
- package/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.d.ts +48 -0
- package/cart/index.d.ts +1 -0
- package/directives/index.d.ts +1 -0
- package/directives/select-on-focusin/sc-select-on-focusin.directive.d.ts +14 -0
- package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +196 -0
- package/esm2022/cart/add-or-editing-cart-item-dialog/index.mjs +3 -0
- package/esm2022/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.mjs +74 -0
- package/esm2022/cart/index.mjs +2 -1
- package/esm2022/directives/index.mjs +2 -1
- package/esm2022/directives/select-on-focusin/sc-select-on-focusin.directive.mjs +31 -0
- package/fesm2022/snabcentr-client-ui.mjs +274 -6
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +1 -1
- package/release_notes.tmp +3 -3
- package/styles/tailwind/tailwind.scss +5 -0
@@ -0,0 +1,97 @@
|
|
1
|
+
import { OnInit } from '@angular/core';
|
2
|
+
import { FormControl, FormGroup } from '@angular/forms';
|
3
|
+
import { ScCartItem, ScINewCartItemBase, ScProduct } from '@snabcentr/client-core';
|
4
|
+
import * as i0 from "@angular/core";
|
5
|
+
/**
|
6
|
+
* Компонент добавления / изменения товара в корзине.
|
7
|
+
*/
|
8
|
+
export declare class ScAddOrEditingCartItemFormComponent implements OnInit {
|
9
|
+
/**
|
10
|
+
* Группа полей добавления / изменения товара в корзине.
|
11
|
+
*/
|
12
|
+
readonly form: FormGroup<{
|
13
|
+
length?: FormControl<number | null>;
|
14
|
+
quantity: FormControl<number | null>;
|
15
|
+
marker: FormControl<string | null>;
|
16
|
+
}>;
|
17
|
+
/**
|
18
|
+
* Данные о товаре.
|
19
|
+
*/
|
20
|
+
readonly product: import("@angular/core").InputSignal<ScProduct>;
|
21
|
+
/**
|
22
|
+
* Данные о товаре в корзине.
|
23
|
+
*/
|
24
|
+
readonly cartItem: import("@angular/core").InputSignal<ScCartItem | undefined>;
|
25
|
+
/**
|
26
|
+
* Признак загрузки данных.
|
27
|
+
*/
|
28
|
+
readonly isLoading: import("@angular/core").InputSignal<boolean>;
|
29
|
+
/**
|
30
|
+
* Признак возможности продажи товара на метраж.
|
31
|
+
*/
|
32
|
+
protected readonly productIsMeasurable: import("@angular/core").Signal<boolean>;
|
33
|
+
/**
|
34
|
+
* Кратность количества для товара.
|
35
|
+
*/
|
36
|
+
protected readonly productMultiplicity: import("@angular/core").Signal<number>;
|
37
|
+
/**
|
38
|
+
* Минимальный метраж для товара.
|
39
|
+
*/
|
40
|
+
protected readonly minLength: import("@angular/core").Signal<number>;
|
41
|
+
/**
|
42
|
+
* Максимальный метраж для товара.
|
43
|
+
*/
|
44
|
+
protected readonly maxLength: import("@angular/core").Signal<number | undefined>;
|
45
|
+
/**
|
46
|
+
* Шаг изменения метража.
|
47
|
+
*/
|
48
|
+
protected readonly lengthStep: import("@angular/core").Signal<number | undefined>;
|
49
|
+
/**
|
50
|
+
* Подсказка по минимальному и максимальному метражу товара.
|
51
|
+
*/
|
52
|
+
protected readonly lengthHint: import("@angular/core").Signal<string>;
|
53
|
+
/**
|
54
|
+
* Итоговая стоимость заказа.
|
55
|
+
*/
|
56
|
+
protected readonly totalCost: import("@angular/core").Signal<number | undefined>;
|
57
|
+
/**
|
58
|
+
* {@link Output} события добавления товара в корзину.
|
59
|
+
*/
|
60
|
+
readonly addToCart: import("@angular/core").OutputEmitterRef<ScINewCartItemBase>;
|
61
|
+
/**
|
62
|
+
* {@link Output} события редактирования товара в корзине.
|
63
|
+
*/
|
64
|
+
readonly editCartItem: import("@angular/core").OutputEmitterRef<Omit<ScINewCartItemBase, "productId">>;
|
65
|
+
/**
|
66
|
+
* Объект-помощник для работы со значениями единиц измерения товара.
|
67
|
+
*/
|
68
|
+
private readonly unitsHelper;
|
69
|
+
/**
|
70
|
+
* Сервис для сбора метрик о действиях пользователей.
|
71
|
+
*/
|
72
|
+
private readonly userMetrikaService;
|
73
|
+
/**
|
74
|
+
* Сервис конвертации данных.
|
75
|
+
*/
|
76
|
+
private readonly convertersService;
|
77
|
+
/** @inheritDoc */
|
78
|
+
ngOnInit(): void;
|
79
|
+
/**
|
80
|
+
* Обработчик события шага метража.
|
81
|
+
*
|
82
|
+
* @param step Шаг.
|
83
|
+
*/
|
84
|
+
protected onStepLength(step: number): void;
|
85
|
+
/**
|
86
|
+
* Обработчик события шага количества.
|
87
|
+
*
|
88
|
+
* @param step Шаг.
|
89
|
+
*/
|
90
|
+
protected onStepQuantity(step: number): void;
|
91
|
+
/**
|
92
|
+
* Обработчик события отправки формы.
|
93
|
+
*/
|
94
|
+
protected onSubmit(): void;
|
95
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScAddOrEditingCartItemFormComponent, never>;
|
96
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScAddOrEditingCartItemFormComponent, "sc-add-or-editing-cart-item-form", never, { "product": { "alias": "product"; "required": true; "isSignal": true; }; "cartItem": { "alias": "cartItem"; "required": false; "isSignal": true; }; "isLoading": { "alias": "isLoading"; "required": false; "isSignal": true; }; }, { "addToCart": "addToCart"; "editCartItem": "editCartItem"; }, never, never, true, never>;
|
97
|
+
}
|
package/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.d.ts
ADDED
@@ -0,0 +1,48 @@
|
|
1
|
+
import { Signal } from '@angular/core';
|
2
|
+
import { ScCartItem, ScINewCartItemBase, ScProduct } from '@snabcentr/client-core';
|
3
|
+
import { TuiDialogContext } from '@taiga-ui/core';
|
4
|
+
import { Observable, Subject } from 'rxjs';
|
5
|
+
import { ScAddOrEditingCartItemFormComponent } from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';
|
6
|
+
import * as i0 from "@angular/core";
|
7
|
+
/**
|
8
|
+
* Компонент добавления / изменения товара в корзине.
|
9
|
+
*/
|
10
|
+
export declare class ScAddOrEditingCartItemDialogComponent {
|
11
|
+
/**
|
12
|
+
* Компонент формы добавления / изменения товара в корзине.
|
13
|
+
*/
|
14
|
+
readonly formComponent: Signal<ScAddOrEditingCartItemFormComponent>;
|
15
|
+
/**
|
16
|
+
* {@link Subject} события отправки формы.
|
17
|
+
*/
|
18
|
+
readonly onSubmit: Subject<ScINewCartItemBase | Omit<ScINewCartItemBase, 'productId'>>;
|
19
|
+
/**
|
20
|
+
* {@link Observable} запроса добавления / изменения товара в корзине.
|
21
|
+
*/
|
22
|
+
readonly submit$: Observable<import("@snabcentr/client-core").ScCart | null>;
|
23
|
+
/**
|
24
|
+
* {@link Observable} изменения состояния загрузки данных.
|
25
|
+
*/
|
26
|
+
readonly loading: Signal<boolean>;
|
27
|
+
/**
|
28
|
+
* Контекст диалогового окна.
|
29
|
+
*/
|
30
|
+
readonly context: TuiDialogContext<boolean, {
|
31
|
+
product: ScProduct;
|
32
|
+
cartItem: ScCartItem | undefined;
|
33
|
+
}>;
|
34
|
+
/**
|
35
|
+
* Данные о товаре.
|
36
|
+
*/
|
37
|
+
readonly product: ScProduct;
|
38
|
+
/**
|
39
|
+
* Данные о товаре в корзине.
|
40
|
+
*/
|
41
|
+
readonly cartItem: ScCartItem | undefined;
|
42
|
+
/**
|
43
|
+
* Сервис для работы с корзиной.
|
44
|
+
*/
|
45
|
+
private readonly cartService;
|
46
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScAddOrEditingCartItemDialogComponent, never>;
|
47
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScAddOrEditingCartItemDialogComponent, "sc-add-or-editing-cart-item-dialog", never, {}, {}, never, never, true, never>;
|
48
|
+
}
|
package/cart/index.d.ts
CHANGED
package/directives/index.d.ts
CHANGED
@@ -4,3 +4,4 @@ export * from './abstract-price-card/abstract-sc-price-card.directive';
|
|
4
4
|
export * from './terminal-link/sc-terminal-link.directive';
|
5
5
|
export * from './links';
|
6
6
|
export * from './sc-date-value-transformer.directive';
|
7
|
+
export * from './select-on-focusin/sc-select-on-focusin.directive';
|
@@ -0,0 +1,14 @@
|
|
1
|
+
import * as i0 from "@angular/core";
|
2
|
+
/**
|
3
|
+
* Директива для обработки события фокуса поля ввода, для последующего выделения содержимого поля ввода.
|
4
|
+
*/
|
5
|
+
export declare class ScSelectOnFocusinDirective {
|
6
|
+
/**
|
7
|
+
* Слушатель и обработчик события `focusin`.
|
8
|
+
*
|
9
|
+
* @param target Целевой объект события `focusin`.
|
10
|
+
*/
|
11
|
+
protected onFocusIn(target: HTMLElement): void;
|
12
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScSelectOnFocusinDirective, never>;
|
13
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<ScSelectOnFocusinDirective, "tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber]", never, {}, {}, never, never, true, never>;
|
14
|
+
}
|
@@ -0,0 +1,196 @@
|
|
1
|
+
/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */
|
2
|
+
import { AsyncPipe, NgIf } from '@angular/common';
|
3
|
+
import { ChangeDetectionStrategy, Component, computed, inject, input, output } from '@angular/core';
|
4
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
5
|
+
import { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';
|
6
|
+
import { ScConvertersService, ScUnitsHelper, ScUserMetrikaGoalsEnum, ScUserMetrikaService, } from '@snabcentr/client-core';
|
7
|
+
import { TuiAmountPipe } from '@taiga-ui/addon-commerce';
|
8
|
+
import { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';
|
9
|
+
import { TuiButton, TuiError, TuiLabel, TuiNumberFormat, 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 { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';
|
13
|
+
import { TuiInputModule } from '@taiga-ui/legacy';
|
14
|
+
import { combineLatest, debounceTime, map, of } 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 { stepValidator } from '../../../validators/step-validator';
|
18
|
+
import * as i0 from "@angular/core";
|
19
|
+
import * as i1 from "@angular/forms";
|
20
|
+
import * as i2 from "../../../directives/next-input-focus/sc-next-input-focus.directive";
|
21
|
+
import * as i3 from "@taiga-ui/legacy";
|
22
|
+
import * as i4 from "@taiga-ui/legacy/components/primitive-textfield";
|
23
|
+
import * as i5 from "@taiga-ui/kit";
|
24
|
+
import * as i6 from "@taiga-ui/core";
|
25
|
+
/**
|
26
|
+
* Компонент добавления / изменения товара в корзине.
|
27
|
+
*/
|
28
|
+
export class ScAddOrEditingCartItemFormComponent {
|
29
|
+
constructor() {
|
30
|
+
/**
|
31
|
+
* Группа полей добавления / изменения товара в корзине.
|
32
|
+
*/
|
33
|
+
this.form = new FormGroup({
|
34
|
+
quantity: new FormControl(null, Validators.required),
|
35
|
+
marker: new FormControl(''),
|
36
|
+
});
|
37
|
+
/**
|
38
|
+
* Данные о товаре.
|
39
|
+
*/
|
40
|
+
this.product = input.required();
|
41
|
+
/**
|
42
|
+
* Данные о товаре в корзине.
|
43
|
+
*/
|
44
|
+
this.cartItem = input();
|
45
|
+
/**
|
46
|
+
* Признак загрузки данных.
|
47
|
+
*/
|
48
|
+
this.isLoading = input(false);
|
49
|
+
/**
|
50
|
+
* Признак возможности продажи товара на метраж.
|
51
|
+
*/
|
52
|
+
this.productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));
|
53
|
+
/**
|
54
|
+
* Кратность количества для товара.
|
55
|
+
*/
|
56
|
+
this.productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));
|
57
|
+
/**
|
58
|
+
* Минимальный метраж для товара.
|
59
|
+
*/
|
60
|
+
this.minLength = computed(() => this.cartItem()?.length ?? this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount) ?? 1);
|
61
|
+
/**
|
62
|
+
* Максимальный метраж для товара.
|
63
|
+
*/
|
64
|
+
this.maxLength = computed(() => this.product().properties?.maxLength);
|
65
|
+
/**
|
66
|
+
* Шаг изменения метража.
|
67
|
+
*/
|
68
|
+
this.lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));
|
69
|
+
/**
|
70
|
+
* Подсказка по минимальному и максимальному метражу товара.
|
71
|
+
*/
|
72
|
+
this.lengthHint = computed(() => {
|
73
|
+
const min = this.minLength();
|
74
|
+
const max = this.maxLength();
|
75
|
+
// eslint-disable-next-line sonarjs/no-nested-conditional, @typescript-eslint/no-unnecessary-condition
|
76
|
+
return (min ?? max) ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';
|
77
|
+
});
|
78
|
+
/**
|
79
|
+
* Итоговая стоимость заказа.
|
80
|
+
*/
|
81
|
+
this.totalCost = toSignal(combineLatest({
|
82
|
+
// eslint-disable-next-line unicorn/explicit-length-check
|
83
|
+
length: this.form.controls.length ? tuiControlValue(this.form.controls.length) : of(1),
|
84
|
+
quantity: tuiControlValue(this.form.controls.quantity),
|
85
|
+
}).pipe(debounceTime(0), // Исправляем ошибку NG0950.
|
86
|
+
map(({ length, quantity }) => (this.product().costRub ?? 0) * length * quantity), map((sum) => Math.round(sum * 100) / 100)));
|
87
|
+
/**
|
88
|
+
* {@link Output} события добавления товара в корзину.
|
89
|
+
*/
|
90
|
+
this.addToCart = output();
|
91
|
+
/**
|
92
|
+
* {@link Output} события редактирования товара в корзине.
|
93
|
+
*/
|
94
|
+
this.editCartItem = output();
|
95
|
+
/**
|
96
|
+
* Объект-помощник для работы со значениями единиц измерения товара.
|
97
|
+
*/
|
98
|
+
this.unitsHelper = inject(ScUnitsHelper);
|
99
|
+
/**
|
100
|
+
* Сервис для сбора метрик о действиях пользователей.
|
101
|
+
*/
|
102
|
+
this.userMetrikaService = inject(ScUserMetrikaService);
|
103
|
+
/**
|
104
|
+
* Сервис конвертации данных.
|
105
|
+
*/
|
106
|
+
this.convertersService = inject(ScConvertersService);
|
107
|
+
}
|
108
|
+
/** @inheritDoc */
|
109
|
+
ngOnInit() {
|
110
|
+
if (this.productIsMeasurable()) {
|
111
|
+
this.form.addControl('length', new FormControl(this.minLength(), [Validators.required, Validators.min(0.01)]));
|
112
|
+
}
|
113
|
+
this.form.patchValue({
|
114
|
+
quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),
|
115
|
+
});
|
116
|
+
this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));
|
117
|
+
this.form.controls.length?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));
|
118
|
+
this.userMetrikaService.emitUserMetrikaEvent({
|
119
|
+
target: ScUserMetrikaGoalsEnum.cartItemAddShow,
|
120
|
+
params: { product_id: this.product().id },
|
121
|
+
});
|
122
|
+
}
|
123
|
+
/**
|
124
|
+
* Обработчик события шага метража.
|
125
|
+
*
|
126
|
+
* @param step Шаг.
|
127
|
+
*/
|
128
|
+
onStepLength(step) {
|
129
|
+
// eslint-disable-next-line unicorn/explicit-length-check
|
130
|
+
if (!this.productIsMeasurable() || !this.form.controls.length) {
|
131
|
+
return;
|
132
|
+
}
|
133
|
+
const length = this.form.controls.length.value ?? 0;
|
134
|
+
this.form.controls.length.setValue(Math.max(this.minLength(), length - (length % step) + step));
|
135
|
+
}
|
136
|
+
/**
|
137
|
+
* Обработчик события шага количества.
|
138
|
+
*
|
139
|
+
* @param step Шаг.
|
140
|
+
*/
|
141
|
+
onStepQuantity(step) {
|
142
|
+
const quantity = this.form.controls.quantity.value ?? 0;
|
143
|
+
this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));
|
144
|
+
}
|
145
|
+
/**
|
146
|
+
* Обработчик события отправки формы.
|
147
|
+
*/
|
148
|
+
onSubmit() {
|
149
|
+
const value = this.convertersService.removeNull(this.form.value);
|
150
|
+
if (!this.cartItem()) {
|
151
|
+
value.productId = this.product().id;
|
152
|
+
}
|
153
|
+
const step = this.lengthStep();
|
154
|
+
if (this.productIsMeasurable() && step) {
|
155
|
+
value.quantity = Math.round(value.quantity * (value.length / step));
|
156
|
+
value.length = step;
|
157
|
+
}
|
158
|
+
tuiMarkControlAsTouchedAndValidate(this.form);
|
159
|
+
if (this.cartItem()) {
|
160
|
+
this.editCartItem.emit(value);
|
161
|
+
}
|
162
|
+
else {
|
163
|
+
this.addToCart.emit(value);
|
164
|
+
}
|
165
|
+
}
|
166
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScAddOrEditingCartItemFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
167
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: ScAddOrEditingCartItemFormComponent, isStandalone: true, selector: "sc-add-or-editing-cart-item-form", inputs: { product: { classPropertyName: "product", publicName: "product", isSignal: true, isRequired: true, transformFunction: null }, cartItem: { classPropertyName: "cartItem", publicName: "cartItem", isSignal: true, isRequired: false, transformFunction: null }, isLoading: { classPropertyName: "isLoading", publicName: "isLoading", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { addToCart: "addToCart", editCartItem: "editCartItem" }, ngImport: i0, template: "<!-- \u0424\u043E\u0440\u043C\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F \u0438\u0437\u043C\u0435\u0440\u044F\u0435\u043C\u043E\u0433\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n<form\n *ngIf=\"product\"\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit()\"\n ScNextInputFocus\n class=\"flex flex-col gap-2\"\n>\n <!-- \u0414\u043B\u0438\u043D\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 (\u043C\u0435\u0442\u0440\u0430\u0436) -->\n <label\n *ngIf=\"productIsMeasurable()\"\n tuiLabel\n >\n \u041C\u0435\u0442\u0440\u0430\u0436, {{ product().unit }} {{ lengthHint() }}\n\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"length\"\n [tuiNumberFormat]=\"{ precision: 2 }\"\n [max]=\"maxLength() || null\"\n [min]=\"minLength() || null\"\n (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n autocomplete=\"length\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n <p\n *ngIf=\"lengthStep()\"\n class=\"tui-form__field-note\"\n >\n \u041C\u0435\u0442\u0440\u0430\u0436 \u0434\u043E\u043B\u0436\u0435\u043D \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u0435\u043D {{ lengthStep() }}\n </p>\n </label>\n\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430 -->\n <label tuiLabel>\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442.\n <tui-textfield>\n @let step = lengthStep();\n <tui-chip\n *ngIf=\"productMultiplicity() && step\"\n size=\"s\"\n appearance=\"negative\"\n class=\"font-bold\"\n >\n x {{ step }} {{ product().unit }}\n </tui-chip>\n\n <input\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n tuiInputNumber\n [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n [min]=\"productMultiplicity()\"\n (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n formControlName=\"quantity\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n <p class=\"tui-form__field-note\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0434\u043E\u043B\u0436\u043D\u043E \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u043D\u043E {{ productMultiplicity() }}</p>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 -->\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-input formControlName=\"marker\">\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <input\n tuiTextfieldLegacy\n autocomplete=\"marker\"\n />\n </tui-input>\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <div class=\"flex flex-col items-center\">\n @let cost = totalCost();\n\n <div *ngIf=\"cost\">\n \u0418\u0442\u043E\u0433\u043E:<span class=\"text-2xl font-bold\">\n {{ cost | tuiAmount: 'RUB' | async }}\n {{ product().currency }}</span\n >\n </div>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <button\n tuiButton\n iconStart=\"@tui.check\"\n [disabled]=\"form.invalid\"\n [loading]=\"isLoading()\"\n type=\"submit\"\n class=\"mt-2\"\n >\n {{ cartItem() ? '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' : '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' }}\n </button>\n </div>\n</form>\n", dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { 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: ScNextInputFocusModule }, { kind: "directive", type: i2.ScNextInputFocusDirective, selector: "form[ScNextInputFocus]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { 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: TuiLabel, selector: "label[tuiLabel]" }, { kind: "directive", type: ScSelectOnFocusinDirective, selector: "tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber]" }, { kind: "directive", type: TuiNumberFormat, selector: "[tuiNumberFormat]", inputs: ["tuiNumberFormat"] }, { kind: "component", type: TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "ngmodule", type: TuiInputModule }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "component", type: i4.TuiTextfieldComponent, selector: "input[tuiTextfieldLegacy], textarea[tuiTextfieldLegacy]" }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "pipe", type: TuiFieldErrorPipe, name: "tuiFieldError" }, { kind: "directive", type: i5.TuiInputNumberDirective, selector: "input[tuiInputNumber]", inputs: ["min", "max", "prefix", "postfix"] }, { kind: "component", type: i6.TuiTextfieldComponent, selector: "tui-textfield", inputs: ["content", "filler"] }, { kind: "pipe", type: TuiAmountPipe, name: "tuiAmount" }, { kind: "directive", type: TuiChip, selector: "tui-chip,[tuiChip]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
168
|
+
}
|
169
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScAddOrEditingCartItemFormComponent, decorators: [{
|
170
|
+
type: Component,
|
171
|
+
args: [{ standalone: true, selector: 'sc-add-or-editing-cart-item-form', imports: [
|
172
|
+
NgIf,
|
173
|
+
FormsModule,
|
174
|
+
ScNextInputFocusModule,
|
175
|
+
ReactiveFormsModule,
|
176
|
+
TuiLabel,
|
177
|
+
ScSelectOnFocusinDirective,
|
178
|
+
TuiNumberFormat,
|
179
|
+
TuiError,
|
180
|
+
TuiInputModule,
|
181
|
+
TuiAppearance,
|
182
|
+
TuiWithAppearance,
|
183
|
+
TuiIcons,
|
184
|
+
TuiWithIcons,
|
185
|
+
TuiButton,
|
186
|
+
TuiButtonLoading,
|
187
|
+
AsyncPipe,
|
188
|
+
TuiFieldErrorPipe,
|
189
|
+
TuiInputNumber,
|
190
|
+
TuiNumberFormat,
|
191
|
+
TuiTextfield,
|
192
|
+
TuiAmountPipe,
|
193
|
+
TuiChip,
|
194
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- \u0424\u043E\u0440\u043C\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F \u0438\u0437\u043C\u0435\u0440\u044F\u0435\u043C\u043E\u0433\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n<form\n *ngIf=\"product\"\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit()\"\n ScNextInputFocus\n class=\"flex flex-col gap-2\"\n>\n <!-- \u0414\u043B\u0438\u043D\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 (\u043C\u0435\u0442\u0440\u0430\u0436) -->\n <label\n *ngIf=\"productIsMeasurable()\"\n tuiLabel\n >\n \u041C\u0435\u0442\u0440\u0430\u0436, {{ product().unit }} {{ lengthHint() }}\n\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"length\"\n [tuiNumberFormat]=\"{ precision: 2 }\"\n [max]=\"maxLength() || null\"\n [min]=\"minLength() || null\"\n (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n autocomplete=\"length\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n <p\n *ngIf=\"lengthStep()\"\n class=\"tui-form__field-note\"\n >\n \u041C\u0435\u0442\u0440\u0430\u0436 \u0434\u043E\u043B\u0436\u0435\u043D \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u0435\u043D {{ lengthStep() }}\n </p>\n </label>\n\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430 -->\n <label tuiLabel>\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442.\n <tui-textfield>\n @let step = lengthStep();\n <tui-chip\n *ngIf=\"productMultiplicity() && step\"\n size=\"s\"\n appearance=\"negative\"\n class=\"font-bold\"\n >\n x {{ step }} {{ product().unit }}\n </tui-chip>\n\n <input\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n tuiInputNumber\n [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n [min]=\"productMultiplicity()\"\n (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n formControlName=\"quantity\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n <p class=\"tui-form__field-note\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0434\u043E\u043B\u0436\u043D\u043E \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u043D\u043E {{ productMultiplicity() }}</p>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 -->\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-input formControlName=\"marker\">\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <input\n tuiTextfieldLegacy\n autocomplete=\"marker\"\n />\n </tui-input>\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <div class=\"flex flex-col items-center\">\n @let cost = totalCost();\n\n <div *ngIf=\"cost\">\n \u0418\u0442\u043E\u0433\u043E:<span class=\"text-2xl font-bold\">\n {{ cost | tuiAmount: 'RUB' | async }}\n {{ product().currency }}</span\n >\n </div>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <button\n tuiButton\n iconStart=\"@tui.check\"\n [disabled]=\"form.invalid\"\n [loading]=\"isLoading()\"\n type=\"submit\"\n class=\"mt-2\"\n >\n {{ cartItem() ? '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' : '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' }}\n </button>\n </div>\n</form>\n" }]
|
195
|
+
}] });
|
196
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { combineLatest, debounceTime, map, of } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(\n        () => this.cartItem()?.length ?? this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount) ?? 1\n    );\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, @typescript-eslint/no-unnecessary-condition\n        return (min ?? max) ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected readonly totalCost = toSignal(\n        combineLatest({\n            // eslint-disable-next-line unicorn/explicit-length-check\n            length: this.form.controls.length ? tuiControlValue<number>(this.form.controls.length) : of(1),\n            quantity: tuiControlValue<number>(this.form.controls.quantity),\n        }).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * length * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        )\n    );\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (this.productIsMeasurable()) {\n            this.form.addControl('length', new FormControl<number | null>(this.minLength(), [Validators.required, Validators.min(0.01)]));\n        }\n\n        this.form.patchValue({\n            quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),\n        });\n\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        this.form.controls.length?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        // eslint-disable-next-line unicorn/explicit-length-check\n        if (!this.productIsMeasurable() || !this.form.controls.length) {\n            return;\n        }\n\n        const length = this.form.controls.length.value ?? 0;\n\n        this.form.controls.length.setValue(Math.max(this.minLength(), length - (length % step) + step));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        if (this.productIsMeasurable() && step) {\n            value.quantity = Math.round(value.quantity * (value.length / step));\n            value.length = step;\n        }\n\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }} {{ lengthHint() }}\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            @let step = lengthStep();\n            <tui-chip\n                *ngIf=\"productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost();\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount: 'RUB' | async }}\n                {{ product().currency }}</span\n            >\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|
@@ -0,0 +1,3 @@
|
|
1
|
+
export * from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';
|
2
|
+
export * from './sc-add-or-editing-cart-item-dialog.component';
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9hZGQtb3ItZWRpdGluZy1jYXJ0LWl0ZW0tZGlhbG9nL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNEVBQTRFLENBQUM7QUFDM0YsY0FBYyxnREFBZ0QsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vYWRkLW9yLWVkaXRpbmctY2FydC1pdGVtLWZvcm0vc2MtYWRkLW9yLWVkaXRpbmctY2FydC1pdGVtLWZvcm0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtYWRkLW9yLWVkaXRpbmctY2FydC1pdGVtLWRpYWxvZy5jb21wb25lbnQnO1xuIl19
|
@@ -0,0 +1,74 @@
|
|
1
|
+
/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method,@typescript-eslint/no-unused-vars */
|
2
|
+
import { HttpErrorResponse } from '@angular/common/http';
|
3
|
+
import { ChangeDetectionStrategy, Component, inject, viewChild } from '@angular/core';
|
4
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
5
|
+
import { ScCartService } from '@snabcentr/client-core';
|
6
|
+
import { tuiIsFalsy } from '@taiga-ui/cdk';
|
7
|
+
import { injectContext } from '@taiga-ui/polymorpheus';
|
8
|
+
import { isObject } from 'lodash-es';
|
9
|
+
import { catchError, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
|
10
|
+
import { ScAddOrEditingCartItemFormComponent } from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';
|
11
|
+
import * as i0 from "@angular/core";
|
12
|
+
/**
|
13
|
+
* Компонент добавления / изменения товара в корзине.
|
14
|
+
*/
|
15
|
+
export class ScAddOrEditingCartItemDialogComponent {
|
16
|
+
constructor() {
|
17
|
+
/**
|
18
|
+
* Компонент формы добавления / изменения товара в корзине.
|
19
|
+
*/
|
20
|
+
this.formComponent = viewChild.required(ScAddOrEditingCartItemFormComponent);
|
21
|
+
/**
|
22
|
+
* {@link Subject} события отправки формы.
|
23
|
+
*/
|
24
|
+
this.onSubmit = new Subject();
|
25
|
+
/**
|
26
|
+
* {@link Observable} запроса добавления / изменения товара в корзине.
|
27
|
+
*/
|
28
|
+
this.submit$ = this.onSubmit.pipe(switchMap((value) => (this.cartItem && !('productId' in value) ? this.cartService.patchCartItem$(this.cartItem.id, value) : this.cartService.addToCart$(value)).pipe(tap(() => {
|
29
|
+
this.context.$implicit.complete();
|
30
|
+
}), catchError((error) => {
|
31
|
+
if (error instanceof HttpErrorResponse) {
|
32
|
+
const { errors, message } = error.error;
|
33
|
+
if (errors && isObject(errors)) {
|
34
|
+
Object.entries(errors).forEach(([k, v]) => {
|
35
|
+
this.formComponent().form.get(k)?.setErrors({ serverResponse: v });
|
36
|
+
});
|
37
|
+
}
|
38
|
+
if (message) {
|
39
|
+
this.formComponent().form.setErrors({ serverResponse: [message] });
|
40
|
+
}
|
41
|
+
this.formComponent().form.updateValueAndValidity();
|
42
|
+
this.formComponent().form.markAsDirty();
|
43
|
+
}
|
44
|
+
return of();
|
45
|
+
}), startWith(null))), startWith(), share());
|
46
|
+
/**
|
47
|
+
* {@link Observable} изменения состояния загрузки данных.
|
48
|
+
*/
|
49
|
+
this.loading = toSignal(this.submit$.pipe(map(tuiIsFalsy)), { initialValue: false });
|
50
|
+
/**
|
51
|
+
* Контекст диалогового окна.
|
52
|
+
*/
|
53
|
+
this.context = injectContext();
|
54
|
+
/**
|
55
|
+
* Данные о товаре.
|
56
|
+
*/
|
57
|
+
this.product = this.context.data.product;
|
58
|
+
/**
|
59
|
+
* Данные о товаре в корзине.
|
60
|
+
*/
|
61
|
+
this.cartItem = this.context.data.cartItem;
|
62
|
+
/**
|
63
|
+
* Сервис для работы с корзиной.
|
64
|
+
*/
|
65
|
+
this.cartService = inject(ScCartService);
|
66
|
+
}
|
67
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScAddOrEditingCartItemDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
68
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ScAddOrEditingCartItemDialogComponent, isStandalone: true, selector: "sc-add-or-editing-cart-item-dialog", viewQueries: [{ propertyName: "formComponent", first: true, predicate: ScAddOrEditingCartItemFormComponent, descendants: true, isSignal: true }], ngImport: i0, template: "@if (product) {\n <sc-add-or-editing-cart-item-form\n [product]=\"product\"\n [cartItem]=\"cartItem\"\n (addToCart)=\"onSubmit.next($event)\"\n (editCartItem)=\"onSubmit.next($event)\"\n [isLoading]=\"loading()\"\n ></sc-add-or-editing-cart-item-form>\n}\n", dependencies: [{ kind: "component", type: ScAddOrEditingCartItemFormComponent, selector: "sc-add-or-editing-cart-item-form", inputs: ["product", "cartItem", "isLoading"], outputs: ["addToCart", "editCartItem"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
69
|
+
}
|
70
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScAddOrEditingCartItemDialogComponent, decorators: [{
|
71
|
+
type: Component,
|
72
|
+
args: [{ standalone: true, selector: 'sc-add-or-editing-cart-item-dialog', imports: [ScAddOrEditingCartItemFormComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (product) {\n <sc-add-or-editing-cart-item-form\n [product]=\"product\"\n [cartItem]=\"cartItem\"\n (addToCart)=\"onSubmit.next($event)\"\n (editCartItem)=\"onSubmit.next($event)\"\n [isLoading]=\"loading()\"\n ></sc-add-or-editing-cart-item-form>\n}\n" }]
|
73
|
+
}] });
|
74
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.ts","../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method,@typescript-eslint/no-unused-vars */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Signal, viewChild } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { ScCartItem, ScCartService, ScINewCartItemBase, ScProduct } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { injectContext } from '@taiga-ui/polymorpheus';\nimport { isObject } from 'lodash-es';\nimport { catchError, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { ScAddOrEditingCartItemFormComponent } from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-dialog',\n    templateUrl: './sc-add-or-editing-cart-item-dialog.component.html',\n    imports: [ScAddOrEditingCartItemFormComponent],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemDialogComponent {\n    /**\n     * Компонент формы добавления / изменения товара в корзине.\n     */\n    public readonly formComponent = viewChild.required<ScAddOrEditingCartItemFormComponent>(ScAddOrEditingCartItemFormComponent);\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<ScINewCartItemBase | Omit<ScINewCartItemBase, 'productId'>> = new Subject<ScINewCartItemBase | Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * {@link Observable} запроса добавления / изменения товара в корзине.\n     */\n    public readonly submit$ = this.onSubmit.pipe(\n        switchMap((value) =>\n            (this.cartItem && !('productId' in value) ? this.cartService.patchCartItem$(this.cartItem.id, value) : this.cartService.addToCart$(value as ScINewCartItemBase)).pipe(\n                tap(() => {\n                    this.context.$implicit.complete();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        if (errors && isObject(errors)) {\n                            Object.entries(errors).forEach(([k, v]) => {\n                                this.formComponent().form.get(k)?.setErrors({ serverResponse: v });\n                            });\n                        }\n\n                        if (message) {\n                            this.formComponent().form.setErrors({ serverResponse: [message] });\n                        }\n\n                        this.formComponent().form.updateValueAndValidity();\n                        this.formComponent().form.markAsDirty();\n                    }\n                    return of();\n                }),\n                startWith(null)\n            )\n        ),\n        startWith(),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных.\n     */\n    public readonly loading: Signal<boolean> = toSignal(this.submit$.pipe(map(tuiIsFalsy)), { initialValue: false });\n\n    /**\n     * Контекст диалогового окна.\n     */\n    public readonly context = injectContext<TuiDialogContext<boolean, { product: ScProduct; cartItem: ScCartItem | undefined }>>();\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = this.context.data.product;\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = this.context.data.cartItem;\n\n    /**\n     * Сервис для работы с корзиной.\n     */\n    private readonly cartService: ScCartService = inject(ScCartService);\n}\n","@if (product) {\n    <sc-add-or-editing-cart-item-form\n        [product]=\"product\"\n        [cartItem]=\"cartItem\"\n        (addToCart)=\"onSubmit.next($event)\"\n        (editCartItem)=\"onSubmit.next($event)\"\n        [isLoading]=\"loading()\"\n    ></sc-add-or-editing-cart-item-form>\n}\n"]}
|
package/esm2022/cart/index.mjs
CHANGED
@@ -1,3 +1,4 @@
|
|
1
1
|
export * from './cart-item/sc-cart-item.component';
|
2
2
|
export * from './sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component';
|
3
|
-
|
3
|
+
export * from './add-or-editing-cart-item-dialog';
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9DQUFvQyxDQUFDO0FBQ25ELGNBQWMscUZBQXFGLENBQUM7QUFDcEcsY0FBYyxtQ0FBbUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vY2FydC1pdGVtL3NjLWNhcnQtaXRlbS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1jYXItYWRkLXByb2R1Y3RzLWZyb20tY3N2LWRpYWxvZy9zYy1jYXItYWRkLXByb2R1Y3RzLWZyb20tY3N2LWRpYWxvZy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9hZGQtb3ItZWRpdGluZy1jYXJ0LWl0ZW0tZGlhbG9nJztcbiJdfQ==
|
@@ -4,4 +4,5 @@ export * from './abstract-price-card/abstract-sc-price-card.directive';
|
|
4
4
|
export * from './terminal-link/sc-terminal-link.directive';
|
5
5
|
export * from './links';
|
6
6
|
export * from './sc-date-value-transformer.directive';
|
7
|
-
|
7
|
+
export * from './select-on-focusin/sc-select-on-focusin.directive';
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvZGlyZWN0aXZlcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtEQUFrRCxDQUFDO0FBQ2pFLGNBQWMsK0NBQStDLENBQUM7QUFDOUQsY0FBYyx3REFBd0QsQ0FBQztBQUN2RSxjQUFjLDRDQUE0QyxDQUFDO0FBQzNELGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsdUNBQXVDLENBQUM7QUFDdEQsY0FBYyxvREFBb0QsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vbmV4dC1pbnB1dC1mb2N1cy9zYy1uZXh0LWlucHV0LWZvY3VzLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL25leHQtaW5wdXQtZm9jdXMvc2MtbmV4dC1pbnB1dC1mb2N1cy5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi9hYnN0cmFjdC1wcmljZS1jYXJkL2Fic3RyYWN0LXNjLXByaWNlLWNhcmQuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vdGVybWluYWwtbGluay9zYy10ZXJtaW5hbC1saW5rLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpbmtzJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtZGF0ZS12YWx1ZS10cmFuc2Zvcm1lci5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9zZWxlY3Qtb24tZm9jdXNpbi9zYy1zZWxlY3Qtb24tZm9jdXNpbi5kaXJlY3RpdmUnO1xuIl19
|
@@ -0,0 +1,31 @@
|
|
1
|
+
import { Directive } from '@angular/core';
|
2
|
+
import * as i0 from "@angular/core";
|
3
|
+
/**
|
4
|
+
* Директива для обработки события фокуса поля ввода, для последующего выделения содержимого поля ввода.
|
5
|
+
*/
|
6
|
+
export class ScSelectOnFocusinDirective {
|
7
|
+
/**
|
8
|
+
* Слушатель и обработчик события `focusin`.
|
9
|
+
*
|
10
|
+
* @param target Целевой объект события `focusin`.
|
11
|
+
*/
|
12
|
+
// eslint-disable-next-line class-methods-use-this
|
13
|
+
onFocusIn(target) {
|
14
|
+
if (target instanceof HTMLInputElement) {
|
15
|
+
target.select();
|
16
|
+
}
|
17
|
+
}
|
18
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScSelectOnFocusinDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
19
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", 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 }); }
|
20
|
+
}
|
21
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScSelectOnFocusinDirective, decorators: [{
|
22
|
+
type: Directive,
|
23
|
+
args: [{
|
24
|
+
standalone: true,
|
25
|
+
selector: 'tui-input-number, tui-input, tui-input-phone, tui-input-date, tui-input-password, input[tuiInputNumber]',
|
26
|
+
host: {
|
27
|
+
'(focusin)': 'onFocusIn($event.target)',
|
28
|
+
},
|
29
|
+
}]
|
30
|
+
}] });
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2Mtc2VsZWN0LW9uLWZvY3VzaW4uZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2RpcmVjdGl2ZXMvc2VsZWN0LW9uLWZvY3VzaW4vc2Mtc2VsZWN0LW9uLWZvY3VzaW4uZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRTFDOztHQUVHO0FBUUgsTUFBTSxPQUFPLDBCQUEwQjtJQUNuQzs7OztPQUlHO0lBQ0gsa0RBQWtEO0lBQ3hDLFNBQVMsQ0FBQyxNQUFtQjtRQUNuQyxJQUFJLE1BQU0sWUFBWSxnQkFBZ0IsRUFBRSxDQUFDO1lBQ3JDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUNwQixDQUFDO0lBQ0wsQ0FBQzsrR0FYUSwwQkFBMEI7bUdBQTFCLDBCQUEwQjs7NEZBQTFCLDBCQUEwQjtrQkFQdEMsU0FBUzttQkFBQztvQkFDUCxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFLHlHQUF5RztvQkFDbkgsSUFBSSxFQUFFO3dCQUNGLFdBQVcsRUFBRSwwQkFBMEI7cUJBQzFDO2lCQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICog0JTQuNGA0LXQutGC0LjQstCwINC00LvRjyDQvtCx0YDQsNCx0L7RgtC60Lgg0YHQvtCx0YvRgtC40Y8g0YTQvtC60YPRgdCwINC/0L7Qu9GPINCy0LLQvtC00LAsINC00LvRjyDQv9C+0YHQu9C10LTRg9GO0YnQtdCz0L4g0LLRi9C00LXQu9C10L3QuNGPINGB0L7QtNC10YDQttC40LzQvtCz0L4g0L/QvtC70Y8g0LLQstC+0LTQsC5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBzZWxlY3RvcjogJ3R1aS1pbnB1dC1udW1iZXIsIHR1aS1pbnB1dCwgdHVpLWlucHV0LXBob25lLCB0dWktaW5wdXQtZGF0ZSwgdHVpLWlucHV0LXBhc3N3b3JkLCBpbnB1dFt0dWlJbnB1dE51bWJlcl0nLFxuICAgIGhvc3Q6IHtcbiAgICAgICAgJyhmb2N1c2luKSc6ICdvbkZvY3VzSW4oJGV2ZW50LnRhcmdldCknLFxuICAgIH0sXG59KVxuZXhwb3J0IGNsYXNzIFNjU2VsZWN0T25Gb2N1c2luRGlyZWN0aXZlIHtcbiAgICAvKipcbiAgICAgKiDQodC70YPRiNCw0YLQtdC70Ywg0Lgg0L7QsdGA0LDQsdC+0YLRh9C40Log0YHQvtCx0YvRgtC40Y8gYGZvY3VzaW5gLlxuICAgICAqXG4gICAgICogQHBhcmFtIHRhcmdldCDQptC10LvQtdCy0L7QuSDQvtCx0YrQtdC60YIg0YHQvtCx0YvRgtC40Y8gYGZvY3VzaW5gLlxuICAgICAqL1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBjbGFzcy1tZXRob2RzLXVzZS10aGlzXG4gICAgcHJvdGVjdGVkIG9uRm9jdXNJbih0YXJnZXQ6IEhUTUxFbGVtZW50KTogdm9pZCB7XG4gICAgICAgIGlmICh0YXJnZXQgaW5zdGFuY2VvZiBIVE1MSW5wdXRFbGVtZW50KSB7XG4gICAgICAgICAgICB0YXJnZXQuc2VsZWN0KCk7XG4gICAgICAgIH1cbiAgICB9XG59XG4iXX0=
|