@snabcentr/client-ui 3.23.0 → 3.24.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/catalog/index.d.ts +1 -0
- package/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.d.ts +68 -0
- package/esm2022/catalog/index.mjs +2 -1
- package/esm2022/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.mjs +126 -0
- package/esm2022/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +2 -2
- package/fesm2022/snabcentr-client-ui.mjs +151 -44
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +2 -2
- package/release_notes.tmp +2 -2
package/catalog/index.d.ts
CHANGED
@@ -1,6 +1,7 @@
|
|
1
1
|
export * from './download-price-list/sc-download-price-list.component';
|
2
2
|
export * from './category-card/sc-category-card.component';
|
3
3
|
export * from './input-quantity/sc-input-quantity.component';
|
4
|
+
export * from './notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component';
|
4
5
|
export * from './price-card/sc-price-card.component';
|
5
6
|
export * from './price-card-inline/sc-price-card-inline.component';
|
6
7
|
export * from './price-history/sc-price-history.component';
|
@@ -0,0 +1,68 @@
|
|
1
|
+
import { Signal } from '@angular/core';
|
2
|
+
import { FormControl, FormGroup } from '@angular/forms';
|
3
|
+
import { ScCatalogService, ScISuggestionType, ScIWarehouse, ScProduct } from '@snabcentr/client-core';
|
4
|
+
import { TuiContext, TuiStringHandler } from '@taiga-ui/cdk';
|
5
|
+
import { TuiDialogContext } from '@taiga-ui/core';
|
6
|
+
import { Subject } from 'rxjs';
|
7
|
+
import * as i0 from "@angular/core";
|
8
|
+
/**
|
9
|
+
* Компонент формы отправки запроса на уведомление о поступлении товара.
|
10
|
+
*/
|
11
|
+
export declare class ScNotifyWhenInStockDialogComponent {
|
12
|
+
/**
|
13
|
+
* Контекст диалогового окна, в котором открыт компонент.
|
14
|
+
*/
|
15
|
+
protected readonly context: TuiDialogContext<void, {
|
16
|
+
product: ScProduct;
|
17
|
+
onClickOfferHandler: () => void;
|
18
|
+
}>;
|
19
|
+
/**
|
20
|
+
* Товар.
|
21
|
+
*/
|
22
|
+
protected readonly product: ScProduct;
|
23
|
+
/**
|
24
|
+
* Форма для отправки запроса на уведомление о поступлении товара.
|
25
|
+
*/
|
26
|
+
protected readonly form: FormGroup<{
|
27
|
+
email: FormControl<string | null>;
|
28
|
+
phone: FormControl<string | null>;
|
29
|
+
verificationCode: FormControl<string | null>;
|
30
|
+
warehouseId: FormControl<string | null>;
|
31
|
+
}>;
|
32
|
+
/**
|
33
|
+
* Список складов.
|
34
|
+
*/
|
35
|
+
protected readonly warehouses: Signal<ScIWarehouse[] | null>;
|
36
|
+
/**
|
37
|
+
* Перечисление типов подсказок.
|
38
|
+
*/
|
39
|
+
protected readonly suggestionType: typeof ScISuggestionType;
|
40
|
+
/**
|
41
|
+
* Сервис для работы с каталогом.
|
42
|
+
*/
|
43
|
+
protected readonly catalogService: ScCatalogService;
|
44
|
+
/**
|
45
|
+
* {@link Subject} события отправки формы.
|
46
|
+
*/
|
47
|
+
protected readonly onSubmit$: Subject<void>;
|
48
|
+
/**
|
49
|
+
* {@link Observable} запроса данных уведомления о поступлении товара.
|
50
|
+
*/
|
51
|
+
private readonly request$;
|
52
|
+
/**
|
53
|
+
* Сигнал изменения состояния загрузки данных.
|
54
|
+
*/
|
55
|
+
protected readonly loading: Signal<boolean>;
|
56
|
+
/**
|
57
|
+
* Проверяет, является ли товар в наличии.
|
58
|
+
*/
|
59
|
+
protected isStock(): boolean;
|
60
|
+
/**
|
61
|
+
* Преобразует объект в значение, отображаемое в поле ввода.
|
62
|
+
*
|
63
|
+
* @param items Выбранные значения.
|
64
|
+
*/
|
65
|
+
protected stringify(items: readonly ScIWarehouse[]): TuiStringHandler<TuiContext<number>>;
|
66
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScNotifyWhenInStockDialogComponent, never>;
|
67
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScNotifyWhenInStockDialogComponent, "sc-notify-when-in-stock-dialog", never, {}, {}, never, never, true, never>;
|
68
|
+
}
|
@@ -1,6 +1,7 @@
|
|
1
1
|
export * from './download-price-list/sc-download-price-list.component';
|
2
2
|
export * from './category-card/sc-category-card.component';
|
3
3
|
export * from './input-quantity/sc-input-quantity.component';
|
4
|
+
export * from './notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component';
|
4
5
|
export * from './price-card/sc-price-card.component';
|
5
6
|
export * from './price-card-inline/sc-price-card-inline.component';
|
6
7
|
export * from './price-history/sc-price-history.component';
|
@@ -8,4 +9,4 @@ export * from './price-list-pagination/sc-price-list-pagination.component';
|
|
8
9
|
export * from './price-warehouse-stock/sc-price-warehouse-stock.component';
|
9
10
|
export * from './sc-favorite-button/sc-favorite-button.component';
|
10
11
|
export * from './sc-catalog.module';
|
11
|
-
//# sourceMappingURL=data:application/json;base64,
|
12
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHdEQUF3RCxDQUFDO0FBQ3ZFLGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyw4Q0FBOEMsQ0FBQztBQUM3RCxjQUFjLHdFQUF3RSxDQUFDO0FBQ3ZGLGNBQWMsc0NBQXNDLENBQUM7QUFDckQsY0FBYyxvREFBb0QsQ0FBQztBQUNuRSxjQUFjLDRDQUE0QyxDQUFDO0FBQzNELGNBQWMsNERBQTRELENBQUM7QUFDM0UsY0FBYyw0REFBNEQsQ0FBQztBQUMzRSxjQUFjLG1EQUFtRCxDQUFDO0FBQ2xFLGNBQWMscUJBQXFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2Rvd25sb2FkLXByaWNlLWxpc3Qvc2MtZG93bmxvYWQtcHJpY2UtbGlzdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jYXRlZ29yeS1jYXJkL3NjLWNhdGVnb3J5LWNhcmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vaW5wdXQtcXVhbnRpdHkvc2MtaW5wdXQtcXVhbnRpdHkuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbm90aWZ5LXdoZW4taW4tc3RvY2stZGlhbG9nL3NjLW5vdGlmeS13aGVuLWluLXN0b2NrLWRpYWxvZy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9wcmljZS1jYXJkL3NjLXByaWNlLWNhcmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2UtY2FyZC1pbmxpbmUvc2MtcHJpY2UtY2FyZC1pbmxpbmUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2UtaGlzdG9yeS9zYy1wcmljZS1oaXN0b3J5LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3ByaWNlLWxpc3QtcGFnaW5hdGlvbi9zYy1wcmljZS1saXN0LXBhZ2luYXRpb24uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2Utd2FyZWhvdXNlLXN0b2NrL3NjLXByaWNlLXdhcmVob3VzZS1zdG9jay5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1mYXZvcml0ZS1idXR0b24vc2MtZmF2b3JpdGUtYnV0dG9uLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWNhdGFsb2cubW9kdWxlJztcbiJdfQ==
|
package/esm2022/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.mjs
ADDED
@@ -0,0 +1,126 @@
|
|
1
|
+
/* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */
|
2
|
+
import { AsyncPipe, NgIf } from '@angular/common';
|
3
|
+
import { HttpErrorResponse } from '@angular/common/http';
|
4
|
+
import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
|
5
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
6
|
+
import { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';
|
7
|
+
import { ScCatalogService, ScISuggestionType, ScWarehouseService } from '@snabcentr/client-core';
|
8
|
+
import { TuiButton, TuiError, TuiLabel, TuiLink, TuiLoader } from '@taiga-ui/core';
|
9
|
+
import { TuiButtonLoading, TuiFieldErrorPipe } from '@taiga-ui/kit';
|
10
|
+
import { TuiInputModule, TuiSelectModule } from '@taiga-ui/legacy';
|
11
|
+
import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
|
12
|
+
import { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
|
13
|
+
import { ScFormFieldsModule } from '../../form-fields/form-fields.module';
|
14
|
+
import { ScVerificationModule } from '../../verification/sc-verification.module';
|
15
|
+
import * as i0 from "@angular/core";
|
16
|
+
import * as i1 from "../../verification/verification-phone-check-form/sc-verification-phone-check-form.component";
|
17
|
+
import * as i2 from "@angular/forms";
|
18
|
+
import * as i3 from "@taiga-ui/legacy";
|
19
|
+
import * as i4 from "../../form-fields/suggestion-field/sc-suggestion-field.component";
|
20
|
+
import * as i5 from "@taiga-ui/core/components/data-list";
|
21
|
+
/**
|
22
|
+
* Компонент формы отправки запроса на уведомление о поступлении товара.
|
23
|
+
*/
|
24
|
+
export class ScNotifyWhenInStockDialogComponent {
|
25
|
+
constructor() {
|
26
|
+
/**
|
27
|
+
* Контекст диалогового окна, в котором открыт компонент.
|
28
|
+
*/
|
29
|
+
this.context = inject(POLYMORPHEUS_CONTEXT);
|
30
|
+
/**
|
31
|
+
* Товар.
|
32
|
+
*/
|
33
|
+
this.product = this.context.data.product;
|
34
|
+
/**
|
35
|
+
* Форма для отправки запроса на уведомление о поступлении товара.
|
36
|
+
*/
|
37
|
+
this.form = new FormGroup({
|
38
|
+
email: new FormControl(null, [Validators.required, Validators.email]),
|
39
|
+
phone: new FormControl(null, [Validators.required, Validators.minLength(12)]),
|
40
|
+
verificationCode: new FormControl(null, [Validators.required, Validators.minLength(6)]),
|
41
|
+
warehouseId: new FormControl(null, Validators.required),
|
42
|
+
});
|
43
|
+
/**
|
44
|
+
* Список складов.
|
45
|
+
*/
|
46
|
+
this.warehouses = toSignal(inject(ScWarehouseService).getWarehouses$(), { initialValue: null });
|
47
|
+
/**
|
48
|
+
* Перечисление типов подсказок.
|
49
|
+
*/
|
50
|
+
this.suggestionType = ScISuggestionType;
|
51
|
+
/**
|
52
|
+
* Сервис для работы с каталогом.
|
53
|
+
*/
|
54
|
+
this.catalogService = inject(ScCatalogService);
|
55
|
+
/**
|
56
|
+
* {@link Subject} события отправки формы.
|
57
|
+
*/
|
58
|
+
this.onSubmit$ = new Subject();
|
59
|
+
/**
|
60
|
+
* {@link Observable} запроса данных уведомления о поступлении товара.
|
61
|
+
*/
|
62
|
+
this.request$ = this.onSubmit$.pipe(filter(() => this.form.valid), map(() => this.form.value), switchMap((value) => this.catalogService.getProductNotifyWhenInStock$(this.product.id, value).pipe(tap(() => {
|
63
|
+
this.context.completeWith();
|
64
|
+
}), catchError((error) => {
|
65
|
+
if (error instanceof HttpErrorResponse) {
|
66
|
+
const { errors, message } = error.error;
|
67
|
+
if (errors) {
|
68
|
+
// eslint-disable-next-line no-shadow
|
69
|
+
Object.entries(errors).forEach(([key, value]) => {
|
70
|
+
this.form.get(key)?.setErrors({ serverResponse: value });
|
71
|
+
});
|
72
|
+
}
|
73
|
+
else if (message) {
|
74
|
+
this.form.setErrors({ serverResponse: [message] });
|
75
|
+
}
|
76
|
+
}
|
77
|
+
return of(undefined);
|
78
|
+
}), startWith(null))), share());
|
79
|
+
/**
|
80
|
+
* Сигнал изменения состояния загрузки данных.
|
81
|
+
*/
|
82
|
+
this.loading = toSignal(this.request$.pipe(map((value) => value === null)), { initialValue: false });
|
83
|
+
}
|
84
|
+
/**
|
85
|
+
* Проверяет, является ли товар в наличии.
|
86
|
+
*/
|
87
|
+
isStock() {
|
88
|
+
const warehouseId = Number(this.form.get('warehouseId')?.value ?? 0);
|
89
|
+
return this.product.isWarehouseStockExist(warehouseId);
|
90
|
+
}
|
91
|
+
/**
|
92
|
+
* Преобразует объект в значение, отображаемое в поле ввода.
|
93
|
+
*
|
94
|
+
* @param items Выбранные значения.
|
95
|
+
*/
|
96
|
+
stringify(items) {
|
97
|
+
const itemsMap = new Map(items.map((item) => {
|
98
|
+
return [item.id, item.name];
|
99
|
+
}));
|
100
|
+
return ({ $implicit }) => itemsMap.get($implicit) ?? '';
|
101
|
+
}
|
102
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScNotifyWhenInStockDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
103
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ScNotifyWhenInStockDialogComponent, isStandalone: true, selector: "sc-notify-when-in-stock-dialog", ngImport: i0, template: "<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit$.next()\"\n class=\"flex flex-col gap-6\"\n>\n <div class=\"flex flex-col gap-3\">\n <sc-verification-phone-check-form [shouldBeConfirmed]=\"true\" />\n\n <label tuiLabel>\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"email\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <label tuiLabel>\n \u0421\u043A\u043B\u0430\u0434\n <tui-select\n formControlName=\"warehouseId\"\n [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n >\n \u0421\u043A\u043B\u0430\u0434\n <tui-data-list *tuiDataList>\n @for (warehouse of warehouses(); track warehouse.id) {\n <button\n tuiOption\n [value]=\"warehouse.id\"\n >\n {{ warehouse.name }}\n </button>\n }\n </tui-data-list>\n </tui-select>\n <tui-error\n formControlName=\"warehouseId\"\n [error]=\"[] | tuiFieldError | async\"\n />\n\n <span\n *ngIf=\"isStock()\"\n [style.color]=\"'var(--tui-text-positive)'\"\n >\u0422\u043E\u0432\u0430\u0440 \u0438\u043C\u0435\u0435\u0442\u0441\u044F \u0432 \u043D\u0430\u043B\u0438\u0447\u0438\u0438 \u043D\u0430 \u0432\u044B\u0431\u0440\u0430\u043D\u043D\u043E\u043C \u0441\u043A\u043B\u0430\u0434\u0435</span\n >\n </label>\n <ng-template #selectLoading> <tui-loader /> </ng-template>\n </div>\n\n <div class=\"flex flex-wrap items-center gap-3\">\n <button\n tuiButton\n type=\"submit\"\n [disabled]=\"form.invalid || isStock()\"\n [loading]=\"loading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n >\n \u041E\u0442\u043F\u0440\u0430\u0432\u0438\u0442\u044C\n </button>\n <div class=\"grow basis-60 text-body-s\">\n \u041D\u0430\u0436\u0438\u043C\u0430\u044F \u043D\u0430 \u043A\u043D\u043E\u043F\u043A\u0443, \u0432\u044B \u0441\u043E\u0433\u043B\u0430\u0448\u0430\u0435\u0442\u0435\u0441\u044C \u0441\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"context.data.onClickOfferHandler()\"\n >\n \u041F\u043E\u043B\u0438\u0442\u0438\u043A\u043E\u0439 \u043A\u043E\u043D\u0444\u0438\u0434\u0435\u043D\u0446\u0438\u0430\u043B\u044C\u043D\u043E\u0441\u0442\u0438\n </button>\n </div>\n </div>\n\n <tui-error\n *ngIf=\"form.errors && form.errors['serverResponse']\"\n [error]=\"[] | tuiFieldError | async\"\n />\n</form>\n", dependencies: [{ kind: "ngmodule", type: ScVerificationModule }, { kind: "component", type: i1.ScVerificationPhoneCheckFormComponent, selector: "sc-verification-phone-check-form", inputs: ["showCodeFields", "readOnly", "shouldBeBusy", "shouldBeConfirmed", "haveCode"], outputs: ["haveCodeChange", "isBusyChange", "isConfirmedChange"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: TuiInputModule }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: TuiLabel, selector: "label[tuiLabel]" }, { kind: "pipe", type: TuiFieldErrorPipe, name: "tuiFieldError" }, { kind: "component", type: TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "ngmodule", type: ScFormFieldsModule }, { kind: "component", type: i4.ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: ["type"], outputs: ["selectedClick"] }, { kind: "ngmodule", type: TuiSelectModule }, { kind: "component", type: i3.TuiSelectComponent, selector: "tui-select", inputs: ["stringify", "identityMatcher", "valueContent"] }, { kind: "directive", type: i3.TuiSelectDirective, selector: "tui-select" }, { kind: "component", type: i5.TuiDataListComponent, selector: "tui-data-list", inputs: ["emptyContent", "size"] }, { kind: "directive", type: i5.TuiDataListDirective, selector: "ng-template[tuiDataList]" }, { kind: "component", type: i5.TuiOption, selector: "button[tuiOption]:not([new]), a[tuiOption]:not([new]), label[tuiOption]:not([new])", inputs: ["disabled", "value"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "component", type: TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
104
|
+
}
|
105
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScNotifyWhenInStockDialogComponent, decorators: [{
|
106
|
+
type: Component,
|
107
|
+
args: [{ standalone: true, selector: 'sc-notify-when-in-stock-dialog', imports: [
|
108
|
+
ScVerificationModule,
|
109
|
+
FormsModule,
|
110
|
+
ReactiveFormsModule,
|
111
|
+
TuiInputModule,
|
112
|
+
TuiLink,
|
113
|
+
TuiButton,
|
114
|
+
TuiLabel,
|
115
|
+
TuiFieldErrorPipe,
|
116
|
+
TuiError,
|
117
|
+
ScVerificationModule,
|
118
|
+
ScFormFieldsModule,
|
119
|
+
TuiSelectModule,
|
120
|
+
TuiButtonLoading,
|
121
|
+
TuiLoader,
|
122
|
+
AsyncPipe,
|
123
|
+
NgIf,
|
124
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit$.next()\"\n class=\"flex flex-col gap-6\"\n>\n <div class=\"flex flex-col gap-3\">\n <sc-verification-phone-check-form [shouldBeConfirmed]=\"true\" />\n\n <label tuiLabel>\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"email\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <label tuiLabel>\n \u0421\u043A\u043B\u0430\u0434\n <tui-select\n formControlName=\"warehouseId\"\n [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n >\n \u0421\u043A\u043B\u0430\u0434\n <tui-data-list *tuiDataList>\n @for (warehouse of warehouses(); track warehouse.id) {\n <button\n tuiOption\n [value]=\"warehouse.id\"\n >\n {{ warehouse.name }}\n </button>\n }\n </tui-data-list>\n </tui-select>\n <tui-error\n formControlName=\"warehouseId\"\n [error]=\"[] | tuiFieldError | async\"\n />\n\n <span\n *ngIf=\"isStock()\"\n [style.color]=\"'var(--tui-text-positive)'\"\n >\u0422\u043E\u0432\u0430\u0440 \u0438\u043C\u0435\u0435\u0442\u0441\u044F \u0432 \u043D\u0430\u043B\u0438\u0447\u0438\u0438 \u043D\u0430 \u0432\u044B\u0431\u0440\u0430\u043D\u043D\u043E\u043C \u0441\u043A\u043B\u0430\u0434\u0435</span\n >\n </label>\n <ng-template #selectLoading> <tui-loader /> </ng-template>\n </div>\n\n <div class=\"flex flex-wrap items-center gap-3\">\n <button\n tuiButton\n type=\"submit\"\n [disabled]=\"form.invalid || isStock()\"\n [loading]=\"loading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n >\n \u041E\u0442\u043F\u0440\u0430\u0432\u0438\u0442\u044C\n </button>\n <div class=\"grow basis-60 text-body-s\">\n \u041D\u0430\u0436\u0438\u043C\u0430\u044F \u043D\u0430 \u043A\u043D\u043E\u043F\u043A\u0443, \u0432\u044B \u0441\u043E\u0433\u043B\u0430\u0448\u0430\u0435\u0442\u0435\u0441\u044C \u0441\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"context.data.onClickOfferHandler()\"\n >\n \u041F\u043E\u043B\u0438\u0442\u0438\u043A\u043E\u0439 \u043A\u043E\u043D\u0444\u0438\u0434\u0435\u043D\u0446\u0438\u0430\u043B\u044C\u043D\u043E\u0441\u0442\u0438\n </button>\n </div>\n </div>\n\n <tui-error\n *ngIf=\"form.errors && form.errors['serverResponse']\"\n [error]=\"[] | tuiFieldError | async\"\n />\n</form>\n" }]
|
125
|
+
}] });
|
126
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-notify-when-in-stock-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.ts","../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { ScCatalogService, ScIProductNotifyWhenInStock, ScISuggestionType, ScIWarehouse, ScProduct, ScWarehouseService } from '@snabcentr/client-core';\nimport { TuiContext, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiError, TuiLabel, TuiLink, TuiLoader } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFieldErrorPipe } from '@taiga-ui/kit';\nimport { TuiInputModule, TuiSelectModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { ScFormFieldsModule } from '../../form-fields/form-fields.module';\nimport { ScVerificationModule } from '../../verification/sc-verification.module';\n\n/**\n * Компонент формы отправки запроса на уведомление о поступлении товара.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-notify-when-in-stock-dialog',\n    templateUrl: './sc-notify-when-in-stock-dialog.component.html',\n    imports: [\n        ScVerificationModule,\n        FormsModule,\n        ReactiveFormsModule,\n        TuiInputModule,\n        TuiLink,\n        TuiButton,\n        TuiLabel,\n        TuiFieldErrorPipe,\n        TuiError,\n        ScVerificationModule,\n        ScFormFieldsModule,\n        TuiSelectModule,\n        TuiButtonLoading,\n        TuiLoader,\n        AsyncPipe,\n        NgIf,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScNotifyWhenInStockDialogComponent {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void }> =\n        inject<TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void }>>(POLYMORPHEUS_CONTEXT);\n\n    /**\n     * Товар.\n     */\n    protected readonly product: ScProduct = this.context.data.product;\n\n    /**\n     * Форма для отправки запроса на уведомление о поступлении товара.\n     */\n    protected readonly form = new FormGroup({\n        email: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        phone: new FormControl<string | null>(null, [Validators.required, Validators.minLength(12)]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n        warehouseId: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * Список складов.\n     */\n    protected readonly warehouses: Signal<ScIWarehouse[] | null> = toSignal(inject(ScWarehouseService).getWarehouses$(), { initialValue: null });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Сервис для работы с каталогом.\n     */\n    protected readonly catalogService: ScCatalogService = inject(ScCatalogService);\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit$: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных уведомления о поступлении товара.\n     */\n    private readonly request$ = this.onSubmit$.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIProductNotifyWhenInStock),\n        switchMap((value) =>\n            this.catalogService.getProductNotifyWhenInStock$(this.product.id, value).pipe(\n                tap(() => {\n                    this.context.completeWith();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        if (errors) {\n                            // eslint-disable-next-line no-shadow\n                            Object.entries(errors).forEach(([key, value]) => {\n                                this.form.get(key)?.setErrors({ serverResponse: value });\n                            });\n                        } else if (message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of(undefined);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * Сигнал изменения состояния загрузки данных.\n     */\n    protected readonly loading: Signal<boolean> = toSignal(this.request$.pipe(map((value) => value === null)), { initialValue: false });\n\n    /**\n     * Проверяет, является ли товар в наличии.\n     */\n    protected isStock(): boolean {\n        const warehouseId = Number(this.form.get('warehouseId')?.value ?? 0);\n\n        return this.product.isWarehouseStockExist(warehouseId);\n    }\n\n    /**\n     * Преобразует объект в значение, отображаемое в поле ввода.\n     *\n     * @param items Выбранные значения.\n     */\n    protected stringify(items: readonly ScIWarehouse[]): TuiStringHandler<TuiContext<number>> {\n        const itemsMap = new Map(\n            items.map((item) => {\n                return [item.id, item.name] as [number, string];\n            })\n        );\n\n        return ({ $implicit }: TuiContext<number>) => itemsMap.get($implicit) ?? '';\n    }\n}\n","<form\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit$.next()\"\n    class=\"flex flex-col gap-6\"\n>\n    <div class=\"flex flex-col gap-3\">\n        <sc-verification-phone-check-form [shouldBeConfirmed]=\"true\" />\n\n        <label tuiLabel>\n            Адрес электронной почты\n            <tui-input formControlName=\"email\">\n                Адрес электронной почты\n                <sc-suggestion-field\n                    *tuiDataList\n                    [type]=\"suggestionType.email\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"email\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        <label tuiLabel>\n            Склад\n            <tui-select\n                formControlName=\"warehouseId\"\n                [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n            >\n                Склад\n                <tui-data-list *tuiDataList>\n                    @for (warehouse of warehouses(); track warehouse.id) {\n                        <button\n                            tuiOption\n                            [value]=\"warehouse.id\"\n                        >\n                            {{ warehouse.name }}\n                        </button>\n                    }\n                </tui-data-list>\n            </tui-select>\n            <tui-error\n                formControlName=\"warehouseId\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n\n            <span\n                *ngIf=\"isStock()\"\n                [style.color]=\"'var(--tui-text-positive)'\"\n                >Товар имеется в наличии на выбранном складе</span\n            >\n        </label>\n        <ng-template #selectLoading> <tui-loader /> </ng-template>\n    </div>\n\n    <div class=\"flex flex-wrap items-center gap-3\">\n        <button\n            tuiButton\n            type=\"submit\"\n            [disabled]=\"form.invalid || isStock()\"\n            [loading]=\"loading()\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n        >\n            Отправить\n        </button>\n        <div class=\"grow basis-60 text-body-s\">\n            Нажимая на кнопку, вы соглашаетесь с\n            <button\n                tuiLink\n                [pseudo]=\"true\"\n                type=\"button\"\n                (click)=\"context.data.onClickOfferHandler()\"\n            >\n                Политикой конфиденциальности\n            </button>\n        </div>\n    </div>\n\n    <tui-error\n        *ngIf=\"form.errors && form.errors['serverResponse']\"\n        [error]=\"[] | tuiFieldError | async\"\n    />\n</form>\n"]}
|
@@ -173,7 +173,7 @@ export class ScVerificationPhoneCheckFormComponent {
|
|
173
173
|
}), switchMap((value) => {
|
174
174
|
if (this.phoneControl.valid && !!value) {
|
175
175
|
return this.verificationService.getPhoneCheck$(value).pipe(map((result) => {
|
176
|
-
if (this.shouldBeBusy() === undefined) {
|
176
|
+
if (this.shouldBeBusy() === undefined && this.shouldBeConfirmed() !== undefined) {
|
177
177
|
this.isBusyChange.emit(result.isBusy);
|
178
178
|
if (!isNil(result.isConfirmed)) {
|
179
179
|
this.isConfirmedChange.emit(result.isConfirmed);
|
@@ -242,4 +242,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
242
242
|
}], propDecorators: { showCodeFields: [{
|
243
243
|
type: Input
|
244
244
|
}] } });
|
245
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-verification-phone-check-form.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/verification/verification-phone-check-form/sc-verification-phone-check-form.component.ts","../../../../../projects/client-ui/verification/verification-phone-check-form/sc-verification-phone-check-form.component.html"],"names":[],"mappings":"AAAA,gEAAgE;AAEhE,OAAO,EAAE,UAAU,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACrE,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,EAAe,KAAK,EAAuB,MAAM,EAA4B,MAAM,eAAe,CAAC;AACpK,OAAO,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AACtD,OAAO,EAA0B,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AAE5E,OAAO,EAAiB,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAC9E,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAEhD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,KAAK,EAAE,MAAM,WAAW,CAAC;AAClC,OAAO,EACH,UAAU,EACV,oBAAoB,EACpB,OAAO,EACP,MAAM,EACN,QAAQ,EACR,GAAG,EAEH,EAAE,EACF,IAAI,EACJ,KAAK,EACL,WAAW,EACX,SAAS,EACT,OAAO,EACP,SAAS,EACT,SAAS,EACT,GAAG,EACH,KAAK,GACR,MAAM,MAAM,CAAC;AAGd,OAAO,EAAE,wCAAwC,EAAE,MAAM,sEAAsE,CAAC;AAChI,OAAO,EAAE,2BAA2B,EAAE,MAAM,aAAa,CAAC;AAC1D,OAAO,EAAE,4BAA4B,EAAE,MAAM,8CAA8C,CAAC;AAC5F,OAAO,EAAE,yBAAyB,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;;;;;;AAE3D;;GAEG;AAOH,MAAM,OAAO,qCAAqC;IANlD;QAuBI;;WAEG;QACgB,wBAAmB,GAAG,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAEvE;;WAEG;QACgB,uBAAkB,GAAG,MAAM,CAAC,kBAAkB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QAEvF;;WAEG;QACgB,SAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAE7C;;WAEG;QACa,aAAQ,GAAyB,KAAK,CAAC,KAAK,CAAC,CAAC;QAE9D;;WAEG;QACa,iBAAY,GAAqC,KAAK,EAAuB,CAAC;QAE9F;;WAEG;QACa,sBAAiB,GAAqC,KAAK,EAAuB,CAAC;QAEnG;;WAEG;QACgB,cAAS,GAAiC,YAAY,EAAE,CAAC;QAE5E;;WAEG;QACgB,4BAAuB,GAA8B,MAAM,CAAC,yBAAyB,CAAC,CAAC;QAE1G;;WAEG;QACgB,iBAAY,GAAwC,QAAQ,CAAC,IAAI,CAAC,uBAAuB,CAAC,gBAAgB,EAAE,CAAC,CAAC;QAEjI;;WAEG;QACa,aAAQ,GAAyB,KAAK,CAAC,KAAK,CAAC,CAAC;QAO9D;;WAEG;QACa,iBAAY,GAAqC,MAAM,EAAkB,CAAC;QAE1F;;WAEG;QACa,sBAAiB,GAAqC,MAAM,EAAkB,CAAC;QAE/F;;WAEG;QACgB,eAAU,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAEnE;;WAEG;QACgB,gBAAW,GAAW,MAAM,CAAC,4BAA4B,CAAC,CAAC;QAE9E;;WAEG;QACgB,wBAAmB,GAAwB,IAAI,CAAC,UAAU,CAAC,IAAI,CAC9E,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,EAClC,MAAM,CAAC,CAAC,KAAK,EAAmB,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,EAC3D,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAChB,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC,IAAI,CACrD,GAAG,CAAC,GAAG,EAAE;YACL,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC7C,CAAC,CAAC,EACF,GAAG,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,EAChB,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,aAAa,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAEtD,MAAM,KAAK,GAAG,eAAe,CAAC;gBAC9B,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;gBAEhD,IAAI,KAAK,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAC5B,MAAM,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;oBACtC,MAAM,OAAO,GAAG,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBAElD,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACpC,CAAC;qBAAM,CAAC;oBACJ,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,aAAa,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBAChH,CAAC;YACL,CAAC;YAED,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE;YACV,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAC3B,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,EACf,WAAW,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC,EAAE,CAAC,CACjD,CACJ,EACD,SAAS,CAAC,KAAK,CAAC,EAChB,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACgB,yBAAoB,GAAmB,2BAA2B,CAAC;QAEtF;;WAEG;QACgB,iBAAY,GAAoB,IAAI,OAAO,EAAU,CAAC;QAEzE;;;;WAIG;QACgB,WAAM,GAA8B,IAAI,CAAC,YAAY,CAAC,IAAI,CACzE,SAAS,CAAC,CAAC,GAAG,EAAE,EAAE,CACd,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,IAAI,CACf,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,KAAK,EAAE,GAAG,CAAC,EAChC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,IAAI,CAAC,CAAC,EAChC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;YACV,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,EAAE,CAAC,CAAC;YACvC,MAAM,OAAO,GAAG,KAAK,GAAG,EAAE,CAAC;YAE3B,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;QACnH,CAAC,CAAC,EACF,OAAO,CAAC,IAAI,CAAC,EACb,SAAS,CAAC,IAAI,CAAC,EACf,oBAAoB,EAAE,CACzB,CACJ,CACJ,CAAC;QAEF;;WAEG;QACgB,yBAAoB,GAAG,MAAM,CAAC,wCAAwC,CAAC,CAAC;QAE3F;;WAEG;QACK,oBAAe,GAAY,IAAI,CAAC;KAuH3C;IArSG;;OAEG;IACH,IACW,cAAc,CAAC,cAAuB;QAC7C,IAAI,CAAC,eAAe,GAAG,cAAc,CAAC;QACtC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,EAAE,CAAC;IACjF,CAAC;IAED;;OAEG;IACH,IAAW,cAAc;QACrB,OAAO,IAAI,CAAC,eAAe,CAAC;IAChC,CAAC;IAkKD;;OAEG;IACH,IAAW,IAAI;QAIX,OAAO,IAAI,CAAC,kBAAkB,EAAE,IAG9B,CAAC;IACP,CAAC;IAED;;OAEG;IACH,IAAW,YAAY;QACnB,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC;IACpC,CAAC;IAED,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,WAAW,GAAG,eAAe,CAAgB,IAAI,CAAC,YAAY,CAAC,CAAC,IAAI,CACrE,GAAG,CAAC,GAAG,EAAE;YACL,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC1B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC7B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClC,IAAI,CAAC,uBAAuB,CAAC,wBAAwB,EAAE,CAAC;YAExD,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,CAAC;gBAC5C,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YACvD,CAAC;QACL,CAAC,CAAC,EACF,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YAChB,IAAI,IAAI,CAAC,YAAY,CAAC,KAAK,IAAI,CAAC,CAAC,KAAK,EAAE,CAAC;gBACrC,OAAO,IAAI,CAAC,mBAAmB,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC,IAAI,CACtD,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE;oBACX,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,SAAS,EAAE,CAAC;wBACpC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;wBAEtC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;4BAC7B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;4BAEhD,OAAO,IAAI,CAAC,iBAAiB,EAAE,KAAK,MAAM,CAAC,WAAW,CAAC;wBAC3D,CAAC;wBAED,OAAO,IAAI,CAAC;oBAChB,CAAC;oBAED,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,SAAS,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,MAAM,CAAC,MAAM,EAAE,CAAC;wBAC7E,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC;4BACxB,cAAc,EAAE;gCACZ,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,sDAAsD,CAAC,CAAC,CAAC,qDAAqD;6BACjI;yBACJ,CAAC,CAAC;wBACH,IAAI,CAAC,YAAY,CAAC,aAAa,EAAE,CAAC;wBAElC,OAAO,KAAK,CAAC;oBACjB,CAAC;oBAED,IAAI,IAAI,CAAC,iBAAiB,EAAE,KAAK,SAAS,IAAI,IAAI,CAAC,iBAAiB,EAAE,KAAK,MAAM,CAAC,WAAW,EAAE,CAAC;wBAC5F,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC;4BACxB,cAAc,EAAE;gCACZ,MAAM,CAAC,WAAW;oCACd,CAAC,CAAC,yBAAyB;oCAC3B,CAAC,CAAC,sJAAsJ;6BAC/J;yBACJ,CAAC,CAAC;wBACH,IAAI,CAAC,YAAY,CAAC,aAAa,EAAE,CAAC;wBAElC,OAAO,KAAK,CAAC;oBACjB,CAAC;oBAED,OAAO,IAAI,CAAC;gBAChB,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;oBAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;wBACrC,MAAM,aAAa,GAAG,KAAK,CAAC,KAAyB,CAAC;wBACtD,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,aAAa,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;oBAChH,CAAC;oBAED,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC;gBACrB,CAAC,CAAC,EACF,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;oBACV,IAAI,CAAC,KAAK,EAAE,CAAC;wBACT,IAAI,CAAC,uBAAuB,CAAC,uBAAuB,CAAC,KAAK,CAAC,CAAC;oBAChE,CAAC;gBACL,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CAAC;YACN,CAAC;YAED,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAExB,OAAO,EAAE,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC,CAAC,EACF,KAAK,EAAE,CACV,CAAC;QAEF,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;IAC5B,CAAC;IAED;;;;OAIG;IACO,WAAW,CAAC,QAAiB;QACnC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAChC,CAAC;IAED;;OAEG;IACO,iBAAiB;QACvB,IAAI,CAAC,uBAAuB,CAAC,wBAAwB,EAAE,CAAC;IAC5D,CAAC;+GArSQ,qCAAqC;mGAArC,qCAAqC,u7BC/ClD,m2IAgHA;;4FDjEa,qCAAqC;kBANjD,SAAS;+BACI,kCAAkC,QAEtC,EAAE,eAAe,EAAE,MAAM,EAAE,mBAChB,uBAAuB,CAAC,MAAM;8BAOpC,cAAc;sBADxB,KAAK","sourcesContent":["/* eslint-disable no-underscore-dangle,lodash/prefer-constant */\n\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Input, input, InputSignal, model, ModelSignal, OnInit, output, OutputEmitterRef, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormGroupDirective } from '@angular/forms';\nimport { MaskitoOptions } from '@maskito/core';\nimport { ScIOrgContact, ScVerificationService } from '@snabcentr/client-core';\nimport { tuiControlValue } from '@taiga-ui/cdk';\nimport type { TuiCountryIsoCode } from '@taiga-ui/i18n';\nimport { getCountries } from 'libphonenumber-js';\nimport { isNil } from 'lodash-es';\nimport {\n    catchError,\n    distinctUntilChanged,\n    endWith,\n    filter,\n    finalize,\n    map,\n    Observable,\n    of,\n    scan,\n    share,\n    shareReplay,\n    startWith,\n    Subject,\n    switchMap,\n    takeWhile,\n    tap,\n    timer,\n} from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT } from '../../contacts/tokens/sc-show-help-notification-in-phone-input.token';\nimport { scPhoneVerificationCodeMask } from '../../masks';\nimport { SC_VERIFICATION_CODE_TIMEOUT } from '../../providers/sc-verification-code-timeout';\nimport { ScHelpNotificationService } from '../../services';\n\n/**\n * Компонент формы проверки телефона и получения кода подтверждения.\n */\n@Component({\n    selector: 'sc-verification-phone-check-form',\n    templateUrl: './sc-verification-phone-check-form.component.html',\n    host: { ngSkipHydration: 'true' },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScVerificationPhoneCheckFormComponent implements OnInit {\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    @Input()\n    public set showCodeFields(showCodeFields: boolean) {\n        this._showCodeFields = showCodeFields;\n        this.form.controls.verificationCode[showCodeFields ? 'enable' : 'disable']();\n    }\n\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    public get showCodeFields(): boolean {\n        return this._showCodeFields;\n    }\n\n    /**\n     * Сервис верификации.\n     */\n    protected readonly verificationService = inject(ScVerificationService);\n\n    /**\n     * Директива c `FormGroup` из DOM.\n     */\n    protected readonly formGroupDirective = inject(FormGroupDirective, { optional: true });\n\n    /**\n     * HTTP-клиент.\n     */\n    protected readonly http = inject(HttpClient);\n\n    /**\n     * Признак, что поле ввода телефона только для чтения.\n     */\n    public readonly readOnly: InputSignal<boolean> = input(false);\n\n    /**\n     * Признак, следует ли телефону быть в системе. От этого признака зависит, в каких случаях выдавать ошибку при проверке занятости телефона.\n     */\n    public readonly shouldBeBusy: InputSignal<boolean | undefined> = input<boolean | undefined>();\n\n    /**\n     * Признак, следует ли проверять, подтверждён ли телефон в системе. При `true` будет вызывать ошибку при наличии телефона в системе без подтверждения.\n     */\n    public readonly shouldBeConfirmed: InputSignal<boolean | undefined> = input<boolean | undefined>();\n\n    /**\n     * Список стран.\n     */\n    protected readonly countries: readonly TuiCountryIsoCode[] = getCountries();\n\n    /**\n     * Сервис для отображения Push-уведомлений с контактами для помощи клиенту.\n     */\n    protected readonly helpNotificationService: ScHelpNotificationService = inject(ScHelpNotificationService);\n\n    /**\n     * Список контактов для отправки push-уведомлений.\n     */\n    protected readonly pushContacts: Signal<ScIOrgContact[] | undefined> = toSignal(this.helpNotificationService.getContactsShow$());\n\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public readonly haveCode: ModelSignal<boolean> = model(false);\n\n    /**\n     * {@link Observable} запроса проверки номера телефона.\n     */\n    public phoneCheck$?: Observable<boolean | null>;\n\n    /**\n     * {@link OutputEmitter} события изменения признака занятости телефона.\n     */\n    public readonly isBusyChange: OutputEmitterRef<boolean | null> = output<boolean | null>();\n\n    /**\n     * {@link OutputEmitter} события изменения признака подтверждения телефона.\n     */\n    public readonly isConfirmedChange: OutputEmitterRef<boolean | null> = output<boolean | null>();\n\n    /**\n     * {@link Subject} события отправки кода подтверждения.\n     */\n    protected readonly onSendCode: Subject<void> = new Subject<void>();\n\n    /**\n     * Таймаут отправки кода подтверждения.\n     */\n    protected readonly codeTimeout: number = inject(SC_VERIFICATION_CODE_TIMEOUT);\n\n    /**\n     * {@link Observable} запроса данных получения кода подтверждения.\n     */\n    protected readonly loadingApproveCode$: Observable<boolean> = this.onSendCode.pipe(\n        map(() => this.phoneControl.value),\n        filter((value): value is string => this.phoneControl.valid),\n        switchMap((value) =>\n            this.verificationService.sendPhoneApproveCode(value).pipe(\n                tap(() => {\n                    this.reloadTimer$.next(this.codeTimeout);\n                }),\n                map(() => false),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const errorResponse = error.error as ApiErrorResponse;\n\n                        const regex = /(\\d{2}):\\d{2}/;\n                        const match = regex.exec(errorResponse.message);\n\n                        if (match && match.length > 1) {\n                            const timeParts = match[0].split(':');\n                            const seconds = Number.parseInt(timeParts[1], 10);\n\n                            this.reloadTimer$.next(seconds);\n                        } else {\n                            this.phoneControl.setErrors({ serverResponse: errorResponse.errors?.['phone'] ?? [errorResponse.message] });\n                        }\n                    }\n\n                    return of(false);\n                }),\n                finalize(() => {\n                    this.setHaveCode(true);\n                }),\n                startWith(true),\n                shareReplay({ refCount: true, bufferSize: 1 })\n            )\n        ),\n        startWith(false),\n        share()\n    );\n\n    /**\n     * Маска поля ввода кода для подтверждения.\n     */\n    protected readonly verificationCodeMask: MaskitoOptions = scPhoneVerificationCodeMask;\n\n    /**\n     * {@link Subject} события запуска/остановки таймера.\n     */\n    protected readonly reloadTimer$: Subject<number> = new Subject<number>();\n\n    /**\n     * {@link Observable} Таймера.\n     *\n     * TODO: Вынести таймер в отдельную директиву TASK:[#9260].\n     */\n    protected readonly timer$: Observable<string | null> = this.reloadTimer$.pipe(\n        switchMap((sec) =>\n            timer(0, 1000).pipe(\n                scan((total) => -1 + total, sec),\n                takeWhile((total) => total >= 0),\n                map((total) => {\n                    const minutes = Math.floor(total / 60);\n                    const seconds = total % 60;\n\n                    return `${Math.round(minutes).toString().padStart(2, '0')}:${Math.round(seconds).toString().padStart(2, '0')}`;\n                }),\n                endWith(null),\n                startWith(null),\n                distinctUntilChanged()\n            )\n        )\n    );\n\n    /**\n     * Токен показа уведомления помощи.\n     */\n    protected readonly showHelpNotification = inject(SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT);\n\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    private _showCodeFields: boolean = true;\n\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public get form(): FormGroup<{\n        phone: FormControl<string | null>;\n        verificationCode: FormControl<string | null>;\n    }> {\n        return this.formGroupDirective?.form as FormGroup<{\n            phone: FormControl<string | null>;\n            verificationCode: FormControl<string | null>;\n        }>;\n    }\n\n    /**\n     * Поле ввода 'Номер телефона'.\n     */\n    public get phoneControl(): FormControl<string | null> {\n        return this.form.controls.phone;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.phoneCheck$ = tuiControlValue<string | null>(this.phoneControl).pipe(\n            tap(() => {\n                this.reloadTimer$.next(0);\n                this.isBusyChange.emit(null);\n                this.isConfirmedChange.emit(null);\n                this.helpNotificationService.closeAllHelpNotification();\n\n                if (this.form.controls.verificationCode.value) {\n                    this.form.controls.verificationCode.setValue(null);\n                }\n            }),\n            switchMap((value) => {\n                if (this.phoneControl.valid && !!value) {\n                    return this.verificationService.getPhoneCheck$(value).pipe(\n                        map((result) => {\n                            if (this.shouldBeBusy() === undefined) {\n                                this.isBusyChange.emit(result.isBusy);\n\n                                if (!isNil(result.isConfirmed)) {\n                                    this.isConfirmedChange.emit(result.isConfirmed);\n\n                                    return this.shouldBeConfirmed() === result.isConfirmed;\n                                }\n\n                                return true;\n                            }\n\n                            if (this.shouldBeBusy() !== undefined && this.shouldBeBusy() !== result.isBusy) {\n                                this.phoneControl.setErrors({\n                                    serverResponse: [\n                                        result.isBusy ? 'Пользователь с таким номером телефона уже существует' : 'Пользователя с таким номером телефона не существует',\n                                    ],\n                                });\n                                this.phoneControl.markAsTouched();\n\n                                return false;\n                            }\n\n                            if (this.shouldBeConfirmed() !== undefined && this.shouldBeConfirmed() !== result.isConfirmed) {\n                                this.phoneControl.setErrors({\n                                    serverResponse: [\n                                        result.isConfirmed\n                                            ? 'Телефон уже подтверждён'\n                                            : 'Номер телефона не подтверждён или вход по нему невозможен. Обратитесь к вашему персональному менеджеру или войдите с использованием e-mail и пароля.',\n                                    ],\n                                });\n                                this.phoneControl.markAsTouched();\n\n                                return false;\n                            }\n\n                            return true;\n                        }),\n                        catchError((error: unknown) => {\n                            if (error instanceof HttpErrorResponse) {\n                                const errorResponse = error.error as ApiErrorResponse;\n                                this.phoneControl.setErrors({ serverResponse: errorResponse.errors?.['phone'] ?? [errorResponse.message] });\n                            }\n\n                            return of(false);\n                        }),\n                        tap((check) => {\n                            if (!check) {\n                                this.helpNotificationService.helpNotificationByPhone(value);\n                            }\n                        }),\n                        startWith(null)\n                    );\n                }\n\n                this.setHaveCode(false);\n\n                return of(false);\n            }),\n            share()\n        );\n\n        this.setHaveCode(false);\n    }\n\n    /**\n     * Устанавливает состояние наличия кода подтверждения у пользователя.\n     *\n     * @param haveCode Признак того есть ли код подтверждения или нет.\n     */\n    protected setHaveCode(haveCode: boolean): void {\n        this.haveCode.set(haveCode);\n    }\n\n    /**\n     * Закрытие уведомления.\n     */\n    protected closeNotification(): void {\n        this.helpNotificationService.closeAllHelpNotification();\n    }\n}\n","@let phoneCheck = phoneCheck$ | async;\n\n<form\n    [formGroup]=\"form\"\n    *tuiLet=\"loadingApproveCode$ | async as loadingApproveCode\"\n    class=\"flex flex-col gap-3\"\n>\n    <label tuiLabel>\n        Номер телефона\n        <tui-textfield>\n            <input\n                formControlName=\"phone\"\n                tuiInputPhoneInternational\n                [readOnly]=\"readOnly()\"\n                [countrySearch]=\"true\"\n                [countries]=\"(countries | tuiSortCountries | async) || []\"\n                autocomplete=\"phone\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"phone\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        @if (showHelpNotification && pushContacts()) {\n            @for (manager of pushContacts(); track manager.id) {\n                <tui-push\n                    type=\"Свяжитесь с нами для решения возникших вопросов\"\n                    (close)=\"closeNotification()\"\n                    class=\"!w-auto\"\n                >\n                    <tui-icon icon=\"@tui.message-square\" />\n                    <sc-manager-card [manager]=\"manager\" />\n                </tui-push>\n            }\n        }\n    </label>\n\n    <ng-container *ngIf=\"showCodeFields && form.enabled && phoneCheck\">\n        <label\n            *ngIf=\"!loadingApproveCode && haveCode()\"\n            tuiLabel\n        >\n            Код из СМС\n            <tui-input formControlName=\"verificationCode\">\n                Код из СМС\n                <input\n                    tuiTextfieldLegacy\n                    [maskito]=\"verificationCodeMask\"\n                    autocomplete=\"new-password\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"verificationCode\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        <div\n            *tuiLet=\"!!loadingApproveCode as loadingApproveCode\"\n            class=\"flex justify-center gap-4\"\n        >\n            <button\n                *ngIf=\"!haveCode()\"\n                tuiButton\n                (click)=\"onSendCode.next()\"\n                [disabled]=\"loadingApproveCode || !phoneCheck || phoneControl.invalid\"\n                [loading]=\"loadingApproveCode\"\n                iconStart=\"@tui.sc.send\"\n            >\n                Получить код\n            </button>\n            <button\n                *ngIf=\"!loadingApproveCode && !haveCode()\"\n                tuiLink\n                [pseudo]=\"true\"\n                [disabled]=\"!phoneCheck || phoneControl.invalid\"\n                (click)=\"setHaveCode(true)\"\n            >\n                У меня есть код\n            </button>\n\n            <ng-container *tuiLet=\"timer$ | async as timer\">\n                <tui-loader\n                    *ngIf=\"haveCode()\"\n                    [showLoader]=\"loadingApproveCode\"\n                >\n                    <button\n                        tuiLink\n                        [pseudo]=\"true\"\n                        [disabled]=\"loadingApproveCode || timer\"\n                        (click)=\"onSendCode.next()\"\n                    >\n                        Повторно отправить код\n                        <ng-container\n                            *ngIf=\"timer\"\n                            class=\"!text-tui-base-08\"\n                        >\n                            (через {{ timer }})\n                        </ng-container>\n                    </button>\n                </tui-loader>\n            </ng-container>\n        </div>\n    </ng-container>\n</form>\n\n<ng-template #checkingPhone>\n    <tui-loader\n        *ngIf=\"!phoneCheck && phoneControl.valid\"\n        class=\"size-4\"\n    />\n</ng-template>\n"]}
|
245
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-verification-phone-check-form.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/verification/verification-phone-check-form/sc-verification-phone-check-form.component.ts","../../../../../projects/client-ui/verification/verification-phone-check-form/sc-verification-phone-check-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-underscore-dangle,lodash/prefer-constant */\n\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Input, input, InputSignal, model, ModelSignal, OnInit, output, OutputEmitterRef, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormGroupDirective } from '@angular/forms';\nimport { MaskitoOptions } from '@maskito/core';\nimport { ScIOrgContact, ScVerificationService } from '@snabcentr/client-core';\nimport { tuiControlValue } from '@taiga-ui/cdk';\nimport type { TuiCountryIsoCode } from '@taiga-ui/i18n';\nimport { getCountries } from 'libphonenumber-js';\nimport { isNil } from 'lodash-es';\nimport {\n    catchError,\n    distinctUntilChanged,\n    endWith,\n    filter,\n    finalize,\n    map,\n    Observable,\n    of,\n    scan,\n    share,\n    shareReplay,\n    startWith,\n    Subject,\n    switchMap,\n    takeWhile,\n    tap,\n    timer,\n} from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT } from '../../contacts/tokens/sc-show-help-notification-in-phone-input.token';\nimport { scPhoneVerificationCodeMask } from '../../masks';\nimport { SC_VERIFICATION_CODE_TIMEOUT } from '../../providers/sc-verification-code-timeout';\nimport { ScHelpNotificationService } from '../../services';\n\n/**\n * Компонент формы проверки телефона и получения кода подтверждения.\n */\n@Component({\n    selector: 'sc-verification-phone-check-form',\n    templateUrl: './sc-verification-phone-check-form.component.html',\n    host: { ngSkipHydration: 'true' },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScVerificationPhoneCheckFormComponent implements OnInit {\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    @Input()\n    public set showCodeFields(showCodeFields: boolean) {\n        this._showCodeFields = showCodeFields;\n        this.form.controls.verificationCode[showCodeFields ? 'enable' : 'disable']();\n    }\n\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    public get showCodeFields(): boolean {\n        return this._showCodeFields;\n    }\n\n    /**\n     * Сервис верификации.\n     */\n    protected readonly verificationService = inject(ScVerificationService);\n\n    /**\n     * Директива c `FormGroup` из DOM.\n     */\n    protected readonly formGroupDirective = inject(FormGroupDirective, { optional: true });\n\n    /**\n     * HTTP-клиент.\n     */\n    protected readonly http = inject(HttpClient);\n\n    /**\n     * Признак, что поле ввода телефона только для чтения.\n     */\n    public readonly readOnly: InputSignal<boolean> = input(false);\n\n    /**\n     * Признак, следует ли телефону быть в системе. От этого признака зависит, в каких случаях выдавать ошибку при проверке занятости телефона.\n     */\n    public readonly shouldBeBusy: InputSignal<boolean | undefined> = input<boolean | undefined>();\n\n    /**\n     * Признак, следует ли проверять, подтверждён ли телефон в системе. При `true` будет вызывать ошибку при наличии телефона в системе без подтверждения.\n     */\n    public readonly shouldBeConfirmed: InputSignal<boolean | undefined> = input<boolean | undefined>();\n\n    /**\n     * Список стран.\n     */\n    protected readonly countries: readonly TuiCountryIsoCode[] = getCountries();\n\n    /**\n     * Сервис для отображения Push-уведомлений с контактами для помощи клиенту.\n     */\n    protected readonly helpNotificationService: ScHelpNotificationService = inject(ScHelpNotificationService);\n\n    /**\n     * Список контактов для отправки push-уведомлений.\n     */\n    protected readonly pushContacts: Signal<ScIOrgContact[] | undefined> = toSignal(this.helpNotificationService.getContactsShow$());\n\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public readonly haveCode: ModelSignal<boolean> = model(false);\n\n    /**\n     * {@link Observable} запроса проверки номера телефона.\n     */\n    public phoneCheck$?: Observable<boolean | null>;\n\n    /**\n     * {@link OutputEmitter} события изменения признака занятости телефона.\n     */\n    public readonly isBusyChange: OutputEmitterRef<boolean | null> = output<boolean | null>();\n\n    /**\n     * {@link OutputEmitter} события изменения признака подтверждения телефона.\n     */\n    public readonly isConfirmedChange: OutputEmitterRef<boolean | null> = output<boolean | null>();\n\n    /**\n     * {@link Subject} события отправки кода подтверждения.\n     */\n    protected readonly onSendCode: Subject<void> = new Subject<void>();\n\n    /**\n     * Таймаут отправки кода подтверждения.\n     */\n    protected readonly codeTimeout: number = inject(SC_VERIFICATION_CODE_TIMEOUT);\n\n    /**\n     * {@link Observable} запроса данных получения кода подтверждения.\n     */\n    protected readonly loadingApproveCode$: Observable<boolean> = this.onSendCode.pipe(\n        map(() => this.phoneControl.value),\n        filter((value): value is string => this.phoneControl.valid),\n        switchMap((value) =>\n            this.verificationService.sendPhoneApproveCode(value).pipe(\n                tap(() => {\n                    this.reloadTimer$.next(this.codeTimeout);\n                }),\n                map(() => false),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const errorResponse = error.error as ApiErrorResponse;\n\n                        const regex = /(\\d{2}):\\d{2}/;\n                        const match = regex.exec(errorResponse.message);\n\n                        if (match && match.length > 1) {\n                            const timeParts = match[0].split(':');\n                            const seconds = Number.parseInt(timeParts[1], 10);\n\n                            this.reloadTimer$.next(seconds);\n                        } else {\n                            this.phoneControl.setErrors({ serverResponse: errorResponse.errors?.['phone'] ?? [errorResponse.message] });\n                        }\n                    }\n\n                    return of(false);\n                }),\n                finalize(() => {\n                    this.setHaveCode(true);\n                }),\n                startWith(true),\n                shareReplay({ refCount: true, bufferSize: 1 })\n            )\n        ),\n        startWith(false),\n        share()\n    );\n\n    /**\n     * Маска поля ввода кода для подтверждения.\n     */\n    protected readonly verificationCodeMask: MaskitoOptions = scPhoneVerificationCodeMask;\n\n    /**\n     * {@link Subject} события запуска/остановки таймера.\n     */\n    protected readonly reloadTimer$: Subject<number> = new Subject<number>();\n\n    /**\n     * {@link Observable} Таймера.\n     *\n     * TODO: Вынести таймер в отдельную директиву TASK:[#9260].\n     */\n    protected readonly timer$: Observable<string | null> = this.reloadTimer$.pipe(\n        switchMap((sec) =>\n            timer(0, 1000).pipe(\n                scan((total) => -1 + total, sec),\n                takeWhile((total) => total >= 0),\n                map((total) => {\n                    const minutes = Math.floor(total / 60);\n                    const seconds = total % 60;\n\n                    return `${Math.round(minutes).toString().padStart(2, '0')}:${Math.round(seconds).toString().padStart(2, '0')}`;\n                }),\n                endWith(null),\n                startWith(null),\n                distinctUntilChanged()\n            )\n        )\n    );\n\n    /**\n     * Токен показа уведомления помощи.\n     */\n    protected readonly showHelpNotification = inject(SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT);\n\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    private _showCodeFields: boolean = true;\n\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public get form(): FormGroup<{\n        phone: FormControl<string | null>;\n        verificationCode: FormControl<string | null>;\n    }> {\n        return this.formGroupDirective?.form as FormGroup<{\n            phone: FormControl<string | null>;\n            verificationCode: FormControl<string | null>;\n        }>;\n    }\n\n    /**\n     * Поле ввода 'Номер телефона'.\n     */\n    public get phoneControl(): FormControl<string | null> {\n        return this.form.controls.phone;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.phoneCheck$ = tuiControlValue<string | null>(this.phoneControl).pipe(\n            tap(() => {\n                this.reloadTimer$.next(0);\n                this.isBusyChange.emit(null);\n                this.isConfirmedChange.emit(null);\n                this.helpNotificationService.closeAllHelpNotification();\n\n                if (this.form.controls.verificationCode.value) {\n                    this.form.controls.verificationCode.setValue(null);\n                }\n            }),\n            switchMap((value) => {\n                if (this.phoneControl.valid && !!value) {\n                    return this.verificationService.getPhoneCheck$(value).pipe(\n                        map((result) => {\n                            if (this.shouldBeBusy() === undefined && this.shouldBeConfirmed() !== undefined) {\n                                this.isBusyChange.emit(result.isBusy);\n\n                                if (!isNil(result.isConfirmed)) {\n                                    this.isConfirmedChange.emit(result.isConfirmed);\n\n                                    return this.shouldBeConfirmed() === result.isConfirmed;\n                                }\n\n                                return true;\n                            }\n\n                            if (this.shouldBeBusy() !== undefined && this.shouldBeBusy() !== result.isBusy) {\n                                this.phoneControl.setErrors({\n                                    serverResponse: [\n                                        result.isBusy ? 'Пользователь с таким номером телефона уже существует' : 'Пользователя с таким номером телефона не существует',\n                                    ],\n                                });\n                                this.phoneControl.markAsTouched();\n\n                                return false;\n                            }\n\n                            if (this.shouldBeConfirmed() !== undefined && this.shouldBeConfirmed() !== result.isConfirmed) {\n                                this.phoneControl.setErrors({\n                                    serverResponse: [\n                                        result.isConfirmed\n                                            ? 'Телефон уже подтверждён'\n                                            : 'Номер телефона не подтверждён или вход по нему невозможен. Обратитесь к вашему персональному менеджеру или войдите с использованием e-mail и пароля.',\n                                    ],\n                                });\n                                this.phoneControl.markAsTouched();\n\n                                return false;\n                            }\n\n                            return true;\n                        }),\n                        catchError((error: unknown) => {\n                            if (error instanceof HttpErrorResponse) {\n                                const errorResponse = error.error as ApiErrorResponse;\n                                this.phoneControl.setErrors({ serverResponse: errorResponse.errors?.['phone'] ?? [errorResponse.message] });\n                            }\n\n                            return of(false);\n                        }),\n                        tap((check) => {\n                            if (!check) {\n                                this.helpNotificationService.helpNotificationByPhone(value);\n                            }\n                        }),\n                        startWith(null)\n                    );\n                }\n\n                this.setHaveCode(false);\n\n                return of(false);\n            }),\n            share()\n        );\n\n        this.setHaveCode(false);\n    }\n\n    /**\n     * Устанавливает состояние наличия кода подтверждения у пользователя.\n     *\n     * @param haveCode Признак того есть ли код подтверждения или нет.\n     */\n    protected setHaveCode(haveCode: boolean): void {\n        this.haveCode.set(haveCode);\n    }\n\n    /**\n     * Закрытие уведомления.\n     */\n    protected closeNotification(): void {\n        this.helpNotificationService.closeAllHelpNotification();\n    }\n}\n","@let phoneCheck = phoneCheck$ | async;\n\n<form\n    [formGroup]=\"form\"\n    *tuiLet=\"loadingApproveCode$ | async as loadingApproveCode\"\n    class=\"flex flex-col gap-3\"\n>\n    <label tuiLabel>\n        Номер телефона\n        <tui-textfield>\n            <input\n                formControlName=\"phone\"\n                tuiInputPhoneInternational\n                [readOnly]=\"readOnly()\"\n                [countrySearch]=\"true\"\n                [countries]=\"(countries | tuiSortCountries | async) || []\"\n                autocomplete=\"phone\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"phone\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        @if (showHelpNotification && pushContacts()) {\n            @for (manager of pushContacts(); track manager.id) {\n                <tui-push\n                    type=\"Свяжитесь с нами для решения возникших вопросов\"\n                    (close)=\"closeNotification()\"\n                    class=\"!w-auto\"\n                >\n                    <tui-icon icon=\"@tui.message-square\" />\n                    <sc-manager-card [manager]=\"manager\" />\n                </tui-push>\n            }\n        }\n    </label>\n\n    <ng-container *ngIf=\"showCodeFields && form.enabled && phoneCheck\">\n        <label\n            *ngIf=\"!loadingApproveCode && haveCode()\"\n            tuiLabel\n        >\n            Код из СМС\n            <tui-input formControlName=\"verificationCode\">\n                Код из СМС\n                <input\n                    tuiTextfieldLegacy\n                    [maskito]=\"verificationCodeMask\"\n                    autocomplete=\"new-password\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"verificationCode\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        <div\n            *tuiLet=\"!!loadingApproveCode as loadingApproveCode\"\n            class=\"flex justify-center gap-4\"\n        >\n            <button\n                *ngIf=\"!haveCode()\"\n                tuiButton\n                (click)=\"onSendCode.next()\"\n                [disabled]=\"loadingApproveCode || !phoneCheck || phoneControl.invalid\"\n                [loading]=\"loadingApproveCode\"\n                iconStart=\"@tui.sc.send\"\n            >\n                Получить код\n            </button>\n            <button\n                *ngIf=\"!loadingApproveCode && !haveCode()\"\n                tuiLink\n                [pseudo]=\"true\"\n                [disabled]=\"!phoneCheck || phoneControl.invalid\"\n                (click)=\"setHaveCode(true)\"\n            >\n                У меня есть код\n            </button>\n\n            <ng-container *tuiLet=\"timer$ | async as timer\">\n                <tui-loader\n                    *ngIf=\"haveCode()\"\n                    [showLoader]=\"loadingApproveCode\"\n                >\n                    <button\n                        tuiLink\n                        [pseudo]=\"true\"\n                        [disabled]=\"loadingApproveCode || timer\"\n                        (click)=\"onSendCode.next()\"\n                    >\n                        Повторно отправить код\n                        <ng-container\n                            *ngIf=\"timer\"\n                            class=\"!text-tui-base-08\"\n                        >\n                            (через {{ timer }})\n                        </ng-container>\n                    </button>\n                </tui-loader>\n            </ng-container>\n        </div>\n    </ng-container>\n</form>\n\n<ng-template #checkingPhone>\n    <tui-loader\n        *ngIf=\"!phoneCheck && phoneControl.valid\"\n        class=\"size-4\"\n    />\n</ng-template>\n"]}
|