@snabcentr/client-ui 3.51.2 → 3.51.5
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.component.d.ts +10 -13
- package/configurators/sandwich/sc-sandwich.component.d.ts +10 -6
- package/esm2022/banner/sc-banner.component.mjs +23 -29
- package/esm2022/configurators/sandwich/sc-sandwich.component.mjs +30 -24
- package/esm2022/order/draft/sc-draft.component.mjs +3 -3
- package/fesm2022/snabcentr-client-ui.mjs +48 -48
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { EventEmitter, QueryList, TemplateRef } from '@angular/core';
|
|
1
|
+
import { EventEmitter, InputSignal, QueryList, TemplateRef } from '@angular/core';
|
|
2
2
|
import { ScBanner } from '@snabcentr/client-core';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
/**
|
|
@@ -16,7 +16,7 @@ export declare class ScBannerComponent {
|
|
|
16
16
|
/**
|
|
17
17
|
* Местоположение баннера.
|
|
18
18
|
*/
|
|
19
|
-
bannerLocation
|
|
19
|
+
bannerLocation: InputSignal<string | undefined>;
|
|
20
20
|
/**
|
|
21
21
|
* Признак, что компонент должен растягиваться.
|
|
22
22
|
*/
|
|
@@ -52,7 +52,14 @@ export declare class ScBannerComponent {
|
|
|
52
52
|
/**
|
|
53
53
|
* Список баннеров.
|
|
54
54
|
*/
|
|
55
|
+
protected readonly banners$: import("rxjs").Observable<ScBanner[]>;
|
|
55
56
|
protected readonly banners: import("@angular/core").Signal<ScBanner[]>;
|
|
57
|
+
/**
|
|
58
|
+
* Общее количество слайдов (баннеры + контент-проекция).
|
|
59
|
+
* Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
|
|
60
|
+
* tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
|
|
61
|
+
*/
|
|
62
|
+
readonly totalItems: import("@angular/core").Signal<number>;
|
|
56
63
|
/**
|
|
57
64
|
* trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,
|
|
58
65
|
* чтобы карусель не видела 0 слайдов между тиками и не вызывала tuiClamp(…, -1).
|
|
@@ -61,16 +68,6 @@ export declare class ScBannerComponent {
|
|
|
61
68
|
* @param banner Данные о баннере.
|
|
62
69
|
*/
|
|
63
70
|
trackByBannerId(_index: number, banner: ScBanner): number;
|
|
64
|
-
/**
|
|
65
|
-
* Общее количество слайдов (баннеры + контент-проекция).
|
|
66
|
-
* Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
|
|
67
|
-
* tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
|
|
68
|
-
*/
|
|
69
|
-
get totalItems(): number;
|
|
70
|
-
/**
|
|
71
|
-
* Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.
|
|
72
|
-
*/
|
|
73
|
-
private get isHidden();
|
|
74
71
|
/**
|
|
75
72
|
* Обработчик нажатия на баннер.
|
|
76
73
|
*
|
|
@@ -78,5 +75,5 @@ export declare class ScBannerComponent {
|
|
|
78
75
|
*/
|
|
79
76
|
onClick(banner: ScBanner): void;
|
|
80
77
|
static ɵfac: i0.ɵɵFactoryDeclaration<ScBannerComponent, never>;
|
|
81
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<ScBannerComponent, "sc-banner", never, { "navigateButton": { "alias": "navigateButton"; "required": false; }; "duration": { "alias": "duration"; "required": false; }; "bannerLocation": { "alias": "bannerLocation"; "required": false; }; "resizable": { "alias": "resizable"; "required": false; }; }, { "loadBannersEvent": "loadBannersEvent"; "clickBannerEvent": "clickBannerEvent"; }, ["bannersListRef"], never, false, never>;
|
|
78
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScBannerComponent, "sc-banner", never, { "navigateButton": { "alias": "navigateButton"; "required": false; }; "duration": { "alias": "duration"; "required": false; }; "bannerLocation": { "alias": "bannerLocation"; "required": false; "isSignal": true; }; "resizable": { "alias": "resizable"; "required": false; }; }, { "loadBannersEvent": "loadBannersEvent"; "clickBannerEvent": "clickBannerEvent"; }, ["bannersListRef"], never, false, never>;
|
|
82
79
|
}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { OnInit } from '@angular/core';
|
|
2
2
|
import { FormArray, FormControl, FormGroup, ValidatorFn } from '@angular/forms';
|
|
3
|
-
import { ScIConfiguratorCalculateResult, ScOrderItem, ScProduct } from '@snabcentr/client-core';
|
|
3
|
+
import { ScCart, ScIConfiguratorCalculateResult, ScIOrderEditorLoader, ScOrderDraft, ScOrderItem, ScProduct } from '@snabcentr/client-core';
|
|
4
4
|
import { Observable } from 'rxjs';
|
|
5
5
|
import { ScISandwichSettings } from '../models/sandwich/sc-i-sandwich-settings';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
@@ -27,7 +27,11 @@ export declare class ScSandwichComponent implements OnInit {
|
|
|
27
27
|
/**
|
|
28
28
|
* Позиция товара/услуги в корзине.
|
|
29
29
|
*/
|
|
30
|
-
readonly
|
|
30
|
+
readonly orderItem: import("@angular/core").InputSignal<ScOrderItem | undefined>;
|
|
31
|
+
/**
|
|
32
|
+
* Идентификатор заказа.
|
|
33
|
+
*/
|
|
34
|
+
orderId?: number;
|
|
31
35
|
/**
|
|
32
36
|
* Событие переключения отображения товаров.
|
|
33
37
|
*/
|
|
@@ -39,7 +43,7 @@ export declare class ScSandwichComponent implements OnInit {
|
|
|
39
43
|
productCategoryId: FormControl<number | null>;
|
|
40
44
|
configurator: FormControl<string | null>;
|
|
41
45
|
quantity: FormControl<number | null>;
|
|
42
|
-
marker: FormControl<string>;
|
|
46
|
+
marker: FormControl<string | null>;
|
|
43
47
|
items: FormArray<FormGroup<{
|
|
44
48
|
width: FormControl<number>;
|
|
45
49
|
length: FormControl<number>;
|
|
@@ -48,9 +52,9 @@ export declare class ScSandwichComponent implements OnInit {
|
|
|
48
52
|
calculationId: FormControl<string | null>;
|
|
49
53
|
}>;
|
|
50
54
|
/**
|
|
51
|
-
* Сервис
|
|
55
|
+
* Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).
|
|
52
56
|
*/
|
|
53
|
-
|
|
57
|
+
protected readonly orderService: ScIOrderEditorLoader<ScOrderItem, ScCart | ScOrderDraft>;
|
|
54
58
|
/**
|
|
55
59
|
* Сервис конвертации данных.
|
|
56
60
|
*/
|
|
@@ -142,5 +146,5 @@ export declare class ScSandwichComponent implements OnInit {
|
|
|
142
146
|
*/
|
|
143
147
|
private setDefaultFormValue;
|
|
144
148
|
static ɵfac: i0.ɵɵFactoryDeclaration<ScSandwichComponent, never>;
|
|
145
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<ScSandwichComponent, "sc-sandwich", never, { "settings": { "alias": "settings"; "required": false; "isSignal": true; }; "categoryId": { "alias": "categoryId"; "required": true; "isSignal": true; }; "editor": { "alias": "editor"; "required": true; "isSignal": true; }; "
|
|
149
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ScSandwichComponent, "sc-sandwich", never, { "settings": { "alias": "settings"; "required": false; "isSignal": true; }; "categoryId": { "alias": "categoryId"; "required": true; "isSignal": true; }; "editor": { "alias": "editor"; "required": true; "isSignal": true; }; "orderItem": { "alias": "orderItem"; "required": true; "isSignal": true; }; "orderId": { "alias": "orderId"; "required": false; }; }, { "toggleShowEvent": "toggleShowEvent"; }, never, never, true, never>;
|
|
146
150
|
}
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, HostBinding, inject, Input, Output } from '@angular/core';
|
|
2
|
-
import { toSignal } from '@angular/core/rxjs-interop';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, computed, ContentChildren, EventEmitter, HostBinding, inject, Input, input, Output, } from '@angular/core';
|
|
2
|
+
import { toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
3
3
|
import { IntersectionObserverService } from '@ng-web-apis/intersection-observer';
|
|
4
4
|
import { ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';
|
|
5
|
-
import { map, shareReplay, tap } from 'rxjs';
|
|
5
|
+
import { map, shareReplay, switchMap, tap } from 'rxjs';
|
|
6
6
|
import { SC_BANNER_DURATION } from '../providers';
|
|
7
7
|
import * as i0 from "@angular/core";
|
|
8
8
|
import * as i1 from "@angular/common";
|
|
@@ -23,6 +23,10 @@ export class ScBannerComponent {
|
|
|
23
23
|
* Интервал автоматической смены слайдов в миллисекундах (используйте 0, чтобы отключить автоматическую смену слайда).
|
|
24
24
|
*/
|
|
25
25
|
this.duration = inject(SC_BANNER_DURATION);
|
|
26
|
+
/**
|
|
27
|
+
* Местоположение баннера.
|
|
28
|
+
*/
|
|
29
|
+
this.bannerLocation = input();
|
|
26
30
|
/**
|
|
27
31
|
* Признак, что компонент должен растягиваться.
|
|
28
32
|
*/
|
|
@@ -54,14 +58,21 @@ export class ScBannerComponent {
|
|
|
54
58
|
/**
|
|
55
59
|
* Список баннеров.
|
|
56
60
|
*/
|
|
57
|
-
this.banners =
|
|
61
|
+
this.banners$ = toObservable(this.bannerLocation).pipe(switchMap((bannerLocation) => this.bannerService.banners$.pipe(map((banners) => banners.filter((banner) => banner.location === bannerLocation)), tap((banners) => {
|
|
58
62
|
if (banners.length > 0 && !this.resizable) {
|
|
59
63
|
this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;
|
|
60
64
|
}
|
|
61
65
|
this.loadBannersEvent.emit(banners.length);
|
|
62
|
-
}),
|
|
66
|
+
}))),
|
|
63
67
|
// eslint-disable-next-line rxjs/no-sharereplay
|
|
64
|
-
shareReplay(1))
|
|
68
|
+
shareReplay(1));
|
|
69
|
+
this.banners = toSignal(this.banners$, { initialValue: [] });
|
|
70
|
+
/**
|
|
71
|
+
* Общее количество слайдов (баннеры + контент-проекция).
|
|
72
|
+
* Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
|
|
73
|
+
* tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
|
|
74
|
+
*/
|
|
75
|
+
this.totalItems = computed(() => this.banners().length + this.bannersListRef.length);
|
|
65
76
|
}
|
|
66
77
|
/**
|
|
67
78
|
* trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,
|
|
@@ -75,21 +86,6 @@ export class ScBannerComponent {
|
|
|
75
86
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
|
76
87
|
return banner.id ?? _index;
|
|
77
88
|
}
|
|
78
|
-
/**
|
|
79
|
-
* Общее количество слайдов (баннеры + контент-проекция).
|
|
80
|
-
* Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
|
|
81
|
-
* tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
|
|
82
|
-
*/
|
|
83
|
-
get totalItems() {
|
|
84
|
-
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
|
85
|
-
return (this.banners()?.length ?? 0) + (this.bannersListRef?.length ?? 0);
|
|
86
|
-
}
|
|
87
|
-
/**
|
|
88
|
-
* Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.
|
|
89
|
-
*/
|
|
90
|
-
get isHidden() {
|
|
91
|
-
return this.totalItems === 0;
|
|
92
|
-
}
|
|
93
89
|
/**
|
|
94
90
|
* Обработчик нажатия на баннер.
|
|
95
91
|
*
|
|
@@ -107,17 +103,18 @@ export class ScBannerComponent {
|
|
|
107
103
|
}
|
|
108
104
|
}
|
|
109
105
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScBannerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
110
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: "navigateButton", duration: "duration", bannerLocation: "bannerLocation", resizable: "resizable" }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerEvent: "clickBannerEvent" }, host: { attributes: { "ngSkipHydration": "true" }, properties: { "
|
|
106
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: { classPropertyName: "navigateButton", publicName: "navigateButton", isSignal: false, isRequired: false, transformFunction: null }, duration: { classPropertyName: "duration", publicName: "duration", isSignal: false, isRequired: false, transformFunction: null }, bannerLocation: { classPropertyName: "bannerLocation", publicName: "bannerLocation", isSignal: true, isRequired: false, transformFunction: null }, resizable: { classPropertyName: "resizable", publicName: "resizable", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerEvent: "clickBannerEvent" }, host: { attributes: { "ngSkipHydration": "true" }, properties: { "class.!hidden": "totalItems() === 0", "style.aspect-ratio": "this.aspectRatio" } }, providers: [IntersectionObserverService], queries: [{ propertyName: "bannersListRef", predicate: ["banner"] }], ngImport: i0, template: "@let effectiveDuration = totalItems() > 1 ? duration : 0;\n\n<tui-carousel\n [duration]=\"effectiveDuration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n>\n @for (banner of banners(); track trackByBannerId($index, banner)) {\n @switch (banner.mediaType) {\n @case ('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 }\n }\n }\n @for (item of bannersListRef; track $index) {\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 }\n</tui-carousel>\n\n@if (navigateButton && duration && banners() && totalItems() > 1) {\n <div\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}\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: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i3.TuiItem, selector: "[tuiItem]" }, { kind: "component", type: i4.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange", "shift"] }, { kind: "pipe", type: i5.ScMediaImageTransformerPipe, name: "scMediaImageTransformer" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
111
107
|
}
|
|
112
108
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScBannerComponent, decorators: [{
|
|
113
109
|
type: Component,
|
|
114
|
-
args: [{ selector: 'sc-banner', providers: [IntersectionObserverService], host: {
|
|
110
|
+
args: [{ selector: 'sc-banner', providers: [IntersectionObserverService], host: {
|
|
111
|
+
ngSkipHydration: 'true',
|
|
112
|
+
'[class.!hidden]': 'totalItems() === 0',
|
|
113
|
+
}, changeDetection: ChangeDetectionStrategy.OnPush, template: "@let effectiveDuration = totalItems() > 1 ? duration : 0;\n\n<tui-carousel\n [duration]=\"effectiveDuration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n>\n @for (banner of banners(); track trackByBannerId($index, banner)) {\n @switch (banner.mediaType) {\n @case ('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 }\n }\n }\n @for (item of bannersListRef; track $index) {\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 }\n</tui-carousel>\n\n@if (navigateButton && duration && banners() && totalItems() > 1) {\n <div\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}\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
114
|
}], propDecorators: { navigateButton: [{
|
|
116
115
|
type: Input
|
|
117
116
|
}], duration: [{
|
|
118
117
|
type: Input
|
|
119
|
-
}], bannerLocation: [{
|
|
120
|
-
type: Input
|
|
121
118
|
}], resizable: [{
|
|
122
119
|
type: Input
|
|
123
120
|
}], loadBannersEvent: [{
|
|
@@ -130,8 +127,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
130
127
|
}], aspectRatio: [{
|
|
131
128
|
type: HostBinding,
|
|
132
129
|
args: ['style.aspect-ratio']
|
|
133
|
-
}], isHidden: [{
|
|
134
|
-
type: HostBinding,
|
|
135
|
-
args: ['class.!hidden']
|
|
136
130
|
}] } });
|
|
137
|
-
//# 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 { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, HostBinding, inject, Input, Output, QueryList, TemplateRef } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { map, shareReplay, tap } from 'rxjs';\n\nimport { SC_BANNER_DURATION } from '../providers';\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 = inject(SC_BANNER_DURATION);\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     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    protected currentBannerId: number = 0;\n\n    /**\n     * Сервис для работы с данными о баннерах.\n     */\n    private readonly bannerService = inject(ScBannerService);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Список баннеров.\n     */\n    protected readonly banners = toSignal(\n        this.bannerService.banners$.pipe(\n            map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()),\n            tap((banners) => {\n                if (banners.length > 0 && !this.resizable) {\n                    this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                }\n\n                this.loadBannersEvent.emit(banners.length);\n            }),\n            // eslint-disable-next-line rxjs/no-sharereplay\n            shareReplay(1)\n        ),\n        { initialValue: [] as ScBanner[] }\n    );\n\n    /**\n     * trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,\n     * чтобы карусель не видела 0 слайдов между тиками и не вызывала tuiClamp(…, -1).\n     *\n     * @param _index Индекс.\n     * @param banner Данные о баннере.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public trackByBannerId(_index: number, banner: ScBanner): number {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        return banner.id ?? _index;\n    }\n\n    /**\n     * Общее количество слайдов (баннеры + контент-проекция).\n     * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать\n     * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.\n     */\n    public get totalItems(): number {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        return (this.banners()?.length ?? 0) + (this.bannersListRef?.length ?? 0);\n    }\n\n    /**\n     * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.\n     */\n    @HostBinding('class.!hidden')\n    private get isHidden(): boolean {\n        return this.totalItems === 0;\n    }\n\n    /**\n     * Обработчик нажатия на баннер.\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","@let effectiveDuration = totalItems > 1 ? duration : 0;\n<tui-carousel\n    [duration]=\"effectiveDuration\"\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; trackBy: trackByBannerId\">\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 && banners() && totalItems > 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"]}
|
|
131
|
+
//# 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    Component,\n    computed,\n    ContentChildren,\n    EventEmitter,\n    HostBinding,\n    inject,\n    Input,\n    input,\n    InputSignal,\n    Output,\n    QueryList,\n    TemplateRef,\n} from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { map, shareReplay, switchMap, tap } from 'rxjs';\n\nimport { SC_BANNER_DURATION } from '../providers';\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: {\n        ngSkipHydration: 'true',\n        '[class.!hidden]': 'totalItems() === 0',\n    },\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 = inject(SC_BANNER_DURATION);\n\n    /**\n     * Местоположение баннера.\n     */\n    public bannerLocation: InputSignal<string | undefined> = input<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     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    protected currentBannerId: number = 0;\n\n    /**\n     * Сервис для работы с данными о баннерах.\n     */\n    private readonly bannerService = inject(ScBannerService);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Список баннеров.\n     */\n    protected readonly banners$ = toObservable(this.bannerLocation).pipe(\n        switchMap((bannerLocation) =>\n            this.bannerService.banners$.pipe(\n                map((banners) => banners.filter((banner) => banner.location === bannerLocation)),\n                tap((banners) => {\n                    if (banners.length > 0 && !this.resizable) {\n                        this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                    }\n\n                    this.loadBannersEvent.emit(banners.length);\n                })\n            )\n        ),\n        // eslint-disable-next-line rxjs/no-sharereplay\n        shareReplay(1)\n    );\n\n    protected readonly banners = toSignal(this.banners$, { initialValue: [] as ScBanner[] });\n\n    /**\n     * Общее количество слайдов (баннеры + контент-проекция).\n     * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать\n     * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.\n     */\n    public readonly totalItems = computed(() => this.banners().length + this.bannersListRef.length);\n\n    /**\n     * trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,\n     * чтобы карусель не видела 0 слайдов между тиками и не вызывала tuiClamp(…, -1).\n     *\n     * @param _index Индекс.\n     * @param banner Данные о баннере.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public trackByBannerId(_index: number, banner: ScBanner): number {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        return banner.id ?? _index;\n    }\n\n    /**\n     * Обработчик нажатия на баннер.\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","@let effectiveDuration = totalItems() > 1 ? duration : 0;\n\n<tui-carousel\n    [duration]=\"effectiveDuration\"\n    #carousel\n    [attr.resizable]=\"resizable\"\n    class=\"size-full overflow-hidden rounded-xl bg-white\"\n    [(index)]=\"currentBannerId\"\n>\n    @for (banner of banners(); track trackByBannerId($index, banner)) {\n        @switch (banner.mediaType) {\n            @case ('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            }\n        }\n    }\n    @for (item of bannersListRef; track $index) {\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    }\n</tui-carousel>\n\n@if (navigateButton && duration && banners() && totalItems() > 1) {\n    <div\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}\n"]}
|