@snabcentr/client-ui 3.44.0 → 3.44.3
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/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.d.ts +5 -0
- package/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.d.ts +5 -0
- package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +10 -2
- package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +10 -2
- package/esm2022/form-fields/addresses-selection-field/sc-addresses-selection-field.component.mjs +6 -4
- package/esm2022/services/sc-help-notification.service.mjs +11 -1
- package/fesm2022/snabcentr-client-ui.mjs +29 -3
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +2 -2
- package/services/sc-help-notification.service.d.ts +6 -0
package/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.d.ts
CHANGED
|
@@ -2,6 +2,7 @@ import { EventEmitter } from '@angular/core';
|
|
|
2
2
|
import { FormGroup } from '@angular/forms';
|
|
3
3
|
import { ScAuthService } from '@snabcentr/client-core';
|
|
4
4
|
import { Observable, Subject } from 'rxjs';
|
|
5
|
+
import { ScHelpNotificationService } from '../../../services/sc-help-notification.service';
|
|
5
6
|
import * as i0 from "@angular/core";
|
|
6
7
|
/**
|
|
7
8
|
* Компонент аутентификации по адресу электронной почты и паролю.
|
|
@@ -16,6 +17,10 @@ export declare class ScSignInFormByEmailComponent {
|
|
|
16
17
|
* {@link Subject} события отправки формы.
|
|
17
18
|
*/
|
|
18
19
|
readonly onSubmit: Subject<void>;
|
|
20
|
+
/**
|
|
21
|
+
* Сервис для отображения Push-уведомлений с контактами для помощи клиенту.
|
|
22
|
+
*/
|
|
23
|
+
protected readonly helpNotificationService: ScHelpNotificationService;
|
|
19
24
|
/**
|
|
20
25
|
* {@link Observable} запроса данных аутентификации.
|
|
21
26
|
*/
|
package/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.d.ts
CHANGED
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { FormControl, FormGroup } from '@angular/forms';
|
|
2
2
|
import { ScAuthService } from '@snabcentr/client-core';
|
|
3
3
|
import { Observable, Subject } from 'rxjs';
|
|
4
|
+
import { ScHelpNotificationService } from '../../../services/sc-help-notification.service';
|
|
4
5
|
import * as i0 from "@angular/core";
|
|
5
6
|
/**
|
|
6
7
|
* Компонент аутентификации по номеру телефона и коду подтверждения.
|
|
@@ -22,6 +23,10 @@ export declare class ScSignInFormByPhoneComponent {
|
|
|
22
23
|
* {@link Subject} события отправки формы.
|
|
23
24
|
*/
|
|
24
25
|
readonly onSubmit: Subject<void>;
|
|
26
|
+
/**
|
|
27
|
+
* Сервис для отображения Push-уведомлений с контактами для помощи клиенту.
|
|
28
|
+
*/
|
|
29
|
+
protected readonly helpNotificationService: ScHelpNotificationService;
|
|
25
30
|
/**
|
|
26
31
|
* {@link Observable} запроса данных аутентификации.
|
|
27
32
|
*/
|
package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */
|
|
2
2
|
import { HttpErrorResponse } from '@angular/common/http';
|
|
3
|
-
import { ChangeDetectionStrategy, Component, EventEmitter, Output } from '@angular/core';
|
|
3
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, inject, Output } from '@angular/core';
|
|
4
4
|
import { FormControl, FormGroup, Validators } from '@angular/forms';
|
|
5
5
|
import { tuiIsFalsy } from '@taiga-ui/cdk';
|
|
6
6
|
import { catchError, filter, map, of, share, startWith, Subject, switchMap } from 'rxjs';
|
|
7
|
+
import { ScHelpNotificationService } from '../../../services/sc-help-notification.service';
|
|
7
8
|
import * as i0 from "@angular/core";
|
|
8
9
|
import * as i1 from "@snabcentr/client-core";
|
|
9
10
|
import * as i2 from "@angular/forms";
|
|
@@ -34,6 +35,10 @@ export class ScSignInFormByEmailComponent {
|
|
|
34
35
|
* {@link Subject} события отправки формы.
|
|
35
36
|
*/
|
|
36
37
|
this.onSubmit = new Subject();
|
|
38
|
+
/**
|
|
39
|
+
* Сервис для отображения Push-уведомлений с контактами для помощи клиенту.
|
|
40
|
+
*/
|
|
41
|
+
this.helpNotificationService = inject(ScHelpNotificationService);
|
|
37
42
|
/**
|
|
38
43
|
* {@link Observable} запроса данных аутентификации.
|
|
39
44
|
*/
|
|
@@ -46,6 +51,9 @@ export class ScSignInFormByEmailComponent {
|
|
|
46
51
|
this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });
|
|
47
52
|
}
|
|
48
53
|
}
|
|
54
|
+
if (message.includes('деактивирован')) {
|
|
55
|
+
this.helpNotificationService.helpNotificationByEmail(value.login);
|
|
56
|
+
}
|
|
49
57
|
if (!errors && message) {
|
|
50
58
|
this.formByEmail.setErrors({ serverResponse: [message] });
|
|
51
59
|
}
|
|
@@ -70,4 +78,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
70
78
|
}], ctorParameters: () => [{ type: i1.ScAuthService }], propDecorators: { forgotPassword: [{
|
|
71
79
|
type: Output
|
|
72
80
|
}] } });
|
|
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"]}
|
|
81
|
+
//# 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, inject, 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 { ScHelpNotificationService } from '../../../services/sc-help-notification.service';\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     * Сервис для отображения Push-уведомлений с контактами для помощи клиенту.\n     */\n    protected readonly helpNotificationService: ScHelpNotificationService = inject(ScHelpNotificationService);\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 (message.includes('деактивирован')) {\n                            this.helpNotificationService.helpNotificationByEmail(value.login);\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"]}
|
package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */
|
|
2
2
|
import { HttpErrorResponse } from '@angular/common/http';
|
|
3
|
-
import { ChangeDetectionStrategy, Component } from '@angular/core';
|
|
3
|
+
import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
|
|
4
4
|
import { FormControl, FormGroup, Validators } from '@angular/forms';
|
|
5
5
|
import { tuiIsFalsy } from '@taiga-ui/cdk';
|
|
6
6
|
import { catchError, filter, map, of, share, startWith, Subject, switchMap } from 'rxjs';
|
|
7
|
+
import { ScHelpNotificationService } from '../../../services/sc-help-notification.service';
|
|
7
8
|
import { phoneValidator } from '../../../validators/sc-phone-validator';
|
|
8
9
|
import * as i0 from "@angular/core";
|
|
9
10
|
import * as i1 from "@snabcentr/client-core";
|
|
@@ -39,6 +40,10 @@ export class ScSignInFormByPhoneComponent {
|
|
|
39
40
|
* {@link Subject} события отправки формы.
|
|
40
41
|
*/
|
|
41
42
|
this.onSubmit = new Subject();
|
|
43
|
+
/**
|
|
44
|
+
* Сервис для отображения Push-уведомлений с контактами для помощи клиенту.
|
|
45
|
+
*/
|
|
46
|
+
this.helpNotificationService = inject(ScHelpNotificationService);
|
|
42
47
|
/**
|
|
43
48
|
* {@link Observable} запроса данных аутентификации.
|
|
44
49
|
*/
|
|
@@ -51,6 +56,9 @@ export class ScSignInFormByPhoneComponent {
|
|
|
51
56
|
this.form.get(key)?.setErrors({ serverResponse: errors[key] });
|
|
52
57
|
}
|
|
53
58
|
}
|
|
59
|
+
if (message.includes('деактивирован')) {
|
|
60
|
+
this.helpNotificationService.helpNotificationByPhone(value.phone);
|
|
61
|
+
}
|
|
54
62
|
if (!errors && message) {
|
|
55
63
|
this.form.setErrors({ serverResponse: [message] });
|
|
56
64
|
}
|
|
@@ -69,4 +77,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
69
77
|
type: Component,
|
|
70
78
|
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
79
|
}], ctorParameters: () => [{ type: i1.ScAuthService }] });
|
|
72
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
80
|
+
//# 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,MAAM,eAAe,CAAC;AAC3E,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,yBAAyB,EAAE,MAAM,gDAAgD,CAAC;AAC3F,OAAO,EAAE,cAAc,EAAE,MAAM,wCAAwC,CAAC;;;;;;;;;AAGxE;;GAEG;AAMH,MAAM,OAAO,4BAA4B;IAiErC;;;;OAIG;IACH,YAAoC,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QArE9D;;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;QACgB,4BAAuB,GAA8B,MAAM,CAAC,yBAAyB,CAAC,CAAC;QAE1G;;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,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;oBACpC,IAAI,CAAC,uBAAuB,CAAC,uBAAuB,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;gBACtE,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;+GAtEzD,4BAA4B;mGAA5B,4BAA4B,gECrBzC,0uBA2BA;;4FDNa,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, inject } 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 { ScHelpNotificationService } from '../../../services/sc-help-notification.service';\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     * Сервис для отображения Push-уведомлений с контактами для помощи клиенту.\n     */\n    protected readonly helpNotificationService: ScHelpNotificationService = inject(ScHelpNotificationService);\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 (message.includes('деактивирован')) {\n                            this.helpNotificationService.helpNotificationByPhone(value.phone);\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"]}
|
package/esm2022/form-fields/addresses-selection-field/sc-addresses-selection-field.component.mjs
CHANGED
|
@@ -100,10 +100,12 @@ export class ScAddressesSelectionFieldComponent {
|
|
|
100
100
|
.getCityById$(Number(this.cityIdControl.value))
|
|
101
101
|
.pipe(takeUntilDestroyed(this.destroyRef))
|
|
102
102
|
.subscribe((city) => {
|
|
103
|
-
this.countryControl.patchValue(city.region.country);
|
|
104
|
-
this.regionControl.patchValue(city.region);
|
|
105
|
-
this.cityControl.patchValue(city);
|
|
106
103
|
this.canSelectCountry.set(city.region.country.id !== this.russiaCountryId);
|
|
104
|
+
setTimeout(() => {
|
|
105
|
+
this.countryControl.patchValue(city.region.country);
|
|
106
|
+
this.regionControl.patchValue(city.region);
|
|
107
|
+
this.cityControl.patchValue(city);
|
|
108
|
+
});
|
|
107
109
|
});
|
|
108
110
|
}
|
|
109
111
|
this.countryControl.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((country) => {
|
|
@@ -152,4 +154,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
152
154
|
type: Inject,
|
|
153
155
|
args: [FormGroupDirective]
|
|
154
156
|
}] }, { type: i1.ScLocationsService }] });
|
|
155
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-addresses-selection-field.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/form-fields/addresses-selection-field/sc-addresses-selection-field.component.ts","../../../../../projects/client-ui/form-fields/addresses-selection-field/sc-addresses-selection-field.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-unused-expressions,class-methods-use-this,@typescript-eslint/unbound-method */\n\nimport { ChangeDetectionStrategy, Component, DestroyRef, effect, Inject, inject, model, OnInit, SkipSelf } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroupDirective, Validators } from '@angular/forms';\nimport { ScICity, ScICityWithRegionAndCountry, ScICountry, ScIRegion, ScLocationsService } from '@snabcentr/client-core';\nimport { tuiControlValue, tuiIsPresent, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { combineLatest, debounceTime, filter, Observable, of, share, startWith, switchMap } from 'rxjs';\n\nimport { CURRENT_COUNTRY_ID } from '../../providers/sc-country-ids';\n\n/**\n * Компонент поля ввода страны/региона/города.\n */\n@Component({\n    selector: 'sc-addresses-selection-field',\n    templateUrl: './sc-addresses-selection-field.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddressesSelectionFieldComponent implements OnInit {\n    /**\n     * Флаг возможности выбора страны.\n     */\n    public readonly canSelectCountry = model<boolean>(false);\n\n    /**\n     * Идентификатор текущей страны системы.\n     */\n    private readonly russiaCountryId: number = inject(CURRENT_COUNTRY_ID);\n\n    /**\n     * FormControl для поля страны.\n     */\n    protected readonly countryControl: FormControl<ScICountry | null> = new FormControl<ScICountry | null>(null, Validators.required);\n\n    /**\n     * FormControl для поля региона.\n     */\n    protected readonly regionControl: FormControl<ScIRegion | null> = new FormControl<ScIRegion | null>(null, Validators.required);\n\n    /**\n     * FormControl для поля города.\n     */\n    protected cityControl: FormControl<ScICity | null> = new FormControl<ScICountry | null>(null, Validators.required);\n\n    /**\n     * {@link Observable} изменения списка стран.\n     */\n    protected countries$: Observable<ScICountry[] | null> = this.locationsService.getCountries$().pipe(share(), startWith(null));\n\n    /**\n     * {@link Observable} изменения списка регионов.\n     */\n    protected regions$: Observable<ScIRegion[] | null> = tuiControlValue<ScIRegion | null>(this.countryControl).pipe(\n        filter(tuiIsPresent),\n        switchMap((country: ScICountry) => this.locationsService.getRegions$(country.id).pipe(startWith(null))),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения списка городов.\n     */\n    protected cities$: Observable<ScICity[] | null> = combineLatest({\n        country: tuiControlValue<ScIRegion | null>(this.countryControl),\n        region: tuiControlValue<ScICity | null>(this.regionControl),\n    }).pipe(\n        debounceTime(0),\n        switchMap(({ country, region }: { country: ScIRegion | null; region: ScICity | null }) =>\n            country && region ? this.locationsService.getCities$(country.id, region.id).pipe(startWith(null)) : of([])\n        ),\n        share()\n    );\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private destroyRef: DestroyRef = inject(DestroyRef);\n\n    /**\n     * Функция преобразования объекта сортировки в значение, отображаемое в поле ввода.\n     *\n     * @param item Выбранный объект выпадающего списка.\n     * @param item.name Название страны/региона/города, которое будет отображено в поле.\n     */\n    protected readonly stringify = (item: { name: string }): string => item.name;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScAddressesSelectionFieldComponent}.\n     *\n     * @param formGroupDirective Директива c `FormGroup` из DOM.\n     * @param locationsService Сервис для получения списков стран, регионов, городов.\n     */\n    public constructor(\n        @SkipSelf() @Inject(FormGroupDirective) private readonly formGroupDirective: FormGroupDirective,\n        private readonly locationsService: ScLocationsService\n    ) {\n        effect(() => {\n            if (this.canSelectCountry()) {\n                this.countryControl.setValue(null);\n            } else {\n                this.countryControl.setValue({\n                    id: this.russiaCountryId,\n                    name: 'Россия',\n                });\n            }\n        });\n    }\n\n    /**\n     * `FormControl` поля идентификатора города.\n     */\n    public get cityIdControl(): FormControl<string | null> {\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        return this.formGroupDirective.form.get('cityId')! as FormControl<string | null>;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (this.cityIdControl.valid) {\n            this.locationsService\n                .getCityById$(Number(this.cityIdControl.value))\n                .pipe(takeUntilDestroyed(this.destroyRef))\n                .subscribe((city: ScICityWithRegionAndCountry) => {\n                    this.countryControl.patchValue(city.region.country);\n                    this.regionControl.patchValue(city.region);\n                    this.cityControl.patchValue(city);\n                    this.canSelectCountry.set(city.region.country.id !== this.russiaCountryId);\n                });\n        }\n\n        this.countryControl.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((country) => {\n            this.regionControl.reset(null);\n            this.cityControl.reset(null);\n            if (country) {\n                this.regionControl.enable();\n            } else {\n                this.regionControl.disable();\n            }\n            this.cityControl.disable();\n        });\n\n        this.regionControl.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((region) => {\n            this.cityControl.reset(null);\n            this.cityIdControl.reset(null);\n            if (region) {\n                this.cityControl.enable();\n            } else {\n                this.cityControl.disable();\n            }\n        });\n    }\n\n    /**\n     * Заполнение полей банковских реквизитов на основе выбранной подсказки.\n     *\n     * @param city Идентификатор записи города.\n     */\n    protected onSelectedCity(city: ScICity | null): void {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        if (this.cityIdControl) {\n            this.cityIdControl.setValue(city?.id.toString() ?? null);\n            tuiMarkControlAsTouchedAndValidate(this.cityIdControl);\n        }\n    }\n}\n","<label\n    *ngIf=\"canSelectCountry()\"\n    tuiLabel\n    class=\"mb-2\"\n>\n    Страна\n    <tui-combo-box\n        *tuiLet=\"(countries$ | async) ?? [] as countries\"\n        [formControl]=\"countryControl\"\n        [stringify]=\"stringify\"\n    >\n        Страна\n        <input\n            tuiTextfieldLegacy\n            autocapitalize=\"off\"\n            autocomplete=\"off\"\n            autocorrect=\"off\"\n            autocomplete=\"new-password\"\n        />\n        <tui-data-list-wrapper\n            *tuiDataList\n            [itemContent]=\"stringify | tuiStringifyContent\"\n            [items]=\"countries | tuiFilterByInput\"\n        ></tui-data-list-wrapper>\n    </tui-combo-box>\n    <tui-error\n        [formControl]=\"countryControl\"\n        [error]=\"[] | tuiFieldError | async\"\n    ></tui-error>\n</label>\n<div class=\"grid items-start gap-4 sm:grid-cols-2\">\n    <label tuiLabel>\n        Регион\n        <tui-combo-box\n            *tuiLet=\"(regions$ | async) ?? [] as regions\"\n            [formControl]=\"regionControl\"\n            [stringify]=\"stringify\"\n        >\n            Регион\n            <input\n                tuiTextfieldLegacy\n                autocapitalize=\"off\"\n                autocomplete=\"off\"\n                autocorrect=\"off\"\n                autocomplete=\"new-password\"\n            />\n            <tui-data-list-wrapper\n                *tuiDataList\n                [itemContent]=\"stringify | tuiStringifyContent\"\n                [items]=\"regions | tuiFilterByInput\"\n            ></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error\n            [formControl]=\"regionControl\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n    <label tuiLabel>\n        Город\n        <tui-combo-box\n            *tuiLet=\"(cities$ | async) ?? [] as cities\"\n            [formControl]=\"cityControl\"\n            [stringify]=\"stringify\"\n            (ngModelChange)=\"onSelectedCity($event)\"\n        >\n            Город\n            <input\n                tuiTextfieldLegacy\n                autocapitalize=\"off\"\n                autocomplete=\"off\"\n                autocorrect=\"off\"\n                autocomplete=\"new-password\"\n            />\n            <tui-data-list-wrapper\n                *tuiDataList\n                [itemContent]=\"stringify | tuiStringifyContent\"\n                [items]=\"cities | tuiFilterByInput\"\n            ></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error\n            [formControl]=\"cityIdControl\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n        <tui-error\n            *ngIf=\"cityIdControl.untouched\"\n            [formControl]=\"cityControl\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n</div>\n"]}
|
|
157
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-addresses-selection-field.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/form-fields/addresses-selection-field/sc-addresses-selection-field.component.ts","../../../../../projects/client-ui/form-fields/addresses-selection-field/sc-addresses-selection-field.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-unused-expressions,class-methods-use-this,@typescript-eslint/unbound-method */\n\nimport { ChangeDetectionStrategy, Component, DestroyRef, effect, Inject, inject, model, OnInit, SkipSelf } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroupDirective, Validators } from '@angular/forms';\nimport { ScICity, ScICityWithRegionAndCountry, ScICountry, ScIRegion, ScLocationsService } from '@snabcentr/client-core';\nimport { tuiControlValue, tuiIsPresent, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { combineLatest, debounceTime, filter, Observable, of, share, startWith, switchMap } from 'rxjs';\n\nimport { CURRENT_COUNTRY_ID } from '../../providers/sc-country-ids';\n\n/**\n * Компонент поля ввода страны/региона/города.\n */\n@Component({\n    selector: 'sc-addresses-selection-field',\n    templateUrl: './sc-addresses-selection-field.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddressesSelectionFieldComponent implements OnInit {\n    /**\n     * Флаг возможности выбора страны.\n     */\n    public readonly canSelectCountry = model<boolean>(false);\n\n    /**\n     * Идентификатор текущей страны системы.\n     */\n    private readonly russiaCountryId: number = inject(CURRENT_COUNTRY_ID);\n\n    /**\n     * FormControl для поля страны.\n     */\n    protected readonly countryControl: FormControl<ScICountry | null> = new FormControl<ScICountry | null>(null, Validators.required);\n\n    /**\n     * FormControl для поля региона.\n     */\n    protected readonly regionControl: FormControl<ScIRegion | null> = new FormControl<ScIRegion | null>(null, Validators.required);\n\n    /**\n     * FormControl для поля города.\n     */\n    protected cityControl: FormControl<ScICity | null> = new FormControl<ScICountry | null>(null, Validators.required);\n\n    /**\n     * {@link Observable} изменения списка стран.\n     */\n    protected countries$: Observable<ScICountry[] | null> = this.locationsService.getCountries$().pipe(share(), startWith(null));\n\n    /**\n     * {@link Observable} изменения списка регионов.\n     */\n    protected regions$: Observable<ScIRegion[] | null> = tuiControlValue<ScIRegion | null>(this.countryControl).pipe(\n        filter(tuiIsPresent),\n        switchMap((country: ScICountry) => this.locationsService.getRegions$(country.id).pipe(startWith(null))),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения списка городов.\n     */\n    protected cities$: Observable<ScICity[] | null> = combineLatest({\n        country: tuiControlValue<ScIRegion | null>(this.countryControl),\n        region: tuiControlValue<ScICity | null>(this.regionControl),\n    }).pipe(\n        debounceTime(0),\n        switchMap(({ country, region }: { country: ScIRegion | null; region: ScICity | null }) =>\n            country && region ? this.locationsService.getCities$(country.id, region.id).pipe(startWith(null)) : of([])\n        ),\n        share()\n    );\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private destroyRef: DestroyRef = inject(DestroyRef);\n\n    /**\n     * Функция преобразования объекта сортировки в значение, отображаемое в поле ввода.\n     *\n     * @param item Выбранный объект выпадающего списка.\n     * @param item.name Название страны/региона/города, которое будет отображено в поле.\n     */\n    protected readonly stringify = (item: { name: string }): string => item.name;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScAddressesSelectionFieldComponent}.\n     *\n     * @param formGroupDirective Директива c `FormGroup` из DOM.\n     * @param locationsService Сервис для получения списков стран, регионов, городов.\n     */\n    public constructor(\n        @SkipSelf() @Inject(FormGroupDirective) private readonly formGroupDirective: FormGroupDirective,\n        private readonly locationsService: ScLocationsService\n    ) {\n        effect(() => {\n            if (this.canSelectCountry()) {\n                this.countryControl.setValue(null);\n            } else {\n                this.countryControl.setValue({\n                    id: this.russiaCountryId,\n                    name: 'Россия',\n                });\n            }\n        });\n    }\n\n    /**\n     * `FormControl` поля идентификатора города.\n     */\n    public get cityIdControl(): FormControl<string | null> {\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        return this.formGroupDirective.form.get('cityId')! as FormControl<string | null>;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (this.cityIdControl.valid) {\n            this.locationsService\n                .getCityById$(Number(this.cityIdControl.value))\n                .pipe(takeUntilDestroyed(this.destroyRef))\n                .subscribe((city: ScICityWithRegionAndCountry) => {\n                    this.canSelectCountry.set(city.region.country.id !== this.russiaCountryId);\n                    setTimeout(() => {\n                        this.countryControl.patchValue(city.region.country);\n                        this.regionControl.patchValue(city.region);\n                        this.cityControl.patchValue(city);\n                    });\n                });\n        }\n\n        this.countryControl.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((country) => {\n            this.regionControl.reset(null);\n            this.cityControl.reset(null);\n\n            if (country) {\n                this.regionControl.enable();\n            } else {\n                this.regionControl.disable();\n            }\n\n            this.cityControl.disable();\n        });\n\n        this.regionControl.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((region) => {\n            this.cityControl.reset(null);\n            this.cityIdControl.reset(null);\n\n            if (region) {\n                this.cityControl.enable();\n            } else {\n                this.cityControl.disable();\n            }\n        });\n    }\n\n    /**\n     * Заполнение полей банковских реквизитов на основе выбранной подсказки.\n     *\n     * @param city Идентификатор записи города.\n     */\n    protected onSelectedCity(city: ScICity | null): void {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        if (this.cityIdControl) {\n            this.cityIdControl.setValue(city?.id.toString() ?? null);\n            tuiMarkControlAsTouchedAndValidate(this.cityIdControl);\n        }\n    }\n}\n","<label\n    *ngIf=\"canSelectCountry()\"\n    tuiLabel\n    class=\"mb-2\"\n>\n    Страна\n    <tui-combo-box\n        *tuiLet=\"(countries$ | async) ?? [] as countries\"\n        [formControl]=\"countryControl\"\n        [stringify]=\"stringify\"\n    >\n        Страна\n        <input\n            tuiTextfieldLegacy\n            autocapitalize=\"off\"\n            autocomplete=\"off\"\n            autocorrect=\"off\"\n            autocomplete=\"new-password\"\n        />\n        <tui-data-list-wrapper\n            *tuiDataList\n            [itemContent]=\"stringify | tuiStringifyContent\"\n            [items]=\"countries | tuiFilterByInput\"\n        ></tui-data-list-wrapper>\n    </tui-combo-box>\n    <tui-error\n        [formControl]=\"countryControl\"\n        [error]=\"[] | tuiFieldError | async\"\n    ></tui-error>\n</label>\n<div class=\"grid items-start gap-4 sm:grid-cols-2\">\n    <label tuiLabel>\n        Регион\n        <tui-combo-box\n            *tuiLet=\"(regions$ | async) ?? [] as regions\"\n            [formControl]=\"regionControl\"\n            [stringify]=\"stringify\"\n        >\n            Регион\n            <input\n                tuiTextfieldLegacy\n                autocapitalize=\"off\"\n                autocomplete=\"off\"\n                autocorrect=\"off\"\n                autocomplete=\"new-password\"\n            />\n            <tui-data-list-wrapper\n                *tuiDataList\n                [itemContent]=\"stringify | tuiStringifyContent\"\n                [items]=\"regions | tuiFilterByInput\"\n            ></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error\n            [formControl]=\"regionControl\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n    <label tuiLabel>\n        Город\n        <tui-combo-box\n            *tuiLet=\"(cities$ | async) ?? [] as cities\"\n            [formControl]=\"cityControl\"\n            [stringify]=\"stringify\"\n            (ngModelChange)=\"onSelectedCity($event)\"\n        >\n            Город\n            <input\n                tuiTextfieldLegacy\n                autocapitalize=\"off\"\n                autocomplete=\"off\"\n                autocorrect=\"off\"\n                autocomplete=\"new-password\"\n            />\n            <tui-data-list-wrapper\n                *tuiDataList\n                [itemContent]=\"stringify | tuiStringifyContent\"\n                [items]=\"cities | tuiFilterByInput\"\n            ></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error\n            [formControl]=\"cityIdControl\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n        <tui-error\n            *ngIf=\"cityIdControl.untouched\"\n            [formControl]=\"cityControl\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n</div>\n"]}
|
|
@@ -78,6 +78,16 @@ export class ScHelpNotificationService {
|
|
|
78
78
|
this.setShowContacts(...contacts);
|
|
79
79
|
});
|
|
80
80
|
}
|
|
81
|
+
/**
|
|
82
|
+
* Показать уведомление по адресу электронной почты.
|
|
83
|
+
*
|
|
84
|
+
* @param email Адрес электронной почты.
|
|
85
|
+
*/
|
|
86
|
+
helpNotificationByEmail(email) {
|
|
87
|
+
this.contactsService.getHelpContacts$({ email }).subscribe((contacts) => {
|
|
88
|
+
this.setShowContacts(...contacts);
|
|
89
|
+
});
|
|
90
|
+
}
|
|
81
91
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScHelpNotificationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
82
92
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScHelpNotificationService, providedIn: 'root' }); }
|
|
83
93
|
}
|
|
@@ -87,4 +97,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
87
97
|
providedIn: 'root',
|
|
88
98
|
}]
|
|
89
99
|
}], ctorParameters: () => [] });
|
|
90
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
100
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-help-notification.service.js","sourceRoot":"","sources":["../../../../projects/client-ui/services/sc-help-notification.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable } from '@angular/core';\nimport { ScContactsService, ScIOrgContact, ScUserService } from '@snabcentr/client-core';\nimport { BehaviorSubject, Observable, of, shareReplay, switchMap } from 'rxjs';\n\nimport { SC_HELP_NOTIFICATION_CLOSE } from '../providers/sc-help-notification-close';\nimport { SC_HELP_NOTIFICATION_LIMIT } from '../providers/sc-help-notification-limit';\n\n/**\n * Сервис для отображения Push-уведомлений с контактами для помощи клиенту.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class ScHelpNotificationService {\n    /**\n     * Список контактов для показа в уведомлении.\n     */\n    private readonly contactsShow: BehaviorSubject<ScIOrgContact[]> = new BehaviorSubject<ScIOrgContact[]>([]);\n\n    /**\n     * Сервис для работы с контактными лицами.\n     */\n    private readonly contactsService = inject(ScContactsService);\n\n    /**\n     * Максимальное количество отображаемых уведомлений.\n     */\n    private readonly notificationLimit = inject(SC_HELP_NOTIFICATION_LIMIT);\n\n    /**\n     * Поток для закрытия уведомлений.\n     */\n    private readonly close$ = inject(SC_HELP_NOTIFICATION_CLOSE);\n\n    /**\n     * Сервис для работы с пользователем.\n     */\n    private readonly userService = inject(ScUserService);\n\n    /**\n     * Список контактов для показа в уведомлении.\n     */\n    public readonly userHelpContacts = this.userService.getUserChange$().pipe(\n        switchMap((user) => (user.isGuest ? of([]) : this.contactsService.getHelpContacts$({ phone: user.contacts.phone.value }))),\n        shareReplay(1)\n    );\n\n    /**\n     * Инициализирует экземпляр класса {@link ScHelpNotificationService}.\n     */\n    public constructor() {\n        this.close$.subscribe(() => {\n            this.closeAllHelpNotification();\n        });\n    }\n\n    /**\n     * Установить список контактов для показа в уведомлении.\n     *\n     * @param contacts Данные контактов.\n     */\n    public setShowContacts(...contacts: ScIOrgContact[]): void {\n        this.contactsShow.next(contacts.slice(0, this.notificationLimit));\n    }\n\n    /**\n     * Возвращает список менеджеров для показа в уведомлении.\n     */\n    public getContactsShow$(): Observable<ScIOrgContact[]> {\n        return this.contactsShow.asObservable();\n    }\n\n    /**\n     * Закрывает уведомление по номеру телефона.\n     *\n     * @param contact Контакт.\n     */\n    public closeHelpNotification(contact: ScIOrgContact): void {\n        this.contactsShow.next(this.contactsShow.getValue().filter((c) => c.id !== contact.id));\n    }\n\n    /**\n     * Закрывает все уведомления.\n     */\n    public closeAllHelpNotification(): void {\n        this.contactsShow.next([]);\n    }\n\n    /**\n     * Показать уведомление по номеру телефона.\n     *\n     * @param phone Номер телефона.\n     */\n    public helpNotificationByPhone(phone: string): void {\n        this.contactsService.getHelpContacts$({ phone }).subscribe((contacts) => {\n            this.setShowContacts(...contacts);\n        });\n    }\n\n    /**\n     * Показать уведомление по адресу электронной почты.\n     *\n     * @param email Адрес электронной почты.\n     */\n    public helpNotificationByEmail(email: string): void {\n        this.contactsService.getHelpContacts$({ email }).subscribe((contacts) => {\n            this.setShowContacts(...contacts);\n        });\n    }\n}\n"]}
|
|
@@ -138,6 +138,16 @@ class ScHelpNotificationService {
|
|
|
138
138
|
this.setShowContacts(...contacts);
|
|
139
139
|
});
|
|
140
140
|
}
|
|
141
|
+
/**
|
|
142
|
+
* Показать уведомление по адресу электронной почты.
|
|
143
|
+
*
|
|
144
|
+
* @param email Адрес электронной почты.
|
|
145
|
+
*/
|
|
146
|
+
helpNotificationByEmail(email) {
|
|
147
|
+
this.contactsService.getHelpContacts$({ email }).subscribe((contacts) => {
|
|
148
|
+
this.setShowContacts(...contacts);
|
|
149
|
+
});
|
|
150
|
+
}
|
|
141
151
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScHelpNotificationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
142
152
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScHelpNotificationService, providedIn: 'root' }); }
|
|
143
153
|
}
|
|
@@ -1621,6 +1631,10 @@ class ScSignInFormByPhoneComponent {
|
|
|
1621
1631
|
* {@link Subject} события отправки формы.
|
|
1622
1632
|
*/
|
|
1623
1633
|
this.onSubmit = new Subject();
|
|
1634
|
+
/**
|
|
1635
|
+
* Сервис для отображения Push-уведомлений с контактами для помощи клиенту.
|
|
1636
|
+
*/
|
|
1637
|
+
this.helpNotificationService = inject(ScHelpNotificationService);
|
|
1624
1638
|
/**
|
|
1625
1639
|
* {@link Observable} запроса данных аутентификации.
|
|
1626
1640
|
*/
|
|
@@ -1633,6 +1647,9 @@ class ScSignInFormByPhoneComponent {
|
|
|
1633
1647
|
this.form.get(key)?.setErrors({ serverResponse: errors[key] });
|
|
1634
1648
|
}
|
|
1635
1649
|
}
|
|
1650
|
+
if (message.includes('деактивирован')) {
|
|
1651
|
+
this.helpNotificationService.helpNotificationByPhone(value.phone);
|
|
1652
|
+
}
|
|
1636
1653
|
if (!errors && message) {
|
|
1637
1654
|
this.form.setErrors({ serverResponse: [message] });
|
|
1638
1655
|
}
|
|
@@ -1675,6 +1692,10 @@ class ScSignInFormByEmailComponent {
|
|
|
1675
1692
|
* {@link Subject} события отправки формы.
|
|
1676
1693
|
*/
|
|
1677
1694
|
this.onSubmit = new Subject();
|
|
1695
|
+
/**
|
|
1696
|
+
* Сервис для отображения Push-уведомлений с контактами для помощи клиенту.
|
|
1697
|
+
*/
|
|
1698
|
+
this.helpNotificationService = inject(ScHelpNotificationService);
|
|
1678
1699
|
/**
|
|
1679
1700
|
* {@link Observable} запроса данных аутентификации.
|
|
1680
1701
|
*/
|
|
@@ -1687,6 +1708,9 @@ class ScSignInFormByEmailComponent {
|
|
|
1687
1708
|
this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });
|
|
1688
1709
|
}
|
|
1689
1710
|
}
|
|
1711
|
+
if (message.includes('деактивирован')) {
|
|
1712
|
+
this.helpNotificationService.helpNotificationByEmail(value.login);
|
|
1713
|
+
}
|
|
1690
1714
|
if (!errors && message) {
|
|
1691
1715
|
this.formByEmail.setErrors({ serverResponse: [message] });
|
|
1692
1716
|
}
|
|
@@ -1962,10 +1986,12 @@ class ScAddressesSelectionFieldComponent {
|
|
|
1962
1986
|
.getCityById$(Number(this.cityIdControl.value))
|
|
1963
1987
|
.pipe(takeUntilDestroyed(this.destroyRef))
|
|
1964
1988
|
.subscribe((city) => {
|
|
1965
|
-
this.countryControl.patchValue(city.region.country);
|
|
1966
|
-
this.regionControl.patchValue(city.region);
|
|
1967
|
-
this.cityControl.patchValue(city);
|
|
1968
1989
|
this.canSelectCountry.set(city.region.country.id !== this.russiaCountryId);
|
|
1990
|
+
setTimeout(() => {
|
|
1991
|
+
this.countryControl.patchValue(city.region.country);
|
|
1992
|
+
this.regionControl.patchValue(city.region);
|
|
1993
|
+
this.cityControl.patchValue(city);
|
|
1994
|
+
});
|
|
1969
1995
|
});
|
|
1970
1996
|
}
|
|
1971
1997
|
this.countryControl.valueChanges.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((country) => {
|