@snabcentr/client-ui 3.26.0 → 3.27.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/catalog/index.d.ts +1 -1
- package/catalog/notify-when-in-stock-dialog/index.d.ts +2 -0
- package/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.d.ts +12 -2
- package/catalog/notify-when-in-stock-dialog/sc-product-in-all-warehouses.pipe.d.ts +22 -0
- package/esm2022/auth/sc-sign-in-form/sc-sign-in-form-by-phone/sc-sign-in-form-by-phone.component.mjs +4 -3
- package/esm2022/auth/sc-simple-sign-up-form/sc-simple-sign-up-form.component.mjs +4 -3
- package/esm2022/auth/sign-up-form/sc-sign-up-form.component.mjs +5 -4
- package/esm2022/catalog/index.mjs +2 -2
- package/esm2022/catalog/notify-when-in-stock-dialog/index.mjs +3 -0
- package/esm2022/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.mjs +41 -16
- package/esm2022/catalog/notify-when-in-stock-dialog/sc-product-in-all-warehouses.pipe.mjs +35 -0
- package/esm2022/contacts/add-contact-dialog/sc-add-contact-dialog.component.mjs +3 -2
- package/esm2022/contragents/add-contragent-dialog/sc-add-contragent-dialog.component.mjs +3 -2
- package/esm2022/contragents/new-contragent-form/sc-new-contragent-form.component.mjs +3 -2
- package/esm2022/delivery-address/add-delivery-address-dialog/sc-add-delivery-address-dialog.component.mjs +3 -2
- package/esm2022/feedback/feedback-form/sc-feedback-form.component.mjs +3 -2
- package/esm2022/user/update-user-info-dialog/sc-update-user-info-dialog.component.mjs +4 -3
- package/esm2022/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.mjs +6 -5
- package/esm2022/validators/index.mjs +3 -1
- package/esm2022/validators/sc-at-least-one-required-validator.mjs +25 -0
- package/esm2022/validators/sc-phone-validator.mjs +14 -0
- package/esm2022/verification/verification-phone-check-form/sc-verification-phone-check-form.component.mjs +16 -3
- package/fesm2022/snabcentr-client-ui.mjs +157 -49
- package/fesm2022/snabcentr-client-ui.mjs.map +1 -1
- package/package.json +2 -2
- package/release_notes.tmp +6 -2
- package/styles/tailwind/tailwind.scss +4 -0
- package/validators/index.d.ts +2 -0
- package/validators/sc-at-least-one-required-validator.d.ts +10 -0
- package/validators/sc-phone-validator.d.ts +5 -0
- package/verification/verification-phone-check-form/sc-verification-phone-check-form.component.d.ts +13 -1
@@ -16,6 +16,7 @@ import { RecaptchaComponent, RecaptchaFormsModule, RecaptchaModule } from 'ng-re
|
|
16
16
|
import { catchError, filter, map, merge, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
|
17
17
|
import { ScFormFieldsModule } from '../../form-fields';
|
18
18
|
import { MAX_FILES_IN_FORM_INPUT } from '../../providers/sc-max-files-in-form-input';
|
19
|
+
import { phoneValidator } from '../../validators/sc-phone-validator';
|
19
20
|
import * as i0 from "@angular/core";
|
20
21
|
import * as i1 from "@angular/forms";
|
21
22
|
import * as i2 from "../../form-fields/suggestion-field/sc-suggestion-field.component";
|
@@ -220,7 +221,7 @@ export class ScFeedbackFormComponent {
|
|
220
221
|
this.form.get('name')?.setValidators([Validators.required, Validators.minLength(3)]);
|
221
222
|
break;
|
222
223
|
case 'phone':
|
223
|
-
this.form.get('phone')?.setValidators([Validators.required,
|
224
|
+
this.form.get('phone')?.setValidators([Validators.required, phoneValidator()]);
|
224
225
|
break;
|
225
226
|
case 'email':
|
226
227
|
this.form.get('email')?.setValidators([Validators.required, Validators.email]);
|
@@ -332,4 +333,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
332
333
|
TuiSortCountriesPipe,
|
333
334
|
], 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" }]
|
334
335
|
}], ctorParameters: () => [], propDecorators: { isValidPhoneNumberForMessenger: [] } });
|
335
|
-
//# 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 { 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.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, Validators.minLength(12)]);\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     * @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,sonarjs/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"]}
|
336
|
+
//# 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.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     * @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,sonarjs/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"]}
|
@@ -6,6 +6,7 @@ import { tuiIsFalsy } from '@taiga-ui/cdk';
|
|
6
6
|
import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
|
7
7
|
import { catchError, filter, map, of, skip, startWith, Subject, switchMap, tap } from 'rxjs';
|
8
8
|
import { SC_USER_INFO } from '../../providers';
|
9
|
+
import { phoneValidator } from '../../validators/sc-phone-validator';
|
9
10
|
import * as i0 from "@angular/core";
|
10
11
|
import * as i1 from "@snabcentr/client-core";
|
11
12
|
import * as i2 from "@angular/common";
|
@@ -46,7 +47,7 @@ export class ScUpdateUserInfoDialogComponent {
|
|
46
47
|
this.form = new FormGroup({
|
47
48
|
name: new FormControl(null, [Validators.required, Validators.minLength(3)]),
|
48
49
|
email: new FormControl(null, [Validators.required, Validators.email]),
|
49
|
-
phone: new FormControl(null, [Validators.required,
|
50
|
+
phone: new FormControl(null, [Validators.required, phoneValidator()]),
|
50
51
|
verificationCode: new FormControl(null, [Validators.required, Validators.minLength(6)]),
|
51
52
|
cityId: new FormControl(null, Validators.required),
|
52
53
|
getNews: new FormControl(false),
|
@@ -100,7 +101,7 @@ export class ScUpdateUserInfoDialogComponent {
|
|
100
101
|
});
|
101
102
|
}
|
102
103
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScUpdateUserInfoDialogComponent, deps: [{ token: SC_USER_INFO }, { token: i1.ScUserService }, { token: i1.ScUserMetrikaService }, { token: POLYMORPHEUS_CONTEXT, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
|
103
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScUpdateUserInfoDialogComponent, selector: "sc-update-user-info-dialog", outputs: { cancelButtonClick: "cancelButtonClick", successUpdateDate: "successUpdateDate" }, ngImport: i0, template: "<form\n [formGroup]=\"form\"\n *ngIf=\"user$ | async as user\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col gap-4\"\n>\n <label tuiLabel\n >{{ user.isOrg ? '\u041D\u0430\u0437\u0432\u0430\u043D\u0438\u0435 \u043E\u0440\u0433\u0430\u043D\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043B\u0438 \u0418\u041F' : '\u0424.\u0418.\u041E.' }}\n <tui-input formControlName=\"name\">\n {{ user.isOrg ? '\u041D\u0430\u0437\u0432\u0430\u043D\u0438\u0435 \u043E\u0440\u0433\u0430\u043D\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043B\u0438 \u0418\u041F' : '\u0424.\u0418.\u041E.' }}\n <ng-container *ngIf=\"user.isOrg\">\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.organization\"\n ></sc-suggestion-field>\n </ng-container>\n </tui-input>\n <tui-error\n formControlName=\"name\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <sc-verification-phone-check-form\n *tuiLet=\"user.contacts.phone.value === form.value.phone as phoneIsChange\"\n [showCodeFields]=\"!phoneIsChange\"\n [shouldBeBusy]=\"phoneIsChange\"\n class=\"w-full\"\n ></sc-verification-phone-check-form>\n <label\n tuiLabel\n class=\"w-full\"\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 ></sc-suggestion-field>\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <sc-addresses-selection-field></sc-addresses-selection-field>\n <label tuiLabel>\n <input\n tuiCheckbox\n type=\"checkbox\"\n formControlName=\"getNews\"\n class=\"w-full\"\n />\n \u042F \u0445\u043E\u0447\u0443 \u043F\u043E\u043B\u0443\u0447\u0430\u0442\u044C \u0440\u0430\u0441\u0441\u044B\u043B\u043A\u0443 \u0421\u043D\u0430\u0431\u0446\u0435\u043D\u0442\u0440 \u043E \u0441\u043A\u0438\u0434\u043A\u0430\u0445 \u0438 \u043D\u043E\u0432\u043E\u0441\u0442\u044F\u0445 \u043F\u043E \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u0435.\n </label>\n <div class=\"flex justify-center gap-4\">\n <button\n tuiButton\n (click)=\"context ? context.$implicit.complete() : cancelButtonClick.emit()\"\n type=\"button\"\n iconStart=\"@tui.x\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n <button\n tuiButton\n [disabled]=\"form.invalid\"\n [loading]=\"!!(loading$ | async)\"\n type=\"submit\"\n iconStart=\"@tui.save\"\n >\n \u0421\u043E\u0445\u0440\u0430\u043D\u0438\u0442\u044C\n </button>\n </div>\n</form>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i3.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i4.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: i4.TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "directive", type: i4.TuiLabel, selector: "label[tuiLabel]" }, { kind: "directive", type: i5.TuiLet, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "component", type: i6.ScVerificationPhoneCheckFormComponent, selector: "sc-verification-phone-check-form", inputs: ["showCodeFields", "readOnly", "shouldBeBusy", "shouldBeConfirmed", "haveCode"], outputs: ["haveCodeChange", "isBusyChange", "isConfirmedChange"] }, { kind: "component", type: i7.ScAddressesSelectionFieldComponent, selector: "sc-addresses-selection-field" }, { kind: "component", type: i8.ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: ["type"], outputs: ["selectedClick"] }, { kind: "component", type: i9.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i9.TuiInputDirective, selector: "tui-input" }, { kind: "component", type: i10.TuiCheckbox, selector: "input[type=\"checkbox\"][tuiCheckbox]", inputs: ["size"] }, { kind: "directive", type: i11.TuiDataListDirective, selector: "ng-template[tuiDataList]" }, { kind: "component", type: i10.TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "pipe", type: i10.TuiFieldErrorPipe, name: "tuiFieldError" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
104
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScUpdateUserInfoDialogComponent, selector: "sc-update-user-info-dialog", outputs: { cancelButtonClick: "cancelButtonClick", successUpdateDate: "successUpdateDate" }, ngImport: i0, template: "<form\n [formGroup]=\"form\"\n *ngIf=\"user$ | async as user\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col gap-4\"\n>\n <label tuiLabel\n >{{ user.isOrg ? '\u041D\u0430\u0437\u0432\u0430\u043D\u0438\u0435 \u043E\u0440\u0433\u0430\u043D\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043B\u0438 \u0418\u041F' : '\u0424.\u0418.\u041E.' }}\n <tui-input formControlName=\"name\">\n {{ user.isOrg ? '\u041D\u0430\u0437\u0432\u0430\u043D\u0438\u0435 \u043E\u0440\u0433\u0430\u043D\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043B\u0438 \u0418\u041F' : '\u0424.\u0418.\u041E.' }}\n <ng-container *ngIf=\"user.isOrg\">\n <sc-suggestion-field\n *tuiDataList\n [type]=\"suggestionType.organization\"\n ></sc-suggestion-field>\n </ng-container>\n </tui-input>\n <tui-error\n formControlName=\"name\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <sc-verification-phone-check-form\n *tuiLet=\"user.contacts.phone.value === form.value.phone as phoneIsChange\"\n [showCodeFields]=\"!phoneIsChange\"\n [shouldBeBusy]=\"phoneIsChange\"\n class=\"w-full\"\n ></sc-verification-phone-check-form>\n <label\n tuiLabel\n class=\"w-full\"\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 ></sc-suggestion-field>\n </tui-input>\n <tui-error\n formControlName=\"email\"\n [error]=\"[] | tuiFieldError | async\"\n ></tui-error>\n </label>\n <sc-addresses-selection-field></sc-addresses-selection-field>\n <label tuiLabel>\n <input\n tuiCheckbox\n type=\"checkbox\"\n formControlName=\"getNews\"\n class=\"w-full\"\n />\n \u042F \u0445\u043E\u0447\u0443 \u043F\u043E\u043B\u0443\u0447\u0430\u0442\u044C \u0440\u0430\u0441\u0441\u044B\u043B\u043A\u0443 \u0421\u043D\u0430\u0431\u0446\u0435\u043D\u0442\u0440 \u043E \u0441\u043A\u0438\u0434\u043A\u0430\u0445 \u0438 \u043D\u043E\u0432\u043E\u0441\u0442\u044F\u0445 \u043F\u043E \u044D\u043B\u0435\u043A\u0442\u0440\u043E\u043D\u043D\u043E\u0439 \u043F\u043E\u0447\u0442\u0435.\n </label>\n <div class=\"flex justify-center gap-4\">\n <button\n tuiButton\n (click)=\"context ? context.$implicit.complete() : cancelButtonClick.emit()\"\n type=\"button\"\n iconStart=\"@tui.x\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n <button\n tuiButton\n [disabled]=\"form.invalid\"\n [loading]=\"!!(loading$ | async)\"\n type=\"submit\"\n iconStart=\"@tui.save\"\n >\n \u0421\u043E\u0445\u0440\u0430\u043D\u0438\u0442\u044C\n </button>\n </div>\n</form>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i3.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i4.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: i4.TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "directive", type: i4.TuiLabel, selector: "label[tuiLabel]" }, { kind: "directive", type: i5.TuiLet, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "component", type: i6.ScVerificationPhoneCheckFormComponent, selector: "sc-verification-phone-check-form", inputs: ["showCodeFields", "readOnly", "pseudoInvalid", "required", "subtitle", "shouldBeBusy", "shouldBeConfirmed", "haveCode"], outputs: ["haveCodeChange", "isBusyChange", "isConfirmedChange"] }, { kind: "component", type: i7.ScAddressesSelectionFieldComponent, selector: "sc-addresses-selection-field" }, { kind: "component", type: i8.ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: ["type"], outputs: ["selectedClick"] }, { kind: "component", type: i9.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i9.TuiInputDirective, selector: "tui-input" }, { kind: "component", type: i10.TuiCheckbox, selector: "input[type=\"checkbox\"][tuiCheckbox]", inputs: ["size"] }, { kind: "directive", type: i11.TuiDataListDirective, selector: "ng-template[tuiDataList]" }, { kind: "component", type: i10.TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }, { kind: "pipe", type: i10.TuiFieldErrorPipe, name: "tuiFieldError" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
104
105
|
}
|
105
106
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScUpdateUserInfoDialogComponent, decorators: [{
|
106
107
|
type: Component,
|
@@ -118,4 +119,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
118
119
|
}], successUpdateDate: [{
|
119
120
|
type: Output
|
120
121
|
}] } });
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-update-user-info-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/user/update-user-info-dialog/sc-update-user-info-dialog.component.ts","../../../../../projects/client-ui/user/update-user-info-dialog/sc-update-user-info-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable guard-for-in,@typescript-eslint/unbound-method,rxjs/no-implicit-any-catch,no-restricted-syntax */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { AfterViewInit, ChangeDetectionStrategy, Component, EventEmitter, Inject, Optional, Output } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { ScIPhoneOnDataApprove, ScISuggestionType, ScIUserOnDataUpdate, ScUser, ScUserMetrikaGoalsEnum, ScUserMetrikaService, ScUserService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, Observable, of, skip, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { SC_USER_INFO } from '../../providers';\n\n/**\n * Диалоговое окно изменения данных пользователя.\n */\n@Component({\n    selector: 'sc-update-user-info-dialog',\n    templateUrl: './sc-update-user-info-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScUpdateUserInfoDialogComponent implements AfterViewInit {\n    /**\n     * Перечисление типов подсказок.\n     */\n    public readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Группа полей ввода формы общих данных пользователя для их редактирования.\n     */\n    public readonly form = new FormGroup({\n        name: new FormControl<string | null>(null, [Validators.required, Validators.minLength(3)]),\n        email: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        phone: new FormControl<string | null>(null, [Validators.required, Validators.minLength(12)]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n        cityId: new FormControl<number | null>(null, Validators.required),\n        getNews: new FormControl<boolean>(false),\n    });\n\n    /**\n     * Сигнал нажатия на кнопку \"Отмена\".\n     */\n    @Output()\n    public readonly cancelButtonClick: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Сигнал события успешного обновления данных пользователя.\n     */\n    @Output()\n    public readonly successUpdateDate: Observable<ScUser> = this.user$.pipe(skip(1));\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса обновления данных пользователя.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIUserOnDataUpdate | ScIPhoneOnDataApprove),\n        switchMap((value) =>\n            this.userService.updateUserInfo$(value).pipe(\n                switchMap(() => this.userService.updateUserChange$()),\n                tap((user) => this.context?.completeWith(user)),\n                catchError((error: HttpErrorResponse) => {\n                    const errorResponse = error.error as ApiErrorResponse;\n\n                    for (const key in errorResponse.errors) {\n                        // eslint-disable-next-line security/detect-object-injection\n                        this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[key] });\n                    }\n\n                    if (!errorResponse.errors && errorResponse.message) {\n                        this.form.setErrors({ serverResponse: [errorResponse.message] });\n                    }\n                    return of({} as ScUser);\n                }),\n                startWith(null)\n            )\n        )\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по email.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScUpdateUserInfoDialogComponent}.\n     *\n     * @param user$ Поток данных о текущем пользователе.\n     * @param userService Сервис данных о пользователе.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        @Inject(SC_USER_INFO) public readonly user$: Observable<ScUser>,\n        private readonly userService: ScUserService,\n        private readonly userMetrikaService: ScUserMetrikaService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context?: TuiDialogContext<ScUser, ScUpdateUserInfoDialogComponent>\n    ) {\n        this.user$\n            .pipe(\n                filter((user) => !user.isGuest),\n                takeUntilDestroyed()\n            )\n            .subscribe((user) => {\n                this.form.patchValue(\n                    {\n                        name: user.name,\n                        email: user.contacts.email.value,\n                        phone: user.contacts.phone.value,\n                        cityId: user.city,\n                        getNews: user.getNews,\n                    },\n                    { emitEvent: false }\n                );\n            });\n    }\n\n    /** @inheritdoc */\n    public ngAfterViewInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.userProfileEditShow,\n        });\n    }\n}\n","<form\n    [formGroup]=\"form\"\n    *ngIf=\"user$ | async as user\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"flex flex-col gap-4\"\n>\n    <label tuiLabel\n        >{{ user.isOrg ? 'Название организации или ИП' : 'Ф.И.О.' }}\n        <tui-input formControlName=\"name\">\n            {{ user.isOrg ? 'Название организации или ИП' : 'Ф.И.О.' }}\n            <ng-container *ngIf=\"user.isOrg\">\n                <sc-suggestion-field\n                    *tuiDataList\n                    [type]=\"suggestionType.organization\"\n                ></sc-suggestion-field>\n            </ng-container>\n        </tui-input>\n        <tui-error\n            formControlName=\"name\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n    <sc-verification-phone-check-form\n        *tuiLet=\"user.contacts.phone.value === form.value.phone as phoneIsChange\"\n        [showCodeFields]=\"!phoneIsChange\"\n        [shouldBeBusy]=\"phoneIsChange\"\n        class=\"w-full\"\n    ></sc-verification-phone-check-form>\n    <label\n        tuiLabel\n        class=\"w-full\"\n        >Адрес электронной почты\n        <tui-input formControlName=\"email\">\n            Адрес электронной почты\n            <sc-suggestion-field\n                *tuiDataList\n                [type]=\"suggestionType.email\"\n            ></sc-suggestion-field>\n        </tui-input>\n        <tui-error\n            formControlName=\"email\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n    <sc-addresses-selection-field></sc-addresses-selection-field>\n    <label tuiLabel>\n        <input\n            tuiCheckbox\n            type=\"checkbox\"\n            formControlName=\"getNews\"\n            class=\"w-full\"\n        />\n        Я хочу получать рассылку Снабцентр о скидках и новостях по электронной почте.\n    </label>\n    <div class=\"flex justify-center gap-4\">\n        <button\n            tuiButton\n            (click)=\"context ? context.$implicit.complete() : cancelButtonClick.emit()\"\n            type=\"button\"\n            iconStart=\"@tui.x\"\n            appearance=\"secondary\"\n        >\n            Отмена\n        </button>\n        <button\n            tuiButton\n            [disabled]=\"form.invalid\"\n            [loading]=\"!!(loading$ | async)\"\n            type=\"submit\"\n            iconStart=\"@tui.save\"\n        >\n            Сохранить\n        </button>\n    </div>\n</form>\n"]}
|
122
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-update-user-info-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/user/update-user-info-dialog/sc-update-user-info-dialog.component.ts","../../../../../projects/client-ui/user/update-user-info-dialog/sc-update-user-info-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable guard-for-in,@typescript-eslint/unbound-method,rxjs/no-implicit-any-catch,no-restricted-syntax */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { AfterViewInit, ChangeDetectionStrategy, Component, EventEmitter, Inject, Optional, Output } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { ScIPhoneOnDataApprove, ScISuggestionType, ScIUserOnDataUpdate, ScUser, ScUserMetrikaGoalsEnum, ScUserMetrikaService, ScUserService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, Observable, of, skip, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { SC_USER_INFO } from '../../providers';\nimport { phoneValidator } from '../../validators/sc-phone-validator';\n\n/**\n * Диалоговое окно изменения данных пользователя.\n */\n@Component({\n    selector: 'sc-update-user-info-dialog',\n    templateUrl: './sc-update-user-info-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScUpdateUserInfoDialogComponent implements AfterViewInit {\n    /**\n     * Перечисление типов подсказок.\n     */\n    public readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Группа полей ввода формы общих данных пользователя для их редактирования.\n     */\n    public readonly form = new FormGroup({\n        name: new FormControl<string | null>(null, [Validators.required, Validators.minLength(3)]),\n        email: new FormControl<string | null>(null, [Validators.required, Validators.email]),\n        phone: new FormControl<string | null>(null, [Validators.required, phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n        cityId: new FormControl<number | null>(null, Validators.required),\n        getNews: new FormControl<boolean>(false),\n    });\n\n    /**\n     * Сигнал нажатия на кнопку \"Отмена\".\n     */\n    @Output()\n    public readonly cancelButtonClick: EventEmitter<void> = new EventEmitter<void>();\n\n    /**\n     * Сигнал события успешного обновления данных пользователя.\n     */\n    @Output()\n    public readonly successUpdateDate: Observable<ScUser> = this.user$.pipe(skip(1));\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса обновления данных пользователя.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIUserOnDataUpdate | ScIPhoneOnDataApprove),\n        switchMap((value) =>\n            this.userService.updateUserInfo$(value).pipe(\n                switchMap(() => this.userService.updateUserChange$()),\n                tap((user) => this.context?.completeWith(user)),\n                catchError((error: HttpErrorResponse) => {\n                    const errorResponse = error.error as ApiErrorResponse;\n\n                    for (const key in errorResponse.errors) {\n                        // eslint-disable-next-line security/detect-object-injection\n                        this.form.get(key)?.setErrors({ serverResponse: errorResponse.errors[key] });\n                    }\n\n                    if (!errorResponse.errors && errorResponse.message) {\n                        this.form.setErrors({ serverResponse: [errorResponse.message] });\n                    }\n                    return of({} as ScUser);\n                }),\n                startWith(null)\n            )\n        )\n    );\n\n    /**\n     * {@link Observable} изменения состояния загрузки данных аутентификации по email.\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScUpdateUserInfoDialogComponent}.\n     *\n     * @param user$ Поток данных о текущем пользователе.\n     * @param userService Сервис данных о пользователе.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        @Inject(SC_USER_INFO) public readonly user$: Observable<ScUser>,\n        private readonly userService: ScUserService,\n        private readonly userMetrikaService: ScUserMetrikaService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context?: TuiDialogContext<ScUser, ScUpdateUserInfoDialogComponent>\n    ) {\n        this.user$\n            .pipe(\n                filter((user) => !user.isGuest),\n                takeUntilDestroyed()\n            )\n            .subscribe((user) => {\n                this.form.patchValue(\n                    {\n                        name: user.name,\n                        email: user.contacts.email.value,\n                        phone: user.contacts.phone.value,\n                        cityId: user.city,\n                        getNews: user.getNews,\n                    },\n                    { emitEvent: false }\n                );\n            });\n    }\n\n    /** @inheritdoc */\n    public ngAfterViewInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.userProfileEditShow,\n        });\n    }\n}\n","<form\n    [formGroup]=\"form\"\n    *ngIf=\"user$ | async as user\"\n    (ngSubmit)=\"onSubmit.next()\"\n    class=\"flex flex-col gap-4\"\n>\n    <label tuiLabel\n        >{{ user.isOrg ? 'Название организации или ИП' : 'Ф.И.О.' }}\n        <tui-input formControlName=\"name\">\n            {{ user.isOrg ? 'Название организации или ИП' : 'Ф.И.О.' }}\n            <ng-container *ngIf=\"user.isOrg\">\n                <sc-suggestion-field\n                    *tuiDataList\n                    [type]=\"suggestionType.organization\"\n                ></sc-suggestion-field>\n            </ng-container>\n        </tui-input>\n        <tui-error\n            formControlName=\"name\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n    <sc-verification-phone-check-form\n        *tuiLet=\"user.contacts.phone.value === form.value.phone as phoneIsChange\"\n        [showCodeFields]=\"!phoneIsChange\"\n        [shouldBeBusy]=\"phoneIsChange\"\n        class=\"w-full\"\n    ></sc-verification-phone-check-form>\n    <label\n        tuiLabel\n        class=\"w-full\"\n        >Адрес электронной почты\n        <tui-input formControlName=\"email\">\n            Адрес электронной почты\n            <sc-suggestion-field\n                *tuiDataList\n                [type]=\"suggestionType.email\"\n            ></sc-suggestion-field>\n        </tui-input>\n        <tui-error\n            formControlName=\"email\"\n            [error]=\"[] | tuiFieldError | async\"\n        ></tui-error>\n    </label>\n    <sc-addresses-selection-field></sc-addresses-selection-field>\n    <label tuiLabel>\n        <input\n            tuiCheckbox\n            type=\"checkbox\"\n            formControlName=\"getNews\"\n            class=\"w-full\"\n        />\n        Я хочу получать рассылку Снабцентр о скидках и новостях по электронной почте.\n    </label>\n    <div class=\"flex justify-center gap-4\">\n        <button\n            tuiButton\n            (click)=\"context ? context.$implicit.complete() : cancelButtonClick.emit()\"\n            type=\"button\"\n            iconStart=\"@tui.x\"\n            appearance=\"secondary\"\n        >\n            Отмена\n        </button>\n        <button\n            tuiButton\n            [disabled]=\"form.invalid\"\n            [loading]=\"!!(loading$ | async)\"\n            type=\"submit\"\n            iconStart=\"@tui.save\"\n        >\n            Сохранить\n        </button>\n    </div>\n</form>\n"]}
|
@@ -7,7 +7,8 @@ import { ScUserMetrikaGoalsEnum } from '@snabcentr/client-core';
|
|
7
7
|
import { tuiIsFalsy } from '@taiga-ui/cdk';
|
8
8
|
import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
|
9
9
|
import { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
|
10
|
-
import { SC_USER_INFO
|
10
|
+
import { SC_USER_INFO } from '../../providers/sc-user-providers';
|
11
|
+
import { phoneValidator } from '../../validators/sc-phone-validator';
|
11
12
|
import * as i0 from "@angular/core";
|
12
13
|
import * as i1 from "@snabcentr/client-core";
|
13
14
|
import * as i2 from "@angular/common";
|
@@ -44,7 +45,7 @@ export class ScUserPhoneApproveDialogComponent {
|
|
44
45
|
* Группа полей ввода для подтверждения номера телефона.
|
45
46
|
*/
|
46
47
|
this.form = new FormGroup({
|
47
|
-
phone: new FormControl(null, [Validators.required,
|
48
|
+
phone: new FormControl(null, [Validators.required, phoneValidator()]),
|
48
49
|
verificationCode: new FormControl(null, [Validators.required, Validators.minLength(6)]),
|
49
50
|
});
|
50
51
|
/**
|
@@ -88,11 +89,11 @@ export class ScUserPhoneApproveDialogComponent {
|
|
88
89
|
});
|
89
90
|
}
|
90
91
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScUserPhoneApproveDialogComponent, deps: [{ token: i1.ScVerificationService }, { token: i1.ScUserMetrikaService }, { token: SC_USER_INFO }, { token: i1.ScUserService }, { token: POLYMORPHEUS_CONTEXT, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
|
91
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScUserPhoneApproveDialogComponent, selector: "sc-user-phone-approve-dialog",
|
92
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScUserPhoneApproveDialogComponent, selector: "sc-user-phone-approve-dialog", ngImport: i0, template: "<ng-container *ngIf=\"user$ | async as user\">\n <form\n [formGroup]=\"form\"\n *tuiLet=\"!!(loading$ | async) as loadingApproveCode\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col gap-3\"\n >\n <sc-verification-phone-check-form\n [(haveCode)]=\"haveCode\"\n [shouldBeBusy]=\"true\"\n [shouldBeConfirmed]=\"false\"\n [showCodeFields]=\"!user.contacts.phone.isApproved\"\n [readOnly]=\"true\"\n class=\"w-full\"\n ></sc-verification-phone-check-form>\n <div\n *ngIf=\"user.contacts.phone.isApproved || haveCode\"\n class=\"flex justify-center gap-4\"\n >\n <button\n *ngIf=\"user.contacts.phone.isApproved && context\"\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u0417\u0430\u043A\u0440\u044B\u0442\u044C\n </button>\n <button\n *ngIf=\"haveCode\"\n tuiButton\n type=\"submit\"\n [loading]=\"loadingApproveCode\"\n [disabled]=\"form.invalid || loadingApproveCode\"\n iconStart=\"@tui.sc.send\"\n >\n \u041F\u043E\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u0442\u044C\n </button>\n </div>\n </form>\n</ng-container>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: i5.TuiLet, selector: "[tuiLet]", inputs: ["tuiLet"] }, { kind: "component", type: i6.ScVerificationPhoneCheckFormComponent, selector: "sc-verification-phone-check-form", inputs: ["showCodeFields", "readOnly", "pseudoInvalid", "required", "subtitle", "shouldBeBusy", "shouldBeConfirmed", "haveCode"], outputs: ["haveCodeChange", "isBusyChange", "isConfirmedChange"] }, { kind: "component", type: i7.TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
92
93
|
}
|
93
94
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScUserPhoneApproveDialogComponent, decorators: [{
|
94
95
|
type: Component,
|
95
|
-
args: [{ selector: 'sc-user-phone-approve-dialog',
|
96
|
+
args: [{ selector: 'sc-user-phone-approve-dialog', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"user$ | async as user\">\n <form\n [formGroup]=\"form\"\n *tuiLet=\"!!(loading$ | async) as loadingApproveCode\"\n (ngSubmit)=\"onSubmit.next()\"\n class=\"flex flex-col gap-3\"\n >\n <sc-verification-phone-check-form\n [(haveCode)]=\"haveCode\"\n [shouldBeBusy]=\"true\"\n [shouldBeConfirmed]=\"false\"\n [showCodeFields]=\"!user.contacts.phone.isApproved\"\n [readOnly]=\"true\"\n class=\"w-full\"\n ></sc-verification-phone-check-form>\n <div\n *ngIf=\"user.contacts.phone.isApproved || haveCode\"\n class=\"flex justify-center gap-4\"\n >\n <button\n *ngIf=\"user.contacts.phone.isApproved && context\"\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u0417\u0430\u043A\u0440\u044B\u0442\u044C\n </button>\n <button\n *ngIf=\"haveCode\"\n tuiButton\n type=\"submit\"\n [loading]=\"loadingApproveCode\"\n [disabled]=\"form.invalid || loadingApproveCode\"\n iconStart=\"@tui.sc.send\"\n >\n \u041F\u043E\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u0442\u044C\n </button>\n </div>\n </form>\n</ng-container>\n" }]
|
96
97
|
}], ctorParameters: () => [{ type: i1.ScVerificationService }, { type: i1.ScUserMetrikaService }, { type: i8.Observable, decorators: [{
|
97
98
|
type: Inject,
|
98
99
|
args: [SC_USER_INFO]
|
@@ -102,4 +103,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
102
103
|
type: Inject,
|
103
104
|
args: [POLYMORPHEUS_CONTEXT]
|
104
105
|
}] }] });
|
105
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-user-phone-approve-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.ts","../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { AfterViewInit, ChangeDetectionStrategy, Component, Inject, Optional } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { ScApprovedData, ScIPhoneOnDataApprove, ScUser, ScUserMetrikaGoalsEnum, ScUserMetrikaService, ScUserService, ScVerificationService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { SC_USER_INFO, SC_USER_PROVIDERS } from '../../providers/sc-user-providers';\n\n/**\n * Диалоговое окно подтверждения номера телефона.\n */\n@Component({\n    selector: 'sc-user-phone-approve-dialog',\n    templateUrl: './sc-user-phone-approve-dialog.component.html',\n    providers: [SC_USER_PROVIDERS],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScUserPhoneApproveDialogComponent implements AfterViewInit {\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public haveCode: boolean = false;\n\n    /**\n     * Группа полей ввода для подтверждения номера телефона.\n     */\n    public readonly form = new FormGroup({\n        phone: new FormControl<string | null>(null, [Validators.required, Validators.minLength(12)]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса подтверждения номера телефона.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIPhoneOnDataApprove),\n        switchMap((value) =>\n            this.verificationService.approvePhone(value).pipe(\n                switchMap(() => this.userService.updateUserChange$()),\n                tap((user) => this.context?.completeWith(user.contacts.phone)),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as ScUser);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния запроса подтверждения номера телефона..\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScUserPhoneApproveDialogComponent}.\n     *\n     * @param verificationService Сервис верификации.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     * @param user$ Поток данных о текущем пользователе.\n     * @param userService Сервис данных о пользователе.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly verificationService: ScVerificationService,\n        private readonly userMetrikaService: ScUserMetrikaService,\n        @Inject(SC_USER_INFO)\n        public readonly user$: Observable<ScUser>,\n        private readonly userService: ScUserService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context?: TuiDialogContext<ScApprovedData, ScUserPhoneApproveDialogComponent>\n    ) {\n        this.user$\n            .pipe(\n                filter((user) => !user.isGuest),\n                takeUntilDestroyed()\n            )\n            .subscribe((user) => {\n                this.form.patchValue({\n                    phone: user.contacts.phone.value,\n                });\n            });\n    }\n\n    /** @inheritdoc */\n    public ngAfterViewInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.userProfileApprovePhoneShow,\n        });\n    }\n}\n","<ng-container *ngIf=\"user$ | async as user\">\n    <form\n        [formGroup]=\"form\"\n        *tuiLet=\"!!(loading$ | async) as loadingApproveCode\"\n        (ngSubmit)=\"onSubmit.next()\"\n        class=\"flex flex-col gap-3\"\n    >\n        <sc-verification-phone-check-form\n            [(haveCode)]=\"haveCode\"\n            [shouldBeBusy]=\"true\"\n            [shouldBeConfirmed]=\"false\"\n            [showCodeFields]=\"!user.contacts.phone.isApproved\"\n            [readOnly]=\"true\"\n            class=\"w-full\"\n        ></sc-verification-phone-check-form>\n        <div\n            *ngIf=\"user.contacts.phone.isApproved || haveCode\"\n            class=\"flex justify-center gap-4\"\n        >\n            <button\n                *ngIf=\"user.contacts.phone.isApproved && context\"\n                tuiButton\n                (click)=\"context.$implicit.complete()\"\n                type=\"button\"\n                appearance=\"secondary\"\n            >\n                Закрыть\n            </button>\n            <button\n                *ngIf=\"haveCode\"\n                tuiButton\n                type=\"submit\"\n                [loading]=\"loadingApproveCode\"\n                [disabled]=\"form.invalid || loadingApproveCode\"\n                iconStart=\"@tui.sc.send\"\n            >\n                Подтвердить\n            </button>\n        </div>\n    </form>\n</ng-container>\n"]}
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-user-phone-approve-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.ts","../../../../../projects/client-ui/user/user-phone-approve-dialog/sc-user-phone-approve-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable no-restricted-syntax,@typescript-eslint/unbound-method */\n\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { AfterViewInit, ChangeDetectionStrategy, Component, Inject, Optional } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, Validators } from '@angular/forms';\nimport { ScApprovedData, ScIPhoneOnDataApprove, ScUser, ScUserMetrikaGoalsEnum, ScUserMetrikaService, ScUserService, ScVerificationService } from '@snabcentr/client-core';\nimport { tuiIsFalsy } from '@taiga-ui/cdk';\nimport { TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, Observable, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth/interfaces/api-error-response';\nimport { SC_USER_INFO } from '../../providers/sc-user-providers';\nimport { phoneValidator } from '../../validators/sc-phone-validator';\n\n/**\n * Диалоговое окно подтверждения номера телефона.\n */\n@Component({\n    selector: 'sc-user-phone-approve-dialog',\n    templateUrl: './sc-user-phone-approve-dialog.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScUserPhoneApproveDialogComponent implements AfterViewInit {\n    /**\n     * Наличие кода подтверждения у пользователя.\n     */\n    public haveCode: boolean = false;\n\n    /**\n     * Группа полей ввода для подтверждения номера телефона.\n     */\n    public readonly form = new FormGroup({\n        phone: new FormControl<string | null>(null, [Validators.required, phoneValidator()]),\n        verificationCode: new FormControl<string | null>(null, [Validators.required, Validators.minLength(6)]),\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    public readonly onSubmit: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса подтверждения номера телефона.\n     */\n    private readonly request$ = this.onSubmit.pipe(\n        filter(() => this.form.valid),\n        map(() => this.form.value as ScIPhoneOnDataApprove),\n        switchMap((value) =>\n            this.verificationService.approvePhone(value).pipe(\n                switchMap(() => this.userService.updateUserChange$()),\n                tap((user) => this.context?.completeWith(user.contacts.phone)),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        for (const key in errors) {\n                            if (Object.hasOwn(errors, key)) {\n                                // eslint-disable-next-line security/detect-object-injection\n                                this.form.get(key)?.setErrors({ serverResponse: errors[key] });\n                            }\n                        }\n\n                        if (!errors && message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of({} as ScUser);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * {@link Observable} изменения состояния запроса подтверждения номера телефона..\n     */\n    public readonly loading$: Observable<boolean> = this.request$.pipe(map(tuiIsFalsy));\n\n    /**\n     * Инициализирует экземпляр класса {@link ScUserPhoneApproveDialogComponent}.\n     *\n     * @param verificationService Сервис верификации.\n     * @param userMetrikaService Сервис для сбора метрик о действиях пользователей.\n     * @param user$ Поток данных о текущем пользователе.\n     * @param userService Сервис данных о пользователе.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly verificationService: ScVerificationService,\n        private readonly userMetrikaService: ScUserMetrikaService,\n        @Inject(SC_USER_INFO)\n        public readonly user$: Observable<ScUser>,\n        private readonly userService: ScUserService,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        public readonly context?: TuiDialogContext<ScApprovedData, ScUserPhoneApproveDialogComponent>\n    ) {\n        this.user$\n            .pipe(\n                filter((user) => !user.isGuest),\n                takeUntilDestroyed()\n            )\n            .subscribe((user) => {\n                this.form.patchValue({\n                    phone: user.contacts.phone.value,\n                });\n            });\n    }\n\n    /** @inheritdoc */\n    public ngAfterViewInit(): void {\n        this.userMetrikaService.emitUserMetrikaEvent({\n            target: ScUserMetrikaGoalsEnum.userProfileApprovePhoneShow,\n        });\n    }\n}\n","<ng-container *ngIf=\"user$ | async as user\">\n    <form\n        [formGroup]=\"form\"\n        *tuiLet=\"!!(loading$ | async) as loadingApproveCode\"\n        (ngSubmit)=\"onSubmit.next()\"\n        class=\"flex flex-col gap-3\"\n    >\n        <sc-verification-phone-check-form\n            [(haveCode)]=\"haveCode\"\n            [shouldBeBusy]=\"true\"\n            [shouldBeConfirmed]=\"false\"\n            [showCodeFields]=\"!user.contacts.phone.isApproved\"\n            [readOnly]=\"true\"\n            class=\"w-full\"\n        ></sc-verification-phone-check-form>\n        <div\n            *ngIf=\"user.contacts.phone.isApproved || haveCode\"\n            class=\"flex justify-center gap-4\"\n        >\n            <button\n                *ngIf=\"user.contacts.phone.isApproved && context\"\n                tuiButton\n                (click)=\"context.$implicit.complete()\"\n                type=\"button\"\n                appearance=\"secondary\"\n            >\n                Закрыть\n            </button>\n            <button\n                *ngIf=\"haveCode\"\n                tuiButton\n                type=\"submit\"\n                [loading]=\"loadingApproveCode\"\n                [disabled]=\"form.invalid || loadingApproveCode\"\n                iconStart=\"@tui.sc.send\"\n            >\n                Подтвердить\n            </button>\n        </div>\n    </form>\n</ng-container>\n"]}
|
@@ -1,5 +1,7 @@
|
|
1
1
|
export * from './sc-bic-validator';
|
2
2
|
export * from './sc-correspondent-account-validator';
|
3
3
|
export * from './sc-password-confirm-matching-validator';
|
4
|
+
export * from './sc-phone-validator';
|
4
5
|
export * from './step-validator';
|
5
|
-
|
6
|
+
export * from './sc-at-least-one-required-validator';
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvdmFsaWRhdG9ycy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9CQUFvQixDQUFDO0FBQ25DLGNBQWMsc0NBQXNDLENBQUM7QUFDckQsY0FBYywwQ0FBMEMsQ0FBQztBQUN6RCxjQUFjLHNCQUFzQixDQUFDO0FBQ3JDLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxzQ0FBc0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vc2MtYmljLXZhbGlkYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3NjLWNvcnJlc3BvbmRlbnQtYWNjb3VudC12YWxpZGF0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1wYXNzd29yZC1jb25maXJtLW1hdGNoaW5nLXZhbGlkYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3NjLXBob25lLXZhbGlkYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3N0ZXAtdmFsaWRhdG9yJztcbmV4cG9ydCAqIGZyb20gJy4vc2MtYXQtbGVhc3Qtb25lLXJlcXVpcmVkLXZhbGlkYXRvcic7XG4iXX0=
|
@@ -0,0 +1,25 @@
|
|
1
|
+
/**
|
2
|
+
* Проверяет, что хотя бы одно поле из списка заполнено.
|
3
|
+
* Поддерживает как отдельные поля, так и группы полей.
|
4
|
+
*
|
5
|
+
* @param fields Массив полей для проверки. Каждый элемент может быть строкой или массивом строк.
|
6
|
+
* @param blockName Имя блока для идентификации ошибки
|
7
|
+
* @returns ValidatorFn
|
8
|
+
*/
|
9
|
+
export function scAtLeastOneRequiredValidator(fields, blockName) {
|
10
|
+
return (control) => {
|
11
|
+
if (fields.length === 0) {
|
12
|
+
return null;
|
13
|
+
}
|
14
|
+
const hasValue = fields.some((fieldKey) => {
|
15
|
+
if (Array.isArray(fieldKey)) {
|
16
|
+
// eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
|
17
|
+
return fieldKey.every((key) => !!control.get(key)?.value?.toString().trim());
|
18
|
+
}
|
19
|
+
// eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
|
20
|
+
return !!control.get(fieldKey)?.value?.toString().trim();
|
21
|
+
});
|
22
|
+
return hasValue ? null : { [blockName ?? 'atLeastOneRequired']: true };
|
23
|
+
};
|
24
|
+
}
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtYXQtbGVhc3Qtb25lLXJlcXVpcmVkLXZhbGlkYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NsaWVudC11aS92YWxpZGF0b3JzL3NjLWF0LWxlYXN0LW9uZS1yZXF1aXJlZC12YWxpZGF0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUE7Ozs7Ozs7R0FPRztBQUNILE1BQU0sVUFBVSw2QkFBNkIsQ0FBQyxNQUFnQyxFQUFFLFNBQWtCO0lBQzlGLE9BQU8sQ0FBQyxPQUF3QixFQUEyQixFQUFFO1FBQ3pELElBQUksTUFBTSxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUUsQ0FBQztZQUN0QixPQUFPLElBQUksQ0FBQztRQUNoQixDQUFDO1FBRUQsTUFBTSxRQUFRLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDLFFBQVEsRUFBRSxFQUFFO1lBQ3RDLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDO2dCQUMxQix5R0FBeUc7Z0JBQ3pHLE9BQU8sUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDLEdBQUcsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7WUFDakYsQ0FBQztZQUVELHlHQUF5RztZQUN6RyxPQUFPLENBQUMsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUM3RCxDQUFDLENBQUMsQ0FBQztRQUVILE9BQU8sUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxTQUFTLElBQUksb0JBQW9CLENBQUMsRUFBRSxJQUFJLEVBQUUsQ0FBQztJQUMzRSxDQUFDLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQWJzdHJhY3RDb250cm9sLCBWYWxpZGF0aW9uRXJyb3JzLCBWYWxpZGF0b3JGbiB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcblxuLyoqXG4gKiDQn9GA0L7QstC10YDRj9C10YIsINGH0YLQviDRhdC+0YLRjyDQsdGLINC+0LTQvdC+INC/0L7Qu9C1INC40Lcg0YHQv9C40YHQutCwINC30LDQv9C+0LvQvdC10L3Qvi5cbiAqINCf0L7QtNC00LXRgNC20LjQstCw0LXRgiDQutCw0Log0L7RgtC00LXQu9GM0L3Ri9C1INC/0L7Qu9GPLCDRgtCw0Log0Lgg0LPRgNGD0L/Qv9GLINC/0L7Qu9C10LkuXG4gKlxuICogQHBhcmFtIGZpZWxkcyDQnNCw0YHRgdC40LIg0L/QvtC70LXQuSDQtNC70Y8g0L/RgNC+0LLQtdGA0LrQuC4g0JrQsNC20LTRi9C5INGN0LvQtdC80LXQvdGCINC80L7QttC10YIg0LHRi9GC0Ywg0YHRgtGA0L7QutC+0Lkg0LjQu9C4INC80LDRgdGB0LjQstC+0Lwg0YHRgtGA0L7Qui5cbiAqIEBwYXJhbSBibG9ja05hbWUg0JjQvNGPINCx0LvQvtC60LAg0LTQu9GPINC40LTQtdC90YLQuNGE0LjQutCw0YbQuNC4INC+0YjQuNCx0LrQuFxuICogQHJldHVybnMgVmFsaWRhdG9yRm5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHNjQXRMZWFzdE9uZVJlcXVpcmVkVmFsaWRhdG9yKGZpZWxkczogQXJyYXk8c3RyaW5nIHwgc3RyaW5nW10+LCBibG9ja05hbWU/OiBzdHJpbmcpOiBWYWxpZGF0b3JGbiB7XG4gICAgcmV0dXJuIChjb250cm9sOiBBYnN0cmFjdENvbnRyb2wpOiBWYWxpZGF0aW9uRXJyb3JzIHwgbnVsbCA9PiB7XG4gICAgICAgIGlmIChmaWVsZHMubGVuZ3RoID09PSAwKSB7XG4gICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGhhc1ZhbHVlID0gZmllbGRzLnNvbWUoKGZpZWxkS2V5KSA9PiB7XG4gICAgICAgICAgICBpZiAoQXJyYXkuaXNBcnJheShmaWVsZEtleSkpIHtcbiAgICAgICAgICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1jYWxsLCBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLW1lbWJlci1hY2Nlc3NcbiAgICAgICAgICAgICAgICByZXR1cm4gZmllbGRLZXkuZXZlcnkoKGtleSkgPT4gISFjb250cm9sLmdldChrZXkpPy52YWx1ZT8udG9TdHJpbmcoKS50cmltKCkpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1jYWxsLCBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLW1lbWJlci1hY2Nlc3NcbiAgICAgICAgICAgIHJldHVybiAhIWNvbnRyb2wuZ2V0KGZpZWxkS2V5KT8udmFsdWU/LnRvU3RyaW5nKCkudHJpbSgpO1xuICAgICAgICB9KTtcblxuICAgICAgICByZXR1cm4gaGFzVmFsdWUgPyBudWxsIDogeyBbYmxvY2tOYW1lID8/ICdhdExlYXN0T25lUmVxdWlyZWQnXTogdHJ1ZSB9O1xuICAgIH07XG59XG4iXX0=
|
@@ -0,0 +1,14 @@
|
|
1
|
+
import { isValidPhoneNumber } from 'libphonenumber-js';
|
2
|
+
/**
|
3
|
+
* Проверяет корректность введенного номера телефона.
|
4
|
+
*/
|
5
|
+
export function phoneValidator() {
|
6
|
+
return (control) => {
|
7
|
+
if (control.value) {
|
8
|
+
const isValid = isValidPhoneNumber(control.value);
|
9
|
+
return isValid ? null : { phone: { value: control.value } };
|
10
|
+
}
|
11
|
+
return null;
|
12
|
+
};
|
13
|
+
}
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtcGhvbmUtdmFsaWRhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xpZW50LXVpL3ZhbGlkYXRvcnMvc2MtcGhvbmUtdmFsaWRhdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBRXZEOztHQUVHO0FBQ0gsTUFBTSxVQUFVLGNBQWM7SUFDMUIsT0FBTyxDQUFDLE9BQXVDLEVBQTJCLEVBQUU7UUFDeEUsSUFBSSxPQUFPLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDaEIsTUFBTSxPQUFPLEdBQUcsa0JBQWtCLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBRWxELE9BQU8sT0FBTyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsS0FBSyxFQUFFLEVBQUUsS0FBSyxFQUFFLE9BQU8sQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDO1FBQ2hFLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQWJzdHJhY3RDb250cm9sLCBWYWxpZGF0aW9uRXJyb3JzLCBWYWxpZGF0b3JGbiB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IGlzVmFsaWRQaG9uZU51bWJlciB9IGZyb20gJ2xpYnBob25lbnVtYmVyLWpzJztcblxuLyoqXG4gKiDQn9GA0L7QstC10YDRj9C10YIg0LrQvtGA0YDQtdC60YLQvdC+0YHRgtGMINCy0LLQtdC00LXQvdC90L7Qs9C+INC90L7QvNC10YDQsCDRgtC10LvQtdGE0L7QvdCwLlxuICovXG5leHBvcnQgZnVuY3Rpb24gcGhvbmVWYWxpZGF0b3IoKTogVmFsaWRhdG9yRm4ge1xuICAgIHJldHVybiAoY29udHJvbDogQWJzdHJhY3RDb250cm9sPHN0cmluZyB8IG51bGw+KTogVmFsaWRhdGlvbkVycm9ycyB8IG51bGwgPT4ge1xuICAgICAgICBpZiAoY29udHJvbC52YWx1ZSkge1xuICAgICAgICAgICAgY29uc3QgaXNWYWxpZCA9IGlzVmFsaWRQaG9uZU51bWJlcihjb250cm9sLnZhbHVlKTtcblxuICAgICAgICAgICAgcmV0dXJuIGlzVmFsaWQgPyBudWxsIDogeyBwaG9uZTogeyB2YWx1ZTogY29udHJvbC52YWx1ZSB9IH07XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9O1xufVxuIl19
|