@snabcentr/client-ui 2.6.1 → 3.1.0
Sign up to get free protection for your applications and to get access to all the features.
- package/cart/cart-item-mobile/sc-cart-item-mobile.component.d.ts +0 -2
- package/catalog/category-card/sc-category-card.component.d.ts +20 -26
- package/catalog/hover-image-carousel/hover-image-carousel.component.d.ts +24 -0
- package/catalog/price-card/sc-price-card.component.d.ts +1 -5
- package/catalog/sc-catalog.module.d.ts +2 -1
- package/directives/abstract-price-card/abstract-sc-price-card.directive.d.ts +6 -0
- package/esm2022/cart/cart-item-mobile/sc-cart-item-mobile.component.mjs +1 -3
- package/esm2022/catalog/category-card/sc-category-card.component.mjs +32 -51
- package/esm2022/catalog/hover-image-carousel/hover-image-carousel.component.mjs +45 -0
- package/esm2022/catalog/price-card/sc-price-card.component.mjs +7 -18
- package/esm2022/catalog/sc-catalog.module.mjs +19 -5
- package/esm2022/directives/abstract-price-card/abstract-sc-price-card.directive.mjs +10 -1
- package/fesm2022/snabcentr-client-ui.mjs +94 -65
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +1 -1
- package/release_notes.tmp +10 -2
- package/styles/tailwind/tailwind.scss +35 -68
@@ -55,8 +55,6 @@ export declare class ScCartItemMobileComponent implements OnInit {
|
|
55
55
|
*
|
56
56
|
* @param unitsHelper Экземпляр класса-помощника для работы со значениями единиц измерения товара.
|
57
57
|
* @param previewDialogService Сервис диалогового окна предварительного просмотра.
|
58
|
-
* @param urls Список ссылок на разделы backend'a.
|
59
|
-
* @param pathImageNotFound Путь до изображения 'Товар не найден'.
|
60
58
|
*/
|
61
59
|
constructor(unitsHelper: ScUnitsHelper, previewDialogService: TuiPreviewDialogService);
|
62
60
|
/**
|
@@ -1,5 +1,5 @@
|
|
1
|
-
import { ChangeDetectorRef, EventEmitter
|
2
|
-
import {
|
1
|
+
import { ChangeDetectorRef, EventEmitter } from '@angular/core';
|
2
|
+
import { ScCategory, ScIUrls } from '@snabcentr/client-core';
|
3
3
|
import { TuiSizeS } from '@taiga-ui/core';
|
4
4
|
import { Observable } from 'rxjs';
|
5
5
|
import * as i0 from "@angular/core";
|
@@ -8,26 +8,16 @@ import * as i0 from "@angular/core";
|
|
8
8
|
*/
|
9
9
|
export declare class ScCategoryCardComponent {
|
10
10
|
private readonly urls;
|
11
|
-
private renderer;
|
12
11
|
private readonly pathImageNotFound;
|
13
|
-
private readonly authService;
|
14
12
|
private readonly cdr;
|
15
13
|
/**
|
16
14
|
* Информация о категории.
|
17
15
|
*/
|
18
16
|
category?: ScCategory;
|
19
|
-
/**
|
20
|
-
* Ссылка на {@link HTMLElement} блока названия.
|
21
|
-
*/
|
22
|
-
private name;
|
23
17
|
/**
|
24
18
|
* Размер карточки категории.
|
25
19
|
*/
|
26
20
|
size: TuiSizeS;
|
27
|
-
/**
|
28
|
-
* Признак, что карточка является скелетоном.
|
29
|
-
*/
|
30
|
-
isSkeleton: boolean;
|
31
21
|
/**
|
32
22
|
* Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.
|
33
23
|
*/
|
@@ -37,9 +27,13 @@ export declare class ScCategoryCardComponent {
|
|
37
27
|
*/
|
38
28
|
readonly authStatus$: Observable<boolean>;
|
39
29
|
/**
|
40
|
-
* Признак
|
30
|
+
* Признак наведения на карточку.
|
41
31
|
*/
|
42
|
-
isHovered: boolean;
|
32
|
+
protected isHovered: boolean;
|
33
|
+
/**
|
34
|
+
* Признак того, отображается этот компонент на мобильном устройстве или нет.
|
35
|
+
*/
|
36
|
+
readonly isMobile: boolean;
|
43
37
|
/**
|
44
38
|
* Событие нажатия на карточку категории.
|
45
39
|
*/
|
@@ -52,28 +46,28 @@ export declare class ScCategoryCardComponent {
|
|
52
46
|
* Инициализирует экземпляр класса {@link CategoryCardComponent}.
|
53
47
|
*
|
54
48
|
* @param urls Список ссылок на разделы backend'a.
|
55
|
-
* @param renderer Экземпляр базового класса для реализации пользовательского рендеринга.
|
56
49
|
* @param pathImageNotFound Путь до изображения 'Товар не найден'.
|
50
|
+
* @param cdr Объект для работы с обнаружением изменений.
|
57
51
|
*/
|
58
|
-
constructor(urls: ScIUrls,
|
52
|
+
constructor(urls: ScIUrls, pathImageNotFound: string, cdr: ChangeDetectorRef);
|
59
53
|
/**
|
60
|
-
*
|
54
|
+
* Обработчик события наведения.
|
61
55
|
*
|
62
|
-
* @param
|
63
|
-
*/
|
64
|
-
getCategoryImgURL(category: ScCategory): string;
|
65
|
-
/**
|
66
|
-
* Обработчик события mousemove.
|
56
|
+
* @param isHovered Признак наведения на карточку.
|
67
57
|
*/
|
68
|
-
|
58
|
+
protected onHovered(isHovered: boolean): void;
|
69
59
|
/**
|
70
|
-
*
|
60
|
+
* Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию ("product_not_found").
|
61
|
+
*
|
62
|
+
* @param category Информация о категории.
|
71
63
|
*/
|
72
|
-
|
64
|
+
protected getCategoryImgURL(category: ScCategory): string;
|
73
65
|
/**
|
74
66
|
* Устанавливает компонент в очередь на обновление.
|
67
|
+
*
|
68
|
+
* @deprecated
|
75
69
|
*/
|
76
70
|
markForCheck(): void;
|
77
71
|
static ɵfac: i0.ɵɵFactoryDeclaration<ScCategoryCardComponent, never>;
|
78
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<ScCategoryCardComponent, "sc-category-card", never, { "category": { "alias": "category"; "required": false; }; "size": { "alias": "size"; "required": false; };
|
72
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScCategoryCardComponent, "sc-category-card", never, { "category": { "alias": "category"; "required": false; }; "size": { "alias": "size"; "required": false; }; }, { "clickOnCardEvent": "clickOnCardEvent"; "clickOnFavoriteEvent": "clickOnFavoriteEvent"; }, never, never, false, never>;
|
79
73
|
}
|
@@ -0,0 +1,24 @@
|
|
1
|
+
import { WritableSignal } from '@angular/core';
|
2
|
+
import * as i0 from "@angular/core";
|
3
|
+
/**
|
4
|
+
* Компонент просмотра изображений через событие наведения.
|
5
|
+
*/
|
6
|
+
export declare class HoverImageCarouselComponent {
|
7
|
+
/**
|
8
|
+
* Массив изображений.
|
9
|
+
*/
|
10
|
+
images: string[];
|
11
|
+
/**
|
12
|
+
* {@link WritableSignal} для отслеживания текущего индекса изображения.
|
13
|
+
*/
|
14
|
+
currentIndex: WritableSignal<number>;
|
15
|
+
/**
|
16
|
+
* Обработчик события mousemove.
|
17
|
+
*
|
18
|
+
* @param offsetX Отступ курсора мыши по оси X от края {@link HTMLElement}.
|
19
|
+
* @param target Объект {@link HTMLElement} в котором лежат изображения
|
20
|
+
*/
|
21
|
+
private onHover;
|
22
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<HoverImageCarouselComponent, never>;
|
23
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<HoverImageCarouselComponent, "sc-hover-image-carousel", never, { "images": { "alias": "images"; "required": false; }; }, {}, never, never, true, never>;
|
24
|
+
}
|
@@ -14,17 +14,13 @@ export declare class ScPriceCardComponent extends AbstractScPriceCard {
|
|
14
14
|
/**
|
15
15
|
* Признак того, что этот компонент отображается на мобильном устройстве.
|
16
16
|
*/
|
17
|
-
|
17
|
+
readonly isMobile: boolean;
|
18
18
|
/**
|
19
19
|
* Инициирует экземпляр класса {@link ScPriceCardComponent}.
|
20
20
|
*
|
21
21
|
* @param cdr Объект для работы с обнаружением изменений.
|
22
22
|
*/
|
23
23
|
constructor(cdr: ChangeDetectorRef);
|
24
|
-
/**
|
25
|
-
* Признак того, что компонент находится в корзине.
|
26
|
-
*/
|
27
|
-
get isInCart(): boolean;
|
28
24
|
/**
|
29
25
|
* Устанавливает компонент в очередь на обновление.
|
30
26
|
*/
|
@@ -15,11 +15,12 @@ import * as i13 from "@angular/forms";
|
|
15
15
|
import * as i14 from "@taiga-ui/kit";
|
16
16
|
import * as i15 from "@taiga-ui/cdk";
|
17
17
|
import * as i16 from "@taiga-ui/addon-charts";
|
18
|
+
import * as i17 from "./hover-image-carousel/hover-image-carousel.component";
|
18
19
|
/**
|
19
20
|
* Модуль каталога.
|
20
21
|
*/
|
21
22
|
export declare class ScCatalogModule {
|
22
23
|
static ɵfac: i0.ɵɵFactoryDeclaration<ScCatalogModule, never>;
|
23
|
-
static ɵmod: i0.ɵɵNgModuleDeclaration<ScCatalogModule, [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent], [typeof i9.CommonModule, typeof i10.RouterModule, typeof i11.TuiButton, typeof i11.TuiIcon, typeof i12.TuiIslandDirective, typeof i12.TuiInputNumberModule, typeof i11.TuiLabel, typeof i12.TuiTextfieldControllerModule, typeof i13.FormsModule, typeof i13.ReactiveFormsModule, typeof i11.TuiHintComponent, typeof i11.TuiHintDirective, typeof i11.TuiHintOptionsDirective, typeof i11.TuiHintUnstyled, typeof i11.TuiHintDriver, typeof i11.TuiHintPosition, typeof i11.TuiHintHover, typeof i11.TuiHintOverflow, typeof i11.TuiHintDescribe, typeof i11.TuiHintHost, typeof i11.TuiHintManual, typeof i11.TuiHintPointer, typeof i14.TuiFieldErrorPipe, typeof i11.TuiLoader, typeof i11.TuiLink, typeof i14.TuiElasticContainer, typeof i15.TuiLet, typeof i15.TuiRepeatTimes, typeof i14.TuiHighlight, typeof i16.TuiLineDaysChart, typeof i16.TuiLineDaysChartHint, typeof i16.TuiAxes, typeof i14.TuiButtonLoading], [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent]>;
|
24
|
+
static ɵmod: i0.ɵɵNgModuleDeclaration<ScCatalogModule, [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent], [typeof i9.CommonModule, typeof i10.RouterModule, typeof i11.TuiButton, typeof i11.TuiIcon, typeof i12.TuiIslandDirective, typeof i12.TuiInputNumberModule, typeof i11.TuiLabel, typeof i12.TuiTextfieldControllerModule, typeof i13.FormsModule, typeof i13.ReactiveFormsModule, typeof i11.TuiHintComponent, typeof i11.TuiHintDirective, typeof i11.TuiHintOptionsDirective, typeof i11.TuiHintUnstyled, typeof i11.TuiHintDriver, typeof i11.TuiHintPosition, typeof i11.TuiHintHover, typeof i11.TuiHintOverflow, typeof i11.TuiHintDescribe, typeof i11.TuiHintHost, typeof i11.TuiHintManual, typeof i11.TuiHintPointer, typeof i14.TuiFieldErrorPipe, typeof i11.TuiLoader, typeof i11.TuiLink, typeof i14.TuiElasticContainer, typeof i15.TuiLet, typeof i15.TuiRepeatTimes, typeof i14.TuiHighlight, typeof i16.TuiLineDaysChart, typeof i16.TuiLineDaysChartHint, typeof i16.TuiAxes, typeof i14.TuiButtonLoading, typeof i14.TuiAvatar, typeof i14.TuiBadge, typeof i17.HoverImageCarouselComponent, typeof i14.TuiLineClamp, typeof i15.TuiHovered], [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent]>;
|
24
25
|
static ɵinj: i0.ɵɵInjectorDeclaration<ScCatalogModule>;
|
25
26
|
}
|
@@ -101,6 +101,12 @@ export declare class AbstractScPriceCard {
|
|
101
101
|
* Возвращает ссылку на preview-изображение карточки товара.
|
102
102
|
*/
|
103
103
|
getCardImagePreview(): string;
|
104
|
+
/**
|
105
|
+
* Возвращает массив превью-изображений товара.
|
106
|
+
*
|
107
|
+
* @param product Объект товара.
|
108
|
+
*/
|
109
|
+
getCardImagePreviewList(product: ScProduct): string[];
|
104
110
|
static ɵfac: i0.ɵɵFactoryDeclaration<AbstractScPriceCard, never>;
|
105
111
|
static ɵdir: i0.ɵɵDirectiveDeclaration<AbstractScPriceCard, never, never, { "cartItem": { "alias": "cartItem"; "required": false; }; "product": { "alias": "product"; "required": false; }; "showQuantityControl": { "alias": "showQuantityControl"; "required": false; }; }, { "clickFavoriteEvent": "clickFavoriteEvent"; "clickAddToCartEvent": "clickAddToCartEvent"; "clickClearEvent": "clickClearEvent"; "clickCardEvent": "clickCardEvent"; "clickPriceHistoryEvent": "clickPriceHistoryEvent"; "quantityValueChanges": "quantityValueChanges"; }, never, never, false, never>;
|
106
112
|
}
|
@@ -38,8 +38,6 @@ export class ScCartItemMobileComponent {
|
|
38
38
|
*
|
39
39
|
* @param unitsHelper Экземпляр класса-помощника для работы со значениями единиц измерения товара.
|
40
40
|
* @param previewDialogService Сервис диалогового окна предварительного просмотра.
|
41
|
-
* @param urls Список ссылок на разделы backend'a.
|
42
|
-
* @param pathImageNotFound Путь до изображения 'Товар не найден'.
|
43
41
|
*/
|
44
42
|
constructor(unitsHelper, previewDialogService) {
|
45
43
|
this.unitsHelper = unitsHelper;
|
@@ -116,4 +114,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImpor
|
|
116
114
|
}], clickCardEvent: [{
|
117
115
|
type: Output
|
118
116
|
}] } });
|
119
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-item-mobile.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.ts","../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.html"],"names":[],"mappings":"AAAA,yCAAyC;AAEzC,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAe,SAAS,EAAE,MAAM,eAAe,CAAC;AAChJ,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAc,aAAa,EAA4B,MAAM,wBAAwB,CAAC;AAE7F,OAAO,EAAE,uBAAuB,EAAE,MAAM,eAAe,CAAC;AACxD,OAAO,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;;;;;;;;;;;AAExC;;GAEG;AAMH,MAAM,OAAO,yBAAyB;IAClC;;OAEG;IACH,IACW,QAAQ,CAAC,IAA4B;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,IAAI,IAAI,EAAE,CAAC;YACP,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACnD,CAAC;IACL,CAAC;IAED;;OAEG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IA+CD;;;;;;;OAOG;IACH,YACoB,WAA0B,EAEzB,oBAA6C;QAF9C,gBAAW,GAAX,WAAW,CAAe;QAEzB,yBAAoB,GAApB,oBAAoB,CAAyB;QAlDlE;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAgB,CAAC,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAE7G;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzG;;WAEG;QAEI,qBAAgB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEvE;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAErE;;WAEG;QACc,gBAAW,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;IAmBlD,CAAC;IAEJ;;OAEG;IACH,IAAW,OAAO;QACd,OAAO,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;IAClC,CAAC;IAED,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;IAClE,CAAC;IAED;;OAEG;IACI,iBAAiB;QACpB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACnF,CAAC;IAED;;;;OAIG;IACI,YAAY,CAAC,OAAkB;QAClC,OAAO,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IAC9C,CAAC;8GAzGQ,yBAAyB,+CA2EtB,uBAAuB;kGA3E1B,yBAAyB,2YCjBtC,2hQAkKA;;2FDjJa,yBAAyB;kBALrC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BA6E1C,MAAM;2BAAC,uBAAuB;yCAtExB,QAAQ;sBADlB,KAAK;gBAoBW,uBAAuB;sBADvC,SAAS;uBAAC,sBAAsB;gBAY1B,oBAAoB;sBAD1B,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAOA,kBAAkB;sBADxB,MAAM;gBAOA,cAAc;sBADpB,MAAM","sourcesContent":["/* eslint-disable no-underscore-dangle */\n\nimport { ChangeDetectionStrategy, Component, EventEmitter, Inject, inject, Input, OnInit, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { ScCartItem, ScImageHelper, ScProduct, ScUnitsHelper } from '@snabcentr/client-core';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { TuiPreviewDialogService } from '@taiga-ui/kit';\nimport { Observable, skip } from 'rxjs';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    selector: 'sc-cart-item-mobile',\n    templateUrl: './sc-cart-item-mobile.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCartItemMobileComponent implements OnInit {\n    /**\n     * Элемент корзины.\n     */\n    @Input()\n    public set cartItem(data: ScCartItem | undefined) {\n        this._cartItem = data;\n\n        if (data) {\n            this.quantityControl.patchValue(data.quantity);\n        }\n    }\n\n    /**\n     * Элемент корзины.\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * Ссылка на представление спецификации.\n     */\n    @ViewChild('specificationPreview')\n    private readonly specificationPreviewRef?: TemplateRef<TuiDialogContext>;\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(0, { updateOn: 'blur' });\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges.pipe(skip(1));\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickDeleteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickSettingsEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на карточку.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Хелпер для работы с изображениями товара.\n     */\n    private readonly imageHelper = inject(ScImageHelper);\n\n    /**\n     * Элемент корзины.\n     */\n    private _cartItem?: ScCartItem;\n\n    /**\n     * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.\n     *\n     * @param unitsHelper Экземпляр класса-помощника для работы со значениями единиц измерения товара.\n     * @param previewDialogService Сервис диалогового окна предварительного просмотра.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: ScUnitsHelper,\n        @Inject(TuiPreviewDialogService)\n        private readonly previewDialogService: TuiPreviewDialogService\n    ) {}\n\n    /**\n     * Продукт элемента корзины.\n     */\n    public get product(): ScProduct | undefined {\n        return this.cartItem?.product;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.quantityValueChanges = this.quantityControl.valueChanges;\n    }\n\n    /**\n     * Отобразить спецификацию.\n     */\n    public showSpecification(): void {\n        this.previewDialogService.open(this.specificationPreviewRef ?? '').subscribe();\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     *\n     * @param product Позиция товара/услуги.\n     */\n    public getCardImage(product: ScProduct): string {\n        return this.imageHelper.getImage(product);\n    }\n}\n","<ng-container *ngIf=\"cartItem && product; else skeleton\">\n    <div class=\"flex min-w-72 gap-2 rounded-xl border border-tui-base-04 bg-white p-2 shadow-sc-2\">\n        <div class=\"flex w-40 items-center\">\n            <img\n                (click)=\"clickCardEvent.emit()\"\n                [src]=\"getCardImage(product)\"\n                [alt]=\"product.name\"\n                class=\"max-h-full w-full rounded-xl object-cover\"\n            />\n        </div>\n        <div class=\"flex w-full flex-col items-center justify-between gap-2 sm:flex-row sm:gap-5\">\n            <div class=\"flex max-w-full flex-col gap-1 self-start\">\n                <a\n                    tuiLink\n                    (click)=\"clickCardEvent.emit()\"\n                >\n                    <span class=\"font-bold\">{{ product.name }}</span>\n                </a>\n\n                <div class=\"flex w-full flex-wrap gap-x-2 gap-y-1 text-xs text-tui-base-07\">\n                    <p *ngIf=\"product.pack\">\n                        Норма упаковки: {{ product.pack }}\n                        <tui-icon\n                            *ngIf=\"product.ignoreMinCountCheck\"\n                            icon=\"@tui.package\"\n                            [tuiHint]=\"minCountHint\"\n                            [tuiHintShowDelay]=\"100\"\n                            tuiHintDirection=\"top\"\n                            class=\"text-sm\"\n                        ></tui-icon>\n                        <ng-template #minCountHint>\n                            Доступен заказ <br />\n                            произвольного <br />\n                            количества\n                        </ng-template>\n                    </p>\n                    <p>Артикул: {{ product.code }}</p>\n                </div>\n\n                <div class=\"flex w-full flex-wrap items-center gap-x-2 gap-y-1\">\n                    <p\n                        *ngIf=\"!product.priceInRub\"\n                        class=\"font-bold\"\n                    >\n                        {{ product.costString }}\n                    </p>\n                    <p class=\"text-xs font-bold text-tui-text-01\">{{ product.costRubString }}</p>\n                    <span\n                        *ngIf=\"product.discount\"\n                        class=\"flex items-center\"\n                    >\n                        <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n                        <span class=\"font-bold text-tui-success-fill\"> -{{ product.discount.percent }}% </span>\n                        <tui-icon\n                            icon=\"@tui.info\"\n                            [tuiHint]=\"discountHint\"\n                            [tuiHintShowDelay]=\"100\"\n                            tuiHintDirection=\"top\"\n                            class=\"!h-4 !text-xs\"\n                        ></tui-icon>\n                        <ng-template #discountHint>\n                            <div class=\"font-bold\">{{ product.discount.name }}</div>\n                            <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n                        </ng-template>\n                    </span>\n                </div>\n\n                <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n\n                <div\n                    *ngIf=\"cartItem.height || cartItem.length || cartItem.width\"\n                    class=\"flex items-center gap-1\"\n                >\n                    <button\n                        tuiIconButton\n                        iconStart=\"@tui.settings\"\n                        (click)=\"clickSettingsEvent.emit()\"\n                        size=\"s\"\n                        appearance=\"secondary\"\n                        class=\"mr-2 !self-center\"\n                    ></button>\n                    <div class=\"flex flex-wrap gap-x-2 text-xs text-tui-base-07\">\n                        <p *ngIf=\"cartItem.marker\">Маркировка: {{ cartItem.marker }}</p>\n                        <p *ngIf=\"cartItem.width\">Ширина: {{ cartItem.width }} м.</p>\n                        <ng-container *ngIf=\"cartItem.height; else length\">\n                            <p>Высота: {{ cartItem.height }} м.</p>\n                        </ng-container>\n                        <ng-template #length>\n                            <p *ngIf=\"cartItem.length\">Длина: {{ cartItem.length }} м.</p>\n                        </ng-template>\n                    </div>\n                </div>\n\n                <a\n                    tuiLink\n                    *ngIf=\"cartItem.specificationImgUrl\"\n                    (click)=\"showSpecification()\"\n                    >Спецификация</a\n                >\n                <ng-template\n                    #specificationPreview\n                    let-preview\n                >\n                    <tui-preview\n                        [rotatable]=\"false\"\n                        [zoomable]=\"false\"\n                    >\n                        <img\n                            *polymorpheusOutlet=\"cartItem.specificationImgUrl as src\"\n                            alt=\"preview\"\n                            [src]=\"cartItem.specificationImgUrl\"\n                        />\n                        <button\n                            iconStart=\"@tui.x\"\n                            title=\"Close\"\n                            tuiIconButton\n                            tuiPreviewAction\n                            type=\"button\"\n                            (click)=\"preview.complete()\"\n                        ></button>\n                    </tui-preview>\n                </ng-template>\n            </div>\n            <div class=\"flex flex-col gap-1 self-start sm:mr-9 sm:self-center\">\n                <div class=\"flex gap-2\">\n                    <sc-input-quantity\n                        #inputQuantity\n                        *ngIf=\"quantityControl\"\n                        [formControl]=\"quantityControl\"\n                        [showCross]=\"false\"\n                        size=\"s\"\n                        [step]=\"unitsHelper.productMultiplicity(product)\"\n                        [ignoreStepValidators]=\"product.ignoreMinCountCheck\"\n                        (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n                        class=\"w-28\"\n                    ></sc-input-quantity>\n                    <button\n                        tuiIconButton\n                        iconStart=\"@tui.trash-2\"\n                        [style.border-radius.%]=\"100\"\n                        size=\"xs\"\n                        (click)=\"clickDeleteEvent.emit()\"\n                    ></button>\n                </div>\n                <p class=\"text-sm font-bold text-tui-text-01\">\n                    Итого: {{ cartItem.costRub | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}\n                </p>\n            </div>\n        </div>\n    </div>\n</ng-container>\n\n<ng-template #skeleton>\n    <div class=\"flex w-full gap-2\">\n        <div class=\"size-20 rounded-xl bg-tui-base-02\"></div>\n        <div class=\"flex grow flex-col gap-2.5 bg-white\">\n            <div class=\"h-4 w-full rounded-xl bg-tui-base-02\"></div>\n            <div class=\"h-4 w-3/5 rounded-xl bg-tui-base-02\"></div>\n            <div class=\"h-4 w-4/5 rounded-xl bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-item-mobile.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.ts","../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.html"],"names":[],"mappings":"AAAA,yCAAyC;AAEzC,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAe,SAAS,EAAE,MAAM,eAAe,CAAC;AAChJ,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAc,aAAa,EAA4B,MAAM,wBAAwB,CAAC;AAE7F,OAAO,EAAE,uBAAuB,EAAE,MAAM,eAAe,CAAC;AACxD,OAAO,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;;;;;;;;;;;AAExC;;GAEG;AAMH,MAAM,OAAO,yBAAyB;IAClC;;OAEG;IACH,IACW,QAAQ,CAAC,IAA4B;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,IAAI,IAAI,EAAE,CAAC;YACP,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACnD,CAAC;IACL,CAAC;IAED;;OAEG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IA+CD;;;;;OAKG;IACH,YACoB,WAA0B,EAEzB,oBAA6C;QAF9C,gBAAW,GAAX,WAAW,CAAe;QAEzB,yBAAoB,GAApB,oBAAoB,CAAyB;QAhDlE;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAgB,CAAC,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAE7G;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzG;;WAEG;QAEI,qBAAgB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEvE;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAErE;;WAEG;QACc,gBAAW,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;IAiBlD,CAAC;IAEJ;;OAEG;IACH,IAAW,OAAO;QACd,OAAO,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;IAClC,CAAC;IAED,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;IAClE,CAAC;IAED;;OAEG;IACI,iBAAiB;QACpB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACnF,CAAC;IAED;;;;OAIG;IACI,YAAY,CAAC,OAAkB;QAClC,OAAO,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IAC9C,CAAC;8GAvGQ,yBAAyB,+CAyEtB,uBAAuB;kGAzE1B,yBAAyB,2YCjBtC,2hQAkKA;;2FDjJa,yBAAyB;kBALrC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BA2E1C,MAAM;2BAAC,uBAAuB;yCApExB,QAAQ;sBADlB,KAAK;gBAoBW,uBAAuB;sBADvC,SAAS;uBAAC,sBAAsB;gBAY1B,oBAAoB;sBAD1B,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAOA,kBAAkB;sBADxB,MAAM;gBAOA,cAAc;sBADpB,MAAM","sourcesContent":["/* eslint-disable no-underscore-dangle */\n\nimport { ChangeDetectionStrategy, Component, EventEmitter, Inject, inject, Input, OnInit, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { ScCartItem, ScImageHelper, ScProduct, ScUnitsHelper } from '@snabcentr/client-core';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { TuiPreviewDialogService } from '@taiga-ui/kit';\nimport { Observable, skip } from 'rxjs';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    selector: 'sc-cart-item-mobile',\n    templateUrl: './sc-cart-item-mobile.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCartItemMobileComponent implements OnInit {\n    /**\n     * Элемент корзины.\n     */\n    @Input()\n    public set cartItem(data: ScCartItem | undefined) {\n        this._cartItem = data;\n\n        if (data) {\n            this.quantityControl.patchValue(data.quantity);\n        }\n    }\n\n    /**\n     * Элемент корзины.\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * Ссылка на представление спецификации.\n     */\n    @ViewChild('specificationPreview')\n    private readonly specificationPreviewRef?: TemplateRef<TuiDialogContext>;\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(0, { updateOn: 'blur' });\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges.pipe(skip(1));\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickDeleteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickSettingsEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на карточку.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Хелпер для работы с изображениями товара.\n     */\n    private readonly imageHelper = inject(ScImageHelper);\n\n    /**\n     * Элемент корзины.\n     */\n    private _cartItem?: ScCartItem;\n\n    /**\n     * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.\n     *\n     * @param unitsHelper Экземпляр класса-помощника для работы со значениями единиц измерения товара.\n     * @param previewDialogService Сервис диалогового окна предварительного просмотра.\n     */\n    public constructor(\n        public readonly unitsHelper: ScUnitsHelper,\n        @Inject(TuiPreviewDialogService)\n        private readonly previewDialogService: TuiPreviewDialogService\n    ) {}\n\n    /**\n     * Продукт элемента корзины.\n     */\n    public get product(): ScProduct | undefined {\n        return this.cartItem?.product;\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.quantityValueChanges = this.quantityControl.valueChanges;\n    }\n\n    /**\n     * Отобразить спецификацию.\n     */\n    public showSpecification(): void {\n        this.previewDialogService.open(this.specificationPreviewRef ?? '').subscribe();\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     *\n     * @param product Позиция товара/услуги.\n     */\n    public getCardImage(product: ScProduct): string {\n        return this.imageHelper.getImage(product);\n    }\n}\n","<ng-container *ngIf=\"cartItem && product; else skeleton\">\n    <div class=\"flex min-w-72 gap-2 rounded-xl border border-tui-base-04 bg-white p-2 shadow-sc-2\">\n        <div class=\"flex w-40 items-center\">\n            <img\n                (click)=\"clickCardEvent.emit()\"\n                [src]=\"getCardImage(product)\"\n                [alt]=\"product.name\"\n                class=\"max-h-full w-full rounded-xl object-cover\"\n            />\n        </div>\n        <div class=\"flex w-full flex-col items-center justify-between gap-2 sm:flex-row sm:gap-5\">\n            <div class=\"flex max-w-full flex-col gap-1 self-start\">\n                <a\n                    tuiLink\n                    (click)=\"clickCardEvent.emit()\"\n                >\n                    <span class=\"font-bold\">{{ product.name }}</span>\n                </a>\n\n                <div class=\"flex w-full flex-wrap gap-x-2 gap-y-1 text-xs text-tui-base-07\">\n                    <p *ngIf=\"product.pack\">\n                        Норма упаковки: {{ product.pack }}\n                        <tui-icon\n                            *ngIf=\"product.ignoreMinCountCheck\"\n                            icon=\"@tui.package\"\n                            [tuiHint]=\"minCountHint\"\n                            [tuiHintShowDelay]=\"100\"\n                            tuiHintDirection=\"top\"\n                            class=\"text-sm\"\n                        ></tui-icon>\n                        <ng-template #minCountHint>\n                            Доступен заказ <br />\n                            произвольного <br />\n                            количества\n                        </ng-template>\n                    </p>\n                    <p>Артикул: {{ product.code }}</p>\n                </div>\n\n                <div class=\"flex w-full flex-wrap items-center gap-x-2 gap-y-1\">\n                    <p\n                        *ngIf=\"!product.priceInRub\"\n                        class=\"font-bold\"\n                    >\n                        {{ product.costString }}\n                    </p>\n                    <p class=\"text-xs font-bold text-tui-text-01\">{{ product.costRubString }}</p>\n                    <span\n                        *ngIf=\"product.discount\"\n                        class=\"flex items-center\"\n                    >\n                        <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n                        <span class=\"font-bold text-tui-success-fill\"> -{{ product.discount.percent }}% </span>\n                        <tui-icon\n                            icon=\"@tui.info\"\n                            [tuiHint]=\"discountHint\"\n                            [tuiHintShowDelay]=\"100\"\n                            tuiHintDirection=\"top\"\n                            class=\"!h-4 !text-xs\"\n                        ></tui-icon>\n                        <ng-template #discountHint>\n                            <div class=\"font-bold\">{{ product.discount.name }}</div>\n                            <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n                        </ng-template>\n                    </span>\n                </div>\n\n                <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n\n                <div\n                    *ngIf=\"cartItem.height || cartItem.length || cartItem.width\"\n                    class=\"flex items-center gap-1\"\n                >\n                    <button\n                        tuiIconButton\n                        iconStart=\"@tui.settings\"\n                        (click)=\"clickSettingsEvent.emit()\"\n                        size=\"s\"\n                        appearance=\"secondary\"\n                        class=\"mr-2 !self-center\"\n                    ></button>\n                    <div class=\"flex flex-wrap gap-x-2 text-xs text-tui-base-07\">\n                        <p *ngIf=\"cartItem.marker\">Маркировка: {{ cartItem.marker }}</p>\n                        <p *ngIf=\"cartItem.width\">Ширина: {{ cartItem.width }} м.</p>\n                        <ng-container *ngIf=\"cartItem.height; else length\">\n                            <p>Высота: {{ cartItem.height }} м.</p>\n                        </ng-container>\n                        <ng-template #length>\n                            <p *ngIf=\"cartItem.length\">Длина: {{ cartItem.length }} м.</p>\n                        </ng-template>\n                    </div>\n                </div>\n\n                <a\n                    tuiLink\n                    *ngIf=\"cartItem.specificationImgUrl\"\n                    (click)=\"showSpecification()\"\n                    >Спецификация</a\n                >\n                <ng-template\n                    #specificationPreview\n                    let-preview\n                >\n                    <tui-preview\n                        [rotatable]=\"false\"\n                        [zoomable]=\"false\"\n                    >\n                        <img\n                            *polymorpheusOutlet=\"cartItem.specificationImgUrl as src\"\n                            alt=\"preview\"\n                            [src]=\"cartItem.specificationImgUrl\"\n                        />\n                        <button\n                            iconStart=\"@tui.x\"\n                            title=\"Close\"\n                            tuiIconButton\n                            tuiPreviewAction\n                            type=\"button\"\n                            (click)=\"preview.complete()\"\n                        ></button>\n                    </tui-preview>\n                </ng-template>\n            </div>\n            <div class=\"flex flex-col gap-1 self-start sm:mr-9 sm:self-center\">\n                <div class=\"flex gap-2\">\n                    <sc-input-quantity\n                        #inputQuantity\n                        *ngIf=\"quantityControl\"\n                        [formControl]=\"quantityControl\"\n                        [showCross]=\"false\"\n                        size=\"s\"\n                        [step]=\"unitsHelper.productMultiplicity(product)\"\n                        [ignoreStepValidators]=\"product.ignoreMinCountCheck\"\n                        (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n                        class=\"w-28\"\n                    ></sc-input-quantity>\n                    <button\n                        tuiIconButton\n                        iconStart=\"@tui.trash-2\"\n                        [style.border-radius.%]=\"100\"\n                        size=\"xs\"\n                        (click)=\"clickDeleteEvent.emit()\"\n                    ></button>\n                </div>\n                <p class=\"text-sm font-bold text-tui-text-01\">\n                    Итого: {{ cartItem.costRub | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}\n                </p>\n            </div>\n        </div>\n    </div>\n</ng-container>\n\n<ng-template #skeleton>\n    <div class=\"flex w-full gap-2\">\n        <div class=\"size-20 rounded-xl bg-tui-base-02\"></div>\n        <div class=\"flex grow flex-col gap-2.5 bg-white\">\n            <div class=\"h-4 w-full rounded-xl bg-tui-base-02\"></div>\n            <div class=\"h-4 w-3/5 rounded-xl bg-tui-base-02\"></div>\n            <div class=\"h-4 w-4/5 rounded-xl bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
|
@@ -1,9 +1,11 @@
|
|
1
|
-
import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding,
|
2
|
-
import { SC_PATH_IMAGE_NOT_FOUND, SC_URLS } from '@snabcentr/client-core';
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, Inject, inject, Input, Output } from '@angular/core';
|
2
|
+
import { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService } from '@snabcentr/client-core';
|
3
|
+
import { TUI_IS_MOBILE } from '@taiga-ui/cdk';
|
3
4
|
import * as i0 from "@angular/core";
|
4
|
-
import * as i1 from "@
|
5
|
-
import * as i2 from "@
|
6
|
-
import * as i3 from "
|
5
|
+
import * as i1 from "@angular/common";
|
6
|
+
import * as i2 from "@taiga-ui/kit";
|
7
|
+
import * as i3 from "@taiga-ui/cdk";
|
8
|
+
import * as i4 from "../sc-favorite-btn/sc-favorite-btn.component";
|
7
9
|
/**
|
8
10
|
* Карточка категории.
|
9
11
|
*/
|
@@ -12,23 +14,17 @@ export class ScCategoryCardComponent {
|
|
12
14
|
* Инициализирует экземпляр класса {@link CategoryCardComponent}.
|
13
15
|
*
|
14
16
|
* @param urls Список ссылок на разделы backend'a.
|
15
|
-
* @param renderer Экземпляр базового класса для реализации пользовательского рендеринга.
|
16
17
|
* @param pathImageNotFound Путь до изображения 'Товар не найден'.
|
18
|
+
* @param cdr Объект для работы с обнаружением изменений.
|
17
19
|
*/
|
18
|
-
constructor(urls,
|
20
|
+
constructor(urls, pathImageNotFound, cdr) {
|
19
21
|
this.urls = urls;
|
20
|
-
this.renderer = renderer;
|
21
22
|
this.pathImageNotFound = pathImageNotFound;
|
22
|
-
this.authService = authService;
|
23
23
|
this.cdr = cdr;
|
24
24
|
/**
|
25
25
|
* Размер карточки категории.
|
26
26
|
*/
|
27
27
|
this.size = 'm';
|
28
|
-
/**
|
29
|
-
* Признак, что карточка является скелетоном.
|
30
|
-
*/
|
31
|
-
this.isSkeleton = false;
|
32
28
|
/**
|
33
29
|
* Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.
|
34
30
|
*/
|
@@ -36,11 +32,15 @@ export class ScCategoryCardComponent {
|
|
36
32
|
/**
|
37
33
|
* {@link Observable} изменения статуса авторизации.
|
38
34
|
*/
|
39
|
-
this.authStatus$ =
|
35
|
+
this.authStatus$ = inject(ScAuthService).getAuthChange();
|
40
36
|
/**
|
41
|
-
* Признак
|
37
|
+
* Признак наведения на карточку.
|
42
38
|
*/
|
43
39
|
this.isHovered = false;
|
40
|
+
/**
|
41
|
+
* Признак того, отображается этот компонент на мобильном устройстве или нет.
|
42
|
+
*/
|
43
|
+
this.isMobile = inject(TUI_IS_MOBILE);
|
44
44
|
/**
|
45
45
|
* Событие нажатия на карточку категории.
|
46
46
|
*/
|
@@ -51,70 +51,51 @@ export class ScCategoryCardComponent {
|
|
51
51
|
this.clickOnFavoriteEvent = new EventEmitter();
|
52
52
|
}
|
53
53
|
/**
|
54
|
-
*
|
54
|
+
* Обработчик события наведения.
|
55
55
|
*
|
56
|
-
* @param
|
56
|
+
* @param isHovered Признак наведения на карточку.
|
57
57
|
*/
|
58
|
-
|
59
|
-
|
58
|
+
onHovered(isHovered) {
|
59
|
+
this.isHovered = isHovered;
|
60
60
|
}
|
61
61
|
/**
|
62
|
-
*
|
63
|
-
|
64
|
-
|
65
|
-
if (this.isHovered) {
|
66
|
-
this.name.nativeElement.style.maxHeight = `${this.name.nativeElement.scrollHeight / 16}rem`;
|
67
|
-
}
|
68
|
-
}
|
69
|
-
/**
|
70
|
-
* Обработчик события mouseleave.
|
62
|
+
* Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию ("product_not_found").
|
63
|
+
*
|
64
|
+
* @param category Информация о категории.
|
71
65
|
*/
|
72
|
-
|
73
|
-
|
74
|
-
this.renderer.setStyle(this.name.nativeElement, 'maxHeight', null);
|
75
|
-
}
|
66
|
+
getCategoryImgURL(category) {
|
67
|
+
return category.properties?.image ? `${this.urls.imgServerUrl}/${category.properties.image}` : this.pathImageNotFound;
|
76
68
|
}
|
77
69
|
/**
|
78
70
|
* Устанавливает компонент в очередь на обновление.
|
71
|
+
*
|
72
|
+
* @deprecated
|
79
73
|
*/
|
80
74
|
markForCheck() {
|
81
75
|
this.cdr.markForCheck();
|
82
76
|
}
|
83
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: ScCategoryCardComponent, deps: [{ token: SC_URLS }, { token:
|
84
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.8", type: ScCategoryCardComponent, selector: "sc-category-card", inputs: { category: "category", size: "size"
|
77
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: ScCategoryCardComponent, deps: [{ token: SC_URLS }, { token: SC_PATH_IMAGE_NOT_FOUND }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
78
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.8", type: ScCategoryCardComponent, selector: "sc-category-card", inputs: { category: "category", size: "size" }, outputs: { clickOnCardEvent: "clickOnCardEvent", clickOnFavoriteEvent: "clickOnFavoriteEvent" }, host: { properties: { "attr.data-size": "this.size" } }, ngImport: i0, template: "<div class=\"relative\">\n <button\n (click)=\"clickOnCardEvent.emit(category)\"\n (tuiHoveredChange)=\"onHovered($event)\"\n class=\"category-btn rounded-tui-radius-m border-tui-base-04 flex flex-col overflow-hidden border\"\n >\n <div class=\"img-wrapper w-full grow overflow-hidden\">\n <img\n *ngIf=\"category && category.properties?.image\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n class=\"size-full object-cover\"\n />\n\n <div\n *ngIf=\"!category\"\n class=\"img-wrapper bg-tui-base-02 size-full\"\n ></div>\n\n <img\n *ngIf=\"category && !category.properties?.image\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n class=\"size-full rounded-xl object-contain p-2\"\n />\n </div>\n\n <div class=\"name flex w-full items-center justify-center\">\n <tui-line-clamp\n *ngIf=\"category; else skeletonName\"\n [content]=\"category.name\"\n class=\"pointer-events-none !h-auto\"\n [linesLimit]=\"isHovered || isMobile ? 4 : 2\"\n />\n </div>\n <ng-template #skeletonName>\n <div class=\"skeleton-name bg-tui-base-02 rounded-tui-radius-s\"></div>\n </ng-template>\n </button>\n <sc-favorite-btn\n *ngIf=\"category && (authStatus$ | async)\"\n [showLoader]=\"favoriteShowLoader\"\n [isFavorite]=\"category.isFavorite\"\n (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n class=\"absolute left-1 top-1\"\n ></sc-favorite-btn>\n</div>\n", styles: [":host{--tui-duration: .15s}:host[data-size=m] button.category-btn{width:100%;height:12.5rem}:host[data-size=m] button.category-btn .img-wrapper{max-height:8.75rem}:host[data-size=m] button.category-btn .name{padding:.25rem 1rem;margin-block:auto;font-size:.9375rem;line-height:1.5rem;font-weight:800}:host[data-size=m] button.category-btn .name .skeleton-name{width:10rem;height:1rem}:host[data-size=s] button.category-btn{width:100%;height:10rem}:host[data-size=s] button.category-btn .img-wrapper{max-height:7rem}:host[data-size=s] button.category-btn .name{padding:.25rem .5rem;margin-block:auto;font-size:.8125rem;line-height:1.25rem;font-weight:800}:host[data-size=s] button.category-btn .name .skeleton-name{width:7rem;height:.75rem}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.TuiLineClamp, selector: "tui-line-clamp", inputs: ["lineHeight", "content", "linesLimit"], outputs: ["overflownChange"] }, { kind: "directive", type: i3.TuiHovered, selector: "[tuiHoveredChange]", outputs: ["tuiHoveredChange"] }, { kind: "component", type: i4.ScFavoriteBtnComponent, selector: "sc-favorite-btn", inputs: ["isFavorite", "showLoader", "disabled"], outputs: ["clickEvent"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
85
79
|
}
|
86
80
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: ScCategoryCardComponent, decorators: [{
|
87
81
|
type: Component,
|
88
|
-
args: [{ selector: 'sc-category-card', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"relative
|
82
|
+
args: [{ selector: 'sc-category-card', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"relative\">\n <button\n (click)=\"clickOnCardEvent.emit(category)\"\n (tuiHoveredChange)=\"onHovered($event)\"\n class=\"category-btn rounded-tui-radius-m border-tui-base-04 flex flex-col overflow-hidden border\"\n >\n <div class=\"img-wrapper w-full grow overflow-hidden\">\n <img\n *ngIf=\"category && category.properties?.image\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n class=\"size-full object-cover\"\n />\n\n <div\n *ngIf=\"!category\"\n class=\"img-wrapper bg-tui-base-02 size-full\"\n ></div>\n\n <img\n *ngIf=\"category && !category.properties?.image\"\n [src]=\"getCategoryImgURL(category)\"\n [alt]=\"category.name\"\n class=\"size-full rounded-xl object-contain p-2\"\n />\n </div>\n\n <div class=\"name flex w-full items-center justify-center\">\n <tui-line-clamp\n *ngIf=\"category; else skeletonName\"\n [content]=\"category.name\"\n class=\"pointer-events-none !h-auto\"\n [linesLimit]=\"isHovered || isMobile ? 4 : 2\"\n />\n </div>\n <ng-template #skeletonName>\n <div class=\"skeleton-name bg-tui-base-02 rounded-tui-radius-s\"></div>\n </ng-template>\n </button>\n <sc-favorite-btn\n *ngIf=\"category && (authStatus$ | async)\"\n [showLoader]=\"favoriteShowLoader\"\n [isFavorite]=\"category.isFavorite\"\n (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n class=\"absolute left-1 top-1\"\n ></sc-favorite-btn>\n</div>\n", styles: [":host{--tui-duration: .15s}:host[data-size=m] button.category-btn{width:100%;height:12.5rem}:host[data-size=m] button.category-btn .img-wrapper{max-height:8.75rem}:host[data-size=m] button.category-btn .name{padding:.25rem 1rem;margin-block:auto;font-size:.9375rem;line-height:1.5rem;font-weight:800}:host[data-size=m] button.category-btn .name .skeleton-name{width:10rem;height:1rem}:host[data-size=s] button.category-btn{width:100%;height:10rem}:host[data-size=s] button.category-btn .img-wrapper{max-height:7rem}:host[data-size=s] button.category-btn .name{padding:.25rem .5rem;margin-block:auto;font-size:.8125rem;line-height:1.25rem;font-weight:800}:host[data-size=s] button.category-btn .name .skeleton-name{width:7rem;height:.75rem}\n"] }]
|
89
83
|
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
90
84
|
type: Inject,
|
91
85
|
args: [SC_URLS]
|
92
|
-
}] }, { type:
|
86
|
+
}] }, { type: undefined, decorators: [{
|
93
87
|
type: Inject,
|
94
88
|
args: [SC_PATH_IMAGE_NOT_FOUND]
|
95
|
-
}] }, { type:
|
89
|
+
}] }, { type: i0.ChangeDetectorRef }], propDecorators: { category: [{
|
96
90
|
type: Input
|
97
|
-
}], name: [{
|
98
|
-
type: ViewChild,
|
99
|
-
args: ['name']
|
100
91
|
}], size: [{
|
101
92
|
type: Input
|
102
93
|
}, {
|
103
94
|
type: HostBinding,
|
104
95
|
args: ['attr.data-size']
|
105
|
-
}], isSkeleton: [{
|
106
|
-
type: Input
|
107
|
-
}], isHovered: [{
|
108
|
-
type: Input
|
109
96
|
}], clickOnCardEvent: [{
|
110
97
|
type: Output
|
111
98
|
}], clickOnFavoriteEvent: [{
|
112
99
|
type: Output
|
113
|
-
}], moveEnterHandler: [{
|
114
|
-
type: HostListener,
|
115
|
-
args: ['mousemove']
|
116
|
-
}], mouseLeaveHandler: [{
|
117
|
-
type: HostListener,
|
118
|
-
args: ['mouseleave']
|
119
100
|
}] } });
|
120
|
-
//# 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,EACH,uBAAuB,EAEvB,SAAS,EAET,YAAY,EACZ,WAAW,EACX,YAAY,EACZ,MAAM,EACN,KAAK,EACL,MAAM,EAEN,SAAS,EACZ,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAsC,MAAM,wBAAwB,CAAC;;;;;AAI9G;;GAEG;AAOH,MAAM,OAAO,uBAAuB;IAsDhC;;;;;;OAMG;IACH,YACsC,IAAa,EACvC,QAAmB,EACuB,iBAAyB,EAC1D,WAA0B,EAC1B,GAAsB;QAJL,SAAI,GAAJ,IAAI,CAAS;QACvC,aAAQ,GAAR,QAAQ,CAAW;QACuB,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,gBAAW,GAAX,WAAW,CAAe;QAC1B,QAAG,GAAH,GAAG,CAAmB;QArD3C;;WAEG;QAGI,SAAI,GAAa,GAAG,CAAC;QAE5B;;WAEG;QAEI,eAAU,GAAY,KAAK,CAAC;QAEnC;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QACa,gBAAW,GAAwB,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;QAEpF;;WAEG;QAEI,cAAS,GAAY,KAAK,CAAC;QAElC;;WAEG;QAEI,qBAAgB,GAA6B,IAAI,YAAY,EAAc,CAAC;QAEnF;;WAEG;QAEI,yBAAoB,GAA6B,IAAI,YAAY,EAAc,CAAC;IAepF,CAAC;IAEJ;;;;OAIG;IACI,iBAAiB,CAAC,QAAoB;QACzC,OAAO,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,GAAG,GAAG,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;IAC3H,CAAC;IAED;;OAEG;IAEK,gBAAgB;QACpB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,SAAS,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,GAAG,EAAE,KAAK,CAAC;QAChG,CAAC;IACL,CAAC;IAED;;OAEG;IAEK,iBAAiB;QACrB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;QACvE,CAAC;IACL,CAAC;IAED;;OAEG;IACI,YAAY;QACf,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;8GAvGQ,uBAAuB,kBA8DpB,OAAO,sCAEP,uBAAuB;kGAhE1B,uBAAuB,yeC3BpC,0vCAwBA;;2FDGa,uBAAuB;kBANnC,SAAS;+BACI,kBAAkB,mBAGX,uBAAuB,CAAC,MAAM;;0BAgE1C,MAAM;2BAAC,OAAO;;0BAEd,MAAM;2BAAC,uBAAuB;qGA3D5B,QAAQ;sBADd,KAAK;gBAOE,IAAI;sBADX,SAAS;uBAAC,MAAM;gBAQV,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB;gBAOtB,UAAU;sBADhB,KAAK;gBAiBC,SAAS;sBADf,KAAK;gBAOC,gBAAgB;sBADtB,MAAM;gBAOA,oBAAoB;sBAD1B,MAAM;gBA+BC,gBAAgB;sBADvB,YAAY;uBAAC,WAAW;gBAWjB,iBAAiB;sBADxB,YAAY;uBAAC,YAAY","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    Output,\n    Renderer2,\n    ViewChild\n} from '@angular/core';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCategory, ScIUrls } from '@snabcentr/client-core';\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     * Ссылка на {@link HTMLElement} блока названия.\n     */\n    @ViewChild('name')\n    private name: ElementRef<HTMLElement>;\n\n    /**\n     * Размер карточки категории.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS = 'm';\n\n    /**\n     * Признак, что карточка является скелетоном.\n     */\n    @Input()\n    public isSkeleton: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = this.authService.getAuthChange();\n\n    /**\n     * Признак что категория имеет поведение наведения и скрытия названия.\n     */\n    @Input()\n    public isHovered: boolean = false;\n\n    /**\n     * Событие нажатия на карточку категории.\n     */\n    @Output()\n    public clickOnCardEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Событие нажатия на кнопку избранной категории.\n     */\n    @Output()\n    public clickOnFavoriteEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Инициализирует экземпляр класса {@link CategoryCardComponent}.\n     *\n     * @param urls Список ссылок на разделы backend'a.\n     * @param renderer Экземпляр базового класса для реализации пользовательского рендеринга.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        private renderer: Renderer2,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly authService: ScAuthService,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию (\"product_not_found\").\n     *\n     * @param category Информация о категории.\n     */\n    public getCategoryImgURL(category: ScCategory): string {\n        return category.properties?.image ? this.urls.imgServerUrl + '/' + category.properties?.image : this.pathImageNotFound;\n    }\n\n    /**\n     * Обработчик события mousemove.\n     */\n    @HostListener('mousemove')\n    private moveEnterHandler(): void {\n        if (this.isHovered) {\n            this.name.nativeElement.style.maxHeight = `${this.name.nativeElement.scrollHeight / 16}rem`;\n        }\n    }\n\n    /**\n     * Обработчик события mouseleave.\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler(): void {\n        if (this.isHovered) {\n            this.renderer.setStyle(this.name.nativeElement, 'maxHeight', null);\n        }\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<div class=\"relative group\">\n    <button\n        (click)=\"clickOnCardEvent.emit(category)\"\n        [class.pointer-events-none]=\"!category || isSkeleton\"\n        class=\"category-btn flex flex-col relative rounded-tui-radius-m overflow-hidden shadow-sc-2\"\n    >\n        <div class=\"img-wrapper w-full grow bg-tui-base-02 overflow-hidden\">\n            <img *ngIf=\"category && !isSkeleton\" [src]=\"getCategoryImgURL(category)\" [alt]=\"category.name\" class=\"w-full\" />\n        </div>\n        <div #name [attr.data-is-hovered]=\"!isHovered\" class=\"name grid items-center justify-center font-bold bg-white w-full overflow-hidden shrink-0 duration-300\">\n            <span *ngIf=\"category && !isSkeleton; else skeletonName\">{{ category.name }}</span>\n            <ng-template #skeletonName>\n                <div class=\"skeleton-name bg-tui-base-02 rounded-tui-radius-s\"></div>\n            </ng-template>\n        </div>\n    </button>\n    <sc-favorite-btn\n        *ngIf=\"category && !isSkeleton && (authStatus$ | async)\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"category.isFavorite\"\n        (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n        class=\"absolute left-1 top-1\"\n    ></sc-favorite-btn>\n</div>\n"]}
|
101
|
+
//# 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;IA8ChC;;;;;;OAMG;IACH,YACsC,IAAa,EACG,iBAAyB,EAC1D,GAAsB;QAFL,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,QAAG,GAAH,GAAG,CAAmB;QAjD3C;;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,cAAS,GAAY,KAAK,CAAC;QAErC;;WAEG;QACa,aAAQ,GAAY,MAAM,CAAC,aAAa,CAAC,CAAC;QAE1D;;WAEG;QAEI,qBAAgB,GAA6B,IAAI,YAAY,EAAc,CAAC;QAEnF;;WAEG;QAEI,yBAAoB,GAA6B,IAAI,YAAY,EAAc,CAAC;IAapF,CAAC;IAEJ;;;;OAIG;IACO,SAAS,CAAC,SAAkB;QAClC,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;IAC/B,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;8GApFQ,uBAAuB,kBAsDpB,OAAO,aACP,uBAAuB;kGAvD1B,uBAAuB,kQCfpC,0wDA+CA;;2FDhCa,uBAAuB;kBANnC,SAAS;+BACI,kBAAkB,mBAGX,uBAAuB,CAAC,MAAM;;0BAwD1C,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;yEAlD5B,QAAQ;sBADd,KAAK;gBAQC,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB;gBA2BtB,gBAAgB;sBADtB,MAAM;gBAOA,oBAAoB;sBAD1B,MAAM","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 isHovered: boolean = false;\n\n    /**\n     * Признак того, отображается этот компонент на мобильном устройстве или нет.\n     */\n    public readonly isMobile: boolean = inject(TUI_IS_MOBILE);\n\n    /**\n     * Событие нажатия на карточку категории.\n     */\n    @Output()\n    public clickOnCardEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Событие нажатия на кнопку избранной категории.\n     */\n    @Output()\n    public clickOnFavoriteEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\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 isHovered Признак наведения на карточку.\n     */\n    protected onHovered(isHovered: boolean): void {\n        this.isHovered = isHovered;\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    <button\n        (click)=\"clickOnCardEvent.emit(category)\"\n        (tuiHoveredChange)=\"onHovered($event)\"\n        class=\"category-btn rounded-tui-radius-m border-tui-base-04 flex flex-col overflow-hidden border\"\n    >\n        <div class=\"img-wrapper w-full grow overflow-hidden\">\n            <img\n                *ngIf=\"category && category.properties?.image\"\n                [src]=\"getCategoryImgURL(category)\"\n                [alt]=\"category.name\"\n                class=\"size-full object-cover\"\n            />\n\n            <div\n                *ngIf=\"!category\"\n                class=\"img-wrapper bg-tui-base-02 size-full\"\n            ></div>\n\n            <img\n                *ngIf=\"category && !category.properties?.image\"\n                [src]=\"getCategoryImgURL(category)\"\n                [alt]=\"category.name\"\n                class=\"size-full rounded-xl object-contain p-2\"\n            />\n        </div>\n\n        <div class=\"name flex w-full items-center justify-center\">\n            <tui-line-clamp\n                *ngIf=\"category; else skeletonName\"\n                [content]=\"category.name\"\n                class=\"pointer-events-none !h-auto\"\n                [linesLimit]=\"isHovered || isMobile ? 4 : 2\"\n            />\n        </div>\n        <ng-template #skeletonName>\n            <div class=\"skeleton-name bg-tui-base-02 rounded-tui-radius-s\"></div>\n        </ng-template>\n    </button>\n    <sc-favorite-btn\n        *ngIf=\"category && (authStatus$ | async)\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"category.isFavorite\"\n        (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n        class=\"absolute left-1 top-1\"\n    ></sc-favorite-btn>\n</div>\n"]}
|
@@ -0,0 +1,45 @@
|
|
1
|
+
import { ChangeDetectionStrategy, Component, HostListener, Input, signal } from '@angular/core';
|
2
|
+
import { TuiPagination } from '@taiga-ui/kit';
|
3
|
+
import * as i0 from "@angular/core";
|
4
|
+
/**
|
5
|
+
* Компонент просмотра изображений через событие наведения.
|
6
|
+
*/
|
7
|
+
export class HoverImageCarouselComponent {
|
8
|
+
constructor() {
|
9
|
+
/**
|
10
|
+
* Массив изображений.
|
11
|
+
*/
|
12
|
+
this.images = [];
|
13
|
+
/**
|
14
|
+
* {@link WritableSignal} для отслеживания текущего индекса изображения.
|
15
|
+
*/
|
16
|
+
this.currentIndex = signal(0);
|
17
|
+
}
|
18
|
+
/**
|
19
|
+
* Обработчик события mousemove.
|
20
|
+
*
|
21
|
+
* @param offsetX Отступ курсора мыши по оси X от края {@link HTMLElement}.
|
22
|
+
* @param target Объект {@link HTMLElement} в котором лежат изображения
|
23
|
+
*/
|
24
|
+
onHover(offsetX, target) {
|
25
|
+
if (this.images.length <= 1 || offsetX < 0) {
|
26
|
+
return;
|
27
|
+
}
|
28
|
+
const elementWidth = target.offsetWidth;
|
29
|
+
const hoverPercentage = offsetX / elementWidth;
|
30
|
+
const newIndex = Math.floor(hoverPercentage * this.images.length);
|
31
|
+
this.currentIndex.set(newIndex);
|
32
|
+
}
|
33
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: HoverImageCarouselComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
34
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.8", type: HoverImageCarouselComponent, isStandalone: true, selector: "sc-hover-image-carousel", inputs: { images: "images" }, host: { listeners: { "mousemove.silent": "onHover($event.offsetX,$event.target)" } }, ngImport: i0, template: "<div\n class=\"overflow-hidden rounded-xl\"\n [class.mb-2]=\"images.length <= 1\"\n [style.aspect-ratio]=\"'20/19'\"\n>\n <img\n [src]=\"images[currentIndex()]\"\n class=\"size-full rounded-xl object-contain\"\n />\n</div>\n\n@if (images.length > 1) {\n <tui-pagination\n size=\"s\"\n [length]=\"images.length\"\n [index]=\"currentIndex()\"\n />\n}\n", dependencies: [{ kind: "component", type: TuiPagination, selector: "tui-pagination", inputs: ["length", "focusable", "size", "disabled", "activePadding", "sidePadding", "content", "index"], outputs: ["indexChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
35
|
+
}
|
36
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: HoverImageCarouselComponent, decorators: [{
|
37
|
+
type: Component,
|
38
|
+
args: [{ selector: 'sc-hover-image-carousel', standalone: true, imports: [TuiPagination], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n class=\"overflow-hidden rounded-xl\"\n [class.mb-2]=\"images.length <= 1\"\n [style.aspect-ratio]=\"'20/19'\"\n>\n <img\n [src]=\"images[currentIndex()]\"\n class=\"size-full rounded-xl object-contain\"\n />\n</div>\n\n@if (images.length > 1) {\n <tui-pagination\n size=\"s\"\n [length]=\"images.length\"\n [index]=\"currentIndex()\"\n />\n}\n" }]
|
39
|
+
}], propDecorators: { images: [{
|
40
|
+
type: Input
|
41
|
+
}], onHover: [{
|
42
|
+
type: HostListener,
|
43
|
+
args: ['mousemove.silent', ['$event.offsetX', '$event.target']]
|
44
|
+
}] } });
|
45
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaG92ZXItaW1hZ2UtY2Fyb3VzZWwuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2NhdGFsb2cvaG92ZXItaW1hZ2UtY2Fyb3VzZWwvaG92ZXItaW1hZ2UtY2Fyb3VzZWwuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2NhdGFsb2cvaG92ZXItaW1hZ2UtY2Fyb3VzZWwvaG92ZXItaW1hZ2UtY2Fyb3VzZWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBa0IsTUFBTSxlQUFlLENBQUM7QUFDaEgsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFFOUM7O0dBRUc7QUFRSCxNQUFNLE9BQU8sMkJBQTJCO0lBUHhDO1FBUUk7O1dBRUc7UUFFSSxXQUFNLEdBQWEsRUFBRSxDQUFDO1FBRTdCOztXQUVHO1FBQ0ksaUJBQVksR0FBMkIsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDO0tBbUIzRDtJQWpCRzs7Ozs7T0FLRztJQUVLLE9BQU8sQ0FBQyxPQUFlLEVBQUUsTUFBbUI7UUFDaEQsSUFBSSxJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sSUFBSSxDQUFDLElBQUksT0FBTyxHQUFHLENBQUMsRUFBRSxDQUFDO1lBQ3pDLE9BQU87UUFDWCxDQUFDO1FBRUQsTUFBTSxZQUFZLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQztRQUN4QyxNQUFNLGVBQWUsR0FBRyxPQUFPLEdBQUcsWUFBWSxDQUFDO1FBQy9DLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsZUFBZSxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDbEUsSUFBSSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDcEMsQ0FBQzs4R0E1QlEsMkJBQTJCO2tHQUEzQiwyQkFBMkIsdU1DYnhDLHdaQWtCQSw0Q0RUYyxhQUFhOzsyRkFJZCwyQkFBMkI7a0JBUHZDLFNBQVM7K0JBQ0kseUJBQXlCLGNBQ3ZCLElBQUksV0FDUCxDQUFDLGFBQWEsQ0FBQyxtQkFFUCx1QkFBdUIsQ0FBQyxNQUFNOzhCQU94QyxNQUFNO3NCQURaLEtBQUs7Z0JBZUUsT0FBTztzQkFEZCxZQUFZO3VCQUFDLGtCQUFrQixFQUFFLENBQUMsZ0JBQWdCLEVBQUUsZUFBZSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgSG9zdExpc3RlbmVyLCBJbnB1dCwgc2lnbmFsLCBXcml0YWJsZVNpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgVHVpUGFnaW5hdGlvbiB9IGZyb20gJ0B0YWlnYS11aS9raXQnO1xuXG4vKipcbiAqINCa0L7QvNC/0L7QvdC10L3RgiDQv9GA0L7RgdC80L7RgtGA0LAg0LjQt9C+0LHRgNCw0LbQtdC90LjQuSDRh9C10YDQtdC3INGB0L7QsdGL0YLQuNC1INC90LDQstC10LTQtdC90LjRjy5cbiAqL1xuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICdzYy1ob3Zlci1pbWFnZS1jYXJvdXNlbCcsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbVHVpUGFnaW5hdGlvbl0sXG4gICAgdGVtcGxhdGVVcmw6ICcuL2hvdmVyLWltYWdlLWNhcm91c2VsLmNvbXBvbmVudC5odG1sJyxcbiAgICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgSG92ZXJJbWFnZUNhcm91c2VsQ29tcG9uZW50IHtcbiAgICAvKipcbiAgICAgKiDQnNCw0YHRgdC40LIg0LjQt9C+0LHRgNCw0LbQtdC90LjQuS5cbiAgICAgKi9cbiAgICBASW5wdXQoKVxuICAgIHB1YmxpYyBpbWFnZXM6IHN0cmluZ1tdID0gW107XG5cbiAgICAvKipcbiAgICAgKiB7QGxpbmsgV3JpdGFibGVTaWduYWx9INC00LvRjyDQvtGC0YHQu9C10LbQuNCy0LDQvdC40Y8g0YLQtdC60YPRidC10LPQviDQuNC90LTQtdC60YHQsCDQuNC30L7QsdGA0LDQttC10L3QuNGPLlxuICAgICAqL1xuICAgIHB1YmxpYyBjdXJyZW50SW5kZXg6IFdyaXRhYmxlU2lnbmFsPG51bWJlcj4gPSBzaWduYWwoMCk7XG5cbiAgICAvKipcbiAgICAgKiDQntCx0YDQsNCx0L7RgtGH0LjQuiDRgdC+0LHRi9GC0LjRjyBtb3VzZW1vdmUuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gb2Zmc2V0WCDQntGC0YHRgtGD0L8g0LrRg9GA0YHQvtGA0LAg0LzRi9GI0Lgg0L/QviDQvtGB0LggWCDQvtGCINC60YDQsNGPIHtAbGluayBIVE1MRWxlbWVudH0uXG4gICAgICogQHBhcmFtIHRhcmdldCDQntCx0YrQtdC60YIge0BsaW5rIEhUTUxFbGVtZW50fSDQsiDQutC+0YLQvtGA0L7QvCDQu9C10LbQsNGCINC40LfQvtCx0YDQsNC20LXQvdC40Y9cbiAgICAgKi9cbiAgICBASG9zdExpc3RlbmVyKCdtb3VzZW1vdmUuc2lsZW50JywgWyckZXZlbnQub2Zmc2V0WCcsICckZXZlbnQudGFyZ2V0J10pXG4gICAgcHJpdmF0ZSBvbkhvdmVyKG9mZnNldFg6IG51bWJlciwgdGFyZ2V0OiBIVE1MRWxlbWVudCkge1xuICAgICAgICBpZiAodGhpcy5pbWFnZXMubGVuZ3RoIDw9IDEgfHwgb2Zmc2V0WCA8IDApIHtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGVsZW1lbnRXaWR0aCA9IHRhcmdldC5vZmZzZXRXaWR0aDtcbiAgICAgICAgY29uc3QgaG92ZXJQZXJjZW50YWdlID0gb2Zmc2V0WCAvIGVsZW1lbnRXaWR0aDtcbiAgICAgICAgY29uc3QgbmV3SW5kZXggPSBNYXRoLmZsb29yKGhvdmVyUGVyY2VudGFnZSAqIHRoaXMuaW1hZ2VzLmxlbmd0aCk7XG4gICAgICAgIHRoaXMuY3VycmVudEluZGV4LnNldChuZXdJbmRleCk7XG4gICAgfVxufVxuIiwiPGRpdlxuICAgIGNsYXNzPVwib3ZlcmZsb3ctaGlkZGVuIHJvdW5kZWQteGxcIlxuICAgIFtjbGFzcy5tYi0yXT1cImltYWdlcy5sZW5ndGggPD0gMVwiXG4gICAgW3N0eWxlLmFzcGVjdC1yYXRpb109XCInMjAvMTknXCJcbj5cbiAgICA8aW1nXG4gICAgICAgIFtzcmNdPVwiaW1hZ2VzW2N1cnJlbnRJbmRleCgpXVwiXG4gICAgICAgIGNsYXNzPVwic2l6ZS1mdWxsIHJvdW5kZWQteGwgb2JqZWN0LWNvbnRhaW5cIlxuICAgIC8+XG48L2Rpdj5cblxuQGlmIChpbWFnZXMubGVuZ3RoID4gMSkge1xuICAgIDx0dWktcGFnaW5hdGlvblxuICAgICAgICBzaXplPVwic1wiXG4gICAgICAgIFtsZW5ndGhdPVwiaW1hZ2VzLmxlbmd0aFwiXG4gICAgICAgIFtpbmRleF09XCJjdXJyZW50SW5kZXgoKVwiXG4gICAgLz5cbn1cbiJdfQ==
|