@snabcentr/client-ui 3.50.0 → 3.50.2

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.
Files changed (23) hide show
  1. package/configurators/sandwich/sc-sandwich.component.d.ts +1 -1
  2. package/directives/abstract-price-card/abstract-sc-price-card.directive.d.ts +2 -2
  3. package/esm2022/catalog/price-card/sc-price-card.component.mjs +3 -3
  4. package/esm2022/catalog/price-card-inline/sc-price-card-inline.component.mjs +3 -3
  5. package/esm2022/configurators/sandwich/sc-sandwich.component.mjs +6 -7
  6. package/esm2022/directives/abstract-price-card/abstract-sc-price-card.directive.mjs +4 -4
  7. package/esm2022/order/directives/index.mjs +3 -0
  8. package/esm2022/order/directives/sc-order-accessor.directive.mjs +28 -0
  9. package/esm2022/order/directives/sc-order-options.mjs +11 -0
  10. package/esm2022/order/draft/sc-draft.component.mjs +6 -5
  11. package/esm2022/order/index.mjs +2 -1
  12. package/esm2022/order/order-item/sc-order-item.component.mjs +5 -4
  13. package/esm2022/order/order-items-list/sc-order-items-list.component.mjs +15 -12
  14. package/esm2022/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.mjs +7 -4
  15. package/esm2022/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.mjs +2 -2
  16. package/fesm2022/snabcentr-client-ui.mjs +74 -35
  17. package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
  18. package/order/directives/index.d.ts +2 -0
  19. package/order/directives/sc-order-accessor.directive.d.ts +13 -0
  20. package/order/directives/sc-order-options.d.ts +14 -0
  21. package/order/index.d.ts +1 -0
  22. package/order/order-items-list/sc-order-items-list.component.d.ts +5 -0
  23. package/package.json +2 -2
@@ -6,9 +6,10 @@ import { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';
6
6
  import { SC_ORDER_LOADER } from '@snabcentr/client-core';
7
7
  import { TuiCurrencyPipe } from '@taiga-ui/addon-commerce';
8
8
  import { TuiButton, TuiFormatNumberPipe } from '@taiga-ui/core';
9
- import { TuiCheckbox } from '@taiga-ui/kit';
9
+ import { TuiButtonLoading, TuiCheckbox } from '@taiga-ui/kit';
10
10
  import { catchError, finalize, map, of } from 'rxjs';
11
11
  import { SC_DIALOG_SERVICE_TOKEN } from '../../providers';
12
+ import { SC_ORDER_OPTIONS } from '../directives/sc-order-options';
12
13
  import { ScOrderItemComponent } from '../order-item/sc-order-item.component';
13
14
  import * as i0 from "@angular/core";
14
15
  import * as i1 from "@angular/forms";
@@ -34,6 +35,10 @@ export class ScOrderItemsListComponent {
34
35
  * Внутренняя группа формы для выбора позиций.
35
36
  */
36
37
  this.selectionFormGroup = new FormGroup({});
38
+ /**
39
+ * Опции заказа (черновика). Предоставляется директивой {@link ScOrderAccessorDirective} в контексте черновика.
40
+ */
41
+ this.orderOptions = inject(SC_ORDER_OPTIONS, { optional: true });
37
42
  /**
38
43
  * Идентификаторы выбранных позиций.
39
44
  */
@@ -118,17 +123,16 @@ export class ScOrderItemsListComponent {
118
123
  if (!this.canEdit()) {
119
124
  return;
120
125
  }
121
- priceCard.quantityShowLoader = true;
126
+ priceCard.showLoader.set(true);
122
127
  this.orderItemsListLoader
123
- .deleteProduct$(item)
128
+ .deleteProduct$(item, this.orderOptions?.orderId())
124
129
  .pipe(catchError(() => {
125
130
  priceCard.quantityControl.patchValue(item.quantity, {
126
131
  emitEvent: false,
127
132
  });
128
133
  return of();
129
134
  }), finalize(() => {
130
- priceCard.quantityShowLoader = false;
131
- priceCard.markForCheck();
135
+ priceCard.showLoader.set(false);
132
136
  }), takeUntilDestroyed(this.destroyRef))
133
137
  .subscribe();
134
138
  }
@@ -142,9 +146,9 @@ export class ScOrderItemsListComponent {
142
146
  if (!this.canEdit() || !priceCard.orderItem || !priceCard.product || !newQuantity) {
143
147
  return;
144
148
  }
145
- priceCard.quantityShowLoader = true;
149
+ priceCard.showLoader.set(true);
146
150
  this.orderItemsListLoader
147
- .updateProduct$(priceCard.orderItem.id, { quantity: newQuantity })
151
+ .updateProduct$(priceCard.orderItem.id, { quantity: newQuantity }, this.orderOptions?.orderId())
148
152
  .pipe(catchError(() => {
149
153
  // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
150
154
  priceCard.quantityControl.patchValue(priceCard.orderItem.quantity, {
@@ -152,8 +156,7 @@ export class ScOrderItemsListComponent {
152
156
  });
153
157
  return of();
154
158
  }), finalize(() => {
155
- priceCard.quantityShowLoader = false;
156
- priceCard.markForCheck();
159
+ priceCard.showLoader.set(false);
157
160
  }), takeUntilDestroyed(this.destroyRef))
158
161
  .subscribe();
159
162
  }
@@ -191,13 +194,13 @@ export class ScOrderItemsListComponent {
191
194
  this.scDialogService.openPriceHistoryChartDialog(product);
192
195
  }
193
196
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
194
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScOrderItemsListComponent, isStandalone: true, selector: "sc-order-items-list", inputs: { selectable: { classPropertyName: "selectable", publicName: "selectable", isSignal: true, isRequired: false, transformFunction: null }, editable: { classPropertyName: "editable", publicName: "editable", isSignal: true, isRequired: false, transformFunction: null }, items: { classPropertyName: "items", publicName: "items", isSignal: false, isRequired: true, transformFunction: null } }, outputs: { selectedItemsIdsChange: "selectedItemsIdsChange" }, ngImport: i0, template: "<div\n [formGroup]=\"selectionFormGroup\"\n class=\"flex flex-col gap-5\"\n>\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\n @for (item of itemsSignal(); track item.id) {\n @if ($index !== 0) {\n <hr class=\"mt-auto h-px w-full bg-tui-clear\" />\n }\n\n <div class=\"flex items-center gap-4 relative\">\n @let isActive = !selectable() || selectionFormGroup.get(item.id)?.value;\n\n <!-- \u0427\u0435\u043A\u0431\u043E\u043A\u0441 \u0432\u044B\u0431\u043E\u0440\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (selectable()) {\n <input\n tuiCheckbox\n type=\"checkbox\"\n size=\"m\"\n [formControlName]=\"item.id\"\n />\n }\n\n <!-- \u041A\u0430\u0440\u0442\u043E\u0447\u043A\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n <sc-order-item\n #itemComponent\n [product]=\"item.product\"\n [orderItem]=\"item\"\n [showQuantityControl]=\"canEdit() && !item.configurator\"\n [isShowEditControls]=\"canEdit()\"\n class=\"w-full duration-200\"\n [class.opacity-60]=\"selectable() && !selectionFormGroup.get(item.id)?.value\"\n (clickCardEvent)=\"openProductInfoDialog(item.product)\"\n (quantityValueChanges)=\"patchItem($event, itemComponent)\"\n (clickPriceHistoryEvent)=\"openPriceHistoryChartDialog(item.product)\"\n (clickSettings)=\"openSettingsDialog(item)\"\n [class.pointer-events-none]=\"!isActive\"\n />\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0443\u0434\u0430\u043B\u0435\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (canEdit()) {\n <button\n tuiIconButton\n iconStart=\"@tui.trash\"\n (click)=\"deleteItem(item, itemComponent)\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"!absolute left-1 top-1 lg:!relative lg:left-0 lg:top-0 lg:!self-center\"\n ></button>\n }\n </div>\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-end gap-2.5\">\n @let totalSum = sumSelectedItems();\n <!-- \u0418\u0442\u043E\u0433\u043E\u0432\u0430\u044F \u0441\u0443\u043C\u043C\u0430. -->\n <div class=\"text-base font-bold\">\u0418\u0442\u043E\u0433\u043E: {{ totalSum | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</div>\n </div>\n</div>\n", dependencies: [{ kind: "component", type: ScOrderItemComponent, selector: "sc-order-item", inputs: ["isShowEditControls"], outputs: ["clickSettings"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: TuiCheckbox, selector: "input[type=\"checkbox\"][tuiCheckbox]", inputs: ["size"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "pipe", type: TuiCurrencyPipe, name: "tuiCurrency" }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "pipe", type: TuiFormatNumberPipe, name: "tuiFormatNumber" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
197
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScOrderItemsListComponent, isStandalone: true, selector: "sc-order-items-list", inputs: { selectable: { classPropertyName: "selectable", publicName: "selectable", isSignal: true, isRequired: false, transformFunction: null }, editable: { classPropertyName: "editable", publicName: "editable", isSignal: true, isRequired: false, transformFunction: null }, items: { classPropertyName: "items", publicName: "items", isSignal: false, isRequired: true, transformFunction: null } }, outputs: { selectedItemsIdsChange: "selectedItemsIdsChange" }, ngImport: i0, template: "<div\n [formGroup]=\"selectionFormGroup\"\n class=\"flex flex-col gap-5\"\n>\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\n @for (item of itemsSignal(); track item.id) {\n @if ($index !== 0) {\n <hr class=\"mt-auto h-px w-full bg-tui-clear\" />\n }\n\n <div class=\"flex items-center gap-4 relative\">\n @let isActive = !selectable() || selectionFormGroup.get(item.id)?.value;\n\n <!-- \u0427\u0435\u043A\u0431\u043E\u043A\u0441 \u0432\u044B\u0431\u043E\u0440\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (selectable()) {\n <input\n tuiCheckbox\n [disabled]=\"itemComponent.showLoader()\"\n type=\"checkbox\"\n size=\"m\"\n [formControlName]=\"item.id\"\n />\n }\n\n <!-- \u041A\u0430\u0440\u0442\u043E\u0447\u043A\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n <sc-order-item\n #itemComponent\n [product]=\"item.product\"\n [orderItem]=\"item\"\n [showQuantityControl]=\"canEdit() && !item.configurator\"\n [isShowEditControls]=\"canEdit()\"\n class=\"w-full duration-200\"\n [class.opacity-60]=\"selectable() && !selectionFormGroup.get(item.id)?.value\"\n (clickCardEvent)=\"openProductInfoDialog(item.product)\"\n (quantityValueChanges)=\"patchItem($event, itemComponent)\"\n (clickPriceHistoryEvent)=\"openPriceHistoryChartDialog(item.product)\"\n (clickSettings)=\"openSettingsDialog(item)\"\n [class.pointer-events-none]=\"!isActive\"\n />\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0443\u0434\u0430\u043B\u0435\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (canEdit()) {\n <button\n tuiIconButton\n iconStart=\"@tui.trash\"\n (click)=\"deleteItem(item, itemComponent)\"\n [loading]=\"itemComponent.showLoader()\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"!absolute left-1 top-1 lg:!relative lg:left-0 lg:top-0 lg:!self-center\"\n ></button>\n }\n </div>\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-end gap-2.5\">\n @let totalSum = sumSelectedItems();\n <!-- \u0418\u0442\u043E\u0433\u043E\u0432\u0430\u044F \u0441\u0443\u043C\u043C\u0430. -->\n <div class=\"text-base font-bold\">\u0418\u0442\u043E\u0433\u043E: {{ totalSum | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</div>\n </div>\n</div>\n", dependencies: [{ kind: "component", type: ScOrderItemComponent, selector: "sc-order-item", inputs: ["isShowEditControls"], outputs: ["clickSettings"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: TuiCheckbox, selector: "input[type=\"checkbox\"][tuiCheckbox]", inputs: ["size"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: TuiCurrencyPipe, name: "tuiCurrency" }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "pipe", type: TuiFormatNumberPipe, name: "tuiFormatNumber" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
195
198
  }
196
199
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListComponent, decorators: [{
197
200
  type: Component,
198
- args: [{ standalone: true, selector: 'sc-order-items-list', imports: [ScOrderItemComponent, ReactiveFormsModule, TuiCheckbox, TuiButton, TuiCurrencyPipe, AsyncPipe, TuiFormatNumberPipe], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n [formGroup]=\"selectionFormGroup\"\n class=\"flex flex-col gap-5\"\n>\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\n @for (item of itemsSignal(); track item.id) {\n @if ($index !== 0) {\n <hr class=\"mt-auto h-px w-full bg-tui-clear\" />\n }\n\n <div class=\"flex items-center gap-4 relative\">\n @let isActive = !selectable() || selectionFormGroup.get(item.id)?.value;\n\n <!-- \u0427\u0435\u043A\u0431\u043E\u043A\u0441 \u0432\u044B\u0431\u043E\u0440\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (selectable()) {\n <input\n tuiCheckbox\n type=\"checkbox\"\n size=\"m\"\n [formControlName]=\"item.id\"\n />\n }\n\n <!-- \u041A\u0430\u0440\u0442\u043E\u0447\u043A\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n <sc-order-item\n #itemComponent\n [product]=\"item.product\"\n [orderItem]=\"item\"\n [showQuantityControl]=\"canEdit() && !item.configurator\"\n [isShowEditControls]=\"canEdit()\"\n class=\"w-full duration-200\"\n [class.opacity-60]=\"selectable() && !selectionFormGroup.get(item.id)?.value\"\n (clickCardEvent)=\"openProductInfoDialog(item.product)\"\n (quantityValueChanges)=\"patchItem($event, itemComponent)\"\n (clickPriceHistoryEvent)=\"openPriceHistoryChartDialog(item.product)\"\n (clickSettings)=\"openSettingsDialog(item)\"\n [class.pointer-events-none]=\"!isActive\"\n />\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0443\u0434\u0430\u043B\u0435\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (canEdit()) {\n <button\n tuiIconButton\n iconStart=\"@tui.trash\"\n (click)=\"deleteItem(item, itemComponent)\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"!absolute left-1 top-1 lg:!relative lg:left-0 lg:top-0 lg:!self-center\"\n ></button>\n }\n </div>\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-end gap-2.5\">\n @let totalSum = sumSelectedItems();\n <!-- \u0418\u0442\u043E\u0433\u043E\u0432\u0430\u044F \u0441\u0443\u043C\u043C\u0430. -->\n <div class=\"text-base font-bold\">\u0418\u0442\u043E\u0433\u043E: {{ totalSum | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</div>\n </div>\n</div>\n" }]
201
+ args: [{ standalone: true, selector: 'sc-order-items-list', imports: [ScOrderItemComponent, ReactiveFormsModule, TuiCheckbox, TuiButton, TuiButtonLoading, TuiCurrencyPipe, AsyncPipe, TuiFormatNumberPipe], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n [formGroup]=\"selectionFormGroup\"\n class=\"flex flex-col gap-5\"\n>\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\n @for (item of itemsSignal(); track item.id) {\n @if ($index !== 0) {\n <hr class=\"mt-auto h-px w-full bg-tui-clear\" />\n }\n\n <div class=\"flex items-center gap-4 relative\">\n @let isActive = !selectable() || selectionFormGroup.get(item.id)?.value;\n\n <!-- \u0427\u0435\u043A\u0431\u043E\u043A\u0441 \u0432\u044B\u0431\u043E\u0440\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (selectable()) {\n <input\n tuiCheckbox\n [disabled]=\"itemComponent.showLoader()\"\n type=\"checkbox\"\n size=\"m\"\n [formControlName]=\"item.id\"\n />\n }\n\n <!-- \u041A\u0430\u0440\u0442\u043E\u0447\u043A\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n <sc-order-item\n #itemComponent\n [product]=\"item.product\"\n [orderItem]=\"item\"\n [showQuantityControl]=\"canEdit() && !item.configurator\"\n [isShowEditControls]=\"canEdit()\"\n class=\"w-full duration-200\"\n [class.opacity-60]=\"selectable() && !selectionFormGroup.get(item.id)?.value\"\n (clickCardEvent)=\"openProductInfoDialog(item.product)\"\n (quantityValueChanges)=\"patchItem($event, itemComponent)\"\n (clickPriceHistoryEvent)=\"openPriceHistoryChartDialog(item.product)\"\n (clickSettings)=\"openSettingsDialog(item)\"\n [class.pointer-events-none]=\"!isActive\"\n />\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0443\u0434\u0430\u043B\u0435\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (canEdit()) {\n <button\n tuiIconButton\n iconStart=\"@tui.trash\"\n (click)=\"deleteItem(item, itemComponent)\"\n [loading]=\"itemComponent.showLoader()\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"!absolute left-1 top-1 lg:!relative lg:left-0 lg:top-0 lg:!self-center\"\n ></button>\n }\n </div>\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-end gap-2.5\">\n @let totalSum = sumSelectedItems();\n <!-- \u0418\u0442\u043E\u0433\u043E\u0432\u0430\u044F \u0441\u0443\u043C\u043C\u0430. -->\n <div class=\"text-base font-bold\">\u0418\u0442\u043E\u0433\u043E: {{ totalSum | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</div>\n </div>\n</div>\n" }]
199
202
  }], propDecorators: { items: [{
200
203
  type: Input,
201
204
  args: [{ required: true }]
202
205
  }] } });
203
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-items-list.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-items-list/sc-order-items-list.component.ts","../../../../../projects/client-ui/order/order-items-list/sc-order-items-list.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-param-reassign */\n\nimport { AsyncPipe } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, DestroyRef, inject, Input, input, signal } from '@angular/core';\nimport { outputFromObservable, takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { SC_ORDER_LOADER, ScIOrderEditorLoader, ScIOrderLoader, ScOrderItem } from '@snabcentr/client-core';\nimport { TuiCurrencyPipe } from '@taiga-ui/addon-commerce';\nimport { TuiButton, TuiFormatNumberPipe } from '@taiga-ui/core';\nimport { TuiCheckbox } from '@taiga-ui/kit';\nimport { catchError, finalize, map, of } from 'rxjs';\n\nimport { AbstractScPriceCard } from '../../directives';\nimport { IScDialogService, SC_DIALOG_SERVICE_TOKEN } from '../../providers';\nimport { ScOrderItemComponent } from '../order-item/sc-order-item.component';\n\n/**\n * Компонент списка товаров корзины, заказа или черновика.\n * Позиции разбиты.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-order-items-list',\n    templateUrl: './sc-order-items-list.component.html',\n    imports: [ScOrderItemComponent, ReactiveFormsModule, TuiCheckbox, TuiButton, TuiCurrencyPipe, AsyncPipe, TuiFormatNumberPipe],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemsListComponent {\n    /**\n     * Внутренний сигнал списка элементов корзины (товаров).\n     */\n    protected readonly itemsSignal = signal<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    protected readonly selectionFormGroup = new FormGroup<Record<string, FormControl<boolean>>>({});\n\n    /**\n     * Идентификаторы выбранных позиций.\n     */\n    protected readonly itemsIds = toSignal(this.selectionFormGroup.valueChanges.pipe(map((value) => Object.entries<boolean>(value as Record<string, boolean>))), {\n        initialValue: [],\n    });\n\n    /**\n     * Сигнал идентификаторов выбранных позиций.\n     */\n    public readonly selectedItemsIds = computed(() => {\n        const selectable = this.selectable();\n\n        return this.itemsIds()\n            .filter(([, checked]) => !selectable || checked)\n            .map(([key]) => key);\n    });\n\n    /**\n     * Выходные данные при изменении выбора позиций.\n     */\n    public readonly selectedItemsIdsChange = outputFromObservable(toObservable(this.selectedItemsIds));\n\n    /**\n     * Сумма стоимости выбранных позиций.\n     */\n    public readonly sumSelectedItems = computed(() => {\n        const items = this.itemsSignal();\n        const selectedItemsIds = this.selectedItemsIds();\n        const selectable = this.selectable();\n\n        return items.filter((item) => !selectable || selectedItemsIds.includes(item.id)).reduce((sum: number, item: ScOrderItem) => sum + item.costRub, 0);\n    });\n\n    /**\n     * Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).\n     * Предоставляется родителем через токен {@link SC_ORDER_LOADER} (реализация {@link ScIOrderLoader}).\n     */\n    protected readonly orderItemsListLoader = inject<ScIOrderLoader | ScIOrderEditorLoader>(SC_ORDER_LOADER);\n\n    /**\n     * Признак, что можно редактировать позиции.\n     */\n    protected readonly canEdit = computed(() => this.editable() && 'updateProduct$' in this.orderItemsListLoader);\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private readonly destroyRef = inject(DestroyRef);\n\n    /**\n     * Сервис для работы с диалоговыми окнами.\n     * Предоставляется родителем через токен {@link SC_DIALOG_SERVICE_TOKEN}.\n     */\n    private readonly scDialogService = inject<IScDialogService>(SC_DIALOG_SERVICE_TOKEN);\n\n    /**\n     * Список элементов корзины (товаров). При установке синхронно обновляет контролы выбора в форме.\n     */\n    @Input({ required: true })\n    set items(value: ScOrderItem[]) {\n        this.itemsSignal.set(value);\n        this.syncSelectionControls();\n    }\n\n    /**\n     * Синхронизирует контролы формы выбора с текущим списком товаров: удаляет контролы для отсутствующих id, добавляет для новых.\n     */\n    private syncSelectionControls(): void {\n        if (!this.selectable()) {\n            return;\n        }\n\n        const currentItems = this.itemsSignal();\n        const currentIds = new Set(currentItems.map((item) => item.id));\n\n        Object.keys(this.selectionFormGroup.controls).forEach((name) => {\n            if (!currentIds.has(name)) {\n                this.selectionFormGroup.removeControl(name as never);\n            }\n        });\n\n        currentItems.forEach((item) => {\n            if (!this.selectionFormGroup.contains(item.id)) {\n                this.selectionFormGroup.addControl(item.id, new FormControl(true, { nonNullable: true }));\n            }\n        });\n    }\n\n    /**\n     * Удаляет позицию.\n     *\n     * @param item Позиция товара для удаления.\n     * @param priceCard Компонент карточки товара.\n     */\n    protected deleteItem(item: ScOrderItem, priceCard: AbstractScPriceCard): void {\n        if (!this.canEdit()) {\n            return;\n        }\n\n        priceCard.quantityShowLoader = true;\n\n        (this.orderItemsListLoader as ScIOrderEditorLoader)\n            .deleteProduct$(item)\n            .pipe(\n                catchError(() => {\n                    priceCard.quantityControl.patchValue(item.quantity, {\n                        emitEvent: false,\n                    });\n\n                    return of();\n                }),\n                finalize(() => {\n                    priceCard.quantityShowLoader = false;\n                    priceCard.markForCheck();\n                }),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe();\n    }\n\n    /**\n     * Обновляет количество товара.\n     *\n     * @param newQuantity Новое значение количества товара.\n     * @param priceCard Компонент карточки товара.\n     */\n    protected patchItem(newQuantity: number | null, priceCard: AbstractScPriceCard): void {\n        if (!this.canEdit() || !priceCard.orderItem || !priceCard.product || !newQuantity) {\n            return;\n        }\n\n        priceCard.quantityShowLoader = true;\n\n        (this.orderItemsListLoader as ScIOrderEditorLoader)\n            .updateProduct$(priceCard.orderItem.id, { quantity: newQuantity })\n            .pipe(\n                catchError(() => {\n                    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n                    priceCard.quantityControl.patchValue(priceCard.orderItem!.quantity, {\n                        emitEvent: false,\n                    });\n\n                    return of();\n                }),\n                finalize(() => {\n                    priceCard.quantityShowLoader = false;\n                    priceCard.markForCheck();\n                }),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe();\n    }\n\n    /**\n     * Открывает модальное окно описания товара.\n     *\n     * @param product Товар/услуга, описание которой необходимо отобразить.\n     */\n    protected openProductInfoDialog(product: ScOrderItem['product']): void {\n        if (this.scDialogService.openProductInfoDialog) {\n            this.scDialogService.openProductInfoDialog(product);\n        }\n    }\n\n    /**\n     * Открывает модальное окно изменения товара или его конфигуратора.\n     *\n     * @param item Позиция товара.\n     */\n    protected openSettingsDialog(item: ScOrderItem): void {\n        const isConfigurator = Boolean(item.configurator);\n        const { isLinear } = item.product.unit;\n\n        if (isConfigurator) {\n            this.scDialogService.openConfiguratorDialog(item.product, item);\n        } else if (isLinear) {\n            this.scDialogService.openAddOrEditingCartItemDialog(item.product, item);\n        }\n    }\n\n    /**\n     * Открывает диалоговое окно \"График изменения стоимости\" для товара.\n     *\n     * @param product Товар/услуга, для которого необходимо отобразить график изменения стоимости.\n     */\n    protected openPriceHistoryChartDialog(product: ScOrderItem['product']): void {\n        this.scDialogService.openPriceHistoryChartDialog(product);\n    }\n}\n","<div\n    [formGroup]=\"selectionFormGroup\"\n    class=\"flex flex-col gap-5\"\n>\n    <!-- Группы товаров по направлениям продаж. -->\n\n    @for (item of itemsSignal(); track item.id) {\n        @if ($index !== 0) {\n            <hr class=\"mt-auto h-px w-full bg-tui-clear\" />\n        }\n\n        <div class=\"flex items-center gap-4 relative\">\n            @let isActive = !selectable() || selectionFormGroup.get(item.id)?.value;\n\n            <!-- Чекбокс выбора товара. -->\n            @if (selectable()) {\n                <input\n                    tuiCheckbox\n                    type=\"checkbox\"\n                    size=\"m\"\n                    [formControlName]=\"item.id\"\n                />\n            }\n\n            <!-- Карточка товара. -->\n            <sc-order-item\n                #itemComponent\n                [product]=\"item.product\"\n                [orderItem]=\"item\"\n                [showQuantityControl]=\"canEdit() && !item.configurator\"\n                [isShowEditControls]=\"canEdit()\"\n                class=\"w-full duration-200\"\n                [class.opacity-60]=\"selectable() && !selectionFormGroup.get(item.id)?.value\"\n                (clickCardEvent)=\"openProductInfoDialog(item.product)\"\n                (quantityValueChanges)=\"patchItem($event, itemComponent)\"\n                (clickPriceHistoryEvent)=\"openPriceHistoryChartDialog(item.product)\"\n                (clickSettings)=\"openSettingsDialog(item)\"\n                [class.pointer-events-none]=\"!isActive\"\n            />\n\n            <!-- Кнопка удаления товара. -->\n            @if (canEdit()) {\n                <button\n                    tuiIconButton\n                    iconStart=\"@tui.trash\"\n                    (click)=\"deleteItem(item, itemComponent)\"\n                    size=\"s\"\n                    appearance=\"secondary\"\n                    class=\"!absolute left-1 top-1 lg:!relative lg:left-0 lg:top-0 lg:!self-center\"\n                ></button>\n            }\n        </div>\n    }\n\n    <!-- Футер группы товаров. -->\n    <div class=\"flex flex-col items-end gap-2.5\">\n        @let totalSum = sumSelectedItems();\n        <!-- Итоговая сумма. -->\n        <div class=\"text-base font-bold\">Итого: {{ totalSum | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</div>\n    </div>\n</div>\n"]}
206
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-items-list.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-items-list/sc-order-items-list.component.ts","../../../../../projects/client-ui/order/order-items-list/sc-order-items-list.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-param-reassign */\n\nimport { AsyncPipe } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, DestroyRef, inject, Input, input, signal } from '@angular/core';\nimport { outputFromObservable, takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { SC_ORDER_LOADER, ScIOrderEditorLoader, ScIOrderLoader, ScOrderItem } from '@snabcentr/client-core';\nimport { TuiCurrencyPipe } from '@taiga-ui/addon-commerce';\nimport { TuiButton, TuiFormatNumberPipe } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiCheckbox } from '@taiga-ui/kit';\nimport { catchError, finalize, map, of } from 'rxjs';\n\nimport { AbstractScPriceCard } from '../../directives';\nimport { IScDialogService, SC_DIALOG_SERVICE_TOKEN } from '../../providers';\nimport { SC_ORDER_OPTIONS, ScOrderOptions } from '../directives/sc-order-options';\nimport { ScOrderItemComponent } from '../order-item/sc-order-item.component';\n\n/**\n * Компонент списка товаров корзины, заказа или черновика.\n * Позиции разбиты.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-order-items-list',\n    templateUrl: './sc-order-items-list.component.html',\n    imports: [ScOrderItemComponent, ReactiveFormsModule, TuiCheckbox, TuiButton, TuiButtonLoading, TuiCurrencyPipe, AsyncPipe, TuiFormatNumberPipe],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemsListComponent {\n    /**\n     * Внутренний сигнал списка элементов корзины (товаров).\n     */\n    protected readonly itemsSignal = signal<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    protected readonly selectionFormGroup = new FormGroup<Record<string, FormControl<boolean>>>({});\n\n    /**\n     * Опции заказа (черновика). Предоставляется директивой {@link ScOrderAccessorDirective} в контексте черновика.\n     */\n    protected readonly orderOptions = inject<ScOrderOptions>(SC_ORDER_OPTIONS, { optional: true });\n\n    /**\n     * Идентификаторы выбранных позиций.\n     */\n    protected readonly itemsIds = toSignal(this.selectionFormGroup.valueChanges.pipe(map((value) => Object.entries<boolean>(value as Record<string, boolean>))), {\n        initialValue: [],\n    });\n\n    /**\n     * Сигнал идентификаторов выбранных позиций.\n     */\n    public readonly selectedItemsIds = computed(() => {\n        const selectable = this.selectable();\n\n        return this.itemsIds()\n            .filter(([, checked]) => !selectable || checked)\n            .map(([key]) => key);\n    });\n\n    /**\n     * Выходные данные при изменении выбора позиций.\n     */\n    public readonly selectedItemsIdsChange = outputFromObservable(toObservable(this.selectedItemsIds));\n\n    /**\n     * Сумма стоимости выбранных позиций.\n     */\n    public readonly sumSelectedItems = computed(() => {\n        const items = this.itemsSignal();\n        const selectedItemsIds = this.selectedItemsIds();\n        const selectable = this.selectable();\n\n        return items.filter((item) => !selectable || selectedItemsIds.includes(item.id)).reduce((sum: number, item: ScOrderItem) => sum + item.costRub, 0);\n    });\n\n    /**\n     * Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).\n     * Предоставляется родителем через токен {@link SC_ORDER_LOADER} (реализация {@link ScIOrderLoader}).\n     */\n    protected readonly orderItemsListLoader = inject<ScIOrderLoader | ScIOrderEditorLoader>(SC_ORDER_LOADER);\n\n    /**\n     * Признак, что можно редактировать позиции.\n     */\n    protected readonly canEdit = computed(() => this.editable() && 'updateProduct$' in this.orderItemsListLoader);\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private readonly destroyRef = inject(DestroyRef);\n\n    /**\n     * Сервис для работы с диалоговыми окнами.\n     * Предоставляется родителем через токен {@link SC_DIALOG_SERVICE_TOKEN}.\n     */\n    private readonly scDialogService = inject<IScDialogService>(SC_DIALOG_SERVICE_TOKEN);\n\n    /**\n     * Список элементов корзины (товаров). При установке синхронно обновляет контролы выбора в форме.\n     */\n    @Input({ required: true })\n    set items(value: ScOrderItem[]) {\n        this.itemsSignal.set(value);\n        this.syncSelectionControls();\n    }\n\n    /**\n     * Синхронизирует контролы формы выбора с текущим списком товаров: удаляет контролы для отсутствующих id, добавляет для новых.\n     */\n    private syncSelectionControls(): void {\n        if (!this.selectable()) {\n            return;\n        }\n\n        const currentItems = this.itemsSignal();\n        const currentIds = new Set(currentItems.map((item) => item.id));\n\n        Object.keys(this.selectionFormGroup.controls).forEach((name) => {\n            if (!currentIds.has(name)) {\n                this.selectionFormGroup.removeControl(name as never);\n            }\n        });\n\n        currentItems.forEach((item) => {\n            if (!this.selectionFormGroup.contains(item.id)) {\n                this.selectionFormGroup.addControl(item.id, new FormControl(true, { nonNullable: true }));\n            }\n        });\n    }\n\n    /**\n     * Удаляет позицию.\n     *\n     * @param item Позиция товара для удаления.\n     * @param priceCard Компонент карточки товара.\n     */\n    protected deleteItem(item: ScOrderItem, priceCard: AbstractScPriceCard): void {\n        if (!this.canEdit()) {\n            return;\n        }\n\n        priceCard.showLoader.set(true);\n\n        (this.orderItemsListLoader as ScIOrderEditorLoader)\n            .deleteProduct$(item, this.orderOptions?.orderId())\n            .pipe(\n                catchError(() => {\n                    priceCard.quantityControl.patchValue(item.quantity, {\n                        emitEvent: false,\n                    });\n\n                    return of();\n                }),\n                finalize(() => {\n                    priceCard.showLoader.set(false);\n                }),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe();\n    }\n\n    /**\n     * Обновляет количество товара.\n     *\n     * @param newQuantity Новое значение количества товара.\n     * @param priceCard Компонент карточки товара.\n     */\n    protected patchItem(newQuantity: number | null, priceCard: AbstractScPriceCard): void {\n        if (!this.canEdit() || !priceCard.orderItem || !priceCard.product || !newQuantity) {\n            return;\n        }\n\n        priceCard.showLoader.set(true);\n\n        (this.orderItemsListLoader as ScIOrderEditorLoader)\n            .updateProduct$(priceCard.orderItem.id, { quantity: newQuantity }, this.orderOptions?.orderId())\n            .pipe(\n                catchError(() => {\n                    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n                    priceCard.quantityControl.patchValue(priceCard.orderItem!.quantity, {\n                        emitEvent: false,\n                    });\n\n                    return of();\n                }),\n                finalize(() => {\n                    priceCard.showLoader.set(false);\n                }),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe();\n    }\n\n    /**\n     * Открывает модальное окно описания товара.\n     *\n     * @param product Товар/услуга, описание которой необходимо отобразить.\n     */\n    protected openProductInfoDialog(product: ScOrderItem['product']): void {\n        if (this.scDialogService.openProductInfoDialog) {\n            this.scDialogService.openProductInfoDialog(product);\n        }\n    }\n\n    /**\n     * Открывает модальное окно изменения товара или его конфигуратора.\n     *\n     * @param item Позиция товара.\n     */\n    protected openSettingsDialog(item: ScOrderItem): void {\n        const isConfigurator = Boolean(item.configurator);\n        const { isLinear } = item.product.unit;\n\n        if (isConfigurator) {\n            this.scDialogService.openConfiguratorDialog(item.product, item);\n        } else if (isLinear) {\n            this.scDialogService.openAddOrEditingCartItemDialog(item.product, item);\n        }\n    }\n\n    /**\n     * Открывает диалоговое окно \"График изменения стоимости\" для товара.\n     *\n     * @param product Товар/услуга, для которого необходимо отобразить график изменения стоимости.\n     */\n    protected openPriceHistoryChartDialog(product: ScOrderItem['product']): void {\n        this.scDialogService.openPriceHistoryChartDialog(product);\n    }\n}\n","<div\n    [formGroup]=\"selectionFormGroup\"\n    class=\"flex flex-col gap-5\"\n>\n    <!-- Группы товаров по направлениям продаж. -->\n\n    @for (item of itemsSignal(); track item.id) {\n        @if ($index !== 0) {\n            <hr class=\"mt-auto h-px w-full bg-tui-clear\" />\n        }\n\n        <div class=\"flex items-center gap-4 relative\">\n            @let isActive = !selectable() || selectionFormGroup.get(item.id)?.value;\n\n            <!-- Чекбокс выбора товара. -->\n            @if (selectable()) {\n                <input\n                    tuiCheckbox\n                    [disabled]=\"itemComponent.showLoader()\"\n                    type=\"checkbox\"\n                    size=\"m\"\n                    [formControlName]=\"item.id\"\n                />\n            }\n\n            <!-- Карточка товара. -->\n            <sc-order-item\n                #itemComponent\n                [product]=\"item.product\"\n                [orderItem]=\"item\"\n                [showQuantityControl]=\"canEdit() && !item.configurator\"\n                [isShowEditControls]=\"canEdit()\"\n                class=\"w-full duration-200\"\n                [class.opacity-60]=\"selectable() && !selectionFormGroup.get(item.id)?.value\"\n                (clickCardEvent)=\"openProductInfoDialog(item.product)\"\n                (quantityValueChanges)=\"patchItem($event, itemComponent)\"\n                (clickPriceHistoryEvent)=\"openPriceHistoryChartDialog(item.product)\"\n                (clickSettings)=\"openSettingsDialog(item)\"\n                [class.pointer-events-none]=\"!isActive\"\n            />\n\n            <!-- Кнопка удаления товара. -->\n            @if (canEdit()) {\n                <button\n                    tuiIconButton\n                    iconStart=\"@tui.trash\"\n                    (click)=\"deleteItem(item, itemComponent)\"\n                    [loading]=\"itemComponent.showLoader()\"\n                    size=\"s\"\n                    appearance=\"secondary\"\n                    class=\"!absolute left-1 top-1 lg:!relative lg:left-0 lg:top-0 lg:!self-center\"\n                ></button>\n            }\n        </div>\n    }\n\n    <!-- Футер группы товаров. -->\n    <div class=\"flex flex-col items-end gap-2.5\">\n        @let totalSum = sumSelectedItems();\n        <!-- Итоговая сумма. -->\n        <div class=\"text-base font-bold\">Итого: {{ totalSum | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</div>\n    </div>\n</div>\n"]}
@@ -60,13 +60,16 @@ export class ScOrderItemsListByDirectionsComponent {
60
60
  * @param direction Направление продаж.
61
61
  */
62
62
  selectedItemsHandler(itemsIds, direction) {
63
- this.selectedItemsIds.update((previous) => [...previous.filter((id) => !direction.items.some((item) => item.id === id)), ...itemsIds]);
63
+ this.selectedItemsIds.update((previous) => [
64
+ ...previous.filter((id) => !direction.items.some((item) => item.id === id) && this.items().some((item) => item.id === id)),
65
+ ...itemsIds,
66
+ ]);
64
67
  }
65
68
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListByDirectionsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
66
- 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 gap-2.5\">\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 appearance=\"primary\"\n [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n iconStart=\"@tui.send\"\n class=\"ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText() }}\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-wrap gap-2.5\">\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=\"@tui.send\"\n class=\"ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText() }}\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 }); }
67
70
  }
68
71
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListByDirectionsComponent, decorators: [{
69
72
  type: Component,
70
- 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 gap-2.5\">\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 appearance=\"primary\"\n [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n iconStart=\"@tui.send\"\n class=\"ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText() }}\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-wrap gap-2.5\">\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=\"@tui.send\"\n class=\"ml-auto\"\n (click)=\"continueClick.emit(direction)\"\n >\n {{ continueButtonText() }}\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" }]
71
74
  }] });
72
- //# 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":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC3G,OAAO,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AACtD,OAAO,EAAkC,mBAAmB,EAAE,MAAM,wBAAwB,CAAC;AAC7F,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAE3C,OAAO,EAAE,uBAAuB,EAAE,MAAM,wCAAwC,CAAC;AACjF,OAAO,EAAE,yBAAyB,EAAE,MAAM,mDAAmD,CAAC;;AAS9F,MAAM,OAAO,qCAAqC;IAPlD;QAQI;;WAEG;QACa,UAAK,GAAG,KAAK,CAAC,QAAQ,EAAiB,CAAC;QAExD;;WAEG;QACa,eAAU,GAAG,KAAK,CAAU,IAAI,CAAC,CAAC;QAElD;;WAEG;QACa,aAAQ,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;QAEjD;;WAEG;QACa,uBAAkB,GAAG,KAAK,CAAU,IAAI,CAAC,CAAC;QAE1D;;WAEG;QACa,uBAAkB,GAAG,KAAK,CAAS,gBAAgB,CAAC,CAAC;QAErE;;WAEG;QACa,kBAAa,GAAG,MAAM,EAA2B,CAAC;QAElE;;WAEG;QACc,eAAU,GAAG,QAAQ,CAAkC,MAAM,CAAC,mBAAmB,CAAC,CAAC,WAAW,EAAE,EAAE,YAAY,EAAE,SAAS,EAAE,CAAC,CAAC;QAE9I;;;WAGG;QACgB,sBAAiB,GAAG,QAAQ,CAA4B,GAAG,EAAE;YAC5E,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;YAC5F,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;YAErC,OAAO,UAAU,EAAE,GAAG,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,IAAI,uBAAuB,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;QACzI,CAAC,CAAC,CAAC;QAEH;;WAEG;QACgB,cAAS,GAAG,QAAQ,CAAgB,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC;QAE1I;;WAEG;QACa,qBAAgB,GAAG,KAAK,CAAW,EAAE,CAAC,CAAC;KAW1D;IATG;;;;;OAKG;IACO,oBAAoB,CAAC,QAAkB,EAAE,SAAkC;QACjF,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC;IAC3I,CAAC;+GAjEQ,qCAAqC;mGAArC,qCAAqC,whCCflD,kiFAiDA,4CDrCc,yBAAyB,kJAAE,SAAS;;4FAGrC,qCAAqC;kBAPjD,SAAS;iCACM,IAAI,YACN,mCAAmC,WAEpC,CAAC,yBAAyB,EAAE,SAAS,CAAC,mBAC9B,uBAAuB,CAAC,MAAM","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<string>('Оформить заказ');\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) => [...previous.filter((id) => !direction.items.some((item) => item.id === id)), ...itemsIds]);\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 gap-2.5\">\n                <ng-content select=\"[footerActions]\"></ng-content>\n\n                <!-- Кнопка продолжения. -->\n                @if (showContinueButton()) {\n                    <button\n                        tuiButton\n                        type=\"button\"\n                        appearance=\"primary\"\n                        [disabled]=\"direction.getItemsByIds(selectedItemsIds()).length === 0\"\n                        iconStart=\"@tui.send\"\n                        class=\"ml-auto\"\n                        (click)=\"continueClick.emit(direction)\"\n                    >\n                        {{ continueButtonText() }}\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<string>('Оформить заказ');\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 gap-2.5\">\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=\"@tui.send\"\n                        class=\"ml-auto\"\n                        (click)=\"continueClick.emit(direction)\"\n                    >\n                        {{ continueButtonText() }}\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"]}
@@ -63,7 +63,7 @@ export class ScOrderItemsListByStockComponent {
63
63
  * @param items Позиции.
64
64
  */
65
65
  selectedItemsHandler(itemsIds, items) {
66
- this.selectedItemsIds.update((previous) => [...previous.filter((id) => !items.some((item) => item.id === id)), ...itemsIds]);
66
+ this.selectedItemsIds.update((previous) => [...previous.filter((id) => !items.some((item) => item.id === id) && this.items().some((item) => item.id === id)), ...itemsIds]);
67
67
  }
68
68
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListByStockComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
69
69
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScOrderItemsListByStockComponent, isStandalone: true, selector: "sc-order-items-list-by-stock", inputs: { items: { classPropertyName: "items", publicName: "items", isSignal: true, isRequired: true, transformFunction: null }, warehouseId: { classPropertyName: "warehouseId", publicName: "warehouseId", 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 }, sectionHeader: { classPropertyName: "sectionHeader", publicName: "sectionHeader", isSignal: true, isRequired: false, transformFunction: null }, selectedItemsIds: { classPropertyName: "selectedItemsIds", publicName: "selectedItemsIds", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { selectedItemsIds: "selectedItemsIdsChange" }, ngImport: i0, template: "<div class=\"flex flex-col gap-5\">\n @if (itemsInWarehouseStock().length > 0) {\n <sc-order-items-list\n [items]=\"itemsInWarehouseStock()\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsInWarehouseStock())\"\n />\n }\n\n @if (itemsOnOrder().length > 0) {\n <div class=\"flex flex-col gap-2.5\">\n @if (sectionHeader()) {\n <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n }\n <sc-order-items-list\n [items]=\"itemsOnOrder()\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsOnOrder())\"\n />\n </div>\n }\n\n @if (itemsNotInWarehouseStock().length > 0) {\n <div class=\"flex flex-col gap-2.5\">\n @if (sectionHeader()) {\n <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n }\n <sc-order-items-list\n [items]=\"itemsNotInWarehouseStock()\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsNotInWarehouseStock())\"\n />\n </div>\n }\n</div>\n", dependencies: [{ kind: "component", type: ScOrderItemsListComponent, selector: "sc-order-items-list", inputs: ["selectable", "editable", "items"], outputs: ["selectedItemsIdsChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
@@ -72,4 +72,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
72
72
  type: Component,
73
73
  args: [{ standalone: true, selector: 'sc-order-items-list-by-stock', imports: [ScOrderItemsListComponent], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col gap-5\">\n @if (itemsInWarehouseStock().length > 0) {\n <sc-order-items-list\n [items]=\"itemsInWarehouseStock()\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsInWarehouseStock())\"\n />\n }\n\n @if (itemsOnOrder().length > 0) {\n <div class=\"flex flex-col gap-2.5\">\n @if (sectionHeader()) {\n <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n }\n <sc-order-items-list\n [items]=\"itemsOnOrder()\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsOnOrder())\"\n />\n </div>\n }\n\n @if (itemsNotInWarehouseStock().length > 0) {\n <div class=\"flex flex-col gap-2.5\">\n @if (sectionHeader()) {\n <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n }\n <sc-order-items-list\n [items]=\"itemsNotInWarehouseStock()\"\n [selectable]=\"selectable()\"\n [editable]=\"editable()\"\n (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsNotInWarehouseStock())\"\n />\n </div>\n }\n</div>\n" }]
74
74
  }] });
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-items-list-by-stock.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.ts","../../../../../projects/client-ui/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,QAAQ,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AAG3F,OAAO,EAAE,yBAAyB,EAAE,MAAM,mDAAmD,CAAC;;AAE9F;;;;;GAKG;AAQH,MAAM,OAAO,gCAAgC;IAP7C;QAQI;;WAEG;QACa,UAAK,GAAG,KAAK,CAAC,QAAQ,EAAiB,CAAC;QAExD;;WAEG;QACa,gBAAW,GAAG,KAAK,CAAC,QAAQ,EAAU,CAAC;QAEvD;;WAEG;QACa,eAAU,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;QAEnD;;WAEG;QACa,aAAQ,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;QAEjD;;WAEG;QACa,kBAAa,GAAG,KAAK,EAAsB,CAAC;QAE5D;;WAEG;QACa,qBAAgB,GAAG,KAAK,CAAW,EAAE,CAAC,CAAC;QAEvD;;WAEG;QACgB,0BAAqB,GAAG,QAAQ,CAAC,GAAG,EAAE;YACrD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;YAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YAEvC,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,qBAAqB,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC5G,CAAC,CAAC,CAAC;QAEH;;WAEG;QACgB,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC5C,OAAO,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC/D,CAAC,CAAC,CAAC;QAEH;;WAEG;QACgB,6BAAwB,GAAG,QAAQ,CAAC,GAAG,EAAE;YACxD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;YAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YAEvC,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,qBAAqB,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC7G,CAAC,CAAC,CAAC;KAWN;IATG;;;;;OAKG;IACO,oBAAoB,CAAC,QAAkB,EAAE,KAAoB;QACnE,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC;IACjI,CAAC;+GAlEQ,gCAAgC;mGAAhC,gCAAgC,88BClB7C,u8CAsCA,4CDvBc,yBAAyB;;4FAG1B,gCAAgC;kBAP5C,SAAS;iCACM,IAAI,YACN,8BAA8B,WAE/B,CAAC,yBAAyB,CAAC,mBACnB,uBAAuB,CAAC,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, input, model } from '@angular/core';\nimport { ScOrderItem } from '@snabcentr/client-core';\n\nimport { ScOrderItemsListComponent } from '../order-items-list/sc-order-items-list.component';\n\n/**\n * Компонент списка товаров, разбитых по наличию на складе:\n * - в наличии на складе;\n * - под заказ;\n * - отсутствуют на складе.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-order-items-list-by-stock',\n    templateUrl: './sc-order-items-list-by-stock.component.html',\n    imports: [ScOrderItemsListComponent],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemsListByStockComponent {\n    /**\n     * Список элементов (товаров) для отображения.\n     */\n    public readonly items = input.required<ScOrderItem[]>();\n\n    /**\n     * Идентификатор склада для проверки наличия.\n     */\n    public readonly warehouseId = input.required<number>();\n\n    /**\n     * Включить выбор позиций чекбоксами.\n     */\n    public readonly selectable = input<boolean>(false);\n\n    /**\n     * Включить редактирование позиций.\n     */\n    public readonly editable = input<boolean>(false);\n\n    /**\n     * Заголовок для секций «под заказ» и «нет на складе». Если не указан — заголовок не выводится.\n     */\n    public readonly sectionHeader = input<string | undefined>();\n\n    /**\n     * Выходные данные при изменении выбора позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Позиции в наличии на выбранном складе (не под заказ).\n     */\n    protected readonly itemsInWarehouseStock = computed(() => {\n        const items = this.items();\n        const warehouseId = this.warehouseId();\n\n        return items.filter((item) => item.product.isWarehouseStockExist(warehouseId) && !item.product.onOrder);\n    });\n\n    /**\n     * Позиции под заказ.\n     */\n    protected readonly itemsOnOrder = computed(() => {\n        return this.items().filter((item) => item.product.onOrder);\n    });\n\n    /**\n     * Позиции, которых нет на выбранном складе (не под заказ).\n     */\n    protected readonly itemsNotInWarehouseStock = computed(() => {\n        const items = this.items();\n        const warehouseId = this.warehouseId();\n\n        return items.filter((item) => !item.product.isWarehouseStockExist(warehouseId) && !item.product.onOrder);\n    });\n\n    /**\n     * Обработчик изменения выбора позиций.\n     *\n     * @param itemsIds Идентификаторы выбранных позиций.\n     * @param items Позиции.\n     */\n    protected selectedItemsHandler(itemsIds: string[], items: ScOrderItem[]): void {\n        this.selectedItemsIds.update((previous) => [...previous.filter((id) => !items.some((item) => item.id === id)), ...itemsIds]);\n    }\n}\n","<div class=\"flex flex-col gap-5\">\n    @if (itemsInWarehouseStock().length > 0) {\n        <sc-order-items-list\n            [items]=\"itemsInWarehouseStock()\"\n            [selectable]=\"selectable()\"\n            [editable]=\"editable()\"\n            (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsInWarehouseStock())\"\n        />\n    }\n\n    @if (itemsOnOrder().length > 0) {\n        <div class=\"flex flex-col gap-2.5\">\n            @if (sectionHeader()) {\n                <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n            }\n            <sc-order-items-list\n                [items]=\"itemsOnOrder()\"\n                [selectable]=\"selectable()\"\n                [editable]=\"editable()\"\n                (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsOnOrder())\"\n            />\n        </div>\n    }\n\n    @if (itemsNotInWarehouseStock().length > 0) {\n        <div class=\"flex flex-col gap-2.5\">\n            @if (sectionHeader()) {\n                <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n            }\n            <sc-order-items-list\n                [items]=\"itemsNotInWarehouseStock()\"\n                [selectable]=\"selectable()\"\n                [editable]=\"editable()\"\n                (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsNotInWarehouseStock())\"\n            />\n        </div>\n    }\n</div>\n"]}
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-items-list-by-stock.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.ts","../../../../../projects/client-ui/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,QAAQ,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AAG3F,OAAO,EAAE,yBAAyB,EAAE,MAAM,mDAAmD,CAAC;;AAE9F;;;;;GAKG;AAQH,MAAM,OAAO,gCAAgC;IAP7C;QAQI;;WAEG;QACa,UAAK,GAAG,KAAK,CAAC,QAAQ,EAAiB,CAAC;QAExD;;WAEG;QACa,gBAAW,GAAG,KAAK,CAAC,QAAQ,EAAU,CAAC;QAEvD;;WAEG;QACa,eAAU,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;QAEnD;;WAEG;QACa,aAAQ,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;QAEjD;;WAEG;QACa,kBAAa,GAAG,KAAK,EAAsB,CAAC;QAE5D;;WAEG;QACa,qBAAgB,GAAG,KAAK,CAAW,EAAE,CAAC,CAAC;QAEvD;;WAEG;QACgB,0BAAqB,GAAG,QAAQ,CAAC,GAAG,EAAE;YACrD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;YAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YAEvC,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,qBAAqB,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC5G,CAAC,CAAC,CAAC;QAEH;;WAEG;QACgB,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC5C,OAAO,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC/D,CAAC,CAAC,CAAC;QAEH;;WAEG;QACgB,6BAAwB,GAAG,QAAQ,CAAC,GAAG,EAAE;YACxD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;YAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;YAEvC,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,qBAAqB,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QAC7G,CAAC,CAAC,CAAC;KAWN;IATG;;;;;OAKG;IACO,oBAAoB,CAAC,QAAkB,EAAE,KAAoB;QACnE,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC;IAChL,CAAC;+GAlEQ,gCAAgC;mGAAhC,gCAAgC,88BClB7C,u8CAsCA,4CDvBc,yBAAyB;;4FAG1B,gCAAgC;kBAP5C,SAAS;iCACM,IAAI,YACN,8BAA8B,WAE/B,CAAC,yBAAyB,CAAC,mBACnB,uBAAuB,CAAC,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, input, model } from '@angular/core';\nimport { ScOrderItem } from '@snabcentr/client-core';\n\nimport { ScOrderItemsListComponent } from '../order-items-list/sc-order-items-list.component';\n\n/**\n * Компонент списка товаров, разбитых по наличию на складе:\n * - в наличии на складе;\n * - под заказ;\n * - отсутствуют на складе.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-order-items-list-by-stock',\n    templateUrl: './sc-order-items-list-by-stock.component.html',\n    imports: [ScOrderItemsListComponent],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemsListByStockComponent {\n    /**\n     * Список элементов (товаров) для отображения.\n     */\n    public readonly items = input.required<ScOrderItem[]>();\n\n    /**\n     * Идентификатор склада для проверки наличия.\n     */\n    public readonly warehouseId = input.required<number>();\n\n    /**\n     * Включить выбор позиций чекбоксами.\n     */\n    public readonly selectable = input<boolean>(false);\n\n    /**\n     * Включить редактирование позиций.\n     */\n    public readonly editable = input<boolean>(false);\n\n    /**\n     * Заголовок для секций «под заказ» и «нет на складе». Если не указан — заголовок не выводится.\n     */\n    public readonly sectionHeader = input<string | undefined>();\n\n    /**\n     * Выходные данные при изменении выбора позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Позиции в наличии на выбранном складе (не под заказ).\n     */\n    protected readonly itemsInWarehouseStock = computed(() => {\n        const items = this.items();\n        const warehouseId = this.warehouseId();\n\n        return items.filter((item) => item.product.isWarehouseStockExist(warehouseId) && !item.product.onOrder);\n    });\n\n    /**\n     * Позиции под заказ.\n     */\n    protected readonly itemsOnOrder = computed(() => {\n        return this.items().filter((item) => item.product.onOrder);\n    });\n\n    /**\n     * Позиции, которых нет на выбранном складе (не под заказ).\n     */\n    protected readonly itemsNotInWarehouseStock = computed(() => {\n        const items = this.items();\n        const warehouseId = this.warehouseId();\n\n        return items.filter((item) => !item.product.isWarehouseStockExist(warehouseId) && !item.product.onOrder);\n    });\n\n    /**\n     * Обработчик изменения выбора позиций.\n     *\n     * @param itemsIds Идентификаторы выбранных позиций.\n     * @param items Позиции.\n     */\n    protected selectedItemsHandler(itemsIds: string[], items: ScOrderItem[]): void {\n        this.selectedItemsIds.update((previous) => [...previous.filter((id) => !items.some((item) => item.id === id) && this.items().some((item) => item.id === id)), ...itemsIds]);\n    }\n}\n","<div class=\"flex flex-col gap-5\">\n    @if (itemsInWarehouseStock().length > 0) {\n        <sc-order-items-list\n            [items]=\"itemsInWarehouseStock()\"\n            [selectable]=\"selectable()\"\n            [editable]=\"editable()\"\n            (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsInWarehouseStock())\"\n        />\n    }\n\n    @if (itemsOnOrder().length > 0) {\n        <div class=\"flex flex-col gap-2.5\">\n            @if (sectionHeader()) {\n                <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n            }\n            <sc-order-items-list\n                [items]=\"itemsOnOrder()\"\n                [selectable]=\"selectable()\"\n                [editable]=\"editable()\"\n                (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsOnOrder())\"\n            />\n        </div>\n    }\n\n    @if (itemsNotInWarehouseStock().length > 0) {\n        <div class=\"flex flex-col gap-2.5\">\n            @if (sectionHeader()) {\n                <div class=\"text-base font-bold text-sc-dark-grey\">{{ sectionHeader() }}</div>\n            }\n            <sc-order-items-list\n                [items]=\"itemsNotInWarehouseStock()\"\n                [selectable]=\"selectable()\"\n                [editable]=\"editable()\"\n                (selectedItemsIdsChange)=\"selectedItemsHandler($event, itemsNotInWarehouseStock())\"\n            />\n        </div>\n    }\n</div>\n"]}