@snabcentr/client-ui 0.23.0 → 0.26.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (59) hide show
  1. package/auth/index.d.ts +2 -1
  2. package/auth/interfaces/{ApiErrorResponse.d.ts → api-error-response.d.ts} +1 -3
  3. package/auth/interfaces/index.d.ts +3 -0
  4. package/auth/interfaces/sc-bank-account-form-group.d.ts +11 -0
  5. package/auth/interfaces/sc-contact-form-group.d.ts +10 -0
  6. package/auth/sc-auth.module.d.ts +13 -8
  7. package/auth/sign-up-form/sc-sign-up-form.component.d.ts +141 -0
  8. package/contacts/add-contact-dialog/sc-add-contact-dialog.component.d.ts +3 -3
  9. package/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.d.ts +2 -2
  10. package/contragents/new-contragent-form/sc-new-contragent-form.component.d.ts +22 -15
  11. package/directives/carousel-item-hidden/sc-carousel-item-hidden.directive.d.ts +4 -4
  12. package/directives/index.d.ts +2 -0
  13. package/directives/next-input-focus/sc-next-input-focus.directive.d.ts +20 -0
  14. package/directives/next-input-focus/sc-next-input-focus.module.d.ts +10 -0
  15. package/esm2020/auth/index.mjs +3 -2
  16. package/esm2020/auth/interfaces/api-error-response.mjs +2 -0
  17. package/esm2020/auth/interfaces/index.mjs +4 -0
  18. package/esm2020/auth/interfaces/sc-bank-account-form-group.mjs +2 -0
  19. package/esm2020/auth/interfaces/sc-contact-form-group.mjs +2 -0
  20. package/esm2020/auth/sc-auth.module.mjs +51 -13
  21. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +13 -7
  22. package/esm2020/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +13 -7
  23. package/esm2020/auth/sign-up-form/sc-sign-up-form.component.mjs +264 -0
  24. package/esm2020/catalog/price-card/sc-price-card.component.mjs +1 -1
  25. package/esm2020/catalog/price-history/sc-price-history.component.mjs +1 -1
  26. package/esm2020/contacts/add-contact-dialog/sc-add-contact-dialog.component.mjs +20 -12
  27. package/esm2020/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.mjs +21 -13
  28. package/esm2020/contragents/add-contragent-dialog/sc-add-contragent-dialog.component.mjs +19 -17
  29. package/esm2020/contragents/new-contragent-form/sc-new-contragent-form.component.mjs +29 -20
  30. package/esm2020/delivery-address/add-delivery-address-dialog/sc-add-delivery-address-dialog.component.mjs +1 -1
  31. package/esm2020/directives/carousel-item-hidden/sc-carousel-item-hidden.directive.mjs +2 -2
  32. package/esm2020/directives/carousel-item-hidden/sc-carousel-item-hidden.module.mjs +2 -2
  33. package/esm2020/directives/index.mjs +3 -1
  34. package/esm2020/directives/next-input-focus/sc-next-input-focus.directive.mjs +40 -0
  35. package/esm2020/directives/next-input-focus/sc-next-input-focus.module.mjs +19 -0
  36. package/esm2020/files/directives/tree-top.directive.mjs +10 -10
  37. package/esm2020/form-fields/suggestion-field/sc-suggestion-field.component.mjs +1 -1
  38. package/esm2020/user/index.mjs +2 -1
  39. package/esm2020/user/sc-user.module.mjs +19 -12
  40. package/esm2020/user/user-managers/sc-user-managers.component.mjs +38 -0
  41. package/esm2020/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.mjs +18 -12
  42. package/esm2020/validators/index.mjs +2 -2
  43. package/esm2020/validators/sc-password-confirm-matching-validator.mjs +17 -0
  44. package/esm2020/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +39 -27
  45. package/fesm2015/snabcentr-client-ui.mjs +2820 -2401
  46. package/fesm2015/snabcentr-client-ui.mjs.map +1 -1
  47. package/fesm2020/snabcentr-client-ui.mjs +2819 -2402
  48. package/fesm2020/snabcentr-client-ui.mjs.map +1 -1
  49. package/package.json +1 -1
  50. package/styles/tailwind/tailwind.scss +17 -0
  51. package/user/index.d.ts +1 -0
  52. package/user/sc-user.module.d.ts +9 -8
  53. package/user/user-managers/sc-user-managers.component.d.ts +26 -0
  54. package/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.d.ts +2 -2
  55. package/validators/index.d.ts +1 -1
  56. package/validators/{scPasswordConfirmMatchingValidator.d.ts → sc-password-confirm-matching-validator.d.ts} +1 -1
  57. package/verification/verification-phone-check-form/sc-verification-phone-check-form.component.d.ts +17 -17
  58. package/esm2020/auth/interfaces/ApiErrorResponse.mjs +0 -2
  59. package/esm2020/validators/scPasswordConfirmMatchingValidator.mjs +0 -17
@@ -172,4 +172,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
172
172
  type: HostBinding,
173
173
  args: ['attr.data-size']
174
174
  }] } });
175
- //# 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":["/* eslint-disable no-underscore-dangle */\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, Inject, inject, Input, Output } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport {\n    SC_PATH_IMAGE_NOT_FOUND,\n    SC_URLS,\n    ScAuthService,\n    ScCartItem,\n    ScImageHelper,\n    ScIUrls,\n    ScIWarehouse,\n    ScProduct,\n    ScWarehouseService,\n    SEARCH_TERM,\n    SEARCH_TERM_PROVIDERS,\n} from '@snabcentr/client-core';\nimport { TUI_IS_MOBILE } from '@taiga-ui/cdk';\nimport { TuiSizeS, TuiSizeXS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\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    providers: [SEARCH_TERM_PROVIDERS],\n})\nexport class ScPriceCardComponent {\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    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    @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     * Позиция товара в корзине.\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 search$ Провайдер поиска терма.\n     * @param cdr Объект для работы с обнаружением изменений.\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        @Inject(SEARCH_TERM) public readonly search$: Observable<string>,\n        private readonly cdr: ChangeDetectorRef,\n        private readonly imageHelper: ScImageHelper\n    ) {}\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    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\n    *ngIf=\"product; else skeleton\"\n    class=\"relative grid h-[19rem] w-auto min-w-40 rounded-xl p-2 text-xs shadow-md\"\n>\n    <div class=\"overflow-hidden\">\n        <img\n            (click)=\"clickCardEvent.emit()\"\n            [src]=\"getCardImagePreview()\"\n            [alt]=\"product.name\"\n            class=\"max-h-full w-full rounded-xl object-cover\"\n        />\n    </div>\n    <sc-favorite-btn\n        *ngIf=\"authStatus$ | async\"\n        class=\"absolute left-0 top-0\"\n        (clickEvent)=\"clickFavoriteEvent.emit()\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"product.isFavorite\"\n        [disabled]=\"!!product.primaryCategory?.isFavorite\"\n    ></sc-favorite-btn>\n    <tui-svg\n        *ngIf=\"product.isPreviouslyOrdered\"\n        src=\"scIconStar\"\n        class=\"absolute right-0 top-0 text-red-700\"\n    ></tui-svg>\n\n    <a\n        class=\"m-0 p-0\"\n        tuiLink\n        iconAlign=\"left\"\n        (click)=\"clickCardEvent.emit()\"\n    >\n        <p\n            class=\"line-clamp-3 text-sm font-bold\"\n            style=\"word-break: break-word\"\n            [tuiHighlight]=\"(search$ | async) ?? ''\"\n        >\n            {{ product.name }}\n        </p>\n    </a>\n    <p\n        *ngIf=\"product?.pack\"\n        class=\"text-tui-text-02\"\n    >\n        Норма упаковки: {{ product.pack }}\n    </p>\n\n    <p\n        [tuiHighlight]=\"(search$ | async) ?? ''\"\n        class=\"text-tui-text-02\"\n    >\n        Артикул: {{ product.code }}\n    </p>\n    <p\n        *ngIf=\"(authStatus$ | async) && product?.costDate\"\n        class=\"text-tui-text-02\"\n    >\n        Дата: {{ product.costDate }}\n    </p>\n    <div\n        *ngIf=\"warehouseSelect$ | async as warehouseSelect\"\n        class=\"flex flex-col\"\n    >\n        <span\n            *ngIf=\"product.discount && !isMobile\"\n            class=\"flex items-center text-tui-text-02\"\n        >\n            <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n            <span class=\"font-bold text-tui-success-fill\"> -{{ product.discount.percent }}% </span>\n            <tui-svg\n                src=\"tuiIconInfoLarge\"\n                [tuiHint]=\"discountHint\"\n                [tuiHintShowDelay]=\"100\"\n                tuiHintDirection=\"top\"\n                class=\"!h-4 !text-xs text-black\"\n            ></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\n            [class.text-tui-text-02]=\"!product.isWarehouseStockExist(warehouseSelect.id)\"\n            class=\"cost font-bold\"\n            >{{ product.costRubString }}</span\n        >\n        <span\n            *ngIf=\"!product.priceInRub\"\n            class=\"hidden text-xs font-bold text-tui-text-02 group-hover:block\"\n            >{{ product.costString }}</span\n        >\n        <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n    </div>\n    <div\n        *ngIf=\"!showQuantityControl\"\n        class=\"mt-1 flex max-w-full items-center justify-stretch gap-2\"\n    >\n        <button\n            tuiButton\n            (click)=\"clickAddToCartEvent.emit(product)\"\n            [showLoader]=\"quantityShowLoader\"\n            [size]=\"size\"\n            class=\"grow\"\n        >\n            <tui-svg\n                src=\"scIconCart\"\n                class=\"!h-4 !text-xs\"\n            >\n            </tui-svg>\n            В корзину\n        </button>\n        <div\n            *ngIf=\"cartItem\"\n            class=\"flex justify-center\"\n        >\n            <tui-svg\n                src=\"tuiIconCheck\"\n                class=\"!h-5 !w-5 rounded-md bg-tui-primary text-black\"\n            ></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=\"mt-2 w-full\"\n        (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n    ></sc-input-quantity>\n</div>\n\n<ng-template #skeleton>\n    <div class=\"card-wrapper flex flex-col overflow-hidden rounded-xl bg-white shadow-sc-1\">\n        <!-- Изображение товара -->\n        <div class=\"h-3/5 w-full rounded-t bg-tui-base-02\"></div>\n        <!-- Краткая информация о товаре -->\n        <div class=\"flex grow flex-col gap-2.5 rounded-b bg-white p-5\">\n            <div class=\"h-4 w-full rounded bg-tui-base-02\"></div>\n            <div class=\"h-4 w-3/5 rounded bg-tui-base-02\"></div>\n            <div class=\"h-4 w-full rounded bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
175
+ //# 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":["/* eslint-disable no-underscore-dangle */\n\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, HostBinding, Inject, inject, Input, Output } from '@angular/core';\nimport { FormControl } from '@angular/forms';\nimport {\n    SC_PATH_IMAGE_NOT_FOUND,\n    SC_URLS,\n    ScAuthService,\n    ScCartItem,\n    ScImageHelper,\n    ScIUrls,\n    ScIWarehouse,\n    ScProduct,\n    ScWarehouseService,\n    SEARCH_TERM,\n    SEARCH_TERM_PROVIDERS,\n} from '@snabcentr/client-core';\nimport { TUI_IS_MOBILE } from '@taiga-ui/cdk';\nimport { TuiSizeS, TuiSizeXS } from '@taiga-ui/core';\nimport { Observable } from 'rxjs';\n\nimport { UnitsHelper } from '../../helpers/sc-units-helper';\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    providers: [SEARCH_TERM_PROVIDERS],\n})\nexport class ScPriceCardComponent {\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    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    @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     * Позиция товара в корзине.\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 search$ Провайдер поиска терма.\n     * @param cdr Объект для работы с обнаружением изменений.\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        @Inject(SEARCH_TERM) public readonly search$: Observable<string>,\n        private readonly cdr: ChangeDetectorRef,\n        private readonly imageHelper: ScImageHelper\n    ) {}\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    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\n    *ngIf=\"product; else skeleton\"\n    class=\"relative grid h-[19rem] w-auto min-w-40 rounded-xl p-2 text-xs shadow-md\"\n>\n    <div class=\"overflow-hidden\">\n        <img\n            (click)=\"clickCardEvent.emit()\"\n            [src]=\"getCardImagePreview()\"\n            [alt]=\"product.name\"\n            class=\"max-h-full w-full rounded-xl object-cover\"\n        />\n    </div>\n    <sc-favorite-btn\n        *ngIf=\"authStatus$ | async\"\n        class=\"absolute left-0 top-0\"\n        (clickEvent)=\"clickFavoriteEvent.emit()\"\n        [showLoader]=\"favoriteShowLoader\"\n        [isFavorite]=\"product.isFavorite\"\n        [disabled]=\"!!product.primaryCategory?.isFavorite\"\n    ></sc-favorite-btn>\n    <tui-svg\n        *ngIf=\"product.isPreviouslyOrdered\"\n        src=\"scIconStar\"\n        class=\"absolute right-0 top-0 text-red-700\"\n    ></tui-svg>\n\n    <a\n        class=\"m-0 p-0\"\n        tuiLink\n        iconAlign=\"left\"\n        (click)=\"clickCardEvent.emit()\"\n    >\n        <p\n            class=\"line-clamp-3 text-sm font-bold\"\n            style=\"word-break: break-word\"\n            [tuiHighlight]=\"(search$ | async) ?? ''\"\n        >\n            {{ product.name }}\n        </p>\n    </a>\n    <p\n        *ngIf=\"product?.pack\"\n        class=\"text-tui-text-02\"\n    >\n        Норма упаковки: {{ product.pack }}\n    </p>\n\n    <p\n        [tuiHighlight]=\"(search$ | async) ?? ''\"\n        class=\"text-tui-text-02\"\n    >\n        Артикул: {{ product.code }}\n    </p>\n    <p\n        *ngIf=\"(authStatus$ | async) && product?.costDate\"\n        class=\"text-tui-text-02\"\n    >\n        Дата: {{ product.costDate }}\n    </p>\n    <div\n        *ngIf=\"warehouseSelect$ | async as warehouseSelect\"\n        class=\"flex flex-col\"\n    >\n        <span\n            *ngIf=\"product.discount && !isMobile\"\n            class=\"flex items-center text-tui-text-02\"\n        >\n            <span class=\"line-through\">{{ product.discountCostString }}</span> &nbsp;\n            <span class=\"font-bold text-tui-success-fill\"> -{{ product.discount.percent }}% </span>\n            <tui-svg\n                src=\"tuiIconInfoLarge\"\n                [tuiHint]=\"discountHint\"\n                [tuiHintShowDelay]=\"100\"\n                tuiHintDirection=\"top\"\n                class=\"!h-4 !text-xs text-black\"\n            ></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\n            [class.text-tui-text-02]=\"!product.isWarehouseStockExist(warehouseSelect.id)\"\n            class=\"cost font-bold\"\n            >{{ product.costRubString }}</span\n        >\n        <span\n            *ngIf=\"!product.priceInRub\"\n            class=\"hidden text-xs font-bold text-tui-text-02 group-hover:block\"\n            >{{ product.costString }}</span\n        >\n        <sc-price-warehouse-stock [product]=\"product\"></sc-price-warehouse-stock>\n    </div>\n    <div\n        *ngIf=\"!showQuantityControl\"\n        class=\"mt-1 flex max-w-full items-center justify-stretch gap-2\"\n    >\n        <button\n            tuiButton\n            (click)=\"clickAddToCartEvent.emit(product)\"\n            [showLoader]=\"quantityShowLoader\"\n            [size]=\"size\"\n            class=\"grow\"\n        >\n            <tui-svg\n                src=\"scIconCart\"\n                class=\"!h-4 !text-xs\"\n            >\n            </tui-svg>\n            В корзину\n        </button>\n        <div\n            *ngIf=\"cartItem\"\n            class=\"flex justify-center\"\n        >\n            <tui-svg\n                src=\"tuiIconCheck\"\n                class=\"!h-5 !w-5 rounded-md bg-tui-primary text-black\"\n            ></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=\"mt-2 w-full\"\n        (keydown.enter)=\"inputQuantity.nativeFocusableElement?.blur()\"\n    ></sc-input-quantity>\n</div>\n\n<ng-template #skeleton>\n    <div class=\"card-wrapper flex flex-col overflow-hidden rounded-xl bg-white shadow-sc-1\">\n        <!-- Изображение товара -->\n        <div class=\"h-3/5 w-full rounded-t bg-tui-base-02\"></div>\n        <!-- Краткая информация о товаре -->\n        <div class=\"flex grow flex-col gap-2.5 rounded-b bg-white p-5\">\n            <div class=\"h-4 w-full rounded bg-tui-base-02\"></div>\n            <div class=\"h-4 w-3/5 rounded bg-tui-base-02\"></div>\n            <div class=\"h-4 w-full rounded bg-tui-base-02\"></div>\n        </div>\n    </div>\n</ng-template>\n"]}
@@ -123,4 +123,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
123
123
  }] }]; }, propDecorators: { product: [{
124
124
  type: Input
125
125
  }], computeLabels$: [], computeValue: [] } });
126
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-history.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.ts","../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,lodash/prefer-lodash-method */\nimport { ChangeDetectionStrategy, Component, Inject, Input, OnInit, Optional } from '@angular/core';\nimport { ScCatalogService, ScProduct, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { TuiDay, TuiDayRange, tuiIsFalsy, tuiIsPresent, TuiMonth, tuiPure, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TUI_MONTHS, TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { filter, map, Observable, shareReplay, startWith, tap } from 'rxjs';\n\n/**\n * График истории цен товара или услуги.\n */\n@Component({\n    selector: 'sc-price-history',\n    templateUrl: './sc-price-history.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceHistoryComponent implements OnInit {\n    /**\n     * {@link Observable} временного промежутка истории цен.\n     */\n    public range$: Observable<TuiDayRange>;\n\n    /**\n     * Товар или услуга, для которого необходимо отобразить историю цен.\n     */\n    @Input() public product: ScProduct;\n\n    /**\n     * {@link Observable} истории цен.\n     */\n    public history$?: Observable<ReadonlyArray<[TuiDay, number]> | null>;\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных истории цен.\n     */\n    public loadingHistory$?: Observable<boolean>;\n\n    /**\n     * Максимальная цена товара.\n     */\n    public priceInfo$?: Observable<{\n        min: number;\n        max: number;\n    }>;\n\n    /**\n     * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.\n     */\n    public readonly xStringify$: Observable<TuiStringHandler<TuiDay>> = this.months$.pipe(\n        map(\n            (months) =>\n                ({ month, day }) =>\n                    // eslint-disable-next-line security/detect-object-injection\n                    `${months[month]}, ${day}`\n        )\n    );\n\n    /**\n     * Функция для преобразования значения number в строку в подсказке по оси Y.\n     *\n     * @param value Значение оси Y.\n     */\n    public readonly yStringify: TuiStringHandler<number> = (value) => `${value.toLocaleString('ru-RU')} ₽`;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.\n     *\n     * @param catalogService Сервис для работы с каталогом.\n     * @param months$ Перечисление месяцев.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly catalogService: ScCatalogService,\n        @Inject(TUI_MONTHS) private readonly months$: Observable<readonly string[]>,\n        private readonly userMetrikaService: ScUserMetrikaService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        private readonly context?: TuiDialogContext<void, { product: ScProduct }>\n    ) {\n        if (context) {\n            this.product = context.data.product;\n        }\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.history$ = this.catalogService.getPriceHistory$(this.product).pipe(\n            map((history) =>\n                Object.keys(history)\n                    .map((item): [TuiDay, number] => [TuiDay.normalizeParse(item, 'DMY'), history[String(item)].cost])\n                    .sort((a, b) => (a[0] > b[0] ? 1 : -1))\n            ),\n            startWith(null),\n            shareReplay({ bufferSize: 1, refCount: true })\n        );\n\n        this.loadingHistory$ = this.history$.pipe(map(tuiIsFalsy));\n\n        this.priceInfo$ = this.history$.pipe(\n            filter(tuiIsPresent),\n            tap(() => {\n                this.userMetrikaService.emitUserMetrikaEvent({\n                    target: ScUserMetrikaGoalsEnum.productPriceHistoryShow,\n                    params: {\n                        product_id: this.product.id,\n                    },\n                });\n            }),\n            filter((history) => history.length > 1),\n            map((history) => history.map((item) => item[1])),\n            map((values) => ({\n                min: Math.min(...values),\n                max: Math.max(...values),\n            }))\n        );\n\n        this.range$ = this.history$.pipe(\n            filter(tuiIsPresent),\n            filter((history) => history.length > 1),\n            map((value: ReadonlyArray<[TuiDay, number]>) => ({ from: value.at(0)?.[0], to: value.at(-1)?.[0] })),\n            filter((range): range is { from: TuiDay; to: TuiDay } => !!range.from && !!range.to),\n            map((range) => {\n                return new TuiDayRange(range.from, range.to);\n            })\n        );\n    }\n\n    /**\n     * Вычисляет подписи даты к оси X.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     */\n    @tuiPure\n    public computeLabels$({ from, to }: TuiDayRange): Observable<readonly string[]> {\n        return this.months$.pipe(map((months) => Array.from({ length: TuiMonth.lengthBetween(from, to) + 1 }, (_, index) => months[from.append({ month: index }).month])));\n    }\n\n    /**\n     * Вычисляет данные для отображения на графике.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     * @param history История цен.\n     */\n    @tuiPure\n    public computeValue({ from, to }: TuiDayRange, history: ReadonlyArray<[TuiDay, number]>): ReadonlyArray<[TuiDay, number]> {\n        return (\n            Array.from({ length: TuiDay.lengthBetween(from, to) + 1 })\n                .fill(0)\n                // eslint-disable-next-line unicorn/no-array-reduce\n                .reduce<ReadonlyArray<[TuiDay, number]>>((array, _, index) => {\n                    const data = from.append({ day: index });\n                    return [\n                        ...array,\n                        [\n                            data,\n                            (history.find((valueHistory, indexHistory, arrayHistory) => data.daySame(valueHistory[0]) || data.dayBefore(arrayHistory[indexHistory + 1][0])) ??\n                                history[0])[1],\n                        ],\n                    ];\n                }, [])\n        );\n    }\n}\n","<tui-loader [showLoader]=\"!!(loadingHistory$ | async)\"> </tui-loader>\n\n<div\n    *ngIf=\"(loadingHistory$ | async) === false\"\n    class=\"flex flex-col items-center\"\n>\n    <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n        <div class=\"w-174 mb-1 text-end text-lg font-bold\">от {{ priceInfo.min.toLocaleString() }} ₽ до {{ priceInfo.max.toLocaleString() }} ₽</div>\n\n        <ng-container *ngIf=\"history$ | async as history\">\n            <div\n                *ngIf=\"range$ | async as range\"\n                class=\"relative h-56 w-full p-5 text-tui-primary\"\n            >\n                <tui-axes\n                    *ngIf=\"computeLabels$(range) | async as labels\"\n                    class=\"h-full\"\n                    [axisXLabels]=\"labels\"\n                    [horizontalLines]=\"4\"\n                    [verticalLines]=\"labels.length\"\n                >\n                    <tui-line-days-chart\n                        class=\"chart\"\n                        [height]=\"priceInfo.max\"\n                        [smoothingFactor]=\"10\"\n                        [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n                        [value]=\"computeValue(range, history)\"\n                        [xStringify]=\"xStringify$ | async\"\n                        [yStringify]=\"yStringify\"\n                        [hintContent]=\"hint\"\n                    ></tui-line-days-chart>\n                </tui-axes>\n            </div>\n        </ng-container>\n    </ng-container>\n    <ng-template #isNotExist>\n        <p class=\"text-base font-bold\">Цена на товар не изменялась</p>\n    </ng-template>\n    <ng-template\n        #hint\n        let-data\n    >\n        <div class=\"font-bold\">{{ data[1].toLocaleString() }} ₽</div>\n        <div>{{ data[0] }}</div>\n    </ng-template>\n</div>\n"]}
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-history.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.ts","../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,lodash/prefer-lodash-method */\n\nimport { ChangeDetectionStrategy, Component, Inject, Input, OnInit, Optional } from '@angular/core';\nimport { ScCatalogService, ScProduct, ScUserMetrikaGoalsEnum, ScUserMetrikaService } from '@snabcentr/client-core';\nimport { TuiDay, TuiDayRange, tuiIsFalsy, tuiIsPresent, TuiMonth, tuiPure, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TUI_MONTHS, TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { filter, map, Observable, shareReplay, startWith, tap } from 'rxjs';\n\n/**\n * График истории цен товара или услуги.\n */\n@Component({\n    selector: 'sc-price-history',\n    templateUrl: './sc-price-history.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceHistoryComponent implements OnInit {\n    /**\n     * {@link Observable} временного промежутка истории цен.\n     */\n    public range$: Observable<TuiDayRange>;\n\n    /**\n     * Товар или услуга, для которого необходимо отобразить историю цен.\n     */\n    @Input() public product: ScProduct;\n\n    /**\n     * {@link Observable} истории цен.\n     */\n    public history$?: Observable<ReadonlyArray<[TuiDay, number]> | null>;\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных истории цен.\n     */\n    public loadingHistory$?: Observable<boolean>;\n\n    /**\n     * Максимальная цена товара.\n     */\n    public priceInfo$?: Observable<{\n        min: number;\n        max: number;\n    }>;\n\n    /**\n     * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.\n     */\n    public readonly xStringify$: Observable<TuiStringHandler<TuiDay>> = this.months$.pipe(\n        map(\n            (months) =>\n                ({ month, day }) =>\n                    // eslint-disable-next-line security/detect-object-injection\n                    `${months[month]}, ${day}`\n        )\n    );\n\n    /**\n     * Функция для преобразования значения number в строку в подсказке по оси Y.\n     *\n     * @param value Значение оси Y.\n     */\n    public readonly yStringify: TuiStringHandler<number> = (value) => `${value.toLocaleString('ru-RU')} ₽`;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.\n     *\n     * @param catalogService Сервис для работы с каталогом.\n     * @param months$ Перечисление месяцев.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly catalogService: ScCatalogService,\n        @Inject(TUI_MONTHS) private readonly months$: Observable<readonly string[]>,\n        private readonly userMetrikaService: ScUserMetrikaService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        private readonly context?: TuiDialogContext<void, { product: ScProduct }>\n    ) {\n        if (context) {\n            this.product = context.data.product;\n        }\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.history$ = this.catalogService.getPriceHistory$(this.product).pipe(\n            map((history) =>\n                Object.keys(history)\n                    .map((item): [TuiDay, number] => [TuiDay.normalizeParse(item, 'DMY'), history[String(item)].cost])\n                    .sort((a, b) => (a[0] > b[0] ? 1 : -1))\n            ),\n            startWith(null),\n            shareReplay({ bufferSize: 1, refCount: true })\n        );\n\n        this.loadingHistory$ = this.history$.pipe(map(tuiIsFalsy));\n\n        this.priceInfo$ = this.history$.pipe(\n            filter(tuiIsPresent),\n            tap(() => {\n                this.userMetrikaService.emitUserMetrikaEvent({\n                    target: ScUserMetrikaGoalsEnum.productPriceHistoryShow,\n                    params: {\n                        product_id: this.product.id,\n                    },\n                });\n            }),\n            filter((history) => history.length > 1),\n            map((history) => history.map((item) => item[1])),\n            map((values) => ({\n                min: Math.min(...values),\n                max: Math.max(...values),\n            }))\n        );\n\n        this.range$ = this.history$.pipe(\n            filter(tuiIsPresent),\n            filter((history) => history.length > 1),\n            map((value: ReadonlyArray<[TuiDay, number]>) => ({ from: value.at(0)?.[0], to: value.at(-1)?.[0] })),\n            filter((range): range is { from: TuiDay; to: TuiDay } => !!range.from && !!range.to),\n            map((range) => {\n                return new TuiDayRange(range.from, range.to);\n            })\n        );\n    }\n\n    /**\n     * Вычисляет подписи даты к оси X.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     */\n    @tuiPure\n    public computeLabels$({ from, to }: TuiDayRange): Observable<readonly string[]> {\n        return this.months$.pipe(map((months) => Array.from({ length: TuiMonth.lengthBetween(from, to) + 1 }, (_, index) => months[from.append({ month: index }).month])));\n    }\n\n    /**\n     * Вычисляет данные для отображения на графике.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     * @param history История цен.\n     */\n    @tuiPure\n    public computeValue({ from, to }: TuiDayRange, history: ReadonlyArray<[TuiDay, number]>): ReadonlyArray<[TuiDay, number]> {\n        return (\n            Array.from({ length: TuiDay.lengthBetween(from, to) + 1 })\n                .fill(0)\n                // eslint-disable-next-line unicorn/no-array-reduce\n                .reduce<ReadonlyArray<[TuiDay, number]>>((array, _, index) => {\n                    const data = from.append({ day: index });\n                    return [\n                        ...array,\n                        [\n                            data,\n                            (history.find((valueHistory, indexHistory, arrayHistory) => data.daySame(valueHistory[0]) || data.dayBefore(arrayHistory[indexHistory + 1][0])) ??\n                                history[0])[1],\n                        ],\n                    ];\n                }, [])\n        );\n    }\n}\n","<tui-loader [showLoader]=\"!!(loadingHistory$ | async)\"> </tui-loader>\n\n<div\n    *ngIf=\"(loadingHistory$ | async) === false\"\n    class=\"flex flex-col items-center\"\n>\n    <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n        <div class=\"w-174 mb-1 text-end text-lg font-bold\">от {{ priceInfo.min.toLocaleString() }} ₽ до {{ priceInfo.max.toLocaleString() }} ₽</div>\n\n        <ng-container *ngIf=\"history$ | async as history\">\n            <div\n                *ngIf=\"range$ | async as range\"\n                class=\"relative h-56 w-full p-5 text-tui-primary\"\n            >\n                <tui-axes\n                    *ngIf=\"computeLabels$(range) | async as labels\"\n                    class=\"h-full\"\n                    [axisXLabels]=\"labels\"\n                    [horizontalLines]=\"4\"\n                    [verticalLines]=\"labels.length\"\n                >\n                    <tui-line-days-chart\n                        class=\"chart\"\n                        [height]=\"priceInfo.max\"\n                        [smoothingFactor]=\"10\"\n                        [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n                        [value]=\"computeValue(range, history)\"\n                        [xStringify]=\"xStringify$ | async\"\n                        [yStringify]=\"yStringify\"\n                        [hintContent]=\"hint\"\n                    ></tui-line-days-chart>\n                </tui-axes>\n            </div>\n        </ng-container>\n    </ng-container>\n    <ng-template #isNotExist>\n        <p class=\"text-base font-bold\">Цена на товар не изменялась</p>\n    </ng-template>\n    <ng-template\n        #hint\n        let-data\n    >\n        <div class=\"font-bold\">{{ data[1].toLocaleString() }} ₽</div>\n        <div>{{ data[0] }}</div>\n    </ng-template>\n</div>\n"]}
@@ -1,8 +1,10 @@
1
+ /* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */
2
+ import { HttpErrorResponse } from '@angular/common/http';
1
3
  import { ChangeDetectionStrategy, Component, Inject } from '@angular/core';
2
- import { FormGroup, FormControl, Validators } from '@angular/forms';
3
- import { Subject, catchError, filter, map, of, share, startWith, switchMap, tap } from 'rxjs';
4
- import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
4
+ import { FormControl, FormGroup, Validators } from '@angular/forms';
5
5
  import { tuiIsFalsy, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';
6
+ import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
7
+ import { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
6
8
  import * as i0 from "@angular/core";
7
9
  import * as i1 from "@snabcentr/client-core";
8
10
  import * as i2 from "@angular/forms";
@@ -31,7 +33,7 @@ export class ScAddContactDialogComponent {
31
33
  name: new FormControl(null, Validators.required),
32
34
  phone: new FormControl(null, [Validators.required, Validators.minLength(12)]),
33
35
  email: new FormControl(null, Validators.email),
34
- position: new FormControl(null, Validators.required)
36
+ position: new FormControl(null, Validators.required),
35
37
  });
36
38
  /**
37
39
  * {@link Subject} события отправки формы.
@@ -41,14 +43,20 @@ export class ScAddContactDialogComponent {
41
43
  * {@link Observable} запроса добавления контактного лица.
42
44
  */
43
45
  this.request$ = this.onSubmit.pipe(map(() => this.form.value), filter((value) => this.form.valid), map((value) => this.convertersService.removeNull({ ...value, ...this.context.data })), switchMap((data) => {
44
- return this.contactsService.createContact$(data).pipe(tap((contact) => this.context?.completeWith(contact)), catchError((error) => {
46
+ return this.contactsService.createContact$(data).pipe(tap((contact) => {
47
+ this.context.completeWith(contact);
48
+ }), catchError((error) => {
45
49
  tuiMarkControlAsTouchedAndValidate(this.form);
46
- const errorResponse = error.error;
47
- for (const key in errorResponse.errors) {
48
- this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[`${key}`] });
49
- }
50
- if (!errorResponse.errors && errorResponse.message) {
51
- this.form.setErrors({ serverResponse: [errorResponse.message] });
50
+ if (error instanceof HttpErrorResponse) {
51
+ const { errors, message } = error.error;
52
+ for (const key in errors) {
53
+ if (Object.hasOwn(errors, key)) {
54
+ this.form.get(key)?.setErrors({ serverResponse: errors[key] });
55
+ }
56
+ }
57
+ if (!errors && message) {
58
+ this.form.setErrors({ serverResponse: [message] });
59
+ }
52
60
  }
53
61
  return of({});
54
62
  }), startWith(null));
@@ -68,4 +76,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
68
76
  type: Inject,
69
77
  args: [POLYMORPHEUS_CONTEXT]
70
78
  }] }]; } });
71
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-contact-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/contacts/add-contact-dialog/sc-add-contact-dialog.component.ts","../../../../../projects/client-ui/contacts/add-contact-dialog/sc-add-contact-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Inject } from '@angular/core';\nimport { FormGroup, FormControl, Validators } from '@angular/forms';\nimport {\n    ScContragentService,\n    ScIBankAccountOnDataCreate,\n    ScIBankAccount,\n    ScIContactOnDataCreate,\n    ScContactsService,\n    ScIContactWithRelations,\n    ScConvertersService,\n    ScIContactBase\n} from '@snabcentr/client-core';\nimport { Observable, Subject, catchError, filter, map, of, share, startWith, switchMap, tap } from 'rxjs';\nimport { scBicValidator, scCorrespondentAccountValidator } from '../../validators';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { tuiIsFalsy, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { ApiErrorResponse } from '../../auth';\n\n/**\n * Компонент формы создания нового контактного лица.\n */\n@Component({\n    selector: 'sc-add-contact-dialog',\n    templateUrl: './sc-add-contact-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScAddContactDialogComponent {\n    /**\n     * Группа с полями контактного лица.\n     */\n    public form: FormGroup<{\n        name: FormControl<string | null>;\n        phone: FormControl<string | null>;\n        email: FormControl<string | null>;\n        position: FormControl<string | null>;\n    }> = new FormGroup({\n            name: new FormControl<string | null>(null, Validators.required),\n            phone: new FormControl<string | null>(null, [Validators.required, Validators.minLength(12)]),\n            email: new FormControl<string | null>(null, Validators.email),\n            position: new FormControl<string | null>(null, Validators.required)\n        });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса добавления контактного лица.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is ScIContactOnDataCreate => this.form.valid),\n        map((value) => this.convertersService.removeNull<ScIContactOnDataCreate>({ ...value, ...this.context.data })),\n        switchMap((data) => {\n            return this.contactsService.createContact$(data).pipe(\n                tap((contact) => this.context?.completeWith(contact)),\n                catchError((error: HttpErrorResponse) => {\n                    tuiMarkControlAsTouchedAndValidate(this.form);\n                    const errorResponse = error.error as ApiErrorResponse;\n\n                    for (const key in errorResponse.errors) {\n                        this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[`${key}`] });\n                    }\n\n                    if (!errorResponse.errors && errorResponse.message) {\n                        this.form.setErrors({ serverResponse: [errorResponse.message] });\n                    }\n\n                    return of({} as ScIBankAccount);\n                }),\n                startWith(null)\n            );\n        }),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScAddContactDialogComponent}.\n     *\n     * @param contactsService Сервис для работы с контактными лицами.\n     * @param convertersService Сервис конвертации данных.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly contactsService: ScContactsService,\n        public readonly convertersService: ScConvertersService,\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context: TuiDialogContext<ScIContactWithRelations, Omit<ScIContactOnDataCreate, keyof ScIContactBase>>\n    ) {}\n}\n","<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit.next()\" class=\"flex flex-col gap-3\">\n    <sc-new-contact-form [form]=\"form\"></sc-new-contact-form>\n    <div class=\"flex gap-4 justify-center\">\n        <button tuiButton (click)=\"context.$implicit.complete()\" type=\"button\" icon=\"tuiIconXLarge\" appearance=\"secondary\">Отмена</button>\n        <button tuiButton tuiMode=\"onLight\" [disabled]=\"form.invalid\" [showLoader]=\"!!(loading$ | async)\" type=\"submit\" icon=\"scIconSave\">Сохранить</button>\n    </div>\n</form>\n"]}
79
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-contact-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/contacts/add-contact-dialog/sc-add-contact-dialog.component.ts","../../../../../projects/client-ui/contacts/add-contact-dialog/sc-add-contact-dialog.component.html"],"names":[],"mappings":"AAAA,2EAA2E;AAE3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEpE,OAAO,EAAE,UAAU,EAAE,kCAAkC,EAAE,MAAM,eAAe,CAAC;AAE/E,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;;;;;;;AAI1G;;GAEG;AAMH,MAAM,OAAO,2BAA2B;IA8DpC;;;;;;OAMG;IACH,YACqB,eAAkC,EACnC,iBAAsC,EAEtC,OAAsG;QAHrG,oBAAe,GAAf,eAAe,CAAmB;QACnC,sBAAiB,GAAjB,iBAAiB,CAAqB;QAEtC,YAAO,GAAP,OAAO,CAA+F;QAxE1H;;WAEG;QACI,SAAI,GAAG,IAAI,SAAS,CAKxB;YACC,IAAI,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC;YAC/D,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC;YAC5F,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,KAAK,CAAC;YAC7D,QAAQ,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC;SACtE,CAAC,CAAC;QAEH;;WAEG;QACa,aAAQ,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAE9D;;WAEG;QACc,aAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAC1C,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAC1B,MAAM,CAAC,CAAC,KAAK,EAAmC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EACnE,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAyB,EAAE,GAAG,KAAK,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,EAC7G,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;YACf,OAAO,IAAI,CAAC,eAAe,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,IAAI,CACjD,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;gBACZ,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YACvC,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;gBAC1B,kCAAkC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAE9C,IAAI,KAAK,YAAY,iBAAiB,EAAE;oBACpC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,KAAyB,CAAC;oBAE5D,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE;wBACtB,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE;4BAC5B,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,EAAE,cAAc,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;yBAClE;qBACJ;oBAED,IAAI,CAAC,MAAM,IAAI,OAAO,EAAE;wBACpB,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;qBACtD;iBACJ;gBACD,OAAO,EAAE,CAAC,EAAoB,CAAC,CAAC;YACpC,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CAAC;QACN,CAAC,CAAC,EACF,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACa,aAAQ,GAAwB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;IAcjF,CAAC;;wHA1EK,2BAA2B,sFAwExB,oBAAoB;4GAxEvB,2BAA2B,6DCrBxC,wmBAOA;2FDca,2BAA2B;kBALvC,SAAS;+BACI,uBAAuB,mBAEhB,uBAAuB,CAAC,MAAM;;0BA0E1C,MAAM;2BAAC,oBAAoB","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, Inject } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { ScContactsService, ScConvertersService, ScIBankAccount, ScIContactBase, ScIContactOnDataCreate, ScIContactWithRelations } from '@snabcentr/client-core';\nimport { tuiIsFalsy, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\n\n/**\n * Компонент формы создания нового контактного лица.\n */\n@Component({\n    selector: 'sc-add-contact-dialog',\n    templateUrl: './sc-add-contact-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddContactDialogComponent {\n    /**\n     * Группа с полями контактного лица.\n     */\n    public form = new FormGroup<{\n        name: FormControl<string | null>;\n        phone: FormControl<string | null>;\n        email: FormControl<string | null>;\n        position: FormControl<string | null>;\n    }>({\n        name: new FormControl<string | null>(null, Validators.required),\n        phone: new FormControl<string | null>(null, [Validators.required, Validators.minLength(12)]),\n        email: new FormControl<string | null>(null, Validators.email),\n        position: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса добавления контактного лица.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is ScIContactOnDataCreate => this.form.valid),\n        map((value) => this.convertersService.removeNull<ScIContactOnDataCreate>({ ...value, ...this.context.data })),\n        switchMap((data) => {\n            return this.contactsService.createContact$(data).pipe(\n                tap((contact) => {\n                    this.context.completeWith(contact);\n                }),\n                catchError((error: unknown) => {\n                    tuiMarkControlAsTouchedAndValidate(this.form);\n\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n                    return of({} as ScIBankAccount);\n                }),\n                startWith(null)\n            );\n        }),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScAddContactDialogComponent}.\n     *\n     * @param contactsService Сервис для работы с контактными лицами.\n     * @param convertersService Сервис конвертации данных.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly contactsService: ScContactsService,\n        public readonly convertersService: ScConvertersService,\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context: TuiDialogContext<ScIContactWithRelations, Omit<ScIContactOnDataCreate, keyof ScIContactBase>>\n    ) {}\n}\n","<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit.next()\" class=\"flex flex-col gap-3\">\n    <sc-new-contact-form [form]=\"form\"></sc-new-contact-form>\n    <div class=\"flex gap-4 justify-center\">\n        <button tuiButton (click)=\"context.$implicit.complete()\" type=\"button\" icon=\"tuiIconXLarge\" appearance=\"secondary\">Отмена</button>\n        <button tuiButton tuiMode=\"onLight\" [disabled]=\"form.invalid\" [showLoader]=\"!!(loading$ | async)\" type=\"submit\" icon=\"scIconSave\">Сохранить</button>\n    </div>\n</form>\n"]}
@@ -1,9 +1,11 @@
1
+ /* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */
2
+ import { HttpErrorResponse } from '@angular/common/http';
1
3
  import { ChangeDetectionStrategy, Component, Inject } from '@angular/core';
2
- import { FormGroup, FormControl, Validators } from '@angular/forms';
3
- import { Subject, catchError, filter, map, of, share, startWith, switchMap, tap } from 'rxjs';
4
- import { scBicValidator, scCorrespondentAccountValidator } from '../../validators';
5
- import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
4
+ import { FormControl, FormGroup, Validators } from '@angular/forms';
6
5
  import { tuiIsFalsy, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';
6
+ import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
7
+ import { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
8
+ import { scBicValidator, scCorrespondentAccountValidator } from '../../validators';
7
9
  import * as i0 from "@angular/core";
8
10
  import * as i1 from "@snabcentr/client-core";
9
11
  import * as i2 from "@angular/forms";
@@ -31,7 +33,7 @@ export class ScAddContragentBankAccountsDialogComponent {
31
33
  bic: new FormControl(null, [Validators.required, Validators.minLength(9), scBicValidator]),
32
34
  accountNumber: new FormControl(null, [Validators.required, Validators.minLength(20)]),
33
35
  correspondentAccount: new FormControl(null, [Validators.required, Validators.minLength(20), scCorrespondentAccountValidator]),
34
- currencyId: new FormControl(null, Validators.required)
36
+ currencyId: new FormControl(null, Validators.required),
35
37
  });
36
38
  /**
37
39
  * {@link Subject} события отправки формы.
@@ -41,14 +43,20 @@ export class ScAddContragentBankAccountsDialogComponent {
41
43
  * {@link Observable} запроса создания банковского счёта контрагента.
42
44
  */
43
45
  this.request$ = this.onSubmit.pipe(map(() => this.form.value), filter((value) => this.form.valid), switchMap((value) => {
44
- return this.contragentService.createContragentBankAccount$(this.context.data.contragentId, value).pipe(tap((bankAccount) => this.context?.completeWith(bankAccount)), catchError((error) => {
46
+ return this.contragentService.createContragentBankAccount$(this.context.data.contragentId, value).pipe(tap((bankAccount) => {
47
+ this.context.completeWith(bankAccount);
48
+ }), catchError((error) => {
45
49
  tuiMarkControlAsTouchedAndValidate(this.form);
46
- const errorResponse = error.error;
47
- for (const key in errorResponse.errors) {
48
- this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[`${key}`] });
49
- }
50
- if (!errorResponse.errors && errorResponse.message) {
51
- this.form.setErrors({ serverResponse: [errorResponse.message] });
50
+ if (error instanceof HttpErrorResponse) {
51
+ const { errors, message } = error.error;
52
+ for (const key in errors) {
53
+ if (Object.hasOwn(errors, key)) {
54
+ this.form.get(key)?.setErrors({ serverResponse: errors[key] });
55
+ }
56
+ }
57
+ if (!errors && message) {
58
+ this.form.setErrors({ serverResponse: [message] });
59
+ }
52
60
  }
53
61
  return of({});
54
62
  }), startWith(null));
@@ -68,4 +76,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
68
76
  type: Inject,
69
77
  args: [POLYMORPHEUS_CONTEXT]
70
78
  }] }]; } });
71
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-contragent-bank-account-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.ts","../../../../../projects/client-ui/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Inject } from '@angular/core';\nimport { FormGroup, FormControl, Validators } from '@angular/forms';\nimport { ScContragentService, ScIBankAccountOnDataCreate, ScIBankAccount } from '@snabcentr/client-core';\nimport { Observable, Subject, catchError, filter, map, of, share, startWith, switchMap, tap } from 'rxjs';\nimport { scBicValidator, scCorrespondentAccountValidator } from '../../validators';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { tuiIsFalsy, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { ApiErrorResponse } from '../../auth';\n\n/**\n * Компонент формы создания нового банковского счета контрагента.\n */\n@Component({\n    selector: 'sc-add-contragent-bank-account-dialog',\n    templateUrl: './sc-add-contragent-bank-account-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScAddContragentBankAccountsDialogComponent {\n    /**\n     * Группа с полем банковских реквизитов.\n     */\n    public form: FormGroup<{\n        bankName: FormControl<string | null>;\n        bic: FormControl<string | null>;\n        accountNumber: FormControl<string | null>;\n        correspondentAccount: FormControl<string | null>;\n        currencyId: FormControl<number | null>;\n    }> = new FormGroup({\n            bankName: new FormControl<string | null>(null, Validators.required),\n            bic: new FormControl<string | null>(null, [Validators.required, Validators.minLength(9), scBicValidator]),\n            accountNumber: new FormControl<string | null>(null, [Validators.required, Validators.minLength(20)]),\n            correspondentAccount: new FormControl<string | null>(null, [Validators.required, Validators.minLength(20), scCorrespondentAccountValidator]),\n            currencyId: new FormControl<number | null>(null, Validators.required)\n        });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса создания банковского счёта контрагента.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is ScIBankAccountOnDataCreate => this.form.valid),\n        switchMap((value) => {\n            return this.contragentService.createContragentBankAccount$(this.context.data.contragentId, value).pipe(\n                tap((bankAccount) => this.context?.completeWith(bankAccount)),\n                catchError((error: HttpErrorResponse) => {\n                    tuiMarkControlAsTouchedAndValidate(this.form);\n                    const errorResponse = error.error as ApiErrorResponse;\n\n                    for (const key in errorResponse.errors) {\n                        this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[`${key}`] });\n                    }\n\n                    if (!errorResponse.errors && errorResponse.message) {\n                        this.form.setErrors({ serverResponse: [errorResponse.message] });\n                    }\n\n                    return of({} as ScIBankAccount);\n                }),\n                startWith(null)\n            );\n        }),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScAddContragentBankAccountsFormComponent}.\n     *\n     * @param contragentService Сервис работы с контрагентами.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly contragentService: ScContragentService,\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context: TuiDialogContext<ScIBankAccount, { contragentId: number }>\n    ) {}\n}\n","<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit.next()\" class=\"flex flex-col gap-3\">\n    <sc-new-contragent-bank-account-form [form]=\"form\"></sc-new-contragent-bank-account-form>\n    <div class=\"flex gap-4 justify-center\">\n        <button tuiButton (click)=\"context.$implicit.complete()\" type=\"button\" icon=\"tuiIconXLarge\" appearance=\"secondary\">Отмена</button>\n        <button tuiButton tuiMode=\"onLight\" [disabled]=\"form.invalid\" [showLoader]=\"!!(loading$ | async)\" type=\"submit\" icon=\"scIconSave\">Сохранить</button>\n    </div>\n</form>"]}
79
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-add-contragent-bank-account-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.ts","../../../../../projects/client-ui/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, Inject } from '@angular/core';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { ScContragentService, ScIBankAccount, ScIBankAccountOnDataCreate } from '@snabcentr/client-core';\nimport { tuiIsFalsy, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { scBicValidator, scCorrespondentAccountValidator } from '../../validators';\n\n/**\n * Компонент формы создания нового банковского счета контрагента.\n */\n@Component({\n    selector: 'sc-add-contragent-bank-account-dialog',\n    templateUrl: './sc-add-contragent-bank-account-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScAddContragentBankAccountsDialogComponent {\n    /**\n     * Группа с полем банковских реквизитов.\n     */\n    public form = new FormGroup<{\n        bankName: FormControl<string | null>;\n        bic: FormControl<string | null>;\n        accountNumber: FormControl<string | null>;\n        correspondentAccount: FormControl<string | null>;\n        currencyId: FormControl<number | null>;\n    }>({\n        bankName: new FormControl<string | null>(null, Validators.required),\n        bic: new FormControl<string | null>(null, [Validators.required, Validators.minLength(9), scBicValidator]),\n        accountNumber: new FormControl<string | null>(null, [Validators.required, Validators.minLength(20)]),\n        correspondentAccount: new FormControl<string | null>(null, [Validators.required, Validators.minLength(20), scCorrespondentAccountValidator]),\n        currencyId: new FormControl<number | null>(null, Validators.required),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса создания банковского счёта контрагента.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        map(() => this.form.value),\n        filter((value): value is ScIBankAccountOnDataCreate => this.form.valid),\n        switchMap((value) => {\n            return this.contragentService.createContragentBankAccount$(this.context.data.contragentId, value).pipe(\n                tap((bankAccount) => {\n                    this.context.completeWith(bankAccount);\n                }),\n                catchError((error: unknown) => {\n                    tuiMarkControlAsTouchedAndValidate(this.form);\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as ScIBankAccount);\n                }),\n                startWith(null)\n            );\n        }),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по номеру телефона.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScAddContragentBankAccountsFormComponent}.\n     *\n     * @param contragentService Сервис работы с контрагентами.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly contragentService: ScContragentService,\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context: TuiDialogContext<ScIBankAccount, { contragentId: number }>\n    ) {}\n}\n","<form [formGroup]=\"form\" (ngSubmit)=\"onSubmit.next()\" class=\"flex flex-col gap-3\">\n    <sc-new-contragent-bank-account-form [form]=\"form\"></sc-new-contragent-bank-account-form>\n    <div class=\"flex gap-4 justify-center\">\n        <button tuiButton (click)=\"context.$implicit.complete()\" type=\"button\" icon=\"tuiIconXLarge\" appearance=\"secondary\">Отмена</button>\n        <button tuiButton tuiMode=\"onLight\" [disabled]=\"form.invalid\" [showLoader]=\"!!(loading$ | async)\" type=\"submit\" icon=\"scIconSave\">Сохранить</button>\n    </div>\n</form>"]}