@snabcentr/client-ui 3.9.3 → 3.10.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -1,5 +1,6 @@
1
- import { ChangeDetectorRef, EventEmitter } from '@angular/core';
1
+ import { ChangeDetectorRef, EventEmitter, InputSignal } from '@angular/core';
2
2
  import { ScCategory, ScIUrls } from '@snabcentr/client-core';
3
+ import { TuiLooseUnion } from '@taiga-ui/cdk';
3
4
  import { TuiSizeS } from '@taiga-ui/core';
4
5
  import { Observable } from 'rxjs';
5
6
  import * as i0 from "@angular/core";
@@ -18,6 +19,10 @@ export declare class ScCategoryCardComponent {
18
19
  * Размер карточки категории.
19
20
  */
20
21
  size: TuiSizeS;
22
+ /**
23
+ * Вид отображения карточки.
24
+ */
25
+ appearance: InputSignal<TuiLooseUnion<'root' | 'normal'>>;
21
26
  /**
22
27
  * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.
23
28
  */
@@ -73,5 +78,5 @@ export declare class ScCategoryCardComponent {
73
78
  */
74
79
  markForCheck(): void;
75
80
  static ɵfac: i0.ɵɵFactoryDeclaration<ScCategoryCardComponent, never>;
76
- static ɵcmp: i0.ɵɵComponentDeclaration<ScCategoryCardComponent, "sc-category-card", never, { "category": { "alias": "category"; "required": false; }; "size": { "alias": "size"; "required": false; }; "enableHover": { "alias": "enableHover"; "required": false; }; "href": { "alias": "href"; "required": false; }; }, { "clickOnFavoriteEvent": "clickOnFavoriteEvent"; }, never, never, false, never>;
81
+ static ɵcmp: i0.ɵɵComponentDeclaration<ScCategoryCardComponent, "sc-category-card", never, { "category": { "alias": "category"; "required": false; }; "size": { "alias": "size"; "required": false; }; "appearance": { "alias": "appearance"; "required": false; "isSignal": true; }; "enableHover": { "alias": "enableHover"; "required": false; }; "href": { "alias": "href"; "required": false; }; }, { "clickOnFavoriteEvent": "clickOnFavoriteEvent"; }, never, never, false, never>;
77
82
  }
@@ -1,4 +1,4 @@
1
- import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Inject, inject, Input, Output } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Inject, inject, Input, input, Output } from '@angular/core';
2
2
  import { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService } from '@snabcentr/client-core';
3
3
  import { TUI_IS_MOBILE } from '@taiga-ui/cdk';
4
4
  import * as i0 from "@angular/core";
@@ -26,6 +26,11 @@ export class ScCategoryCardComponent {
26
26
  * Размер карточки категории.
27
27
  */
28
28
  this.size = 'm';
29
+ // TODO: Переделать на HostBinding.
30
+ /**
31
+ * Вид отображения карточки.
32
+ */
33
+ this.appearance = input('normal');
29
34
  /**
30
35
  * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.
31
36
  */
@@ -76,11 +81,11 @@ export class ScCategoryCardComponent {
76
81
  this.cdr.markForCheck();
77
82
  }
78
83
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ScCategoryCardComponent, deps: [{ token: SC_URLS }, { token: SC_PATH_IMAGE_NOT_FOUND }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
79
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: ScCategoryCardComponent, selector: "sc-category-card", inputs: { category: "category", size: "size", enableHover: "enableHover", href: "href" }, outputs: { clickOnFavoriteEvent: "clickOnFavoriteEvent" }, host: { properties: { "attr.data-size": "this.size" } }, ngImport: i0, template: "<div class=\"relative\">\n <a\n (tuiHoveredChange)=\"onHovered($event)\"\n [routerLink]=\"href ?? null\"\n class=\"category-button flex flex-col overflow-hidden rounded-tui-radius-m border border-tui-base-04 text-center\"\n >\n <div class=\"img-wrapper w-full grow overflow-hidden\">\n <img\n *ngIf=\"category\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n class=\"size-full rounded-xl object-contain p-2\"\n />\n\n <!-- \u0411\u043B\u043E\u043A \u0434\u043B\u044F \u0441\u043A\u0435\u043B\u0435\u0442\u043E\u043D\u0430 \u043A\u0430\u0440\u0442\u043E\u0447\u043A\u0438 -->\n <div\n *ngIf=\"!category\"\n class=\"img-wrapper size-full bg-tui-base-02\"\n ></div>\n </div>\n\n <div class=\"name flex w-full items-center justify-center\">\n @if (category) {\n @if (enableHover && !isMobile) {\n <tui-line-clamp\n [content]=\"category.name\"\n class=\"pointer-events-none\"\n [lineHeight]=\"size === 'm' ? 26 : 24\"\n [linesLimit]=\"isHover ? 4 : 2\"\n />\n } @else {\n {{ category.name }}\n }\n } @else {\n <div class=\"skeleton-name rounded-tui-radius-s bg-tui-base-02\"></div>\n }\n </div>\n </a>\n <sc-favorite-button\n *ngIf=\"category && (authStatus$ | async)\"\n [showLoader]=\"favoriteShowLoader\"\n [isFavorite]=\"category.isFavorite\"\n (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n class=\"absolute left-1 top-1\"\n />\n</div>\n", styles: [":host{--tui-duration: .15s}:host[data-size=m] a.category-button{width:100%;height:12.5rem}:host[data-size=m] a.category-button .img-wrapper{max-height:8.75rem}:host[data-size=m] a.category-button .name{padding:.25rem 1rem;margin-block:auto;font-size:.9375rem;line-height:1.5rem;font-weight:800}:host[data-size=m] a.category-button .name .skeleton-name{width:10rem;height:1rem}:host[data-size=s] a.category-button{width:100%;height:10rem}:host[data-size=s] a.category-button .img-wrapper{max-height:7rem}:host[data-size=s] a.category-button .name{padding:.25rem .5rem;margin-block:auto;font-size:.8125rem;line-height:1.25rem;font-weight:800}:host[data-size=s] a.category-button .name .skeleton-name{width:7rem;height:.75rem}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "component", type: i3.TuiLineClamp, selector: "tui-line-clamp", inputs: ["lineHeight", "content", "linesLimit"], outputs: ["overflownChange"] }, { kind: "directive", type: i4.TuiHovered, selector: "[tuiHoveredChange]", outputs: ["tuiHoveredChange"] }, { kind: "component", type: i5.ScFavoriteButtonComponent, selector: "sc-favorite-button", inputs: ["isFavorite", "showLoader", "disabled"], outputs: ["clickEvent"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
84
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: ScCategoryCardComponent, selector: "sc-category-card", inputs: { category: { classPropertyName: "category", publicName: "category", isSignal: false, isRequired: false, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: false, isRequired: false, transformFunction: null }, appearance: { classPropertyName: "appearance", publicName: "appearance", isSignal: true, isRequired: false, transformFunction: null }, enableHover: { classPropertyName: "enableHover", publicName: "enableHover", isSignal: false, isRequired: false, transformFunction: null }, href: { classPropertyName: "href", publicName: "href", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { clickOnFavoriteEvent: "clickOnFavoriteEvent" }, host: { properties: { "attr.data-size": "this.size" } }, ngImport: i0, template: "<div class=\"relative\">\n <a\n (tuiHoveredChange)=\"onHovered($event)\"\n [routerLink]=\"href ?? null\"\n class=\"category-button rounded-tui-radius-m border-tui-base-04 flex flex-col overflow-hidden border text-center\"\n >\n <div class=\"img-wrapper w-full grow overflow-hidden\">\n <img\n *ngIf=\"category\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n [ngClass]=\"{ '!object-contain p-2': appearance() === 'normal' || !getCategoryImgURL(category) }\"\n class=\"size-full rounded-xl object-cover\"\n />\n\n <!-- \u0411\u043B\u043E\u043A \u0434\u043B\u044F \u0441\u043A\u0435\u043B\u0435\u0442\u043E\u043D\u0430 \u043A\u0430\u0440\u0442\u043E\u0447\u043A\u0438 -->\n <div\n *ngIf=\"!category\"\n class=\"img-wrapper bg-tui-base-02 size-full\"\n ></div>\n </div>\n\n <div class=\"name flex w-full items-center justify-center\">\n @if (category) {\n @if (enableHover && !isMobile) {\n <tui-line-clamp\n [content]=\"category.name\"\n class=\"pointer-events-none\"\n [lineHeight]=\"size === 'm' ? 26 : 24\"\n [linesLimit]=\"isHover ? 4 : 2\"\n />\n } @else {\n {{ category.name }}\n }\n } @else {\n <div class=\"skeleton-name rounded-tui-radius-s bg-tui-base-02\"></div>\n }\n </div>\n </a>\n <sc-favorite-button\n *ngIf=\"category && (authStatus$ | async)\"\n [showLoader]=\"favoriteShowLoader\"\n [isFavorite]=\"category.isFavorite\"\n (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n class=\"absolute left-1 top-1\"\n />\n</div>\n", styles: [":host{--tui-duration: .15s}:host[data-size=m] a.category-button{width:100%;height:12.5rem}:host[data-size=m] a.category-button .img-wrapper{max-height:8.75rem}:host[data-size=m] a.category-button .name{padding:.25rem 1rem;margin-block:auto;font-size:.9375rem;line-height:1.5rem;font-weight:800}:host[data-size=m] a.category-button .name .skeleton-name{width:10rem;height:1rem}:host[data-size=s] a.category-button{width:100%;height:10rem}:host[data-size=s] a.category-button .img-wrapper{max-height:7rem}:host[data-size=s] a.category-button .name{padding:.25rem .5rem;margin-block:auto;font-size:.8125rem;line-height:1.25rem;font-weight:800}:host[data-size=s] a.category-button .name .skeleton-name{width:7rem;height:.75rem}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "component", type: i3.TuiLineClamp, selector: "tui-line-clamp", inputs: ["lineHeight", "content", "linesLimit"], outputs: ["overflownChange"] }, { kind: "directive", type: i4.TuiHovered, selector: "[tuiHoveredChange]", outputs: ["tuiHoveredChange"] }, { kind: "component", type: i5.ScFavoriteButtonComponent, selector: "sc-favorite-button", inputs: ["isFavorite", "showLoader", "disabled"], outputs: ["clickEvent"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
80
85
  }
81
86
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ScCategoryCardComponent, decorators: [{
82
87
  type: Component,
83
- args: [{ selector: 'sc-category-card', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"relative\">\n <a\n (tuiHoveredChange)=\"onHovered($event)\"\n [routerLink]=\"href ?? null\"\n class=\"category-button flex flex-col overflow-hidden rounded-tui-radius-m border border-tui-base-04 text-center\"\n >\n <div class=\"img-wrapper w-full grow overflow-hidden\">\n <img\n *ngIf=\"category\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n class=\"size-full rounded-xl object-contain p-2\"\n />\n\n <!-- \u0411\u043B\u043E\u043A \u0434\u043B\u044F \u0441\u043A\u0435\u043B\u0435\u0442\u043E\u043D\u0430 \u043A\u0430\u0440\u0442\u043E\u0447\u043A\u0438 -->\n <div\n *ngIf=\"!category\"\n class=\"img-wrapper size-full bg-tui-base-02\"\n ></div>\n </div>\n\n <div class=\"name flex w-full items-center justify-center\">\n @if (category) {\n @if (enableHover && !isMobile) {\n <tui-line-clamp\n [content]=\"category.name\"\n class=\"pointer-events-none\"\n [lineHeight]=\"size === 'm' ? 26 : 24\"\n [linesLimit]=\"isHover ? 4 : 2\"\n />\n } @else {\n {{ category.name }}\n }\n } @else {\n <div class=\"skeleton-name rounded-tui-radius-s bg-tui-base-02\"></div>\n }\n </div>\n </a>\n <sc-favorite-button\n *ngIf=\"category && (authStatus$ | async)\"\n [showLoader]=\"favoriteShowLoader\"\n [isFavorite]=\"category.isFavorite\"\n (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n class=\"absolute left-1 top-1\"\n />\n</div>\n", styles: [":host{--tui-duration: .15s}:host[data-size=m] a.category-button{width:100%;height:12.5rem}:host[data-size=m] a.category-button .img-wrapper{max-height:8.75rem}:host[data-size=m] a.category-button .name{padding:.25rem 1rem;margin-block:auto;font-size:.9375rem;line-height:1.5rem;font-weight:800}:host[data-size=m] a.category-button .name .skeleton-name{width:10rem;height:1rem}:host[data-size=s] a.category-button{width:100%;height:10rem}:host[data-size=s] a.category-button .img-wrapper{max-height:7rem}:host[data-size=s] a.category-button .name{padding:.25rem .5rem;margin-block:auto;font-size:.8125rem;line-height:1.25rem;font-weight:800}:host[data-size=s] a.category-button .name .skeleton-name{width:7rem;height:.75rem}\n"] }]
88
+ args: [{ selector: 'sc-category-card', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"relative\">\n <a\n (tuiHoveredChange)=\"onHovered($event)\"\n [routerLink]=\"href ?? null\"\n class=\"category-button rounded-tui-radius-m border-tui-base-04 flex flex-col overflow-hidden border text-center\"\n >\n <div class=\"img-wrapper w-full grow overflow-hidden\">\n <img\n *ngIf=\"category\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n [ngClass]=\"{ '!object-contain p-2': appearance() === 'normal' || !getCategoryImgURL(category) }\"\n class=\"size-full rounded-xl object-cover\"\n />\n\n <!-- \u0411\u043B\u043E\u043A \u0434\u043B\u044F \u0441\u043A\u0435\u043B\u0435\u0442\u043E\u043D\u0430 \u043A\u0430\u0440\u0442\u043E\u0447\u043A\u0438 -->\n <div\n *ngIf=\"!category\"\n class=\"img-wrapper bg-tui-base-02 size-full\"\n ></div>\n </div>\n\n <div class=\"name flex w-full items-center justify-center\">\n @if (category) {\n @if (enableHover && !isMobile) {\n <tui-line-clamp\n [content]=\"category.name\"\n class=\"pointer-events-none\"\n [lineHeight]=\"size === 'm' ? 26 : 24\"\n [linesLimit]=\"isHover ? 4 : 2\"\n />\n } @else {\n {{ category.name }}\n }\n } @else {\n <div class=\"skeleton-name rounded-tui-radius-s bg-tui-base-02\"></div>\n }\n </div>\n </a>\n <sc-favorite-button\n *ngIf=\"category && (authStatus$ | async)\"\n [showLoader]=\"favoriteShowLoader\"\n [isFavorite]=\"category.isFavorite\"\n (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n class=\"absolute left-1 top-1\"\n />\n</div>\n", styles: [":host{--tui-duration: .15s}:host[data-size=m] a.category-button{width:100%;height:12.5rem}:host[data-size=m] a.category-button .img-wrapper{max-height:8.75rem}:host[data-size=m] a.category-button .name{padding:.25rem 1rem;margin-block:auto;font-size:.9375rem;line-height:1.5rem;font-weight:800}:host[data-size=m] a.category-button .name .skeleton-name{width:10rem;height:1rem}:host[data-size=s] a.category-button{width:100%;height:10rem}:host[data-size=s] a.category-button .img-wrapper{max-height:7rem}:host[data-size=s] a.category-button .name{padding:.25rem .5rem;margin-block:auto;font-size:.8125rem;line-height:1.25rem;font-weight:800}:host[data-size=s] a.category-button .name .skeleton-name{width:7rem;height:.75rem}\n"] }]
84
89
  }], ctorParameters: () => [{ type: undefined, decorators: [{
85
90
  type: Inject,
86
91
  args: [SC_URLS]
@@ -101,4 +106,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImpo
101
106
  }], href: [{
102
107
  type: Input
103
108
  }] } });
104
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-category-card.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.ts","../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAqB,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAChJ,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAE,aAAa,EAAuB,MAAM,wBAAwB,CAAC;AAC9G,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;;;;;;;AAI9C;;GAEG;AAOH,MAAM,OAAO,uBAAuB;IAoDhC;;;;;;OAMG;IACH,YACsC,IAAa,EACG,iBAAyB,EAC1D,GAAsB;QAFL,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,QAAG,GAAH,GAAG,CAAmB;QAvD3C;;WAEG;QAGI,SAAI,GAAa,GAAG,CAAC;QAE5B;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QACa,gBAAW,GAAwB,MAAM,CAAC,aAAa,CAAC,CAAC,aAAa,EAAE,CAAC;QAEzF;;WAEG;QACO,YAAO,GAAY,KAAK,CAAC;QAEnC;;WAEG;QAEI,gBAAW,GAAY,KAAK,CAAC;QAEpC;;WAEG;QACa,aAAQ,GAAY,MAAM,CAAC,aAAa,CAAC,CAAC;QAE1D;;WAEG;QAEI,yBAAoB,GAA6B,IAAI,YAAY,EAAc,CAAC;IAmBpF,CAAC;IAEJ;;;;OAIG;IACO,SAAS,CAAC,OAAgB;QAChC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;IAC3B,CAAC;IAED;;;;OAIG;IACO,iBAAiB,CAAC,QAAoB;QAC5C,OAAO,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,QAAQ,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;IAC1H,CAAC;IAED;;;;OAIG;IACI,YAAY;QACf,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;+GA1FQ,uBAAuB,kBA4DpB,OAAO,aACP,uBAAuB;mGA7D1B,uBAAuB,sQCfpC,szDA8CA;;4FD/Ba,uBAAuB;kBANnC,SAAS;+BACI,kBAAkB,mBAGX,uBAAuB,CAAC,MAAM;;0BA8D1C,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;yEAxD5B,QAAQ;sBADd,KAAK;gBAQC,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB;gBAsBtB,WAAW;sBADjB,KAAK;gBAYC,oBAAoB;sBAD1B,MAAM;gBAOA,IAAI;sBADV,KAAK","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, Inject, inject, Input, Output } from '@angular/core';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCategory, ScIUrls } from '@snabcentr/client-core';\nimport { TUI_IS_MOBILE } from '@taiga-ui/cdk';\nimport { TuiSizeS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\n/**\n * Карточка категории.\n */\n@Component({\n    selector: 'sc-category-card',\n    templateUrl: './sc-category-card.component.html',\n    styleUrls: ['./sc-category-card.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCategoryCardComponent {\n    /**\n     * Данные о категории.\n     */\n    @Input()\n    public category?: ScCategory;\n\n    /**\n     * Размер карточки категории.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS = 'm';\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = inject(ScAuthService).getAuthChange();\n\n    /**\n     * Признак наведения на карточку.\n     */\n    protected isHover: boolean = false;\n\n    /**\n     * Признак что категория имеет поведение наведения и скрытия названия.\n     */\n    @Input()\n    public enableHover: boolean = false;\n\n    /**\n     * Признак того, отображается этот компонент на мобильном устройстве или нет.\n     */\n    public readonly isMobile: boolean = inject(TUI_IS_MOBILE);\n\n    /**\n     * Событие нажатия на кнопку избранной категории.\n     */\n    @Output()\n    public clickOnFavoriteEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Ссылка на страницу категории. Используется именно `href`, так как остановить событие клика для `routerLink` не вышло.\n     */\n    @Input()\n    public href?: string;\n\n    /**\n     * Инициализирует экземпляр класса {@link CategoryCardComponent}.\n     *\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     * @param cdr Объект для работы с обнаружением изменений.\n     */\n    public constructor(\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Обработчик события наведения.\n     *\n     * @param isHover Признак наведения на карточку.\n     */\n    protected onHovered(isHover: boolean): void {\n        this.isHover = isHover;\n    }\n\n    /**\n     * Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию (\"product_not_found\").\n     *\n     * @param category Данные о категории.\n     */\n    protected getCategoryImgURL(category: ScCategory): string {\n        return category.properties?.image ? `${this.urls.imgServerUrl}/${category.properties.image}` : this.pathImageNotFound;\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     *\n     * @deprecated\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<div class=\"relative\">\n    <a\n        (tuiHoveredChange)=\"onHovered($event)\"\n        [routerLink]=\"href ?? null\"\n        class=\"category-button flex flex-col overflow-hidden rounded-tui-radius-m border border-tui-base-04 text-center\"\n    >\n        <div class=\"img-wrapper w-full grow overflow-hidden\">\n            <img\n                *ngIf=\"category\"\n                [src]=\"getCategoryImgURL(category)\"\n                [alt]=\"category.name\"\n                class=\"size-full rounded-xl object-contain p-2\"\n            />\n\n            <!-- Блок для скелетона карточки -->\n            <div\n                *ngIf=\"!category\"\n                class=\"img-wrapper size-full bg-tui-base-02\"\n            ></div>\n        </div>\n\n        <div class=\"name flex w-full items-center justify-center\">\n            @if (category) {\n                @if (enableHover && !isMobile) {\n                    <tui-line-clamp\n                        [content]=\"category.name\"\n                        class=\"pointer-events-none\"\n                        [lineHeight]=\"size === 'm' ? 26 : 24\"\n                        [linesLimit]=\"isHover ? 4 : 2\"\n                    />\n                } @else {\n                    {{ category.name }}\n                }\n            } @else {\n                <div class=\"skeleton-name rounded-tui-radius-s bg-tui-base-02\"></div>\n            }\n        </div>\n    </a>\n    <sc-favorite-button\n        *ngIf=\"category && (authStatus$ | async)\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"category.isFavorite\"\n        (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n        class=\"absolute left-1 top-1\"\n    />\n</div>\n"]}
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-category-card.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.ts","../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAqB,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,EAAe,MAAM,EAAE,MAAM,eAAe,CAAC;AACpK,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAE,aAAa,EAAuB,MAAM,wBAAwB,CAAC;AAC9G,OAAO,EAAE,aAAa,EAAiB,MAAM,eAAe,CAAC;;;;;;;AAI7D;;GAEG;AAOH,MAAM,OAAO,uBAAuB;IA0DhC;;;;;;OAMG;IACH,YACsC,IAAa,EACG,iBAAyB,EAC1D,GAAsB;QAFL,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,QAAG,GAAH,GAAG,CAAmB;QA7D3C;;WAEG;QAGI,SAAI,GAAa,GAAG,CAAC;QAE5B,mCAAmC;QACnC;;WAEG;QACI,eAAU,GAAkD,KAAK,CAAmC,QAAQ,CAAC,CAAC;QAErH;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QACa,gBAAW,GAAwB,MAAM,CAAC,aAAa,CAAC,CAAC,aAAa,EAAE,CAAC;QAEzF;;WAEG;QACO,YAAO,GAAY,KAAK,CAAC;QAEnC;;WAEG;QAEI,gBAAW,GAAY,KAAK,CAAC;QAEpC;;WAEG;QACa,aAAQ,GAAY,MAAM,CAAC,aAAa,CAAC,CAAC;QAE1D;;WAEG;QAEI,yBAAoB,GAA6B,IAAI,YAAY,EAAc,CAAC;IAmBpF,CAAC;IAEJ;;;;OAIG;IACO,SAAS,CAAC,OAAgB;QAChC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;IAC3B,CAAC;IAED;;;;OAIG;IACO,iBAAiB,CAAC,QAAoB;QAC5C,OAAO,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,QAAQ,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;IAC1H,CAAC;IAED;;;;OAIG;IACI,YAAY;QACf,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;+GAhGQ,uBAAuB,kBAkEpB,OAAO,aACP,uBAAuB;mGAnE1B,uBAAuB,wzBCfpC,o6DA+CA;;4FDhCa,uBAAuB;kBANnC,SAAS;+BACI,kBAAkB,mBAGX,uBAAuB,CAAC,MAAM;;0BAoE1C,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;yEA9D5B,QAAQ;sBADd,KAAK;gBAQC,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB;gBA4BtB,WAAW;sBADjB,KAAK;gBAYC,oBAAoB;sBAD1B,MAAM;gBAOA,IAAI;sBADV,KAAK","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, Inject, inject, Input, input, InputSignal, Output } from '@angular/core';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCategory, ScIUrls } from '@snabcentr/client-core';\nimport { TUI_IS_MOBILE, TuiLooseUnion } from '@taiga-ui/cdk';\nimport { TuiSizeS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\n/**\n * Карточка категории.\n */\n@Component({\n    selector: 'sc-category-card',\n    templateUrl: './sc-category-card.component.html',\n    styleUrls: ['./sc-category-card.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCategoryCardComponent {\n    /**\n     * Данные о категории.\n     */\n    @Input()\n    public category?: ScCategory;\n\n    /**\n     * Размер карточки категории.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS = 'm';\n\n    // TODO: Переделать на HostBinding.\n    /**\n     * Вид отображения карточки.\n     */\n    public appearance: InputSignal<TuiLooseUnion<'root' | 'normal'>> = input<TuiLooseUnion<'root' | 'normal'>>('normal');\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = inject(ScAuthService).getAuthChange();\n\n    /**\n     * Признак наведения на карточку.\n     */\n    protected isHover: boolean = false;\n\n    /**\n     * Признак что категория имеет поведение наведения и скрытия названия.\n     */\n    @Input()\n    public enableHover: boolean = false;\n\n    /**\n     * Признак того, отображается этот компонент на мобильном устройстве или нет.\n     */\n    public readonly isMobile: boolean = inject(TUI_IS_MOBILE);\n\n    /**\n     * Событие нажатия на кнопку избранной категории.\n     */\n    @Output()\n    public clickOnFavoriteEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Ссылка на страницу категории. Используется именно `href`, так как остановить событие клика для `routerLink` не вышло.\n     */\n    @Input()\n    public href?: string;\n\n    /**\n     * Инициализирует экземпляр класса {@link CategoryCardComponent}.\n     *\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     * @param cdr Объект для работы с обнаружением изменений.\n     */\n    public constructor(\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Обработчик события наведения.\n     *\n     * @param isHover Признак наведения на карточку.\n     */\n    protected onHovered(isHover: boolean): void {\n        this.isHover = isHover;\n    }\n\n    /**\n     * Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию (\"product_not_found\").\n     *\n     * @param category Данные о категории.\n     */\n    protected getCategoryImgURL(category: ScCategory): string {\n        return category.properties?.image ? `${this.urls.imgServerUrl}/${category.properties.image}` : this.pathImageNotFound;\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     *\n     * @deprecated\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<div class=\"relative\">\n    <a\n        (tuiHoveredChange)=\"onHovered($event)\"\n        [routerLink]=\"href ?? null\"\n        class=\"category-button rounded-tui-radius-m border-tui-base-04 flex flex-col overflow-hidden border text-center\"\n    >\n        <div class=\"img-wrapper w-full grow overflow-hidden\">\n            <img\n                *ngIf=\"category\"\n                [src]=\"getCategoryImgURL(category)\"\n                [alt]=\"category.name\"\n                [ngClass]=\"{ '!object-contain p-2': appearance() === 'normal' || !getCategoryImgURL(category) }\"\n                class=\"size-full rounded-xl object-cover\"\n            />\n\n            <!-- Блок для скелетона карточки -->\n            <div\n                *ngIf=\"!category\"\n                class=\"img-wrapper bg-tui-base-02 size-full\"\n            ></div>\n        </div>\n\n        <div class=\"name flex w-full items-center justify-center\">\n            @if (category) {\n                @if (enableHover && !isMobile) {\n                    <tui-line-clamp\n                        [content]=\"category.name\"\n                        class=\"pointer-events-none\"\n                        [lineHeight]=\"size === 'm' ? 26 : 24\"\n                        [linesLimit]=\"isHover ? 4 : 2\"\n                    />\n                } @else {\n                    {{ category.name }}\n                }\n            } @else {\n                <div class=\"skeleton-name rounded-tui-radius-s bg-tui-base-02\"></div>\n            }\n        </div>\n    </a>\n    <sc-favorite-button\n        *ngIf=\"category && (authStatus$ | async)\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"category.isFavorite\"\n        (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n        class=\"absolute left-1 top-1\"\n    />\n</div>\n"]}
@@ -0,0 +1,267 @@
1
+ /* eslint-disable @typescript-eslint/unbound-method */
2
+ import { AsyncPipe, NgClass, NgFor, NgIf } from '@angular/common';
3
+ import { ChangeDetectionStrategy, Component, effect, inject, input, output } from '@angular/core';
4
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
5
+ import { FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';
6
+ import { ScConvertersService, ScFeedbackService, ScISuggestionType, ScUploadedFile, ScUserService } from '@snabcentr/client-core';
7
+ import { TuiValidationError } from '@taiga-ui/cdk';
8
+ import { TuiButton, TuiError, TuiLabel, TuiLink } from '@taiga-ui/core';
9
+ import { TuiButtonLoading, TuiFieldErrorPipe, TuiFiles, tuiFilesAccepted } from '@taiga-ui/kit';
10
+ import { TuiInputModule, TuiInputPhoneModule, TuiTextareaModule } from '@taiga-ui/legacy';
11
+ import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
12
+ import { RecaptchaFormsModule, RecaptchaModule } from 'ng-recaptcha-2';
13
+ import { ScFormFieldsModule } from 'projects/client-ui/form-fields';
14
+ import { MAX_FILES_IN_FORM_INPUT } from 'projects/client-ui/providers/sc-max-files-in-form-input';
15
+ import { catchError, filter, map, merge, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
16
+ import * as i0 from "@angular/core";
17
+ import * as i1 from "@angular/forms";
18
+ import * as i2 from "../../form-fields/suggestion-field/sc-suggestion-field.component";
19
+ import * as i3 from "@taiga-ui/legacy";
20
+ import * as i4 from "ng-recaptcha-2";
21
+ import * as i5 from "@taiga-ui/kit";
22
+ /**
23
+ * Компонент формы обратной связи.
24
+ */
25
+ export class ScFeedbackFormComponent {
26
+ /**
27
+ * Инициализирует экземпляр класса {@link ScFeedbackFormComponent}.
28
+ */
29
+ constructor() {
30
+ /**
31
+ * Условное название блока где расположена ссылка.
32
+ */
33
+ this.formSlag = input.required();
34
+ /**
35
+ * Событие нажатия на кнопку договора оферты.
36
+ */
37
+ this.onClickOffer = output();
38
+ /**
39
+ * Событие успешной отправки формы.
40
+ */
41
+ this.onSendSuccess = output();
42
+ /**
43
+ * Сервис обратной связи.
44
+ */
45
+ this.feedbackService = inject(ScFeedbackService);
46
+ /**
47
+ * Сервис конвертации данных.
48
+ */
49
+ this.convertersService = inject(ScConvertersService);
50
+ /**
51
+ * Максимальное количество файлов, отправляемых в форме.
52
+ */
53
+ this.maxFilesInForm = inject(MAX_FILES_IN_FORM_INPUT);
54
+ /**
55
+ * Сервис данных о пользователе.
56
+ */
57
+ this.user$ = inject(ScUserService)
58
+ .getUserChange$()
59
+ .pipe(tap((user) => {
60
+ if (user.isGuest) {
61
+ this.form.addControl('recaptcha', this.recaptchaControl);
62
+ this.resetForm();
63
+ }
64
+ else {
65
+ this.form.removeControl('recaptcha');
66
+ // eslint-disable-next-line default-case
67
+ switch (this.formSlag()) {
68
+ case 'common':
69
+ this.form.patchValue({ name: user.name, email: user.contacts.email.value });
70
+ break;
71
+ case 'contacts':
72
+ case 'vacancies':
73
+ this.form.patchValue({ name: user.name, phone: user.contacts.phone.value });
74
+ break;
75
+ case 'samples':
76
+ this.form.patchValue({ name: user.name, phone: user.contacts.phone.value, email: user.contacts.email.value });
77
+ break;
78
+ }
79
+ }
80
+ }));
81
+ /**
82
+ * Контекст диалогового окна, в котором открыт компонент.
83
+ */
84
+ this.context = inject(POLYMORPHEUS_CONTEXT, { optional: true });
85
+ /**
86
+ * Поле ввода для recaptcha.
87
+ */
88
+ this.recaptchaControl = new FormControl(null);
89
+ /**
90
+ * Группа полей ввода для формы «Пожелания и предложения по улучшению сайта».
91
+ */
92
+ this.form = new FormGroup({
93
+ name: new FormControl(null, [Validators.required, Validators.minLength(3)]),
94
+ phone: new FormControl(null),
95
+ email: new FormControl(null),
96
+ message: new FormControl(null, []),
97
+ files: new FormControl([], [this.maxFilesLength(5)]),
98
+ });
99
+ /**
100
+ * Перечисление типов подсказок.
101
+ */
102
+ this.suggestionType = ScISuggestionType;
103
+ /**
104
+ * {@link Subject} события отправки формы.
105
+ */
106
+ this.onSubmit = new Subject();
107
+ /**
108
+ * {@link Observable} запроса на отправку данных.
109
+ */
110
+ this.request$ = this.onSubmit.pipe(filter(() => this.form.valid), map(() => this.form.value), switchMap((value) => this.feedbackService.sendFeedback(this.context?.data.formSlag ?? this.formSlag(), this.convertersService.removeNullRecursive(value)).pipe(tap(() => {
111
+ if (this.context) {
112
+ this.context.completeWith();
113
+ }
114
+ else {
115
+ this.resetForm();
116
+ this.onSendSuccess.emit();
117
+ }
118
+ }),
119
+ // eslint-disable-next-line rxjs/no-implicit-any-catch
120
+ catchError((error) => {
121
+ this.form.markAllAsTouched();
122
+ const { errors, message } = error.error;
123
+ if (errors) {
124
+ Object.keys(errors).forEach((key) => {
125
+ const control = this.form.get(key);
126
+ if (control) {
127
+ // eslint-disable-next-line security/detect-object-injection
128
+ control.setErrors({ serverResponse: errors[key] });
129
+ }
130
+ });
131
+ }
132
+ else if (message) {
133
+ this.form.setErrors({ serverResponse: [message] });
134
+ }
135
+ return of(null);
136
+ }),
137
+ // eslint-disable-next-line unicorn/no-useless-undefined
138
+ startWith(undefined))), share());
139
+ /**
140
+ * {@link Observable} изменения состояния загрузки данных.
141
+ */
142
+ this.loading$ = this.request$.pipe(map((value) => value === undefined));
143
+ /**
144
+ * Управляет состоянием загруженных файлов.
145
+ */
146
+ this.loadedFile = new FormControl([], [this.maxFilesLength()]);
147
+ /**
148
+ * Observable, который отслеживает, приняты ли файлы в соответствии с требованиями по типу размеру и т.д.
149
+ */
150
+ this.accepted$ = this.loadedFile.valueChanges.pipe(map(() => tuiFilesAccepted(this.loadedFile)));
151
+ /**
152
+ * Список файлов, которые были отклонены при загрузке.
153
+ */
154
+ this.rejected = [];
155
+ merge(this.accepted$, this.user$, this.loadedFile.valueChanges.pipe(tap((files) => {
156
+ this.form.controls.files.reset([]);
157
+ this.onUploadFile(files);
158
+ })))
159
+ .pipe(takeUntilDestroyed())
160
+ .subscribe();
161
+ // Валидация полей ввода в зависимости от формы.
162
+ effect(() => {
163
+ switch (this.formSlag()) {
164
+ case 'common':
165
+ this.form.get('email')?.setValidators([Validators.required, Validators.email]);
166
+ this.form.get('phone')?.clearValidators();
167
+ break;
168
+ case 'contacts':
169
+ case 'vacancies':
170
+ this.form.get('email')?.clearValidators();
171
+ this.form.get('phone')?.setValidators([Validators.required, Validators.minLength(12)]);
172
+ break;
173
+ case 'samples':
174
+ this.form.get('email')?.setValidators([Validators.required, Validators.email]);
175
+ this.form.get('phone')?.setValidators([Validators.required, Validators.minLength(12)]);
176
+ break;
177
+ default:
178
+ this.form.get('email')?.clearValidators();
179
+ this.form.get('phone')?.clearValidators();
180
+ break;
181
+ }
182
+ // Обновляем состояние валидации
183
+ this.form.get('email')?.updateValueAndValidity();
184
+ this.form.get('phone')?.updateValueAndValidity();
185
+ });
186
+ }
187
+ /**
188
+ * Обрабатывает отклоненные файлы.
189
+ *
190
+ * @param files Список файлов, которые были отклонены.
191
+ */
192
+ onReject(files) {
193
+ this.rejected = [...new Set([...this.rejected, ...files])];
194
+ }
195
+ /**
196
+ * Удаляет файл из списка загруженных и отклоненных файлов.
197
+ *
198
+ * @param file Файл, который нужно удалить.
199
+ */
200
+ onRemove(file) {
201
+ this.rejected = this.rejected.filter((rejected) => rejected !== file);
202
+ this.loadedFile.setValue(this.loadedFile.value?.filter((current) => current !== file) ?? []);
203
+ }
204
+ /**
205
+ * Загружает файл в форму обратной связи.
206
+ *
207
+ * @param files Набор файлов.
208
+ */
209
+ onUploadFile(files) {
210
+ files?.forEach((file) => {
211
+ const reader = new FileReader();
212
+ reader.addEventListener('load', () => {
213
+ if (reader.result) {
214
+ // eslint-disable-next-line @typescript-eslint/no-base-to-string,sonarjs/no-base-to-string
215
+ this.form.controls.files.setValue([...(this.form.controls.files.value ?? []), new ScUploadedFile(file.name, reader.result.toString())]);
216
+ }
217
+ });
218
+ reader.readAsDataURL(file);
219
+ });
220
+ }
221
+ /**
222
+ * Проверяет количество файлов загруженных пользователем.
223
+ *
224
+ * @param maxLength максимальное допустимое количество файлов для отправки.
225
+ */
226
+ maxFilesLength(maxLength = this.maxFilesInForm) {
227
+ return ({ value }) => value.length > maxLength ? { maxLength: new TuiValidationError(`Максимальное количество файлов - ${maxLength}`) } : null;
228
+ }
229
+ /**
230
+ * Выполняет очистку полей формы.
231
+ */
232
+ resetForm() {
233
+ this.form.reset({
234
+ email: null,
235
+ name: null,
236
+ phone: null,
237
+ message: null,
238
+ files: [],
239
+ });
240
+ }
241
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ScFeedbackFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
242
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: ScFeedbackFormComponent, isStandalone: true, selector: "sc-feedback-form", inputs: { formSlag: { classPropertyName: "formSlag", publicName: "formSlag", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { onClickOffer: "onClickOffer", onSendSuccess: "onSendSuccess" }, ngImport: i0, template: "@let formSlug = formSlag();\n<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col justify-between gap-4\"\n>\n <!-- \u0424\u0418\u041E -->\n <label tuiLabel>\n \u0424\u0418\u041E\n <tui-input formControlName=\"name\"> \u0423\u043A\u0430\u0436\u0438\u0442\u0435 \u0412\u0430\u0448\u0438 \u0424.\u0418.\u041E. </tui-input>\n <tui-error\n formControlName=\"name\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0422\u0435\u043B\u0435\u0444\u043E\u043D -->\n <label\n *ngIf=\"formSlug === 'contacts' || formSlug === 'vacancies'\"\n tuiLabel\n >\n \u0422\u0435\u043B\u0435\u0444\u043E\u043D\n <tui-input-phone formControlName=\"phone\"> \u0422\u0435\u043B\u0435\u0444\u043E\u043D </tui-input-phone>\n <tui-error\n formControlName=\"phone\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B -->\n <label\n *ngIf=\"formSlug !== 'contacts' && formSlug !== 'vacancies'\"\n tuiLabel\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 formControlName=\"email\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439 -->\n <label\n *ngIf=\"formSlug === 'common'\"\n tuiLabel\n >\n \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439\n <tui-textarea formControlName=\"message\"> \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439 </tui-textarea>\n <tui-error\n formControlName=\"message\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0424\u0430\u0439\u043B\u044B -->\n <div\n *ngIf=\"formSlug === 'vacancies'\"\n class=\"flex flex-col gap-1\"\n >\n <label tuiInputFiles>\n <input\n #validator=\"tuiInputFilesValidator\"\n accept=\".doc,.docx,.pdf,image/*\"\n tuiInputFiles\n [formControl]=\"loadedFile\"\n [multiple]=\"true\"\n (reject)=\"onReject($event)\"\n />\n </label>\n\n <tui-error\n *ngIf=\"loadedFile.invalid\"\n [error]=\"['maxLength'] | tuiFieldError | async\"\n [formControl]=\"loadedFile\"\n />\n\n @let accepted = accepted$ | async;\n <tui-files\n *ngIf=\"accepted?.length || rejected.length\"\n class=\"tui-space_top-1\"\n >\n <tui-file\n *ngFor=\"let file of accepted\"\n [file]=\"file\"\n (remove)=\"onRemove(file)\"\n />\n <tui-file\n *ngFor=\"let file of rejected\"\n state=\"error\"\n [file]=\"(file | tuiFileRejected: validator | async) || file\"\n (remove)=\"onRemove(file)\"\n />\n </tui-files>\n </div>\n\n <!-- recaptcha -->\n @if (form.get('recaptcha')) {\n <div\n class=\"mb-3 max-h-20 overflow-hidden duration-500\"\n [ngClass]=\"{ '!max-h-0 !-m-2': form.get('recaptcha')?.valid }\"\n >\n <re-captcha formControlName=\"recaptcha\" />\n </div>\n }\n\n <!-- \u042D\u043B\u0435\u043C\u0435\u043D\u0442\u044B \u0443\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F -->\n <div class=\"flex gap-3\">\n <button\n tuiButton\n [disabled]=\"form.invalid\"\n [loading]=\"!!(loading$ | async)\"\n type=\"submit\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n >\n \u041E\u0441\u0442\u0430\u0432\u0438\u0442\u044C\n </button>\n <div class=\"flex items-center\">\n <span class=\"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(); onClickOffer.emit()\"\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 .\n </span>\n </div>\n </div>\n</form>\n", dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgFor, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "pipe", type: TuiFieldErrorPipe, name: "tuiFieldError" }, { kind: "ngmodule", type: ScFormFieldsModule }, { kind: "component", type: i2.ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: ["type"], outputs: ["selectedClick"] }, { kind: "ngmodule", type: TuiInputModule }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "ngmodule", type: TuiInputPhoneModule }, { kind: "component", type: i3.TuiInputPhoneComponent, selector: "tui-input-phone", inputs: ["phoneMaskAfterCountryCode", "allowText", "search", "countryCode"], outputs: ["searchChange"] }, { kind: "directive", type: i3.TuiInputPhoneDirective, selector: "tui-input-phone" }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "directive", type: TuiLabel, selector: "label[tuiLabel]" }, { kind: "ngmodule", type: RecaptchaFormsModule }, { kind: "directive", type: i4.RecaptchaValueAccessorDirective, selector: "re-captcha[formControlName],re-captcha[formControl],re-captcha[ngModel]" }, { kind: "ngmodule", type: RecaptchaModule }, { kind: "component", type: i4.RecaptchaComponent, selector: "re-captcha", inputs: ["id", "siteKey", "theme", "type", "size", "tabIndex", "badge", "errorMode"], outputs: ["resolved", "error", "errored"], exportAs: ["reCaptcha"] }, { kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: TuiTextareaModule }, { kind: "component", type: i3.TuiTextareaComponent, selector: "tui-textarea", inputs: ["rows", "maxLength", "expandable"] }, { kind: "directive", type: i3.TuiTextareaDirective, selector: "tui-textarea" }, { kind: "component", type: i5.TuiFile, selector: "tui-file,a[tuiFile],button[tuiFile]", inputs: ["file", "state", "size", "showDelete", "showSize", "leftContent"], outputs: ["remove"] }, { kind: "component", type: i5.TuiInputFiles, selector: "label[tuiInputFiles]" }, { kind: "component", type: i5.TuiFilesComponent, selector: "tui-files", inputs: ["max", "expanded"], outputs: ["expandedChange"] }, { kind: "pipe", type: i5.TuiFileRejectedPipe, name: "tuiFileRejected" }, { kind: "directive", type: i5.TuiInputFilesDirective, selector: "input[tuiInputFiles]", outputs: ["reject"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
243
+ }
244
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ScFeedbackFormComponent, decorators: [{
245
+ type: Component,
246
+ args: [{ standalone: true, selector: 'sc-feedback-form', imports: [
247
+ AsyncPipe,
248
+ ReactiveFormsModule,
249
+ NgClass,
250
+ NgFor,
251
+ TuiError,
252
+ TuiFieldErrorPipe,
253
+ ScFormFieldsModule,
254
+ TuiInputModule,
255
+ TuiInputPhoneModule,
256
+ TuiButton,
257
+ TuiButtonLoading,
258
+ TuiLabel,
259
+ RecaptchaFormsModule,
260
+ RecaptchaModule,
261
+ TuiLink,
262
+ NgIf,
263
+ TuiTextareaModule,
264
+ TuiFiles,
265
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let formSlug = formSlag();\n<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col justify-between gap-4\"\n>\n <!-- \u0424\u0418\u041E -->\n <label tuiLabel>\n \u0424\u0418\u041E\n <tui-input formControlName=\"name\"> \u0423\u043A\u0430\u0436\u0438\u0442\u0435 \u0412\u0430\u0448\u0438 \u0424.\u0418.\u041E. </tui-input>\n <tui-error\n formControlName=\"name\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0422\u0435\u043B\u0435\u0444\u043E\u043D -->\n <label\n *ngIf=\"formSlug === 'contacts' || formSlug === 'vacancies'\"\n tuiLabel\n >\n \u0422\u0435\u043B\u0435\u0444\u043E\u043D\n <tui-input-phone formControlName=\"phone\"> \u0422\u0435\u043B\u0435\u0444\u043E\u043D </tui-input-phone>\n <tui-error\n formControlName=\"phone\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B -->\n <label\n *ngIf=\"formSlug !== 'contacts' && formSlug !== 'vacancies'\"\n tuiLabel\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 formControlName=\"email\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439 -->\n <label\n *ngIf=\"formSlug === 'common'\"\n tuiLabel\n >\n \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439\n <tui-textarea formControlName=\"message\"> \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439 </tui-textarea>\n <tui-error\n formControlName=\"message\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0424\u0430\u0439\u043B\u044B -->\n <div\n *ngIf=\"formSlug === 'vacancies'\"\n class=\"flex flex-col gap-1\"\n >\n <label tuiInputFiles>\n <input\n #validator=\"tuiInputFilesValidator\"\n accept=\".doc,.docx,.pdf,image/*\"\n tuiInputFiles\n [formControl]=\"loadedFile\"\n [multiple]=\"true\"\n (reject)=\"onReject($event)\"\n />\n </label>\n\n <tui-error\n *ngIf=\"loadedFile.invalid\"\n [error]=\"['maxLength'] | tuiFieldError | async\"\n [formControl]=\"loadedFile\"\n />\n\n @let accepted = accepted$ | async;\n <tui-files\n *ngIf=\"accepted?.length || rejected.length\"\n class=\"tui-space_top-1\"\n >\n <tui-file\n *ngFor=\"let file of accepted\"\n [file]=\"file\"\n (remove)=\"onRemove(file)\"\n />\n <tui-file\n *ngFor=\"let file of rejected\"\n state=\"error\"\n [file]=\"(file | tuiFileRejected: validator | async) || file\"\n (remove)=\"onRemove(file)\"\n />\n </tui-files>\n </div>\n\n <!-- recaptcha -->\n @if (form.get('recaptcha')) {\n <div\n class=\"mb-3 max-h-20 overflow-hidden duration-500\"\n [ngClass]=\"{ '!max-h-0 !-m-2': form.get('recaptcha')?.valid }\"\n >\n <re-captcha formControlName=\"recaptcha\" />\n </div>\n }\n\n <!-- \u042D\u043B\u0435\u043C\u0435\u043D\u0442\u044B \u0443\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F -->\n <div class=\"flex gap-3\">\n <button\n tuiButton\n [disabled]=\"form.invalid\"\n [loading]=\"!!(loading$ | async)\"\n type=\"submit\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n >\n \u041E\u0441\u0442\u0430\u0432\u0438\u0442\u044C\n </button>\n <div class=\"flex items-center\">\n <span class=\"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(); onClickOffer.emit()\"\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 .\n </span>\n </div>\n </div>\n</form>\n" }]
266
+ }], ctorParameters: () => [] });
267
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-feedback-form.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/feedback/feedback-form/sc-feedback-form.component.ts","../../../../../projects/client-ui/feedback/feedback-form/sc-feedback-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgClass, NgFor, NgIf } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, effect, inject, input, InputSignal, output, OutputEmitterRef } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { AbstractControl, FormControl, FormGroup, ReactiveFormsModule, ValidatorFn, Validators } from '@angular/forms';\nimport { ScConvertersService, ScFeedbackForm, ScFeedbackService, ScIFeedbackMessage, ScISuggestionType, ScUploadedFile, ScUser, ScUserService } from '@snabcentr/client-core';\nimport { TuiValidationError } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiError, TuiLabel, TuiLink } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFieldErrorPipe, TuiFileLike, TuiFiles, tuiFilesAccepted } from '@taiga-ui/kit';\nimport { TuiInputModule, TuiInputPhoneModule, TuiTextareaModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { RecaptchaFormsModule, RecaptchaModule } from 'ng-recaptcha-2';\nimport { ApiErrorResponse } from 'projects/client-ui/auth';\nimport { ScFormFieldsModule } from 'projects/client-ui/form-fields';\nimport { MAX_FILES_IN_FORM_INPUT } from 'projects/client-ui/providers/sc-max-files-in-form-input';\nimport { catchError, filter, map, merge, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\n/**\n * Компонент формы обратной связи.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-feedback-form',\n    templateUrl: './sc-feedback-form.component.html',\n    imports: [\n        AsyncPipe,\n        ReactiveFormsModule,\n        NgClass,\n        NgFor,\n        TuiError,\n        TuiFieldErrorPipe,\n        ScFormFieldsModule,\n        TuiInputModule,\n        TuiInputPhoneModule,\n        TuiButton,\n        TuiButtonLoading,\n        TuiLabel,\n        RecaptchaFormsModule,\n        RecaptchaModule,\n        TuiLink,\n        NgIf,\n        TuiTextareaModule,\n        TuiFiles,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScFeedbackFormComponent {\n    /**\n     * Условное название блока где расположена ссылка.\n     */\n    public readonly formSlag: InputSignal<keyof typeof ScFeedbackForm> = input.required<keyof typeof ScFeedbackForm>();\n\n    /**\n     * Событие нажатия на кнопку договора оферты.\n     */\n    public readonly onClickOffer: OutputEmitterRef<void> = output();\n\n    /**\n     * Событие успешной отправки формы.\n     */\n    public readonly onSendSuccess: OutputEmitterRef<void> = output();\n\n    /**\n     * Сервис обратной связи.\n     */\n    private readonly feedbackService: ScFeedbackService = inject(ScFeedbackService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /**\n     * Максимальное количество файлов, отправляемых в форме.\n     */\n    private readonly maxFilesInForm: number = inject(MAX_FILES_IN_FORM_INPUT);\n\n    /**\n     * Сервис данных о пользователе.\n     */\n    private readonly user$: Observable<ScUser> = inject(ScUserService)\n        .getUserChange$()\n        .pipe(\n            tap((user) => {\n                if (user.isGuest) {\n                    (this.form as FormGroup).addControl('recaptcha', this.recaptchaControl);\n                    this.resetForm();\n                } else {\n                    (this.form as FormGroup).removeControl('recaptcha');\n                    // eslint-disable-next-line default-case\n                    switch (this.formSlag()) {\n                        case 'common':\n                            this.form.patchValue({ name: user.name, email: user.contacts.email.value });\n                            break;\n\n                        case 'contacts':\n                        case 'vacancies':\n                            this.form.patchValue({ name: user.name, phone: user.contacts.phone.value });\n                            break;\n\n                        case 'samples':\n                            this.form.patchValue({ name: user.name, phone: user.contacts.phone.value, email: user.contacts.email.value });\n                            break;\n                    }\n                }\n            })\n        );\n\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { onClickOfferHandler: () => void; formSlag: keyof typeof ScFeedbackForm }> | null = inject<\n        TuiDialogContext<void, { onClickOfferHandler: () => void; formSlag: keyof typeof ScFeedbackForm }>\n    >(POLYMORPHEUS_CONTEXT, { optional: true });\n\n    /**\n     * Поле ввода для recaptcha.\n     */\n    protected readonly recaptchaControl: FormControl<string | null> = new FormControl<string | null>(null);\n\n    /**\n     * Группа полей ввода для формы «Пожелания и предложения по улучшению сайта».\n     */\n    protected readonly form = new FormGroup({\n        name: new FormControl<string | null>(null, [Validators.required, Validators.minLength(3)]),\n        phone: new FormControl<string | null>(null),\n        email: new FormControl<string | null>(null),\n        message: new FormControl<string | null>(null, []),\n        files: new FormControl<ScUploadedFile[] | null>([], [this.maxFilesLength(5)]),\n    });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на отправку данных.\n     */\n    protected readonly request$: Observable<undefined | null> = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIFeedbackMessage),\n        switchMap((value) =>\n            this.feedbackService.sendFeedback(this.context?.data.formSlag ?? this.formSlag(), this.convertersService.removeNullRecursive(value)).pipe(\n                tap(() => {\n                    if (this.context) {\n                        this.context.completeWith();\n                    } else {\n                        this.resetForm();\n                        this.onSendSuccess.emit();\n                    }\n                }),\n                // eslint-disable-next-line rxjs/no-implicit-any-catch\n                catchError((error: HttpErrorResponse) => {\n                    this.form.markAllAsTouched();\n\n                    const { errors, message } = error.error as ApiErrorResponse;\n\n                    if (errors) {\n                        Object.keys(errors).forEach((key) => {\n                            const control = this.form.get(key);\n\n                            if (control) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                control.setErrors({ serverResponse: errors[key] });\n                            }\n                        });\n                    } else if (message) {\n                        this.form.setErrors({ serverResponse: [message] });\n                    }\n\n                    return of(null);\n                }),\n                // eslint-disable-next-line unicorn/no-useless-undefined\n                startWith(undefined)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных.\n     */\n    protected readonly loading$: Observable<boolean> = this.request$.pipe(map((value) => value === undefined));\n\n    /**\n     * Управляет состоянием загруженных файлов.\n     */\n    protected readonly loadedFile: FormControl<TuiFileLike[] | null> = new FormControl<TuiFileLike[] | null>([], [this.maxFilesLength()]);\n\n    /**\n     * Observable, который отслеживает, приняты ли файлы в соответствии с требованиями по типу размеру и т.д.\n     */\n    protected readonly accepted$: Observable<File[]> = this.loadedFile.valueChanges.pipe(map(() => tuiFilesAccepted(this.loadedFile)));\n\n    /**\n     * Список файлов, которые были отклонены при загрузке.\n     */\n    protected rejected: readonly File[] = [];\n\n    /**\n     * Инициализирует экземпляр класса {@link ScFeedbackFormComponent}.\n     */\n    public constructor() {\n        merge(\n            this.accepted$,\n            this.user$,\n            this.loadedFile.valueChanges.pipe(\n                tap((files) => {\n                    this.form.controls.files.reset([]);\n                    this.onUploadFile(files);\n                })\n            )\n        )\n            .pipe(takeUntilDestroyed())\n            .subscribe();\n\n        // Валидация полей ввода в зависимости от формы.\n        effect(() => {\n            switch (this.formSlag()) {\n                case 'common':\n                    this.form.get('email')?.setValidators([Validators.required, Validators.email]);\n                    this.form.get('phone')?.clearValidators();\n                    break;\n\n                case 'contacts':\n                case 'vacancies':\n                    this.form.get('email')?.clearValidators();\n                    this.form.get('phone')?.setValidators([Validators.required, Validators.minLength(12)]);\n                    break;\n\n                case 'samples':\n                    this.form.get('email')?.setValidators([Validators.required, Validators.email]);\n                    this.form.get('phone')?.setValidators([Validators.required, Validators.minLength(12)]);\n                    break;\n\n                default:\n                    this.form.get('email')?.clearValidators();\n                    this.form.get('phone')?.clearValidators();\n                    break;\n            }\n\n            // Обновляем состояние валидации\n            this.form.get('email')?.updateValueAndValidity();\n            this.form.get('phone')?.updateValueAndValidity();\n        });\n    }\n\n    /**\n     * Обрабатывает отклоненные файлы.\n     *\n     * @param files Список файлов, которые были отклонены.\n     */\n    protected onReject(files: readonly File[]): void {\n        this.rejected = [...new Set([...this.rejected, ...files])];\n    }\n\n    /**\n     * Удаляет файл из списка загруженных и отклоненных файлов.\n     *\n     * @param file Файл, который нужно удалить.\n     */\n    protected onRemove(file: File): void {\n        this.rejected = this.rejected.filter((rejected) => rejected !== file);\n        this.loadedFile.setValue(this.loadedFile.value?.filter((current) => current !== file) ?? []);\n    }\n\n    /**\n     * Загружает файл в форму обратной связи.\n     *\n     * @param files Набор файлов.\n     */\n    private onUploadFile(files: TuiFileLike[] | null): void {\n        files?.forEach((file) => {\n            const reader = new FileReader();\n\n            reader.addEventListener('load', () => {\n                if (reader.result) {\n                    // eslint-disable-next-line @typescript-eslint/no-base-to-string,sonarjs/no-base-to-string\n                    this.form.controls.files.setValue([...(this.form.controls.files.value ?? []), new ScUploadedFile(file.name, reader.result.toString())]);\n                }\n            });\n\n            reader.readAsDataURL(file as unknown as Blob);\n        });\n    }\n\n    /**\n     * Проверяет количество файлов загруженных пользователем.\n     *\n     * @param maxLength максимальное допустимое количество файлов для отправки.\n     */\n    private maxFilesLength(maxLength: number = this.maxFilesInForm): ValidatorFn {\n        return ({ value }: AbstractControl<TuiFileLike[]>) =>\n            value.length > maxLength ? { maxLength: new TuiValidationError(`Максимальное количество файлов - ${maxLength}`) } : null;\n    }\n\n    /**\n     * Выполняет очистку полей формы.\n     */\n    private resetForm(): void {\n        this.form.reset({\n            email: null,\n            name: null,\n            phone: null,\n            message: null,\n            files: [],\n        });\n    }\n}\n","@let formSlug = formSlag();\n<form\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"flex flex-col justify-between gap-4\"\n>\n    <!-- ФИО -->\n    <label tuiLabel>\n        ФИО\n        <tui-input formControlName=\"name\"> Укажите Ваши Ф.И.О. </tui-input>\n        <tui-error\n            formControlName=\"name\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Телефон -->\n    <label\n        *ngIf=\"formSlug === 'contacts' || formSlug === 'vacancies'\"\n        tuiLabel\n    >\n        Телефон\n        <tui-input-phone formControlName=\"phone\"> Телефон </tui-input-phone>\n        <tui-error\n            formControlName=\"phone\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Адрес электронной почты -->\n    <label\n        *ngIf=\"formSlug !== 'contacts' && formSlug !== 'vacancies'\"\n        tuiLabel\n    >\n        Адрес электронной почты\n        <tui-input formControlName=\"email\">\n            Адрес электронной почты\n            <sc-suggestion-field\n                *tuiDataList\n                [type]=\"suggestionType.email\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"email\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Комментарий -->\n    <label\n        *ngIf=\"formSlug === 'common'\"\n        tuiLabel\n    >\n        Комментарий\n        <tui-textarea formControlName=\"message\"> Комментарий </tui-textarea>\n        <tui-error\n            formControlName=\"message\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Файлы -->\n    <div\n        *ngIf=\"formSlug === 'vacancies'\"\n        class=\"flex flex-col gap-1\"\n    >\n        <label tuiInputFiles>\n            <input\n                #validator=\"tuiInputFilesValidator\"\n                accept=\".doc,.docx,.pdf,image/*\"\n                tuiInputFiles\n                [formControl]=\"loadedFile\"\n                [multiple]=\"true\"\n                (reject)=\"onReject($event)\"\n            />\n        </label>\n\n        <tui-error\n            *ngIf=\"loadedFile.invalid\"\n            [error]=\"['maxLength'] | tuiFieldError | async\"\n            [formControl]=\"loadedFile\"\n        />\n\n        @let accepted = accepted$ | async;\n        <tui-files\n            *ngIf=\"accepted?.length || rejected.length\"\n            class=\"tui-space_top-1\"\n        >\n            <tui-file\n                *ngFor=\"let file of accepted\"\n                [file]=\"file\"\n                (remove)=\"onRemove(file)\"\n            />\n            <tui-file\n                *ngFor=\"let file of rejected\"\n                state=\"error\"\n                [file]=\"(file | tuiFileRejected: validator | async) || file\"\n                (remove)=\"onRemove(file)\"\n            />\n        </tui-files>\n    </div>\n\n    <!-- recaptcha -->\n    @if (form.get('recaptcha')) {\n        <div\n            class=\"mb-3 max-h-20 overflow-hidden duration-500\"\n            [ngClass]=\"{ '!max-h-0 !-m-2': form.get('recaptcha')?.valid }\"\n        >\n            <re-captcha formControlName=\"recaptcha\" />\n        </div>\n    }\n\n    <!-- Элементы управления  -->\n    <div class=\"flex gap-3\">\n        <button\n            tuiButton\n            [disabled]=\"form.invalid\"\n            [loading]=\"!!(loading$ | async)\"\n            type=\"submit\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n        >\n            Оставить\n        </button>\n        <div class=\"flex items-center\">\n            <span class=\"text-body-s\">\n                Нажимая на кнопку, вы соглашаетесь с\n                <button\n                    tuiLink\n                    [pseudo]=\"true\"\n                    type=\"button\"\n                    (click)=\"context?.data?.onClickOfferHandler(); onClickOffer.emit()\"\n                >\n                    Политикой конфиденциальности\n                </button>\n                .\n            </span>\n        </div>\n    </div>\n</form>\n"]}
@@ -0,0 +1,32 @@
1
+ import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
2
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
+ import { TuiIcon } from '@taiga-ui/core';
4
+ import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
5
+ import { EMPTY, of, switchMap, timer } from 'rxjs';
6
+ import * as i0 from "@angular/core";
7
+ /**
8
+ * Компонент «Спасибо!».
9
+ */
10
+ export class ScGratitudeComponent {
11
+ /**
12
+ * Инициализирует экземпляр класса {@link ScGratitudeComponent}.
13
+ */
14
+ constructor() {
15
+ /**
16
+ * Контекст диалогового окна, в котором открыт компонент.
17
+ */
18
+ this.context = inject(POLYMORPHEUS_CONTEXT, { optional: true });
19
+ of(this.context?.data.autoClose)
20
+ .pipe(switchMap((autoClose) => (autoClose ? timer(autoClose) : EMPTY)), takeUntilDestroyed())
21
+ .subscribe(() => {
22
+ this.context?.$implicit.complete();
23
+ });
24
+ }
25
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ScGratitudeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
26
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.12", type: ScGratitudeComponent, isStandalone: true, selector: "sc-gratitude", ngImport: i0, template: "<div class=\"flex flex-col items-center gap-4 py-14 text-center\">\n <tui-icon\n icon=\"@tui.circle-check\"\n class=\"text-tui-success-fill !size-20 !text-5xl\"\n />\n <p class=\"text-h5\">\u0421\u043F\u0430\u0441\u0438\u0431\u043E!</p>\n <p\n [innerHTML]=\"context?.data?.subTitle ?? '\u0412\u0430\u0448 \u0434\u0430\u043D\u043D\u044B\u0435 \u0443\u0441\u043F\u0435\u0448\u043D\u043E \u043E\u0442\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u044B. \u041C\u044B \u0441\u0432\u044F\u0436\u0435\u043C\u0441\u044F \u0441 \u0432\u0430\u043C\u0438 \u0432 \u0431\u043B\u0438\u0436\u0430\u0439\u0448\u0435\u0435 \u0432\u0440\u0435\u043C\u044F.'\"\n class=\"text-body-m mx-2\"\n ></p>\n</div>\n", dependencies: [{ kind: "component", type: TuiIcon, selector: "tui-icon", inputs: ["icon", "background"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
27
+ }
28
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ScGratitudeComponent, decorators: [{
29
+ type: Component,
30
+ args: [{ standalone: true, selector: 'sc-gratitude', imports: [TuiIcon], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center gap-4 py-14 text-center\">\n <tui-icon\n icon=\"@tui.circle-check\"\n class=\"text-tui-success-fill !size-20 !text-5xl\"\n />\n <p class=\"text-h5\">\u0421\u043F\u0430\u0441\u0438\u0431\u043E!</p>\n <p\n [innerHTML]=\"context?.data?.subTitle ?? '\u0412\u0430\u0448 \u0434\u0430\u043D\u043D\u044B\u0435 \u0443\u0441\u043F\u0435\u0448\u043D\u043E \u043E\u0442\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u044B. \u041C\u044B \u0441\u0432\u044F\u0436\u0435\u043C\u0441\u044F \u0441 \u0432\u0430\u043C\u0438 \u0432 \u0431\u043B\u0438\u0436\u0430\u0439\u0448\u0435\u0435 \u0432\u0440\u0435\u043C\u044F.'\"\n class=\"text-body-m mx-2\"\n ></p>\n</div>\n" }]
31
+ }], ctorParameters: () => [] });
32
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtZ3JhdGl0dWRlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9mZWVkYmFjay9ncmF0aXR1ZGUvc2MtZ3JhdGl0dWRlLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9mZWVkYmFjay9ncmF0aXR1ZGUvc2MtZ3JhdGl0dWRlLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNFLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ2hFLE9BQU8sRUFBb0IsT0FBTyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDM0QsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDOUQsT0FBTyxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLE1BQU0sQ0FBQzs7QUFFbkQ7O0dBRUc7QUFRSCxNQUFNLE9BQU8sb0JBQW9CO0lBTTdCOztPQUVHO0lBQ0g7UUFSQTs7V0FFRztRQUNnQixZQUFPLEdBQUcsTUFBTSxDQUFrRSxvQkFBb0IsRUFBRSxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDO1FBTTNJLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUM7YUFDM0IsSUFBSSxDQUNELFNBQVMsQ0FBQyxDQUFDLFNBQVMsRUFBRSxFQUFFLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsRUFDaEUsa0JBQWtCLEVBQUUsQ0FDdkI7YUFDQSxTQUFTLENBQUMsR0FBRyxFQUFFO1lBQ1osSUFBSSxDQUFDLE9BQU8sRUFBRSxTQUFTLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDdkMsQ0FBQyxDQUFDLENBQUM7SUFDWCxDQUFDOytHQWxCUSxvQkFBb0I7bUdBQXBCLG9CQUFvQix3RUNoQmpDLHl0QkFXQSw0Q0RFYyxPQUFPOzs0RkFHUixvQkFBb0I7a0JBUGhDLFNBQVM7aUNBQ00sSUFBSSxZQUNOLGNBQWMsV0FFZixDQUFDLE9BQU8sQ0FBQyxtQkFDRCx1QkFBdUIsQ0FBQyxNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyB0YWtlVW50aWxEZXN0cm95ZWQgfSBmcm9tICdAYW5ndWxhci9jb3JlL3J4anMtaW50ZXJvcCc7XG5pbXBvcnQgeyBUdWlEaWFsb2dDb250ZXh0LCBUdWlJY29uIH0gZnJvbSAnQHRhaWdhLXVpL2NvcmUnO1xuaW1wb3J0IHsgUE9MWU1PUlBIRVVTX0NPTlRFWFQgfSBmcm9tICdAdGFpZ2EtdWkvcG9seW1vcnBoZXVzJztcbmltcG9ydCB7IEVNUFRZLCBvZiwgc3dpdGNoTWFwLCB0aW1lciB9IGZyb20gJ3J4anMnO1xuXG4vKipcbiAqINCa0L7QvNC/0L7QvdC10L3RgiDCq9Ch0L/QsNGB0LjQsdC+IcK7LlxuICovXG5AQ29tcG9uZW50KHtcbiAgICBzdGFuZGFsb25lOiB0cnVlLFxuICAgIHNlbGVjdG9yOiAnc2MtZ3JhdGl0dWRlJyxcbiAgICB0ZW1wbGF0ZVVybDogJy4vc2MtZ3JhdGl0dWRlLmNvbXBvbmVudC5odG1sJyxcbiAgICBpbXBvcnRzOiBbVHVpSWNvbl0sXG4gICAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIFNjR3JhdGl0dWRlQ29tcG9uZW50IHtcbiAgICAvKipcbiAgICAgKiDQmtC+0L3RgtC10LrRgdGCINC00LjQsNC70L7Qs9C+0LLQvtCz0L4g0L7QutC90LAsINCyINC60L7RgtC+0YDQvtC8INC+0YLQutGA0YvRgiDQutC+0LzQv9C+0L3QtdC90YIuXG4gICAgICovXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IGNvbnRleHQgPSBpbmplY3Q8VHVpRGlhbG9nQ29udGV4dDx2b2lkLCB7IGF1dG9DbG9zZTogbnVtYmVyOyBzdWJUaXRsZTogc3RyaW5nIH0+PihQT0xZTU9SUEhFVVNfQ09OVEVYVCwgeyBvcHRpb25hbDogdHJ1ZSB9KTtcblxuICAgIC8qKlxuICAgICAqINCY0L3QuNGG0LjQsNC70LjQt9C40YDRg9C10YIg0Y3QutC30LXQvNC/0LvRj9GAINC60LvQsNGB0YHQsCB7QGxpbmsgU2NHcmF0aXR1ZGVDb21wb25lbnR9LlxuICAgICAqL1xuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcigpIHtcbiAgICAgICAgb2YodGhpcy5jb250ZXh0Py5kYXRhLmF1dG9DbG9zZSlcbiAgICAgICAgICAgIC5waXBlKFxuICAgICAgICAgICAgICAgIHN3aXRjaE1hcCgoYXV0b0Nsb3NlKSA9PiAoYXV0b0Nsb3NlID8gdGltZXIoYXV0b0Nsb3NlKSA6IEVNUFRZKSksXG4gICAgICAgICAgICAgICAgdGFrZVVudGlsRGVzdHJveWVkKClcbiAgICAgICAgICAgIClcbiAgICAgICAgICAgIC5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgICAgICAgICAgIHRoaXMuY29udGV4dD8uJGltcGxpY2l0LmNvbXBsZXRlKCk7XG4gICAgICAgICAgICB9KTtcbiAgICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwiZmxleCBmbGV4LWNvbCBpdGVtcy1jZW50ZXIgZ2FwLTQgcHktMTQgdGV4dC1jZW50ZXJcIj5cbiAgICA8dHVpLWljb25cbiAgICAgICAgaWNvbj1cIkB0dWkuY2lyY2xlLWNoZWNrXCJcbiAgICAgICAgY2xhc3M9XCJ0ZXh0LXR1aS1zdWNjZXNzLWZpbGwgIXNpemUtMjAgIXRleHQtNXhsXCJcbiAgICAvPlxuICAgIDxwIGNsYXNzPVwidGV4dC1oNVwiPtCh0L/QsNGB0LjQsdC+ITwvcD5cbiAgICA8cFxuICAgICAgICBbaW5uZXJIVE1MXT1cImNvbnRleHQ/LmRhdGE/LnN1YlRpdGxlID8/ICfQktCw0Ygg0LTQsNC90L3Ri9C1INGD0YHQv9C10YjQvdC+INC+0YLQv9GA0LDQstC70LXQvdGLLiDQnNGLINGB0LLRj9C20LXQvNGB0Y8g0YEg0LLQsNC80Lgg0LIg0LHQu9C40LbQsNC50YjQtdC1INCy0YDQtdC80Y8uJ1wiXG4gICAgICAgIGNsYXNzPVwidGV4dC1ib2R5LW0gbXgtMlwiXG4gICAgPjwvcD5cbjwvZGl2PlxuIl19
@@ -0,0 +1,3 @@
1
+ export * from './feedback-form/sc-feedback-form.component';
2
+ export * from './gratitude/sc-gratitude.component';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvZmVlZGJhY2svaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyw0Q0FBNEMsQ0FBQztBQUMzRCxjQUFjLG9DQUFvQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9mZWVkYmFjay1mb3JtL3NjLWZlZWRiYWNrLWZvcm0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vZ3JhdGl0dWRlL3NjLWdyYXRpdHVkZS5jb21wb25lbnQnO1xuIl19