@snabcentr/client-ui 3.32.3 → 3.33.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/banner/sc-banner.module.d.ts +2 -1
- package/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.d.ts +1 -1
- package/cart/index.d.ts +1 -1
- package/cart/{sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component.d.ts → sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component.d.ts} +3 -3
- package/catalog/category-card/sc-category-card.component.d.ts +0 -6
- package/catalog/hover-image-carousel/{hover-image-carousel.component.d.ts → sc-hover-image-carousel.component.d.ts} +16 -7
- package/catalog/index.d.ts +1 -0
- package/catalog/sc-catalog.module.d.ts +3 -2
- package/directives/abstract-price-card/abstract-sc-price-card.directive.d.ts +4 -4
- package/esm2022/banner/sc-banner.component.mjs +3 -3
- package/esm2022/banner/sc-banner.module.mjs +4 -3
- package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +6 -6
- package/esm2022/cart/cart-item/sc-cart-item.component.mjs +4 -4
- package/esm2022/cart/index.mjs +2 -2
- package/esm2022/cart/sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component.mjs +110 -0
- package/esm2022/catalog/category-card/sc-category-card.component.mjs +4 -11
- package/esm2022/catalog/hover-image-carousel/sc-hover-image-carousel.component.mjs +76 -0
- package/esm2022/catalog/index.mjs +2 -1
- package/esm2022/catalog/price-card/sc-price-card.component.mjs +2 -2
- package/esm2022/catalog/price-card-inline/sc-price-card-inline.component.mjs +4 -4
- package/esm2022/catalog/sc-catalog.module.mjs +8 -5
- package/esm2022/directives/abstract-price-card/abstract-sc-price-card.directive.mjs +5 -6
- package/esm2022/news/news-card/sc-news-card.component.mjs +4 -8
- package/esm2022/news/sc-news.module.mjs +4 -3
- package/esm2022/order/order-item-mobile/sc-order-item-mobile.component.mjs +12 -20
- package/fesm2022/snabcentr-client-ui.mjs +67 -86
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/news/news-card/sc-news-card.component.d.ts +1 -5
- package/news/sc-news.module.d.ts +2 -1
- package/order/order-item-mobile/sc-order-item-mobile.component.d.ts +8 -14
- package/package.json +2 -2
- package/release_notes.tmp +5 -9
- package/styles/tailwind/tailwind.scss +0 -4
- package/esm2022/cart/sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component.mjs +0 -110
- package/esm2022/catalog/hover-image-carousel/hover-image-carousel.component.mjs +0 -76
@@ -7,11 +7,12 @@ import * as i5 from "@taiga-ui/cdk/directives/item";
|
|
7
7
|
import * as i6 from "@taiga-ui/kit";
|
8
8
|
import * as i7 from "@ng-web-apis/intersection-observer";
|
9
9
|
import * as i8 from "@taiga-ui/cdk";
|
10
|
+
import * as i9 from "@snabcentr/client-core";
|
10
11
|
/**
|
11
12
|
* Модуль баннеров.
|
12
13
|
*/
|
13
14
|
export declare class ScBannerModule {
|
14
15
|
static ɵfac: i0.ɵɵFactoryDeclaration<ScBannerModule, never>;
|
15
|
-
static ɵmod: i0.ɵɵNgModuleDeclaration<ScBannerModule, [typeof i1.ScBannerComponent], [typeof i2.CommonModule, typeof i3.RouterModule, typeof i4.TuiButton, typeof i5.TuiItem, typeof i6.TuiCarouselComponent, typeof i6.TuiCarouselDirective, typeof i6.TuiCarouselAutoscroll, typeof i6.TuiCarouselButtons, typeof i6.TuiCarouselScroll, typeof i4.TuiLoader, typeof i7.WaIntersectionObserverDirective, typeof i7.WaIntersectionObservee, typeof i7.WaIntersectionRoot, typeof i8.TuiLet], [typeof i1.ScBannerComponent]>;
|
16
|
+
static ɵmod: i0.ɵɵNgModuleDeclaration<ScBannerModule, [typeof i1.ScBannerComponent], [typeof i2.CommonModule, typeof i3.RouterModule, typeof i4.TuiButton, typeof i5.TuiItem, typeof i6.TuiCarouselComponent, typeof i6.TuiCarouselDirective, typeof i6.TuiCarouselAutoscroll, typeof i6.TuiCarouselButtons, typeof i6.TuiCarouselScroll, typeof i4.TuiLoader, typeof i7.WaIntersectionObserverDirective, typeof i7.WaIntersectionObservee, typeof i7.WaIntersectionRoot, typeof i8.TuiLet, typeof i9.ScMediaImageTransformerPipe], [typeof i1.ScBannerComponent]>;
|
16
17
|
static ɵinj: i0.ɵɵInjectorDeclaration<ScBannerModule>;
|
17
18
|
}
|
@@ -42,7 +42,7 @@ export declare class ScAddOrEditingCartItemFormComponent implements OnInit {
|
|
42
42
|
/**
|
43
43
|
* Минимальный метраж для товара.
|
44
44
|
*/
|
45
|
-
protected readonly minLength: import("@angular/core").Signal<number>;
|
45
|
+
protected readonly minLength: import("@angular/core").Signal<number | undefined>;
|
46
46
|
/**
|
47
47
|
* Максимальный метраж для товара.
|
48
48
|
*/
|
package/cart/index.d.ts
CHANGED
@@ -1,3 +1,3 @@
|
|
1
1
|
export * from './cart-item/sc-cart-item.component';
|
2
|
-
export * from './sc-
|
2
|
+
export * from './sc-cart-add-products-from-csv-dialog/sc-cart-add-products-from-csv-dialog.component';
|
3
3
|
export * from './add-or-editing-cart-item-dialog';
|
@@ -6,7 +6,7 @@ import * as i0 from "@angular/core";
|
|
6
6
|
/**
|
7
7
|
* Компонент диалога для добавления товаров в корзину из CSV файла.
|
8
8
|
*/
|
9
|
-
export declare class
|
9
|
+
export declare class ScCartAddProductsFromCsvDialogComponent {
|
10
10
|
/**
|
11
11
|
* Контекст диалогового окна, в котором открыт компонент.
|
12
12
|
*/
|
@@ -57,6 +57,6 @@ export declare class ScCarAddProductsFromCsvDialogComponent {
|
|
57
57
|
* @param blob Бинарный объект.
|
58
58
|
*/
|
59
59
|
private downloadExampleFile;
|
60
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<
|
61
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<
|
60
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScCartAddProductsFromCsvDialogComponent, never>;
|
61
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScCartAddProductsFromCsvDialogComponent, "sc-cart-add-products-from-csv-dialog", never, {}, {}, never, never, true, never>;
|
62
62
|
}
|
@@ -65,12 +65,6 @@ export declare class ScCategoryCardComponent {
|
|
65
65
|
* @param isHover Признак наведения на карточку.
|
66
66
|
*/
|
67
67
|
protected onHovered(isHover: boolean): void;
|
68
|
-
/**
|
69
|
-
* Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию ("product_not_found").
|
70
|
-
*
|
71
|
-
* @param category Данные о категории.
|
72
|
-
*/
|
73
|
-
protected getCategoryImgURL(category: ScCategory): string;
|
74
68
|
/**
|
75
69
|
* Устанавливает компонент в очередь на обновление.
|
76
70
|
*
|
@@ -1,21 +1,30 @@
|
|
1
|
-
import { WritableSignal } from '@angular/core';
|
1
|
+
import { InputSignal, Signal, WritableSignal } from '@angular/core';
|
2
|
+
import { ScIImage } from '@snabcentr/client-core';
|
2
3
|
import * as i0 from "@angular/core";
|
3
4
|
/**
|
4
5
|
* Компонент просмотра изображений через событие наведения.
|
5
6
|
*/
|
6
|
-
export declare class
|
7
|
+
export declare class ScHoverImageCarouselComponent {
|
7
8
|
/**
|
8
9
|
* Массив изображений.
|
9
10
|
*/
|
10
|
-
images:
|
11
|
+
readonly images: InputSignal<ScIImage[]>;
|
11
12
|
/**
|
12
13
|
* Признак что необходимо отобразить элементы управления.
|
13
14
|
*/
|
14
|
-
isShowActions: boolean
|
15
|
+
readonly isShowActions: InputSignal<boolean>;
|
15
16
|
/**
|
16
17
|
* {@link WritableSignal} для отслеживания текущего индекса изображения.
|
17
18
|
*/
|
18
|
-
currentIndex: WritableSignal<number>;
|
19
|
+
readonly currentIndex: WritableSignal<number>;
|
20
|
+
/**
|
21
|
+
* {@link WritableSignal} для отслеживания текущего индекса изображения.
|
22
|
+
*/
|
23
|
+
readonly currentImage: Signal<ScIImage>;
|
24
|
+
/**
|
25
|
+
* Название товара.
|
26
|
+
*/
|
27
|
+
readonly productName: InputSignal<string | undefined>;
|
19
28
|
/**
|
20
29
|
* Обработчик события `mousemove.silent`.
|
21
30
|
*
|
@@ -34,6 +43,6 @@ export declare class HoverImageCarouselComponent {
|
|
34
43
|
* @param index Индекс изображения, который соответствует точке.
|
35
44
|
*/
|
36
45
|
protected onDotHovered(hovered: boolean, index: number): void;
|
37
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<
|
38
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<
|
46
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<ScHoverImageCarouselComponent, never>;
|
47
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScHoverImageCarouselComponent, "sc-hover-image-carousel", never, { "images": { "alias": "images"; "required": true; "isSignal": true; }; "isShowActions": { "alias": "isShowActions"; "required": false; "isSignal": true; }; "productName": { "alias": "productName"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
39
48
|
}
|
package/catalog/index.d.ts
CHANGED
@@ -1,5 +1,6 @@
|
|
1
1
|
export * from './download-price-list/sc-download-price-list.component';
|
2
2
|
export * from './category-card/sc-category-card.component';
|
3
|
+
export * from './hover-image-carousel/sc-hover-image-carousel.component';
|
3
4
|
export * from './input-quantity/sc-input-quantity.component';
|
4
5
|
export * from './notify-when-in-stock-dialog';
|
5
6
|
export * from './price-card/sc-price-card.component';
|
@@ -11,18 +11,19 @@ import * as i9 from "@angular/forms";
|
|
11
11
|
import * as i10 from "@taiga-ui/kit";
|
12
12
|
import * as i11 from "@taiga-ui/cdk";
|
13
13
|
import * as i12 from "@taiga-ui/addon-charts";
|
14
|
-
import * as i13 from "./hover-image-carousel/hover-image-carousel.component";
|
14
|
+
import * as i13 from "./hover-image-carousel/sc-hover-image-carousel.component";
|
15
15
|
import * as i14 from "./cost-with-discount/cost-with-discount.component";
|
16
16
|
import * as i15 from "./price-card-inline/sc-price-card-inline.component";
|
17
17
|
import * as i16 from "./price-warehouse-stock/sc-price-warehouse-stock.component";
|
18
18
|
import * as i17 from "./sc-favorite-button/sc-favorite-button.component";
|
19
19
|
import * as i18 from "./input-quantity/sc-input-quantity.component";
|
20
20
|
import * as i19 from "../pipes/sc-format-date";
|
21
|
+
import * as i20 from "@snabcentr/client-core";
|
21
22
|
/**
|
22
23
|
* Модуль каталога.
|
23
24
|
*/
|
24
25
|
export declare class ScCatalogModule {
|
25
26
|
static ɵfac: i0.ɵɵFactoryDeclaration<ScCatalogModule, never>;
|
26
|
-
static ɵmod: i0.ɵɵNgModuleDeclaration<ScCatalogModule, [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScPriceCardComponent, typeof i4.ScPriceHistoryComponent], [typeof i5.CommonModule, typeof i6.RouterModule, typeof i7.TuiButton, typeof i7.TuiIcon, typeof i8.TuiIslandDirective, typeof i8.TuiInputNumberModule, typeof i7.TuiLabel, typeof i8.TuiTextfieldControllerModule, typeof i9.FormsModule, typeof i9.ReactiveFormsModule, typeof i7.TuiHintComponent, typeof i7.TuiHintDirective, typeof i7.TuiHintOptionsDirective, typeof i7.TuiHintUnstyled, typeof i7.TuiHintDriver, typeof i7.TuiHintPosition, typeof i7.TuiHintHover, typeof i7.TuiHintOverflow, typeof i7.TuiHintDescribe, typeof i7.TuiHintHost, typeof i7.TuiHintManual, typeof i7.TuiHintPointer, typeof i10.TuiFieldErrorPipe, typeof i7.TuiLoader, typeof i7.TuiLink, typeof i10.TuiElasticContainer, typeof i11.TuiLet, typeof i11.TuiRepeatTimes, typeof i10.TuiHighlight, typeof i12.TuiLineDaysChart, typeof i11.TuiAutoFocus, typeof i12.TuiLineDaysChartHint, typeof i12.TuiAxes, typeof i10.TuiButtonLoading, typeof i10.TuiAvatar, typeof i10.TuiBadge, typeof i13.
|
27
|
+
static ɵmod: i0.ɵɵNgModuleDeclaration<ScCatalogModule, [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScPriceCardComponent, typeof i4.ScPriceHistoryComponent], [typeof i5.CommonModule, typeof i6.RouterModule, typeof i7.TuiButton, typeof i7.TuiIcon, typeof i8.TuiIslandDirective, typeof i8.TuiInputNumberModule, typeof i7.TuiLabel, typeof i8.TuiTextfieldControllerModule, typeof i9.FormsModule, typeof i9.ReactiveFormsModule, typeof i7.TuiHintComponent, typeof i7.TuiHintDirective, typeof i7.TuiHintOptionsDirective, typeof i7.TuiHintUnstyled, typeof i7.TuiHintDriver, typeof i7.TuiHintPosition, typeof i7.TuiHintHover, typeof i7.TuiHintOverflow, typeof i7.TuiHintDescribe, typeof i7.TuiHintHost, typeof i7.TuiHintManual, typeof i7.TuiHintPointer, typeof i10.TuiFieldErrorPipe, typeof i7.TuiLoader, typeof i7.TuiLink, typeof i10.TuiElasticContainer, typeof i11.TuiLet, typeof i11.TuiRepeatTimes, typeof i10.TuiHighlight, typeof i12.TuiLineDaysChart, typeof i11.TuiAutoFocus, typeof i12.TuiLineDaysChartHint, typeof i12.TuiAxes, typeof i10.TuiButtonLoading, typeof i10.TuiAvatar, typeof i10.TuiBadge, typeof i13.ScHoverImageCarouselComponent, typeof i10.TuiLineClamp, typeof i11.TuiHovered, typeof i14.CostWithDiscountComponent, typeof i15.ScPriceCardInlineComponent, typeof i16.ScPriceWarehouseStockComponent, typeof i17.ScFavoriteButtonComponent, typeof i18.ScInputQuantityComponent, typeof i19.ScFormatDatePipe, typeof i20.ScMediaImageTransformerPipe], [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScPriceCardComponent, typeof i4.ScPriceHistoryComponent]>;
|
27
28
|
static ɵinj: i0.ɵɵInjectorDeclaration<ScCatalogModule>;
|
28
29
|
}
|
@@ -1,6 +1,6 @@
|
|
1
1
|
import { EventEmitter, WritableSignal } from '@angular/core';
|
2
2
|
import { FormControl } from '@angular/forms';
|
3
|
-
import { ScCartItem, ScImageHelper, ScProduct, ScUnitsHelper } from '@snabcentr/client-core';
|
3
|
+
import { ScCartItem, ScIImage, ScImageHelper, ScProduct, ScUnitsHelper } from '@snabcentr/client-core';
|
4
4
|
import { Observable } from 'rxjs';
|
5
5
|
import * as i0 from "@angular/core";
|
6
6
|
/**
|
@@ -118,15 +118,15 @@ export declare class AbstractScPriceCard {
|
|
118
118
|
*/
|
119
119
|
get skeletonVisible(): boolean;
|
120
120
|
/**
|
121
|
-
* Возвращает
|
121
|
+
* Возвращает данные для preview-изображения карточки товара.
|
122
122
|
*/
|
123
|
-
getCardImagePreview():
|
123
|
+
getCardImagePreview(): ScIImage;
|
124
124
|
/**
|
125
125
|
* Возвращает массив превью-изображений товара.
|
126
126
|
*
|
127
127
|
* @param product Объект товара.
|
128
128
|
*/
|
129
|
-
getCardImagePreviewList(product: ScProduct):
|
129
|
+
getCardImagePreviewList(product: ScProduct): ScIImage[];
|
130
130
|
/**
|
131
131
|
* Устанавливает компонент в очередь на обновление.
|
132
132
|
*/
|
@@ -107,11 +107,11 @@ export class ScBannerComponent {
|
|
107
107
|
}
|
108
108
|
}
|
109
109
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScBannerComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.ScBannerService }, { token: IntersectionObserverService }, { token: ElementRef }, { token: i2.ScPxConverter }, { token: i1.ScUserMetrikaService }], target: i0.ɵɵFactoryTarget.Component }); }
|
110
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
110
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: "navigateButton", duration: "duration", bannerLocation: "bannerLocation", resizable: "resizable" }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerEvent: "clickBannerEvent" }, host: { attributes: { "ngSkipHydration": "true" }, properties: { "style.aspect-ratio": "this.aspectRatio", "class.!hidden": "this.isHidden" } }, providers: [IntersectionObserverService], queries: [{ propertyName: "bannersListRef", predicate: ["banner"] }], ngImport: i0, template: "<ng-container *tuiLet=\"banners$ | async\">\n <tui-carousel\n [duration]=\"duration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n >\n <ng-container *ngFor=\"let banner of banners; let index = index\">\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'image'\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n class=\"size-full\"\n >\n <picture>\n @if (banner.mediaFileWebp) {\n <source\n type=\"image/webp\"\n [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n />\n }\n <img\n [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n [alt]=\"banner.title\"\n class=\"size-full object-cover\"\n />\n </picture>\n </a>\n </ng-container>\n </ng-container>\n </ng-container>\n <ng-container *ngFor=\"let item of bannersListRef\">\n <div\n *tuiItem\n [style.aspect-ratio]=\"aspectRatio\"\n class=\"size-full overflow-hidden\"\n >\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n </tui-carousel>\n\n <div\n *ngIf=\"navigateButton && duration && this.banners && this.bannersListRef.length + this.banners.length > 1\"\n tuiTheme=\"light\"\n class=\"flex items-center\"\n >\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-left\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.prev()\"\n class=\"!absolute left-2\"\n ></button>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-right\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.next()\"\n class=\"!absolute right-2\"\n ></button>\n </div>\n</ng-container>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}::ng-deep tui-carousel[resizable=true] .t-scroller,::ng-deep tui-carousel[resizable=true] .t-items{width:100%;height:100%}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i3.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i4.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i5.TuiItem, selector: "[tuiItem]" }, { kind: "component", type: i6.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange", "shift"] }, { kind: "directive", type: i7.TuiLet, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.ScMediaImageTransformerPipe, name: "scMediaImageTransformer" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
111
111
|
}
|
112
112
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScBannerComponent, decorators: [{
|
113
113
|
type: Component,
|
114
|
-
args: [{ selector: 'sc-banner', providers: [IntersectionObserverService], host: { ngSkipHydration: 'true' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *tuiLet=\"banners$ | async\">\n <tui-carousel\n [duration]=\"duration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n >\n <ng-container *ngFor=\"let banner of banners; let index = index\">\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'image'\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n
|
114
|
+
args: [{ selector: 'sc-banner', providers: [IntersectionObserverService], host: { ngSkipHydration: 'true' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *tuiLet=\"banners$ | async\">\n <tui-carousel\n [duration]=\"duration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n >\n <ng-container *ngFor=\"let banner of banners; let index = index\">\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'image'\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n class=\"size-full\"\n >\n <picture>\n @if (banner.mediaFileWebp) {\n <source\n type=\"image/webp\"\n [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n />\n }\n <img\n [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n [alt]=\"banner.title\"\n class=\"size-full object-cover\"\n />\n </picture>\n </a>\n </ng-container>\n </ng-container>\n </ng-container>\n <ng-container *ngFor=\"let item of bannersListRef\">\n <div\n *tuiItem\n [style.aspect-ratio]=\"aspectRatio\"\n class=\"size-full overflow-hidden\"\n >\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n </tui-carousel>\n\n <div\n *ngIf=\"navigateButton && duration && this.banners && this.bannersListRef.length + this.banners.length > 1\"\n tuiTheme=\"light\"\n class=\"flex items-center\"\n >\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-left\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.prev()\"\n class=\"!absolute left-2\"\n ></button>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-right\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.next()\"\n class=\"!absolute right-2\"\n ></button>\n </div>\n</ng-container>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}::ng-deep tui-carousel[resizable=true] .t-scroller,::ng-deep tui-carousel[resizable=true] .t-items{width:100%;height:100%}\n"] }]
|
115
115
|
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.ScBannerService }, { type: i8.IntersectionObserverService, decorators: [{
|
116
116
|
type: Inject,
|
117
117
|
args: [IntersectionObserverService]
|
@@ -140,4 +140,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
140
140
|
type: HostBinding,
|
141
141
|
args: ['class.!hidden']
|
142
142
|
}] } });
|
143
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-banner.component.js","sourceRoot":"","sources":["../../../../projects/client-ui/banner/sc-banner.component.ts","../../../../projects/client-ui/banner/sc-banner.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,uBAAuB,EAEvB,SAAS,EACT,eAAe,EACf,UAAU,EACV,YAAY,EACZ,WAAW,EACX,MAAM,EACN,KAAK,EACL,MAAM,GAGT,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,2BAA2B,EAAE,MAAM,oCAAoC,CAAC;AACjF,OAAO,EAA6B,sBAAsB,EAAwB,MAAM,wBAAwB,CAAC;AACjH,OAAO,EAAE,GAAG,EAAc,WAAW,EAAE,OAAO,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;;;;;;;;;;AAIlE;;GAEG;AASH,MAAM,OAAO,iBAAiB;IAsF1B;;;;;;;;;OASG;IACH,YACqB,GAAsB,EACtB,aAA8B,EACO,QAAqC,EACtD,OAA4B,EACzD,WAA0B,EACjB,kBAAwC;QALxC,QAAG,GAAH,GAAG,CAAmB;QACtB,kBAAa,GAAb,aAAa,CAAiB;QACO,aAAQ,GAAR,QAAQ,CAA6B;QACtD,YAAO,GAAP,OAAO,CAAqB;QACzD,gBAAW,GAAX,WAAW,CAAe;QACjB,uBAAkB,GAAlB,kBAAkB,CAAsB;QArG7D;;WAEG;QAEI,mBAAc,GAAY,IAAI,CAAC;QAEtC;;WAEG;QAEI,aAAQ,GAAW,IAAI,CAAC;QAQ/B;;WAEG;QAEI,cAAS,GAAY,KAAK,CAAC;QAElC;;WAEG;QAEI,qBAAgB,GAAyB,IAAI,YAAY,EAAU,CAAC;QAE3E;;WAEG;QAEI,qBAAgB,GAA2B,IAAI,YAAY,EAAY,CAAC;QAS/E;;WAEG;QACI,oBAAe,GAAW,CAAC,CAAC;QAEnC;;WAEG;QACa,iBAAY,GAAqB,IAAI,OAAO,EAAW,CAAC;QAExE;;WAEG;QACa,aAAQ,GAA2B,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAC/E,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,KAAK,IAAI,CAAC,cAAc,CAAC,CAAC,OAAO,EAAE,CAAC,EAC/F,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;YACZ,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACrB,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;oBAClB,IAAI,CAAC,WAAW,GAAG,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,MAAM,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;gBACpE,CAAC;gBAED,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;gBACvB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACjC,CAAC;YAED,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAC/C,CAAC,CAAC;QACF,+CAA+C;QAC/C,WAAW,CAAC,CAAC,CAAC,CACjB,CAAC;QAEF;;WAEG;QACI,YAAO,GAAgB,EAAE,CAAC;QAEjC;;WAEG;QAEI,gBAAW,GAAW,EAAE,CAAC;IAmB7B,CAAC;IAEJ;;OAEG;IACH,IACY,QAAQ;QAChB,OAAO,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC;IACnF,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAC,MAAgB;QAC3B,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,CAAC;YACzC,MAAM,EAAE,sBAAsB,CAAC,WAAW;YAC1C,MAAM,EAAE;gBACJ,SAAS,EAAE,MAAM,CAAC,EAAE;aACvB;SACJ,CAAC,CAAC;QAEH,IAAI,MAAM,CAAC,GAAG,EAAE,CAAC;YACb,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACvC,CAAC;IACL,CAAC;+GAjIQ,iBAAiB,kFAmGd,2BAA2B,aAC3B,UAAU;mGApGb,iBAAiB,4YAJf,CAAC,2BAA2B,CAAC,gGC3B5C,ouEA6DA;;4FD9Ba,iBAAiB;kBAR7B,SAAS;+BACI,WAAW,aAGV,CAAC,2BAA2B,CAAC,QAClC,EAAE,eAAe,EAAE,MAAM,EAAE,mBAChB,uBAAuB,CAAC,MAAM;;0BAqG1C,MAAM;2BAAC,2BAA2B;;0BAClC,MAAM;2BAAC,UAAU;wGA/Ff,cAAc;sBADpB,KAAK;gBAOC,QAAQ;sBADd,KAAK;gBAOC,cAAc;sBADpB,KAAK;gBAOC,SAAS;sBADf,KAAK;gBAOC,gBAAgB;sBADtB,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAQA,cAAc;sBAFpB,eAAe;uBAAC,QAAQ;gBA4ClB,WAAW;sBADjB,WAAW;uBAAC,oBAAoB;gBA0BrB,QAAQ;sBADnB,WAAW;uBAAC,eAAe","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    Inject,\n    Input,\n    Output,\n    QueryList,\n    TemplateRef,\n} from '@angular/core';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { map, Observable, shareReplay, Subject, tap } from 'rxjs';\n\nimport { ScPxConverter } from '../helpers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    providers: [IntersectionObserverService],\n    host: { ngSkipHydration: 'true' },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScBannerComponent {\n    /**\n     * Признак, что необходимо показывать кнопки навигации.\n     */\n    @Input()\n    public navigateButton: boolean = true;\n\n    /**\n     * Интервал автоматической смены слайдов в миллисекундах (используйте 0, чтобы отключить автоматическую смену слайда).\n     */\n    @Input()\n    public duration: number = 5000;\n\n    /**\n     * Местоположение баннера.\n     */\n    @Input()\n    public bannerLocation?: string;\n\n    /**\n     * Признак, что компонент должен растягиваться.\n     */\n    @Input()\n    public resizable: boolean = false;\n\n    /**\n     * Событие загрузки баннеров с количеством полученных баннеров.\n     */\n    @Output()\n    public loadBannersEvent: EventEmitter<number> = new EventEmitter<number>();\n\n    /**\n     * Событие нажатия на изображение баннера.\n     */\n    @Output()\n    public clickBannerEvent: EventEmitter<ScBanner> = new EventEmitter<ScBanner>();\n\n    /**\n     * Список ссылок на элемент представлений шаблонов.\n     */\n    @ContentChildren('banner')\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    public bannersListRef: QueryList<TemplateRef<any>>;\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    public currentBannerId: number = 0;\n\n    /**\n     * {@link Subject} изменения состояния таймера.\n     */\n    public readonly toggleTimer$: Subject<boolean> = new Subject<boolean>();\n\n    /**\n     * {@link Observable} обновления списка баннеров.\n     */\n    public readonly banners$: Observable<ScBanner[]> = this.bannerService.banners$.pipe(\n        map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()),\n        tap((banners) => {\n            if (banners.length > 0) {\n                if (!this.resizable) {\n                    this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                }\n\n                this.banners = banners;\n                this.toggleTimer$.next(true);\n            }\n\n            this.loadBannersEvent.emit(banners.length);\n        }),\n        // eslint-disable-next-line rxjs/no-sharereplay\n        shareReplay(1)\n    );\n\n    /**\n     * Список баннеров.\n     */\n    public banners?: ScBanner[] = [];\n\n    /**\n     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Инициализирует экземпляр класса {@link ScBannerComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param bannerService Сервис для работы с баннерами.\n     * @param entries$ Intersection Observer Service.\n     * @param element Элемент баннера.\n     * @param pxConverter Экземпляр класса-помощника для конвертации пикселей.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     */\n    public constructor(\n        private readonly cdr: ChangeDetectorRef,\n        private readonly bannerService: ScBannerService,\n        @Inject(IntersectionObserverService) private readonly entries$: IntersectionObserverService,\n        @Inject(ElementRef) private readonly element: ElementRef<Element>,\n        private pxConverter: ScPxConverter,\n        private readonly userMetrikaService: ScUserMetrikaService\n    ) {}\n\n    /**\n     * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.\n     */\n    @HostBinding('class.!hidden')\n    private get isHidden(): boolean {\n        return !this.banners || this.bannersListRef.length + this.banners.length === 0;\n    }\n\n    /**\n     * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.\n     *\n     * @param banner Баннер, по ссылке которого совершён переход.\n     */\n    public onClick(banner: ScBanner): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.bannerClick,\n            params: {\n                banner_id: banner.id,\n            },\n        });\n\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\n}\n","<ng-container *tuiLet=\"banners$ | async\">\n    <tui-carousel\n        [duration]=\"duration\"\n        #carousel\n        [attr.resizable]=\"resizable\"\n        class=\"size-full overflow-hidden rounded-xl bg-white\"\n        [(index)]=\"currentBannerId\"\n    >\n        <ng-container *ngFor=\"let banner of banners; let index = index\">\n            <ng-container [ngSwitch]=\"banner.mediaType\">\n                <ng-container *ngSwitchCase=\"'image'\">\n                    <a\n                        *tuiItem\n                        (click)=\"onClick(banner)\"\n                        target=\"_blank\"\n                        [title]=\"banner.title\"\n                        [style.aspect-ratio]=\"aspectRatio\"\n                        [attr.href]=\"banner.url ? banner.url : null\"\n                        [style.background-image]=\"'url(' + banner.mediaFile + ')'\"\n                        class=\"size-full bg-[length:100%_100%]\"\n                    >\n                    </a>\n                </ng-container>\n            </ng-container>\n        </ng-container>\n        <ng-container *ngFor=\"let item of bannersListRef\">\n            <div\n                *tuiItem\n                [style.aspect-ratio]=\"aspectRatio\"\n                class=\"size-full overflow-hidden\"\n            >\n                <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n            </div>\n        </ng-container>\n    </tui-carousel>\n\n    <div\n        *ngIf=\"navigateButton && duration && this.banners && this.bannersListRef.length + this.banners.length > 1\"\n        tuiTheme=\"light\"\n        class=\"flex items-center\"\n    >\n        <button\n            tuiIconButton\n            iconStart=\"@tui.chevron-left\"\n            size=\"m\"\n            [style.border-radius.%]=\"100\"\n            appearance=\"flat\"\n            (click)=\"carousel.prev()\"\n            class=\"!absolute left-2\"\n        ></button>\n        <button\n            tuiIconButton\n            iconStart=\"@tui.chevron-right\"\n            size=\"m\"\n            [style.border-radius.%]=\"100\"\n            appearance=\"flat\"\n            (click)=\"carousel.next()\"\n            class=\"!absolute right-2\"\n        ></button>\n    </div>\n</ng-container>\n"]}
|
143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-banner.component.js","sourceRoot":"","sources":["../../../../projects/client-ui/banner/sc-banner.component.ts","../../../../projects/client-ui/banner/sc-banner.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    Inject,\n    Input,\n    Output,\n    QueryList,\n    TemplateRef,\n} from '@angular/core';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { map, Observable, shareReplay, Subject, tap } from 'rxjs';\n\nimport { ScPxConverter } from '../helpers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    providers: [IntersectionObserverService],\n    host: { ngSkipHydration: 'true' },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScBannerComponent {\n    /**\n     * Признак, что необходимо показывать кнопки навигации.\n     */\n    @Input()\n    public navigateButton: boolean = true;\n\n    /**\n     * Интервал автоматической смены слайдов в миллисекундах (используйте 0, чтобы отключить автоматическую смену слайда).\n     */\n    @Input()\n    public duration: number = 5000;\n\n    /**\n     * Местоположение баннера.\n     */\n    @Input()\n    public bannerLocation?: string;\n\n    /**\n     * Признак, что компонент должен растягиваться.\n     */\n    @Input()\n    public resizable: boolean = false;\n\n    /**\n     * Событие загрузки баннеров с количеством полученных баннеров.\n     */\n    @Output()\n    public loadBannersEvent: EventEmitter<number> = new EventEmitter<number>();\n\n    /**\n     * Событие нажатия на изображение баннера.\n     */\n    @Output()\n    public clickBannerEvent: EventEmitter<ScBanner> = new EventEmitter<ScBanner>();\n\n    /**\n     * Список ссылок на элемент представлений шаблонов.\n     */\n    @ContentChildren('banner')\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    public bannersListRef: QueryList<TemplateRef<any>>;\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    public currentBannerId: number = 0;\n\n    /**\n     * {@link Subject} изменения состояния таймера.\n     */\n    public readonly toggleTimer$: Subject<boolean> = new Subject<boolean>();\n\n    /**\n     * {@link Observable} обновления списка баннеров.\n     */\n    public readonly banners$: Observable<ScBanner[]> = this.bannerService.banners$.pipe(\n        map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()),\n        tap((banners) => {\n            if (banners.length > 0) {\n                if (!this.resizable) {\n                    this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                }\n\n                this.banners = banners;\n                this.toggleTimer$.next(true);\n            }\n\n            this.loadBannersEvent.emit(banners.length);\n        }),\n        // eslint-disable-next-line rxjs/no-sharereplay\n        shareReplay(1)\n    );\n\n    /**\n     * Список баннеров.\n     */\n    public banners?: ScBanner[] = [];\n\n    /**\n     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Инициализирует экземпляр класса {@link ScBannerComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param bannerService Сервис для работы с баннерами.\n     * @param entries$ Intersection Observer Service.\n     * @param element Элемент баннера.\n     * @param pxConverter Экземпляр класса-помощника для конвертации пикселей.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     */\n    public constructor(\n        private readonly cdr: ChangeDetectorRef,\n        private readonly bannerService: ScBannerService,\n        @Inject(IntersectionObserverService) private readonly entries$: IntersectionObserverService,\n        @Inject(ElementRef) private readonly element: ElementRef<Element>,\n        private pxConverter: ScPxConverter,\n        private readonly userMetrikaService: ScUserMetrikaService\n    ) {}\n\n    /**\n     * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.\n     */\n    @HostBinding('class.!hidden')\n    private get isHidden(): boolean {\n        return !this.banners || this.bannersListRef.length + this.banners.length === 0;\n    }\n\n    /**\n     * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.\n     *\n     * @param banner Баннер, по ссылке которого совершён переход.\n     */\n    public onClick(banner: ScBanner): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.bannerClick,\n            params: {\n                banner_id: banner.id,\n            },\n        });\n\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\n}\n","<ng-container *tuiLet=\"banners$ | async\">\n    <tui-carousel\n        [duration]=\"duration\"\n        #carousel\n        [attr.resizable]=\"resizable\"\n        class=\"size-full overflow-hidden rounded-xl bg-white\"\n        [(index)]=\"currentBannerId\"\n    >\n        <ng-container *ngFor=\"let banner of banners; let index = index\">\n            <ng-container [ngSwitch]=\"banner.mediaType\">\n                <ng-container *ngSwitchCase=\"'image'\">\n                    <a\n                        *tuiItem\n                        (click)=\"onClick(banner)\"\n                        target=\"_blank\"\n                        [title]=\"banner.title\"\n                        [style.aspect-ratio]=\"aspectRatio\"\n                        [attr.href]=\"banner.url ? banner.url : null\"\n                        class=\"size-full\"\n                    >\n                        <picture>\n                            @if (banner.mediaFileWebp) {\n                                <source\n                                    type=\"image/webp\"\n                                    [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n                                />\n                            }\n                            <img\n                                [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n                                [alt]=\"banner.title\"\n                                class=\"size-full object-cover\"\n                            />\n                        </picture>\n                    </a>\n                </ng-container>\n            </ng-container>\n        </ng-container>\n        <ng-container *ngFor=\"let item of bannersListRef\">\n            <div\n                *tuiItem\n                [style.aspect-ratio]=\"aspectRatio\"\n                class=\"size-full overflow-hidden\"\n            >\n                <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n            </div>\n        </ng-container>\n    </tui-carousel>\n\n    <div\n        *ngIf=\"navigateButton && duration && this.banners && this.bannersListRef.length + this.banners.length > 1\"\n        tuiTheme=\"light\"\n        class=\"flex items-center\"\n    >\n        <button\n            tuiIconButton\n            iconStart=\"@tui.chevron-left\"\n            size=\"m\"\n            [style.border-radius.%]=\"100\"\n            appearance=\"flat\"\n            (click)=\"carousel.prev()\"\n            class=\"!absolute left-2\"\n        ></button>\n        <button\n            tuiIconButton\n            iconStart=\"@tui.chevron-right\"\n            size=\"m\"\n            [style.border-radius.%]=\"100\"\n            appearance=\"flat\"\n            (click)=\"carousel.next()\"\n            class=\"!absolute right-2\"\n        ></button>\n    </div>\n</ng-container>\n"]}
|
@@ -2,6 +2,7 @@ import { CommonModule } from '@angular/common';
|
|
2
2
|
import { NgModule } from '@angular/core';
|
3
3
|
import { RouterModule } from '@angular/router';
|
4
4
|
import { WaIntersectionObserver } from '@ng-web-apis/intersection-observer';
|
5
|
+
import { ScMediaImageTransformerPipe } from '@snabcentr/client-core';
|
5
6
|
import { TuiLet } from '@taiga-ui/cdk';
|
6
7
|
import { TuiButton, TuiLoader } from '@taiga-ui/core';
|
7
8
|
import { TuiCarousel } from '@taiga-ui/kit';
|
@@ -15,15 +16,15 @@ import * as i3 from "@ng-web-apis/intersection-observer";
|
|
15
16
|
*/
|
16
17
|
export class ScBannerModule {
|
17
18
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScBannerModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
|
18
|
-
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.13", ngImport: i0, type: ScBannerModule, declarations: [ScBannerComponent], imports: [CommonModule, RouterModule, TuiButton, i1.TuiItem, i2.TuiCarouselComponent, i2.TuiCarouselDirective, i2.TuiCarouselAutoscroll, i2.TuiCarouselButtons, i2.TuiCarouselScroll, TuiLoader, i3.WaIntersectionObserverDirective, i3.WaIntersectionObservee, i3.WaIntersectionRoot, TuiLet], exports: [ScBannerComponent] }); }
|
19
|
+
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.13", ngImport: i0, type: ScBannerModule, declarations: [ScBannerComponent], imports: [CommonModule, RouterModule, TuiButton, i1.TuiItem, i2.TuiCarouselComponent, i2.TuiCarouselDirective, i2.TuiCarouselAutoscroll, i2.TuiCarouselButtons, i2.TuiCarouselScroll, TuiLoader, i3.WaIntersectionObserverDirective, i3.WaIntersectionObservee, i3.WaIntersectionRoot, TuiLet, ScMediaImageTransformerPipe], exports: [ScBannerComponent] }); }
|
19
20
|
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScBannerModule, imports: [CommonModule, RouterModule, i2.TuiCarouselComponent, TuiLoader] }); }
|
20
21
|
}
|
21
22
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScBannerModule, decorators: [{
|
22
23
|
type: NgModule,
|
23
24
|
args: [{
|
24
25
|
declarations: [ScBannerComponent],
|
25
|
-
imports: [CommonModule, RouterModule, TuiButton, ...TuiCarousel, TuiLoader, ...WaIntersectionObserver, TuiLet],
|
26
|
+
imports: [CommonModule, RouterModule, TuiButton, ...TuiCarousel, TuiLoader, ...WaIntersectionObserver, TuiLet, ScMediaImageTransformerPipe],
|
26
27
|
exports: [ScBannerComponent],
|
27
28
|
}]
|
28
29
|
}] });
|
29
|
-
//# sourceMappingURL=data:application/json;base64,
|
30
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtYmFubmVyLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9iYW5uZXIvc2MtYmFubmVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFDNUUsT0FBTyxFQUFFLDJCQUEyQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDckUsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN2QyxPQUFPLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3RELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFNUMsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sdUJBQXVCLENBQUM7Ozs7O0FBRTFEOztHQUVHO0FBTUgsTUFBTSxPQUFPLGNBQWM7K0dBQWQsY0FBYztnSEFBZCxjQUFjLGlCQUpSLGlCQUFpQixhQUN0QixZQUFZLEVBQUUsWUFBWSxFQUFFLFNBQVMsdUlBQWtCLFNBQVMsd0ZBQTZCLE1BQU0sRUFBRSwyQkFBMkIsYUFDaEksaUJBQWlCO2dIQUVsQixjQUFjLFlBSGIsWUFBWSxFQUFFLFlBQVksMkJBQTZCLFNBQVM7OzRGQUdqRSxjQUFjO2tCQUwxQixRQUFRO21CQUFDO29CQUNOLFlBQVksRUFBRSxDQUFDLGlCQUFpQixDQUFDO29CQUNqQyxPQUFPLEVBQUUsQ0FBQyxZQUFZLEVBQUUsWUFBWSxFQUFFLFNBQVMsRUFBRSxHQUFHLFdBQVcsRUFBRSxTQUFTLEVBQUUsR0FBRyxzQkFBc0IsRUFBRSxNQUFNLEVBQUUsMkJBQTJCLENBQUM7b0JBQzNJLE9BQU8sRUFBRSxDQUFDLGlCQUFpQixDQUFDO2lCQUMvQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgUm91dGVyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvcm91dGVyJztcbmltcG9ydCB7IFdhSW50ZXJzZWN0aW9uT2JzZXJ2ZXIgfSBmcm9tICdAbmctd2ViLWFwaXMvaW50ZXJzZWN0aW9uLW9ic2VydmVyJztcbmltcG9ydCB7IFNjTWVkaWFJbWFnZVRyYW5zZm9ybWVyUGlwZSB9IGZyb20gJ0BzbmFiY2VudHIvY2xpZW50LWNvcmUnO1xuaW1wb3J0IHsgVHVpTGV0IH0gZnJvbSAnQHRhaWdhLXVpL2Nkayc7XG5pbXBvcnQgeyBUdWlCdXR0b24sIFR1aUxvYWRlciB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7IFR1aUNhcm91c2VsIH0gZnJvbSAnQHRhaWdhLXVpL2tpdCc7XG5cbmltcG9ydCB7IFNjQmFubmVyQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1iYW5uZXIuY29tcG9uZW50JztcblxuLyoqXG4gKiDQnNC+0LTRg9C70Ywg0LHQsNC90L3QtdGA0L7Qsi5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtTY0Jhbm5lckNvbXBvbmVudF0sXG4gICAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgUm91dGVyTW9kdWxlLCBUdWlCdXR0b24sIC4uLlR1aUNhcm91c2VsLCBUdWlMb2FkZXIsIC4uLldhSW50ZXJzZWN0aW9uT2JzZXJ2ZXIsIFR1aUxldCwgU2NNZWRpYUltYWdlVHJhbnNmb3JtZXJQaXBlXSxcbiAgICBleHBvcnRzOiBbU2NCYW5uZXJDb21wb25lbnRdLFxufSlcbmV4cG9ydCBjbGFzcyBTY0Jhbm5lck1vZHVsZSB7fVxuIl19
|
@@ -61,7 +61,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
61
61
|
/**
|
62
62
|
* Минимальный метраж для товара.
|
63
63
|
*/
|
64
|
-
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount)
|
64
|
+
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));
|
65
65
|
/**
|
66
66
|
* Максимальный метраж для товара.
|
67
67
|
*/
|
@@ -76,8 +76,8 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
76
76
|
this.lengthHint = computed(() => {
|
77
77
|
const min = this.minLength();
|
78
78
|
const max = this.maxLength();
|
79
|
-
// eslint-disable-next-line sonarjs/no-nested-conditional,
|
80
|
-
return
|
79
|
+
// eslint-disable-next-line sonarjs/no-nested-conditional, sonarjs/prefer-nullish-coalescing
|
80
|
+
return min || max ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';
|
81
81
|
});
|
82
82
|
/**
|
83
83
|
* {@link Output} события добавления товара в корзину.
|
@@ -108,7 +108,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
108
108
|
});
|
109
109
|
if (this.productIsMeasurable()) {
|
110
110
|
// Добавляем поле ввода длины, если товар измеряемый.
|
111
|
-
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));
|
111
|
+
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength() ?? 0.01, [Validators.required, Validators.min(0.01)]));
|
112
112
|
}
|
113
113
|
// Устанавливаем количество из корзины или кратности товара.
|
114
114
|
this.form.patchValue({
|
@@ -161,7 +161,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
161
161
|
return;
|
162
162
|
}
|
163
163
|
const length = lengthControl.value ?? 0;
|
164
|
-
let newLength = Math.max(this.minLength(), length + step);
|
164
|
+
let newLength = Math.max(this.minLength() ?? step, length + step);
|
165
165
|
// Округляем до ближайшего кратного числа.
|
166
166
|
newLength = Math.round(newLength / step) * step;
|
167
167
|
lengthControl.setValue(Number(newLength.toFixed(2)));
|
@@ -230,4 +230,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
230
230
|
TuiChip,
|
231
231
|
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- \u0424\u043E\u0440\u043C\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F \u0438\u0437\u043C\u0435\u0440\u044F\u0435\u043C\u043E\u0433\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n<form\n *ngIf=\"product\"\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit()\"\n ScNextInputFocus\n class=\"flex flex-col gap-2\"\n>\n @let step = lengthStep();\n\n <!-- \u0414\u043B\u0438\u043D\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 (\u043C\u0435\u0442\u0440\u0430\u0436) -->\n <label\n *ngIf=\"productIsMeasurable()\"\n tuiLabel\n >\n \u041C\u0435\u0442\u0440\u0430\u0436, {{ product().unit }}\n @if (!step || maxLength()) {\n {{ lengthHint() }}\n }\n\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"length\"\n [tuiNumberFormat]=\"{ precision: 2 }\"\n [max]=\"maxLength() || null\"\n [min]=\"minLength() || null\"\n (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n autocomplete=\"length\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n <p\n *ngIf=\"lengthStep()\"\n class=\"tui-form__field-note\"\n >\n \u041C\u0435\u0442\u0440\u0430\u0436 \u0434\u043E\u043B\u0436\u0435\u043D \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u0435\u043D {{ lengthStep() }}\n </p>\n </label>\n\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430 -->\n <label tuiLabel>\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442.\n <tui-textfield>\n <tui-chip\n *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n size=\"s\"\n appearance=\"negative\"\n class=\"font-bold\"\n >\n x {{ step }} {{ product().unit }}\n </tui-chip>\n\n <input\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n tuiInputNumber\n [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n [min]=\"productMultiplicity()\"\n (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n formControlName=\"quantity\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n <p class=\"tui-form__field-note\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0434\u043E\u043B\u0436\u043D\u043E \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u043D\u043E {{ productMultiplicity() }}</p>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 -->\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-input formControlName=\"marker\">\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <input\n tuiTextfieldLegacy\n autocomplete=\"marker\"\n />\n </tui-input>\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <div class=\"flex flex-col items-center\">\n @let cost = totalCost$ | async;\n\n <div *ngIf=\"cost\">\n \u0418\u0442\u043E\u0433\u043E:<span class=\"text-2xl font-bold\">\n {{ cost | tuiAmount | async }}\n {{ product().currency }}\n </span>\n </div>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <button\n tuiButton\n iconStart=\"@tui.check\"\n [disabled]=\"form.invalid\"\n [loading]=\"isLoading()\"\n type=\"submit\"\n class=\"mt-2\"\n >\n {{ cartItem() ? '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' : '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' }}\n </button>\n </div>\n</form>\n" }]
|
232
232
|
}] });
|
233
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { isUndefined } from 'lodash-es';\nimport { debounceTime, distinctUntilChanged, filter, map, Observable, pairwise, startWith } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Признак того, что для указанной категории или продукта действует конфигуратор длины.\n     */\n    protected readonly isLengthConfigurator = computed(() => Boolean(this.product().properties?.['isLengthConfigurator']));\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount) ?? 1);\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, @typescript-eslint/no-unnecessary-condition\n        return (min ?? max) ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected totalCost$: Observable<number>;\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n\n        if (this.productIsMeasurable()) {\n            // Добавляем поле ввода длины, если товар измеряемый.\n            this.form.addControl('length', new FormControl<number | null>(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));\n        }\n\n        // Устанавливаем количество из корзины или кратности товара.\n        this.form.patchValue({\n            quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),\n            marker: this.cartItem()?.marker ?? '',\n        });\n\n        const step = this.lengthStep();\n\n        if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {\n            // Если есть конфигуратор длины, рассчитываем начально значение длины.\n            this.form.controls.length?.patchValue(this.form.controls.quantity.value * step);\n        }\n\n        // Добавляем валидацию шага для количества.\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        // Добавляем валидацию шага для длины.\n        this.form.get('length')?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        // Считаем итоговую стоимость заказа.\n        this.totalCost$ = tuiControlValue<{\n            length?: number;\n            quantity: number;\n        }>(this.form).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            filter(() => this.form.valid),\n            distinctUntilChanged((previous, current) => previous.length === current.length && previous.quantity === current.quantity), // Только при изменении значений\n            startWith(this.form.value as { length?: number; quantity: number }),\n            pairwise(),\n            map(([previous, current]) => {\n                const lengthControl = this.form.get('length');\n\n                // Если нет конфигуратора длины или некорректные данные — возвращаем без изменений.\n                if (!this.isLengthConfigurator() || isUndefined(current.length) || !lengthControl || this.form.invalid || !step) {\n                    return current;\n                }\n\n                // Если изменилось количество — пересчитываем длину.\n                if (previous.quantity !== current.quantity && this.form.controls.quantity.value) {\n                    const newValue = this.form.controls.quantity.value * step;\n                    lengthControl.patchValue(newValue);\n                }\n\n                // Если изменилась длина — пересчитываем количество и возвращаем значение длины.\n                if (previous.length !== current.length && lengthControl.value) {\n                    const newValue = lengthControl.value / step;\n                    this.form.controls.quantity.patchValue(newValue);\n\n                    return { length: step, quantity: newValue };\n                }\n\n                // Возвращаем итоговые значения для расчёта стоимости.\n                return { length: step, quantity: current.quantity };\n            }),\n            // Считаем стоимость: цена * длина * количество.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * (length ?? 1) * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        );\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        const lengthControl = this.form.get('length');\n\n        if (!this.productIsMeasurable() || !lengthControl) {\n            return;\n        }\n\n        const length = lengthControl.value ?? 0;\n        let newLength = Math.max(this.minLength(), length + step);\n        // Округляем до ближайшего кратного числа.\n        newLength = Math.round(newLength / step) * step;\n        lengthControl.setValue(Number(newLength.toFixed(2)));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        // Удаляем null-поля из значения формы.\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        // Если это добавление нового товара — добавляем productId.\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        // Если товар измеряемый и есть конфигуратор длины — устанавливаем длину по шагу (количество уже будет вычислено по формуле Метраж/min-count)\n        if (this.productIsMeasurable() && this.isLengthConfigurator() && step) {\n            value.length = step;\n        }\n\n        // Помечаем все поля формы как \"touch\" и запускаем валидацию для отображения ошибок с сервера.\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        // В зависимости от режима — редактируем или добавляем товар.\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    @let step = lengthStep();\n\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }}\n        @if (!step || maxLength()) {\n            {{ lengthHint() }}\n        }\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            <tui-chip\n                *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost$ | async;\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount | async }}\n                {{ product().currency }}\n            </span>\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|
233
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { isUndefined } from 'lodash-es';\nimport { debounceTime, distinctUntilChanged, filter, map, Observable, pairwise, startWith } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Признак того, что для указанной категории или продукта действует конфигуратор длины.\n     */\n    protected readonly isLengthConfigurator = computed(() => Boolean(this.product().properties?.['isLengthConfigurator']));\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, sonarjs/prefer-nullish-coalescing\n        return min || max ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected totalCost$: Observable<number>;\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n\n        if (this.productIsMeasurable()) {\n            // Добавляем поле ввода длины, если товар измеряемый.\n            this.form.addControl('length', new FormControl<number | null>(this.cartItem()?.length ?? this.minLength() ?? 0.01, [Validators.required, Validators.min(0.01)]));\n        }\n\n        // Устанавливаем количество из корзины или кратности товара.\n        this.form.patchValue({\n            quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),\n            marker: this.cartItem()?.marker ?? '',\n        });\n\n        const step = this.lengthStep();\n\n        if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {\n            // Если есть конфигуратор длины, рассчитываем начально значение длины.\n            this.form.controls.length?.patchValue(this.form.controls.quantity.value * step);\n        }\n\n        // Добавляем валидацию шага для количества.\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        // Добавляем валидацию шага для длины.\n        this.form.get('length')?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        // Считаем итоговую стоимость заказа.\n        this.totalCost$ = tuiControlValue<{\n            length?: number;\n            quantity: number;\n        }>(this.form).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            filter(() => this.form.valid),\n            distinctUntilChanged((previous, current) => previous.length === current.length && previous.quantity === current.quantity), // Только при изменении значений\n            startWith(this.form.value as { length?: number; quantity: number }),\n            pairwise(),\n            map(([previous, current]) => {\n                const lengthControl = this.form.get('length');\n\n                // Если нет конфигуратора длины или некорректные данные — возвращаем без изменений.\n                if (!this.isLengthConfigurator() || isUndefined(current.length) || !lengthControl || this.form.invalid || !step) {\n                    return current;\n                }\n\n                // Если изменилось количество — пересчитываем длину.\n                if (previous.quantity !== current.quantity && this.form.controls.quantity.value) {\n                    const newValue = this.form.controls.quantity.value * step;\n                    lengthControl.patchValue(newValue);\n                }\n\n                // Если изменилась длина — пересчитываем количество и возвращаем значение длины.\n                if (previous.length !== current.length && lengthControl.value) {\n                    const newValue = lengthControl.value / step;\n                    this.form.controls.quantity.patchValue(newValue);\n\n                    return { length: step, quantity: newValue };\n                }\n\n                // Возвращаем итоговые значения для расчёта стоимости.\n                return { length: step, quantity: current.quantity };\n            }),\n            // Считаем стоимость: цена * длина * количество.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * (length ?? 1) * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        );\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        const lengthControl = this.form.get('length');\n\n        if (!this.productIsMeasurable() || !lengthControl) {\n            return;\n        }\n\n        const length = lengthControl.value ?? 0;\n        let newLength = Math.max(this.minLength() ?? step, length + step);\n        // Округляем до ближайшего кратного числа.\n        newLength = Math.round(newLength / step) * step;\n        lengthControl.setValue(Number(newLength.toFixed(2)));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        // Удаляем null-поля из значения формы.\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        // Если это добавление нового товара — добавляем productId.\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        // Если товар измеряемый и есть конфигуратор длины — устанавливаем длину по шагу (количество уже будет вычислено по формуле Метраж/min-count)\n        if (this.productIsMeasurable() && this.isLengthConfigurator() && step) {\n            value.length = step;\n        }\n\n        // Помечаем все поля формы как \"touch\" и запускаем валидацию для отображения ошибок с сервера.\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        // В зависимости от режима — редактируем или добавляем товар.\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    @let step = lengthStep();\n\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }}\n        @if (!step || maxLength()) {\n            {{ lengthHint() }}\n        }\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            <tui-chip\n                *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost$ | async;\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount | async }}\n                {{ product().currency }}\n            </span>\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|