@snabcentr/client-ui 3.51.1 → 3.51.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,4 @@
1
- import { EventEmitter, QueryList, TemplateRef } from '@angular/core';
1
+ import { EventEmitter, InputSignal, QueryList, TemplateRef } from '@angular/core';
2
2
  import { ScBanner } from '@snabcentr/client-core';
3
3
  import * as i0 from "@angular/core";
4
4
  /**
@@ -16,7 +16,7 @@ export declare class ScBannerComponent {
16
16
  /**
17
17
  * Местоположение баннера.
18
18
  */
19
- bannerLocation?: string;
19
+ bannerLocation: InputSignal<string | undefined>;
20
20
  /**
21
21
  * Признак, что компонент должен растягиваться.
22
22
  */
@@ -52,7 +52,14 @@ export declare class ScBannerComponent {
52
52
  /**
53
53
  * Список баннеров.
54
54
  */
55
+ protected readonly banners$: import("rxjs").Observable<ScBanner[]>;
55
56
  protected readonly banners: import("@angular/core").Signal<ScBanner[]>;
57
+ /**
58
+ * Общее количество слайдов (баннеры + контент-проекция).
59
+ * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
60
+ * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
61
+ */
62
+ readonly totalItems: import("@angular/core").Signal<number>;
56
63
  /**
57
64
  * trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,
58
65
  * чтобы карусель не видела 0 слайдов между тиками и не вызывала tuiClamp(…, -1).
@@ -61,16 +68,6 @@ export declare class ScBannerComponent {
61
68
  * @param banner Данные о баннере.
62
69
  */
63
70
  trackByBannerId(_index: number, banner: ScBanner): number;
64
- /**
65
- * Общее количество слайдов (баннеры + контент-проекция).
66
- * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
67
- * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
68
- */
69
- get totalItems(): number;
70
- /**
71
- * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.
72
- */
73
- private get isHidden();
74
71
  /**
75
72
  * Обработчик нажатия на баннер.
76
73
  *
@@ -78,5 +75,5 @@ export declare class ScBannerComponent {
78
75
  */
79
76
  onClick(banner: ScBanner): void;
80
77
  static ɵfac: i0.ɵɵFactoryDeclaration<ScBannerComponent, never>;
81
- static ɵcmp: i0.ɵɵComponentDeclaration<ScBannerComponent, "sc-banner", never, { "navigateButton": { "alias": "navigateButton"; "required": false; }; "duration": { "alias": "duration"; "required": false; }; "bannerLocation": { "alias": "bannerLocation"; "required": false; }; "resizable": { "alias": "resizable"; "required": false; }; }, { "loadBannersEvent": "loadBannersEvent"; "clickBannerEvent": "clickBannerEvent"; }, ["bannersListRef"], never, false, never>;
78
+ static ɵcmp: i0.ɵɵComponentDeclaration<ScBannerComponent, "sc-banner", never, { "navigateButton": { "alias": "navigateButton"; "required": false; }; "duration": { "alias": "duration"; "required": false; }; "bannerLocation": { "alias": "bannerLocation"; "required": false; "isSignal": true; }; "resizable": { "alias": "resizable"; "required": false; }; }, { "loadBannersEvent": "loadBannersEvent"; "clickBannerEvent": "clickBannerEvent"; }, ["bannersListRef"], never, false, never>;
82
79
  }
@@ -1,5 +1,5 @@
1
1
  import { Signal } from '@angular/core';
2
- import { ScINewOrderItemBase, ScOrderItem, ScProduct } from '@snabcentr/client-core';
2
+ import { ScCart, ScINewOrderItemBase, ScIOrderEditorLoader, ScOrderDraft, ScOrderItem, ScProduct } from '@snabcentr/client-core';
3
3
  import { TuiDialogContext } from '@taiga-ui/core';
4
4
  import { Observable, Subject } from 'rxjs';
5
5
  import { ScAddOrEditingCartItemFormComponent } from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';
@@ -19,7 +19,7 @@ export declare class ScAddOrEditingCartItemDialogComponent {
19
19
  /**
20
20
  * {@link Observable} запроса добавления / изменения товара в корзине.
21
21
  */
22
- readonly submit$: Observable<import("@snabcentr/client-core").ScCart>;
22
+ readonly submit$: Observable<{} | ScCart | ScOrderDraft>;
23
23
  /**
24
24
  * {@link Observable} изменения состояния загрузки данных.
25
25
  */
@@ -30,6 +30,7 @@ export declare class ScAddOrEditingCartItemDialogComponent {
30
30
  readonly context: TuiDialogContext<boolean, {
31
31
  product: ScProduct;
32
32
  orderItem: ScOrderItem | undefined;
33
+ orderId: number | undefined;
33
34
  }>;
34
35
  /**
35
36
  * Данные о товаре.
@@ -40,9 +41,9 @@ export declare class ScAddOrEditingCartItemDialogComponent {
40
41
  */
41
42
  readonly orderItem: ScOrderItem | undefined;
42
43
  /**
43
- * Сервис для работы с корзиной.
44
+ * Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).
44
45
  */
45
- private readonly cartService;
46
+ protected readonly orderService: ScIOrderEditorLoader<ScOrderItem, ScCart | ScOrderDraft>;
46
47
  static ɵfac: i0.ɵɵFactoryDeclaration<ScAddOrEditingCartItemDialogComponent, never>;
47
48
  static ɵcmp: i0.ɵɵComponentDeclaration<ScAddOrEditingCartItemDialogComponent, "sc-add-or-editing-cart-item-dialog", never, {}, {}, never, never, true, never>;
48
49
  }
@@ -1,8 +1,8 @@
1
- import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, HostBinding, inject, Input, Output } from '@angular/core';
2
- import { toSignal } from '@angular/core/rxjs-interop';
1
+ import { ChangeDetectionStrategy, Component, computed, ContentChildren, EventEmitter, HostBinding, inject, Input, input, Output, } from '@angular/core';
2
+ import { toObservable, toSignal } from '@angular/core/rxjs-interop';
3
3
  import { IntersectionObserverService } from '@ng-web-apis/intersection-observer';
4
4
  import { ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';
5
- import { map, shareReplay, tap } from 'rxjs';
5
+ import { map, shareReplay, switchMap, tap } from 'rxjs';
6
6
  import { SC_BANNER_DURATION } from '../providers';
7
7
  import * as i0 from "@angular/core";
8
8
  import * as i1 from "@angular/common";
@@ -23,6 +23,10 @@ export class ScBannerComponent {
23
23
  * Интервал автоматической смены слайдов в миллисекундах (используйте 0, чтобы отключить автоматическую смену слайда).
24
24
  */
25
25
  this.duration = inject(SC_BANNER_DURATION);
26
+ /**
27
+ * Местоположение баннера.
28
+ */
29
+ this.bannerLocation = input();
26
30
  /**
27
31
  * Признак, что компонент должен растягиваться.
28
32
  */
@@ -54,14 +58,21 @@ export class ScBannerComponent {
54
58
  /**
55
59
  * Список баннеров.
56
60
  */
57
- this.banners = toSignal(this.bannerService.banners$.pipe(map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()), tap((banners) => {
61
+ this.banners$ = toObservable(this.bannerLocation).pipe(switchMap((bannerLocation) => this.bannerService.banners$.pipe(map((banners) => banners.filter((banner) => banner.location === bannerLocation)), tap((banners) => {
58
62
  if (banners.length > 0 && !this.resizable) {
59
63
  this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;
60
64
  }
61
65
  this.loadBannersEvent.emit(banners.length);
62
- }),
66
+ }))),
63
67
  // eslint-disable-next-line rxjs/no-sharereplay
64
- shareReplay(1)), { initialValue: [] });
68
+ shareReplay(1));
69
+ this.banners = toSignal(this.banners$, { initialValue: [] });
70
+ /**
71
+ * Общее количество слайдов (баннеры + контент-проекция).
72
+ * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
73
+ * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
74
+ */
75
+ this.totalItems = computed(() => this.banners().length + this.bannersListRef.length);
65
76
  }
66
77
  /**
67
78
  * trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,
@@ -75,21 +86,6 @@ export class ScBannerComponent {
75
86
  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
76
87
  return banner.id ?? _index;
77
88
  }
78
- /**
79
- * Общее количество слайдов (баннеры + контент-проекция).
80
- * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать
81
- * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.
82
- */
83
- get totalItems() {
84
- // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
85
- return (this.banners()?.length ?? 0) + (this.bannersListRef?.length ?? 0);
86
- }
87
- /**
88
- * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.
89
- */
90
- get isHidden() {
91
- return this.totalItems === 0;
92
- }
93
89
  /**
94
90
  * Обработчик нажатия на баннер.
95
91
  *
@@ -107,17 +103,18 @@ export class ScBannerComponent {
107
103
  }
108
104
  }
109
105
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScBannerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
110
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: "navigateButton", duration: "duration", bannerLocation: "bannerLocation", resizable: "resizable" }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerEvent: "clickBannerEvent" }, host: { attributes: { "ngSkipHydration": "true" }, properties: { "style.aspect-ratio": "this.aspectRatio", "class.!hidden": "this.isHidden" } }, providers: [IntersectionObserverService], queries: [{ propertyName: "bannersListRef", predicate: ["banner"] }], ngImport: i0, template: "@let effectiveDuration = totalItems > 1 ? duration : 0;\n<tui-carousel\n [duration]=\"effectiveDuration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n>\n <ng-container *ngFor=\"let banner of banners(); let index = index; trackBy: trackByBannerId\">\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'image'\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n class=\"size-full\"\n >\n <picture>\n @if (banner.mediaFileWebp) {\n <source\n type=\"image/webp\"\n [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n />\n }\n <img\n [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n [alt]=\"banner.title\"\n class=\"size-full object-cover\"\n />\n </picture>\n </a>\n </ng-container>\n </ng-container>\n </ng-container>\n <ng-container *ngFor=\"let item of bannersListRef\">\n <div\n *tuiItem\n [style.aspect-ratio]=\"aspectRatio\"\n class=\"size-full overflow-hidden\"\n >\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n</tui-carousel>\n\n<div\n *ngIf=\"navigateButton && duration && banners() && totalItems > 1\"\n tuiTheme=\"light\"\n class=\"flex items-center\"\n>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-left\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.prev()\"\n class=\"!absolute left-2\"\n ></button>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-right\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.next()\"\n class=\"!absolute right-2\"\n ></button>\n</div>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}::ng-deep tui-carousel[resizable=true] .t-scroller,::ng-deep tui-carousel[resizable=true] .t-items{width:100%;height:100%}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i1.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i2.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i3.TuiItem, selector: "[tuiItem]" }, { kind: "component", type: i4.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange", "shift"] }, { kind: "pipe", type: i5.ScMediaImageTransformerPipe, name: "scMediaImageTransformer" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
106
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScBannerComponent, selector: "sc-banner", inputs: { navigateButton: { classPropertyName: "navigateButton", publicName: "navigateButton", isSignal: false, isRequired: false, transformFunction: null }, duration: { classPropertyName: "duration", publicName: "duration", isSignal: false, isRequired: false, transformFunction: null }, bannerLocation: { classPropertyName: "bannerLocation", publicName: "bannerLocation", isSignal: true, isRequired: false, transformFunction: null }, resizable: { classPropertyName: "resizable", publicName: "resizable", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { loadBannersEvent: "loadBannersEvent", clickBannerEvent: "clickBannerEvent" }, host: { attributes: { "ngSkipHydration": "true" }, properties: { "class.!hidden": "totalItems() === 0", "style.aspect-ratio": "this.aspectRatio" } }, providers: [IntersectionObserverService], queries: [{ propertyName: "bannersListRef", predicate: ["banner"] }], ngImport: i0, template: "@let effectiveDuration = totalItems() > 1 ? duration : 0;\n\n<tui-carousel\n [duration]=\"effectiveDuration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n>\n @for (banner of banners(); track trackByBannerId($index, banner)) {\n @switch (banner.mediaType) {\n @case ('image') {\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n class=\"size-full\"\n >\n <picture>\n @if (banner.mediaFileWebp) {\n <source\n type=\"image/webp\"\n [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n />\n }\n <img\n [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n [alt]=\"banner.title\"\n class=\"size-full object-cover\"\n />\n </picture>\n </a>\n }\n }\n }\n @for (item of bannersListRef; track $index) {\n <div\n *tuiItem\n [style.aspect-ratio]=\"aspectRatio\"\n class=\"size-full overflow-hidden\"\n >\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n }\n</tui-carousel>\n\n@if (navigateButton && duration && banners() && totalItems() > 1) {\n <div\n tuiTheme=\"light\"\n class=\"flex items-center\"\n >\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-left\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.prev()\"\n class=\"!absolute left-2\"\n ></button>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-right\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.next()\"\n class=\"!absolute right-2\"\n ></button>\n </div>\n}\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}::ng-deep tui-carousel[resizable=true] .t-scroller,::ng-deep tui-carousel[resizable=true] .t-items{width:100%;height:100%}\n"], dependencies: [{ kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i3.TuiItem, selector: "[tuiItem]" }, { kind: "component", type: i4.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange", "shift"] }, { kind: "pipe", type: i5.ScMediaImageTransformerPipe, name: "scMediaImageTransformer" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
111
107
  }
112
108
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScBannerComponent, decorators: [{
113
109
  type: Component,
114
- args: [{ selector: 'sc-banner', providers: [IntersectionObserverService], host: { ngSkipHydration: 'true' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "@let effectiveDuration = totalItems > 1 ? duration : 0;\n<tui-carousel\n [duration]=\"effectiveDuration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n>\n <ng-container *ngFor=\"let banner of banners(); let index = index; trackBy: trackByBannerId\">\n <ng-container [ngSwitch]=\"banner.mediaType\">\n <ng-container *ngSwitchCase=\"'image'\">\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n class=\"size-full\"\n >\n <picture>\n @if (banner.mediaFileWebp) {\n <source\n type=\"image/webp\"\n [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n />\n }\n <img\n [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n [alt]=\"banner.title\"\n class=\"size-full object-cover\"\n />\n </picture>\n </a>\n </ng-container>\n </ng-container>\n </ng-container>\n <ng-container *ngFor=\"let item of bannersListRef\">\n <div\n *tuiItem\n [style.aspect-ratio]=\"aspectRatio\"\n class=\"size-full overflow-hidden\"\n >\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n </ng-container>\n</tui-carousel>\n\n<div\n *ngIf=\"navigateButton && duration && banners() && totalItems > 1\"\n tuiTheme=\"light\"\n class=\"flex items-center\"\n>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-left\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.prev()\"\n class=\"!absolute left-2\"\n ></button>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-right\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.next()\"\n class=\"!absolute right-2\"\n ></button>\n</div>\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}::ng-deep tui-carousel[resizable=true] .t-scroller,::ng-deep tui-carousel[resizable=true] .t-items{width:100%;height:100%}\n"] }]
110
+ args: [{ selector: 'sc-banner', providers: [IntersectionObserverService], host: {
111
+ ngSkipHydration: 'true',
112
+ '[class.!hidden]': 'totalItems() === 0',
113
+ }, changeDetection: ChangeDetectionStrategy.OnPush, template: "@let effectiveDuration = totalItems() > 1 ? duration : 0;\n\n<tui-carousel\n [duration]=\"effectiveDuration\"\n #carousel\n [attr.resizable]=\"resizable\"\n class=\"size-full overflow-hidden rounded-xl bg-white\"\n [(index)]=\"currentBannerId\"\n>\n @for (banner of banners(); track trackByBannerId($index, banner)) {\n @switch (banner.mediaType) {\n @case ('image') {\n <a\n *tuiItem\n (click)=\"onClick(banner)\"\n target=\"_blank\"\n [title]=\"banner.title\"\n [style.aspect-ratio]=\"aspectRatio\"\n [attr.href]=\"banner.url ? banner.url : null\"\n class=\"size-full\"\n >\n <picture>\n @if (banner.mediaFileWebp) {\n <source\n type=\"image/webp\"\n [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n />\n }\n <img\n [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n [alt]=\"banner.title\"\n class=\"size-full object-cover\"\n />\n </picture>\n </a>\n }\n }\n }\n @for (item of bannersListRef; track $index) {\n <div\n *tuiItem\n [style.aspect-ratio]=\"aspectRatio\"\n class=\"size-full overflow-hidden\"\n >\n <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n </div>\n }\n</tui-carousel>\n\n@if (navigateButton && duration && banners() && totalItems() > 1) {\n <div\n tuiTheme=\"light\"\n class=\"flex items-center\"\n >\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-left\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.prev()\"\n class=\"!absolute left-2\"\n ></button>\n <button\n tuiIconButton\n iconStart=\"@tui.chevron-right\"\n size=\"m\"\n [style.border-radius.%]=\"100\"\n appearance=\"flat\"\n (click)=\"carousel.next()\"\n class=\"!absolute right-2\"\n ></button>\n </div>\n}\n", styles: [":host{--tui-carousel-padding: 0;display:flex;position:relative}::ng-deep tui-carousel[resizable=true] .t-scroller,::ng-deep tui-carousel[resizable=true] .t-items{width:100%;height:100%}\n"] }]
115
114
  }], propDecorators: { navigateButton: [{
116
115
  type: Input
117
116
  }], duration: [{
118
117
  type: Input
119
- }], bannerLocation: [{
120
- type: Input
121
118
  }], resizable: [{
122
119
  type: Input
123
120
  }], loadBannersEvent: [{
@@ -130,8 +127,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
130
127
  }], aspectRatio: [{
131
128
  type: HostBinding,
132
129
  args: ['style.aspect-ratio']
133
- }], isHidden: [{
134
- type: HostBinding,
135
- args: ['class.!hidden']
136
130
  }] } });
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-banner.component.js","sourceRoot":"","sources":["../../../../projects/client-ui/banner/sc-banner.component.ts","../../../../projects/client-ui/banner/sc-banner.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, HostBinding, inject, Input, Output, QueryList, TemplateRef } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { map, shareReplay, tap } from 'rxjs';\n\nimport { SC_BANNER_DURATION } from '../providers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    providers: [IntersectionObserverService],\n    host: { ngSkipHydration: 'true' },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScBannerComponent {\n    /**\n     * Признак, что необходимо показывать кнопки навигации.\n     */\n    @Input()\n    public navigateButton: boolean = true;\n\n    /**\n     * Интервал автоматической смены слайдов в миллисекундах (используйте 0, чтобы отключить автоматическую смену слайда).\n     */\n    @Input()\n    public duration: number = inject(SC_BANNER_DURATION);\n\n    /**\n     * Местоположение баннера.\n     */\n    @Input()\n    public bannerLocation?: string;\n\n    /**\n     * Признак, что компонент должен растягиваться.\n     */\n    @Input()\n    public resizable: boolean = false;\n\n    /**\n     * Событие загрузки баннеров с количеством полученных баннеров.\n     */\n    @Output()\n    public loadBannersEvent: EventEmitter<number> = new EventEmitter<number>();\n\n    /**\n     * Событие нажатия на изображение баннера.\n     */\n    @Output()\n    public clickBannerEvent: EventEmitter<ScBanner> = new EventEmitter<ScBanner>();\n\n    /**\n     * Список ссылок на элемент представлений шаблонов.\n     */\n    @ContentChildren('banner')\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    public bannersListRef: QueryList<TemplateRef<any>>;\n\n    /**\n     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    protected currentBannerId: number = 0;\n\n    /**\n     * Сервис для работы с данными о баннерах.\n     */\n    private readonly bannerService = inject(ScBannerService);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Список баннеров.\n     */\n    protected readonly banners = toSignal(\n        this.bannerService.banners$.pipe(\n            map((banners) => banners.filter((banner) => banner.location === this.bannerLocation).reverse()),\n            tap((banners) => {\n                if (banners.length > 0 && !this.resizable) {\n                    this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                }\n\n                this.loadBannersEvent.emit(banners.length);\n            }),\n            // eslint-disable-next-line rxjs/no-sharereplay\n            shareReplay(1)\n        ),\n        { initialValue: [] as ScBanner[] }\n    );\n\n    /**\n     * trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,\n     * чтобы карусель не видела 0 слайдов между тиками и не вызывала tuiClamp(…, -1).\n     *\n     * @param _index Индекс.\n     * @param banner Данные о баннере.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public trackByBannerId(_index: number, banner: ScBanner): number {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        return banner.id ?? _index;\n    }\n\n    /**\n     * Общее количество слайдов (баннеры + контент-проекция).\n     * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать\n     * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.\n     */\n    public get totalItems(): number {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        return (this.banners()?.length ?? 0) + (this.bannersListRef?.length ?? 0);\n    }\n\n    /**\n     * Свойство, от которого зависит наличие класса `!hidden` у `:host` компонента.\n     */\n    @HostBinding('class.!hidden')\n    private get isHidden(): boolean {\n        return this.totalItems === 0;\n    }\n\n    /**\n     * Обработчик нажатия на баннер.\n     *\n     * @param banner Баннер, по ссылке которого совершён переход.\n     */\n    public onClick(banner: ScBanner): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.bannerClick,\n            params: {\n                banner_id: banner.id,\n            },\n        });\n\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\n}\n","@let effectiveDuration = totalItems > 1 ? duration : 0;\n<tui-carousel\n    [duration]=\"effectiveDuration\"\n    #carousel\n    [attr.resizable]=\"resizable\"\n    class=\"size-full overflow-hidden rounded-xl bg-white\"\n    [(index)]=\"currentBannerId\"\n>\n    <ng-container *ngFor=\"let banner of banners(); let index = index; trackBy: trackByBannerId\">\n        <ng-container [ngSwitch]=\"banner.mediaType\">\n            <ng-container *ngSwitchCase=\"'image'\">\n                <a\n                    *tuiItem\n                    (click)=\"onClick(banner)\"\n                    target=\"_blank\"\n                    [title]=\"banner.title\"\n                    [style.aspect-ratio]=\"aspectRatio\"\n                    [attr.href]=\"banner.url ? banner.url : null\"\n                    class=\"size-full\"\n                >\n                    <picture>\n                        @if (banner.mediaFileWebp) {\n                            <source\n                                type=\"image/webp\"\n                                [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n                            />\n                        }\n                        <img\n                            [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n                            [alt]=\"banner.title\"\n                            class=\"size-full object-cover\"\n                        />\n                    </picture>\n                </a>\n            </ng-container>\n        </ng-container>\n    </ng-container>\n    <ng-container *ngFor=\"let item of bannersListRef\">\n        <div\n            *tuiItem\n            [style.aspect-ratio]=\"aspectRatio\"\n            class=\"size-full overflow-hidden\"\n        >\n            <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n        </div>\n    </ng-container>\n</tui-carousel>\n\n<div\n    *ngIf=\"navigateButton && duration && banners() && totalItems > 1\"\n    tuiTheme=\"light\"\n    class=\"flex items-center\"\n>\n    <button\n        tuiIconButton\n        iconStart=\"@tui.chevron-left\"\n        size=\"m\"\n        [style.border-radius.%]=\"100\"\n        appearance=\"flat\"\n        (click)=\"carousel.prev()\"\n        class=\"!absolute left-2\"\n    ></button>\n    <button\n        tuiIconButton\n        iconStart=\"@tui.chevron-right\"\n        size=\"m\"\n        [style.border-radius.%]=\"100\"\n        appearance=\"flat\"\n        (click)=\"carousel.next()\"\n        class=\"!absolute right-2\"\n    ></button>\n</div>\n"]}
131
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-banner.component.js","sourceRoot":"","sources":["../../../../projects/client-ui/banner/sc-banner.component.ts","../../../../projects/client-ui/banner/sc-banner.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    ContentChildren,\n    EventEmitter,\n    HostBinding,\n    inject,\n    Input,\n    input,\n    InputSignal,\n    Output,\n    QueryList,\n    TemplateRef,\n} from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { IntersectionObserverService } from '@ng-web-apis/intersection-observer';\nimport { ScBanner, ScBannerService, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { map, shareReplay, switchMap, tap } from 'rxjs';\n\nimport { SC_BANNER_DURATION } from '../providers';\n\n/**\n * Баннер с прокруткой переданных {@link TemplateRef} элементов, и баннеров локации.\n */\n@Component({\n    selector: 'sc-banner',\n    templateUrl: './sc-banner.component.html',\n    styleUrls: ['./sc-banner.component.scss'],\n    providers: [IntersectionObserverService],\n    host: {\n        ngSkipHydration: 'true',\n        '[class.!hidden]': 'totalItems() === 0',\n    },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScBannerComponent {\n    /**\n     * Признак, что необходимо показывать кнопки навигации.\n     */\n    @Input()\n    public navigateButton: boolean = true;\n\n    /**\n     * Интервал автоматической смены слайдов в миллисекундах (используйте 0, чтобы отключить автоматическую смену слайда).\n     */\n    @Input()\n    public duration: number = inject(SC_BANNER_DURATION);\n\n    /**\n     * Местоположение баннера.\n     */\n    public bannerLocation: InputSignal<string | undefined> = input<string>();\n\n    /**\n     * Признак, что компонент должен растягиваться.\n     */\n    @Input()\n    public resizable: boolean = false;\n\n    /**\n     * Событие загрузки баннеров с количеством полученных баннеров.\n     */\n    @Output()\n    public loadBannersEvent: EventEmitter<number> = new EventEmitter<number>();\n\n    /**\n     * Событие нажатия на изображение баннера.\n     */\n    @Output()\n    public clickBannerEvent: EventEmitter<ScBanner> = new EventEmitter<ScBanner>();\n\n    /**\n     * Список ссылок на элемент представлений шаблонов.\n     */\n    @ContentChildren('banner')\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    public bannersListRef: QueryList<TemplateRef<any>>;\n\n    /**\n     * Свойство, от которого зависит соотношение `:host` компонента.\n     */\n    @HostBinding('style.aspect-ratio')\n    public aspectRatio: string = '';\n\n    /**\n     * Идентификатор текущего баннера.\n     */\n    protected currentBannerId: number = 0;\n\n    /**\n     * Сервис для работы с данными о баннерах.\n     */\n    private readonly bannerService = inject(ScBannerService);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Список баннеров.\n     */\n    protected readonly banners$ = toObservable(this.bannerLocation).pipe(\n        switchMap((bannerLocation) =>\n            this.bannerService.banners$.pipe(\n                map((banners) => banners.filter((banner) => banner.location === bannerLocation)),\n                tap((banners) => {\n                    if (banners.length > 0 && !this.resizable) {\n                        this.aspectRatio = `${banners[0].width} / ${banners[0].height}`;\n                    }\n\n                    this.loadBannersEvent.emit(banners.length);\n                })\n            )\n        ),\n        // eslint-disable-next-line rxjs/no-sharereplay\n        shareReplay(1)\n    );\n\n    protected readonly banners = toSignal(this.banners$, { initialValue: [] as ScBanner[] });\n\n    /**\n     * Общее количество слайдов (баннеры + контент-проекция).\n     * Используется для отключения автоскролла при 0–1 слайдах, чтобы не вызывать\n     * tuiClamp с max < min в TuiCarouselComponent.onAutoscroll → updateIndex.\n     */\n    public readonly totalItems = computed(() => this.banners().length + this.bannersListRef.length);\n\n    /**\n     * trackBy для *ngFor баннеров: сохраняет узлы при обновлении списка,\n     * чтобы карусель не видела 0 слайдов между тиками и не вызывала tuiClamp(…, -1).\n     *\n     * @param _index Индекс.\n     * @param banner Данные о баннере.\n     */\n    // eslint-disable-next-line class-methods-use-this\n    public trackByBannerId(_index: number, banner: ScBanner): number {\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        return banner.id ?? _index;\n    }\n\n    /**\n     * Обработчик нажатия на баннер.\n     *\n     * @param banner Баннер, по ссылке которого совершён переход.\n     */\n    public onClick(banner: ScBanner): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.bannerClick,\n            params: {\n                banner_id: banner.id,\n            },\n        });\n\n        if (banner.url) {\n            this.clickBannerEvent.emit(banner);\n        }\n    }\n}\n","@let effectiveDuration = totalItems() > 1 ? duration : 0;\n\n<tui-carousel\n    [duration]=\"effectiveDuration\"\n    #carousel\n    [attr.resizable]=\"resizable\"\n    class=\"size-full overflow-hidden rounded-xl bg-white\"\n    [(index)]=\"currentBannerId\"\n>\n    @for (banner of banners(); track trackByBannerId($index, banner)) {\n        @switch (banner.mediaType) {\n            @case ('image') {\n                <a\n                    *tuiItem\n                    (click)=\"onClick(banner)\"\n                    target=\"_blank\"\n                    [title]=\"banner.title\"\n                    [style.aspect-ratio]=\"aspectRatio\"\n                    [attr.href]=\"banner.url ? banner.url : null\"\n                    class=\"size-full\"\n                >\n                    <picture>\n                        @if (banner.mediaFileWebp) {\n                            <source\n                                type=\"image/webp\"\n                                [srcset]=\"banner.mediaFileWebp | scMediaImageTransformer\"\n                            />\n                        }\n                        <img\n                            [src]=\"banner.mediaFile | scMediaImageTransformer: true\"\n                            [alt]=\"banner.title\"\n                            class=\"size-full object-cover\"\n                        />\n                    </picture>\n                </a>\n            }\n        }\n    }\n    @for (item of bannersListRef; track $index) {\n        <div\n            *tuiItem\n            [style.aspect-ratio]=\"aspectRatio\"\n            class=\"size-full overflow-hidden\"\n        >\n            <ng-container [ngTemplateOutlet]=\"item\"></ng-container>\n        </div>\n    }\n</tui-carousel>\n\n@if (navigateButton && duration && banners() && totalItems() > 1) {\n    <div\n        tuiTheme=\"light\"\n        class=\"flex items-center\"\n    >\n        <button\n            tuiIconButton\n            iconStart=\"@tui.chevron-left\"\n            size=\"m\"\n            [style.border-radius.%]=\"100\"\n            appearance=\"flat\"\n            (click)=\"carousel.prev()\"\n            class=\"!absolute left-2\"\n        ></button>\n        <button\n            tuiIconButton\n            iconStart=\"@tui.chevron-right\"\n            size=\"m\"\n            [style.border-radius.%]=\"100\"\n            appearance=\"flat\"\n            (click)=\"carousel.next()\"\n            class=\"!absolute right-2\"\n        ></button>\n    </div>\n}\n"]}
@@ -2,7 +2,7 @@
2
2
  import { HttpErrorResponse } from '@angular/common/http';
3
3
  import { ChangeDetectionStrategy, Component, inject, viewChild } from '@angular/core';
4
4
  import { toSignal } from '@angular/core/rxjs-interop';
5
- import { ScCartService } from '@snabcentr/client-core';
5
+ import { SC_ORDER_LOADER } from '@snabcentr/client-core';
6
6
  import { tuiIsFalsy } from '@taiga-ui/cdk';
7
7
  import { injectContext } from '@taiga-ui/polymorpheus';
8
8
  import { isObject } from 'lodash-es';
@@ -25,7 +25,9 @@ export class ScAddOrEditingCartItemDialogComponent {
25
25
  /**
26
26
  * {@link Observable} запроса добавления / изменения товара в корзине.
27
27
  */
28
- this.submit$ = this.onSubmit.pipe(switchMap((value) => (this.orderItem && !('productId' in value) ? this.cartService.updateProduct$(this.orderItem.id, value) : this.cartService.addProduct$(value)).pipe(tap(() => {
28
+ this.submit$ = this.onSubmit.pipe(switchMap((value) => (this.orderItem && !('productId' in value) && 'updateProduct$' in this.orderService
29
+ ? this.orderService.updateProduct$(this.orderItem.id, value, this.context.data.orderId)
30
+ : this.orderService.addProduct$(value, this.context.data.orderId)).pipe(tap(() => {
29
31
  this.context.$implicit.complete();
30
32
  }), catchError((error) => {
31
33
  if (error instanceof HttpErrorResponse) {
@@ -60,9 +62,9 @@ export class ScAddOrEditingCartItemDialogComponent {
60
62
  */
61
63
  this.orderItem = this.context.data.orderItem;
62
64
  /**
63
- * Сервис для работы с корзиной.
65
+ * Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).
64
66
  */
65
- this.cartService = inject(ScCartService);
67
+ this.orderService = inject(SC_ORDER_LOADER);
66
68
  }
67
69
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScAddOrEditingCartItemDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
68
70
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScAddOrEditingCartItemDialogComponent, isStandalone: true, selector: "sc-add-or-editing-cart-item-dialog", viewQueries: [{ propertyName: "formComponent", first: true, predicate: ScAddOrEditingCartItemFormComponent, descendants: true, isSignal: true }], ngImport: i0, template: "@if (product) {\n <sc-add-or-editing-cart-item-form\n [product]=\"product\"\n [orderItem]=\"orderItem\"\n (addProduct)=\"onSubmit.next($event)\"\n (editCartItem)=\"onSubmit.next($event)\"\n [isLoading]=\"loading()\"\n ></sc-add-or-editing-cart-item-form>\n}\n", dependencies: [{ kind: "component", type: ScAddOrEditingCartItemFormComponent, selector: "sc-add-or-editing-cart-item-form", inputs: ["product", "orderItem", "isLoading"], outputs: ["addProduct", "editCartItem"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
@@ -71,4 +73,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
71
73
  type: Component,
72
74
  args: [{ standalone: true, selector: 'sc-add-or-editing-cart-item-dialog', imports: [ScAddOrEditingCartItemFormComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (product) {\n <sc-add-or-editing-cart-item-form\n [product]=\"product\"\n [orderItem]=\"orderItem\"\n (addProduct)=\"onSubmit.next($event)\"\n (editCartItem)=\"onSubmit.next($event)\"\n [isLoading]=\"loading()\"\n ></sc-add-or-editing-cart-item-form>\n}\n" }]
73
75
  }] });
74
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.ts","../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method,@typescript-eslint/no-unused-vars */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Signal, viewChild } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { ScCartService, ScIApiErrorResponse, ScINewOrderItemBase, ScOrderItem, ScProduct } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { injectContext } from '@taiga-ui/polymorpheus';\nimport { isObject } from 'lodash-es';\nimport { catchError, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ScAddOrEditingCartItemFormComponent } from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-dialog',\n    templateUrl: './sc-add-or-editing-cart-item-dialog.component.html',\n    imports: [ScAddOrEditingCartItemFormComponent],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemDialogComponent {\n    /**\n     * Компонент формы добавления / изменения товара в корзине.\n     */\n    public readonly formComponent = viewChild.required<ScAddOrEditingCartItemFormComponent>(ScAddOrEditingCartItemFormComponent);\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<ScINewOrderItemBase> = new Subject<ScINewOrderItemBase>();\n\n    /**\n     * {@link Observable} запроса добавления / изменения товара в корзине.\n     */\n    public readonly submit$ = this.onSubmit.pipe(\n        switchMap((value) =>\n            (this.orderItem && !('productId' in value) ? this.cartService.updateProduct$(this.orderItem.id, value) : this.cartService.addProduct$(value)).pipe(\n                tap(() => {\n                    this.context.$implicit.complete();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ScIApiErrorResponse;\n\n                        if (errors && isObject(errors)) {\n                            Object.entries(errors).forEach(([k, v]) => {\n                                this.formComponent().form.get(k)?.setErrors({ serverResponse: v });\n                            });\n                        }\n\n                        if (message) {\n                            this.formComponent().form.setErrors({ serverResponse: [message] });\n                        }\n\n                        this.formComponent().form.updateValueAndValidity();\n                        this.formComponent().form.markAsDirty();\n                    }\n                    return of();\n                })\n            )\n        ),\n        startWith(),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных.\n     */\n    public readonly loading: Signal<boolean> = toSignal(this.submit$.pipe(map(tuiIsFalsy)), { initialValue: false });\n\n    /**\n     * Контекст диалогового окна.\n     */\n    public readonly context = injectContext<TuiDialogContext<boolean, { product: ScProduct; orderItem: ScOrderItem | undefined }>>();\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = this.context.data.product;\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly orderItem = this.context.data.orderItem;\n\n    /**\n     * Сервис для работы с корзиной.\n     */\n    private readonly cartService: ScCartService = inject(ScCartService);\n}\n","@if (product) {\n    <sc-add-or-editing-cart-item-form\n        [product]=\"product\"\n        [orderItem]=\"orderItem\"\n        (addProduct)=\"onSubmit.next($event)\"\n        (editCartItem)=\"onSubmit.next($event)\"\n        [isLoading]=\"loading()\"\n    ></sc-add-or-editing-cart-item-form>\n}\n"]}
76
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.ts","../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method,@typescript-eslint/no-unused-vars */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Signal, viewChild } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { SC_ORDER_LOADER, ScCart, ScIApiErrorResponse, ScINewOrderItemBase, ScIOrderEditorLoader, ScOrderDraft, ScOrderItem, ScProduct } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { injectContext } from '@taiga-ui/polymorpheus';\nimport { isObject } from 'lodash-es';\nimport { catchError, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ScAddOrEditingCartItemFormComponent } from './add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-dialog',\n    templateUrl: './sc-add-or-editing-cart-item-dialog.component.html',\n    imports: [ScAddOrEditingCartItemFormComponent],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemDialogComponent {\n    /**\n     * Компонент формы добавления / изменения товара в корзине.\n     */\n    public readonly formComponent = viewChild.required<ScAddOrEditingCartItemFormComponent>(ScAddOrEditingCartItemFormComponent);\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<ScINewOrderItemBase> = new Subject<ScINewOrderItemBase>();\n\n    /**\n     * {@link Observable} запроса добавления / изменения товара в корзине.\n     */\n    public readonly submit$ = this.onSubmit.pipe(\n        switchMap((value) =>\n            (this.orderItem && !('productId' in value) && 'updateProduct$' in this.orderService\n                ? this.orderService.updateProduct$(this.orderItem.id, value, this.context.data.orderId)\n                : this.orderService.addProduct$(value, this.context.data.orderId)\n            ).pipe(\n                tap(() => {\n                    this.context.$implicit.complete();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ScIApiErrorResponse;\n\n                        if (errors && isObject(errors)) {\n                            Object.entries(errors).forEach(([k, v]) => {\n                                this.formComponent().form.get(k)?.setErrors({ serverResponse: v });\n                            });\n                        }\n\n                        if (message) {\n                            this.formComponent().form.setErrors({ serverResponse: [message] });\n                        }\n\n                        this.formComponent().form.updateValueAndValidity();\n                        this.formComponent().form.markAsDirty();\n                    }\n                    return of();\n                })\n            )\n        ),\n        startWith(),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных.\n     */\n    public readonly loading: Signal<boolean> = toSignal(this.submit$.pipe(map(tuiIsFalsy)), { initialValue: false });\n\n    /**\n     * Контекст диалогового окна.\n     */\n    public readonly context = injectContext<TuiDialogContext<boolean, { product: ScProduct; orderItem: ScOrderItem | undefined; orderId: number | undefined }>>();\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = this.context.data.product;\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly orderItem = this.context.data.orderItem;\n\n    /**\n     * Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).\n     */\n    protected readonly orderService = inject<ScIOrderEditorLoader<ScOrderItem, ScCart | ScOrderDraft>>(SC_ORDER_LOADER);\n}\n","@if (product) {\n    <sc-add-or-editing-cart-item-form\n        [product]=\"product\"\n        [orderItem]=\"orderItem\"\n        (addProduct)=\"onSubmit.next($event)\"\n        (editCartItem)=\"onSubmit.next($event)\"\n        [isLoading]=\"loading()\"\n    ></sc-add-or-editing-cart-item-form>\n}\n"]}
@@ -104,7 +104,7 @@ export class ScDraftComponent {
104
104
  provide: SC_ORDER_LOADER,
105
105
  useExisting: ScOrderDraftsService,
106
106
  },
107
- ], ngImport: i0, template: "@let items = draft()?.products;\n\n@if (items) {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @if (items.length) {\n <sc-order-items-list-by-directions\n scOrder\n [orderId]=\"draft()?.id\"\n [items]=\"items\"\n [editable]=\"isDraftPage()\"\n [selectable]=\"true\"\n (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n (continueClick)=\"goToNewOrder($event)\"\n >\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n </sc-order-items-list-by-directions>\n } @else {\n <p class=\"text-base font-bold\">\u0427\u0435\u0440\u043D\u043E\u0432\u0438\u043A \u043F\u0443\u0441\u0442.</p>\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n }\n </div>\n </div>\n} @else {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @for (item of [1, 2, 3, 4]; track item) {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n }\n <div class=\"flex gap-2\">\n <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n </div>\n </div>\n </div>\n}\n", dependencies: [{ kind: "component", type: ScOrderItemsListByDirectionsComponent, selector: "sc-order-items-list-by-directions", inputs: ["items", "selectable", "editable", "showContinueButton", "continueButtonText", "selectedItemsIds"], outputs: ["continueClick", "selectedItemsIdsChange"] }, { kind: "directive", type: ScOrderAccessorDirective, selector: "[scOrder]", inputs: ["orderId"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
107
+ ], ngImport: i0, template: "@let items = draft()?.products;\n\n@if (items) {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @if (items.length) {\n <sc-order-items-list-by-directions\n scOrder\n [orderId]=\"draft()?.id\"\n [items]=\"items\"\n [editable]=\"isDraftPage()\"\n [continueButtonText]=\"{ title: '\u041E\u0444\u043E\u0440\u043C\u0438\u0442\u044C \u0437\u0430\u043A\u0430\u0437', iconStart: '@tui.sc.send' }\"\n [selectable]=\"true\"\n (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n (continueClick)=\"goToNewOrder($event)\"\n >\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n </sc-order-items-list-by-directions>\n } @else {\n <p class=\"text-base font-bold\">\u0427\u0435\u0440\u043D\u043E\u0432\u0438\u043A \u043F\u0443\u0441\u0442.</p>\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n }\n </div>\n </div>\n} @else {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @for (item of [1, 2, 3, 4]; track item) {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n }\n <div class=\"flex gap-2\">\n <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n </div>\n </div>\n </div>\n}\n", dependencies: [{ kind: "component", type: ScOrderItemsListByDirectionsComponent, selector: "sc-order-items-list-by-directions", inputs: ["items", "selectable", "editable", "showContinueButton", "continueButtonText", "selectedItemsIds"], outputs: ["continueClick", "selectedItemsIdsChange"] }, { kind: "directive", type: ScOrderAccessorDirective, selector: "[scOrder]", inputs: ["orderId"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
108
108
  }
109
109
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScDraftComponent, decorators: [{
110
110
  type: Component,
@@ -113,6 +113,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
113
113
  provide: SC_ORDER_LOADER,
114
114
  useExisting: ScOrderDraftsService,
115
115
  },
116
- ], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let items = draft()?.products;\n\n@if (items) {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @if (items.length) {\n <sc-order-items-list-by-directions\n scOrder\n [orderId]=\"draft()?.id\"\n [items]=\"items\"\n [editable]=\"isDraftPage()\"\n [selectable]=\"true\"\n (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n (continueClick)=\"goToNewOrder($event)\"\n >\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n </sc-order-items-list-by-directions>\n } @else {\n <p class=\"text-base font-bold\">\u0427\u0435\u0440\u043D\u043E\u0432\u0438\u043A \u043F\u0443\u0441\u0442.</p>\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n }\n </div>\n </div>\n} @else {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @for (item of [1, 2, 3, 4]; track item) {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n }\n <div class=\"flex gap-2\">\n <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n </div>\n </div>\n </div>\n}\n" }]
116
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let items = draft()?.products;\n\n@if (items) {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @if (items.length) {\n <sc-order-items-list-by-directions\n scOrder\n [orderId]=\"draft()?.id\"\n [items]=\"items\"\n [editable]=\"isDraftPage()\"\n [continueButtonText]=\"{ title: '\u041E\u0444\u043E\u0440\u043C\u0438\u0442\u044C \u0437\u0430\u043A\u0430\u0437', iconStart: '@tui.sc.send' }\"\n [selectable]=\"true\"\n (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n (continueClick)=\"goToNewOrder($event)\"\n >\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n </sc-order-items-list-by-directions>\n } @else {\n <p class=\"text-base font-bold\">\u0427\u0435\u0440\u043D\u043E\u0432\u0438\u043A \u043F\u0443\u0441\u0442.</p>\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n }\n </div>\n </div>\n} @else {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @for (item of [1, 2, 3, 4]; track item) {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n }\n <div class=\"flex gap-2\">\n <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n </div>\n </div>\n </div>\n}\n" }]
117
117
  }] });
118
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-draft.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/draft/sc-draft.component.ts","../../../../../projects/client-ui/order/draft/sc-draft.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, DestroyRef, inject, model, OnInit, output } from '@angular/core';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { SC_ORDER_LOADER, ScOrderDraft, ScOrderDraftsService, scOrderIsLoaded, ScRouteKeys, ScSeoService } from '@snabcentr/client-core';\nimport { TuiButton, TuiDialogService, TuiIcons } from '@taiga-ui/core';\nimport { TUI_CONFIRM } from '@taiga-ui/kit';\nimport { distinctUntilChanged, filter, map, noop, switchMap } from 'rxjs';\n\nimport { ScOrderAccessorDirective } from '../directives/sc-order-accessor.directive';\nimport { ScOrderItemsByDirection } from '../models/sc-i-cart-items-by-direction';\nimport { ScOrderItemsListByDirectionsComponent } from '../order-items-list-by-directions/sc-order-items-list-by-directions.component';\n\n/**\n * Компонент страницы черновика заказа (аналог корзины для одного черновика).\n */\n@Component({\n    standalone: true,\n    selector: 'sc-draft',\n    templateUrl: './sc-draft.component.html',\n    imports: [ScOrderItemsListByDirectionsComponent, ScOrderAccessorDirective, TuiButton, TuiIcons],\n    providers: [\n        {\n            provide: SC_ORDER_LOADER,\n            useExisting: ScOrderDraftsService,\n        },\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScDraftComponent implements OnInit {\n    /**\n     * Сервис маршрутизации.\n     */\n    private readonly router = inject(Router);\n\n    /**\n     * Сервис предоставляющий доступ к данным о маршруте.\n     */\n    private readonly route: ActivatedRoute = inject(ActivatedRoute);\n\n    /**\n     * Сервис для работы с черновиками заказов.\n     */\n    private readonly draftsService = inject(ScOrderDraftsService);\n\n    /**\n     * Сигнал с данными текущего черновика заказа (из параметра offerId в URL).\n     */\n    public readonly draft = model<ScOrderDraft | null>(null);\n\n    /**\n     * Идентификаторы выбранных позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Источник события удаления или отмены заказа.\n     */\n    public readonly deleteEvent = output();\n\n    /**\n     * Сервис для работы с диалоговыми окнами.\n     */\n    private readonly dialogService = inject<TuiDialogService>(TuiDialogService);\n\n    /**\n     * Текущий ключ маршрута.\n     */\n    protected readonly isDraftPage = toSignal(\n        this.route.data.pipe(\n            map((data) => (data['routes_key'] as ScRouteKeys) === ScRouteKeys.draft),\n            distinctUntilChanged()\n        ),\n        { initialValue: false }\n    );\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private readonly destroyRef: DestroyRef = inject(DestroyRef);\n\n    /**\n     * Сервис установки SEO-данных приложения.\n     */\n    private readonly seoService: ScSeoService = inject(ScSeoService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (!this.draft()) {\n            this.draftsService\n                .get$(Number.parseInt(this.route.snapshot.paramMap.get('offerId') ?? '', 10))\n                .pipe(filter(scOrderIsLoaded), takeUntilDestroyed(this.destroyRef))\n                .subscribe((draft) => {\n                    this.draft.set(draft);\n                    this.seoService.setByDefaultAsNoIndexed('Черновики заказов');\n                });\n        }\n    }\n\n    /**\n     * Переход к оформлению заказа по выбранному направлению продаж.\n     *\n     * @param direction Группа позиций по направлению продаж.\n     */\n    protected goToNewOrder(direction: ScOrderItemsByDirection): void {\n        const ids = this.selectedItemsIds().filter((id) => direction.items.some((item) => item.id === id));\n\n        this.router.navigate(['orders', 'new', this.draft()?.id, direction.direction.id, 'draft'], { queryParams: { itemsIds: ids } }).catch(noop);\n    }\n\n    /**\n     * Удаляет черновик и перенаправляет к списку черновиков.\n     */\n    protected clearDraft(): void {\n        const { id } = this.draft() ?? {};\n\n        if (!id) {\n            return;\n        }\n\n        this.dialogService\n            .open<boolean>(TUI_CONFIRM, {\n                label: `Удалить черновик №${id}?`,\n                data: {\n                    content: `Это действие удалит черновик №${id}.`,\n                    yes: 'Да, удалить',\n                    no: 'Нет, оставить',\n                },\n            })\n            .pipe(\n                filter(Boolean),\n                switchMap(() => this.draftsService.delete$(id)),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe(() => {\n                this.deleteEvent.emit();\n                this.router.navigate(['/drafts']).catch(noop);\n            });\n    }\n}\n","@let items = draft()?.products;\n\n@if (items) {\n    <div class=\"flex flex-col gap-12\">\n        <div class=\"flex flex-col gap-5\">\n            @if (items.length) {\n                <sc-order-items-list-by-directions\n                    scOrder\n                    [orderId]=\"draft()?.id\"\n                    [items]=\"items\"\n                    [editable]=\"isDraftPage()\"\n                    [selectable]=\"true\"\n                    (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n                    (continueClick)=\"goToNewOrder($event)\"\n                >\n                    <div\n                        footerActions\n                        class=\"flex gap-2\"\n                    >\n                        <button\n                            tuiButton\n                            appearance=\"secondary\"\n                            (click)=\"clearDraft()\"\n                            iconStart=\"@tui.trash-2\"\n                        >\n                            Удалить черновик\n                        </button>\n                    </div>\n                </sc-order-items-list-by-directions>\n            } @else {\n                <p class=\"text-base font-bold\">Черновик пуст.</p>\n                <div\n                    footerActions\n                    class=\"flex gap-2\"\n                >\n                    <button\n                        tuiButton\n                        appearance=\"secondary\"\n                        (click)=\"clearDraft()\"\n                        iconStart=\"@tui.trash-2\"\n                    >\n                        Удалить черновик\n                    </button>\n                </div>\n            }\n        </div>\n    </div>\n} @else {\n    <div class=\"flex flex-col gap-12\">\n        <div class=\"flex flex-col gap-5\">\n            @for (item of [1, 2, 3, 4]; track item) {\n                <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n            }\n            <div class=\"flex gap-2\">\n                <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n                <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n            </div>\n        </div>\n    </div>\n}\n"]}
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-draft.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/draft/sc-draft.component.ts","../../../../../projects/client-ui/order/draft/sc-draft.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, DestroyRef, inject, model, OnInit, output } from '@angular/core';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { SC_ORDER_LOADER, ScOrderDraft, ScOrderDraftsService, scOrderIsLoaded, ScRouteKeys, ScSeoService } from '@snabcentr/client-core';\nimport { TuiButton, TuiDialogService, TuiIcons } from '@taiga-ui/core';\nimport { TUI_CONFIRM } from '@taiga-ui/kit';\nimport { distinctUntilChanged, filter, map, noop, switchMap } from 'rxjs';\n\nimport { ScOrderAccessorDirective } from '../directives/sc-order-accessor.directive';\nimport { ScOrderItemsByDirection } from '../models/sc-i-cart-items-by-direction';\nimport { ScOrderItemsListByDirectionsComponent } from '../order-items-list-by-directions/sc-order-items-list-by-directions.component';\n\n/**\n * Компонент страницы черновика заказа (аналог корзины для одного черновика).\n */\n@Component({\n    standalone: true,\n    selector: 'sc-draft',\n    templateUrl: './sc-draft.component.html',\n    imports: [ScOrderItemsListByDirectionsComponent, ScOrderAccessorDirective, TuiButton, TuiIcons],\n    providers: [\n        {\n            provide: SC_ORDER_LOADER,\n            useExisting: ScOrderDraftsService,\n        },\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScDraftComponent implements OnInit {\n    /**\n     * Сервис маршрутизации.\n     */\n    private readonly router = inject(Router);\n\n    /**\n     * Сервис предоставляющий доступ к данным о маршруте.\n     */\n    private readonly route: ActivatedRoute = inject(ActivatedRoute);\n\n    /**\n     * Сервис для работы с черновиками заказов.\n     */\n    private readonly draftsService = inject(ScOrderDraftsService);\n\n    /**\n     * Сигнал с данными текущего черновика заказа (из параметра offerId в URL).\n     */\n    public readonly draft = model<ScOrderDraft | null>(null);\n\n    /**\n     * Идентификаторы выбранных позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Источник события удаления или отмены заказа.\n     */\n    public readonly deleteEvent = output();\n\n    /**\n     * Сервис для работы с диалоговыми окнами.\n     */\n    private readonly dialogService = inject<TuiDialogService>(TuiDialogService);\n\n    /**\n     * Текущий ключ маршрута.\n     */\n    protected readonly isDraftPage = toSignal(\n        this.route.data.pipe(\n            map((data) => (data['routes_key'] as ScRouteKeys) === ScRouteKeys.draft),\n            distinctUntilChanged()\n        ),\n        { initialValue: false }\n    );\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private readonly destroyRef: DestroyRef = inject(DestroyRef);\n\n    /**\n     * Сервис установки SEO-данных приложения.\n     */\n    private readonly seoService: ScSeoService = inject(ScSeoService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (!this.draft()) {\n            this.draftsService\n                .get$(Number.parseInt(this.route.snapshot.paramMap.get('offerId') ?? '', 10))\n                .pipe(filter(scOrderIsLoaded), takeUntilDestroyed(this.destroyRef))\n                .subscribe((draft) => {\n                    this.draft.set(draft);\n                    this.seoService.setByDefaultAsNoIndexed('Черновики заказов');\n                });\n        }\n    }\n\n    /**\n     * Переход к оформлению заказа по выбранному направлению продаж.\n     *\n     * @param direction Группа позиций по направлению продаж.\n     */\n    protected goToNewOrder(direction: ScOrderItemsByDirection): void {\n        const ids = this.selectedItemsIds().filter((id) => direction.items.some((item) => item.id === id));\n\n        this.router.navigate(['orders', 'new', this.draft()?.id, direction.direction.id, 'draft'], { queryParams: { itemsIds: ids } }).catch(noop);\n    }\n\n    /**\n     * Удаляет черновик и перенаправляет к списку черновиков.\n     */\n    protected clearDraft(): void {\n        const { id } = this.draft() ?? {};\n\n        if (!id) {\n            return;\n        }\n\n        this.dialogService\n            .open<boolean>(TUI_CONFIRM, {\n                label: `Удалить черновик №${id}?`,\n                data: {\n                    content: `Это действие удалит черновик №${id}.`,\n                    yes: 'Да, удалить',\n                    no: 'Нет, оставить',\n                },\n            })\n            .pipe(\n                filter(Boolean),\n                switchMap(() => this.draftsService.delete$(id)),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe(() => {\n                this.deleteEvent.emit();\n                this.router.navigate(['/drafts']).catch(noop);\n            });\n    }\n}\n","@let items = draft()?.products;\n\n@if (items) {\n    <div class=\"flex flex-col gap-12\">\n        <div class=\"flex flex-col gap-5\">\n            @if (items.length) {\n                <sc-order-items-list-by-directions\n                    scOrder\n                    [orderId]=\"draft()?.id\"\n                    [items]=\"items\"\n                    [editable]=\"isDraftPage()\"\n                    [continueButtonText]=\"{ title: 'Оформить заказ', iconStart: '@tui.sc.send' }\"\n                    [selectable]=\"true\"\n                    (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n                    (continueClick)=\"goToNewOrder($event)\"\n                >\n                    <div\n                        footerActions\n                        class=\"flex gap-2\"\n                    >\n                        <button\n                            tuiButton\n                            appearance=\"secondary\"\n                            (click)=\"clearDraft()\"\n                            iconStart=\"@tui.trash-2\"\n                        >\n                            Удалить черновик\n                        </button>\n                    </div>\n                </sc-order-items-list-by-directions>\n            } @else {\n                <p class=\"text-base font-bold\">Черновик пуст.</p>\n                <div\n                    footerActions\n                    class=\"flex gap-2\"\n                >\n                    <button\n                        tuiButton\n                        appearance=\"secondary\"\n                        (click)=\"clearDraft()\"\n                        iconStart=\"@tui.trash-2\"\n                    >\n                        Удалить черновик\n                    </button>\n                </div>\n            }\n        </div>\n    </div>\n} @else {\n    <div class=\"flex flex-col gap-12\">\n        <div class=\"flex flex-col gap-5\">\n            @for (item of [1, 2, 3, 4]; track item) {\n                <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n            }\n            <div class=\"flex gap-2\">\n                <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n                <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n            </div>\n        </div>\n    </div>\n}\n"]}
@@ -66,10 +66,10 @@ export class ScOrderItemsListByDirectionsComponent {
66
66
  ]);
67
67
  }
68
68
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListByDirectionsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
69
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScOrderItemsListByDirectionsComponent, isStandalone: true, selector: "sc-order-items-list-by-directions", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: true, transformFunction: null }, selectable: { classPropertyName: "selectable", publicName: "selectable", isSignal: true, isRequired: false, transformFunction: null }, editable: { classPropertyName: "editable", publicName: "editable", isSignal: true, isRequired: false, transformFunction: null }, showContinueButton: { classPropertyName: "showContinueButton", publicName: "showContinueButton", isSignal: true, isRequired: false, transformFunction: null }, continueButtonText: { classPropertyName: "continueButtonText", publicName: "continueButtonText", isSignal: true, isRequired: false, transformFunction: null }, selectedItemsIds: { classPropertyName: "selectedItemsIds", publicName: "selectedItemsIds", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { continueClick: "continueClick", selectedItemsIds: "selectedItemsIdsChange" }, ngImport: i0, template: "<div class=\"flex flex-col gap-5\">\n <!-- \u0413\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432 \u043F\u043E \u043D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F\u043C \u043F\u0440\u043E\u0434\u0430\u0436. -->\n @for (direction of itemsByDirections(); track direction.direction?.id; let groupIndex = $index) {\n <div class=\"flex flex-col gap-2.5\">\n @if (direction.direction && itemsByDirections().length > 1) {\n <div class=\"text-base font-bold\">\u041D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u0435 \u043F\u0440\u043E\u0434\u0430\u0436: {{ direction.direction.description }}</div>\n }\n\n <sc-order-items-list\n [items]=\"direction.items\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, direction)\"\n />\n\n <!-- \u0424\u0443\u0442\u0435\u0440 \u0433\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432. -->\n <div class=\"flex flex-wrap-reverse grow gap-x-2.5 gap-y-4\">\n <ng-content select=\"[footerActions]\"></ng-content>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u043F\u0440\u043E\u0434\u043E\u043B\u0436\u0435\u043D\u0438\u044F. -->\n @if (showContinueButton()) {\n <button\n tuiButton\n type=\"button\"\n [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n [iconStart]=\"continueButtonText().iconStart\"\n class=\"ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText().title }}\n </button>\n }\n </div>\n </div>\n }\n\n @if (nullItems().length) {\n <div class=\"flex flex-col gap-2.5\">\n <div class=\"text-base font-bold\">\u0422\u043E\u0432\u0430\u0440\u044B \u0438 \u0443\u0441\u043B\u0443\u0433\u0438, \u043A\u043E\u0442\u043E\u0440\u044B\u0435 \u0431\u043E\u043B\u044C\u0448\u0435 \u043D\u0435 \u043F\u0440\u0435\u0434\u043E\u0441\u0442\u0430\u0432\u043B\u044F\u044E\u0442\u0441\u044F.</div>\n\n <sc-order-items-list\n [items]=\"nullItems()\"\n [selectable]=\"false\"\n [editable]=\"editable()\"\n />\n </div>\n }\n</div>\n", dependencies: [{ kind: "component", type: ScOrderItemsListComponent, selector: "sc-order-items-list", inputs: ["selectable", "editable", "items"], outputs: ["selectedItemsIdsChange"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
69
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScOrderItemsListByDirectionsComponent, isStandalone: true, selector: "sc-order-items-list-by-directions", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: true, transformFunction: null }, selectable: { classPropertyName: "selectable", publicName: "selectable", isSignal: true, isRequired: false, transformFunction: null }, editable: { classPropertyName: "editable", publicName: "editable", isSignal: true, isRequired: false, transformFunction: null }, showContinueButton: { classPropertyName: "showContinueButton", publicName: "showContinueButton", isSignal: true, isRequired: false, transformFunction: null }, continueButtonText: { classPropertyName: "continueButtonText", publicName: "continueButtonText", isSignal: true, isRequired: false, transformFunction: null }, selectedItemsIds: { classPropertyName: "selectedItemsIds", publicName: "selectedItemsIds", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { continueClick: "continueClick", selectedItemsIds: "selectedItemsIdsChange" }, ngImport: i0, template: "<div class=\"flex flex-col gap-5\">\n <!-- \u0413\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432 \u043F\u043E \u043D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F\u043C \u043F\u0440\u043E\u0434\u0430\u0436. -->\n @for (direction of itemsByDirections(); track direction.direction?.id; let groupIndex = $index) {\n <div class=\"flex flex-col gap-2.5\">\n @if (direction.direction && itemsByDirections().length > 1) {\n <div class=\"text-base font-bold\">\u041D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u0435 \u043F\u0440\u043E\u0434\u0430\u0436: {{ direction.direction.description }}</div>\n }\n\n <sc-order-items-list\n [items]=\"direction.items\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, direction)\"\n />\n\n <!-- \u0424\u0443\u0442\u0435\u0440 \u0433\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432. -->\n <div class=\"flex flex-col items-center sm:flex-row sm:items-start sm:flex-wrap-reverse grow gap-x-2.5 gap-y-4\">\n <ng-content select=\"[footerActions]\" />\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u043F\u0440\u043E\u0434\u043E\u043B\u0436\u0435\u043D\u0438\u044F. -->\n @if (showContinueButton()) {\n <button\n tuiButton\n type=\"button\"\n [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n [iconStart]=\"continueButtonText().iconStart\"\n class=\"sm:ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText().title }}\n </button>\n }\n </div>\n </div>\n }\n\n @if (nullItems().length) {\n <div class=\"flex flex-col gap-2.5\">\n <div class=\"text-base font-bold\">\u0422\u043E\u0432\u0430\u0440\u044B \u0438 \u0443\u0441\u043B\u0443\u0433\u0438, \u043A\u043E\u0442\u043E\u0440\u044B\u0435 \u0431\u043E\u043B\u044C\u0448\u0435 \u043D\u0435 \u043F\u0440\u0435\u0434\u043E\u0441\u0442\u0430\u0432\u043B\u044F\u044E\u0442\u0441\u044F.</div>\n\n <sc-order-items-list\n [items]=\"nullItems()\"\n [selectable]=\"false\"\n [editable]=\"editable()\"\n />\n </div>\n }\n</div>\n", dependencies: [{ kind: "component", type: ScOrderItemsListComponent, selector: "sc-order-items-list", inputs: ["selectable", "editable", "items"], outputs: ["selectedItemsIdsChange"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
70
70
  }
71
71
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListByDirectionsComponent, decorators: [{
72
72
  type: Component,
73
- args: [{ standalone: true, selector: 'sc-order-items-list-by-directions', imports: [ScOrderItemsListComponent, TuiButton], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col gap-5\">\n <!-- \u0413\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432 \u043F\u043E \u043D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F\u043C \u043F\u0440\u043E\u0434\u0430\u0436. -->\n @for (direction of itemsByDirections(); track direction.direction?.id; let groupIndex = $index) {\n <div class=\"flex flex-col gap-2.5\">\n @if (direction.direction && itemsByDirections().length > 1) {\n <div class=\"text-base font-bold\">\u041D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u0435 \u043F\u0440\u043E\u0434\u0430\u0436: {{ direction.direction.description }}</div>\n }\n\n <sc-order-items-list\n [items]=\"direction.items\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, direction)\"\n />\n\n <!-- \u0424\u0443\u0442\u0435\u0440 \u0433\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432. -->\n <div class=\"flex flex-wrap-reverse grow gap-x-2.5 gap-y-4\">\n <ng-content select=\"[footerActions]\"></ng-content>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u043F\u0440\u043E\u0434\u043E\u043B\u0436\u0435\u043D\u0438\u044F. -->\n @if (showContinueButton()) {\n <button\n tuiButton\n type=\"button\"\n [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n [iconStart]=\"continueButtonText().iconStart\"\n class=\"ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText().title }}\n </button>\n }\n </div>\n </div>\n }\n\n @if (nullItems().length) {\n <div class=\"flex flex-col gap-2.5\">\n <div class=\"text-base font-bold\">\u0422\u043E\u0432\u0430\u0440\u044B \u0438 \u0443\u0441\u043B\u0443\u0433\u0438, \u043A\u043E\u0442\u043E\u0440\u044B\u0435 \u0431\u043E\u043B\u044C\u0448\u0435 \u043D\u0435 \u043F\u0440\u0435\u0434\u043E\u0441\u0442\u0430\u0432\u043B\u044F\u044E\u0442\u0441\u044F.</div>\n\n <sc-order-items-list\n [items]=\"nullItems()\"\n [selectable]=\"false\"\n [editable]=\"editable()\"\n />\n </div>\n }\n</div>\n" }]
73
+ args: [{ standalone: true, selector: 'sc-order-items-list-by-directions', imports: [ScOrderItemsListComponent, TuiButton], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col gap-5\">\n <!-- \u0413\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432 \u043F\u043E \u043D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F\u043C \u043F\u0440\u043E\u0434\u0430\u0436. -->\n @for (direction of itemsByDirections(); track direction.direction?.id; let groupIndex = $index) {\n <div class=\"flex flex-col gap-2.5\">\n @if (direction.direction && itemsByDirections().length > 1) {\n <div class=\"text-base font-bold\">\u041D\u0430\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u0435 \u043F\u0440\u043E\u0434\u0430\u0436: {{ direction.direction.description }}</div>\n }\n\n <sc-order-items-list\n [items]=\"direction.items\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, direction)\"\n />\n\n <!-- \u0424\u0443\u0442\u0435\u0440 \u0433\u0440\u0443\u043F\u043F\u044B \u0442\u043E\u0432\u0430\u0440\u043E\u0432. -->\n <div class=\"flex flex-col items-center sm:flex-row sm:items-start sm:flex-wrap-reverse grow gap-x-2.5 gap-y-4\">\n <ng-content select=\"[footerActions]\" />\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u043F\u0440\u043E\u0434\u043E\u043B\u0436\u0435\u043D\u0438\u044F. -->\n @if (showContinueButton()) {\n <button\n tuiButton\n type=\"button\"\n [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n [iconStart]=\"continueButtonText().iconStart\"\n class=\"sm:ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText().title }}\n </button>\n }\n </div>\n </div>\n }\n\n @if (nullItems().length) {\n <div class=\"flex flex-col gap-2.5\">\n <div class=\"text-base font-bold\">\u0422\u043E\u0432\u0430\u0440\u044B \u0438 \u0443\u0441\u043B\u0443\u0433\u0438, \u043A\u043E\u0442\u043E\u0440\u044B\u0435 \u0431\u043E\u043B\u044C\u0448\u0435 \u043D\u0435 \u043F\u0440\u0435\u0434\u043E\u0441\u0442\u0430\u0432\u043B\u044F\u044E\u0442\u0441\u044F.</div>\n\n <sc-order-items-list\n [items]=\"nullItems()\"\n [selectable]=\"false\"\n [editable]=\"editable()\"\n />\n </div>\n }\n</div>\n" }]
74
74
  }] });
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-items-list-by-directions.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.ts","../../../../../projects/client-ui/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, inject, input, model, output } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { ScISalesDirection, ScOrderItem, ScReferencesService } from '@snabcentr/client-core';\nimport { TuiButton } from '@taiga-ui/core';\n\nimport { ScOrderItemsByDirection } from '../models/sc-i-cart-items-by-direction';\nimport { ScOrderItemsListComponent } from '../order-items-list/sc-order-items-list.component';\n\n@Component({\n    standalone: true,\n    selector: 'sc-order-items-list-by-directions',\n    templateUrl: './sc-order-items-list-by-directions.component.html',\n    imports: [ScOrderItemsListComponent, TuiButton],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemsListByDirectionsComponent {\n    /**\n     * Внутренний сигнал списка элементов корзины (товаров).\n     */\n    public readonly items = input.required<ScOrderItem[]>();\n\n    /**\n     * Включить выбор позиций чекбоксами.\n     */\n    public readonly selectable = input<boolean>(true);\n\n    /**\n     * Включить редактирование позиций.\n     */\n    public readonly editable = input<boolean>(false);\n\n    /**\n     * Включить отображение кнопки продолжения.\n     */\n    public readonly showContinueButton = input<boolean>(true);\n\n    /**\n     * Текст кнопки продолжения.\n     */\n    public readonly continueButtonText = input<{ title: string; iconStart: string }>({ title: 'Заказать', iconStart: '@tui.shopping-cart' });\n\n    /**\n     * Событие клика на кнопку продолжения.\n     */\n    public readonly continueClick = output<ScOrderItemsByDirection>();\n\n    /**\n     * Список направлений продаж из справочника.\n     */\n    private readonly directions = toSignal<ScISalesDirection[] | undefined>(inject(ScReferencesService).directions$, { initialValue: undefined });\n\n    /**\n     * Позиции корзины, сгруппированные по направлениям продаж.\n     * Если передан cartDirections, используется он, иначе группировка выполняется автоматически.\n     */\n    protected readonly itemsByDirections = computed<ScOrderItemsByDirection[]>(() => {\n        const items = this.items().filter((item) => !item.product.isNull && !item.product.isHidden);\n        const directions = this.directions();\n\n        return directions?.map((direction) => new ScOrderItemsByDirection(direction, items)).filter((group) => group.items.length > 0) ?? [];\n    });\n\n    /**\n     * Позиции корзины, которые больше не предоставляются.\n     */\n    protected readonly nullItems = computed<ScOrderItem[]>(() => this.items().filter((item) => item.product.isNull || item.product.isHidden));\n\n    /**\n     * Выходные данные при изменении выбора позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Обработчик изменения выбора позиций.\n     *\n     * @param itemsIds Идентификаторы выбранных позиций.\n     * @param direction Направление продаж.\n     */\n    protected selectedItemsHandler(itemsIds: string[], direction: ScOrderItemsByDirection): void {\n        this.selectedItemsIds.update((previous) => [\n            ...previous.filter((id) => !direction.items.some((item) => item.id === id) && this.items().some((item) => item.id === id)),\n            ...itemsIds,\n        ]);\n    }\n}\n","<div class=\"flex flex-col gap-5\">\n    <!-- Группы товаров по направлениям продаж. -->\n    @for (direction of itemsByDirections(); track direction.direction?.id; let groupIndex = $index) {\n        <div class=\"flex flex-col gap-2.5\">\n            @if (direction.direction && itemsByDirections().length > 1) {\n                <div class=\"text-base font-bold\">Направление продаж: {{ direction.direction.description }}</div>\n            }\n\n            <sc-order-items-list\n                [items]=\"direction.items\"\n                [selectable]=\"selectable()\"\n                [editable]=\"editable()\"\n                (selectedItemsIdsChange)=\"selectedItemsHandler($event, direction)\"\n            />\n\n            <!-- Футер группы товаров. -->\n            <div class=\"flex flex-wrap-reverse grow gap-x-2.5 gap-y-4\">\n                <ng-content select=\"[footerActions]\"></ng-content>\n\n                <!-- Кнопка продолжения. -->\n                @if (showContinueButton()) {\n                    <button\n                        tuiButton\n                        type=\"button\"\n                        [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n                        [iconStart]=\"continueButtonText().iconStart\"\n                        class=\"ml-auto\"\n                        (click)=\"continueClick.emit(direction)\"\n                    >\n                        {{ continueButtonText().title }}\n                    </button>\n                }\n            </div>\n        </div>\n    }\n\n    @if (nullItems().length) {\n        <div class=\"flex flex-col gap-2.5\">\n            <div class=\"text-base font-bold\">Товары и услуги, которые больше не предоставляются.</div>\n\n            <sc-order-items-list\n                [items]=\"nullItems()\"\n                [selectable]=\"false\"\n                [editable]=\"editable()\"\n            />\n        </div>\n    }\n</div>\n"]}
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-items-list-by-directions.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.ts","../../../../../projects/client-ui/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, inject, input, model, output } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { ScISalesDirection, ScOrderItem, ScReferencesService } from '@snabcentr/client-core';\nimport { TuiButton } from '@taiga-ui/core';\n\nimport { ScOrderItemsByDirection } from '../models/sc-i-cart-items-by-direction';\nimport { ScOrderItemsListComponent } from '../order-items-list/sc-order-items-list.component';\n\n@Component({\n    standalone: true,\n    selector: 'sc-order-items-list-by-directions',\n    templateUrl: './sc-order-items-list-by-directions.component.html',\n    imports: [ScOrderItemsListComponent, TuiButton],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemsListByDirectionsComponent {\n    /**\n     * Внутренний сигнал списка элементов корзины (товаров).\n     */\n    public readonly items = input.required<ScOrderItem[]>();\n\n    /**\n     * Включить выбор позиций чекбоксами.\n     */\n    public readonly selectable = input<boolean>(true);\n\n    /**\n     * Включить редактирование позиций.\n     */\n    public readonly editable = input<boolean>(false);\n\n    /**\n     * Включить отображение кнопки продолжения.\n     */\n    public readonly showContinueButton = input<boolean>(true);\n\n    /**\n     * Текст кнопки продолжения.\n     */\n    public readonly continueButtonText = input<{ title: string; iconStart: string }>({ title: 'Заказать', iconStart: '@tui.shopping-cart' });\n\n    /**\n     * Событие клика на кнопку продолжения.\n     */\n    public readonly continueClick = output<ScOrderItemsByDirection>();\n\n    /**\n     * Список направлений продаж из справочника.\n     */\n    private readonly directions = toSignal<ScISalesDirection[] | undefined>(inject(ScReferencesService).directions$, { initialValue: undefined });\n\n    /**\n     * Позиции корзины, сгруппированные по направлениям продаж.\n     * Если передан cartDirections, используется он, иначе группировка выполняется автоматически.\n     */\n    protected readonly itemsByDirections = computed<ScOrderItemsByDirection[]>(() => {\n        const items = this.items().filter((item) => !item.product.isNull && !item.product.isHidden);\n        const directions = this.directions();\n\n        return directions?.map((direction) => new ScOrderItemsByDirection(direction, items)).filter((group) => group.items.length > 0) ?? [];\n    });\n\n    /**\n     * Позиции корзины, которые больше не предоставляются.\n     */\n    protected readonly nullItems = computed<ScOrderItem[]>(() => this.items().filter((item) => item.product.isNull || item.product.isHidden));\n\n    /**\n     * Выходные данные при изменении выбора позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Обработчик изменения выбора позиций.\n     *\n     * @param itemsIds Идентификаторы выбранных позиций.\n     * @param direction Направление продаж.\n     */\n    protected selectedItemsHandler(itemsIds: string[], direction: ScOrderItemsByDirection): void {\n        this.selectedItemsIds.update((previous) => [\n            ...previous.filter((id) => !direction.items.some((item) => item.id === id) && this.items().some((item) => item.id === id)),\n            ...itemsIds,\n        ]);\n    }\n}\n","<div class=\"flex flex-col gap-5\">\n    <!-- Группы товаров по направлениям продаж. -->\n    @for (direction of itemsByDirections(); track direction.direction?.id; let groupIndex = $index) {\n        <div class=\"flex flex-col gap-2.5\">\n            @if (direction.direction && itemsByDirections().length > 1) {\n                <div class=\"text-base font-bold\">Направление продаж: {{ direction.direction.description }}</div>\n            }\n\n            <sc-order-items-list\n                [items]=\"direction.items\"\n                [selectable]=\"selectable()\"\n                [editable]=\"editable()\"\n                (selectedItemsIdsChange)=\"selectedItemsHandler($event, direction)\"\n            />\n\n            <!-- Футер группы товаров. -->\n            <div class=\"flex flex-col items-center sm:flex-row sm:items-start sm:flex-wrap-reverse grow gap-x-2.5 gap-y-4\">\n                <ng-content select=\"[footerActions]\" />\n\n                <!-- Кнопка продолжения. -->\n                @if (showContinueButton()) {\n                    <button\n                        tuiButton\n                        type=\"button\"\n                        [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n                        [iconStart]=\"continueButtonText().iconStart\"\n                        class=\"sm:ml-auto\"\n                        (click)=\"continueClick.emit(direction)\"\n                    >\n                        {{ continueButtonText().title }}\n                    </button>\n                }\n            </div>\n        </div>\n    }\n\n    @if (nullItems().length) {\n        <div class=\"flex flex-col gap-2.5\">\n            <div class=\"text-base font-bold\">Товары и услуги, которые больше не предоставляются.</div>\n\n            <sc-order-items-list\n                [items]=\"nullItems()\"\n                [selectable]=\"false\"\n                [editable]=\"editable()\"\n            />\n        </div>\n    }\n</div>\n"]}