@snabcentr/client-ui 0.2.1 → 0.2.3

Sign up to get free protection for your applications and to get access to all the features.
@@ -1,18 +1,18 @@
1
- import { ChangeDetectorRef, OnInit, QueryList, TemplateRef, EventEmitter } from '@angular/core';
2
- import { ScBannerService, ScBanner } from '@snabcentr/client-core';
1
+ import { AfterViewInit, EventEmitter, QueryList, TemplateRef, ElementRef, ChangeDetectorRef } from '@angular/core';
2
+ import { ScBanner, ScBannerService } from '@snabcentr/client-core';
3
+ import { Observable, Subject } from 'rxjs';
3
4
  import { ScPxConverter } from '../helpers';
5
+ import { IntersectionObserverService } from '@ng-web-apis/intersection-observer';
4
6
  import * as i0 from "@angular/core";
5
7
  /**
6
8
  * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.
7
9
  */
8
- export declare class ScBannerComponent implements OnInit {
10
+ export declare class ScBannerComponent implements AfterViewInit {
9
11
  private readonly cdr;
10
- private bannerService;
12
+ private readonly bannerService;
13
+ private readonly entries$;
14
+ private readonly element;
11
15
  private pxConverter;
12
- /**
13
- * Список ссылок на элемент представлений шаблонов.
14
- */
15
- bannersListRef: QueryList<TemplateRef<any>>;
16
16
  /**
17
17
  * Признак, что необходимо показывать кнопки навигации.
18
18
  */
@@ -40,35 +40,51 @@ export declare class ScBannerComponent implements OnInit {
40
40
  /**
41
41
  * Событие нажатия на изображение баннера.
42
42
  */
43
- clickBannerImgEvent: EventEmitter<ScBanner>;
43
+ clickBannerEvent: EventEmitter<ScBanner>;
44
+ /**
45
+ * Список ссылок на элемент представлений шаблонов.
46
+ */
47
+ bannersListRef: QueryList<TemplateRef<any>>;
48
+ /**
49
+ * Ссылка на {@link HTMLVideoElement} элемента отображения видео.
50
+ */
51
+ private videoRef;
44
52
  /**
45
53
  * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.
46
54
  */
47
- get isHidden(): boolean;
55
+ private get isHidden();
48
56
  /**
49
- * Свойство, от которого зависит высота `:host` компонента.
57
+ * Идентификатор текущего баннера.
50
58
  */
51
- height: string;
59
+ currentBannerId: number;
52
60
  /**
53
- * Свойство, от которого зависит ширина `:host` компонента.
61
+ * {@link Subject} изменения состояния таймера.
54
62
  */
55
- width: string;
63
+ readonly toggleTimer$: Subject<boolean>;
56
64
  /**
57
- * Свойство, от которого зависит соотношение `:host` компонента.
65
+ * Признак, что необходимо показывать кнопки старта видео баннера.
58
66
  */
59
- aspectRatio: string;
67
+ showPlayBtn: boolean;
60
68
  /**
61
- * Идентификатор текущего баннера.
69
+ * {@link Observable} Обновления списка баннеров.
62
70
  */
63
- currentBannerId: number;
71
+ readonly banners$: Observable<ScBanner[]>;
64
72
  /**
65
73
  * Список баннеров.
66
74
  */
67
- banners: ScBanner[];
75
+ banners?: ScBanner[];
76
+ /**
77
+ * Свойство, от которого зависит высота `:host` компонента.
78
+ */
79
+ height: string;
80
+ /**
81
+ * Свойство, от которого зависит ширина `:host` компонента.
82
+ */
83
+ width: string;
68
84
  /**
69
- * Признак, что указатель мыши находится над баннером.
85
+ * Свойство, от которого зависит соотношение `:host` компонента.
70
86
  */
71
- private isMouseOver;
87
+ aspectRatio: string;
72
88
  /**
73
89
  * Инициализирует экземпляр класса {@link ScBannerComponent}.
74
90
  *
@@ -76,31 +92,39 @@ export declare class ScBannerComponent implements OnInit {
76
92
  * @param bannerService Сервис для работы с баннерами.
77
93
  * @param pxConverter Класс хэлпер для конвертации пикселей.
78
94
  */
79
- constructor(cdr: ChangeDetectorRef, bannerService: ScBannerService, pxConverter: ScPxConverter);
95
+ constructor(cdr: ChangeDetectorRef, bannerService: ScBannerService, entries$: IntersectionObserverService, element: ElementRef<Element>, pxConverter: ScPxConverter);
80
96
  /** @inheritDoc */
81
- ngOnInit(): void;
97
+ ngAfterViewInit(): void;
82
98
  /**
83
- * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.
84
- *
85
- * @param banner Баннер, по ссылке которого совершён переход.
86
- */
87
- onClick(banner: ScBanner): void;
88
- /**
89
- * Переключиться на предыдущий баннер.
99
+ * Переключает на предыдущий баннер.
90
100
  */
91
101
  onPreviousBanner(): void;
92
102
  /**
93
- * Переключиться на следующий баннер.
103
+ * Переключает на следующий баннер.
94
104
  */
95
105
  onNextBanner(): void;
106
+ /**
107
+ * Переключает на следующий баннер.
108
+ */
109
+ onEndedVideo(): void;
110
+ /**
111
+ * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.
112
+ *
113
+ * @param banner Баннер, по ссылке которого совершён переход.
114
+ */
115
+ onClick(banner: ScBanner): void;
96
116
  /**
97
117
  * Обработчик события mouseenter.
118
+ *
119
+ * @private
98
120
  */
99
121
  private mouseEnterHandler;
100
122
  /**
101
123
  * Обработчик события mouseleave.
124
+ *
125
+ * @private
102
126
  */
103
127
  private mouseLeaveHandler;
104
128
  static ɵfac: i0.ɵɵFactoryDeclaration<ScBannerComponent, never>;
105
- static ɵcmp: i0.ɵɵComponentDeclaration<ScBannerComponent, "sc-banner", never, { "navigateButton": "navigateButton"; "playerInterval": "playerInterval"; "disabled": "disabled"; "bannerLocation": "bannerLocation"; "resizable": "resizable"; }, { "loadBannersEvent": "loadBannersEvent"; "clickBannerImgEvent": "clickBannerImgEvent"; }, ["bannersListRef"], never, false>;
129
+ static ɵcmp: i0.ɵɵComponentDeclaration<ScBannerComponent, "sc-banner", never, { "navigateButton": "navigateButton"; "playerInterval": "playerInterval"; "disabled": "disabled"; "bannerLocation": "bannerLocation"; "resizable": "resizable"; }, { "loadBannersEvent": "loadBannersEvent"; "clickBannerEvent": "clickBannerEvent"; }, ["bannersListRef"], never, false>;
106
130
  }
@@ -5,11 +5,13 @@ import * as i3 from "@angular/router";
5
5
  import * as i4 from "@taiga-ui/core";
6
6
  import * as i5 from "@angular/material/core";
7
7
  import * as i6 from "@taiga-ui/kit";
8
+ import * as i7 from "@ng-web-apis/intersection-observer";
9
+ import * as i8 from "@taiga-ui/cdk";
8
10
  /**
9
11
  * Модуль баннеров.
10
12
  */
11
13
  export declare class ScBannerModule {
12
14
  static ɵfac: i0.ɵɵFactoryDeclaration<ScBannerModule, never>;
13
- static ɵmod: i0.ɵɵNgModuleDeclaration<ScBannerModule, [typeof i1.ScBannerComponent], [typeof i2.CommonModule, typeof i3.RouterModule, typeof i4.TuiButtonModule, typeof i5.MatRippleModule, typeof i6.TuiCarouselModule, typeof i6.TuiPaginationModule, typeof i4.TuiModeModule], [typeof i1.ScBannerComponent]>;
15
+ static ɵmod: i0.ɵɵNgModuleDeclaration<ScBannerModule, [typeof i1.ScBannerComponent], [typeof i2.CommonModule, typeof i3.RouterModule, typeof i4.TuiButtonModule, typeof i5.MatRippleModule, typeof i6.TuiCarouselModule, typeof i4.TuiModeModule, typeof i4.TuiLoaderModule, typeof i7.IntersectionObserverModule, typeof i8.TuiLetModule], [typeof i1.ScBannerComponent]>;
14
16
  static ɵinj: i0.ɵɵInjectorDeclaration<ScBannerModule>;
15
17
  }
@@ -1,14 +1,18 @@
1
1
  import { __decorate } from "tslib";
2
- import { ChangeDetectionStrategy, Component, HostListener, Input, ContentChildren, HostBinding, Output, EventEmitter, } from '@angular/core';
2
+ import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter, ContentChildren, ViewChild, ElementRef, HostBinding, Inject, HostListener, } from '@angular/core';
3
+ import { map, tap, shareReplay, filter, switchMap, interval, takeUntil, Subject } from 'rxjs';
3
4
  import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
4
- import { filter, interval, map } from 'rxjs';
5
+ import { IntersectionObserverService } from '@ng-web-apis/intersection-observer';
6
+ import { tuiFadeIn } from '@taiga-ui/core';
5
7
  import * as i0 from "@angular/core";
6
8
  import * as i1 from "@snabcentr/client-core";
7
9
  import * as i2 from "../helpers";
8
10
  import * as i3 from "@angular/common";
9
11
  import * as i4 from "@taiga-ui/core";
10
- import * as i5 from "@taiga-ui/kit";
11
- import * as i6 from "@taiga-ui/cdk";
12
+ import * as i5 from "@angular/material/core";
13
+ import * as i6 from "@taiga-ui/kit";
14
+ import * as i7 from "@taiga-ui/cdk";
15
+ import * as i8 from "@ng-web-apis/intersection-observer";
12
16
  /**
13
17
  * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.
14
18
  */
@@ -20,9 +24,11 @@ let ScBannerComponent = class ScBannerComponent {
20
24
  * @param bannerService Сервис для работы с баннерами.
21
25
  * @param pxConverter Класс хэлпер для конвертации пикселей.
22
26
  */
23
- constructor(cdr, bannerService, pxConverter) {
27
+ constructor(cdr, bannerService, entries$, element, pxConverter) {
24
28
  this.cdr = cdr;
25
29
  this.bannerService = bannerService;
30
+ this.entries$ = entries$;
31
+ this.element = element;
26
32
  this.pxConverter = pxConverter;
27
33
  /**
28
34
  * Признак, что необходимо показывать кнопки навигации.
@@ -31,7 +37,7 @@ let ScBannerComponent = class ScBannerComponent {
31
37
  /**
32
38
  * Интервал показа слайдов в миллисекундах.
33
39
  */
34
- this.playerInterval = 8000;
40
+ this.playerInterval = 5000;
35
41
  /**
36
42
  * Признак, что прокрутка выключена.
37
43
  */
@@ -47,111 +53,134 @@ let ScBannerComponent = class ScBannerComponent {
47
53
  /**
48
54
  * Событие нажатия на изображение баннера.
49
55
  */
50
- this.clickBannerImgEvent = new EventEmitter();
56
+ this.clickBannerEvent = new EventEmitter();
51
57
  /**
52
- * Свойство, от которого зависит высота `:host` компонента.
58
+ * Идентификатор текущего баннера.
53
59
  */
54
- this.height = '';
60
+ this.currentBannerId = 0;
55
61
  /**
56
- * Свойство, от которого зависит ширина `:host` компонента.
62
+ * {@link Subject} изменения состояния таймера.
57
63
  */
58
- this.width = '';
64
+ this.toggleTimer$ = new Subject();
59
65
  /**
60
- * Свойство, от которого зависит соотношение `:host` компонента.
66
+ * Признак, что необходимо показывать кнопки старта видео баннера.
61
67
  */
62
- this.aspectRatio = '';
68
+ this.showPlayBtn = false;
63
69
  /**
64
- * Идентификатор текущего баннера.
70
+ * {@link Observable} Обновления списка баннеров.
65
71
  */
66
- this.currentBannerId = 0;
72
+ this.banners$ = this.bannerService.banners$.pipe(map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()), tap((banners) => {
73
+ if (banners.length) {
74
+ if (this.resizable) {
75
+ this.width = `100%`;
76
+ }
77
+ else {
78
+ this.height = `${this.pxConverter.pxToRem(banners[0].height)}rem`;
79
+ this.width = `${this.pxConverter.pxToRem(banners[0].width)}rem`;
80
+ }
81
+ this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;
82
+ this.banners = banners;
83
+ this.toggleTimer$.next(true);
84
+ }
85
+ this.loadBannersEvent.emit(banners.length);
86
+ }), shareReplay(1));
67
87
  /**
68
88
  * Список баннеров.
69
89
  */
70
90
  this.banners = [];
71
91
  /**
72
- * Признак, что указатель мыши находится над баннером.
92
+ * Свойство, от которого зависит высота `:host` компонента.
73
93
  */
74
- this.isMouseOver = false;
94
+ this.height = 'auto';
95
+ /**
96
+ * Свойство, от которого зависит ширина `:host` компонента.
97
+ */
98
+ this.width = '';
99
+ /**
100
+ * Свойство, от которого зависит соотношение `:host` компонента.
101
+ */
102
+ this.aspectRatio = '';
75
103
  /**
76
104
  * Обработчик события mouseenter.
105
+ *
106
+ * @private
77
107
  */
78
- this.mouseEnterHandler = () => (this.isMouseOver = true);
108
+ this.mouseEnterHandler = () => this.toggleTimer$.next(false);
79
109
  /**
80
110
  * Обработчик события mouseleave.
111
+ *
112
+ * @private
81
113
  */
82
- this.mouseLeaveHandler = () => (this.isMouseOver = false);
114
+ this.mouseLeaveHandler = () => this.toggleTimer$.next(true);
83
115
  }
84
116
  /**
85
117
  * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.
86
118
  */
87
119
  get isHidden() {
88
- return this.bannersListRef.length + this.banners.length === 0;
120
+ return !this.banners || this.bannersListRef.length + this.banners.length === 0;
89
121
  }
90
122
  /** @inheritDoc */
91
- ngOnInit() {
92
- interval(this.playerInterval)
93
- .pipe(filter(() => !this.disabled && !this.isMouseOver), untilDestroyed(this))
123
+ ngAfterViewInit() {
124
+ this.toggleTimer$
125
+ .pipe(filter((toggle) => toggle), switchMap(() => interval(this.playerInterval).pipe(takeUntil(this.toggleTimer$))), filter(() => !this.disabled && (this.banners?.[this.currentBannerId]?.mediaType === 'image' || this.showPlayBtn)), untilDestroyed(this))
94
126
  .subscribe(() => this.onNextBanner());
95
- if (this.bannerLocation) {
96
- this.bannerService.banners$
97
- .pipe(map((banners) => banners.filter((banner) => banner.location === this.bannerLocation)), untilDestroyed(this))
98
- .subscribe((banners) => {
99
- this.banners = banners;
100
- if (banners.length) {
101
- if (this.resizable) {
102
- this.width = `100%`;
103
- }
104
- else {
105
- this.height = `${this.pxConverter.pxToRem(banners[0].height)}rem`;
106
- this.width = `${this.pxConverter.pxToRem(banners[0].width)}rem`;
107
- }
108
- this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;
109
- }
110
- this.loadBannersEvent.emit(this.banners.length);
111
- this.cdr.markForCheck();
112
- });
113
- }
127
+ // Отслеживание пересечения компонента с экраном пользователя.
128
+ // Если баннера не находится в поле видимости пользователя, то он перестаёт переключатся, а видео останавливается.
129
+ this.entries$.pipe(map((entries) => entries.find((item) => item.target === this.element.nativeElement))).subscribe((entry) => {
130
+ this.toggleTimer$.next(!!entry?.isIntersecting);
131
+ });
114
132
  }
115
133
  /**
116
- * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.
117
- *
118
- * @param banner Баннер, по ссылке которого совершён переход.
119
- */
120
- onClick(banner) {
121
- if (banner.url) {
122
- this.clickBannerImgEvent.emit(banner);
123
- }
124
- }
125
- /**
126
- * Переключиться на предыдущий баннер.
134
+ * Переключает на предыдущий баннер.
127
135
  */
128
136
  onPreviousBanner() {
129
137
  const previous = this.currentBannerId - 1;
130
- this.currentBannerId = previous < 0 ? this.bannersListRef.length + this.banners.length - 1 : previous;
138
+ this.currentBannerId = previous < 0 ? this.bannersListRef.length + (this.banners?.length ?? 0) - 1 : previous;
131
139
  this.cdr.markForCheck();
132
140
  }
133
141
  /**
134
- * Переключиться на следующий баннер.
142
+ * Переключает на следующий баннер.
135
143
  */
136
144
  onNextBanner() {
137
145
  const next = this.currentBannerId + 1;
138
- this.currentBannerId = next === this.bannersListRef.length + this.banners.length ? 0 : next;
146
+ this.currentBannerId = next === this.bannersListRef.length + (this.banners?.length ?? 0) ? 0 : next;
139
147
  this.cdr.markForCheck();
140
148
  }
149
+ /**
150
+ * Переключает на следующий баннер.
151
+ */
152
+ onEndedVideo() {
153
+ this.onNextBanner();
154
+ this.toggleTimer$.next(true);
155
+ this.showPlayBtn = true;
156
+ }
157
+ /**
158
+ * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.
159
+ *
160
+ * @param banner Баннер, по ссылке которого совершён переход.
161
+ */
162
+ onClick(banner) {
163
+ if (banner.url) {
164
+ this.clickBannerEvent.emit(banner);
165
+ }
166
+ }
141
167
  };
142
- ScBannerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.ScBannerService }, { token: i2.ScPxConverter }], target: i0.ɵɵFactoryTarget.Component });
143
- ScBannerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: "navigateButton", playerInterval: "playerInterval", disabled: "disabled", bannerLocation: "bannerLocation", resizable: "resizable" }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerImgEvent: "clickBannerImgEvent" }, host: { listeners: { "mouseenter": "mouseEnterHandler()", "mouseleave": "mouseLeaveHandler()" }, properties: { "class.!hidden": "this.isHidden", "style.height": "this.height", "style.width": "this.width", "style.aspect-ratio": "this.aspectRatio" } }, queries: [{ propertyName: "bannersListRef", predicate: ["banner"] }], ngImport: i0, template: "<div class=\"flex relative\" [style.aspect-ratio]=\"aspectRatio\">\n <tui-carousel class=\"bg-white w-full shadow-sc-2 !absolute rounded overflow-hidden\" [(index)]=\"currentBannerId\">\n <ng-container *ngFor=\"let item of bannersListRef; index as index\">\n <div *tuiItem>\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n <ng-container *ngFor=\"let banner of banners; index as index\">\n <a *tuiItem (click)=\"onClick(banner)\" [attr.href]=\"banner.url ? banner.url : null\" target=\"_blank\" [title]=\"banner.title\" class=\"\">\n <img [src]=\"banner.mediaFile\" alt=\"\u0411\u0430\u043D\u043D\u0435\u0440\" [style.aspect-ratio]=\"aspectRatio\" class=\"pointer-events-none object-cover h-full w-full\" />\n </a>\n </ng-container>\n </tui-carousel>\n <div *ngIf=\"navigateButton && !disabled && (this.bannersListRef.length + this.banners.length) > 1\" class=\"w-full h-full flex items-center grow justify-between p-3\">\n <button tuiIconButton icon=\"tuiIconChevronLeftLarge\" tuiMode=\"onLight\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onPreviousBanner()\" class=\"arrow-btn\"></button>\n <button tuiIconButton icon=\"tuiIconChevronRightLarge\" tuiMode=\"onLight\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onNextBanner()\" class=\"arrow-btn\"></button>\n </div>\n</div>\n<tui-pagination\n *ngIf=\"navigateButton && !disabled && (this.bannersListRef.length + this.banners.length) > 1\"\n size=\"s\"\n class=\"mt-3\"\n [length]=\"this.bannersListRef.length + this.banners.length\"\n [(index)]=\"currentBannerId\"\n></tui-pagination>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;flex-direction:column}:host tui-pagination{color:var(--tui-primary)}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i4.TuiButtonComponent, selector: "button[tuiButton], button[tuiIconButton], a[tuiButton], a[tuiIconButton]", inputs: ["appearance", "disabled", "icon", "iconRight", "shape", "showLoader", "size"] }, { kind: "component", type: i5.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange"] }, { kind: "directive", type: i5.TuiCarouselDirective, selector: "tui-carousel", inputs: ["duration", "index"] }, { kind: "directive", type: i6.TuiItemDirective, selector: "[tuiItem]" }, { kind: "component", type: i5.TuiPaginationComponent, selector: "tui-pagination", inputs: ["length", "size", "disabled", "activePadding", "sidePadding", "content", "index"], outputs: ["indexChange"] }, { kind: "directive", type: i4.TuiModeDirective, selector: "[tuiMode]", inputs: ["tuiMode"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
168
+ ScBannerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.ScBannerService }, { token: IntersectionObserverService }, { token: ElementRef }, { token: i2.ScPxConverter }], target: i0.ɵɵFactoryTarget.Component });
169
+ ScBannerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: "navigateButton", playerInterval: "playerInterval", disabled: "disabled", bannerLocation: "bannerLocation", resizable: "resizable" }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerEvent: "clickBannerEvent" }, host: { listeners: { "mouseenter": "mouseEnterHandler()", "mouseleave": "mouseLeaveHandler()" }, properties: { "class.!hidden": "this.isHidden", "style.height": "this.height", "style.width": "this.width", "style.aspect-ratio": "this.aspectRatio" } }, providers: [IntersectionObserverService], queries: [{ propertyName: "bannersListRef", predicate: ["banner"] }], viewQueries: [{ propertyName: "videoRef", first: true, predicate: ["videoPlayer"], descendants: true }], ngImport: i0, template: "<ng-container *tuiLet=\"banners$ | async\">\n <tui-carousel class=\"bg-white w-full h-full shadow-sc-2 rounded-xl overflow-hidden\" [(index)]=\"currentBannerId\">\n <ng-container *ngFor=\"let banner of banners; let index = index\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n target=\"_blank\"\n [title]=\"banner.title\"\n class=\"relative\"\n >\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'video'\">\n <video\n #videoPlayer\n (suspend)=\"showPlayBtn = true\"\n (play)=\"showPlayBtn = false\"\n [src]=\"banner.mediaFile\"\n (ended)=\"onEndedVideo()\"\n (mouseover)=\"videoPlayer.pause()\"\n (mouseout)=\"videoPlayer.play()\"\n class=\"object-cover h-full\"\n muted\n autoplay\n ></video>\n <button\n *ngIf=\"showPlayBtn\"\n tuiIconButton\n [@tuiFadeIn]=\"200\"\n matRipple\n (click)=\"$event.preventDefault(); videoPlayer.play()\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"!absolute left-8 bottom-4\"\n >\n <i class=\"icon-refresh text-black\"></i>\n </button>\n </ng-container>\n\n <img *ngSwitchCase=\"'image'\" [src]=\"banner.mediaFile\" alt=\"\u0411\u0430\u043D\u043D\u0435\u0440\" class=\"object-cover h-full\" />\n </ng-container>\n </a>\n </ng-container>\n <ng-container *ngFor=\"let item of bannersListRef\">\n <div *tuiItem [style.height]=\"height\" [style.width]=\"width\" [style.aspect-ratio]=\"aspectRatio\" class=\"overflow-hidden\">\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n </tui-carousel>\n</ng-container>\n<div *ngIf=\"navigateButton && !disabled && this.banners && (this.bannersListRef.length + this.banners.length) > 1\" tuiMode=\"onLight\" class=\"flex items-center\">\n <button tuiIconButton icon=\"tuiIconChevronLeftLarge\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onPreviousBanner()\" class=\"!absolute left-2\"></button>\n <button tuiIconButton icon=\"tuiIconChevronRightLarge\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onNextBanner()\" class=\"!absolute right-2\"></button>\n</div>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i3.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i3.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i4.TuiButtonComponent, selector: "button[tuiButton], button[tuiIconButton], a[tuiButton], a[tuiIconButton]", inputs: ["appearance", "disabled", "icon", "iconRight", "shape", "showLoader", "size"] }, { kind: "directive", type: i5.MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }, { kind: "component", type: i6.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange"] }, { kind: "directive", type: i6.TuiCarouselDirective, selector: "tui-carousel", inputs: ["duration", "index"] }, { kind: "directive", type: i7.TuiItemDirective, selector: "[tuiItem]" }, { kind: "directive", type: i4.TuiModeDirective, selector: "[tuiMode]", inputs: ["tuiMode"] }, { kind: "directive", type: i7.TuiLetDirective, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }], animations: [tuiFadeIn], changeDetection: i0.ChangeDetectionStrategy.OnPush });
144
170
  ScBannerComponent = __decorate([
145
171
  UntilDestroy({ checkProperties: true })
146
172
  ], ScBannerComponent);
147
173
  export { ScBannerComponent };
148
174
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerComponent, decorators: [{
149
175
  type: Component,
150
- args: [{ selector: 'sc-banner', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex relative\" [style.aspect-ratio]=\"aspectRatio\">\n <tui-carousel class=\"bg-white w-full shadow-sc-2 !absolute rounded overflow-hidden\" [(index)]=\"currentBannerId\">\n <ng-container *ngFor=\"let item of bannersListRef; index as index\">\n <div *tuiItem>\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n <ng-container *ngFor=\"let banner of banners; index as index\">\n <a *tuiItem (click)=\"onClick(banner)\" [attr.href]=\"banner.url ? banner.url : null\" target=\"_blank\" [title]=\"banner.title\" class=\"\">\n <img [src]=\"banner.mediaFile\" alt=\"\u0411\u0430\u043D\u043D\u0435\u0440\" [style.aspect-ratio]=\"aspectRatio\" class=\"pointer-events-none object-cover h-full w-full\" />\n </a>\n </ng-container>\n </tui-carousel>\n <div *ngIf=\"navigateButton && !disabled && (this.bannersListRef.length + this.banners.length) > 1\" class=\"w-full h-full flex items-center grow justify-between p-3\">\n <button tuiIconButton icon=\"tuiIconChevronLeftLarge\" tuiMode=\"onLight\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onPreviousBanner()\" class=\"arrow-btn\"></button>\n <button tuiIconButton icon=\"tuiIconChevronRightLarge\" tuiMode=\"onLight\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onNextBanner()\" class=\"arrow-btn\"></button>\n </div>\n</div>\n<tui-pagination\n *ngIf=\"navigateButton && !disabled && (this.bannersListRef.length + this.banners.length) > 1\"\n size=\"s\"\n class=\"mt-3\"\n [length]=\"this.bannersListRef.length + this.banners.length\"\n [(index)]=\"currentBannerId\"\n></tui-pagination>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;flex-direction:column}:host tui-pagination{color:var(--tui-primary)}\n"] }]
151
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.ScBannerService }, { type: i2.ScPxConverter }]; }, propDecorators: { bannersListRef: [{
152
- type: ContentChildren,
153
- args: ['banner']
154
- }], navigateButton: [{
176
+ args: [{ selector: 'sc-banner', changeDetection: ChangeDetectionStrategy.OnPush, providers: [IntersectionObserverService], animations: [tuiFadeIn], template: "<ng-container *tuiLet=\"banners$ | async\">\n <tui-carousel class=\"bg-white w-full h-full shadow-sc-2 rounded-xl overflow-hidden\" [(index)]=\"currentBannerId\">\n <ng-container *ngFor=\"let banner of banners; let index = index\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n target=\"_blank\"\n [title]=\"banner.title\"\n class=\"relative\"\n >\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'video'\">\n <video\n #videoPlayer\n (suspend)=\"showPlayBtn = true\"\n (play)=\"showPlayBtn = false\"\n [src]=\"banner.mediaFile\"\n (ended)=\"onEndedVideo()\"\n (mouseover)=\"videoPlayer.pause()\"\n (mouseout)=\"videoPlayer.play()\"\n class=\"object-cover h-full\"\n muted\n autoplay\n ></video>\n <button\n *ngIf=\"showPlayBtn\"\n tuiIconButton\n [@tuiFadeIn]=\"200\"\n matRipple\n (click)=\"$event.preventDefault(); videoPlayer.play()\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"!absolute left-8 bottom-4\"\n >\n <i class=\"icon-refresh text-black\"></i>\n </button>\n </ng-container>\n\n <img *ngSwitchCase=\"'image'\" [src]=\"banner.mediaFile\" alt=\"\u0411\u0430\u043D\u043D\u0435\u0440\" class=\"object-cover h-full\" />\n </ng-container>\n </a>\n </ng-container>\n <ng-container *ngFor=\"let item of bannersListRef\">\n <div *tuiItem [style.height]=\"height\" [style.width]=\"width\" [style.aspect-ratio]=\"aspectRatio\" class=\"overflow-hidden\">\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n </tui-carousel>\n</ng-container>\n<div *ngIf=\"navigateButton && !disabled && this.banners && (this.bannersListRef.length + this.banners.length) > 1\" tuiMode=\"onLight\" class=\"flex items-center\">\n <button tuiIconButton icon=\"tuiIconChevronLeftLarge\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onPreviousBanner()\" class=\"!absolute left-2\"></button>\n <button tuiIconButton icon=\"tuiIconChevronRightLarge\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onNextBanner()\" class=\"!absolute right-2\"></button>\n</div>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}\n"] }]
177
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.ScBannerService }, { type: i8.IntersectionObserverService, decorators: [{
178
+ type: Inject,
179
+ args: [IntersectionObserverService]
180
+ }] }, { type: i0.ElementRef, decorators: [{
181
+ type: Inject,
182
+ args: [ElementRef]
183
+ }] }, { type: i2.ScPxConverter }]; }, propDecorators: { navigateButton: [{
155
184
  type: Input
156
185
  }], playerInterval: [{
157
186
  type: Input
@@ -163,8 +192,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
163
192
  type: Input
164
193
  }], loadBannersEvent: [{
165
194
  type: Output
166
- }], clickBannerImgEvent: [{
195
+ }], clickBannerEvent: [{
167
196
  type: Output
197
+ }], bannersListRef: [{
198
+ type: ContentChildren,
199
+ args: ['banner']
200
+ }], videoRef: [{
201
+ type: ViewChild,
202
+ args: ['videoPlayer']
168
203
  }], isHidden: [{
169
204
  type: HostBinding,
170
205
  args: ['class.!hidden']
@@ -184,4 +219,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
184
219
  type: HostListener,
185
220
  args: ['mouseleave']
186
221
  }] } });
187
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-banner.component.js","sourceRoot":"","sources":["../../../../projects/client-ui/banner/sc-banner.component.ts","../../../../projects/client-ui/banner/sc-banner.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    HostListener,\n    Input,\n    OnInit,\n    QueryList,\n    TemplateRef,\n    ContentChildren,\n    HostBinding,\n    Output,\n    EventEmitter,\n} from '@angular/core';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { filter, interval, map } from 'rxjs';\nimport { ScBannerService, ScBanner } from '@snabcentr/client-core';\nimport { ScPxConverter } from '../helpers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\n@UntilDestroy({ checkProperties: true })\nexport class ScBannerComponent implements OnInit {\n    /**\n     * Список ссылок на элемент представлений шаблонов.\n     */\n    @ContentChildren('banner')\n    public bannersListRef: QueryList<TemplateRef<any>>;\n\n    /**\n     * Признак, что необходимо показывать кнопки навигации.\n     */\n    @Input()\n    public navigateButton: boolean = true;\n\n    /**\n     * Интервал показа слайдов в миллисекундах.\n     */\n    @Input()\n    public playerInterval: number = 8000;\n\n    /**\n     * Признак, что прокрутка выключена.\n     */\n    @Input()\n    public disabled: boolean = false;\n\n    /**\n     * Местоположение баннера.\n     */\n    @Input()\n    public bannerLocation?: string;\n\n    /**\n     * Признак, что компонент должен растягиваться.\n     */\n    @Input()\n    public resizable: boolean = true;\n\n    /**\n     * Событие загрузки баннеров с количеством полученных баннеров.\n     */\n    @Output()\n    public loadBannersEvent: EventEmitter<number> = new EventEmitter<number>();\n\n    /**\n     * Событие нажатия на изображение баннера.\n     */\n    @Output()\n    public clickBannerImgEvent: EventEmitter<ScBanner> = new EventEmitter<ScBanner>();\n\n    /**\n     * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.\n     */\n    @HostBinding('class.!hidden')\n    public get isHidden(): boolean {\n        return this.bannersListRef.length + this.banners.length === 0;\n    }\n\n    /**\n     * Свойство, от которого зависит высота `:host` компонента.\n     */\n    @HostBinding('style.height')\n    public height: string = '';\n\n    /**\n     * Свойство, от которого зависит ширина `:host` компонента.\n     */\n    @HostBinding('style.width')\n    public width: string = '';\n\n    /**\n     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    public currentBannerId: number = 0;\n\n    /**\n     * Список баннеров.\n     */\n    public banners: ScBanner[] = [];\n\n    /**\n     * Признак, что указатель мыши находится над баннером.\n     */\n    private isMouseOver: boolean = false;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScBannerComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param bannerService Сервис для работы с баннерами.\n     * @param pxConverter Класс хэлпер для конвертации пикселей.\n     */\n    public constructor(private readonly cdr: ChangeDetectorRef, private bannerService: ScBannerService, private pxConverter: ScPxConverter) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        interval(this.playerInterval)\n            .pipe(\n                filter(() => !this.disabled && !this.isMouseOver),\n                untilDestroyed(this)\n            )\n            .subscribe(() => this.onNextBanner());\n\n        if (this.bannerLocation) {\n            this.bannerService.banners$\n                .pipe(\n                    map((banners) => banners.filter((banner) => banner.location === this.bannerLocation)),\n                    untilDestroyed(this)\n                )\n                .subscribe((banners) => {\n                    this.banners = banners;\n\n                    if (banners.length) {\n                        if (this.resizable) {\n                            this.width = `100%`;\n                        } else {\n                            this.height = `${this.pxConverter.pxToRem(banners[0].height)}rem`;\n                            this.width = `${this.pxConverter.pxToRem(banners[0].width)}rem`;\n                        }\n\n                        this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                    }\n\n                    this.loadBannersEvent.emit(this.banners.length);\n                    this.cdr.markForCheck();\n                });\n        }\n    }\n\n    /**\n     * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.\n     *\n     * @param banner Баннер, по ссылке которого совершён переход.\n     */\n    public onClick(banner: ScBanner): void {\n        if (banner.url) {\n            this.clickBannerImgEvent.emit(banner);\n        }\n    }\n\n    /**\n     * Переключиться на предыдущий баннер.\n     */\n    public onPreviousBanner(): void {\n        const previous: number = this.currentBannerId - 1;\n        this.currentBannerId = previous < 0 ? this.bannersListRef.length + this.banners.length - 1 : previous;\n        this.cdr.markForCheck();\n    }\n\n    /**\n     * Переключиться на следующий баннер.\n     */\n    public onNextBanner(): void {\n        const next: number = this.currentBannerId + 1;\n        this.currentBannerId = next === this.bannersListRef.length + this.banners.length ? 0 : next;\n        this.cdr.markForCheck();\n    }\n\n    /**\n     * Обработчик события mouseenter.\n     */\n    @HostListener('mouseenter')\n    private mouseEnterHandler: () => boolean = () => (this.isMouseOver = true);\n\n    /**\n     * Обработчик события mouseleave.\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler: () => boolean = () => (this.isMouseOver = false);\n}\n","<div class=\"flex relative\" [style.aspect-ratio]=\"aspectRatio\">\n    <tui-carousel class=\"bg-white w-full shadow-sc-2 !absolute rounded overflow-hidden\" [(index)]=\"currentBannerId\">\n        <ng-container *ngFor=\"let item of bannersListRef; index as index\">\n            <div *tuiItem>\n                <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n            </div>\n        </ng-container>\n        <ng-container *ngFor=\"let banner of banners; index as index\">\n            <a *tuiItem (click)=\"onClick(banner)\" [attr.href]=\"banner.url ? banner.url : null\" target=\"_blank\" [title]=\"banner.title\" class=\"\">\n                <img [src]=\"banner.mediaFile\" alt=\"Баннер\" [style.aspect-ratio]=\"aspectRatio\" class=\"pointer-events-none object-cover h-full w-full\" />\n            </a>\n        </ng-container>\n    </tui-carousel>\n    <div *ngIf=\"navigateButton && !disabled && (this.bannersListRef.length + this.banners.length) > 1\" class=\"w-full h-full flex items-center grow justify-between p-3\">\n        <button tuiIconButton icon=\"tuiIconChevronLeftLarge\" tuiMode=\"onLight\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onPreviousBanner()\" class=\"arrow-btn\"></button>\n        <button tuiIconButton icon=\"tuiIconChevronRightLarge\" tuiMode=\"onLight\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onNextBanner()\" class=\"arrow-btn\"></button>\n    </div>\n</div>\n<tui-pagination\n    *ngIf=\"navigateButton && !disabled && (this.bannersListRef.length + this.banners.length) > 1\"\n    size=\"s\"\n    class=\"mt-3\"\n    [length]=\"this.bannersListRef.length + this.banners.length\"\n    [(index)]=\"currentBannerId\"\n></tui-pagination>\n"]}
222
+ //# 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    Component,\n    ChangeDetectionStrategy,\n    AfterViewInit,\n    Input,\n    Output,\n    EventEmitter,\n    ContentChildren,\n    QueryList,\n    TemplateRef,\n    ViewChild,\n    ElementRef,\n    HostBinding,\n    ChangeDetectorRef,\n    Inject,\n    HostListener,\n} from '@angular/core';\nimport { ScBanner, ScBannerService } from '@snabcentr/client-core';\nimport { Observable, map, tap, shareReplay, filter, switchMap, interval, takeUntil, Subject } from 'rxjs';\nimport { ScPxConverter } from '../helpers';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { tuiFadeIn } from '@taiga-ui/core';\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    changeDetection: ChangeDetectionStrategy.OnPush,\n    providers: [IntersectionObserverService],\n    animations: [tuiFadeIn],\n})\n@UntilDestroy({ checkProperties: true })\nexport class ScBannerComponent implements AfterViewInit {\n    /**\n     * Признак, что необходимо показывать кнопки навигации.\n     */\n    @Input()\n    public navigateButton: boolean = true;\n\n    /**\n     * Интервал показа слайдов в миллисекундах.\n     */\n    @Input()\n    public playerInterval: number = 5000;\n\n    /**\n     * Признак, что прокрутка выключена.\n     */\n    @Input()\n    public disabled: boolean = false;\n\n    /**\n     * Местоположение баннера.\n     */\n    @Input()\n    public bannerLocation?: string;\n\n    /**\n     * Признак, что компонент должен растягиваться.\n     */\n    @Input()\n    public resizable: boolean = true;\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    public bannersListRef: QueryList<TemplateRef<any>>;\n\n    /**\n     * Ссылка на {@link HTMLVideoElement} элемента отображения видео.\n     */\n    @ViewChild('videoPlayer') private videoRef: ElementRef<HTMLVideoElement>;\n\n    /**\n     * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.\n     */\n    @HostBinding('class.!hidden')\n    private get isHidden(): boolean {\n        return !this.banners || this.bannersListRef.length + this.banners.length === 0;\n    }\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    public currentBannerId: number = 0;\n\n    /**\n     * {@link Subject} изменения состояния таймера.\n     */\n    public readonly toggleTimer$: Subject<boolean> = new Subject<boolean>();\n\n    /**\n     * Признак, что необходимо показывать кнопки старта видео баннера.\n     */\n    public showPlayBtn: boolean = false;\n\n    /**\n     * {@link Observable} Обновления списка баннеров.\n     */\n    public readonly banners$: Observable<ScBanner[]> = this.bannerService.banners$.pipe(\n        map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()),\n        tap((banners) => {\n            if (banners.length) {\n                if (this.resizable) {\n                    this.width = `100%`;\n                } else {\n                    this.height = `${this.pxConverter.pxToRem(banners[0].height)}rem`;\n                    this.width = `${this.pxConverter.pxToRem(banners[0].width)}rem`;\n                }\n\n                this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                this.banners = banners;\n                this.toggleTimer$.next(true);\n            }\n\n            this.loadBannersEvent.emit(banners.length);\n        }),\n        shareReplay(1)\n    );\n\n    /**\n     * Список баннеров.\n     */\n    public banners?: ScBanner[] = [];\n\n    /**\n     * Свойство, от которого зависит высота `:host` компонента.\n     */\n    @HostBinding('style.height')\n    public height: string = 'auto';\n\n    /**\n     * Свойство, от которого зависит ширина `:host` компонента.\n     */\n    @HostBinding('style.width')\n    public width: string = '';\n\n    /**\n     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Инициализирует экземпляр класса {@link ScBannerComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param bannerService Сервис для работы с баннерами.\n     * @param pxConverter Класс хэлпер для конвертации пикселей.\n     */\n    public constructor(\n        private readonly cdr: ChangeDetectorRef,\n        private readonly bannerService: ScBannerService,\n        @Inject(IntersectionObserverService) private readonly entries$: IntersectionObserverService,\n        @Inject(ElementRef) private readonly element: ElementRef<Element>,\n        private pxConverter: ScPxConverter\n    ) {}\n\n    /** @inheritDoc */\n    public ngAfterViewInit(): void {\n        this.toggleTimer$\n            .pipe(\n                filter((toggle) => toggle),\n                switchMap(() => interval(this.playerInterval).pipe(takeUntil(this.toggleTimer$))),\n                filter(() => !this.disabled && (this.banners?.[this.currentBannerId]?.mediaType === 'image' || this.showPlayBtn)),\n                untilDestroyed(this)\n            )\n            .subscribe(() => this.onNextBanner());\n\n        // Отслеживание пересечения компонента с экраном пользователя.\n        // Если баннера не находится в поле видимости пользователя, то он перестаёт переключатся, а видео останавливается.\n        this.entries$.pipe(map((entries) => entries.find((item) => item.target === this.element.nativeElement))).subscribe((entry) => {\n            this.toggleTimer$.next(!!entry?.isIntersecting);\n        });\n    }\n\n    /**\n     * Переключает на предыдущий баннер.\n     */\n    public onPreviousBanner(): void {\n        const previous: number = this.currentBannerId - 1;\n        this.currentBannerId = previous < 0 ? this.bannersListRef.length + (this.banners?.length ?? 0) - 1 : previous;\n\n        this.cdr.markForCheck();\n    }\n\n    /**\n     * Переключает на следующий баннер.\n     */\n    public onNextBanner(): void {\n        const next: number = this.currentBannerId + 1;\n        this.currentBannerId = next === this.bannersListRef.length + (this.banners?.length ?? 0) ? 0 : next;\n\n        this.cdr.markForCheck();\n    }\n\n    /**\n     * Переключает на следующий баннер.\n     */\n    public onEndedVideo(): void {\n        this.onNextBanner();\n        this.toggleTimer$.next(true);\n        this.showPlayBtn = true;\n    }\n\n    /**\n     * Обработчик нажатия на баннер, генерирующий событие {@link clickBannerImgEvent}.\n     *\n     * @param banner Баннер, по ссылке которого совершён переход.\n     */\n    public onClick(banner: ScBanner): void {\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\n\n    /**\n     * Обработчик события mouseenter.\n     *\n     * @private\n     */\n    @HostListener('mouseenter')\n    private mouseEnterHandler: () => void = () => this.toggleTimer$.next(false);\n\n    /**\n     * Обработчик события mouseleave.\n     *\n     * @private\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler: () => void = () => this.toggleTimer$.next(true);\n}\n","<ng-container *tuiLet=\"banners$ | async\">\n    <tui-carousel class=\"bg-white w-full h-full shadow-sc-2 rounded-xl overflow-hidden\" [(index)]=\"currentBannerId\">\n        <ng-container *ngFor=\"let banner of banners; let index = index\">\n            <a\n                *tuiItem\n                (click)=\"onClick(banner)\"\n                [style.aspect-ratio]=\"aspectRatio\"\n                [attr.href]=\"banner.url ? banner.url : null\"\n                target=\"_blank\"\n                [title]=\"banner.title\"\n                class=\"relative\"\n            >\n                <ng-container [ngSwitch]=\"banner.mediaType\">\n                    <ng-container *ngSwitchCase=\"'video'\">\n                        <video\n                            #videoPlayer\n                            (suspend)=\"showPlayBtn = true\"\n                            (play)=\"showPlayBtn = false\"\n                            [src]=\"banner.mediaFile\"\n                            (ended)=\"onEndedVideo()\"\n                            (mouseover)=\"videoPlayer.pause()\"\n                            (mouseout)=\"videoPlayer.play()\"\n                            class=\"object-cover h-full\"\n                            muted\n                            autoplay\n                        ></video>\n                        <button\n                            *ngIf=\"showPlayBtn\"\n                            tuiIconButton\n                            [@tuiFadeIn]=\"200\"\n                            matRipple\n                            (click)=\"$event.preventDefault(); videoPlayer.play()\"\n                            size=\"s\"\n                            appearance=\"secondary\"\n                            class=\"!absolute left-8 bottom-4\"\n                        >\n                            <i class=\"icon-refresh text-black\"></i>\n                        </button>\n                    </ng-container>\n\n                    <img *ngSwitchCase=\"'image'\" [src]=\"banner.mediaFile\" alt=\"Баннер\" class=\"object-cover h-full\" />\n                </ng-container>\n            </a>\n        </ng-container>\n        <ng-container *ngFor=\"let item of bannersListRef\">\n            <div *tuiItem [style.height]=\"height\" [style.width]=\"width\" [style.aspect-ratio]=\"aspectRatio\" class=\"overflow-hidden\">\n                <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n            </div>\n        </ng-container>\n    </tui-carousel>\n</ng-container>\n<div *ngIf=\"navigateButton && !disabled && this.banners && (this.bannersListRef.length + this.banners.length) > 1\" tuiMode=\"onLight\" class=\"flex items-center\">\n    <button tuiIconButton icon=\"tuiIconChevronLeftLarge\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onPreviousBanner()\" class=\"!absolute left-2\"></button>\n    <button tuiIconButton icon=\"tuiIconChevronRightLarge\" size=\"m\" shape=\"rounded\" appearance=\"flat\" (click)=\"onNextBanner()\" class=\"!absolute right-2\"></button>\n</div>\n"]}
@@ -2,9 +2,11 @@ import { NgModule } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import { ScBannerComponent } from './sc-banner.component';
4
4
  import { RouterModule } from '@angular/router';
5
- import { TuiButtonModule, TuiModeModule } from '@taiga-ui/core';
5
+ import { TuiButtonModule, TuiLoaderModule, TuiModeModule } from '@taiga-ui/core';
6
6
  import { MatRippleModule } from '@angular/material/core';
7
- import { TuiCarouselModule, TuiPaginationModule } from '@taiga-ui/kit';
7
+ import { TuiCarouselModule } from '@taiga-ui/kit';
8
+ import { IntersectionObserverModule } from '@ng-web-apis/intersection-observer';
9
+ import { TuiLetModule } from '@taiga-ui/cdk';
8
10
  import * as i0 from "@angular/core";
9
11
  /**
10
12
  * Модуль баннеров.
@@ -12,14 +14,14 @@ import * as i0 from "@angular/core";
12
14
  export class ScBannerModule {
13
15
  }
14
16
  ScBannerModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
15
- ScBannerModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "14.3.0", ngImport: i0, type: ScBannerModule, declarations: [ScBannerComponent], imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiPaginationModule, TuiModeModule], exports: [ScBannerComponent] });
16
- ScBannerModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerModule, imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiPaginationModule, TuiModeModule] });
17
+ ScBannerModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "14.3.0", ngImport: i0, type: ScBannerModule, declarations: [ScBannerComponent], imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiModeModule, TuiLoaderModule, IntersectionObserverModule, TuiLetModule], exports: [ScBannerComponent] });
18
+ ScBannerModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerModule, imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiModeModule, TuiLoaderModule, IntersectionObserverModule, TuiLetModule] });
17
19
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScBannerModule, decorators: [{
18
20
  type: NgModule,
19
21
  args: [{
20
22
  declarations: [ScBannerComponent],
21
- imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiPaginationModule, TuiModeModule],
23
+ imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiModeModule, TuiLoaderModule, IntersectionObserverModule, TuiLetModule],
22
24
  exports: [ScBannerComponent],
23
25
  }]
24
26
  }] });
25
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtYmFubmVyLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9iYW5uZXIvc2MtYmFubmVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxhQUFhLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNoRSxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDekQsT0FBTyxFQUFFLGlCQUFpQixFQUFFLG1CQUFtQixFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUV2RTs7R0FFRztBQU1ILE1BQU0sT0FBTyxjQUFjOzsyR0FBZCxjQUFjOzRHQUFkLGNBQWMsaUJBSlIsaUJBQWlCLGFBQ3RCLFlBQVksRUFBRSxZQUFZLEVBQUUsZUFBZSxFQUFFLGVBQWUsRUFBRSxpQkFBaUIsRUFBRSxtQkFBbUIsRUFBRSxhQUFhLGFBQ25ILGlCQUFpQjs0R0FFbEIsY0FBYyxZQUhiLFlBQVksRUFBRSxZQUFZLEVBQUUsZUFBZSxFQUFFLGVBQWUsRUFBRSxpQkFBaUIsRUFBRSxtQkFBbUIsRUFBRSxhQUFhOzJGQUdwSCxjQUFjO2tCQUwxQixRQUFRO21CQUFDO29CQUNOLFlBQVksRUFBRSxDQUFDLGlCQUFpQixDQUFDO29CQUNqQyxPQUFPLEVBQUUsQ0FBQyxZQUFZLEVBQUUsWUFBWSxFQUFFLGVBQWUsRUFBRSxlQUFlLEVBQUUsaUJBQWlCLEVBQUUsbUJBQW1CLEVBQUUsYUFBYSxDQUFDO29CQUM5SCxPQUFPLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztpQkFDL0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFNjQmFubmVyQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1iYW5uZXIuY29tcG9uZW50JztcbmltcG9ydCB7IFJvdXRlck1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBUdWlCdXR0b25Nb2R1bGUsIFR1aU1vZGVNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY29yZSc7XG5pbXBvcnQgeyBNYXRSaXBwbGVNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9jb3JlJztcbmltcG9ydCB7IFR1aUNhcm91c2VsTW9kdWxlLCBUdWlQYWdpbmF0aW9uTW9kdWxlIH0gZnJvbSAnQHRhaWdhLXVpL2tpdCc7XG5cbi8qKlxuICog0JzQvtC00YPQu9GMINCx0LDQvdC90LXRgNC+0LIuXG4gKi9cbkBOZ01vZHVsZSh7XG4gICAgZGVjbGFyYXRpb25zOiBbU2NCYW5uZXJDb21wb25lbnRdLFxuICAgIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFJvdXRlck1vZHVsZSwgVHVpQnV0dG9uTW9kdWxlLCBNYXRSaXBwbGVNb2R1bGUsIFR1aUNhcm91c2VsTW9kdWxlLCBUdWlQYWdpbmF0aW9uTW9kdWxlLCBUdWlNb2RlTW9kdWxlXSxcbiAgICBleHBvcnRzOiBbU2NCYW5uZXJDb21wb25lbnRdLFxufSlcbmV4cG9ydCBjbGFzcyBTY0Jhbm5lck1vZHVsZSB7fVxuIl19
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtYmFubmVyLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9iYW5uZXIvc2MtYmFubmVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDakYsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3pELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNsRCxPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUNoRixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUU3Qzs7R0FFRztBQU1ILE1BQU0sT0FBTyxjQUFjOzsyR0FBZCxjQUFjOzRHQUFkLGNBQWMsaUJBSlIsaUJBQWlCLGFBQ3RCLFlBQVksRUFBRSxZQUFZLEVBQUUsZUFBZSxFQUFFLGVBQWUsRUFBRSxpQkFBaUIsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLDBCQUEwQixFQUFFLFlBQVksYUFDekosaUJBQWlCOzRHQUVsQixjQUFjLFlBSGIsWUFBWSxFQUFFLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLGlCQUFpQixFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsMEJBQTBCLEVBQUUsWUFBWTsyRkFHMUosY0FBYztrQkFMMUIsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztvQkFDakMsT0FBTyxFQUFFLENBQUMsWUFBWSxFQUFFLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLGlCQUFpQixFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsMEJBQTBCLEVBQUUsWUFBWSxDQUFDO29CQUNwSyxPQUFPLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztpQkFDL0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFNjQmFubmVyQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1iYW5uZXIuY29tcG9uZW50JztcbmltcG9ydCB7IFJvdXRlck1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBUdWlCdXR0b25Nb2R1bGUsIFR1aUxvYWRlck1vZHVsZSwgVHVpTW9kZU1vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7IE1hdFJpcHBsZU1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHsgVHVpQ2Fyb3VzZWxNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkva2l0JztcbmltcG9ydCB7IEludGVyc2VjdGlvbk9ic2VydmVyTW9kdWxlIH0gZnJvbSAnQG5nLXdlYi1hcGlzL2ludGVyc2VjdGlvbi1vYnNlcnZlcic7XG5pbXBvcnQgeyBUdWlMZXRNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY2RrJztcblxuLyoqXG4gKiDQnNC+0LTRg9C70Ywg0LHQsNC90L3QtdGA0L7Qsi5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtTY0Jhbm5lckNvbXBvbmVudF0sXG4gICAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgUm91dGVyTW9kdWxlLCBUdWlCdXR0b25Nb2R1bGUsIE1hdFJpcHBsZU1vZHVsZSwgVHVpQ2Fyb3VzZWxNb2R1bGUsIFR1aU1vZGVNb2R1bGUsIFR1aUxvYWRlck1vZHVsZSwgSW50ZXJzZWN0aW9uT2JzZXJ2ZXJNb2R1bGUsIFR1aUxldE1vZHVsZV0sXG4gICAgZXhwb3J0czogW1NjQmFubmVyQ29tcG9uZW50XSxcbn0pXG5leHBvcnQgY2xhc3MgU2NCYW5uZXJNb2R1bGUge31cbiJdfQ==