@snabcentr/client-ui 3.35.0 → 3.37.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,3 +1,4 @@
1
+ import { HttpErrorResponse } from '@angular/common/http';
1
2
  import { OnInit, Signal } from '@angular/core';
2
3
  import { FormControl, FormGroup } from '@angular/forms';
3
4
  import { ScCatalogService, ScISuggestionType, ScIWarehouse, ScProduct, ScUser } from '@snabcentr/client-core';
@@ -15,6 +16,7 @@ export declare class ScNotifyWhenInStockDialogComponent implements OnInit {
15
16
  protected readonly context: TuiDialogContext<void, {
16
17
  product: ScProduct;
17
18
  onClickOfferHandler: () => void;
19
+ onErrorHandler: (error: HttpErrorResponse) => void;
18
20
  }>;
19
21
  /**
20
22
  * Товар.
@@ -27,7 +27,7 @@ export class ScSignInFormByEmailComponent {
27
27
  * Группа полей ввода для формы «Вход на сайт».
28
28
  */
29
29
  this.formByEmail = new FormGroup({
30
- login: new FormControl(null, [Validators.required, Validators.email]),
30
+ username: new FormControl(null, [Validators.required, Validators.email]),
31
31
  password: new FormControl(null, Validators.required),
32
32
  });
33
33
  /**
@@ -37,17 +37,31 @@ export class ScSignInFormByEmailComponent {
37
37
  /**
38
38
  * {@link Observable} запроса данных аутентификации.
39
39
  */
40
- this.emailRequest$ = this.onSubmit.pipe(filter(() => this.formByEmail.valid), map(() => this.formByEmail.value), switchMap((value) => this.authService.getSignIn$(value).pipe(catchError((error) => {
40
+ this.emailRequest$ = this.onSubmit.pipe(filter(() => this.formByEmail.valid), map(() => this.formByEmail.value), switchMap((value) => this.authService.getSignIn$(value).pipe(
41
+ // eslint-disable-next-line sonarjs/cognitive-complexity
42
+ catchError((error) => {
41
43
  if (error instanceof HttpErrorResponse) {
42
- const { errors, message } = error.error;
43
- for (const key in errors) {
44
- if (Object.hasOwn(errors, key)) {
45
- // eslint-disable-next-line security/detect-object-injection
46
- this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });
44
+ if ('error' in error.error) {
45
+ const errorResponse = error.error;
46
+ if (errorResponse.error.includes('invalid_credentials')) {
47
+ this.formByEmail.setErrors({ serverResponse: ['Пользователь с указанными учетными данными не найден. Проверьте правильность ввода.'] });
48
+ }
49
+ else {
50
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
51
+ this.formByEmail.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });
47
52
  }
48
53
  }
49
- if (!errors && message) {
50
- this.formByEmail.setErrors({ serverResponse: [message] });
54
+ else {
55
+ const { errors, message } = error.error;
56
+ for (const key in errors) {
57
+ if (Object.hasOwn(errors, key)) {
58
+ // eslint-disable-next-line security/detect-object-injection
59
+ this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });
60
+ }
61
+ }
62
+ if (!errors && message) {
63
+ this.formByEmail.setErrors({ serverResponse: [message] });
64
+ }
51
65
  }
52
66
  }
53
67
  return of({});
@@ -62,12 +76,12 @@ export class ScSignInFormByEmailComponent {
62
76
  this.forgotPassword = new EventEmitter();
63
77
  }
64
78
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScSignInFormByEmailComponent, deps: [{ token: i1.ScAuthService }], target: i0.ɵɵFactoryTarget.Component }); }
65
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScSignInFormByEmailComponent, selector: "sc-sign-in-form-by-email", outputs: { forgotPassword: "forgotPassword" }, ngImport: i0, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\n <label tuiLabel\n >\u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"login\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <input\n tuiTextfieldLegacy\n autocomplete=\"email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"login\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <label tuiLabel\n >\u041F\u0430\u0440\u043E\u043B\u044C\n <tui-input-password formControlName=\"password\">\n \u041F\u0430\u0440\u043E\u043B\u044C\n <input\n tuiTextfieldLegacy\n autocomplete=\"current-password\"\n />\n </tui-input-password>\n <tui-error\n formControlName=\"password\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n </div>\n <div class=\"mb-4 flex flex-col items-center gap-4\">\n <a\n tuiLink\n [pseudo]=\"true\"\n (click)=\"forgotPassword.emit()\"\n class=\"text-base\"\n >\u0417\u0430\u0431\u044B\u043B\u0438 \u043F\u0430\u0440\u043E\u043B\u044C?</a\n >\n <button\n tuiButton\n type=\"submit\"\n [loading]=\"!!(loadingEmailAuth$ | async)\"\n [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n </div>\n</form>\n", dependencies: [{ kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i3.TuiInputPasswordComponent, selector: "tui-input-password" }, { kind: "directive", type: i3.TuiInputPasswordDirective, selector: "tui-input-password" }, { kind: "component", type: i4.TuiTextfieldComponent, selector: "input[tuiTextfieldLegacy], textarea[tuiTextfieldLegacy]" }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "directive", type: i5.TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: i5.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i5.TuiLabel, selector: "label[tuiLabel]" }, { kind: "component", type: i5.TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "component", type: i6.TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: i7.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.TuiFieldErrorPipe, name: "tuiFieldError" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
79
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScSignInFormByEmailComponent, selector: "sc-sign-in-form-by-email", outputs: { forgotPassword: "forgotPassword" }, ngImport: i0, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\n <label tuiLabel\n >\u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"username\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <input\n tuiTextfieldLegacy\n autocomplete=\"email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"username\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <label tuiLabel\n >\u041F\u0430\u0440\u043E\u043B\u044C\n <tui-input-password formControlName=\"password\">\n \u041F\u0430\u0440\u043E\u043B\u044C\n <input\n tuiTextfieldLegacy\n autocomplete=\"current-password\"\n />\n </tui-input-password>\n <tui-error\n formControlName=\"password\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n </div>\n <div class=\"mb-4 flex flex-col items-center gap-4\">\n <a\n tuiLink\n [pseudo]=\"true\"\n (click)=\"forgotPassword.emit()\"\n class=\"text-base\"\n >\u0417\u0430\u0431\u044B\u043B\u0438 \u043F\u0430\u0440\u043E\u043B\u044C?</a\n >\n <button\n tuiButton\n type=\"submit\"\n [loading]=\"!!(loadingEmailAuth$ | async)\"\n [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n </div>\n</form>\n", dependencies: [{ kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i3.TuiInputPasswordComponent, selector: "tui-input-password" }, { kind: "directive", type: i3.TuiInputPasswordDirective, selector: "tui-input-password" }, { kind: "component", type: i4.TuiTextfieldComponent, selector: "input[tuiTextfieldLegacy], textarea[tuiTextfieldLegacy]" }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "directive", type: i5.TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: i5.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i5.TuiLabel, selector: "label[tuiLabel]" }, { kind: "component", type: i5.TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "component", type: i6.TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: i7.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.TuiFieldErrorPipe, name: "tuiFieldError" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
66
80
  }
67
81
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScSignInFormByEmailComponent, decorators: [{
68
82
  type: Component,
69
- args: [{ selector: 'sc-sign-in-form-by-email', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\n <label tuiLabel\n >\u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"login\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <input\n tuiTextfieldLegacy\n autocomplete=\"email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"login\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <label tuiLabel\n >\u041F\u0430\u0440\u043E\u043B\u044C\n <tui-input-password formControlName=\"password\">\n \u041F\u0430\u0440\u043E\u043B\u044C\n <input\n tuiTextfieldLegacy\n autocomplete=\"current-password\"\n />\n </tui-input-password>\n <tui-error\n formControlName=\"password\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n </div>\n <div class=\"mb-4 flex flex-col items-center gap-4\">\n <a\n tuiLink\n [pseudo]=\"true\"\n (click)=\"forgotPassword.emit()\"\n class=\"text-base\"\n >\u0417\u0430\u0431\u044B\u043B\u0438 \u043F\u0430\u0440\u043E\u043B\u044C?</a\n >\n <button\n tuiButton\n type=\"submit\"\n [loading]=\"!!(loadingEmailAuth$ | async)\"\n [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n </div>\n</form>\n" }]
83
+ args: [{ selector: 'sc-sign-in-form-by-email', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"formByEmail\"\n (ngSubmit)=\"onSubmit.next()\"\n>\n <div class=\"mb-8 flex flex-col gap-4\">\n <label tuiLabel\n >\u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"username\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <input\n tuiTextfieldLegacy\n autocomplete=\"email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"username\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <label tuiLabel\n >\u041F\u0430\u0440\u043E\u043B\u044C\n <tui-input-password formControlName=\"password\">\n \u041F\u0430\u0440\u043E\u043B\u044C\n <input\n tuiTextfieldLegacy\n autocomplete=\"current-password\"\n />\n </tui-input-password>\n <tui-error\n formControlName=\"password\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n </div>\n <div class=\"mb-4 flex flex-col items-center gap-4\">\n <a\n tuiLink\n [pseudo]=\"true\"\n (click)=\"forgotPassword.emit()\"\n class=\"text-base\"\n >\u0417\u0430\u0431\u044B\u043B\u0438 \u043F\u0430\u0440\u043E\u043B\u044C?</a\n >\n <button\n tuiButton\n type=\"submit\"\n [loading]=\"!!(loadingEmailAuth$ | async)\"\n [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n </div>\n</form>\n" }]
70
84
  }], ctorParameters: () => [{ type: i1.ScAuthService }], propDecorators: { forgotPassword: [{
71
85
  type: Output
72
86
  }] } });
73
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-email.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.html"],"names":[],"mappings":"AAAA,2EAA2E;AAE3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACzF,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEpE,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;;;;;;;;;AAIrG;;GAEG;AAMH,MAAM,OAAO,4BAA4B;IAwDrC;;;;OAIG;IACH,YAAoC,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QA5D9D;;WAEG;QACa,gBAAW,GAAc,IAAI,SAAS,CAAC;YACnD,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC;YACpF,QAAQ,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC;SACtE,CAAC,CAAC;QAEH;;WAEG;QACa,aAAQ,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAE9D;;WAEG;QACc,kBAAa,GAAkC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAC9E,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EACpC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,KAAe,CAAC,EAC3C,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAChB,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CACnC,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAE5D,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;oBACvB,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,CAAC;wBAC7B,4DAA4D;wBAC5D,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,EAAE,cAAc,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;oBAC1E,CAAC;gBACL,CAAC;gBAED,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE,CAAC;oBACrB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBAC9D,CAAC;YACL,CAAC;YAED,OAAO,EAAE,CAAC,EAAgB,CAAC,CAAC;QAChC,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CACJ,EACD,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACa,sBAAiB,GAAwB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;QAElG;;WAEG;QACuB,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAOvB,CAAC;+GA7DzD,4BAA4B;mGAA5B,4BAA4B,+GCnBzC,qjEAsDA;;4FDnCa,4BAA4B;kBALxC,SAAS;+BACI,0BAA0B,mBAEnB,uBAAuB,CAAC,MAAM;kFAwDrB,cAAc;sBAAvC,MAAM","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Output } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, ILogin, ScAuthService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по адресу электронной почты и паролю.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-email',\n    templateUrl: './sc-sign-in-form-by-email.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByEmailComponent {\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly formByEmail: FormGroup = new FormGroup({\n        login: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        password: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly emailRequest$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        filter(() => this.formByEmail.valid),\n        map(() => this.formByEmail.value as ILogin),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.formByEmail.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по email.\n     */\n    public readonly loadingEmailAuth$: Observable<boolean> = this.emailRequest$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Сигнал нажатия на кнопку \"Забыли пароль\".\n     */\n    @Output() public readonly forgotPassword: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByEmailComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"formByEmail\"\n    (ngSubmit)=\"onSubmit.next()\"\n>\n    <div class=\"mb-8 flex flex-col gap-4\">\n        <label tuiLabel\n            >Адрес электронной почты\n            <tui-input formControlName=\"login\">\n                Адрес электронной почты\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"email\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"login\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <label tuiLabel\n            >Пароль\n            <tui-input-password formControlName=\"password\">\n                Пароль\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"current-password\"\n                />\n            </tui-input-password>\n            <tui-error\n                formControlName=\"password\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </div>\n    <div class=\"mb-4 flex flex-col items-center gap-4\">\n        <a\n            tuiLink\n            [pseudo]=\"true\"\n            (click)=\"forgotPassword.emit()\"\n            class=\"text-base\"\n            >Забыли пароль?</a\n        >\n        <button\n            tuiButton\n            type=\"submit\"\n            [loading]=\"!!(loadingEmailAuth$ | async)\"\n            [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n            iconStart=\"@tui.sc.circle-arrow-in-right\"\n        >\n            Войти\n        </button>\n    </div>\n</form>\n"]}
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-email.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Output } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, ILogin, ScAuthService, ScIKeycloakErrorResponse } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по адресу электронной почты и паролю.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-email',\n    templateUrl: './sc-sign-in-form-by-email.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByEmailComponent {\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly formByEmail: FormGroup = new FormGroup({\n        username: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        password: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly emailRequest$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        filter(() => this.formByEmail.valid),\n        map(() => this.formByEmail.value as ILogin),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                // eslint-disable-next-line sonarjs/cognitive-complexity\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        if ('error' in error.error) {\n                            const errorResponse = error.error as ScIKeycloakErrorResponse;\n\n                            if (errorResponse.error.includes('invalid_credentials')) {\n                                this.formByEmail.setErrors({ serverResponse: ['Пользователь с указанными учетными данными не найден. Проверьте правильность ввода.'] });\n                            } else {\n                                // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                                this.formByEmail.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });\n                            }\n                        } else {\n                            const { errors, message } = error.error as ApiErrorResponse;\n\n                            for (const key in errors) {\n                                if (Object.hasOwn(errors, key)) {\n                                    // eslint-disable-next-line security/detect-object-injection\n                                    this.formByEmail.get(key)?.setErrors({ serverResponse: errors[key] });\n                                }\n                            }\n\n                            if (!errors && message) {\n                                this.formByEmail.setErrors({ serverResponse: [message] });\n                            }\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по email.\n     */\n    public readonly loadingEmailAuth$: Observable<boolean> = this.emailRequest$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Сигнал нажатия на кнопку \"Забыли пароль\".\n     */\n    @Output() public readonly forgotPassword: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByEmailComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"formByEmail\"\n    (ngSubmit)=\"onSubmit.next()\"\n>\n    <div class=\"mb-8 flex flex-col gap-4\">\n        <label tuiLabel\n            >Адрес электронной почты\n            <tui-input formControlName=\"username\">\n                Адрес электронной почты\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"email\"\n                />\n            </tui-input>\n            <tui-error\n                formControlName=\"username\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <label tuiLabel\n            >Пароль\n            <tui-input-password formControlName=\"password\">\n                Пароль\n                <input\n                    tuiTextfieldLegacy\n                    autocomplete=\"current-password\"\n                />\n            </tui-input-password>\n            <tui-error\n                formControlName=\"password\"\n                [error]=\"[] | tuiFieldError | async\"\n            ></tui-error>\n        </label>\n        <tui-error [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </div>\n    <div class=\"mb-4 flex flex-col items-center gap-4\">\n        <a\n            tuiLink\n            [pseudo]=\"true\"\n            (click)=\"forgotPassword.emit()\"\n            class=\"text-base\"\n            >Забыли пароль?</a\n        >\n        <button\n            tuiButton\n            type=\"submit\"\n            [loading]=\"!!(loadingEmailAuth$ | async)\"\n            [disabled]=\"formByEmail.invalid || !!(loadingEmailAuth$ | async)\"\n            iconStart=\"@tui.sc.circle-arrow-in-right\"\n        >\n            Войти\n        </button>\n    </div>\n</form>\n"]}
@@ -42,17 +42,31 @@ export class ScSignInFormByPhoneComponent {
42
42
  /**
43
43
  * {@link Observable} запроса данных аутентификации.
44
44
  */
45
- this.request$ = this.onSubmit.pipe(map(() => this.form.value), filter((value) => this.form.valid), switchMap((value) => this.authService.getSignIn$(value).pipe(catchError((error) => {
45
+ this.request$ = this.onSubmit.pipe(map(() => this.form.value), filter((value) => this.form.valid), switchMap((value) => this.authService.getSignIn$(value).pipe(
46
+ // eslint-disable-next-line sonarjs/cognitive-complexity
47
+ catchError((error) => {
46
48
  if (error instanceof HttpErrorResponse) {
47
- const { errors, message } = error.error;
48
- for (const key in errors) {
49
- if (Object.hasOwn(errors, key)) {
50
- // eslint-disable-next-line security/detect-object-injection
51
- this.form.get(key)?.setErrors({ serverResponse: errors[key] });
49
+ if ('error' in error.error) {
50
+ const errorResponse = error.error;
51
+ if (errorResponse.error.includes('invalid_credentials')) {
52
+ this.form.setErrors({ serverResponse: ['Неверный код подтверждения.'] });
53
+ }
54
+ else {
55
+ // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
56
+ this.form.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });
52
57
  }
53
58
  }
54
- if (!errors && message) {
55
- this.form.setErrors({ serverResponse: [message] });
59
+ else {
60
+ const { errors, message } = error.error;
61
+ for (const key in errors) {
62
+ if (Object.hasOwn(errors, key)) {
63
+ // eslint-disable-next-line security/detect-object-injection
64
+ this.form.get(key)?.setErrors({ serverResponse: errors[key] });
65
+ }
66
+ }
67
+ if (!errors && message) {
68
+ this.form.setErrors({ serverResponse: [message] });
69
+ }
56
70
  }
57
71
  }
58
72
  return of({});
@@ -69,4 +83,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
69
83
  type: Component,
70
84
  args: [{ selector: 'sc-sign-in-form-by-phone', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"form\"\n *tuiLet=\"!!(loading$ | async) as loading\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"mb-4 flex flex-col items-center gap-4\"\n>\n <sc-verification-phone-check-form\n [(haveCode)]=\"haveCode\"\n [shouldBeBusy]=\"true\"\n [shouldBeConfirmed]=\"true\"\n class=\"w-full\"\n />\n <tui-error\n [error]=\"[] | tuiFieldError | async\"\n class=\"self-center\"\n />\n <button\n *ngIf=\"haveCode\"\n tuiButton\n type=\"submit\"\n [loading]=\"loading\"\n [disabled]=\"form.invalid || loading\"\n iconStart=\"@tui.sc.circle-arrow-in-right\"\n >\n \u0412\u043E\u0439\u0442\u0438\n </button>\n</form>\n" }]
71
85
  }], ctorParameters: () => [{ type: i1.ScAuthService }] });
72
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-phone.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.html"],"names":[],"mappings":"AAAA,2EAA2E;AAE3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEpE,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AAErG,OAAO,EAAE,cAAc,EAAE,MAAM,wCAAwC,CAAC;;;;;;;;;AAGxE;;GAEG;AAMH,MAAM,OAAO,4BAA4B;IAwDrC;;;;OAIG;IACH,YAAoC,WAA0B;QAA1B,gBAAW,GAAX,WAAW,CAAe;QA5D9D;;WAEG;QACI,aAAQ,GAAY,KAAK,CAAC;QAEjC;;WAEG;QACa,SAAI,GAAG,IAAI,SAAS,CAAC;YACjC,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,cAAc,EAAE,CAAC,CAAC;YACpF,gBAAgB,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;SACzG,CAAC,CAAC;QAEH;;WAEG;QACa,aAAQ,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAE9D;;WAEG;QACc,aAAQ,GAAkC,IAAI,CAAC,QAAQ,CAAC,IAAI,CACzE,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAC1B,MAAM,CAAC,CAAC,KAAK,EAAwB,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EACxD,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAChB,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CACnC,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAE5D,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;oBACvB,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,CAAC;wBAC7B,4DAA4D;wBAC5D,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,EAAE,cAAc,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;oBACnE,CAAC;gBACL,CAAC;gBAED,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE,CAAC;oBACrB,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACvD,CAAC;YACL,CAAC;YAED,OAAO,EAAE,CAAC,EAAgB,CAAC,CAAC;QAChC,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CACJ,EACD,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACa,aAAQ,GAAwB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;IAOnB,CAAC;+GA7DzD,4BAA4B;mGAA5B,4BAA4B,gECpBzC,0uBA2BA;;4FDPa,4BAA4B;kBALxC,SAAS;+BACI,0BAA0B,mBAEnB,uBAAuB,CAAC,MAAM","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, IPhoneLogin, ScAuthService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { phoneValidator } from '../../../validators/sc-phone-validator';\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по номеру телефона и коду подтверждения.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-phone',\n    templateUrl: './sc-sign-in-form-by-phone.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByPhoneComponent {\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public haveCode: boolean = false;\n\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly form = new FormGroup({\n        phone: new FormControl<string | null>(null, [Validators.required, phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly request$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is IPhoneLogin => this.form.valid),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByPhoneComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"form\"\n    *tuiLet=\"!!(loading$ | async) as loading\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"mb-4 flex flex-col items-center gap-4\"\n>\n    <sc-verification-phone-check-form\n        [(haveCode)]=\"haveCode\"\n        [shouldBeBusy]=\"true\"\n        [shouldBeConfirmed]=\"true\"\n        class=\"w-full\"\n    />\n    <tui-error\n        [error]=\"[] | tuiFieldError | async\"\n        class=\"self-center\"\n    />\n    <button\n        *ngIf=\"haveCode\"\n        tuiButton\n        type=\"submit\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid || loading\"\n        iconStart=\"@tui.sc.circle-arrow-in-right\"\n    >\n        Войти\n    </button>\n</form>\n"]}
86
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-sign-in-form-by-phone.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.ts","../../../../../../projects/client-ui/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { IAuthToken, IPhoneLogin, ScAuthService, ScIKeycloakErrorResponse } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap } from 'rxjs';\n\nimport { phoneValidator } from '../../../validators/sc-phone-validator';\nimport { ApiErrorResponse } from '../../interfaces/api-error-response';\n\n/**\n * Компонент аутентификации по номеру телефона и коду подтверждения.\n */\n@Component({\n    selector: 'sc-sign-in-form-by-phone',\n    templateUrl: './sc-sign-in-form-by-phone.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSignInFormByPhoneComponent {\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public haveCode: boolean = false;\n\n    /**\n     * Группа полей ввода для формы «Вход на сайт».\n     */\n    public readonly form = new FormGroup({\n        phone: new FormControl<string | null>(null, [Validators.required, phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных аутентификации.\n     */\n    private readonly request$: Observable<IAuthToken | null> = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is IPhoneLogin => this.form.valid),\n        switchMap((value) =>\n            this.authService.getSignIn$(value).pipe(\n                // eslint-disable-next-line sonarjs/cognitive-complexity\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        if ('error' in error.error) {\n                            const errorResponse = error.error as ScIKeycloakErrorResponse;\n\n                            if (errorResponse.error.includes('invalid_credentials')) {\n                                this.form.setErrors({ serverResponse: ['Неверный код подтверждения.'] });\n                            } else {\n                                // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                                this.form.setErrors({ serverResponse: [errorResponse.errorDescription ?? errorResponse.error] });\n                            }\n                        } else {\n                            const { errors, message } = error.error as ApiErrorResponse;\n\n                            for (const key in errors) {\n                                if (Object.hasOwn(errors, key)) {\n                                    // eslint-disable-next-line security/detect-object-injection\n                                    this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                                }\n                            }\n\n                            if (!errors && message) {\n                                this.form.setErrors({ serverResponse: [message] });\n                            }\n                        }\n                    }\n\n                    return of({} as IAuthToken);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSignInFormByPhoneComponent}.\n     *\n     * @param authService Сервис аутентификации.\n     */\n    public constructor(private readonly authService: ScAuthService) {}\n}\n","<form\n    [formGroup]=\"form\"\n    *tuiLet=\"!!(loading$ | async) as loading\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"mb-4 flex flex-col items-center gap-4\"\n>\n    <sc-verification-phone-check-form\n        [(haveCode)]=\"haveCode\"\n        [shouldBeBusy]=\"true\"\n        [shouldBeConfirmed]=\"true\"\n        class=\"w-full\"\n    />\n    <tui-error\n        [error]=\"[] | tuiFieldError | async\"\n        class=\"self-center\"\n    />\n    <button\n        *ngIf=\"haveCode\"\n        tuiButton\n        type=\"submit\"\n        [loading]=\"loading\"\n        [disabled]=\"form.invalid || loading\"\n        iconStart=\"@tui.sc.circle-arrow-in-right\"\n    >\n        Войти\n    </button>\n</form>\n"]}
@@ -1,5 +1,5 @@
1
1
  /* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */
2
- import { AsyncPipe, NgIf, NgTemplateOutlet } from '@angular/common';
2
+ import { AsyncPipe, NgIf } from '@angular/common';
3
3
  import { HttpErrorResponse } from '@angular/common/http';
4
4
  import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
5
5
  import { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';
@@ -101,6 +101,7 @@ export class ScNotifyWhenInStockDialogComponent {
101
101
  else if (message) {
102
102
  this.form.setErrors({ serverResponse: [message] });
103
103
  }
104
+ this.context.data.onErrorHandler(error);
104
105
  }
105
106
  return of(undefined);
106
107
  }), startWith(null))), share());
@@ -165,7 +166,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
165
166
  TuiCardLarge,
166
167
  TuiIcon,
167
168
  TuiTooltip,
168
- NgTemplateOutlet,
169
169
  ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit$.next()\"\n class=\"flex flex-col gap-6\"\n>\n @let isPhoneVisible = requiredFields.includes('phone');\n @let isEmailVisible = requiredFields.includes('email');\n @let phoneTouched = form.get('phone')?.touched;\n @let emailTouched = form.get('email')?.touched;\n @let isShowError = form.errors?.['atLeastOneRequired'] && ((isPhoneVisible && phoneTouched) || (isEmailVisible && emailTouched));\n @let user = user$ | async;\n @let userPhone = user?.contacts?.phone;\n @let isShowCodeFields = !!form.controls.phone.value && (user?.isGuest || userPhone?.value !== form.controls.phone.value || !userPhone?.isApproved);\n\n <div\n *ngIf=\"user\"\n class=\"flex flex-col gap-3\"\n >\n <label tuiLabel>\n <div\n [tuiAppearance]=\"isPhoneVisible && isEmailVisible ? 'outline-grayscale' : 'flat'\"\n [tuiAppearanceMode]=\"isShowError ? 'invalid' : null\"\n tuiCardLarge\n class=\"flex flex-col !gap-1\"\n [style.padding]=\"requiredFields.length <= 1 ? '0' : undefined\"\n [style.border-radius]=\"requiredFields.length <= 1 ? '0' : undefined\"\n >\n <sc-verification-phone-check-form\n *ngIf=\"isPhoneVisible\"\n [required]=\"!form.controls.email.value\"\n [showCodeFields]=\"isShowCodeFields\"\n [pseudoInvalid]=\"isShowError\"\n />\n <div\n *ngIf=\"isPhoneVisible && isEmailVisible\"\n class=\"tui-island__paragraph flex items-center gap-3\"\n [style.color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n >\n <div\n class=\"h-px flex-1\"\n [style.background-color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n ></div>\n <div class=\"text-body-s font-bold\">\n \u0418\u043B\u0438(\u0438)\n <tui-icon\n [appearance]=\"isShowError ? 'negative' : 'grayscale'\"\n tuiHintDirection=\"right\"\n tuiHintAppearance=\"dark\"\n tuiTooltip=\"\u041F\u043E\u0436\u0430\u043B\u0443\u0439\u0441\u0442\u0430, \u0443\u043A\u0430\u0436\u0438\u0442\u0435 \u043D\u043E\u043C\u0435\u0440 \u0442\u0435\u043B\u0435\u0444\u043E\u043D\u0430 \u0438\u043B\u0438(\u0438) \u0430\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\"\n />\n </div>\n <div\n class=\"h-px flex-1\"\n [style.background-color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n ></div>\n </div>\n <label\n tuiLabel\n *ngIf=\"isEmailVisible\"\n >\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input\n [pseudoInvalid]=\"isShowError\"\n formControlName=\"email\"\n >\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n </div>\n <tui-error\n *ngIf=\"isShowError\"\n error=\"\u041F\u043E\u0436\u0430\u043B\u0443\u0439\u0441\u0442\u0430, \u0443\u043A\u0430\u0436\u0438\u0442\u0435 \u043D\u043E\u043C\u0435\u0440 \u0442\u0435\u043B\u0435\u0444\u043E\u043D\u0430 \u0438(\u0438\u043B\u0438) \u0430\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B \u0434\u043B\u044F \u0443\u0432\u0435\u0434\u043E\u043C\u043B\u0435\u043D\u0438\u044F \u043E \u043F\u043E\u0441\u0442\u0443\u043F\u043B\u0435\u043D\u0438\u0438 \u0442\u043E\u0432\u0430\u0440\u0430\"\n />\n </label>\n <label tuiLabel>\n \u0421\u043A\u043B\u0430\u0434\n <tui-select\n formControlName=\"warehouseId\"\n [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n >\n \u0421\u043A\u043B\u0430\u0434\n <tui-data-list *tuiDataList>\n @for (warehouse of warehouses(); track warehouse.id) {\n @if (!product.isWarehouseStockExist(warehouse.id)) {\n <button\n tuiOption\n [value]=\"warehouse.id\"\n >\n {{ warehouse.name }}\n </button>\n }\n }\n </tui-data-list>\n </tui-select>\n <tui-error\n formControlName=\"warehouseId\"\n [error]=\"[] | tuiFieldError | async\"\n />\n\n <span\n *ngIf=\"isStock()\"\n [style.color]=\"'var(--tui-text-positive)'\"\n >\n \u0422\u043E\u0432\u0430\u0440 \u0438\u043C\u0435\u0435\u0442\u0441\u044F \u0432 \u043D\u0430\u043B\u0438\u0447\u0438\u0438 \u043D\u0430 \u0432\u044B\u0431\u0440\u0430\u043D\u043D\u043E\u043C \u0441\u043A\u043B\u0430\u0434\u0435\n </span>\n </label>\n <ng-template #selectLoading> <tui-loader /> </ng-template>\n </div>\n\n <div class=\"flex flex-wrap items-center gap-3\">\n <button\n tuiButton\n type=\"submit\"\n [disabled]=\"form.invalid || isStock() || (isShowCodeFields && !form.controls.verificationCode.value && form.controls.phone.valid)\"\n [loading]=\"loading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n >\n \u041E\u0442\u043F\u0440\u0430\u0432\u0438\u0442\u044C\n </button>\n <div class=\"grow basis-60 text-body-s\">\n \u041D\u0430\u0436\u0438\u043C\u0430\u044F \u043D\u0430 \u043A\u043D\u043E\u043F\u043A\u0443, \u0432\u044B \u0441\u043E\u0433\u043B\u0430\u0448\u0430\u0435\u0442\u0435\u0441\u044C \u0441\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"context.data.onClickOfferHandler()\"\n >\n \u041F\u043E\u043B\u0438\u0442\u0438\u043A\u043E\u0439 \u043A\u043E\u043D\u0444\u0438\u0434\u0435\u043D\u0446\u0438\u0430\u043B\u044C\u043D\u043E\u0441\u0442\u0438\n </button>\n </div>\n </div>\n\n <tui-error\n *ngIf=\"form.errors && form.errors['serverResponse']\"\n [error]=\"[] | tuiFieldError | async\"\n />\n</form>\n" }]
170
170
  }], ctorParameters: () => [] });
171
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-notify-when-in-stock-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.ts","../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf, NgTemplateOutlet } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, OnInit, Signal } from '@angular/core';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCatalogService,\n    ScConvertersService,\n    ScIProductNotifyWhenInStock,\n    ScISuggestionType,\n    ScIWarehouse,\n    ScProduct,\n    ScUser,\n    ScUserService,\n    ScWarehouseService,\n} from '@snabcentr/client-core';\nimport { TuiContext, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TuiAppearance, TuiButton, TuiDialogContext, TuiError, TuiIcon, TuiLabel, TuiLink, TuiLoader } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFieldErrorPipe, TuiTooltip } from '@taiga-ui/kit';\nimport { TuiCardLarge } from '@taiga-ui/layout';\nimport { TuiInputModule, TuiSelectModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { ScFormFieldsModule } from '../../form-fields/form-fields.module';\nimport { SC_NOTIFY_WHEN_IN_STOCK_REQUIRED_FIELDS } from '../../providers';\nimport { scAtLeastOneRequiredValidator } from '../../validators/sc-at-least-one-required-validator';\nimport { phoneValidator } from '../../validators/sc-phone-validator';\nimport { ScVerificationModule } from '../../verification/sc-verification.module';\n\n/**\n * Компонент формы отправки запроса на уведомление о поступлении товара.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-notify-when-in-stock-dialog',\n    templateUrl: './sc-notify-when-in-stock-dialog.component.html',\n    imports: [\n        ScVerificationModule,\n        FormsModule,\n        ReactiveFormsModule,\n        TuiInputModule,\n        TuiLink,\n        TuiButton,\n        TuiLabel,\n        TuiFieldErrorPipe,\n        TuiError,\n        ScVerificationModule,\n        ScFormFieldsModule,\n        TuiSelectModule,\n        TuiButtonLoading,\n        TuiLoader,\n        AsyncPipe,\n        NgIf,\n        TuiAppearance,\n        TuiCardLarge,\n        TuiIcon,\n        TuiTooltip,\n        NgTemplateOutlet,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScNotifyWhenInStockDialogComponent implements OnInit {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void }> =\n        inject<TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void }>>(POLYMORPHEUS_CONTEXT);\n\n    /**\n     * Товар.\n     */\n    protected readonly product: ScProduct = this.context.data.product;\n\n    /**\n     * Форма для отправки запроса на уведомление о поступлении товара.\n     */\n    protected readonly form = new FormGroup({\n        email: new FormControl<string | null>(null, [Validators.email]),\n        phone: new FormControl<string | null>(null, [phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.minLength(6)]),\n        warehouseId: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * Список складов.\n     */\n    protected readonly warehouses: Signal<ScIWarehouse[] | null> = toSignal(inject(ScWarehouseService).getWarehouses$(), { initialValue: null });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Массив полей, которые должны быть заполнены хотя бы одним значением.\n     */\n    protected readonly requiredFields = inject(SC_NOTIFY_WHEN_IN_STOCK_REQUIRED_FIELDS);\n\n    /**\n     * Сервис для работы с каталогом.\n     */\n    protected readonly catalogService: ScCatalogService = inject(ScCatalogService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /**\n     * {@link Observable} данных о пользователе.\n     */\n    protected readonly user$: Observable<ScUser | null> = inject(ScUserService)\n        .getUserChange$()\n        .pipe(\n            tap((user) => {\n                if (!user.isGuest) {\n                    this.form.patchValue({\n                        phone: user.contacts.phone.value,\n                        email: user.contacts.email.value,\n                    });\n                }\n            })\n        );\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit$: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных уведомления о поступлении товара.\n     */\n    private readonly request$ = this.onSubmit$.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.getRawValue() as ScIProductNotifyWhenInStock),\n        switchMap((value) =>\n            this.catalogService.getProductNotifyWhenInStock$(this.product.id, this.convertersService.removeNullRecursive(value)).pipe(\n                tap(() => {\n                    this.context.completeWith();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        if (errors) {\n                            // eslint-disable-next-line no-shadow\n                            Object.entries(errors).forEach(([key, value]) => {\n                                this.form.get(key)?.setErrors({ serverResponse: value });\n                            });\n                        } else if (message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of(undefined);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * Сигнал изменения состояния загрузки данных.\n     */\n    protected readonly loading: Signal<boolean> = toSignal(this.request$.pipe(map((value) => value === null)), { initialValue: false });\n\n    /**\n     * Инициализирует экземпляр класса {@link ScNotifyWhenInStockDialogComponent}.\n     */\n    public constructor() {\n        this.user$.pipe(takeUntilDestroyed()).subscribe();\n    }\n\n    /**\n     * Инициализация компонента.\n     */\n    public ngOnInit(): void {\n        if (this.requiredFields.length === 1) {\n            this.form.get(this.requiredFields[0])?.setValidators(Validators.required);\n        } else if (this.requiredFields.length > 1) {\n            this.form.setValidators(scAtLeastOneRequiredValidator(this.requiredFields));\n        }\n    }\n\n    /**\n     * Проверяет, является ли товар в наличии.\n     */\n    protected isStock(): boolean {\n        const warehouseId = Number(this.form.get('warehouseId')?.value ?? 0);\n\n        return this.product.isWarehouseStockExist(warehouseId);\n    }\n\n    /**\n     * Преобразует объект в значение, отображаемое в поле ввода.\n     *\n     * @param items Выбранные значения.\n     */\n    protected stringify(items: readonly ScIWarehouse[]): TuiStringHandler<TuiContext<number>> {\n        const itemsMap = new Map(\n            items.map((item) => {\n                return [item.id, item.name] as [number, string];\n            })\n        );\n\n        return ({ $implicit }: TuiContext<number>) => itemsMap.get($implicit) ?? '';\n    }\n}\n","<form\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit$.next()\"\n    class=\"flex flex-col gap-6\"\n>\n    @let isPhoneVisible = requiredFields.includes('phone');\n    @let isEmailVisible = requiredFields.includes('email');\n    @let phoneTouched = form.get('phone')?.touched;\n    @let emailTouched = form.get('email')?.touched;\n    @let isShowError = form.errors?.['atLeastOneRequired'] && ((isPhoneVisible && phoneTouched) || (isEmailVisible && emailTouched));\n    @let user = user$ | async;\n    @let userPhone = user?.contacts?.phone;\n    @let isShowCodeFields = !!form.controls.phone.value && (user?.isGuest || userPhone?.value !== form.controls.phone.value || !userPhone?.isApproved);\n\n    <div\n        *ngIf=\"user\"\n        class=\"flex flex-col gap-3\"\n    >\n        <label tuiLabel>\n            <div\n                [tuiAppearance]=\"isPhoneVisible && isEmailVisible ? 'outline-grayscale' : 'flat'\"\n                [tuiAppearanceMode]=\"isShowError ? 'invalid' : null\"\n                tuiCardLarge\n                class=\"flex flex-col !gap-1\"\n                [style.padding]=\"requiredFields.length <= 1 ? '0' : undefined\"\n                [style.border-radius]=\"requiredFields.length <= 1 ? '0' : undefined\"\n            >\n                <sc-verification-phone-check-form\n                    *ngIf=\"isPhoneVisible\"\n                    [required]=\"!form.controls.email.value\"\n                    [showCodeFields]=\"isShowCodeFields\"\n                    [pseudoInvalid]=\"isShowError\"\n                />\n                <div\n                    *ngIf=\"isPhoneVisible && isEmailVisible\"\n                    class=\"tui-island__paragraph flex items-center gap-3\"\n                    [style.color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n                >\n                    <div\n                        class=\"h-px flex-1\"\n                        [style.background-color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n                    ></div>\n                    <div class=\"text-body-s font-bold\">\n                        Или(и)\n                        <tui-icon\n                            [appearance]=\"isShowError ? 'negative' : 'grayscale'\"\n                            tuiHintDirection=\"right\"\n                            tuiHintAppearance=\"dark\"\n                            tuiTooltip=\"Пожалуйста, укажите номер телефона или(и) адрес электронной почты\"\n                        />\n                    </div>\n                    <div\n                        class=\"h-px flex-1\"\n                        [style.background-color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n                    ></div>\n                </div>\n                <label\n                    tuiLabel\n                    *ngIf=\"isEmailVisible\"\n                >\n                    Адрес электронной почты\n                    <tui-input\n                        [pseudoInvalid]=\"isShowError\"\n                        formControlName=\"email\"\n                    >\n                        Адрес электронной почты\n                        <sc-suggestion-field\n                            *tuiDataList\n                            [type]=\"suggestionType.email\"\n                        />\n                    </tui-input>\n                    <tui-error\n                        formControlName=\"email\"\n                        [error]=\"[] | tuiFieldError | async\"\n                    />\n                </label>\n            </div>\n            <tui-error\n                *ngIf=\"isShowError\"\n                error=\"Пожалуйста, укажите номер телефона и(или) адрес электронной почты для уведомления о поступлении товара\"\n            />\n        </label>\n        <label tuiLabel>\n            Склад\n            <tui-select\n                formControlName=\"warehouseId\"\n                [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n            >\n                Склад\n                <tui-data-list *tuiDataList>\n                    @for (warehouse of warehouses(); track warehouse.id) {\n                        @if (!product.isWarehouseStockExist(warehouse.id)) {\n                            <button\n                                tuiOption\n                                [value]=\"warehouse.id\"\n                            >\n                                {{ warehouse.name }}\n                            </button>\n                        }\n                    }\n                </tui-data-list>\n            </tui-select>\n            <tui-error\n                formControlName=\"warehouseId\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n\n            <span\n                *ngIf=\"isStock()\"\n                [style.color]=\"'var(--tui-text-positive)'\"\n            >\n                Товар имеется в наличии на выбранном складе\n            </span>\n        </label>\n        <ng-template #selectLoading> <tui-loader /> </ng-template>\n    </div>\n\n    <div class=\"flex flex-wrap items-center gap-3\">\n        <button\n            tuiButton\n            type=\"submit\"\n            [disabled]=\"form.invalid || isStock() || (isShowCodeFields && !form.controls.verificationCode.value && form.controls.phone.valid)\"\n            [loading]=\"loading()\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n        >\n            Отправить\n        </button>\n        <div class=\"grow basis-60 text-body-s\">\n            Нажимая на кнопку, вы соглашаетесь с\n            <button\n                tuiLink\n                [pseudo]=\"true\"\n                type=\"button\"\n                (click)=\"context.data.onClickOfferHandler()\"\n            >\n                Политикой конфиденциальности\n            </button>\n        </div>\n    </div>\n\n    <tui-error\n        *ngIf=\"form.errors && form.errors['serverResponse']\"\n        [error]=\"[] | tuiFieldError | async\"\n    />\n</form>\n"]}
171
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-notify-when-in-stock-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.ts","../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, OnInit, Signal } from '@angular/core';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCatalogService,\n    ScConvertersService,\n    ScIProductNotifyWhenInStock,\n    ScISuggestionType,\n    ScIWarehouse,\n    ScProduct,\n    ScUser,\n    ScUserService,\n    ScWarehouseService,\n} from '@snabcentr/client-core';\nimport { TuiContext, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TuiAppearance, TuiButton, TuiDialogContext, TuiError, TuiIcon, TuiLabel, TuiLink, TuiLoader } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFieldErrorPipe, TuiTooltip } from '@taiga-ui/kit';\nimport { TuiCardLarge } from '@taiga-ui/layout';\nimport { TuiInputModule, TuiSelectModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { ScFormFieldsModule } from '../../form-fields/form-fields.module';\nimport { SC_NOTIFY_WHEN_IN_STOCK_REQUIRED_FIELDS } from '../../providers';\nimport { scAtLeastOneRequiredValidator } from '../../validators/sc-at-least-one-required-validator';\nimport { phoneValidator } from '../../validators/sc-phone-validator';\nimport { ScVerificationModule } from '../../verification/sc-verification.module';\n\n/**\n * Компонент формы отправки запроса на уведомление о поступлении товара.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-notify-when-in-stock-dialog',\n    templateUrl: './sc-notify-when-in-stock-dialog.component.html',\n    imports: [\n        ScVerificationModule,\n        FormsModule,\n        ReactiveFormsModule,\n        TuiInputModule,\n        TuiLink,\n        TuiButton,\n        TuiLabel,\n        TuiFieldErrorPipe,\n        TuiError,\n        ScVerificationModule,\n        ScFormFieldsModule,\n        TuiSelectModule,\n        TuiButtonLoading,\n        TuiLoader,\n        AsyncPipe,\n        NgIf,\n        TuiAppearance,\n        TuiCardLarge,\n        TuiIcon,\n        TuiTooltip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScNotifyWhenInStockDialogComponent implements OnInit {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void; onErrorHandler: (error: HttpErrorResponse) => void }> =\n        inject<TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void; onErrorHandler: (error: HttpErrorResponse) => void }>>(POLYMORPHEUS_CONTEXT);\n\n    /**\n     * Товар.\n     */\n    protected readonly product: ScProduct = this.context.data.product;\n\n    /**\n     * Форма для отправки запроса на уведомление о поступлении товара.\n     */\n    protected readonly form = new FormGroup({\n        email: new FormControl<string | null>(null, [Validators.email]),\n        phone: new FormControl<string | null>(null, [phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.minLength(6)]),\n        warehouseId: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * Список складов.\n     */\n    protected readonly warehouses: Signal<ScIWarehouse[] | null> = toSignal(inject(ScWarehouseService).getWarehouses$(), { initialValue: null });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Массив полей, которые должны быть заполнены хотя бы одним значением.\n     */\n    protected readonly requiredFields = inject(SC_NOTIFY_WHEN_IN_STOCK_REQUIRED_FIELDS);\n\n    /**\n     * Сервис для работы с каталогом.\n     */\n    protected readonly catalogService: ScCatalogService = inject(ScCatalogService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /**\n     * {@link Observable} данных о пользователе.\n     */\n    protected readonly user$: Observable<ScUser | null> = inject(ScUserService)\n        .getUserChange$()\n        .pipe(\n            tap((user) => {\n                if (!user.isGuest) {\n                    this.form.patchValue({\n                        phone: user.contacts.phone.value,\n                        email: user.contacts.email.value,\n                    });\n                }\n            })\n        );\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit$: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса данных уведомления о поступлении товара.\n     */\n    private readonly request$ = this.onSubmit$.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.getRawValue() as ScIProductNotifyWhenInStock),\n        switchMap((value) =>\n            this.catalogService.getProductNotifyWhenInStock$(this.product.id, this.convertersService.removeNullRecursive(value)).pipe(\n                tap(() => {\n                    this.context.completeWith();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        if (errors) {\n                            // eslint-disable-next-line no-shadow\n                            Object.entries(errors).forEach(([key, value]) => {\n                                this.form.get(key)?.setErrors({ serverResponse: value });\n                            });\n                        } else if (message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n\n                        this.context.data.onErrorHandler(error);\n                    }\n\n                    return of(undefined);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * Сигнал изменения состояния загрузки данных.\n     */\n    protected readonly loading: Signal<boolean> = toSignal(this.request$.pipe(map((value) => value === null)), { initialValue: false });\n\n    /**\n     * Инициализирует экземпляр класса {@link ScNotifyWhenInStockDialogComponent}.\n     */\n    public constructor() {\n        this.user$.pipe(takeUntilDestroyed()).subscribe();\n    }\n\n    /**\n     * Инициализация компонента.\n     */\n    public ngOnInit(): void {\n        if (this.requiredFields.length === 1) {\n            this.form.get(this.requiredFields[0])?.setValidators(Validators.required);\n        } else if (this.requiredFields.length > 1) {\n            this.form.setValidators(scAtLeastOneRequiredValidator(this.requiredFields));\n        }\n    }\n\n    /**\n     * Проверяет, является ли товар в наличии.\n     */\n    protected isStock(): boolean {\n        const warehouseId = Number(this.form.get('warehouseId')?.value ?? 0);\n\n        return this.product.isWarehouseStockExist(warehouseId);\n    }\n\n    /**\n     * Преобразует объект в значение, отображаемое в поле ввода.\n     *\n     * @param items Выбранные значения.\n     */\n    protected stringify(items: readonly ScIWarehouse[]): TuiStringHandler<TuiContext<number>> {\n        const itemsMap = new Map(\n            items.map((item) => {\n                return [item.id, item.name] as [number, string];\n            })\n        );\n\n        return ({ $implicit }: TuiContext<number>) => itemsMap.get($implicit) ?? '';\n    }\n}\n","<form\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit$.next()\"\n    class=\"flex flex-col gap-6\"\n>\n    @let isPhoneVisible = requiredFields.includes('phone');\n    @let isEmailVisible = requiredFields.includes('email');\n    @let phoneTouched = form.get('phone')?.touched;\n    @let emailTouched = form.get('email')?.touched;\n    @let isShowError = form.errors?.['atLeastOneRequired'] && ((isPhoneVisible && phoneTouched) || (isEmailVisible && emailTouched));\n    @let user = user$ | async;\n    @let userPhone = user?.contacts?.phone;\n    @let isShowCodeFields = !!form.controls.phone.value && (user?.isGuest || userPhone?.value !== form.controls.phone.value || !userPhone?.isApproved);\n\n    <div\n        *ngIf=\"user\"\n        class=\"flex flex-col gap-3\"\n    >\n        <label tuiLabel>\n            <div\n                [tuiAppearance]=\"isPhoneVisible && isEmailVisible ? 'outline-grayscale' : 'flat'\"\n                [tuiAppearanceMode]=\"isShowError ? 'invalid' : null\"\n                tuiCardLarge\n                class=\"flex flex-col !gap-1\"\n                [style.padding]=\"requiredFields.length <= 1 ? '0' : undefined\"\n                [style.border-radius]=\"requiredFields.length <= 1 ? '0' : undefined\"\n            >\n                <sc-verification-phone-check-form\n                    *ngIf=\"isPhoneVisible\"\n                    [required]=\"!form.controls.email.value\"\n                    [showCodeFields]=\"isShowCodeFields\"\n                    [pseudoInvalid]=\"isShowError\"\n                />\n                <div\n                    *ngIf=\"isPhoneVisible && isEmailVisible\"\n                    class=\"tui-island__paragraph flex items-center gap-3\"\n                    [style.color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n                >\n                    <div\n                        class=\"h-px flex-1\"\n                        [style.background-color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n                    ></div>\n                    <div class=\"text-body-s font-bold\">\n                        Или(и)\n                        <tui-icon\n                            [appearance]=\"isShowError ? 'negative' : 'grayscale'\"\n                            tuiHintDirection=\"right\"\n                            tuiHintAppearance=\"dark\"\n                            tuiTooltip=\"Пожалуйста, укажите номер телефона или(и) адрес электронной почты\"\n                        />\n                    </div>\n                    <div\n                        class=\"h-px flex-1\"\n                        [style.background-color]=\"isShowError ? 'var(--tui-text-negative)' : 'var(--tui-text-tertiary)'\"\n                    ></div>\n                </div>\n                <label\n                    tuiLabel\n                    *ngIf=\"isEmailVisible\"\n                >\n                    Адрес электронной почты\n                    <tui-input\n                        [pseudoInvalid]=\"isShowError\"\n                        formControlName=\"email\"\n                    >\n                        Адрес электронной почты\n                        <sc-suggestion-field\n                            *tuiDataList\n                            [type]=\"suggestionType.email\"\n                        />\n                    </tui-input>\n                    <tui-error\n                        formControlName=\"email\"\n                        [error]=\"[] | tuiFieldError | async\"\n                    />\n                </label>\n            </div>\n            <tui-error\n                *ngIf=\"isShowError\"\n                error=\"Пожалуйста, укажите номер телефона и(или) адрес электронной почты для уведомления о поступлении товара\"\n            />\n        </label>\n        <label tuiLabel>\n            Склад\n            <tui-select\n                formControlName=\"warehouseId\"\n                [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n            >\n                Склад\n                <tui-data-list *tuiDataList>\n                    @for (warehouse of warehouses(); track warehouse.id) {\n                        @if (!product.isWarehouseStockExist(warehouse.id)) {\n                            <button\n                                tuiOption\n                                [value]=\"warehouse.id\"\n                            >\n                                {{ warehouse.name }}\n                            </button>\n                        }\n                    }\n                </tui-data-list>\n            </tui-select>\n            <tui-error\n                formControlName=\"warehouseId\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n\n            <span\n                *ngIf=\"isStock()\"\n                [style.color]=\"'var(--tui-text-positive)'\"\n            >\n                Товар имеется в наличии на выбранном складе\n            </span>\n        </label>\n        <ng-template #selectLoading> <tui-loader /> </ng-template>\n    </div>\n\n    <div class=\"flex flex-wrap items-center gap-3\">\n        <button\n            tuiButton\n            type=\"submit\"\n            [disabled]=\"form.invalid || isStock() || (isShowCodeFields && !form.controls.verificationCode.value && form.controls.phone.valid)\"\n            [loading]=\"loading()\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n        >\n            Отправить\n        </button>\n        <div class=\"grow basis-60 text-body-s\">\n            Нажимая на кнопку, вы соглашаетесь с\n            <button\n                tuiLink\n                [pseudo]=\"true\"\n                type=\"button\"\n                (click)=\"context.data.onClickOfferHandler()\"\n            >\n                Политикой конфиденциальности\n            </button>\n        </div>\n    </div>\n\n    <tui-error\n        *ngIf=\"form.errors && form.errors['serverResponse']\"\n        [error]=\"[] | tuiFieldError | async\"\n    />\n</form>\n"]}
@@ -1,3 +1,4 @@
1
1
  export * from './verification-phone-check-form/sc-verification-phone-check-form.component';
2
2
  export * from './sc-verification.module';
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmVyaWZpY2F0aW9uL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNEVBQTRFLENBQUM7QUFDM0YsY0FBYywwQkFBMEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdmVyaWZpY2F0aW9uLXBob25lLWNoZWNrLWZvcm0vc2MtdmVyaWZpY2F0aW9uLXBob25lLWNoZWNrLWZvcm0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtdmVyaWZpY2F0aW9uLm1vZHVsZSc7XG4iXX0=
3
+ export * from './providers';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmVyaWZpY2F0aW9uL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsNEVBQTRFLENBQUM7QUFDM0YsY0FBYywwQkFBMEIsQ0FBQztBQUN6QyxjQUFjLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdmVyaWZpY2F0aW9uLXBob25lLWNoZWNrLWZvcm0vc2MtdmVyaWZpY2F0aW9uLXBob25lLWNoZWNrLWZvcm0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtdmVyaWZpY2F0aW9uLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL3Byb3ZpZGVycyc7XG4iXX0=
@@ -0,0 +1,3 @@
1
+ export * from './sc-phone-approve-code-sender.token';
2
+ export * from './sc-phone-approve-code-sender.provider';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmVyaWZpY2F0aW9uL3Byb3ZpZGVycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMseUNBQXlDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3NjLXBob25lLWFwcHJvdmUtY29kZS1zZW5kZXIudG9rZW4nO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1waG9uZS1hcHByb3ZlLWNvZGUtc2VuZGVyLnByb3ZpZGVyJztcbiJdfQ==
@@ -0,0 +1,12 @@
1
+ import { ScVerificationService } from '@snabcentr/client-core';
2
+ import { SC_PHONE_APPROVE_CODE_SENDER } from './sc-phone-approve-code-sender.token';
3
+ /**
4
+ * Провайдер по умолчанию для получения кода подтверждения телефона.
5
+ * Использует ScVerificationService для отправки кода.
6
+ */
7
+ export const SC_PHONE_APPROVE_CODE_SENDER_PROVIDER = {
8
+ provide: SC_PHONE_APPROVE_CODE_SENDER,
9
+ useFactory: (verificationService) => (phoneNumber) => verificationService.sendPhoneApproveCode(phoneNumber),
10
+ deps: [ScVerificationService],
11
+ };
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcGhvbmUtYXBwcm92ZS1jb2RlLXNlbmRlci5wcm92aWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS92ZXJpZmljYXRpb24vcHJvdmlkZXJzL3NjLXBob25lLWFwcHJvdmUtY29kZS1zZW5kZXIucHJvdmlkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFFL0QsT0FBTyxFQUFFLDRCQUE0QixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFFcEY7OztHQUdHO0FBQ0gsTUFBTSxDQUFDLE1BQU0scUNBQXFDLEdBQWE7SUFDM0QsT0FBTyxFQUFFLDRCQUE0QjtJQUNyQyxVQUFVLEVBQUUsQ0FBQyxtQkFBMEMsRUFBRSxFQUFFLENBQUMsQ0FBQyxXQUFtQixFQUFFLEVBQUUsQ0FBQyxtQkFBbUIsQ0FBQyxvQkFBb0IsQ0FBQyxXQUFXLENBQUM7SUFDMUksSUFBSSxFQUFFLENBQUMscUJBQXFCLENBQUM7Q0FDaEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFByb3ZpZGVyIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTY1ZlcmlmaWNhdGlvblNlcnZpY2UgfSBmcm9tICdAc25hYmNlbnRyL2NsaWVudC1jb3JlJztcblxuaW1wb3J0IHsgU0NfUEhPTkVfQVBQUk9WRV9DT0RFX1NFTkRFUiB9IGZyb20gJy4vc2MtcGhvbmUtYXBwcm92ZS1jb2RlLXNlbmRlci50b2tlbic7XG5cbi8qKlxuICog0J/RgNC+0LLQsNC50LTQtdGAINC/0L4g0YPQvNC+0LvRh9Cw0L3QuNGOINC00LvRjyDQv9C+0LvRg9GH0LXQvdC40Y8g0LrQvtC00LAg0L/QvtC00YLQstC10YDQttC00LXQvdC40Y8g0YLQtdC70LXRhNC+0L3QsC5cbiAqINCY0YHQv9C+0LvRjNC30YPQtdGCIFNjVmVyaWZpY2F0aW9uU2VydmljZSDQtNC70Y8g0L7RgtC/0YDQsNCy0LrQuCDQutC+0LTQsC5cbiAqL1xuZXhwb3J0IGNvbnN0IFNDX1BIT05FX0FQUFJPVkVfQ09ERV9TRU5ERVJfUFJPVklERVI6IFByb3ZpZGVyID0ge1xuICAgIHByb3ZpZGU6IFNDX1BIT05FX0FQUFJPVkVfQ09ERV9TRU5ERVIsXG4gICAgdXNlRmFjdG9yeTogKHZlcmlmaWNhdGlvblNlcnZpY2U6IFNjVmVyaWZpY2F0aW9uU2VydmljZSkgPT4gKHBob25lTnVtYmVyOiBzdHJpbmcpID0+IHZlcmlmaWNhdGlvblNlcnZpY2Uuc2VuZFBob25lQXBwcm92ZUNvZGUocGhvbmVOdW1iZXIpLFxuICAgIGRlcHM6IFtTY1ZlcmlmaWNhdGlvblNlcnZpY2VdLFxufTtcbiJdfQ==
@@ -0,0 +1,7 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ /**
3
+ * Токен для получения кода подтверждения телефона.
4
+ * Позволяет подключать внешнюю логику получения кода подтверждения.
5
+ */
6
+ export const SC_PHONE_APPROVE_CODE_SENDER = new InjectionToken('SC_PHONE_APPROVE_CODE_SENDER');
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcGhvbmUtYXBwcm92ZS1jb2RlLXNlbmRlci50b2tlbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS92ZXJpZmljYXRpb24vcHJvdmlkZXJzL3NjLXBob25lLWFwcHJvdmUtY29kZS1zZW5kZXIudG9rZW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUcvQzs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyxJQUFJLGNBQWMsQ0FBNEMsOEJBQThCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5cbi8qKlxuICog0KLQvtC60LXQvSDQtNC70Y8g0L/QvtC70YPRh9C10L3QuNGPINC60L7QtNCwINC/0L7QtNGC0LLQtdGA0LbQtNC10L3QuNGPINGC0LXQu9C10YTQvtC90LAuXG4gKiDQn9C+0LfQstC+0LvRj9C10YIg0L/QvtC00LrQu9GO0YfQsNGC0Ywg0LLQvdC10YjQvdGO0Y4g0LvQvtCz0LjQutGDINC/0L7Qu9GD0YfQtdC90LjRjyDQutC+0LTQsCDQv9C+0LTRgtCy0LXRgNC20LTQtdC90LjRjy5cbiAqL1xuZXhwb3J0IGNvbnN0IFNDX1BIT05FX0FQUFJPVkVfQ09ERV9TRU5ERVIgPSBuZXcgSW5qZWN0aW9uVG9rZW48KHBob25lTnVtYmVyOiBzdHJpbmcpID0+IE9ic2VydmFibGU8dm9pZD4+KCdTQ19QSE9ORV9BUFBST1ZFX0NPREVfU0VOREVSJyk7XG4iXX0=