@ecodev/natural 62.1.1 → 62.1.3

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.
@@ -83,6 +83,11 @@ export class NaturalPanelsService {
83
83
  this.updateComponentsPosition();
84
84
  });
85
85
  }
86
+ /**
87
+ * Notify the service to start listening to route changes to open panels
88
+ *
89
+ * @internal
90
+ */
86
91
  start(route) {
87
92
  NaturalPanelsService._opened = true;
88
93
  this.routeSub = route.url.subscribe(segments => {
@@ -136,6 +141,11 @@ export class NaturalPanelsService {
136
141
  this.router.errorHandler = originalErrorHandler;
137
142
  });
138
143
  }
144
+ /**
145
+ * Notify the service that all panels were closed
146
+ *
147
+ * @internal
148
+ */
139
149
  stop() {
140
150
  NaturalPanelsService._opened = false;
141
151
  this.routeSub?.unsubscribe();
@@ -146,6 +156,8 @@ export class NaturalPanelsService {
146
156
  }
147
157
  /**
148
158
  * Go to panel matching given component. Causes an url change.
159
+ *
160
+ * @internal
149
161
  */
150
162
  goToPanelByComponent(component) {
151
163
  this.goToPanelByIndex(this.getPanelIndex(component));
@@ -289,6 +301,13 @@ export class NaturalPanelsService {
289
301
  }
290
302
  return this.dialog.openDialogs.findIndex(dialog => dialog.componentInstance === component);
291
303
  }
304
+ /**
305
+ * Whether the given panel is currently the top, visible, panel. If there are no panels opened at all, then any panel given is considered top, visible, panel.
306
+ */
307
+ isTopPanel(component) {
308
+ const length = this.dialog.openDialogs.length;
309
+ return !length || this.dialog.openDialogs[length - 1]?.componentInstance === component;
310
+ }
292
311
  /**
293
312
  * Repositions panels from start until given index
294
313
  */
@@ -326,4 +345,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
326
345
  providedIn: 'root',
327
346
  }]
328
347
  }], ctorParameters: () => [] });
329
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"panels.service.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/panels/panels.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {BreakpointObserver, Breakpoints} from '@angular/cdk/layout';\nimport {ComponentType} from '@angular/cdk/portal';\nimport {inject, Injectable, Injector, runInInjectionContext} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {MatDialog, MatDialogConfig} from '@angular/material/dialog';\nimport {ActivatedRoute, DefaultUrlSerializer, NavigationError, Router, UrlSegment} from '@angular/router';\nimport {differenceWith, flatten, isEqual} from 'lodash-es';\nimport {forkJoin, Observable, of, Subject, Subscription} from 'rxjs';\nimport {NaturalAbstractPanel} from './abstract-panel';\nimport {getStackConfig} from './panels.urlmatcher';\nimport {\n    NaturalPanelConfig,\n    NaturalPanelData,\n    NaturalPanelsBeforeOpenPanel,\n    NaturalPanelsRouterRule,\n    PanelsHooksConfig,\n} from './types';\n\nfunction segmentsToString(segments: UrlSegment[]): string {\n    return segments.map(s => s.toString()).join('/');\n}\n\nfunction compareConfigs(a: NaturalPanelConfig, b: NaturalPanelConfig): boolean {\n    return a.route.path === b.route.path && a.rule === b.rule && isEqual(a.params, b.params);\n}\n\n/**\n * TODO: implement route update when closing dialog with escape\n * @dynamic\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalPanelsService {\n    private readonly router = inject(Router);\n    private readonly dialog = inject(MatDialog);\n    private readonly injector = inject(Injector);\n    private hooksConfig = inject(PanelsHooksConfig);\n\n    private readonly panelWidth = '960px';\n\n    /**\n     * Because of this static property Panels are **not** compatible with SSR.\n     * And we cannot make it non-static, because `UrlMatcher` cannot be injected.\n     */\n    private static _opened = false;\n    public static get opened(): boolean {\n        return this._opened;\n    }\n\n    /**\n     * Stream that emits when all open dialog have finished closing\n     */\n    public readonly afterAllClosed = new Subject<void>();\n    /**\n     * Cache for panels counter. Works more like an ID.\n     * Is used to give an unique identifier to multiple similar panels configurations\n     */\n    private counter = 1;\n    /**\n     * Class applied to dialog overlay related with panels\n     * If change, change CSS too\n     */\n    private panelClass = 'panel';\n\n    /**\n     * Cache for panels setup before navigation change.\n     * Used to detect panels openings/closings and adapt for new configuration\n     */\n    private oldFullConfig: NaturalPanelConfig[] = [];\n\n    /**\n     * Cache for subscription stop\n     */\n    private routeSub?: Subscription;\n\n    /**\n     * Cache for subscription stop\n     */\n    private navSub?: Subscription;\n\n    /**\n     * Horizontal gaps between panels\n     */\n    private panelsOffsetH = 35;\n\n    /**\n     * Vertical gaps between panels\n     */\n    private panelsOffsetV = 40;\n\n    /**\n     * Cache of previous screen size\n     * Used to change panels stack orientation on small screens\n     */\n    private isVertical = false;\n\n    public constructor() {\n        const breakpointObserver = inject(BreakpointObserver);\n\n        //  Watch media to know if display panels horizontally or vertically\n        breakpointObserver\n            .observe(Breakpoints.XSmall)\n            .pipe(takeUntilDestroyed())\n            .subscribe(result => {\n                this.isVertical = result.matches;\n                this.updateComponentsPosition();\n            });\n    }\n\n    public start(route: ActivatedRoute): void {\n        NaturalPanelsService._opened = true;\n        this.routeSub = route.url.subscribe(segments => {\n            this.updatePanels(segments, route.snapshot.data.panelsRoutes);\n        });\n\n        this.navSub = this.router.events.subscribe(ev => {\n            if (!(ev instanceof NavigationError)) {\n                return;\n            }\n\n            // Abort and propagate navigation error\n            if (ev.url.endsWith('/')) {\n                this.stop();\n\n                // This is a bit hackish: we hardcode an impossible url to trigger normal navigation error mechanism\n                this.router.navigateByUrl('panels-reached-invalid-url');\n\n                return;\n            }\n\n            this.counter++;\n\n            // pc stands for \"panel counter\", required to give an identification to panels with exact same config\n            // E.g /new\n            const wantedUrl = ev.url + ';pc=' + this.counter;\n\n            // Segments matching from wanted url. E.g ~ ['new']\n            const wantedUrSegments = new DefaultUrlSerializer().parse(wantedUrl).root.children.primary.segments;\n\n            // Don't match any config\n            const wantedConfig = getStackConfig(wantedUrSegments, route.snapshot.data.panelsRoutes, this.injector);\n\n            if (wantedConfig.length) {\n                return this.appendConfigToCurrentUrl(wantedConfig);\n            }\n\n            // Currently activated routes. E.g ['objective', 'objective', 123, 'risk']\n            const currentSegments = flatten(\n                this.dialog.openDialogs.map(d => d.componentInstance.panelData.config.route.segments),\n            );\n\n            // Last segment. E.g ['risk']\n            const lastOfCurrentSegments = currentSegments.slice(-1);\n\n            // Config for ['risk', 'new']\n            const currentAndWantedConfig = getStackConfig(\n                lastOfCurrentSegments.concat(wantedUrSegments),\n                route.snapshot.data.panelsRoutes,\n                this.injector,\n            );\n\n            return this.appendConfigToCurrentUrl(currentAndWantedConfig);\n        });\n    }\n\n    /**\n     * Uses given configuration to add in the end of current url\n     * Neutralizes router error handling\n     */\n    private appendConfigToCurrentUrl(config: NaturalPanelConfig[]): void {\n        const originalErrorHandler = this.router.errorHandler;\n\n        // Nullify error handler (will be de-neutralized after route redirection)\n        if (config) {\n            this.router.errorHandler = () => undefined;\n        }\n\n        // Navigate to same url + /risk/new Result : /risk/risk/new\n        const newUrl = config.map(conf => segmentsToString(conf.route.segments)).join('/');\n\n        this.router.navigateByUrl(this.router.url + '/' + newUrl).then(() => {\n            // After navigation has ended, restore original error handler because he's not a bad guy\n            this.router.errorHandler = originalErrorHandler;\n        });\n    }\n\n    public stop(): void {\n        NaturalPanelsService._opened = false;\n        this.routeSub?.unsubscribe();\n        this.navSub?.unsubscribe();\n        this.dialog.closeAll();\n        this.oldFullConfig = [];\n        this.afterAllClosed.next();\n    }\n\n    /**\n     * Go to panel matching given component. Causes an url change.\n     */\n    public goToPanelByComponent(component: NaturalAbstractPanel): void {\n        this.goToPanelByIndex(this.getPanelIndex(component));\n    }\n\n    /**\n     * Go to panel matching given component. Causes an url change.\n     */\n    public goToPenultimatePanel(): void {\n        this.goToPanelByIndex(this.dialog.openDialogs.length - 2);\n    }\n\n    /**\n     * Calls the new url that only includes the segments from the panels we want to stay open\n     */\n    private goToPanelByIndex(index: number): void {\n        // Extracts url segments from next panel until last one\n        const url = this.dialog.openDialogs\n            .slice(index + 1)\n            .map(dialog => {\n                return segmentsToString(dialog.componentInstance.panelData.config.route.segments);\n            })\n            .join('/');\n\n        // Remove extra segments and redirects to root\n        this.router.navigateByUrl(this.router.url.replace('/' + url, ''));\n    }\n\n    /**\n     * Selecting a panel is equivalent to close all those that are in front of him\n     * @param index of panel in stack. The most behind (the first one) is 0.\n     */\n    private selectPanelByIndex(index: number): Observable<void> {\n        const lastDialog = this.dialog.openDialogs[this.dialog.openDialogs.length - 1];\n\n        // Update new panels set positions\n        this.updateComponentsPosition();\n\n        for (let i = this.dialog.openDialogs.length - 1; i >= index + 1; i--) {\n            this.dialog.openDialogs[i].close();\n        }\n\n        return lastDialog.afterClosed();\n    }\n\n    /**\n     * Open new panels if url has changed with new segments\n     */\n    private updatePanels(segments: UrlSegment[], routes: NaturalPanelsRouterRule[]): void {\n        // Transform url segments into a config with component name and ID if provided in next segment\n        // Returns an array of configs, each config represents the content relative to a panel\n        const newFullConfig = getStackConfig(segments, routes, this.injector);\n        const configsToRemove = differenceWith(this.oldFullConfig, newFullConfig, compareConfigs);\n        const configsToAdd = differenceWith(newFullConfig, this.oldFullConfig, compareConfigs);\n\n        const indexOfNextPanel = this.oldFullConfig.length - configsToRemove.length - 1;\n\n        if (configsToRemove.length && configsToAdd.length) {\n            // Add and remove panels\n            this.selectPanelByIndex(indexOfNextPanel).subscribe(() => {\n                this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());\n            });\n        } else if (configsToRemove.length && !configsToAdd.length) {\n            // only remove panels\n            this.selectPanelByIndex(indexOfNextPanel).subscribe(() => this.updateComponentsPosition());\n        } else if (!configsToRemove.length && configsToAdd.length) {\n            // only add panels\n            this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());\n        }\n\n        this.oldFullConfig = newFullConfig;\n    }\n\n    /**\n     * Resolve all services, then open panels\n     */\n    private openPanels(\n        newItemsConfig: NaturalPanelConfig[],\n        fullConfig: NaturalPanelConfig[],\n    ): Observable<Observable<any> | null> {\n        const subject = new Subject<Observable<any> | null>();\n\n        // Resolve everything before opening a single panel\n        const resolves = newItemsConfig.map(conf => this.getResolvedData(conf));\n\n        // ForkJoin emits when all promises are executed;\n        forkJoin(resolves).subscribe(resolvedResult => {\n            // For each new config entry, open a new panel\n            for (let i = 0; i < newItemsConfig.length; i++) {\n                const config = newItemsConfig[i];\n                let itemData: NaturalPanelData = {\n                    // Config of actual panel route\n                    config: config,\n\n                    // Data resolved by service\n                    // Use in component by calling this.panelData.data.xyz\n                    data: resolvedResult[i],\n                    linkableObjects: [],\n                };\n\n                if (this.hooksConfig?.beforeOpenPanel) {\n                    const event: NaturalPanelsBeforeOpenPanel = {\n                        itemData: itemData,\n                        panelConfig: config,\n                        resolvedResult: resolvedResult,\n                        fullPanelsConfig: fullConfig,\n                    };\n\n                    itemData = this.hooksConfig.beforeOpenPanel(this.injector, event);\n                }\n\n                this.openPanel(config.component, itemData);\n            }\n\n            this.dialog.openDialogs[this.dialog.openDialogs.length - 1].afterOpened().subscribe(() => {\n                subject.next(null);\n            });\n        });\n\n        return subject;\n    }\n\n    private getResolvedData(config: NaturalPanelConfig): Observable<Record<string, unknown>> {\n        if (!config.resolve || (config.resolve && Object.keys(config.resolve).length === 0)) {\n            return of({});\n        }\n\n        const resolveKeys = Object.keys(config.resolve);\n        const resolvedData: Record<string, Observable<unknown>> = {};\n        const injector = config.injector;\n\n        if (injector) {\n            resolveKeys.forEach(key => {\n                resolvedData[key] = runInInjectionContext(injector, () => config.resolve[key](config));\n            });\n        }\n\n        return forkJoin(resolvedData);\n    }\n\n    private openPanel(componentOrTemplateRef: ComponentType<NaturalAbstractPanel>, data: NaturalPanelData): void {\n        const conf: MatDialogConfig = {\n            panelClass: this.panelClass,\n            closeOnNavigation: false,\n            hasBackdrop: this.dialog.openDialogs.length === 0,\n            height: '100%',\n            width: this.panelWidth,\n            position: {\n                top: '0',\n                right: '0',\n            },\n        };\n\n        const dialogRef = this.dialog.open<NaturalAbstractPanel>(componentOrTemplateRef, conf);\n\n        // Panelisable interface attributes/functions\n        dialogRef.componentInstance.initPanel(data);\n\n        dialogRef.beforeClosed().subscribe(() => {\n            const index = this.getPanelIndex(dialogRef.componentInstance);\n            this.goToPanelByIndex(index - 1);\n        });\n    }\n\n    /**\n     * Return panel position (index) by searching matching component\n     */\n    private getPanelIndex(component: NaturalAbstractPanel): number {\n        if (!component) {\n            return -1;\n        }\n\n        return this.dialog.openDialogs.findIndex(dialog => dialog.componentInstance === component);\n    }\n\n    /**\n     * Repositions panels from start until given index\n     */\n    private updateComponentsPosition(): void {\n        if (!this.dialog.openDialogs.length) {\n            return;\n        }\n\n        // Select the panels that are still opened, ignore the others because they'll be closed\n        const affectedElements = this.dialog.openDialogs;\n        for (let i = 0; i < affectedElements.length; i++) {\n            const dialog = affectedElements[i];\n\n            // Set all panels as \"hidden\" except last one. IsFrontPanel attribute causes a CSS that hides body via hostbinding\n            dialog.componentInstance.isFrontPanel = i === affectedElements.length - 1;\n            dialog.componentInstance.panelService = this;\n\n            // Assign offset\n            const deep = affectedElements.length - 1 - i;\n\n            let position: any = {right: deep * this.panelsOffsetH + 'px'};\n            if (this.isVertical && affectedElements.length > 1) {\n                const top = i * this.panelsOffsetV + 'px';\n                position = {top: top, right: '0px'};\n                dialog.updateSize(this.panelWidth, `calc(100% - ${top})`); // call before updatePosition\n            } else {\n                dialog.updateSize(this.panelWidth, `100%`);\n            }\n\n            dialog.updatePosition(position);\n        }\n    }\n}\n"]}
348
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"panels.service.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/panels/panels.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {BreakpointObserver, Breakpoints} from '@angular/cdk/layout';\nimport {ComponentType} from '@angular/cdk/portal';\nimport {inject, Injectable, Injector, runInInjectionContext} from '@angular/core';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {MatDialog, MatDialogConfig} from '@angular/material/dialog';\nimport {ActivatedRoute, DefaultUrlSerializer, NavigationError, Router, UrlSegment} from '@angular/router';\nimport {differenceWith, flatten, isEqual} from 'lodash-es';\nimport {forkJoin, Observable, of, Subject, Subscription} from 'rxjs';\nimport {NaturalAbstractPanel} from './abstract-panel';\nimport {getStackConfig} from './panels.urlmatcher';\nimport {\n    NaturalPanelConfig,\n    NaturalPanelData,\n    NaturalPanelsBeforeOpenPanel,\n    NaturalPanelsRouterRule,\n    PanelsHooksConfig,\n} from './types';\n\nfunction segmentsToString(segments: UrlSegment[]): string {\n    return segments.map(s => s.toString()).join('/');\n}\n\nfunction compareConfigs(a: NaturalPanelConfig, b: NaturalPanelConfig): boolean {\n    return a.route.path === b.route.path && a.rule === b.rule && isEqual(a.params, b.params);\n}\n\n/**\n * TODO: implement route update when closing dialog with escape\n * @dynamic\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalPanelsService {\n    private readonly router = inject(Router);\n    private readonly dialog = inject(MatDialog);\n    private readonly injector = inject(Injector);\n    private hooksConfig = inject(PanelsHooksConfig);\n\n    private readonly panelWidth = '960px';\n\n    /**\n     * Because of this static property Panels are **not** compatible with SSR.\n     * And we cannot make it non-static, because `UrlMatcher` cannot be injected.\n     */\n    private static _opened = false;\n    public static get opened(): boolean {\n        return this._opened;\n    }\n\n    /**\n     * Stream that emits when all open dialog have finished closing\n     */\n    public readonly afterAllClosed = new Subject<void>();\n    /**\n     * Cache for panels counter. Works more like an ID.\n     * Is used to give an unique identifier to multiple similar panels configurations\n     */\n    private counter = 1;\n    /**\n     * Class applied to dialog overlay related with panels\n     * If change, change CSS too\n     */\n    private panelClass = 'panel';\n\n    /**\n     * Cache for panels setup before navigation change.\n     * Used to detect panels openings/closings and adapt for new configuration\n     */\n    private oldFullConfig: NaturalPanelConfig[] = [];\n\n    /**\n     * Cache for subscription stop\n     */\n    private routeSub?: Subscription;\n\n    /**\n     * Cache for subscription stop\n     */\n    private navSub?: Subscription;\n\n    /**\n     * Horizontal gaps between panels\n     */\n    private panelsOffsetH = 35;\n\n    /**\n     * Vertical gaps between panels\n     */\n    private panelsOffsetV = 40;\n\n    /**\n     * Cache of previous screen size\n     * Used to change panels stack orientation on small screens\n     */\n    private isVertical = false;\n\n    public constructor() {\n        const breakpointObserver = inject(BreakpointObserver);\n\n        //  Watch media to know if display panels horizontally or vertically\n        breakpointObserver\n            .observe(Breakpoints.XSmall)\n            .pipe(takeUntilDestroyed())\n            .subscribe(result => {\n                this.isVertical = result.matches;\n                this.updateComponentsPosition();\n            });\n    }\n\n    /**\n     * Notify the service to start listening to route changes to open panels\n     *\n     * @internal\n     */\n    public start(route: ActivatedRoute): void {\n        NaturalPanelsService._opened = true;\n        this.routeSub = route.url.subscribe(segments => {\n            this.updatePanels(segments, route.snapshot.data.panelsRoutes);\n        });\n\n        this.navSub = this.router.events.subscribe(ev => {\n            if (!(ev instanceof NavigationError)) {\n                return;\n            }\n\n            // Abort and propagate navigation error\n            if (ev.url.endsWith('/')) {\n                this.stop();\n\n                // This is a bit hackish: we hardcode an impossible url to trigger normal navigation error mechanism\n                this.router.navigateByUrl('panels-reached-invalid-url');\n\n                return;\n            }\n\n            this.counter++;\n\n            // pc stands for \"panel counter\", required to give an identification to panels with exact same config\n            // E.g /new\n            const wantedUrl = ev.url + ';pc=' + this.counter;\n\n            // Segments matching from wanted url. E.g ~ ['new']\n            const wantedUrSegments = new DefaultUrlSerializer().parse(wantedUrl).root.children.primary.segments;\n\n            // Don't match any config\n            const wantedConfig = getStackConfig(wantedUrSegments, route.snapshot.data.panelsRoutes, this.injector);\n\n            if (wantedConfig.length) {\n                return this.appendConfigToCurrentUrl(wantedConfig);\n            }\n\n            // Currently activated routes. E.g ['objective', 'objective', 123, 'risk']\n            const currentSegments = flatten(\n                this.dialog.openDialogs.map(d => d.componentInstance.panelData.config.route.segments),\n            );\n\n            // Last segment. E.g ['risk']\n            const lastOfCurrentSegments = currentSegments.slice(-1);\n\n            // Config for ['risk', 'new']\n            const currentAndWantedConfig = getStackConfig(\n                lastOfCurrentSegments.concat(wantedUrSegments),\n                route.snapshot.data.panelsRoutes,\n                this.injector,\n            );\n\n            return this.appendConfigToCurrentUrl(currentAndWantedConfig);\n        });\n    }\n\n    /**\n     * Uses given configuration to add in the end of current url\n     * Neutralizes router error handling\n     */\n    private appendConfigToCurrentUrl(config: NaturalPanelConfig[]): void {\n        const originalErrorHandler = this.router.errorHandler;\n\n        // Nullify error handler (will be de-neutralized after route redirection)\n        if (config) {\n            this.router.errorHandler = () => undefined;\n        }\n\n        // Navigate to same url + /risk/new Result : /risk/risk/new\n        const newUrl = config.map(conf => segmentsToString(conf.route.segments)).join('/');\n\n        this.router.navigateByUrl(this.router.url + '/' + newUrl).then(() => {\n            // After navigation has ended, restore original error handler because he's not a bad guy\n            this.router.errorHandler = originalErrorHandler;\n        });\n    }\n\n    /**\n     * Notify the service that all panels were closed\n     *\n     * @internal\n     */\n    public stop(): void {\n        NaturalPanelsService._opened = false;\n        this.routeSub?.unsubscribe();\n        this.navSub?.unsubscribe();\n        this.dialog.closeAll();\n        this.oldFullConfig = [];\n        this.afterAllClosed.next();\n    }\n\n    /**\n     * Go to panel matching given component. Causes an url change.\n     *\n     * @internal\n     */\n    public goToPanelByComponent(component: NaturalAbstractPanel): void {\n        this.goToPanelByIndex(this.getPanelIndex(component));\n    }\n\n    /**\n     * Go to panel matching given component. Causes an url change.\n     */\n    public goToPenultimatePanel(): void {\n        this.goToPanelByIndex(this.dialog.openDialogs.length - 2);\n    }\n\n    /**\n     * Calls the new url that only includes the segments from the panels we want to stay open\n     */\n    private goToPanelByIndex(index: number): void {\n        // Extracts url segments from next panel until last one\n        const url = this.dialog.openDialogs\n            .slice(index + 1)\n            .map(dialog => {\n                return segmentsToString(dialog.componentInstance.panelData.config.route.segments);\n            })\n            .join('/');\n\n        // Remove extra segments and redirects to root\n        this.router.navigateByUrl(this.router.url.replace('/' + url, ''));\n    }\n\n    /**\n     * Selecting a panel is equivalent to close all those that are in front of him\n     * @param index of panel in stack. The most behind (the first one) is 0.\n     */\n    private selectPanelByIndex(index: number): Observable<void> {\n        const lastDialog = this.dialog.openDialogs[this.dialog.openDialogs.length - 1];\n\n        // Update new panels set positions\n        this.updateComponentsPosition();\n\n        for (let i = this.dialog.openDialogs.length - 1; i >= index + 1; i--) {\n            this.dialog.openDialogs[i].close();\n        }\n\n        return lastDialog.afterClosed();\n    }\n\n    /**\n     * Open new panels if url has changed with new segments\n     */\n    private updatePanels(segments: UrlSegment[], routes: NaturalPanelsRouterRule[]): void {\n        // Transform url segments into a config with component name and ID if provided in next segment\n        // Returns an array of configs, each config represents the content relative to a panel\n        const newFullConfig = getStackConfig(segments, routes, this.injector);\n        const configsToRemove = differenceWith(this.oldFullConfig, newFullConfig, compareConfigs);\n        const configsToAdd = differenceWith(newFullConfig, this.oldFullConfig, compareConfigs);\n\n        const indexOfNextPanel = this.oldFullConfig.length - configsToRemove.length - 1;\n\n        if (configsToRemove.length && configsToAdd.length) {\n            // Add and remove panels\n            this.selectPanelByIndex(indexOfNextPanel).subscribe(() => {\n                this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());\n            });\n        } else if (configsToRemove.length && !configsToAdd.length) {\n            // only remove panels\n            this.selectPanelByIndex(indexOfNextPanel).subscribe(() => this.updateComponentsPosition());\n        } else if (!configsToRemove.length && configsToAdd.length) {\n            // only add panels\n            this.openPanels(configsToAdd, newFullConfig).subscribe(() => this.updateComponentsPosition());\n        }\n\n        this.oldFullConfig = newFullConfig;\n    }\n\n    /**\n     * Resolve all services, then open panels\n     */\n    private openPanels(\n        newItemsConfig: NaturalPanelConfig[],\n        fullConfig: NaturalPanelConfig[],\n    ): Observable<Observable<any> | null> {\n        const subject = new Subject<Observable<any> | null>();\n\n        // Resolve everything before opening a single panel\n        const resolves = newItemsConfig.map(conf => this.getResolvedData(conf));\n\n        // ForkJoin emits when all promises are executed;\n        forkJoin(resolves).subscribe(resolvedResult => {\n            // For each new config entry, open a new panel\n            for (let i = 0; i < newItemsConfig.length; i++) {\n                const config = newItemsConfig[i];\n                let itemData: NaturalPanelData = {\n                    // Config of actual panel route\n                    config: config,\n\n                    // Data resolved by service\n                    // Use in component by calling this.panelData.data.xyz\n                    data: resolvedResult[i],\n                    linkableObjects: [],\n                };\n\n                if (this.hooksConfig?.beforeOpenPanel) {\n                    const event: NaturalPanelsBeforeOpenPanel = {\n                        itemData: itemData,\n                        panelConfig: config,\n                        resolvedResult: resolvedResult,\n                        fullPanelsConfig: fullConfig,\n                    };\n\n                    itemData = this.hooksConfig.beforeOpenPanel(this.injector, event);\n                }\n\n                this.openPanel(config.component, itemData);\n            }\n\n            this.dialog.openDialogs[this.dialog.openDialogs.length - 1].afterOpened().subscribe(() => {\n                subject.next(null);\n            });\n        });\n\n        return subject;\n    }\n\n    private getResolvedData(config: NaturalPanelConfig): Observable<Record<string, unknown>> {\n        if (!config.resolve || (config.resolve && Object.keys(config.resolve).length === 0)) {\n            return of({});\n        }\n\n        const resolveKeys = Object.keys(config.resolve);\n        const resolvedData: Record<string, Observable<unknown>> = {};\n        const injector = config.injector;\n\n        if (injector) {\n            resolveKeys.forEach(key => {\n                resolvedData[key] = runInInjectionContext(injector, () => config.resolve[key](config));\n            });\n        }\n\n        return forkJoin(resolvedData);\n    }\n\n    private openPanel(componentOrTemplateRef: ComponentType<NaturalAbstractPanel>, data: NaturalPanelData): void {\n        const conf: MatDialogConfig = {\n            panelClass: this.panelClass,\n            closeOnNavigation: false,\n            hasBackdrop: this.dialog.openDialogs.length === 0,\n            height: '100%',\n            width: this.panelWidth,\n            position: {\n                top: '0',\n                right: '0',\n            },\n        };\n\n        const dialogRef = this.dialog.open<NaturalAbstractPanel>(componentOrTemplateRef, conf);\n\n        // Panelisable interface attributes/functions\n        dialogRef.componentInstance.initPanel(data);\n\n        dialogRef.beforeClosed().subscribe(() => {\n            const index = this.getPanelIndex(dialogRef.componentInstance);\n            this.goToPanelByIndex(index - 1);\n        });\n    }\n\n    /**\n     * Return panel position (index) by searching matching component\n     */\n    private getPanelIndex(component: NaturalAbstractPanel): number {\n        if (!component) {\n            return -1;\n        }\n\n        return this.dialog.openDialogs.findIndex(dialog => dialog.componentInstance === component);\n    }\n\n    /**\n     * Whether the given panel is currently the top, visible, panel. If there are no panels opened at all, then any panel given is considered top, visible, panel.\n     */\n    public isTopPanel(component: NaturalAbstractPanel): boolean {\n        const length = this.dialog.openDialogs.length;\n        return !length || this.dialog.openDialogs[length - 1]?.componentInstance === component;\n    }\n\n    /**\n     * Repositions panels from start until given index\n     */\n    private updateComponentsPosition(): void {\n        if (!this.dialog.openDialogs.length) {\n            return;\n        }\n\n        // Select the panels that are still opened, ignore the others because they'll be closed\n        const affectedElements = this.dialog.openDialogs;\n        for (let i = 0; i < affectedElements.length; i++) {\n            const dialog = affectedElements[i];\n\n            // Set all panels as \"hidden\" except last one. IsFrontPanel attribute causes a CSS that hides body via hostbinding\n            dialog.componentInstance.isFrontPanel = i === affectedElements.length - 1;\n            dialog.componentInstance.panelService = this;\n\n            // Assign offset\n            const deep = affectedElements.length - 1 - i;\n\n            let position: any = {right: deep * this.panelsOffsetH + 'px'};\n            if (this.isVertical && affectedElements.length > 1) {\n                const top = i * this.panelsOffsetV + 'px';\n                position = {top: top, right: '0px'};\n                dialog.updateSize(this.panelWidth, `calc(100% - ${top})`); // call before updatePosition\n            } else {\n                dialog.updateSize(this.panelWidth, `100%`);\n            }\n\n            dialog.updatePosition(position);\n        }\n    }\n}\n"]}
@@ -1,5 +1,5 @@
1
1
  import { coerceBooleanProperty } from '@angular/cdk/coercion';
2
- import { Directive, EventEmitter, inject, Input, Output } from '@angular/core';
2
+ import { computed, Directive, EventEmitter, inject, input, Input, Output } from '@angular/core';
3
3
  import { FormControl, FormControlDirective, FormControlName, NgControl, Validators, } from '@angular/forms';
4
4
  import { ErrorStateMatcher } from '@angular/material/core';
5
5
  import * as i0 from "@angular/core";
@@ -26,9 +26,11 @@ class ExternalFormControlMatcher extends ErrorStateMatcher {
26
26
  export class AbstractSelect {
27
27
  placeholder;
28
28
  /**
29
- * Mat-hint
29
+ * Mat-hint, if given, and it is non-empty, then `subscriptSizing` will
30
+ * automatically be set to `dynamic` to allow for long, wrapping text.
30
31
  */
31
- hint = null;
32
+ hint = input();
33
+ subscriptSizing = computed(() => (this.hint() ? 'dynamic' : 'fixed'));
32
34
  /**
33
35
  * If given an error message, it will be displayed in a `<mat-error>`, but only if the control
34
36
  * is actually in an error state via one of its validators.
@@ -199,15 +201,13 @@ export class AbstractSelect {
199
201
  this.internalCtrl.updateValueAndValidity();
200
202
  }
201
203
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: AbstractSelect, deps: [], target: i0.ɵɵFactoryTarget.Directive });
202
- static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: AbstractSelect, isStandalone: true, inputs: { placeholder: "placeholder", hint: "hint", error: "error", required: "required", navigateTo: "navigateTo", clearLabel: "clearLabel", showIcon: "showIcon", icon: "icon", displayWith: "displayWith", disabled: "disabled" }, outputs: { selectionChange: "selectionChange", blur: "blur" }, ngImport: i0 });
204
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.3", type: AbstractSelect, isStandalone: true, inputs: { placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: false, isRequired: false, transformFunction: null }, hint: { classPropertyName: "hint", publicName: "hint", isSignal: true, isRequired: false, transformFunction: null }, error: { classPropertyName: "error", publicName: "error", isSignal: false, isRequired: false, transformFunction: null }, required: { classPropertyName: "required", publicName: "required", isSignal: false, isRequired: false, transformFunction: null }, navigateTo: { classPropertyName: "navigateTo", publicName: "navigateTo", isSignal: false, isRequired: false, transformFunction: null }, clearLabel: { classPropertyName: "clearLabel", publicName: "clearLabel", isSignal: false, isRequired: false, transformFunction: null }, showIcon: { classPropertyName: "showIcon", publicName: "showIcon", isSignal: false, isRequired: false, transformFunction: null }, icon: { classPropertyName: "icon", publicName: "icon", isSignal: false, isRequired: false, transformFunction: null }, displayWith: { classPropertyName: "displayWith", publicName: "displayWith", isSignal: false, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { selectionChange: "selectionChange", blur: "blur" }, ngImport: i0 });
203
205
  }
204
206
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: AbstractSelect, decorators: [{
205
207
  type: Directive,
206
208
  args: [{ standalone: true }]
207
209
  }], ctorParameters: () => [], propDecorators: { placeholder: [{
208
210
  type: Input
209
- }], hint: [{
210
- type: Input
211
211
  }], error: [{
212
212
  type: Input
213
213
  }], required: [{
@@ -229,4 +229,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
229
229
  }], disabled: [{
230
230
  type: Input
231
231
  }] } });
232
- //# 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    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 given an error message, it will be displayed in a `<mat-error>`, but only if the control\n     * is actually in an error state via one of its validators.\n     *\n     * This is not needed for the special case of the required validator, because the message is hardcoded.\n     */\n    @Input() public error: 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?.hasValidator(Validators.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.hasValidator(Validators.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"]}
232
+ //# 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 {computed, Directive, DoCheck, EventEmitter, inject, input, Input, OnInit, Output} from '@angular/core';\nimport {\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, if given, and it is non-empty, then `subscriptSizing` will\n     * automatically be set to `dynamic` to allow for long, wrapping text.\n     */\n    public readonly hint = input<string | null>();\n\n    protected readonly subscriptSizing = computed(() => (this.hint() ? 'dynamic' : 'fixed'));\n\n    /**\n     * If given an error message, it will be displayed in a `<mat-error>`, but only if the control\n     * is actually in an error state via one of its validators.\n     *\n     * This is not needed for the special case of the required validator, because the message is hardcoded.\n     */\n    @Input() public error: 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?.hasValidator(Validators.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.hasValidator(Validators.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"]}
@@ -268,7 +268,7 @@ export class NaturalSelectComponent extends AbstractSelect {
268
268
  return this.variablesManager.variables.value;
269
269
  }
270
270
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSelectComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
271
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.3", type: NaturalSelectComponent, isStandalone: true, selector: "natural-select", inputs: { service: "service", optionRequired: "optionRequired", searchField: "searchField", searchOperator: "searchOperator", filter: "filter", disabled: "disabled" }, queries: [{ propertyName: "itemTemplate", first: true, predicate: TemplateRef, descendants: true }], viewQueries: [{ propertyName: "autoTrigger", first: true, predicate: MatAutocompleteTrigger, descendants: true }], usesInheritance: true, ngImport: i0, template: "<!-- 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\u00E9sultats</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\u00E9lection\"\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\u00E9s\u00E9lectionner\">\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 } @else if (error) {\n <mat-error>{{ error }}</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", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"], dependencies: [{ kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i1.MatAutocomplete, selector: "mat-autocomplete", inputs: ["aria-label", "aria-labelledby", "displayWith", "autoActiveFirstOption", "autoSelectActiveOption", "requireSelection", "panelWidth", "disableRipple", "class", "hideSingleSelectionIndicator"], outputs: ["optionSelected", "opened", "closed", "optionActivated"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i2.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "directive", type: i1.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", inputs: ["matAutocomplete", "matAutocompletePosition", "matAutocompleteConnectedTo", "autocomplete", "matAutocompleteDisabled"], exportAs: ["matAutocompleteTrigger"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "directive", type: i4.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i4.MatPrefix, selector: "[matPrefix], [matIconPrefix], [matTextPrefix]", inputs: ["matTextPrefix"] }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i5.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i6.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i8.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i9.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i9.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i10.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }] });
271
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.3", type: NaturalSelectComponent, isStandalone: true, selector: "natural-select", inputs: { service: "service", optionRequired: "optionRequired", searchField: "searchField", searchOperator: "searchOperator", filter: "filter", disabled: "disabled" }, queries: [{ propertyName: "itemTemplate", first: true, predicate: TemplateRef, descendants: true }], viewQueries: [{ propertyName: "autoTrigger", first: true, predicate: MatAutocompleteTrigger, descendants: true }], usesInheritance: true, ngImport: i0, template: "<!-- 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\u00E9sultats</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 [subscriptSizing]=\"subscriptSizing()\">\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\u00E9lection\"\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\u00E9s\u00E9lectionner\">\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 } @else if (error) {\n <mat-error>{{ error }}</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", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"], dependencies: [{ kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i1.MatAutocomplete, selector: "mat-autocomplete", inputs: ["aria-label", "aria-labelledby", "displayWith", "autoActiveFirstOption", "autoSelectActiveOption", "requireSelection", "panelWidth", "disableRipple", "class", "hideSingleSelectionIndicator"], outputs: ["optionSelected", "opened", "closed", "optionActivated"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i2.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "directive", type: i1.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", inputs: ["matAutocomplete", "matAutocompletePosition", "matAutocompleteConnectedTo", "autocomplete", "matAutocompleteDisabled"], exportAs: ["matAutocompleteTrigger"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "directive", type: i4.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i4.MatPrefix, selector: "[matPrefix], [matIconPrefix], [matTextPrefix]", inputs: ["matTextPrefix"] }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i5.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i6.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i8.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i9.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i9.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i10.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }] });
272
272
  }
273
273
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSelectComponent, decorators: [{
274
274
  type: Component,
@@ -286,7 +286,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
286
286
  MatButtonModule,
287
287
  MatTooltipModule,
288
288
  RouterLink,
289
- ], template: "<!-- 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\u00E9sultats</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\u00E9lection\"\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\u00E9s\u00E9lectionner\">\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 } @else if (error) {\n <mat-error>{{ error }}</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", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"] }]
289
+ ], template: "<!-- 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\u00E9sultats</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 [subscriptSizing]=\"subscriptSizing()\">\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\u00E9lection\"\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\u00E9s\u00E9lectionner\">\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 } @else if (error) {\n <mat-error>{{ error }}</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", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"] }]
290
290
  }], propDecorators: { autoTrigger: [{
291
291
  type: ViewChild,
292
292
  args: [MatAutocompleteTrigger]
@@ -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\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    } @else if (error) {\n        <mat-error>{{ error }}</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 [subscriptSizing]=\"subscriptSizing()\">\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    } @else if (error) {\n        <mat-error>{{ error }}</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"]}