@snabcentr/client-ui 3.31.2 → 3.32.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 +1 -1
- package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +7 -10
- package/esm2022/models/i-sc-image.mjs +2 -0
- package/esm2022/models/index.mjs +2 -0
- package/esm2022/public-api.mjs +2 -1
- package/esm2022/samples/interfaces/sc-i-preview-sample.mjs +1 -1
- package/esm2022/samples/interfaces/sc-preview-mosquito-config.mjs +1 -1
- package/esm2022/samples/preview-sample/sc-preview-sample.component.mjs +3 -3
- package/esm2022/samples/sc-preview-samples-mosquito/sc-preview-samples-mosquito.component.mjs +3 -3
- package/fesm2022/snabcentr-client-ui.mjs +10 -13
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/models/i-sc-image.d.ts +17 -0
- package/models/index.d.ts +1 -0
- package/package.json +12 -12
- package/public-api.d.ts +1 -0
- package/release_notes.tmp +2 -12
- package/samples/interfaces/sc-i-preview-sample.d.ts +3 -2
- package/samples/interfaces/sc-preview-mosquito-config.d.ts +3 -2
- package/styles/tailwind/tailwind.scss +10 -4
@@ -42,7 +42,7 @@ export declare class ScAddOrEditingCartItemFormComponent implements OnInit {
|
|
42
42
|
/**
|
43
43
|
* Минимальный метраж для товара.
|
44
44
|
*/
|
45
|
-
protected readonly minLength: import("@angular/core").Signal<number
|
45
|
+
protected readonly minLength: import("@angular/core").Signal<number>;
|
46
46
|
/**
|
47
47
|
* Максимальный метраж для товара.
|
48
48
|
*/
|
@@ -61,7 +61,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
61
61
|
/**
|
62
62
|
* Минимальный метраж для товара.
|
63
63
|
*/
|
64
|
-
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));
|
64
|
+
this.minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount) ?? 1);
|
65
65
|
/**
|
66
66
|
* Максимальный метраж для товара.
|
67
67
|
*/
|
@@ -76,8 +76,8 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
76
76
|
this.lengthHint = computed(() => {
|
77
77
|
const min = this.minLength();
|
78
78
|
const max = this.maxLength();
|
79
|
-
// eslint-disable-next-line sonarjs/no-nested-conditional,
|
80
|
-
return min
|
79
|
+
// eslint-disable-next-line sonarjs/no-nested-conditional, @typescript-eslint/no-unnecessary-condition
|
80
|
+
return (min ?? max) ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';
|
81
81
|
});
|
82
82
|
/**
|
83
83
|
* {@link Output} события добавления товара в корзину.
|
@@ -108,13 +108,10 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
108
108
|
});
|
109
109
|
if (this.productIsMeasurable()) {
|
110
110
|
// Добавляем поле ввода длины, если товар измеряемый.
|
111
|
-
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength()
|
111
|
+
this.form.addControl('length', new FormControl(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));
|
112
112
|
}
|
113
113
|
// Устанавливаем количество из корзины или кратности товара.
|
114
|
-
this.form.patchValue(
|
115
|
-
quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),
|
116
|
-
marker: this.cartItem()?.marker ?? '',
|
117
|
-
});
|
114
|
+
this.form.controls.quantity.patchValue(this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()));
|
118
115
|
const step = this.lengthStep();
|
119
116
|
if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {
|
120
117
|
// Если есть конфигуратор длины, рассчитываем начально значение длины.
|
@@ -161,7 +158,7 @@ export class ScAddOrEditingCartItemFormComponent {
|
|
161
158
|
return;
|
162
159
|
}
|
163
160
|
const length = lengthControl.value ?? 0;
|
164
|
-
let newLength = Math.max(this.minLength()
|
161
|
+
let newLength = Math.max(this.minLength(), length + step);
|
165
162
|
// Округляем до ближайшего кратного числа.
|
166
163
|
newLength = Math.round(newLength / step) * step;
|
167
164
|
lengthControl.setValue(Number(newLength.toFixed(2)));
|
@@ -230,4 +227,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
230
227
|
TuiChip,
|
231
228
|
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- \u0424\u043E\u0440\u043C\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F \u0438\u0437\u043C\u0435\u0440\u044F\u0435\u043C\u043E\u0433\u043E \u0442\u043E\u0432\u0430\u0440\u0430. -->\n<form\n *ngIf=\"product\"\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit()\"\n ScNextInputFocus\n class=\"flex flex-col gap-2\"\n>\n @let step = lengthStep();\n\n <!-- \u0414\u043B\u0438\u043D\u0430 \u0442\u043E\u0432\u0430\u0440\u0430 (\u043C\u0435\u0442\u0440\u0430\u0436) -->\n <label\n *ngIf=\"productIsMeasurable()\"\n tuiLabel\n >\n \u041C\u0435\u0442\u0440\u0430\u0436, {{ product().unit }}\n @if (!step || maxLength()) {\n {{ lengthHint() }}\n }\n\n <tui-textfield>\n <input\n tuiInputNumber\n formControlName=\"length\"\n [tuiNumberFormat]=\"{ precision: 2 }\"\n [max]=\"maxLength() || null\"\n [min]=\"minLength() || null\"\n (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n autocomplete=\"length\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"length\"\n [error]=\"[] | tuiFieldError | async\"\n />\n <p\n *ngIf=\"lengthStep()\"\n class=\"tui-form__field-note\"\n >\n \u041C\u0435\u0442\u0440\u0430\u0436 \u0434\u043E\u043B\u0436\u0435\u043D \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u0435\u043D {{ lengthStep() }}\n </p>\n </label>\n\n <!-- \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0442\u043E\u0432\u0430\u0440\u0430 -->\n <label tuiLabel>\n \u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E, \u0448\u0442.\n <tui-textfield>\n <tui-chip\n *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n size=\"s\"\n appearance=\"negative\"\n class=\"font-bold\"\n >\n x {{ step }} {{ product().unit }}\n </tui-chip>\n\n <input\n placeholder=\"\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E\"\n tuiInputNumber\n [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n [min]=\"productMultiplicity()\"\n (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n formControlName=\"quantity\"\n autocomplete=\"quantity\"\n />\n </tui-textfield>\n <p class=\"tui-form__field-note\">\u041A\u043E\u043B\u0438\u0447\u0435\u0441\u0442\u0432\u043E \u0434\u043E\u043B\u0436\u043D\u043E \u0431\u044B\u0442\u044C \u043A\u0440\u0430\u0442\u043D\u043E {{ productMultiplicity() }}</p>\n <tui-error\n formControlName=\"quantity\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430 -->\n <label tuiLabel>\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <tui-input formControlName=\"marker\">\n \u041C\u0430\u0440\u043A\u0438\u0440\u043E\u0432\u043A\u0430\n <input\n tuiTextfieldLegacy\n autocomplete=\"marker\"\n />\n </tui-input>\n <tui-error\n formControlName=\"marker\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <div class=\"flex flex-col items-center\">\n @let cost = totalCost$ | async;\n\n <div *ngIf=\"cost\">\n \u0418\u0442\u043E\u0433\u043E:<span class=\"text-2xl font-bold\">\n {{ cost | tuiAmount | async }}\n {{ product().currency }}\n </span>\n </div>\n\n <!-- \u041A\u043D\u043E\u043F\u043A\u0430 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u044F / \u0440\u0435\u0434\u0430\u043A\u0442\u0438\u0440\u043E\u0432\u0430\u043D\u0438\u044F \u0442\u043E\u0432\u0430\u0440\u0430 \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443 -->\n <button\n tuiButton\n iconStart=\"@tui.check\"\n [disabled]=\"form.invalid\"\n [loading]=\"isLoading()\"\n type=\"submit\"\n class=\"mt-2\"\n >\n {{ cartItem() ? '\u0418\u0437\u043C\u0435\u043D\u0438\u0442\u044C' : '\u0414\u043E\u0431\u0430\u0432\u0438\u0442\u044C \u0432 \u043A\u043E\u0440\u0437\u0438\u043D\u0443' }}\n </button>\n </div>\n</form>\n" }]
|
232
229
|
}] });
|
233
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { isUndefined } from 'lodash-es';\nimport { debounceTime, distinctUntilChanged, filter, map, Observable, pairwise, startWith } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Признак того, что для указанной категории или продукта действует конфигуратор длины.\n     */\n    protected readonly isLengthConfigurator = computed(() => Boolean(this.product().properties?.['isLengthConfigurator']));\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, sonarjs/prefer-nullish-coalescing\n        return min || max ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected totalCost$: Observable<number>;\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n\n        if (this.productIsMeasurable()) {\n            // Добавляем поле ввода длины, если товар измеряемый.\n            this.form.addControl('length', new FormControl<number | null>(this.cartItem()?.length ?? this.minLength() ?? 0.01, [Validators.required, Validators.min(0.01)]));\n        }\n\n        // Устанавливаем количество из корзины или кратности товара.\n        this.form.patchValue({\n            quantity: this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()),\n            marker: this.cartItem()?.marker ?? '',\n        });\n\n        const step = this.lengthStep();\n\n        if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {\n            // Если есть конфигуратор длины, рассчитываем начально значение длины.\n            this.form.controls.length?.patchValue(this.form.controls.quantity.value * step);\n        }\n\n        // Добавляем валидацию шага для количества.\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        // Добавляем валидацию шага для длины.\n        this.form.get('length')?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        // Считаем итоговую стоимость заказа.\n        this.totalCost$ = tuiControlValue<{\n            length?: number;\n            quantity: number;\n        }>(this.form).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            filter(() => this.form.valid),\n            distinctUntilChanged((previous, current) => previous.length === current.length && previous.quantity === current.quantity), // Только при изменении значений\n            startWith(this.form.value as { length?: number; quantity: number }),\n            pairwise(),\n            map(([previous, current]) => {\n                const lengthControl = this.form.get('length');\n\n                // Если нет конфигуратора длины или некорректные данные — возвращаем без изменений.\n                if (!this.isLengthConfigurator() || isUndefined(current.length) || !lengthControl || this.form.invalid || !step) {\n                    return current;\n                }\n\n                // Если изменилось количество — пересчитываем длину.\n                if (previous.quantity !== current.quantity && this.form.controls.quantity.value) {\n                    const newValue = this.form.controls.quantity.value * step;\n                    lengthControl.patchValue(newValue);\n                }\n\n                // Если изменилась длина — пересчитываем количество и возвращаем значение длины.\n                if (previous.length !== current.length && lengthControl.value) {\n                    const newValue = lengthControl.value / step;\n                    this.form.controls.quantity.patchValue(newValue);\n\n                    return { length: step, quantity: newValue };\n                }\n\n                // Возвращаем итоговые значения для расчёта стоимости.\n                return { length: step, quantity: current.quantity };\n            }),\n            // Считаем стоимость: цена * длина * количество.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * (length ?? 1) * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        );\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        const lengthControl = this.form.get('length');\n\n        if (!this.productIsMeasurable() || !lengthControl) {\n            return;\n        }\n\n        const length = lengthControl.value ?? 0;\n        let newLength = Math.max(this.minLength() ?? step, length + step);\n        // Округляем до ближайшего кратного числа.\n        newLength = Math.round(newLength / step) * step;\n        lengthControl.setValue(Number(newLength.toFixed(2)));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        // Удаляем null-поля из значения формы.\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        // Если это добавление нового товара — добавляем productId.\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        // Если товар измеряемый и есть конфигуратор длины — устанавливаем длину по шагу (количество уже будет вычислено по формуле Метраж/min-count)\n        if (this.productIsMeasurable() && this.isLengthConfigurator() && step) {\n            value.length = step;\n        }\n\n        // Помечаем все поля формы как \"touch\" и запускаем валидацию для отображения ошибок с сервера.\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        // В зависимости от режима — редактируем или добавляем товар.\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    @let step = lengthStep();\n\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }}\n        @if (!step || maxLength()) {\n            {{ lengthHint() }}\n        }\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            <tui-chip\n                *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost$ | async;\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount | async }}\n                {{ product().currency }}\n            </span>\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|
230
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-or-editing-cart-item-form.component.js","sourceRoot":"","sources":["../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.ts","../../../../../../projects/client-ui/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable sonarjs/no-nested-template-literals,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, computed, inject, input, OnInit, output } from '@angular/core';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport {\n    ScCartItem,\n    ScConvertersService,\n    ScINewCartItemBase,\n    ScINewCartItemMeasured,\n    ScProduct,\n    ScUnitsHelper,\n    ScUserMetrikaGoalsEnum,\n    ScUserMetrikaService,\n} from '@snabcentr/client-core';\nimport { TuiAmountPipe } from '@taiga-ui/addon-commerce';\nimport { tuiControlValue, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiButton, TuiError, TuiLabel, TuiNumberFormat, TuiTextfield } from '@taiga-ui/core';\nimport { TuiAppearance, TuiWithAppearance } from '@taiga-ui/core/directives/appearance';\nimport { TuiIcons, TuiWithIcons } from '@taiga-ui/core/directives/icons';\nimport { TuiButtonLoading, TuiChip, TuiFieldErrorPipe, TuiInputNumber } from '@taiga-ui/kit';\nimport { TuiInputModule } from '@taiga-ui/legacy';\nimport { isUndefined } from 'lodash-es';\nimport { debounceTime, distinctUntilChanged, filter, map, Observable, pairwise, startWith } from 'rxjs';\n\nimport { ScNextInputFocusModule } from '../../../directives/next-input-focus/sc-next-input-focus.module';\nimport { ScSelectOnFocusinDirective } from '../../../directives/select-on-focusin/sc-select-on-focusin.directive';\nimport { stepValidator } from '../../../validators/step-validator';\n\n/**\n * Компонент добавления / изменения товара в корзине.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-add-or-editing-cart-item-form',\n    templateUrl: './sc-add-or-editing-cart-item-form.component.html',\n    imports: [\n        NgIf,\n        FormsModule,\n        ScNextInputFocusModule,\n        ReactiveFormsModule,\n        TuiLabel,\n        ScSelectOnFocusinDirective,\n        TuiNumberFormat,\n        TuiError,\n        TuiInputModule,\n        TuiAppearance,\n        TuiWithAppearance,\n        TuiIcons,\n        TuiWithIcons,\n        TuiButton,\n        TuiButtonLoading,\n        AsyncPipe,\n        TuiFieldErrorPipe,\n        TuiInputNumber,\n        TuiNumberFormat,\n        TuiTextfield,\n        TuiAmountPipe,\n        TuiChip,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddOrEditingCartItemFormComponent implements OnInit {\n    /**\n     * Группа полей добавления / изменения товара в корзине.\n     */\n    public readonly form = new FormGroup<{\n        length?: FormControl<number | null>;\n        quantity: FormControl<number | null>;\n        marker: FormControl<string | null>;\n    }>({\n        quantity: new FormControl<number | null>(null, Validators.required),\n        marker: new FormControl<string | null>(''),\n    });\n\n    /**\n     * Данные о товаре.\n     */\n    public readonly product = input.required<ScProduct>();\n\n    /**\n     * Признак того, что для указанной категории или продукта действует конфигуратор длины.\n     */\n    protected readonly isLengthConfigurator = computed(() => Boolean(this.product().properties?.['isLengthConfigurator']));\n\n    /**\n     * Данные о товаре в корзине.\n     */\n    public readonly cartItem = input<ScCartItem | undefined>();\n\n    /**\n     * Признак загрузки данных.\n     */\n    public readonly isLoading = input(false);\n\n    /**\n     * Признак возможности продажи товара на метраж.\n     */\n    protected readonly productIsMeasurable = computed(() => this.unitsHelper.productIsMeasurable(this.product()));\n\n    /**\n     * Кратность количества для товара.\n     */\n    protected readonly productMultiplicity = computed(() => this.unitsHelper.productMultiplicity(this.product()));\n\n    /**\n     * Минимальный метраж для товара.\n     */\n    protected readonly minLength = computed(() => this.product().properties?.minLength ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount) ?? 1);\n\n    /**\n     * Максимальный метраж для товара.\n     */\n    protected readonly maxLength = computed(() => this.product().properties?.maxLength);\n\n    /**\n     * Шаг изменения метража.\n     */\n    protected readonly lengthStep = computed(() => this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : this.product().minCount));\n\n    /**\n     * Подсказка по минимальному и максимальному метражу товара.\n     */\n    protected readonly lengthHint = computed(() => {\n        const min = this.minLength();\n        const max = this.maxLength();\n\n        // eslint-disable-next-line sonarjs/no-nested-conditional, @typescript-eslint/no-unnecessary-condition\n        return (min ?? max) ? `(${min ? `от ${min}` : ''}${min && max ? ' ' : ''}${max ? `до ${max}` : ''} ${this.product().unit})` : '';\n    });\n\n    /**\n     * Итоговая стоимость заказа.\n     */\n    protected totalCost$: Observable<number>;\n\n    /**\n     * {@link Output} события добавления товара в корзину.\n     */\n    public readonly addToCart = output<ScINewCartItemBase>();\n\n    /**\n     * {@link Output} события редактирования товара в корзине.\n     */\n    public readonly editCartItem = output<Omit<ScINewCartItemBase, 'productId'>>();\n\n    /**\n     * Объект-помощник для работы со значениями единиц измерения товара.\n     */\n    private readonly unitsHelper: ScUnitsHelper = inject(ScUnitsHelper);\n\n    /**\n     * Сервис для сбора метрик о действиях пользователей.\n     */\n    private readonly userMetrikaService: ScUserMetrikaService = inject(ScUserMetrikaService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.cartItemAddShow,\n            params: { product_id: this.product().id },\n        });\n\n        if (this.productIsMeasurable()) {\n            // Добавляем поле ввода длины, если товар измеряемый.\n            this.form.addControl('length', new FormControl<number | null>(this.cartItem()?.length ?? this.minLength(), [Validators.required, Validators.min(0.01)]));\n        }\n\n        // Устанавливаем количество из корзины или кратности товара.\n        this.form.controls.quantity.patchValue(this.cartItem()?.quantity ?? this.unitsHelper.productMultiplicity(this.product()));\n        const step = this.lengthStep();\n\n        if (this.isLengthConfigurator() && this.lengthStep() && this.form.controls.quantity.value && step) {\n            // Если есть конфигуратор длины, рассчитываем начально значение длины.\n            this.form.controls.length?.patchValue(this.form.controls.quantity.value * step);\n        }\n\n        // Добавляем валидацию шага для количества.\n        this.form.controls.quantity.addValidators(stepValidator(this.unitsHelper.productStepForValidator(this.product())));\n        // Добавляем валидацию шага для длины.\n        this.form.get('length')?.addValidators(stepValidator(this.product().properties?.lengthStep ?? (this.product().ignoreMinCountCheck ? 0 : (this.product().minCount ?? 0))));\n\n        // Считаем итоговую стоимость заказа.\n        this.totalCost$ = tuiControlValue<{\n            length?: number;\n            quantity: number;\n        }>(this.form).pipe(\n            debounceTime(0), // Исправляем ошибку NG0950.\n            filter(() => this.form.valid),\n            distinctUntilChanged((previous, current) => previous.length === current.length && previous.quantity === current.quantity), // Только при изменении значений\n            startWith(this.form.value as { length?: number; quantity: number }),\n            pairwise(),\n            map(([previous, current]) => {\n                const lengthControl = this.form.get('length');\n\n                // Если нет конфигуратора длины или некорректные данные — возвращаем без изменений.\n                if (!this.isLengthConfigurator() || isUndefined(current.length) || !lengthControl || this.form.invalid || !step) {\n                    return current;\n                }\n\n                // Если изменилось количество — пересчитываем длину.\n                if (previous.quantity !== current.quantity && this.form.controls.quantity.value) {\n                    const newValue = this.form.controls.quantity.value * step;\n                    lengthControl.patchValue(newValue);\n                }\n\n                // Если изменилась длина — пересчитываем количество и возвращаем значение длины.\n                if (previous.length !== current.length && lengthControl.value) {\n                    const newValue = lengthControl.value / step;\n                    this.form.controls.quantity.patchValue(newValue);\n\n                    return { length: step, quantity: newValue };\n                }\n\n                // Возвращаем итоговые значения для расчёта стоимости.\n                return { length: step, quantity: current.quantity };\n            }),\n            // Считаем стоимость: цена * длина * количество.\n            map(({ length, quantity }) => (this.product().costRub ?? 0) * (length ?? 1) * quantity),\n            map((sum) => Math.round(sum * 100) / 100)\n        );\n    }\n\n    /**\n     * Обработчик события шага метража.\n     *\n     * @param step Шаг.\n     */\n    protected onStepLength(step: number): void {\n        const lengthControl = this.form.get('length');\n\n        if (!this.productIsMeasurable() || !lengthControl) {\n            return;\n        }\n\n        const length = lengthControl.value ?? 0;\n        let newLength = Math.max(this.minLength(), length + step);\n        // Округляем до ближайшего кратного числа.\n        newLength = Math.round(newLength / step) * step;\n        lengthControl.setValue(Number(newLength.toFixed(2)));\n    }\n\n    /**\n     * Обработчик события шага количества.\n     *\n     * @param step Шаг.\n     */\n    protected onStepQuantity(step: number): void {\n        const quantity = this.form.controls.quantity.value ?? 0;\n\n        this.form.controls.quantity.setValue(Math.max(this.productMultiplicity(), quantity - (quantity % step) + step));\n    }\n\n    /**\n     * Обработчик события отправки формы.\n     */\n    protected onSubmit(): void {\n        // Удаляем null-поля из значения формы.\n        const value = this.convertersService.removeNull<ScINewCartItemMeasured>(this.form.value as ScINewCartItemMeasured);\n\n        // Если это добавление нового товара — добавляем productId.\n        if (!this.cartItem()) {\n            value.productId = this.product().id;\n        }\n\n        const step = this.lengthStep();\n\n        // Если товар измеряемый и есть конфигуратор длины — устанавливаем длину по шагу (количество уже будет вычислено по формуле Метраж/min-count)\n        if (this.productIsMeasurable() && this.isLengthConfigurator() && step) {\n            value.length = step;\n        }\n\n        // Помечаем все поля формы как \"touch\" и запускаем валидацию для отображения ошибок с сервера.\n        tuiMarkControlAsTouchedAndValidate(this.form);\n\n        // В зависимости от режима — редактируем или добавляем товар.\n        if (this.cartItem()) {\n            this.editCartItem.emit(value);\n        } else {\n            this.addToCart.emit(value);\n        }\n    }\n}\n","<!-- Форма добавления измеряемого товара. -->\n<form\n    *ngIf=\"product\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit()\"\n    ScNextInputFocus\n    class=\"flex flex-col gap-2\"\n>\n    @let step = lengthStep();\n\n    <!-- Длина товара (метраж) -->\n    <label\n        *ngIf=\"productIsMeasurable()\"\n        tuiLabel\n    >\n        Метраж, {{ product().unit }}\n        @if (!step || maxLength()) {\n            {{ lengthHint() }}\n        }\n\n        <tui-textfield>\n            <input\n                tuiInputNumber\n                formControlName=\"length\"\n                [tuiNumberFormat]=\"{ precision: 2 }\"\n                [max]=\"maxLength() || null\"\n                [min]=\"minLength() || null\"\n                (keydown.arrowDown)=\"onStepLength(-(lengthStep() ?? 0.01))\"\n                (keydown.arrowUp)=\"onStepLength(lengthStep() ?? 0.01)\"\n                autocomplete=\"length\"\n            />\n        </tui-textfield>\n        <tui-error\n            formControlName=\"length\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n        <p\n            *ngIf=\"lengthStep()\"\n            class=\"tui-form__field-note\"\n        >\n            Метраж должен быть кратен {{ lengthStep() }}\n        </p>\n    </label>\n\n    <!-- Количество товара -->\n    <label tuiLabel>\n        Количество, шт.\n        <tui-textfield>\n            <tui-chip\n                *ngIf=\"isLengthConfigurator() && productMultiplicity() && step\"\n                size=\"s\"\n                appearance=\"negative\"\n                class=\"font-bold\"\n            >\n                x {{ step }} {{ product().unit }}\n            </tui-chip>\n\n            <input\n                placeholder=\"Количество\"\n                tuiInputNumber\n                [tuiNumberFormat]=\"{ decimalMode: 'not-zero' }\"\n                [min]=\"productMultiplicity()\"\n                (keydown.arrowDown)=\"onStepQuantity(-productMultiplicity())\"\n                (keydown.arrowUp)=\"onStepQuantity(productMultiplicity())\"\n                formControlName=\"quantity\"\n                autocomplete=\"quantity\"\n            />\n        </tui-textfield>\n        <p class=\"tui-form__field-note\">Количество должно быть кратно {{ productMultiplicity() }}</p>\n        <tui-error\n            formControlName=\"quantity\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Маркировка -->\n    <label tuiLabel>\n        Маркировка\n        <tui-input formControlName=\"marker\">\n            Маркировка\n            <input\n                tuiTextfieldLegacy\n                autocomplete=\"marker\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"marker\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Кнопка добавления / редактирования товара в корзину -->\n    <div class=\"flex flex-col items-center\">\n        @let cost = totalCost$ | async;\n\n        <div *ngIf=\"cost\">\n            Итого:<span class=\"text-2xl font-bold\">\n                {{ cost | tuiAmount | async }}\n                {{ product().currency }}\n            </span>\n        </div>\n\n        <!-- Кнопка добавления / редактирования товара в корзину -->\n        <button\n            tuiButton\n            iconStart=\"@tui.check\"\n            [disabled]=\"form.invalid\"\n            [loading]=\"isLoading()\"\n            type=\"submit\"\n            class=\"mt-2\"\n        >\n            {{ cartItem() ? 'Изменить' : 'Добавить в корзину' }}\n        </button>\n    </div>\n</form>\n"]}
|
@@ -0,0 +1,2 @@
|
|
1
|
+
export {};
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaS1zYy1pbWFnZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9tb2RlbHMvaS1zYy1pbWFnZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiDQnNC+0LTQtdC70Ywg0LTQsNC90L3Ri9GFINC40LfQvtCx0YDQsNC20LXQvdC40Y8uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSVNjSW1hZ2Uge1xuICAgIC8qKlxuICAgICAqINCh0YHRi9C70LrQsCDQvdCwINC40LfQvtCx0YDQsNC20LXQvdC40LUg0LIg0YTQvtGA0LzQsNGC0LUg0L/QviDRg9C80L7Qu9GH0LDQvdC40Y4uXG4gICAgICovXG4gICAgZGVmYXVsdEZvcm1hdDogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICog0KHRgdGL0LvQutCwINC90LAg0LjQt9C+0LHRgNCw0LbQtdC90LjQtSDQsiDRhNC+0YDQvNCw0YLQtSB3ZWJwLlxuICAgICAqL1xuICAgIHdlYnA6IHN0cmluZztcblxuICAgIC8qKlxuICAgICAqINCa0YDQsNGC0LrQvtC1INC+0L/QuNGB0LDQvdC40LUg0LjQt9C+0LHRgNCw0LbQtdC90LjRjywg0LXRgdC70Lgg0LLQvtC30L3QuNC60LvQsCDQv9GA0L7QsdC70LXQvNCwINC30LDQs9GA0YPQt9C60LguXG4gICAgICovXG4gICAgYWx0Pzogc3RyaW5nO1xufVxuIl19
|
@@ -0,0 +1,2 @@
|
|
1
|
+
export * from './i-sc-image';
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvbW9kZWxzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9pLXNjLWltYWdlJztcbiJdfQ==
|
package/esm2022/public-api.mjs
CHANGED
@@ -1,6 +1,7 @@
|
|
1
1
|
/*
|
2
2
|
* Public API Surface of ui
|
3
3
|
*/
|
4
|
+
export * from './models';
|
4
5
|
export * from './services';
|
5
6
|
export * from './auth';
|
6
7
|
export * from './accordion';
|
@@ -31,4 +32,4 @@ export * from './user';
|
|
31
32
|
export * from './validators';
|
32
33
|
export * from './verification';
|
33
34
|
export * from './profile';
|
34
|
-
//# sourceMappingURL=data:application/json;base64,
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxvQkFBb0IsQ0FBQztBQUNuQyxjQUFjLFdBQVcsQ0FBQztBQUMxQixjQUFjLGNBQWMsQ0FBQztBQUM3QixjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsVUFBVSxDQUFDO0FBQ3pCLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsUUFBUSxDQUFDO0FBQ3ZCLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsV0FBVyxDQUFDO0FBQzFCLGNBQWMsYUFBYSxDQUFDO0FBQzVCLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsVUFBVSxDQUFDO0FBQ3pCLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLFdBQVcsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgdWlcbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL21vZGVscyc7XG5leHBvcnQgKiBmcm9tICcuL3NlcnZpY2VzJztcbmV4cG9ydCAqIGZyb20gJy4vYXV0aCc7XG5leHBvcnQgKiBmcm9tICcuL2FjY29yZGlvbic7XG5leHBvcnQgKiBmcm9tICcuL2Jhbm5lcic7XG5leHBvcnQgKiBmcm9tICcuL2NhcnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jYXRhbG9nJztcbmV4cG9ydCAqIGZyb20gJy4vY29udHJhZ2VudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9jb250YWN0cyc7XG5leHBvcnQgKiBmcm9tICcuL2RlbGl2ZXJ5LWFkZHJlc3MnO1xuZXhwb3J0ICogZnJvbSAnLi9kaWFsb2dzJztcbmV4cG9ydCAqIGZyb20gJy4vZGlyZWN0aXZlcyc7XG5leHBvcnQgKiBmcm9tICcuL2Vycm9yLWhhbmRsZXInO1xuZXhwb3J0ICogZnJvbSAnLi9maWxlcyc7XG5leHBvcnQgKiBmcm9tICcuL2Zvcm0tZmllbGRzJztcbmV4cG9ydCAqIGZyb20gJy4vaWNvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9sb2FkZXInO1xuZXhwb3J0ICogZnJvbSAnLi9icmFuZHMtbGlzdCc7XG5leHBvcnQgKiBmcm9tICcuL25ld3MnO1xuZXhwb3J0ICogZnJvbSAnLi9vcmRlcic7XG5leHBvcnQgKiBmcm9tICcuL3BhZ2VzJztcbmV4cG9ydCAqIGZyb20gJy4vcGlwZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9zYW1wbGVzJztcbmV4cG9ydCAqIGZyb20gJy4vcHJvdmlkZXJzJztcbmV4cG9ydCAqIGZyb20gJy4vZmVlZGJhY2snO1xuZXhwb3J0ICogZnJvbSAnLi9xcmNvZGUnO1xuZXhwb3J0ICogZnJvbSAnLi9zaGFyZS1idXR0b24nO1xuZXhwb3J0ICogZnJvbSAnLi91c2VyJztcbmV4cG9ydCAqIGZyb20gJy4vdmFsaWRhdG9ycyc7XG5leHBvcnQgKiBmcm9tICcuL3ZlcmlmaWNhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL3Byb2ZpbGUnO1xuIl19
|
@@ -1,2 +1,2 @@
|
|
1
1
|
export {};
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtaS1wcmV2aWV3LXNhbXBsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9zYW1wbGVzL2ludGVyZmFjZXMvc2MtaS1wcmV2aWV3LXNhbXBsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSVNjSW1hZ2UgfSBmcm9tICcuLi8uLi9tb2RlbHMnO1xuXG4vKipcbiAqINCc0L7QtNC10LvRjCDQtNCw0L3QvdGL0YUg0L7QsdGA0LDQt9GG0L7QsiDRgtC+0LLQsNGA0LAuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2NJUHJldmlld1NhbXBsZSB7XG4gICAgLyoqXG4gICAgICog0JfQsNCz0L7Qu9C+0LLQvtC6LlxuICAgICAqL1xuICAgIHRpdGxlOiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiDQntC/0LjRgdCw0L3QuNC1ICjQv9C+0LTQt9Cw0LPQvtC70L7QstC+0LopLlxuICAgICAqL1xuICAgIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiDQrdC60LfQtdC80L/Qu9GP0YDRiyDQvtCx0YDQsNC30YbQvtCyLlxuICAgICAqL1xuICAgIGl0ZW1zPzogQXJyYXk8e1xuICAgICAgICAvKipcbiAgICAgICAgICog0KHRgdGL0LvQutCwINC90LAg0LjQt9C+0LHRgNCw0LbQtdC90LjQtSDQutC90L7Qv9C60Lgg0L/QtdGA0LXQutC70Y7Rh9Cw0YLQtdC70Y8uXG4gICAgICAgICAqL1xuICAgICAgICBidXR0b25JbWFnZVVybDogSVNjSW1hZ2U7XG5cbiAgICAgICAgLyoqXG4gICAgICAgICAqINCh0YHRi9C70LrQsCDQvdCwINC40LfQvtCx0YDQsNC20LXQvdC40LUg0LIg0LPQsNC70LXRgNC10LUsINGB0L7QvtGC0LLQtdGC0YHRgtCy0YPRjtGJ0LXQtSDQutC90L7Qv9C60LUuXG4gICAgICAgICAqL1xuICAgICAgICBjYXJvdXNlbEl0ZW1JbWFnZTogSVNjSW1hZ2U7XG5cbiAgICAgICAgLyoqXG4gICAgICAgICAqINCd0LDQuNC80LXQvdC+0LLQsNC90LjQtSDRhtCy0LXRgtCwINC+0LHRgNCw0LfRhtCwLlxuICAgICAgICAgKi9cbiAgICAgICAgY29sb3I6IHN0cmluZztcblxuICAgICAgICAvKipcbiAgICAgICAgICog0K3Qu9C10LzQtdC90YLRiyDQutC90L7Qv9C+0Log0LLQt9Cw0LjQvNC+0LTQtdC50YHRgtCy0LjRjy5cbiAgICAgICAgICovXG4gICAgICAgIGFjdGlvbnM/OiBBcnJheTx7XG4gICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAqINCX0L3QsNGH0L7QuiDRjdC70LXQvNC10L3RgtCwINCy0LfQsNC40LzQvtC00LXQudGB0YLQstC40Y8uXG4gICAgICAgICAgICAgKi9cbiAgICAgICAgICAgIGljb246IHN0cmluZztcblxuICAgICAgICAgICAgLyoqXG4gICAgICAgICAgICAgKiDQn9C+0LTQv9C40YHRjCDRjdC70LXQvNC10L3RgtCwINCy0LfQsNC40LzQvtC00LXQudGB0YLQstC40Y8uXG4gICAgICAgICAgICAgKi9cbiAgICAgICAgICAgIGxhYmVsOiBzdHJpbmc7XG5cbiAgICAgICAgICAgIC8qKlxuICAgICAgICAgICAgICog0KHRgdGL0LvQutCwLlxuICAgICAgICAgICAgICovXG4gICAgICAgICAgICBocmVmPzogc3RyaW5nO1xuXG4gICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAqINCe0LHRgNCw0LHQvtGC0YfQuNC6INGN0LvQtdC80LXQvdGC0LAg0LLQt9Cw0LjQvNC+0LTQtdC50YHRgtCy0LjRjy5cbiAgICAgICAgICAgICAqL1xuICAgICAgICAgICAgaGFuZGxlcj86ICguLi5hdHI6IHVua25vd25bXSkgPT4gdW5rbm93bjtcbiAgICAgICAgfT47XG4gICAgfT47XG5cbiAgICAvKipcbiAgICAgKiDQodGB0YvQu9C60LAg0L3QsCDQuNC30L7QsdGA0LDQttC10L3QuNC1INGC0L7QstCw0YDQsCDQsiDQvtC/0LjRgdCw0L3QuNC4LlxuICAgICAqL1xuICAgIGRlc2NyaXB0aW9uSW1hZ2VVcmw/OiBzdHJpbmc7XG59XG4iXX0=
|
@@ -1,2 +1,2 @@
|
|
1
1
|
export {};
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcHJldmlldy1tb3NxdWl0by1jb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvc2FtcGxlcy9pbnRlcmZhY2VzL3NjLXByZXZpZXctbW9zcXVpdG8tY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJU2NJbWFnZSB9IGZyb20gJy4uLy4uL21vZGVscyc7XG5cbi8qKlxuICog0JrQvtC90YTQuNCz0YPRgNCw0YbQuNGPINC00LvRjyDQvtGC0L7QsdGA0LDQttC10L3QuNGPINC+0LHRgNCw0LfRhtCwINC80L7RgdC60LjRgtC90L7QuSDRgdC10YLQutC4LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFNjUHJldmlld01vc3F1aXRvQ29uZmlnIHtcbiAgICAvKipcbiAgICAgKiDQotC40L/RiyDQvNC+0YHQutC40YLQvdGL0YUg0YHQtdGC0L7Qui5cbiAgICAgKi9cbiAgICBuZXRUeXBlOiBzdHJpbmdbXTtcblxuICAgIC8qKlxuICAgICAqINCm0LLQtdGC0LAg0L/RgNC+0YTQuNC70Y8uXG4gICAgICovXG4gICAgY29sb3I6IHN0cmluZ1tdO1xuXG4gICAgLyoqXG4gICAgICog0KLQuNC/0Ysg0LrRgNC10L/Qu9C10L3QuNC5INC80L7RgdC60LjRgtC90YvRhSDRgdC10YLQvtC6INC6INC+0LrQvdGDLlxuICAgICAqL1xuICAgIG1vdW50VHlwZTogc3RyaW5nW107XG5cbiAgICAvKipcbiAgICAgKiDQodC/0YDQsNCy0L7Rh9C90LjQuiDQstGB0LXRhSDQstC+0LfQvNC+0LbQvdGL0YUg0LfQvdCw0YfQtdC90LjQuSDQvtC/0LjRgdCw0L3QuNC5INC60L7QvdGE0LjQs9GD0YDQsNGG0LjQuS5cbiAgICAgKi9cbiAgICBkZXNjcmlwdGlvbnM6IFJlY29yZDxzdHJpbmcsIFJlY29yZDxzdHJpbmcsIHN0cmluZz4+O1xuXG4gICAgLyoqXG4gICAgICog0J/QvtC00LfQsNCz0L7Qu9C+0LLQvtC6INC00LvRjyDQvtC/0LjRgdCw0L3QuNGPINGB0LXRgtC60LguXG4gICAgICovXG4gICAgc3VidGl0bGVzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuXG4gICAgLyoqXG4gICAgICog0JjQutC+0L3QutC4INC00LvRjyDQvtGC0L7QsdGA0LDQttC10L3QuNGPINC90LAg0LrQvdC+0L/QutCw0YUuXG4gICAgICovXG4gICAgaWNvbnM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG5cbiAgICAvKipcbiAgICAgKiDQnNC40L3QuNC80LDQu9GM0L3QsNGPINCy0YvRgdC+0YLQsCDRgdCy0LXRgtC+0LLQvtCz0L4g0L/RgNC+0ZHQvNCwINCyINC80LjQu9C70LjQvNC10YLRgNCw0YUuXG4gICAgICovXG4gICAgbWluSGVpZ2h0OiBudW1iZXI7XG5cbiAgICAvKipcbiAgICAgKiDQodC/0LjRgdC+0Log0LzQsNC60YHQuNC80LDQu9GM0L3QviDQstC+0LfQvNC+0LbQvdGL0YUg0LLRi9GB0L7RgiDRgdCy0LXRgtC+0LLQvtCz0L4g0L/RgNC+0ZHQvNCwINCyINC80LjQu9C70LjQvNC10YLRgNCw0YUg0LIg0LfQsNCy0LjRgdC40LzQvtGB0YLQuCDQvtGCINGC0LjQv9CwINGB0LXRgtC60LguXG4gICAgICovXG4gICAgbWF4SGVpZ2h0czogUmVjb3JkPHN0cmluZywgbnVtYmVyPjtcblxuICAgIC8qKlxuICAgICAqINCc0LjQvdC40LzQsNC70YzQvdCw0Y8g0YjQuNGA0LjQvdCwINGB0LLQtdGC0L7QstC+0LPQviDQv9GA0L7RkdC80LAg0LIg0LzQuNC70LvQuNC80LXRgtGA0LDRhS5cbiAgICAgKi9cbiAgICBtaW5XaWR0aDogbnVtYmVyO1xuXG4gICAgLyoqXG4gICAgICog0KHQv9C40YHQvtC6INC80LDQutGB0LjQvNCw0LvRjNC90L4g0LLQvtC30LzQvtC20L3Ri9GFINGI0LjRgNC40L0g0YHQstC10YLQvtCy0L7Qs9C+INC/0YDQvtGR0LzQsCDQsiDQvNC40LvQu9C40LzQtdGC0YDQsNGFINCyINC30LDQstC40YHQuNC80L7RgdGC0Lgg0L7RgiDRgtC40L/QsCDRgdC10YLQutC4LlxuICAgICAqL1xuICAgIG1heFdpZHRoczogUmVjb3JkPHN0cmluZywgbnVtYmVyPjtcblxuICAgIC8qKlxuICAgICAqINCi0LXQs9C4LCDQutC+0YLQvtGA0YvQtSDQsdGD0LTRg9GCINC+0YLQvtCx0YDQsNC20LDRgtGM0YHRjyDQvdCwINC60L3QvtC/0LrQsNGFLlxuICAgICAqL1xuICAgIHRhZ3M6IFJlY29yZDxzdHJpbmcsIHN0cmluZ1tdPjtcblxuICAgIC8qKlxuICAgICAqINCY0LfQvtCx0YDQsNC20LXQvdC40Y8g0LTQu9GPINC+0YLQvtCx0YDQsNC20LXQvdC40Y8g0LIg0LrQsNGA0YPRgdC10LvQuCDQv9C+INGG0LLQtdGC0YMg0L/RgNC+0YTQuNC70Y8uXG4gICAgICovXG4gICAgaW1hZ2VQcmV2aWV3c0J5Q29sb3I6IFJlY29yZDxzdHJpbmcsIElTY0ltYWdlW10+O1xuXG4gICAgLyoqXG4gICAgICog0JjQt9C+0LHRgNCw0LbQtdC90LjRjyDRhtCy0LXRgtC+0LIg0L/RgNC+0YTQuNC70Y8uXG4gICAgICovXG4gICAgY29sb3JzSWNvbnM6IFJlY29yZDxzdHJpbmcsIElTY0ltYWdlPjtcblxuICAgIC8qKlxuICAgICAqINCh0YHRi9C70LrQsCDQvdCwINC60L7QvdGE0LjQs9GD0YDQsNGC0L7RgCDQvNC+0YHQutC40YLQvdC+0Lkg0YHQtdGC0LrQuC5cbiAgICAgKi9cbiAgICBtb3NxdWl0b0NvbmZpZ3VyYXRvckxpbms6IHN0cmluZztcbn1cbiJdfQ==
|
@@ -33,10 +33,10 @@ export class ScPreviewSampleComponent {
|
|
33
33
|
this.isReverse = input(false);
|
34
34
|
}
|
35
35
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScPreviewSampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
36
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: ScPreviewSampleComponent, selector: "sc-preview-sample", inputs: { index: { classPropertyName: "index", publicName: "index", isSignal: true, isRequired: false, transformFunction: null }, duration: { classPropertyName: "duration", publicName: "duration", isSignal: true, isRequired: false, transformFunction: null }, sample: { classPropertyName: "sample", publicName: "sample", isSignal: true, isRequired: true, transformFunction: null }, isShowCarouselSwitches: { classPropertyName: "isShowCarouselSwitches", publicName: "isShowCarouselSwitches", isSignal: true, isRequired: false, transformFunction: null }, isReverse: { classPropertyName: "isReverse", publicName: "isReverse", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { index: "indexChange" }, host: { attributes: { "ngSkipHydration": "true" } }, ngImport: i0, template: "<div\n *ngIf=\"sample() as sample\"\n class=\"mb-4 flex flex-col-reverse gap-4 md:mb-0 md:flex-row md:gap-8\"\n [class.md:!flex-row-reverse]=\"isReverse()\"\n>\n @let isShownSwitches = sample.items && sample.items.length > 1;\n\n <div class=\"flex max-w-96 flex-1 flex-col justify-center self-center md:max-w-none md:gap-4\">\n <div class=\"flex w-full flex-col gap-3 md:gap-4 md:px-8\">\n <div class=\"text-body-l-bold\">{{ sample.title }}</div>\n <div\n *ngIf=\"isShownSwitches\"\n class=\"flex gap-4\"\n >\n <tui-avatar\n [
|
36
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: ScPreviewSampleComponent, selector: "sc-preview-sample", inputs: { index: { classPropertyName: "index", publicName: "index", isSignal: true, isRequired: false, transformFunction: null }, duration: { classPropertyName: "duration", publicName: "duration", isSignal: true, isRequired: false, transformFunction: null }, sample: { classPropertyName: "sample", publicName: "sample", isSignal: true, isRequired: true, transformFunction: null }, isShowCarouselSwitches: { classPropertyName: "isShowCarouselSwitches", publicName: "isShowCarouselSwitches", isSignal: true, isRequired: false, transformFunction: null }, isReverse: { classPropertyName: "isReverse", publicName: "isReverse", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { index: "indexChange" }, host: { attributes: { "ngSkipHydration": "true" } }, ngImport: i0, template: "<div\n *ngIf=\"sample() as sample\"\n class=\"mb-4 flex flex-col-reverse gap-4 md:mb-0 md:flex-row md:gap-8\"\n [class.md:!flex-row-reverse]=\"isReverse()\"\n>\n @let isShownSwitches = sample.items && sample.items.length > 1;\n\n <div class=\"flex max-w-96 flex-1 flex-col justify-center self-center md:max-w-none md:gap-4\">\n <div class=\"flex w-full flex-col gap-3 md:gap-4 md:px-8\">\n <div class=\"text-body-l-bold\">{{ sample.title }}</div>\n <div\n *ngIf=\"isShownSwitches\"\n class=\"flex gap-4\"\n >\n <tui-avatar\n [round]=\"false\"\n *ngFor=\"let item of sample.items; let itemIndex = index\"\n (tuiHoveredChange)=\"index.set(itemIndex)\"\n [class.active]=\"index() === itemIndex\"\n >\n <picture>\n <source\n type=\"image/webp\"\n [srcset]=\"item.buttonImageUrl.webp\"\n />\n <img\n [src]=\"item.buttonImageUrl.defaultFormat\"\n [alt]=\"item.buttonImageUrl.alt ?? '\u0418\u0437\u043E\u0431\u0440\u0430\u0436\u0435\u043D\u0438\u0435 \u043A\u043D\u043E\u043F\u043A\u0438 \u043F\u0435\u0440\u0435\u043A\u043B\u044E\u0447\u0430\u0442\u0435\u043B\u044F ' + sample.title\"\n />\n </picture>\n </tui-avatar>\n </div>\n <div\n *ngIf=\"sample.items as items\"\n class=\"text-body-m-bold\"\n >\n {{ items[index()].color }}\n </div>\n <div>{{ sample.description }}</div>\n <img\n *ngIf=\"sample.descriptionImageUrl\"\n [src]=\"sample.descriptionImageUrl\"\n alt=\"\u041A\u043E\u043D\u0441\u0442\u0440\u0443\u043A\u0446\u0438\u044F \u043E\u0431\u0440\u0430\u0437\u0446\u0430 {{ sample.title }}\"\n class=\"w-fit\"\n />\n <div\n *ngIf=\"sample.items && sample.items[index()].actions?.length\"\n class=\"flex flex-wrap gap-3 md:gap-4\"\n >\n <a\n *ngFor=\"let action of sample.items[index()].actions\"\n tuiButton\n [attr.href]=\"action.href\"\n (click)=\"action.handler ? $event.preventDefault() : {}; action.handler?.()\"\n [iconStart]=\"action.icon\"\n appearance=\"secondary\"\n >\n {{ action.label }}\n </a>\n </div>\n </div>\n </div>\n\n <div class=\"relative h-[19em] flex-1 overflow-hidden rounded-xl shadow-tui-shadow md:h-96 lg:h-[28rem]\">\n <button\n *ngIf=\"isShownSwitches && isShowCarouselSwitches()\"\n tuiIconButton\n (click)=\"carousel.prev()\"\n iconStart=\"@tui.chevron-left\"\n appearance=\"icon\"\n class=\"!absolute left-0 top-1/2 z-10 ml-2 -translate-y-1/2\"\n ></button>\n\n <tui-carousel\n #carousel\n [(index)]=\"index\"\n [duration]=\"duration()\"\n class=\"h-[19em] w-full md:h-96 lg:h-[28rem]\"\n >\n <ng-container *ngFor=\"let item of sample.items\">\n <picture *tuiItem>\n <source\n type=\"image/webp\"\n [srcset]=\"item.carouselItemImage.webp\"\n />\n <img\n [src]=\"item.carouselItemImage.defaultFormat\"\n [alt]=\"item.carouselItemImage.alt ?? '\u041A\u043E\u043D\u0441\u0442\u0440\u0443\u043A\u0446\u0438\u044F \u043E\u0431\u0440\u0430\u0437\u0446\u0430 ' + sample.title\"\n class=\"h-[19em] w-full object-cover object-right md:h-96 lg:h-[28rem]\"\n />\n </picture>\n </ng-container>\n </tui-carousel>\n <button\n *ngIf=\"isShownSwitches && isShowCarouselSwitches()\"\n tuiIconButton\n (click)=\"carousel.next()\"\n iconStart=\"@tui.chevron-right\"\n appearance=\"icon\"\n class=\"!absolute right-0 top-1/2 z-10 mr-2 -translate-y-1/2\"\n ></button>\n </div>\n</div>\n", styles: [":host{--tui-carousel-padding: 0}tui-avatar.active{--tw-ring-color: var(--tui-background-accent-1);--tw-ring-offset-width: 2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 rgba(0, 0, 0, 0))}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.TuiItem, selector: "[tuiItem]" }, { kind: "component", type: i3.TuiCarouselComponent, selector: "tui-carousel", inputs: ["draggable", "itemsCount", "index"], outputs: ["indexChange", "shift"] }, { kind: "component", type: i3.TuiAvatar, selector: "tui-avatar,button[tuiAvatar],a[tuiAvatar]", inputs: ["size", "round", "src"] }, { kind: "directive", type: i4.TuiHovered, selector: "[tuiHoveredChange]", outputs: ["tuiHoveredChange"] }, { kind: "directive", type: i5.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
37
37
|
}
|
38
38
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScPreviewSampleComponent, decorators: [{
|
39
39
|
type: Component,
|
40
|
-
args: [{ selector: 'sc-preview-sample', host: { ngSkipHydration: 'true' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n *ngIf=\"sample() as sample\"\n class=\"mb-4 flex flex-col-reverse gap-4 md:mb-0 md:flex-row md:gap-8\"\n [class.md:!flex-row-reverse]=\"isReverse()\"\n>\n @let isShownSwitches = sample.items && sample.items.length > 1;\n\n <div class=\"flex max-w-96 flex-1 flex-col justify-center self-center md:max-w-none md:gap-4\">\n <div class=\"flex w-full flex-col gap-3 md:gap-4 md:px-8\">\n <div class=\"text-body-l-bold\">{{ sample.title }}</div>\n <div\n *ngIf=\"isShownSwitches\"\n class=\"flex gap-4\"\n >\n <tui-avatar\n [
|
40
|
+
args: [{ selector: 'sc-preview-sample', host: { ngSkipHydration: 'true' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n *ngIf=\"sample() as sample\"\n class=\"mb-4 flex flex-col-reverse gap-4 md:mb-0 md:flex-row md:gap-8\"\n [class.md:!flex-row-reverse]=\"isReverse()\"\n>\n @let isShownSwitches = sample.items && sample.items.length > 1;\n\n <div class=\"flex max-w-96 flex-1 flex-col justify-center self-center md:max-w-none md:gap-4\">\n <div class=\"flex w-full flex-col gap-3 md:gap-4 md:px-8\">\n <div class=\"text-body-l-bold\">{{ sample.title }}</div>\n <div\n *ngIf=\"isShownSwitches\"\n class=\"flex gap-4\"\n >\n <tui-avatar\n [round]=\"false\"\n *ngFor=\"let item of sample.items; let itemIndex = index\"\n (tuiHoveredChange)=\"index.set(itemIndex)\"\n [class.active]=\"index() === itemIndex\"\n >\n <picture>\n <source\n type=\"image/webp\"\n [srcset]=\"item.buttonImageUrl.webp\"\n />\n <img\n [src]=\"item.buttonImageUrl.defaultFormat\"\n [alt]=\"item.buttonImageUrl.alt ?? '\u0418\u0437\u043E\u0431\u0440\u0430\u0436\u0435\u043D\u0438\u0435 \u043A\u043D\u043E\u043F\u043A\u0438 \u043F\u0435\u0440\u0435\u043A\u043B\u044E\u0447\u0430\u0442\u0435\u043B\u044F ' + sample.title\"\n />\n </picture>\n </tui-avatar>\n </div>\n <div\n *ngIf=\"sample.items as items\"\n class=\"text-body-m-bold\"\n >\n {{ items[index()].color }}\n </div>\n <div>{{ sample.description }}</div>\n <img\n *ngIf=\"sample.descriptionImageUrl\"\n [src]=\"sample.descriptionImageUrl\"\n alt=\"\u041A\u043E\u043D\u0441\u0442\u0440\u0443\u043A\u0446\u0438\u044F \u043E\u0431\u0440\u0430\u0437\u0446\u0430 {{ sample.title }}\"\n class=\"w-fit\"\n />\n <div\n *ngIf=\"sample.items && sample.items[index()].actions?.length\"\n class=\"flex flex-wrap gap-3 md:gap-4\"\n >\n <a\n *ngFor=\"let action of sample.items[index()].actions\"\n tuiButton\n [attr.href]=\"action.href\"\n (click)=\"action.handler ? $event.preventDefault() : {}; action.handler?.()\"\n [iconStart]=\"action.icon\"\n appearance=\"secondary\"\n >\n {{ action.label }}\n </a>\n </div>\n </div>\n </div>\n\n <div class=\"relative h-[19em] flex-1 overflow-hidden rounded-xl shadow-tui-shadow md:h-96 lg:h-[28rem]\">\n <button\n *ngIf=\"isShownSwitches && isShowCarouselSwitches()\"\n tuiIconButton\n (click)=\"carousel.prev()\"\n iconStart=\"@tui.chevron-left\"\n appearance=\"icon\"\n class=\"!absolute left-0 top-1/2 z-10 ml-2 -translate-y-1/2\"\n ></button>\n\n <tui-carousel\n #carousel\n [(index)]=\"index\"\n [duration]=\"duration()\"\n class=\"h-[19em] w-full md:h-96 lg:h-[28rem]\"\n >\n <ng-container *ngFor=\"let item of sample.items\">\n <picture *tuiItem>\n <source\n type=\"image/webp\"\n [srcset]=\"item.carouselItemImage.webp\"\n />\n <img\n [src]=\"item.carouselItemImage.defaultFormat\"\n [alt]=\"item.carouselItemImage.alt ?? '\u041A\u043E\u043D\u0441\u0442\u0440\u0443\u043A\u0446\u0438\u044F \u043E\u0431\u0440\u0430\u0437\u0446\u0430 ' + sample.title\"\n class=\"h-[19em] w-full object-cover object-right md:h-96 lg:h-[28rem]\"\n />\n </picture>\n </ng-container>\n </tui-carousel>\n <button\n *ngIf=\"isShownSwitches && isShowCarouselSwitches()\"\n tuiIconButton\n (click)=\"carousel.next()\"\n iconStart=\"@tui.chevron-right\"\n appearance=\"icon\"\n class=\"!absolute right-0 top-1/2 z-10 mr-2 -translate-y-1/2\"\n ></button>\n </div>\n</div>\n", styles: [":host{--tui-carousel-padding: 0}tui-avatar.active{--tw-ring-color: var(--tui-background-accent-1);--tw-ring-offset-width: 2px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 rgba(0, 0, 0, 0))}\n"] }]
|
41
41
|
}] });
|
42
|
-
//# sourceMappingURL=data:application/json;base64,
|
42
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-preview-sample.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/samples/preview-sample/sc-preview-sample.component.ts","../../../../../projects/client-ui/samples/preview-sample/sc-preview-sample.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAe,KAAK,EAAe,MAAM,eAAe,CAAC;AACnH,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;;;;;;;AAI9C;;GAEG;AAQH,MAAM,OAAO,wBAAwB;IAPrC;QAQI;;WAEG;QACa,UAAK,GAAwB,KAAK,CAAS,CAAC,CAAC,CAAC;QAE9D;;WAEG;QACI,aAAQ,GAAwB,KAAK,CAAS,IAAI,CAAC,CAAC;QAE3D;;WAEG;QACI,WAAM,GAAkC,KAAK,CAAC,QAAQ,EAAoB,CAAC;QAElF;;WAEG;QACI,2BAAsB,GAAyB,KAAK,CAAC,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC;QAEpF;;WAEG;QACI,cAAS,GAAyB,KAAK,CAAC,KAAK,CAAC,CAAC;KACzD;+GAzBY,wBAAwB;mGAAxB,wBAAwB,+zBCfrC,08IAuGA;;4FDxFa,wBAAwB;kBAPpC,SAAS;+BACI,mBAAmB,QAGvB,EAAE,eAAe,EAAE,MAAM,EAAE,mBAChB,uBAAuB,CAAC,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, inject, input, InputSignal, model, ModelSignal } from '@angular/core';\nimport { TUI_IS_MOBILE } from '@taiga-ui/cdk';\n\nimport { ScIPreviewSample } from '../interfaces/sc-i-preview-sample';\n\n/**\n * Компонент представления образцов товара.\n */\n@Component({\n    selector: 'sc-preview-sample',\n    templateUrl: './sc-preview-sample.component.html',\n    styleUrls: ['./sc-preview-sample.component.scss'],\n    host: { ngSkipHydration: 'true' },\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPreviewSampleComponent {\n    /**\n     * Индекс элемента в галерее.\n     */\n    public readonly index: ModelSignal<number> = model<number>(0);\n\n    /**\n     * Длительность переключения карусели.\n     */\n    public duration: InputSignal<number> = input<number>(5000);\n\n    /**\n     * Объект данных товара.\n     */\n    public sample: InputSignal<ScIPreviewSample> = input.required<ScIPreviewSample>();\n\n    /**\n     * Признак, что необходимо отображать переключатели на карусели.\n     */\n    public isShowCarouselSwitches: InputSignal<boolean> = input(!inject(TUI_IS_MOBILE));\n\n    /**\n     * Признак, что при горизонтальном отображении компонент должен быть развёрнут.\n     */\n    public isReverse: InputSignal<boolean> = input(false);\n}\n","<div\n    *ngIf=\"sample() as sample\"\n    class=\"mb-4 flex flex-col-reverse gap-4 md:mb-0 md:flex-row md:gap-8\"\n    [class.md:!flex-row-reverse]=\"isReverse()\"\n>\n    @let isShownSwitches = sample.items && sample.items.length > 1;\n\n    <div class=\"flex max-w-96 flex-1 flex-col justify-center self-center md:max-w-none md:gap-4\">\n        <div class=\"flex w-full flex-col gap-3 md:gap-4 md:px-8\">\n            <div class=\"text-body-l-bold\">{{ sample.title }}</div>\n            <div\n                *ngIf=\"isShownSwitches\"\n                class=\"flex gap-4\"\n            >\n                <tui-avatar\n                    [round]=\"false\"\n                    *ngFor=\"let item of sample.items; let itemIndex = index\"\n                    (tuiHoveredChange)=\"index.set(itemIndex)\"\n                    [class.active]=\"index() === itemIndex\"\n                >\n                    <picture>\n                        <source\n                            type=\"image/webp\"\n                            [srcset]=\"item.buttonImageUrl.webp\"\n                        />\n                        <img\n                            [src]=\"item.buttonImageUrl.defaultFormat\"\n                            [alt]=\"item.buttonImageUrl.alt ?? 'Изображение кнопки переключателя ' + sample.title\"\n                        />\n                    </picture>\n                </tui-avatar>\n            </div>\n            <div\n                *ngIf=\"sample.items as items\"\n                class=\"text-body-m-bold\"\n            >\n                {{ items[index()].color }}\n            </div>\n            <div>{{ sample.description }}</div>\n            <img\n                *ngIf=\"sample.descriptionImageUrl\"\n                [src]=\"sample.descriptionImageUrl\"\n                alt=\"Конструкция образца {{ sample.title }}\"\n                class=\"w-fit\"\n            />\n            <div\n                *ngIf=\"sample.items && sample.items[index()].actions?.length\"\n                class=\"flex flex-wrap gap-3 md:gap-4\"\n            >\n                <a\n                    *ngFor=\"let action of sample.items[index()].actions\"\n                    tuiButton\n                    [attr.href]=\"action.href\"\n                    (click)=\"action.handler ? $event.preventDefault() : {}; action.handler?.()\"\n                    [iconStart]=\"action.icon\"\n                    appearance=\"secondary\"\n                >\n                    {{ action.label }}\n                </a>\n            </div>\n        </div>\n    </div>\n\n    <div class=\"relative h-[19em] flex-1 overflow-hidden rounded-xl shadow-tui-shadow md:h-96 lg:h-[28rem]\">\n        <button\n            *ngIf=\"isShownSwitches && isShowCarouselSwitches()\"\n            tuiIconButton\n            (click)=\"carousel.prev()\"\n            iconStart=\"@tui.chevron-left\"\n            appearance=\"icon\"\n            class=\"!absolute left-0 top-1/2 z-10 ml-2 -translate-y-1/2\"\n        ></button>\n\n        <tui-carousel\n            #carousel\n            [(index)]=\"index\"\n            [duration]=\"duration()\"\n            class=\"h-[19em] w-full md:h-96 lg:h-[28rem]\"\n        >\n            <ng-container *ngFor=\"let item of sample.items\">\n                <picture *tuiItem>\n                    <source\n                        type=\"image/webp\"\n                        [srcset]=\"item.carouselItemImage.webp\"\n                    />\n                    <img\n                        [src]=\"item.carouselItemImage.defaultFormat\"\n                        [alt]=\"item.carouselItemImage.alt ?? 'Конструкция образца ' + sample.title\"\n                        class=\"h-[19em] w-full object-cover object-right md:h-96 lg:h-[28rem]\"\n                    />\n                </picture>\n            </ng-container>\n        </tui-carousel>\n        <button\n            *ngIf=\"isShownSwitches && isShowCarouselSwitches()\"\n            tuiIconButton\n            (click)=\"carousel.next()\"\n            iconStart=\"@tui.chevron-right\"\n            appearance=\"icon\"\n            class=\"!absolute right-0 top-1/2 z-10 mr-2 -translate-y-1/2\"\n        ></button>\n    </div>\n</div>\n"]}
|