@snabcentr/client-ui 0.11.4 → 0.14.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (105) hide show
  1. package/auth/sc-sign-in-form/sc-sign-in-form.component.d.ts +8 -4
  2. package/banner/sc-banner.component.d.ts +7 -3
  3. package/cart/cart-item-mobile/sc-cart-item-mobile.component.d.ts +1 -1
  4. package/contacts/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.d.ts +47 -0
  5. package/contacts/index.d.ts +2 -0
  6. package/contacts/new-contragent-bank-account-form/sc-new-contragent-bank-account-form.component.d.ts +55 -0
  7. package/contacts/sc-contacts.module.d.ts +11 -5
  8. package/contragents/contragents-accordion/contragents-accordion-item/sc-contragents-accordion-item.component.d.ts +2 -3
  9. package/contragents/contragents-accordion/sc-contragents-accordion.component.d.ts +1 -1
  10. package/esm2020/accordion/sc-accordion-content.directive.mjs +2 -2
  11. package/esm2020/accordion/sc-accordion.component.mjs +1 -1
  12. package/esm2020/accordion/sc-accordion.module.mjs +2 -2
  13. package/esm2020/auth/constants/phone-approve-code-mask.mjs +2 -2
  14. package/esm2020/auth/sc-auth.module.mjs +3 -3
  15. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +2 -2
  16. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +2 -2
  17. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form.component.mjs +14 -5
  18. package/esm2020/banner/sc-banner.component.mjs +17 -8
  19. package/esm2020/banner/sc-banner.module.mjs +2 -2
  20. package/esm2020/cart/cart-item-mobile/sc-cart-item-mobile.component.mjs +2 -2
  21. package/esm2020/cart/sc-cart.module.mjs +3 -3
  22. package/esm2020/catalog/categories-list/sc-categories-list.component.mjs +1 -1
  23. package/esm2020/catalog/category-card/sc-category-card.component.mjs +2 -2
  24. package/esm2020/catalog/input-quantity/sc-input-quantity.component.mjs +3 -3
  25. package/esm2020/catalog/price-card/sc-price-card.component.mjs +1 -1
  26. package/esm2020/catalog/price-history/sc-chart-option.mjs +18 -18
  27. package/esm2020/catalog/price-history/sc-lang-RU.mjs +25 -25
  28. package/esm2020/catalog/price-history/sc-price-history.component.mjs +2 -2
  29. package/esm2020/catalog/price-list-pagination/sc-price-list-pagination.component.mjs +1 -1
  30. package/esm2020/catalog/price-warehouse-stock/sc-price-warehouse-stock.component.mjs +1 -1
  31. package/esm2020/catalog/sc-catalog.module.mjs +5 -5
  32. package/esm2020/catalog/sc-favorite-btn/sc-favorite-btn.component.mjs +1 -1
  33. package/esm2020/contacts/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.mjs +72 -0
  34. package/esm2020/contacts/contacts-accordion/sc-contacts-accordion.component.mjs +1 -1
  35. package/esm2020/contacts/index.mjs +3 -1
  36. package/esm2020/contacts/new-contragent-bank-account-form/sc-new-contragent-bank-account-form.component.mjs +61 -0
  37. package/esm2020/contacts/sc-contacts.module.mjs +75 -7
  38. package/esm2020/contragents/contragents-accordion/contragents-accordion-item/sc-contragents-accordion-item.component.mjs +3 -3
  39. package/esm2020/contragents/contragents-accordion/sc-contragents-accordion.component.mjs +3 -3
  40. package/esm2020/contragents/sc-contragents.module.mjs +2 -2
  41. package/esm2020/delivery-address/delivery-address-accordion/delivery-address-accordion-item/sc-delivery-address-accordion-item.component.mjs +1 -1
  42. package/esm2020/delivery-address/delivery-address-accordion/sc-delivery-address-accordion.component.mjs +1 -1
  43. package/esm2020/delivery-address/sc-delivery-address.module.mjs +2 -2
  44. package/esm2020/files/directives/tree-top.directive.mjs +8 -8
  45. package/esm2020/files/directives/tree.directive.mjs +2 -2
  46. package/esm2020/files/file-tree-item/file-tree-item.component.mjs +2 -2
  47. package/esm2020/files/files-and-documents.component.mjs +6 -6
  48. package/esm2020/files/files-and-documents.module.mjs +2 -2
  49. package/esm2020/files/services/tree-icon.service.mjs +3 -3
  50. package/esm2020/files/services/tree-loader.service.mjs +2 -2
  51. package/esm2020/form-fields/addresses-selection-field/sc-addresses-selection-field.component.mjs +21 -16
  52. package/esm2020/form-fields/form-fields.module.mjs +4 -4
  53. package/esm2020/form-fields/suggestion-field/sc-suggestion-field.component.mjs +5 -5
  54. package/esm2020/helpers/sc-px-converter.mjs +2 -2
  55. package/esm2020/helpers/sc-units-helper.mjs +2 -2
  56. package/esm2020/icons/sc-client-ui-icons-name.mjs +2 -2
  57. package/esm2020/masks/index.mjs +4 -0
  58. package/esm2020/masks/sc-account-number-mask.mjs +7 -0
  59. package/esm2020/masks/sc-bic-mask.mjs +7 -0
  60. package/esm2020/masks/sc-correspondent-account-mask.mjs +7 -0
  61. package/esm2020/news/news-card/sc-news-card.component.mjs +1 -1
  62. package/esm2020/news/news-card-skeleton/sc-news-card-skeleton.component.mjs +1 -1
  63. package/esm2020/news/sc-news.module.mjs +2 -2
  64. package/esm2020/order/index.mjs +2 -1
  65. package/esm2020/order/order-item-mobile/order-item-mobile.component.mjs +1 -1
  66. package/esm2020/order/sc-order.module.mjs +8 -6
  67. package/esm2020/order/sc-payment-status/sc-payment-status.component.mjs +35 -0
  68. package/esm2020/profile/sc-profile-accordions-content/sc-profile-accordions-content.component.mjs +9 -3
  69. package/esm2020/profile/sc-profile.module.mjs +3 -3
  70. package/esm2020/providers/productsPagination.mjs +11 -10
  71. package/esm2020/providers/scUserProviders.mjs +3 -3
  72. package/esm2020/share-button/sc-share-button.component.mjs +1 -1
  73. package/esm2020/share-button/sc-share-button.module.mjs +2 -2
  74. package/esm2020/tokens/sc-linear-values-token.mjs +2 -2
  75. package/esm2020/user/reset-user-password/sc-reset-user-password.component.mjs +5 -5
  76. package/esm2020/user/sc-user.module.mjs +3 -3
  77. package/esm2020/user/update-user-info-dialog/sc-update-user-info-dialog.component.mjs +4 -4
  78. package/esm2020/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.mjs +3 -3
  79. package/esm2020/validators/index.mjs +3 -1
  80. package/esm2020/validators/scBicValidator.mjs +7 -0
  81. package/esm2020/validators/scCorrespondentAccountValidator.mjs +7 -0
  82. package/esm2020/validators/scPasswordConfirmMatchingValidator.mjs +8 -9
  83. package/esm2020/verification/sc-verification.module.mjs +3 -3
  84. package/esm2020/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +7 -7
  85. package/fesm2015/snabcentr-client-ui.mjs +878 -609
  86. package/fesm2015/snabcentr-client-ui.mjs.map +1 -1
  87. package/fesm2020/snabcentr-client-ui.mjs +875 -609
  88. package/fesm2020/snabcentr-client-ui.mjs.map +1 -1
  89. package/files/services/tree-icon.service.d.ts +2 -2
  90. package/form-fields/addresses-selection-field/sc-addresses-selection-field.component.d.ts +1 -1
  91. package/masks/index.d.ts +3 -0
  92. package/masks/sc-account-number-mask.d.ts +5 -0
  93. package/masks/sc-bic-mask.d.ts +5 -0
  94. package/masks/sc-correspondent-account-mask.d.ts +5 -0
  95. package/order/index.d.ts +1 -0
  96. package/order/sc-order.module.d.ts +6 -4
  97. package/order/sc-payment-status/sc-payment-status.component.d.ts +25 -0
  98. package/package.json +1 -1
  99. package/profile/sc-profile-accordions-content/sc-profile-accordions-content.component.d.ts +5 -1
  100. package/providers/productsPagination.d.ts +3 -3
  101. package/styles/tailwind/tailwind.scss +55 -8
  102. package/validators/index.d.ts +2 -0
  103. package/validators/scBicValidator.d.ts +5 -0
  104. package/validators/scCorrespondentAccountValidator.d.ts +5 -0
  105. package/validators/scPasswordConfirmMatchingValidator.d.ts +2 -1
@@ -1,5 +1,6 @@
1
1
  import { __decorate } from "tslib";
2
- import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter, ContentChildren, ViewChild, ElementRef, HostBinding, Inject, HostListener, } from '@angular/core';
2
+ import { Component, ChangeDetectionStrategy, Input, Output, EventEmitter, ContentChildren, ViewChild, ElementRef, HostBinding, Inject, HostListener } from '@angular/core';
3
+ import { ScUserMetrikaGoalsEnum } from '@snabcentr/client-core';
3
4
  import { map, tap, shareReplay, filter, switchMap, interval, takeUntil, Subject } from 'rxjs';
4
5
  import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
5
6
  import { IntersectionObserverService } from '@ng-web-apis/intersection-observer';
@@ -22,14 +23,18 @@ let ScBannerComponent = class ScBannerComponent {
22
23
  *
23
24
  * @param cdr Объект для работы с обнаружением изменений.
24
25
  * @param bannerService Сервис для работы с баннерами.
25
- * @param pxConverter Класс хэлпер для конвертации пикселей.
26
+ * @param entries$
27
+ * @param element
28
+ * @param pxConverter Экземпляр класса-помощника для конвертации пикселей.
29
+ * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.
26
30
  */
27
- constructor(cdr, bannerService, entries$, element, pxConverter) {
31
+ constructor(cdr, bannerService, entries$, element, pxConverter, userMetrikaService) {
28
32
  this.cdr = cdr;
29
33
  this.bannerService = bannerService;
30
34
  this.entries$ = entries$;
31
35
  this.element = element;
32
36
  this.pxConverter = pxConverter;
37
+ this.userMetrikaService = userMetrikaService;
33
38
  /**
34
39
  * Признак, что необходимо показывать кнопки навигации.
35
40
  */
@@ -72,7 +77,7 @@ let ScBannerComponent = class ScBannerComponent {
72
77
  this.banners$ = this.bannerService.banners$.pipe(map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()), tap((banners) => {
73
78
  if (banners.length) {
74
79
  if (this.resizable) {
75
- this.width = `100%`;
80
+ this.width = '100%';
76
81
  }
77
82
  else {
78
83
  this.height = `${this.pxConverter.pxToRem(banners[0].height)}rem`;
@@ -125,7 +130,7 @@ let ScBannerComponent = class ScBannerComponent {
125
130
  .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))
126
131
  .subscribe(() => this.onNextBanner());
127
132
  // Отслеживание пересечения компонента с экраном пользователя.
128
- // Если баннера не находится в поле видимости пользователя, то он перестаёт переключатся, а видео останавливается.
133
+ // Если баннера не находится в поле видимости пользователя, то он перестаёт переключаться, а видео останавливается.
129
134
  this.entries$.pipe(map((entries) => entries.find((item) => item.target === this.element.nativeElement))).subscribe((entry) => {
130
135
  this.toggleTimer$.next(!!entry?.isIntersecting);
131
136
  });
@@ -160,12 +165,16 @@ let ScBannerComponent = class ScBannerComponent {
160
165
  * @param banner Баннер, по ссылке которого совершён переход.
161
166
  */
162
167
  onClick(banner) {
168
+ this.userMetrikaService.emitUserMetrikaEvent({
169
+ target: ScUserMetrikaGoalsEnum.bannerClick,
170
+ params: { banner_id: banner.id }
171
+ });
163
172
  if (banner.url) {
164
173
  this.clickBannerEvent.emit(banner);
165
174
  }
166
175
  }
167
176
  };
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 });
177
+ 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 }, { token: i1.ScUserMetrikaService }], target: i0.ɵɵFactoryTarget.Component });
169
178
  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 });
170
179
  ScBannerComponent = __decorate([
171
180
  UntilDestroy({ checkProperties: true })
@@ -180,7 +189,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
180
189
  }] }, { type: i0.ElementRef, decorators: [{
181
190
  type: Inject,
182
191
  args: [ElementRef]
183
- }] }, { type: i2.ScPxConverter }]; }, propDecorators: { navigateButton: [{
192
+ }] }, { type: i2.ScPxConverter }, { type: i1.ScUserMetrikaService }]; }, propDecorators: { navigateButton: [{
184
193
  type: Input
185
194
  }], playerInterval: [{
186
195
  type: Input
@@ -219,4 +228,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
219
228
  type: HostListener,
220
229
  args: ['mouseleave']
221
230
  }] } });
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"]}
231
+ //# 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, ScUserMetrikaGoalsEnum, ScUserMetrikaService } 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 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(() => 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        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    /**\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"]}
@@ -21,7 +21,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
21
21
  args: [{
22
22
  declarations: [ScBannerComponent],
23
23
  imports: [CommonModule, RouterModule, TuiButtonModule, MatRippleModule, TuiCarouselModule, TuiModeModule, TuiLoaderModule, IntersectionObserverModule, TuiLetModule],
24
- exports: [ScBannerComponent],
24
+ exports: [ScBannerComponent]
25
25
  }]
26
26
  }] });
27
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtYmFubmVyLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9iYW5uZXIvc2MtYmFubmVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDakYsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3pELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNsRCxPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUNoRixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUU3Qzs7R0FFRztBQU1ILE1BQU0sT0FBTyxjQUFjOzsyR0FBZCxjQUFjOzRHQUFkLGNBQWMsaUJBSlIsaUJBQWlCLGFBQ3RCLFlBQVksRUFBRSxZQUFZLEVBQUUsZUFBZSxFQUFFLGVBQWUsRUFBRSxpQkFBaUIsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLDBCQUEwQixFQUFFLFlBQVksYUFDekosaUJBQWlCOzRHQUVsQixjQUFjLFlBSGIsWUFBWSxFQUFFLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLGlCQUFpQixFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsMEJBQTBCLEVBQUUsWUFBWTsyRkFHMUosY0FBYztrQkFMMUIsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztvQkFDakMsT0FBTyxFQUFFLENBQUMsWUFBWSxFQUFFLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLGlCQUFpQixFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsMEJBQTBCLEVBQUUsWUFBWSxDQUFDO29CQUNwSyxPQUFPLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztpQkFDL0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFNjQmFubmVyQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1iYW5uZXIuY29tcG9uZW50JztcbmltcG9ydCB7IFJvdXRlck1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBUdWlCdXR0b25Nb2R1bGUsIFR1aUxvYWRlck1vZHVsZSwgVHVpTW9kZU1vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7IE1hdFJpcHBsZU1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHsgVHVpQ2Fyb3VzZWxNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkva2l0JztcbmltcG9ydCB7IEludGVyc2VjdGlvbk9ic2VydmVyTW9kdWxlIH0gZnJvbSAnQG5nLXdlYi1hcGlzL2ludGVyc2VjdGlvbi1vYnNlcnZlcic7XG5pbXBvcnQgeyBUdWlMZXRNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY2RrJztcblxuLyoqXG4gKiDQnNC+0LTRg9C70Ywg0LHQsNC90L3QtdGA0L7Qsi5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtTY0Jhbm5lckNvbXBvbmVudF0sXG4gICAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgUm91dGVyTW9kdWxlLCBUdWlCdXR0b25Nb2R1bGUsIE1hdFJpcHBsZU1vZHVsZSwgVHVpQ2Fyb3VzZWxNb2R1bGUsIFR1aU1vZGVNb2R1bGUsIFR1aUxvYWRlck1vZHVsZSwgSW50ZXJzZWN0aW9uT2JzZXJ2ZXJNb2R1bGUsIFR1aUxldE1vZHVsZV0sXG4gICAgZXhwb3J0czogW1NjQmFubmVyQ29tcG9uZW50XSxcbn0pXG5leHBvcnQgY2xhc3MgU2NCYW5uZXJNb2R1bGUge31cbiJdfQ==
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtYmFubmVyLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9iYW5uZXIvc2MtYmFubmVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMxRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDakYsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3pELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNsRCxPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUNoRixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUU3Qzs7R0FFRztBQU1ILE1BQU0sT0FBTyxjQUFjOzsyR0FBZCxjQUFjOzRHQUFkLGNBQWMsaUJBSlIsaUJBQWlCLGFBQ3RCLFlBQVksRUFBRSxZQUFZLEVBQUUsZUFBZSxFQUFFLGVBQWUsRUFBRSxpQkFBaUIsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLDBCQUEwQixFQUFFLFlBQVksYUFDekosaUJBQWlCOzRHQUVsQixjQUFjLFlBSGIsWUFBWSxFQUFFLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLGlCQUFpQixFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsMEJBQTBCLEVBQUUsWUFBWTsyRkFHMUosY0FBYztrQkFMMUIsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztvQkFDakMsT0FBTyxFQUFFLENBQUMsWUFBWSxFQUFFLFlBQVksRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLGlCQUFpQixFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsMEJBQTBCLEVBQUUsWUFBWSxDQUFDO29CQUNwSyxPQUFPLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQztpQkFDL0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFNjQmFubmVyQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1iYW5uZXIuY29tcG9uZW50JztcbmltcG9ydCB7IFJvdXRlck1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQgeyBUdWlCdXR0b25Nb2R1bGUsIFR1aUxvYWRlck1vZHVsZSwgVHVpTW9kZU1vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7IE1hdFJpcHBsZU1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHsgVHVpQ2Fyb3VzZWxNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkva2l0JztcbmltcG9ydCB7IEludGVyc2VjdGlvbk9ic2VydmVyTW9kdWxlIH0gZnJvbSAnQG5nLXdlYi1hcGlzL2ludGVyc2VjdGlvbi1vYnNlcnZlcic7XG5pbXBvcnQgeyBUdWlMZXRNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY2RrJztcblxuLyoqXG4gKiDQnNC+0LTRg9C70Ywg0LHQsNC90L3QtdGA0L7Qsi5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtTY0Jhbm5lckNvbXBvbmVudF0sXG4gICAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgUm91dGVyTW9kdWxlLCBUdWlCdXR0b25Nb2R1bGUsIE1hdFJpcHBsZU1vZHVsZSwgVHVpQ2Fyb3VzZWxNb2R1bGUsIFR1aU1vZGVNb2R1bGUsIFR1aUxvYWRlck1vZHVsZSwgSW50ZXJzZWN0aW9uT2JzZXJ2ZXJNb2R1bGUsIFR1aUxldE1vZHVsZV0sXG4gICAgZXhwb3J0czogW1NjQmFubmVyQ29tcG9uZW50XVxufSlcbmV4cG9ydCBjbGFzcyBTY0Jhbm5lck1vZHVsZSB7fVxuIl19
@@ -20,7 +20,7 @@ export class ScCartItemMobileComponent {
20
20
  /**
21
21
  * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.
22
22
  *
23
- * @param unitsHelper Объект-хэлпер для работы со значениями единиц измерения товара.
23
+ * @param unitsHelper Экземпляр класса-помощника для работы со значениями единиц измерения товара.
24
24
  * @param previewDialogService Сервис диалогового окна предварительного просмотра.
25
25
  * @param urls Список ссылок на разделы backend'a.
26
26
  * @param pathImageNotFound Путь до изображения 'Товар не найден'.
@@ -117,4 +117,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
117
117
  }], clickCardEvent: [{
118
118
  type: Output
119
119
  }] } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-item-mobile.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.ts","../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAe,SAAS,EAAE,MAAM,eAAe,CAAC;AACxI,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAkC,MAAM,wBAAwB,CAAC;AAC1G,OAAO,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;AACxC,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;;;;;;;;;;;AAIlE;;GAEG;AAMH,MAAM,OAAO,yBAAyB;IAmElC;;;;;;;OAOG;IACH,YACoB,WAAwB,EAEvB,oBAA6C,EAC5B,IAAa,EACG,iBAAyB;QAJ3D,gBAAW,GAAX,WAAW,CAAa;QAEvB,yBAAoB,GAApB,oBAAoB,CAAyB;QAC5B,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QApE/E;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAgB,CAAC,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAqB7G;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzG;;WAEG;QAEI,qBAAgB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEvE;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAuBlE,CAAC;IAhEJ;;OAEG;IACH,IACW,QAAQ,CAAC,IAA4B;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,IAAI,IAAI,EAAE;YACN,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAClD;IACL,CAAC;IAED;;OAEG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IA0BD;;OAEG;IACH,IAAW,OAAO;QACd,OAAO,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;IAClC,CAAC;IAkBD,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;IAClE,CAAC;IAED;;OAEG;IACI,iBAAiB;QACpB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACnF,CAAC;IAED;;OAEG;IACI,YAAY;QACf,OAAO,IAAI,CAAC,OAAO,EAAE,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC;IACpH,CAAC;;sHApGQ,yBAAyB,6CA6EtB,uBAAuB,aAEvB,OAAO,aACP,uBAAuB;0GAhF1B,yBAAyB,2YChBtC,i7KAoFA;2FDpEa,yBAAyB;kBALrC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BA+E1C,MAAM;2BAAC,uBAAuB;;0BAE9B,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;4CAtElB,uBAAuB;sBADvC,SAAS;uBAAC,sBAAsB;gBAYtB,QAAQ;sBADlB,KAAK;gBAoBC,oBAAoB;sBAD1B,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAOA,kBAAkB;sBADxB,MAAM;gBAOA,cAAc;sBADpB,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, OnInit, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScCartItem, ScIUrls, ScProduct } from '@snabcentr/client-core';\nimport { Observable, skip } from 'rxjs';\nimport { TuiPreviewDialogService } from '@taiga-ui/addon-preview';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    selector: 'sc-cart-item-mobile',\n    templateUrl: './sc-cart-item-mobile.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCartItemMobileComponent implements OnInit {\n    /**\n     * Элемент корзины.\n     */\n    private _cartItem?: ScCartItem;\n\n    /**\n     * Ссылка на представление спецификации.\n     */\n    @ViewChild('specificationPreview')\n    private readonly specificationPreviewRef?: TemplateRef<TuiDialogContext>;\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(0, { updateOn: 'blur' });\n\n    /**\n     * Элемент корзины.\n     */\n    @Input()\n    public set cartItem(data: ScCartItem | undefined) {\n        this._cartItem = data;\n\n        if (data) {\n            this.quantityControl.patchValue(data.quantity);\n        }\n    }\n\n    /**\n     * Элемент корзины.\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges.pipe(skip(1));\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickDeleteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickSettingsEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на карточку.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Продукт элемента корзины.\n     */\n    public get product(): ScProduct | undefined {\n        return this.cartItem?.product;\n    }\n\n    /**\n     * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.\n     *\n     * @param unitsHelper Объект-хэлпер для работы со значениями единиц измерения товара.\n     * @param previewDialogService Сервис диалогового окна предварительного просмотра.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: UnitsHelper,\n        @Inject(TuiPreviewDialogService)\n        private readonly previewDialogService: TuiPreviewDialogService,\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string\n    ) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.quantityValueChanges = this.quantityControl.valueChanges;\n    }\n\n    /**\n     * Отобразить спецификацию.\n     */\n    public showSpecification(): void {\n        this.previewDialogService.open(this.specificationPreviewRef || '').subscribe();\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     */\n    public getCardImage(): string {\n        return this.product?.getImagePreview(this.urls.imgServerUrl) ?? this.urls.imgServerUrl + this.pathImageNotFound;\n    }\n}\n","<ng-container *ngIf=\"cartItem && product; else skeleton\">\n    <!-- TODO: Необходима реализация функционала.\n        <button tuiIconButton size=\"m\" icon=\"scIconVerticalThreeDots\" appearance=\"float\" tuiMode=\"onLight\" class=\"!absolute right-0 top-0\"></button> -->\n    <div class=\"flex bg-white border border-tui-base-04 shadow-sc-2 rounded-xl min-w-72 p-4\">\n        <div class=\"flex mr-5 max-h-52 min-w-[30%] items-center\">\n            <img (click)=\"clickCardEvent.emit()\" [src]=\"getCardImage()\" [alt]=\"product.name\" class=\"rounded-xl w-full max-h-full object-cover\" />\n        </div>\n        <div class=\"flex flex-col items-start w-full\">\n            <a tuiLink (click)=\"clickCardEvent.emit()\" class=\"mb-1\">\n                <span class=\"font-bold\">{{ product.name }}</span>\n            </a>\n            <div class=\"text-tui-text-02 text-xs\">\n                <p *ngIf=\"product.pack\">Норма упаковки: {{ product.pack }}</p>\n                <p>Артикул: {{ product.code }}</p>\n                <div class=\"flex flex-col self-center gap-y-0.5\">\n                    <p class=\"flex flex-col items-baseline gap-x-2 font-bold\">\n                        <span>{{ product.costRubString }}</span>\n                        <span *ngIf=\"!product.priceInRub\">{{ product.costString }}</span>\n                    </p>\n                    <span *ngIf=\"product.discount\" class=\"flex items-center\">\n                        <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n                        <span class=\"text-tui-success-fill font-bold\"> -{{ product.discount.percent }}% </span>\n                        <tui-svg src=\"tuiIconInfoLarge\" [tuiHint]=\"discountHint\" [tuiHintShowDelay]=\"100\" tuiHintDirection=\"top\" class=\"!text-xs !h-4\"></tui-svg>\n                        <ng-template #discountHint>\n                            <div class=\"font-bold\">{{ product.discount.name }}</div>\n                            <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n                        </ng-template>\n                    </span>\n                    <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n                </div>\n\n                <div *ngIf=\"cartItem.height || cartItem.length || cartItem.width\" class=\"flex my-0.5 items-center\">\n                    <button\n                        tuiIconButton\n                        icon=\"tuiIconSettings\"\n                        (click)=\"clickSettingsEvent.emit()\"\n                        size=\"s\"\n                        appearance=\"secondary\"\n                        tuiMode=\"onLight\"\n                        class=\"mr-2 !self-center\"\n                    ></button>\n                    <div>\n                        <ng-container *ngIf=\"cartItem.height; else length\">\n                            <p>Высота: {{ cartItem.height }} м.</p>\n                        </ng-container>\n                        <ng-template #length>\n                            <p *ngIf=\"cartItem.length\">Длина: {{ cartItem.length }} м.</p>\n                        </ng-template>\n                        <p *ngIf=\"cartItem.width\">Ширина: {{ cartItem.width }} м.</p>\n                    </div>\n                </div>\n\n                <a tuiLink *ngIf=\"cartItem.specificationImgUrl\" (click)=\"showSpecification()\">Спецификация</a>\n                <ng-template #specificationPreview let-preview>\n                    <tui-preview [rotatable]=\"false\" [zoomable]=\"false\">\n                        <img *polymorpheusOutlet=\"cartItem.specificationImgUrl as src\" alt=\"preview\" [src]=\"cartItem.specificationImgUrl\" />\n                        <button icon=\"tuiIconClose\" title=\"Close\" tuiIconButton tuiPreviewAction type=\"button\" (click)=\"preview.complete()\"></button>\n                    </tui-preview>\n                </ng-template>\n                <p class=\"font-bold text-tui-text-01 text-sm\">Итого: {{ cartItem.costRub | tuiFormatNumber }} {{ 'RUB' | tuiCurrency }}</p>\n            </div>\n            <sc-input-quantity\n                #inputQuantity\n                *ngIf=\"quantityControl\"\n                [formControl]=\"quantityControl\"\n                size=\"s\"\n                [step]=\"unitsHelper.productMultiplicity(product)\"\n                (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n                class=\"w-full mt-2\"\n            ></sc-input-quantity>\n        </div>\n    </div>\n</ng-container>\n\n<ng-template #skeleton>\n    <div class=\"flex gap-2 w-full\">\n        <div class=\"h-20 w-20 bg-tui-base-02 rounded-xl\"></div>\n        <div class=\"flex flex-col grow gap-2.5 bg-white\">\n            <div class=\"w-full h-4 rounded-xl bg-tui-base-02\"></div>\n            <div class=\"w-3/5 h-4 rounded-xl bg-tui-base-02\"></div>\n            <div class=\"w-4/5 h-4 rounded-xl bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-item-mobile.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.ts","../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAe,SAAS,EAAE,MAAM,eAAe,CAAC;AACxI,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAkC,MAAM,wBAAwB,CAAC;AAC1G,OAAO,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;AACxC,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;;;;;;;;;;;AAIlE;;GAEG;AAMH,MAAM,OAAO,yBAAyB;IAmElC;;;;;;;OAOG;IACH,YACoB,WAAwB,EAEvB,oBAA6C,EAC5B,IAAa,EACG,iBAAyB;QAJ3D,gBAAW,GAAX,WAAW,CAAa;QAEvB,yBAAoB,GAApB,oBAAoB,CAAyB;QAC5B,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QApE/E;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAgB,CAAC,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAqB7G;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzG;;WAEG;QAEI,qBAAgB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEvE;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAuBlE,CAAC;IAhEJ;;OAEG;IACH,IACW,QAAQ,CAAC,IAA4B;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,IAAI,IAAI,EAAE;YACN,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAClD;IACL,CAAC;IAED;;OAEG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IA0BD;;OAEG;IACH,IAAW,OAAO;QACd,OAAO,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;IAClC,CAAC;IAkBD,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;IAClE,CAAC;IAED;;OAEG;IACI,iBAAiB;QACpB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACnF,CAAC;IAED;;OAEG;IACI,YAAY;QACf,OAAO,IAAI,CAAC,OAAO,EAAE,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC;IACpH,CAAC;;sHApGQ,yBAAyB,6CA6EtB,uBAAuB,aAEvB,OAAO,aACP,uBAAuB;0GAhF1B,yBAAyB,2YChBtC,i7KAoFA;2FDpEa,yBAAyB;kBALrC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BA+E1C,MAAM;2BAAC,uBAAuB;;0BAE9B,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;4CAtElB,uBAAuB;sBADvC,SAAS;uBAAC,sBAAsB;gBAYtB,QAAQ;sBADlB,KAAK;gBAoBC,oBAAoB;sBAD1B,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAOA,kBAAkB;sBADxB,MAAM;gBAOA,cAAc;sBADpB,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, OnInit, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScCartItem, ScIUrls, ScProduct } from '@snabcentr/client-core';\nimport { Observable, skip } from 'rxjs';\nimport { TuiPreviewDialogService } from '@taiga-ui/addon-preview';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    selector: 'sc-cart-item-mobile',\n    templateUrl: './sc-cart-item-mobile.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScCartItemMobileComponent implements OnInit {\n    /**\n     * Элемент корзины.\n     */\n    private _cartItem?: ScCartItem;\n\n    /**\n     * Ссылка на представление спецификации.\n     */\n    @ViewChild('specificationPreview')\n    private readonly specificationPreviewRef?: TemplateRef<TuiDialogContext>;\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(0, { updateOn: 'blur' });\n\n    /**\n     * Элемент корзины.\n     */\n    @Input()\n    public set cartItem(data: ScCartItem | undefined) {\n        this._cartItem = data;\n\n        if (data) {\n            this.quantityControl.patchValue(data.quantity);\n        }\n    }\n\n    /**\n     * Элемент корзины.\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges.pipe(skip(1));\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickDeleteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickSettingsEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на карточку.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Продукт элемента корзины.\n     */\n    public get product(): ScProduct | undefined {\n        return this.cartItem?.product;\n    }\n\n    /**\n     * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.\n     *\n     * @param unitsHelper Экземпляр класса-помощника для работы со значениями единиц измерения товара.\n     * @param previewDialogService Сервис диалогового окна предварительного просмотра.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: UnitsHelper,\n        @Inject(TuiPreviewDialogService)\n        private readonly previewDialogService: TuiPreviewDialogService,\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string\n    ) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.quantityValueChanges = this.quantityControl.valueChanges;\n    }\n\n    /**\n     * Отобразить спецификацию.\n     */\n    public showSpecification(): void {\n        this.previewDialogService.open(this.specificationPreviewRef || '').subscribe();\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     */\n    public getCardImage(): string {\n        return this.product?.getImagePreview(this.urls.imgServerUrl) ?? this.urls.imgServerUrl + this.pathImageNotFound;\n    }\n}\n","<ng-container *ngIf=\"cartItem && product; else skeleton\">\n    <!-- TODO: Необходима реализация функционала.\n        <button tuiIconButton size=\"m\" icon=\"scIconVerticalThreeDots\" appearance=\"float\" tuiMode=\"onLight\" class=\"!absolute right-0 top-0\"></button> -->\n    <div class=\"flex bg-white border border-tui-base-04 shadow-sc-2 rounded-xl min-w-72 p-4\">\n        <div class=\"flex mr-5 max-h-52 min-w-[30%] items-center\">\n            <img (click)=\"clickCardEvent.emit()\" [src]=\"getCardImage()\" [alt]=\"product.name\" class=\"rounded-xl w-full max-h-full object-cover\" />\n        </div>\n        <div class=\"flex flex-col items-start w-full\">\n            <a tuiLink (click)=\"clickCardEvent.emit()\" class=\"mb-1\">\n                <span class=\"font-bold\">{{ product.name }}</span>\n            </a>\n            <div class=\"text-tui-text-02 text-xs\">\n                <p *ngIf=\"product.pack\">Норма упаковки: {{ product.pack }}</p>\n                <p>Артикул: {{ product.code }}</p>\n                <div class=\"flex flex-col self-center gap-y-0.5\">\n                    <p class=\"flex flex-col items-baseline gap-x-2 font-bold\">\n                        <span>{{ product.costRubString }}</span>\n                        <span *ngIf=\"!product.priceInRub\">{{ product.costString }}</span>\n                    </p>\n                    <span *ngIf=\"product.discount\" class=\"flex items-center\">\n                        <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n                        <span class=\"text-tui-success-fill font-bold\"> -{{ product.discount.percent }}% </span>\n                        <tui-svg src=\"tuiIconInfoLarge\" [tuiHint]=\"discountHint\" [tuiHintShowDelay]=\"100\" tuiHintDirection=\"top\" class=\"!text-xs !h-4\"></tui-svg>\n                        <ng-template #discountHint>\n                            <div class=\"font-bold\">{{ product.discount.name }}</div>\n                            <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n                        </ng-template>\n                    </span>\n                    <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n                </div>\n\n                <div *ngIf=\"cartItem.height || cartItem.length || cartItem.width\" class=\"flex my-0.5 items-center\">\n                    <button\n                        tuiIconButton\n                        icon=\"tuiIconSettings\"\n                        (click)=\"clickSettingsEvent.emit()\"\n                        size=\"s\"\n                        appearance=\"secondary\"\n                        tuiMode=\"onLight\"\n                        class=\"mr-2 !self-center\"\n                    ></button>\n                    <div>\n                        <ng-container *ngIf=\"cartItem.height; else length\">\n                            <p>Высота: {{ cartItem.height }} м.</p>\n                        </ng-container>\n                        <ng-template #length>\n                            <p *ngIf=\"cartItem.length\">Длина: {{ cartItem.length }} м.</p>\n                        </ng-template>\n                        <p *ngIf=\"cartItem.width\">Ширина: {{ cartItem.width }} м.</p>\n                    </div>\n                </div>\n\n                <a tuiLink *ngIf=\"cartItem.specificationImgUrl\" (click)=\"showSpecification()\">Спецификация</a>\n                <ng-template #specificationPreview let-preview>\n                    <tui-preview [rotatable]=\"false\" [zoomable]=\"false\">\n                        <img *polymorpheusOutlet=\"cartItem.specificationImgUrl as src\" alt=\"preview\" [src]=\"cartItem.specificationImgUrl\" />\n                        <button icon=\"tuiIconClose\" title=\"Close\" tuiIconButton tuiPreviewAction type=\"button\" (click)=\"preview.complete()\"></button>\n                    </tui-preview>\n                </ng-template>\n                <p class=\"font-bold text-tui-text-01 text-sm\">Итого: {{ cartItem.costRub | tuiFormatNumber }} {{ 'RUB' | tuiCurrency }}</p>\n            </div>\n            <sc-input-quantity\n                #inputQuantity\n                *ngIf=\"quantityControl\"\n                [formControl]=\"quantityControl\"\n                size=\"s\"\n                [step]=\"unitsHelper.productMultiplicity(product)\"\n                (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n                class=\"w-full mt-2\"\n            ></sc-input-quantity>\n        </div>\n    </div>\n</ng-container>\n\n<ng-template #skeleton>\n    <div class=\"flex gap-2 w-full\">\n        <div class=\"h-20 w-20 bg-tui-base-02 rounded-xl\"></div>\n        <div class=\"flex flex-col grow gap-2.5 bg-white\">\n            <div class=\"w-full h-4 rounded-xl bg-tui-base-02\"></div>\n            <div class=\"w-3/5 h-4 rounded-xl bg-tui-base-02\"></div>\n            <div class=\"w-4/5 h-4 rounded-xl bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
@@ -58,8 +58,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
58
58
  FormsModule,
59
59
  ReactiveFormsModule,
60
60
  TuiPreviewModule,
61
- PolymorpheusModule,
62
- ],
61
+ PolymorpheusModule
62
+ ]
63
63
  }]
64
64
  }] });
65
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2FydC5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9zYy1jYXJ0Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxrREFBa0QsQ0FBQztBQUM3RixPQUFPLEVBQUUsZUFBZSxFQUFFLHlCQUF5QixFQUFFLGFBQWEsRUFBRSxhQUFhLEVBQUUsYUFBYSxFQUFFLFlBQVksRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3ZJLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFDN0MsT0FBTyxFQUFFLFdBQVcsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ2xFLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ2pFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQzNELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDBCQUEwQixDQUFDOztBQUU5RDs7R0FFRztBQW9CSCxNQUFNLE9BQU8sWUFBWTs7eUdBQVosWUFBWTswR0FBWixZQUFZLGlCQWxCTix5QkFBeUIsYUFHcEMsWUFBWTtRQUNaLGFBQWE7UUFDYixlQUFlO1FBQ2YsZUFBZTtRQUNmLHFCQUFxQjtRQUNyQix5QkFBeUI7UUFDekIsYUFBYTtRQUNiLFlBQVk7UUFDWixhQUFhO1FBQ2IsV0FBVztRQUNYLG1CQUFtQjtRQUNuQixnQkFBZ0I7UUFDaEIsa0JBQWtCLGFBZFoseUJBQXlCOzBHQWlCMUIsWUFBWSxZQWZqQixZQUFZO1FBQ1osYUFBYTtRQUNiLGVBQWU7UUFDZixlQUFlO1FBQ2YscUJBQXFCO1FBQ3JCLHlCQUF5QjtRQUN6QixhQUFhO1FBQ2IsWUFBWTtRQUNaLGFBQWE7UUFDYixXQUFXO1FBQ1gsbUJBQW1CO1FBQ25CLGdCQUFnQjtRQUNoQixrQkFBa0I7MkZBR2IsWUFBWTtrQkFuQnhCLFFBQVE7bUJBQUM7b0JBQ04sWUFBWSxFQUFFLENBQUMseUJBQXlCLENBQUM7b0JBQ3pDLE9BQU8sRUFBRSxDQUFDLHlCQUF5QixDQUFDO29CQUNwQyxPQUFPLEVBQUU7d0JBQ0wsWUFBWTt3QkFDWixhQUFhO3dCQUNiLGVBQWU7d0JBQ2YsZUFBZTt3QkFDZixxQkFBcUI7d0JBQ3JCLHlCQUF5Qjt3QkFDekIsYUFBYTt3QkFDYixZQUFZO3dCQUNaLGFBQWE7d0JBQ2IsV0FBVzt3QkFDWCxtQkFBbUI7d0JBQ25CLGdCQUFnQjt3QkFDaEIsa0JBQWtCO3FCQUNyQjtpQkFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgU2NDYXJ0SXRlbU1vYmlsZUNvbXBvbmVudCB9IGZyb20gJy4vY2FydC1pdGVtLW1vYmlsZS9zYy1jYXJ0LWl0ZW0tbW9iaWxlLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBUdWlCdXR0b25Nb2R1bGUsIFR1aUZvcm1hdE51bWJlclBpcGVNb2R1bGUsIFR1aUhpbnRNb2R1bGUsIFR1aUxpbmtNb2R1bGUsIFR1aU1vZGVNb2R1bGUsIFR1aVN2Z01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7IFNjQ2F0YWxvZ01vZHVsZSB9IGZyb20gJy4uL2NhdGFsb2cnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBUdWlDdXJyZW5jeVBpcGVNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvYWRkb24tY29tbWVyY2UnO1xuaW1wb3J0IHsgVHVpUHJldmlld01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9hZGRvbi1wcmV2aWV3JztcbmltcG9ydCB7IFBvbHltb3JwaGV1c01vZHVsZSB9IGZyb20gJ0B0aW5rb2ZmL25nLXBvbHltb3JwaGV1cyc7XG5cbi8qKlxuICog0JzQvtC00YPQu9GMINC60L7RgNC30LjQvdGLLlxuICovXG5ATmdNb2R1bGUoe1xuICAgIGRlY2xhcmF0aW9uczogW1NjQ2FydEl0ZW1Nb2JpbGVDb21wb25lbnRdLFxuICAgIGV4cG9ydHM6IFtTY0NhcnRJdGVtTW9iaWxlQ29tcG9uZW50XSxcbiAgICBpbXBvcnRzOiBbXG4gICAgICAgIENvbW1vbk1vZHVsZSxcbiAgICAgICAgVHVpTGlua01vZHVsZSxcbiAgICAgICAgU2NDYXRhbG9nTW9kdWxlLFxuICAgICAgICBUdWlCdXR0b25Nb2R1bGUsXG4gICAgICAgIFR1aUN1cnJlbmN5UGlwZU1vZHVsZSxcbiAgICAgICAgVHVpRm9ybWF0TnVtYmVyUGlwZU1vZHVsZSxcbiAgICAgICAgVHVpTW9kZU1vZHVsZSxcbiAgICAgICAgVHVpU3ZnTW9kdWxlLFxuICAgICAgICBUdWlIaW50TW9kdWxlLFxuICAgICAgICBGb3Jtc01vZHVsZSxcbiAgICAgICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICAgICAgVHVpUHJldmlld01vZHVsZSxcbiAgICAgICAgUG9seW1vcnBoZXVzTW9kdWxlLFxuICAgIF0sXG59KVxuZXhwb3J0IGNsYXNzIFNjQ2FydE1vZHVsZSB7fVxuIl19
65
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2FydC5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9zYy1jYXJ0Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxrREFBa0QsQ0FBQztBQUM3RixPQUFPLEVBQUUsZUFBZSxFQUFFLHlCQUF5QixFQUFFLGFBQWEsRUFBRSxhQUFhLEVBQUUsYUFBYSxFQUFFLFlBQVksRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3ZJLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFDN0MsT0FBTyxFQUFFLFdBQVcsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ2xFLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ2pFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQzNELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDBCQUEwQixDQUFDOztBQUU5RDs7R0FFRztBQW9CSCxNQUFNLE9BQU8sWUFBWTs7eUdBQVosWUFBWTswR0FBWixZQUFZLGlCQWxCTix5QkFBeUIsYUFHcEMsWUFBWTtRQUNaLGFBQWE7UUFDYixlQUFlO1FBQ2YsZUFBZTtRQUNmLHFCQUFxQjtRQUNyQix5QkFBeUI7UUFDekIsYUFBYTtRQUNiLFlBQVk7UUFDWixhQUFhO1FBQ2IsV0FBVztRQUNYLG1CQUFtQjtRQUNuQixnQkFBZ0I7UUFDaEIsa0JBQWtCLGFBZFoseUJBQXlCOzBHQWlCMUIsWUFBWSxZQWZqQixZQUFZO1FBQ1osYUFBYTtRQUNiLGVBQWU7UUFDZixlQUFlO1FBQ2YscUJBQXFCO1FBQ3JCLHlCQUF5QjtRQUN6QixhQUFhO1FBQ2IsWUFBWTtRQUNaLGFBQWE7UUFDYixXQUFXO1FBQ1gsbUJBQW1CO1FBQ25CLGdCQUFnQjtRQUNoQixrQkFBa0I7MkZBR2IsWUFBWTtrQkFuQnhCLFFBQVE7bUJBQUM7b0JBQ04sWUFBWSxFQUFFLENBQUMseUJBQXlCLENBQUM7b0JBQ3pDLE9BQU8sRUFBRSxDQUFDLHlCQUF5QixDQUFDO29CQUNwQyxPQUFPLEVBQUU7d0JBQ0wsWUFBWTt3QkFDWixhQUFhO3dCQUNiLGVBQWU7d0JBQ2YsZUFBZTt3QkFDZixxQkFBcUI7d0JBQ3JCLHlCQUF5Qjt3QkFDekIsYUFBYTt3QkFDYixZQUFZO3dCQUNaLGFBQWE7d0JBQ2IsV0FBVzt3QkFDWCxtQkFBbUI7d0JBQ25CLGdCQUFnQjt3QkFDaEIsa0JBQWtCO3FCQUNyQjtpQkFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgU2NDYXJ0SXRlbU1vYmlsZUNvbXBvbmVudCB9IGZyb20gJy4vY2FydC1pdGVtLW1vYmlsZS9zYy1jYXJ0LWl0ZW0tbW9iaWxlLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBUdWlCdXR0b25Nb2R1bGUsIFR1aUZvcm1hdE51bWJlclBpcGVNb2R1bGUsIFR1aUhpbnRNb2R1bGUsIFR1aUxpbmtNb2R1bGUsIFR1aU1vZGVNb2R1bGUsIFR1aVN2Z01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7IFNjQ2F0YWxvZ01vZHVsZSB9IGZyb20gJy4uL2NhdGFsb2cnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBUdWlDdXJyZW5jeVBpcGVNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvYWRkb24tY29tbWVyY2UnO1xuaW1wb3J0IHsgVHVpUHJldmlld01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9hZGRvbi1wcmV2aWV3JztcbmltcG9ydCB7IFBvbHltb3JwaGV1c01vZHVsZSB9IGZyb20gJ0B0aW5rb2ZmL25nLXBvbHltb3JwaGV1cyc7XG5cbi8qKlxuICog0JzQvtC00YPQu9GMINC60L7RgNC30LjQvdGLLlxuICovXG5ATmdNb2R1bGUoe1xuICAgIGRlY2xhcmF0aW9uczogW1NjQ2FydEl0ZW1Nb2JpbGVDb21wb25lbnRdLFxuICAgIGV4cG9ydHM6IFtTY0NhcnRJdGVtTW9iaWxlQ29tcG9uZW50XSxcbiAgICBpbXBvcnRzOiBbXG4gICAgICAgIENvbW1vbk1vZHVsZSxcbiAgICAgICAgVHVpTGlua01vZHVsZSxcbiAgICAgICAgU2NDYXRhbG9nTW9kdWxlLFxuICAgICAgICBUdWlCdXR0b25Nb2R1bGUsXG4gICAgICAgIFR1aUN1cnJlbmN5UGlwZU1vZHVsZSxcbiAgICAgICAgVHVpRm9ybWF0TnVtYmVyUGlwZU1vZHVsZSxcbiAgICAgICAgVHVpTW9kZU1vZHVsZSxcbiAgICAgICAgVHVpU3ZnTW9kdWxlLFxuICAgICAgICBUdWlIaW50TW9kdWxlLFxuICAgICAgICBGb3Jtc01vZHVsZSxcbiAgICAgICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICAgICAgVHVpUHJldmlld01vZHVsZSxcbiAgICAgICAgUG9seW1vcnBoZXVzTW9kdWxlXG4gICAgXVxufSlcbmV4cG9ydCBjbGFzcyBTY0NhcnRNb2R1bGUge31cbiJdfQ==
@@ -47,4 +47,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
47
47
  }], clickCategoryEvent: [{
48
48
  type: Output
49
49
  }] } });
50
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2F0ZWdvcmllcy1saXN0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9jYXRhbG9nL2NhdGVnb3JpZXMtbGlzdC9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2NhdGFsb2cvY2F0ZWdvcmllcy1saXN0L3NjLWNhdGVnb3JpZXMtbGlzdC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDaEgsT0FBTyxFQUFFLE9BQU8sRUFBVyxNQUFNLHdCQUF3QixDQUFDO0FBQzFELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxlQUFlLENBQUM7Ozs7O0FBRTlDOztHQUVHO0FBTUgsTUFBTSxPQUFPLHlCQUF5QjtJQWtCbEM7Ozs7T0FJRztJQUNILFlBQXFELElBQWE7UUFBYixTQUFJLEdBQUosSUFBSSxDQUFTO1FBaEJsRTs7V0FFRztRQUNhLGFBQVEsR0FBWSxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFMUQ7O1dBRUc7UUFFSSx1QkFBa0IsR0FBb0IsSUFBSSxZQUFZLEVBQUssQ0FBQztJQU9FLENBQUM7SUFFdEU7Ozs7T0FJRztJQUNJLGdCQUFnQixDQUFDLE9BQWU7UUFDbkMsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksR0FBRyxHQUFHLEdBQUcsT0FBTyxDQUFDO0lBQ2xELENBQUM7O3NIQWhDUSx5QkFBeUIsa0JBdUJQLE9BQU87MEdBdkJ6Qix5QkFBeUIsdUpDWnRDLHU0REE0QkE7MkZEaEJhLHlCQUF5QjtrQkFMckMsU0FBUzsrQkFDSSxvQkFBb0IsbUJBRWIsdUJBQXVCLENBQUMsTUFBTTs7MEJBeUIzQixNQUFNOzJCQUFDLE9BQU87NENBbEIzQixVQUFVO3NCQURoQixLQUFLO2dCQVlDLGtCQUFrQjtzQkFEeEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgSW5qZWN0LCBJbnB1dCwgT3V0cHV0LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFNDX1VSTFMsIFNjSVVybHMgfSBmcm9tICdAc25hYmNlbnRyL2NsaWVudC1jb3JlJztcbmltcG9ydCB7IFRVSV9JU19NT0JJTEUgfSBmcm9tICdAdGFpZ2EtdWkvY2RrJztcblxuLyoqXG4gKiDQmtC+0LzQv9C+0L3QtdC90YIg0YHQv9C40YHQutCwINC60LDRgtC10LPQvtGA0LjQuS5cbiAqL1xuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICdzYy1jYXRlZ29yaWVzLWxpc3QnLFxuICAgIHRlbXBsYXRlVXJsOiAnLi9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50Lmh0bWwnLFxuICAgIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBTY0NhdGVnb3JpZXNMaXN0Q29tcG9uZW50PFQ+IHtcbiAgICAvKipcbiAgICAgKiDQodC/0LjRgdC+0Log0LrQsNGC0LXQs9C+0YDQuNC5LlxuICAgICAqL1xuICAgIEBJbnB1dCgpXG4gICAgcHVibGljIGNhdGVnb3JpZXM6IFRbXSB8IHVuZGVmaW5lZCB8IG51bGw7XG5cbiAgICAvKipcbiAgICAgKiDQn9GA0LjQt9C90LDQuiDRgtC+0LPQviwg0L7RgtC+0LHRgNCw0LbQsNC10YLRgdGPINGN0YLQvtGCINC60L7QvNC/0L7QvdC10L3RgiDQvdCwINC80L7QsdC40LvRjNC90L7QvCDRg9GB0YLRgNC+0LnRgdGC0LLQtSDQuNC70Lgg0L3QtdGCLlxuICAgICAqL1xuICAgIHB1YmxpYyByZWFkb25seSBpc01vYmlsZTogYm9vbGVhbiA9IGluamVjdChUVUlfSVNfTU9CSUxFKTtcblxuICAgIC8qKlxuICAgICAqINCh0L7QsdGL0YLQuNC1INC90LDQttCw0YLQuNGPINC90LAg0LrQsNGC0LXQs9C+0YDQuNGOLlxuICAgICAqL1xuICAgIEBPdXRwdXQoKVxuICAgIHB1YmxpYyBjbGlja0NhdGVnb3J5RXZlbnQ6IEV2ZW50RW1pdHRlcjxUPiA9IG5ldyBFdmVudEVtaXR0ZXI8VD4oKTtcblxuICAgIC8qKlxuICAgICAqINCY0L3QuNGG0LjQsNC70LjQt9C40YDRg9C10YIg0Y3QutC30LXQvNC/0LvRj9GAINC60LvQsNGB0YHQsCB7QGxpbmsgU2NDYXRlZ29yaWVzTGlzdENvbXBvbmVudH0uXG4gICAgICpcbiAgICAgKiBAcGFyYW0gdXJscyDQvtCx0YrQtdC60YIg0LjQvdGE0L7RgNC80LDRhtC40Lgg0L4g0LHQsNC30L7QstC+0Lwg0YHQv9C40YHQutC1INGB0YHRi9C70L7QuiDQv9GA0LjQu9C+0LbQtdC90LjRjy5cbiAgICAgKi9cbiAgICBwdWJsaWMgY29uc3RydWN0b3IoQEluamVjdChTQ19VUkxTKSBwcml2YXRlIHJlYWRvbmx5IHVybHM6IFNjSVVybHMpIHt9XG5cbiAgICAvKipcbiAgICAgKiDQktC+0LfQstGA0LDRidCw0LXRgiDQv9GD0YLRjCDQtNC+INC40LfQvtCx0YDQsNC20LXQvdC40Y8g0L3QsCDRgdC10YDQstC10YDQtS5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBpbWdQYXRoINC/0YPRgtGMLCDQs9C00LUg0YXRgNCw0L3QuNGC0YHRjyDQuNC30L7QsdGA0LDQttC10L3QuNC1LlxuICAgICAqL1xuICAgIHB1YmxpYyBnZXRDYXRlZ29yeUltYWdlKGltZ1BhdGg6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgICAgIHJldHVybiB0aGlzLnVybHMuaW1nU2VydmVyVXJsICsgJy8nICsgaW1nUGF0aDtcbiAgICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwiZ3JpZCBncmlkLWNvbHMtW3JlcGVhdChhdXRvLWZpbGwsbWlubWF4KDIxcmVtLDFmcikpXSBnYXAteC04IGdhcC15LTJcIj5cbiAgICA8bmctY29udGFpbmVyICpuZ0lmPVwiY2F0ZWdvcmllczsgZWxzZSBjYXRlZ29yaWVzU2tlbGV0b25cIj5cbiAgICAgICAgPGJ1dHRvblxuICAgICAgICAgICAgKm5nRm9yPVwibGV0IGl0ZW0gb2YgJGFueShjYXRlZ29yaWVzKVwiXG4gICAgICAgICAgICBjbGFzcz1cImdyb3VwIGZsZXggZ2FwLTIgcHktMS41IHB4LTMgYmctdHVpLWJhc2UtMDIgcm91bmRlZC14bCBoLTExIGl0ZW1zLWNlbnRlciB0ZXh0LWxlZnQgaG92ZXItaG92ZXI6aG92ZXI6cmluZy0yIGhvdmVyLW5vbmU6YWN0aXZlOnJpbmctMiByaW5nLXR1aS1wcmltYXJ5IGR1cmF0aW9uLTE1MFwiXG4gICAgICAgICAgICAoY2xpY2spPVwiY2xpY2tDYXRlZ29yeUV2ZW50LmVtaXQoaXRlbSlcIlxuICAgICAgICA+XG4gICAgICAgICAgICA8aW1nICpuZ0lmPVwiaXRlbS5wcm9wZXJ0aWVzPy5pbWFnZVwiIFtzcmNdPVwiZ2V0Q2F0ZWdvcnlJbWFnZShpdGVtLnByb3BlcnRpZXM/LmltYWdlKVwiIFthbHRdPVwiaXRlbS5uYW1lXCIgY2xhc3M9XCJ3LTggaC04IG9iamVjdC1jb3ZlciByb3VuZGVkIGJnLXR1aS1iYXNlLTAzXCIgLz5cbiAgICAgICAgICAgIDx0dWktc3ZnICpuZ0lmPVwiIWl0ZW0ucHJvcGVydGllcz8uaW1hZ2VcIiBzcmM9XCJ0dWlJY29uQ2FtZXJhT2ZmTGFyZ2VcIiBjbGFzcz1cImJnLXR1aS1iYXNlLTAzICF3LTggIWgtOCByb3VuZGVkIHRleHQtdHVpLWJhc2UtMDVcIj48L3R1aS1zdmc+XG4gICAgICAgICAgICA8ZGl2IGNsYXNzPVwiZmxleCBncm93IHRydW5jYXRlXCI+XG4gICAgICAgICAgICAgICAgPHAgY2xhc3M9XCJ0ZXh0LWJhc2UgZm9udC1ib2xkIHRydW5jYXRlXCI+e3sgaXRlbS5uYW1lIH19PC9wPlxuICAgICAgICAgICAgICAgIDx0dWktc3ZnICpuZ0lmPVwiaXRlbS5pc0Zhdm9yaXRlXCIgc3JjPVwic2NJY29uRmF2b3JpdGVGaWxsXCIgY2xhc3M9XCJ0ZXh0LXR1aS1wcmltYXJ5ICF0ZXh0LXNtIG1yLWF1dG9cIj48L3R1aS1zdmc+XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICAgIDx0dWktc3ZnIHNyYz1cInR1aUljb25DaGV2cm9uUmlnaHRMYXJnZVwiIGNsYXNzPVwiaG92ZXItaG92ZXI6Z3JvdXAtaG92ZXI6dGV4dC10dWktcHJpbWFyeSBob3Zlci1ub25lOmdyb3VwLWFjdGl2ZTp0ZXh0LXR1aS1wcmltYXJ5IGR1cmF0aW9uLTE1MFwiPjwvdHVpLXN2Zz5cbiAgICAgICAgPC9idXR0b24+XG4gICAgPC9uZy1jb250YWluZXI+XG48L2Rpdj5cblxuPG5nLXRlbXBsYXRlICNjYXRlZ29yaWVzU2tlbGV0b24+XG4gICAgPGJ1dHRvblxuICAgICAgICAqdHVpUmVwZWF0VGltZXM9XCJsZXQgaW5kZXggb2YgaXNNb2JpbGUgPyAzIDogNlwiXG4gICAgICAgIGNsYXNzPVwiZ3JvdXAgZmxleCBnYXAtMiBweS0xLjUgcHgtMyBiZy10dWktYmFzZS0wMiByb3VuZGVkLXhsIGgtMTEgaXRlbXMtY2VudGVyIHRleHQtbGVmdCBob3Zlci1ob3Zlcjpob3ZlcjpyaW5nLTIgaG92ZXItbm9uZTphY3RpdmU6cmluZy0yIHJpbmctdHVpLWJhc2UtMDUgZHVyYXRpb24tMTUwXCJcbiAgICA+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJ3LTggaC04IHR1aS1za2VsZXRvblwiPjwvZGl2PlxuICAgICAgICA8ZGl2IGNsYXNzPVwiaC00IGdyb3cgdHVpLXNrZWxldG9uXCI+PC9kaXY+XG4gICAgICAgIDx0dWktc3ZnIHNyYz1cInR1aUljb25DaGV2cm9uUmlnaHRMYXJnZVwiIGNsYXNzPVwiaG92ZXItaG92ZXI6Z3JvdXAtaG92ZXI6dGV4dC10dWktcHJpbWFyeSBob3Zlci1ub25lOmdyb3VwLWFjdGl2ZTp0ZXh0LXR1aS1wcmltYXJ5IGR1cmF0aW9uLTE1MCAgdHVpLXNrZWxldG9uXCI+PC90dWktc3ZnPlxuICAgIDwvYnV0dG9uPlxuPC9uZy10ZW1wbGF0ZT5cbiJdfQ==
50
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2F0ZWdvcmllcy1saXN0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9jYXRhbG9nL2NhdGVnb3JpZXMtbGlzdC9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2NhdGFsb2cvY2F0ZWdvcmllcy1saXN0L3NjLWNhdGVnb3JpZXMtbGlzdC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDaEgsT0FBTyxFQUFFLE9BQU8sRUFBVyxNQUFNLHdCQUF3QixDQUFDO0FBQzFELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxlQUFlLENBQUM7Ozs7O0FBRTlDOztHQUVHO0FBTUgsTUFBTSxPQUFPLHlCQUF5QjtJQWtCbEM7Ozs7T0FJRztJQUNILFlBQXFELElBQWE7UUFBYixTQUFJLEdBQUosSUFBSSxDQUFTO1FBaEJsRTs7V0FFRztRQUNhLGFBQVEsR0FBWSxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7UUFFMUQ7O1dBRUc7UUFFSSx1QkFBa0IsR0FBb0IsSUFBSSxZQUFZLEVBQUssQ0FBQztJQU9FLENBQUM7SUFFdEU7Ozs7T0FJRztJQUNJLGdCQUFnQixDQUFDLE9BQWU7UUFDbkMsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksR0FBRyxHQUFHLEdBQUcsT0FBTyxDQUFDO0lBQ2xELENBQUM7O3NIQWhDUSx5QkFBeUIsa0JBdUJQLE9BQU87MEdBdkJ6Qix5QkFBeUIsdUpDWnRDLHU0REE0QkE7MkZEaEJhLHlCQUF5QjtrQkFMckMsU0FBUzsrQkFDSSxvQkFBb0IsbUJBRWIsdUJBQXVCLENBQUMsTUFBTTs7MEJBeUIzQixNQUFNOzJCQUFDLE9BQU87NENBbEIzQixVQUFVO3NCQURoQixLQUFLO2dCQVlDLGtCQUFrQjtzQkFEeEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgSW5qZWN0LCBJbnB1dCwgT3V0cHV0LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFNDX1VSTFMsIFNjSVVybHMgfSBmcm9tICdAc25hYmNlbnRyL2NsaWVudC1jb3JlJztcbmltcG9ydCB7IFRVSV9JU19NT0JJTEUgfSBmcm9tICdAdGFpZ2EtdWkvY2RrJztcblxuLyoqXG4gKiDQmtC+0LzQv9C+0L3QtdC90YIg0YHQv9C40YHQutCwINC60LDRgtC10LPQvtGA0LjQuS5cbiAqL1xuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICdzYy1jYXRlZ29yaWVzLWxpc3QnLFxuICAgIHRlbXBsYXRlVXJsOiAnLi9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50Lmh0bWwnLFxuICAgIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoXG59KVxuZXhwb3J0IGNsYXNzIFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQ8VD4ge1xuICAgIC8qKlxuICAgICAqINCh0L/QuNGB0L7QuiDQutCw0YLQtdCz0L7RgNC40LkuXG4gICAgICovXG4gICAgQElucHV0KClcbiAgICBwdWJsaWMgY2F0ZWdvcmllczogVFtdIHwgdW5kZWZpbmVkIHwgbnVsbDtcblxuICAgIC8qKlxuICAgICAqINCf0YDQuNC30L3QsNC6INGC0L7Qs9C+LCDQvtGC0L7QsdGA0LDQttCw0LXRgtGB0Y8g0Y3RgtC+0YIg0LrQvtC80L/QvtC90LXQvdGCINC90LAg0LzQvtCx0LjQu9GM0L3QvtC8INGD0YHRgtGA0L7QudGB0YLQstC1INC40LvQuCDQvdC10YIuXG4gICAgICovXG4gICAgcHVibGljIHJlYWRvbmx5IGlzTW9iaWxlOiBib29sZWFuID0gaW5qZWN0KFRVSV9JU19NT0JJTEUpO1xuXG4gICAgLyoqXG4gICAgICog0KHQvtCx0YvRgtC40LUg0L3QsNC20LDRgtC40Y8g0L3QsCDQutCw0YLQtdCz0L7RgNC40Y4uXG4gICAgICovXG4gICAgQE91dHB1dCgpXG4gICAgcHVibGljIGNsaWNrQ2F0ZWdvcnlFdmVudDogRXZlbnRFbWl0dGVyPFQ+ID0gbmV3IEV2ZW50RW1pdHRlcjxUPigpO1xuXG4gICAgLyoqXG4gICAgICog0JjQvdC40YbQuNCw0LvQuNC30LjRgNGD0LXRgiDRjdC60LfQtdC80L/Qu9GP0YAg0LrQu9Cw0YHRgdCwIHtAbGluayBTY0NhdGVnb3JpZXNMaXN0Q29tcG9uZW50fS5cbiAgICAgKlxuICAgICAqIEBwYXJhbSB1cmxzINC+0LHRitC10LrRgiDQuNC90YTQvtGA0LzQsNGG0LjQuCDQviDQsdCw0LfQvtCy0L7QvCDRgdC/0LjRgdC60LUg0YHRgdGL0LvQvtC6INC/0YDQuNC70L7QttC10L3QuNGPLlxuICAgICAqL1xuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcihASW5qZWN0KFNDX1VSTFMpIHByaXZhdGUgcmVhZG9ubHkgdXJsczogU2NJVXJscykge31cblxuICAgIC8qKlxuICAgICAqINCS0L7Qt9Cy0YDQsNGJ0LDQtdGCINC/0YPRgtGMINC00L4g0LjQt9C+0LHRgNCw0LbQtdC90LjRjyDQvdCwINGB0LXRgNCy0LXRgNC1LlxuICAgICAqXG4gICAgICogQHBhcmFtIGltZ1BhdGgg0L/Rg9GC0YwsINCz0LTQtSDRhdGA0LDQvdC40YLRgdGPINC40LfQvtCx0YDQsNC20LXQvdC40LUuXG4gICAgICovXG4gICAgcHVibGljIGdldENhdGVnb3J5SW1hZ2UoaW1nUGF0aDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICAgICAgcmV0dXJuIHRoaXMudXJscy5pbWdTZXJ2ZXJVcmwgKyAnLycgKyBpbWdQYXRoO1xuICAgIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJncmlkIGdyaWQtY29scy1bcmVwZWF0KGF1dG8tZmlsbCxtaW5tYXgoMjFyZW0sMWZyKSldIGdhcC14LTggZ2FwLXktMlwiPlxuICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCJjYXRlZ29yaWVzOyBlbHNlIGNhdGVnb3JpZXNTa2VsZXRvblwiPlxuICAgICAgICA8YnV0dG9uXG4gICAgICAgICAgICAqbmdGb3I9XCJsZXQgaXRlbSBvZiAkYW55KGNhdGVnb3JpZXMpXCJcbiAgICAgICAgICAgIGNsYXNzPVwiZ3JvdXAgZmxleCBnYXAtMiBweS0xLjUgcHgtMyBiZy10dWktYmFzZS0wMiByb3VuZGVkLXhsIGgtMTEgaXRlbXMtY2VudGVyIHRleHQtbGVmdCBob3Zlci1ob3Zlcjpob3ZlcjpyaW5nLTIgaG92ZXItbm9uZTphY3RpdmU6cmluZy0yIHJpbmctdHVpLXByaW1hcnkgZHVyYXRpb24tMTUwXCJcbiAgICAgICAgICAgIChjbGljayk9XCJjbGlja0NhdGVnb3J5RXZlbnQuZW1pdChpdGVtKVwiXG4gICAgICAgID5cbiAgICAgICAgICAgIDxpbWcgKm5nSWY9XCJpdGVtLnByb3BlcnRpZXM/LmltYWdlXCIgW3NyY109XCJnZXRDYXRlZ29yeUltYWdlKGl0ZW0ucHJvcGVydGllcz8uaW1hZ2UpXCIgW2FsdF09XCJpdGVtLm5hbWVcIiBjbGFzcz1cInctOCBoLTggb2JqZWN0LWNvdmVyIHJvdW5kZWQgYmctdHVpLWJhc2UtMDNcIiAvPlxuICAgICAgICAgICAgPHR1aS1zdmcgKm5nSWY9XCIhaXRlbS5wcm9wZXJ0aWVzPy5pbWFnZVwiIHNyYz1cInR1aUljb25DYW1lcmFPZmZMYXJnZVwiIGNsYXNzPVwiYmctdHVpLWJhc2UtMDMgIXctOCAhaC04IHJvdW5kZWQgdGV4dC10dWktYmFzZS0wNVwiPjwvdHVpLXN2Zz5cbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJmbGV4IGdyb3cgdHJ1bmNhdGVcIj5cbiAgICAgICAgICAgICAgICA8cCBjbGFzcz1cInRleHQtYmFzZSBmb250LWJvbGQgdHJ1bmNhdGVcIj57eyBpdGVtLm5hbWUgfX08L3A+XG4gICAgICAgICAgICAgICAgPHR1aS1zdmcgKm5nSWY9XCJpdGVtLmlzRmF2b3JpdGVcIiBzcmM9XCJzY0ljb25GYXZvcml0ZUZpbGxcIiBjbGFzcz1cInRleHQtdHVpLXByaW1hcnkgIXRleHQtc20gbXItYXV0b1wiPjwvdHVpLXN2Zz5cbiAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgICAgPHR1aS1zdmcgc3JjPVwidHVpSWNvbkNoZXZyb25SaWdodExhcmdlXCIgY2xhc3M9XCJob3Zlci1ob3Zlcjpncm91cC1ob3Zlcjp0ZXh0LXR1aS1wcmltYXJ5IGhvdmVyLW5vbmU6Z3JvdXAtYWN0aXZlOnRleHQtdHVpLXByaW1hcnkgZHVyYXRpb24tMTUwXCI+PC90dWktc3ZnPlxuICAgICAgICA8L2J1dHRvbj5cbiAgICA8L25nLWNvbnRhaW5lcj5cbjwvZGl2PlxuXG48bmctdGVtcGxhdGUgI2NhdGVnb3JpZXNTa2VsZXRvbj5cbiAgICA8YnV0dG9uXG4gICAgICAgICp0dWlSZXBlYXRUaW1lcz1cImxldCBpbmRleCBvZiBpc01vYmlsZSA/IDMgOiA2XCJcbiAgICAgICAgY2xhc3M9XCJncm91cCBmbGV4IGdhcC0yIHB5LTEuNSBweC0zIGJnLXR1aS1iYXNlLTAyIHJvdW5kZWQteGwgaC0xMSBpdGVtcy1jZW50ZXIgdGV4dC1sZWZ0IGhvdmVyLWhvdmVyOmhvdmVyOnJpbmctMiBob3Zlci1ub25lOmFjdGl2ZTpyaW5nLTIgcmluZy10dWktYmFzZS0wNSBkdXJhdGlvbi0xNTBcIlxuICAgID5cbiAgICAgICAgPGRpdiBjbGFzcz1cInctOCBoLTggdHVpLXNrZWxldG9uXCI+PC9kaXY+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJoLTQgZ3JvdyB0dWktc2tlbGV0b25cIj48L2Rpdj5cbiAgICAgICAgPHR1aS1zdmcgc3JjPVwidHVpSWNvbkNoZXZyb25SaWdodExhcmdlXCIgY2xhc3M9XCJob3Zlci1ob3Zlcjpncm91cC1ob3Zlcjp0ZXh0LXR1aS1wcmltYXJ5IGhvdmVyLW5vbmU6Z3JvdXAtYWN0aXZlOnRleHQtdHVpLXByaW1hcnkgZHVyYXRpb24tMTUwICB0dWktc2tlbGV0b25cIj48L3R1aS1zdmc+XG4gICAgPC9idXR0b24+XG48L25nLXRlbXBsYXRlPlxuIl19
@@ -1,4 +1,4 @@
1
- import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, HostListener, Inject, Input, Output, ViewChild, } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, HostBinding, HostListener, Inject, Input, Output, ViewChild } from '@angular/core';
2
2
  import { SC_PATH_IMAGE_NOT_FOUND, SC_URLS } from '@snabcentr/client-core';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "@snabcentr/client-core";
@@ -117,4 +117,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
117
117
  type: HostListener,
118
118
  args: ['mouseleave']
119
119
  }] } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-category-card.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.ts","../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,uBAAuB,EAEvB,SAAS,EAET,YAAY,EACZ,WAAW,EACX,YAAY,EACZ,MAAM,EACN,KAAK,EACL,MAAM,EAEN,SAAS,GACZ,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAsC,MAAM,wBAAwB,CAAC;;;;;AAI9G;;GAEG;AAOH,MAAM,OAAO,uBAAuB;IAsDhC;;;;;;OAMG;IACH,YACsC,IAAa,EACvC,QAAmB,EACuB,iBAAyB,EAC1D,WAA0B,EAC1B,GAAsB;QAJL,SAAI,GAAJ,IAAI,CAAS;QACvC,aAAQ,GAAR,QAAQ,CAAW;QACuB,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,gBAAW,GAAX,WAAW,CAAe;QAC1B,QAAG,GAAH,GAAG,CAAmB;QArD3C;;WAEG;QAGI,SAAI,GAAa,GAAG,CAAC;QAE5B;;WAEG;QAEI,eAAU,GAAY,KAAK,CAAC;QAEnC;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QACa,gBAAW,GAAwB,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;QAEpF;;WAEG;QAEI,cAAS,GAAY,KAAK,CAAC;QAElC;;WAEG;QAEI,qBAAgB,GAA6B,IAAI,YAAY,EAAc,CAAC;QAEnF;;WAEG;QAEI,yBAAoB,GAA6B,IAAI,YAAY,EAAc,CAAC;IAepF,CAAC;IAEJ;;;;OAIG;IACI,iBAAiB,CAAC,QAAoB;QACzC,OAAO,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,GAAG,GAAG,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;IAC3H,CAAC;IAED;;OAEG;IAEK,gBAAgB;QACpB,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,SAAS,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,GAAG,EAAE,KAAK,CAAC;SAC/F;IACL,CAAC;IAED;;OAEG;IAEK,iBAAiB;QACrB,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;SACtE;IACL,CAAC;IAED;;OAEG;IACI,YAAY;QACf,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;;oHAvGQ,uBAAuB,kBA8DpB,OAAO,sCAEP,uBAAuB;wGAhE1B,uBAAuB,yeC3BpC,0vCAwBA;2FDGa,uBAAuB;kBANnC,SAAS;+BACI,kBAAkB,mBAGX,uBAAuB,CAAC,MAAM;;0BAgE1C,MAAM;2BAAC,OAAO;;0BAEd,MAAM;2BAAC,uBAAuB;wGA3D5B,QAAQ;sBADd,KAAK;gBAOE,IAAI;sBADX,SAAS;uBAAC,MAAM;gBAQV,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB;gBAOtB,UAAU;sBADhB,KAAK;gBAiBC,SAAS;sBADf,KAAK;gBAOC,gBAAgB;sBADtB,MAAM;gBAOA,oBAAoB;sBAD1B,MAAM;gBA+BC,gBAAgB;sBADvB,YAAY;uBAAC,WAAW;gBAWjB,iBAAiB;sBADxB,YAAY;uBAAC,YAAY","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    Output,\n    Renderer2,\n    ViewChild,\n} from '@angular/core';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCategory, ScIUrls } from '@snabcentr/client-core';\nimport { TuiSizeS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\n/**\n * Карточка категории.\n */\n@Component({\n    selector: 'sc-category-card',\n    templateUrl: './sc-category-card.component.html',\n    styleUrls: ['./sc-category-card.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCategoryCardComponent {\n    /**\n     * Информация о категории.\n     */\n    @Input()\n    public category?: ScCategory;\n\n    /**\n     * Ссылка на {@link HTMLElement} блока названия.\n     */\n    @ViewChild('name')\n    private name: ElementRef<HTMLElement>;\n\n    /**\n     * Размер карточки категории.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS = 'm';\n\n    /**\n     * Признак, что карточка является скелетоном.\n     */\n    @Input()\n    public isSkeleton: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = this.authService.getAuthChange();\n\n    /**\n     * Признак что категория имеет поведение наведения и скрытия названия.\n     */\n    @Input()\n    public isHovered: boolean = false;\n\n    /**\n     * Событие нажатия на карточку категории.\n     */\n    @Output()\n    public clickOnCardEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Событие нажатия на кнопку избранной категории.\n     */\n    @Output()\n    public clickOnFavoriteEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Инициализирует экземпляр класса {@link CategoryCardComponent}.\n     *\n     * @param urls Список ссылок на разделы backend'a.\n     * @param renderer Экземпляр базового класса для реализации пользовательского рендеринга.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        private renderer: Renderer2,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly authService: ScAuthService,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию (\"product_not_found\").\n     *\n     * @param category Информация о категории.\n     */\n    public getCategoryImgURL(category: ScCategory): string {\n        return category.properties?.image ? this.urls.imgServerUrl + '/' + category.properties?.image : this.pathImageNotFound;\n    }\n\n    /**\n     * Обработчик события mousemove.\n     */\n    @HostListener('mousemove')\n    private moveEnterHandler(): void {\n        if (this.isHovered) {\n            this.name.nativeElement.style.maxHeight = `${this.name.nativeElement.scrollHeight / 16}rem`;\n        }\n    }\n\n    /**\n     * Обработчик события mouseleave.\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler(): void {\n        if (this.isHovered) {\n            this.renderer.setStyle(this.name.nativeElement, 'maxHeight', null);\n        }\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<div class=\"relative group\">\n    <button\n        (click)=\"clickOnCardEvent.emit(category)\"\n        [class.pointer-events-none]=\"!category || isSkeleton\"\n        class=\"category-btn flex flex-col relative rounded-tui-radius-m overflow-hidden shadow-sc-2\"\n    >\n        <div class=\"img-wrapper w-full grow bg-tui-base-02 overflow-hidden\">\n            <img *ngIf=\"category && !isSkeleton\" [src]=\"getCategoryImgURL(category)\" [alt]=\"category.name\" class=\"w-full\" />\n        </div>\n        <div #name [attr.data-is-hovered]=\"!isHovered\" class=\"name grid items-center justify-center font-bold bg-white w-full overflow-hidden shrink-0 duration-300\">\n            <span *ngIf=\"category && !isSkeleton; else skeletonName\">{{ category.name }}</span>\n            <ng-template #skeletonName>\n                <div class=\"skeleton-name bg-tui-base-02 rounded-tui-radius-s\"></div>\n            </ng-template>\n        </div>\n    </button>\n    <sc-favorite-btn\n        *ngIf=\"category && !isSkeleton && (authStatus$ | async)\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"category.isFavorite\"\n        (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n        class=\"absolute left-1 top-1\"\n    ></sc-favorite-btn>\n</div>\n"]}
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-category-card.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.ts","../../../../../projects/client-ui/catalog/category-card/sc-category-card.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,uBAAuB,EAEvB,SAAS,EAET,YAAY,EACZ,WAAW,EACX,YAAY,EACZ,MAAM,EACN,KAAK,EACL,MAAM,EAEN,SAAS,EACZ,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAsC,MAAM,wBAAwB,CAAC;;;;;AAI9G;;GAEG;AAOH,MAAM,OAAO,uBAAuB;IAsDhC;;;;;;OAMG;IACH,YACsC,IAAa,EACvC,QAAmB,EACuB,iBAAyB,EAC1D,WAA0B,EAC1B,GAAsB;QAJL,SAAI,GAAJ,IAAI,CAAS;QACvC,aAAQ,GAAR,QAAQ,CAAW;QACuB,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,gBAAW,GAAX,WAAW,CAAe;QAC1B,QAAG,GAAH,GAAG,CAAmB;QArD3C;;WAEG;QAGI,SAAI,GAAa,GAAG,CAAC;QAE5B;;WAEG;QAEI,eAAU,GAAY,KAAK,CAAC;QAEnC;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QACa,gBAAW,GAAwB,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;QAEpF;;WAEG;QAEI,cAAS,GAAY,KAAK,CAAC;QAElC;;WAEG;QAEI,qBAAgB,GAA6B,IAAI,YAAY,EAAc,CAAC;QAEnF;;WAEG;QAEI,yBAAoB,GAA6B,IAAI,YAAY,EAAc,CAAC;IAepF,CAAC;IAEJ;;;;OAIG;IACI,iBAAiB,CAAC,QAAoB;QACzC,OAAO,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,GAAG,GAAG,QAAQ,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;IAC3H,CAAC;IAED;;OAEG;IAEK,gBAAgB;QACpB,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,SAAS,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,GAAG,EAAE,KAAK,CAAC;SAC/F;IACL,CAAC;IAED;;OAEG;IAEK,iBAAiB;QACrB,IAAI,IAAI,CAAC,SAAS,EAAE;YAChB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;SACtE;IACL,CAAC;IAED;;OAEG;IACI,YAAY;QACf,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;;oHAvGQ,uBAAuB,kBA8DpB,OAAO,sCAEP,uBAAuB;wGAhE1B,uBAAuB,yeC3BpC,0vCAwBA;2FDGa,uBAAuB;kBANnC,SAAS;+BACI,kBAAkB,mBAGX,uBAAuB,CAAC,MAAM;;0BAgE1C,MAAM;2BAAC,OAAO;;0BAEd,MAAM;2BAAC,uBAAuB;wGA3D5B,QAAQ;sBADd,KAAK;gBAOE,IAAI;sBADX,SAAS;uBAAC,MAAM;gBAQV,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB;gBAOtB,UAAU;sBADhB,KAAK;gBAiBC,SAAS;sBADf,KAAK;gBAOC,gBAAgB;sBADtB,MAAM;gBAOA,oBAAoB;sBAD1B,MAAM;gBA+BC,gBAAgB;sBADvB,YAAY;uBAAC,WAAW;gBAWjB,iBAAiB;sBADxB,YAAY;uBAAC,YAAY","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    Output,\n    Renderer2,\n    ViewChild\n} from '@angular/core';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCategory, ScIUrls } from '@snabcentr/client-core';\nimport { TuiSizeS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\n/**\n * Карточка категории.\n */\n@Component({\n    selector: 'sc-category-card',\n    templateUrl: './sc-category-card.component.html',\n    styleUrls: ['./sc-category-card.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScCategoryCardComponent {\n    /**\n     * Информация о категории.\n     */\n    @Input()\n    public category?: ScCategory;\n\n    /**\n     * Ссылка на {@link HTMLElement} блока названия.\n     */\n    @ViewChild('name')\n    private name: ElementRef<HTMLElement>;\n\n    /**\n     * Размер карточки категории.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS = 'm';\n\n    /**\n     * Признак, что карточка является скелетоном.\n     */\n    @Input()\n    public isSkeleton: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = this.authService.getAuthChange();\n\n    /**\n     * Признак что категория имеет поведение наведения и скрытия названия.\n     */\n    @Input()\n    public isHovered: boolean = false;\n\n    /**\n     * Событие нажатия на карточку категории.\n     */\n    @Output()\n    public clickOnCardEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Событие нажатия на кнопку избранной категории.\n     */\n    @Output()\n    public clickOnFavoriteEvent: EventEmitter<ScCategory> = new EventEmitter<ScCategory>();\n\n    /**\n     * Инициализирует экземпляр класса {@link CategoryCardComponent}.\n     *\n     * @param urls Список ссылок на разделы backend'a.\n     * @param renderer Экземпляр базового класса для реализации пользовательского рендеринга.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        private renderer: Renderer2,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly authService: ScAuthService,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Возвращает путь к изображению категории. Если путь отсутствует, то вернёт изображение по-умолчанию (\"product_not_found\").\n     *\n     * @param category Информация о категории.\n     */\n    public getCategoryImgURL(category: ScCategory): string {\n        return category.properties?.image ? this.urls.imgServerUrl + '/' + category.properties?.image : this.pathImageNotFound;\n    }\n\n    /**\n     * Обработчик события mousemove.\n     */\n    @HostListener('mousemove')\n    private moveEnterHandler(): void {\n        if (this.isHovered) {\n            this.name.nativeElement.style.maxHeight = `${this.name.nativeElement.scrollHeight / 16}rem`;\n        }\n    }\n\n    /**\n     * Обработчик события mouseleave.\n     */\n    @HostListener('mouseleave')\n    private mouseLeaveHandler(): void {\n        if (this.isHovered) {\n            this.renderer.setStyle(this.name.nativeElement, 'maxHeight', null);\n        }\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<div class=\"relative group\">\n    <button\n        (click)=\"clickOnCardEvent.emit(category)\"\n        [class.pointer-events-none]=\"!category || isSkeleton\"\n        class=\"category-btn flex flex-col relative rounded-tui-radius-m overflow-hidden shadow-sc-2\"\n    >\n        <div class=\"img-wrapper w-full grow bg-tui-base-02 overflow-hidden\">\n            <img *ngIf=\"category && !isSkeleton\" [src]=\"getCategoryImgURL(category)\" [alt]=\"category.name\" class=\"w-full\" />\n        </div>\n        <div #name [attr.data-is-hovered]=\"!isHovered\" class=\"name grid items-center justify-center font-bold bg-white w-full overflow-hidden shrink-0 duration-300\">\n            <span *ngIf=\"category && !isSkeleton; else skeletonName\">{{ category.name }}</span>\n            <ng-template #skeletonName>\n                <div class=\"skeleton-name bg-tui-base-02 rounded-tui-radius-s\"></div>\n            </ng-template>\n        </div>\n    </button>\n    <sc-favorite-btn\n        *ngIf=\"category && !isSkeleton && (authStatus$ | async)\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"category.isFavorite\"\n        (clickEvent)=\"clickOnFavoriteEvent.emit()\"\n        class=\"absolute left-1 top-1\"\n    ></sc-favorite-btn>\n</div>\n"]}
@@ -1,6 +1,6 @@
1
- import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, HostListener, Inject, Input, Optional, Output, Self, ViewChild, } from '@angular/core';
1
+ import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, HostListener, Inject, Input, Optional, Output, Self, ViewChild } from '@angular/core';
2
2
  import { NgControl } from '@angular/forms';
3
- import { stepValidator } from '../../validators/stepValidator';
3
+ import { stepValidator } from '../../validators';
4
4
  import { TUI_NUMBER_VALUE_TRANSFORMER, TuiInputNumberComponent } from '@taiga-ui/kit';
5
5
  import { AbstractTuiNullableControl } from '@taiga-ui/cdk';
6
6
  import * as i0 from "@angular/core";
@@ -155,4 +155,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
155
155
  type: HostListener,
156
156
  args: ['keydown.arrowUp', ['step']]
157
157
  }] } });
158
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-input-quantity.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/input-quantity/sc-input-quantity.component.ts","../../../../../projects/client-ui/catalog/input-quantity/sc-input-quantity.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    OnInit,\n    Optional,\n    Output,\n    Self,\n    ViewChild,\n} from '@angular/core';\nimport { FormControl, NgControl } from '@angular/forms';\nimport { stepValidator } from '../../validators/stepValidator';\nimport { TUI_NUMBER_VALUE_TRANSFORMER, TuiInputNumberComponent } from '@taiga-ui/kit';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nimport { TuiLoaderComponent, TuiSizeL, TuiSizeM, TuiSizeS } from '@taiga-ui/core';\nimport { AbstractTuiNullableControl, AbstractTuiValueTransformer } from '@taiga-ui/cdk';\n\n@Component({\n    selector: 'sc-input-quantity',\n    templateUrl: './sc-input-quantity.component.html',\n    styleUrls: ['./sc-input-quantity.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScInputQuantityComponent extends AbstractTuiNullableControl<number> implements OnInit {\n    /**\n     * Компонент поля ввода.\n     */\n    @ViewChild(TuiInputNumberComponent)\n    private readonly numberInput: TuiInputNumberComponent;\n\n    /**\n     * Шаг увеличения количества. Отвечает за проверку кратности ввода.\n     */\n    @Input()\n    public step: number = 1;\n\n    /**\n     * Признак, что необходимо отобразить {@link TuiLoaderComponent} над компонентом.\n     */\n    @Input()\n    public showLoader: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить кнопку очистки поля ввода.\n     */\n    @Input()\n    public showCross: boolean = true;\n\n    /**\n     * Формат внешнего вида компонента.\n     */\n    @Input()\n    @HostBinding('attr.data-appearance')\n    public appearance: 'primary' | 'secondary' = 'primary';\n\n    /**\n     * Признак, что компонент деактивирован.\n     */\n    @Input()\n    @HostBinding('attr.data-disabled')\n    public isDisabled: boolean = false;\n\n    /**\n     * Размер компонента.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeL | TuiSizeM | TuiSizeS = 'm';\n\n    /**\n     * Событие нажатия на кнопку \"Очистить\".\n     */\n    @Output()\n    public clickClearEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Возвращает элемент, который может быть сфокусирован.\n     */\n    public get nativeFocusableElement(): HTMLInputElement | null {\n        return this.numberInput.nativeFocusableElement;\n    }\n\n    /**\n     * Инициализирует экземпляр класса {@link ScInputQuantityComponent}.\n     *\n     * @param ngControl Объект поля ввода для настройки валидации.\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param transformer Преобразователь значений элемента формы.\n     */\n    public constructor(\n        @Optional()\n        @Self()\n        @Inject(NgControl)\n        ngControl: NgControl | null,\n        @Inject(ChangeDetectorRef)\n        cdr: ChangeDetectorRef,\n        @Optional()\n        @Inject(TUI_NUMBER_VALUE_TRANSFORMER)\n        transformer: AbstractTuiValueTransformer<number | null>\n    ) {\n        super(ngControl, cdr, transformer);\n    }\n\n    public get focused(): boolean {\n        return !!this.numberInput?.focused;\n    }\n\n    public get formControl(): FormControl<any> | null {\n        return this.control as FormControl<any> | null;\n    }\n\n    /** @inheritDoc */\n    public override ngOnInit(): void {\n        if (this.control) {\n            this.control.setValidators(stepValidator(this.step));\n        }\n    }\n\n    /**\n     * Обработчик события нажатия стрелок клавиатуры.\n     *\n     * @param step Шаг изменения количества.\n     */\n    @HostListener('keydown.arrowDown', ['-step'])\n    @HostListener('keydown.arrowUp', ['step'])\n    public onArrow(step: number | null): void {\n        if (!step) {\n            return;\n        }\n\n        if (step > 0) {\n            this.incident();\n        } else {\n            this.decrement();\n        }\n    }\n\n    /**\n     * Увеличивает значение в поле ввода на 1 шаг. Если число в поле ввода не кратно шагу, то увеличит до ближайшего кратного значения.\n     */\n    public incident(): void {\n        this.numberInput.onArrow(this.step - ((this.numberInput.value || 0) % this.step));\n    }\n\n    /**\n     * Уменьшает значение в поле ввода на 1 шаг. Если число в поле ввода не кратно шагу, то уменьшит до ближайшего кратного значения.\n     */\n    public decrement(): void {\n        this.numberInput.onArrow(-((this.numberInput.value || 0) % this.step) || -this.step);\n    }\n\n    /**\n     * Очищает поля ввода.\n     */\n    public clear(): void {\n        this.control?.patchValue(null);\n        this.clickClearEvent.emit();\n    }\n}\n","<tui-loader *ngIf=\"formControl\" class=\"w-full\" [overlay]=\"true\" [showLoader]=\"showLoader\" [size]=\"size\">\n    <div class=\"flex items-center text-center gap-1\">\n        <div class=\"field-with-button flex grow rounded-xl\">\n            <button\n                tuiIconButton\n                tuiMode=\"onLight\"\n                [appearance]=\"appearance\"\n                [disabled]=\"!numberInput.canDecrement\"\n                [focusable]=\"false\"\n                [size]=\"size\"\n                (click.prevent)=\"decrement()\"\n                (mousedown.prevent)=\"numberInput.nativeFocusableElement?.focus()\"\n            >\n                <tui-svg src=\"tuiIconMinusLarge\"></tui-svg>\n            </button>\n            <tui-input-number\n                #numberInput\n                [formControl]=\"formControl\"\n                [tuiHint]=\"([] | tuiFieldError | async )?.message\"\n                [tuiTextfieldLabelOutside]=\"true\"\n                [min]=\"step\"\n                [tuiTextfieldSize]=\"size\"\n                [required]=\"true\"\n                [style.text-align]=\"'center'\"\n                [style.font-weight]=\"700\"\n                class=\"grow\"\n            >\n            </tui-input-number>\n            <button\n                tuiIconButton\n                tuiMode=\"onLight\"\n                [appearance]=\"appearance\"\n                [size]=\"size\"\n                [disabled]=\"!numberInput.canIncrement\"\n                [focusable]=\"false\"\n                (click.prevent)=\"incident()\"\n                (mousedown.prevent)=\"numberInput.nativeFocusableElement?.focus()\"\n            >\n                <tui-svg src=\"tuiIconPlusLarge\"></tui-svg>\n            </button>\n        </div>\n        <button\n            *ngIf=\"showCross\"\n            tuiIconButton\n            tuiMode=\"onLight\"\n            [appearance]=\"appearance\"\n            [size]=\"size\"\n            [disabled]=\"!numberInput.canIncrement\"\n            [focusable]=\"false\"\n            (click.prevent)=\"clear()\"\n            (mousedown.prevent)=\"numberInput.nativeFocusableElement?.focus()\"\n        >\n            <tui-svg src=\"tuiIconCloseLarge\"></tui-svg>\n        </button>\n    </div>\n</tui-loader>\n"]}
158
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-input-quantity.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/input-quantity/sc-input-quantity.component.ts","../../../../../projects/client-ui/catalog/input-quantity/sc-input-quantity.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    HostBinding,\n    HostListener,\n    Inject,\n    Input,\n    OnInit,\n    Optional,\n    Output,\n    Self,\n    ViewChild\n} from '@angular/core';\nimport { FormControl, NgControl } from '@angular/forms';\nimport { stepValidator } from '../../validators';\nimport { TUI_NUMBER_VALUE_TRANSFORMER, TuiInputNumberComponent } from '@taiga-ui/kit';\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nimport { TuiLoaderComponent, TuiSizeL, TuiSizeM, TuiSizeS } from '@taiga-ui/core';\nimport { AbstractTuiNullableControl, AbstractTuiValueTransformer } from '@taiga-ui/cdk';\n\n@Component({\n    selector: 'sc-input-quantity',\n    templateUrl: './sc-input-quantity.component.html',\n    styleUrls: ['./sc-input-quantity.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScInputQuantityComponent extends AbstractTuiNullableControl<number> implements OnInit {\n    /**\n     * Компонент поля ввода.\n     */\n    @ViewChild(TuiInputNumberComponent)\n    private readonly numberInput: TuiInputNumberComponent;\n\n    /**\n     * Шаг увеличения количества. Отвечает за проверку кратности ввода.\n     */\n    @Input()\n    public step: number = 1;\n\n    /**\n     * Признак, что необходимо отобразить {@link TuiLoaderComponent} над компонентом.\n     */\n    @Input()\n    public showLoader: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить кнопку очистки поля ввода.\n     */\n    @Input()\n    public showCross: boolean = true;\n\n    /**\n     * Формат внешнего вида компонента.\n     */\n    @Input()\n    @HostBinding('attr.data-appearance')\n    public appearance: 'primary' | 'secondary' = 'primary';\n\n    /**\n     * Признак, что компонент деактивирован.\n     */\n    @Input()\n    @HostBinding('attr.data-disabled')\n    public isDisabled: boolean = false;\n\n    /**\n     * Размер компонента.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeL | TuiSizeM | TuiSizeS = 'm';\n\n    /**\n     * Событие нажатия на кнопку \"Очистить\".\n     */\n    @Output()\n    public clickClearEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Возвращает элемент, который может быть сфокусирован.\n     */\n    public get nativeFocusableElement(): HTMLInputElement | null {\n        return this.numberInput.nativeFocusableElement;\n    }\n\n    /**\n     * Инициализирует экземпляр класса {@link ScInputQuantityComponent}.\n     *\n     * @param ngControl Объект поля ввода для настройки валидации.\n     * @param cdr Объект для работы с обнаружением изменений.\n     * @param transformer Преобразователь значений элемента формы.\n     */\n    public constructor(\n        @Optional()\n        @Self()\n        @Inject(NgControl)\n            ngControl: NgControl | null,\n        @Inject(ChangeDetectorRef)\n            cdr: ChangeDetectorRef,\n        @Optional()\n        @Inject(TUI_NUMBER_VALUE_TRANSFORMER)\n            transformer: AbstractTuiValueTransformer<number | null>\n    ) {\n        super(ngControl, cdr, transformer);\n    }\n\n    public get focused(): boolean {\n        return !!this.numberInput?.focused;\n    }\n\n    public get formControl(): FormControl<any> | null {\n        return this.control as FormControl<any> | null;\n    }\n\n    /** @inheritDoc */\n    public override ngOnInit(): void {\n        if (this.control) {\n            this.control.setValidators(stepValidator(this.step));\n        }\n    }\n\n    /**\n     * Обработчик события нажатия стрелок клавиатуры.\n     *\n     * @param step Шаг изменения количества.\n     */\n    @HostListener('keydown.arrowDown', ['-step'])\n    @HostListener('keydown.arrowUp', ['step'])\n    public onArrow(step: number | null): void {\n        if (!step) {\n            return;\n        }\n\n        if (step > 0) {\n            this.incident();\n        } else {\n            this.decrement();\n        }\n    }\n\n    /**\n     * Увеличивает значение в поле ввода на 1 шаг. Если число в поле ввода не кратно шагу, то увеличит до ближайшего кратного значения.\n     */\n    public incident(): void {\n        this.numberInput.onArrow(this.step - ((this.numberInput.value || 0) % this.step));\n    }\n\n    /**\n     * Уменьшает значение в поле ввода на 1 шаг. Если число в поле ввода не кратно шагу, то уменьшит до ближайшего кратного значения.\n     */\n    public decrement(): void {\n        this.numberInput.onArrow(-((this.numberInput.value || 0) % this.step) || -this.step);\n    }\n\n    /**\n     * Очищает поля ввода.\n     */\n    public clear(): void {\n        this.control?.patchValue(null);\n        this.clickClearEvent.emit();\n    }\n}\n","<tui-loader *ngIf=\"formControl\" class=\"w-full\" [overlay]=\"true\" [showLoader]=\"showLoader\" [size]=\"size\">\n    <div class=\"flex items-center text-center gap-1\">\n        <div class=\"field-with-button flex grow rounded-xl\">\n            <button\n                tuiIconButton\n                tuiMode=\"onLight\"\n                [appearance]=\"appearance\"\n                [disabled]=\"!numberInput.canDecrement\"\n                [focusable]=\"false\"\n                [size]=\"size\"\n                (click.prevent)=\"decrement()\"\n                (mousedown.prevent)=\"numberInput.nativeFocusableElement?.focus()\"\n            >\n                <tui-svg src=\"tuiIconMinusLarge\"></tui-svg>\n            </button>\n            <tui-input-number\n                #numberInput\n                [formControl]=\"formControl\"\n                [tuiHint]=\"([] | tuiFieldError | async )?.message\"\n                [tuiTextfieldLabelOutside]=\"true\"\n                [min]=\"step\"\n                [tuiTextfieldSize]=\"size\"\n                [required]=\"true\"\n                [style.text-align]=\"'center'\"\n                [style.font-weight]=\"700\"\n                class=\"grow\"\n            >\n            </tui-input-number>\n            <button\n                tuiIconButton\n                tuiMode=\"onLight\"\n                [appearance]=\"appearance\"\n                [size]=\"size\"\n                [disabled]=\"!numberInput.canIncrement\"\n                [focusable]=\"false\"\n                (click.prevent)=\"incident()\"\n                (mousedown.prevent)=\"numberInput.nativeFocusableElement?.focus()\"\n            >\n                <tui-svg src=\"tuiIconPlusLarge\"></tui-svg>\n            </button>\n        </div>\n        <button\n            *ngIf=\"showCross\"\n            tuiIconButton\n            tuiMode=\"onLight\"\n            [appearance]=\"appearance\"\n            [size]=\"size\"\n            [disabled]=\"!numberInput.canIncrement\"\n            [focusable]=\"false\"\n            (click.prevent)=\"clear()\"\n            (mousedown.prevent)=\"numberInput.nativeFocusableElement?.focus()\"\n        >\n            <tui-svg src=\"tuiIconCloseLarge\"></tui-svg>\n        </button>\n    </div>\n</tui-loader>\n"]}