@snabcentr/client-ui 0.23.0 → 0.26.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (59) hide show
  1. package/auth/index.d.ts +2 -1
  2. package/auth/interfaces/{ApiErrorResponse.d.ts → api-error-response.d.ts} +1 -3
  3. package/auth/interfaces/index.d.ts +3 -0
  4. package/auth/interfaces/sc-bank-account-form-group.d.ts +11 -0
  5. package/auth/interfaces/sc-contact-form-group.d.ts +10 -0
  6. package/auth/sc-auth.module.d.ts +13 -8
  7. package/auth/sign-up-form/sc-sign-up-form.component.d.ts +141 -0
  8. package/contacts/add-contact-dialog/sc-add-contact-dialog.component.d.ts +3 -3
  9. package/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.d.ts +2 -2
  10. package/contragents/new-contragent-form/sc-new-contragent-form.component.d.ts +22 -15
  11. package/directives/carousel-item-hidden/sc-carousel-item-hidden.directive.d.ts +4 -4
  12. package/directives/index.d.ts +2 -0
  13. package/directives/next-input-focus/sc-next-input-focus.directive.d.ts +20 -0
  14. package/directives/next-input-focus/sc-next-input-focus.module.d.ts +10 -0
  15. package/esm2020/auth/index.mjs +3 -2
  16. package/esm2020/auth/interfaces/api-error-response.mjs +2 -0
  17. package/esm2020/auth/interfaces/index.mjs +4 -0
  18. package/esm2020/auth/interfaces/sc-bank-account-form-group.mjs +2 -0
  19. package/esm2020/auth/interfaces/sc-contact-form-group.mjs +2 -0
  20. package/esm2020/auth/sc-auth.module.mjs +51 -13
  21. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +13 -7
  22. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +13 -7
  23. package/esm2020/auth/sign-up-form/sc-sign-up-form.component.mjs +264 -0
  24. package/esm2020/catalog/price-card/sc-price-card.component.mjs +1 -1
  25. package/esm2020/catalog/price-history/sc-price-history.component.mjs +1 -1
  26. package/esm2020/contacts/add-contact-dialog/sc-add-contact-dialog.component.mjs +20 -12
  27. package/esm2020/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.mjs +21 -13
  28. package/esm2020/contragents/add-contragent-dialog/sc-add-contragent-dialog.component.mjs +19 -17
  29. package/esm2020/contragents/new-contragent-form/sc-new-contragent-form.component.mjs +29 -20
  30. package/esm2020/delivery-address/add-delivery-address-dialog/sc-add-delivery-address-dialog.component.mjs +1 -1
  31. package/esm2020/directives/carousel-item-hidden/sc-carousel-item-hidden.directive.mjs +2 -2
  32. package/esm2020/directives/carousel-item-hidden/sc-carousel-item-hidden.module.mjs +2 -2
  33. package/esm2020/directives/index.mjs +3 -1
  34. package/esm2020/directives/next-input-focus/sc-next-input-focus.directive.mjs +40 -0
  35. package/esm2020/directives/next-input-focus/sc-next-input-focus.module.mjs +19 -0
  36. package/esm2020/files/directives/tree-top.directive.mjs +10 -10
  37. package/esm2020/form-fields/suggestion-field/sc-suggestion-field.component.mjs +1 -1
  38. package/esm2020/user/index.mjs +2 -1
  39. package/esm2020/user/sc-user.module.mjs +19 -12
  40. package/esm2020/user/user-managers/sc-user-managers.component.mjs +38 -0
  41. package/esm2020/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.mjs +18 -12
  42. package/esm2020/validators/index.mjs +2 -2
  43. package/esm2020/validators/sc-password-confirm-matching-validator.mjs +17 -0
  44. package/esm2020/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +39 -27
  45. package/fesm2015/snabcentr-client-ui.mjs +2820 -2401
  46. package/fesm2015/snabcentr-client-ui.mjs.map +1 -1
  47. package/fesm2020/snabcentr-client-ui.mjs +2819 -2402
  48. package/fesm2020/snabcentr-client-ui.mjs.map +1 -1
  49. package/package.json +1 -1
  50. package/styles/tailwind/tailwind.scss +17 -0
  51. package/user/index.d.ts +1 -0
  52. package/user/sc-user.module.d.ts +9 -8
  53. package/user/user-managers/sc-user-managers.component.d.ts +26 -0
  54. package/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.d.ts +2 -2
  55. package/validators/index.d.ts +1 -1
  56. package/validators/{scPasswordConfirmMatchingValidator.d.ts → sc-password-confirm-matching-validator.d.ts} +1 -1
  57. package/verification/verification-phone-check-form/sc-verification-phone-check-form.component.d.ts +17 -17
  58. package/esm2020/auth/interfaces/ApiErrorResponse.mjs +0 -2
  59. package/esm2020/validators/scPasswordConfirmMatchingValidator.mjs +0 -17
@@ -1,11 +1,13 @@
1
1
  import { __decorate } from "tslib";
2
+ /* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */
3
+ import { HttpErrorResponse } from '@angular/common/http';
2
4
  import { ChangeDetectionStrategy, Component, Inject, Optional } from '@angular/core';
3
- import { FormGroup, FormControl, Validators } from '@angular/forms';
4
- import { Subject, filter, map, switchMap, catchError, of, startWith, share, tap } from 'rxjs';
5
+ import { FormControl, FormGroup, Validators } from '@angular/forms';
6
+ import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
5
7
  import { tuiIsFalsy } from '@taiga-ui/cdk';
6
8
  import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
9
+ import { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
7
10
  import { SC_USER_INFO, SC_USER_PROVIDERS } from '../../providers/scUserProviders';
8
- import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
9
11
  import * as i0 from "@angular/core";
10
12
  import * as i1 from "@snabcentr/client-core";
11
13
  import * as i2 from "@angular/common";
@@ -40,7 +42,7 @@ let ScUserPhoneApproveDialogComponent = class ScUserPhoneApproveDialogComponent
40
42
  */
41
43
  this.form = new FormGroup({
42
44
  phone: new FormControl(null, [Validators.required, Validators.minLength(12)]),
43
- verificationCode: new FormControl(null, [Validators.required, Validators.minLength(6)])
45
+ verificationCode: new FormControl(null, [Validators.required, Validators.minLength(6)]),
44
46
  });
45
47
  /**
46
48
  * {@link Subject} события отправки формы.
@@ -50,12 +52,16 @@ let ScUserPhoneApproveDialogComponent = class ScUserPhoneApproveDialogComponent
50
52
  * {@link Observable} запроса подтверждения номера телефона.
51
53
  */
52
54
  this.request$ = this.onSubmit.pipe(filter(() => this.form.valid), map(() => this.form.value), switchMap((value) => this.verificationService.approvePhone(value).pipe(switchMap(() => this.userService.updateUserChange$()), tap((user) => this.context?.completeWith(user.contacts.phone)), catchError((error) => {
53
- const errorResponse = error.error;
54
- for (const key in errorResponse.errors) {
55
- this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[`${key}`] });
56
- }
57
- if (!errorResponse.errors && errorResponse.message) {
58
- this.form.setErrors({ serverResponse: [errorResponse.message] });
55
+ if (error instanceof HttpErrorResponse) {
56
+ const { errors, message } = error.error;
57
+ for (const key in errors) {
58
+ if (Object.hasOwn(errors, key)) {
59
+ this.form.get(key)?.setErrors({ serverResponse: errors[key] });
60
+ }
61
+ }
62
+ if (!errors && message) {
63
+ this.form.setErrors({ serverResponse: [message] });
64
+ }
59
65
  }
60
66
  return of({});
61
67
  }), startWith(null))), share());
@@ -65,7 +71,7 @@ let ScUserPhoneApproveDialogComponent = class ScUserPhoneApproveDialogComponent
65
71
  this.loading$ = this.request$.pipe(map(tuiIsFalsy));
66
72
  this.user$.pipe(untilDestroyed(this)).subscribe((user) => {
67
73
  this.form.patchValue({
68
- phone: user.contacts.phone.value
74
+ phone: user.contacts.phone.value,
69
75
  });
70
76
  });
71
77
  }
@@ -88,4 +94,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
88
94
  type: Inject,
89
95
  args: [POLYMORPHEUS_CONTEXT]
90
96
  }] }]; } });
91
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-user-phone-approve-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.ts","../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, Inject, Optional } from '@angular/core';\nimport { FormGroup, FormControl, Validators } from '@angular/forms';\nimport { ApiErrorResponse } from '../../auth/interfaces/ApiErrorResponse';\nimport { Subject, Observable, filter, map, switchMap, catchError, of, startWith, share, tap } from 'rxjs';\nimport { ScApprovedData, ScIPhoneOnDataApprove, ScUser, ScUserService, ScVerificationService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { SC_USER_INFO, SC_USER_PROVIDERS } from '../../providers/scUserProviders';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\n\n/**\n * Диалоговое окно подтверждения номера телефона.\n */\n@Component({\n    selector: 'sc-user-phone-approve-dialog',\n    templateUrl: './sc-user-phone-approve-dialog.component.html',\n    providers: [SC_USER_PROVIDERS],\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\n@UntilDestroy({ checkProperties: true })\nexport class ScUserPhoneApproveDialogComponent {\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, Validators.minLength(12)]),\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$ = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIPhoneOnDataApprove),\n        switchMap((value) =>\n            this.verificationService.approvePhone(value).pipe(\n                switchMap(() => this.userService.updateUserChange$()),\n                tap((user) => this.context?.completeWith(user.contacts.phone)),\n                catchError((error: HttpErrorResponse) => {\n                    const errorResponse = error.error as ApiErrorResponse;\n\n                    for (const key in errorResponse.errors) {\n                        this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[`${key}`] });\n                    }\n\n                    if (!errorResponse.errors && errorResponse.message) {\n                        this.form.setErrors({ serverResponse: [errorResponse.message] });\n                    }\n                    return of({} as ScUser);\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 ScUserPhoneApproveDialogComponent}.\n     *\n     * @param verificationService Сервис верификации.\n     * @param user$ Поток информации о текущем пользователе.\n     * @param userService Сервис информации о пользователе.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly verificationService: ScVerificationService,\n        @Inject(SC_USER_INFO) public readonly user$: Observable<ScUser>,\n        private readonly userService: ScUserService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context?: TuiDialogContext<ScApprovedData, ScUserPhoneApproveDialogComponent>\n    ) {\n        this.user$.pipe(untilDestroyed(this)).subscribe((user) => {\n            this.form.patchValue({\n                phone: user.contacts.phone.value\n            });\n        });\n    }\n}\n","<ng-container *ngIf=\"user$ | async as user\">\n    <form [formGroup]=\"form\" *tuiLet=\"!!(loading$ | async) as loadingApproveCode\" (ngSubmit)=\"onSubmit.next()\" class=\"flex flex-col gap-3\">\n        <sc-verification-phone-check-form\n            [(haveCode)]=\"haveCode\"\n            [shouldBeConfirmed]=\"false\"\n            [showCodeFields]=\"!user.contacts.phone.isApproved\"\n            [readOnly]=\"true\"\n            class=\"w-full\"\n        ></sc-verification-phone-check-form>\n        <div *ngIf=\"user.contacts.phone.isApproved || haveCode\" class=\"flex gap-4 justify-center\">\n            <button *ngIf=\"user.contacts.phone.isApproved && context\" tuiButton (click)=\"context.$implicit.complete()\" type=\"button\" appearance=\"secondary\">Закрыть</button>\n            <button *ngIf=\"haveCode\" tuiButton type=\"submit\" [showLoader]=\"loadingApproveCode\" [disabled]=\"form.invalid || loadingApproveCode\" icon=\"scIconLogIn\">\n                Подтвердить\n            </button>\n        </div>\n    </form>\n</ng-container>\n"]}
97
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-user-phone-approve-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.ts","../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.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, Inject, Optional } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { ScApprovedData, ScIPhoneOnDataApprove, ScUser, ScUserService, ScVerificationService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { SC_USER_INFO, SC_USER_PROVIDERS } from '../../providers/scUserProviders';\n\n/**\n * Диалоговое окно подтверждения номера телефона.\n */\n@Component({\n    selector: 'sc-user-phone-approve-dialog',\n    templateUrl: './sc-user-phone-approve-dialog.component.html',\n    providers: [SC_USER_PROVIDERS],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\n@UntilDestroy({ checkProperties: true })\nexport class ScUserPhoneApproveDialogComponent {\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, Validators.minLength(12)]),\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$ = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIPhoneOnDataApprove),\n        switchMap((value) =>\n            this.verificationService.approvePhone(value).pipe(\n                switchMap(() => this.userService.updateUserChange$()),\n                tap((user) => this.context?.completeWith(user.contacts.phone)),\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                                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 ScUser);\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 ScUserPhoneApproveDialogComponent}.\n     *\n     * @param verificationService Сервис верификации.\n     * @param user$ Поток информации о текущем пользователе.\n     * @param userService Сервис информации о пользователе.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly verificationService: ScVerificationService,\n        @Inject(SC_USER_INFO) public readonly user$: Observable<ScUser>,\n        private readonly userService: ScUserService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context?: TuiDialogContext<ScApprovedData, ScUserPhoneApproveDialogComponent>\n    ) {\n        this.user$.pipe(untilDestroyed(this)).subscribe((user) => {\n            this.form.patchValue({\n                phone: user.contacts.phone.value,\n            });\n        });\n    }\n}\n","<ng-container *ngIf=\"user$ | async as user\">\n    <form [formGroup]=\"form\" *tuiLet=\"!!(loading$ | async) as loadingApproveCode\" (ngSubmit)=\"onSubmit.next()\" class=\"flex flex-col gap-3\">\n        <sc-verification-phone-check-form\n            [(haveCode)]=\"haveCode\"\n            [shouldBeConfirmed]=\"false\"\n            [showCodeFields]=\"!user.contacts.phone.isApproved\"\n            [readOnly]=\"true\"\n            class=\"w-full\"\n        ></sc-verification-phone-check-form>\n        <div *ngIf=\"user.contacts.phone.isApproved || haveCode\" class=\"flex gap-4 justify-center\">\n            <button *ngIf=\"user.contacts.phone.isApproved && context\" tuiButton (click)=\"context.$implicit.complete()\" type=\"button\" appearance=\"secondary\">Закрыть</button>\n            <button *ngIf=\"haveCode\" tuiButton type=\"submit\" [showLoader]=\"loadingApproveCode\" [disabled]=\"form.invalid || loadingApproveCode\" icon=\"scIconLogIn\">\n                Подтвердить\n            </button>\n        </div>\n    </form>\n</ng-container>\n"]}
@@ -1,5 +1,5 @@
1
1
  export * from './scBicValidator';
2
2
  export * from './scCorrespondentAccountValidator';
3
- export * from './scPasswordConfirmMatchingValidator';
3
+ export * from './sc-password-confirm-matching-validator';
4
4
  export * from './stepValidator';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmFsaWRhdG9ycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYyxzQ0FBc0MsQ0FBQztBQUNyRCxjQUFjLGlCQUFpQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9zY0JpY1ZhbGlkYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3NjQ29ycmVzcG9uZGVudEFjY291bnRWYWxpZGF0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9zY1Bhc3N3b3JkQ29uZmlybU1hdGNoaW5nVmFsaWRhdG9yJztcbmV4cG9ydCAqIGZyb20gJy4vc3RlcFZhbGlkYXRvcic7XG4iXX0=
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmFsaWRhdG9ycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtCQUFrQixDQUFDO0FBQ2pDLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYywwQ0FBMEMsQ0FBQztBQUN6RCxjQUFjLGlCQUFpQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9zY0JpY1ZhbGlkYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3NjQ29ycmVzcG9uZGVudEFjY291bnRWYWxpZGF0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1wYXNzd29yZC1jb25maXJtLW1hdGNoaW5nLXZhbGlkYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3N0ZXBWYWxpZGF0b3InO1xuIl19
@@ -0,0 +1,17 @@
1
+ /**
2
+ * Проверяет повторный ввод пароля на эквивалентность паролю.
3
+ *
4
+ * @param passwordKey Ключ поля ввода пароля.
5
+ * @param passwordConfirmKey Ключ поля повторного ввода пароля.
6
+ */
7
+ export function scPasswordConfirmMatchingValidator(passwordKey, passwordConfirmKey) {
8
+ return (control) => {
9
+ const passwordInput = control.get(passwordKey);
10
+ const passwordConfirmInput = control.get(passwordConfirmKey);
11
+ if (!passwordInput || !passwordConfirmInput || passwordInput.invalid || passwordConfirmInput.invalid || !passwordInput.value || !passwordConfirmInput.value) {
12
+ return null;
13
+ }
14
+ return passwordInput.value === passwordConfirmInput.value ? null : { passwordNotEquivalent: true };
15
+ };
16
+ }
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcGFzc3dvcmQtY29uZmlybS1tYXRjaGluZy12YWxpZGF0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmFsaWRhdG9ycy9zYy1wYXNzd29yZC1jb25maXJtLW1hdGNoaW5nLXZhbGlkYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQTs7Ozs7R0FLRztBQUNILE1BQU0sVUFBVSxrQ0FBa0MsQ0FBQyxXQUFtQixFQUFFLGtCQUEwQjtJQUM5RixPQUFPLENBQUMsT0FBd0IsRUFBMkIsRUFBRTtRQUN6RCxNQUFNLGFBQWEsR0FBMEMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUN0RixNQUFNLG9CQUFvQixHQUEwQyxPQUFPLENBQUMsR0FBRyxDQUFDLGtCQUFrQixDQUFDLENBQUM7UUFFcEcsSUFBSSxDQUFDLGFBQWEsSUFBSSxDQUFDLG9CQUFvQixJQUFJLGFBQWEsQ0FBQyxPQUFPLElBQUksb0JBQW9CLENBQUMsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDLEtBQUssSUFBSSxDQUFDLG9CQUFvQixDQUFDLEtBQUssRUFBRTtZQUN6SixPQUFPLElBQUksQ0FBQztTQUNmO1FBRUQsT0FBTyxhQUFhLENBQUMsS0FBSyxLQUFLLG9CQUFvQixDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLHFCQUFxQixFQUFFLElBQUksRUFBRSxDQUFDO0lBQ3ZHLENBQUMsQ0FBQztBQUNOLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBYnN0cmFjdENvbnRyb2wsIFZhbGlkYXRpb25FcnJvcnMsIFZhbGlkYXRvckZuIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG4vKipcbiAqINCf0YDQvtCy0LXRgNGP0LXRgiDQv9C+0LLRgtC+0YDQvdGL0Lkg0LLQstC+0LQg0L/QsNGA0L7Qu9GPINC90LAg0Y3QutCy0LjQstCw0LvQtdC90YLQvdC+0YHRgtGMINC/0LDRgNC+0LvRji5cbiAqXG4gKiBAcGFyYW0gcGFzc3dvcmRLZXkg0JrQu9GO0Ycg0L/QvtC70Y8g0LLQstC+0LTQsCDQv9Cw0YDQvtC70Y8uXG4gKiBAcGFyYW0gcGFzc3dvcmRDb25maXJtS2V5INCa0LvRjtGHINC/0L7Qu9GPINC/0L7QstGC0L7RgNC90L7Qs9C+INCy0LLQvtC00LAg0L/QsNGA0L7Qu9GPLlxuICovXG5leHBvcnQgZnVuY3Rpb24gc2NQYXNzd29yZENvbmZpcm1NYXRjaGluZ1ZhbGlkYXRvcihwYXNzd29yZEtleTogc3RyaW5nLCBwYXNzd29yZENvbmZpcm1LZXk6IHN0cmluZyk6IFZhbGlkYXRvckZuIHwgbnVsbCB7XG4gICAgcmV0dXJuIChjb250cm9sOiBBYnN0cmFjdENvbnRyb2wpOiBWYWxpZGF0aW9uRXJyb3JzIHwgbnVsbCA9PiB7XG4gICAgICAgIGNvbnN0IHBhc3N3b3JkSW5wdXQ6IEFic3RyYWN0Q29udHJvbDxzdHJpbmcgfCBudWxsPiB8IG51bGwgPSBjb250cm9sLmdldChwYXNzd29yZEtleSk7XG4gICAgICAgIGNvbnN0IHBhc3N3b3JkQ29uZmlybUlucHV0OiBBYnN0cmFjdENvbnRyb2w8c3RyaW5nIHwgbnVsbD4gfCBudWxsID0gY29udHJvbC5nZXQocGFzc3dvcmRDb25maXJtS2V5KTtcblxuICAgICAgICBpZiAoIXBhc3N3b3JkSW5wdXQgfHwgIXBhc3N3b3JkQ29uZmlybUlucHV0IHx8IHBhc3N3b3JkSW5wdXQuaW52YWxpZCB8fCBwYXNzd29yZENvbmZpcm1JbnB1dC5pbnZhbGlkIHx8ICFwYXNzd29yZElucHV0LnZhbHVlIHx8ICFwYXNzd29yZENvbmZpcm1JbnB1dC52YWx1ZSkge1xuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gcGFzc3dvcmRJbnB1dC52YWx1ZSA9PT0gcGFzc3dvcmRDb25maXJtSW5wdXQudmFsdWUgPyBudWxsIDogeyBwYXNzd29yZE5vdEVxdWl2YWxlbnQ6IHRydWUgfTtcbiAgICB9O1xufVxuIl19
@@ -1,7 +1,9 @@
1
+ /* eslint-disable no-underscore-dangle,lodash/prefer-constant */
2
+ import { HttpErrorResponse } from '@angular/common/http';
1
3
  import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, Output } from '@angular/core';
2
4
  import { FormGroupDirective } from '@angular/forms';
3
- import { Subject, filter, map, switchMap, catchError, of, startWith, share, tap, finalize, timer, scan, takeWhile, endWith, distinctUntilChanged } from 'rxjs';
4
5
  import { tuiControlValue } from '@taiga-ui/cdk';
6
+ import { catchError, distinctUntilChanged, endWith, filter, finalize, map, of, scan, share, startWith, Subject, switchMap, takeWhile, tap, timer } from 'rxjs';
5
7
  import { scPhoneVerificationCodeMask } from '../../masks';
6
8
  import * as i0 from "@angular/core";
7
9
  import * as i1 from "@snabcentr/client-core";
@@ -24,10 +26,6 @@ export class ScVerificationPhoneCheckFormComponent {
24
26
  constructor(verificationService, formGroupDirective) {
25
27
  this.verificationService = verificationService;
26
28
  this.formGroupDirective = formGroupDirective;
27
- /**
28
- * Признак, следует ли отображать функционал кода подтверждения.
29
- */
30
- this._showCodeFields = true;
31
29
  /**
32
30
  * Признак, что поле ввода телефона только для чтения.
33
31
  */
@@ -55,20 +53,26 @@ export class ScVerificationPhoneCheckFormComponent {
55
53
  /**
56
54
  * {@link Observable} запроса данных получения кода подтверждения.
57
55
  */
58
- this.loadingApproveCode$ = this.onSendCode.pipe(filter(() => this.phoneControl.valid), map(() => this.phoneControl.value), switchMap((value) => this.verificationService.sendPhoneApproveCode(value).pipe(tap(() => this.reloadTimer$.next(60)), map(() => false), catchError((error) => {
59
- const errorResponse = error.error;
60
- const regex = /(\d{2}):\d{2}/;
61
- const match = errorResponse.message.match(regex);
62
- if (match && match.length > 1) {
63
- const timeParts = match[0].split(':');
64
- const seconds = parseInt(timeParts[1], 10);
65
- this.reloadTimer$.next(seconds);
66
- }
67
- else {
68
- this.phoneControl.setErrors({ serverResponse: errorResponse?.errors?.['phone'] ?? [errorResponse.message] });
56
+ this.loadingApproveCode$ = this.onSendCode.pipe(map(() => this.phoneControl.value), filter((value) => this.phoneControl.valid), switchMap((value) => this.verificationService.sendPhoneApproveCode(value).pipe(tap(() => {
57
+ this.reloadTimer$.next(60);
58
+ }), map(() => false), catchError((error) => {
59
+ if (error instanceof HttpErrorResponse) {
60
+ const errorResponse = error.error;
61
+ const regex = /(\d{2}):\d{2}/;
62
+ const match = errorResponse.message.match(regex);
63
+ if (match && match.length > 1) {
64
+ const timeParts = match[0].split(':');
65
+ const seconds = Number.parseInt(timeParts[1], 10);
66
+ this.reloadTimer$.next(seconds);
67
+ }
68
+ else {
69
+ this.phoneControl.setErrors({ serverResponse: errorResponse.errors?.['phone'] ?? [errorResponse.message] });
70
+ }
69
71
  }
70
72
  return of(false);
71
- }), finalize(() => this.setHaveCode(true)), startWith(true))), startWith(false), share());
73
+ }), finalize(() => {
74
+ this.setHaveCode(true);
75
+ }), startWith(true))), startWith(false), share());
72
76
  /**
73
77
  * Маска поля ввода кода для подтверждения.
74
78
  */
@@ -82,11 +86,15 @@ export class ScVerificationPhoneCheckFormComponent {
82
86
  *
83
87
  * TODO: Вынести таймер в отдельную директиву TASK:[#9260].
84
88
  */
85
- this.timer$ = this.reloadTimer$.pipe(switchMap((sec) => timer(0, 1000).pipe(scan((total) => --total, sec), takeWhile((total) => total >= 0), map((total) => {
89
+ this.timer$ = this.reloadTimer$.pipe(switchMap((sec) => timer(0, 1000).pipe(scan((total) => -1 + total, sec), takeWhile((total) => total >= 0), map((total) => {
86
90
  const minutes = Math.floor(total / 60);
87
91
  const seconds = total % 60;
88
92
  return `${Math.round(minutes).toString().padStart(2, '0')}:${Math.round(seconds).toString().padStart(2, '0')}`;
89
93
  }), endWith(null), startWith(null), distinctUntilChanged())));
94
+ /**
95
+ * Признак, следует ли отображать функционал кода подтверждения.
96
+ */
97
+ this._showCodeFields = true;
90
98
  }
91
99
  /**
92
100
  * Признак, следует ли отображать функционал кода подтверждения.
@@ -111,18 +119,20 @@ export class ScVerificationPhoneCheckFormComponent {
111
119
  * Поле ввода 'Номер телефона'.
112
120
  */
113
121
  get phoneControl() {
114
- return this.form?.controls.phone;
122
+ return this.form.controls.phone;
115
123
  }
116
124
  /** @inheritDoc */
117
125
  ngOnInit() {
118
- this.phoneCheck$ = tuiControlValue(this.phoneControl).pipe(tap(() => this.reloadTimer$.next(0)), switchMap((value) => {
126
+ this.phoneCheck$ = tuiControlValue(this.phoneControl).pipe(tap(() => {
127
+ this.reloadTimer$.next(0);
128
+ }), switchMap((value) => {
119
129
  if (this.phoneControl.valid && !!value) {
120
130
  return this.verificationService.getPhoneCheck$(value).pipe(map((result) => {
121
131
  if (this.shouldBeBusy !== result.isBusy) {
122
132
  this.phoneControl.setErrors({
123
133
  serverResponse: [
124
- result.isBusy ? 'Пользователь с таким номером телефона уже существует' : 'Пользователя с таким номером телефона не существует'
125
- ]
134
+ result.isBusy ? 'Пользователь с таким номером телефона уже существует' : 'Пользователя с таким номером телефона не существует',
135
+ ],
126
136
  });
127
137
  this.phoneControl.markAsTouched();
128
138
  return false;
@@ -132,16 +142,18 @@ export class ScVerificationPhoneCheckFormComponent {
132
142
  serverResponse: [
133
143
  result.isConfirmed
134
144
  ? 'Телефон уже подтверждён'
135
- : 'Номер телефона не подтверждён или вход по нему невозможен. Обратитесь к вашему персональному менеджеру или войдите с использованием e-mail и пароля.'
136
- ]
145
+ : 'Номер телефона не подтверждён или вход по нему невозможен. Обратитесь к вашему персональному менеджеру или войдите с использованием e-mail и пароля.',
146
+ ],
137
147
  });
138
148
  this.phoneControl.markAsTouched();
139
149
  return false;
140
150
  }
141
151
  return true;
142
152
  }), catchError((error) => {
143
- const errorResponse = error.error;
144
- this.phoneControl.setErrors({ serverResponse: errorResponse?.errors?.['phone'] ?? [errorResponse.message] });
153
+ if (error instanceof HttpErrorResponse) {
154
+ const errorResponse = error.error;
155
+ this.phoneControl.setErrors({ serverResponse: errorResponse.errors?.['phone'] ?? [errorResponse.message] });
156
+ }
145
157
  return of(false);
146
158
  }), startWith(null));
147
159
  }
@@ -181,4 +193,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
181
193
  }], haveCodeChange: [{
182
194
  type: Output
183
195
  }] } });
184
- //# 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":["import { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, OnInit, Output } from '@angular/core';\nimport { FormGroup, FormControl, FormGroupDirective } from '@angular/forms';\nimport { MaskitoOptions } from '@maskito/core';\nimport { ScVerificationService } from '@snabcentr/client-core';\nimport { ApiErrorResponse } from '../../auth/interfaces/ApiErrorResponse';\nimport { Subject, Observable, filter, map, switchMap, catchError, of, startWith, share, tap, finalize, timer, scan, takeWhile, endWith, distinctUntilChanged } from 'rxjs';\nimport { tuiControlValue } from '@taiga-ui/cdk';\nimport { scPhoneVerificationCodeMask } from '../../masks';\n\n/**\n * Компонент формы проверки телефона и получения кода подтверждения.\n */\n@Component({\n    selector: 'sc-verification-phone-check-form',\n    templateUrl: './sc-verification-phone-check-form.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScVerificationPhoneCheckFormComponent implements OnInit {\n    /**\n     * Признак, следует ли отображать функционал кода подтверждения.\n     */\n    private _showCodeFields: boolean = true;\n\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    @Input()\n    public readOnly: boolean = false;\n\n    /**\n     * Признак, следует ли телефону быть в системе. От этого признака зависит в каких случаях выдавать ошибку при проверки занятости телефона.\n     */\n    @Input()\n    public shouldBeBusy: boolean = true;\n\n    /**\n     * Признак, следует ли проверять подтверждён ли телефона в системе. При `true` будет вызывать ошибку при наличие телефона в системе без подтверждения.\n     */\n    @Input()\n    public shouldBeConfirmed?: boolean = undefined;\n\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    @Input()\n    public haveCode: boolean = false;\n\n    /**\n     * Событие для обратной привязки наличия кода подтверждения\n     */\n    @Output()\n    public haveCodeChange = new EventEmitter<boolean>();\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    /**\n     * {@link Observable} запроса проверки номера телефона.\n     */\n    public phoneCheck$?: Observable<boolean | null>;\n\n    /**\n     * {@link Subject} события отправки кода подтверждения.\n     */\n    public readonly onSendCode: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных получения кода подтверждения.\n     */\n    public readonly loadingApproveCode$: Observable<boolean> = this.onSendCode.pipe(\n        filter(() => this.phoneControl.valid),\n        map(() => this.phoneControl.value as string),\n        switchMap((value) =>\n            this.verificationService.sendPhoneApproveCode(value).pipe(\n                tap(() => this.reloadTimer$.next(60)),\n                map(() => false),\n                catchError((error: HttpErrorResponse) => {\n                    const errorResponse = error.error as ApiErrorResponse;\n\n                    const regex = /(\\d{2}):\\d{2}/;\n                    const match = errorResponse.message.match(regex);\n\n                    if (match && match.length > 1) {\n                        const timeParts = match[0].split(':');\n                        const seconds = 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                    return of(false);\n                }),\n                finalize(() => this.setHaveCode(true)),\n                startWith(true)\n            )\n        ),\n        startWith(false),\n        share()\n    );\n\n    /**\n     * Маска поля ввода кода для подтверждения.\n     */\n    public readonly verificationCodeMask: MaskitoOptions = scPhoneVerificationCodeMask;\n\n    /**\n     * {@link Subject} события запуска/остановки таймера.\n     */\n    public readonly reloadTimer$: Subject<number> = new Subject<number>();\n\n    /**\n     * {@link Observable} Таймера.\n     *\n     * TODO: Вынести таймер в отдельную директиву TASK:[#9260].\n     */\n    public readonly timer$: Observable<string | null> = this.reloadTimer$.pipe(\n        switchMap((sec) =>\n            timer(0, 1000).pipe(\n                scan((total) => --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     * Инициализирует экземпляр класса {@link ScVerificationPhoneCheckFormComponent}.\n     *\n     * @param verificationService Сервис верификации.\n     * @param formGroupDirective Директива c `FormGroup` из DOM.\n     */\n    public constructor(\n        private readonly verificationService: ScVerificationService,\n        @Inject(FormGroupDirective)\n        private readonly formGroupDirective?: FormGroupDirective\n    ) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.phoneCheck$ = tuiControlValue<string | null>(this.phoneControl).pipe(\n            tap(() => this.reloadTimer$.next(0)),\n            switchMap((value) => {\n                if (this.phoneControl.valid && !!value) {\n                    return this.verificationService.getPhoneCheck$(value).pipe(\n                        map((result) => {\n                            if (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: HttpErrorResponse) => {\n                            const errorResponse = error.error as ApiErrorResponse;\n                            this.phoneControl.setErrors({ serverResponse: errorResponse?.errors?.['phone'] ?? [errorResponse.message] });\n                            return of(false);\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    public setHaveCode(haveCode: boolean): void {\n        this.haveCode = haveCode;\n        this.haveCodeChange.emit(haveCode);\n    }\n}\n","<form [formGroup]=\"form\" *tuiLet=\"(loadingApproveCode$ | async) as loadingApproveCode\" class=\"flex flex-col gap-3\">\n    <label tuiLabel=\"Номер телефона\">\n        <tui-input-phone formControlName=\"phone\" [readOnly]=\"readOnly\" [tuiTextfieldCustomContent]=\"checkingPhone\">\n            Номер телефона\n            <input tuiTextfield autocomplete=\"phone\" />\n        </tui-input-phone>\n        <tui-error formControlName=\"phone\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </label>\n    <ng-container *ngIf=\"showCodeFields && form.enabled\">\n        <label *ngIf=\"loadingApproveCode === false && haveCode\" tuiLabel=\"Код из СМС\">\n            <tui-input formControlName=\"verificationCode\">\n                Код из СМС\n                <input tuiTextfield [maskito]=\"verificationCodeMask\" autocomplete=\"new-password\" />\n            </tui-input>\n            <tui-error formControlName=\"verificationCode\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n        </label>\n\n        <div *tuiLet=\"!!loadingApproveCode as loadingApproveCode\" class=\"flex gap-4 justify-center\">\n            <button\n                *ngIf=\"!haveCode\"\n                tuiButton\n                (click)=\"onSendCode.next()\"\n                [disabled]=\"loadingApproveCode || !!!(phoneCheck$ | async) || phoneControl.invalid\"\n                [showLoader]=\"loadingApproveCode\"\n                icon=\"scIconLogIn\"\n            >\n                Получить код\n            </button>\n            <button *ngIf=\"!loadingApproveCode && !haveCode\" tuiLink [pseudo]=\"true\" [disabled]=\"!!!(phoneCheck$ | async) || phoneControl.invalid\" (click)=\"setHaveCode(true)\">\n                У меня есть код\n            </button>\n\n            <ng-container *tuiLet=\"timer$ | async as timer\">\n                <tui-loader *ngIf=\"haveCode\" [showLoader]=\"loadingApproveCode\">\n                    <button tuiLink [pseudo]=\"true\" [disabled]=\"loadingApproveCode || timer\" (click)=\"onSendCode.next()\">\n                        Повторно отправить код\n                        <ng-container *ngIf=\"timer\" class=\"!text-tui-base-08\">(через {{ timer }})</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 *ngIf=\"!!!(phoneCheck$ | async) && phoneControl.valid\" class=\"w-4 h-4\"> </tui-loader>\n</ng-template>\n"]}
196
+ //# 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 { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, OnInit, Output } from '@angular/core';\nimport { FormControl, FormGroup, FormGroupDirective } from '@angular/forms';\nimport { MaskitoOptions } from '@maskito/core';\nimport { ScVerificationService } from '@snabcentr/client-core';\nimport { tuiControlValue } from '@taiga-ui/cdk';\nimport { catchError, distinctUntilChanged, endWith, filter, finalize, map, Observable, of, scan, share, startWith, Subject, switchMap, takeWhile, tap, timer } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { scPhoneVerificationCodeMask } from '../../masks';\n\n/**\n * Компонент формы проверки телефона и получения кода подтверждения.\n */\n@Component({\n    selector: 'sc-verification-phone-check-form',\n    templateUrl: './sc-verification-phone-check-form.component.html',\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    @Input()\n    public readOnly: boolean = false;\n\n    /**\n     * Признак, следует ли телефону быть в системе. От этого признака зависит в каких случаях выдавать ошибку при проверки занятости телефона.\n     */\n    @Input()\n    public shouldBeBusy: boolean = true;\n\n    /**\n     * Признак, следует ли проверять подтверждён ли телефона в системе. При `true` будет вызывать ошибку при наличие телефона в системе без подтверждения.\n     */\n    @Input()\n    public shouldBeConfirmed?: boolean = undefined;\n\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    @Input()\n    public haveCode: boolean = false;\n\n    /**\n     * Событие для обратной привязки наличия кода подтверждения\n     */\n    @Output()\n    public haveCodeChange = new EventEmitter<boolean>();\n\n    /**\n     * {@link Observable} запроса проверки номера телефона.\n     */\n    public phoneCheck$?: Observable<boolean | null>;\n\n    /**\n     * {@link Subject} события отправки кода подтверждения.\n     */\n    public readonly onSendCode: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных получения кода подтверждения.\n     */\n    public 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(60);\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 = errorResponse.message.match(regex);\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            )\n        ),\n        startWith(false),\n        share()\n    );\n\n    /**\n     * Маска поля ввода кода для подтверждения.\n     */\n    public readonly verificationCodeMask: MaskitoOptions = scPhoneVerificationCodeMask;\n\n    /**\n     * {@link Subject} события запуска/остановки таймера.\n     */\n    public readonly reloadTimer$: Subject<number> = new Subject<number>();\n\n    /**\n     * {@link Observable} Таймера.\n     *\n     * TODO: Вынести таймер в отдельную директиву TASK:[#9260].\n     */\n    public 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    private _showCodeFields: boolean = true;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScVerificationPhoneCheckFormComponent}.\n     *\n     * @param verificationService Сервис верификации.\n     * @param formGroupDirective Директива c `FormGroup` из DOM.\n     */\n    public constructor(\n        private readonly verificationService: ScVerificationService,\n        @Inject(FormGroupDirective)\n        private readonly formGroupDirective?: FormGroupDirective\n    ) {}\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            }),\n            switchMap((value) => {\n                if (this.phoneControl.valid && !!value) {\n                    return this.verificationService.getPhoneCheck$(value).pipe(\n                        map((result) => {\n                            if (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                        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    public setHaveCode(haveCode: boolean): void {\n        this.haveCode = haveCode;\n        this.haveCodeChange.emit(haveCode);\n    }\n}\n","<form [formGroup]=\"form\" *tuiLet=\"(loadingApproveCode$ | async) as loadingApproveCode\" class=\"flex flex-col gap-3\">\n    <label tuiLabel=\"Номер телефона\">\n        <tui-input-phone formControlName=\"phone\" [readOnly]=\"readOnly\" [tuiTextfieldCustomContent]=\"checkingPhone\">\n            Номер телефона\n            <input tuiTextfield autocomplete=\"phone\" />\n        </tui-input-phone>\n        <tui-error formControlName=\"phone\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </label>\n    <ng-container *ngIf=\"showCodeFields && form.enabled\">\n        <label *ngIf=\"loadingApproveCode === false && haveCode\" tuiLabel=\"Код из СМС\">\n            <tui-input formControlName=\"verificationCode\">\n                Код из СМС\n                <input tuiTextfield [maskito]=\"verificationCodeMask\" autocomplete=\"new-password\" />\n            </tui-input>\n            <tui-error formControlName=\"verificationCode\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n        </label>\n\n        <div *tuiLet=\"!!loadingApproveCode as loadingApproveCode\" class=\"flex gap-4 justify-center\">\n            <button\n                *ngIf=\"!haveCode\"\n                tuiButton\n                (click)=\"onSendCode.next()\"\n                [disabled]=\"loadingApproveCode || !!!(phoneCheck$ | async) || phoneControl.invalid\"\n                [showLoader]=\"loadingApproveCode\"\n                icon=\"scIconLogIn\"\n            >\n                Получить код\n            </button>\n            <button *ngIf=\"!loadingApproveCode && !haveCode\" tuiLink [pseudo]=\"true\" [disabled]=\"!!!(phoneCheck$ | async) || phoneControl.invalid\" (click)=\"setHaveCode(true)\">\n                У меня есть код\n            </button>\n\n            <ng-container *tuiLet=\"timer$ | async as timer\">\n                <tui-loader *ngIf=\"haveCode\" [showLoader]=\"loadingApproveCode\">\n                    <button tuiLink [pseudo]=\"true\" [disabled]=\"loadingApproveCode || timer\" (click)=\"onSendCode.next()\">\n                        Повторно отправить код\n                        <ng-container *ngIf=\"timer\" class=\"!text-tui-base-08\">(через {{ timer }})</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 *ngIf=\"!!!(phoneCheck$ | async) && phoneControl.valid\" class=\"w-4 h-4\"> </tui-loader>\n</ng-template>\n"]}