@ecodev/natural 61.1.2 → 61.3.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.
@@ -155,7 +155,8 @@ export class NaturalAbstractDetail extends NaturalAbstractPanel {
155
155
  return this.router.navigateByUrl(newUrl); // replace /new by /123
156
156
  }
157
157
  else {
158
- return this.router.navigate(['..', model.id], { relativeTo: this.route });
158
+ const queryParams = this.route.snapshot.queryParams;
159
+ return this.router.navigate(['..', model.id], { relativeTo: this.route, queryParams });
159
160
  }
160
161
  }
161
162
  return EMPTY;
@@ -225,4 +226,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
225
226
  type: Directive,
226
227
  args: [{ standalone: true }]
227
228
  }], ctorParameters: () => [{ type: undefined }, { type: undefined }] });
228
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-detail.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/abstract-detail.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase} from 'lodash-es';\nimport {NaturalAlertService} from '../modules/alert/alert.service';\nimport {NaturalAbstractPanel} from '../modules/panels/abstract-panel';\nimport {NaturalAbstractModelService, WithId} from '../services/abstract-model.service';\nimport {ExtractResolve, ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {EMPTY, endWith, finalize, last, Observable, switchMap, tap} from 'rxjs';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {CumulativeChanges} from './cumulative-changes';\nimport {MAT_DIALOG_DATA} from '@angular/material/dialog';\nimport {NaturalDialogTriggerProvidedData} from '../modules/dialog-trigger/dialog-trigger.component';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * `Data` contains in `model` either the model fetched from DB or default values (without ID). And besides `model`,\n * any other extra keys defined by Extra.\n */\ntype Data<TService, Extra> = {model: {id?: string} & ExtractResolve<TService>} & Extra;\n\nfunction isNaturalDialogTriggerProvidedData(\n    dialogData: unknown,\n): dialogData is NaturalDialogTriggerProvidedData<never> {\n    return (\n        !!dialogData &&\n        typeof dialogData === 'object' &&\n        'activatedRoute' in dialogData &&\n        dialogData.activatedRoute instanceof ActivatedRoute\n    );\n}\n\n// @dynamic\n@Directive({standalone: true})\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            {id: string},\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n        ExtraResolve extends Literal = Record<never, never>,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Data retrieved by the server via route resolvers.\n     *\n     * The key `model` is special. It is readonly and represents the model being updated\n     * as it exists on server. The value is kept up to date when Apollo mutates it on server.\n     *\n     * The only time when `model` is not readonly is during creation. Only then can we modify the model values directly.\n     *\n     * Other keys, if present, are whatever is returned from route resolvers as-is.\n     */\n    public override data: Data<TService, ExtraResolve>;\n\n    /**\n     * Form that manages the data from the controller\n     */\n    public form: UntypedFormGroup = new UntypedFormGroup({});\n\n    /**\n     * Show / hides the bottom fab button (mostly to hide it when we are on other tabs where semantic of button can conflict with ...\n     * semantic of data on other tab, like relations that list other objects)\n     */\n    public showFabButton = true;\n\n    protected readonly alertService = inject(NaturalAlertService);\n\n    protected readonly router = inject(Router);\n\n    protected readonly route = inject(ActivatedRoute);\n\n    private readonly _dialogData: unknown = inject(MAT_DIALOG_DATA, {optional: true});\n\n    /**\n     * Once set, this must not change anymore, especially not right after the creation mutation,\n     * so the form does not switch from creation mode to update mode without an actual reload of\n     * model from DB (by navigating to update page).\n     */\n    private _isUpdatePage = false;\n    private readonly changes = new CumulativeChanges<typeof this.form.getRawValue>();\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n    ) {\n        super();\n\n        this.data = {\n            model: this.service.getDefaultForServer(),\n        } as Data<TService, ExtraResolve>;\n    }\n\n    /**\n     * You probably should not override this method. Instead, consider overriding `initForm()`.\n     */\n    public ngOnInit(): void {\n        if (this.isPanel) {\n            this.initForm();\n        } else {\n            const route = isNaturalDialogTriggerProvidedData(this._dialogData)\n                ? this._dialogData.activatedRoute\n                : this.route;\n            this.#subscribeToModelFromResolvedData(route);\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    #subscribeToModelFromResolvedData(route: ActivatedRoute): void {\n        let previousId = -1;\n        route.data\n            .pipe(\n                switchMap(data => {\n                    if (!(data.model instanceof Observable)) {\n                        throw new Error(\n                            'Resolved data must include the key `model`, and it must be an observable (usually one from Apollo).',\n                        );\n                    }\n\n                    // Subscribe to model to know when Apollo cache is changed, so we can reflect it into `data.model`\n                    return data.model.pipe(\n                        tap((model: ExtractResolve<TService>) => {\n                            this.data = {\n                                ...data,\n                                model: model,\n                            } as Data<TService, ExtraResolve>;\n\n                            if (previousId !== model.id) {\n                                previousId = model.id;\n                                this.initForm();\n                            }\n                        }),\n                    );\n                }),\n                takeUntilDestroyed(this.destroyRef),\n            )\n            .subscribe();\n    }\n\n    /**\n     * Returns whether `data.model` was fetched from DB, so we are on an update page, or if it is a new object\n     * with (only) default values, so we are on a creation page.\n     *\n     * This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing\n     */\n    protected isUpdatePage(): this is {data: {model: ExtractTone<TService>}} {\n        return this._isUpdatePage;\n    }\n\n    /**\n     * Update the object on the server with the values from the form fields that were modified since\n     * the initialization, or since the previous successful update.\n     *\n     * Form fields that are never modified are **not** sent to the server, unless if you specify `submitAllFields`.\n     */\n    public update(now = false, submitAllFields = false): void {\n        if (!this.isUpdatePage()) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            const newValues = this.form.getRawValue();\n            if (submitAllFields) {\n                this.changes.initialize({});\n            }\n\n            const toSubmit = {\n                id: this.data.model.id,\n                ...this.changes.differences(newValues),\n            };\n\n            const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);\n            update.subscribe(model => {\n                this.changes.commit(newValues);\n                this.alertService.info($localize`Mis à jour`);\n                this.postUpdate(model);\n            });\n        });\n    }\n\n    public create(redirect = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        const newValues = this.form.getRawValue();\n        this.form.disable();\n\n        this.service\n            .create(newValues)\n            .pipe(\n                switchMap(model => {\n                    this.alertService.info($localize`Créé`);\n\n                    return this.postCreate(model).pipe(endWith(model), last());\n                }),\n                switchMap((model: WithId<object>) => {\n                    if (redirect) {\n                        if (this.isPanel) {\n                            const oldUrl = this.router.url;\n                            const nextUrl: string = this.panelData?.config.params.nextRoute;\n                            const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');\n                            return this.router.navigateByUrl(newUrl); // replace /new by /123\n                        } else {\n                            return this.router.navigate(['..', model.id], {relativeTo: this.route});\n                        }\n                    }\n\n                    return EMPTY;\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    /**\n     * `confirmer` can be used to open a custom dialog, or anything else, to confirm the deletion, instead of the standard dialog\n     */\n    public delete(redirectionRoute?: unknown[], confirmer?: Observable<boolean | undefined>): void {\n        this.form.disable();\n\n        (\n            confirmer ??\n            this.alertService.confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n        )\n            .pipe(\n                switchMap(confirmed => {\n                    if (!confirmed || !this.isUpdatePage()) {\n                        return EMPTY;\n                    }\n\n                    this.preDelete(this.data.model);\n\n                    return this.service.delete([this.data.model]).pipe(\n                        switchMap(() => {\n                            this.alertService.info($localize`Supprimé`);\n\n                            if (this.isPanel) {\n                                this.panelService?.goToPenultimatePanel();\n\n                                return EMPTY;\n                            } else {\n                                const defaultRoute = ['../../' + kebabCase(this.key)];\n                                return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {\n                                    relativeTo: this.route,\n                                });\n                            }\n                        }),\n                    );\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postUpdate(model: ExtractTupdate<TService>): void {\n        // noop\n    }\n\n    /**\n     * Returns an observable that will be subscribed to immediately and the\n     * redirect navigation will only happen after the observable completes.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected preDelete(model: ExtractTone<TService>): void {\n        // noop\n    }\n\n    /**\n     * Initialize the form whenever it is needed.\n     *\n     * You should override this method, and not `ngOnInit()` if you need to customize the form. Because this will\n     * correctly be called more than one time per component instance if needed, when the route changes. But `ngOnInit()`\n     * will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.\n     */\n    protected initForm(): void {\n        this._isUpdatePage = !!this.data.model.id;\n        this.form = this.service.getFormGroup(this.data.model);\n        this.changes.initialize(this.form.getRawValue());\n    }\n}\n"]}
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-detail.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/abstract-detail.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, inject, OnInit} from '@angular/core';\nimport {UntypedFormGroup} from '@angular/forms';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {kebabCase} from 'lodash-es';\nimport {NaturalAlertService} from '../modules/alert/alert.service';\nimport {NaturalAbstractPanel} from '../modules/panels/abstract-panel';\nimport {NaturalAbstractModelService, WithId} from '../services/abstract-model.service';\nimport {ExtractResolve, ExtractTcreate, ExtractTone, ExtractTupdate, Literal} from '../types/types';\nimport {EMPTY, endWith, finalize, last, Observable, switchMap, tap} from 'rxjs';\nimport {ifValid, validateAllFormControls} from './validators';\nimport {PaginatedData} from './data-source';\nimport {QueryVariables} from './query-variable-manager';\nimport {CumulativeChanges} from './cumulative-changes';\nimport {MAT_DIALOG_DATA} from '@angular/material/dialog';\nimport {NaturalDialogTriggerProvidedData} from '../modules/dialog-trigger/dialog-trigger.component';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * `Data` contains in `model` either the model fetched from DB or default values (without ID). And besides `model`,\n * any other extra keys defined by Extra.\n */\ntype Data<TService, Extra> = {model: {id?: string} & ExtractResolve<TService>} & Extra;\n\nfunction isNaturalDialogTriggerProvidedData(\n    dialogData: unknown,\n): dialogData is NaturalDialogTriggerProvidedData<never> {\n    return (\n        !!dialogData &&\n        typeof dialogData === 'object' &&\n        'activatedRoute' in dialogData &&\n        dialogData.activatedRoute instanceof ActivatedRoute\n    );\n}\n\n// @dynamic\n@Directive({standalone: true})\nexport class NaturalAbstractDetail<\n        TService extends NaturalAbstractModelService<\n            {id: string},\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            unknown,\n            any\n        >,\n        ExtraResolve extends Literal = Record<never, never>,\n    >\n    extends NaturalAbstractPanel\n    implements OnInit\n{\n    /**\n     * Data retrieved by the server via route resolvers.\n     *\n     * The key `model` is special. It is readonly and represents the model being updated\n     * as it exists on server. The value is kept up to date when Apollo mutates it on server.\n     *\n     * The only time when `model` is not readonly is during creation. Only then can we modify the model values directly.\n     *\n     * Other keys, if present, are whatever is returned from route resolvers as-is.\n     */\n    public override data: Data<TService, ExtraResolve>;\n\n    /**\n     * Form that manages the data from the controller\n     */\n    public form: UntypedFormGroup = new UntypedFormGroup({});\n\n    /**\n     * Show / hides the bottom fab button (mostly to hide it when we are on other tabs where semantic of button can conflict with ...\n     * semantic of data on other tab, like relations that list other objects)\n     */\n    public showFabButton = true;\n\n    protected readonly alertService = inject(NaturalAlertService);\n\n    protected readonly router = inject(Router);\n\n    protected readonly route = inject(ActivatedRoute);\n\n    private readonly _dialogData: unknown = inject(MAT_DIALOG_DATA, {optional: true});\n\n    /**\n     * Once set, this must not change anymore, especially not right after the creation mutation,\n     * so the form does not switch from creation mode to update mode without an actual reload of\n     * model from DB (by navigating to update page).\n     */\n    private _isUpdatePage = false;\n    private readonly changes = new CumulativeChanges<typeof this.form.getRawValue>();\n\n    public constructor(\n        protected readonly key: string,\n        public readonly service: TService,\n    ) {\n        super();\n\n        this.data = {\n            model: this.service.getDefaultForServer(),\n        } as Data<TService, ExtraResolve>;\n    }\n\n    /**\n     * You probably should not override this method. Instead, consider overriding `initForm()`.\n     */\n    public ngOnInit(): void {\n        if (this.isPanel) {\n            this.initForm();\n        } else {\n            const route = isNaturalDialogTriggerProvidedData(this._dialogData)\n                ? this._dialogData.activatedRoute\n                : this.route;\n            this.#subscribeToModelFromResolvedData(route);\n        }\n    }\n\n    public changeTab(index: number): void {\n        this.showFabButton = index === 0;\n    }\n\n    #subscribeToModelFromResolvedData(route: ActivatedRoute): void {\n        let previousId = -1;\n        route.data\n            .pipe(\n                switchMap(data => {\n                    if (!(data.model instanceof Observable)) {\n                        throw new Error(\n                            'Resolved data must include the key `model`, and it must be an observable (usually one from Apollo).',\n                        );\n                    }\n\n                    // Subscribe to model to know when Apollo cache is changed, so we can reflect it into `data.model`\n                    return data.model.pipe(\n                        tap((model: ExtractResolve<TService>) => {\n                            this.data = {\n                                ...data,\n                                model: model,\n                            } as Data<TService, ExtraResolve>;\n\n                            if (previousId !== model.id) {\n                                previousId = model.id;\n                                this.initForm();\n                            }\n                        }),\n                    );\n                }),\n                takeUntilDestroyed(this.destroyRef),\n            )\n            .subscribe();\n    }\n\n    /**\n     * Returns whether `data.model` was fetched from DB, so we are on an update page, or if it is a new object\n     * with (only) default values, so we are on a creation page.\n     *\n     * This should be used instead of checking `data.model.id` directly, in order to type guard and get proper typing\n     */\n    protected isUpdatePage(): this is {data: {model: ExtractTone<TService>}} {\n        return this._isUpdatePage;\n    }\n\n    /**\n     * Update the object on the server with the values from the form fields that were modified since\n     * the initialization, or since the previous successful update.\n     *\n     * Form fields that are never modified are **not** sent to the server, unless if you specify `submitAllFields`.\n     */\n    public update(now = false, submitAllFields = false): void {\n        if (!this.isUpdatePage()) {\n            return;\n        }\n\n        validateAllFormControls(this.form);\n\n        ifValid(this.form).subscribe(() => {\n            const newValues = this.form.getRawValue();\n            if (submitAllFields) {\n                this.changes.initialize({});\n            }\n\n            const toSubmit = {\n                id: this.data.model.id,\n                ...this.changes.differences(newValues),\n            };\n\n            const update = now ? this.service.updateNow(toSubmit) : this.service.update(toSubmit);\n            update.subscribe(model => {\n                this.changes.commit(newValues);\n                this.alertService.info($localize`Mis à jour`);\n                this.postUpdate(model);\n            });\n        });\n    }\n\n    public create(redirect = true): void {\n        validateAllFormControls(this.form);\n\n        if (!this.form.valid) {\n            return;\n        }\n\n        const newValues = this.form.getRawValue();\n        this.form.disable();\n\n        this.service\n            .create(newValues)\n            .pipe(\n                switchMap(model => {\n                    this.alertService.info($localize`Créé`);\n\n                    return this.postCreate(model).pipe(endWith(model), last());\n                }),\n                switchMap((model: WithId<object>) => {\n                    if (redirect) {\n                        if (this.isPanel) {\n                            const oldUrl = this.router.url;\n                            const nextUrl: string = this.panelData?.config.params.nextRoute;\n                            const newUrl = oldUrl.replace('/new', '/' + model.id) + (nextUrl ? '/' + nextUrl : '');\n                            return this.router.navigateByUrl(newUrl); // replace /new by /123\n                        } else {\n                            const queryParams = this.route.snapshot.queryParams;\n                            return this.router.navigate(['..', model.id], {relativeTo: this.route, queryParams});\n                        }\n                    }\n\n                    return EMPTY;\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    /**\n     * `confirmer` can be used to open a custom dialog, or anything else, to confirm the deletion, instead of the standard dialog\n     */\n    public delete(redirectionRoute?: unknown[], confirmer?: Observable<boolean | undefined>): void {\n        this.form.disable();\n\n        (\n            confirmer ??\n            this.alertService.confirm(\n                $localize`Suppression`,\n                $localize`Voulez-vous supprimer définitivement cet élément ?`,\n                $localize`Supprimer définitivement`,\n            )\n        )\n            .pipe(\n                switchMap(confirmed => {\n                    if (!confirmed || !this.isUpdatePage()) {\n                        return EMPTY;\n                    }\n\n                    this.preDelete(this.data.model);\n\n                    return this.service.delete([this.data.model]).pipe(\n                        switchMap(() => {\n                            this.alertService.info($localize`Supprimé`);\n\n                            if (this.isPanel) {\n                                this.panelService?.goToPenultimatePanel();\n\n                                return EMPTY;\n                            } else {\n                                const defaultRoute = ['../../' + kebabCase(this.key)];\n                                return this.router.navigate(redirectionRoute ? redirectionRoute : defaultRoute, {\n                                    relativeTo: this.route,\n                                });\n                            }\n                        }),\n                    );\n                }),\n                finalize(() => this.form.enable()),\n            )\n            .subscribe();\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postUpdate(model: ExtractTupdate<TService>): void {\n        // noop\n    }\n\n    /**\n     * Returns an observable that will be subscribed to immediately and the\n     * redirect navigation will only happen after the observable completes.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected postCreate(model: ExtractTcreate<TService>): Observable<unknown> {\n        return EMPTY;\n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    protected preDelete(model: ExtractTone<TService>): void {\n        // noop\n    }\n\n    /**\n     * Initialize the form whenever it is needed.\n     *\n     * You should override this method, and not `ngOnInit()` if you need to customize the form. Because this will\n     * correctly be called more than one time per component instance if needed, when the route changes. But `ngOnInit()`\n     * will incorrectly be called exactly 1 time per component instance, even if the object changes via route navigation.\n     */\n    protected initForm(): void {\n        this._isUpdatePage = !!this.data.model.id;\n        this.form = this.service.getFormGroup(this.data.model);\n        this.changes.initialize(this.form.getRawValue());\n    }\n}\n"]}
@@ -27,7 +27,7 @@ export class NaturalFileComponent {
27
27
  *
28
28
  * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept
29
29
  */
30
- accept = 'image/bmp,image/gif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';
30
+ accept = 'image/avif,image/bmp,image/gif,image/heic,image/heif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';
31
31
  /**
32
32
  * If given, it will be called when a new file is selected. The callback should typically upload the file
33
33
  * to the server and link the newly uploaded file to the existing related object.
@@ -170,4 +170,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
170
170
  }], modelChange: [{
171
171
  type: Output
172
172
  }] } });
173
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/file/component/file.component.ts","../../../../../../../projects/natural/src/lib/modules/file/component/file.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    EventEmitter,\n    HostBinding,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    SimpleChanges,\n    inject,\n} from '@angular/core';\nimport {AbstractControl} from '@angular/forms';\nimport {Observable, of, Subject, tap} from 'rxjs';\nimport {NaturalFileService} from '../file.service';\nimport {CommonModule, DOCUMENT} from '@angular/common';\nimport {FileModel} from '../types';\nimport {NaturalAlertService} from '../../alert/alert.service';\nimport {NaturalCapitalizePipe} from '../../common/pipes/capitalize.pipe';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatRippleModule} from '@angular/material/core';\nimport {NaturalFileDropDirective} from '../file-drop.directive';\nimport {NaturalBackgroundDensityDirective} from '../../common/directives/background-density.directive';\n\n// @dynamic\n@Component({\n    selector: 'natural-file',\n    templateUrl: './file.component.html',\n    styleUrl: './file.component.scss',\n    standalone: true,\n    imports: [\n        NaturalFileDropDirective,\n        MatRippleModule,\n        CommonModule,\n        MatIconModule,\n        NaturalIconDirective,\n        NaturalCapitalizePipe,\n        NaturalBackgroundDensityDirective,\n    ],\n})\nexport class NaturalFileComponent implements OnInit, OnChanges {\n    private readonly naturalFileService = inject(NaturalFileService);\n    private readonly alertService = inject(NaturalAlertService);\n    private readonly document = inject(DOCUMENT);\n\n    @HostBinding('style.height.px') @Input() public height = 250;\n\n    @Input() public action: 'upload' | 'download' | null = null;\n\n    @Input() public backgroundSize = 'contain';\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element,\n     * it can be a mix of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept = 'image/bmp,image/gif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';\n\n    /**\n     * If given, it will be called when a new file is selected. The callback should typically upload the file\n     * to the server and link the newly uploaded file to the existing related object.\n     *\n     * The callback **must** be able to run even if the calling component has been destroyed. That means in most\n     * cases you **must** `bind()` the callback explicitly, like so:\n     *\n     * ```html\n     * <natural-file [uploader]=\"myCallback.bind(this)\" />\n     * ```\n     *\n     * Also, you probably **should** set a `[formCtrl]` so that the form is updated automatically, instead of doing\n     * it manually within the callback.\n     */\n    @Input() public uploader?: (file: File) => Observable<FileModel>;\n\n    @Input() public model: FileModel | null = null;\n\n    /**\n     * If provided, its value will get updated when the model changes.\n     * But its value is never read, so if you want to set a value use `[model]` instead.\n     */\n    @Input() public formCtrl: AbstractControl | null | undefined = null;\n\n    /**\n     * This **must not** be used to mutate the server, because it is very likely it will never be called if the\n     * human navigates away from the page before the upload is finished. Instead, you should use `[uploader]`.\n     */\n    @Output() public readonly modelChange = new EventEmitter<FileModel>();\n\n    public imagePreview = '';\n    public filePreview: string | null = null;\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.model && changes.model.previousValue !== changes.model.currentValue) {\n            this.updateImage();\n        }\n    }\n\n    public ngOnInit(): void {\n        this.updateImage();\n    }\n\n    public upload(file: File): void {\n        this.model = {file: file};\n        this.updateImage();\n\n        if (this.formCtrl) {\n            this.formCtrl.setValue(this.model);\n        }\n\n        const observable =\n            this.uploader?.(file).pipe(tap(() => this.alertService.info($localize`Mis à jour`))) ?? of(this.model);\n\n        observable.subscribe(result => {\n            this.model = result;\n            if (this.formCtrl) {\n                this.formCtrl.setValue(this.model);\n            }\n\n            this.modelChange.emit(this.model);\n        });\n    }\n\n    public getDownloadLink(): null | string {\n        if (this.action !== 'download') {\n            return null;\n        }\n\n        return this.naturalFileService.getDownloadLink(this.model);\n    }\n\n    private updateImage(): void {\n        this.imagePreview = '';\n        this.filePreview = null;\n        if (!this.model) {\n            return;\n        }\n\n        if (this.model.file?.type.includes('image/')) {\n            // Model from upload (before saving)\n            this.getBase64(this.model.file).subscribe(result => {\n                if (this.model?.file?.type) {\n                    const content = 'url(data:' + this.model?.file?.type + ';base64,' + result + ')';\n                    this.imagePreview = content;\n                }\n            });\n        } else if (this.model.file) {\n            this.filePreview = this.model.file.type.split('/')[1];\n        } else if (this.model.__typename === 'Image' && this.model.id) {\n            // Model image with id, use specific API to render image by size\n            const window = this.document.defaultView;\n            if (!window) {\n                throw new Error('Could not show image preview because `window` is undefined');\n            }\n\n            const loc = window.location;\n            const height = this.height ? '/' + this.height : '';\n\n            // create image url without port to stay compatible with dev mode\n            const image = loc.protocol + '//' + loc.hostname + '/api/image/' + this.model.id + height;\n            this.imagePreview = image;\n        } else if (this.model?.mime && ['File', 'AccountingDocument'].includes(this.model.__typename || '')) {\n            this.filePreview = this.model.mime.split('/')[1];\n        } else if (this.model.src) {\n            // external url\n            this.imagePreview = this.model.src;\n        }\n    }\n\n    private getBase64(file: File | null): Observable<string> {\n        if (!file) {\n            return of('');\n        }\n\n        const subject = new Subject<string>();\n\n        const reader = new FileReader();\n        reader.addEventListener('load', (ev: any) => {\n            subject.next(btoa(ev.target.result));\n            subject.complete();\n        });\n        reader.readAsBinaryString(file);\n\n        return subject.asObservable();\n    }\n}\n","<a\n    (fileChange)=\"upload($event)\"\n    naturalFileDrop\n    [selectable]=\"true\"\n    [accept]=\"accept\"\n    [attr.href]=\"getDownloadLink()\"\n    [class.has-action]=\"!!action\"\n    [class.suggest-upload]=\"!model && action === 'upload'\"\n    [fileSelectionDisabled]=\"action !== 'upload'\"\n    [matRippleDisabled]=\"!action\"\n    [naturalBackgroundDensity]=\"imagePreview\"\n    [style.backgroundSize]=\"backgroundSize\"\n    matRipple\n    target=\"_blank\"\n>\n    @if (filePreview) {\n        <div class=\"file-preview\">\n            <mat-icon [size]=\"height * 0.33\" naturalIcon=\"attachment\" />\n            {{ filePreview | uppercase }}\n        </div>\n    }\n\n    <div class=\"action-overlay\">\n        @if (action === 'upload') {\n            <mat-icon [size]=\"height * 0.66\" naturalIcon=\"cloud_upload\" />\n        }\n        @if (action === 'download') {\n            <mat-icon [size]=\"height * 0.66\" naturalIcon=\"get_app\" />\n        }\n        {{ action | capitalize }}\n    </div>\n</a>\n"]}
173
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/file/component/file.component.ts","../../../../../../../projects/natural/src/lib/modules/file/component/file.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    EventEmitter,\n    HostBinding,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    SimpleChanges,\n    inject,\n} from '@angular/core';\nimport {AbstractControl} from '@angular/forms';\nimport {Observable, of, Subject, tap} from 'rxjs';\nimport {NaturalFileService} from '../file.service';\nimport {CommonModule, DOCUMENT} from '@angular/common';\nimport {FileModel} from '../types';\nimport {NaturalAlertService} from '../../alert/alert.service';\nimport {NaturalCapitalizePipe} from '../../common/pipes/capitalize.pipe';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatRippleModule} from '@angular/material/core';\nimport {NaturalFileDropDirective} from '../file-drop.directive';\nimport {NaturalBackgroundDensityDirective} from '../../common/directives/background-density.directive';\n\n// @dynamic\n@Component({\n    selector: 'natural-file',\n    templateUrl: './file.component.html',\n    styleUrl: './file.component.scss',\n    standalone: true,\n    imports: [\n        NaturalFileDropDirective,\n        MatRippleModule,\n        CommonModule,\n        MatIconModule,\n        NaturalIconDirective,\n        NaturalCapitalizePipe,\n        NaturalBackgroundDensityDirective,\n    ],\n})\nexport class NaturalFileComponent implements OnInit, OnChanges {\n    private readonly naturalFileService = inject(NaturalFileService);\n    private readonly alertService = inject(NaturalAlertService);\n    private readonly document = inject(DOCUMENT);\n\n    @HostBinding('style.height.px') @Input() public height = 250;\n\n    @Input() public action: 'upload' | 'download' | null = null;\n\n    @Input() public backgroundSize = 'contain';\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element,\n     * it can be a mix of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept =\n        'image/avif,image/bmp,image/gif,image/heic,image/heif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';\n\n    /**\n     * If given, it will be called when a new file is selected. The callback should typically upload the file\n     * to the server and link the newly uploaded file to the existing related object.\n     *\n     * The callback **must** be able to run even if the calling component has been destroyed. That means in most\n     * cases you **must** `bind()` the callback explicitly, like so:\n     *\n     * ```html\n     * <natural-file [uploader]=\"myCallback.bind(this)\" />\n     * ```\n     *\n     * Also, you probably **should** set a `[formCtrl]` so that the form is updated automatically, instead of doing\n     * it manually within the callback.\n     */\n    @Input() public uploader?: (file: File) => Observable<FileModel>;\n\n    @Input() public model: FileModel | null = null;\n\n    /**\n     * If provided, its value will get updated when the model changes.\n     * But its value is never read, so if you want to set a value use `[model]` instead.\n     */\n    @Input() public formCtrl: AbstractControl | null | undefined = null;\n\n    /**\n     * This **must not** be used to mutate the server, because it is very likely it will never be called if the\n     * human navigates away from the page before the upload is finished. Instead, you should use `[uploader]`.\n     */\n    @Output() public readonly modelChange = new EventEmitter<FileModel>();\n\n    public imagePreview = '';\n    public filePreview: string | null = null;\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.model && changes.model.previousValue !== changes.model.currentValue) {\n            this.updateImage();\n        }\n    }\n\n    public ngOnInit(): void {\n        this.updateImage();\n    }\n\n    public upload(file: File): void {\n        this.model = {file: file};\n        this.updateImage();\n\n        if (this.formCtrl) {\n            this.formCtrl.setValue(this.model);\n        }\n\n        const observable =\n            this.uploader?.(file).pipe(tap(() => this.alertService.info($localize`Mis à jour`))) ?? of(this.model);\n\n        observable.subscribe(result => {\n            this.model = result;\n            if (this.formCtrl) {\n                this.formCtrl.setValue(this.model);\n            }\n\n            this.modelChange.emit(this.model);\n        });\n    }\n\n    public getDownloadLink(): null | string {\n        if (this.action !== 'download') {\n            return null;\n        }\n\n        return this.naturalFileService.getDownloadLink(this.model);\n    }\n\n    private updateImage(): void {\n        this.imagePreview = '';\n        this.filePreview = null;\n        if (!this.model) {\n            return;\n        }\n\n        if (this.model.file?.type.includes('image/')) {\n            // Model from upload (before saving)\n            this.getBase64(this.model.file).subscribe(result => {\n                if (this.model?.file?.type) {\n                    const content = 'url(data:' + this.model?.file?.type + ';base64,' + result + ')';\n                    this.imagePreview = content;\n                }\n            });\n        } else if (this.model.file) {\n            this.filePreview = this.model.file.type.split('/')[1];\n        } else if (this.model.__typename === 'Image' && this.model.id) {\n            // Model image with id, use specific API to render image by size\n            const window = this.document.defaultView;\n            if (!window) {\n                throw new Error('Could not show image preview because `window` is undefined');\n            }\n\n            const loc = window.location;\n            const height = this.height ? '/' + this.height : '';\n\n            // create image url without port to stay compatible with dev mode\n            const image = loc.protocol + '//' + loc.hostname + '/api/image/' + this.model.id + height;\n            this.imagePreview = image;\n        } else if (this.model?.mime && ['File', 'AccountingDocument'].includes(this.model.__typename || '')) {\n            this.filePreview = this.model.mime.split('/')[1];\n        } else if (this.model.src) {\n            // external url\n            this.imagePreview = this.model.src;\n        }\n    }\n\n    private getBase64(file: File | null): Observable<string> {\n        if (!file) {\n            return of('');\n        }\n\n        const subject = new Subject<string>();\n\n        const reader = new FileReader();\n        reader.addEventListener('load', (ev: any) => {\n            subject.next(btoa(ev.target.result));\n            subject.complete();\n        });\n        reader.readAsBinaryString(file);\n\n        return subject.asObservable();\n    }\n}\n","<a\n    (fileChange)=\"upload($event)\"\n    naturalFileDrop\n    [selectable]=\"true\"\n    [accept]=\"accept\"\n    [attr.href]=\"getDownloadLink()\"\n    [class.has-action]=\"!!action\"\n    [class.suggest-upload]=\"!model && action === 'upload'\"\n    [fileSelectionDisabled]=\"action !== 'upload'\"\n    [matRippleDisabled]=\"!action\"\n    [naturalBackgroundDensity]=\"imagePreview\"\n    [style.backgroundSize]=\"backgroundSize\"\n    matRipple\n    target=\"_blank\"\n>\n    @if (filePreview) {\n        <div class=\"file-preview\">\n            <mat-icon [size]=\"height * 0.33\" naturalIcon=\"attachment\" />\n            {{ filePreview | uppercase }}\n        </div>\n    }\n\n    <div class=\"action-overlay\">\n        @if (action === 'upload') {\n            <mat-icon [size]=\"height * 0.66\" naturalIcon=\"cloud_upload\" />\n        }\n        @if (action === 'download') {\n            <mat-icon [size]=\"height * 0.66\" naturalIcon=\"get_app\" />\n        }\n        {{ action | capitalize }}\n    </div>\n</a>\n"]}
@@ -8,7 +8,7 @@ import * as i1 from "@angular/material/button";
8
8
  import * as i2 from "@angular/material/icon";
9
9
  export class NaturalFixedButtonComponent {
10
10
  icon;
11
- link = [];
11
+ link = null;
12
12
  color = 'accent';
13
13
  disabled = false;
14
14
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFixedButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
@@ -27,4 +27,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
27
27
  }], disabled: [{
28
28
  type: Input
29
29
  }] } });
30
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZml4ZWQtYnV0dG9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2ZpeGVkLWJ1dHRvbi9maXhlZC1idXR0b24uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZml4ZWQtYnV0dG9uL2ZpeGVkLWJ1dHRvbi5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFFLEtBQUssRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUUvQyxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0saUJBQWlCLENBQUM7QUFDM0MsT0FBTyxFQUFDLG9CQUFvQixFQUFDLE1BQU0sd0JBQXdCLENBQUM7QUFDNUQsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQ3JELE9BQU8sRUFBQyxlQUFlLEVBQUMsTUFBTSwwQkFBMEIsQ0FBQzs7OztBQVN6RCxNQUFNLE9BQU8sMkJBQTJCO0lBQ0osSUFBSSxDQUFVO0lBQzlCLElBQUksR0FBNkIsRUFBRSxDQUFDO0lBQ3BDLEtBQUssR0FBaUIsUUFBUSxDQUFDO0lBQy9CLFFBQVEsR0FBRyxLQUFLLENBQUM7dUdBSnhCLDJCQUEyQjsyRkFBM0IsMkJBQTJCLDhKQ2R4Qyx3SUFHQSw2SERTYyxlQUFlLHlKQUFFLFVBQVUsbU9BQUUsYUFBYSxvTEFBRSxvQkFBb0I7OzJGQUVqRSwyQkFBMkI7a0JBUHZDLFNBQVM7K0JBQ0ksc0JBQXNCLGNBR3BCLElBQUksV0FDUCxDQUFDLGVBQWUsRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLG9CQUFvQixDQUFDOzhCQUczQyxJQUFJO3NCQUFuQyxLQUFLO3VCQUFDLEVBQUMsUUFBUSxFQUFFLElBQUksRUFBQztnQkFDUCxJQUFJO3NCQUFuQixLQUFLO2dCQUNVLEtBQUs7c0JBQXBCLEtBQUs7Z0JBQ1UsUUFBUTtzQkFBdkIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29tcG9uZW50LCBJbnB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge1RoZW1lUGFsZXR0ZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvY29yZSc7XG5pbXBvcnQge1JvdXRlckxpbmt9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQge05hdHVyYWxJY29uRGlyZWN0aXZlfSBmcm9tICcuLi9pY29uL2ljb24uZGlyZWN0aXZlJztcbmltcG9ydCB7TWF0SWNvbk1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvaWNvbic7XG5pbXBvcnQge01hdEJ1dHRvbk1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvYnV0dG9uJztcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICduYXR1cmFsLWZpeGVkLWJ1dHRvbicsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2ZpeGVkLWJ1dHRvbi5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmw6ICcuL2ZpeGVkLWJ1dHRvbi5jb21wb25lbnQuc2NzcycsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbTWF0QnV0dG9uTW9kdWxlLCBSb3V0ZXJMaW5rLCBNYXRJY29uTW9kdWxlLCBOYXR1cmFsSWNvbkRpcmVjdGl2ZV0sXG59KVxuZXhwb3J0IGNsYXNzIE5hdHVyYWxGaXhlZEJ1dHRvbkNvbXBvbmVudCB7XG4gICAgQElucHV0KHtyZXF1aXJlZDogdHJ1ZX0pIHB1YmxpYyBpY29uITogc3RyaW5nO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBsaW5rOiBSb3V0ZXJMaW5rWydyb3V0ZXJMaW5rJ10gPSBbXTtcbiAgICBASW5wdXQoKSBwdWJsaWMgY29sb3I6IFRoZW1lUGFsZXR0ZSA9ICdhY2NlbnQnO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBkaXNhYmxlZCA9IGZhbHNlO1xufVxuIiwiPGJ1dHRvbiBbY29sb3JdPVwiY29sb3JcIiBbZGlzYWJsZWRdPVwiZGlzYWJsZWRcIiBbcm91dGVyTGlua109XCJsaW5rXCIgbWF0LWZhYj5cbiAgICA8bWF0LWljb24gW25hdHVyYWxJY29uXT1cImljb25cIiAvPlxuPC9idXR0b24+XG4iXX0=
30
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZml4ZWQtYnV0dG9uLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2ZpeGVkLWJ1dHRvbi9maXhlZC1idXR0b24uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvZml4ZWQtYnV0dG9uL2ZpeGVkLWJ1dHRvbi5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFFLEtBQUssRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUUvQyxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0saUJBQWlCLENBQUM7QUFDM0MsT0FBTyxFQUFDLG9CQUFvQixFQUFDLE1BQU0sd0JBQXdCLENBQUM7QUFDNUQsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQ3JELE9BQU8sRUFBQyxlQUFlLEVBQUMsTUFBTSwwQkFBMEIsQ0FBQzs7OztBQVN6RCxNQUFNLE9BQU8sMkJBQTJCO0lBQ0osSUFBSSxDQUFVO0lBQzlCLElBQUksR0FBb0MsSUFBSSxDQUFDO0lBQzdDLEtBQUssR0FBaUIsUUFBUSxDQUFDO0lBQy9CLFFBQVEsR0FBRyxLQUFLLENBQUM7dUdBSnhCLDJCQUEyQjsyRkFBM0IsMkJBQTJCLDhKQ2R4Qyx3SUFHQSw2SERTYyxlQUFlLHlKQUFFLFVBQVUsbU9BQUUsYUFBYSxvTEFBRSxvQkFBb0I7OzJGQUVqRSwyQkFBMkI7a0JBUHZDLFNBQVM7K0JBQ0ksc0JBQXNCLGNBR3BCLElBQUksV0FDUCxDQUFDLGVBQWUsRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLG9CQUFvQixDQUFDOzhCQUczQyxJQUFJO3NCQUFuQyxLQUFLO3VCQUFDLEVBQUMsUUFBUSxFQUFFLElBQUksRUFBQztnQkFDUCxJQUFJO3NCQUFuQixLQUFLO2dCQUNVLEtBQUs7c0JBQXBCLEtBQUs7Z0JBQ1UsUUFBUTtzQkFBdkIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29tcG9uZW50LCBJbnB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge1RoZW1lUGFsZXR0ZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvY29yZSc7XG5pbXBvcnQge1JvdXRlckxpbmt9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5pbXBvcnQge05hdHVyYWxJY29uRGlyZWN0aXZlfSBmcm9tICcuLi9pY29uL2ljb24uZGlyZWN0aXZlJztcbmltcG9ydCB7TWF0SWNvbk1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvaWNvbic7XG5pbXBvcnQge01hdEJ1dHRvbk1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvYnV0dG9uJztcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICduYXR1cmFsLWZpeGVkLWJ1dHRvbicsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2ZpeGVkLWJ1dHRvbi5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmw6ICcuL2ZpeGVkLWJ1dHRvbi5jb21wb25lbnQuc2NzcycsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbTWF0QnV0dG9uTW9kdWxlLCBSb3V0ZXJMaW5rLCBNYXRJY29uTW9kdWxlLCBOYXR1cmFsSWNvbkRpcmVjdGl2ZV0sXG59KVxuZXhwb3J0IGNsYXNzIE5hdHVyYWxGaXhlZEJ1dHRvbkNvbXBvbmVudCB7XG4gICAgQElucHV0KHtyZXF1aXJlZDogdHJ1ZX0pIHB1YmxpYyBpY29uITogc3RyaW5nO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBsaW5rOiBSb3V0ZXJMaW5rWydyb3V0ZXJMaW5rJ10gfCBudWxsID0gbnVsbDtcbiAgICBASW5wdXQoKSBwdWJsaWMgY29sb3I6IFRoZW1lUGFsZXR0ZSA9ICdhY2NlbnQnO1xuICAgIEBJbnB1dCgpIHB1YmxpYyBkaXNhYmxlZCA9IGZhbHNlO1xufVxuIiwiPGJ1dHRvbiBbY29sb3JdPVwiY29sb3JcIiBbZGlzYWJsZWRdPVwiZGlzYWJsZWRcIiBbcm91dGVyTGlua109XCJsaW5rXCIgbWF0LWZhYj5cbiAgICA8bWF0LWljb24gW25hdHVyYWxJY29uXT1cImljb25cIiAvPlxuPC9idXR0b24+XG4iXX0=
@@ -72,12 +72,12 @@ export class AbstractSelect {
72
72
  /**
73
73
  * Contains internal representation for current selection AND searched text (for autocomplete)
74
74
  *
75
- * It is **not** necessarily `V | null`.
75
+ * It is **not** necessarily `TValue | null`.
76
76
  *
77
- * - NaturalSelectComponent: `string | V | null`. We allow `string`
78
- * only when `optionRequired` is false, so most of the time it is `V | null`.
77
+ * - NaturalSelectComponent: `string | TValue | null`. We allow `string`
78
+ * only when `optionRequired` is false, so most of the time it is `TValue | null`.
79
79
  * - NaturalSelectHierarchicComponent: `string | null`.
80
- * - NaturalSelectEnumComponent: `V | null`.
80
+ * - NaturalSelectEnumComponent: `TValue | null`.
81
81
  *
82
82
  * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :
83
83
  * - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed
@@ -220,4 +220,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
220
220
  }], disabled: [{
221
221
  type: Input
222
222
  }] } });
223
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-select.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/select/abstract-select.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,qBAAqB,EAAC,MAAM,uBAAuB,CAAC;AAC5D,OAAO,EAAC,SAAS,EAAW,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,MAAM,EAAC,MAAM,eAAe,CAAC;AAC9F,OAAO,EAGH,WAAW,EACX,oBAAoB,EACpB,eAAe,EACf,SAAS,EACT,UAAU,GACb,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAC,iBAAiB,EAAC,MAAM,wBAAwB,CAAC;;AAEzD;;;;;GAKG;AACH,MAAM,0BAAiC,SAAQ,iBAAiB;IACxB;IAApC,YAAoC,SAA+B;QAC/D,KAAK,EAAE,CAAC;QADwB,cAAS,GAAT,SAAS,CAAsB;IAEnE,CAAC;IAEe,YAAY;QACxB,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,OAAO,IAAI,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC;QACtF,IAAI,YAAY,EAAE,CAAC;YACf,OAAO,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,IAAI,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC;QACnF,CAAC;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;CACJ;AAGD,MAAM,OAAgB,cAAc;IAChB,WAAW,CAAU;IAErC;;OAEG;IACa,IAAI,GAAkB,IAAI,CAAC;IAE3C;;OAEG;IACH,IACW,QAAQ,CAAC,KAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAC9C,IAAI,CAAC,aAAa,EAAE,CAAC;IACzB,CAAC;IAED,IAAW,QAAQ;QACf,OAAO,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC;IAC5B,CAAC;IAEO,SAAS,CAAsB;IAEvC;;OAEG;IACa,UAAU,CAAyB;IAEnD;;OAEG;IACa,UAAU,CAAU;IAEpC;;OAEG;IACa,QAAQ,GAAG,IAAI,CAAC;IAEhC;;OAEG;IACa,IAAI,GAAG,QAAQ,CAAC;IAEhC;;OAEG;IACa,WAAW,CAA8B;IAEzD;;OAEG;IACuB,eAAe,GAAG,IAAI,YAAY,EAAY,CAAC;IAEzE;;OAEG;IACH,4DAA4D;IAClC,IAAI,GAAG,IAAI,YAAY,EAAQ,CAAC;IAE1D;;;;;;;;;;;;;OAaG;IACa,YAAY,GAAG,IAAI,WAAW,CAAW,IAAI,CAAC,CAAC;IAE/D;;;OAGG;IACI,QAAQ,CAA4B;IAE3C;;;OAGG;IACI,SAAS,CAAc;IAEd,OAAO,CAAmC;IAC1C,SAAS,GAAG,MAAM,CAAC,SAAS,EAAE,EAAC,QAAQ,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAC,CAAC,CAAC;IAE5E;QACI,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,SAAS,CAAC,aAAa,GAAG,IAAI,CAAC;QACxC,CAAC;QAED,IAAI,CAAC,OAAO,GAAG,IAAI,0BAA0B,CAAC,IAAI,CAAC,CAAC;IACxD,CAAC;IAEM,SAAS;QACZ,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,aAAa,EAAE,CAAC;QACzB,CAAC;IACL,CAAC;IAEM,UAAU,CAAC,KAAe;QAC7B,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAEM,QAAQ;QACX,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,YAAY,oBAAoB,IAAI,IAAI,CAAC,SAAS,YAAY,eAAe,CAAC;QAC/G,IAAI,UAAU,IAAI,OAAO,IAAI,CAAC,SAAS,KAAK,WAAW,EAAE,CAAC;YACtD,OAAO,CAAC,IAAI,CAAC,yFAAyF,CAAC,CAAC;QAC5G,CAAC;IACL,CAAC;IAED;;OAEG;IACH,IACW,QAAQ,CAAC,QAAiB;QACjC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC;IACxE,CAAC;IAEM,gBAAgB,CAAC,EAA4B;QAChD,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACvB,CAAC;IAEM,iBAAiB,CAAC,EAAc;QACnC,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACxB,CAAC;IAID;;;OAGG;IACI,KAAK;QACR,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED;;OAEG;IACI,MAAM;QACT,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;IACrB,CAAC;IAED;;OAEG;IACI,cAAc,CAAC,KAAe;QACjC,6FAA6F;QAC7F,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAChB,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACzB,CAAC;QAED,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACrC,CAAC;IAEM,gBAAgB,CAAC,UAAmB;QACvC,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;IAC/B,CAAC;IAEM,eAAe;QAClB,OAAO,IAAI,CAAC,YAAY,EAAE,OAAO,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;IACxF,CAAC;IAEM,KAAK;QACR,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,SAAS,EAAE,CAAC;QACrB,CAAC;IACL,CAAC;IAEM,gBAAgB;QACnB,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC;QAEtF,OAAO,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;IACxC,CAAC;IAED;;OAEG;IACK,aAAa;QACjB,sCAAsC;QACtC,MAAM,mBAAmB,GAAG,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC,EAAqB,CAAC,EAAE,QAAQ,CAAC;QAEnG,sDAAsD;QACtD,MAAM,iBAAiB,GAAG,OAAO,IAAI,CAAC,SAAS,KAAK,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,mBAAmB,CAAC;QAEvG,oCAAoC;QACpC,MAAM,qBAAqB,GAAG,IAAI,CAAC,YAAY,EAAE,SAAS,EAAE,CAAC,EAAqB,CAAC,EAAE,QAAQ,CAAC;QAE9F,gEAAgE;QAChE,IAAI,qBAAqB,KAAK,iBAAiB,EAAE,CAAC;YAC9C,OAAO;QACX,CAAC;QAED,wBAAwB;QACxB,IAAI,iBAAiB,EAAE,CAAC;YACpB,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QACzD,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE,CAAC;QACxC,CAAC;QAED,IAAI,CAAC,YAAY,CAAC,sBAAsB,EAAE,CAAC;IAC/C,CAAC;uGAhNiB,cAAc;2FAAd,cAAc;;2FAAd,cAAc;kBADnC,SAAS;mBAAC,EAAC,UAAU,EAAE,IAAI,EAAC;wDAET,WAAW;sBAA1B,KAAK;gBAKU,IAAI;sBAAnB,KAAK;gBAMK,QAAQ;sBADlB,KAAK;gBAeU,UAAU;sBAAzB,KAAK;gBAKU,UAAU;sBAAzB,KAAK;gBAKU,QAAQ;sBAAvB,KAAK;gBAKU,IAAI;sBAAnB,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAKoB,eAAe;sBAAxC,MAAM;gBAMmB,IAAI;sBAA7B,MAAM;gBA8DI,QAAQ;sBADlB,KAAK","sourcesContent":["import {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {Directive, DoCheck, EventEmitter, inject, Input, OnInit, Output} from '@angular/core';\nimport {\n    AbstractControl,\n    ControlValueAccessor,\n    FormControl,\n    FormControlDirective,\n    FormControlName,\n    NgControl,\n    Validators,\n} from '@angular/forms';\nimport {ErrorStateMatcher} from '@angular/material/core';\n\n/**\n * This will completely ignore internal formControl and instead use the one from the component\n * which comes from outside of this component. This basically allows us to **not** depend on\n * touched status propagation between outside and inside world, and thus get rid of our legacy\n * custom FormControl class (\"NaturalFormControl\").\n */\nclass ExternalFormControlMatcher<T, I> extends ErrorStateMatcher {\n    public constructor(private readonly component: AbstractSelect<T, I>) {\n        super();\n    }\n\n    public override isErrorState(): boolean {\n        const externalCtrl = this.component.ngControl?.control || this.component.internalCtrl;\n        if (externalCtrl) {\n            return !!(externalCtrl.errors && (externalCtrl.touched || externalCtrl.dirty));\n        }\n\n        return false;\n    }\n}\n\n@Directive({standalone: true})\nexport abstract class AbstractSelect<V, I> implements OnInit, ControlValueAccessor, DoCheck {\n    @Input() public placeholder?: string;\n\n    /**\n     * Mat-hint\n     */\n    @Input() public hint: string | null = null;\n\n    /**\n     * If the field is required\n     */\n    @Input()\n    public set required(value: boolean) {\n        this._required = coerceBooleanProperty(value);\n        this.applyRequired();\n    }\n\n    public get required(): boolean {\n        return !!this._required;\n    }\n\n    private _required: boolean | undefined;\n\n    /**\n     * Add a suffix button that is a link to given destination\n     */\n    @Input() public navigateTo?: any[] | string | null;\n\n    /**\n     * If provided cause a new clear button to appear\n     */\n    @Input() public clearLabel?: string;\n\n    /**\n     * Whether to show the search icon\n     */\n    @Input() public showIcon = true;\n\n    /**\n     * Icon name\n     */\n    @Input() public icon = 'search';\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: V | null) => string;\n\n    /**\n     * Emit the selected value whenever it changes\n     */\n    @Output() public readonly selectionChange = new EventEmitter<V | null>();\n\n    /**\n     * Emits when internal input is blurred\n     */\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() public readonly blur = new EventEmitter<void>();\n\n    /**\n     * Contains internal representation for current selection AND searched text (for autocomplete)\n     *\n     * It is **not** necessarily `V | null`.\n     *\n     * - NaturalSelectComponent: `string | V | null`. We allow `string`\n     *   only when `optionRequired` is false, so most of the time it is `V | null`.\n     * - NaturalSelectHierarchicComponent: `string | null`.\n     * - NaturalSelectEnumComponent: `V | null`.\n     *\n     * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :\n     *  - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed\n     *  - Dirty status (changed value) means we are in search/autocomplete mode\n     */\n    public readonly internalCtrl = new FormControl<I | null>(null);\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onChange?: (item: V | null) => void;\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onTouched?: () => void;\n\n    public readonly matcher: ExternalFormControlMatcher<V, I>;\n    public readonly ngControl = inject(NgControl, {optional: true, self: true});\n\n    public constructor() {\n        if (this.ngControl) {\n            this.ngControl.valueAccessor = this;\n        }\n\n        this.matcher = new ExternalFormControlMatcher(this);\n    }\n\n    public ngDoCheck(): void {\n        if (this.ngControl) {\n            this.applyRequired();\n        }\n    }\n\n    public writeValue(value: I | null): void {\n        this.internalCtrl.setValue(value);\n    }\n\n    public ngOnInit(): void {\n        const isReactive = this.ngControl instanceof FormControlDirective || this.ngControl instanceof FormControlName;\n        if (isReactive && typeof this._required !== 'undefined') {\n            console.warn('<natural-select-*> should not be used as ReactiveForm and with the [required] attribute');\n        }\n    }\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public registerOnChange(fn: (item: V | null) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public abstract getDisplayFn(): (item: V | null) => string;\n\n    /**\n     * Commit the model to null\n     * Emit and event to update the model\n     */\n    public clear(): void {\n        this.internalCtrl.setValue(null);\n        this.propagateValue(null);\n    }\n\n    /**\n     * If input is dirty (search running) restore to model value\n     */\n    public onBlur(): void {\n        this.touch();\n        this.blur.emit();\n    }\n\n    /**\n     * Commit the model change\n     */\n    public propagateValue(value: V | null): void {\n        // before selectionChange to allow formControl to update before change is effectively emitted\n        if (this.onChange) {\n            this.onChange(value);\n        }\n\n        this.selectionChange.emit(value);\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    public touch(): void {\n        if (this.onTouched) {\n            this.onTouched();\n        }\n    }\n\n    public hasRequiredError(): boolean {\n        const control = this.ngControl?.control ? this.ngControl?.control : this.internalCtrl;\n\n        return control.hasError('required');\n    }\n\n    /**\n     * Apply Validators.required on the internal form, based on ngControl or [required] attribute, giving priority to attribute.\n     */\n    private applyRequired(): void {\n        // Required status on parent validator\n        const outerRequiredStatus = this?.ngControl?.control?.validator?.({} as AbstractControl)?.required;\n\n        // Wanted required status, giving priority to template\n        const newRequiredStatus = typeof this._required !== 'undefined' ? this._required : outerRequiredStatus;\n\n        // Actual internal validation status\n        const currentRequiredStatus = this.internalCtrl?.validator?.({} as AbstractControl)?.required;\n\n        // If wanted status is similar to actual status, stop everything\n        if (currentRequiredStatus === newRequiredStatus) {\n            return;\n        }\n\n        // Apply only if changed\n        if (newRequiredStatus) {\n            this.internalCtrl.setValidators(Validators.required);\n        } else {\n            this.internalCtrl.clearValidators();\n        }\n\n        this.internalCtrl.updateValueAndValidity();\n    }\n}\n"]}
223
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-select.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/select/abstract-select.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {Directive, DoCheck, EventEmitter, inject, Input, OnInit, Output} from '@angular/core';\nimport {\n    AbstractControl,\n    ControlValueAccessor,\n    FormControl,\n    FormControlDirective,\n    FormControlName,\n    NgControl,\n    Validators,\n} from '@angular/forms';\nimport {ErrorStateMatcher} from '@angular/material/core';\n\n/**\n * This will completely ignore internal formControl and instead use the one from the component\n * which comes from outside of this component. This basically allows us to **not** depend on\n * touched status propagation between outside and inside world, and thus get rid of our legacy\n * custom FormControl class (\"NaturalFormControl\").\n */\nclass ExternalFormControlMatcher<TValue, TInput> extends ErrorStateMatcher {\n    public constructor(private readonly component: AbstractSelect<TValue, TInput>) {\n        super();\n    }\n\n    public override isErrorState(): boolean {\n        const externalCtrl = this.component.ngControl?.control || this.component.internalCtrl;\n        if (externalCtrl) {\n            return !!(externalCtrl.errors && (externalCtrl.touched || externalCtrl.dirty));\n        }\n\n        return false;\n    }\n}\n\n@Directive({standalone: true})\nexport abstract class AbstractSelect<TValue, TInput> implements OnInit, ControlValueAccessor, DoCheck {\n    @Input() public placeholder?: string;\n\n    /**\n     * Mat-hint\n     */\n    @Input() public hint: string | null = null;\n\n    /**\n     * If the field is required\n     */\n    @Input()\n    public set required(value: boolean) {\n        this._required = coerceBooleanProperty(value);\n        this.applyRequired();\n    }\n\n    public get required(): boolean {\n        return !!this._required;\n    }\n\n    private _required: boolean | undefined;\n\n    /**\n     * Add a suffix button that is a link to given destination\n     */\n    @Input() public navigateTo?: any[] | string | null;\n\n    /**\n     * If provided cause a new clear button to appear\n     */\n    @Input() public clearLabel?: string;\n\n    /**\n     * Whether to show the search icon\n     */\n    @Input() public showIcon = true;\n\n    /**\n     * Icon name\n     */\n    @Input() public icon = 'search';\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: TValue | null) => string;\n\n    /**\n     * Emit the selected value whenever it changes\n     */\n    @Output() public readonly selectionChange = new EventEmitter<TValue | null>();\n\n    /**\n     * Emits when internal input is blurred\n     */\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() public readonly blur = new EventEmitter<void>();\n\n    /**\n     * Contains internal representation for current selection AND searched text (for autocomplete)\n     *\n     * It is **not** necessarily `TValue | null`.\n     *\n     * - NaturalSelectComponent: `string | TValue | null`. We allow `string`\n     *   only when `optionRequired` is false, so most of the time it is `TValue | null`.\n     * - NaturalSelectHierarchicComponent: `string | null`.\n     * - NaturalSelectEnumComponent: `TValue | null`.\n     *\n     * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :\n     *  - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed\n     *  - Dirty status (changed value) means we are in search/autocomplete mode\n     */\n    public readonly internalCtrl = new FormControl<TInput | null>(null);\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onChange?: (item: TValue | null) => void;\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onTouched?: () => void;\n\n    public readonly matcher: ExternalFormControlMatcher<TValue, TInput>;\n    public readonly ngControl = inject(NgControl, {optional: true, self: true});\n\n    public constructor() {\n        if (this.ngControl) {\n            this.ngControl.valueAccessor = this;\n        }\n\n        this.matcher = new ExternalFormControlMatcher(this);\n    }\n\n    public ngDoCheck(): void {\n        if (this.ngControl) {\n            this.applyRequired();\n        }\n    }\n\n    public writeValue(value: TInput | null): void {\n        this.internalCtrl.setValue(value);\n    }\n\n    public ngOnInit(): void {\n        const isReactive = this.ngControl instanceof FormControlDirective || this.ngControl instanceof FormControlName;\n        if (isReactive && typeof this._required !== 'undefined') {\n            console.warn('<natural-select-*> should not be used as ReactiveForm and with the [required] attribute');\n        }\n    }\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public registerOnChange(fn: (item: TValue | null) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public abstract getDisplayFn(): (item: TValue | null) => string;\n\n    /**\n     * Commit the model to null\n     * Emit and event to update the model\n     */\n    public clear(): void {\n        this.internalCtrl.setValue(null);\n        this.propagateValue(null);\n    }\n\n    /**\n     * If input is dirty (search running) restore to model value\n     */\n    public onBlur(): void {\n        this.touch();\n        this.blur.emit();\n    }\n\n    /**\n     * Commit the model change\n     */\n    public propagateValue(value: TValue | null): void {\n        // before selectionChange to allow formControl to update before change is effectively emitted\n        if (this.onChange) {\n            this.onChange(value);\n        }\n\n        this.selectionChange.emit(value);\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    public touch(): void {\n        if (this.onTouched) {\n            this.onTouched();\n        }\n    }\n\n    public hasRequiredError(): boolean {\n        const control = this.ngControl?.control ? this.ngControl?.control : this.internalCtrl;\n\n        return control.hasError('required');\n    }\n\n    /**\n     * Apply Validators.required on the internal form, based on ngControl or [required] attribute, giving priority to attribute.\n     */\n    private applyRequired(): void {\n        // Required status on parent validator\n        const outerRequiredStatus = this?.ngControl?.control?.validator?.({} as AbstractControl)?.required;\n\n        // Wanted required status, giving priority to template\n        const newRequiredStatus = typeof this._required !== 'undefined' ? this._required : outerRequiredStatus;\n\n        // Actual internal validation status\n        const currentRequiredStatus = this.internalCtrl?.validator?.({} as AbstractControl)?.required;\n\n        // If wanted status is similar to actual status, stop everything\n        if (currentRequiredStatus === newRequiredStatus) {\n            return;\n        }\n\n        // Apply only if changed\n        if (newRequiredStatus) {\n            this.internalCtrl.setValidators(Validators.required);\n        } else {\n            this.internalCtrl.clearValidators();\n        }\n\n        this.internalCtrl.updateValueAndValidity();\n    }\n}\n"]}
@@ -307,4 +307,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
307
307
  }], disabled: [{
308
308
  type: Input
309
309
  }] } });
310
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/select/select/select.component.ts","../../../../../../../projects/natural/src/lib/modules/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    Component,\n    ContentChild,\n    DestroyRef,\n    inject,\n    Input,\n    OnInit,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatAutocompleteModule, MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {merge} from 'lodash-es';\nimport {Observable} from 'rxjs';\nimport {debounceTime, distinctUntilChanged, finalize, map} from 'rxjs/operators';\nimport {PaginatedData} from '../../../classes/data-source';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {NaturalAbstractModelService} from '../../../services/abstract-model.service';\nimport {ExtractTallOne, ExtractVall, Literal} from '../../../types/types';\nimport {AbstractSelect} from '../abstract-select.component';\nimport {RouterLink} from '@angular/router';\nimport {MatTooltipModule} from '@angular/material/tooltip';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatOptionModule} from '@angular/material/core';\nimport {CommonModule} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\ntype V<TService> = string | ExtractTallOne<TService>;\n\n/**\n * Default usage:\n * ```html\n * <natural-select [service]=\"myServiceInstance\" [(model)]=\"myModel\" (modelChange)=myChangeFn($event) />\n * ```\n *\n * Custom template usage :\n * ```html\n * <natural-select [service]=\"svc\" [(ngModel)]=\"model\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-select>\n * ```\n *\n * `[(ngModel)]` and `(ngModelChange)` are optional.\n *\n * Placeholder :\n * ```html\n * <natural-select placeholder=\"my placeholder\" />\n * ```\n *\n * Search with like %xxx% on specified field `name` instead of custom filter on whole object\n * ```html\n * <natural-select searchField=\"name\" />\n * ```\n *\n * Allows to input free string without selecting an option from autocomplete suggestions\n * ```html\n * <natural-select [optionRequired]=\"false\" />\n * ```\n */\n@Component({\n    selector: 'natural-select',\n    templateUrl: './select.component.html',\n    styleUrl: './select.component.scss',\n    standalone: true,\n    imports: [\n        MatAutocompleteModule,\n        CommonModule,\n        MatOptionModule,\n        MatFormFieldModule,\n        MatInputModule,\n        FormsModule,\n        ReactiveFormsModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatProgressSpinnerModule,\n        MatButtonModule,\n        MatTooltipModule,\n        RouterLink,\n    ],\n})\nexport class NaturalSelectComponent<\n        TService extends NaturalAbstractModelService<\n            any,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            any,\n            any\n        >,\n    >\n    extends AbstractSelect<V<TService>, V<TService>>\n    implements OnInit, ControlValueAccessor, AfterViewInit\n{\n    private readonly destroyRef = inject(DestroyRef);\n    @ViewChild(MatAutocompleteTrigger) public autoTrigger!: MatAutocompleteTrigger;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    /**\n     * Service with watchAll function that accepts queryVariables.\n     */\n    @Input({required: true}) public service!: TService;\n\n    /**\n     * If false, allows to input free string without selecting an option from autocomplete suggestions\n     */\n    @Input() public optionRequired = true;\n\n    /**\n     * The field on which to search for, default to 'custom'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchField: 'custom' | string = 'custom';\n\n    /**\n     * The operator with which to search for, default to 'search' if `searchField` is 'custom', else 'like'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchOperator: 'search' | string | null = null;\n\n    /**\n     * Cache the committed value during search mode.\n     * It's used to be restored in case we cancel the selection\n     */\n    private lastValidValue: V<TService> | null = null;\n\n    /**\n     * Additional filter for query\n     */\n    @Input()\n    public set filter(filter: ExtractVall<TService>['filter'] | null | undefined) {\n        this.variablesManager.set('additional-filter', {filter: filter});\n    }\n\n    /**\n     * Items returned by server to show in listing\n     */\n    public items: null | Observable<readonly any[]> = null;\n\n    /**\n     * Whether we are searching something\n     */\n    public loading = false;\n\n    /**\n     * If some items are not shown in result list\n     * Shows a message after list if true\n     */\n    public hasMoreItems = false;\n\n    public nbTotal = 0;\n\n    /**\n     * Default page size\n     */\n    private pageSize = 10;\n\n    /**\n     * Init search options\n     */\n    private readonly variablesManager = new NaturalQueryVariablesManager<QueryVariables>();\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public override set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.initService();\n    }\n\n    public ngAfterViewInit(): void {\n        this.internalCtrl.valueChanges\n            .pipe(takeUntilDestroyed(this.destroyRef), distinctUntilChanged(), debounceTime(300))\n            .subscribe(val => this.search(val));\n    }\n\n    public onInternalFormChange(): void {\n        // If we allow free string typing, then we propagate it as it is being typed\n        if (!this.optionRequired) {\n            this.propagateValue(this.internalCtrl.value);\n        }\n    }\n\n    public override onBlur(): void {\n        if (this.internalCtrl.dirty) {\n            this.reset();\n        }\n        super.onBlur();\n    }\n\n    /**\n     * Reset form to it's initial value\n     * Discard searched text (in autocomplete use case)\n     * Doest not commit the change to the model (no change event is emitted)\n     */\n    public reset(): void {\n        this.internalCtrl.setValue(this.lastValidValue);\n        this.internalCtrl.markAsPristine();\n    }\n\n    /**\n     * Enter semantic means we want to validate something.\n     * If we hit ENTER while typing a text, the stroke is ignored because the value is invalid (it's accepted in free text mode)\n     * If we hit ENTER while the input field is empty, we validate the unselection (empty is a valid value)\n     */\n    public onKeyEnter(): void {\n        if (!this.internalCtrl.value) {\n            this.clear();\n            this.autoTrigger.closePanel();\n        }\n    }\n\n    public override writeValue(value: V<TService> | null): void {\n        super.writeValue(value);\n        this.lastValidValue = this.internalCtrl.value;\n    }\n\n    private initService(): void {\n        // Assert given service has a watchAll function\n        if (typeof this.service.watchAll !== 'function') {\n            throw new TypeError('Provided service does not contain watchAll function');\n        }\n\n        const defaultPagination = {\n            pagination: {\n                pageIndex: 0,\n                pageSize: this.pageSize,\n            },\n        };\n\n        const variables = merge(defaultPagination, this.getSearchFilter(null));\n        this.variablesManager.set('variables', variables);\n    }\n\n    public startSearch(): void {\n        // Start search only once\n        if (this.items) {\n            return;\n        }\n\n        // Init query, and when query results arrive, finish loading, and count items\n        this.items = this.service.watchAll(this.variablesManager).pipe(\n            takeUntilDestroyed(this.destroyRef),\n            finalize(() => (this.loading = false)),\n            map(data => {\n                this.loading = false;\n                this.nbTotal = data.length;\n                const nbListed = Math.min(data.length, this.pageSize);\n                this.hasMoreItems = this.nbTotal > nbListed;\n\n                return data.items;\n            }),\n        );\n\n        this.loading = true;\n        this.items.subscribe();\n    }\n\n    /**\n     * Commit the model change\n     * Set internal form as pristine to reflect that the visible value match the model\n     */\n    public override propagateValue(value: V<TService> | null): void {\n        this.internalCtrl.markAsPristine();\n        this.lastValidValue = this.internalCtrl.value;\n        this.loading = false;\n\n        // If we cleared value via button, but we allow free string typing, then force to empty string\n        if (!this.optionRequired && value === null) {\n            value = '';\n        }\n\n        super.propagateValue(value);\n    }\n\n    /**\n     * Very important to return something, above all if [select]='displayedValue' attribute value is used\n     */\n    public getDisplayFn(): (item: V<TService> | null) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return (item: any) => {\n            if (!item) {\n                return '';\n            }\n\n            if (typeof item === 'string') {\n                return item;\n            }\n\n            return item.fullName || item.name || item.iban || item[this.searchField] || item.id;\n        };\n    }\n\n    public override clear(): void {\n        this.search(null);\n        super.clear();\n    }\n\n    public search(term: V<TService> | null): void {\n        if (typeof term === 'string' || term === null) {\n            if (term) {\n                this.loading = !!this.items;\n            }\n\n            this.variablesManager.merge('variables', this.getSearchFilter(term));\n        }\n    }\n\n    public override showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    private getSearchFilter(term: string | null): QueryVariables {\n        const searchOperator = this.searchOperator ?? (this.searchField === 'custom' ? 'search' : 'like');\n        if (term && searchOperator === 'like') {\n            term = '%' + term + '%';\n        }\n\n        return {\n            filter: {\n                groups: [\n                    {\n                        conditions: [\n                            {\n                                [this.searchField]: term\n                                    ? {\n                                          [searchOperator]: {value: term},\n                                      }\n                                    : null,\n                            },\n                        ],\n                    },\n                ],\n            },\n        };\n    }\n\n    public getVariablesForDebug(): Readonly<QueryVariables> | undefined {\n        return this.variablesManager.variables.value;\n    }\n}\n","<!-- Autocomplete menu -->\n<mat-autocomplete\n    #ac=\"matAutocomplete\"\n    (optionSelected)=\"propagateValue($event.option.value)\"\n    [displayWith]=\"getDisplayFn()\"\n    panelWidth=\"auto !important\"\n>\n    @for (item of items | async; track $index) {\n        <mat-option [value]=\"item\">\n            <ng-template\n                [ngTemplateOutletContext]=\"{item: item}\"\n                [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n            />\n        </mat-option>\n    }\n    @if (hasMoreItems) {\n        <div class=\"mat-caption\" i18n style=\"padding: 5px 10px\">Saisir pour chercher parmi {{ nbTotal }} résultats</div>\n    }\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n    <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n    <mat-label>{{ placeholder }}</mat-label>\n\n    <input\n        (blur)=\"onBlur()\"\n        (change)=\"onInternalFormChange()\"\n        (click)=\"autoTrigger.openPanel()\"\n        (focus)=\"startSearch()\"\n        (keydown.esc)=\"reset()\"\n        (keydown.enter)=\"onKeyEnter()\"\n        [formControl]=\"internalCtrl\"\n        [matAutocomplete]=\"ac\"\n        aria-label=\"Recherche et sélection\"\n        i18n-aria-label\n        matInput\n        [errorStateMatcher]=\"matcher\"\n    />\n\n    @if (hint) {\n        <mat-hint>{{ hint }}</mat-hint>\n    }\n\n    <!-- Meta data -->\n    @if (!loading && showIcon) {\n        <mat-icon [naturalIcon]=\"icon\" matIconPrefix />\n    }\n\n    @if (loading) {\n        <div class=\"loading-wrapper\" matIconPrefix>\n            <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n        </div>\n    }\n\n    <!-- Clear button -->\n    <div matIconSuffix>\n        @if (internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel) {\n            <button (click)=\"clear()\" mat-icon-button i18n-matTooltip matTooltip=\"Désélectionner\">\n                <mat-icon naturalIcon=\"close\" />\n            </button>\n        }\n        @if (internalCtrl.dirty && internalCtrl.enabled && optionRequired) {\n            <button (click)=\"reset()\" mat-icon-button i18n-matTooltip matTooltip=\"Annuler la recherche\">\n                <mat-icon naturalIcon=\"undo\" />\n            </button>\n        }\n        @if (internalCtrl.pristine && internalCtrl.value && navigateTo) {\n            <button\n                [routerLink]=\"navigateTo\"\n                (click)=\"$event.stopPropagation()\"\n                mat-icon-button\n                i18n-matTooltip\n                matTooltip=\"Naviguer vers\"\n            >\n                <mat-icon naturalIcon=\"open_in_browser\" />\n            </button>\n        }\n    </div>\n\n    @if (hasRequiredError()) {\n        <mat-error i18n>Ce champ est requis</mat-error>\n    }\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n@if (showClearButton()) {\n    <div class=\"external-buttons\">\n        @if (showClearButton()) {\n            <button (click)=\"clear()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n        }\n    </div>\n}\n"]}
310
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/select/select/select.component.ts","../../../../../../../projects/natural/src/lib/modules/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    Component,\n    ContentChild,\n    DestroyRef,\n    inject,\n    Input,\n    OnInit,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatAutocompleteModule, MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {merge} from 'lodash-es';\nimport {Observable} from 'rxjs';\nimport {debounceTime, distinctUntilChanged, finalize, map} from 'rxjs/operators';\nimport {PaginatedData} from '../../../classes/data-source';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {NaturalAbstractModelService} from '../../../services/abstract-model.service';\nimport {ExtractTallOne, ExtractVall, Literal} from '../../../types/types';\nimport {AbstractSelect} from '../abstract-select.component';\nimport {RouterLink} from '@angular/router';\nimport {MatTooltipModule} from '@angular/material/tooltip';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatOptionModule} from '@angular/material/core';\nimport {CommonModule} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n// Return the type of TValue for the given TService\ntype ValueTypeFor<TService> = string | ExtractTallOne<TService>;\n\n/**\n * Default usage:\n * ```html\n * <natural-select [service]=\"myServiceInstance\" [(model)]=\"myModel\" (modelChange)=myChangeFn($event) />\n * ```\n *\n * Custom template usage :\n * ```html\n * <natural-select [service]=\"svc\" [(ngModel)]=\"model\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-select>\n * ```\n *\n * `[(ngModel)]` and `(ngModelChange)` are optional.\n *\n * Placeholder :\n * ```html\n * <natural-select placeholder=\"my placeholder\" />\n * ```\n *\n * Search with like %xxx% on specified field `name` instead of custom filter on whole object\n * ```html\n * <natural-select searchField=\"name\" />\n * ```\n *\n * Allows to input free string without selecting an option from autocomplete suggestions\n * ```html\n * <natural-select [optionRequired]=\"false\" />\n * ```\n */\n@Component({\n    selector: 'natural-select',\n    templateUrl: './select.component.html',\n    styleUrl: './select.component.scss',\n    standalone: true,\n    imports: [\n        MatAutocompleteModule,\n        CommonModule,\n        MatOptionModule,\n        MatFormFieldModule,\n        MatInputModule,\n        FormsModule,\n        ReactiveFormsModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatProgressSpinnerModule,\n        MatButtonModule,\n        MatTooltipModule,\n        RouterLink,\n    ],\n})\nexport class NaturalSelectComponent<\n        TService extends NaturalAbstractModelService<\n            any,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            any,\n            any\n        >,\n    >\n    extends AbstractSelect<ValueTypeFor<TService>, ValueTypeFor<TService>>\n    implements OnInit, ControlValueAccessor, AfterViewInit\n{\n    private readonly destroyRef = inject(DestroyRef);\n    @ViewChild(MatAutocompleteTrigger) public autoTrigger!: MatAutocompleteTrigger;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    /**\n     * Service with watchAll function that accepts queryVariables.\n     */\n    @Input({required: true}) public service!: TService;\n\n    /**\n     * If false, allows to input free string without selecting an option from autocomplete suggestions\n     */\n    @Input() public optionRequired = true;\n\n    /**\n     * The field on which to search for, default to 'custom'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchField: 'custom' | string = 'custom';\n\n    /**\n     * The operator with which to search for, default to 'search' if `searchField` is 'custom', else 'like'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchOperator: 'search' | string | null = null;\n\n    /**\n     * Cache the committed value during search mode.\n     * It's used to be restored in case we cancel the selection\n     */\n    private lastValidValue: ValueTypeFor<TService> | null = null;\n\n    /**\n     * Additional filter for query\n     */\n    @Input()\n    public set filter(filter: ExtractVall<TService>['filter'] | null | undefined) {\n        this.variablesManager.set('additional-filter', {filter: filter});\n    }\n\n    /**\n     * Items returned by server to show in listing\n     */\n    public items: null | Observable<readonly any[]> = null;\n\n    /**\n     * Whether we are searching something\n     */\n    public loading = false;\n\n    /**\n     * If some items are not shown in result list\n     * Shows a message after list if true\n     */\n    public hasMoreItems = false;\n\n    public nbTotal = 0;\n\n    /**\n     * Default page size\n     */\n    private pageSize = 10;\n\n    /**\n     * Init search options\n     */\n    private readonly variablesManager = new NaturalQueryVariablesManager<QueryVariables>();\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public override set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.initService();\n    }\n\n    public ngAfterViewInit(): void {\n        this.internalCtrl.valueChanges\n            .pipe(takeUntilDestroyed(this.destroyRef), distinctUntilChanged(), debounceTime(300))\n            .subscribe(val => this.search(val));\n    }\n\n    public onInternalFormChange(): void {\n        // If we allow free string typing, then we propagate it as it is being typed\n        if (!this.optionRequired) {\n            this.propagateValue(this.internalCtrl.value);\n        }\n    }\n\n    public override onBlur(): void {\n        if (this.internalCtrl.dirty) {\n            this.reset();\n        }\n        super.onBlur();\n    }\n\n    /**\n     * Reset form to it's initial value\n     * Discard searched text (in autocomplete use case)\n     * Doest not commit the change to the model (no change event is emitted)\n     */\n    public reset(): void {\n        this.internalCtrl.setValue(this.lastValidValue);\n        this.internalCtrl.markAsPristine();\n    }\n\n    /**\n     * Enter semantic means we want to validate something.\n     * If we hit ENTER while typing a text, the stroke is ignored because the value is invalid (it's accepted in free text mode)\n     * If we hit ENTER while the input field is empty, we validate the unselection (empty is a valid value)\n     */\n    public onKeyEnter(): void {\n        if (!this.internalCtrl.value) {\n            this.clear();\n            this.autoTrigger.closePanel();\n        }\n    }\n\n    public override writeValue(value: ValueTypeFor<TService> | null): void {\n        super.writeValue(value);\n        this.lastValidValue = this.internalCtrl.value;\n    }\n\n    private initService(): void {\n        // Assert given service has a watchAll function\n        if (typeof this.service.watchAll !== 'function') {\n            throw new TypeError('Provided service does not contain watchAll function');\n        }\n\n        const defaultPagination = {\n            pagination: {\n                pageIndex: 0,\n                pageSize: this.pageSize,\n            },\n        };\n\n        const variables = merge(defaultPagination, this.getSearchFilter(null));\n        this.variablesManager.set('variables', variables);\n    }\n\n    public startSearch(): void {\n        // Start search only once\n        if (this.items) {\n            return;\n        }\n\n        // Init query, and when query results arrive, finish loading, and count items\n        this.items = this.service.watchAll(this.variablesManager).pipe(\n            takeUntilDestroyed(this.destroyRef),\n            finalize(() => (this.loading = false)),\n            map(data => {\n                this.loading = false;\n                this.nbTotal = data.length;\n                const nbListed = Math.min(data.length, this.pageSize);\n                this.hasMoreItems = this.nbTotal > nbListed;\n\n                return data.items;\n            }),\n        );\n\n        this.loading = true;\n        this.items.subscribe();\n    }\n\n    /**\n     * Commit the model change\n     * Set internal form as pristine to reflect that the visible value match the model\n     */\n    public override propagateValue(value: ValueTypeFor<TService> | null): void {\n        this.internalCtrl.markAsPristine();\n        this.lastValidValue = this.internalCtrl.value;\n        this.loading = false;\n\n        // If we cleared value via button, but we allow free string typing, then force to empty string\n        if (!this.optionRequired && value === null) {\n            value = '';\n        }\n\n        super.propagateValue(value);\n    }\n\n    /**\n     * Very important to return something, above all if [select]='displayedValue' attribute value is used\n     */\n    public getDisplayFn(): (item: ValueTypeFor<TService> | null) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return (item: any) => {\n            if (!item) {\n                return '';\n            }\n\n            if (typeof item === 'string') {\n                return item;\n            }\n\n            return item.fullName || item.name || item.iban || item[this.searchField] || item.id;\n        };\n    }\n\n    public override clear(): void {\n        this.search(null);\n        super.clear();\n    }\n\n    public search(term: ValueTypeFor<TService> | null): void {\n        if (typeof term === 'string' || term === null) {\n            if (term) {\n                this.loading = !!this.items;\n            }\n\n            this.variablesManager.merge('variables', this.getSearchFilter(term));\n        }\n    }\n\n    public override showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    private getSearchFilter(term: string | null): QueryVariables {\n        const searchOperator = this.searchOperator ?? (this.searchField === 'custom' ? 'search' : 'like');\n        if (term && searchOperator === 'like') {\n            term = '%' + term + '%';\n        }\n\n        return {\n            filter: {\n                groups: [\n                    {\n                        conditions: [\n                            {\n                                [this.searchField]: term\n                                    ? {\n                                          [searchOperator]: {value: term},\n                                      }\n                                    : null,\n                            },\n                        ],\n                    },\n                ],\n            },\n        };\n    }\n\n    public getVariablesForDebug(): Readonly<QueryVariables> | undefined {\n        return this.variablesManager.variables.value;\n    }\n}\n","<!-- Autocomplete menu -->\n<mat-autocomplete\n    #ac=\"matAutocomplete\"\n    (optionSelected)=\"propagateValue($event.option.value)\"\n    [displayWith]=\"getDisplayFn()\"\n    panelWidth=\"auto !important\"\n>\n    @for (item of items | async; track $index) {\n        <mat-option [value]=\"item\">\n            <ng-template\n                [ngTemplateOutletContext]=\"{item: item}\"\n                [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n            />\n        </mat-option>\n    }\n    @if (hasMoreItems) {\n        <div class=\"mat-caption\" i18n style=\"padding: 5px 10px\">Saisir pour chercher parmi {{ nbTotal }} résultats</div>\n    }\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n    <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n    <mat-label>{{ placeholder }}</mat-label>\n\n    <input\n        (blur)=\"onBlur()\"\n        (change)=\"onInternalFormChange()\"\n        (click)=\"autoTrigger.openPanel()\"\n        (focus)=\"startSearch()\"\n        (keydown.esc)=\"reset()\"\n        (keydown.enter)=\"onKeyEnter()\"\n        [formControl]=\"internalCtrl\"\n        [matAutocomplete]=\"ac\"\n        aria-label=\"Recherche et sélection\"\n        i18n-aria-label\n        matInput\n        [errorStateMatcher]=\"matcher\"\n    />\n\n    @if (hint) {\n        <mat-hint>{{ hint }}</mat-hint>\n    }\n\n    <!-- Meta data -->\n    @if (!loading && showIcon) {\n        <mat-icon [naturalIcon]=\"icon\" matIconPrefix />\n    }\n\n    @if (loading) {\n        <div class=\"loading-wrapper\" matIconPrefix>\n            <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n        </div>\n    }\n\n    <!-- Clear button -->\n    <div matIconSuffix>\n        @if (internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel) {\n            <button (click)=\"clear()\" mat-icon-button i18n-matTooltip matTooltip=\"Désélectionner\">\n                <mat-icon naturalIcon=\"close\" />\n            </button>\n        }\n        @if (internalCtrl.dirty && internalCtrl.enabled && optionRequired) {\n            <button (click)=\"reset()\" mat-icon-button i18n-matTooltip matTooltip=\"Annuler la recherche\">\n                <mat-icon naturalIcon=\"undo\" />\n            </button>\n        }\n        @if (internalCtrl.pristine && internalCtrl.value && navigateTo) {\n            <button\n                [routerLink]=\"navigateTo\"\n                (click)=\"$event.stopPropagation()\"\n                mat-icon-button\n                i18n-matTooltip\n                matTooltip=\"Naviguer vers\"\n            >\n                <mat-icon naturalIcon=\"open_in_browser\" />\n            </button>\n        }\n    </div>\n\n    @if (hasRequiredError()) {\n        <mat-error i18n>Ce champ est requis</mat-error>\n    }\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n@if (showClearButton()) {\n    <div class=\"external-buttons\">\n        @if (showClearButton()) {\n            <button (click)=\"clear()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n        }\n    </div>\n}\n"]}
@@ -54,4 +54,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
54
54
  }], multiple: [{
55
55
  type: Input
56
56
  }] } });
57
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZWN0LWVudW0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvc2VsZWN0L3NlbGVjdC1lbnVtL3NlbGVjdC1lbnVtLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL3NlbGVjdC9zZWxlY3QtZW51bS9zZWxlY3QtZW51bS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQVMsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUF1QixXQUFXLEVBQUUsbUJBQW1CLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUV0RixPQUFPLEVBQVEsa0JBQWtCLEVBQUMsTUFBTSxnQ0FBZ0MsQ0FBQztBQUN6RSxPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sOEJBQThCLENBQUM7QUFDNUQsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQ3ZELE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsZUFBZSxFQUFDLE1BQU0sMEJBQTBCLENBQUM7QUFDekQsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0sOEJBQThCLENBQUM7Ozs7Ozs7QUFXaEUsTUFBTSxPQUFPLDBCQUEyQixTQUFRLGNBQW9CO0lBQy9DLFdBQVcsR0FBRyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztJQUUxRDs7T0FFRztJQUM2QixRQUFRLENBQVU7SUFFbEQ7O09BRUc7SUFDYSxTQUFTLENBQVU7SUFFbkM7O09BRUc7SUFDYSxjQUFjLENBQTRCO0lBRTFEOztPQUVHO0lBQ2EsUUFBUSxHQUFHLEtBQUssQ0FBQztJQUUxQixLQUFLLENBQXVCO0lBRW5CLFFBQVE7UUFDcEIsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3JELENBQUM7SUFFTSxZQUFZO1FBQ2YsTUFBTSxJQUFJLEtBQUssQ0FBQyw2QkFBNkIsQ0FBQyxDQUFDO0lBQ25ELENBQUM7dUdBaENRLDBCQUEwQjsyRkFBMUIsMEJBQTBCLHdOQ25CdkMsNjBCQTJCQSxvR0RWYyxrQkFBa0IsK2RBQUUsZUFBZSxtckJBQUUsV0FBVyxzSUFBRSxtQkFBbUIsaU5BQUUsWUFBWSxtRkFBRSxlQUFlOzsyRkFFckcsMEJBQTBCO2tCQVB0QyxTQUFTOytCQUNJLHFCQUFxQixjQUduQixJQUFJLFdBQ1AsQ0FBQyxrQkFBa0IsRUFBRSxlQUFlLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLFlBQVksRUFBRSxlQUFlLENBQUM7OEJBUS9FLFFBQVE7c0JBQXZDLEtBQUs7dUJBQUMsRUFBQyxRQUFRLEVBQUUsSUFBSSxFQUFDO2dCQUtQLFNBQVM7c0JBQXhCLEtBQUs7Z0JBS1UsY0FBYztzQkFBN0IsS0FBSztnQkFLVSxRQUFRO3NCQUF2QixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtDb21wb25lbnQsIGluamVjdCwgSW5wdXQsIE9uSW5pdH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0NvbnRyb2xWYWx1ZUFjY2Vzc29yLCBGb3Jtc01vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHtPYnNlcnZhYmxlfSBmcm9tICdyeGpzJztcbmltcG9ydCB7SUVudW0sIE5hdHVyYWxFbnVtU2VydmljZX0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvZW51bS5zZXJ2aWNlJztcbmltcG9ydCB7QWJzdHJhY3RTZWxlY3R9IGZyb20gJy4uL2Fic3RyYWN0LXNlbGVjdC5jb21wb25lbnQnO1xuaW1wb3J0IHtNYXRPcHRpb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHtDb21tb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQge01hdFNlbGVjdE1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvc2VsZWN0JztcbmltcG9ydCB7TWF0Rm9ybUZpZWxkTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9mb3JtLWZpZWxkJztcblxudHlwZSBWID0gSUVudW1bJ3ZhbHVlJ10gfCBJRW51bVsndmFsdWUnXVtdO1xuXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ25hdHVyYWwtc2VsZWN0LWVudW0nLFxuICAgIHRlbXBsYXRlVXJsOiAnLi9zZWxlY3QtZW51bS5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmw6ICcuL3NlbGVjdC1lbnVtLmNvbXBvbmVudC5zY3NzJyxcbiAgICBzdGFuZGFsb25lOiB0cnVlLFxuICAgIGltcG9ydHM6IFtNYXRGb3JtRmllbGRNb2R1bGUsIE1hdFNlbGVjdE1vZHVsZSwgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUsIENvbW1vbk1vZHVsZSwgTWF0T3B0aW9uTW9kdWxlXSxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbFNlbGVjdEVudW1Db21wb25lbnQgZXh0ZW5kcyBBYnN0cmFjdFNlbGVjdDxWLCBWPiBpbXBsZW1lbnRzIE9uSW5pdCwgQ29udHJvbFZhbHVlQWNjZXNzb3Ige1xuICAgIHByaXZhdGUgcmVhZG9ubHkgZW51bVNlcnZpY2UgPSBpbmplY3QoTmF0dXJhbEVudW1TZXJ2aWNlKTtcblxuICAgIC8qKlxuICAgICAqIFRoZSBuYW1lIG9mIHRoZSBlbnVtIHR5cGUsIGVnOiBgXCJBY3Rpb25TdGF0dXNcImBcbiAgICAgKi9cbiAgICBASW5wdXQoe3JlcXVpcmVkOiB0cnVlfSkgcHVibGljIGVudW1OYW1lITogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogSWYgZ2l2ZW4gYW4gZXh0cmEgb3B0aW9uIGlzIGFkZGVkIHRvIHNlbGVjdCBgbnVsbGAgd2l0aCBnaXZlbiBsYWJlbFxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBudWxsTGFiZWw/OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBGdW5jdGlvbnMgdGhhdCByZWNlaXZlcyBhbiBlbnVtIHZhbHVlIGFuZCByZXR1cm5zIHdoZXRoZXIgdGhhdCB2YWx1ZSBpcyBkaXNhYmxlZFxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBvcHRpb25EaXNhYmxlZD86IChpdGVtOiBJRW51bSkgPT4gYm9vbGVhbjtcblxuICAgIC8qKlxuICAgICAqIFdoZXRoZXIgdGhlIHVzZXIgc2hvdWxkIGJlIGFsbG93ZWQgdG8gc2VsZWN0IG11bHRpcGxlIG9wdGlvbnNcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbXVsdGlwbGUgPSBmYWxzZTtcblxuICAgIHB1YmxpYyBpdGVtcz86IE9ic2VydmFibGU8SUVudW1bXT47XG5cbiAgICBwdWJsaWMgb3ZlcnJpZGUgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgICAgIHN1cGVyLm5nT25Jbml0KCk7XG4gICAgICAgIHRoaXMuaXRlbXMgPSB0aGlzLmVudW1TZXJ2aWNlLmdldCh0aGlzLmVudW1OYW1lKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0RGlzcGxheUZuKCk6IChpdGVtOiBWIHwgbnVsbCkgPT4gc3RyaW5nIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdUaGlzIHNob3VsZCBuZXZlciBiZSBjYWxsZWQnKTtcbiAgICB9XG59XG4iLCI8bWF0LWZvcm0tZmllbGQ+XG4gICAgPG1hdC1sYWJlbD57eyBwbGFjZWhvbGRlciB9fTwvbWF0LWxhYmVsPlxuICAgIDxtYXQtc2VsZWN0XG4gICAgICAgIChzZWxlY3Rpb25DaGFuZ2UpPVwicHJvcGFnYXRlVmFsdWUoJGV2ZW50LnZhbHVlKVwiXG4gICAgICAgIFtmb3JtQ29udHJvbF09XCJpbnRlcm5hbEN0cmxcIlxuICAgICAgICAoYmx1cik9XCJvbkJsdXIoKVwiXG4gICAgICAgIFtlcnJvclN0YXRlTWF0Y2hlcl09XCJtYXRjaGVyXCJcbiAgICAgICAgW211bHRpcGxlXT1cIm11bHRpcGxlXCJcbiAgICA+XG4gICAgICAgIEBpZiAobnVsbExhYmVsKSB7XG4gICAgICAgICAgICA8bWF0LW9wdGlvbiBbdmFsdWVdPVwibnVsbFwiPnt7IG51bGxMYWJlbCB9fTwvbWF0LW9wdGlvbj5cbiAgICAgICAgfVxuICAgICAgICBAZm9yIChpdGVtIG9mIGl0ZW1zIHwgYXN5bmM7IHRyYWNrIGl0ZW0udmFsdWUpIHtcbiAgICAgICAgICAgIDxtYXQtb3B0aW9uIFt2YWx1ZV09XCJpdGVtLnZhbHVlXCIgW2Rpc2FibGVkXT1cIm9wdGlvbkRpc2FibGVkID8gb3B0aW9uRGlzYWJsZWQoaXRlbSkgOiBmYWxzZVwiPlxuICAgICAgICAgICAgICAgIHt7IGl0ZW0ubmFtZSB9fVxuICAgICAgICAgICAgPC9tYXQtb3B0aW9uPlxuICAgICAgICB9XG4gICAgPC9tYXQtc2VsZWN0PlxuXG4gICAgQGlmIChoaW50KSB7XG4gICAgICAgIDxtYXQtaGludD57eyBoaW50IH19PC9tYXQtaGludD5cbiAgICB9XG5cbiAgICBAaWYgKGhhc1JlcXVpcmVkRXJyb3IoKSkge1xuICAgICAgICA8bWF0LWVycm9yIGkxOG4+Q2UgY2hhbXAgZXN0IHJlcXVpczwvbWF0LWVycm9yPlxuICAgIH1cbjwvbWF0LWZvcm0tZmllbGQ+XG4iXX0=
57
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZWN0LWVudW0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvc2VsZWN0L3NlbGVjdC1lbnVtL3NlbGVjdC1lbnVtLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL3NlbGVjdC9zZWxlY3QtZW51bS9zZWxlY3QtZW51bS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQVMsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUF1QixXQUFXLEVBQUUsbUJBQW1CLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUV0RixPQUFPLEVBQVEsa0JBQWtCLEVBQUMsTUFBTSxnQ0FBZ0MsQ0FBQztBQUN6RSxPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sOEJBQThCLENBQUM7QUFDNUQsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQ3ZELE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsZUFBZSxFQUFDLE1BQU0sMEJBQTBCLENBQUM7QUFDekQsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0sOEJBQThCLENBQUM7Ozs7Ozs7QUFXaEUsTUFBTSxPQUFPLDBCQUEyQixTQUFRLGNBQTRCO0lBQ3ZELFdBQVcsR0FBRyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztJQUUxRDs7T0FFRztJQUM2QixRQUFRLENBQVU7SUFFbEQ7O09BRUc7SUFDYSxTQUFTLENBQVU7SUFFbkM7O09BRUc7SUFDYSxjQUFjLENBQTRCO0lBRTFEOztPQUVHO0lBQ2EsUUFBUSxHQUFHLEtBQUssQ0FBQztJQUUxQixLQUFLLENBQXVCO0lBRW5CLFFBQVE7UUFDcEIsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3JELENBQUM7SUFFTSxZQUFZO1FBQ2YsTUFBTSxJQUFJLEtBQUssQ0FBQyw2QkFBNkIsQ0FBQyxDQUFDO0lBQ25ELENBQUM7dUdBaENRLDBCQUEwQjsyRkFBMUIsMEJBQTBCLHdOQ25CdkMsNjBCQTJCQSxvR0RWYyxrQkFBa0IsK2RBQUUsZUFBZSxtckJBQUUsV0FBVyxzSUFBRSxtQkFBbUIsaU5BQUUsWUFBWSxtRkFBRSxlQUFlOzsyRkFFckcsMEJBQTBCO2tCQVB0QyxTQUFTOytCQUNJLHFCQUFxQixjQUduQixJQUFJLFdBQ1AsQ0FBQyxrQkFBa0IsRUFBRSxlQUFlLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLFlBQVksRUFBRSxlQUFlLENBQUM7OEJBUS9FLFFBQVE7c0JBQXZDLEtBQUs7dUJBQUMsRUFBQyxRQUFRLEVBQUUsSUFBSSxFQUFDO2dCQUtQLFNBQVM7c0JBQXhCLEtBQUs7Z0JBS1UsY0FBYztzQkFBN0IsS0FBSztnQkFLVSxRQUFRO3NCQUF2QixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtDb21wb25lbnQsIGluamVjdCwgSW5wdXQsIE9uSW5pdH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0NvbnRyb2xWYWx1ZUFjY2Vzc29yLCBGb3Jtc01vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHtPYnNlcnZhYmxlfSBmcm9tICdyeGpzJztcbmltcG9ydCB7SUVudW0sIE5hdHVyYWxFbnVtU2VydmljZX0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvZW51bS5zZXJ2aWNlJztcbmltcG9ydCB7QWJzdHJhY3RTZWxlY3R9IGZyb20gJy4uL2Fic3RyYWN0LXNlbGVjdC5jb21wb25lbnQnO1xuaW1wb3J0IHtNYXRPcHRpb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHtDb21tb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQge01hdFNlbGVjdE1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvc2VsZWN0JztcbmltcG9ydCB7TWF0Rm9ybUZpZWxkTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9mb3JtLWZpZWxkJztcblxudHlwZSBWYWx1ZSA9IElFbnVtWyd2YWx1ZSddIHwgSUVudW1bJ3ZhbHVlJ11bXTtcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICduYXR1cmFsLXNlbGVjdC1lbnVtJyxcbiAgICB0ZW1wbGF0ZVVybDogJy4vc2VsZWN0LWVudW0uY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsOiAnLi9zZWxlY3QtZW51bS5jb21wb25lbnQuc2NzcycsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbTWF0Rm9ybUZpZWxkTW9kdWxlLCBNYXRTZWxlY3RNb2R1bGUsIEZvcm1zTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlLCBDb21tb25Nb2R1bGUsIE1hdE9wdGlvbk1vZHVsZV0sXG59KVxuZXhwb3J0IGNsYXNzIE5hdHVyYWxTZWxlY3RFbnVtQ29tcG9uZW50IGV4dGVuZHMgQWJzdHJhY3RTZWxlY3Q8VmFsdWUsIFZhbHVlPiBpbXBsZW1lbnRzIE9uSW5pdCwgQ29udHJvbFZhbHVlQWNjZXNzb3Ige1xuICAgIHByaXZhdGUgcmVhZG9ubHkgZW51bVNlcnZpY2UgPSBpbmplY3QoTmF0dXJhbEVudW1TZXJ2aWNlKTtcblxuICAgIC8qKlxuICAgICAqIFRoZSBuYW1lIG9mIHRoZSBlbnVtIHR5cGUsIGVnOiBgXCJBY3Rpb25TdGF0dXNcImBcbiAgICAgKi9cbiAgICBASW5wdXQoe3JlcXVpcmVkOiB0cnVlfSkgcHVibGljIGVudW1OYW1lITogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogSWYgZ2l2ZW4gYW4gZXh0cmEgb3B0aW9uIGlzIGFkZGVkIHRvIHNlbGVjdCBgbnVsbGAgd2l0aCBnaXZlbiBsYWJlbFxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBudWxsTGFiZWw/OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBGdW5jdGlvbnMgdGhhdCByZWNlaXZlcyBhbiBlbnVtIHZhbHVlIGFuZCByZXR1cm5zIHdoZXRoZXIgdGhhdCB2YWx1ZSBpcyBkaXNhYmxlZFxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBvcHRpb25EaXNhYmxlZD86IChpdGVtOiBJRW51bSkgPT4gYm9vbGVhbjtcblxuICAgIC8qKlxuICAgICAqIFdoZXRoZXIgdGhlIHVzZXIgc2hvdWxkIGJlIGFsbG93ZWQgdG8gc2VsZWN0IG11bHRpcGxlIG9wdGlvbnNcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbXVsdGlwbGUgPSBmYWxzZTtcblxuICAgIHB1YmxpYyBpdGVtcz86IE9ic2VydmFibGU8SUVudW1bXT47XG5cbiAgICBwdWJsaWMgb3ZlcnJpZGUgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgICAgIHN1cGVyLm5nT25Jbml0KCk7XG4gICAgICAgIHRoaXMuaXRlbXMgPSB0aGlzLmVudW1TZXJ2aWNlLmdldCh0aGlzLmVudW1OYW1lKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0RGlzcGxheUZuKCk6IChpdGVtOiBWYWx1ZSB8IG51bGwpID0+IHN0cmluZyB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignVGhpcyBzaG91bGQgbmV2ZXIgYmUgY2FsbGVkJyk7XG4gICAgfVxufVxuIiwiPG1hdC1mb3JtLWZpZWxkPlxuICAgIDxtYXQtbGFiZWw+e3sgcGxhY2Vob2xkZXIgfX08L21hdC1sYWJlbD5cbiAgICA8bWF0LXNlbGVjdFxuICAgICAgICAoc2VsZWN0aW9uQ2hhbmdlKT1cInByb3BhZ2F0ZVZhbHVlKCRldmVudC52YWx1ZSlcIlxuICAgICAgICBbZm9ybUNvbnRyb2xdPVwiaW50ZXJuYWxDdHJsXCJcbiAgICAgICAgKGJsdXIpPVwib25CbHVyKClcIlxuICAgICAgICBbZXJyb3JTdGF0ZU1hdGNoZXJdPVwibWF0Y2hlclwiXG4gICAgICAgIFttdWx0aXBsZV09XCJtdWx0aXBsZVwiXG4gICAgPlxuICAgICAgICBAaWYgKG51bGxMYWJlbCkge1xuICAgICAgICAgICAgPG1hdC1vcHRpb24gW3ZhbHVlXT1cIm51bGxcIj57eyBudWxsTGFiZWwgfX08L21hdC1vcHRpb24+XG4gICAgICAgIH1cbiAgICAgICAgQGZvciAoaXRlbSBvZiBpdGVtcyB8IGFzeW5jOyB0cmFjayBpdGVtLnZhbHVlKSB7XG4gICAgICAgICAgICA8bWF0LW9wdGlvbiBbdmFsdWVdPVwiaXRlbS52YWx1ZVwiIFtkaXNhYmxlZF09XCJvcHRpb25EaXNhYmxlZCA/IG9wdGlvbkRpc2FibGVkKGl0ZW0pIDogZmFsc2VcIj5cbiAgICAgICAgICAgICAgICB7eyBpdGVtLm5hbWUgfX1cbiAgICAgICAgICAgIDwvbWF0LW9wdGlvbj5cbiAgICAgICAgfVxuICAgIDwvbWF0LXNlbGVjdD5cblxuICAgIEBpZiAoaGludCkge1xuICAgICAgICA8bWF0LWhpbnQ+e3sgaGludCB9fTwvbWF0LWhpbnQ+XG4gICAgfVxuXG4gICAgQGlmIChoYXNSZXF1aXJlZEVycm9yKCkpIHtcbiAgICAgICAgPG1hdC1lcnJvciBpMThuPkNlIGNoYW1wIGVzdCByZXF1aXM8L21hdC1lcnJvcj5cbiAgICB9XG48L21hdC1mb3JtLWZpZWxkPlxuIl19
@@ -2471,7 +2471,8 @@ class NaturalAbstractDetail extends NaturalAbstractPanel {
2471
2471
  return this.router.navigateByUrl(newUrl); // replace /new by /123
2472
2472
  }
2473
2473
  else {
2474
- return this.router.navigate(['..', model.id], { relativeTo: this.route });
2474
+ const queryParams = this.route.snapshot.queryParams;
2475
+ return this.router.navigate(['..', model.id], { relativeTo: this.route, queryParams });
2475
2476
  }
2476
2477
  }
2477
2478
  return EMPTY;
@@ -6638,12 +6639,12 @@ class AbstractSelect {
6638
6639
  /**
6639
6640
  * Contains internal representation for current selection AND searched text (for autocomplete)
6640
6641
  *
6641
- * It is **not** necessarily `V | null`.
6642
+ * It is **not** necessarily `TValue | null`.
6642
6643
  *
6643
- * - NaturalSelectComponent: `string | V | null`. We allow `string`
6644
- * only when `optionRequired` is false, so most of the time it is `V | null`.
6644
+ * - NaturalSelectComponent: `string | TValue | null`. We allow `string`
6645
+ * only when `optionRequired` is false, so most of the time it is `TValue | null`.
6645
6646
  * - NaturalSelectHierarchicComponent: `string | null`.
6646
- * - NaturalSelectEnumComponent: `V | null`.
6647
+ * - NaturalSelectEnumComponent: `TValue | null`.
6647
6648
  *
6648
6649
  * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :
6649
6650
  * - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed
@@ -9209,7 +9210,7 @@ class NaturalFileComponent {
9209
9210
  *
9210
9211
  * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept
9211
9212
  */
9212
- accept = 'image/bmp,image/gif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';
9213
+ accept = 'image/avif,image/bmp,image/gif,image/heic,image/heif,image/jpeg,image/pjpeg,image/png,image/svg+xml,image/svg,image/webp';
9213
9214
  /**
9214
9215
  * If given, it will be called when a new file is selected. The callback should typically upload the file
9215
9216
  * to the server and link the newly uploaded file to the existing related object.
@@ -9359,7 +9360,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
9359
9360
 
9360
9361
  class NaturalFixedButtonComponent {
9361
9362
  icon;
9362
- link = [];
9363
+ link = null;
9363
9364
  color = 'accent';
9364
9365
  disabled = false;
9365
9366
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalFixedButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });