@snabcentr/client-ui 0.3.2-clients → 0.3.3-clients
Sign up to get free protection for your applications and to get access to all the features.
- package/esm2020/cart/cart-item-mobile/sc-cart-item-mobile.component.mjs +2 -2
- package/esm2020/catalog/price-card/sc-price-card.component.mjs +2 -2
- package/esm2020/icons/sc-client-ui-icons-name.mjs +2 -1
- package/esm2020/validators/stepValidator.mjs +4 -6
- package/fesm2015/snabcentr-client-ui.mjs +6 -7
- package/fesm2015/snabcentr-client-ui.mjs.map +1 -1
- package/fesm2020/snabcentr-client-ui.mjs +6 -7
- package/fesm2020/snabcentr-client-ui.mjs.map +1 -1
- package/icons/svg-pack/scIconPlasticProfile.svg +2 -2
- package/icons/svg-pack/scIconSandwichPanels.svg +11 -0
- package/icons/svg-pack/scIconSiding.svg +1 -1
- package/icons/svg-pack/scIconSmile.svg +4 -4
- package/package.json +2 -2
- package/validators/stepValidator.d.ts +2 -2
@@ -33,7 +33,7 @@ export class ScCartItemMobileComponent {
|
|
33
33
|
/**
|
34
34
|
* {@link FormControl} поля ввода количества товара в корзине.
|
35
35
|
*/
|
36
|
-
this.quantityControl = new FormControl(0);
|
36
|
+
this.quantityControl = new FormControl(0, { updateOn: 'blur' });
|
37
37
|
/**
|
38
38
|
* {@link Observable} изменения количества товара в корзине.
|
39
39
|
*/
|
@@ -117,4 +117,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
117
117
|
}], clickCardEvent: [{
|
118
118
|
type: Output
|
119
119
|
}] } });
|
120
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-item-mobile.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.ts","../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAe,SAAS,EAAE,MAAM,eAAe,CAAC;AACxI,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAkC,MAAM,wBAAwB,CAAC;AAC1G,OAAO,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;AACxC,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;;;;;;;;;;;AAIlE;;GAEG;AAMH,MAAM,OAAO,yBAAyB;IAkElC;;;;;;;OAOG;IACH,YACoB,WAAwB,EAEvB,oBAA6C,EAC5B,IAAa,EACG,iBAAyB;QAJ3D,gBAAW,GAAX,WAAW,CAAa;QAEvB,yBAAoB,GAApB,oBAAoB,CAAyB;QAC5B,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QAnE/E;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAC,CAAC,CAAC,CAAC;QAoBxE;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzG;;WAEG;QAEI,qBAAgB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEvE;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAuBlE,CAAC;IA/DJ;;OAEG;IACH,IACW,QAAQ,CAAC,IAA4B;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,IAAI,EAAE;YACN,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAClD;IACL,CAAC;IAED;;OAEG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IA0BD;;OAEG;IACH,IAAW,OAAO;QACd,OAAO,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;IAClC,CAAC;IAkBD,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;IAClE,CAAC;IAED;;OAEG;IACI,iBAAiB;QACpB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACnF,CAAC;IAED;;OAEG;IACI,YAAY;QACf,OAAO,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC;IAC7G,CAAC;;sHAnGQ,yBAAyB,6CA4EtB,uBAAuB,aAEvB,OAAO,aACP,uBAAuB;0GA/E1B,yBAAyB,2YChBtC,m5MAiGA;2FDjFa,yBAAyB;kBALrC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BA8E1C,MAAM;2BAAC,uBAAuB;;0BAE9B,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;4CArElB,uBAAuB;sBADvC,SAAS;uBAAC,sBAAsB;gBAYtB,QAAQ;sBADlB,KAAK;gBAmBC,oBAAoB;sBAD1B,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAOA,kBAAkB;sBADxB,MAAM;gBAOA,cAAc;sBADpB,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, OnInit, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScCartItem, ScIUrls, ScProduct } from '@snabcentr/client-core';\nimport { Observable, skip } from 'rxjs';\nimport { TuiPreviewDialogService } from '@taiga-ui/addon-preview';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    selector: 'sc-cart-item-mobile',\n    templateUrl: './sc-cart-item-mobile.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCartItemMobileComponent implements OnInit {\n    /**\n     * Элемент корзины.\n     */\n    private _cartItem?: ScCartItem;\n\n    /**\n     * Ссылка на представление спецификации.\n     */\n    @ViewChild('specificationPreview')\n    private readonly specificationPreviewRef?: TemplateRef<TuiDialogContext>;\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl(0);\n\n    /**\n     * Элемент корзины.\n     */\n    @Input()\n    public set cartItem(data: ScCartItem | undefined) {\n        this._cartItem = data;\n        if (data) {\n            this.quantityControl.patchValue(data.quantity);\n        }\n    }\n\n    /**\n     * Элемент корзины.\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges.pipe(skip(1));\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickDeleteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickSettingsEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на карточку.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Продукт элемента корзины.\n     */\n    public get product(): ScProduct | undefined {\n        return this.cartItem?.product;\n    }\n\n    /**\n     * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.\n     *\n     * @param unitsHelper Объект-хэлпер для работы со значениями единиц измерения товара.\n     * @param previewDialogService Сервис диалогового окна предварительного просмотра.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: UnitsHelper,\n        @Inject(TuiPreviewDialogService)\n        private readonly previewDialogService: TuiPreviewDialogService,\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string\n    ) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.quantityValueChanges = this.quantityControl.valueChanges;\n    }\n\n    /**\n     * Отобразить спецификацию.\n     */\n    public showSpecification(): void {\n        this.previewDialogService.open(this.specificationPreviewRef || '').subscribe();\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     */\n    public getCardImage(): string {\n        return this.product?.getImage(this.urls.imgServerUrl) ?? this.urls.imgServerUrl + this.pathImageNotFound;\n    }\n}\n","<div class=\"ml-2 mt-2 relative p-4 gap-y-2 gap-x-4 flex flex-wrap bg-white border border-tui-base-04 shadow-sc-2 rounded-xl\">\n    <ng-container *ngIf=\"cartItem && product; else skeleton\">\n        <!--\n        TODO: Необходима реализация функционала.\n        <button tuiIconButton size=\"m\" icon=\"scIconVerticalThreeDots\" appearance=\"float\" tuiMode=\"onLight\" class=\"!absolute right-0 top-0\"></button>\n        -->\n        <button\n            tuiIconButton\n            size=\"m\"\n            icon=\"tuiIconTrash2Large\"\n            appearance=\"secondary\"\n            tuiMode=\"onLight\"\n            (click)=\"clickDeleteEvent.emit()\"\n            class=\"shadow-sc-2 !absolute -left-2 -top-2\"\n        ></button>\n        <div class=\"flex gap-2\">\n            <div class=\"flex shrink-0 h-20 w-20 justify-center items-center overflow-hidden\">\n                <img (click)=\"clickCardEvent.emit()\" [src]=\"getCardImage()\" [alt]=\"product.name\" [class.p-5]=\"!product.images?.length\" class=\"cursor-pointer\" />\n            </div>\n\n            <div class=\"flex flex-wrap self-center gap-x-8 gap-y-0.5\">\n                <div class=\"w-[13rem]\">\n                    <a tuiLink (click)=\"clickCardEvent.emit()\">\n                        <span class=\"font-bold\">{{ product.name }}</span>\n                    </a>\n                    <div class=\"text-tui-text-02 text-xs\">\n                        <p>Артикул: {{ product.code }}</p>\n                        <p *ngIf=\"product.pack\">Норма упаковки: {{ product.pack }}</p>\n                        <a tuiLink *ngIf=\"cartItem.specificationImgUrl\" (click)=\"showSpecification()\">Спецификация</a>\n                        <ng-template #specificationPreview let-preview>\n                            <tui-preview [rotatable]=\"false\" [zoomable]=\"false\">\n                                <img *polymorpheusOutlet=\"cartItem.specificationImgUrl as src\" alt=\"preview\" [src]=\"cartItem.specificationImgUrl\" />\n                                <button icon=\"tuiIconClose\" title=\"Close\" tuiIconButton tuiPreviewAction type=\"button\" (click)=\"preview.complete()\"></button>\n                            </tui-preview>\n                        </ng-template>\n                    </div>\n                </div>\n                <div class=\"flex flex-col self-center w-[10rem] gap-x-8 gap-y-0.5\">\n                    <span *ngIf=\"product.discount\" class=\"flex items-center text-xs text-tui-text-02\">\n                        <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n                        <span class=\"text-tui-success-fill font-bold\"> -{{ product.discount.percent }}% </span>\n                        <tui-svg src=\"tuiIconInfoLarge\" [tuiHint]=\"discountHint\" [tuiHintShowDelay]=\"100\" tuiHintDirection=\"top\" class=\"text-tui-text-01 !text-xs !h-4\"></tui-svg>\n                        <ng-template #discountHint>\n                            <div class=\"font-bold\">{{ product.discount.name }}</div>\n                            <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n                        </ng-template>\n                    </span>\n                    <p class=\"flex flex-col items-baseline gap-x-2 font-bold\">\n                        <span>{{ product.costRubString }}</span>\n                        <span *ngIf=\"!product.priceInRub\" class=\"text-xs text-tui-text-02\">{{ product.costString }}</span>\n                    </p>\n                    <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n                </div>\n            </div>\n        </div>\n\n        <div class=\"flex gap-2 items-center w-44 text-xs text-tui-text-02\">\n            <!--\n            TODO: Необходима реализация функционала.\n            <button tuiIconButton (click)=\"clickSettingsEvent.emit()\" size=\"m\" icon=\"scIconSettings\" appearance=\"secondary\" tuiMode=\"onLight\"></button>\n            -->\n            <div>\n                <ng-container *ngIf=\"unitsHelper.productIsMeasurable(product); else notMeasurable\">\n                    <p>Длина: {{ cartItem.length }} {{ product.unit }}</p>\n                </ng-container>\n                <ng-template #notMeasurable>\n                    <p *ngIf=\"cartItem.length && !unitsHelper.productIsMeasurable(product)\">Длина: {{ cartItem.length }} м.</p>\n                    <p *ngIf=\"cartItem.width\">Ширина: {{ cartItem.width }} м.</p>\n                    <p *ngIf=\"cartItem.height\">Высота: {{ cartItem.height }} м.</p>\n                </ng-template>\n                <p class=\"font-bold text-tui-text-01\">Итого: {{ cartItem.costRub | tuiFormatNumber }} {{ 'RUB' | tuiCurrency }}</p>\n            </div>\n        </div>\n        <div class=\"flex items-center grow justify-end\">\n            <sc-input-quantity\n                class=\"w-28\"\n                *ngIf=\"quantityControl\"\n                [formControl]=\"quantityControl\"\n                [quantityUnit]=\"unitsHelper.productIsMeasurable(product) ? 'шт' : product.quantityUnit\"\n                [showCross]=\"false\"\n                size=\"s\"\n                [step]=\"unitsHelper.productMultiplicity(product)\"\n            ></sc-input-quantity>\n        </div>\n    </ng-container>\n\n    <ng-template #skeleton>\n        <div class=\"flex gap-2 w-full\">\n            <div class=\"h-20 w-20 bg-tui-base-02 rounded-xl \"></div>\n            <div class=\"flex flex-col grow gap-2.5 bg-white\">\n                <div class=\"w-full h-4 rounded-xl bg-tui-base-02\"></div>\n                <div class=\"w-3/5 h-4 rounded-xl bg-tui-base-02\"></div>\n                <div class=\"w-4/5 h-4 rounded-xl bg-tui-base-02\"></div>\n            </div>\n        </div>\n    </ng-template>\n</div>\n"]}
|
120
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-cart-item-mobile.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.ts","../../../../../projects/client-ui/cart/cart-item-mobile/sc-cart-item-mobile.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAe,SAAS,EAAE,MAAM,eAAe,CAAC;AACxI,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAkC,MAAM,wBAAwB,CAAC;AAC1G,OAAO,EAAc,IAAI,EAAE,MAAM,MAAM,CAAC;AACxC,OAAO,EAAE,uBAAuB,EAAE,MAAM,yBAAyB,CAAC;;;;;;;;;;;AAIlE;;GAEG;AAMH,MAAM,OAAO,yBAAyB;IAmElC;;;;;;;OAOG;IACH,YACoB,WAAwB,EAEvB,oBAA6C,EAC5B,IAAa,EACG,iBAAyB;QAJ3D,gBAAW,GAAX,WAAW,CAAa;QAEvB,yBAAoB,GAApB,oBAAoB,CAAyB;QAC5B,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QApE/E;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAgB,CAAC,EAAE,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC,CAAC;QAqB7G;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzG;;WAEG;QAEI,qBAAgB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEvE;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;IAuBlE,CAAC;IAhEJ;;OAEG;IACH,IACW,QAAQ,CAAC,IAA4B;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,IAAI,IAAI,EAAE;YACN,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAClD;IACL,CAAC;IAED;;OAEG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IA0BD;;OAEG;IACH,IAAW,OAAO;QACd,OAAO,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;IAClC,CAAC;IAkBD,kBAAkB;IACX,QAAQ;QACX,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;IAClE,CAAC;IAED;;OAEG;IACI,iBAAiB;QACpB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,uBAAuB,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC;IACnF,CAAC;IAED;;OAEG;IACI,YAAY;QACf,OAAO,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC;IAC7G,CAAC;;sHApGQ,yBAAyB,6CA6EtB,uBAAuB,aAEvB,OAAO,aACP,uBAAuB;0GAhF1B,yBAAyB,2YChBtC,m5MAiGA;2FDjFa,yBAAyB;kBALrC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BA+E1C,MAAM;2BAAC,uBAAuB;;0BAE9B,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;4CAtElB,uBAAuB;sBADvC,SAAS;uBAAC,sBAAsB;gBAYtB,QAAQ;sBADlB,KAAK;gBAoBC,oBAAoB;sBAD1B,MAAM;gBAOA,gBAAgB;sBADtB,MAAM;gBAOA,kBAAkB;sBADxB,MAAM;gBAOA,cAAc;sBADpB,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, OnInit, Output, TemplateRef, ViewChild } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScCartItem, ScIUrls, ScProduct } from '@snabcentr/client-core';\nimport { Observable, skip } from 'rxjs';\nimport { TuiPreviewDialogService } from '@taiga-ui/addon-preview';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\n\n/**\n * Компонент карточки элемента корзины.\n */\n@Component({\n    selector: 'sc-cart-item-mobile',\n    templateUrl: './sc-cart-item-mobile.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCartItemMobileComponent implements OnInit {\n    /**\n     * Элемент корзины.\n     */\n    private _cartItem?: ScCartItem;\n\n    /**\n     * Ссылка на представление спецификации.\n     */\n    @ViewChild('specificationPreview')\n    private readonly specificationPreviewRef?: TemplateRef<TuiDialogContext>;\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(0, { updateOn: 'blur' });\n\n    /**\n     * Элемент корзины.\n     */\n    @Input()\n    public set cartItem(data: ScCartItem | undefined) {\n        this._cartItem = data;\n\n        if (data) {\n            this.quantityControl.patchValue(data.quantity);\n        }\n    }\n\n    /**\n     * Элемент корзины.\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges.pipe(skip(1));\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickDeleteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"Удалить из корзины\".\n     */\n    @Output()\n    public clickSettingsEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на карточку.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Продукт элемента корзины.\n     */\n    public get product(): ScProduct | undefined {\n        return this.cartItem?.product;\n    }\n\n    /**\n     * Инициирует экземпляр класса {@link ScCartItemMobileComponent}.\n     *\n     * @param unitsHelper Объект-хэлпер для работы со значениями единиц измерения товара.\n     * @param previewDialogService Сервис диалогового окна предварительного просмотра.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: UnitsHelper,\n        @Inject(TuiPreviewDialogService)\n        private readonly previewDialogService: TuiPreviewDialogService,\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string\n    ) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.quantityValueChanges = this.quantityControl.valueChanges;\n    }\n\n    /**\n     * Отобразить спецификацию.\n     */\n    public showSpecification(): void {\n        this.previewDialogService.open(this.specificationPreviewRef || '').subscribe();\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     */\n    public getCardImage(): string {\n        return this.product?.getImage(this.urls.imgServerUrl) ?? this.urls.imgServerUrl + this.pathImageNotFound;\n    }\n}\n","<div class=\"ml-2 mt-2 relative p-4 gap-y-2 gap-x-4 flex flex-wrap bg-white border border-tui-base-04 shadow-sc-2 rounded-xl\">\n    <ng-container *ngIf=\"cartItem && product; else skeleton\">\n        <!--\n        TODO: Необходима реализация функционала.\n        <button tuiIconButton size=\"m\" icon=\"scIconVerticalThreeDots\" appearance=\"float\" tuiMode=\"onLight\" class=\"!absolute right-0 top-0\"></button>\n        -->\n        <button\n            tuiIconButton\n            size=\"m\"\n            icon=\"tuiIconTrash2Large\"\n            appearance=\"secondary\"\n            tuiMode=\"onLight\"\n            (click)=\"clickDeleteEvent.emit()\"\n            class=\"shadow-sc-2 !absolute -left-2 -top-2\"\n        ></button>\n        <div class=\"flex gap-2\">\n            <div class=\"flex shrink-0 h-20 w-20 justify-center items-center overflow-hidden\">\n                <img (click)=\"clickCardEvent.emit()\" [src]=\"getCardImage()\" [alt]=\"product.name\" [class.p-5]=\"!product.images?.length\" class=\"cursor-pointer\" />\n            </div>\n\n            <div class=\"flex flex-wrap self-center gap-x-8 gap-y-0.5\">\n                <div class=\"w-[13rem]\">\n                    <a tuiLink (click)=\"clickCardEvent.emit()\">\n                        <span class=\"font-bold\">{{ product.name }}</span>\n                    </a>\n                    <div class=\"text-tui-text-02 text-xs\">\n                        <p>Артикул: {{ product.code }}</p>\n                        <p *ngIf=\"product.pack\">Норма упаковки: {{ product.pack }}</p>\n                        <a tuiLink *ngIf=\"cartItem.specificationImgUrl\" (click)=\"showSpecification()\">Спецификация</a>\n                        <ng-template #specificationPreview let-preview>\n                            <tui-preview [rotatable]=\"false\" [zoomable]=\"false\">\n                                <img *polymorpheusOutlet=\"cartItem.specificationImgUrl as src\" alt=\"preview\" [src]=\"cartItem.specificationImgUrl\" />\n                                <button icon=\"tuiIconClose\" title=\"Close\" tuiIconButton tuiPreviewAction type=\"button\" (click)=\"preview.complete()\"></button>\n                            </tui-preview>\n                        </ng-template>\n                    </div>\n                </div>\n                <div class=\"flex flex-col self-center w-[10rem] gap-x-8 gap-y-0.5\">\n                    <span *ngIf=\"product.discount\" class=\"flex items-center text-xs text-tui-text-02\">\n                        <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n                        <span class=\"text-tui-success-fill font-bold\"> -{{ product.discount.percent }}% </span>\n                        <tui-svg src=\"tuiIconInfoLarge\" [tuiHint]=\"discountHint\" [tuiHintShowDelay]=\"100\" tuiHintDirection=\"top\" class=\"text-tui-text-01 !text-xs !h-4\"></tui-svg>\n                        <ng-template #discountHint>\n                            <div class=\"font-bold\">{{ product.discount.name }}</div>\n                            <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n                        </ng-template>\n                    </span>\n                    <p class=\"flex flex-col items-baseline gap-x-2 font-bold\">\n                        <span>{{ product.costRubString }}</span>\n                        <span *ngIf=\"!product.priceInRub\" class=\"text-xs text-tui-text-02\">{{ product.costString }}</span>\n                    </p>\n                    <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n                </div>\n            </div>\n        </div>\n\n        <div class=\"flex gap-2 items-center w-44 text-xs text-tui-text-02\">\n            <!--\n            TODO: Необходима реализация функционала.\n            <button tuiIconButton (click)=\"clickSettingsEvent.emit()\" size=\"m\" icon=\"scIconSettings\" appearance=\"secondary\" tuiMode=\"onLight\"></button>\n            -->\n            <div>\n                <ng-container *ngIf=\"unitsHelper.productIsMeasurable(product); else notMeasurable\">\n                    <p>Длина: {{ cartItem.length }} {{ product.unit }}</p>\n                </ng-container>\n                <ng-template #notMeasurable>\n                    <p *ngIf=\"cartItem.length && !unitsHelper.productIsMeasurable(product)\">Длина: {{ cartItem.length }} м.</p>\n                    <p *ngIf=\"cartItem.width\">Ширина: {{ cartItem.width }} м.</p>\n                    <p *ngIf=\"cartItem.height\">Высота: {{ cartItem.height }} м.</p>\n                </ng-template>\n                <p class=\"font-bold text-tui-text-01\">Итого: {{ cartItem.costRub | tuiFormatNumber }} {{ 'RUB' | tuiCurrency }}</p>\n            </div>\n        </div>\n        <div class=\"flex items-center grow justify-end\">\n            <sc-input-quantity\n                class=\"w-28\"\n                *ngIf=\"quantityControl\"\n                [formControl]=\"quantityControl\"\n                [quantityUnit]=\"unitsHelper.productIsMeasurable(product) ? 'шт' : product.quantityUnit\"\n                [showCross]=\"false\"\n                size=\"s\"\n                [step]=\"unitsHelper.productMultiplicity(product)\"\n            ></sc-input-quantity>\n        </div>\n    </ng-container>\n\n    <ng-template #skeleton>\n        <div class=\"flex gap-2 w-full\">\n            <div class=\"h-20 w-20 bg-tui-base-02 rounded-xl \"></div>\n            <div class=\"flex flex-col grow gap-2.5 bg-white\">\n                <div class=\"w-full h-4 rounded-xl bg-tui-base-02\"></div>\n                <div class=\"w-3/5 h-4 rounded-xl bg-tui-base-02\"></div>\n                <div class=\"w-4/5 h-4 rounded-xl bg-tui-base-02\"></div>\n            </div>\n        </div>\n    </ng-template>\n</div>\n"]}
|
@@ -65,7 +65,7 @@ export class ScPriceCardComponent {
|
|
65
65
|
/**
|
66
66
|
* {@link FormControl} поля ввода количества товара в корзине.
|
67
67
|
*/
|
68
|
-
this.quantityControl = new FormControl(null);
|
68
|
+
this.quantityControl = new FormControl(null, { updateOn: 'blur' });
|
69
69
|
/**
|
70
70
|
* {@link Observable} изменения количества товара в корзине.
|
71
71
|
*/
|
@@ -156,4 +156,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
156
156
|
type: HostBinding,
|
157
157
|
args: ['attr.data-size']
|
158
158
|
}] } });
|
159
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-card.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-card/sc-price-card.component.ts","../../../../../projects/client-ui/catalog/price-card/sc-price-card.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAqB,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACxI,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,OAAO,EAAE,uBAAuB,EAAE,OAAO,EAAmF,MAAM,wBAAwB,CAAC;;;;;;;;;;AAI3J;;GAEG;AAOH,MAAM,OAAO,oBAAoB;IAkH7B;;;;;;;;OAQG;IACH,YACoB,WAAwB,EACvB,WAA0B,EAC1B,gBAAoC,EACnB,IAAa,EACG,iBAAyB,EAC1D,GAAsB;QALvB,gBAAW,GAAX,WAAW,CAAa;QACvB,gBAAW,GAAX,WAAW,CAAe;QAC1B,qBAAgB,GAAhB,gBAAgB,CAAoB;QACnB,SAAI,GAAJ,IAAI,CAAS;QACG,sBAAiB,GAAjB,iBAAiB,CAAQ;QAC1D,QAAG,GAAH,GAAG,CAAmB;QAhI3C;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QACI,uBAAkB,GAAY,KAAK,CAAC;QAE3C;;WAEG;QAEI,wBAAmB,GAAY,KAAK,CAAC;QAgC5C;;WAEG;QAEI,uBAAkB,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAEzE;;WAEG;QAEI,wBAAmB,GAA4B,IAAI,YAAY,EAAa,CAAC;QAEpF;;WAEG;QAEI,oBAAe,GAA6B,IAAI,YAAY,EAAc,CAAC;QAElF;;WAEG;QAEI,mBAAc,GAAuB,IAAI,YAAY,EAAQ,CAAC;QAErE;;WAEG;QACI,qBAAgB,GAAoC,IAAI,CAAC,gBAAgB,CAAC,yBAAyB,EAAE,CAAC;QAE7G;;WAEG;QACI,oBAAe,GAA+B,IAAI,WAAW,CAAgB,IAAI,CAAC,CAAC;QAE1F;;WAEG;QAEI,yBAAoB,GAA8B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC;QAE3F;;WAEG;QACa,gBAAW,GAAwB,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;QAEpF;;WAEG;QAGI,SAAI,GAAyB,GAAG,CAAC;IAiCrC,CAAC;IAjHJ;;;;OAIG;IACH,IAAW,QAAQ;QACf,OAAO,IAAI,CAAC,SAAS,CAAC;IAC1B,CAAC;IAED;;;;OAIG;IACH,IACW,QAAQ,CAAC,KAA6B;QAC7C,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QAEvB,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,IAAI,IAAI,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;QACxF,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;IACpC,CAAC;IA8DD;;;OAGG;IACH,IAAW,eAAe;QACtB,OAAO,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;IAChD,CAAC;IA2BD;;OAEG;IACH,IAAW,eAAe;QACtB,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC;IACzB,CAAC;IAED;;OAEG;IACI,YAAY;QACf,OAAO,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,iBAAiB,CAAC;IACpF,CAAC;IAED;;OAEG;IACI,YAAY;QACf,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;;iHAvJQ,oBAAoB,4GA+HjB,OAAO,aACP,uBAAuB;qGAhI1B,oBAAoB,qbChBjC,qvKAqEA;2FDrDa,oBAAoB;kBANhC,SAAS;+BACI,eAAe,mBAGR,uBAAuB,CAAC,MAAM;;0BAiI1C,MAAM;2BAAC,OAAO;;0BACd,MAAM;2BAAC,uBAAuB;4EAjH5B,mBAAmB;sBADzB,KAAK;gBAkBK,QAAQ;sBADlB,KAAK;gBAcC,OAAO;sBADb,KAAK;gBAOC,kBAAkB;sBADxB,MAAM;gBAOA,mBAAmB;sBADzB,MAAM;gBAOA,eAAe;sBADrB,MAAM;gBAOA,cAAc;sBADpB,MAAM;gBAiBA,oBAAoB;sBAD1B,MAAM;gBAaA,IAAI;sBAFV,KAAK;;sBACL,WAAW;uBAAC,gBAAgB","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, Inject, Input, Output } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCartItem, ScIUrls, ScProduct, ScIWarehouse, ScWarehouseService } from '@snabcentr/client-core';\nimport { TuiSizeS, TuiSizeXS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\n/**\n * Компонент карточки товара.\n */\n@Component({\n    selector: 'sc-price-card',\n    templateUrl: './sc-price-card.component.html',\n    styleUrls: ['./sc-price-card.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceCardComponent {\n    /**\n     * Признак, что необходимо отобразить лоадер для поля ввода количества товара.\n     */\n    public quantityShowLoader: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить поле ввода количества товара.\n     */\n    @Input()\n    public showQuantityControl: boolean = false;\n\n    /**\n     * Позиция товара в корзине.\n     *\n     * TODO: Сделать товар в корзине наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * Позиция товара в корзине\n     *\n     * TODO: Сделать товар в корзине наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    @Input()\n    public set cartItem(value: ScCartItem | undefined) {\n        this._cartItem = value;\n\n        this.quantityControl.patchValue(this._cartItem?.quantity ?? null, { emitEvent: false });\n        this.quantityShowLoader = false;\n    }\n\n    /**\n     * Объект товара.\n     *\n     * TODO: Сделать товар наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    @Input()\n    public product?: ScProduct;\n\n    /**\n     * Событие нажатия на кнопку \"В избранное\".\n     */\n    @Output()\n    public clickFavoriteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"В корзину\".\n     */\n    @Output()\n    public clickAddToCartEvent: EventEmitter<ScProduct> = new EventEmitter<ScProduct>();\n\n    /**\n     * Событие нажатия на кнопку очистки количества товара.\n     */\n    @Output()\n    public clickClearEvent: EventEmitter<ScCartItem> = new EventEmitter<ScCartItem>();\n\n    /**\n     * Событие нажатия на карточку товара.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * {@link Observable} изменения выбранного склада.\n     */\n    public warehouseSelect$: Observable<ScIWarehouse | null> = this.warehouseService.getWarehouseSelectChange$();\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(null);\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = this.authService.getAuthChange();\n\n    /**\n     * Размер компонента.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS | TuiSizeXS = 'm';\n\n    /**\n     * Конвертация размера для компонента sc-input-quantity.\n     * TODO: Решить проблему с функционалом size для разных платформ.\n     */\n    public get getQuantitySize(): TuiSizeS {\n        return this.size === 'xs' ? 's' : this.size;\n    }\n\n    /**\n     * Позиция товара в корзине.\n     *\n     * TODO: Сделать товар в корзине наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    private _cartItem?: ScCartItem | undefined;\n\n    /**\n     * Инициирует экземпляр класса {@link ScPriceCardComponent}.\n     *\n     * @param unitsHelper Объект-хэлпер для работы со значениями единиц измерения товара.\n     * @param authService Сервис аутентификации пользователей.\n     * @param warehouseService Сервис для работы со складами.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: UnitsHelper,\n        private readonly authService: ScAuthService,\n        private readonly warehouseService: ScWarehouseService,\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Признак, что нужно показать скелетон.\n     */\n    public get skeletonVisible(): boolean {\n        return !this.product;\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     */\n    public getCardImage(): string {\n        return this.product?.getImage(this.urls.imgServerUrl) ?? this.pathImageNotFound;\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<!-- TODO: реализовать данный компонент и для десктопа и для мобильного приложения. Исправить все глобальные тайговские стили, привести их в порядок согласно с дизайном. -->\n<div *ngIf=\"product; else skeleton\" class=\"min-w-[10rem] shadow-md w-auto rounded-xl p-2 grid h-[19rem] text-xs relative\">\n    <div class=\"overflow-hidden\">\n        <img (click)=\"clickCardEvent.emit()\" [src]=\"getCardImage()\" [alt]=\"product.name\" class=\"rounded-xl w-full max-h-full object-cover\" />\n    </div>\n    <sc-favorite-btn\n        *ngIf=\"authStatus$ | async\"\n        class=\"top-0 left-0 absolute\"\n        (clickEvent)=\"clickFavoriteEvent.emit()\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"product.isFavorite\"\n        [disabled]=\"!!product.primaryCategory?.isFavorite\"\n    ></sc-favorite-btn>\n    <tui-svg *ngIf=\"product.isPreviouslyOrdered\" src=\"scIconStar\" class=\"top-0 right-0 absolute text-red-700\"></tui-svg>\n\n    <a class=\"p-0 m-0\" tuiLink iconAlign=\"left\" (click)=\"clickCardEvent.emit()\">\n        <p class=\"font-bold text-sm line-clamp-3\" style=\"word-break: break-word;\">{{ product.name }}</p>\n    </a>\n    <p *ngIf=\"product?.pack\" class=\"text-tui-text-02\">Норма упаковки: {{ product.pack }}</p>\n\n    <p class=\"text-tui-text-02\">Артикул: {{ product.code }}</p>\n    <p *ngIf=\"(authStatus$ | async) && product?.costDate\" class=\"text-tui-text-02\">Дата: {{ product.costDate }}</p>\n    <div *ngIf=\"warehouseSelect$ | async as warehouseSelect\" class=\"flex flex-col\">\n        <span *ngIf=\"product.discount\" class=\"flex items-center text-tui-text-02\">\n            <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n            <span class=\"text-tui-success-fill font-bold\"> -{{ product.discount.percent }}% </span>\n            <tui-svg src=\"tuiIconInfoLarge\" [tuiHint]=\"discountHint\" [tuiHintShowDelay]=\"100\" tuiHintDirection=\"top\" class=\"text-black !text-xs !h-4\"></tui-svg>\n            <ng-template #discountHint>\n                <div class=\"font-bold\">{{ product.discount.name }}</div>\n                <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n            </ng-template>\n        </span>\n        <span [class.text-tui-text-02]=\"!product.isWarehouseStockExist(warehouseSelect.id)\" class=\"cost font-bold\">{{ product.costRubString }}</span>\n        <span *ngIf=\"!product.priceInRub\" class=\"text-xs font-bold text-tui-text-02 hidden group-hover:block\">{{ product.costString }}</span>\n        <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n    </div>\n    <div *ngIf=\"!showQuantityControl\" class=\"flex gap-2 max-w-full items-center justify-stretch mt-1\">\n        <button tuiButton (click)=\"clickAddToCartEvent.emit(product)\" [showLoader]=\"quantityShowLoader\" [size]=\"size\" class=\"grow\">\n            <tui-svg src=\"scIconCart\" class=\"!text-xs !h-4\"> </tui-svg>\n            В корзину\n        </button>\n        <div *ngIf=\"cartItem\" class=\"flex justify-center\">\n            <tui-svg src=\"tuiIconCheck\" class=\"!h-5 !w-5 bg-tui-primary text-black rounded-md\"></tui-svg>\n        </div>\n    </div>\n    <sc-input-quantity\n        *ngIf=\"showQuantityControl\"\n        [formControl]=\"quantityControl\"\n        [quantityUnit]=\"product.quantityUnit\"\n        [size]=\"getQuantitySize\"\n        [step]=\"unitsHelper.productMultiplicity(product)\"\n        [showLoader]=\"quantityShowLoader\"\n        (clickClearEvent)=\"clickClearEvent.emit(cartItem)\"\n        class=\"w-full\"\n    ></sc-input-quantity>\n</div>\n\n<ng-template #skeleton>\n    <div class=\"flex flex-col card-wrapper bg-white rounded-xl overflow-hidden shadow-sc-1\">\n        <!-- Изображение товара -->\n        <div class=\"w-full h-3/5 rounded-t bg-tui-base-02\"></div>\n        <!-- Краткая информация о товаре -->\n        <div class=\"flex flex-col grow gap-2.5 bg-white rounded-b p-5\">\n            <div class=\"w-full h-4 rounded bg-tui-base-02\"></div>\n            <div class=\"w-3/5 h-4 rounded bg-tui-base-02\"></div>\n            <div class=\"w-full h-4 rounded bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
|
159
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-card.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-card/sc-price-card.component.ts","../../../../../projects/client-ui/catalog/price-card/sc-price-card.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, Inject, Input, Output } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\nimport { SC_PATH_IMAGE_NOT_FOUND, SC_URLS, ScAuthService, ScCartItem, ScIUrls, ScProduct, ScIWarehouse, ScWarehouseService } from '@snabcentr/client-core';\nimport { TuiSizeS, TuiSizeXS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\n/**\n * Компонент карточки товара.\n */\n@Component({\n    selector: 'sc-price-card',\n    templateUrl: './sc-price-card.component.html',\n    styleUrls: ['./sc-price-card.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceCardComponent {\n    /**\n     * Признак, что необходимо отобразить лоадер для поля ввода количества товара.\n     */\n    public quantityShowLoader: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить лоадер для кнопки избранных товаров и категорий.\n     */\n    public favoriteShowLoader: boolean = false;\n\n    /**\n     * Признак, что необходимо отобразить поле ввода количества товара.\n     */\n    @Input()\n    public showQuantityControl: boolean = false;\n\n    /**\n     * Позиция товара в корзине.\n     *\n     * TODO: Сделать товар в корзине наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    public get cartItem(): ScCartItem | undefined {\n        return this._cartItem;\n    }\n\n    /**\n     * Позиция товара в корзине\n     *\n     * TODO: Сделать товар в корзине наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    @Input()\n    public set cartItem(value: ScCartItem | undefined) {\n        this._cartItem = value;\n\n        this.quantityControl.patchValue(this._cartItem?.quantity ?? null, { emitEvent: false });\n        this.quantityShowLoader = false;\n    }\n\n    /**\n     * Объект товара.\n     *\n     * TODO: Сделать товар наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    @Input()\n    public product?: ScProduct;\n\n    /**\n     * Событие нажатия на кнопку \"В избранное\".\n     */\n    @Output()\n    public clickFavoriteEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Событие нажатия на кнопку \"В корзину\".\n     */\n    @Output()\n    public clickAddToCartEvent: EventEmitter<ScProduct> = new EventEmitter<ScProduct>();\n\n    /**\n     * Событие нажатия на кнопку очистки количества товара.\n     */\n    @Output()\n    public clickClearEvent: EventEmitter<ScCartItem> = new EventEmitter<ScCartItem>();\n\n    /**\n     * Событие нажатия на карточку товара.\n     */\n    @Output()\n    public clickCardEvent: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * {@link Observable} изменения выбранного склада.\n     */\n    public warehouseSelect$: Observable<ScIWarehouse | null> = this.warehouseService.getWarehouseSelectChange$();\n\n    /**\n     * {@link FormControl} поля ввода количества товара в корзине.\n     */\n    public quantityControl: FormControl<number | null> = new FormControl<number | null>(null, { updateOn: 'blur' });\n\n    /**\n     * {@link Observable} изменения количества товара в корзине.\n     */\n    @Output()\n    public quantityValueChanges: Observable<number | null> = this.quantityControl.valueChanges;\n\n    /**\n     * {@link Observable} изменения статуса авторизации.\n     */\n    public readonly authStatus$: Observable<boolean> = this.authService.getAuthChange();\n\n    /**\n     * Размер компонента.\n     */\n    @Input()\n    @HostBinding('attr.data-size')\n    public size: TuiSizeS | TuiSizeXS = 'm';\n\n    /**\n     * Конвертация размера для компонента sc-input-quantity.\n     * TODO: Решить проблему с функционалом size для разных платформ.\n     */\n    public get getQuantitySize(): TuiSizeS {\n        return this.size === 'xs' ? 's' : this.size;\n    }\n\n    /**\n     * Позиция товара в корзине.\n     *\n     * TODO: Сделать товар в корзине наблюдаемой переменной после реализации TASK:[#7144].\n     */\n    private _cartItem?: ScCartItem | undefined;\n\n    /**\n     * Инициирует экземпляр класса {@link ScPriceCardComponent}.\n     *\n     * @param unitsHelper Объект-хэлпер для работы со значениями единиц измерения товара.\n     * @param authService Сервис аутентификации пользователей.\n     * @param warehouseService Сервис для работы со складами.\n     * @param urls Список ссылок на разделы backend'a.\n     * @param pathImageNotFound Путь до изображения 'Товар не найден'.\n     */\n    public constructor(\n        public readonly unitsHelper: UnitsHelper,\n        private readonly authService: ScAuthService,\n        private readonly warehouseService: ScWarehouseService,\n        @Inject(SC_URLS) private readonly urls: ScIUrls,\n        @Inject(SC_PATH_IMAGE_NOT_FOUND) private readonly pathImageNotFound: string,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    /**\n     * Признак, что нужно показать скелетон.\n     */\n    public get skeletonVisible(): boolean {\n        return !this.product;\n    }\n\n    /**\n     * Возвращает ссылку на изображение карточки товара.\n     */\n    public getCardImage(): string {\n        return this.product?.getImage(this.urls.imgServerUrl) ?? this.pathImageNotFound;\n    }\n\n    /**\n     * Устанавливает компонент в очередь на обновление.\n     */\n    public markForCheck(): void {\n        this.cdr.markForCheck();\n    }\n}\n","<!-- TODO: реализовать данный компонент и для десктопа и для мобильного приложения. Исправить все глобальные тайговские стили, привести их в порядок согласно с дизайном. -->\n<div *ngIf=\"product; else skeleton\" class=\"min-w-[10rem] shadow-md w-auto rounded-xl p-2 grid h-[19rem] text-xs relative\">\n    <div class=\"overflow-hidden\">\n        <img (click)=\"clickCardEvent.emit()\" [src]=\"getCardImage()\" [alt]=\"product.name\" class=\"rounded-xl w-full max-h-full object-cover\" />\n    </div>\n    <sc-favorite-btn\n        *ngIf=\"authStatus$ | async\"\n        class=\"top-0 left-0 absolute\"\n        (clickEvent)=\"clickFavoriteEvent.emit()\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"product.isFavorite\"\n        [disabled]=\"!!product.primaryCategory?.isFavorite\"\n    ></sc-favorite-btn>\n    <tui-svg *ngIf=\"product.isPreviouslyOrdered\" src=\"scIconStar\" class=\"top-0 right-0 absolute text-red-700\"></tui-svg>\n\n    <a class=\"p-0 m-0\" tuiLink iconAlign=\"left\" (click)=\"clickCardEvent.emit()\">\n        <p class=\"font-bold text-sm line-clamp-3\" style=\"word-break: break-word;\">{{ product.name }}</p>\n    </a>\n    <p *ngIf=\"product?.pack\" class=\"text-tui-text-02\">Норма упаковки: {{ product.pack }}</p>\n\n    <p class=\"text-tui-text-02\">Артикул: {{ product.code }}</p>\n    <p *ngIf=\"(authStatus$ | async) && product?.costDate\" class=\"text-tui-text-02\">Дата: {{ product.costDate }}</p>\n    <div *ngIf=\"warehouseSelect$ | async as warehouseSelect\" class=\"flex flex-col\">\n        <span *ngIf=\"product.discount\" class=\"flex items-center text-tui-text-02\">\n            <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n            <span class=\"text-tui-success-fill font-bold\"> -{{ product.discount.percent }}% </span>\n            <tui-svg src=\"tuiIconInfoLarge\" [tuiHint]=\"discountHint\" [tuiHintShowDelay]=\"100\" tuiHintDirection=\"top\" class=\"text-black !text-xs !h-4\"></tui-svg>\n            <ng-template #discountHint>\n                <div class=\"font-bold\">{{ product.discount.name }}</div>\n                <div *ngIf=\"product.discount.expiredAt as expiredAt\">Дата окончания: {{ expiredAt }}</div>\n            </ng-template>\n        </span>\n        <span [class.text-tui-text-02]=\"!product.isWarehouseStockExist(warehouseSelect.id)\" class=\"cost font-bold\">{{ product.costRubString }}</span>\n        <span *ngIf=\"!product.priceInRub\" class=\"text-xs font-bold text-tui-text-02 hidden group-hover:block\">{{ product.costString }}</span>\n        <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n    </div>\n    <div *ngIf=\"!showQuantityControl\" class=\"flex gap-2 max-w-full items-center justify-stretch mt-1\">\n        <button tuiButton (click)=\"clickAddToCartEvent.emit(product)\" [showLoader]=\"quantityShowLoader\" [size]=\"size\" class=\"grow\">\n            <tui-svg src=\"scIconCart\" class=\"!text-xs !h-4\"> </tui-svg>\n            В корзину\n        </button>\n        <div *ngIf=\"cartItem\" class=\"flex justify-center\">\n            <tui-svg src=\"tuiIconCheck\" class=\"!h-5 !w-5 bg-tui-primary text-black rounded-md\"></tui-svg>\n        </div>\n    </div>\n    <sc-input-quantity\n        *ngIf=\"showQuantityControl\"\n        [formControl]=\"quantityControl\"\n        [quantityUnit]=\"product.quantityUnit\"\n        [size]=\"getQuantitySize\"\n        [step]=\"unitsHelper.productMultiplicity(product)\"\n        [showLoader]=\"quantityShowLoader\"\n        (clickClearEvent)=\"clickClearEvent.emit(cartItem)\"\n        class=\"w-full\"\n    ></sc-input-quantity>\n</div>\n\n<ng-template #skeleton>\n    <div class=\"flex flex-col card-wrapper bg-white rounded-xl overflow-hidden shadow-sc-1\">\n        <!-- Изображение товара -->\n        <div class=\"w-full h-3/5 rounded-t bg-tui-base-02\"></div>\n        <!-- Краткая информация о товаре -->\n        <div class=\"flex flex-col grow gap-2.5 bg-white rounded-b p-5\">\n            <div class=\"w-full h-4 rounded bg-tui-base-02\"></div>\n            <div class=\"w-3/5 h-4 rounded bg-tui-base-02\"></div>\n            <div class=\"w-full h-4 rounded bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
|
@@ -5,6 +5,7 @@ export const scClientUiIconsName = [
|
|
5
5
|
'scIconWordFile',
|
6
6
|
'scIconWindowsill',
|
7
7
|
'scIconWarning',
|
8
|
+
'scIconSandwichPanels',
|
8
9
|
'scIconWallet',
|
9
10
|
'scIconViewTree',
|
10
11
|
'scIconViewInline',
|
@@ -87,4 +88,4 @@ export const scClientUiIconsName = [
|
|
87
88
|
'scIconApplication',
|
88
89
|
'scIconAddProfile',
|
89
90
|
];
|
90
|
-
//# sourceMappingURL=data:application/json;base64,
|
91
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2xpZW50LXVpLWljb25zLW5hbWUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvaWNvbnMvc2MtY2xpZW50LXVpLWljb25zLW5hbWUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRztJQUMvQixnQkFBZ0I7SUFDaEIsa0JBQWtCO0lBQ2xCLGVBQWU7SUFDZixzQkFBc0I7SUFDdEIsY0FBYztJQUNkLGdCQUFnQjtJQUNoQixrQkFBa0I7SUFDbEIsV0FBVztJQUNYLGlCQUFpQjtJQUNqQixhQUFhO0lBQ2IsZ0JBQWdCO0lBQ2hCLFlBQVk7SUFDWixzQkFBc0I7SUFDdEIsZ0JBQWdCO0lBQ2hCLG1CQUFtQjtJQUNuQixzQkFBc0I7SUFDdEIsZ0JBQWdCO0lBQ2hCLHVCQUF1QjtJQUN2QixzQkFBc0I7SUFDdEIsbUJBQW1CO0lBQ25CLGtCQUFrQjtJQUNsQixhQUFhO0lBQ2IsY0FBYztJQUNkLGFBQWE7SUFDYixZQUFZO0lBQ1osY0FBYztJQUNkLFlBQVk7SUFDWixnQkFBZ0I7SUFDaEIsY0FBYztJQUNkLGtCQUFrQjtJQUNsQixjQUFjO0lBQ2QsbUJBQW1CO0lBQ25CLGtCQUFrQjtJQUNsQixzQkFBc0I7SUFDdEIsZUFBZTtJQUNmLGFBQWE7SUFDYixzQkFBc0I7SUFDdEIsa0JBQWtCO0lBQ2xCLGVBQWU7SUFDZixlQUFlO0lBQ2YsYUFBYTtJQUNiLFlBQVk7SUFDWixtQkFBbUI7SUFDbkIsYUFBYTtJQUNiLFlBQVk7SUFDWixhQUFhO0lBQ2IsZ0JBQWdCO0lBQ2hCLFlBQVk7SUFDWixZQUFZO0lBQ1osWUFBWTtJQUNaLGVBQWU7SUFDZixpQkFBaUI7SUFDakIsYUFBYTtJQUNiLGFBQWE7SUFDYixrQkFBa0I7SUFDbEIsY0FBYztJQUNkLFlBQVk7SUFDWixtQkFBbUI7SUFDbkIsY0FBYztJQUNkLFlBQVk7SUFDWixvQkFBb0I7SUFDcEIsZ0JBQWdCO0lBQ2hCLGdCQUFnQjtJQUNoQixpQkFBaUI7SUFDakIsYUFBYTtJQUNiLFlBQVk7SUFDWixxQkFBcUI7SUFDckIsZ0JBQWdCO0lBQ2hCLG9CQUFvQjtJQUNwQixhQUFhO0lBQ2IsZUFBZTtJQUNmLG1CQUFtQjtJQUNuQixZQUFZO0lBQ1osY0FBYztJQUNkLGdCQUFnQjtJQUNoQixZQUFZO0lBQ1osWUFBWTtJQUNaLFdBQVc7SUFDWCxpQkFBaUI7SUFDakIsbUJBQW1CO0lBQ25CLHVCQUF1QjtJQUN2QixxQkFBcUI7SUFDckIsb0JBQW9CO0lBQ3BCLG1CQUFtQjtJQUNuQixrQkFBa0I7Q0FDckIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICog0KHQv9C40YHQvtC6INC40LrQvtC90L7Qui5cbiAqL1xuZXhwb3J0IGNvbnN0IHNjQ2xpZW50VWlJY29uc05hbWUgPSBbXG4gICAgJ3NjSWNvbldvcmRGaWxlJyxcbiAgICAnc2NJY29uV2luZG93c2lsbCcsXG4gICAgJ3NjSWNvbldhcm5pbmcnLFxuICAgICdzY0ljb25TYW5kd2ljaFBhbmVscycsXG4gICAgJ3NjSWNvbldhbGxldCcsXG4gICAgJ3NjSWNvblZpZXdUcmVlJyxcbiAgICAnc2NJY29uVmlld0lubGluZScsXG4gICAgJ3NjSWNvblRvcCcsXG4gICAgJ3NjSWNvblRlbXBsYXRlcycsXG4gICAgJ3NjSWNvblRhc2tzJyxcbiAgICAnc2NJY29uU3VpdGNhc2UnLFxuICAgICdzY0ljb25TdGFyJyxcbiAgICAnc2NJY29uU29jaWFsV2hhdHNhcHAnLFxuICAgICdzY0ljb25Tb2NpYWxWaycsXG4gICAgJ3NjSWNvblNvY2lhbFZpYmVyJyxcbiAgICAnc2NJY29uU29jaWFsVGVsZWdyYW0nLFxuICAgICdzY0ljb25Tb2NpYWxRUicsXG4gICAgJ3NjSWNvblNvY2lhbEluc3RhZ3JhbScsXG4gICAgJ3NjSWNvblNvY2lhbEZhY2Vib29rJyxcbiAgICAnc2NJY29uU29jaWFsRW1haWwnLFxuICAgICdzY0ljb25Tb2NpYWxDb3B5JyxcbiAgICAnc2NJY29uU21pbGUnLFxuICAgICdzY0ljb25TaWRpbmcnLFxuICAgICdzY0ljb25TaGFyZScsXG4gICAgJ3NjSWNvblNlbmQnLFxuICAgICdzY0ljb25TZWFyY2gnLFxuICAgICdzY0ljb25TYXZlJyxcbiAgICAnc2NJY29uU2FuZHdpY2gnLFxuICAgICdzY0ljb25Sb2NrZXQnLFxuICAgICdzY0ljb25SZXF1aXNpdGVzJyxcbiAgICAnc2NJY29uUmVwZWF0JyxcbiAgICAnc2NJY29uUmVjbGFtYXRpb24nLFxuICAgICdzY0ljb25RUkNvZGVTY2FuJyxcbiAgICAnc2NJY29uUHJvbW9NYXRlcmlhbHMnLFxuICAgICdzY0ljb25Qcm9maWxlJyxcbiAgICAnc2NJY29uUHJpY2UnLFxuICAgICdzY0ljb25QbGFzdGljUHJvZmlsZScsXG4gICAgJ3NjSWNvblBlcmNlbnRhZ2UnLFxuICAgICdzY0ljb25QZGZGaWxlJyxcbiAgICAnc2NJY29uUGFsZXR0ZScsXG4gICAgJ3NjSWNvbk9mZmVyJyxcbiAgICAnc2NJY29uTmV3cycsXG4gICAgJ3NjSWNvbk1vc3F1aXRvTmV0JyxcbiAgICAnc2NJY29uTW9uZXknLFxuICAgICdzY0ljb25NYWlsJyxcbiAgICAnc2NJY29uTG9nSW4nLFxuICAgICdzY0ljb25Mb2NrT3BlbicsXG4gICAgJ3NjSWNvbkxvY2snLFxuICAgICdzY0ljb25MaXN0JyxcbiAgICAnc2NJY29uSG9tZScsXG4gICAgJ3NjSWNvbkhpc3RvcnknLFxuICAgICdzY0ljb25IZWFydEZpbGwnLFxuICAgICdzY0ljb25IZWFydCcsXG4gICAgJ3NjSWNvbkdyYXBoJyxcbiAgICAnc2NJY29uRm9sZGVyT3BlbicsXG4gICAgJ3NjSWNvbkZvbGRlcicsXG4gICAgJ3NjSWNvbkZvYW0nLFxuICAgICdzY0ljb25Gb2FtQmFsbG9vbicsXG4gICAgJ3NjSWNvbkZpbHRlcicsXG4gICAgJ3NjSWNvbkZpbGUnLFxuICAgICdzY0ljb25GYXZvcml0ZUZpbGwnLFxuICAgICdzY0ljb25GYXZvcml0ZScsXG4gICAgJ3NjSWNvbkZhc3RlbmVyJyxcbiAgICAnc2NJY29uRXhjZWxGaWxlJyxcbiAgICAnc2NJY29uRXJyb3InLFxuICAgICdzY0ljb25FZGl0JyxcbiAgICAnc2NJY29uRGVsZXRlUHJvZmlsZScsXG4gICAgJ3NjSWNvbkNvbnRhY3RzJyxcbiAgICAnc2NJY29uQ29uZmlndXJhdG9yJyxcbiAgICAnc2NJY29uQ2xvY2snLFxuICAgICdzY0ljb25DbGllbnRzJyxcbiAgICAnc2NJY29uQ2F0YWxvZ0JhY2snLFxuICAgICdzY0ljb25DYXJ0JyxcbiAgICAnc2NJY29uQ2Fub3B5JyxcbiAgICAnc2NJY29uQ2FsZW5kYXInLFxuICAgICdzY0ljb25Cb29rJyxcbiAgICAnc2NJY29uQmVsbCcsXG4gICAgJ3NjSWNvbkJhZycsXG4gICAgJ3NjSWNvbkF0dGVudGlvbicsXG4gICAgJ3NjSWNvbkFycm93UmV0dXJuJyxcbiAgICAnc2NJY29uQXJyb3dMYXJnZVN0YXJ0JyxcbiAgICAnc2NJY29uQXJyb3dMYXJnZUVuZCcsXG4gICAgJ3NjSWNvbkFycm93Rm9yd2FyZCcsXG4gICAgJ3NjSWNvbkFwcGxpY2F0aW9uJyxcbiAgICAnc2NJY29uQWRkUHJvZmlsZScsXG5dO1xuIl19
|
@@ -2,16 +2,14 @@
|
|
2
2
|
* Проверяет кратность значения.
|
3
3
|
*
|
4
4
|
* @param step Шаг кратности.
|
5
|
-
* @param
|
5
|
+
* @param precision Точность, количество чисел после запятой.
|
6
6
|
*/
|
7
|
-
export function stepValidator(step,
|
7
|
+
export function stepValidator(step, precision = 3) {
|
8
8
|
return ({ value }) => {
|
9
9
|
if (value) {
|
10
|
-
|
11
|
-
const closestMultiple = Math.round(controlValue / step) * step;
|
12
|
-
return Math.abs(controlValue - closestMultiple) > tolerance ? { error: `Число не кратно ${step}` } : null;
|
10
|
+
return Number((value / step).toFixed(precision)) % 1 !== 0 ? { error: `Число не кратно ${step}` } : null;
|
13
11
|
}
|
14
12
|
return null;
|
15
13
|
};
|
16
14
|
}
|
17
|
-
//# sourceMappingURL=data:application/json;base64,
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcFZhbGlkYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS92YWxpZGF0b3JzL3N0ZXBWYWxpZGF0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUE7Ozs7O0dBS0c7QUFDSCxNQUFNLFVBQVUsYUFBYSxDQUFDLElBQVksRUFBRSxZQUFvQixDQUFDO0lBQzdELE9BQU8sQ0FBQyxFQUFFLEtBQUssRUFBa0MsRUFBMkIsRUFBRTtRQUMxRSxJQUFJLEtBQUssRUFBRTtZQUNQLE9BQU8sTUFBTSxDQUFDLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsS0FBSyxFQUFFLG1CQUFtQixJQUFJLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7U0FDNUc7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQWJzdHJhY3RDb250cm9sLCBWYWxpZGF0b3JGbiwgVmFsaWRhdGlvbkVycm9ycyB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcblxuLyoqXG4gKiDQn9GA0L7QstC10YDRj9C10YIg0LrRgNCw0YLQvdC+0YHRgtGMINC30L3QsNGH0LXQvdC40Y8uXG4gKlxuICogQHBhcmFtIHN0ZXAg0KjQsNCzINC60YDQsNGC0L3QvtGB0YLQuC5cbiAqIEBwYXJhbSBwcmVjaXNpb24g0KLQvtGH0L3QvtGB0YLRjCwg0LrQvtC70LjRh9C10YHRgtCy0L4g0YfQuNGB0LXQuyDQv9C+0YHQu9C1INC30LDQv9GP0YLQvtC5LlxuICovXG5leHBvcnQgZnVuY3Rpb24gc3RlcFZhbGlkYXRvcihzdGVwOiBudW1iZXIsIHByZWNpc2lvbjogbnVtYmVyID0gMyk6IFZhbGlkYXRvckZuIHtcbiAgICByZXR1cm4gKHsgdmFsdWUgfTogQWJzdHJhY3RDb250cm9sPG51bWJlciB8IG51bGw+KTogVmFsaWRhdGlvbkVycm9ycyB8IG51bGwgPT4ge1xuICAgICAgICBpZiAodmFsdWUpIHtcbiAgICAgICAgICAgIHJldHVybiBOdW1iZXIoKHZhbHVlIC8gc3RlcCkudG9GaXhlZChwcmVjaXNpb24pKSAlIDEgIT09IDAgPyB7IGVycm9yOiBg0KfQuNGB0LvQviDQvdC1INC60YDQsNGC0L3QviAke3N0ZXB9YCB9IDogbnVsbDtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBudWxsO1xuICAgIH07XG59XG4iXX0=
|
@@ -341,14 +341,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
341
341
|
* Проверяет кратность значения.
|
342
342
|
*
|
343
343
|
* @param step Шаг кратности.
|
344
|
-
* @param
|
344
|
+
* @param precision Точность, количество чисел после запятой.
|
345
345
|
*/
|
346
|
-
function stepValidator(step,
|
346
|
+
function stepValidator(step, precision = 3) {
|
347
347
|
return ({ value }) => {
|
348
348
|
if (value) {
|
349
|
-
|
350
|
-
const closestMultiple = Math.round(controlValue / step) * step;
|
351
|
-
return Math.abs(controlValue - closestMultiple) > tolerance ? { error: `Число не кратно ${step}` } : null;
|
349
|
+
return Number((value / step).toFixed(precision)) % 1 !== 0 ? { error: `Число не кратно ${step}` } : null;
|
352
350
|
}
|
353
351
|
return null;
|
354
352
|
};
|
@@ -559,7 +557,7 @@ class ScCartItemMobileComponent {
|
|
559
557
|
/**
|
560
558
|
* {@link FormControl} поля ввода количества товара в корзине.
|
561
559
|
*/
|
562
|
-
this.quantityControl = new FormControl(0);
|
560
|
+
this.quantityControl = new FormControl(0, { updateOn: 'blur' });
|
563
561
|
/**
|
564
562
|
* {@link Observable} изменения количества товара в корзине.
|
565
563
|
*/
|
@@ -858,7 +856,7 @@ class ScPriceCardComponent {
|
|
858
856
|
/**
|
859
857
|
* {@link FormControl} поля ввода количества товара в корзине.
|
860
858
|
*/
|
861
|
-
this.quantityControl = new FormControl(null);
|
859
|
+
this.quantityControl = new FormControl(null, { updateOn: 'blur' });
|
862
860
|
/**
|
863
861
|
* {@link Observable} изменения количества товара в корзине.
|
864
862
|
*/
|
@@ -1656,6 +1654,7 @@ const scClientUiIconsName = [
|
|
1656
1654
|
'scIconWordFile',
|
1657
1655
|
'scIconWindowsill',
|
1658
1656
|
'scIconWarning',
|
1657
|
+
'scIconSandwichPanels',
|
1659
1658
|
'scIconWallet',
|
1660
1659
|
'scIconViewTree',
|
1661
1660
|
'scIconViewInline',
|