@snabcentr/client-ui 0.17.4 → 0.19.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (23) hide show
  1. package/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.d.ts +4 -8
  2. package/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.d.ts +3 -8
  3. package/auth/sc-sign-in-form/sc-sign-in-form.component.d.ts +8 -4
  4. package/catalog/price-history/sc-price-history.component.d.ts +44 -43
  5. package/catalog/sc-catalog.module.d.ts +3 -3
  6. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +4 -16
  7. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +4 -16
  8. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form.component.mjs +24 -6
  9. package/esm2020/banner/sc-banner.component.mjs +4 -2
  10. package/esm2020/catalog/price-history/sc-price-history.component.mjs +85 -79
  11. package/esm2020/catalog/sc-catalog.module.mjs +12 -15
  12. package/fesm2015/snabcentr-client-ui.mjs +125 -321
  13. package/fesm2015/snabcentr-client-ui.mjs.map +1 -1
  14. package/fesm2020/snabcentr-client-ui.mjs +123 -320
  15. package/fesm2020/snabcentr-client-ui.mjs.map +1 -1
  16. package/package.json +3 -5
  17. package/styles/tailwind/tailwind.scss +0 -24
  18. package/catalog/price-history/sc-chart-option.d.ts +0 -7
  19. package/catalog/price-history/sc-i-chart-data-item.d.ts +0 -13
  20. package/catalog/price-history/sc-lang-ru.d.ts +0 -115
  21. package/esm2020/catalog/price-history/sc-chart-option.mjs +0 -79
  22. package/esm2020/catalog/price-history/sc-i-chart-data-item.mjs +0 -2
  23. package/esm2020/catalog/price-history/sc-lang-ru.mjs +0 -115
@@ -168,7 +168,9 @@ let ScBannerComponent = class ScBannerComponent {
168
168
  onClick(banner) {
169
169
  this.userMetrikaService.emitUserMetrikaEvent({
170
170
  target: ScUserMetrikaGoalsEnum.bannerClick,
171
- params: { banner_id: banner.id },
171
+ params: {
172
+ banner_id: banner.id,
173
+ },
172
174
  });
173
175
  if (banner.url) {
174
176
  this.clickBannerEvent.emit(banner);
@@ -229,4 +231,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
229
231
  type: HostListener,
230
232
  args: ['mouseleave']
231
233
  }] } });
232
- //# 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    AfterViewInit,\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    Output,\n    QueryList,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { tuiFadeIn } from '@taiga-ui/core';\nimport { filter, interval, map, Observable, shareReplay, Subject, switchMap, takeUntil, tap } from 'rxjs';\n\nimport { ScPxConverter } from '../helpers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    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')\n    private videoRef: ElementRef<HTMLVideoElement>;\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 > 0) {\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     * Свойство, от которого зависит наличие класса `!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     * Свойство, от которого зависит высота `: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     * Обработчик события mouseenter.\n     */\n    @HostListener('mouseenter')\n    private mouseEnterHandler: () => void = () => {\n        this.toggleTimer$.next(false);\n    };\n\n    /**\n     * Обработчик события mouseleave.\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler: () => void = () => {\n        this.toggleTimer$.next(true);\n    };\n\n    /**\n     * Инициализирует экземпляр класса {@link ScBannerComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param bannerService Сервис для работы с баннерами.\n     * @param entries$\n     * @param element Элемент баннера.\n     * @param pxConverter Экземпляр класса-помощника для конвертации пикселей.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     */\n    public constructor(\n        private readonly cdr: ChangeDetectorRef,\n        private readonly bannerService: ScBannerService,\n        @Inject(IntersectionObserverService) private readonly entries$: IntersectionObserverService,\n        @Inject(ElementRef) private readonly element: ElementRef<Element>,\n        private pxConverter: ScPxConverter,\n        private readonly userMetrikaService: ScUserMetrikaService\n    ) {}\n\n    /** @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(() => {\n                this.onNextBanner();\n            });\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        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.bannerClick,\n            params: { banner_id: banner.id },\n        });\n\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\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"]}
234
+ //# 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    AfterViewInit,\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    Output,\n    QueryList,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { tuiFadeIn } from '@taiga-ui/core';\nimport { filter, interval, map, Observable, shareReplay, Subject, switchMap, takeUntil, tap } from 'rxjs';\n\nimport { ScPxConverter } from '../helpers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    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')\n    private videoRef: ElementRef<HTMLVideoElement>;\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 > 0) {\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     * Свойство, от которого зависит наличие класса `!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     * Свойство, от которого зависит высота `: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     * Обработчик события mouseenter.\n     */\n    @HostListener('mouseenter')\n    private mouseEnterHandler: () => void = () => {\n        this.toggleTimer$.next(false);\n    };\n\n    /**\n     * Обработчик события mouseleave.\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler: () => void = () => {\n        this.toggleTimer$.next(true);\n    };\n\n    /**\n     * Инициализирует экземпляр класса {@link ScBannerComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param bannerService Сервис для работы с баннерами.\n     * @param entries$\n     * @param element Элемент баннера.\n     * @param pxConverter Экземпляр класса-помощника для конвертации пикселей.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     */\n    public constructor(\n        private readonly cdr: ChangeDetectorRef,\n        private readonly bannerService: ScBannerService,\n        @Inject(IntersectionObserverService) private readonly entries$: IntersectionObserverService,\n        @Inject(ElementRef) private readonly element: ElementRef<Element>,\n        private pxConverter: ScPxConverter,\n        private readonly userMetrikaService: ScUserMetrikaService\n    ) {}\n\n    /** @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(() => {\n                this.onNextBanner();\n            });\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        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.bannerClick,\n            params: {\n                banner_id: banner.id,\n            },\n        });\n\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\n}\n","<ng-container *tuiLet=\"banners$ | async\">\n    <tui-carousel 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"]}
@@ -1,108 +1,114 @@
1
- import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
2
- import { scChartOption } from './sc-chart-option';
1
+ import { __decorate } from "tslib";
2
+ /* eslint-disable class-methods-use-this,lodash/prefer-lodash-method */
3
+ import { ChangeDetectionStrategy, Component, Inject, Input, Optional } from '@angular/core';
4
+ import { TuiDay, TuiDayRange, TuiMonth, tuiPure } from '@taiga-ui/cdk';
5
+ import { TUI_MONTHS } from '@taiga-ui/core';
6
+ import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
7
+ import { filter, map, shareReplay } from 'rxjs';
3
8
  import * as i0 from "@angular/core";
4
- import * as i1 from "@angular/common";
5
- import * as i2 from "ngx-echarts";
9
+ import * as i1 from "@snabcentr/client-core";
10
+ import * as i2 from "@angular/common";
11
+ import * as i3 from "@taiga-ui/addon-charts";
12
+ import * as i4 from "rxjs";
6
13
  /**
7
14
  * График истории цен товара или услуги.
8
- * TODO: TASK[#7482] Перепроверить возможность использования TuiLineDaysChart.
9
15
  */
10
16
  export class ScPriceHistoryComponent {
11
17
  /**
12
18
  * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.
13
19
  *
14
- * @param cdr Объект для работы с обнаружением изменений.
20
+ * @param catalogService Сервис для работы с каталогом.
21
+ * @param months$ Перечисление месяцев.
22
+ * @param context Контекст диалогового окна, в котором открыт компонент.
15
23
  */
16
- constructor(cdr) {
17
- this.cdr = cdr;
24
+ constructor(catalogService, months$, context) {
25
+ this.catalogService = catalogService;
26
+ this.months$ = months$;
27
+ this.context = context;
18
28
  /**
19
- * Настройки графика.
29
+ * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.
20
30
  */
21
- this.initOption = { locale: 'RU' };
31
+ this.xStringify$ = this.months$.pipe(map((months) => ({ month, day }) =>
32
+ // eslint-disable-next-line security/detect-object-injection
33
+ `${months[month]}, ${day}`));
22
34
  /**
23
- * Параметры отрисовки графика истории цены.
35
+ * Функция для преобразования значения number в строку в подсказке по оси Y.
36
+ *
37
+ * @param value Значение оси Y.
24
38
  */
25
- this.chartOption = scChartOption;
26
- /**
27
- * Данные об истории цены на товар или услугу.
28
- */
29
- this.data = [];
39
+ this.yStringify = (value) => `${value.toLocaleString('ru-RU')} ₽`;
40
+ if (context) {
41
+ this.product = context.data.product;
42
+ }
30
43
  }
31
44
  /** @inheritDoc */
32
45
  ngOnInit() {
33
- if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {
34
- this.chartOption.series.data = [];
35
- }
36
- Object.keys(this.history)
37
- .map((key) => {
38
- // ? Можно избежать переведя даты в api в формат ECMAScript® 2023: https://tc39.es/ecma262/#sec-date-time-string-format
39
- const dataString = key.split('.').reverse();
40
- const data = new Date(+dataString[0], +dataString[1] - 1, +dataString[2]);
41
- return { data: data, cost: this.history[String(key)].cost };
42
- })
43
- .sort((a, b) => +a.data - +b.data)
44
- .forEach((item, index, array) => {
45
- let nextDate;
46
- // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
47
- if (array[index + 1]?.data) {
48
- nextDate = new Date(array[index + 1]?.data);
49
- nextDate.setDate(nextDate.getDate() - 1);
50
- }
51
- else {
52
- nextDate = new Date();
53
- }
54
- this.pushDataItem(item.cost, item.data, nextDate);
55
- });
56
- // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
57
- if (this.eChartsInstance) {
58
- this.setChartData();
59
- }
46
+ this.history$ = this.catalogService.getPriceHistory$(this.product).pipe(map((history) => Object.keys(history)
47
+ .map((item) => [TuiDay.normalizeParse(item, 'DMY'), history[String(item)].cost])
48
+ .sort((a, b) => (a[0] > b[0] ? 1 : -1))), shareReplay({ bufferSize: 1, refCount: true }), filter((history) => history.length > 1));
49
+ this.priceInfo$ = this.history$.pipe(map((history) => history.map((item) => item[1])), map((values) => ({
50
+ min: Math.min(...values),
51
+ max: Math.max(...values),
52
+ })));
53
+ this.range$ = this.history$.pipe(map((value) => ({ from: value.at(0)?.[0], to: value.at(-1)?.[0] })), filter((range) => !!range.from && !!range.to), map((range) => {
54
+ return new TuiDayRange(range.from, range.to);
55
+ }));
60
56
  }
61
57
  /**
62
- * Перехватчик жизненного цикла {@link ECharts}, который вызывается при его инициализации.
58
+ * Вычисляет подписи даты к оси X.
63
59
  *
64
- * @param eChartsInstance Экземпляр {@link ECharts}.
60
+ * @param param0 {@link TuiDayRange} Временной промежуток истории цен.
61
+ * @param param0.from {@link TuiDay} Дата начала истории цен.
62
+ * @param param0.to {@link TuiDay} Дата конца истории цен.
65
63
  */
66
- // eslint-disable-next-line unicorn/prevent-abbreviations
67
- onChartInit(eChartsInstance) {
68
- this.eChartsInstance = eChartsInstance;
69
- if (this.data.length > 0) {
70
- this.setChartData();
71
- }
72
- }
73
- /**
74
- * Устанавливает новые данные {@link ScPriceHistoryComponent.eChartsInstance}.
75
- */
76
- setChartData() {
77
- if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {
78
- this.maxPrice = Math.max(...this.data.map((item) => item.value[1]));
79
- this.minPrice = Math.min(...this.data.map((item) => item.value[1]));
80
- this.chartOption.series.data = this.data;
81
- this.eChartsInstance.clear();
82
- this.eChartsInstance.setOption(this.chartOption, true, true);
83
- this.cdr.markForCheck();
84
- }
64
+ computeLabels$({ from, to }) {
65
+ return this.months$.pipe(map((months) => Array.from({ length: TuiMonth.lengthBetween(from, to) + 1 }, (_, index) => months[from.append({ month: index }).month])));
85
66
  }
86
67
  /**
87
- * Добавляет значение цены товара в массив истории цен.
68
+ * Вычисляет данные для отображения на графике.
88
69
  *
89
- * @param value Цена товара или услуги.
90
- * @param date Дата установки цены.
91
- * @param nextDate Следующая дата установки цены.
70
+ * @param param0 {@link TuiDayRange} Временной промежуток истории цен.
71
+ * @param param0.from {@link TuiDay} Дата начала истории цен.
72
+ * @param param0.to {@link TuiDay} Дата конца истории цен.
73
+ * @param history История цен.
92
74
  */
93
- pushDataItem(value, date, nextDate) {
94
- this.data.push({
95
- name: date.toString(),
96
- value: [date, value, nextDate],
97
- });
75
+ computeValue({ from, to }, history) {
76
+ return (Array.from({ length: TuiDay.lengthBetween(from, to) + 1 })
77
+ .fill(0)
78
+ // eslint-disable-next-line unicorn/no-array-reduce
79
+ .reduce((array, _, index) => {
80
+ const data = from.append({ day: index });
81
+ return [
82
+ ...array,
83
+ [
84
+ data,
85
+ (history.find((valueHistory, indexHistory, arrayHistory) => data.daySame(valueHistory[0]) || data.dayBefore(arrayHistory[indexHistory + 1][0])) ??
86
+ history[0])[1],
87
+ ],
88
+ ];
89
+ }, []));
98
90
  }
99
91
  }
100
- ScPriceHistoryComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScPriceHistoryComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
101
- ScPriceHistoryComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScPriceHistoryComponent, selector: "sc-price-history", inputs: { history: "history" }, ngImport: i0, template: "<div class=\"flex flex-col items-center\">\n <div *ngIf=\"maxPrice && minPrice\" class=\"w-full font-bold text-end text-lg mb-1\">\u043E\u0442 {{ minPrice.toLocaleString() }} \u20BD \u0434\u043E {{ maxPrice.toLocaleString() }} \u20BD</div>\n <div class=\"relative w-full h-56\">\n <div class=\"h-48 bg-tui-base-02 mt-2 absolute rounded right-0 left-16\"></div>\n <div echarts [initOpts]=\"initOption\" (chartInit)=\"onChartInit($event)\" [options]=\"chartOption\" class=\"w-full !h-full touch-none\"></div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgxEchartsDirective, selector: "echarts, [echarts]", inputs: ["options", "theme", "loading", "initOpts", "merge", "autoResize", "loadingType", "loadingOpts"], outputs: ["chartInit", "optionsError", "chartClick", "chartDblClick", "chartMouseDown", "chartMouseMove", "chartMouseUp", "chartMouseOver", "chartMouseOut", "chartGlobalOut", "chartContextMenu", "chartLegendSelectChanged", "chartLegendSelected", "chartLegendUnselected", "chartLegendScroll", "chartDataZoom", "chartDataRangeSelected", "chartTimelineChanged", "chartTimelinePlayChanged", "chartRestore", "chartDataViewChanged", "chartMagicTypeChanged", "chartPieSelectChanged", "chartPieSelected", "chartPieUnselected", "chartMapSelectChanged", "chartMapSelected", "chartMapUnselected", "chartAxisAreaSelected", "chartFocusNodeAdjacency", "chartUnfocusNodeAdjacency", "chartBrush", "chartBrushEnd", "chartBrushSelected", "chartRendered", "chartFinished"], exportAs: ["echarts"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
92
+ ScPriceHistoryComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScPriceHistoryComponent, deps: [{ token: i1.ScCatalogService }, { token: TUI_MONTHS }, { token: POLYMORPHEUS_CONTEXT, optional: true }], target: i0.ɵɵFactoryTarget.Component });
93
+ ScPriceHistoryComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScPriceHistoryComponent, selector: "sc-price-history", inputs: { product: "product" }, ngImport: i0, template: "<div class=\"flex flex-col items-center\">\n <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n <div class=\"w-174 mb-1 text-end text-lg font-bold\">\u043E\u0442 {{ priceInfo.min.toLocaleString() }} \u20BD \u0434\u043E {{ priceInfo.max.toLocaleString() }} \u20BD</div>\n\n <ng-container *ngIf=\"history$ | async as history\">\n <div\n *ngIf=\"range$ | async as range\"\n class=\"relative h-56 w-full p-5 text-tui-primary\"\n >\n <tui-axes\n *ngIf=\"computeLabels$(range) | async as labels\"\n class=\"h-full\"\n [axisXLabels]=\"labels\"\n [horizontalLines]=\"4\"\n [verticalLines]=\"labels.length\"\n >\n <tui-line-days-chart\n class=\"chart\"\n [height]=\"priceInfo.max\"\n [smoothingFactor]=\"10\"\n [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n [value]=\"computeValue(range, history)\"\n [xStringify]=\"xStringify$ | async\"\n [yStringify]=\"yStringify\"\n [hintContent]=\"hint\"\n ></tui-line-days-chart>\n </tui-axes>\n </div>\n </ng-container>\n </ng-container>\n <ng-template #isNotExist>\n <p class=\"text-base font-bold\">\u0426\u0435\u043D\u0430 \u043D\u0430 \u0442\u043E\u0432\u0430\u0440 \u043D\u0435 \u0438\u0437\u043C\u0435\u043D\u044F\u043B\u0430\u0441\u044C</p>\n </ng-template>\n <ng-template\n #hint\n let-data\n >\n <div class=\"font-bold\">{{ data[1].toLocaleString() }} \u20BD</div>\n <div>{{ data[0] }}</div>\n </ng-template>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.TuiLineDaysChartComponent, selector: "tui-line-days-chart", inputs: ["value", "y", "height", "smoothingFactor", "hintContent", "xStringify", "yStringify", "dots"] }, { kind: "component", type: i3.TuiAxesComponent, selector: "tui-axes", inputs: ["axisX", "axisXLabels", "axisY", "axisYInset", "axisYLabels", "axisYName", "axisYSecondaryInset", "axisYSecondaryLabels", "axisYSecondaryName", "horizontalLines", "horizontalLinesHandler", "verticalLines", "verticalLinesHandler"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
94
+ __decorate([
95
+ tuiPure
96
+ ], ScPriceHistoryComponent.prototype, "computeLabels$", null);
97
+ __decorate([
98
+ tuiPure
99
+ ], ScPriceHistoryComponent.prototype, "computeValue", null);
102
100
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScPriceHistoryComponent, decorators: [{
103
101
  type: Component,
104
- args: [{ selector: 'sc-price-history', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center\">\n <div *ngIf=\"maxPrice && minPrice\" class=\"w-full font-bold text-end text-lg mb-1\">\u043E\u0442 {{ minPrice.toLocaleString() }} \u20BD \u0434\u043E {{ maxPrice.toLocaleString() }} \u20BD</div>\n <div class=\"relative w-full h-56\">\n <div class=\"h-48 bg-tui-base-02 mt-2 absolute rounded right-0 left-16\"></div>\n <div echarts [initOpts]=\"initOption\" (chartInit)=\"onChartInit($event)\" [options]=\"chartOption\" class=\"w-full !h-full touch-none\"></div>\n </div>\n</div>\n" }]
105
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { history: [{
102
+ args: [{ selector: 'sc-price-history', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center\">\n <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n <div class=\"w-174 mb-1 text-end text-lg font-bold\">\u043E\u0442 {{ priceInfo.min.toLocaleString() }} \u20BD \u0434\u043E {{ priceInfo.max.toLocaleString() }} \u20BD</div>\n\n <ng-container *ngIf=\"history$ | async as history\">\n <div\n *ngIf=\"range$ | async as range\"\n class=\"relative h-56 w-full p-5 text-tui-primary\"\n >\n <tui-axes\n *ngIf=\"computeLabels$(range) | async as labels\"\n class=\"h-full\"\n [axisXLabels]=\"labels\"\n [horizontalLines]=\"4\"\n [verticalLines]=\"labels.length\"\n >\n <tui-line-days-chart\n class=\"chart\"\n [height]=\"priceInfo.max\"\n [smoothingFactor]=\"10\"\n [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n [value]=\"computeValue(range, history)\"\n [xStringify]=\"xStringify$ | async\"\n [yStringify]=\"yStringify\"\n [hintContent]=\"hint\"\n ></tui-line-days-chart>\n </tui-axes>\n </div>\n </ng-container>\n </ng-container>\n <ng-template #isNotExist>\n <p class=\"text-base font-bold\">\u0426\u0435\u043D\u0430 \u043D\u0430 \u0442\u043E\u0432\u0430\u0440 \u043D\u0435 \u0438\u0437\u043C\u0435\u043D\u044F\u043B\u0430\u0441\u044C</p>\n </ng-template>\n <ng-template\n #hint\n let-data\n >\n <div class=\"font-bold\">{{ data[1].toLocaleString() }} \u20BD</div>\n <div>{{ data[0] }}</div>\n </ng-template>\n</div>\n" }]
103
+ }], ctorParameters: function () { return [{ type: i1.ScCatalogService }, { type: i4.Observable, decorators: [{
104
+ type: Inject,
105
+ args: [TUI_MONTHS]
106
+ }] }, { type: undefined, decorators: [{
107
+ type: Optional
108
+ }, {
109
+ type: Inject,
110
+ args: [POLYMORPHEUS_CONTEXT]
111
+ }] }]; }, propDecorators: { product: [{
106
112
  type: Input
107
- }] } });
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-history.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.ts","../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport { ScPriceHistory } from '@snabcentr/client-core';\nimport { ECharts, EChartsOption } from 'echarts';\n\nimport { scChartOption } from './sc-chart-option';\nimport { ScIChartDataItem } from './sc-i-chart-data-item';\n\n/**\n * График истории цен товара или услуги.\n * TODO: TASK[#7482] Перепроверить возможность использования TuiLineDaysChart.\n */\n@Component({\n    selector: 'sc-price-history',\n    templateUrl: './sc-price-history.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceHistoryComponent implements OnInit {\n    /**\n     * История цен на товар/услугу.\n     */\n    @Input() public history: ScPriceHistory;\n\n    /**\n     * Максимальная цена товара.\n     */\n    public maxPrice?: number;\n\n    /**\n     * Минимальная цена товара.\n     */\n    public minPrice?: number;\n\n    /**\n     * Настройки графика.\n     */\n    public initOption: {\n        devicePixelRatio?: number;\n        renderer?: string;\n        width?: number | string;\n        height?: number | string;\n        locale?: string;\n    } = { locale: 'RU' };\n\n    /**\n     * Параметры отрисовки графика истории цены.\n     */\n    public chartOption: EChartsOption = scChartOption;\n\n    /**\n     * Данные об истории цены на товар или услугу.\n     */\n    private data: ScIChartDataItem[] = [];\n\n    /**\n     * Экземпляр {@link ECharts} в шаблоне.\n     */\n    private eChartsInstance: ECharts;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     */\n    public constructor(private readonly cdr: ChangeDetectorRef) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {\n            this.chartOption.series.data = [];\n        }\n\n        Object.keys(this.history)\n            .map((key) => {\n                // ? Можно избежать переведя даты в api в формат ECMAScript® 2023: https://tc39.es/ecma262/#sec-date-time-string-format\n                const dataString = key.split('.').reverse();\n                const data = new Date(+dataString[0], +dataString[1] - 1, +dataString[2]);\n\n                return { data: data, cost: this.history[String(key)].cost };\n            })\n            .sort((a, b) => +a.data - +b.data)\n            .forEach((item, index, array) => {\n                let nextDate: Date;\n\n                // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                if (array[index + 1]?.data) {\n                    nextDate = new Date(array[index + 1]?.data);\n                    nextDate.setDate(nextDate.getDate() - 1);\n                } else {\n                    nextDate = new Date();\n                }\n\n                this.pushDataItem(item.cost, item.data, nextDate);\n            });\n\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        if (this.eChartsInstance) {\n            this.setChartData();\n        }\n    }\n\n    /**\n     * Перехватчик жизненного цикла {@link ECharts}, который вызывается при его инициализации.\n     *\n     * @param eChartsInstance Экземпляр {@link ECharts}.\n     */\n    // eslint-disable-next-line unicorn/prevent-abbreviations\n    public onChartInit(eChartsInstance: ECharts) {\n        this.eChartsInstance = eChartsInstance;\n\n        if (this.data.length > 0) {\n            this.setChartData();\n        }\n    }\n\n    /**\n     * Устанавливает новые данные {@link ScPriceHistoryComponent.eChartsInstance}.\n     */\n    private setChartData(): void {\n        if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {\n            this.maxPrice = Math.max(...this.data.map((item) => item.value[1]));\n            this.minPrice = Math.min(...this.data.map((item) => item.value[1]));\n            this.chartOption.series.data = this.data;\n            this.eChartsInstance.clear();\n            this.eChartsInstance.setOption(this.chartOption, true, true);\n            this.cdr.markForCheck();\n        }\n    }\n\n    /**\n     * Добавляет значение цены товара в массив истории цен.\n     *\n     * @param value Цена товара или услуги.\n     * @param date Дата установки цены.\n     * @param nextDate Следующая дата установки цены.\n     */\n    private pushDataItem(value: number, date: Date, nextDate: Date): void {\n        this.data.push({\n            name: date.toString(),\n            value: [date, value, nextDate],\n        });\n    }\n}\n","<div class=\"flex flex-col items-center\">\n    <div *ngIf=\"maxPrice && minPrice\" class=\"w-full font-bold text-end text-lg mb-1\">от {{ minPrice.toLocaleString() }} ₽ до {{ maxPrice.toLocaleString() }} ₽</div>\n    <div class=\"relative w-full h-56\">\n        <div class=\"h-48 bg-tui-base-02 mt-2 absolute rounded right-0 left-16\"></div>\n        <div echarts [initOpts]=\"initOption\" (chartInit)=\"onChartInit($event)\" [options]=\"chartOption\" class=\"w-full !h-full touch-none\"></div>\n    </div>\n</div>\n"]}
113
+ }], computeLabels$: [], computeValue: [] } });
114
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-history.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.ts","../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,lodash/prefer-lodash-method */\nimport { ChangeDetectionStrategy, Component, Inject, Input, OnInit, Optional } from '@angular/core';\nimport { ScCatalogService, ScProduct } from '@snabcentr/client-core';\nimport { TuiDay, TuiDayRange, TuiMonth, tuiPure, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TUI_MONTHS, TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { filter, map, Observable, shareReplay } from 'rxjs';\n\n/**\n * График истории цен товара или услуги.\n */\n@Component({\n    selector: 'sc-price-history',\n    templateUrl: './sc-price-history.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceHistoryComponent implements OnInit {\n    /**\n     * {@link Observable} временного промежутка истории цен.\n     */\n    public range$: Observable<TuiDayRange>;\n\n    /**\n     * Товар или услуга, для которого необходимо отобразить историю цен.\n     */\n    @Input() public product: ScProduct;\n\n    /**\n     * {@link Observable} истории цен.\n     */\n    public history$?: Observable<ReadonlyArray<[TuiDay, number]>>;\n\n    /**\n     * Максимальная цена товара.\n     */\n    public priceInfo$?: Observable<{\n        min: number;\n        max: number;\n    }>;\n\n    /**\n     * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.\n     */\n    public readonly xStringify$: Observable<TuiStringHandler<TuiDay>> = this.months$.pipe(\n        map(\n            (months) =>\n                ({ month, day }) =>\n                    // eslint-disable-next-line security/detect-object-injection\n                    `${months[month]}, ${day}`\n        )\n    );\n\n    /**\n     * Функция для преобразования значения number в строку в подсказке по оси Y.\n     *\n     * @param value Значение оси Y.\n     */\n    public readonly yStringify: TuiStringHandler<number> = (value) => `${value.toLocaleString('ru-RU')} ₽`;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.\n     *\n     * @param catalogService Сервис для работы с каталогом.\n     * @param months$ Перечисление месяцев.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly catalogService: ScCatalogService,\n        @Inject(TUI_MONTHS) private readonly months$: Observable<readonly string[]>,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        private readonly context?: TuiDialogContext<void, { product: ScProduct }>\n    ) {\n        if (context) {\n            this.product = context.data.product;\n        }\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.history$ = this.catalogService.getPriceHistory$(this.product).pipe(\n            map((history) =>\n                Object.keys(history)\n                    .map((item): [TuiDay, number] => [TuiDay.normalizeParse(item, 'DMY'), history[String(item)].cost])\n                    .sort((a, b) => (a[0] > b[0] ? 1 : -1))\n            ),\n            shareReplay({ bufferSize: 1, refCount: true }),\n            filter((history) => history.length > 1)\n        );\n\n        this.priceInfo$ = this.history$.pipe(\n            map((history) => history.map((item) => item[1])),\n            map((values) => ({\n                min: Math.min(...values),\n                max: Math.max(...values),\n            }))\n        );\n\n        this.range$ = this.history$.pipe(\n            map((value: ReadonlyArray<[TuiDay, number]>) => ({ from: value.at(0)?.[0], to: value.at(-1)?.[0] })),\n            filter((range): range is { from: TuiDay; to: TuiDay } => !!range.from && !!range.to),\n            map((range) => {\n                return new TuiDayRange(range.from, range.to);\n            })\n        );\n    }\n\n    /**\n     * Вычисляет подписи даты к оси X.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     */\n    @tuiPure\n    public computeLabels$({ from, to }: TuiDayRange): Observable<readonly string[]> {\n        return this.months$.pipe(map((months) => Array.from({ length: TuiMonth.lengthBetween(from, to) + 1 }, (_, index) => months[from.append({ month: index }).month])));\n    }\n\n    /**\n     * Вычисляет данные для отображения на графике.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     * @param history История цен.\n     */\n    @tuiPure\n    public computeValue({ from, to }: TuiDayRange, history: ReadonlyArray<[TuiDay, number]>): ReadonlyArray<[TuiDay, number]> {\n        return (\n            Array.from({ length: TuiDay.lengthBetween(from, to) + 1 })\n                .fill(0)\n                // eslint-disable-next-line unicorn/no-array-reduce\n                .reduce<ReadonlyArray<[TuiDay, number]>>((array, _, index) => {\n                    const data = from.append({ day: index });\n                    return [\n                        ...array,\n                        [\n                            data,\n                            (history.find((valueHistory, indexHistory, arrayHistory) => data.daySame(valueHistory[0]) || data.dayBefore(arrayHistory[indexHistory + 1][0])) ??\n                                history[0])[1],\n                        ],\n                    ];\n                }, [])\n        );\n    }\n}\n","<div class=\"flex flex-col items-center\">\n    <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n        <div class=\"w-174 mb-1 text-end text-lg font-bold\">от {{ priceInfo.min.toLocaleString() }} ₽ до {{ priceInfo.max.toLocaleString() }} ₽</div>\n\n        <ng-container *ngIf=\"history$ | async as history\">\n            <div\n                *ngIf=\"range$ | async as range\"\n                class=\"relative h-56 w-full p-5 text-tui-primary\"\n            >\n                <tui-axes\n                    *ngIf=\"computeLabels$(range) | async as labels\"\n                    class=\"h-full\"\n                    [axisXLabels]=\"labels\"\n                    [horizontalLines]=\"4\"\n                    [verticalLines]=\"labels.length\"\n                >\n                    <tui-line-days-chart\n                        class=\"chart\"\n                        [height]=\"priceInfo.max\"\n                        [smoothingFactor]=\"10\"\n                        [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n                        [value]=\"computeValue(range, history)\"\n                        [xStringify]=\"xStringify$ | async\"\n                        [yStringify]=\"yStringify\"\n                        [hintContent]=\"hint\"\n                    ></tui-line-days-chart>\n                </tui-axes>\n            </div>\n        </ng-container>\n    </ng-container>\n    <ng-template #isNotExist>\n        <p class=\"text-base font-bold\">Цена на товар не изменялась</p>\n    </ng-template>\n    <ng-template\n        #hint\n        let-data\n    >\n        <div class=\"font-bold\">{{ data[1].toLocaleString() }} ₽</div>\n        <div>{{ data[0] }}</div>\n    </ng-template>\n</div>\n"]}
@@ -2,27 +2,19 @@ import { CommonModule } from '@angular/common';
2
2
  import { NgModule } from '@angular/core';
3
3
  import { FormsModule, ReactiveFormsModule } from '@angular/forms';
4
4
  import { RouterModule } from '@angular/router';
5
+ import { TuiAxesModule, TuiLineDaysChartModule } from '@taiga-ui/addon-charts';
5
6
  import { TuiLetModule, TuiRepeatTimesModule } from '@taiga-ui/cdk';
6
7
  import { TuiButtonModule, TuiHintModule, TuiLabelModule, TuiLinkModule, TuiLoaderModule, TuiModeModule, TuiSvgModule, TuiTextfieldControllerModule } from '@taiga-ui/core';
7
8
  import { TuiElasticContainerModule, TuiFieldErrorPipeModule, TuiHighlightModule, TuiInputNumberModule, TuiIslandModule } from '@taiga-ui/kit';
8
- import { LineChart } from 'echarts/charts';
9
- import { GridComponent, TitleComponent, TooltipComponent } from 'echarts/components';
10
- import * as echarts from 'echarts/core';
11
- import { SVGRenderer } from 'echarts/renderers';
12
- import { NgxEchartsModule } from 'ngx-echarts';
13
9
  import { ScCategoriesListComponent } from './categories-list/sc-categories-list.component';
14
10
  import { ScCategoryCardComponent } from './category-card/sc-category-card.component';
15
11
  import { ScInputQuantityComponent } from './input-quantity/sc-input-quantity.component';
16
12
  import { ScPriceCardComponent } from './price-card/sc-price-card.component';
17
- import scLangRU from './price-history/sc-lang-ru';
18
13
  import { ScPriceHistoryComponent } from './price-history/sc-price-history.component';
19
14
  import { ScPriceListPaginationComponent } from './price-list-pagination/sc-price-list-pagination.component';
20
15
  import { ScPriceWarehouseStockComponent } from './price-warehouse-stock/sc-price-warehouse-stock.component';
21
16
  import { ScFavoriteBtnComponent as ScFavoriteButtonComponent } from './sc-favorite-btn/sc-favorite-btn.component';
22
17
  import * as i0 from "@angular/core";
23
- import * as i1 from "ngx-echarts";
24
- echarts.registerLocale('RU', scLangRU);
25
- echarts.use([TitleComponent, TooltipComponent, GridComponent, LineChart, SVGRenderer]);
26
18
  /**
27
19
  * Модуль каталога.
28
20
  */
@@ -51,9 +43,12 @@ ScCatalogModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version
51
43
  TuiFieldErrorPipeModule,
52
44
  TuiLoaderModule,
53
45
  TuiLinkModule,
54
- TuiElasticContainerModule, i1.NgxEchartsModule, TuiLetModule,
46
+ TuiElasticContainerModule,
47
+ TuiLetModule,
55
48
  TuiRepeatTimesModule,
56
- TuiHighlightModule], exports: [ScPriceListPaginationComponent,
49
+ TuiHighlightModule,
50
+ TuiLineDaysChartModule,
51
+ TuiAxesModule], exports: [ScPriceListPaginationComponent,
57
52
  ScCategoryCardComponent,
58
53
  ScFavoriteButtonComponent,
59
54
  ScInputQuantityComponent,
@@ -77,10 +72,11 @@ ScCatalogModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version
77
72
  TuiLoaderModule,
78
73
  TuiLinkModule,
79
74
  TuiElasticContainerModule,
80
- NgxEchartsModule.forRoot({ echarts }),
81
75
  TuiLetModule,
82
76
  TuiRepeatTimesModule,
83
- TuiHighlightModule] });
77
+ TuiHighlightModule,
78
+ TuiLineDaysChartModule,
79
+ TuiAxesModule] });
84
80
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScCatalogModule, decorators: [{
85
81
  type: NgModule,
86
82
  args: [{
@@ -121,11 +117,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
121
117
  TuiLoaderModule,
122
118
  TuiLinkModule,
123
119
  TuiElasticContainerModule,
124
- NgxEchartsModule.forRoot({ echarts }),
125
120
  TuiLetModule,
126
121
  TuiRepeatTimesModule,
127
122
  TuiHighlightModule,
123
+ TuiLineDaysChartModule,
124
+ TuiAxesModule,
128
125
  ],
129
126
  }]
130
127
  }] });
131
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2F0YWxvZy5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9zYy1jYXRhbG9nLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxZQUFZLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkUsT0FBTyxFQUFFLGVBQWUsRUFBRSxhQUFhLEVBQUUsY0FBYyxFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLFlBQVksRUFBRSw0QkFBNEIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzNLLE9BQU8sRUFBRSx5QkFBeUIsRUFBRSx1QkFBdUIsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxlQUFlLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDOUksT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzNDLE9BQU8sRUFBRSxhQUFhLEVBQUUsY0FBYyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDckYsT0FBTyxLQUFLLE9BQU8sTUFBTSxjQUFjLENBQUM7QUFDeEMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQ2hELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUUvQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxnREFBZ0QsQ0FBQztBQUMzRixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFB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QUFvQjt3QkFDcEIsa0JBQWtCO3FCQUNyQjtpQkFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBSb3V0ZXJNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgVHVpTGV0TW9kdWxlLCBUdWlSZXBlYXRUaW1lc01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jZGsnO1xuaW1wb3J0IHsgVHVpQnV0dG9uTW9kdWxlLCBUdWlIaW50TW9kdWxlLCBUdWlMYWJlbE1vZHVsZSwgVHVpTGlua01vZHVsZSwgVHVpTG9hZGVyTW9kdWxlLCBUdWlNb2RlTW9kdWxlLCBUdWlTdmdNb2R1bGUsIFR1aVRleHRmaWVsZENvbnRyb2xsZXJNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY29yZSc7XG5pbXBvcnQgeyBUdWlFbGFzdGljQ29udGFpbmVyTW9kdWxlLCBUdWlGaWVsZEVycm9yUGlwZU1vZHVsZSwgVHVpSGlnaGxpZ2h0TW9kdWxlLCBUdWlJbnB1dE51bWJlck1vZHVsZSwgVHVpSXNsYW5kTW9kdWxlIH0gZnJvbSAnQHRhaWdhLXVpL2tpdCc7XG5pbXBvcnQgeyBMaW5lQ2hhcnQgfSBmcm9tICdlY2hhcnRzL2NoYXJ0cyc7XG5pbXBvcnQgeyBHcmlkQ29tcG9uZW50LCBUaXRsZUNvbXBvbmVudCwgVG9vbHRpcENvbXBvbmVudCB9IGZyb20gJ2VjaGFydHMvY29tcG9uZW50cyc7XG5pbXBvcnQgKiBhcyBlY2hhcnRzIGZyb20gJ2VjaGFydHMvY29yZSc7XG5pbXBvcnQgeyBTVkdSZW5kZXJlciB9IGZyb20gJ2VjaGFydHMvcmVuZGVyZXJzJztcbmltcG9ydCB7IE5neEVjaGFydHNNb2R1bGUgfSBmcm9tICduZ3gtZWNoYXJ0cyc7XG5cbmltcG9ydCB7IFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQgfSBmcm9tICcuL2NhdGVnb3JpZXMtbGlzdC9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IFNjQ2F0ZWdvcnlDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9jYXRlZ29yeS1jYXJkL3NjLWNhdGVnb3J5LWNhcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCB9IGZyb20gJy4vaW5wdXQtcXVhbnRpdHkvc2MtaW5wdXQtcXVhbnRpdHkuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS1jYXJkL3NjLXByaWNlLWNhcmQuY29tcG9uZW50JztcbmltcG9ydCBzY0xhbmdSVSBmcm9tICcuL3ByaWNlLWhpc3Rvcnkvc2MtbGFuZy1ydSc7XG5pbXBvcnQgeyBTY1ByaWNlSGlzdG9yeUNvbXBvbmVudCB9IGZyb20gJy4vcHJpY2UtaGlzdG9yeS9zYy1wcmljZS1oaXN0b3J5LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTY1ByaWNlTGlzdFBhZ2luYXRpb25Db21wb25lbnQgfSBmcm9tICcuL3ByaWNlLWxpc3QtcGFnaW5hdGlvbi9zYy1wcmljZS1saXN0LXBhZ2luYXRpb24uY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VXYXJlaG91c2VTdG9ja0NvbXBvbmVudCB9IGZyb20gJy4vcHJpY2Utd2FyZWhvdXNlLXN0b2NrL3NjLXByaWNlLXdhcmVob3VzZS1zdG9jay5jb21wb25lbnQnO1xuaW1wb3J0IHsgU2NGYXZvcml0ZUJ0bkNvbXBvbmVudCBhcyBTY0Zhdm9yaXRlQnV0dG9uQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1mYXZvcml0ZS1idG4vc2MtZmF2b3JpdGUtYnRuLmNvbXBvbmVudCc7XG5cbmVjaGFydHMucmVnaXN0ZXJMb2NhbGUoJ1JVJywgc2NMYW5nUlUpO1xuZWNoYXJ0cy51c2UoW1RpdGxlQ29tcG9uZW50LCBUb29sdGlwQ29tcG9uZW50LCBHcmlkQ29tcG9uZW50LCBMaW5lQ2hhcnQsIFNWR1JlbmRlcmVyXSk7XG5cbi8qKlxuICog0JzQvtC00YPQu9GMINC60LDRgtCw0LvQvtCz0LAuXG4gKi9cbkBOZ01vZHVsZSh7XG4gICAgZGVjbGFyYXRpb25zOiBbXG4gICAgICAgIFNjUHJpY2VMaXN0UGFnaW5hdGlvbkNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yeUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjRmF2b3JpdGVCdXR0b25Db21wb25lbnQsXG4gICAgICAgIFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjUHJpY2VXYXJlaG91c2VTdG9ja0NvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUhpc3RvcnlDb21wb25lbnQsXG4gICAgICAgIFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQsXG4gICAgXSxcbiAgICBleHBvcnRzOiBbXG4gICAgICAgIFNjUHJpY2VMaXN0UGFnaW5hdGlvbkNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yeUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjRmF2b3JpdGVCdXR0b25Db21wb25lbnQsXG4gICAgICAgIFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjUHJpY2VXYXJlaG91c2VTdG9ja0NvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUhpc3RvcnlDb21wb25lbnQsXG4gICAgICAgIFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQsXG4gICAgXSxcbiAgICBpbXBvcnRzOiBbXG4gICAgICAgIENvbW1vbk1vZHVsZSxcbiAgICAgICAgUm91dGVyTW9kdWxlLFxuICAgICAgICBUdWlCdXR0b25Nb2R1bGUsXG4gICAgICAgIFR1aVN2Z01vZHVsZSxcbiAgICAgICAgVHVpSXNsYW5kTW9kdWxlLFxuICAgICAgICBUdWlJbnB1dE51bWJlck1vZHVsZSxcbiAgICAgICAgVHVpTGFiZWxNb2R1bGUsXG4gICAgICAgIFR1aVRleHRmaWVsZENvbnRyb2xsZXJNb2R1bGUsXG4gICAgICAgIEZvcm1zTW9kdWxlLFxuICAgICAgICBSZWFjdGl2ZUZvcm1zTW9kdWxlLFxuICAgICAgICBUdWlIaW50TW9kdWxlLFxuICAgICAgICBUdWlNb2RlTW9kdWxlLFxuICAgICAgICBUdWlGaWVsZEVycm9yUGlwZU1vZHVsZSxcbiAgICAgICAgVHVpTG9hZGVyTW9kdWxlLFxuICAgICAgICBUdWlMaW5rTW9kdWxlLFxuICAgICAgICBUdWlFbGFzdGljQ29udGFpbmVyTW9kdWxlLFxuICAgICAgICBOZ3hFY2hhcnRzTW9kdWxlLmZvclJvb3QoeyBlY2hhcnRzIH0pLFxuICAgICAgICBUdWlMZXRNb2R1bGUsXG4gICAgICAgIFR1aVJlcGVhdFRpbWVzTW9kdWxlLFxuICAgICAgICBUdWlIaWdobGlnaHRNb2R1bGUsXG4gICAgXSxcbn0pXG5leHBvcnQgY2xhc3MgU2NDYXRhbG9nTW9kdWxlIHt9XG4iXX0=
128
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2F0YWxvZy5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9zYy1jYXRhbG9nLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxhQUFhLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUMvRSxPQUFPLEVBQUUsWUFBWSxFQUFFLG9CQUFvQixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25FLE9BQU8sRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLGNBQWMsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLGFBQWEsRUFBRSxZQUFZLEVBQUUsNEJBQTRCLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMzSyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsdUJBQXVCLEVBQUUsa0JBQWtCLEVBQUUsb0JBQW9CLEVBQUUsZUFBZSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRTlJLE9BQU8sRUFBRSx5QkFBeUIsRUFBRSxNQUFNLGdEQUFnRCxDQUFDO0FBQzNGLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLDRDQUE0QyxDQUFDO0FBQ3JGLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLDhDQUE4QyxDQUFDO0FBQ3hGLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHNDQUFzQyxDQUFDO0FBQzVFLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLDRDQUE0QyxDQUFDO0FBQ3JGLE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLDREQUE0RCxDQUFDO0FBQzVHLE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLDREQUE0RCxDQUFDO0FBQzVHLE9BQU8sRUFBRSxzQkFBc0IsSUFBSSx5QkFBeUIsRUFBRSxNQUFNLDZDQUE2QyxDQUFDOztBQUVsSDs7R0FFRztBQThDSCxNQUFNLE9BQU8sZUFBZTs7NEdBQWYsZUFBZTs2R0FBZixlQUFlLGlCQTNDcEIsOEJBQThCO1FBQzlCLHVCQUF1QjtRQUN2Qix5QkFBeUI7UUFDekIsd0JBQXdCO1FBQ3hCLG9CQUFvQjtRQUNwQiw4QkFBOEI7UUFDOUIsdUJBQXVCO1FBQ3ZCLHlCQUF5QixhQWF6QixZQUFZO1FBQ1osWUFBWTtRQUNaLGVBQWU7UUFDZixZQUFZO1FBQ1osZUFBZTtRQUNmLG9CQUFvQjtRQUNwQixjQUFjO1FBQ2QsNEJBQTRCO1FBQzVCLFdBQVc7UUFDWCxtQkFBbUI7UUFDbkIsYUFBYTtRQUNiLGFBQWE7UUFDYix1QkFBdUI7UUFDdkIsZUFBZTtRQUNmLGFBQWE7UUFDYix5QkFBeUI7UUFDekIsWUFBWTtRQUNaLG9CQUFvQjtRQUNwQixrQkFBa0I7UUFDbEIsc0JBQXNCO1FBQ3RCLGFBQWEsYUE5QmIsOEJBQThCO1FBQzlCLHVCQUF1QjtRQUN2Qix5QkFBeUI7UUFDekIsd0JBQXdCO1FBQ3hCLG9CQUFvQjtRQUNwQiw4QkFBOEI7UUFDOUIsdUJBQXVCO1FBQ3ZCLHlCQUF5Qjs2R0EwQnBCLGVBQWUsWUF2QnBCLFlBQVk7UUFDWixZQUFZO1FBQ1osZUFBZTtRQUNmLFlBQVk7UUFDWixlQUFlO1FBQ2Ysb0JBQW9CO1FBQ3BCLGNBQWM7UUFDZCw0QkFBNEI7UUFDNUIsV0FBVztRQUNYLG1CQUFtQjtRQUNuQixhQUFhO1FBQ2IsYUFBYTtRQUNiLHVCQUF1QjtRQUN2QixlQUFlO1FBQ2YsYUFBYTtRQUNiLHlCQUF5QjtRQUN6QixZQUFZO1FBQ1osb0JBQW9CO1FBQ3BCLGtCQUFrQjtRQUNsQixzQkFBc0I7UUFDdEIsYUFBYTsyRkFHUixlQUFlO2tCQTdDM0IsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUU7d0JBQ1YsOEJBQThCO3dCQUM5Qix1QkFBdUI7d0JBQ3ZCLHlCQUF5Qjt3QkFDekIsd0JBQXdCO3dCQUN4QixvQkFBb0I7d0JBQ3BCLDhCQUE4Qjt3QkFDOUIsdUJBQXVCO3dCQUN2Qix5QkFBeUI7cUJBQzVCO29CQUNELE9BQU8sRUFBRTt3QkFDTCw4QkFBOEI7d0JBQzlCLHVCQUF1Qjt3QkFDdkIseUJBQXlCO3dCQUN6Qix3QkFBd0I7d0JBQ3hCLG9CQUFvQjt3QkFDcEIsOEJBQThCO3dCQUM5Qix1QkFBdUI7d0JBQ3ZCLHlCQUF5QjtxQkFDNUI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNMLFlBQVk7d0JBQ1osWUFBWTt3QkFDWixlQUFlO3dCQUNmLFlBQVk7d0JBQ1osZUFBZTt3QkFDZixvQkFBb0I7d0JBQ3BCLGNBQWM7d0JBQ2QsNEJBQTRCO3dCQUM1QixXQUFXO3dCQUNYLG1CQUFtQjt3QkFDbkIsYUFBYTt3QkFDYixhQUFhO3dCQUNiLHVCQUF1Qjt3QkFDdkIsZUFBZTt3QkFDZixhQUFhO3dCQUNiLHlCQUF5Qjt3QkFDekIsWUFBWTt3QkFDWixvQkFBb0I7d0JBQ3BCLGtCQUFrQjt3QkFDbEIsc0JBQXNCO3dCQUN0QixhQUFhO3FCQUNoQjtpQkFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBSb3V0ZXJNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgVHVpQXhlc01vZHVsZSwgVHVpTGluZURheXNDaGFydE1vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9hZGRvbi1jaGFydHMnO1xuaW1wb3J0IHsgVHVpTGV0TW9kdWxlLCBUdWlSZXBlYXRUaW1lc01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jZGsnO1xuaW1wb3J0IHsgVHVpQnV0dG9uTW9kdWxlLCBUdWlIaW50TW9kdWxlLCBUdWlMYWJlbE1vZHVsZSwgVHVpTGlua01vZHVsZSwgVHVpTG9hZGVyTW9kdWxlLCBUdWlNb2RlTW9kdWxlLCBUdWlTdmdNb2R1bGUsIFR1aVRleHRmaWVsZENvbnRyb2xsZXJNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY29yZSc7XG5pbXBvcnQgeyBUdWlFbGFzdGljQ29udGFpbmVyTW9kdWxlLCBUdWlGaWVsZEVycm9yUGlwZU1vZHVsZSwgVHVpSGlnaGxpZ2h0TW9kdWxlLCBUdWlJbnB1dE51bWJlck1vZHVsZSwgVHVpSXNsYW5kTW9kdWxlIH0gZnJvbSAnQHRhaWdhLXVpL2tpdCc7XG5cbmltcG9ydCB7IFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQgfSBmcm9tICcuL2NhdGVnb3JpZXMtbGlzdC9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IFNjQ2F0ZWdvcnlDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9jYXRlZ29yeS1jYXJkL3NjLWNhdGVnb3J5LWNhcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCB9IGZyb20gJy4vaW5wdXQtcXVhbnRpdHkvc2MtaW5wdXQtcXVhbnRpdHkuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS1jYXJkL3NjLXByaWNlLWNhcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VIaXN0b3J5Q29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS1oaXN0b3J5L3NjLXByaWNlLWhpc3RvcnkuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VMaXN0UGFnaW5hdGlvbkNvbXBvbmVudCB9IGZyb20gJy4vcHJpY2UtbGlzdC1wYWdpbmF0aW9uL3NjLXByaWNlLWxpc3QtcGFnaW5hdGlvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgU2NQcmljZVdhcmVob3VzZVN0b2NrQ29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS13YXJlaG91c2Utc3RvY2svc2MtcHJpY2Utd2FyZWhvdXNlLXN0b2NrLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTY0Zhdm9yaXRlQnRuQ29tcG9uZW50IGFzIFNjRmF2b3JpdGVCdXR0b25Db21wb25lbnQgfSBmcm9tICcuL3NjLWZhdm9yaXRlLWJ0bi9zYy1mYXZvcml0ZS1idG4uY29tcG9uZW50JztcblxuLyoqXG4gKiDQnNC+0LTRg9C70Ywg0LrQsNGC0LDQu9C+0LPQsC5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtcbiAgICAgICAgU2NQcmljZUxpc3RQYWdpbmF0aW9uQ29tcG9uZW50LFxuICAgICAgICBTY0NhdGVnb3J5Q2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NGYXZvcml0ZUJ1dHRvbkNvbXBvbmVudCxcbiAgICAgICAgU2NJbnB1dFF1YW50aXR5Q29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlQ2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZVdhcmVob3VzZVN0b2NrQ29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlSGlzdG9yeUNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yaWVzTGlzdENvbXBvbmVudCxcbiAgICBdLFxuICAgIGV4cG9ydHM6IFtcbiAgICAgICAgU2NQcmljZUxpc3RQYWdpbmF0aW9uQ29tcG9uZW50LFxuICAgICAgICBTY0NhdGVnb3J5Q2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NGYXZvcml0ZUJ1dHRvbkNvbXBvbmVudCxcbiAgICAgICAgU2NJbnB1dFF1YW50aXR5Q29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlQ2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZVdhcmVob3VzZVN0b2NrQ29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlSGlzdG9yeUNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yaWVzTGlzdENvbXBvbmVudCxcbiAgICBdLFxuICAgIGltcG9ydHM6IFtcbiAgICAgICAgQ29tbW9uTW9kdWxlLFxuICAgICAgICBSb3V0ZXJNb2R1bGUsXG4gICAgICAgIFR1aUJ1dHRvbk1vZHVsZSxcbiAgICAgICAgVHVpU3ZnTW9kdWxlLFxuICAgICAgICBUdWlJc2xhbmRNb2R1bGUsXG4gICAgICAgIFR1aUlucHV0TnVtYmVyTW9kdWxlLFxuICAgICAgICBUdWlMYWJlbE1vZHVsZSxcbiAgICAgICAgVHVpVGV4dGZpZWxkQ29udHJvbGxlck1vZHVsZSxcbiAgICAgICAgRm9ybXNNb2R1bGUsXG4gICAgICAgIFJlYWN0aXZlRm9ybXNNb2R1bGUsXG4gICAgICAgIFR1aUhpbnRNb2R1bGUsXG4gICAgICAgIFR1aU1vZGVNb2R1bGUsXG4gICAgICAgIFR1aUZpZWxkRXJyb3JQaXBlTW9kdWxlLFxuICAgICAgICBUdWlMb2FkZXJNb2R1bGUsXG4gICAgICAgIFR1aUxpbmtNb2R1bGUsXG4gICAgICAgIFR1aUVsYXN0aWNDb250YWluZXJNb2R1bGUsXG4gICAgICAgIFR1aUxldE1vZHVsZSxcbiAgICAgICAgVHVpUmVwZWF0VGltZXNNb2R1bGUsXG4gICAgICAgIFR1aUhpZ2hsaWdodE1vZHVsZSxcbiAgICAgICAgVHVpTGluZURheXNDaGFydE1vZHVsZSxcbiAgICAgICAgVHVpQXhlc01vZHVsZSxcbiAgICBdLFxufSlcbmV4cG9ydCBjbGFzcyBTY0NhdGFsb2dNb2R1bGUge31cbiJdfQ==