@snabcentr/client-ui 3.36.0 → 3.37.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/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +26 -12
- package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +23 -9
- package/esm2022/verification/index.mjs +2 -1
- package/esm2022/verification/providers/index.mjs +3 -0
- package/esm2022/verification/providers/sc-phone-approve-code-sender.provider.mjs +12 -0
- package/esm2022/verification/providers/sc-phone-approve-code-sender.token.mjs +7 -0
- package/esm2022/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +27 -9
- package/fesm2022/snabcentr-client-ui.mjs +91 -30
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +2 -2
- package/release_notes.tmp +2 -2
- package/verification/index.d.ts +1 -0
- package/verification/providers/index.d.ts +2 -0
- package/verification/providers/sc-phone-approve-code-sender.provider.d.ts +6 -0
- package/verification/providers/sc-phone-approve-code-sender.token.d.ts +7 -0
- package/verification/verification-phone-check-form/sc-verification-phone-check-form.component.d.ts +4 -0
package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs
CHANGED
@@ -27,7 +27,7 @@ export class ScSignInFormByEmailComponent {
|
|
27
27
|
* Группа полей ввода для формы «Вход на сайт».
|
28
28
|
*/
|
29
29
|
this.formByEmail = new FormGroup({
|
30
|
-
|
30
|
+
username: new FormControl(null, [Validators.required, Validators.email]),
|
31
31
|
password: new FormControl(null, Validators.required),
|
32
32
|
});
|
33
33
|
/**
|
@@ -37,17 +37,31 @@ export class ScSignInFormByEmailComponent {
|
|
37
37
|
/**
|
38
38
|
* {@link Observable} запроса данных аутентификации.
|
39
39
|
*/
|
40
|
-
this.emailRequest$ = this.onSubmit.pipe(filter(() => this.formByEmail.valid), map(() => this.formByEmail.value), switchMap((value) => this.authService.getSignIn$(value).pipe(
|
40
|
+
this.emailRequest$ = this.onSubmit.pipe(filter(() => this.formByEmail.valid), map(() => this.formByEmail.value), switchMap((value) => this.authService.getSignIn$(value).pipe(
|
41
|
+
// eslint-disable-next-line sonarjs/cognitive-complexity
|
42
|
+
catchError((error) => {
|
41
43
|
if (error instanceof HttpErrorResponse) {
|
42
|
-
|
43
|
-
|
44
|
-
if (
|
45
|
-
|
46
|
-
|
44
|
+
if ('error' in error.error) {
|
45
|
+
const errorResponse = error.error;
|
46
|
+
if (errorResponse.error.includes('invalid_credentials')) {
|
47
|
+
this.formByEmail.setErrors({ serverResponse: ['Пользователь с указанными учетными данными не найден. Проверьте правильность ввода.'] });
|
48
|
+
}
|
49
|
+
else {
|
50
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
51
|
+
this.formByEmail.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });
|
47
52
|
}
|
48
53
|
}
|
49
|
-
|
50
|
-
|
54
|
+
else {
|
55
|
+
const { errors, message } = error.error;
|
56
|
+
for (const key in errors) {
|
57
|
+
if (Object.hasOwn(errors, key)) {
|
58
|
+
// eslint-disable-next-line security/detect-object-injection
|
59
|
+
this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });
|
60
|
+
}
|
61
|
+
}
|
62
|
+
if (!errors && message) {
|
63
|
+
this.formByEmail.setErrors({ serverResponse: [message] });
|
64
|
+
}
|
51
65
|
}
|
52
66
|
}
|
53
67
|
return of({});
|
@@ -62,12 +76,12 @@ export class ScSignInFormByEmailComponent {
|
|
62
76
|
this.forgotPassword = new EventEmitter();
|
63
77
|
}
|
64
78
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScSignInFormByEmailComponent, deps: [{ token: i1.ScAuthService }], target: i0.ɵɵFactoryTarget.Component }); }
|
65
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScSignInFormByEmailComponent, selector: "sc-sign-in-form-by-email", outputs: { forgotPassword: "forgotPassword" }, ngImport: i0, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\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=\"
|
79
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScSignInFormByEmailComponent, selector: "sc-sign-in-form-by-email", outputs: { forgotPassword: "forgotPassword" }, ngImport: i0, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\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=\"username\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <input\n tuiTextfieldLegacy\n autocomplete=\"email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"username\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <label tuiLabel\n >\u041F\u0430\u0440\u043E\u043B\u044C\n <tui-input-password formControlName=\"password\">\n \u041F\u0430\u0440\u043E\u043B\u044C\n <input\n tuiTextfieldLegacy\n autocomplete=\"current-password\"\n />\n </tui-input-password>\n <tui-error\n formControlName=\"password\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n </div>\n <div class=\"mb-4 flex flex-col items-center gap-4\">\n <a\n tuiLink\n [pseudo]=\"true\"\n (click)=\"forgotPassword.emit()\"\n class=\"text-base\"\n >\u0417\u0430\u0431\u044B\u043B\u0438 \u043F\u0430\u0440\u043E\u043B\u044C?</a\n >\n <button\n tuiButton\n type=\"submit\"\n [loading]=\"!!(loadingEmailAuth$ | async)\"\n [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n </div>\n</form>\n", dependencies: [{ 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: "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: "component", type: i3.TuiInputPasswordComponent, selector: "tui-input-password" }, { kind: "directive", type: i3.TuiInputPasswordDirective, selector: "tui-input-password" }, { kind: "component", type: i4.TuiTextfieldComponent, selector: "input[tuiTextfieldLegacy], textarea[tuiTextfieldLegacy]" }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "directive", type: i5.TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: i5.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i5.TuiLabel, selector: "label[tuiLabel]" }, { kind: "component", type: i5.TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "component", type: i6.TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: i7.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.TuiFieldErrorPipe, name: "tuiFieldError" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
66
80
|
}
|
67
81
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScSignInFormByEmailComponent, decorators: [{
|
68
82
|
type: Component,
|
69
|
-
args: [{ selector: 'sc-sign-in-form-by-email', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\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=\"
|
83
|
+
args: [{ selector: 'sc-sign-in-form-by-email', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\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=\"username\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <input\n tuiTextfieldLegacy\n autocomplete=\"email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"username\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <label tuiLabel\n >\u041F\u0430\u0440\u043E\u043B\u044C\n <tui-input-password formControlName=\"password\">\n \u041F\u0430\u0440\u043E\u043B\u044C\n <input\n tuiTextfieldLegacy\n autocomplete=\"current-password\"\n />\n </tui-input-password>\n <tui-error\n formControlName=\"password\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n </div>\n <div class=\"mb-4 flex flex-col items-center gap-4\">\n <a\n tuiLink\n [pseudo]=\"true\"\n (click)=\"forgotPassword.emit()\"\n class=\"text-base\"\n >\u0417\u0430\u0431\u044B\u043B\u0438 \u043F\u0430\u0440\u043E\u043B\u044C?</a\n >\n <button\n tuiButton\n type=\"submit\"\n [loading]=\"!!(loadingEmailAuth$ | async)\"\n [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n </div>\n</form>\n" }]
|
70
84
|
}], ctorParameters: () => [{ type: i1.ScAuthService }], propDecorators: { forgotPassword: [{
|
71
85
|
type: Output
|
72
86
|
}] } });
|
73
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-email.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.html"],"names":[],"mappings":"AAAA,2EAA2E;AAE3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACzF,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEpE,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;;;;;;;;;AAIrG;;GAEG;AAMH,MAAM,OAAO,4BAA4B;IAwDrC;;;;OAIG;IACH,YAAoC,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QA5D9D;;WAEG;QACa,gBAAW,GAAc,IAAI,SAAS,CAAC;YACnD,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC;YACpF,QAAQ,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC;SACtE,CAAC,CAAC;QAEH;;WAEG;QACa,aAAQ,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAE9D;;WAEG;QACc,kBAAa,GAAkC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAC9E,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EACpC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAe,CAAC,EAC3C,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAChB,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CACnC,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAE5D,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;oBACvB,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,CAAC;wBAC7B,4DAA4D;wBAC5D,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,EAAE,cAAc,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;oBAC1E,CAAC;gBACL,CAAC;gBAED,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE,CAAC;oBACrB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBAC9D,CAAC;YACL,CAAC;YAED,OAAO,EAAE,CAAC,EAAgB,CAAC,CAAC;QAChC,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CACJ,EACD,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACa,sBAAiB,GAAwB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;QAElG;;WAEG;QACuB,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAOvB,CAAC;+GA7DzD,4BAA4B;mGAA5B,4BAA4B,+GCnBzC,qjEAsDA;;4FDnCa,4BAA4B;kBALxC,SAAS;+BACI,0BAA0B,mBAEnB,uBAAuB,CAAC,MAAM;kFAwDrB,cAAc;sBAAvC,MAAM","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Output } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, ILogin, ScAuthService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по адресу электронной почты и паролю.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-email',\n    templateUrl: './sc-sign-in-form-by-email.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByEmailComponent {\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly formByEmail: FormGroup = new FormGroup({\n        login: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        password: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly emailRequest$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        filter(() => this.formByEmail.valid),\n        map(() => this.formByEmail.value as ILogin),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.formByEmail.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по email.\n     */\n    public readonly loadingEmailAuth$: Observable<boolean> = this.emailRequest$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Сигнал нажатия на кнопку \"Забыли пароль\".\n     */\n    @Output() public readonly forgotPassword: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByEmailComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"formByEmail\"\n    (ngSubmit)=\"onSubmit.next()\"\n>\n    <div class=\"mb-8 flex flex-col gap-4\">\n        <label tuiLabel\n            >Адрес электронной почты\n            <tui-input formControlName=\"login\">\n                Адрес электронной почты\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"email\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"login\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <label tuiLabel\n            >Пароль\n            <tui-input-password formControlName=\"password\">\n                Пароль\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"current-password\"\n                />\n            </tui-input-password>\n            <tui-error\n                formControlName=\"password\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </div>\n    <div class=\"mb-4 flex flex-col items-center gap-4\">\n        <a\n            tuiLink\n            [pseudo]=\"true\"\n            (click)=\"forgotPassword.emit()\"\n            class=\"text-base\"\n            >Забыли пароль?</a\n        >\n        <button\n            tuiButton\n            type=\"submit\"\n            [loading]=\"!!(loadingEmailAuth$ | async)\"\n            [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n            iconStart=\"@tui.sc.circle-arrow-in-right\"\n        >\n            Войти\n        </button>\n    </div>\n</form>\n"]}
|
87
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-email.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Output } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, ILogin, ScAuthService, ScIKeycloakErrorResponse } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по адресу электронной почты и паролю.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-email',\n    templateUrl: './sc-sign-in-form-by-email.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByEmailComponent {\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly formByEmail: FormGroup = new FormGroup({\n        username: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        password: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly emailRequest$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        filter(() => this.formByEmail.valid),\n        map(() => this.formByEmail.value as ILogin),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                // eslint-disable-next-line sonarjs/cognitive-complexity\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        if ('error' in error.error) {\n                            const errorResponse = error.error as ScIKeycloakErrorResponse;\n\n                            if (errorResponse.error.includes('invalid_credentials')) {\n                                this.formByEmail.setErrors({ serverResponse: ['Пользователь с указанными учетными данными не найден. Проверьте правильность ввода.'] });\n                            } else {\n                                // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                                this.formByEmail.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });\n                            }\n                        } else {\n                            const { errors, message } = error.error as ApiErrorResponse;\n\n                            for (const key in errors) {\n                                if (Object.hasOwn(errors, key)) {\n                                    // eslint-disable-next-line security/detect-object-injection\n                                    this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });\n                                }\n                            }\n\n                            if (!errors && message) {\n                                this.formByEmail.setErrors({ serverResponse: [message] });\n                            }\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по email.\n     */\n    public readonly loadingEmailAuth$: Observable<boolean> = this.emailRequest$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Сигнал нажатия на кнопку \"Забыли пароль\".\n     */\n    @Output() public readonly forgotPassword: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByEmailComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"formByEmail\"\n    (ngSubmit)=\"onSubmit.next()\"\n>\n    <div class=\"mb-8 flex flex-col gap-4\">\n        <label tuiLabel\n            >Адрес электронной почты\n            <tui-input formControlName=\"username\">\n                Адрес электронной почты\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"email\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"username\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <label tuiLabel\n            >Пароль\n            <tui-input-password formControlName=\"password\">\n                Пароль\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"current-password\"\n                />\n            </tui-input-password>\n            <tui-error\n                formControlName=\"password\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </div>\n    <div class=\"mb-4 flex flex-col items-center gap-4\">\n        <a\n            tuiLink\n            [pseudo]=\"true\"\n            (click)=\"forgotPassword.emit()\"\n            class=\"text-base\"\n            >Забыли пароль?</a\n        >\n        <button\n            tuiButton\n            type=\"submit\"\n            [loading]=\"!!(loadingEmailAuth$ | async)\"\n            [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n            iconStart=\"@tui.sc.circle-arrow-in-right\"\n        >\n            Войти\n        </button>\n    </div>\n</form>\n"]}
|
package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs
CHANGED
@@ -42,17 +42,31 @@ export class ScSignInFormByPhoneComponent {
|
|
42
42
|
/**
|
43
43
|
* {@link Observable} запроса данных аутентификации.
|
44
44
|
*/
|
45
|
-
this.request$ = this.onSubmit.pipe(map(() => this.form.value), filter((value) => this.form.valid), switchMap((value) => this.authService.getSignIn$(value).pipe(
|
45
|
+
this.request$ = this.onSubmit.pipe(map(() => this.form.value), filter((value) => this.form.valid), switchMap((value) => this.authService.getSignIn$(value).pipe(
|
46
|
+
// eslint-disable-next-line sonarjs/cognitive-complexity
|
47
|
+
catchError((error) => {
|
46
48
|
if (error instanceof HttpErrorResponse) {
|
47
|
-
|
48
|
-
|
49
|
-
if (
|
50
|
-
|
51
|
-
|
49
|
+
if ('error' in error.error) {
|
50
|
+
const errorResponse = error.error;
|
51
|
+
if (errorResponse.error.includes('invalid_credentials')) {
|
52
|
+
this.form.setErrors({ serverResponse: ['Неверный код подтверждения.'] });
|
53
|
+
}
|
54
|
+
else {
|
55
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
56
|
+
this.form.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });
|
52
57
|
}
|
53
58
|
}
|
54
|
-
|
55
|
-
|
59
|
+
else {
|
60
|
+
const { errors, message } = error.error;
|
61
|
+
for (const key in errors) {
|
62
|
+
if (Object.hasOwn(errors, key)) {
|
63
|
+
// eslint-disable-next-line security/detect-object-injection
|
64
|
+
this.form.get(key)?.setErrors({ serverResponse: errors[key] });
|
65
|
+
}
|
66
|
+
}
|
67
|
+
if (!errors && message) {
|
68
|
+
this.form.setErrors({ serverResponse: [message] });
|
69
|
+
}
|
56
70
|
}
|
57
71
|
}
|
58
72
|
return of({});
|
@@ -69,4 +83,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
69
83
|
type: Component,
|
70
84
|
args: [{ selector: 'sc-sign-in-form-by-phone', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"form\"\n *tuiLet=\"!!(loading$ | async) as loading\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"mb-4 flex flex-col items-center gap-4\"\n>\n <sc-verification-phone-check-form\n [(haveCode)]=\"haveCode\"\n [shouldBeBusy]=\"true\"\n [shouldBeConfirmed]=\"true\"\n class=\"w-full\"\n />\n <tui-error\n [error]=\"[] | tuiFieldError | async\"\n class=\"self-center\"\n />\n <button\n *ngIf=\"haveCode\"\n tuiButton\n type=\"submit\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid || loading\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n</form>\n" }]
|
71
85
|
}], ctorParameters: () => [{ type: i1.ScAuthService }] });
|
72
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-phone.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.html"],"names":[],"mappings":"AAAA,2EAA2E;AAE3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEpE,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AAErG,OAAO,EAAE,cAAc,EAAE,MAAM,wCAAwC,CAAC;;;;;;;;;AAGxE;;GAEG;AAMH,MAAM,OAAO,4BAA4B;IAwDrC;;;;OAIG;IACH,YAAoC,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QA5D9D;;WAEG;QACI,aAAQ,GAAY,KAAK,CAAC;QAEjC;;WAEG;QACa,SAAI,GAAG,IAAI,SAAS,CAAC;YACjC,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,cAAc,EAAE,CAAC,CAAC;YACpF,gBAAgB,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;SACzG,CAAC,CAAC;QAEH;;WAEG;QACa,aAAQ,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAE9D;;WAEG;QACc,aAAQ,GAAkC,IAAI,CAAC,QAAQ,CAAC,IAAI,CACzE,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAC1B,MAAM,CAAC,CAAC,KAAK,EAAwB,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EACxD,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAChB,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CACnC,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAE5D,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;oBACvB,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,CAAC;wBAC7B,4DAA4D;wBAC5D,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,EAAE,cAAc,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;oBACnE,CAAC;gBACL,CAAC;gBAED,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE,CAAC;oBACrB,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACvD,CAAC;YACL,CAAC;YAED,OAAO,EAAE,CAAC,EAAgB,CAAC,CAAC;QAChC,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CACJ,EACD,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACa,aAAQ,GAAwB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;IAOnB,CAAC;+GA7DzD,4BAA4B;mGAA5B,4BAA4B,gECpBzC,0uBA2BA;;4FDPa,4BAA4B;kBALxC,SAAS;+BACI,0BAA0B,mBAEnB,uBAAuB,CAAC,MAAM","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, IPhoneLogin, ScAuthService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { phoneValidator } from '../../../validators/sc-phone-validator';\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по номеру телефона и коду подтверждения.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-phone',\n    templateUrl: './sc-sign-in-form-by-phone.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByPhoneComponent {\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public haveCode: boolean = false;\n\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly form = new FormGroup({\n        phone: new FormControl<string | null>(null, [Validators.required, phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly request$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is IPhoneLogin => this.form.valid),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByPhoneComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"form\"\n    *tuiLet=\"!!(loading$ | async) as loading\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"mb-4 flex flex-col items-center gap-4\"\n>\n    <sc-verification-phone-check-form\n        [(haveCode)]=\"haveCode\"\n        [shouldBeBusy]=\"true\"\n        [shouldBeConfirmed]=\"true\"\n        class=\"w-full\"\n    />\n    <tui-error\n        [error]=\"[] | tuiFieldError | async\"\n        class=\"self-center\"\n    />\n    <button\n        *ngIf=\"haveCode\"\n        tuiButton\n        type=\"submit\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid || loading\"\n        iconStart=\"@tui.sc.circle-arrow-in-right\"\n    >\n        Войти\n    </button>\n</form>\n"]}
|
86
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-phone.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, IPhoneLogin, ScAuthService, ScIKeycloakErrorResponse } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { phoneValidator } from '../../../validators/sc-phone-validator';\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по номеру телефона и коду подтверждения.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-phone',\n    templateUrl: './sc-sign-in-form-by-phone.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByPhoneComponent {\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public haveCode: boolean = false;\n\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly form = new FormGroup({\n        phone: new FormControl<string | null>(null, [Validators.required, phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly request$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is IPhoneLogin => this.form.valid),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                // eslint-disable-next-line sonarjs/cognitive-complexity\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        if ('error' in error.error) {\n                            const errorResponse = error.error as ScIKeycloakErrorResponse;\n\n                            if (errorResponse.error.includes('invalid_credentials')) {\n                                this.form.setErrors({ serverResponse: ['Неверный код подтверждения.'] });\n                            } else {\n                                // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                                this.form.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });\n                            }\n                        } else {\n                            const { errors, message } = error.error as ApiErrorResponse;\n\n                            for (const key in errors) {\n                                if (Object.hasOwn(errors, key)) {\n                                    // eslint-disable-next-line security/detect-object-injection\n                                    this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                                }\n                            }\n\n                            if (!errors && message) {\n                                this.form.setErrors({ serverResponse: [message] });\n                            }\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByPhoneComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"form\"\n    *tuiLet=\"!!(loading$ | async) as loading\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"mb-4 flex flex-col items-center gap-4\"\n>\n    <sc-verification-phone-check-form\n        [(haveCode)]=\"haveCode\"\n        [shouldBeBusy]=\"true\"\n        [shouldBeConfirmed]=\"true\"\n        class=\"w-full\"\n    />\n    <tui-error\n        [error]=\"[] | tuiFieldError | async\"\n        class=\"self-center\"\n    />\n    <button\n        *ngIf=\"haveCode\"\n        tuiButton\n        type=\"submit\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid || loading\"\n        iconStart=\"@tui.sc.circle-arrow-in-right\"\n    >\n        Войти\n    </button>\n</form>\n"]}
|
@@ -1,3 +1,4 @@
|
|
1
1
|
export * from './verification-phone-check-form/sc-verification-phone-check-form.component';
|
2
2
|
export * from './sc-verification.module';
|
3
|
-
|
3
|
+
export * from './providers';
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmVyaWZpY2F0aW9uL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNEVBQTRFLENBQUM7QUFDM0YsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdmVyaWZpY2F0aW9uLXBob25lLWNoZWNrLWZvcm0vc2MtdmVyaWZpY2F0aW9uLXBob25lLWNoZWNrLWZvcm0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtdmVyaWZpY2F0aW9uLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL3Byb3ZpZGVycyc7XG4iXX0=
|
@@ -0,0 +1,3 @@
|
|
1
|
+
export * from './sc-phone-approve-code-sender.token';
|
2
|
+
export * from './sc-phone-approve-code-sender.provider';
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmVyaWZpY2F0aW9uL3Byb3ZpZGVycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMseUNBQXlDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3NjLXBob25lLWFwcHJvdmUtY29kZS1zZW5kZXIudG9rZW4nO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1waG9uZS1hcHByb3ZlLWNvZGUtc2VuZGVyLnByb3ZpZGVyJztcbiJdfQ==
|
@@ -0,0 +1,12 @@
|
|
1
|
+
import { ScVerificationService } from '@snabcentr/client-core';
|
2
|
+
import { SC_PHONE_APPROVE_CODE_SENDER } from './sc-phone-approve-code-sender.token';
|
3
|
+
/**
|
4
|
+
* Провайдер по умолчанию для получения кода подтверждения телефона.
|
5
|
+
* Использует ScVerificationService для отправки кода.
|
6
|
+
*/
|
7
|
+
export const SC_PHONE_APPROVE_CODE_SENDER_PROVIDER = {
|
8
|
+
provide: SC_PHONE_APPROVE_CODE_SENDER,
|
9
|
+
useFactory: (verificationService) => (phoneNumber) => verificationService.sendPhoneApproveCode(phoneNumber),
|
10
|
+
deps: [ScVerificationService],
|
11
|
+
};
|
12
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcGhvbmUtYXBwcm92ZS1jb2RlLXNlbmRlci5wcm92aWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS92ZXJpZmljYXRpb24vcHJvdmlkZXJzL3NjLXBob25lLWFwcHJvdmUtY29kZS1zZW5kZXIucHJvdmlkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFFL0QsT0FBTyxFQUFFLDRCQUE0QixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFFcEY7OztHQUdHO0FBQ0gsTUFBTSxDQUFDLE1BQU0scUNBQXFDLEdBQWE7SUFDM0QsT0FBTyxFQUFFLDRCQUE0QjtJQUNyQyxVQUFVLEVBQUUsQ0FBQyxtQkFBMEMsRUFBRSxFQUFFLENBQUMsQ0FBQyxXQUFtQixFQUFFLEVBQUUsQ0FBQyxtQkFBbUIsQ0FBQyxvQkFBb0IsQ0FBQyxXQUFXLENBQUM7SUFDMUksSUFBSSxFQUFFLENBQUMscUJBQXFCLENBQUM7Q0FDaEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFByb3ZpZGVyIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTY1ZlcmlmaWNhdGlvblNlcnZpY2UgfSBmcm9tICdAc25hYmNlbnRyL2NsaWVudC1jb3JlJztcblxuaW1wb3J0IHsgU0NfUEhPTkVfQVBQUk9WRV9DT0RFX1NFTkRFUiB9IGZyb20gJy4vc2MtcGhvbmUtYXBwcm92ZS1jb2RlLXNlbmRlci50b2tlbic7XG5cbi8qKlxuICog0J/RgNC+0LLQsNC50LTQtdGAINC/0L4g0YPQvNC+0LvRh9Cw0L3QuNGOINC00LvRjyDQv9C+0LvRg9GH0LXQvdC40Y8g0LrQvtC00LAg0L/QvtC00YLQstC10YDQttC00LXQvdC40Y8g0YLQtdC70LXRhNC+0L3QsC5cbiAqINCY0YHQv9C+0LvRjNC30YPQtdGCIFNjVmVyaWZpY2F0aW9uU2VydmljZSDQtNC70Y8g0L7RgtC/0YDQsNCy0LrQuCDQutC+0LTQsC5cbiAqL1xuZXhwb3J0IGNvbnN0IFNDX1BIT05FX0FQUFJPVkVfQ09ERV9TRU5ERVJfUFJPVklERVI6IFByb3ZpZGVyID0ge1xuICAgIHByb3ZpZGU6IFNDX1BIT05FX0FQUFJPVkVfQ09ERV9TRU5ERVIsXG4gICAgdXNlRmFjdG9yeTogKHZlcmlmaWNhdGlvblNlcnZpY2U6IFNjVmVyaWZpY2F0aW9uU2VydmljZSkgPT4gKHBob25lTnVtYmVyOiBzdHJpbmcpID0+IHZlcmlmaWNhdGlvblNlcnZpY2Uuc2VuZFBob25lQXBwcm92ZUNvZGUocGhvbmVOdW1iZXIpLFxuICAgIGRlcHM6IFtTY1ZlcmlmaWNhdGlvblNlcnZpY2VdLFxufTtcbiJdfQ==
|
@@ -0,0 +1,7 @@
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
2
|
+
/**
|
3
|
+
* Токен для получения кода подтверждения телефона.
|
4
|
+
* Позволяет подключать внешнюю логику получения кода подтверждения.
|
5
|
+
*/
|
6
|
+
export const SC_PHONE_APPROVE_CODE_SENDER = new InjectionToken('SC_PHONE_APPROVE_CODE_SENDER');
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcGhvbmUtYXBwcm92ZS1jb2RlLXNlbmRlci50b2tlbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS92ZXJpZmljYXRpb24vcHJvdmlkZXJzL3NjLXBob25lLWFwcHJvdmUtY29kZS1zZW5kZXIudG9rZW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUcvQzs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyxJQUFJLGNBQWMsQ0FBNEMsOEJBQThCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5cbi8qKlxuICog0KLQvtC60LXQvSDQtNC70Y8g0L/QvtC70YPRh9C10L3QuNGPINC60L7QtNCwINC/0L7QtNGC0LLQtdGA0LbQtNC10L3QuNGPINGC0LXQu9C10YTQvtC90LAuXG4gKiDQn9C+0LfQstC+0LvRj9C10YIg0L/QvtC00LrQu9GO0YfQsNGC0Ywg0LLQvdC10YjQvdGO0Y4g0LvQvtCz0LjQutGDINC/0L7Qu9GD0YfQtdC90LjRjyDQutC+0LTQsCDQv9C+0LTRgtCy0LXRgNC20LTQtdC90LjRjy5cbiAqL1xuZXhwb3J0IGNvbnN0IFNDX1BIT05FX0FQUFJPVkVfQ09ERV9TRU5ERVIgPSBuZXcgSW5qZWN0aW9uVG9rZW48KHBob25lTnVtYmVyOiBzdHJpbmcpID0+IE9ic2VydmFibGU8dm9pZD4+KCdTQ19QSE9ORV9BUFBST1ZFX0NPREVfU0VOREVSJyk7XG4iXX0=
|
@@ -12,6 +12,7 @@ import { SC_SHOW_HELP_NOTIFICATION_IN_PHONE_INPUT } from '../../contacts/tokens/
|
|
12
12
|
import { scPhoneVerificationCodeMask } from '../../masks';
|
13
13
|
import { SC_VERIFICATION_CODE_TIMEOUT } from '../../providers/sc-verification-code-timeout';
|
14
14
|
import { ScHelpNotificationService } from '../../services';
|
15
|
+
import { SC_PHONE_APPROVE_CODE_SENDER } from '../providers';
|
15
16
|
import * as i0 from "@angular/core";
|
16
17
|
import * as i1 from "@angular/common";
|
17
18
|
import * as i2 from "@angular/forms";
|
@@ -33,6 +34,10 @@ export class ScVerificationPhoneCheckFormComponent {
|
|
33
34
|
* Сервис верификации.
|
34
35
|
*/
|
35
36
|
this.verificationService = inject(ScVerificationService);
|
37
|
+
/**
|
38
|
+
* Функция отправки кода подтверждения телефона.
|
39
|
+
*/
|
40
|
+
this.sendPhoneApproveCode = inject(SC_PHONE_APPROVE_CODE_SENDER);
|
36
41
|
/**
|
37
42
|
* Директива c `FormGroup` из DOM.
|
38
43
|
*/
|
@@ -100,20 +105,33 @@ export class ScVerificationPhoneCheckFormComponent {
|
|
100
105
|
/**
|
101
106
|
* {@link Observable} запроса данных получения кода подтверждения.
|
102
107
|
*/
|
103
|
-
this.loadingApproveCode$ = this.onSendCode.pipe(map(() => this.phoneControl.value), filter((value) => this.phoneControl.valid), switchMap((value) => this.
|
108
|
+
this.loadingApproveCode$ = this.onSendCode.pipe(map(() => this.phoneControl.value), filter((value) => this.phoneControl.valid), switchMap((value) => this.sendPhoneApproveCode(value).pipe(tap(() => {
|
104
109
|
this.reloadTimer$.next(this.codeTimeout);
|
105
110
|
}), map(() => false), catchError((error) => {
|
106
111
|
if (error instanceof HttpErrorResponse) {
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
const
|
112
|
-
|
112
|
+
let seconds = null;
|
113
|
+
if ('error' in error.error) {
|
114
|
+
const errorResponse = error.error;
|
115
|
+
const regex = /(\d{1,3})\s+seconds/;
|
116
|
+
const match = regex.exec(errorResponse.error);
|
117
|
+
if (match) {
|
118
|
+
seconds = Number.parseInt(match[1], 10);
|
119
|
+
}
|
120
|
+
}
|
121
|
+
else {
|
122
|
+
const errorResponse = error.error;
|
123
|
+
const regex = /(\d{2}):\d{2}/;
|
124
|
+
const match = regex.exec(errorResponse.message);
|
125
|
+
if (match && match.length > 1) {
|
126
|
+
const timeParts = match[0].split(':');
|
127
|
+
seconds = Number.parseInt(timeParts[1], 10);
|
128
|
+
}
|
129
|
+
}
|
130
|
+
if (seconds) {
|
113
131
|
this.reloadTimer$.next(seconds);
|
114
132
|
}
|
115
133
|
else {
|
116
|
-
this.phoneControl.setErrors({ serverResponse:
|
134
|
+
this.phoneControl.setErrors({ serverResponse: ['Неизвестная ошибка, попробуйте позже.'] });
|
117
135
|
}
|
118
136
|
}
|
119
137
|
return of(false);
|
@@ -254,4 +272,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
254
272
|
}], propDecorators: { showCodeFields: [{
|
255
273
|
type: Input
|
256
274
|
}] } });
|
257
|
-
//# 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 pseudoInvalid: InputSignal<boolean> = input(false);\n\n    /**\n     * Признак, что поле ввода телефона и кода подтверждения обязательные.\n     */\n    public readonly required: InputSignal<boolean> = input(false);\n\n    /**\n     * Текст подсказки.\n     */\n    public readonly subtitle: InputSignal<string> = input('');\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                [invalid]=\"pseudoInvalid() || null\"\n                [countrySearch]=\"true\"\n                [countries]=\"(countries | tuiSortCountries | async) || []\"\n                autocomplete=\"phone\"\n            />\n        </tui-textfield>\n        <p\n            *ngIf=\"subtitle()\"\n            class=\"text-body-s text-tui-text-tertiary\"\n        >\n            {{ subtitle() }}\n        </p>\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\n                formControlName=\"verificationCode\"\n                [required]=\"required() || showCodeFields\"\n            >\n                Код из СМС\n                <input\n                    tuiAutoFocus\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"]}
|
275
|
+
//# 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 { ScIKeycloakErrorResponse, 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';\nimport { SC_PHONE_APPROVE_CODE_SENDER } from '../providers';\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     * Функция отправки кода подтверждения телефона.\n     */\n    protected readonly sendPhoneApproveCode = inject(SC_PHONE_APPROVE_CODE_SENDER);\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 pseudoInvalid: InputSignal<boolean> = input(false);\n\n    /**\n     * Признак, что поле ввода телефона и кода подтверждения обязательные.\n     */\n    public readonly required: InputSignal<boolean> = input(false);\n\n    /**\n     * Текст подсказки.\n     */\n    public readonly subtitle: InputSignal<string> = input('');\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.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                        let seconds: number | null = null;\n\n                        if ('error' in error.error) {\n                            const errorResponse = error.error as ScIKeycloakErrorResponse;\n\n                            const regex = /(\\d{1,3})\\s+seconds/;\n                            const match = regex.exec(errorResponse.error);\n\n                            if (match) {\n                                seconds = Number.parseInt(match[1], 10);\n                            }\n                        } else {\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                                seconds = Number.parseInt(timeParts[1], 10);\n                            }\n                        }\n\n                        if (seconds) {\n                            this.reloadTimer$.next(seconds);\n                        } else {\n                            this.phoneControl.setErrors({ serverResponse: ['Неизвестная ошибка, попробуйте позже.'] });\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                [invalid]=\"pseudoInvalid() || null\"\n                [countrySearch]=\"true\"\n                [countries]=\"(countries | tuiSortCountries | async) || []\"\n                autocomplete=\"phone\"\n            />\n        </tui-textfield>\n        <p\n            *ngIf=\"subtitle()\"\n            class=\"text-body-s text-tui-text-tertiary\"\n        >\n            {{ subtitle() }}\n        </p>\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\n                formControlName=\"verificationCode\"\n                [required]=\"required() || showCodeFields\"\n            >\n                Код из СМС\n                <input\n                    tuiAutoFocus\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"]}
|