@snabcentr/client-ui 0.8.7 → 0.10.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.
Potentially problematic release.
This version of @snabcentr/client-ui might be problematic. Click here for more details.
- package/auth/constants/phone-approve-code-mask.d.ts +1 -1
- package/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.d.ts +3 -3
- package/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.d.ts +10 -7
- package/banner/sc-banner.component.d.ts +1 -1
- package/esm2020/auth/constants/phone-approve-code-mask.mjs +2 -2
- package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +6 -6
- package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +14 -13
- package/esm2020/auth/sc-sign-in-form/sc-sign-in-form.component.mjs +3 -3
- package/esm2020/banner/sc-banner.component.mjs +2 -2
- package/esm2020/catalog/price-card/sc-price-card.component.mjs +4 -1
- package/esm2020/form-fields/addresses-selection-field/sc-addresses-selection-field.component.mjs +117 -0
- package/esm2020/form-fields/form-fields.module.mjs +80 -0
- package/esm2020/form-fields/suggestion-field/sc-suggestion-field.component.mjs +111 -0
- package/esm2020/profile/sc-profile-accordions-content/sc-profile-accordions-content.component.mjs +29 -16
- package/esm2020/user/index.mjs +3 -1
- package/esm2020/user/reset-user-password/sc-reset-user-password.component.mjs +4 -4
- package/esm2020/user/sc-user.module.mjs +42 -8
- package/esm2020/user/update-user-info-form/sc-update-user-info-form.component.mjs +97 -0
- package/esm2020/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.mjs +91 -0
- package/esm2020/verification/index.mjs +2 -2
- package/esm2020/verification/sc-verification.module.mjs +6 -6
- package/esm2020/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +184 -0
- package/fesm2015/snabcentr-client-ui.mjs +608 -94
- package/fesm2015/snabcentr-client-ui.mjs.map +1 -1
- package/fesm2020/snabcentr-client-ui.mjs +594 -94
- package/fesm2020/snabcentr-client-ui.mjs.map +1 -1
- package/form-fields/addresses-selection-field/sc-addresses-selection-field.component.d.ts +65 -0
- package/form-fields/form-fields.module.d.ts +17 -0
- package/form-fields/suggestion-field/sc-suggestion-field.component.d.ts +45 -0
- package/package.json +2 -2
- package/profile/sc-profile-accordions-content/sc-profile-accordions-content.component.d.ts +24 -10
- package/styles/tailwind/tailwind.scss +22 -0
- package/user/index.d.ts +2 -0
- package/user/reset-user-password/sc-reset-user-password.component.d.ts +3 -3
- package/user/sc-user.module.d.ts +10 -6
- package/user/update-user-info-form/sc-update-user-info-form.component.d.ts +56 -0
- package/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.d.ts +48 -0
- package/verification/index.d.ts +1 -1
- package/verification/sc-verification.module.d.ts +2 -2
- package/verification/{phone-approve-form/phone-approve-form.component.d.ts → verification-phone-check-form/sc-verification-phone-check-form.component.d.ts} +27 -11
- package/esm2020/verification/phone-approve-form/phone-approve-form.component.mjs +0 -156
@@ -121,6 +121,9 @@ export class ScPriceCardComponent {
|
|
121
121
|
* Возвращает ссылку на preview-изображение карточки товара.
|
122
122
|
*/
|
123
123
|
getCardImagePreview() {
|
124
|
+
if (this.product && this.product.images) {
|
125
|
+
this.product.images.sort((img1, img2) => Number(img2.isDefault) - Number(img1.isDefault));
|
126
|
+
}
|
124
127
|
return this.product ? this.imageHelper.getImagePreview(this.product) : this.pathImageNotFound;
|
125
128
|
}
|
126
129
|
/**
|
@@ -163,4 +166,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
163
166
|
type: HostBinding,
|
164
167
|
args: ['attr.data-size']
|
165
168
|
}] } });
|
166
|
-
//# 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, 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, ScImageHelper } from '@snabcentr/client-core';\nimport { TuiSizeS, TuiSizeXS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\nimport { TUI_IS_MOBILE } from '@taiga-ui/cdk';\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 readonly isMobile: boolean = inject(TUI_IS_MOBILE);\n\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     * @param imageHelper Хелпер для работы с изображениями товара.\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        private readonly imageHelper: ScImageHelper\n    ) {}\n\n    /**\n     * Признак, что нужно показать скелетон.\n     */\n    public get skeletonVisible(): boolean {\n        return !this.product;\n    }\n\n    /**\n     * Возвращает ссылку на preview-изображение карточки товара.\n     */\n    public getCardImagePreview(): string {\n        return this.product ? this.imageHelper.getImagePreview(this.product) : 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]=\"getCardImagePreview()\" [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 && !isMobile\" 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        #inputQuantity\n        *ngIf=\"showQuantityControl\"\n        [formControl]=\"quantityControl\"\n        [size]=\"getQuantitySize\"\n        [step]=\"unitsHelper.productMultiplicity(product)\"\n        [showLoader]=\"quantityShowLoader\"\n        (clickClearEvent)=\"clickClearEvent.emit(cartItem)\"\n        class=\"w-full\"\n        (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\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"]}
|
169
|
+
//# 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, 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, ScImageHelper } from '@snabcentr/client-core';\nimport { TuiSizeS, TuiSizeXS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\nimport { TUI_IS_MOBILE } from '@taiga-ui/cdk';\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 readonly isMobile: boolean = inject(TUI_IS_MOBILE);\n\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     * @param imageHelper Хелпер для работы с изображениями товара.\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        private readonly imageHelper: ScImageHelper\n    ) {}\n\n    /**\n     * Признак, что нужно показать скелетон.\n     */\n    public get skeletonVisible(): boolean {\n        return !this.product;\n    }\n\n    /**\n     * Возвращает ссылку на preview-изображение карточки товара.\n     */\n    public getCardImagePreview(): string {\n        if (this.product && this.product.images) {\n            this.product.images.sort((img1, img2) => Number(img2.isDefault) - Number(img1.isDefault));\n        }\n\n        return this.product ? this.imageHelper.getImagePreview(this.product) : 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]=\"getCardImagePreview()\" [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 && !isMobile\" 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        #inputQuantity\n        *ngIf=\"showQuantityControl\"\n        [formControl]=\"quantityControl\"\n        [size]=\"getQuantitySize\"\n        [step]=\"unitsHelper.productMultiplicity(product)\"\n        [showLoader]=\"quantityShowLoader\"\n        (clickClearEvent)=\"clickClearEvent.emit(cartItem)\"\n        class=\"w-full\"\n        (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\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"]}
|
package/esm2020/form-fields/addresses-selection-field/sc-addresses-selection-field.component.mjs
ADDED
@@ -0,0 +1,117 @@
|
|
1
|
+
import { __decorate } from "tslib";
|
2
|
+
import { ChangeDetectionStrategy, Component, Inject, SkipSelf } from '@angular/core';
|
3
|
+
import { FormControl, FormGroupDirective, Validators } from '@angular/forms';
|
4
|
+
import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
|
5
|
+
import { tuiControlValue, tuiIsPresent, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';
|
6
|
+
import { combineLatest, debounceTime, filter, of, share, startWith, switchMap, tap } from 'rxjs';
|
7
|
+
import * as i0 from "@angular/core";
|
8
|
+
import * as i1 from "@snabcentr/client-core";
|
9
|
+
import * as i2 from "@angular/common";
|
10
|
+
import * as i3 from "@angular/forms";
|
11
|
+
import * as i4 from "@taiga-ui/core";
|
12
|
+
import * as i5 from "@taiga-ui/cdk";
|
13
|
+
import * as i6 from "@taiga-ui/kit";
|
14
|
+
/**
|
15
|
+
* Компонент поля ввода страны/региона/города.
|
16
|
+
*/
|
17
|
+
let ScAddressesSelectionFieldComponent = class ScAddressesSelectionFieldComponent {
|
18
|
+
/**
|
19
|
+
* Инициализирует экземпляр класса {@link BankSuggestionFieldsComponent}.
|
20
|
+
*
|
21
|
+
* @param formGroupDirective Директива c `FormGroup` из DOM.
|
22
|
+
* @param locationsService Сервис для получения списков стран, регионов, городов.
|
23
|
+
*/
|
24
|
+
constructor(formGroupDirective, locationsService) {
|
25
|
+
this.formGroupDirective = formGroupDirective;
|
26
|
+
this.locationsService = locationsService;
|
27
|
+
/**
|
28
|
+
* FormControl для поля страны.
|
29
|
+
*/
|
30
|
+
this.countryControl = new FormControl({
|
31
|
+
id: 3159,
|
32
|
+
name: 'Россия',
|
33
|
+
}, Validators.required);
|
34
|
+
/**
|
35
|
+
* FormControl для поля региона.
|
36
|
+
*/
|
37
|
+
this.regionControl = new FormControl(null, Validators.required);
|
38
|
+
/**
|
39
|
+
* FormControl для поля города.
|
40
|
+
*/
|
41
|
+
this.cityControl = new FormControl(null, Validators.required);
|
42
|
+
/**
|
43
|
+
* {@link Observable} изменения списка стран.
|
44
|
+
*/
|
45
|
+
this.countries$ = this.locationsService.getCountries$().pipe(share(), startWith(null));
|
46
|
+
/**
|
47
|
+
* {@link Observable} изменения списка регионов.
|
48
|
+
*/
|
49
|
+
this.regions$ = tuiControlValue(this.countryControl).pipe(filter(tuiIsPresent), switchMap((country) => this.locationsService.getRegions$(country.id).pipe(startWith(null))));
|
50
|
+
/**
|
51
|
+
* {@link Observable} изменения списка городов.
|
52
|
+
*/
|
53
|
+
this.cities$ = combineLatest({
|
54
|
+
country: tuiControlValue(this.countryControl).pipe(tap((country) => (country ? this.regionControl.enable() : this.regionControl.disable())), tap(() => {
|
55
|
+
this.regionControl.reset(null);
|
56
|
+
this.cityControl.reset(null);
|
57
|
+
this.cityControl.disable();
|
58
|
+
})),
|
59
|
+
region: tuiControlValue(this.regionControl).pipe(tap(() => {
|
60
|
+
this.cityControl.reset(null);
|
61
|
+
this.cityIdControl.reset(null);
|
62
|
+
}), tap((region) => (region ? this.cityControl.enable() : this.cityControl.disable()))),
|
63
|
+
}).pipe(debounceTime(0), switchMap(({ country, region }) => country && region ? this.locationsService.getCities$(country.id, region.id).pipe(startWith(null)) : of([])), share());
|
64
|
+
/**
|
65
|
+
* Функция преобразования объекта сортировки в значение, отображаемое в поле ввода.
|
66
|
+
*
|
67
|
+
* @param item Выбранный объект выпадающего списка.
|
68
|
+
*/
|
69
|
+
this.stringify = (item) => `${item.name}`;
|
70
|
+
}
|
71
|
+
/**
|
72
|
+
* `FormControl` поля идентификатора города.
|
73
|
+
*/
|
74
|
+
get cityIdControl() {
|
75
|
+
return this.formGroupDirective.form.get('cityId');
|
76
|
+
}
|
77
|
+
/** @inheritDoc */
|
78
|
+
ngOnInit() {
|
79
|
+
if (this.cityIdControl && this.cityIdControl.valid) {
|
80
|
+
this.locationsService
|
81
|
+
.getUserCity$(Number(this.cityIdControl?.value))
|
82
|
+
.pipe(filter((city) => city?.region?.country?.id === 3159), untilDestroyed(this))
|
83
|
+
.subscribe((city) => {
|
84
|
+
this.countryControl.patchValue(city?.region?.country ?? null);
|
85
|
+
this.regionControl.patchValue(city?.region ?? null);
|
86
|
+
this.cityControl.patchValue(city);
|
87
|
+
});
|
88
|
+
}
|
89
|
+
}
|
90
|
+
/**
|
91
|
+
* Заполнение полей банковских реквизитов на основе выбранной подсказки.
|
92
|
+
*
|
93
|
+
* @param suggestion Объект подсказки по банку.
|
94
|
+
*/
|
95
|
+
onSelectedCity(city) {
|
96
|
+
if (this.cityIdControl) {
|
97
|
+
this.cityIdControl.setValue(city?.id.toString() ?? null);
|
98
|
+
tuiMarkControlAsTouchedAndValidate(this.cityIdControl);
|
99
|
+
}
|
100
|
+
}
|
101
|
+
};
|
102
|
+
ScAddressesSelectionFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScAddressesSelectionFieldComponent, deps: [{ token: FormGroupDirective, skipSelf: true }, { token: i1.ScLocationsService }], target: i0.ɵɵFactoryTarget.Component });
|
103
|
+
ScAddressesSelectionFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScAddressesSelectionFieldComponent, selector: "sc-addresses-selection-field", ngImport: i0, template: "<div class=\"flex flex-col sm:flex-row gap-4 items-start\">\n <!-- TODO: \u041E\u0442\u043A\u0440\u044B\u0442\u044C \u043F\u043E\u043B\u0435 \u0432\u044B\u0431\u043E\u0440\u0430 \u0441\u0442\u0440\u0430\u043D\u044B \u043F\u0440\u0438 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u0438 \u0432\u043E\u0437\u043C\u043E\u0436\u043D\u043E\u0441\u0442\u0438 \u0432\u044B\u0431\u043E\u0440\u0430 \u0441\u0442\u0440\u0430\u043D\u044B \u0434\u043B\u044F \u043A\u043B\u0438\u0435\u043D\u0442\u0430. -->\n <!-- <label tuiLabel=\"\u0421\u0442\u0440\u0430\u043D\u0430\" class=\"w-full\">\n <tui-combo-box *tuiLet=\"countries$ | async as countries\" [formControl]=\"countryControl\" [stringify]=\"stringify\">\n \u0421\u0442\u0440\u0430\u043D\u0430\n <input tuiTextfield autocomplete=\"new-password\" />\n <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"countries | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n </tui-combo-box>\n <tui-error [formControl]=\"countryControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n </label> -->\n <!-- TODO: [TASK:9383]: \u0414\u043E\u0440\u0430\u0431\u043E\u0442\u0430\u0442\u044C \u043A\u043E\u043C\u043F\u043E\u043D\u0435\u043D\u0442 \u0432\u044B\u0431\u043E\u0440\u0430 \u0433\u043E\u0440\u043E\u0434\u0430, \u043A\u0430\u043A \u043D\u0430 \u043F\u043E\u0440\u0442\u0430\u043B\u0435 -->\n <label tuiLabel=\"\u0420\u0435\u0433\u0438\u043E\u043D\" class=\"w-full\">\n <tui-combo-box *tuiLet=\"regions$ | async as regions\" [formControl]=\"regionControl\" [stringify]=\"stringify\">\n \u0420\u0435\u0433\u0438\u043E\u043D\n <input tuiTextfield autocapitalize=\"off\" autocomplete=\"off\" autocorrect=\"off\" />\n <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"regions | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n </tui-combo-box>\n <tui-error [formControl]=\"regionControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n </label>\n <label tuiLabel=\"\u0413\u043E\u0440\u043E\u0434\" class=\"w-full\">\n <tui-combo-box *tuiLet=\"cities$ | async as cities\" [formControl]=\"cityControl\" [stringify]=\"stringify\" (ngModelChange)=\"onSelectedCity($event)\">\n \u0413\u043E\u0440\u043E\u0434\n <input tuiTextfield autocapitalize=\"off\" autocomplete=\"off\" autocorrect=\"off\" />\n <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"cities | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n </tui-combo-box>\n <tui-error [formControl]=\"cityIdControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n <tui-error *ngIf=\"cityIdControl.untouched\" [formControl]=\"cityControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n </label>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i4.TuiTextfieldComponent, selector: "input[tuiTextfield], textarea[tuiTextfield]" }, { kind: "component", type: i4.TuiLabelComponent, selector: "label[tuiLabel]", inputs: ["tuiLabel", "context"] }, { kind: "directive", type: i5.TuiLetDirective, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "component", type: i4.TuiErrorComponent, selector: "tui-error", inputs: ["error"] }, { kind: "directive", type: i4.TuiDataListDirective, selector: "ng-template[tuiDataList]" }, { kind: "component", type: i6.TuiComboBoxComponent, selector: "tui-combo-box", inputs: ["stringify", "strictMatcher", "identityMatcher", "valueContent", "strict", "search"], outputs: ["searchChange"] }, { kind: "directive", type: i6.TuiComboBoxDirective, selector: "tui-combo-box" }, { kind: "component", type: i6.TuiDataListWrapperComponent, selector: "tui-data-list-wrapper:not([labels])", inputs: ["items"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.TuiFieldErrorPipe, name: "tuiFieldError" }, { kind: "pipe", type: i6.TuiFilterByInputWithPipe, name: "tuiFilterByInputWith" }, { kind: "pipe", type: i6.TuiStringifyContentPipe, name: "tuiStringifyContent" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
104
|
+
ScAddressesSelectionFieldComponent = __decorate([
|
105
|
+
UntilDestroy({ checkProperties: true })
|
106
|
+
], ScAddressesSelectionFieldComponent);
|
107
|
+
export { ScAddressesSelectionFieldComponent };
|
108
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScAddressesSelectionFieldComponent, decorators: [{
|
109
|
+
type: Component,
|
110
|
+
args: [{ selector: 'sc-addresses-selection-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col sm:flex-row gap-4 items-start\">\n <!-- TODO: \u041E\u0442\u043A\u0440\u044B\u0442\u044C \u043F\u043E\u043B\u0435 \u0432\u044B\u0431\u043E\u0440\u0430 \u0441\u0442\u0440\u0430\u043D\u044B \u043F\u0440\u0438 \u0434\u043E\u0431\u0430\u0432\u043B\u0435\u043D\u0438\u0438 \u0432\u043E\u0437\u043C\u043E\u0436\u043D\u043E\u0441\u0442\u0438 \u0432\u044B\u0431\u043E\u0440\u0430 \u0441\u0442\u0440\u0430\u043D\u044B \u0434\u043B\u044F \u043A\u043B\u0438\u0435\u043D\u0442\u0430. -->\n <!-- <label tuiLabel=\"\u0421\u0442\u0440\u0430\u043D\u0430\" class=\"w-full\">\n <tui-combo-box *tuiLet=\"countries$ | async as countries\" [formControl]=\"countryControl\" [stringify]=\"stringify\">\n \u0421\u0442\u0440\u0430\u043D\u0430\n <input tuiTextfield autocomplete=\"new-password\" />\n <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"countries | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n </tui-combo-box>\n <tui-error [formControl]=\"countryControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n </label> -->\n <!-- TODO: [TASK:9383]: \u0414\u043E\u0440\u0430\u0431\u043E\u0442\u0430\u0442\u044C \u043A\u043E\u043C\u043F\u043E\u043D\u0435\u043D\u0442 \u0432\u044B\u0431\u043E\u0440\u0430 \u0433\u043E\u0440\u043E\u0434\u0430, \u043A\u0430\u043A \u043D\u0430 \u043F\u043E\u0440\u0442\u0430\u043B\u0435 -->\n <label tuiLabel=\"\u0420\u0435\u0433\u0438\u043E\u043D\" class=\"w-full\">\n <tui-combo-box *tuiLet=\"regions$ | async as regions\" [formControl]=\"regionControl\" [stringify]=\"stringify\">\n \u0420\u0435\u0433\u0438\u043E\u043D\n <input tuiTextfield autocapitalize=\"off\" autocomplete=\"off\" autocorrect=\"off\" />\n <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"regions | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n </tui-combo-box>\n <tui-error [formControl]=\"regionControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n </label>\n <label tuiLabel=\"\u0413\u043E\u0440\u043E\u0434\" class=\"w-full\">\n <tui-combo-box *tuiLet=\"cities$ | async as cities\" [formControl]=\"cityControl\" [stringify]=\"stringify\" (ngModelChange)=\"onSelectedCity($event)\">\n \u0413\u043E\u0440\u043E\u0434\n <input tuiTextfield autocapitalize=\"off\" autocomplete=\"off\" autocorrect=\"off\" />\n <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"cities | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n </tui-combo-box>\n <tui-error [formControl]=\"cityIdControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n <tui-error *ngIf=\"cityIdControl.untouched\" [formControl]=\"cityControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n </label>\n</div>\n" }]
|
111
|
+
}], ctorParameters: function () { return [{ type: i3.FormGroupDirective, decorators: [{
|
112
|
+
type: SkipSelf
|
113
|
+
}, {
|
114
|
+
type: Inject,
|
115
|
+
args: [FormGroupDirective]
|
116
|
+
}] }, { type: i1.ScLocationsService }]; } });
|
117
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-addresses-selection-field.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/form-fields/addresses-selection-field/sc-addresses-selection-field.component.ts","../../../../../projects/client-ui/form-fields/addresses-selection-field/sc-addresses-selection-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Inject, OnInit, SkipSelf } from '@angular/core';\nimport { FormControl, FormGroupDirective, Validators } from '@angular/forms';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { ScICity, ScICountry, ScIRegion, ScLocationsService } from '@snabcentr/client-core';\nimport { tuiControlValue, tuiIsPresent, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { Observable, combineLatest, debounceTime, filter, of, share, startWith, switchMap, tap } from 'rxjs';\n\n/**\n * Компонент поля ввода страны/региона/города.\n */\n@Component({\n    selector: 'sc-addresses-selection-field',\n    templateUrl: './sc-addresses-selection-field.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\n@UntilDestroy({ checkProperties: true })\nexport class ScAddressesSelectionFieldComponent implements OnInit {\n    /**\n     * FormControl для поля страны.\n     */\n    public countryControl: FormControl<ScICountry | null> = new FormControl<ScICountry | null>(\n        {\n            id: 3159,\n            name: 'Россия',\n        },\n        Validators.required\n    );\n\n    /**\n     * FormControl для поля региона.\n     */\n    public regionControl: FormControl<ScIRegion | null> = new FormControl<ScIRegion | null>(null, Validators.required);\n\n    /**\n     * FormControl для поля города.\n     */\n    public cityControl: FormControl<ScICity | null> = new FormControl<ScICountry | null>(null, Validators.required);\n\n    /**\n     * {@link Observable} изменения списка стран.\n     */\n    public countries$: Observable<ScICountry[] | null> = this.locationsService.getCountries$().pipe(share(), startWith(null));\n\n    /**\n     * {@link Observable} изменения списка регионов.\n     */\n    public regions$: Observable<ScIRegion[] | null> = tuiControlValue<ScIRegion | null>(this.countryControl).pipe(\n        filter(tuiIsPresent),\n        switchMap((country: ScICountry) => this.locationsService.getRegions$(country.id).pipe(startWith(null)))\n    );\n\n    /**\n     * {@link Observable} изменения списка городов.\n     */\n    public cities$: Observable<ScICity[] | null> = combineLatest({\n        country: tuiControlValue<ScIRegion | null>(this.countryControl).pipe(\n            tap((country) => (country ? this.regionControl.enable() : this.regionControl.disable())),\n            tap(() => {\n                this.regionControl.reset(null);\n                this.cityControl.reset(null);\n                this.cityControl.disable();\n            })\n        ),\n        region: tuiControlValue<ScICity | null>(this.regionControl).pipe(\n            tap(() => {\n                this.cityControl.reset(null);\n                this.cityIdControl.reset(null);\n            }),\n            tap((region) => (region ? this.cityControl.enable() : this.cityControl.disable()))\n        ),\n    }).pipe(\n        debounceTime(0),\n        switchMap(({ country, region }: { country: ScIRegion | null; region: ScICity | null }) =>\n            country && region ? this.locationsService.getCities$(country.id, region.id).pipe(startWith(null)) : of([])\n        ),\n        share()\n    );\n\n    /**\n     * `FormControl` поля идентификатора города.\n     */\n    public get cityIdControl(): FormControl<string | null> {\n        return this.formGroupDirective.form.get('cityId')! as FormControl<string | null>;\n    }\n\n    /**\n     * Инициализирует экземпляр класса {@link BankSuggestionFieldsComponent}.\n     *\n     * @param formGroupDirective Директива c `FormGroup` из DOM.\n     * @param locationsService Сервис для получения списков стран, регионов, городов.\n     */\n    public constructor(@SkipSelf() @Inject(FormGroupDirective) private readonly formGroupDirective: FormGroupDirective, private readonly locationsService: ScLocationsService) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (this.cityIdControl && this.cityIdControl.valid) {\n            this.locationsService\n                .getUserCity$(Number(this.cityIdControl?.value))\n                .pipe(\n                    filter((city: ScICity) => city?.region?.country?.id === 3159),\n                    untilDestroyed(this)\n                )\n                .subscribe((city: ScICity) => {\n                    this.countryControl.patchValue(city?.region?.country ?? null);\n                    this.regionControl.patchValue(city?.region ?? null);\n                    this.cityControl.patchValue(city);\n                });\n        }\n    }\n\n    /**\n     * Заполнение полей банковских реквизитов на основе выбранной подсказки.\n     *\n     * @param suggestion Объект подсказки по банку.\n     */\n    public onSelectedCity(city: ScICity | null): void {\n        if (this.cityIdControl) {\n            this.cityIdControl.setValue(city?.id.toString() ?? null);\n            tuiMarkControlAsTouchedAndValidate(this.cityIdControl);\n        }\n    }\n\n    /**\n     * Функция преобразования объекта сортировки в значение, отображаемое в поле ввода.\n     *\n     * @param item Выбранный объект выпадающего списка.\n     */\n    public readonly stringify = (item: { name: string }): string => `${item.name}`;\n}\n","<div class=\"flex flex-col sm:flex-row gap-4 items-start\">\n    <!-- TODO: Открыть поле выбора страны при добавлении возможности выбора страны для клиента. -->\n    <!-- <label tuiLabel=\"Страна\" class=\"w-full\">\n        <tui-combo-box *tuiLet=\"countries$ | async as countries\" [formControl]=\"countryControl\" [stringify]=\"stringify\">\n            Страна\n            <input tuiTextfield autocomplete=\"new-password\" />\n            <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"countries | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error [formControl]=\"countryControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </label> -->\n    <!-- TODO: [TASK:9383]: Доработать компонент выбора города, как на портале  -->\n    <label tuiLabel=\"Регион\" class=\"w-full\">\n        <tui-combo-box *tuiLet=\"regions$ | async as regions\" [formControl]=\"regionControl\" [stringify]=\"stringify\">\n            Регион\n            <input tuiTextfield autocapitalize=\"off\" autocomplete=\"off\" autocorrect=\"off\" />\n            <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"regions | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error [formControl]=\"regionControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </label>\n    <label tuiLabel=\"Город\" class=\"w-full\">\n        <tui-combo-box *tuiLet=\"cities$ | async as cities\" [formControl]=\"cityControl\" [stringify]=\"stringify\" (ngModelChange)=\"onSelectedCity($event)\">\n            Город\n            <input tuiTextfield autocapitalize=\"off\" autocomplete=\"off\" autocorrect=\"off\" />\n            <tui-data-list-wrapper *tuiDataList [itemContent]=\"stringify | tuiStringifyContent\" [items]=\"cities | tuiFilterByInputWith: stringify\"></tui-data-list-wrapper>\n        </tui-combo-box>\n        <tui-error [formControl]=\"cityIdControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n        <tui-error *ngIf=\"cityIdControl.untouched\" [formControl]=\"cityControl\" [error]=\"[] | tuiFieldError | async\"></tui-error>\n    </label>\n</div>\n"]}
|
@@ -0,0 +1,80 @@
|
|
1
|
+
import { NgModule } from '@angular/core';
|
2
|
+
import { CommonModule } from '@angular/common';
|
3
|
+
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
4
|
+
import { TuiLetModule } from '@taiga-ui/cdk';
|
5
|
+
import { TuiTextfieldControllerModule, TuiLabelModule, TuiErrorModule, TuiDataListModule, TuiLoaderModule, TuiButtonModule } from '@taiga-ui/core';
|
6
|
+
import { TuiInputModule, TuiFieldErrorPipeModule, TuiComboBoxModule, TuiDataListWrapperModule, TuiFilterByInputPipeModule, TuiStringifyContentPipeModule, TuiInputPhoneModule, } from '@taiga-ui/kit';
|
7
|
+
import { MaskitoModule } from '@maskito/angular';
|
8
|
+
import { ScAddressesSelectionFieldComponent } from './addresses-selection-field/sc-addresses-selection-field.component';
|
9
|
+
import { ScSuggestionFieldComponent } from './suggestion-field/sc-suggestion-field.component';
|
10
|
+
import * as i0 from "@angular/core";
|
11
|
+
/**
|
12
|
+
* Модуль полей ввода.
|
13
|
+
*/
|
14
|
+
export class ScFormFieldsModule {
|
15
|
+
}
|
16
|
+
ScFormFieldsModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScFormFieldsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
17
|
+
ScFormFieldsModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "14.3.0", ngImport: i0, type: ScFormFieldsModule, declarations: [ScAddressesSelectionFieldComponent, ScSuggestionFieldComponent], imports: [CommonModule,
|
18
|
+
FormsModule,
|
19
|
+
ReactiveFormsModule,
|
20
|
+
TuiInputModule,
|
21
|
+
TuiTextfieldControllerModule,
|
22
|
+
TuiLabelModule,
|
23
|
+
TuiLetModule,
|
24
|
+
TuiFieldErrorPipeModule,
|
25
|
+
TuiErrorModule,
|
26
|
+
TuiDataListModule,
|
27
|
+
TuiLoaderModule,
|
28
|
+
TuiComboBoxModule,
|
29
|
+
TuiDataListWrapperModule,
|
30
|
+
TuiFilterByInputPipeModule,
|
31
|
+
TuiStringifyContentPipeModule,
|
32
|
+
TuiInputPhoneModule,
|
33
|
+
TuiButtonModule,
|
34
|
+
MaskitoModule], exports: [ScAddressesSelectionFieldComponent, ScSuggestionFieldComponent] });
|
35
|
+
ScFormFieldsModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScFormFieldsModule, imports: [CommonModule,
|
36
|
+
FormsModule,
|
37
|
+
ReactiveFormsModule,
|
38
|
+
TuiInputModule,
|
39
|
+
TuiTextfieldControllerModule,
|
40
|
+
TuiLabelModule,
|
41
|
+
TuiLetModule,
|
42
|
+
TuiFieldErrorPipeModule,
|
43
|
+
TuiErrorModule,
|
44
|
+
TuiDataListModule,
|
45
|
+
TuiLoaderModule,
|
46
|
+
TuiComboBoxModule,
|
47
|
+
TuiDataListWrapperModule,
|
48
|
+
TuiFilterByInputPipeModule,
|
49
|
+
TuiStringifyContentPipeModule,
|
50
|
+
TuiInputPhoneModule,
|
51
|
+
TuiButtonModule,
|
52
|
+
MaskitoModule] });
|
53
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScFormFieldsModule, decorators: [{
|
54
|
+
type: NgModule,
|
55
|
+
args: [{
|
56
|
+
declarations: [ScAddressesSelectionFieldComponent, ScSuggestionFieldComponent],
|
57
|
+
imports: [
|
58
|
+
CommonModule,
|
59
|
+
FormsModule,
|
60
|
+
ReactiveFormsModule,
|
61
|
+
TuiInputModule,
|
62
|
+
TuiTextfieldControllerModule,
|
63
|
+
TuiLabelModule,
|
64
|
+
TuiLetModule,
|
65
|
+
TuiFieldErrorPipeModule,
|
66
|
+
TuiErrorModule,
|
67
|
+
TuiDataListModule,
|
68
|
+
TuiLoaderModule,
|
69
|
+
TuiComboBoxModule,
|
70
|
+
TuiDataListWrapperModule,
|
71
|
+
TuiFilterByInputPipeModule,
|
72
|
+
TuiStringifyContentPipeModule,
|
73
|
+
TuiInputPhoneModule,
|
74
|
+
TuiButtonModule,
|
75
|
+
MaskitoModule,
|
76
|
+
],
|
77
|
+
exports: [ScAddressesSelectionFieldComponent, ScSuggestionFieldComponent],
|
78
|
+
}]
|
79
|
+
}] });
|
80
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybS1maWVsZHMubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL2Zvcm0tZmllbGRzL2Zvcm0tZmllbGRzLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUM3QyxPQUFPLEVBQUUsNEJBQTRCLEVBQUUsY0FBYyxFQUFFLGNBQWMsRUFBRSxpQkFBaUIsRUFBRSxlQUFlLEVBQUUsZUFBZSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbkosT0FBTyxFQUNILGNBQWMsRUFDZCx1QkFBdUIsRUFDdkIsaUJBQWlCLEVBQ2pCLHdCQUF3QixFQUN4QiwwQkFBMEIsRUFDMUIsNkJBQTZCLEVBQzdCLG1CQUFtQixHQUN0QixNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFDakQsT0FBTyxFQUFFLGtDQUFrQyxFQUFFLE1BQU0sb0VBQW9FLENBQUM7QUFDeEgsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sa0RBQWtELENBQUM7O0FBRTlGOztHQUVHO0FBeUJILE1BQU0sT0FBTyxrQkFBa0I7OytHQUFsQixrQkFBa0I7Z0hBQWxCLGtCQUFrQixpQkF2Qlosa0NBQWtDLEVBQUUsMEJBQTBCLGFBRXpFLFlBQVk7UUFDWixXQUFXO1FBQ1gsbUJBQW1CO1FBQ25CLGNBQWM7UUFDZCw0QkFBNEI7UUFDNUIsY0FBYztRQUNkLFlBQVk7UUFDWix1QkFBdUI7UUFDdkIsY0FBYztRQUNkLGlCQUFpQjtRQUNqQixlQUFlO1FBQ2YsaUJBQWlCO1FBQ2pCLHdCQUF3QjtRQUN4QiwwQkFBMEI7UUFDMUIsNkJBQTZCO1FBQzdCLG1CQUFtQjtRQUNuQixlQUFlO1FBQ2YsYUFBYSxhQUVQLGtDQUFrQyxFQUFFLDBCQUEwQjtnSEFFL0Qsa0JBQWtCLFlBckJ2QixZQUFZO1FBQ1osV0FBVztRQUNYLG1CQUFtQjtRQUNuQixjQUFjO1FBQ2QsNEJBQTRCO1FBQzVCLGNBQWM7UUFDZCxZQUFZO1FBQ1osdUJBQXVCO1FBQ3ZCLGNBQWM7UUFDZCxpQkFBaUI7UUFDakIsZUFBZTtRQUNmLGlCQUFpQjtRQUNqQix3QkFBd0I7UUFDeEIsMEJBQTBCO1FBQzFCLDZCQUE2QjtRQUM3QixtQkFBbUI7UUFDbkIsZUFBZTtRQUNmLGFBQWE7MkZBSVIsa0JBQWtCO2tCQXhCOUIsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUUsQ0FBQyxrQ0FBa0MsRUFBRSwwQkFBMEIsQ0FBQztvQkFDOUUsT0FBTyxFQUFFO3dCQUNMLFlBQVk7d0JBQ1osV0FBVzt3QkFDWCxtQkFBbUI7d0JBQ25CLGNBQWM7d0JBQ2QsNEJBQTRCO3dCQUM1QixjQUFjO3dCQUNkLFlBQVk7d0JBQ1osdUJBQXVCO3dCQUN2QixjQUFjO3dCQUNkLGlCQUFpQjt3QkFDakIsZUFBZTt3QkFDZixpQkFBaUI7d0JBQ2pCLHdCQUF3Qjt3QkFDeEIsMEJBQTBCO3dCQUMxQiw2QkFBNkI7d0JBQzdCLG1CQUFtQjt3QkFDbkIsZUFBZTt3QkFDZixhQUFhO3FCQUNoQjtvQkFDRCxPQUFPLEVBQUUsQ0FBQyxrQ0FBa0MsRUFBRSwwQkFBMEIsQ0FBQztpQkFDNUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEZvcm1zTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgVHVpTGV0TW9kdWxlIH0gZnJvbSAnQHRhaWdhLXVpL2Nkayc7XG5pbXBvcnQgeyBUdWlUZXh0ZmllbGRDb250cm9sbGVyTW9kdWxlLCBUdWlMYWJlbE1vZHVsZSwgVHVpRXJyb3JNb2R1bGUsIFR1aURhdGFMaXN0TW9kdWxlLCBUdWlMb2FkZXJNb2R1bGUsIFR1aUJ1dHRvbk1vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jb3JlJztcbmltcG9ydCB7XG4gICAgVHVpSW5wdXRNb2R1bGUsXG4gICAgVHVpRmllbGRFcnJvclBpcGVNb2R1bGUsXG4gICAgVHVpQ29tYm9Cb3hNb2R1bGUsXG4gICAgVHVpRGF0YUxpc3RXcmFwcGVyTW9kdWxlLFxuICAgIFR1aUZpbHRlckJ5SW5wdXRQaXBlTW9kdWxlLFxuICAgIFR1aVN0cmluZ2lmeUNvbnRlbnRQaXBlTW9kdWxlLFxuICAgIFR1aUlucHV0UGhvbmVNb2R1bGUsXG59IGZyb20gJ0B0YWlnYS11aS9raXQnO1xuaW1wb3J0IHsgTWFza2l0b01vZHVsZSB9IGZyb20gJ0BtYXNraXRvL2FuZ3VsYXInO1xuaW1wb3J0IHsgU2NBZGRyZXNzZXNTZWxlY3Rpb25GaWVsZENvbXBvbmVudCB9IGZyb20gJy4vYWRkcmVzc2VzLXNlbGVjdGlvbi1maWVsZC9zYy1hZGRyZXNzZXMtc2VsZWN0aW9uLWZpZWxkLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTY1N1Z2dlc3Rpb25GaWVsZENvbXBvbmVudCB9IGZyb20gJy4vc3VnZ2VzdGlvbi1maWVsZC9zYy1zdWdnZXN0aW9uLWZpZWxkLmNvbXBvbmVudCc7XG5cbi8qKlxuICog0JzQvtC00YPQu9GMINC/0L7Qu9C10Lkg0LLQstC+0LTQsC5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtTY0FkZHJlc3Nlc1NlbGVjdGlvbkZpZWxkQ29tcG9uZW50LCBTY1N1Z2dlc3Rpb25GaWVsZENvbXBvbmVudF0sXG4gICAgaW1wb3J0czogW1xuICAgICAgICBDb21tb25Nb2R1bGUsXG4gICAgICAgIEZvcm1zTW9kdWxlLFxuICAgICAgICBSZWFjdGl2ZUZvcm1zTW9kdWxlLFxuICAgICAgICBUdWlJbnB1dE1vZHVsZSxcbiAgICAgICAgVHVpVGV4dGZpZWxkQ29udHJvbGxlck1vZHVsZSxcbiAgICAgICAgVHVpTGFiZWxNb2R1bGUsXG4gICAgICAgIFR1aUxldE1vZHVsZSxcbiAgICAgICAgVHVpRmllbGRFcnJvclBpcGVNb2R1bGUsXG4gICAgICAgIFR1aUVycm9yTW9kdWxlLFxuICAgICAgICBUdWlEYXRhTGlzdE1vZHVsZSxcbiAgICAgICAgVHVpTG9hZGVyTW9kdWxlLFxuICAgICAgICBUdWlDb21ib0JveE1vZHVsZSxcbiAgICAgICAgVHVpRGF0YUxpc3RXcmFwcGVyTW9kdWxlLFxuICAgICAgICBUdWlGaWx0ZXJCeUlucHV0UGlwZU1vZHVsZSxcbiAgICAgICAgVHVpU3RyaW5naWZ5Q29udGVudFBpcGVNb2R1bGUsXG4gICAgICAgIFR1aUlucHV0UGhvbmVNb2R1bGUsXG4gICAgICAgIFR1aUJ1dHRvbk1vZHVsZSxcbiAgICAgICAgTWFza2l0b01vZHVsZSxcbiAgICBdLFxuICAgIGV4cG9ydHM6IFtTY0FkZHJlc3Nlc1NlbGVjdGlvbkZpZWxkQ29tcG9uZW50LCBTY1N1Z2dlc3Rpb25GaWVsZENvbXBvbmVudF0sXG59KVxuZXhwb3J0IGNsYXNzIFNjRm9ybUZpZWxkc01vZHVsZSB7fVxuIl19
|
@@ -0,0 +1,111 @@
|
|
1
|
+
import { Component, Inject, Input, SkipSelf } from '@angular/core';
|
2
|
+
import { NgControl } from '@angular/forms';
|
3
|
+
import { ScISuggestionType } from '@snabcentr/client-core';
|
4
|
+
import { tuiIsPresent, tuiIsFalsy, tuiMarkControlAsTouchedAndValidate, tuiControlValue } from '@taiga-ui/cdk';
|
5
|
+
import { catchError, debounceTime, filter, map, of, share, startWith, switchMap, throwError } from 'rxjs';
|
6
|
+
import * as i0 from "@angular/core";
|
7
|
+
import * as i1 from "@snabcentr/client-core";
|
8
|
+
import * as i2 from "@angular/common";
|
9
|
+
import * as i3 from "@taiga-ui/cdk";
|
10
|
+
import * as i4 from "@taiga-ui/core";
|
11
|
+
import * as i5 from "@angular/forms";
|
12
|
+
/**
|
13
|
+
* Компонент подсказок для поля ввода.
|
14
|
+
*/
|
15
|
+
export class ScSuggestionFieldComponent {
|
16
|
+
/**
|
17
|
+
* Инициализирует экземпляр класса {@link ScSuggestionFieldComponent}.
|
18
|
+
*
|
19
|
+
* @param control Контрол поля ввода.
|
20
|
+
* @param suggestionService Сервис работы с API подсказок.
|
21
|
+
*/
|
22
|
+
constructor(control, suggestionService) {
|
23
|
+
this.control = control;
|
24
|
+
this.suggestionService = suggestionService;
|
25
|
+
/**
|
26
|
+
* Тип подсказок.
|
27
|
+
*/
|
28
|
+
this.type = ScISuggestionType.fio;
|
29
|
+
/**
|
30
|
+
* Перечисление типов подсказок.
|
31
|
+
*/
|
32
|
+
this.suggestionType = ScISuggestionType;
|
33
|
+
}
|
34
|
+
/** @inheritDoc */
|
35
|
+
ngOnInit() {
|
36
|
+
const request$ = tuiControlValue(this.control).pipe(debounceTime(300), filter(tuiIsPresent), filter((term) => term.length >= 3), switchMap((term) => this.suggestionService.getData(term, this.type).pipe(startWith(null))), catchError((error) => {
|
37
|
+
const errorResponse = error.error;
|
38
|
+
if (this.control.control) {
|
39
|
+
tuiMarkControlAsTouchedAndValidate(this.control.control);
|
40
|
+
this.control.control.setErrors({
|
41
|
+
serverResponse: [errorResponse.message || `Ошибка сервера: пожалуйста, перезагрузите страницу и/или выполните запрос позже.`],
|
42
|
+
});
|
43
|
+
}
|
44
|
+
return throwError(() => of([]));
|
45
|
+
}), share());
|
46
|
+
this.suggestions$ = request$?.pipe(filter(tuiIsPresent), startWith(null));
|
47
|
+
this.loading$ = request$.pipe(map(tuiIsFalsy));
|
48
|
+
}
|
49
|
+
/**
|
50
|
+
* Заполнение полей основе выбранной подсказки.
|
51
|
+
*
|
52
|
+
* @param suggestion Объект подсказки.
|
53
|
+
*/
|
54
|
+
onSelected(suggestion) {
|
55
|
+
switch (this.type) {
|
56
|
+
case ScISuggestionType.fio:
|
57
|
+
if ('fio' in suggestion) {
|
58
|
+
this.control.control?.patchValue(suggestion.fio);
|
59
|
+
}
|
60
|
+
break;
|
61
|
+
case ScISuggestionType.address:
|
62
|
+
if ('addressString' in suggestion) {
|
63
|
+
this.control.control?.patchValue(suggestion.addressString, { emitEvent: false });
|
64
|
+
}
|
65
|
+
break;
|
66
|
+
case ScISuggestionType.bank:
|
67
|
+
if ('inn' in suggestion && 'bic' in suggestion) {
|
68
|
+
this.control.control?.parent?.patchValue({
|
69
|
+
bankName: suggestion.name,
|
70
|
+
bic: suggestion.bic,
|
71
|
+
correspondentAccount: suggestion.correspondentAccount,
|
72
|
+
}, { emitEvent: false });
|
73
|
+
}
|
74
|
+
break;
|
75
|
+
case ScISuggestionType.organization:
|
76
|
+
if ('inn' in suggestion && !('bic' in suggestion)) {
|
77
|
+
this.control.control?.parent?.patchValue({
|
78
|
+
name: suggestion.name,
|
79
|
+
inn: suggestion.inn,
|
80
|
+
kpp: suggestion.kpp,
|
81
|
+
okpo: suggestion.okpo,
|
82
|
+
directorName: suggestion.directorName,
|
83
|
+
directorPosition: suggestion.directorPosition,
|
84
|
+
legalAddress: suggestion.legalAddress,
|
85
|
+
postalAddress: suggestion.postalAddress,
|
86
|
+
actualAddress: suggestion.actualAddress,
|
87
|
+
}, { emitEvent: false });
|
88
|
+
}
|
89
|
+
break;
|
90
|
+
case ScISuggestionType.email:
|
91
|
+
if ('email' in suggestion) {
|
92
|
+
this.control.control?.patchValue(suggestion.email, { emitEvent: false });
|
93
|
+
}
|
94
|
+
break;
|
95
|
+
}
|
96
|
+
}
|
97
|
+
}
|
98
|
+
ScSuggestionFieldComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScSuggestionFieldComponent, deps: [{ token: NgControl, skipSelf: true }, { token: i1.ScSuggestionService }], target: i0.ɵɵFactoryTarget.Component });
|
99
|
+
ScSuggestionFieldComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: { type: "type" }, ngImport: i0, template: "<tui-loader size=\"m\" [overlay]=\"true\" [showLoader]=\"!!(loading$ | async)\">\n <tui-data-list *tuiLet=\"suggestions$ | async as options\">\n <button *ngFor=\"let option of options\" tuiOption (click)=\"onSelected(option)\" class=\"flex-col !items-start\">\n <ng-container *tuiLet=\"$any(option) as option\">\n <ng-container [ngSwitch]=\"type\">\n <ng-container *ngSwitchCase=\"suggestionType.fio\">\n <span class=\"text-tui-link\">{{ option.fio }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.organization\">\n <span class=\"text-tui-link\">\n {{ option.name }}\n <span class=\"ml-2 text-tui-text-02\">\u0438\u043D\u043D: {{ option.inn }}</span>\n </span>\n <span class=\"text-tui-text-02\">{{ option?.legalAddress }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.bank\">\n <span class=\"text-tui-link\">{{ option.name }}</span>\n <span class=\"text-tui-text-02\">{{ option.bic }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.email\">\n <span class=\"text-tui-link\">{{ option.email }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.address\">\n <span class=\"text-tui-link\">{{ option.addressString }}</span>\n </ng-container>\n </ng-container>\n </ng-container>\n </button>\n </tui-data-list>\n</tui-loader>\n", dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i2.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i3.TuiLetDirective, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "component", type: i4.TuiDataListComponent, selector: "tui-data-list", inputs: ["role", "emptyContent", "size"] }, { kind: "component", type: i4.TuiOptionComponent, selector: "button[tuiOption], a[tuiOption]", inputs: ["size", "role", "disabled", "value"] }, { kind: "component", type: i4.TuiLoaderComponent, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }] });
|
100
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScSuggestionFieldComponent, decorators: [{
|
101
|
+
type: Component,
|
102
|
+
args: [{ selector: 'sc-suggestion-field', template: "<tui-loader size=\"m\" [overlay]=\"true\" [showLoader]=\"!!(loading$ | async)\">\n <tui-data-list *tuiLet=\"suggestions$ | async as options\">\n <button *ngFor=\"let option of options\" tuiOption (click)=\"onSelected(option)\" class=\"flex-col !items-start\">\n <ng-container *tuiLet=\"$any(option) as option\">\n <ng-container [ngSwitch]=\"type\">\n <ng-container *ngSwitchCase=\"suggestionType.fio\">\n <span class=\"text-tui-link\">{{ option.fio }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.organization\">\n <span class=\"text-tui-link\">\n {{ option.name }}\n <span class=\"ml-2 text-tui-text-02\">\u0438\u043D\u043D: {{ option.inn }}</span>\n </span>\n <span class=\"text-tui-text-02\">{{ option?.legalAddress }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.bank\">\n <span class=\"text-tui-link\">{{ option.name }}</span>\n <span class=\"text-tui-text-02\">{{ option.bic }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.email\">\n <span class=\"text-tui-link\">{{ option.email }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.address\">\n <span class=\"text-tui-link\">{{ option.addressString }}</span>\n </ng-container>\n </ng-container>\n </ng-container>\n </button>\n </tui-data-list>\n</tui-loader>\n" }]
|
103
|
+
}], ctorParameters: function () { return [{ type: i5.NgControl, decorators: [{
|
104
|
+
type: SkipSelf
|
105
|
+
}, {
|
106
|
+
type: Inject,
|
107
|
+
args: [NgControl]
|
108
|
+
}] }, { type: i1.ScSuggestionService }]; }, propDecorators: { type: [{
|
109
|
+
type: Input
|
110
|
+
}] } });
|
111
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-suggestion-field.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/form-fields/suggestion-field/sc-suggestion-field.component.ts","../../../../../projects/client-ui/form-fields/suggestion-field/sc-suggestion-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import { HttpErrorResponse } from '@angular/common/http';\nimport { Component, Inject, Input, OnInit, SkipSelf } from '@angular/core';\nimport { FormGroup, NgControl } from '@angular/forms';\nimport { ScISuggestionType, ScSuggestion, ScSuggestionService } from '@snabcentr/client-core';\nimport { tuiIsPresent, tuiIsFalsy, tuiMarkControlAsTouchedAndValidate, tuiControlValue } from '@taiga-ui/cdk';\nimport { ApiErrorResponse } from '../../auth';\nimport { Observable, catchError, debounceTime, filter, map, of, share, startWith, switchMap, throwError } from 'rxjs';\n\n/**\n * Компонент подсказок для поля ввода.\n */\n@Component({\n    selector: 'sc-suggestion-field',\n    templateUrl: './sc-suggestion-field.component.html',\n})\nexport class ScSuggestionFieldComponent implements OnInit {\n    /**\n     * Тип подсказок.\n     */\n    @Input()\n    public type: ScISuggestionType = ScISuggestionType.fio;\n\n    /**\n     * {@link Observable} подсказок.\n     */\n    public suggestions$?: Observable<ScSuggestion[] | null>;\n\n    /**\n     * {@link Observable} состояние запроса подсказок.\n     */\n    public loading$?: Observable<boolean>;\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    public readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScSuggestionFieldComponent}.\n     *\n     * @param control Контрол поля ввода.\n     * @param suggestionService Сервис работы с API подсказок.\n     */\n    public constructor(\n        @SkipSelf()\n        @Inject(NgControl)\n        public readonly control: NgControl,\n        private readonly suggestionService: ScSuggestionService\n    ) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        const request$ = tuiControlValue<string | null>(this.control).pipe(\n            debounceTime(300),\n            filter(tuiIsPresent),\n            filter((term) => term.length >= 3),\n            switchMap((term: string) => this.suggestionService.getData<ScSuggestion>(term, this.type).pipe(startWith(null))),\n            catchError((error: HttpErrorResponse) => {\n                const errorResponse = error.error as ApiErrorResponse;\n\n                if (this.control.control) {\n                    tuiMarkControlAsTouchedAndValidate(this.control.control);\n\n                    this.control.control.setErrors({\n                        serverResponse: [errorResponse.message || `Ошибка сервера: пожалуйста, перезагрузите страницу и/или выполните запрос позже.`],\n                    });\n                }\n\n                return throwError(() => of([]));\n            }),\n            share()\n        );\n\n        this.suggestions$ = request$?.pipe(filter(tuiIsPresent), startWith(null));\n        this.loading$ = request$.pipe(map(tuiIsFalsy));\n    }\n\n    /**\n     * Заполнение полей основе выбранной подсказки.\n     *\n     * @param suggestion Объект подсказки.\n     */\n    public onSelected(suggestion: ScSuggestion): void {\n        switch (this.type) {\n            case ScISuggestionType.fio:\n                if ('fio' in suggestion) {\n                    this.control.control?.patchValue(suggestion.fio);\n                }\n                break;\n            case ScISuggestionType.address:\n                if ('addressString' in suggestion) {\n                    this.control.control?.patchValue(suggestion.addressString, { emitEvent: false });\n                }\n                break;\n            case ScISuggestionType.bank:\n                if ('inn' in suggestion && 'bic' in suggestion) {\n                    (this.control.control?.parent as FormGroup)?.patchValue(\n                        {\n                            bankName: suggestion.name,\n                            bic: suggestion.bic,\n                            correspondentAccount: suggestion.correspondentAccount,\n                        },\n                        { emitEvent: false }\n                    );\n                }\n                break;\n            case ScISuggestionType.organization:\n                if ('inn' in suggestion && !('bic' in suggestion)) {\n                    (this.control.control?.parent as FormGroup)?.patchValue(\n                        {\n                            name: suggestion.name,\n                            inn: suggestion.inn,\n                            kpp: suggestion.kpp,\n                            okpo: suggestion.okpo,\n                            directorName: suggestion.directorName,\n                            directorPosition: suggestion.directorPosition,\n                            legalAddress: suggestion.legalAddress,\n                            postalAddress: suggestion.postalAddress,\n                            actualAddress: suggestion.actualAddress,\n                        },\n                        { emitEvent: false }\n                    );\n                }\n                break;\n            case ScISuggestionType.email:\n                if ('email' in suggestion) {\n                    this.control.control?.patchValue(suggestion.email, { emitEvent: false });\n                }\n                break;\n        }\n    }\n}\n","<tui-loader size=\"m\" [overlay]=\"true\" [showLoader]=\"!!(loading$ | async)\">\n    <tui-data-list *tuiLet=\"suggestions$ | async as options\">\n        <button *ngFor=\"let option of options\" tuiOption (click)=\"onSelected(option)\" class=\"flex-col !items-start\">\n            <ng-container *tuiLet=\"$any(option) as option\">\n                <ng-container [ngSwitch]=\"type\">\n                    <ng-container *ngSwitchCase=\"suggestionType.fio\">\n                        <span class=\"text-tui-link\">{{ option.fio }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.organization\">\n                        <span class=\"text-tui-link\">\n                            {{ option.name }}\n                            <span class=\"ml-2 text-tui-text-02\">инн: {{ option.inn }}</span>\n                        </span>\n                        <span class=\"text-tui-text-02\">{{ option?.legalAddress }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.bank\">\n                        <span class=\"text-tui-link\">{{ option.name }}</span>\n                        <span class=\"text-tui-text-02\">{{ option.bic }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.email\">\n                        <span class=\"text-tui-link\">{{ option.email }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.address\">\n                        <span class=\"text-tui-link\">{{ option.addressString }}</span>\n                    </ng-container>\n                </ng-container>\n            </ng-container>\n        </button>\n    </tui-data-list>\n</tui-loader>\n"]}
|