@snabcentr/client-ui 3.22.0 → 3.24.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/cart/index.d.ts CHANGED
@@ -1 +1,2 @@
1
1
  export * from './cart-item/sc-cart-item.component';
2
+ export * from './sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component';
@@ -0,0 +1,62 @@
1
+ import { Signal } from '@angular/core';
2
+ import { FormControl } from '@angular/forms';
3
+ import { TuiDialogContext } from '@taiga-ui/core';
4
+ import { Observable, Subject } from 'rxjs';
5
+ import * as i0 from "@angular/core";
6
+ /**
7
+ * Компонент диалога для добавления товаров в корзину из CSV файла.
8
+ */
9
+ export declare class ScCarAddProductsFromCsvDialogComponent {
10
+ /**
11
+ * Контекст диалогового окна, в котором открыт компонент.
12
+ */
13
+ protected readonly context: TuiDialogContext;
14
+ /**
15
+ * Сервис для работы с корзиной.
16
+ */
17
+ private readonly cartService;
18
+ /**
19
+ * Объект окна.
20
+ */
21
+ private readonly window;
22
+ /**
23
+ * {@link Subject} события скачивания цен каталога.
24
+ */
25
+ protected readonly onDownloadClick: Subject<void>;
26
+ /**
27
+ * {@link Observable} запроса на скачивание цен каталога.
28
+ */
29
+ protected readonly downloadRequest$: Observable<Blob | null | undefined>;
30
+ /**
31
+ * Признак того, что запрос выполняется.
32
+ */
33
+ protected readonly isDownloadLoading: Signal<boolean>;
34
+ /**
35
+ * {@link Subject} события отправки формы.
36
+ */
37
+ protected readonly onSubmit$: Subject<void>;
38
+ /**
39
+ * {@link Observable} запроса на добавление товаров из CSV файла.
40
+ */
41
+ protected readonly submitRequest$: Observable<import("@snabcentr/client-core").ScCart | null>;
42
+ /**
43
+ * Признак того, что запрос на добавление товаров выполняется.
44
+ */
45
+ protected readonly isSubmitLoading: Signal<boolean>;
46
+ /**
47
+ * Поле для загрузки файла.
48
+ */
49
+ protected readonly control: FormControl<File | null>;
50
+ /**
51
+ * Метод для удаления загруженного файла.
52
+ */
53
+ protected removeFile(): void;
54
+ /**
55
+ * Метод для скачивания примера файла.
56
+ *
57
+ * @param blob Бинарный объект.
58
+ */
59
+ private downloadExampleFile;
60
+ static ɵfac: i0.ɵɵFactoryDeclaration<ScCarAddProductsFromCsvDialogComponent, never>;
61
+ static ɵcmp: i0.ɵɵComponentDeclaration<ScCarAddProductsFromCsvDialogComponent, "sc-car-add-products-from-csv-dialog", never, {}, {}, never, never, true, never>;
62
+ }
@@ -1,6 +1,7 @@
1
1
  export * from './download-price-list/sc-download-price-list.component';
2
2
  export * from './category-card/sc-category-card.component';
3
3
  export * from './input-quantity/sc-input-quantity.component';
4
+ export * from './notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component';
4
5
  export * from './price-card/sc-price-card.component';
5
6
  export * from './price-card-inline/sc-price-card-inline.component';
6
7
  export * from './price-history/sc-price-history.component';
@@ -0,0 +1,68 @@
1
+ import { Signal } from '@angular/core';
2
+ import { FormControl, FormGroup } from '@angular/forms';
3
+ import { ScCatalogService, ScISuggestionType, ScIWarehouse, ScProduct } from '@snabcentr/client-core';
4
+ import { TuiContext, TuiStringHandler } from '@taiga-ui/cdk';
5
+ import { TuiDialogContext } from '@taiga-ui/core';
6
+ import { Subject } from 'rxjs';
7
+ import * as i0 from "@angular/core";
8
+ /**
9
+ * Компонент формы отправки запроса на уведомление о поступлении товара.
10
+ */
11
+ export declare class ScNotifyWhenInStockDialogComponent {
12
+ /**
13
+ * Контекст диалогового окна, в котором открыт компонент.
14
+ */
15
+ protected readonly context: TuiDialogContext<void, {
16
+ product: ScProduct;
17
+ onClickOfferHandler: () => void;
18
+ }>;
19
+ /**
20
+ * Товар.
21
+ */
22
+ protected readonly product: ScProduct;
23
+ /**
24
+ * Форма для отправки запроса на уведомление о поступлении товара.
25
+ */
26
+ protected readonly form: FormGroup<{
27
+ email: FormControl<string | null>;
28
+ phone: FormControl<string | null>;
29
+ verificationCode: FormControl<string | null>;
30
+ warehouseId: FormControl<string | null>;
31
+ }>;
32
+ /**
33
+ * Список складов.
34
+ */
35
+ protected readonly warehouses: Signal<ScIWarehouse[] | null>;
36
+ /**
37
+ * Перечисление типов подсказок.
38
+ */
39
+ protected readonly suggestionType: typeof ScISuggestionType;
40
+ /**
41
+ * Сервис для работы с каталогом.
42
+ */
43
+ protected readonly catalogService: ScCatalogService;
44
+ /**
45
+ * {@link Subject} события отправки формы.
46
+ */
47
+ protected readonly onSubmit$: Subject<void>;
48
+ /**
49
+ * {@link Observable} запроса данных уведомления о поступлении товара.
50
+ */
51
+ private readonly request$;
52
+ /**
53
+ * Сигнал изменения состояния загрузки данных.
54
+ */
55
+ protected readonly loading: Signal<boolean>;
56
+ /**
57
+ * Проверяет, является ли товар в наличии.
58
+ */
59
+ protected isStock(): boolean;
60
+ /**
61
+ * Преобразует объект в значение, отображаемое в поле ввода.
62
+ *
63
+ * @param items Выбранные значения.
64
+ */
65
+ protected stringify(items: readonly ScIWarehouse[]): TuiStringHandler<TuiContext<number>>;
66
+ static ɵfac: i0.ɵɵFactoryDeclaration<ScNotifyWhenInStockDialogComponent, never>;
67
+ static ɵcmp: i0.ɵɵComponentDeclaration<ScNotifyWhenInStockDialogComponent, "sc-notify-when-in-stock-dialog", never, {}, {}, never, never, true, never>;
68
+ }
@@ -1,2 +1,3 @@
1
1
  export * from './cart-item/sc-cart-item.component';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9DQUFvQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9jYXJ0LWl0ZW0vc2MtY2FydC1pdGVtLmNvbXBvbmVudCc7XG4iXX0=
2
+ export * from './sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2FydC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG9DQUFvQyxDQUFDO0FBQ25ELGNBQWMscUZBQXFGLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2NhcnQtaXRlbS9zYy1jYXJ0LWl0ZW0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtY2FyLWFkZC1wcm9kdWN0cy1mcm9tLWNzdi1kaWFsb2cvc2MtY2FyLWFkZC1wcm9kdWN0cy1mcm9tLWNzdi1kaWFsb2cuY29tcG9uZW50JztcbiJdfQ==
@@ -0,0 +1,110 @@
1
+ import { NgIf } from '@angular/common';
2
+ import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
3
+ import { toSignal } from '@angular/core/rxjs-interop';
4
+ import { FormControl, ReactiveFormsModule } from '@angular/forms';
5
+ import { WA_WINDOW } from '@ng-web-apis/common';
6
+ import { ScCartService, ScUploadedFile } from '@snabcentr/client-core';
7
+ import { tuiIsPresent } from '@taiga-ui/cdk';
8
+ import { TuiButton, TuiLink, TuiLoader } from '@taiga-ui/core';
9
+ import { TuiButtonLoading, TuiFiles } from '@taiga-ui/kit';
10
+ import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
11
+ import { filter, map, Observable, share, startWith, Subject, switchMap, tap } from 'rxjs';
12
+ import * as i0 from "@angular/core";
13
+ import * as i1 from "@taiga-ui/kit";
14
+ import * as i2 from "@angular/forms";
15
+ /**
16
+ * Компонент диалога для добавления товаров в корзину из CSV файла.
17
+ */
18
+ export class ScCarAddProductsFromCsvDialogComponent {
19
+ constructor() {
20
+ /**
21
+ * Контекст диалогового окна, в котором открыт компонент.
22
+ */
23
+ this.context = inject(POLYMORPHEUS_CONTEXT);
24
+ /**
25
+ * Сервис для работы с корзиной.
26
+ */
27
+ this.cartService = inject(ScCartService);
28
+ /**
29
+ * Объект окна.
30
+ */
31
+ this.window = inject(WA_WINDOW);
32
+ /**
33
+ * {@link Subject} события скачивания цен каталога.
34
+ */
35
+ this.onDownloadClick = new Subject();
36
+ /**
37
+ * {@link Observable} запроса на скачивание цен каталога.
38
+ */
39
+ this.downloadRequest$ = this.onDownloadClick.pipe(switchMap(() => this.cartService.getCartCsvExample$().pipe(tap((blob) => {
40
+ this.downloadExampleFile(blob);
41
+ }), startWith(null))), share(),
42
+ // eslint-disable-next-line unicorn/no-useless-undefined
43
+ startWith(undefined));
44
+ /**
45
+ * Признак того, что запрос выполняется.
46
+ */
47
+ this.isDownloadLoading = toSignal(this.downloadRequest$.pipe(map((value) => value === null)), {
48
+ initialValue: false,
49
+ });
50
+ /**
51
+ * {@link Subject} события отправки формы.
52
+ */
53
+ this.onSubmit$ = new Subject();
54
+ /**
55
+ * {@link Observable} запроса на добавление товаров из CSV файла.
56
+ */
57
+ this.submitRequest$ = this.onSubmit$.pipe(map(() => this.control.value), filter(tuiIsPresent), switchMap((file) => new Observable((observer) => {
58
+ const reader = new FileReader();
59
+ reader.addEventListener('load', () => {
60
+ observer.next(new ScUploadedFile(file.name, reader.result));
61
+ });
62
+ reader.addEventListener('error', (e) => {
63
+ observer.error(e);
64
+ });
65
+ reader.readAsDataURL(file);
66
+ return () => {
67
+ reader.abort();
68
+ };
69
+ })), switchMap((file) => this.cartService.addProductsFromCsv$(file).pipe(tap(() => {
70
+ this.context.$implicit.complete();
71
+ }), startWith(null))));
72
+ /**
73
+ * Признак того, что запрос на добавление товаров выполняется.
74
+ */
75
+ this.isSubmitLoading = toSignal(this.submitRequest$.pipe(map((value) => value === null)), {
76
+ initialValue: false,
77
+ });
78
+ /**
79
+ * Поле для загрузки файла.
80
+ */
81
+ this.control = new FormControl(null);
82
+ }
83
+ /**
84
+ * Метод для удаления загруженного файла.
85
+ */
86
+ removeFile() {
87
+ this.control.setValue(null);
88
+ }
89
+ /**
90
+ * Метод для скачивания примера файла.
91
+ *
92
+ * @param blob Бинарный объект.
93
+ */
94
+ downloadExampleFile(blob) {
95
+ const url = this.window.URL.createObjectURL(blob);
96
+ // используем ссылку и download, чтобы указать название файла.
97
+ const a = this.window.document.createElement('a');
98
+ a.href = url;
99
+ a.download = 'Пример файла добавления товаров в корзину.csv';
100
+ a.click();
101
+ this.window.URL.revokeObjectURL(url);
102
+ }
103
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScCarAddProductsFromCsvDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
104
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ScCarAddProductsFromCsvDialogComponent, isStandalone: true, selector: "sc-car-add-products-from-csv-dialog", ngImport: i0, template: "<div class=\"flex flex-col items-center gap-8\">\n <tui-loader\n [overlay]=\"true\"\n [showLoader]=\"isDownloadLoading()\"\n size=\"s\"\n class=\"mt-8\"\n >\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"onDownloadClick.next()\"\n >\n \u041F\u0440\u0438\u043C\u0435\u0440 .csv \u0444\u0430\u0439\u043B\u0430\n </button>\n </tui-loader>\n\n <div class=\"flex w-full flex-col gap-1\">\n <label\n tuiInputFiles\n class=\"w-full\"\n >\n <input\n accept=\"text/csv\"\n tuiInputFiles\n [formControl]=\"control\"\n />\n </label>\n\n <tui-files class=\"tui-space_top-1\">\n <tui-file\n *ngIf=\"control.value as file\"\n [file]=\"file\"\n (remove)=\"removeFile()\"\n />\n </tui-files>\n </div>\n <div class=\"flex gap-2\">\n <button\n tuiButton\n [disabled]=\"!control.value\"\n [loading]=\"isSubmitLoading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n (click)=\"onSubmit$.next()\"\n >\n \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044C\n </button>\n <button\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "component", type: TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "component", type: i1.TuiFile, selector: "tui-file,a[tuiFile],button[tuiFile]", inputs: ["file", "state", "size", "showDelete", "showSize", "leftContent"], outputs: ["remove"] }, { kind: "component", type: i1.TuiInputFiles, selector: "label[tuiInputFiles]" }, { kind: "component", type: i1.TuiFilesComponent, selector: "tui-files", inputs: ["max", "expanded"], outputs: ["expandedChange"] }, { kind: "directive", type: i1.TuiInputFilesDirective, selector: "input[tuiInputFiles]", outputs: ["reject"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
105
+ }
106
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScCarAddProductsFromCsvDialogComponent, decorators: [{
107
+ type: Component,
108
+ args: [{ standalone: true, selector: 'sc-car-add-products-from-csv-dialog', imports: [TuiLink, TuiLoader, TuiFiles, ReactiveFormsModule, NgIf, TuiButton, TuiButtonLoading], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center gap-8\">\n <tui-loader\n [overlay]=\"true\"\n [showLoader]=\"isDownloadLoading()\"\n size=\"s\"\n class=\"mt-8\"\n >\n <button\n tuiLink\n [pseudo]=\"true\"\n type=\"button\"\n (click)=\"onDownloadClick.next()\"\n >\n \u041F\u0440\u0438\u043C\u0435\u0440 .csv \u0444\u0430\u0439\u043B\u0430\n </button>\n </tui-loader>\n\n <div class=\"flex w-full flex-col gap-1\">\n <label\n tuiInputFiles\n class=\"w-full\"\n >\n <input\n accept=\"text/csv\"\n tuiInputFiles\n [formControl]=\"control\"\n />\n </label>\n\n <tui-files class=\"tui-space_top-1\">\n <tui-file\n *ngIf=\"control.value as file\"\n [file]=\"file\"\n (remove)=\"removeFile()\"\n />\n </tui-files>\n </div>\n <div class=\"flex gap-2\">\n <button\n tuiButton\n [disabled]=\"!control.value\"\n [loading]=\"isSubmitLoading()\"\n iconStart=\"@tui.sc.send\"\n class=\"self-center\"\n (click)=\"onSubmit$.next()\"\n >\n \u0417\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u044C\n </button>\n <button\n tuiButton\n (click)=\"context.$implicit.complete()\"\n type=\"button\"\n appearance=\"secondary\"\n >\n \u041E\u0442\u043C\u0435\u043D\u0430\n </button>\n </div>\n</div>\n" }]
109
+ }] });
110
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-car-add-products-from-csv-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/cart/sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component.ts","../../../../../projects/client-ui/cart/sc-car-add-products-from-csv-dialog/sc-car-add-products-from-csv-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, inject, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { WA_WINDOW } from '@ng-web-apis/common';\nimport { ScCartService, ScUploadedFile } from '@snabcentr/client-core';\nimport { tuiIsPresent } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiLink, TuiLoader } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFiles } from '@taiga-ui/kit';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { filter, map, Observable, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\n/**\n * Компонент диалога для добавления товаров в корзину из CSV файла.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-car-add-products-from-csv-dialog',\n    templateUrl: './sc-car-add-products-from-csv-dialog.component.html',\n    imports: [TuiLink, TuiLoader, TuiFiles, ReactiveFormsModule, NgIf, TuiButton, TuiButtonLoading],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScCarAddProductsFromCsvDialogComponent {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext = inject<TuiDialogContext>(POLYMORPHEUS_CONTEXT);\n\n    /**\n     * Сервис для работы с корзиной.\n     */\n    private readonly cartService: ScCartService = inject(ScCartService);\n\n    /**\n     * Объект окна.\n     */\n    private readonly window: Window & typeof globalThis = inject<Window & typeof globalThis>(WA_WINDOW);\n\n    /**\n     * {@link Subject} события скачивания цен каталога.\n     */\n    protected readonly onDownloadClick: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на скачивание цен каталога.\n     */\n    protected readonly downloadRequest$ = this.onDownloadClick.pipe(\n        switchMap(() =>\n            this.cartService.getCartCsvExample$().pipe(\n                tap((blob: Blob) => {\n                    this.downloadExampleFile(blob);\n                }),\n                startWith(null)\n            )\n        ),\n        share(),\n        // eslint-disable-next-line unicorn/no-useless-undefined\n        startWith(undefined)\n    );\n\n    /**\n     * Признак того, что запрос выполняется.\n     */\n    protected readonly isDownloadLoading: Signal<boolean> = toSignal(this.downloadRequest$.pipe(map((value) => value === null)), {\n        initialValue: false,\n    });\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected readonly onSubmit$: Subject<void> = new Subject<void>();\n\n    /**\n     * {@link Observable} запроса на добавление товаров из CSV файла.\n     */\n    protected readonly submitRequest$ = this.onSubmit$.pipe(\n        map(() => this.control.value),\n        filter(tuiIsPresent),\n        switchMap(\n            (file: File) =>\n                new Observable<ScUploadedFile>((observer) => {\n                    const reader = new FileReader();\n\n                    reader.addEventListener('load', () => {\n                        observer.next(new ScUploadedFile(file.name, reader.result as string));\n                    });\n\n                    reader.addEventListener('error', (e) => {\n                        observer.error(e);\n                    });\n\n                    reader.readAsDataURL(file);\n\n                    return () => {\n                        reader.abort();\n                    };\n                })\n        ),\n        switchMap((file) =>\n            this.cartService.addProductsFromCsv$(file).pipe(\n                tap(() => {\n                    this.context.$implicit.complete();\n                }),\n                startWith(null)\n            )\n        )\n    );\n\n    /**\n     * Признак того, что запрос на добавление товаров выполняется.\n     */\n    protected readonly isSubmitLoading: Signal<boolean> = toSignal(this.submitRequest$.pipe(map((value) => value === null)), {\n        initialValue: false,\n    });\n\n    /**\n     * Поле для загрузки файла.\n     */\n    protected readonly control: FormControl<File | null> = new FormControl<File | null>(null);\n\n    /**\n     * Метод для удаления загруженного файла.\n     */\n    protected removeFile(): void {\n        this.control.setValue(null);\n    }\n\n    /**\n     * Метод для скачивания примера файла.\n     *\n     * @param blob Бинарный объект.\n     */\n    private downloadExampleFile(blob: Blob): void {\n        const url = this.window.URL.createObjectURL(blob);\n        // используем ссылку и download, чтобы указать название файла.\n        const a = this.window.document.createElement('a');\n\n        a.href = url;\n        a.download = 'Пример файла добавления товаров в корзину.csv';\n        a.click();\n        this.window.URL.revokeObjectURL(url);\n    }\n}\n","<div class=\"flex flex-col items-center gap-8\">\n    <tui-loader\n        [overlay]=\"true\"\n        [showLoader]=\"isDownloadLoading()\"\n        size=\"s\"\n        class=\"mt-8\"\n    >\n        <button\n            tuiLink\n            [pseudo]=\"true\"\n            type=\"button\"\n            (click)=\"onDownloadClick.next()\"\n        >\n            Пример .csv файла\n        </button>\n    </tui-loader>\n\n    <div class=\"flex w-full flex-col gap-1\">\n        <label\n            tuiInputFiles\n            class=\"w-full\"\n        >\n            <input\n                accept=\"text/csv\"\n                tuiInputFiles\n                [formControl]=\"control\"\n            />\n        </label>\n\n        <tui-files class=\"tui-space_top-1\">\n            <tui-file\n                *ngIf=\"control.value as file\"\n                [file]=\"file\"\n                (remove)=\"removeFile()\"\n            />\n        </tui-files>\n    </div>\n    <div class=\"flex gap-2\">\n        <button\n            tuiButton\n            [disabled]=\"!control.value\"\n            [loading]=\"isSubmitLoading()\"\n            iconStart=\"@tui.sc.send\"\n            class=\"self-center\"\n            (click)=\"onSubmit$.next()\"\n        >\n            Загрузить\n        </button>\n        <button\n            tuiButton\n            (click)=\"context.$implicit.complete()\"\n            type=\"button\"\n            appearance=\"secondary\"\n        >\n            Отмена\n        </button>\n    </div>\n</div>\n"]}
@@ -1,6 +1,7 @@
1
1
  export * from './download-price-list/sc-download-price-list.component';
2
2
  export * from './category-card/sc-category-card.component';
3
3
  export * from './input-quantity/sc-input-quantity.component';
4
+ export * from './notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component';
4
5
  export * from './price-card/sc-price-card.component';
5
6
  export * from './price-card-inline/sc-price-card-inline.component';
6
7
  export * from './price-history/sc-price-history.component';
@@ -8,4 +9,4 @@ export * from './price-list-pagination/sc-price-list-pagination.component';
8
9
  export * from './price-warehouse-stock/sc-price-warehouse-stock.component';
9
10
  export * from './sc-favorite-button/sc-favorite-button.component';
10
11
  export * from './sc-catalog.module';
11
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHdEQUF3RCxDQUFDO0FBQ3ZFLGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyw4Q0FBOEMsQ0FBQztBQUM3RCxjQUFjLHNDQUFzQyxDQUFDO0FBQ3JELGNBQWMsb0RBQW9ELENBQUM7QUFDbkUsY0FBYyw0Q0FBNEMsQ0FBQztBQUMzRCxjQUFjLDREQUE0RCxDQUFDO0FBQzNFLGNBQWMsNERBQTRELENBQUM7QUFDM0UsY0FBYyxtREFBbUQsQ0FBQztBQUNsRSxjQUFjLHFCQUFxQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9kb3dubG9hZC1wcmljZS1saXN0L3NjLWRvd25sb2FkLXByaWNlLWxpc3QuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vY2F0ZWdvcnktY2FyZC9zYy1jYXRlZ29yeS1jYXJkLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2lucHV0LXF1YW50aXR5L3NjLWlucHV0LXF1YW50aXR5LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3ByaWNlLWNhcmQvc2MtcHJpY2UtY2FyZC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9wcmljZS1jYXJkLWlubGluZS9zYy1wcmljZS1jYXJkLWlubGluZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9wcmljZS1oaXN0b3J5L3NjLXByaWNlLWhpc3RvcnkuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2UtbGlzdC1wYWdpbmF0aW9uL3NjLXByaWNlLWxpc3QtcGFnaW5hdGlvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9wcmljZS13YXJlaG91c2Utc3RvY2svc2MtcHJpY2Utd2FyZWhvdXNlLXN0b2NrLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWZhdm9yaXRlLWJ1dHRvbi9zYy1mYXZvcml0ZS1idXR0b24uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vc2MtY2F0YWxvZy5tb2R1bGUnO1xuIl19
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHdEQUF3RCxDQUFDO0FBQ3ZFLGNBQWMsNENBQTRDLENBQUM7QUFDM0QsY0FBYyw4Q0FBOEMsQ0FBQztBQUM3RCxjQUFjLHdFQUF3RSxDQUFDO0FBQ3ZGLGNBQWMsc0NBQXNDLENBQUM7QUFDckQsY0FBYyxvREFBb0QsQ0FBQztBQUNuRSxjQUFjLDRDQUE0QyxDQUFDO0FBQzNELGNBQWMsNERBQTRELENBQUM7QUFDM0UsY0FBYyw0REFBNEQsQ0FBQztBQUMzRSxjQUFjLG1EQUFtRCxDQUFDO0FBQ2xFLGNBQWMscUJBQXFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2Rvd25sb2FkLXByaWNlLWxpc3Qvc2MtZG93bmxvYWQtcHJpY2UtbGlzdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9jYXRlZ29yeS1jYXJkL3NjLWNhdGVnb3J5LWNhcmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vaW5wdXQtcXVhbnRpdHkvc2MtaW5wdXQtcXVhbnRpdHkuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbm90aWZ5LXdoZW4taW4tc3RvY2stZGlhbG9nL3NjLW5vdGlmeS13aGVuLWluLXN0b2NrLWRpYWxvZy5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9wcmljZS1jYXJkL3NjLXByaWNlLWNhcmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2UtY2FyZC1pbmxpbmUvc2MtcHJpY2UtY2FyZC1pbmxpbmUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2UtaGlzdG9yeS9zYy1wcmljZS1oaXN0b3J5LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3ByaWNlLWxpc3QtcGFnaW5hdGlvbi9zYy1wcmljZS1saXN0LXBhZ2luYXRpb24uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vcHJpY2Utd2FyZWhvdXNlLXN0b2NrL3NjLXByaWNlLXdhcmVob3VzZS1zdG9jay5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9zYy1mYXZvcml0ZS1idXR0b24vc2MtZmF2b3JpdGUtYnV0dG9uLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3NjLWNhdGFsb2cubW9kdWxlJztcbiJdfQ==
@@ -0,0 +1,126 @@
1
+ /* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */
2
+ import { AsyncPipe, NgIf } from '@angular/common';
3
+ import { HttpErrorResponse } from '@angular/common/http';
4
+ import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
5
+ import { toSignal } from '@angular/core/rxjs-interop';
6
+ import { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';
7
+ import { ScCatalogService, ScISuggestionType, ScWarehouseService } from '@snabcentr/client-core';
8
+ import { TuiButton, TuiError, TuiLabel, TuiLink, TuiLoader } from '@taiga-ui/core';
9
+ import { TuiButtonLoading, TuiFieldErrorPipe } from '@taiga-ui/kit';
10
+ import { TuiInputModule, TuiSelectModule } from '@taiga-ui/legacy';
11
+ import { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';
12
+ import { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';
13
+ import { ScFormFieldsModule } from '../../form-fields/form-fields.module';
14
+ import { ScVerificationModule } from '../../verification/sc-verification.module';
15
+ import * as i0 from "@angular/core";
16
+ import * as i1 from "../../verification/verification-phone-check-form/sc-verification-phone-check-form.component";
17
+ import * as i2 from "@angular/forms";
18
+ import * as i3 from "@taiga-ui/legacy";
19
+ import * as i4 from "../../form-fields/suggestion-field/sc-suggestion-field.component";
20
+ import * as i5 from "@taiga-ui/core/components/data-list";
21
+ /**
22
+ * Компонент формы отправки запроса на уведомление о поступлении товара.
23
+ */
24
+ export class ScNotifyWhenInStockDialogComponent {
25
+ constructor() {
26
+ /**
27
+ * Контекст диалогового окна, в котором открыт компонент.
28
+ */
29
+ this.context = inject(POLYMORPHEUS_CONTEXT);
30
+ /**
31
+ * Товар.
32
+ */
33
+ this.product = this.context.data.product;
34
+ /**
35
+ * Форма для отправки запроса на уведомление о поступлении товара.
36
+ */
37
+ this.form = new FormGroup({
38
+ email: new FormControl(null, [Validators.required, Validators.email]),
39
+ phone: new FormControl(null, [Validators.required, Validators.minLength(12)]),
40
+ verificationCode: new FormControl(null, [Validators.required, Validators.minLength(6)]),
41
+ warehouseId: new FormControl(null, Validators.required),
42
+ });
43
+ /**
44
+ * Список складов.
45
+ */
46
+ this.warehouses = toSignal(inject(ScWarehouseService).getWarehouses$(), { initialValue: null });
47
+ /**
48
+ * Перечисление типов подсказок.
49
+ */
50
+ this.suggestionType = ScISuggestionType;
51
+ /**
52
+ * Сервис для работы с каталогом.
53
+ */
54
+ this.catalogService = inject(ScCatalogService);
55
+ /**
56
+ * {@link Subject} события отправки формы.
57
+ */
58
+ this.onSubmit$ = new Subject();
59
+ /**
60
+ * {@link Observable} запроса данных уведомления о поступлении товара.
61
+ */
62
+ this.request$ = this.onSubmit$.pipe(filter(() => this.form.valid), map(() => this.form.value), switchMap((value) => this.catalogService.getProductNotifyWhenInStock$(this.product.id, value).pipe(tap(() => {
63
+ this.context.completeWith();
64
+ }), catchError((error) => {
65
+ if (error instanceof HttpErrorResponse) {
66
+ const { errors, message } = error.error;
67
+ if (errors) {
68
+ // eslint-disable-next-line no-shadow
69
+ Object.entries(errors).forEach(([key, value]) => {
70
+ this.form.get(key)?.setErrors({ serverResponse: value });
71
+ });
72
+ }
73
+ else if (message) {
74
+ this.form.setErrors({ serverResponse: [message] });
75
+ }
76
+ }
77
+ return of(undefined);
78
+ }), startWith(null))), share());
79
+ /**
80
+ * Сигнал изменения состояния загрузки данных.
81
+ */
82
+ this.loading = toSignal(this.request$.pipe(map((value) => value === null)), { initialValue: false });
83
+ }
84
+ /**
85
+ * Проверяет, является ли товар в наличии.
86
+ */
87
+ isStock() {
88
+ const warehouseId = Number(this.form.get('warehouseId')?.value ?? 0);
89
+ return this.product.isWarehouseStockExist(warehouseId);
90
+ }
91
+ /**
92
+ * Преобразует объект в значение, отображаемое в поле ввода.
93
+ *
94
+ * @param items Выбранные значения.
95
+ */
96
+ stringify(items) {
97
+ const itemsMap = new Map(items.map((item) => {
98
+ return [item.id, item.name];
99
+ }));
100
+ return ({ $implicit }) => itemsMap.get($implicit) ?? '';
101
+ }
102
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScNotifyWhenInStockDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
103
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ScNotifyWhenInStockDialogComponent, isStandalone: true, selector: "sc-notify-when-in-stock-dialog", ngImport: i0, template: "<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit$.next()\"\n class=\"flex flex-col gap-6\"\n>\n <div class=\"flex flex-col gap-3\">\n <sc-verification-phone-check-form [shouldBeConfirmed]=\"true\" />\n\n <label tuiLabel>\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 <label tuiLabel>\n \u0421\u043A\u043B\u0430\u0434\n <tui-select\n formControlName=\"warehouseId\"\n [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n >\n \u0421\u043A\u043B\u0430\u0434\n <tui-data-list *tuiDataList>\n @for (warehouse of warehouses(); track warehouse.id) {\n <button\n tuiOption\n [value]=\"warehouse.id\"\n >\n {{ warehouse.name }}\n </button>\n }\n </tui-data-list>\n </tui-select>\n <tui-error\n formControlName=\"warehouseId\"\n [error]=\"[] | tuiFieldError | async\"\n />\n\n <span\n *ngIf=\"isStock()\"\n [style.color]=\"'var(--tui-text-positive)'\"\n >\u0422\u043E\u0432\u0430\u0440 \u0438\u043C\u0435\u0435\u0442\u0441\u044F \u0432 \u043D\u0430\u043B\u0438\u0447\u0438\u0438 \u043D\u0430 \u0432\u044B\u0431\u0440\u0430\u043D\u043D\u043E\u043C \u0441\u043A\u043B\u0430\u0434\u0435</span\n >\n </label>\n <ng-template #selectLoading> <tui-loader /> </ng-template>\n </div>\n\n <div class=\"flex flex-wrap items-center gap-3\">\n <button\n tuiButton\n type=\"submit\"\n [disabled]=\"form.invalid || isStock()\"\n [loading]=\"loading()\"\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()\"\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\n <tui-error\n *ngIf=\"form.errors && form.errors['serverResponse']\"\n [error]=\"[] | tuiFieldError | async\"\n />\n</form>\n", dependencies: [{ kind: "ngmodule", type: ScVerificationModule }, { kind: "component", type: i1.ScVerificationPhoneCheckFormComponent, selector: "sc-verification-phone-check-form", inputs: ["showCodeFields", "readOnly", "shouldBeBusy", "shouldBeConfirmed", "haveCode"], outputs: ["haveCodeChange", "isBusyChange", "isConfirmedChange"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: TuiInputModule }, { kind: "component", type: i3.TuiInputComponent, selector: "tui-input" }, { kind: "directive", type: i3.TuiInputDirective, selector: "tui-input" }, { kind: "directive", type: TuiLink, selector: "a[tuiLink], button[tuiLink]", inputs: ["pseudo"] }, { kind: "directive", type: TuiButton, selector: "a[tuiButton],button[tuiButton],a[tuiIconButton],button[tuiIconButton]", inputs: ["size"] }, { kind: "directive", type: TuiLabel, selector: "label[tuiLabel]" }, { kind: "pipe", type: TuiFieldErrorPipe, name: "tuiFieldError" }, { kind: "component", type: TuiError, selector: "tui-error", inputs: ["error"] }, { kind: "ngmodule", type: ScFormFieldsModule }, { kind: "component", type: i4.ScSuggestionFieldComponent, selector: "sc-suggestion-field", inputs: ["type"], outputs: ["selectedClick"] }, { kind: "ngmodule", type: TuiSelectModule }, { kind: "component", type: i3.TuiSelectComponent, selector: "tui-select", inputs: ["stringify", "identityMatcher", "valueContent"] }, { kind: "directive", type: i3.TuiSelectDirective, selector: "tui-select" }, { kind: "component", type: i5.TuiDataListComponent, selector: "tui-data-list", inputs: ["emptyContent", "size"] }, { kind: "directive", type: i5.TuiDataListDirective, selector: "ng-template[tuiDataList]" }, { kind: "component", type: i5.TuiOption, selector: "button[tuiOption]:not([new]), a[tuiOption]:not([new]), label[tuiOption]:not([new])", inputs: ["disabled", "value"] }, { kind: "component", type: TuiButtonLoading, selector: "[tuiButton][loading],[tuiIconButton][loading]", inputs: ["size", "loading"] }, { kind: "component", type: TuiLoader, selector: "tui-loader", inputs: ["size", "inheritColor", "overlay", "textContent", "showLoader"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
104
+ }
105
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScNotifyWhenInStockDialogComponent, decorators: [{
106
+ type: Component,
107
+ args: [{ standalone: true, selector: 'sc-notify-when-in-stock-dialog', imports: [
108
+ ScVerificationModule,
109
+ FormsModule,
110
+ ReactiveFormsModule,
111
+ TuiInputModule,
112
+ TuiLink,
113
+ TuiButton,
114
+ TuiLabel,
115
+ TuiFieldErrorPipe,
116
+ TuiError,
117
+ ScVerificationModule,
118
+ ScFormFieldsModule,
119
+ TuiSelectModule,
120
+ TuiButtonLoading,
121
+ TuiLoader,
122
+ AsyncPipe,
123
+ NgIf,
124
+ ], changeDetection: ChangeDetectionStrategy.OnPush, template: "<form\n [formGroup]=\"form\"\n (ngSubmit)=\"onSubmit$.next()\"\n class=\"flex flex-col gap-6\"\n>\n <div class=\"flex flex-col gap-3\">\n <sc-verification-phone-check-form [shouldBeConfirmed]=\"true\" />\n\n <label tuiLabel>\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 <label tuiLabel>\n \u0421\u043A\u043B\u0430\u0434\n <tui-select\n formControlName=\"warehouseId\"\n [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n >\n \u0421\u043A\u043B\u0430\u0434\n <tui-data-list *tuiDataList>\n @for (warehouse of warehouses(); track warehouse.id) {\n <button\n tuiOption\n [value]=\"warehouse.id\"\n >\n {{ warehouse.name }}\n </button>\n }\n </tui-data-list>\n </tui-select>\n <tui-error\n formControlName=\"warehouseId\"\n [error]=\"[] | tuiFieldError | async\"\n />\n\n <span\n *ngIf=\"isStock()\"\n [style.color]=\"'var(--tui-text-positive)'\"\n >\u0422\u043E\u0432\u0430\u0440 \u0438\u043C\u0435\u0435\u0442\u0441\u044F \u0432 \u043D\u0430\u043B\u0438\u0447\u0438\u0438 \u043D\u0430 \u0432\u044B\u0431\u0440\u0430\u043D\u043D\u043E\u043C \u0441\u043A\u043B\u0430\u0434\u0435</span\n >\n </label>\n <ng-template #selectLoading> <tui-loader /> </ng-template>\n </div>\n\n <div class=\"flex flex-wrap items-center gap-3\">\n <button\n tuiButton\n type=\"submit\"\n [disabled]=\"form.invalid || isStock()\"\n [loading]=\"loading()\"\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()\"\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\n <tui-error\n *ngIf=\"form.errors && form.errors['serverResponse']\"\n [error]=\"[] | tuiFieldError | async\"\n />\n</form>\n" }]
125
+ }] });
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-notify-when-in-stock-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.ts","../../../../../projects/client-ui/catalog/notify-when-in-stock-dialog/sc-notify-when-in-stock-dialog.component.html"],"names":[],"mappings":"AAAA,2GAA2G;AAE3G,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,MAAM,EAAU,MAAM,eAAe,CAAC;AACnF,OAAO,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AACtD,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,WAAW,EAAE,mBAAmB,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AACtG,OAAO,EAAE,gBAAgB,EAA+B,iBAAiB,EAA2B,kBAAkB,EAAE,MAAM,wBAAwB,CAAC;AAEvJ,OAAO,EAAE,SAAS,EAAoB,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACrG,OAAO,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AACpE,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,kBAAkB,CAAC;AACnE,OAAO,EAAE,oBAAoB,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,EAAE,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AAG9F,OAAO,EAAE,kBAAkB,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,EAAE,oBAAoB,EAAE,MAAM,2CAA2C,CAAC;;;;;;;AAEjF;;GAEG;AAyBH,MAAM,OAAO,kCAAkC;IAxB/C;QAyBI;;WAEG;QACgB,YAAO,GACtB,MAAM,CAAkF,oBAAoB,CAAC,CAAC;QAElH;;WAEG;QACgB,YAAO,GAAc,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC;QAElE;;WAEG;QACgB,SAAI,GAAG,IAAI,SAAS,CAAC;YACpC,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,KAAK,CAAC,CAAC;YACpF,KAAK,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC;YAC5F,gBAAgB,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;YACtG,WAAW,EAAE,IAAI,WAAW,CAAgB,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC;SACzE,CAAC,CAAC;QAEH;;WAEG;QACgB,eAAU,GAAkC,QAAQ,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,cAAc,EAAE,EAAE,EAAE,YAAY,EAAE,IAAI,EAAE,CAAC,CAAC;QAE7I;;WAEG;QACgB,mBAAc,GAA6B,iBAAiB,CAAC;QAEhF;;WAEG;QACgB,mBAAc,GAAqB,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAE/E;;WAEG;QACgB,cAAS,GAAkB,IAAI,OAAO,EAAQ,CAAC;QAElE;;WAEG;QACc,aAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAC3C,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAC7B,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,KAAoC,CAAC,EACzD,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAChB,IAAI,CAAC,cAAc,CAAC,4BAA4B,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC,IAAI,CACzE,GAAG,CAAC,GAAG,EAAE;YACL,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;QAChC,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,KAAc,EAAE,EAAE;YAC1B,IAAI,KAAK,YAAY,iBAAiB,EAAE,CAAC;gBACrC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,KAAK,CAAC,KAAyB,CAAC;gBAE5D,IAAI,MAAM,EAAE,CAAC;oBACT,qCAAqC;oBACrC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE;wBAC5C,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,CAAC,EAAE,cAAc,EAAE,KAAK,EAAE,CAAC,CAAC;oBAC7D,CAAC,CAAC,CAAC;gBACP,CAAC;qBAAM,IAAI,OAAO,EAAE,CAAC;oBACjB,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,cAAc,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACvD,CAAC;YACL,CAAC;YAED,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC;QACzB,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,CAClB,CACJ,EACD,KAAK,EAAE,CACV,CAAC;QAEF;;WAEG;QACgB,YAAO,GAAoB,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,KAAK,IAAI,CAAC,CAAC,EAAE,EAAE,YAAY,EAAE,KAAK,EAAE,CAAC,CAAC;KAyBvI;IAvBG;;OAEG;IACO,OAAO;QACb,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC;QAErE,OAAO,IAAI,CAAC,OAAO,CAAC,qBAAqB,CAAC,WAAW,CAAC,CAAC;IAC3D,CAAC;IAED;;;;OAIG;IACO,SAAS,CAAC,KAA8B;QAC9C,MAAM,QAAQ,GAAG,IAAI,GAAG,CACpB,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;YACf,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,IAAI,CAAqB,CAAC;QACpD,CAAC,CAAC,CACL,CAAC;QAEF,OAAO,CAAC,EAAE,SAAS,EAAsB,EAAE,EAAE,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IAChF,CAAC;+GAtGQ,kCAAkC;mGAAlC,kCAAkC,0FC9C/C,s7GAoFA,2CDzDQ,oBAAoB,8SACpB,WAAW,2YACX,mBAAmB,+UACnB,cAAc,mLACd,OAAO,4FACP,SAAS,oIACT,QAAQ,uDACR,iBAAiB,sDACjB,QAAQ,wEAER,kBAAkB,yKAClB,eAAe,0mBACf,gBAAgB,uHAChB,SAAS,4HACT,SAAS,8CACT,IAAI;;4FAIC,kCAAkC;kBAxB9C,SAAS;iCACM,IAAI,YACN,gCAAgC,WAEjC;wBACL,oBAAoB;wBACpB,WAAW;wBACX,mBAAmB;wBACnB,cAAc;wBACd,OAAO;wBACP,SAAS;wBACT,QAAQ;wBACR,iBAAiB;wBACjB,QAAQ;wBACR,oBAAoB;wBACpB,kBAAkB;wBAClB,eAAe;wBACf,gBAAgB;wBAChB,SAAS;wBACT,SAAS;wBACT,IAAI;qBACP,mBACgB,uBAAuB,CAAC,MAAM","sourcesContent":["/* eslint-disable class-methods-use-this,unicorn/no-useless-undefined, @typescript-eslint/unbound-method */\n\nimport { AsyncPipe, NgIf } from '@angular/common';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport { ChangeDetectionStrategy, Component, inject, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { FormControl, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { ScCatalogService, ScIProductNotifyWhenInStock, ScISuggestionType, ScIWarehouse, ScProduct, ScWarehouseService } from '@snabcentr/client-core';\nimport { TuiContext, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TuiButton, TuiDialogContext, TuiError, TuiLabel, TuiLink, TuiLoader } from '@taiga-ui/core';\nimport { TuiButtonLoading, TuiFieldErrorPipe } from '@taiga-ui/kit';\nimport { TuiInputModule, TuiSelectModule } from '@taiga-ui/legacy';\nimport { POLYMORPHEUS_CONTEXT } from '@taiga-ui/polymorpheus';\nimport { catchError, filter, map, of, share, startWith, Subject, switchMap, tap } from 'rxjs';\n\nimport { ApiErrorResponse } from '../../auth';\nimport { ScFormFieldsModule } from '../../form-fields/form-fields.module';\nimport { ScVerificationModule } from '../../verification/sc-verification.module';\n\n/**\n * Компонент формы отправки запроса на уведомление о поступлении товара.\n */\n@Component({\n    standalone: true,\n    selector: 'sc-notify-when-in-stock-dialog',\n    templateUrl: './sc-notify-when-in-stock-dialog.component.html',\n    imports: [\n        ScVerificationModule,\n        FormsModule,\n        ReactiveFormsModule,\n        TuiInputModule,\n        TuiLink,\n        TuiButton,\n        TuiLabel,\n        TuiFieldErrorPipe,\n        TuiError,\n        ScVerificationModule,\n        ScFormFieldsModule,\n        TuiSelectModule,\n        TuiButtonLoading,\n        TuiLoader,\n        AsyncPipe,\n        NgIf,\n    ],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScNotifyWhenInStockDialogComponent {\n    /**\n     * Контекст диалогового окна, в котором открыт компонент.\n     */\n    protected readonly context: TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void }> =\n        inject<TuiDialogContext<void, { product: ScProduct; onClickOfferHandler: () => void }>>(POLYMORPHEUS_CONTEXT);\n\n    /**\n     * Товар.\n     */\n    protected readonly product: ScProduct = this.context.data.product;\n\n    /**\n     * Форма для отправки запроса на уведомление о поступлении товара.\n     */\n    protected readonly form = new FormGroup({\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        warehouseId: new FormControl<string | null>(null, Validators.required),\n    });\n\n    /**\n     * Список складов.\n     */\n    protected readonly warehouses: Signal<ScIWarehouse[] | null> = toSignal(inject(ScWarehouseService).getWarehouses$(), { initialValue: null });\n\n    /**\n     * Перечисление типов подсказок.\n     */\n    protected readonly suggestionType: typeof ScISuggestionType = ScISuggestionType;\n\n    /**\n     * Сервис для работы с каталогом.\n     */\n    protected readonly catalogService: ScCatalogService = inject(ScCatalogService);\n\n    /**\n     * {@link Subject} события отправки формы.\n     */\n    protected 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 ScIProductNotifyWhenInStock),\n        switchMap((value) =>\n            this.catalogService.getProductNotifyWhenInStock$(this.product.id, value).pipe(\n                tap(() => {\n                    this.context.completeWith();\n                }),\n                catchError((error: unknown) => {\n                    if (error instanceof HttpErrorResponse) {\n                        const { errors, message } = error.error as ApiErrorResponse;\n\n                        if (errors) {\n                            // eslint-disable-next-line no-shadow\n                            Object.entries(errors).forEach(([key, value]) => {\n                                this.form.get(key)?.setErrors({ serverResponse: value });\n                            });\n                        } else if (message) {\n                            this.form.setErrors({ serverResponse: [message] });\n                        }\n                    }\n\n                    return of(undefined);\n                }),\n                startWith(null)\n            )\n        ),\n        share()\n    );\n\n    /**\n     * Сигнал изменения состояния загрузки данных.\n     */\n    protected readonly loading: Signal<boolean> = toSignal(this.request$.pipe(map((value) => value === null)), { initialValue: false });\n\n    /**\n     * Проверяет, является ли товар в наличии.\n     */\n    protected isStock(): boolean {\n        const warehouseId = Number(this.form.get('warehouseId')?.value ?? 0);\n\n        return this.product.isWarehouseStockExist(warehouseId);\n    }\n\n    /**\n     * Преобразует объект в значение, отображаемое в поле ввода.\n     *\n     * @param items Выбранные значения.\n     */\n    protected stringify(items: readonly ScIWarehouse[]): TuiStringHandler<TuiContext<number>> {\n        const itemsMap = new Map(\n            items.map((item) => {\n                return [item.id, item.name] as [number, string];\n            })\n        );\n\n        return ({ $implicit }: TuiContext<number>) => itemsMap.get($implicit) ?? '';\n    }\n}\n","<form\n    [formGroup]=\"form\"\n    (ngSubmit)=\"onSubmit$.next()\"\n    class=\"flex flex-col gap-6\"\n>\n    <div class=\"flex flex-col gap-3\">\n        <sc-verification-phone-check-form [shouldBeConfirmed]=\"true\" />\n\n        <label tuiLabel>\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        <label tuiLabel>\n            Склад\n            <tui-select\n                formControlName=\"warehouseId\"\n                [valueContent]=\"warehouses() ? stringify(warehouses()!) : selectLoading\"\n            >\n                Склад\n                <tui-data-list *tuiDataList>\n                    @for (warehouse of warehouses(); track warehouse.id) {\n                        <button\n                            tuiOption\n                            [value]=\"warehouse.id\"\n                        >\n                            {{ warehouse.name }}\n                        </button>\n                    }\n                </tui-data-list>\n            </tui-select>\n            <tui-error\n                formControlName=\"warehouseId\"\n                [error]=\"[] | tuiFieldError | async\"\n            />\n\n            <span\n                *ngIf=\"isStock()\"\n                [style.color]=\"'var(--tui-text-positive)'\"\n                >Товар имеется в наличии на выбранном складе</span\n            >\n        </label>\n        <ng-template #selectLoading> <tui-loader /> </ng-template>\n    </div>\n\n    <div class=\"flex flex-wrap items-center gap-3\">\n        <button\n            tuiButton\n            type=\"submit\"\n            [disabled]=\"form.invalid || isStock()\"\n            [loading]=\"loading()\"\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()\"\n            >\n                Политикой конфиденциальности\n            </button>\n        </div>\n    </div>\n\n    <tui-error\n        *ngIf=\"form.errors && form.errors['serverResponse']\"\n        [error]=\"[] | tuiFieldError | async\"\n    />\n</form>\n"]}