@snabcentr/client-ui 3.49.3 → 3.50.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (69) hide show
  1. package/auth/interfaces/index.d.ts +0 -1
  2. package/banner/sc-banner.component.d.ts +23 -29
  3. package/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.d.ts +5 -5
  4. package/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.d.ts +5 -5
  5. package/cart/index.d.ts +0 -1
  6. package/catalog/price-card-inline/sc-price-card-inline.component.d.ts +1 -1
  7. package/configurators/sandwich/sc-i-new-cart-item-sandwich.d.ts +2 -2
  8. package/configurators/sandwich/sc-sandwich.component.d.ts +2 -2
  9. package/directives/abstract-price-card/abstract-sc-price-card.directive.d.ts +6 -6
  10. package/esm2022/auth/interfaces/index.mjs +1 -2
  11. package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-email/sc-sign-in-form-by-email.component.mjs +1 -1
  12. package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +1 -1
  13. package/esm2022/auth/sc-simple-sign-up-form/sc-simple-sign-up-form.component.mjs +1 -1
  14. package/esm2022/auth/sign-up-form/sc-sign-up-form.component.mjs +3 -3
  15. package/esm2022/banner/sc-banner.component.mjs +53 -59
  16. package/esm2022/cart/add-or-editing-cart-item-dialog/add-or-editing-cart-item-form/sc-add-or-editing-cart-item-form.component.mjs +11 -11
  17. package/esm2022/cart/add-or-editing-cart-item-dialog/sc-add-or-editing-cart-item-dialog.component.mjs +6 -6
  18. package/esm2022/cart/index.mjs +1 -2
  19. package/esm2022/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.mjs +1 -1
  20. package/esm2022/catalog/price-card/sc-price-card.component.mjs +3 -3
  21. package/esm2022/catalog/price-card-inline/sc-price-card-inline.component.mjs +3 -3
  22. package/esm2022/catalog/price-history/sc-price-history.component.mjs +3 -3
  23. package/esm2022/configurators/sandwich/sc-i-new-cart-item-sandwich.mjs +1 -1
  24. package/esm2022/configurators/sandwich/sc-sandwich.component.mjs +6 -13
  25. package/esm2022/contacts/add-contact-dialog/sc-add-contact-dialog.component.mjs +1 -1
  26. package/esm2022/contragents/add-contragent-bank-account-dialog/sc-add-contragent-bank-account-dialog.component.mjs +1 -1
  27. package/esm2022/contragents/add-contragent-dialog/sc-add-contragent-dialog.component.mjs +1 -1
  28. package/esm2022/delivery-address/add-delivery-address-dialog/sc-add-delivery-address-dialog.component.mjs +1 -1
  29. package/esm2022/directives/abstract-price-card/abstract-sc-price-card.directive.mjs +8 -8
  30. package/esm2022/feedback/feedback-form/sc-feedback-form.component.mjs +1 -1
  31. package/esm2022/form-fields/suggestion-field/sc-suggestion-field.component.mjs +3 -3
  32. package/esm2022/order/draft/sc-draft.component.mjs +117 -0
  33. package/esm2022/order/index.mjs +7 -2
  34. package/esm2022/order/models/sc-i-cart-items-by-direction.mjs +22 -0
  35. package/esm2022/order/order-item/sc-order-item.component.mjs +85 -0
  36. package/esm2022/order/order-items-list/sc-order-items-list.component.mjs +203 -0
  37. package/esm2022/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.mjs +72 -0
  38. package/esm2022/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.mjs +75 -0
  39. package/esm2022/order/sc-order.module.mjs +5 -6
  40. package/esm2022/providers/index.mjs +2 -1
  41. package/esm2022/providers/sc-dialog-service.providers.mjs +6 -0
  42. package/esm2022/user/reset-user-password/sc-reset-user-password.component.mjs +1 -1
  43. package/esm2022/user/update-user-info-dialog/sc-update-user-info-dialog.component.mjs +2 -2
  44. package/esm2022/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.mjs +2 -2
  45. package/esm2022/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +1 -1
  46. package/fesm2022/snabcentr-client-ui.mjs +1971 -1580
  47. package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
  48. package/order/draft/sc-draft.component.d.ts +63 -0
  49. package/order/index.d.ts +6 -1
  50. package/order/models/sc-i-cart-items-by-direction.d.ts +25 -0
  51. package/{cart/cart-item/sc-cart-item.component.d.ts → order/order-item/sc-order-item.component.d.ts} +7 -3
  52. package/order/order-items-list/sc-order-items-list.component.d.ts +102 -0
  53. package/order/order-items-list-by-directions/sc-order-items-list-by-directions.component.d.ts +55 -0
  54. package/order/order-items-list-by-stock/sc-order-items-list-by-stock.component.d.ts +55 -0
  55. package/order/sc-order.module.d.ts +10 -11
  56. package/package.json +3 -2
  57. package/providers/index.d.ts +1 -0
  58. package/providers/sc-dialog-service.providers.d.ts +41 -0
  59. package/styles/taiga/snabcentr-tailwind-preset.js +2 -3
  60. package/styles/tailwind/tailwind.scss +96 -100
  61. package/auth/interfaces/api-error-response.d.ts +0 -13
  62. package/esm2022/auth/interfaces/api-error-response.mjs +0 -2
  63. package/esm2022/cart/cart-item/sc-cart-item.component.mjs +0 -71
  64. package/esm2022/helpers/index.mjs +0 -2
  65. package/esm2022/helpers/sc-px-converter.mjs +0 -27
  66. package/esm2022/order/order-item-mobile/sc-order-item-mobile.component.mjs +0 -60
  67. package/helpers/index.d.ts +0 -1
  68. package/helpers/sc-px-converter.d.ts +0 -15
  69. package/order/order-item-mobile/sc-order-item-mobile.component.d.ts +0 -39
@@ -341,4 +341,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
341
341
  TuiSortCountriesPipe,
342
342
  ], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let fieldsList = fields();\n<form\n *ngIf=\"fieldsList.length > 0\"\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col justify-between gap-4\"\n>\n <!-- \u0424\u0418\u041E -->\n <label\n *ngIf=\"fieldsList.includes('name')\"\n tuiLabel\n >\n \u0424\u0418\u041E\n <tui-input formControlName=\"name\"> \u0423\u043A\u0430\u0436\u0438\u0442\u0435 \u0412\u0430\u0448\u0438 \u0424.\u0418.\u041E. </tui-input>\n <tui-error\n formControlName=\"name\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0422\u0435\u043B\u0435\u0444\u043E\u043D -->\n @if (fieldsList.includes('phone')) {\n <label tuiLabel>\n \u0422\u0435\u043B\u0435\u0444\u043E\u043D\n <tui-textfield>\n <input\n formControlName=\"phone\"\n tuiInputPhoneInternational\n [countrySearch]=\"true\"\n [countries]=\"(countries | tuiSortCountries | async) || []\"\n autocomplete=\"phone\"\n />\n </tui-textfield>\n <tui-error\n formControlName=\"phone\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <label\n *ngIf=\"fieldsList.includes('isShouldUseMessenger') && form.controls.phone.value && isValidPhoneNumberForMessenger(form.controls.phone.value)\"\n tuiLabel\n >\n <input\n formControlName=\"isShouldUseMessenger\"\n tuiCheckbox\n type=\"checkbox\"\n />\n <span tuiTitle> \u041D\u0430\u043F\u0438\u0441\u0430\u0442\u044C \u0432 \u043C\u0435\u0441\u0441\u0435\u043D\u0434\u0436\u0435\u0440\u0430\u0445, \u0430 \u043D\u0435 \u0437\u0432\u043E\u043D\u0438\u0442\u044C </span>\n </label>\n }\n <!-- \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B -->\n <label\n *ngIf=\"fieldsList.includes('email')\"\n tuiLabel\n >\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <tui-input formControlName=\"email\">\n \u0410\u0434\u0440\u0435\u0441 \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u044B\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.email\"\n />\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439 -->\n <label\n *ngIf=\"fieldsList.includes('message')\"\n tuiLabel\n >\n \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439\n <tui-textarea formControlName=\"message\"> \u041A\u043E\u043C\u043C\u0435\u043D\u0442\u0430\u0440\u0438\u0439 </tui-textarea>\n <tui-error\n formControlName=\"message\"\n [error]=\"[] | tuiFieldError | async\"\n />\n </label>\n\n <!-- \u0424\u0430\u0439\u043B\u044B -->\n <div\n *ngIf=\"fieldsList.includes('files')\"\n class=\"flex flex-col gap-1\"\n >\n <label tuiInputFiles>\n <input\n #validator=\"tuiInputFilesValidator\"\n accept=\".doc,.docx,.pdf,image/*\"\n tuiInputFiles\n [formControl]=\"loadedFile\"\n [multiple]=\"true\"\n (reject)=\"onReject($event)\"\n />\n </label>\n\n <tui-error\n *ngIf=\"loadedFile.invalid\"\n [error]=\"['maxLength'] | tuiFieldError | async\"\n [formControl]=\"loadedFile\"\n />\n\n @let accepted = accepted$ | async;\n <tui-files\n *ngIf=\"accepted?.length || rejected.length\"\n class=\"tui-space_top-1\"\n >\n <tui-file\n *ngFor=\"let file of accepted\"\n [file]=\"file\"\n (remove)=\"onRemove(file)\"\n />\n <tui-file\n *ngFor=\"let file of rejected\"\n state=\"error\"\n [file]=\"(file | tuiFileRejected: validator | async) || file\"\n (remove)=\"onRemove(file)\"\n />\n </tui-files>\n </div>\n\n <!-- recaptcha -->\n @if (form.get('recaptcha')) {\n <div\n class=\"mb-3 max-h-20 overflow-hidden duration-500\"\n [ngClass]=\"{ '!max-h-0 !-m-2': form.get('recaptcha')?.valid }\"\n >\n <re-captcha formControlName=\"recaptcha\" />\n </div>\n }\n\n <!-- \u042D\u043B\u0435\u043C\u0435\u043D\u0442\u044B \u0443\u043F\u0440\u0430\u0432\u043B\u0435\u043D\u0438\u044F -->\n <div class=\"flex flex-wrap items-center gap-3\">\n <button\n tuiButton\n [disabled]=\"form.invalid\"\n [loading]=\"!!(loading$ | async)\"\n type=\"submit\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n >\n \u041E\u0442\u043F\u0440\u0430\u0432\u0438\u0442\u044C\n </button>\n <div class=\"grow basis-60 text-body-s\">\n \u041D\u0430\u0436\u0438\u043C\u0430\u044F \u043D\u0430 \u043A\u043D\u043E\u043F\u043A\u0443, \u0432\u044B \u0441\u043E\u0433\u043B\u0430\u0448\u0430\u0435\u0442\u0435\u0441\u044C \u0441\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"context?.data?.onClickOfferHandler(); clickOffer.emit()\"\n >\n \u041F\u043E\u043B\u0438\u0442\u0438\u043A\u043E\u0439 \u043A\u043E\u043D\u0444\u0438\u0434\u0435\u043D\u0446\u0438\u0430\u043B\u044C\u043D\u043E\u0441\u0442\u0438\n </button>\n </div>\n </div>\n</form>\n" }]
343
343
  }], ctorParameters: () => [], propDecorators: { isValidPhoneNumberForMessenger: [] } });
344
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-feedback-form.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/feedback/feedback-form/sc-feedback-form.component.ts","../../../../../projects/client-ui/feedback/feedback-form/sc-feedback-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgClass, NgFor, NgIf } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, computed, effect, inject, input, InputSignal, output, OutputEmitterRef, Signal } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { AbstractControl, FormControl, FormGroup, ReactiveFormsModule, ValidatorFn, Validators } from '@angular/forms';\nimport {\n    ScConvertersService,\n    ScFeedbackForms,\n    ScFeedbackService,\n    ScIFeedbackMessage,\n    ScISuggestionType,\n    ScPhoneService,\n    ScUploadedFile,\n    ScUser,\n    ScUserService,\n} from '@snabcentr/client-core';\nimport { TuiLooseUnion, tuiPure, TuiValidationError } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiError, TuiLabel, TuiLink, TuiTextfield, TuiTitle } from '@taiga-ui/core';\nimport { TuiInputPhoneInternational } from '@taiga-ui/experimental';\nimport { TuiCountryIsoCode } from '@taiga-ui/i18n';\nimport { TuiButtonLoading, TuiCheckbox, TuiFieldErrorPipe, TuiFileLike, TuiFiles, tuiFilesAccepted, TuiSortCountriesPipe } from '@taiga-ui/kit';\nimport { TuiInputModule, TuiInputPhoneModule, TuiTextareaModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { getCountries } from 'libphonenumber-js';\nimport { RecaptchaComponent, RecaptchaFormsModule, RecaptchaModule } from 'ng-recaptcha-2';\nimport { catchError, filter, map, merge, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { ScFormFieldsModule } from '../../form-fields';\nimport { MAX_FILES_IN_FORM_INPUT } from '../../providers/sc-max-files-in-form-input';\nimport { phoneValidator } from '../../validators/sc-phone-validator';\nimport { ScFeedbackFormConfig, ScFeedbackFormFieldType } from '../interfaces';\n\n/**\n * Компонент формы обратной связи.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-feedback-form',\n    templateUrl: './sc-feedback-form.component.html',\n    imports: [\n        AsyncPipe,\n        ReactiveFormsModule,\n        NgClass,\n        NgFor,\n        TuiError,\n        TuiFieldErrorPipe,\n        ScFormFieldsModule,\n        TuiInputModule,\n        TuiInputPhoneModule,\n        TuiButton,\n        TuiButtonLoading,\n        TuiLabel,\n        RecaptchaFormsModule,\n        RecaptchaModule,\n        TuiLink,\n        NgIf,\n        TuiTextareaModule,\n        TuiCheckbox,\n        TuiTitle,\n        TuiFiles,\n        TuiInputPhoneInternational,\n        ...TuiTextfield,\n        TuiSortCountriesPipe,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScFeedbackFormComponent {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { onClickOfferHandler: () => void } & ScFeedbackFormConfig> | null = inject<\n        TuiDialogContext<void, { onClickOfferHandler: () => void } & ScFeedbackFormConfig>\n    >(POLYMORPHEUS_CONTEXT, { optional: true });\n\n    /**\n     * Символьное обозначение (slug) формы.\n     */\n    // eslint-disable-next-line @angular-eslint/no-input-rename\n    public readonly slugInput: InputSignal<TuiLooseUnion<ScFeedbackForms>> = input<TuiLooseUnion<ScFeedbackForms>>('common', { alias: 'slug' });\n\n    /**\n     * Список полей, доступных в данной форме.\n     */\n    // eslint-disable-next-line @angular-eslint/no-input-rename\n    public readonly fieldsInput: InputSignal<ScFeedbackFormFieldType[]> = input<ScFeedbackFormFieldType[]>(['name', 'email', 'phone', 'message', 'files'], { alias: 'fields' });\n\n    /**\n     * Символьное обозначение (slug) формы.\n     */\n    protected readonly slug: Signal<TuiLooseUnion<ScFeedbackForms>> = computed(() => this.context?.data.slug ?? this.slugInput());\n\n    /**\n     * Список полей, доступных в данной форме.\n     */\n    protected readonly fields: Signal<ScFeedbackFormFieldType[]> = computed(() => this.context?.data.fields ?? this.fieldsInput());\n\n    /**\n     * Событие нажатия на кнопку договора оферты.\n     */\n    public readonly clickOffer: OutputEmitterRef<void> = output();\n\n    /**\n     * Событие успешной отправки формы.\n     */\n    public readonly sendSuccess: OutputEmitterRef<void> = output();\n\n    /**\n     * Сервис обратной связи.\n     */\n    private readonly feedbackService: ScFeedbackService = inject(ScFeedbackService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /**\n     * Максимальное количество файлов, отправляемых в форме.\n     */\n    private readonly maxFilesInForm: number = inject(MAX_FILES_IN_FORM_INPUT);\n\n    /**\n     * Сервис данных о пользователе.\n     */\n    private readonly user$: Observable<ScUser> = inject(ScUserService)\n        .getUserChange$()\n        .pipe(\n            tap((user) => {\n                if (user.isGuest) {\n                    (this.form as FormGroup).addControl('recaptcha', this.recaptchaControl);\n                    this.resetForm();\n                } else {\n                    (this.form as FormGroup).removeControl('recaptcha');\n\n                    const fields = this.fields();\n                    this.form.patchValue({\n                        name: fields.includes('name') ? user.name : null,\n                        phone: fields.includes('phone') ? user.contacts.phone.value : null,\n                        email: fields.includes('email') ? user.contacts.email.value : null,\n                    });\n                }\n            })\n        );\n\n    /**\n     * Поле ввода для recaptcha.\n     */\n    protected readonly recaptchaControl: FormControl<string | null> = new FormControl<string | null>(null, Validators.required);\n\n    /**\n     * Группа полей ввода для формы «Пожелания и предложения по улучшению сайта».\n     */\n    protected readonly form = new FormGroup({\n        name: new FormControl<string | null>(null),\n        phone: new FormControl<string | null>(null),\n        email: new FormControl<string | null>(null),\n        message: new FormControl<string | null>(null),\n        files: new FormControl<ScUploadedFile[] | null>([]),\n        isShouldUseMessenger: new FormControl<boolean | null>(false),\n    });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на отправку данных.\n     */\n    protected readonly request$: Observable<undefined | null> = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIFeedbackMessage),\n        tap((value: ScIFeedbackMessage) => {\n            if (value.phone && !ScPhoneService.isValidPhoneNumberForMessenger(value.phone)) {\n                // eslint-disable-next-line no-param-reassign\n                delete value.isShouldUseMessenger;\n            }\n        }),\n        switchMap((value) =>\n            this.feedbackService.sendFeedback(this.slug() as ScFeedbackForms, this.convertersService.removeNullRecursive(value)).pipe(\n                tap(() => {\n                    if (this.context) {\n                        this.context.completeWith();\n                    } else {\n                        this.resetForm();\n                        this.onRemoveAll();\n                        this.sendSuccess.emit();\n                    }\n                }),\n                // eslint-disable-next-line rxjs/no-implicit-any-catch\n                catchError((error: HttpErrorResponse) => {\n                    this.form.markAllAsTouched();\n\n                    const { errors, message } = error.error as ApiErrorResponse;\n\n                    if (errors) {\n                        Object.keys(errors).forEach((key) => {\n                            const control = this.form.get(key);\n\n                            if (control) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                control.setErrors({ serverResponse: errors[key] });\n                            }\n                        });\n                    } else if (message) {\n                        this.form.setErrors({ serverResponse: [message] });\n                    }\n\n                    return of(null);\n                }),\n                // eslint-disable-next-line unicorn/no-useless-undefined\n                startWith(undefined)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных.\n     */\n    protected readonly loading$: Observable<boolean> = this.request$.pipe(map((value) => value === undefined));\n\n    /**\n     * Управляет состоянием загруженных файлов.\n     */\n    protected readonly loadedFile: FormControl<TuiFileLike[] | null> = new FormControl<TuiFileLike[] | null>([], [this.maxFilesLength()]);\n\n    /**\n     * Observable, который отслеживает, приняты ли файлы в соответствии с требованиями по типу размеру и т.д.\n     */\n    protected readonly accepted$: Observable<File[]> = this.loadedFile.valueChanges.pipe(map(() => tuiFilesAccepted(this.loadedFile)));\n\n    /**\n     * Список файлов, которые были отклонены при загрузке.\n     */\n    protected rejected: readonly File[] = [];\n\n    /**\n     * Список стран.\n     */\n    protected readonly countries: readonly TuiCountryIsoCode[] = getCountries();\n\n    /**\n     * Инициализирует экземпляр класса {@link ScFeedbackFormComponent}.\n     */\n    public constructor() {\n        /**\n         * Это решение ошибки #4940, было @see [предложено](https://github.com/DethAriel/ng-recaptcha/issues/123#issuecomment-426112101) автором {@link RecaptchaComponent}.\n         */\n        RecaptchaComponent.prototype.ngOnDestroy = function (): void {\n            // eslint-disable-next-line @typescript-eslint/dot-notation\n            if (this['subscription']) {\n                // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call, @typescript-eslint/dot-notation\n                this['subscription'].unsubscribe();\n            }\n        };\n\n        merge(\n            this.accepted$,\n            this.user$,\n            this.loadedFile.valueChanges.pipe(\n                tap((files) => {\n                    this.form.controls.files.reset([]);\n                    this.onUploadFile(files);\n                })\n            )\n        )\n            .pipe(takeUntilDestroyed())\n            .subscribe();\n\n        // Валидация полей ввода в зависимости от формы.\n        effect(() => {\n            this.clearAllValidators();\n            this.setAllValidators(this.fields());\n        });\n    }\n\n    /**\n     * Удаляет валидацию со всех полей ввода.\n     */\n    private clearAllValidators(): void {\n        Object.keys(this.form.controls).forEach((key) => {\n            const control = this.form.get(key);\n\n            if (control && control !== this.recaptchaControl) {\n                control.clearValidators();\n                control.updateValueAndValidity();\n            }\n        });\n    }\n\n    /**\n     * Устанавливает валидацию со для указанных полей ввода.\n     *\n     * @param fields Поля, для которых нужно установить валидацию.\n     */\n    private setAllValidators(fields: ScFeedbackFormFieldType[]): void {\n        fields.forEach((filed: ScFeedbackFormFieldType) => {\n            // eslint-disable-next-line default-case\n            switch (filed) {\n                case 'name':\n                    this.form.get('name')?.setValidators([Validators.required, Validators.minLength(3)]);\n                    break;\n                case 'phone':\n                    this.form.get('phone')?.setValidators([Validators.required, phoneValidator()]);\n                    break;\n                case 'email':\n                    this.form.get('email')?.setValidators([Validators.required, Validators.email]);\n                    break;\n                case 'message':\n                    this.form.get('message')?.setValidators([Validators.required]);\n                    break;\n                case 'files':\n                    this.form.get('files')?.setValidators([Validators.required, this.maxFilesLength(5)]);\n                    break;\n            }\n        });\n    }\n\n    /**\n     * Обрабатывает отклоненные файлы.\n     *\n     * @param files Список файлов, которые были отклонены.\n     */\n    protected onReject(files: readonly File[]): void {\n        this.rejected = [...new Set([...this.rejected, ...files])];\n    }\n\n    /**\n     * Удаляет файл из списка загруженных и отклоненных файлов.\n     *\n     * @param file Файл, который нужно удалить.\n     */\n    protected onRemove(file: File): void {\n        this.rejected = this.rejected.filter((rejected) => rejected !== file);\n        this.loadedFile.setValue(this.loadedFile.value?.filter((current) => current !== file) ?? []);\n    }\n\n    /**\n     * Очищает список файлов.\n     */\n    protected onRemoveAll(): void {\n        this.rejected = [];\n        this.loadedFile.setValue([]);\n    }\n\n    /**\n     * Загружает файл в форму обратной связи.\n     *\n     * @param files Набор файлов.\n     */\n    private onUploadFile(files: TuiFileLike[] | null): void {\n        files?.forEach((file) => {\n            const reader = new FileReader();\n\n            reader.addEventListener('load', () => {\n                if (reader.result) {\n                    // eslint-disable-next-line @typescript-eslint/no-base-to-string\n                    this.form.controls.files.setValue([...(this.form.controls.files.value ?? []), new ScUploadedFile(file.name, reader.result.toString())]);\n                }\n            });\n\n            reader.readAsDataURL(file as unknown as Blob);\n        });\n    }\n\n    /**\n     * Проверяет количество файлов загруженных пользователем.\n     *\n     * @param maxLength максимальное допустимое количество файлов для отправки.\n     */\n    private maxFilesLength(maxLength: number = this.maxFilesInForm): ValidatorFn {\n        return ({ value }: AbstractControl<TuiFileLike[]>) =>\n            value.length > maxLength ? { maxLength: new TuiValidationError(`Максимальное количество файлов - ${maxLength}`) } : null;\n    }\n\n    /**\n     * Выполняет очистку полей формы.\n     */\n    private resetForm(): void {\n        this.form.reset({\n            email: null,\n            name: null,\n            phone: null,\n            message: null,\n            files: [],\n            isShouldUseMessenger: false,\n        });\n    }\n\n    /**\n     * Проверяет, что номер телефона подходит для использования в мессенджерах.\n     *\n     * @param value Номер телефона, который необходимо проверить.\n     * @returns Признак того, что номер телефона подходит для мессенджеров.\n     */\n    @tuiPure\n    protected isValidPhoneNumberForMessenger(value: string): boolean {\n        return ScPhoneService.isValidPhoneNumberForMessenger(value);\n    }\n}\n","@let fieldsList = fields();\n<form\n    *ngIf=\"fieldsList.length > 0\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"flex flex-col justify-between gap-4\"\n>\n    <!-- ФИО -->\n    <label\n        *ngIf=\"fieldsList.includes('name')\"\n        tuiLabel\n    >\n        ФИО\n        <tui-input formControlName=\"name\"> Укажите Ваши Ф.И.О. </tui-input>\n        <tui-error\n            formControlName=\"name\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Телефон -->\n    @if (fieldsList.includes('phone')) {\n        <label tuiLabel>\n            Телефон\n            <tui-textfield>\n                <input\n                    formControlName=\"phone\"\n                    tuiInputPhoneInternational\n                    [countrySearch]=\"true\"\n                    [countries]=\"(countries | tuiSortCountries | async) || []\"\n                    autocomplete=\"phone\"\n                />\n            </tui-textfield>\n            <tui-error\n                formControlName=\"phone\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        <label\n            *ngIf=\"fieldsList.includes('isShouldUseMessenger') && form.controls.phone.value && isValidPhoneNumberForMessenger(form.controls.phone.value)\"\n            tuiLabel\n        >\n            <input\n                formControlName=\"isShouldUseMessenger\"\n                tuiCheckbox\n                type=\"checkbox\"\n            />\n            <span tuiTitle> Написать в мессенджерах, а не звонить </span>\n        </label>\n    }\n    <!-- Адрес электронной почты -->\n    <label\n        *ngIf=\"fieldsList.includes('email')\"\n        tuiLabel\n    >\n        Адрес электронной почты\n        <tui-input formControlName=\"email\">\n            Адрес электронной почты\n            <sc-suggestion-field\n                *tuiDataList\n                [type]=\"suggestionType.email\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"email\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Комментарий -->\n    <label\n        *ngIf=\"fieldsList.includes('message')\"\n        tuiLabel\n    >\n        Комментарий\n        <tui-textarea formControlName=\"message\"> Комментарий </tui-textarea>\n        <tui-error\n            formControlName=\"message\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Файлы -->\n    <div\n        *ngIf=\"fieldsList.includes('files')\"\n        class=\"flex flex-col gap-1\"\n    >\n        <label tuiInputFiles>\n            <input\n                #validator=\"tuiInputFilesValidator\"\n                accept=\".doc,.docx,.pdf,image/*\"\n                tuiInputFiles\n                [formControl]=\"loadedFile\"\n                [multiple]=\"true\"\n                (reject)=\"onReject($event)\"\n            />\n        </label>\n\n        <tui-error\n            *ngIf=\"loadedFile.invalid\"\n            [error]=\"['maxLength'] | tuiFieldError | async\"\n            [formControl]=\"loadedFile\"\n        />\n\n        @let accepted = accepted$ | async;\n        <tui-files\n            *ngIf=\"accepted?.length || rejected.length\"\n            class=\"tui-space_top-1\"\n        >\n            <tui-file\n                *ngFor=\"let file of accepted\"\n                [file]=\"file\"\n                (remove)=\"onRemove(file)\"\n            />\n            <tui-file\n                *ngFor=\"let file of rejected\"\n                state=\"error\"\n                [file]=\"(file | tuiFileRejected: validator | async) || file\"\n                (remove)=\"onRemove(file)\"\n            />\n        </tui-files>\n    </div>\n\n    <!-- recaptcha -->\n    @if (form.get('recaptcha')) {\n        <div\n            class=\"mb-3 max-h-20 overflow-hidden duration-500\"\n            [ngClass]=\"{ '!max-h-0 !-m-2': form.get('recaptcha')?.valid }\"\n        >\n            <re-captcha formControlName=\"recaptcha\" />\n        </div>\n    }\n\n    <!-- Элементы управления  -->\n    <div class=\"flex flex-wrap items-center gap-3\">\n        <button\n            tuiButton\n            [disabled]=\"form.invalid\"\n            [loading]=\"!!(loading$ | async)\"\n            type=\"submit\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n        >\n            Отправить\n        </button>\n        <div class=\"grow basis-60 text-body-s\">\n            Нажимая на кнопку, вы соглашаетесь с\n            <button\n                tuiLink\n                [pseudo]=\"true\"\n                type=\"button\"\n                (click)=\"context?.data?.onClickOfferHandler(); clickOffer.emit()\"\n            >\n                Политикой конфиденциальности\n            </button>\n        </div>\n    </div>\n</form>\n"]}
344
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-feedback-form.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/feedback/feedback-form/sc-feedback-form.component.ts","../../../../../projects/client-ui/feedback/feedback-form/sc-feedback-form.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,@typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgClass, NgFor, NgIf } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, computed, effect, inject, input, InputSignal, output, OutputEmitterRef, Signal } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { AbstractControl, FormControl, FormGroup, ReactiveFormsModule, ValidatorFn, Validators } from '@angular/forms';\nimport {\n    ScConvertersService,\n    ScFeedbackForms,\n    ScFeedbackService,\n    ScIApiErrorResponse,\n    ScIFeedbackMessage,\n    ScISuggestionType,\n    ScPhoneService,\n    ScUploadedFile,\n    ScUser,\n    ScUserService,\n} from '@snabcentr/client-core';\nimport { TuiLooseUnion, tuiPure, TuiValidationError } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiError, TuiLabel, TuiLink, TuiTextfield, TuiTitle } from '@taiga-ui/core';\nimport { TuiInputPhoneInternational } from '@taiga-ui/experimental';\nimport { TuiCountryIsoCode } from '@taiga-ui/i18n';\nimport { TuiButtonLoading, TuiCheckbox, TuiFieldErrorPipe, TuiFileLike, TuiFiles, tuiFilesAccepted, TuiSortCountriesPipe } from '@taiga-ui/kit';\nimport { TuiInputModule, TuiInputPhoneModule, TuiTextareaModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { getCountries } from 'libphonenumber-js';\nimport { RecaptchaComponent, RecaptchaFormsModule, RecaptchaModule } from 'ng-recaptcha-2';\nimport { catchError, filter, map, merge, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ScFormFieldsModule } from '../../form-fields';\nimport { MAX_FILES_IN_FORM_INPUT } from '../../providers/sc-max-files-in-form-input';\nimport { phoneValidator } from '../../validators/sc-phone-validator';\nimport { ScFeedbackFormConfig, ScFeedbackFormFieldType } from '../interfaces';\n\n/**\n * Компонент формы обратной связи.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-feedback-form',\n    templateUrl: './sc-feedback-form.component.html',\n    imports: [\n        AsyncPipe,\n        ReactiveFormsModule,\n        NgClass,\n        NgFor,\n        TuiError,\n        TuiFieldErrorPipe,\n        ScFormFieldsModule,\n        TuiInputModule,\n        TuiInputPhoneModule,\n        TuiButton,\n        TuiButtonLoading,\n        TuiLabel,\n        RecaptchaFormsModule,\n        RecaptchaModule,\n        TuiLink,\n        NgIf,\n        TuiTextareaModule,\n        TuiCheckbox,\n        TuiTitle,\n        TuiFiles,\n        TuiInputPhoneInternational,\n        ...TuiTextfield,\n        TuiSortCountriesPipe,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScFeedbackFormComponent {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { onClickOfferHandler: () => void } & ScFeedbackFormConfig> | null = inject<\n        TuiDialogContext<void, { onClickOfferHandler: () => void } & ScFeedbackFormConfig>\n    >(POLYMORPHEUS_CONTEXT, { optional: true });\n\n    /**\n     * Символьное обозначение (slug) формы.\n     */\n    // eslint-disable-next-line @angular-eslint/no-input-rename\n    public readonly slugInput: InputSignal<TuiLooseUnion<ScFeedbackForms>> = input<TuiLooseUnion<ScFeedbackForms>>('common', { alias: 'slug' });\n\n    /**\n     * Список полей, доступных в данной форме.\n     */\n    // eslint-disable-next-line @angular-eslint/no-input-rename\n    public readonly fieldsInput: InputSignal<ScFeedbackFormFieldType[]> = input<ScFeedbackFormFieldType[]>(['name', 'email', 'phone', 'message', 'files'], { alias: 'fields' });\n\n    /**\n     * Символьное обозначение (slug) формы.\n     */\n    protected readonly slug: Signal<TuiLooseUnion<ScFeedbackForms>> = computed(() => this.context?.data.slug ?? this.slugInput());\n\n    /**\n     * Список полей, доступных в данной форме.\n     */\n    protected readonly fields: Signal<ScFeedbackFormFieldType[]> = computed(() => this.context?.data.fields ?? this.fieldsInput());\n\n    /**\n     * Событие нажатия на кнопку договора оферты.\n     */\n    public readonly clickOffer: OutputEmitterRef<void> = output();\n\n    /**\n     * Событие успешной отправки формы.\n     */\n    public readonly sendSuccess: OutputEmitterRef<void> = output();\n\n    /**\n     * Сервис обратной связи.\n     */\n    private readonly feedbackService: ScFeedbackService = inject(ScFeedbackService);\n\n    /**\n     * Сервис конвертации данных.\n     */\n    private readonly convertersService: ScConvertersService = inject(ScConvertersService);\n\n    /**\n     * Максимальное количество файлов, отправляемых в форме.\n     */\n    private readonly maxFilesInForm: number = inject(MAX_FILES_IN_FORM_INPUT);\n\n    /**\n     * Сервис данных о пользователе.\n     */\n    private readonly user$: Observable<ScUser> = inject(ScUserService)\n        .getUserChange$()\n        .pipe(\n            tap((user) => {\n                if (user.isGuest) {\n                    (this.form as FormGroup).addControl('recaptcha', this.recaptchaControl);\n                    this.resetForm();\n                } else {\n                    (this.form as FormGroup).removeControl('recaptcha');\n\n                    const fields = this.fields();\n                    this.form.patchValue({\n                        name: fields.includes('name') ? user.name : null,\n                        phone: fields.includes('phone') ? user.contacts.phone.value : null,\n                        email: fields.includes('email') ? user.contacts.email.value : null,\n                    });\n                }\n            })\n        );\n\n    /**\n     * Поле ввода для recaptcha.\n     */\n    protected readonly recaptchaControl: FormControl<string | null> = new FormControl<string | null>(null, Validators.required);\n\n    /**\n     * Группа полей ввода для формы «Пожелания и предложения по улучшению сайта».\n     */\n    protected readonly form = new FormGroup({\n        name: new FormControl<string | null>(null),\n        phone: new FormControl<string | null>(null),\n        email: new FormControl<string | null>(null),\n        message: new FormControl<string | null>(null),\n        files: new FormControl<ScUploadedFile[] | null>([]),\n        isShouldUseMessenger: new FormControl<boolean | null>(false),\n    });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на отправку данных.\n     */\n    protected readonly request$: Observable<undefined | null> = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIFeedbackMessage),\n        tap((value: ScIFeedbackMessage) => {\n            if (value.phone && !ScPhoneService.isValidPhoneNumberForMessenger(value.phone)) {\n                // eslint-disable-next-line no-param-reassign\n                delete value.isShouldUseMessenger;\n            }\n        }),\n        switchMap((value) =>\n            this.feedbackService.sendFeedback(this.slug() as ScFeedbackForms, this.convertersService.removeNullRecursive(value)).pipe(\n                tap(() => {\n                    if (this.context) {\n                        this.context.completeWith();\n                    } else {\n                        this.resetForm();\n                        this.onRemoveAll();\n                        this.sendSuccess.emit();\n                    }\n                }),\n                // eslint-disable-next-line rxjs/no-implicit-any-catch\n                catchError((error: HttpErrorResponse) => {\n                    this.form.markAllAsTouched();\n\n                    const { errors, message } = error.error as ScIApiErrorResponse;\n\n                    if (errors) {\n                        Object.keys(errors).forEach((key) => {\n                            const control = this.form.get(key);\n\n                            if (control) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                control.setErrors({ serverResponse: errors[key] });\n                            }\n                        });\n                    } else if (message) {\n                        this.form.setErrors({ serverResponse: [message] });\n                    }\n\n                    return of(null);\n                }),\n                // eslint-disable-next-line unicorn/no-useless-undefined\n                startWith(undefined)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных.\n     */\n    protected readonly loading$: Observable<boolean> = this.request$.pipe(map((value) => value === undefined));\n\n    /**\n     * Управляет состоянием загруженных файлов.\n     */\n    protected readonly loadedFile: FormControl<TuiFileLike[] | null> = new FormControl<TuiFileLike[] | null>([], [this.maxFilesLength()]);\n\n    /**\n     * Observable, который отслеживает, приняты ли файлы в соответствии с требованиями по типу размеру и т.д.\n     */\n    protected readonly accepted$: Observable<File[]> = this.loadedFile.valueChanges.pipe(map(() => tuiFilesAccepted(this.loadedFile)));\n\n    /**\n     * Список файлов, которые были отклонены при загрузке.\n     */\n    protected rejected: readonly File[] = [];\n\n    /**\n     * Список стран.\n     */\n    protected readonly countries: readonly TuiCountryIsoCode[] = getCountries();\n\n    /**\n     * Инициализирует экземпляр класса {@link ScFeedbackFormComponent}.\n     */\n    public constructor() {\n        /**\n         * Это решение ошибки #4940, было @see [предложено](https://github.com/DethAriel/ng-recaptcha/issues/123#issuecomment-426112101) автором {@link RecaptchaComponent}.\n         */\n        RecaptchaComponent.prototype.ngOnDestroy = function (): void {\n            // eslint-disable-next-line @typescript-eslint/dot-notation\n            if (this['subscription']) {\n                // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call, @typescript-eslint/dot-notation\n                this['subscription'].unsubscribe();\n            }\n        };\n\n        merge(\n            this.accepted$,\n            this.user$,\n            this.loadedFile.valueChanges.pipe(\n                tap((files) => {\n                    this.form.controls.files.reset([]);\n                    this.onUploadFile(files);\n                })\n            )\n        )\n            .pipe(takeUntilDestroyed())\n            .subscribe();\n\n        // Валидация полей ввода в зависимости от формы.\n        effect(() => {\n            this.clearAllValidators();\n            this.setAllValidators(this.fields());\n        });\n    }\n\n    /**\n     * Удаляет валидацию со всех полей ввода.\n     */\n    private clearAllValidators(): void {\n        Object.keys(this.form.controls).forEach((key) => {\n            const control = this.form.get(key);\n\n            if (control && control !== this.recaptchaControl) {\n                control.clearValidators();\n                control.updateValueAndValidity();\n            }\n        });\n    }\n\n    /**\n     * Устанавливает валидацию со для указанных полей ввода.\n     *\n     * @param fields Поля, для которых нужно установить валидацию.\n     */\n    private setAllValidators(fields: ScFeedbackFormFieldType[]): void {\n        fields.forEach((filed: ScFeedbackFormFieldType) => {\n            // eslint-disable-next-line default-case\n            switch (filed) {\n                case 'name':\n                    this.form.get('name')?.setValidators([Validators.required, Validators.minLength(3)]);\n                    break;\n                case 'phone':\n                    this.form.get('phone')?.setValidators([Validators.required, phoneValidator()]);\n                    break;\n                case 'email':\n                    this.form.get('email')?.setValidators([Validators.required, Validators.email]);\n                    break;\n                case 'message':\n                    this.form.get('message')?.setValidators([Validators.required]);\n                    break;\n                case 'files':\n                    this.form.get('files')?.setValidators([Validators.required, this.maxFilesLength(5)]);\n                    break;\n            }\n        });\n    }\n\n    /**\n     * Обрабатывает отклоненные файлы.\n     *\n     * @param files Список файлов, которые были отклонены.\n     */\n    protected onReject(files: readonly File[]): void {\n        this.rejected = [...new Set([...this.rejected, ...files])];\n    }\n\n    /**\n     * Удаляет файл из списка загруженных и отклоненных файлов.\n     *\n     * @param file Файл, который нужно удалить.\n     */\n    protected onRemove(file: File): void {\n        this.rejected = this.rejected.filter((rejected) => rejected !== file);\n        this.loadedFile.setValue(this.loadedFile.value?.filter((current) => current !== file) ?? []);\n    }\n\n    /**\n     * Очищает список файлов.\n     */\n    protected onRemoveAll(): void {\n        this.rejected = [];\n        this.loadedFile.setValue([]);\n    }\n\n    /**\n     * Загружает файл в форму обратной связи.\n     *\n     * @param files Набор файлов.\n     */\n    private onUploadFile(files: TuiFileLike[] | null): void {\n        files?.forEach((file) => {\n            const reader = new FileReader();\n\n            reader.addEventListener('load', () => {\n                if (reader.result) {\n                    // eslint-disable-next-line @typescript-eslint/no-base-to-string\n                    this.form.controls.files.setValue([...(this.form.controls.files.value ?? []), new ScUploadedFile(file.name, reader.result.toString())]);\n                }\n            });\n\n            reader.readAsDataURL(file as unknown as Blob);\n        });\n    }\n\n    /**\n     * Проверяет количество файлов загруженных пользователем.\n     *\n     * @param maxLength максимальное допустимое количество файлов для отправки.\n     */\n    private maxFilesLength(maxLength: number = this.maxFilesInForm): ValidatorFn {\n        return ({ value }: AbstractControl<TuiFileLike[]>) =>\n            value.length > maxLength ? { maxLength: new TuiValidationError(`Максимальное количество файлов - ${maxLength}`) } : null;\n    }\n\n    /**\n     * Выполняет очистку полей формы.\n     */\n    private resetForm(): void {\n        this.form.reset({\n            email: null,\n            name: null,\n            phone: null,\n            message: null,\n            files: [],\n            isShouldUseMessenger: false,\n        });\n    }\n\n    /**\n     * Проверяет, что номер телефона подходит для использования в мессенджерах.\n     *\n     * @param value Номер телефона, который необходимо проверить.\n     * @returns Признак того, что номер телефона подходит для мессенджеров.\n     */\n    @tuiPure\n    protected isValidPhoneNumberForMessenger(value: string): boolean {\n        return ScPhoneService.isValidPhoneNumberForMessenger(value);\n    }\n}\n","@let fieldsList = fields();\n<form\n    *ngIf=\"fieldsList.length > 0\"\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"flex flex-col justify-between gap-4\"\n>\n    <!-- ФИО -->\n    <label\n        *ngIf=\"fieldsList.includes('name')\"\n        tuiLabel\n    >\n        ФИО\n        <tui-input formControlName=\"name\"> Укажите Ваши Ф.И.О. </tui-input>\n        <tui-error\n            formControlName=\"name\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Телефон -->\n    @if (fieldsList.includes('phone')) {\n        <label tuiLabel>\n            Телефон\n            <tui-textfield>\n                <input\n                    formControlName=\"phone\"\n                    tuiInputPhoneInternational\n                    [countrySearch]=\"true\"\n                    [countries]=\"(countries | tuiSortCountries | async) || []\"\n                    autocomplete=\"phone\"\n                />\n            </tui-textfield>\n            <tui-error\n                formControlName=\"phone\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n        </label>\n\n        <label\n            *ngIf=\"fieldsList.includes('isShouldUseMessenger') && form.controls.phone.value && isValidPhoneNumberForMessenger(form.controls.phone.value)\"\n            tuiLabel\n        >\n            <input\n                formControlName=\"isShouldUseMessenger\"\n                tuiCheckbox\n                type=\"checkbox\"\n            />\n            <span tuiTitle> Написать в мессенджерах, а не звонить </span>\n        </label>\n    }\n    <!-- Адрес электронной почты -->\n    <label\n        *ngIf=\"fieldsList.includes('email')\"\n        tuiLabel\n    >\n        Адрес электронной почты\n        <tui-input formControlName=\"email\">\n            Адрес электронной почты\n            <sc-suggestion-field\n                *tuiDataList\n                [type]=\"suggestionType.email\"\n            />\n        </tui-input>\n        <tui-error\n            formControlName=\"email\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Комментарий -->\n    <label\n        *ngIf=\"fieldsList.includes('message')\"\n        tuiLabel\n    >\n        Комментарий\n        <tui-textarea formControlName=\"message\"> Комментарий </tui-textarea>\n        <tui-error\n            formControlName=\"message\"\n            [error]=\"[] | tuiFieldError | async\"\n        />\n    </label>\n\n    <!-- Файлы -->\n    <div\n        *ngIf=\"fieldsList.includes('files')\"\n        class=\"flex flex-col gap-1\"\n    >\n        <label tuiInputFiles>\n            <input\n                #validator=\"tuiInputFilesValidator\"\n                accept=\".doc,.docx,.pdf,image/*\"\n                tuiInputFiles\n                [formControl]=\"loadedFile\"\n                [multiple]=\"true\"\n                (reject)=\"onReject($event)\"\n            />\n        </label>\n\n        <tui-error\n            *ngIf=\"loadedFile.invalid\"\n            [error]=\"['maxLength'] | tuiFieldError | async\"\n            [formControl]=\"loadedFile\"\n        />\n\n        @let accepted = accepted$ | async;\n        <tui-files\n            *ngIf=\"accepted?.length || rejected.length\"\n            class=\"tui-space_top-1\"\n        >\n            <tui-file\n                *ngFor=\"let file of accepted\"\n                [file]=\"file\"\n                (remove)=\"onRemove(file)\"\n            />\n            <tui-file\n                *ngFor=\"let file of rejected\"\n                state=\"error\"\n                [file]=\"(file | tuiFileRejected: validator | async) || file\"\n                (remove)=\"onRemove(file)\"\n            />\n        </tui-files>\n    </div>\n\n    <!-- recaptcha -->\n    @if (form.get('recaptcha')) {\n        <div\n            class=\"mb-3 max-h-20 overflow-hidden duration-500\"\n            [ngClass]=\"{ '!max-h-0 !-m-2': form.get('recaptcha')?.valid }\"\n        >\n            <re-captcha formControlName=\"recaptcha\" />\n        </div>\n    }\n\n    <!-- Элементы управления  -->\n    <div class=\"flex flex-wrap items-center gap-3\">\n        <button\n            tuiButton\n            [disabled]=\"form.invalid\"\n            [loading]=\"!!(loading$ | async)\"\n            type=\"submit\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n        >\n            Отправить\n        </button>\n        <div class=\"grow basis-60 text-body-s\">\n            Нажимая на кнопку, вы соглашаетесь с\n            <button\n                tuiLink\n                [pseudo]=\"true\"\n                type=\"button\"\n                (click)=\"context?.data?.onClickOfferHandler(); clickOffer.emit()\"\n            >\n                Политикой конфиденциальности\n            </button>\n        </div>\n    </div>\n</form>\n"]}
@@ -156,11 +156,11 @@ export class ScSuggestionFieldComponent {
156
156
  return suggestion.rawData?.['state']?.status === 'LIQUIDATED';
157
157
  }
158
158
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSuggestionFieldComponent, deps: [{ token: NgControl, skipSelf: true }, { token: i1.ScSuggestionService }], target: i0.ɵɵFactoryTarget.Component }); }
159
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: { type: "type" }, outputs: { selectedClick: "selectedClick" }, ngImport: i0, template: "<tui-loader\n size=\"m\"\n [overlay]=\"true\"\n [showLoader]=\"!!(loading$ | async)\"\n>\n @let options = suggestions$ | async;\n <tui-data-list>\n <button\n *ngFor=\"let option of options\"\n tuiOption\n (click)=\"onSelected(option)\"\n [disabled]=\"isDisabledSuggestion(option)\"\n class=\"flex-col !items-start\"\n >\n @let optionValue = $any(option);\n <ng-container>\n <ng-container [ngSwitch]=\"type\">\n <ng-container *ngSwitchCase=\"suggestionType.fio\">\n <span class=\"text-tui-link\">{{ optionValue.fio }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.organization\">\n <span class=\"text-tui-link\">\n {{ optionValue.name }}\n <span class=\"ml-1 text-tui-text-02\">\u0438\u043D\u043D: {{ optionValue.inn }}</span>\n <span\n *ngIf=\"isOrgTerminated(optionValue)\"\n class=\"text-tui-negative\"\n >\n \u041B\u0438\u043A\u0432\u0438\u0434\u0438\u0440\u043E\u0432\u0430\u043D\u043E\n </span>\n </span>\n <span class=\"text-tui-text-02\">{{ optionValue?.legalAddress }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.bank\">\n <span class=\"text-tui-link\">{{ optionValue.name }}</span>\n <span class=\"text-tui-text-02\">{{ optionValue.bic }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.email\">\n <span class=\"text-tui-link\">{{ optionValue.email }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.address\">\n <span class=\"text-tui-link\">{{ optionValue.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.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i2.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i3.TuiDataListComponent, selector: "tui-data-list", inputs: ["emptyContent", "size"] }, { kind: "component", type: i3.TuiOption, selector: "button[tuiOption]:not([new]), a[tuiOption]:not([new]), label[tuiOption]:not([new])", inputs: ["disabled", "value"] }, { kind: "component", type: i3.TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
159
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: { type: "type" }, outputs: { selectedClick: "selectedClick" }, ngImport: i0, template: "<tui-loader\n size=\"m\"\n [overlay]=\"true\"\n [showLoader]=\"!!(loading$ | async)\"\n>\n @let options = suggestions$ | async;\n <tui-data-list>\n <button\n *ngFor=\"let option of options\"\n tuiOption\n (click)=\"onSelected(option)\"\n [disabled]=\"isDisabledSuggestion(option)\"\n class=\"flex-col !items-start\"\n >\n @let optionValue = $any(option);\n <ng-container [ngSwitch]=\"type\">\n <ng-container *ngSwitchCase=\"suggestionType.fio\">\n <span class=\"text-tui-link\">{{ optionValue.fio }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.organization\">\n <span class=\"text-tui-link\">\n {{ optionValue.name }}\n <span class=\"ml-1 text-tui-text-02\">\u0438\u043D\u043D: {{ optionValue.inn }}</span>\n <span\n *ngIf=\"isOrgTerminated(optionValue)\"\n class=\"text-tui-negative\"\n >\n \u041B\u0438\u043A\u0432\u0438\u0434\u0438\u0440\u043E\u0432\u0430\u043D\u043E\n </span>\n </span>\n <span class=\"text-tui-text-02\">{{ optionValue?.legalAddress }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.bank\">\n <span class=\"text-tui-link\">{{ optionValue.name }}</span>\n <span class=\"text-tui-text-02\">{{ optionValue.bic }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.email\">\n <span class=\"text-tui-link\">{{ optionValue.email }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.address\">\n <span class=\"text-tui-link\">{{ optionValue.addressString }}</span>\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.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i2.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "component", type: i3.TuiDataListComponent, selector: "tui-data-list", inputs: ["emptyContent", "size"] }, { kind: "component", type: i3.TuiOption, selector: "button[tuiOption]:not([new]), a[tuiOption]:not([new]), label[tuiOption]:not([new])", inputs: ["disabled", "value"] }, { kind: "component", type: i3.TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
160
160
  }
161
161
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScSuggestionFieldComponent, decorators: [{
162
162
  type: Component,
163
- args: [{ selector: 'sc-suggestion-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<tui-loader\n size=\"m\"\n [overlay]=\"true\"\n [showLoader]=\"!!(loading$ | async)\"\n>\n @let options = suggestions$ | async;\n <tui-data-list>\n <button\n *ngFor=\"let option of options\"\n tuiOption\n (click)=\"onSelected(option)\"\n [disabled]=\"isDisabledSuggestion(option)\"\n class=\"flex-col !items-start\"\n >\n @let optionValue = $any(option);\n <ng-container>\n <ng-container [ngSwitch]=\"type\">\n <ng-container *ngSwitchCase=\"suggestionType.fio\">\n <span class=\"text-tui-link\">{{ optionValue.fio }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.organization\">\n <span class=\"text-tui-link\">\n {{ optionValue.name }}\n <span class=\"ml-1 text-tui-text-02\">\u0438\u043D\u043D: {{ optionValue.inn }}</span>\n <span\n *ngIf=\"isOrgTerminated(optionValue)\"\n class=\"text-tui-negative\"\n >\n \u041B\u0438\u043A\u0432\u0438\u0434\u0438\u0440\u043E\u0432\u0430\u043D\u043E\n </span>\n </span>\n <span class=\"text-tui-text-02\">{{ optionValue?.legalAddress }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.bank\">\n <span class=\"text-tui-link\">{{ optionValue.name }}</span>\n <span class=\"text-tui-text-02\">{{ optionValue.bic }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.email\">\n <span class=\"text-tui-link\">{{ optionValue.email }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.address\">\n <span class=\"text-tui-link\">{{ optionValue.addressString }}</span>\n </ng-container>\n </ng-container>\n </ng-container>\n </button>\n </tui-data-list>\n</tui-loader>\n" }]
163
+ args: [{ selector: 'sc-suggestion-field', changeDetection: ChangeDetectionStrategy.OnPush, template: "<tui-loader\n size=\"m\"\n [overlay]=\"true\"\n [showLoader]=\"!!(loading$ | async)\"\n>\n @let options = suggestions$ | async;\n <tui-data-list>\n <button\n *ngFor=\"let option of options\"\n tuiOption\n (click)=\"onSelected(option)\"\n [disabled]=\"isDisabledSuggestion(option)\"\n class=\"flex-col !items-start\"\n >\n @let optionValue = $any(option);\n <ng-container [ngSwitch]=\"type\">\n <ng-container *ngSwitchCase=\"suggestionType.fio\">\n <span class=\"text-tui-link\">{{ optionValue.fio }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.organization\">\n <span class=\"text-tui-link\">\n {{ optionValue.name }}\n <span class=\"ml-1 text-tui-text-02\">\u0438\u043D\u043D: {{ optionValue.inn }}</span>\n <span\n *ngIf=\"isOrgTerminated(optionValue)\"\n class=\"text-tui-negative\"\n >\n \u041B\u0438\u043A\u0432\u0438\u0434\u0438\u0440\u043E\u0432\u0430\u043D\u043E\n </span>\n </span>\n <span class=\"text-tui-text-02\">{{ optionValue?.legalAddress }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.bank\">\n <span class=\"text-tui-link\">{{ optionValue.name }}</span>\n <span class=\"text-tui-text-02\">{{ optionValue.bic }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.email\">\n <span class=\"text-tui-link\">{{ optionValue.email }}</span>\n </ng-container>\n <ng-container *ngSwitchCase=\"suggestionType.address\">\n <span class=\"text-tui-link\">{{ optionValue.addressString }}</span>\n </ng-container>\n </ng-container>\n </button>\n </tui-data-list>\n</tui-loader>\n" }]
164
164
  }], ctorParameters: () => [{ type: i4.NgControl, decorators: [{
165
165
  type: SkipSelf
166
166
  }, {
@@ -171,4 +171,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
171
171
  }], selectedClick: [{
172
172
  type: Output
173
173
  }] } });
174
- //# 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":"AAAA,2CAA2C;AAE3C,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAClI,OAAO,EAAa,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACtD,OAAO,EAAE,yBAAyB,EAA6B,iBAAiB,EAAqC,MAAM,wBAAwB,CAAC;AACpJ,OAAO,EAAE,eAAe,EAAE,UAAU,EAAE,YAAY,EAAE,kCAAkC,EAAE,MAAM,eAAe,CAAC;AAC9G,OAAO,EAAE,kBAAkB,EAAmB,MAAM,gBAAgB,CAAC;AACrE,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAGtH,OAAO,EAAE,0BAA0B,EAAE,MAAM,4CAA4C,CAAC;;;;;;AAExF;;GAEG;AAMH,MAAM,OAAO,0BAA0B;IA6CnC;;;;;OAKG;IACH,YAGoB,OAAkB,EACjB,iBAAsC;QADvC,YAAO,GAAP,OAAO,CAAW;QACjB,sBAAiB,GAAjB,iBAAiB,CAAqB;QAtD3D;;WAEG;QAEI,SAAI,GAAsB,iBAAiB,CAAC,GAAG,CAAC;QAEvD;;WAEG;QAEI,kBAAa,GAA+B,IAAI,YAAY,EAAgB,CAAC;QAYpF;;WAEG;QACa,mBAAc,GAA6B,iBAAiB,CAAC;QAE7E;;WAEG;QACc,SAAI,GAAoC,MAAM,CAA2B,kBAAkB,EAAE;YAC1G,QAAQ,EAAE,IAAI;SACjB,CAAC,CAAC;QAEH;;WAEG;QACc,wBAAmB,GAAW,MAAM,CAAC,yBAAyB,CAAC,CAAC;QAEjF;;WAEG;QACc,0BAAqB,GAAY,MAAM,CAAC,0BAA0B,CAAC,CAAC;IAalF,CAAC;IAEJ,kBAAkB;IACX,QAAQ;QACX,MAAM,QAAQ,GAAG,eAAe,CAAgB,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,CAC9D,YAAY,CAAC,GAAG,CAAC,EACjB,MAAM,CAAC,YAAY,CAAC,EACpB,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,EAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,mBAAmB,CAAC,EACzD,SAAS,CAAC,CAAC,IAAY,EAAE,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAe,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAChH,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,aAAa,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAEtD,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;oBACvB,kCAAkC,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;oBAEzD,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC;wBAC3B,cAAc,EAAE,CAAC,aAAa,CAAC,OAAO,IAAI,kFAAkF,CAAC;qBAChI,CAAC,CAAC;gBACP,CAAC;YACL,CAAC;YAED,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QACpC,CAAC,CAAC,EACF,KAAK,EAAE,CACV,CAAC;QAEF,IAAI,CAAC,YAAY,GAAG,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QACzE,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;IACnD,CAAC;IAED;;;;OAIG;IACI,UAAU,CAAC,UAAwB;QACtC,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAChB,KAAK,iBAAiB,CAAC,GAAG;gBACtB,IAAI,KAAK,IAAI,UAAU,EAAE,CAAC;oBACtB,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC3E,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,OAAO;gBAC1B,IAAI,eAAe,IAAI,UAAU,EAAE,CAAC;oBAChC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC,aAAa,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBACrF,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,IAAI;gBACvB,IAAI,KAAK,IAAI,UAAU,IAAI,KAAK,IAAI,UAAU,EAAE,CAAC;oBAC7C,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAoB,CAAA,CAAC,UAAU,CAClD;wBACI,QAAQ,EAAE,UAAU,CAAC,IAAI;wBACzB,GAAG,EAAE,UAAU,CAAC,GAAG;wBACnB,oBAAoB,EAAE,UAAU,CAAC,oBAAoB;qBACxD,EACD,EAAE,SAAS,EAAE,KAAK,EAAE,CACvB,CAAC;gBACN,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,YAAY;gBAC/B,IAAI,KAAK,IAAI,UAAU,IAAI,CAAC,CAAC,KAAK,IAAI,UAAU,CAAC,EAAE,CAAC;oBAChD,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAoB,CAAA,CAAC,UAAU,CAClD;wBACI,IAAI,EAAE,UAAU,CAAC,IAAI;wBACrB,GAAG,EAAE,UAAU,CAAC,GAAG;wBACnB,GAAG,EAAE,UAAU,CAAC,GAAG;wBACnB,IAAI,EAAE,UAAU,CAAC,IAAI;wBACrB,YAAY,EAAE,UAAU,CAAC,YAAY;wBACrC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB;wBAC7C,YAAY,EAAE,UAAU,CAAC,YAAY;wBACrC,aAAa,EAAE,UAAU,CAAC,aAAa;wBACvC,aAAa,EAAE,UAAU,CAAC,aAAa;qBAC1C,EACD,EAAE,SAAS,EAAE,KAAK,EAAE,CACvB,CAAC;gBACN,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,KAAK;gBACxB,IAAI,OAAO,IAAI,UAAU,EAAE,CAAC;oBACxB,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC7E,CAAC;gBACD,MAAM;YACV;gBACI,MAAM;QACd,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,EAAE,YAAY,EAAE,CAAC;YAC1B,+HAA+H;YAC/H,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAC/C,CAAC;QACD,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACxC,CAAC;IAED;;;;OAIG;IACI,oBAAoB,CAAC,UAAwB;QAChD,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAChB,KAAK,iBAAiB,CAAC,GAAG,CAAC;YAC3B,KAAK,iBAAiB,CAAC,OAAO,CAAC;YAC/B,KAAK,iBAAiB,CAAC,IAAI,CAAC;YAC5B,KAAK,iBAAiB,CAAC,KAAK;gBACxB,OAAO,KAAK,CAAC;YACjB,KAAK,iBAAiB,CAAC,YAAY;gBAC/B,IAAI,KAAK,IAAI,UAAU,IAAI,CAAC,CAAC,KAAK,IAAI,UAAU,CAAC,EAAE,CAAC;oBAChD,OAAO,CAAC,IAAI,CAAC,qBAAqB,IAAI,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;gBAC3E,CAAC;gBACD,MAAM;YACV;gBACI,OAAO,KAAK,CAAC;QACrB,CAAC;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;IAED;;;;OAIG;IACI,eAAe,CAAC,UAAqC;QACxD,sEAAsE;QACtE,OAAO,UAAU,CAAC,OAAO,EAAE,CAAC,OAAO,CAAC,EAAE,MAAM,KAAK,YAAY,CAAC;IAClE,CAAC;+GAvLQ,0BAA0B,kBAqDvB,SAAS;mGArDZ,0BAA0B,kICrBvC,mzEAgDA;;4FD3Ba,0BAA0B;kBALtC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BAsD1C,QAAQ;;0BACR,MAAM;2BAAC,SAAS;2EAhDd,IAAI;sBADV,KAAK;gBAOC,aAAa;sBADnB,MAAM","sourcesContent":["/* eslint-disable class-methods-use-this */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Inject, inject, Input, OnInit, Output, SkipSelf } from '@angular/core';\nimport { FormGroup, NgControl } from '@angular/forms';\nimport { SC_MIN_LENGTH_SEARCH_TERM, ScISuggestionOrganization, ScISuggestionType, ScSuggestion, ScSuggestionService } from '@snabcentr/client-core';\nimport { tuiControlValue, tuiIsFalsy, tuiIsPresent, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TUI_DATA_LIST_HOST, TuiDataListHost } from '@taiga-ui/core';\nimport { catchError, debounceTime, filter, map, Observable, of, share, startWith, switchMap, throwError } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { SC_ALLOW_SELECT_TERMINATED } from '../../providers/sc-allow-select-terminated';\n\n/**\n * Компонент подсказок для поля ввода.\n */\n@Component({\n    selector: 'sc-suggestion-field',\n    templateUrl: './sc-suggestion-field.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSuggestionFieldComponent implements OnInit {\n    /**\n     * Тип подсказок.\n     */\n    @Input()\n    public type: ScISuggestionType = ScISuggestionType.fio;\n\n    /**\n     * Событие выбора результата подсказки.\n     */\n    @Output()\n    public selectedClick: EventEmitter<ScSuggestion> = new EventEmitter<ScSuggestion>();\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     * Компонент обёртка выпадающего списка.\n     */\n    private readonly host: TuiDataListHost<unknown> | null = inject<TuiDataListHost<unknown>>(TUI_DATA_LIST_HOST, {\n        optional: true,\n    });\n\n    /**\n     * Минимальная длина поискового запроса для получения данных.\n     */\n    private readonly minLengthSearchTerm: number = inject(SC_MIN_LENGTH_SEARCH_TERM);\n\n    /**\n     * Признак, что разрешено выбирать записи ликвидированных организаций.\n     */\n    private readonly allowSelectTerminated: boolean = inject(SC_ALLOW_SELECT_TERMINATED);\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(500),\n            filter(tuiIsPresent),\n            map((term) => term.trim()),\n            filter((term) => term.length >= this.minLengthSearchTerm),\n            switchMap((term: string) => this.suggestionService.getData<ScSuggestion>(term, this.type).pipe(startWith(null))),\n            catchError((error: unknown) => {\n                if (error instanceof 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\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, { emitEvent: false });\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            default:\n                break;\n        }\n\n        if (this.host?.handleOption) {\n            // Дёргаем внутренний обработчик выбора элемента выпадающего списка для решения проблемы закрытия выпадающего списка подсказки.\n            this.host.handleOption(this.control.value);\n        }\n        this.selectedClick.emit(suggestion);\n    }\n\n    /**\n     * Возвращает признак, что элемент выпадающего списка подсказок отключён.\n     *\n     * @param suggestion Объект подсказки.\n     */\n    public isDisabledSuggestion(suggestion: ScSuggestion): boolean {\n        switch (this.type) {\n            case ScISuggestionType.fio:\n            case ScISuggestionType.address:\n            case ScISuggestionType.bank:\n            case ScISuggestionType.email:\n                return false;\n            case ScISuggestionType.organization:\n                if ('inn' in suggestion && !('bic' in suggestion)) {\n                    return !this.allowSelectTerminated && this.isOrgTerminated(suggestion);\n                }\n                break;\n            default:\n                return false;\n        }\n\n        return false;\n    }\n\n    /**\n     * Проверяет прекращена ли деятельность указанной организации.\n     *\n     * @param suggestion Объект подсказки по организации.\n     */\n    public isOrgTerminated(suggestion: ScISuggestionOrganization): boolean {\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n        return suggestion.rawData?.['state']?.status === 'LIQUIDATED';\n    }\n}\n","<tui-loader\n    size=\"m\"\n    [overlay]=\"true\"\n    [showLoader]=\"!!(loading$ | async)\"\n>\n    @let options = suggestions$ | async;\n    <tui-data-list>\n        <button\n            *ngFor=\"let option of options\"\n            tuiOption\n            (click)=\"onSelected(option)\"\n            [disabled]=\"isDisabledSuggestion(option)\"\n            class=\"flex-col !items-start\"\n        >\n            @let optionValue = $any(option);\n            <ng-container>\n                <ng-container [ngSwitch]=\"type\">\n                    <ng-container *ngSwitchCase=\"suggestionType.fio\">\n                        <span class=\"text-tui-link\">{{ optionValue.fio }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.organization\">\n                        <span class=\"text-tui-link\">\n                            {{ optionValue.name }}\n                            <span class=\"ml-1 text-tui-text-02\">инн: {{ optionValue.inn }}</span>\n                            <span\n                                *ngIf=\"isOrgTerminated(optionValue)\"\n                                class=\"text-tui-negative\"\n                            >\n                                Ликвидировано\n                            </span>\n                        </span>\n                        <span class=\"text-tui-text-02\">{{ optionValue?.legalAddress }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.bank\">\n                        <span class=\"text-tui-link\">{{ optionValue.name }}</span>\n                        <span class=\"text-tui-text-02\">{{ optionValue.bic }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.email\">\n                        <span class=\"text-tui-link\">{{ optionValue.email }}</span>\n                    </ng-container>\n                    <ng-container *ngSwitchCase=\"suggestionType.address\">\n                        <span class=\"text-tui-link\">{{ optionValue.addressString }}</span>\n                    </ng-container>\n                </ng-container>\n            </ng-container>\n        </button>\n    </tui-data-list>\n</tui-loader>\n"]}
174
+ //# 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":"AAAA,2CAA2C;AAE3C,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAClI,OAAO,EAAa,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACtD,OAAO,EAAE,yBAAyB,EAAkD,iBAAiB,EAAqC,MAAM,wBAAwB,CAAC;AACzK,OAAO,EAAE,eAAe,EAAE,UAAU,EAAE,YAAY,EAAE,kCAAkC,EAAE,MAAM,eAAe,CAAC;AAC9G,OAAO,EAAE,kBAAkB,EAAmB,MAAM,gBAAgB,CAAC;AACrE,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,EAAE,GAAG,EAAc,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAEtH,OAAO,EAAE,0BAA0B,EAAE,MAAM,4CAA4C,CAAC;;;;;;AAExF;;GAEG;AAMH,MAAM,OAAO,0BAA0B;IA6CnC;;;;;OAKG;IACH,YAGoB,OAAkB,EACjB,iBAAsC;QADvC,YAAO,GAAP,OAAO,CAAW;QACjB,sBAAiB,GAAjB,iBAAiB,CAAqB;QAtD3D;;WAEG;QAEI,SAAI,GAAsB,iBAAiB,CAAC,GAAG,CAAC;QAEvD;;WAEG;QAEI,kBAAa,GAA+B,IAAI,YAAY,EAAgB,CAAC;QAYpF;;WAEG;QACa,mBAAc,GAA6B,iBAAiB,CAAC;QAE7E;;WAEG;QACc,SAAI,GAAoC,MAAM,CAA2B,kBAAkB,EAAE;YAC1G,QAAQ,EAAE,IAAI;SACjB,CAAC,CAAC;QAEH;;WAEG;QACc,wBAAmB,GAAW,MAAM,CAAC,yBAAyB,CAAC,CAAC;QAEjF;;WAEG;QACc,0BAAqB,GAAY,MAAM,CAAC,0BAA0B,CAAC,CAAC;IAalF,CAAC;IAEJ,kBAAkB;IACX,QAAQ;QACX,MAAM,QAAQ,GAAG,eAAe,CAAgB,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,CAC9D,YAAY,CAAC,GAAG,CAAC,EACjB,MAAM,CAAC,YAAY,CAAC,EACpB,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,EAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,mBAAmB,CAAC,EACzD,SAAS,CAAC,CAAC,IAAY,EAAE,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAe,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAChH,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,aAAa,GAAG,KAAK,CAAC,KAA4B,CAAC;gBAEzD,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;oBACvB,kCAAkC,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;oBAEzD,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC;wBAC3B,cAAc,EAAE,CAAC,aAAa,CAAC,OAAO,IAAI,kFAAkF,CAAC;qBAChI,CAAC,CAAC;gBACP,CAAC;YACL,CAAC;YAED,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QACpC,CAAC,CAAC,EACF,KAAK,EAAE,CACV,CAAC;QAEF,IAAI,CAAC,YAAY,GAAG,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,EAAE,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QACzE,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;IACnD,CAAC;IAED;;;;OAIG;IACI,UAAU,CAAC,UAAwB;QACtC,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAChB,KAAK,iBAAiB,CAAC,GAAG;gBACtB,IAAI,KAAK,IAAI,UAAU,EAAE,CAAC;oBACtB,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC3E,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,OAAO;gBAC1B,IAAI,eAAe,IAAI,UAAU,EAAE,CAAC;oBAChC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC,aAAa,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBACrF,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,IAAI;gBACvB,IAAI,KAAK,IAAI,UAAU,IAAI,KAAK,IAAI,UAAU,EAAE,CAAC;oBAC7C,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAoB,CAAA,CAAC,UAAU,CAClD;wBACI,QAAQ,EAAE,UAAU,CAAC,IAAI;wBACzB,GAAG,EAAE,UAAU,CAAC,GAAG;wBACnB,oBAAoB,EAAE,UAAU,CAAC,oBAAoB;qBACxD,EACD,EAAE,SAAS,EAAE,KAAK,EAAE,CACvB,CAAC;gBACN,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,YAAY;gBAC/B,IAAI,KAAK,IAAI,UAAU,IAAI,CAAC,CAAC,KAAK,IAAI,UAAU,CAAC,EAAE,CAAC;oBAChD,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAoB,CAAA,CAAC,UAAU,CAClD;wBACI,IAAI,EAAE,UAAU,CAAC,IAAI;wBACrB,GAAG,EAAE,UAAU,CAAC,GAAG;wBACnB,GAAG,EAAE,UAAU,CAAC,GAAG;wBACnB,IAAI,EAAE,UAAU,CAAC,IAAI;wBACrB,YAAY,EAAE,UAAU,CAAC,YAAY;wBACrC,gBAAgB,EAAE,UAAU,CAAC,gBAAgB;wBAC7C,YAAY,EAAE,UAAU,CAAC,YAAY;wBACrC,aAAa,EAAE,UAAU,CAAC,aAAa;wBACvC,aAAa,EAAE,UAAU,CAAC,aAAa;qBAC1C,EACD,EAAE,SAAS,EAAE,KAAK,EAAE,CACvB,CAAC;gBACN,CAAC;gBACD,MAAM;YACV,KAAK,iBAAiB,CAAC,KAAK;gBACxB,IAAI,OAAO,IAAI,UAAU,EAAE,CAAC;oBACxB,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC7E,CAAC;gBACD,MAAM;YACV;gBACI,MAAM;QACd,CAAC;QAED,IAAI,IAAI,CAAC,IAAI,EAAE,YAAY,EAAE,CAAC;YAC1B,+HAA+H;YAC/H,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAC/C,CAAC;QACD,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACxC,CAAC;IAED;;;;OAIG;IACI,oBAAoB,CAAC,UAAwB;QAChD,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;YAChB,KAAK,iBAAiB,CAAC,GAAG,CAAC;YAC3B,KAAK,iBAAiB,CAAC,OAAO,CAAC;YAC/B,KAAK,iBAAiB,CAAC,IAAI,CAAC;YAC5B,KAAK,iBAAiB,CAAC,KAAK;gBACxB,OAAO,KAAK,CAAC;YACjB,KAAK,iBAAiB,CAAC,YAAY;gBAC/B,IAAI,KAAK,IAAI,UAAU,IAAI,CAAC,CAAC,KAAK,IAAI,UAAU,CAAC,EAAE,CAAC;oBAChD,OAAO,CAAC,IAAI,CAAC,qBAAqB,IAAI,IAAI,CAAC,eAAe,CAAC,UAAU,CAAC,CAAC;gBAC3E,CAAC;gBACD,MAAM;YACV;gBACI,OAAO,KAAK,CAAC;QACrB,CAAC;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;IAED;;;;OAIG;IACI,eAAe,CAAC,UAAqC;QACxD,sEAAsE;QACtE,OAAO,UAAU,CAAC,OAAO,EAAE,CAAC,OAAO,CAAC,EAAE,MAAM,KAAK,YAAY,CAAC;IAClE,CAAC;+GAvLQ,0BAA0B,kBAqDvB,SAAS;mGArDZ,0BAA0B,kICpBvC,0oEA8CA;;4FD1Ba,0BAA0B;kBALtC,SAAS;+BACI,qBAAqB,mBAEd,uBAAuB,CAAC,MAAM;;0BAsD1C,QAAQ;;0BACR,MAAM;2BAAC,SAAS;2EAhDd,IAAI;sBADV,KAAK;gBAOC,aAAa;sBADnB,MAAM","sourcesContent":["/* eslint-disable class-methods-use-this */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, EventEmitter, Inject, inject, Input, OnInit, Output, SkipSelf } from '@angular/core';\nimport { FormGroup, NgControl } from '@angular/forms';\nimport { SC_MIN_LENGTH_SEARCH_TERM, ScIApiErrorResponse, ScISuggestionOrganization, ScISuggestionType, ScSuggestion, ScSuggestionService } from '@snabcentr/client-core';\nimport { tuiControlValue, tuiIsFalsy, tuiIsPresent, tuiMarkControlAsTouchedAndValidate } from '@taiga-ui/cdk';\nimport { TUI_DATA_LIST_HOST, TuiDataListHost } from '@taiga-ui/core';\nimport { catchError, debounceTime, filter, map, Observable, of, share, startWith, switchMap, throwError } from 'rxjs';\n\nimport { SC_ALLOW_SELECT_TERMINATED } from '../../providers/sc-allow-select-terminated';\n\n/**\n * Компонент подсказок для поля ввода.\n */\n@Component({\n    selector: 'sc-suggestion-field',\n    templateUrl: './sc-suggestion-field.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScSuggestionFieldComponent implements OnInit {\n    /**\n     * Тип подсказок.\n     */\n    @Input()\n    public type: ScISuggestionType = ScISuggestionType.fio;\n\n    /**\n     * Событие выбора результата подсказки.\n     */\n    @Output()\n    public selectedClick: EventEmitter<ScSuggestion> = new EventEmitter<ScSuggestion>();\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     * Компонент обёртка выпадающего списка.\n     */\n    private readonly host: TuiDataListHost<unknown> | null = inject<TuiDataListHost<unknown>>(TUI_DATA_LIST_HOST, {\n        optional: true,\n    });\n\n    /**\n     * Минимальная длина поискового запроса для получения данных.\n     */\n    private readonly minLengthSearchTerm: number = inject(SC_MIN_LENGTH_SEARCH_TERM);\n\n    /**\n     * Признак, что разрешено выбирать записи ликвидированных организаций.\n     */\n    private readonly allowSelectTerminated: boolean = inject(SC_ALLOW_SELECT_TERMINATED);\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(500),\n            filter(tuiIsPresent),\n            map((term) => term.trim()),\n            filter((term) => term.length >= this.minLengthSearchTerm),\n            switchMap((term: string) => this.suggestionService.getData<ScSuggestion>(term, this.type).pipe(startWith(null))),\n            catchError((error: unknown) => {\n                if (error instanceof HttpErrorResponse) {\n                    const errorResponse = error.error as ScIApiErrorResponse;\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\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, { emitEvent: false });\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            default:\n                break;\n        }\n\n        if (this.host?.handleOption) {\n            // Дёргаем внутренний обработчик выбора элемента выпадающего списка для решения проблемы закрытия выпадающего списка подсказки.\n            this.host.handleOption(this.control.value);\n        }\n        this.selectedClick.emit(suggestion);\n    }\n\n    /**\n     * Возвращает признак, что элемент выпадающего списка подсказок отключён.\n     *\n     * @param suggestion Объект подсказки.\n     */\n    public isDisabledSuggestion(suggestion: ScSuggestion): boolean {\n        switch (this.type) {\n            case ScISuggestionType.fio:\n            case ScISuggestionType.address:\n            case ScISuggestionType.bank:\n            case ScISuggestionType.email:\n                return false;\n            case ScISuggestionType.organization:\n                if ('inn' in suggestion && !('bic' in suggestion)) {\n                    return !this.allowSelectTerminated && this.isOrgTerminated(suggestion);\n                }\n                break;\n            default:\n                return false;\n        }\n\n        return false;\n    }\n\n    /**\n     * Проверяет прекращена ли деятельность указанной организации.\n     *\n     * @param suggestion Объект подсказки по организации.\n     */\n    public isOrgTerminated(suggestion: ScISuggestionOrganization): boolean {\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n        return suggestion.rawData?.['state']?.status === 'LIQUIDATED';\n    }\n}\n","<tui-loader\n    size=\"m\"\n    [overlay]=\"true\"\n    [showLoader]=\"!!(loading$ | async)\"\n>\n    @let options = suggestions$ | async;\n    <tui-data-list>\n        <button\n            *ngFor=\"let option of options\"\n            tuiOption\n            (click)=\"onSelected(option)\"\n            [disabled]=\"isDisabledSuggestion(option)\"\n            class=\"flex-col !items-start\"\n        >\n            @let optionValue = $any(option);\n            <ng-container [ngSwitch]=\"type\">\n                <ng-container *ngSwitchCase=\"suggestionType.fio\">\n                    <span class=\"text-tui-link\">{{ optionValue.fio }}</span>\n                </ng-container>\n                <ng-container *ngSwitchCase=\"suggestionType.organization\">\n                    <span class=\"text-tui-link\">\n                        {{ optionValue.name }}\n                        <span class=\"ml-1 text-tui-text-02\">инн: {{ optionValue.inn }}</span>\n                        <span\n                            *ngIf=\"isOrgTerminated(optionValue)\"\n                            class=\"text-tui-negative\"\n                        >\n                            Ликвидировано\n                        </span>\n                    </span>\n                    <span class=\"text-tui-text-02\">{{ optionValue?.legalAddress }}</span>\n                </ng-container>\n                <ng-container *ngSwitchCase=\"suggestionType.bank\">\n                    <span class=\"text-tui-link\">{{ optionValue.name }}</span>\n                    <span class=\"text-tui-text-02\">{{ optionValue.bic }}</span>\n                </ng-container>\n                <ng-container *ngSwitchCase=\"suggestionType.email\">\n                    <span class=\"text-tui-link\">{{ optionValue.email }}</span>\n                </ng-container>\n                <ng-container *ngSwitchCase=\"suggestionType.address\">\n                    <span class=\"text-tui-link\">{{ optionValue.addressString }}</span>\n                </ng-container>\n            </ng-container>\n        </button>\n    </tui-data-list>\n</tui-loader>\n"]}
@@ -0,0 +1,117 @@
1
+ import { ChangeDetectionStrategy, Component, DestroyRef, inject, model, output } from '@angular/core';
2
+ import { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';
3
+ import { ActivatedRoute, Router, RouterLink } from '@angular/router';
4
+ import { SC_ORDER_LOADER, ScOrderDraftsService, scOrderIsLoaded, ScRouteKeys, ScSeoService } from '@snabcentr/client-core';
5
+ import { TuiButton, TuiDialogService, TuiIcons } from '@taiga-ui/core';
6
+ import { TUI_CONFIRM } from '@taiga-ui/kit';
7
+ import { distinctUntilChanged, filter, map, noop, switchMap } from 'rxjs';
8
+ import { ScOrderItemsListByDirectionsComponent } from '../order-items-list-by-directions/sc-order-items-list-by-directions.component';
9
+ import * as i0 from "@angular/core";
10
+ /**
11
+ * Компонент страницы черновика заказа (аналог корзины для одного черновика).
12
+ */
13
+ export class ScDraftComponent {
14
+ constructor() {
15
+ /**
16
+ * Сервис маршрутизации.
17
+ */
18
+ this.router = inject(Router);
19
+ /**
20
+ * Сервис предоставляющий доступ к данным о маршруте.
21
+ */
22
+ this.route = inject(ActivatedRoute);
23
+ /**
24
+ * Сервис для работы с черновиками заказов.
25
+ */
26
+ this.draftsService = inject(ScOrderDraftsService);
27
+ /**
28
+ * Сигнал с данными текущего черновика заказа (из параметра offerId в URL).
29
+ */
30
+ this.draft = model(null);
31
+ /**
32
+ * Идентификаторы выбранных позиций.
33
+ */
34
+ this.selectedItemsIds = model([]);
35
+ /**
36
+ * Источник события удаления или отмены заказа.
37
+ */
38
+ this.deleteEvent = output();
39
+ /**
40
+ * Сервис для работы с диалоговыми окнами.
41
+ */
42
+ this.dialogService = inject(TuiDialogService);
43
+ /**
44
+ * Текущий ключ маршрута.
45
+ */
46
+ this.isDraftPage = toSignal(this.route.data.pipe(map((data) => data['routes_key'] === ScRouteKeys.draft), distinctUntilChanged()), { initialValue: false });
47
+ /**
48
+ * Ссылка для автоматического управления уничтожением зависимостей.
49
+ */
50
+ this.destroyRef = inject(DestroyRef);
51
+ /**
52
+ * Сервис установки SEO-данных приложения.
53
+ */
54
+ this.seoService = inject(ScSeoService);
55
+ }
56
+ /** @inheritDoc */
57
+ ngOnInit() {
58
+ if (!this.draft()) {
59
+ this.draftsService
60
+ .get$(Number.parseInt(this.route.snapshot.paramMap.get('offerId') ?? '', 10))
61
+ .pipe(filter(scOrderIsLoaded), takeUntilDestroyed(this.destroyRef))
62
+ .subscribe((draft) => {
63
+ this.draft.set(draft);
64
+ this.seoService.setByDefaultAsNoIndexed('Черновики заказов');
65
+ });
66
+ }
67
+ }
68
+ /**
69
+ * Переход к оформлению заказа по выбранному направлению продаж.
70
+ *
71
+ * @param direction Группа позиций по направлению продаж.
72
+ */
73
+ goToNewOrder(direction) {
74
+ const ids = this.selectedItemsIds().filter((id) => direction.items.some((item) => item.id === id));
75
+ this.router.navigate(['orders', 'new', this.draft()?.id, direction.direction.id, 'draft'], { queryParams: { itemsIds: ids } }).catch(noop);
76
+ }
77
+ /**
78
+ * Удаляет черновик и перенаправляет к списку черновиков.
79
+ */
80
+ clearDraft() {
81
+ const { id } = this.draft() ?? {};
82
+ if (!id) {
83
+ return;
84
+ }
85
+ this.dialogService
86
+ .open(TUI_CONFIRM, {
87
+ label: `Удалить черновик №${id}?`,
88
+ data: {
89
+ content: `Это действие удалит черновик №${id}.`,
90
+ yes: 'Да, удалить',
91
+ no: 'Нет, оставить',
92
+ },
93
+ })
94
+ .pipe(filter(Boolean), switchMap(() => this.draftsService.delete$(id)), takeUntilDestroyed(this.destroyRef))
95
+ .subscribe(() => {
96
+ this.deleteEvent.emit();
97
+ this.router.navigate(['/drafts']).catch(noop);
98
+ });
99
+ }
100
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScDraftComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
101
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ScDraftComponent, isStandalone: true, selector: "sc-draft", inputs: { draft: { classPropertyName: "draft", publicName: "draft", isSignal: true, isRequired: false, transformFunction: null }, selectedItemsIds: { classPropertyName: "selectedItemsIds", publicName: "selectedItemsIds", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { draft: "draftChange", selectedItemsIds: "selectedItemsIdsChange", deleteEvent: "deleteEvent" }, providers: [
102
+ {
103
+ provide: SC_ORDER_LOADER,
104
+ useExisting: ScOrderDraftsService,
105
+ },
106
+ ], ngImport: i0, template: "@let items = draft()?.products;\n\n@if (items) {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @if (items.length) {\n <sc-order-items-list-by-directions\n [items]=\"items\"\n [editable]=\"isDraftPage()\"\n [selectable]=\"true\"\n [continueButtonText]=\"'\u041E\u0444\u043E\u0440\u043C\u0438\u0442\u044C \u0437\u0430\u043A\u0430\u0437'\"\n (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n (continueClick)=\"goToNewOrder($event)\"\n >\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <a\n tuiButton\n appearance=\"secondary\"\n iconStart=\"@tui.shopping-cart\"\n routerLink=\"/catalog\"\n class=\"self-center\"\n >\n \u041F\u0435\u0440\u0435\u0439\u0442\u0438 \u0432 \u043A\u0430\u0442\u0430\u043B\u043E\u0433\n </a>\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n </sc-order-items-list-by-directions>\n } @else {\n <p class=\"text-base font-bold\">\u0427\u0435\u0440\u043D\u043E\u0432\u0438\u043A \u043F\u0443\u0441\u0442.</p>\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <a\n tuiButton\n appearance=\"secondary\"\n iconStart=\"@tui.shopping-cart\"\n routerLink=\"/catalog\"\n class=\"self-center\"\n >\n \u041F\u0435\u0440\u0435\u0439\u0442\u0438 \u0432 \u043A\u0430\u0442\u0430\u043B\u043E\u0433\n </a>\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n }\n </div>\n </div>\n} @else {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @for (item of [1, 2, 3, 4]; track item) {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n }\n <div class=\"flex gap-2\">\n <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n </div>\n </div>\n </div>\n}\n", dependencies: [{ kind: "component", type: ScOrderItemsListByDirectionsComponent, selector: "sc-order-items-list-by-directions", inputs: ["items", "selectable", "editable", "showContinueButton", "continueButtonText", "selectedItemsIds"], outputs: ["continueClick", "selectedItemsIdsChange"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
107
+ }
108
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ScDraftComponent, decorators: [{
109
+ type: Component,
110
+ args: [{ standalone: true, selector: 'sc-draft', imports: [ScOrderItemsListByDirectionsComponent, RouterLink, TuiButton, TuiIcons], providers: [
111
+ {
112
+ provide: SC_ORDER_LOADER,
113
+ useExisting: ScOrderDraftsService,
114
+ },
115
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "@let items = draft()?.products;\n\n@if (items) {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @if (items.length) {\n <sc-order-items-list-by-directions\n [items]=\"items\"\n [editable]=\"isDraftPage()\"\n [selectable]=\"true\"\n [continueButtonText]=\"'\u041E\u0444\u043E\u0440\u043C\u0438\u0442\u044C \u0437\u0430\u043A\u0430\u0437'\"\n (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n (continueClick)=\"goToNewOrder($event)\"\n >\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <a\n tuiButton\n appearance=\"secondary\"\n iconStart=\"@tui.shopping-cart\"\n routerLink=\"/catalog\"\n class=\"self-center\"\n >\n \u041F\u0435\u0440\u0435\u0439\u0442\u0438 \u0432 \u043A\u0430\u0442\u0430\u043B\u043E\u0433\n </a>\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n </sc-order-items-list-by-directions>\n } @else {\n <p class=\"text-base font-bold\">\u0427\u0435\u0440\u043D\u043E\u0432\u0438\u043A \u043F\u0443\u0441\u0442.</p>\n <div\n footerActions\n class=\"flex gap-2\"\n >\n <a\n tuiButton\n appearance=\"secondary\"\n iconStart=\"@tui.shopping-cart\"\n routerLink=\"/catalog\"\n class=\"self-center\"\n >\n \u041F\u0435\u0440\u0435\u0439\u0442\u0438 \u0432 \u043A\u0430\u0442\u0430\u043B\u043E\u0433\n </a>\n <button\n tuiButton\n appearance=\"secondary\"\n (click)=\"clearDraft()\"\n iconStart=\"@tui.trash-2\"\n >\n \u0423\u0434\u0430\u043B\u0438\u0442\u044C \u0447\u0435\u0440\u043D\u043E\u0432\u0438\u043A\n </button>\n </div>\n }\n </div>\n </div>\n} @else {\n <div class=\"flex flex-col gap-12\">\n <div class=\"flex flex-col gap-5\">\n @for (item of [1, 2, 3, 4]; track item) {\n <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n }\n <div class=\"flex gap-2\">\n <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n </div>\n </div>\n </div>\n}\n" }]
116
+ }] });
117
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-draft.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/order/draft/sc-draft.component.ts","../../../../../projects/client-ui/order/draft/sc-draft.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, DestroyRef, inject, model, OnInit, output } from '@angular/core';\nimport { takeUntilDestroyed, toSignal } from '@angular/core/rxjs-interop';\nimport { ActivatedRoute, Router, RouterLink } from '@angular/router';\nimport { SC_ORDER_LOADER, ScOrderDraft, ScOrderDraftsService, scOrderIsLoaded, ScRouteKeys, ScSeoService } from '@snabcentr/client-core';\nimport { TuiButton, TuiDialogService, TuiIcons } from '@taiga-ui/core';\nimport { TUI_CONFIRM } from '@taiga-ui/kit';\nimport { distinctUntilChanged, filter, map, noop, switchMap } from 'rxjs';\n\nimport { ScOrderItemsByDirection } from '../models/sc-i-cart-items-by-direction';\nimport { ScOrderItemsListByDirectionsComponent } from '../order-items-list-by-directions/sc-order-items-list-by-directions.component';\n\n/**\n * Компонент страницы черновика заказа (аналог корзины для одного черновика).\n */\n@Component({\n    standalone: true,\n    selector: 'sc-draft',\n    templateUrl: './sc-draft.component.html',\n    imports: [ScOrderItemsListByDirectionsComponent, RouterLink, TuiButton, TuiIcons],\n    providers: [\n        {\n            provide: SC_ORDER_LOADER,\n            useExisting: ScOrderDraftsService,\n        },\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScDraftComponent implements OnInit {\n    /**\n     * Сервис маршрутизации.\n     */\n    private readonly router = inject(Router);\n\n    /**\n     * Сервис предоставляющий доступ к данным о маршруте.\n     */\n    private readonly route: ActivatedRoute = inject(ActivatedRoute);\n\n    /**\n     * Сервис для работы с черновиками заказов.\n     */\n    private readonly draftsService = inject(ScOrderDraftsService);\n\n    /**\n     * Сигнал с данными текущего черновика заказа (из параметра offerId в URL).\n     */\n    public readonly draft = model<ScOrderDraft | null>(null);\n\n    /**\n     * Идентификаторы выбранных позиций.\n     */\n    public readonly selectedItemsIds = model<string[]>([]);\n\n    /**\n     * Источник события удаления или отмены заказа.\n     */\n    public readonly deleteEvent = output();\n\n    /**\n     * Сервис для работы с диалоговыми окнами.\n     */\n    private readonly dialogService = inject<TuiDialogService>(TuiDialogService);\n\n    /**\n     * Текущий ключ маршрута.\n     */\n    protected readonly isDraftPage = toSignal(\n        this.route.data.pipe(\n            map((data) => (data['routes_key'] as ScRouteKeys) === ScRouteKeys.draft),\n            distinctUntilChanged()\n        ),\n        { initialValue: false }\n    );\n\n    /**\n     * Ссылка для автоматического управления уничтожением зависимостей.\n     */\n    private readonly destroyRef: DestroyRef = inject(DestroyRef);\n\n    /**\n     * Сервис установки SEO-данных приложения.\n     */\n    private readonly seoService: ScSeoService = inject(ScSeoService);\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (!this.draft()) {\n            this.draftsService\n                .get$(Number.parseInt(this.route.snapshot.paramMap.get('offerId') ?? '', 10))\n                .pipe(filter(scOrderIsLoaded), takeUntilDestroyed(this.destroyRef))\n                .subscribe((draft) => {\n                    this.draft.set(draft);\n                    this.seoService.setByDefaultAsNoIndexed('Черновики заказов');\n                });\n        }\n    }\n\n    /**\n     * Переход к оформлению заказа по выбранному направлению продаж.\n     *\n     * @param direction Группа позиций по направлению продаж.\n     */\n    protected goToNewOrder(direction: ScOrderItemsByDirection): void {\n        const ids = this.selectedItemsIds().filter((id) => direction.items.some((item) => item.id === id));\n\n        this.router.navigate(['orders', 'new', this.draft()?.id, direction.direction.id, 'draft'], { queryParams: { itemsIds: ids } }).catch(noop);\n    }\n\n    /**\n     * Удаляет черновик и перенаправляет к списку черновиков.\n     */\n    protected clearDraft(): void {\n        const { id } = this.draft() ?? {};\n\n        if (!id) {\n            return;\n        }\n\n        this.dialogService\n            .open<boolean>(TUI_CONFIRM, {\n                label: `Удалить черновик №${id}?`,\n                data: {\n                    content: `Это действие удалит черновик №${id}.`,\n                    yes: 'Да, удалить',\n                    no: 'Нет, оставить',\n                },\n            })\n            .pipe(\n                filter(Boolean),\n                switchMap(() => this.draftsService.delete$(id)),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe(() => {\n                this.deleteEvent.emit();\n                this.router.navigate(['/drafts']).catch(noop);\n            });\n    }\n}\n","@let items = draft()?.products;\n\n@if (items) {\n    <div class=\"flex flex-col gap-12\">\n        <div class=\"flex flex-col gap-5\">\n            @if (items.length) {\n                <sc-order-items-list-by-directions\n                    [items]=\"items\"\n                    [editable]=\"isDraftPage()\"\n                    [selectable]=\"true\"\n                    [continueButtonText]=\"'Оформить заказ'\"\n                    (selectedItemsIdsChange)=\"selectedItemsIds.set($event)\"\n                    (continueClick)=\"goToNewOrder($event)\"\n                >\n                    <div\n                        footerActions\n                        class=\"flex gap-2\"\n                    >\n                        <a\n                            tuiButton\n                            appearance=\"secondary\"\n                            iconStart=\"@tui.shopping-cart\"\n                            routerLink=\"/catalog\"\n                            class=\"self-center\"\n                        >\n                            Перейти в каталог\n                        </a>\n                        <button\n                            tuiButton\n                            appearance=\"secondary\"\n                            (click)=\"clearDraft()\"\n                            iconStart=\"@tui.trash-2\"\n                        >\n                            Удалить черновик\n                        </button>\n                    </div>\n                </sc-order-items-list-by-directions>\n            } @else {\n                <p class=\"text-base font-bold\">Черновик пуст.</p>\n                <div\n                    footerActions\n                    class=\"flex gap-2\"\n                >\n                    <a\n                        tuiButton\n                        appearance=\"secondary\"\n                        iconStart=\"@tui.shopping-cart\"\n                        routerLink=\"/catalog\"\n                        class=\"self-center\"\n                    >\n                        Перейти в каталог\n                    </a>\n                    <button\n                        tuiButton\n                        appearance=\"secondary\"\n                        (click)=\"clearDraft()\"\n                        iconStart=\"@tui.trash-2\"\n                    >\n                        Удалить черновик\n                    </button>\n                </div>\n            }\n        </div>\n    </div>\n} @else {\n    <div class=\"flex flex-col gap-12\">\n        <div class=\"flex flex-col gap-5\">\n            @for (item of [1, 2, 3, 4]; track item) {\n                <div class=\"tui-skeleton flex h-[8.25rem] w-full overflow-hidden rounded-xl\"></div>\n            }\n            <div class=\"flex gap-2\">\n                <div class=\"tui-skeleton h-10 w-40 rounded-md\"></div>\n                <div class=\"tui-skeleton h-10 w-36 rounded-md\"></div>\n            </div>\n        </div>\n    </div>\n}\n"]}
@@ -1,4 +1,9 @@
1
- export * from './order-item-mobile/sc-order-item-mobile.component';
1
+ export * from './models/sc-i-cart-items-by-direction';
2
+ export * from './order-item/sc-order-item.component';
3
+ export * from './order-items-list/sc-order-items-list.component';
4
+ export * from './order-items-list-by-directions/sc-order-items-list-by-directions.component';
5
+ export * from './order-items-list-by-stock/sc-order-items-list-by-stock.component';
6
+ export * from './draft/sc-draft.component';
2
7
  export * from './sc-payment-status/sc-payment-status.component';
3
8
  export * from './sc-order.module';
4
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvb3JkZXIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxvREFBb0QsQ0FBQztBQUNuRSxjQUFjLGlEQUFpRCxDQUFDO0FBQ2hFLGNBQWMsbUJBQW1CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL29yZGVyLWl0ZW0tbW9iaWxlL3NjLW9yZGVyLWl0ZW0tbW9iaWxlLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLXBheW1lbnQtc3RhdHVzL3NjLXBheW1lbnQtc3RhdHVzLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLW9yZGVyLm1vZHVsZSc7XG4iXX0=
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvb3JkZXIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyx1Q0FBdUMsQ0FBQztBQUN0RCxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsa0RBQWtELENBQUM7QUFDakUsY0FBYyw4RUFBOEUsQ0FBQztBQUM3RixjQUFjLG9FQUFvRSxDQUFDO0FBQ25GLGNBQWMsNEJBQTRCLENBQUM7QUFDM0MsY0FBYyxpREFBaUQsQ0FBQztBQUNoRSxjQUFjLG1CQUFtQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9tb2RlbHMvc2MtaS1jYXJ0LWl0ZW1zLWJ5LWRpcmVjdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL29yZGVyLWl0ZW0vc2Mtb3JkZXItaXRlbS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9vcmRlci1pdGVtcy1saXN0L3NjLW9yZGVyLWl0ZW1zLWxpc3QuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vb3JkZXItaXRlbXMtbGlzdC1ieS1kaXJlY3Rpb25zL3NjLW9yZGVyLWl0ZW1zLWxpc3QtYnktZGlyZWN0aW9ucy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9vcmRlci1pdGVtcy1saXN0LWJ5LXN0b2NrL3NjLW9yZGVyLWl0ZW1zLWxpc3QtYnktc3RvY2suY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vZHJhZnQvc2MtZHJhZnQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtcGF5bWVudC1zdGF0dXMvc2MtcGF5bWVudC1zdGF0dXMuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2Mtb3JkZXIubW9kdWxlJztcbiJdfQ==
@@ -0,0 +1,22 @@
1
+ /**
2
+ * Группа позиций заказа по направлению продаж.
3
+ */
4
+ export class ScOrderItemsByDirection {
5
+ /**
6
+ * @param direction Направление продаж.
7
+ * @param allItems Все элементы заказа.
8
+ */
9
+ constructor(direction, allItems) {
10
+ this.direction = direction;
11
+ this.items = allItems.filter((item) => item.salesDirection === direction.id);
12
+ }
13
+ /**
14
+ * Получает список позиций заказа по идентификаторам.
15
+ *
16
+ * @param itemsIds Идентификаторы позиций.
17
+ */
18
+ getItemsByIds(itemsIds) {
19
+ return this.items.filter((item) => itemsIds.includes(item.id));
20
+ }
21
+ }
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtaS1jYXJ0LWl0ZW1zLWJ5LWRpcmVjdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS9vcmRlci9tb2RlbHMvc2MtaS1jYXJ0LWl0ZW1zLWJ5LWRpcmVjdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQTs7R0FFRztBQUNILE1BQU0sT0FBTyx1QkFBdUI7SUFXaEM7OztPQUdHO0lBQ0gsWUFBWSxTQUE0QixFQUFFLFFBQWdDO1FBQ3RFLElBQUksQ0FBQyxTQUFTLEdBQUcsU0FBUyxDQUFDO1FBQzNCLElBQUksQ0FBQyxLQUFLLEdBQUcsUUFBUSxDQUFDLE1BQU0sQ0FBQyxDQUFDLElBQUksRUFBRSxFQUFFLENBQUMsSUFBSSxDQUFDLGNBQWMsS0FBSyxTQUFTLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDakYsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxhQUFhLENBQUMsUUFBa0I7UUFDbkMsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDLElBQUksRUFBRSxFQUFFLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztJQUNuRSxDQUFDO0NBQ0oiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTY0lTYWxlc0RpcmVjdGlvbiwgU2NPcmRlckl0ZW0gfSBmcm9tICdAc25hYmNlbnRyL2NsaWVudC1jb3JlJztcblxuLyoqXG4gKiDQk9GA0YPQv9C/0LAg0L/QvtC30LjRhtC40Lkg0LfQsNC60LDQt9CwINC/0L4g0L3QsNC/0YDQsNCy0LvQtdC90LjRjiDQv9GA0L7QtNCw0LYuXG4gKi9cbmV4cG9ydCBjbGFzcyBTY09yZGVySXRlbXNCeURpcmVjdGlvbiB7XG4gICAgLyoqXG4gICAgICog0J3QsNC/0YDQsNCy0LvQtdC90LjQtSDQv9GA0L7QtNCw0LYg0LjQu9C4IHVuZGVmaW5lZCDQtNC70Y8g0L/QvtC30LjRhtC40Lkg0LHQtdC3INC90LDQv9GA0LDQstC70LXQvdC40Y8uXG4gICAgICovXG4gICAgcHVibGljIHJlYWRvbmx5IGRpcmVjdGlvbjogU2NJU2FsZXNEaXJlY3Rpb247XG5cbiAgICAvKipcbiAgICAgKiDQodC/0LjRgdC+0Log0L/QvtC30LjRhtC40Lkg0LfQsNC60LDQt9CwINCyINC00LDQvdC90L7QuSDQs9GA0YPQv9C/0LUuXG4gICAgICovXG4gICAgcHVibGljIHJlYWRvbmx5IGl0ZW1zOiBTY09yZGVySXRlbVtdO1xuXG4gICAgLyoqXG4gICAgICogQHBhcmFtIGRpcmVjdGlvbiDQndCw0L/RgNCw0LLQu9C10L3QuNC1INC/0YDQvtC00LDQti5cbiAgICAgKiBAcGFyYW0gYWxsSXRlbXMg0JLRgdC1INGN0LvQtdC80LXQvdGC0Ysg0LfQsNC60LDQt9CwLlxuICAgICAqL1xuICAgIGNvbnN0cnVjdG9yKGRpcmVjdGlvbjogU2NJU2FsZXNEaXJlY3Rpb24sIGFsbEl0ZW1zOiByZWFkb25seSBTY09yZGVySXRlbVtdKSB7XG4gICAgICAgIHRoaXMuZGlyZWN0aW9uID0gZGlyZWN0aW9uO1xuICAgICAgICB0aGlzLml0ZW1zID0gYWxsSXRlbXMuZmlsdGVyKChpdGVtKSA9PiBpdGVtLnNhbGVzRGlyZWN0aW9uID09PSBkaXJlY3Rpb24uaWQpO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqINCf0L7Qu9GD0YfQsNC10YIg0YHQv9C40YHQvtC6INC/0L7Qt9C40YbQuNC5INC30LDQutCw0LfQsCDQv9C+INC40LTQtdC90YLQuNGE0LjQutCw0YLQvtGA0LDQvC5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBpdGVtc0lkcyDQmNC00LXQvdGC0LjRhNC40LrQsNGC0L7RgNGLINC/0L7Qt9C40YbQuNC5LlxuICAgICAqL1xuICAgIHB1YmxpYyBnZXRJdGVtc0J5SWRzKGl0ZW1zSWRzOiBzdHJpbmdbXSk6IFNjT3JkZXJJdGVtW10ge1xuICAgICAgICByZXR1cm4gdGhpcy5pdGVtcy5maWx0ZXIoKGl0ZW0pID0+IGl0ZW1zSWRzLmluY2x1ZGVzKGl0ZW0uaWQpKTtcbiAgICB9XG59XG4iXX0=