@snabcentr/client-ui 3.49.4 → 3.50.0
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.
- package/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.d.ts +5 -5
- package/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.d.ts +5 -5
- package/cart/index.d.ts +0 -1
- package/catalog/price-card-inline/sc-price-card-inline.component.d.ts +1 -1
- package/configurators/sandwich/sc-i-new-cart-item-sandwich.d.ts +2 -2
- package/configurators/sandwich/sc-sandwich.component.d.ts +2 -2
- package/directives/abstract-price-card/abstract-sc-price-card.directive.d.ts +6 -6
- package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +11 -11
- package/esm2022/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.mjs +6 -6
- package/esm2022/cart/index.mjs +1 -2
- package/esm2022/catalog/price-card/sc-price-card.component.mjs +3 -3
- package/esm2022/catalog/price-card-inline/sc-price-card-inline.component.mjs +3 -3
- package/esm2022/configurators/sandwich/sc-i-new-cart-item-sandwich.mjs +1 -1
- package/esm2022/configurators/sandwich/sc-sandwich.component.mjs +5 -5
- package/esm2022/directives/abstract-price-card/abstract-sc-price-card.directive.mjs +8 -8
- package/esm2022/order/draft/sc-draft.component.mjs +117 -0
- package/esm2022/order/index.mjs +7 -2
- package/esm2022/order/models/sc-i-cart-items-by-direction.mjs +22 -0
- package/esm2022/order/order-item/sc-order-item.component.mjs +85 -0
- package/esm2022/order/order-items-list/sc-order-items-list.component.mjs +203 -0
- package/esm2022/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.mjs +72 -0
- package/esm2022/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.mjs +75 -0
- package/esm2022/order/sc-order.module.mjs +5 -6
- package/esm2022/providers/index.mjs +2 -1
- package/esm2022/providers/sc-dialog-service.providers.mjs +6 -0
- package/fesm2022/snabcentr-client-ui.mjs +1836 -1410
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/order/draft/sc-draft.component.d.ts +63 -0
- package/order/index.d.ts +6 -1
- package/order/models/sc-i-cart-items-by-direction.d.ts +25 -0
- package/{cart/cart-item/sc-cart-item.component.d.ts → order/order-item/sc-order-item.component.d.ts} +7 -3
- package/order/order-items-list/sc-order-items-list.component.d.ts +102 -0
- package/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.d.ts +55 -0
- package/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.d.ts +55 -0
- package/order/sc-order.module.d.ts +10 -11
- package/package.json +2 -1
- package/providers/index.d.ts +1 -0
- package/providers/sc-dialog-service.providers.d.ts +41 -0
- package/styles/taiga/snabcentr-tailwind-preset.js +2 -3
- package/styles/tailwind/tailwind.scss +96 -100
- package/esm2022/cart/cart-item/sc-cart-item.component.mjs +0 -71
- package/esm2022/order/order-item-mobile/sc-order-item-mobile.component.mjs +0 -60
- package/order/order-item-mobile/sc-order-item-mobile.component.d.ts +0 -39
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { ChangeDetectionStrategy, Component, inject, input, output, signal } from '@angular/core';
|
|
3
|
+
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
4
|
+
import { RouterModule } from '@angular/router';
|
|
5
|
+
import { SEARCH_TERM_PROVIDERS } from '@snabcentr/client-core';
|
|
6
|
+
import { TuiCurrencyPipe } from '@taiga-ui/addon-commerce';
|
|
7
|
+
import { TuiAutoFocus } from '@taiga-ui/cdk';
|
|
8
|
+
import { TuiAppearance, TuiButton, TuiFormatNumberPipe, TuiHint, TuiIcon, TuiLink, tuiTextfieldOptionsProvider } from '@taiga-ui/core';
|
|
9
|
+
import { TuiHighlight, TuiPreview, TuiPreviewDialogService } from '@taiga-ui/kit';
|
|
10
|
+
import { TuiTextfieldControllerModule } from '@taiga-ui/legacy';
|
|
11
|
+
import { CostWithDiscountComponent, ScHoverImageCarouselComponent, ScInputQuantityComponent, ScPriceWarehouseStockComponent } from '../../catalog';
|
|
12
|
+
import { AbstractScPriceCard } from '../../directives';
|
|
13
|
+
import { ScNoindexWrapperComponent } from '../../noindex-wrapper';
|
|
14
|
+
import * as i0 from "@angular/core";
|
|
15
|
+
import * as i1 from "@angular/common";
|
|
16
|
+
import * as i2 from "@taiga-ui/polymorpheus";
|
|
17
|
+
import * as i3 from "@angular/forms";
|
|
18
|
+
import * as i4 from "@taiga-ui/core";
|
|
19
|
+
import * as i5 from "@taiga-ui/kit";
|
|
20
|
+
/**
|
|
21
|
+
* Компонент карточки элемента корзины.
|
|
22
|
+
*/
|
|
23
|
+
export class ScOrderItemComponent extends AbstractScPriceCard {
|
|
24
|
+
constructor() {
|
|
25
|
+
super(...arguments);
|
|
26
|
+
/**
|
|
27
|
+
* Событие нажатия на кнопку редактирования конфигурации.
|
|
28
|
+
*/
|
|
29
|
+
this.clickSettings = output();
|
|
30
|
+
/**
|
|
31
|
+
* Сервис диалогового окна предварительного просмотра.
|
|
32
|
+
*/
|
|
33
|
+
this.previewDialogService = inject(TuiPreviewDialogService);
|
|
34
|
+
/**
|
|
35
|
+
* Признак, что необходимо отобразить элементы редактирования.
|
|
36
|
+
*/
|
|
37
|
+
this.isShowEditControls = input(true);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Отображает спецификацию.
|
|
41
|
+
*
|
|
42
|
+
* @param specificationPreviewReference Шаблон спецификации.
|
|
43
|
+
*/
|
|
44
|
+
showSpecification(specificationPreviewReference) {
|
|
45
|
+
this.previewDialogService.open(specificationPreviewReference).subscribe();
|
|
46
|
+
}
|
|
47
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
48
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScOrderItemComponent, isStandalone: true, selector: "sc-order-item", inputs: { isShowEditControls: { classPropertyName: "isShowEditControls", publicName: "isShowEditControls", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { clickSettings: "clickSettings" }, providers: [
|
|
49
|
+
SEARCH_TERM_PROVIDERS,
|
|
50
|
+
tuiTextfieldOptionsProvider({
|
|
51
|
+
appearance: signal('textfield'),
|
|
52
|
+
}),
|
|
53
|
+
], usesInheritance: true, ngImport: i0, template: "@if (product && orderItem) {\n <div class=\"@container relative flex w-full gap-4 rounded-xl p-1 text-body-s hover:bg-tui-background-neutral-1\">\n <sc-noindex-wrapper>\n @let isDisabled = product.isHidden || product.isNull;\n <sc-hover-image-carousel\n (click)=\"isDisabled ? null : clickCardEvent.emit()\"\n [images]=\"!isMobile ? getCardImagePreviewList(product) : [getCardImagePreview()]\"\n [isShowActions]=\"false\"\n class=\"aspect-square w-24 shrink-0 cursor-pointer self-start\"\n />\n <div class=\"flex w-full flex-col gap-y-1 gap-x-4 @5xl:flex-row @5xl:items-center @5xl:gap-5\">\n <div class=\"flex min-w-0 flex-1 flex-col gap-1\">\n <div\n [tuiHighlight]=\"(search$ | async) ?? ''\"\n class=\"text-tui-text-02\"\n >\n \u0410\u0440\u0442\u0438\u043A\u0443\u043B: {{ product.code }}\n </div>\n <div class=\"flex flex-col gap-1\">\n <a\n tuiLink\n [attr.href]=\"href ?? null\"\n [tuiHighlight]=\"(search$ | async) ?? ''\"\n (click)=\"$event.preventDefault(); clickCardEvent.emit()\"\n [class.disabled]=\"isDisabled\"\n class=\"!text-body-m-bold\"\n >\n {{ product.name }}\n </a>\n @if (product.supplierSku) {\n <span class=\"text-tui-text-02\"> \u0410\u0440\u0442\u0438\u043A\u0443\u043B \u043F\u0440\u043E\u0438\u0437\u0432\u043E\u0434\u0438\u0442\u0435\u043B\u044F: {{ product.supplierSku }} </span>\n }\n @if (product.pack) {\n <div class=\"flex items-center gap-1 text-tui-text-02\">\n \u041D\u043E\u0440\u043C\u0430 \u0443\u043F\u0430\u043A\u043E\u0432\u043A\u0438: {{ product.pack }}\n @if (product.ignoreMinCountCheck) {\n <tui-icon\n icon=\"@tui.package\"\n [tuiHint]=\"minCountHint\"\n [tuiHintShowDelay]=\"100\"\n tuiHintDirection=\"top\"\n class=\"text-body-xl text-tui-text-01 opacity-90\"\n />\n <ng-template #minCountHint>\n \u0414\u043E\u0441\u0442\u0443\u043F\u0435\u043D \u0437\u0430\u043A\u0430\u0437 <br />\n \u043F\u0440\u043E\u0438\u0437\u0432\u043E\u043B\u044C\u043D\u043E\u0433\u043E <br />\n \u043A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u0430\n </ng-template>\n }\n </div>\n }\n\n <sc-cost-with-discount [product]=\"product\" />\n </div>\n @if (orderItem.specificationImgUrl) {\n <a\n tuiLink\n (click)=\"showSpecification(specificationPreview)\"\n [class.disabled]=\"isDisabled\"\n >\n \u0421\u043F\u0435\u0446\u0438\u0444\u0438\u043A\u0430\u0446\u0438\u044F\n </a>\n }\n <ng-template\n #specificationPreview\n let-preview\n >\n <tui-preview\n [rotatable]=\"false\"\n [zoomable]=\"false\"\n >\n <img\n *polymorpheusOutlet=\"orderItem.specificationImgUrl as src\"\n alt=\"preview\"\n [src]=\"orderItem.specificationImgUrl\"\n />\n <button\n iconStart=\"@tui.x\"\n title=\"Close\"\n tuiIconButton\n tuiPreviewAction\n type=\"button\"\n (click)=\"preview.complete()\"\n ></button>\n </tui-preview>\n </ng-template>\n </div>\n <div class=\"grid grid-cols-1 @lg:grid-cols-3 gap-2 @2xl:gap-4 @2xl:grid-rows-1\">\n <div class=\"grid grid-cols-1 items-center @lg:gap-2 @2xl:grid-cols-2 @lg:col-span-2\">\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (isShowEditControls() && showQuantityControl) {\n <!-- \u041F\u043E\u043B\u0435 \u0432\u0432\u043E\u0434\u0430 \u043A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n <sc-input-quantity\n #inputQuantity\n [tuiAutoFocus]=\"autoFocuseQuantityInput()\"\n [formControl]=\"quantityControl\"\n size=\"m\"\n [showCross]=\"false\"\n [isDisabled]=\"isDisabled\"\n [step]=\"unitsHelper.productMultiplicity(product)\"\n [ignoreStepValidators]=\"product.ignoreMinCountCheck\"\n [showLoader]=\"quantityShowLoader\"\n (clickClearEvent)=\"clickClearEvent.emit(orderItem)\"\n (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n class=\"w-32 shrink-0\"\n />\n } @else {\n <div class=\"text-body-m\">\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E:\n <strong> {{ orderItem.quantity }} \u0448\u0442.</strong>\n </div>\n }\n\n @let showAdditionalInfo = orderItem.height || orderItem.length || orderItem.width || orderItem.marker;\n @if (showAdditionalInfo) {\n <div class=\"flex items-center gap-2 @lg:gap-3\">\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (isShowEditControls()) {\n <button\n tuiIconButton\n iconStart=\"@tui.settings\"\n [disabled]=\"isDisabled\"\n (click)=\"clickSettings.emit()\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"shrink-0\"\n ></button>\n }\n\n <!-- \u0414\u043E\u043F\u043E\u043B\u043D\u0438\u0442\u0435\u043B\u044C\u043D\u0430\u044F \u0438\u043D\u0444\u043E\u0440\u043C\u0430\u0446\u0438\u044F \u043E \u0442\u043E\u0432\u0430\u0440\u0435. -->\n <div\n class=\"flex flex-col gap-x-1 text-tui-base-07\"\n [class.text-body-m]=\"!isShowEditControls()\"\n >\n @if (orderItem.marker) {\n <p>\u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430: {{ orderItem.marker }}</p>\n }\n @if (orderItem.width) {\n <p>\u0428\u0438\u0440\u0438\u043D\u0430: {{ orderItem.width }} \u043C.</p>\n }\n @if (orderItem.height) {\n <p>\u0412\u044B\u0441\u043E\u0442\u0430: {{ orderItem.height }} \u043C.</p>\n } @else if (orderItem.length) {\n <p>\u0414\u043B\u0438\u043D\u0430: {{ orderItem.length }} \u043C.</p>\n }\n </div>\n </div>\n }\n </div>\n\n <!-- \u0421\u0443\u043C\u043C\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 \u0438 \u0441\u043A\u043B\u0430\u0434\u0441\u043A\u043E\u0435 \u043D\u0430\u043B\u0438\u0447\u0438\u0435. -->\n <div class=\"flex min-w-[11rem] flex-col @lg:justify-end @2xl:!justify-center text-body-m\">\n <p class=\"whitespace-nowrap\">\n \u0421\u0443\u043C\u043C\u0430:\n <strong> {{ orderItem.costRub | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</strong>\n </p>\n <sc-price-warehouse-stock [product]=\"product\" />\n </div>\n </div>\n </div>\n </sc-noindex-wrapper>\n </div>\n} @else {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n}\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: RouterModule }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiIcon, selector: "tui-icon", inputs: ["icon", "background"] }, { kind: "ngmodule", type: TuiTextfieldControllerModule }, { kind: "directive", type: i2.PolymorpheusOutlet, selector: "[polymorpheusOutlet]", inputs: ["polymorpheusOutlet", "polymorpheusOutletContext"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.TuiHintDirective, selector: "[tuiHint]:not(ng-container):not(ng-template)", inputs: ["tuiHintContext", "tuiHintAppearance", "tuiHint"], outputs: ["tuiHintVisible"] }, { kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: TuiHighlight, selector: "[tuiHighlight]", inputs: ["tuiHighlight", "tuiHighlightColor"] }, { kind: "directive", type: TuiAutoFocus, selector: "[tuiAutoFocus]", inputs: ["tuiAutoFocus"] }, { kind: "component", type: ScHoverImageCarouselComponent, selector: "sc-hover-image-carousel", inputs: ["images", "isShowActions", "productName"] }, { kind: "component", type: CostWithDiscountComponent, selector: "sc-cost-with-discount", inputs: ["product", "size"] }, { kind: "component", type: ScPriceWarehouseStockComponent, selector: "sc-price-warehouse-stock", inputs: ["classList", "product", "withStockHint", "fromMain"] }, { kind: "component", type: ScInputQuantityComponent, selector: "sc-input-quantity", inputs: ["step", "showLoader", "showCross", "ignoreStepValidators", "appearance", "isDisabled", "size"], outputs: ["clickClearEvent"] }, { kind: "pipe", type: TuiCurrencyPipe, name: "tuiCurrency" }, { kind: "pipe", type: TuiFormatNumberPipe, name: "tuiFormatNumber" }, { kind: "component", type: i5.TuiPreviewComponent, selector: "tui-preview", inputs: ["zoomable", "rotatable", "initialScale"] }, { kind: "directive", type: i5.TuiPreviewAction, selector: "[tuiPreviewAction]" }, { kind: "component", type: ScNoindexWrapperComponent, selector: "sc-noindex-wrapper" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
54
|
+
}
|
|
55
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemComponent, decorators: [{
|
|
56
|
+
type: Component,
|
|
57
|
+
args: [{ standalone: true, selector: 'sc-order-item', imports: [
|
|
58
|
+
CommonModule,
|
|
59
|
+
RouterModule,
|
|
60
|
+
TuiButton,
|
|
61
|
+
TuiIcon,
|
|
62
|
+
TuiTextfieldControllerModule,
|
|
63
|
+
FormsModule,
|
|
64
|
+
ReactiveFormsModule,
|
|
65
|
+
TuiHint,
|
|
66
|
+
TuiLink,
|
|
67
|
+
TuiHighlight,
|
|
68
|
+
TuiAutoFocus,
|
|
69
|
+
ScHoverImageCarouselComponent,
|
|
70
|
+
CostWithDiscountComponent,
|
|
71
|
+
ScPriceWarehouseStockComponent,
|
|
72
|
+
ScInputQuantityComponent,
|
|
73
|
+
TuiCurrencyPipe,
|
|
74
|
+
TuiFormatNumberPipe,
|
|
75
|
+
TuiPreview,
|
|
76
|
+
ScNoindexWrapperComponent,
|
|
77
|
+
TuiAppearance,
|
|
78
|
+
], providers: [
|
|
79
|
+
SEARCH_TERM_PROVIDERS,
|
|
80
|
+
tuiTextfieldOptionsProvider({
|
|
81
|
+
appearance: signal('textfield'),
|
|
82
|
+
}),
|
|
83
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (product && orderItem) {\n <div class=\"@container relative flex w-full gap-4 rounded-xl p-1 text-body-s hover:bg-tui-background-neutral-1\">\n <sc-noindex-wrapper>\n @let isDisabled = product.isHidden || product.isNull;\n <sc-hover-image-carousel\n (click)=\"isDisabled ? null : clickCardEvent.emit()\"\n [images]=\"!isMobile ? getCardImagePreviewList(product) : [getCardImagePreview()]\"\n [isShowActions]=\"false\"\n class=\"aspect-square w-24 shrink-0 cursor-pointer self-start\"\n />\n <div class=\"flex w-full flex-col gap-y-1 gap-x-4 @5xl:flex-row @5xl:items-center @5xl:gap-5\">\n <div class=\"flex min-w-0 flex-1 flex-col gap-1\">\n <div\n [tuiHighlight]=\"(search$ | async) ?? ''\"\n class=\"text-tui-text-02\"\n >\n \u0410\u0440\u0442\u0438\u043A\u0443\u043B: {{ product.code }}\n </div>\n <div class=\"flex flex-col gap-1\">\n <a\n tuiLink\n [attr.href]=\"href ?? null\"\n [tuiHighlight]=\"(search$ | async) ?? ''\"\n (click)=\"$event.preventDefault(); clickCardEvent.emit()\"\n [class.disabled]=\"isDisabled\"\n class=\"!text-body-m-bold\"\n >\n {{ product.name }}\n </a>\n @if (product.supplierSku) {\n <span class=\"text-tui-text-02\"> \u0410\u0440\u0442\u0438\u043A\u0443\u043B \u043F\u0440\u043E\u0438\u0437\u0432\u043E\u0434\u0438\u0442\u0435\u043B\u044F: {{ product.supplierSku }} </span>\n }\n @if (product.pack) {\n <div class=\"flex items-center gap-1 text-tui-text-02\">\n \u041D\u043E\u0440\u043C\u0430 \u0443\u043F\u0430\u043A\u043E\u0432\u043A\u0438: {{ product.pack }}\n @if (product.ignoreMinCountCheck) {\n <tui-icon\n icon=\"@tui.package\"\n [tuiHint]=\"minCountHint\"\n [tuiHintShowDelay]=\"100\"\n tuiHintDirection=\"top\"\n class=\"text-body-xl text-tui-text-01 opacity-90\"\n />\n <ng-template #minCountHint>\n \u0414\u043E\u0441\u0442\u0443\u043F\u0435\u043D \u0437\u0430\u043A\u0430\u0437 <br />\n \u043F\u0440\u043E\u0438\u0437\u0432\u043E\u043B\u044C\u043D\u043E\u0433\u043E <br />\n \u043A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u0430\n </ng-template>\n }\n </div>\n }\n\n <sc-cost-with-discount [product]=\"product\" />\n </div>\n @if (orderItem.specificationImgUrl) {\n <a\n tuiLink\n (click)=\"showSpecification(specificationPreview)\"\n [class.disabled]=\"isDisabled\"\n >\n \u0421\u043F\u0435\u0446\u0438\u0444\u0438\u043A\u0430\u0446\u0438\u044F\n </a>\n }\n <ng-template\n #specificationPreview\n let-preview\n >\n <tui-preview\n [rotatable]=\"false\"\n [zoomable]=\"false\"\n >\n <img\n *polymorpheusOutlet=\"orderItem.specificationImgUrl as src\"\n alt=\"preview\"\n [src]=\"orderItem.specificationImgUrl\"\n />\n <button\n iconStart=\"@tui.x\"\n title=\"Close\"\n tuiIconButton\n tuiPreviewAction\n type=\"button\"\n (click)=\"preview.complete()\"\n ></button>\n </tui-preview>\n </ng-template>\n </div>\n <div class=\"grid grid-cols-1 @lg:grid-cols-3 gap-2 @2xl:gap-4 @2xl:grid-rows-1\">\n <div class=\"grid grid-cols-1 items-center @lg:gap-2 @2xl:grid-cols-2 @lg:col-span-2\">\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (isShowEditControls() && showQuantityControl) {\n <!-- \u041F\u043E\u043B\u0435 \u0432\u0432\u043E\u0434\u0430 \u043A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u0430 \u0442\u043E\u0432\u0430\u0440\u0430. -->\n <sc-input-quantity\n #inputQuantity\n [tuiAutoFocus]=\"autoFocuseQuantityInput()\"\n [formControl]=\"quantityControl\"\n size=\"m\"\n [showCross]=\"false\"\n [isDisabled]=\"isDisabled\"\n [step]=\"unitsHelper.productMultiplicity(product)\"\n [ignoreStepValidators]=\"product.ignoreMinCountCheck\"\n [showLoader]=\"quantityShowLoader\"\n (clickClearEvent)=\"clickClearEvent.emit(orderItem)\"\n (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n class=\"w-32 shrink-0\"\n />\n } @else {\n <div class=\"text-body-m\">\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E:\n <strong> {{ orderItem.quantity }} \u0448\u0442.</strong>\n </div>\n }\n\n @let showAdditionalInfo = orderItem.height || orderItem.length || orderItem.width || orderItem.marker;\n @if (showAdditionalInfo) {\n <div class=\"flex items-center gap-2 @lg:gap-3\">\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430. -->\n @if (isShowEditControls()) {\n <button\n tuiIconButton\n iconStart=\"@tui.settings\"\n [disabled]=\"isDisabled\"\n (click)=\"clickSettings.emit()\"\n size=\"s\"\n appearance=\"secondary\"\n class=\"shrink-0\"\n ></button>\n }\n\n <!-- \u0414\u043E\u043F\u043E\u043B\u043D\u0438\u0442\u0435\u043B\u044C\u043D\u0430\u044F \u0438\u043D\u0444\u043E\u0440\u043C\u0430\u0446\u0438\u044F \u043E \u0442\u043E\u0432\u0430\u0440\u0435. -->\n <div\n class=\"flex flex-col gap-x-1 text-tui-base-07\"\n [class.text-body-m]=\"!isShowEditControls()\"\n >\n @if (orderItem.marker) {\n <p>\u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430: {{ orderItem.marker }}</p>\n }\n @if (orderItem.width) {\n <p>\u0428\u0438\u0440\u0438\u043D\u0430: {{ orderItem.width }} \u043C.</p>\n }\n @if (orderItem.height) {\n <p>\u0412\u044B\u0441\u043E\u0442\u0430: {{ orderItem.height }} \u043C.</p>\n } @else if (orderItem.length) {\n <p>\u0414\u043B\u0438\u043D\u0430: {{ orderItem.length }} \u043C.</p>\n }\n </div>\n </div>\n }\n </div>\n\n <!-- \u0421\u0443\u043C\u043C\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 \u0438 \u0441\u043A\u043B\u0430\u0434\u0441\u043A\u043E\u0435 \u043D\u0430\u043B\u0438\u0447\u0438\u0435. -->\n <div class=\"flex min-w-[11rem] flex-col @lg:justify-end @2xl:!justify-center text-body-m\">\n <p class=\"whitespace-nowrap\">\n \u0421\u0443\u043C\u043C\u0430:\n <strong> {{ orderItem.costRub | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</strong>\n </p>\n <sc-price-warehouse-stock [product]=\"product\" />\n </div>\n </div>\n </div>\n </sc-noindex-wrapper>\n </div>\n} @else {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n}\n" }]
|
|
84
|
+
}] });
|
|
85
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-order-item.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/order-item/sc-order-item.component.ts","../../../../../projects/client-ui/order/order-item/sc-order-item.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAoB,MAAM,EAAe,MAAM,eAAe,CAAC;AACjI,OAAO,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAClE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAC/D,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,YAAY,EAAE,MAAM,eAAe,CAAC;AAC7C,OAAO,EAAE,aAAa,EAAE,SAAS,EAAoB,mBAAmB,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,2BAA2B,EAAE,MAAM,gBAAgB,CAAC;AACzJ,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,uBAAuB,EAAE,MAAM,eAAe,CAAC;AAClF,OAAO,EAAE,4BAA4B,EAAE,MAAM,kBAAkB,CAAC;AAEhE,OAAO,EAAE,yBAAyB,EAAE,6BAA6B,EAAE,wBAAwB,EAAE,8BAA8B,EAAE,MAAM,eAAe,CAAC;AACnJ,OAAO,EAAE,mBAAmB,EAAE,MAAM,kBAAkB,CAAC;AACvD,OAAO,EAAE,yBAAyB,EAAE,MAAM,uBAAuB,CAAC;;;;;;;AAElE;;GAEG;AAmCH,MAAM,OAAO,oBAAqB,SAAQ,mBAAmB;IAlC7D;;QAmCI;;WAEG;QACa,kBAAa,GAA2B,MAAM,EAAE,CAAC;QAEjE;;WAEG;QACc,yBAAoB,GAA4B,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAEjG;;WAEG;QACa,uBAAkB,GAAG,KAAK,CAAU,IAAI,CAAC,CAAC;KAU7D;IARG;;;;OAIG;IACO,iBAAiB,CAAC,6BAA4D;QACpF,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC,SAAS,EAAE,CAAC;IAC9E,CAAC;+GAvBQ,oBAAoB;mGAApB,oBAAoB,oRARlB;YACP,qBAAqB;YACrB,2BAA2B,CAAC;gBACxB,UAAU,EAAE,MAAM,CAAC,WAAW,CAAC;aAClC,CAAC;SACL,iDCjDL,+hUAqKA,2CD9IQ,YAAY,mFACZ,YAAY,+BACZ,SAAS,oIACT,OAAO,oFACP,4BAA4B,iLAC5B,WAAW,sIACX,mBAAmB,uZAEnB,OAAO,4FACP,YAAY,0GACZ,YAAY,qFACZ,6BAA6B,wHAC7B,yBAAyB,+FACzB,8BAA8B,oIAC9B,wBAAwB,gMACxB,eAAe,+CACf,mBAAmB,2QAEnB,yBAAyB;;4FAWpB,oBAAoB;kBAlChC,SAAS;iCACM,IAAI,YACN,eAAe,WAEhB;wBACL,YAAY;wBACZ,YAAY;wBACZ,SAAS;wBACT,OAAO;wBACP,4BAA4B;wBAC5B,WAAW;wBACX,mBAAmB;wBACnB,OAAO;wBACP,OAAO;wBACP,YAAY;wBACZ,YAAY;wBACZ,6BAA6B;wBAC7B,yBAAyB;wBACzB,8BAA8B;wBAC9B,wBAAwB;wBACxB,eAAe;wBACf,mBAAmB;wBACnB,UAAU;wBACV,yBAAyB;wBACzB,aAAa;qBAChB,aACU;wBACP,qBAAqB;wBACrB,2BAA2B,CAAC;4BACxB,UAAU,EAAE,MAAM,CAAC,WAAW,CAAC;yBAClC,CAAC;qBACL,mBACgB,uBAAuB,CAAC,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, inject, input, output, OutputEmitterRef, signal, TemplateRef } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { RouterModule } from '@angular/router';\nimport { SEARCH_TERM_PROVIDERS } from '@snabcentr/client-core';\nimport { TuiCurrencyPipe } from '@taiga-ui/addon-commerce';\nimport { TuiAutoFocus } from '@taiga-ui/cdk';\nimport { TuiAppearance, TuiButton, TuiDialogContext, TuiFormatNumberPipe, TuiHint, TuiIcon, TuiLink, tuiTextfieldOptionsProvider } from '@taiga-ui/core';\nimport { TuiHighlight, TuiPreview, TuiPreviewDialogService } from '@taiga-ui/kit';\nimport { TuiTextfieldControllerModule } from '@taiga-ui/legacy';\n\nimport { CostWithDiscountComponent, ScHoverImageCarouselComponent, ScInputQuantityComponent, ScPriceWarehouseStockComponent } from '../../catalog';\nimport { AbstractScPriceCard } from '../../directives';\nimport { ScNoindexWrapperComponent } from '../../noindex-wrapper';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-order-item',\n    templateUrl: './sc-order-item.component.html',\n    imports: [\n        CommonModule,\n        RouterModule,\n        TuiButton,\n        TuiIcon,\n        TuiTextfieldControllerModule,\n        FormsModule,\n        ReactiveFormsModule,\n        TuiHint,\n        TuiLink,\n        TuiHighlight,\n        TuiAutoFocus,\n        ScHoverImageCarouselComponent,\n        CostWithDiscountComponent,\n        ScPriceWarehouseStockComponent,\n        ScInputQuantityComponent,\n        TuiCurrencyPipe,\n        TuiFormatNumberPipe,\n        TuiPreview,\n        ScNoindexWrapperComponent,\n        TuiAppearance,\n    ],\n    providers: [\n        SEARCH_TERM_PROVIDERS,\n        tuiTextfieldOptionsProvider({\n            appearance: signal('textfield'),\n        }),\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScOrderItemComponent extends AbstractScPriceCard {\n    /**\n     * Событие нажатия на кнопку редактирования конфигурации.\n     */\n    public readonly clickSettings: OutputEmitterRef<void> = output();\n\n    /**\n     * Сервис диалогового окна предварительного просмотра.\n     */\n    private readonly previewDialogService: TuiPreviewDialogService = inject(TuiPreviewDialogService);\n\n    /**\n     * Признак, что необходимо отобразить элементы редактирования.\n     */\n    public readonly isShowEditControls = input<boolean>(true);\n\n    /**\n     * Отображает спецификацию.\n     *\n     * @param specificationPreviewReference Шаблон спецификации.\n     */\n    protected showSpecification(specificationPreviewReference: TemplateRef<TuiDialogContext>): void {\n        this.previewDialogService.open(specificationPreviewReference).subscribe();\n    }\n}\n","@if (product && orderItem) {\n    <div class=\"@container relative flex w-full gap-4 rounded-xl p-1 text-body-s hover:bg-tui-background-neutral-1\">\n        <sc-noindex-wrapper>\n            @let isDisabled = product.isHidden || product.isNull;\n            <sc-hover-image-carousel\n                (click)=\"isDisabled ? null : clickCardEvent.emit()\"\n                [images]=\"!isMobile ? getCardImagePreviewList(product) : [getCardImagePreview()]\"\n                [isShowActions]=\"false\"\n                class=\"aspect-square w-24 shrink-0 cursor-pointer self-start\"\n            />\n            <div class=\"flex w-full flex-col gap-y-1 gap-x-4 @5xl:flex-row @5xl:items-center @5xl:gap-5\">\n                <div class=\"flex min-w-0 flex-1 flex-col gap-1\">\n                    <div\n                        [tuiHighlight]=\"(search$ | async) ?? ''\"\n                        class=\"text-tui-text-02\"\n                    >\n                        Артикул: {{ product.code }}\n                    </div>\n                    <div class=\"flex flex-col gap-1\">\n                        <a\n                            tuiLink\n                            [attr.href]=\"href ?? null\"\n                            [tuiHighlight]=\"(search$ | async) ?? ''\"\n                            (click)=\"$event.preventDefault(); clickCardEvent.emit()\"\n                            [class.disabled]=\"isDisabled\"\n                            class=\"!text-body-m-bold\"\n                        >\n                            {{ product.name }}\n                        </a>\n                        @if (product.supplierSku) {\n                            <span class=\"text-tui-text-02\"> Артикул производителя: {{ product.supplierSku }} </span>\n                        }\n                        @if (product.pack) {\n                            <div class=\"flex items-center gap-1 text-tui-text-02\">\n                                Норма упаковки: {{ product.pack }}\n                                @if (product.ignoreMinCountCheck) {\n                                    <tui-icon\n                                        icon=\"@tui.package\"\n                                        [tuiHint]=\"minCountHint\"\n                                        [tuiHintShowDelay]=\"100\"\n                                        tuiHintDirection=\"top\"\n                                        class=\"text-body-xl text-tui-text-01 opacity-90\"\n                                    />\n                                    <ng-template #minCountHint>\n                                        Доступен заказ <br />\n                                        произвольного <br />\n                                        количества\n                                    </ng-template>\n                                }\n                            </div>\n                        }\n\n                        <sc-cost-with-discount [product]=\"product\" />\n                    </div>\n                    @if (orderItem.specificationImgUrl) {\n                        <a\n                            tuiLink\n                            (click)=\"showSpecification(specificationPreview)\"\n                            [class.disabled]=\"isDisabled\"\n                        >\n                            Спецификация\n                        </a>\n                    }\n                    <ng-template\n                        #specificationPreview\n                        let-preview\n                    >\n                        <tui-preview\n                            [rotatable]=\"false\"\n                            [zoomable]=\"false\"\n                        >\n                            <img\n                                *polymorpheusOutlet=\"orderItem.specificationImgUrl as src\"\n                                alt=\"preview\"\n                                [src]=\"orderItem.specificationImgUrl\"\n                            />\n                            <button\n                                iconStart=\"@tui.x\"\n                                title=\"Close\"\n                                tuiIconButton\n                                tuiPreviewAction\n                                type=\"button\"\n                                (click)=\"preview.complete()\"\n                            ></button>\n                        </tui-preview>\n                    </ng-template>\n                </div>\n                <div class=\"grid grid-cols-1 @lg:grid-cols-3 gap-2 @2xl:gap-4 @2xl:grid-rows-1\">\n                    <div class=\"grid grid-cols-1 items-center @lg:gap-2 @2xl:grid-cols-2 @lg:col-span-2\">\n                        <!-- Количество товара. -->\n                        @if (isShowEditControls() && showQuantityControl) {\n                            <!-- Поле ввода количества товара. -->\n                            <sc-input-quantity\n                                #inputQuantity\n                                [tuiAutoFocus]=\"autoFocuseQuantityInput()\"\n                                [formControl]=\"quantityControl\"\n                                size=\"m\"\n                                [showCross]=\"false\"\n                                [isDisabled]=\"isDisabled\"\n                                [step]=\"unitsHelper.productMultiplicity(product)\"\n                                [ignoreStepValidators]=\"product.ignoreMinCountCheck\"\n                                [showLoader]=\"quantityShowLoader\"\n                                (clickClearEvent)=\"clickClearEvent.emit(orderItem)\"\n                                (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n                                class=\"w-32 shrink-0\"\n                            />\n                        } @else {\n                            <div class=\"text-body-m\">\n                                Количество:\n                                <strong> {{ orderItem.quantity }} шт.</strong>\n                            </div>\n                        }\n\n                        @let showAdditionalInfo = orderItem.height || orderItem.length || orderItem.width || orderItem.marker;\n                        @if (showAdditionalInfo) {\n                            <div class=\"flex items-center gap-2 @lg:gap-3\">\n                                <!-- Кнопка редактирования товара. -->\n                                @if (isShowEditControls()) {\n                                    <button\n                                        tuiIconButton\n                                        iconStart=\"@tui.settings\"\n                                        [disabled]=\"isDisabled\"\n                                        (click)=\"clickSettings.emit()\"\n                                        size=\"s\"\n                                        appearance=\"secondary\"\n                                        class=\"shrink-0\"\n                                    ></button>\n                                }\n\n                                <!-- Дополнительная информация о товаре. -->\n                                <div\n                                    class=\"flex flex-col gap-x-1 text-tui-base-07\"\n                                    [class.text-body-m]=\"!isShowEditControls()\"\n                                >\n                                    @if (orderItem.marker) {\n                                        <p>Маркировка: {{ orderItem.marker }}</p>\n                                    }\n                                    @if (orderItem.width) {\n                                        <p>Ширина: {{ orderItem.width }} м.</p>\n                                    }\n                                    @if (orderItem.height) {\n                                        <p>Высота: {{ orderItem.height }} м.</p>\n                                    } @else if (orderItem.length) {\n                                        <p>Длина: {{ orderItem.length }} м.</p>\n                                    }\n                                </div>\n                            </div>\n                        }\n                    </div>\n\n                    <!-- Сумма товара и складское наличие. -->\n                    <div class=\"flex min-w-[11rem] flex-col @lg:justify-end @2xl:!justify-center text-body-m\">\n                        <p class=\"whitespace-nowrap\">\n                            Сумма:\n                            <strong> {{ orderItem.costRub | tuiFormatNumber: { precision: 2, decimalSeparator: '.', rounding: 'ceil' } | async }} {{ 'RUB' | tuiCurrency }}</strong>\n                        </p>\n                        <sc-price-warehouse-stock [product]=\"product\" />\n                    </div>\n                </div>\n            </div>\n        </sc-noindex-wrapper>\n    </div>\n} @else {\n    <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n}\n"]}
|
|
@@ -0,0 +1,203 @@
|
|
|
1
|
+
/* eslint-disable no-param-reassign */
|
|
2
|
+
import { AsyncPipe } from '@angular/common';
|
|
3
|
+
import { ChangeDetectionStrategy, Component, computed, DestroyRef, inject, Input, input, signal } from '@angular/core';
|
|
4
|
+
import { outputFromObservable, takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
5
|
+
import { FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';
|
|
6
|
+
import { SC_ORDER_LOADER } from '@snabcentr/client-core';
|
|
7
|
+
import { TuiCurrencyPipe } from '@taiga-ui/addon-commerce';
|
|
8
|
+
import { TuiButton, TuiFormatNumberPipe } from '@taiga-ui/core';
|
|
9
|
+
import { TuiCheckbox } from '@taiga-ui/kit';
|
|
10
|
+
import { catchError, finalize, map, of } from 'rxjs';
|
|
11
|
+
import { SC_DIALOG_SERVICE_TOKEN } from '../../providers';
|
|
12
|
+
import { ScOrderItemComponent } from '../order-item/sc-order-item.component';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
import * as i1 from "@angular/forms";
|
|
15
|
+
/**
|
|
16
|
+
* Компонент списка товаров корзины, заказа или черновика.
|
|
17
|
+
* Позиции разбиты.
|
|
18
|
+
*/
|
|
19
|
+
export class ScOrderItemsListComponent {
|
|
20
|
+
constructor() {
|
|
21
|
+
/**
|
|
22
|
+
* Внутренний сигнал списка элементов корзины (товаров).
|
|
23
|
+
*/
|
|
24
|
+
this.itemsSignal = signal([]);
|
|
25
|
+
/**
|
|
26
|
+
* Включить выбор позиций чекбоксами.
|
|
27
|
+
*/
|
|
28
|
+
this.selectable = input(true);
|
|
29
|
+
/**
|
|
30
|
+
* Включить редактирование позиций.
|
|
31
|
+
*/
|
|
32
|
+
this.editable = input(false);
|
|
33
|
+
/**
|
|
34
|
+
* Внутренняя группа формы для выбора позиций.
|
|
35
|
+
*/
|
|
36
|
+
this.selectionFormGroup = new FormGroup({});
|
|
37
|
+
/**
|
|
38
|
+
* Идентификаторы выбранных позиций.
|
|
39
|
+
*/
|
|
40
|
+
this.itemsIds = toSignal(this.selectionFormGroup.valueChanges.pipe(map((value) => Object.entries(value))), {
|
|
41
|
+
initialValue: [],
|
|
42
|
+
});
|
|
43
|
+
/**
|
|
44
|
+
* Сигнал идентификаторов выбранных позиций.
|
|
45
|
+
*/
|
|
46
|
+
this.selectedItemsIds = computed(() => {
|
|
47
|
+
const selectable = this.selectable();
|
|
48
|
+
return this.itemsIds()
|
|
49
|
+
.filter(([, checked]) => !selectable || checked)
|
|
50
|
+
.map(([key]) => key);
|
|
51
|
+
});
|
|
52
|
+
/**
|
|
53
|
+
* Выходные данные при изменении выбора позиций.
|
|
54
|
+
*/
|
|
55
|
+
this.selectedItemsIdsChange = outputFromObservable(toObservable(this.selectedItemsIds));
|
|
56
|
+
/**
|
|
57
|
+
* Сумма стоимости выбранных позиций.
|
|
58
|
+
*/
|
|
59
|
+
this.sumSelectedItems = computed(() => {
|
|
60
|
+
const items = this.itemsSignal();
|
|
61
|
+
const selectedItemsIds = this.selectedItemsIds();
|
|
62
|
+
const selectable = this.selectable();
|
|
63
|
+
return items.filter((item) => !selectable || selectedItemsIds.includes(item.id)).reduce((sum, item) => sum + item.costRub, 0);
|
|
64
|
+
});
|
|
65
|
+
/**
|
|
66
|
+
* Сервис получения и редактирования списка товаров заказа (корзина / заказ / черновик = заказ).
|
|
67
|
+
* Предоставляется родителем через токен {@link SC_ORDER_LOADER} (реализация {@link ScIOrderLoader}).
|
|
68
|
+
*/
|
|
69
|
+
this.orderItemsListLoader = inject(SC_ORDER_LOADER);
|
|
70
|
+
/**
|
|
71
|
+
* Признак, что можно редактировать позиции.
|
|
72
|
+
*/
|
|
73
|
+
this.canEdit = computed(() => this.editable() && 'updateProduct$' in this.orderItemsListLoader);
|
|
74
|
+
/**
|
|
75
|
+
* Ссылка для автоматического управления уничтожением зависимостей.
|
|
76
|
+
*/
|
|
77
|
+
this.destroyRef = inject(DestroyRef);
|
|
78
|
+
/**
|
|
79
|
+
* Сервис для работы с диалоговыми окнами.
|
|
80
|
+
* Предоставляется родителем через токен {@link SC_DIALOG_SERVICE_TOKEN}.
|
|
81
|
+
*/
|
|
82
|
+
this.scDialogService = inject(SC_DIALOG_SERVICE_TOKEN);
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Список элементов корзины (товаров). При установке синхронно обновляет контролы выбора в форме.
|
|
86
|
+
*/
|
|
87
|
+
set items(value) {
|
|
88
|
+
this.itemsSignal.set(value);
|
|
89
|
+
this.syncSelectionControls();
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Синхронизирует контролы формы выбора с текущим списком товаров: удаляет контролы для отсутствующих id, добавляет для новых.
|
|
93
|
+
*/
|
|
94
|
+
syncSelectionControls() {
|
|
95
|
+
if (!this.selectable()) {
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
const currentItems = this.itemsSignal();
|
|
99
|
+
const currentIds = new Set(currentItems.map((item) => item.id));
|
|
100
|
+
Object.keys(this.selectionFormGroup.controls).forEach((name) => {
|
|
101
|
+
if (!currentIds.has(name)) {
|
|
102
|
+
this.selectionFormGroup.removeControl(name);
|
|
103
|
+
}
|
|
104
|
+
});
|
|
105
|
+
currentItems.forEach((item) => {
|
|
106
|
+
if (!this.selectionFormGroup.contains(item.id)) {
|
|
107
|
+
this.selectionFormGroup.addControl(item.id, new FormControl(true, { nonNullable: true }));
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Удаляет позицию.
|
|
113
|
+
*
|
|
114
|
+
* @param item Позиция товара для удаления.
|
|
115
|
+
* @param priceCard Компонент карточки товара.
|
|
116
|
+
*/
|
|
117
|
+
deleteItem(item, priceCard) {
|
|
118
|
+
if (!this.canEdit()) {
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
priceCard.quantityShowLoader = true;
|
|
122
|
+
this.orderItemsListLoader
|
|
123
|
+
.deleteProduct$(item)
|
|
124
|
+
.pipe(catchError(() => {
|
|
125
|
+
priceCard.quantityControl.patchValue(item.quantity, {
|
|
126
|
+
emitEvent: false,
|
|
127
|
+
});
|
|
128
|
+
return of();
|
|
129
|
+
}), finalize(() => {
|
|
130
|
+
priceCard.quantityShowLoader = false;
|
|
131
|
+
priceCard.markForCheck();
|
|
132
|
+
}), takeUntilDestroyed(this.destroyRef))
|
|
133
|
+
.subscribe();
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* Обновляет количество товара.
|
|
137
|
+
*
|
|
138
|
+
* @param newQuantity Новое значение количества товара.
|
|
139
|
+
* @param priceCard Компонент карточки товара.
|
|
140
|
+
*/
|
|
141
|
+
patchItem(newQuantity, priceCard) {
|
|
142
|
+
if (!this.canEdit() || !priceCard.orderItem || !priceCard.product || !newQuantity) {
|
|
143
|
+
return;
|
|
144
|
+
}
|
|
145
|
+
priceCard.quantityShowLoader = true;
|
|
146
|
+
this.orderItemsListLoader
|
|
147
|
+
.updateProduct$(priceCard.orderItem.id, { quantity: newQuantity })
|
|
148
|
+
.pipe(catchError(() => {
|
|
149
|
+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
150
|
+
priceCard.quantityControl.patchValue(priceCard.orderItem.quantity, {
|
|
151
|
+
emitEvent: false,
|
|
152
|
+
});
|
|
153
|
+
return of();
|
|
154
|
+
}), finalize(() => {
|
|
155
|
+
priceCard.quantityShowLoader = false;
|
|
156
|
+
priceCard.markForCheck();
|
|
157
|
+
}), takeUntilDestroyed(this.destroyRef))
|
|
158
|
+
.subscribe();
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Открывает модальное окно описания товара.
|
|
162
|
+
*
|
|
163
|
+
* @param product Товар/услуга, описание которой необходимо отобразить.
|
|
164
|
+
*/
|
|
165
|
+
openProductInfoDialog(product) {
|
|
166
|
+
if (this.scDialogService.openProductInfoDialog) {
|
|
167
|
+
this.scDialogService.openProductInfoDialog(product);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Открывает модальное окно изменения товара или его конфигуратора.
|
|
172
|
+
*
|
|
173
|
+
* @param item Позиция товара.
|
|
174
|
+
*/
|
|
175
|
+
openSettingsDialog(item) {
|
|
176
|
+
const isConfigurator = Boolean(item.configurator);
|
|
177
|
+
const { isLinear } = item.product.unit;
|
|
178
|
+
if (isConfigurator) {
|
|
179
|
+
this.scDialogService.openConfiguratorDialog(item.product, item);
|
|
180
|
+
}
|
|
181
|
+
else if (isLinear) {
|
|
182
|
+
this.scDialogService.openAddOrEditingCartItemDialog(item.product, item);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Открывает диалоговое окно "График изменения стоимости" для товара.
|
|
187
|
+
*
|
|
188
|
+
* @param product Товар/услуга, для которого необходимо отобразить график изменения стоимости.
|
|
189
|
+
*/
|
|
190
|
+
openPriceHistoryChartDialog(product) {
|
|
191
|
+
this.scDialogService.openPriceHistoryChartDialog(product);
|
|
192
|
+
}
|
|
193
|
+
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 }); }
|
|
195
|
+
}
|
|
196
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListComponent, decorators: [{
|
|
197
|
+
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" }]
|
|
199
|
+
}], propDecorators: { items: [{
|
|
200
|
+
type: Input,
|
|
201
|
+
args: [{ required: true }]
|
|
202
|
+
}] } });
|
|
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"]}
|
package/esm2022/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.mjs
ADDED
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, computed, inject, input, model, output } from '@angular/core';
|
|
2
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
|
3
|
+
import { ScReferencesService } from '@snabcentr/client-core';
|
|
4
|
+
import { TuiButton } from '@taiga-ui/core';
|
|
5
|
+
import { ScOrderItemsByDirection } from '../models/sc-i-cart-items-by-direction';
|
|
6
|
+
import { ScOrderItemsListComponent } from '../order-items-list/sc-order-items-list.component';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export class ScOrderItemsListByDirectionsComponent {
|
|
9
|
+
constructor() {
|
|
10
|
+
/**
|
|
11
|
+
* Внутренний сигнал списка элементов корзины (товаров).
|
|
12
|
+
*/
|
|
13
|
+
this.items = input.required();
|
|
14
|
+
/**
|
|
15
|
+
* Включить выбор позиций чекбоксами.
|
|
16
|
+
*/
|
|
17
|
+
this.selectable = input(true);
|
|
18
|
+
/**
|
|
19
|
+
* Включить редактирование позиций.
|
|
20
|
+
*/
|
|
21
|
+
this.editable = input(false);
|
|
22
|
+
/**
|
|
23
|
+
* Включить отображение кнопки продолжения.
|
|
24
|
+
*/
|
|
25
|
+
this.showContinueButton = input(true);
|
|
26
|
+
/**
|
|
27
|
+
* Текст кнопки продолжения.
|
|
28
|
+
*/
|
|
29
|
+
this.continueButtonText = input('Оформить заказ');
|
|
30
|
+
/**
|
|
31
|
+
* Событие клика на кнопку продолжения.
|
|
32
|
+
*/
|
|
33
|
+
this.continueClick = output();
|
|
34
|
+
/**
|
|
35
|
+
* Список направлений продаж из справочника.
|
|
36
|
+
*/
|
|
37
|
+
this.directions = toSignal(inject(ScReferencesService).directions$, { initialValue: undefined });
|
|
38
|
+
/**
|
|
39
|
+
* Позиции корзины, сгруппированные по направлениям продаж.
|
|
40
|
+
* Если передан cartDirections, используется он, иначе группировка выполняется автоматически.
|
|
41
|
+
*/
|
|
42
|
+
this.itemsByDirections = computed(() => {
|
|
43
|
+
const items = this.items().filter((item) => !item.product.isNull && !item.product.isHidden);
|
|
44
|
+
const directions = this.directions();
|
|
45
|
+
return directions?.map((direction) => new ScOrderItemsByDirection(direction, items)).filter((group) => group.items.length > 0) ?? [];
|
|
46
|
+
});
|
|
47
|
+
/**
|
|
48
|
+
* Позиции корзины, которые больше не предоставляются.
|
|
49
|
+
*/
|
|
50
|
+
this.nullItems = computed(() => this.items().filter((item) => item.product.isNull || item.product.isHidden));
|
|
51
|
+
/**
|
|
52
|
+
* Выходные данные при изменении выбора позиций.
|
|
53
|
+
*/
|
|
54
|
+
this.selectedItemsIds = model([]);
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Обработчик изменения выбора позиций.
|
|
58
|
+
*
|
|
59
|
+
* @param itemsIds Идентификаторы выбранных позиций.
|
|
60
|
+
* @param direction Направление продаж.
|
|
61
|
+
*/
|
|
62
|
+
selectedItemsHandler(itemsIds, direction) {
|
|
63
|
+
this.selectedItemsIds.update((previous) => [...previous.filter((id) => !direction.items.some((item) => item.id === id)), ...itemsIds]);
|
|
64
|
+
}
|
|
65
|
+
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 }); }
|
|
67
|
+
}
|
|
68
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScOrderItemsListByDirectionsComponent, decorators: [{
|
|
69
|
+
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" }]
|
|
71
|
+
}] });
|
|
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"]}
|