@ecodev/natural 61.1.1 → 61.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/classes/abstract-detail.mjs +3 -2
- package/esm2022/lib/modules/fixed-button/fixed-button.component.mjs +2 -2
- package/esm2022/lib/modules/hierarchic-selector/classes/hierarchic-configuration.mjs +1 -1
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.component.mjs +3 -3
- package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs +12 -20
- package/esm2022/lib/modules/select/abstract-select.component.mjs +11 -17
- package/esm2022/lib/modules/select/select/select.component.mjs +1 -1
- package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +5 -9
- package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +4 -8
- package/fesm2022/ecodev-natural.mjs +29 -49
- package/fesm2022/ecodev-natural.mjs.map +1 -1
- package/lib/modules/fixed-button/fixed-button.component.d.ts +1 -1
- package/lib/modules/hierarchic-selector/classes/hierarchic-configuration.d.ts +0 -4
- package/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.d.ts +0 -1
- package/lib/modules/select/abstract-select.component.d.ts +19 -19
- package/lib/modules/select/select/select.component.d.ts +6 -6
- package/lib/modules/select/select-enum/select-enum.component.d.ts +3 -4
- package/lib/modules/select/select-hierarchic/select-hierarchic.component.d.ts +0 -1
- package/package.json +1 -1
package/esm2022/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Injectable, Injector
|
|
1
|
+
import { inject, Injectable, Injector } from '@angular/core';
|
|
2
2
|
import { intersection } from 'lodash-es';
|
|
3
3
|
import { BehaviorSubject, first, forkJoin } from 'rxjs';
|
|
4
4
|
import { finalize, map } from 'rxjs/operators';
|
|
@@ -25,7 +25,7 @@ export class NaturalHierarchicSelectorService {
|
|
|
25
25
|
*/
|
|
26
26
|
init(config, contextFilter = null, searchVariables = null) {
|
|
27
27
|
this.validateConfiguration(config);
|
|
28
|
-
this.configuration =
|
|
28
|
+
this.configuration = config;
|
|
29
29
|
return this.getList(null, contextFilter, searchVariables).pipe(map(data => this.dataChange.next(data)));
|
|
30
30
|
}
|
|
31
31
|
/**
|
|
@@ -58,7 +58,7 @@ export class NaturalHierarchicSelectorService {
|
|
|
58
58
|
*/
|
|
59
59
|
getList(node = null, contextFilters = null, searchVariables = null) {
|
|
60
60
|
const configurations = this.getContextualizedConfigs(node, contextFilters, searchVariables);
|
|
61
|
-
const observables = configurations.map(c => c.
|
|
61
|
+
const observables = configurations.map(c => c.injectedService.getAll(c.variablesManager));
|
|
62
62
|
// Fire queries, and merge results, transforming apollo items into Node Object.
|
|
63
63
|
return forkJoin(observables).pipe(map(results => {
|
|
64
64
|
const listing = [];
|
|
@@ -74,7 +74,7 @@ export class NaturalHierarchicSelectorService {
|
|
|
74
74
|
}
|
|
75
75
|
countItems(node, contextFilters = null) {
|
|
76
76
|
const configurations = this.getContextualizedConfigs(node, contextFilters, null);
|
|
77
|
-
const observables = configurations.map(c => c.
|
|
77
|
+
const observables = configurations.map(c => c.injectedService.count(c.variablesManager).pipe(first()));
|
|
78
78
|
forkJoin(observables).subscribe(results => {
|
|
79
79
|
const totalItems = results.reduce((total, length) => total + length, 0);
|
|
80
80
|
node.expandable = totalItems > 0;
|
|
@@ -91,10 +91,9 @@ export class NaturalHierarchicSelectorService {
|
|
|
91
91
|
const configs = node ? this.getNextConfigs(node.node.config) : this.configuration;
|
|
92
92
|
const pagination = { pageIndex: 0, pageSize: 999 };
|
|
93
93
|
for (const config of configs) {
|
|
94
|
-
const item = {};
|
|
95
94
|
const contextFilter = this.getFilterByService(config, contextFilters);
|
|
96
95
|
const filter = this.getServiceFilter(node, config, contextFilter, !!searchVariables);
|
|
97
|
-
if (!filter
|
|
96
|
+
if (!filter) {
|
|
98
97
|
continue;
|
|
99
98
|
}
|
|
100
99
|
const variablesManager = new NaturalQueryVariablesManager();
|
|
@@ -103,11 +102,12 @@ export class NaturalHierarchicSelectorService {
|
|
|
103
102
|
if (searchVariables) {
|
|
104
103
|
variablesManager.set('natural-search', searchVariables);
|
|
105
104
|
}
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
105
|
+
const injectedService = this.injector.get(config.service);
|
|
106
|
+
configsAndServices.push({
|
|
107
|
+
configuration: config,
|
|
108
|
+
injectedService: injectedService,
|
|
109
|
+
variablesManager: variablesManager,
|
|
110
|
+
});
|
|
111
111
|
}
|
|
112
112
|
return configsAndServices;
|
|
113
113
|
}
|
|
@@ -147,14 +147,6 @@ export class NaturalHierarchicSelectorService {
|
|
|
147
147
|
}
|
|
148
148
|
return result;
|
|
149
149
|
}
|
|
150
|
-
injectServicesInConfiguration(configurations) {
|
|
151
|
-
for (const config of configurations) {
|
|
152
|
-
if (!config.injectedService) {
|
|
153
|
-
config.injectedService = this.injector.get(config.service);
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
return configurations;
|
|
157
|
-
}
|
|
158
150
|
/**
|
|
159
151
|
* Checks that each configuration.selectableAtKey attribute is unique
|
|
160
152
|
*/
|
|
@@ -248,4 +240,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
|
|
|
248
240
|
type: Injectable,
|
|
249
241
|
args: [{ providedIn: 'root' }]
|
|
250
242
|
}] });
|
|
251
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchic-selector.service.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {Injectable, Injector, inject} from '@angular/core';\nimport {intersection} from 'lodash-es';\nimport {BehaviorSubject, first, forkJoin, Observable} from 'rxjs';\nimport {finalize, map} from 'rxjs/operators';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {HierarchicFlatNode} from '../classes/flat-node';\nimport {\n    NaturalHierarchicConfiguration,\n    NaturalHierarchicServiceConfiguration,\n} from '../classes/hierarchic-configuration';\nimport {\n    HierarchicFilterConfiguration,\n    HierarchicFiltersConfiguration,\n} from '../classes/hierarchic-filters-configuration';\nimport {HierarchicModel, HierarchicModelNode} from '../classes/model-node';\nimport {Literal} from '../../../types/types';\nimport {FilterGroupCondition} from '../../search/classes/graphql-doctrine.types';\n\nexport type OrganizedModelSelection = Record<string, any[]>;\n\ntype ContextualizedConfig = {\n    configuration: NaturalHierarchicServiceConfiguration;\n    variablesManager: NaturalQueryVariablesManager;\n};\n\n@Injectable({providedIn: 'root'})\nexport class NaturalHierarchicSelectorService {\n    private readonly injector = inject(Injector);\n\n    /**\n     * Stores the global result of the tree\n     * This observable contains Node.\n     * When it's updated, the TreeController and TreeFlattener process the new array to generate the flat tree.\n     */\n    public readonly dataChange: BehaviorSubject<HierarchicModelNode[]> = new BehaviorSubject<HierarchicModelNode[]>([]);\n\n    /**\n     * Configuration for relations and selection constraints\n     *\n     * The list should be sorted in the order of the hierarchic (list first parent rules, then child rules)\n     */\n    private configuration: NaturalHierarchicConfiguration[] = [];\n\n    /**\n     * Init component by saving the complete configuration, and then retrieving root elements.\n     * Updates **another** observable (this.dataChange) when data is retrieved.\n     */\n    public init(\n        config: NaturalHierarchicConfiguration[],\n        contextFilter: HierarchicFiltersConfiguration | null = null,\n        searchVariables: QueryVariables | null = null,\n    ): Observable<unknown> {\n        this.validateConfiguration(config);\n        this.configuration = this.injectServicesInConfiguration(config);\n        return this.getList(null, contextFilter, searchVariables).pipe(map(data => this.dataChange.next(data)));\n    }\n\n    /**\n     * Get list of children, considering given FlatNode id as a parent.\n     * Mark loading status individually on nodes.\n     */\n    public loadChildren(\n        flatNode: HierarchicFlatNode,\n        contextFilter: HierarchicFiltersConfiguration | null = null,\n    ): void {\n        // Dont refetch children. Improve performances\n        // Prevents interferences between HierarchicModelNode structure and angular components navigation.\n        // Prevents a bug where grand children were lost if closing root\n        if (flatNode.node.children.length) {\n            return;\n        }\n\n        flatNode.loading = true;\n        this.getList(flatNode, contextFilter)\n            .pipe(finalize(() => (flatNode.loading = false)))\n            .subscribe(items => {\n                flatNode.node.childrenChange.next(items);\n                this.dataChange.next(this.dataChange.value);\n            });\n    }\n\n    public search(searchVariables: QueryVariables, contextFilter: HierarchicFiltersConfiguration | null = null): void {\n        this.getList(null, contextFilter, searchVariables).subscribe(items => {\n            this.dataChange.next(items);\n        });\n    }\n\n    /**\n     * Retrieve elements from the server\n     * Get root elements if node is null, or child elements if node is given\n     */\n    private getList(\n        node: HierarchicFlatNode | null = null,\n        contextFilters: HierarchicFiltersConfiguration | null = null,\n        searchVariables: QueryVariables | null = null,\n    ): Observable<HierarchicModelNode[]> {\n        const configurations = this.getContextualizedConfigs(node, contextFilters, searchVariables);\n        const observables = configurations.map(c => c.configuration.injectedService.getAll(c.variablesManager));\n\n        // Fire queries, and merge results, transforming apollo items into Node Object.\n        return forkJoin(observables).pipe(\n            map(results => {\n                const listing: HierarchicModelNode[] = [];\n\n                // For each result of an observable\n                for (let i = 0; i < results.length; i++) {\n                    // For each item of the result, convert into Node object\n                    for (const item of results[i].items) {\n                        listing.push(this.getOrCreateModelNode(item, configurations[i].configuration));\n                    }\n                }\n\n                return listing;\n            }),\n        );\n    }\n\n    public countItems(node: HierarchicFlatNode, contextFilters: HierarchicFiltersConfiguration | null = null): void {\n        const configurations = this.getContextualizedConfigs(node, contextFilters, null);\n        const observables = configurations.map(c =>\n            c.configuration.injectedService.count(c.variablesManager).pipe(first()),\n        );\n\n        forkJoin(observables).subscribe(results => {\n            const totalItems = results.reduce((total, length) => total + length, 0);\n            node.expandable = totalItems > 0;\n        });\n    }\n\n    private getContextualizedConfigs(\n        node: HierarchicFlatNode | null = null,\n        contextFilters: HierarchicFiltersConfiguration | null = null,\n        searchVariables: QueryVariables | null = null,\n    ): ContextualizedConfig[] {\n        const configsAndServices: ContextualizedConfig[] = [];\n\n        // Considering the whole configuration may cause queries with no/wrong results we have imperatively to avoid !\n        // e.g there are cross dependencies between equipments and taxonomies filters. Both have \"parents\" and \"taxonomies\" filters...\n        // When clicking on a equipment, the configuration of taxonomies with match \"parents\" filter, but use the id of the equipment\n        // To fix this, we should only consider configuration after the one given by the node passed as argument.\n        // That would mean : no child can affect parent.\n        // That would mean : sorting in the configuration have semantic/hierarchy implications\n        const configs = node ? this.getNextConfigs(node.node.config) : this.configuration;\n\n        const pagination = {pageIndex: 0, pageSize: 999};\n\n        for (const config of configs) {\n            const item: ContextualizedConfig = {} as ContextualizedConfig;\n            const contextFilter = this.getFilterByService(config, contextFilters);\n            const filter = this.getServiceFilter(node, config, contextFilter, !!searchVariables);\n\n            if (!filter || !config.injectedService) {\n                continue;\n            }\n\n            const variablesManager = new NaturalQueryVariablesManager();\n\n            variablesManager.set('variables', {filter: filter, pagination: pagination});\n            variablesManager.set('config-filter', {filter: config.filter});\n\n            if (searchVariables) {\n                variablesManager.set('natural-search', searchVariables);\n            }\n\n            // Cast NaturalHierarchicServiceConfiguration because the undefined\n            // injectedServices are filtered earlier and we can validate value\n            item.configuration = config as NaturalHierarchicServiceConfiguration;\n            item.variablesManager = variablesManager;\n            configsAndServices.push(item);\n        }\n\n        return configsAndServices;\n    }\n\n    /**\n     * Return models matching given FlatNodes\n     * Returns a Literal of models grouped by their configuration attribute \"selectableAtKey\"\n     */\n    public toOrganizedSelection(nodes: HierarchicModelNode[]): OrganizedModelSelection {\n        const selection = this.configuration.reduce<Literal>((group, config) => {\n            if (config.selectableAtKey) {\n                group[config.selectableAtKey] = [];\n            }\n            return group;\n        }, {});\n\n        for (const node of nodes) {\n            if (node.config.selectableAtKey) {\n                selection[node.config.selectableAtKey].push(node.model);\n            }\n        }\n\n        return selection;\n    }\n\n    /**\n     * Transforms an OrganizedModelSelection into a list of ModelNodes\n     */\n    public fromOrganizedSelection(organizedModelSelection: OrganizedModelSelection): HierarchicModelNode[] {\n        if (!organizedModelSelection) {\n            return [];\n        }\n\n        const result: HierarchicModelNode[] = [];\n        for (const selectableAtKey of Object.keys(organizedModelSelection)) {\n            const config = this.getConfigurationBySelectableKey(selectableAtKey);\n            if (config) {\n                for (const model of organizedModelSelection[selectableAtKey]) {\n                    result.push(new HierarchicModelNode(model, config));\n                }\n            }\n        }\n        return result;\n    }\n\n    private injectServicesInConfiguration(\n        configurations: NaturalHierarchicConfiguration[],\n    ): NaturalHierarchicConfiguration[] {\n        for (const config of configurations) {\n            if (!config.injectedService) {\n                config.injectedService = this.injector.get(config.service);\n            }\n        }\n\n        return configurations;\n    }\n\n    /**\n     * Checks that each configuration.selectableAtKey attribute is unique\n     */\n    private validateConfiguration(configurations: NaturalHierarchicConfiguration[]): void {\n        const selectableAtKeyAttributes: string[] = [];\n        for (const config of configurations) {\n            if (config.selectableAtKey) {\n                const keyIndex = selectableAtKeyAttributes.indexOf(config.selectableAtKey);\n\n                if (keyIndex === -1 && config.selectableAtKey) {\n                    selectableAtKeyAttributes.push(config.selectableAtKey);\n                }\n\n                // TODO : remove ?\n                // This behavior maybe dangerous in case we re-open hierarchical selector with the last returned config having non-unique\n                // keys\n                if (keyIndex < -1) {\n                    console.warn('Invalid hierarchic configuration : selectableAtKey attribute should be unique');\n                }\n            }\n        }\n    }\n\n    /**\n     * Return configurations setup in the list after the given one\n     */\n    private getNextConfigs(config: NaturalHierarchicConfiguration): NaturalHierarchicConfiguration[] {\n        const configIndex = this.configuration.findIndex(c => c === config);\n        return this.configuration.slice(configIndex);\n    }\n\n    /**\n     * Builds queryVariables filter for children query\n     */\n    private getServiceFilter(\n        flatNode: HierarchicFlatNode | null,\n        config: NaturalHierarchicConfiguration,\n        contextFilter: HierarchicFilterConfiguration['filter'] | null = null,\n        allDeeps = false,\n    ): HierarchicFilterConfiguration['filter'] | null {\n        const fieldCondition: FilterGroupCondition = {};\n\n        // if no parent, filter empty elements\n        if (!flatNode) {\n            if (!config.parentsRelationNames) {\n                return contextFilter ? contextFilter : {};\n            }\n\n            if (!allDeeps) {\n                config.parentsRelationNames.forEach(f => {\n                    fieldCondition[f] = {empty: {}};\n                });\n            }\n        } else {\n            if (!flatNode.node.config.childrenRelationNames || !config.parentsRelationNames) {\n                return null;\n            }\n\n            const matchingFilters = intersection(\n                flatNode.node.config.childrenRelationNames,\n                config.parentsRelationNames,\n            );\n            if (!matchingFilters.length) {\n                return null;\n            }\n            fieldCondition[matchingFilters[0]] = {have: {values: [flatNode.node.model.id]}};\n        }\n\n        const filters = {groups: [{conditions: [fieldCondition]}]};\n\n        // todo : is it right ? shouldn't it be managed with QueryVariablesManager's channels ? ?\n        if (contextFilter) {\n            filters.groups.push(...contextFilter.groups);\n        }\n\n        return filters;\n    }\n\n    /**\n     * Return a context filter applicable to the service for given config\n     *\n     * @param config Applicable config\n     * @param contextFilters List of context filters\n     */\n    private getFilterByService(\n        config: NaturalHierarchicConfiguration,\n        contextFilters: HierarchicFilterConfiguration[] | null,\n    ): HierarchicFilterConfiguration['filter'] | null {\n        if (!contextFilters || !config) {\n            return null;\n        }\n\n        const filter = contextFilters.find(f => f.service === config.service);\n        return filter ? filter.filter : null;\n    }\n\n    /**\n     * Search in configurations.selectableAtKey attribute to find given key and return the configuration\n     */\n    private getConfigurationBySelectableKey(\n        key: NaturalHierarchicConfiguration['selectableAtKey'],\n    ): NaturalHierarchicConfiguration | null {\n        if (!this.configuration) {\n            return null;\n        }\n\n        return this.configuration.find(conf => conf.selectableAtKey === key) || null;\n    }\n\n    private getOrCreateModelNode(\n        item: HierarchicModel,\n        configuration: NaturalHierarchicConfiguration,\n    ): HierarchicModelNode {\n        const node = this.dataChange.value.find(n => n.model.id === item.id && n.model.__typename === item.__typename);\n        return node || new HierarchicModelNode(item, configuration);\n    }\n}\n"]}
|
|
243
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hierarchic-selector.service.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/hierarchic-selector/hierarchic-selector/hierarchic-selector.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAC,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAC,YAAY,EAAC,MAAM,WAAW,CAAC;AACvC,OAAO,EAAC,eAAe,EAAE,KAAK,EAAE,QAAQ,EAAa,MAAM,MAAM,CAAC;AAClE,OAAO,EAAC,QAAQ,EAAE,GAAG,EAAC,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAC,4BAA4B,EAAiB,MAAM,yCAAyC,CAAC;AAOrG,OAAO,EAAkB,mBAAmB,EAAC,MAAM,uBAAuB,CAAC;;AAa3E,MAAM,OAAO,gCAAgC;IACxB,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;IAE7C;;;;OAIG;IACa,UAAU,GAA2C,IAAI,eAAe,CAAwB,EAAE,CAAC,CAAC;IAEpH;;;;OAIG;IACK,aAAa,GAAqC,EAAE,CAAC;IAE7D;;;OAGG;IACI,IAAI,CACP,MAAwC,EACxC,gBAAuD,IAAI,EAC3D,kBAAyC,IAAI;QAE7C,IAAI,CAAC,qBAAqB,CAAC,MAAM,CAAC,CAAC;QACnC,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;QAC5B,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,aAAa,EAAE,eAAe,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAC5G,CAAC;IAED;;;OAGG;IACI,YAAY,CACf,QAA4B,EAC5B,gBAAuD,IAAI;QAE3D,8CAA8C;QAC9C,kGAAkG;QAClG,gEAAgE;QAChE,IAAI,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;YAChC,OAAO;QACX,CAAC;QAED,QAAQ,CAAC,OAAO,GAAG,IAAI,CAAC;QACxB,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,aAAa,CAAC;aAChC,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,QAAQ,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC,CAAC;aAChD,SAAS,CAAC,KAAK,CAAC,EAAE;YACf,QAAQ,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACzC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;IACX,CAAC;IAEM,MAAM,CAAC,eAA+B,EAAE,gBAAuD,IAAI;QACtG,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,aAAa,EAAE,eAAe,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YACjE,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;OAGG;IACK,OAAO,CACX,OAAkC,IAAI,EACtC,iBAAwD,IAAI,EAC5D,kBAAyC,IAAI;QAE7C,MAAM,cAAc,GAAG,IAAI,CAAC,wBAAwB,CAAC,IAAI,EAAE,cAAc,EAAE,eAAe,CAAC,CAAC;QAC5F,MAAM,WAAW,GAAG,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC;QAE1F,+EAA+E;QAC/E,OAAO,QAAQ,CAAC,WAAW,CAAC,CAAC,IAAI,CAC7B,GAAG,CAAC,OAAO,CAAC,EAAE;YACV,MAAM,OAAO,GAA0B,EAAE,CAAC;YAE1C,mCAAmC;YACnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACtC,wDAAwD;gBACxD,KAAK,MAAM,IAAI,IAAI,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC;oBAClC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,oBAAoB,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC;gBACnF,CAAC;YACL,CAAC;YAED,OAAO,OAAO,CAAC;QACnB,CAAC,CAAC,CACL,CAAC;IACN,CAAC;IAEM,UAAU,CAAC,IAAwB,EAAE,iBAAwD,IAAI;QACpG,MAAM,cAAc,GAAG,IAAI,CAAC,wBAAwB,CAAC,IAAI,EAAE,cAAc,EAAE,IAAI,CAAC,CAAC;QACjF,MAAM,WAAW,GAAG,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QAEvG,QAAQ,CAAC,WAAW,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE;YACtC,MAAM,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,CAAC,KAAK,GAAG,MAAM,EAAE,CAAC,CAAC,CAAC;YACxE,IAAI,CAAC,UAAU,GAAG,UAAU,GAAG,CAAC,CAAC;QACrC,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,wBAAwB,CAC5B,OAAkC,IAAI,EACtC,iBAAwD,IAAI,EAC5D,kBAAyC,IAAI;QAE7C,MAAM,kBAAkB,GAA2B,EAAE,CAAC;QAEtD,8GAA8G;QAC9G,8HAA8H;QAC9H,6HAA6H;QAC7H,yGAAyG;QACzG,gDAAgD;QAChD,sFAAsF;QACtF,MAAM,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC;QAElF,MAAM,UAAU,GAAG,EAAC,SAAS,EAAE,CAAC,EAAE,QAAQ,EAAE,GAAG,EAAC,CAAC;QAEjD,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC3B,MAAM,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC;YACtE,MAAM,MAAM,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC,CAAC,eAAe,CAAC,CAAC;YAErF,IAAI,CAAC,MAAM,EAAE,CAAC;gBACV,SAAS;YACb,CAAC;YAED,MAAM,gBAAgB,GAAG,IAAI,4BAA4B,EAAE,CAAC;YAE5D,gBAAgB,CAAC,GAAG,CAAC,WAAW,EAAE,EAAC,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAC,CAAC,CAAC;YAC5E,gBAAgB,CAAC,GAAG,CAAC,eAAe,EAAE,EAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAC,CAAC,CAAC;YAE/D,IAAI,eAAe,EAAE,CAAC;gBAClB,gBAAgB,CAAC,GAAG,CAAC,gBAAgB,EAAE,eAAe,CAAC,CAAC;YAC5D,CAAC;YAED,MAAM,eAAe,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAC1D,kBAAkB,CAAC,IAAI,CAAC;gBACpB,aAAa,EAAE,MAAM;gBACrB,eAAe,EAAE,eAAe;gBAChC,gBAAgB,EAAE,gBAAgB;aACrC,CAAC,CAAC;QACP,CAAC;QAED,OAAO,kBAAkB,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACI,oBAAoB,CAAC,KAA4B;QACpD,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAU,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;YACnE,IAAI,MAAM,CAAC,eAAe,EAAE,CAAC;gBACzB,KAAK,CAAC,MAAM,CAAC,eAAe,CAAC,GAAG,EAAE,CAAC;YACvC,CAAC;YACD,OAAO,KAAK,CAAC;QACjB,CAAC,EAAE,EAAE,CAAC,CAAC;QAEP,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACvB,IAAI,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC;gBAC9B,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5D,CAAC;QACL,CAAC;QAED,OAAO,SAAS,CAAC;IACrB,CAAC;IAED;;OAEG;IACI,sBAAsB,CAAC,uBAAgD;QAC1E,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAC3B,OAAO,EAAE,CAAC;QACd,CAAC;QAED,MAAM,MAAM,GAA0B,EAAE,CAAC;QACzC,KAAK,MAAM,eAAe,IAAI,MAAM,CAAC,IAAI,CAAC,uBAAuB,CAAC,EAAE,CAAC;YACjE,MAAM,MAAM,GAAG,IAAI,CAAC,+BAA+B,CAAC,eAAe,CAAC,CAAC;YACrE,IAAI,MAAM,EAAE,CAAC;gBACT,KAAK,MAAM,KAAK,IAAI,uBAAuB,CAAC,eAAe,CAAC,EAAE,CAAC;oBAC3D,MAAM,CAAC,IAAI,CAAC,IAAI,mBAAmB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC;gBACxD,CAAC;YACL,CAAC;QACL,CAAC;QACD,OAAO,MAAM,CAAC;IAClB,CAAC;IAED;;OAEG;IACK,qBAAqB,CAAC,cAAgD;QAC1E,MAAM,yBAAyB,GAAa,EAAE,CAAC;QAC/C,KAAK,MAAM,MAAM,IAAI,cAAc,EAAE,CAAC;YAClC,IAAI,MAAM,CAAC,eAAe,EAAE,CAAC;gBACzB,MAAM,QAAQ,GAAG,yBAAyB,CAAC,OAAO,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;gBAE3E,IAAI,QAAQ,KAAK,CAAC,CAAC,IAAI,MAAM,CAAC,eAAe,EAAE,CAAC;oBAC5C,yBAAyB,CAAC,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;gBAC3D,CAAC;gBAED,kBAAkB;gBAClB,yHAAyH;gBACzH,OAAO;gBACP,IAAI,QAAQ,GAAG,CAAC,CAAC,EAAE,CAAC;oBAChB,OAAO,CAAC,IAAI,CAAC,+EAA+E,CAAC,CAAC;gBAClG,CAAC;YACL,CAAC;QACL,CAAC;IACL,CAAC;IAED;;OAEG;IACK,cAAc,CAAC,MAAsC;QACzD,MAAM,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,MAAM,CAAC,CAAC;QACpE,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;IACjD,CAAC;IAED;;OAEG;IACK,gBAAgB,CACpB,QAAmC,EACnC,MAAsC,EACtC,gBAAgE,IAAI,EACpE,QAAQ,GAAG,KAAK;QAEhB,MAAM,cAAc,GAAyB,EAAE,CAAC;QAEhD,sCAAsC;QACtC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACZ,IAAI,CAAC,MAAM,CAAC,oBAAoB,EAAE,CAAC;gBAC/B,OAAO,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,CAAC;YAC9C,CAAC;YAED,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACZ,MAAM,CAAC,oBAAoB,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;oBACpC,cAAc,CAAC,CAAC,CAAC,GAAG,EAAC,KAAK,EAAE,EAAE,EAAC,CAAC;gBACpC,CAAC,CAAC,CAAC;YACP,CAAC;QACL,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,qBAAqB,IAAI,CAAC,MAAM,CAAC,oBAAoB,EAAE,CAAC;gBAC9E,OAAO,IAAI,CAAC;YAChB,CAAC;YAED,MAAM,eAAe,GAAG,YAAY,CAChC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,qBAAqB,EAC1C,MAAM,CAAC,oBAAoB,CAC9B,CAAC;YACF,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC;gBAC1B,OAAO,IAAI,CAAC;YAChB,CAAC;YACD,cAAc,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,GAAG,EAAC,IAAI,EAAE,EAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,EAAC,EAAC,CAAC;QACpF,CAAC;QAED,MAAM,OAAO,GAAG,EAAC,MAAM,EAAE,CAAC,EAAC,UAAU,EAAE,CAAC,cAAc,CAAC,EAAC,CAAC,EAAC,CAAC;QAE3D,yFAAyF;QACzF,IAAI,aAAa,EAAE,CAAC;YAChB,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC;QACjD,CAAC;QAED,OAAO,OAAO,CAAC;IACnB,CAAC;IAED;;;;;OAKG;IACK,kBAAkB,CACtB,MAAsC,EACtC,cAAsD;QAEtD,IAAI,CAAC,cAAc,IAAI,CAAC,MAAM,EAAE,CAAC;YAC7B,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,MAAM,MAAM,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,MAAM,CAAC,OAAO,CAAC,CAAC;QACtE,OAAO,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC;IACzC,CAAC;IAED;;OAEG;IACK,+BAA+B,CACnC,GAAsD;QAEtD,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,OAAO,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,KAAK,GAAG,CAAC,IAAI,IAAI,CAAC;IACjF,CAAC;IAEO,oBAAoB,CACxB,IAAqB,EACrB,aAA6C;QAE7C,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,IAAI,CAAC,CAAC,KAAK,CAAC,UAAU,KAAK,IAAI,CAAC,UAAU,CAAC,CAAC;QAC/G,OAAO,IAAI,IAAI,IAAI,mBAAmB,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;IAChE,CAAC;uGA9SQ,gCAAgC;2GAAhC,gCAAgC,cADpB,MAAM;;2FAClB,gCAAgC;kBAD5C,UAAU;mBAAC,EAAC,UAAU,EAAE,MAAM,EAAC","sourcesContent":["import {inject, Injectable, Injector} from '@angular/core';\nimport {intersection} from 'lodash-es';\nimport {BehaviorSubject, first, forkJoin, Observable} from 'rxjs';\nimport {finalize, map} from 'rxjs/operators';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {HierarchicFlatNode} from '../classes/flat-node';\nimport {NaturalHierarchicConfiguration} from '../classes/hierarchic-configuration';\nimport {\n    HierarchicFilterConfiguration,\n    HierarchicFiltersConfiguration,\n} from '../classes/hierarchic-filters-configuration';\nimport {HierarchicModel, HierarchicModelNode} from '../classes/model-node';\nimport {Literal, UntypedModelService} from '../../../types/types';\nimport {FilterGroupCondition} from '../../search/classes/graphql-doctrine.types';\n\nexport type OrganizedModelSelection = Record<string, any[]>;\n\ntype ContextualizedConfig<T extends UntypedModelService = UntypedModelService> = {\n    configuration: NaturalHierarchicConfiguration;\n    injectedService: T;\n    variablesManager: NaturalQueryVariablesManager;\n};\n\n@Injectable({providedIn: 'root'})\nexport class NaturalHierarchicSelectorService {\n    private readonly injector = inject(Injector);\n\n    /**\n     * Stores the global result of the tree\n     * This observable contains Node.\n     * When it's updated, the TreeController and TreeFlattener process the new array to generate the flat tree.\n     */\n    public readonly dataChange: BehaviorSubject<HierarchicModelNode[]> = new BehaviorSubject<HierarchicModelNode[]>([]);\n\n    /**\n     * Configuration for relations and selection constraints\n     *\n     * The list should be sorted in the order of the hierarchic (list first parent rules, then child rules)\n     */\n    private configuration: NaturalHierarchicConfiguration[] = [];\n\n    /**\n     * Init component by saving the complete configuration, and then retrieving root elements.\n     * Updates **another** observable (this.dataChange) when data is retrieved.\n     */\n    public init(\n        config: NaturalHierarchicConfiguration[],\n        contextFilter: HierarchicFiltersConfiguration | null = null,\n        searchVariables: QueryVariables | null = null,\n    ): Observable<unknown> {\n        this.validateConfiguration(config);\n        this.configuration = config;\n        return this.getList(null, contextFilter, searchVariables).pipe(map(data => this.dataChange.next(data)));\n    }\n\n    /**\n     * Get list of children, considering given FlatNode id as a parent.\n     * Mark loading status individually on nodes.\n     */\n    public loadChildren(\n        flatNode: HierarchicFlatNode,\n        contextFilter: HierarchicFiltersConfiguration | null = null,\n    ): void {\n        // Dont refetch children. Improve performances\n        // Prevents interferences between HierarchicModelNode structure and angular components navigation.\n        // Prevents a bug where grand children were lost if closing root\n        if (flatNode.node.children.length) {\n            return;\n        }\n\n        flatNode.loading = true;\n        this.getList(flatNode, contextFilter)\n            .pipe(finalize(() => (flatNode.loading = false)))\n            .subscribe(items => {\n                flatNode.node.childrenChange.next(items);\n                this.dataChange.next(this.dataChange.value);\n            });\n    }\n\n    public search(searchVariables: QueryVariables, contextFilter: HierarchicFiltersConfiguration | null = null): void {\n        this.getList(null, contextFilter, searchVariables).subscribe(items => {\n            this.dataChange.next(items);\n        });\n    }\n\n    /**\n     * Retrieve elements from the server\n     * Get root elements if node is null, or child elements if node is given\n     */\n    private getList(\n        node: HierarchicFlatNode | null = null,\n        contextFilters: HierarchicFiltersConfiguration | null = null,\n        searchVariables: QueryVariables | null = null,\n    ): Observable<HierarchicModelNode[]> {\n        const configurations = this.getContextualizedConfigs(node, contextFilters, searchVariables);\n        const observables = configurations.map(c => c.injectedService.getAll(c.variablesManager));\n\n        // Fire queries, and merge results, transforming apollo items into Node Object.\n        return forkJoin(observables).pipe(\n            map(results => {\n                const listing: HierarchicModelNode[] = [];\n\n                // For each result of an observable\n                for (let i = 0; i < results.length; i++) {\n                    // For each item of the result, convert into Node object\n                    for (const item of results[i].items) {\n                        listing.push(this.getOrCreateModelNode(item, configurations[i].configuration));\n                    }\n                }\n\n                return listing;\n            }),\n        );\n    }\n\n    public countItems(node: HierarchicFlatNode, contextFilters: HierarchicFiltersConfiguration | null = null): void {\n        const configurations = this.getContextualizedConfigs(node, contextFilters, null);\n        const observables = configurations.map(c => c.injectedService.count(c.variablesManager).pipe(first()));\n\n        forkJoin(observables).subscribe(results => {\n            const totalItems = results.reduce((total, length) => total + length, 0);\n            node.expandable = totalItems > 0;\n        });\n    }\n\n    private getContextualizedConfigs(\n        node: HierarchicFlatNode | null = null,\n        contextFilters: HierarchicFiltersConfiguration | null = null,\n        searchVariables: QueryVariables | null = null,\n    ): ContextualizedConfig[] {\n        const configsAndServices: ContextualizedConfig[] = [];\n\n        // Considering the whole configuration may cause queries with no/wrong results we have imperatively to avoid !\n        // e.g there are cross dependencies between equipments and taxonomies filters. Both have \"parents\" and \"taxonomies\" filters...\n        // When clicking on a equipment, the configuration of taxonomies with match \"parents\" filter, but use the id of the equipment\n        // To fix this, we should only consider configuration after the one given by the node passed as argument.\n        // That would mean : no child can affect parent.\n        // That would mean : sorting in the configuration have semantic/hierarchy implications\n        const configs = node ? this.getNextConfigs(node.node.config) : this.configuration;\n\n        const pagination = {pageIndex: 0, pageSize: 999};\n\n        for (const config of configs) {\n            const contextFilter = this.getFilterByService(config, contextFilters);\n            const filter = this.getServiceFilter(node, config, contextFilter, !!searchVariables);\n\n            if (!filter) {\n                continue;\n            }\n\n            const variablesManager = new NaturalQueryVariablesManager();\n\n            variablesManager.set('variables', {filter: filter, pagination: pagination});\n            variablesManager.set('config-filter', {filter: config.filter});\n\n            if (searchVariables) {\n                variablesManager.set('natural-search', searchVariables);\n            }\n\n            const injectedService = this.injector.get(config.service);\n            configsAndServices.push({\n                configuration: config,\n                injectedService: injectedService,\n                variablesManager: variablesManager,\n            });\n        }\n\n        return configsAndServices;\n    }\n\n    /**\n     * Return models matching given FlatNodes\n     * Returns a Literal of models grouped by their configuration attribute \"selectableAtKey\"\n     */\n    public toOrganizedSelection(nodes: HierarchicModelNode[]): OrganizedModelSelection {\n        const selection = this.configuration.reduce<Literal>((group, config) => {\n            if (config.selectableAtKey) {\n                group[config.selectableAtKey] = [];\n            }\n            return group;\n        }, {});\n\n        for (const node of nodes) {\n            if (node.config.selectableAtKey) {\n                selection[node.config.selectableAtKey].push(node.model);\n            }\n        }\n\n        return selection;\n    }\n\n    /**\n     * Transforms an OrganizedModelSelection into a list of ModelNodes\n     */\n    public fromOrganizedSelection(organizedModelSelection: OrganizedModelSelection): HierarchicModelNode[] {\n        if (!organizedModelSelection) {\n            return [];\n        }\n\n        const result: HierarchicModelNode[] = [];\n        for (const selectableAtKey of Object.keys(organizedModelSelection)) {\n            const config = this.getConfigurationBySelectableKey(selectableAtKey);\n            if (config) {\n                for (const model of organizedModelSelection[selectableAtKey]) {\n                    result.push(new HierarchicModelNode(model, config));\n                }\n            }\n        }\n        return result;\n    }\n\n    /**\n     * Checks that each configuration.selectableAtKey attribute is unique\n     */\n    private validateConfiguration(configurations: NaturalHierarchicConfiguration[]): void {\n        const selectableAtKeyAttributes: string[] = [];\n        for (const config of configurations) {\n            if (config.selectableAtKey) {\n                const keyIndex = selectableAtKeyAttributes.indexOf(config.selectableAtKey);\n\n                if (keyIndex === -1 && config.selectableAtKey) {\n                    selectableAtKeyAttributes.push(config.selectableAtKey);\n                }\n\n                // TODO : remove ?\n                // This behavior maybe dangerous in case we re-open hierarchical selector with the last returned config having non-unique\n                // keys\n                if (keyIndex < -1) {\n                    console.warn('Invalid hierarchic configuration : selectableAtKey attribute should be unique');\n                }\n            }\n        }\n    }\n\n    /**\n     * Return configurations setup in the list after the given one\n     */\n    private getNextConfigs(config: NaturalHierarchicConfiguration): NaturalHierarchicConfiguration[] {\n        const configIndex = this.configuration.findIndex(c => c === config);\n        return this.configuration.slice(configIndex);\n    }\n\n    /**\n     * Builds queryVariables filter for children query\n     */\n    private getServiceFilter(\n        flatNode: HierarchicFlatNode | null,\n        config: NaturalHierarchicConfiguration,\n        contextFilter: HierarchicFilterConfiguration['filter'] | null = null,\n        allDeeps = false,\n    ): HierarchicFilterConfiguration['filter'] | null {\n        const fieldCondition: FilterGroupCondition = {};\n\n        // if no parent, filter empty elements\n        if (!flatNode) {\n            if (!config.parentsRelationNames) {\n                return contextFilter ? contextFilter : {};\n            }\n\n            if (!allDeeps) {\n                config.parentsRelationNames.forEach(f => {\n                    fieldCondition[f] = {empty: {}};\n                });\n            }\n        } else {\n            if (!flatNode.node.config.childrenRelationNames || !config.parentsRelationNames) {\n                return null;\n            }\n\n            const matchingFilters = intersection(\n                flatNode.node.config.childrenRelationNames,\n                config.parentsRelationNames,\n            );\n            if (!matchingFilters.length) {\n                return null;\n            }\n            fieldCondition[matchingFilters[0]] = {have: {values: [flatNode.node.model.id]}};\n        }\n\n        const filters = {groups: [{conditions: [fieldCondition]}]};\n\n        // todo : is it right ? shouldn't it be managed with QueryVariablesManager's channels ? ?\n        if (contextFilter) {\n            filters.groups.push(...contextFilter.groups);\n        }\n\n        return filters;\n    }\n\n    /**\n     * Return a context filter applicable to the service for given config\n     *\n     * @param config Applicable config\n     * @param contextFilters List of context filters\n     */\n    private getFilterByService(\n        config: NaturalHierarchicConfiguration,\n        contextFilters: HierarchicFilterConfiguration[] | null,\n    ): HierarchicFilterConfiguration['filter'] | null {\n        if (!contextFilters || !config) {\n            return null;\n        }\n\n        const filter = contextFilters.find(f => f.service === config.service);\n        return filter ? filter.filter : null;\n    }\n\n    /**\n     * Search in configurations.selectableAtKey attribute to find given key and return the configuration\n     */\n    private getConfigurationBySelectableKey(\n        key: NaturalHierarchicConfiguration['selectableAtKey'],\n    ): NaturalHierarchicConfiguration | null {\n        if (!this.configuration) {\n            return null;\n        }\n\n        return this.configuration.find(conf => conf.selectableAtKey === key) || null;\n    }\n\n    private getOrCreateModelNode(\n        item: HierarchicModel,\n        configuration: NaturalHierarchicConfiguration,\n    ): HierarchicModelNode {\n        const node = this.dataChange.value.find(n => n.model.id === item.id && n.model.__typename === item.__typename);\n        return node || new HierarchicModelNode(item, configuration);\n    }\n}\n"]}
|
|
@@ -1,9 +1,8 @@
|
|
|
1
1
|
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
2
|
-
import { Directive, EventEmitter,
|
|
3
|
-
import { FormControl, FormControlDirective, FormControlName, Validators, } from '@angular/forms';
|
|
2
|
+
import { Directive, EventEmitter, inject, Input, Output } from '@angular/core';
|
|
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";
|
|
6
|
-
import * as i1 from "@angular/forms";
|
|
7
6
|
/**
|
|
8
7
|
* This will completely ignore internal formControl and instead use the one from the component
|
|
9
8
|
* which comes from outside of this component. This basically allows us to **not** depend on
|
|
@@ -25,7 +24,6 @@ class ExternalFormControlMatcher extends ErrorStateMatcher {
|
|
|
25
24
|
}
|
|
26
25
|
}
|
|
27
26
|
export class AbstractSelect {
|
|
28
|
-
ngControl;
|
|
29
27
|
placeholder;
|
|
30
28
|
/**
|
|
31
29
|
* Mat-hint
|
|
@@ -74,12 +72,12 @@ export class AbstractSelect {
|
|
|
74
72
|
/**
|
|
75
73
|
* Contains internal representation for current selection AND searched text (for autocomplete)
|
|
76
74
|
*
|
|
77
|
-
* It is **not** necessarily `
|
|
75
|
+
* It is **not** necessarily `TValue | null`.
|
|
78
76
|
*
|
|
79
|
-
* - NaturalSelectComponent: `string |
|
|
80
|
-
* only when `optionRequired` is false, so most of the time it is `
|
|
77
|
+
* - NaturalSelectComponent: `string | TValue | null`. We allow `string`
|
|
78
|
+
* only when `optionRequired` is false, so most of the time it is `TValue | null`.
|
|
81
79
|
* - NaturalSelectHierarchicComponent: `string | null`.
|
|
82
|
-
* - NaturalSelectEnumComponent: `
|
|
80
|
+
* - NaturalSelectEnumComponent: `TValue | null`.
|
|
83
81
|
*
|
|
84
82
|
* In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :
|
|
85
83
|
* - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed
|
|
@@ -97,8 +95,8 @@ export class AbstractSelect {
|
|
|
97
95
|
*/
|
|
98
96
|
onTouched;
|
|
99
97
|
matcher;
|
|
100
|
-
|
|
101
|
-
|
|
98
|
+
ngControl = inject(NgControl, { optional: true, self: true });
|
|
99
|
+
constructor() {
|
|
102
100
|
if (this.ngControl) {
|
|
103
101
|
this.ngControl.valueAccessor = this;
|
|
104
102
|
}
|
|
@@ -193,17 +191,13 @@ export class AbstractSelect {
|
|
|
193
191
|
}
|
|
194
192
|
this.internalCtrl.updateValueAndValidity();
|
|
195
193
|
}
|
|
196
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: AbstractSelect, deps: [
|
|
194
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: AbstractSelect, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
197
195
|
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.3", type: AbstractSelect, isStandalone: true, inputs: { placeholder: "placeholder", hint: "hint", required: "required", navigateTo: "navigateTo", clearLabel: "clearLabel", showIcon: "showIcon", icon: "icon", displayWith: "displayWith", disabled: "disabled" }, outputs: { selectionChange: "selectionChange", blur: "blur" }, ngImport: i0 });
|
|
198
196
|
}
|
|
199
197
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: AbstractSelect, decorators: [{
|
|
200
198
|
type: Directive,
|
|
201
199
|
args: [{ standalone: true }]
|
|
202
|
-
}], ctorParameters: () => [
|
|
203
|
-
type: Optional
|
|
204
|
-
}, {
|
|
205
|
-
type: Self
|
|
206
|
-
}] }], propDecorators: { placeholder: [{
|
|
200
|
+
}], ctorParameters: () => [], propDecorators: { placeholder: [{
|
|
207
201
|
type: Input
|
|
208
202
|
}], hint: [{
|
|
209
203
|
type: Input
|
|
@@ -226,4 +220,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
|
|
|
226
220
|
}], disabled: [{
|
|
227
221
|
type: Input
|
|
228
222
|
}] } });
|
|
229
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-select.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/select/abstract-select.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,qBAAqB,EAAC,MAAM,uBAAuB,CAAC;AAC5D,OAAO,EAAC,SAAS,EAAW,YAAY,EAAE,KAAK,EAAU,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAC,MAAM,eAAe,CAAC;AACtG,OAAO,EAGH,WAAW,EACX,oBAAoB,EACpB,eAAe,EAEf,UAAU,GACb,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAC,iBAAiB,EAAC,MAAM,wBAAwB,CAAC;;;AAEzD;;;;;GAKG;AACH,MAAM,0BAAiC,SAAQ,iBAAiB;IACxB;IAApC,YAAoC,SAA+B;QAC/D,KAAK,EAAE,CAAC;QADwB,cAAS,GAAT,SAAS,CAAsB;IAEnE,CAAC;IAEe,YAAY;QACxB,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,EAAE,OAAO,IAAI,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC;QACtF,IAAI,YAAY,EAAE,CAAC;YACf,OAAO,CAAC,CAAC,CAAC,YAAY,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,IAAI,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC;QACnF,CAAC;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;CACJ;AAGD,MAAM,OAAgB,cAAc;IAyFuB;IAxFvC,WAAW,CAAU;IAErC;;OAEG;IACa,IAAI,GAAkB,IAAI,CAAC;IAE3C;;OAEG;IACH,IACW,QAAQ,CAAC,KAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAC9C,IAAI,CAAC,aAAa,EAAE,CAAC;IACzB,CAAC;IAED,IAAW,QAAQ;QACf,OAAO,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC;IAC5B,CAAC;IAEO,SAAS,CAAsB;IAEvC;;OAEG;IACa,UAAU,CAAyB;IAEnD;;OAEG;IACa,UAAU,CAAU;IAEpC;;OAEG;IACa,QAAQ,GAAG,IAAI,CAAC;IAEhC;;OAEG;IACa,IAAI,GAAG,QAAQ,CAAC;IAEhC;;OAEG;IACa,WAAW,CAA8B;IAEzD;;OAEG;IACuB,eAAe,GAAG,IAAI,YAAY,EAAY,CAAC;IAEzE;;OAEG;IACH,4DAA4D;IAClC,IAAI,GAAG,IAAI,YAAY,EAAQ,CAAC;IAE1D;;;;;;;;;;;;;OAaG;IACa,YAAY,GAAG,IAAI,WAAW,CAAW,IAAI,CAAC,CAAC;IAE/D;;;OAGG;IACI,QAAQ,CAA4B;IAE3C;;;OAGG;IACI,SAAS,CAAc;IAEd,OAAO,CAAmC;IAE1D,YAAuD,SAA2B;QAA3B,cAAS,GAAT,SAAS,CAAkB;QAC9E,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,SAAS,CAAC,aAAa,GAAG,IAAI,CAAC;QACxC,CAAC;QAED,IAAI,CAAC,OAAO,GAAG,IAAI,0BAA0B,CAAC,IAAI,CAAC,CAAC;IACxD,CAAC;IAEM,SAAS;QACZ,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,aAAa,EAAE,CAAC;QACzB,CAAC;IACL,CAAC;IAEM,UAAU,CAAC,KAAe;QAC7B,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACtC,CAAC;IAEM,QAAQ;QACX,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,YAAY,oBAAoB,IAAI,IAAI,CAAC,SAAS,YAAY,eAAe,CAAC;QAC/G,IAAI,UAAU,IAAI,OAAO,IAAI,CAAC,SAAS,KAAK,WAAW,EAAE,CAAC;YACtD,OAAO,CAAC,IAAI,CAAC,yFAAyF,CAAC,CAAC;QAC5G,CAAC;IACL,CAAC;IAED;;OAEG;IACH,IACW,QAAQ,CAAC,QAAiB;QACjC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC;IACxE,CAAC;IAEM,gBAAgB,CAAC,EAA4B;QAChD,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACvB,CAAC;IAEM,iBAAiB,CAAC,EAAc;QACnC,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACxB,CAAC;IAID;;;OAGG;IACI,KAAK;QACR,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED;;OAEG;IACI,MAAM;QACT,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;IACrB,CAAC;IAED;;OAEG;IACI,cAAc,CAAC,KAAe;QACjC,6FAA6F;QAC7F,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAChB,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACzB,CAAC;QAED,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACrC,CAAC;IAEM,gBAAgB,CAAC,UAAmB;QACvC,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;IAC/B,CAAC;IAEM,eAAe;QAClB,OAAO,IAAI,CAAC,YAAY,EAAE,OAAO,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;IACxF,CAAC;IAEM,KAAK;QACR,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,SAAS,EAAE,CAAC;QACrB,CAAC;IACL,CAAC;IAEM,gBAAgB;QACnB,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC;QAEtF,OAAO,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;IACxC,CAAC;IAED;;OAEG;IACK,aAAa;QACjB,sCAAsC;QACtC,MAAM,mBAAmB,GAAG,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC,EAAqB,CAAC,EAAE,QAAQ,CAAC;QAEnG,sDAAsD;QACtD,MAAM,iBAAiB,GAAG,OAAO,IAAI,CAAC,SAAS,KAAK,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,mBAAmB,CAAC;QAEvG,oCAAoC;QACpC,MAAM,qBAAqB,GAAG,IAAI,CAAC,YAAY,EAAE,SAAS,EAAE,CAAC,EAAqB,CAAC,EAAE,QAAQ,CAAC;QAE9F,gEAAgE;QAChE,IAAI,qBAAqB,KAAK,iBAAiB,EAAE,CAAC;YAC9C,OAAO;QACX,CAAC;QAED,wBAAwB;QACxB,IAAI,iBAAiB,EAAE,CAAC;YACpB,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QACzD,CAAC;aAAM,CAAC;YACJ,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE,CAAC;QACxC,CAAC;QAED,IAAI,CAAC,YAAY,CAAC,sBAAsB,EAAE,CAAC;IAC/C,CAAC;uGA/MiB,cAAc;2FAAd,cAAc;;2FAAd,cAAc;kBADnC,SAAS;mBAAC,EAAC,UAAU,EAAE,IAAI,EAAC;;0BA0FL,QAAQ;;0BAAI,IAAI;yCAxFpB,WAAW;sBAA1B,KAAK;gBAKU,IAAI;sBAAnB,KAAK;gBAMK,QAAQ;sBADlB,KAAK;gBAeU,UAAU;sBAAzB,KAAK;gBAKU,UAAU;sBAAzB,KAAK;gBAKU,QAAQ;sBAAvB,KAAK;gBAKU,IAAI;sBAAnB,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAKoB,eAAe;sBAAxC,MAAM;gBAMmB,IAAI;sBAA7B,MAAM;gBA6DI,QAAQ;sBADlB,KAAK","sourcesContent":["import {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {Directive, DoCheck, EventEmitter, Input, OnInit, Optional, Output, Self} from '@angular/core';\nimport {\n    AbstractControl,\n    ControlValueAccessor,\n    FormControl,\n    FormControlDirective,\n    FormControlName,\n    NgControl,\n    Validators,\n} from '@angular/forms';\nimport {ErrorStateMatcher} from '@angular/material/core';\n\n/**\n * This will completely ignore internal formControl and instead use the one from the component\n * which comes from outside of this component. This basically allows us to **not** depend on\n * touched status propagation between outside and inside world, and thus get rid of our legacy\n * custom FormControl class (\"NaturalFormControl\").\n */\nclass ExternalFormControlMatcher<T, I> extends ErrorStateMatcher {\n    public constructor(private readonly component: AbstractSelect<T, I>) {\n        super();\n    }\n\n    public override isErrorState(): boolean {\n        const externalCtrl = this.component.ngControl?.control || this.component.internalCtrl;\n        if (externalCtrl) {\n            return !!(externalCtrl.errors && (externalCtrl.touched || externalCtrl.dirty));\n        }\n\n        return false;\n    }\n}\n\n@Directive({standalone: true})\nexport abstract class AbstractSelect<V, I> implements OnInit, ControlValueAccessor, DoCheck {\n    @Input() public placeholder?: string;\n\n    /**\n     * Mat-hint\n     */\n    @Input() public hint: string | null = null;\n\n    /**\n     * If the field is required\n     */\n    @Input()\n    public set required(value: boolean) {\n        this._required = coerceBooleanProperty(value);\n        this.applyRequired();\n    }\n\n    public get required(): boolean {\n        return !!this._required;\n    }\n\n    private _required: boolean | undefined;\n\n    /**\n     * Add a suffix button that is a link to given destination\n     */\n    @Input() public navigateTo?: any[] | string | null;\n\n    /**\n     * If provided cause a new clear button to appear\n     */\n    @Input() public clearLabel?: string;\n\n    /**\n     * Whether to show the search icon\n     */\n    @Input() public showIcon = true;\n\n    /**\n     * Icon name\n     */\n    @Input() public icon = 'search';\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: V | null) => string;\n\n    /**\n     * Emit the selected value whenever it changes\n     */\n    @Output() public readonly selectionChange = new EventEmitter<V | null>();\n\n    /**\n     * Emits when internal input is blurred\n     */\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() public readonly blur = new EventEmitter<void>();\n\n    /**\n     * Contains internal representation for current selection AND searched text (for autocomplete)\n     *\n     * It is **not** necessarily `V | null`.\n     *\n     * - NaturalSelectComponent: `string | V | null`. We allow `string`\n     *   only when `optionRequired` is false, so most of the time it is `V | null`.\n     * - NaturalSelectHierarchicComponent: `string | null`.\n     * - NaturalSelectEnumComponent: `V | null`.\n     *\n     * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :\n     *  - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed\n     *  - Dirty status (changed value) means we are in search/autocomplete mode\n     */\n    public readonly internalCtrl = new FormControl<I | null>(null);\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onChange?: (item: V | null) => void;\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onTouched?: () => void;\n\n    public readonly matcher: ExternalFormControlMatcher<V, I>;\n\n    public constructor(@Optional() @Self() public readonly ngControl: NgControl | null) {\n        if (this.ngControl) {\n            this.ngControl.valueAccessor = this;\n        }\n\n        this.matcher = new ExternalFormControlMatcher(this);\n    }\n\n    public ngDoCheck(): void {\n        if (this.ngControl) {\n            this.applyRequired();\n        }\n    }\n\n    public writeValue(value: I | null): void {\n        this.internalCtrl.setValue(value);\n    }\n\n    public ngOnInit(): void {\n        const isReactive = this.ngControl instanceof FormControlDirective || this.ngControl instanceof FormControlName;\n        if (isReactive && typeof this._required !== 'undefined') {\n            console.warn('<natural-select-*> should not be used as ReactiveForm and with the [required] attribute');\n        }\n    }\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public registerOnChange(fn: (item: V | null) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public abstract getDisplayFn(): (item: V | null) => string;\n\n    /**\n     * Commit the model to null\n     * Emit and event to update the model\n     */\n    public clear(): void {\n        this.internalCtrl.setValue(null);\n        this.propagateValue(null);\n    }\n\n    /**\n     * If input is dirty (search running) restore to model value\n     */\n    public onBlur(): void {\n        this.touch();\n        this.blur.emit();\n    }\n\n    /**\n     * Commit the model change\n     */\n    public propagateValue(value: V | null): void {\n        // before selectionChange to allow formControl to update before change is effectively emitted\n        if (this.onChange) {\n            this.onChange(value);\n        }\n\n        this.selectionChange.emit(value);\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    public touch(): void {\n        if (this.onTouched) {\n            this.onTouched();\n        }\n    }\n\n    public hasRequiredError(): boolean {\n        const control = this.ngControl?.control ? this.ngControl?.control : this.internalCtrl;\n\n        return control.hasError('required');\n    }\n\n    /**\n     * Apply Validators.required on the internal form, based on ngControl or [required] attribute, giving priority to attribute.\n     */\n    private applyRequired(): void {\n        // Required status on parent validator\n        const outerRequiredStatus = this?.ngControl?.control?.validator?.({} as AbstractControl)?.required;\n\n        // Wanted required status, giving priority to template\n        const newRequiredStatus = typeof this._required !== 'undefined' ? this._required : outerRequiredStatus;\n\n        // Actual internal validation status\n        const currentRequiredStatus = this.internalCtrl?.validator?.({} as AbstractControl)?.required;\n\n        // If wanted status is similar to actual status, stop everything\n        if (currentRequiredStatus === newRequiredStatus) {\n            return;\n        }\n\n        // Apply only if changed\n        if (newRequiredStatus) {\n            this.internalCtrl.setValidators(Validators.required);\n        } else {\n            this.internalCtrl.clearValidators();\n        }\n\n        this.internalCtrl.updateValueAndValidity();\n    }\n}\n"]}
|
|
223
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-select.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/select/abstract-select.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {Directive, DoCheck, EventEmitter, inject, Input, OnInit, Output} from '@angular/core';\nimport {\n    AbstractControl,\n    ControlValueAccessor,\n    FormControl,\n    FormControlDirective,\n    FormControlName,\n    NgControl,\n    Validators,\n} from '@angular/forms';\nimport {ErrorStateMatcher} from '@angular/material/core';\n\n/**\n * This will completely ignore internal formControl and instead use the one from the component\n * which comes from outside of this component. This basically allows us to **not** depend on\n * touched status propagation between outside and inside world, and thus get rid of our legacy\n * custom FormControl class (\"NaturalFormControl\").\n */\nclass ExternalFormControlMatcher<TValue, TInput> extends ErrorStateMatcher {\n    public constructor(private readonly component: AbstractSelect<TValue, TInput>) {\n        super();\n    }\n\n    public override isErrorState(): boolean {\n        const externalCtrl = this.component.ngControl?.control || this.component.internalCtrl;\n        if (externalCtrl) {\n            return !!(externalCtrl.errors && (externalCtrl.touched || externalCtrl.dirty));\n        }\n\n        return false;\n    }\n}\n\n@Directive({standalone: true})\nexport abstract class AbstractSelect<TValue, TInput> implements OnInit, ControlValueAccessor, DoCheck {\n    @Input() public placeholder?: string;\n\n    /**\n     * Mat-hint\n     */\n    @Input() public hint: string | null = null;\n\n    /**\n     * If the field is required\n     */\n    @Input()\n    public set required(value: boolean) {\n        this._required = coerceBooleanProperty(value);\n        this.applyRequired();\n    }\n\n    public get required(): boolean {\n        return !!this._required;\n    }\n\n    private _required: boolean | undefined;\n\n    /**\n     * Add a suffix button that is a link to given destination\n     */\n    @Input() public navigateTo?: any[] | string | null;\n\n    /**\n     * If provided cause a new clear button to appear\n     */\n    @Input() public clearLabel?: string;\n\n    /**\n     * Whether to show the search icon\n     */\n    @Input() public showIcon = true;\n\n    /**\n     * Icon name\n     */\n    @Input() public icon = 'search';\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: TValue | null) => string;\n\n    /**\n     * Emit the selected value whenever it changes\n     */\n    @Output() public readonly selectionChange = new EventEmitter<TValue | null>();\n\n    /**\n     * Emits when internal input is blurred\n     */\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() public readonly blur = new EventEmitter<void>();\n\n    /**\n     * Contains internal representation for current selection AND searched text (for autocomplete)\n     *\n     * It is **not** necessarily `TValue | null`.\n     *\n     * - NaturalSelectComponent: `string | TValue | null`. We allow `string`\n     *   only when `optionRequired` is false, so most of the time it is `TValue | null`.\n     * - NaturalSelectHierarchicComponent: `string | null`.\n     * - NaturalSelectEnumComponent: `TValue | null`.\n     *\n     * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :\n     *  - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed\n     *  - Dirty status (changed value) means we are in search/autocomplete mode\n     */\n    public readonly internalCtrl = new FormControl<TInput | null>(null);\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onChange?: (item: TValue | null) => void;\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onTouched?: () => void;\n\n    public readonly matcher: ExternalFormControlMatcher<TValue, TInput>;\n    public readonly ngControl = inject(NgControl, {optional: true, self: true});\n\n    public constructor() {\n        if (this.ngControl) {\n            this.ngControl.valueAccessor = this;\n        }\n\n        this.matcher = new ExternalFormControlMatcher(this);\n    }\n\n    public ngDoCheck(): void {\n        if (this.ngControl) {\n            this.applyRequired();\n        }\n    }\n\n    public writeValue(value: TInput | null): void {\n        this.internalCtrl.setValue(value);\n    }\n\n    public ngOnInit(): void {\n        const isReactive = this.ngControl instanceof FormControlDirective || this.ngControl instanceof FormControlName;\n        if (isReactive && typeof this._required !== 'undefined') {\n            console.warn('<natural-select-*> should not be used as ReactiveForm and with the [required] attribute');\n        }\n    }\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public registerOnChange(fn: (item: TValue | null) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public abstract getDisplayFn(): (item: TValue | null) => string;\n\n    /**\n     * Commit the model to null\n     * Emit and event to update the model\n     */\n    public clear(): void {\n        this.internalCtrl.setValue(null);\n        this.propagateValue(null);\n    }\n\n    /**\n     * If input is dirty (search running) restore to model value\n     */\n    public onBlur(): void {\n        this.touch();\n        this.blur.emit();\n    }\n\n    /**\n     * Commit the model change\n     */\n    public propagateValue(value: TValue | null): void {\n        // before selectionChange to allow formControl to update before change is effectively emitted\n        if (this.onChange) {\n            this.onChange(value);\n        }\n\n        this.selectionChange.emit(value);\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    public touch(): void {\n        if (this.onTouched) {\n            this.onTouched();\n        }\n    }\n\n    public hasRequiredError(): boolean {\n        const control = this.ngControl?.control ? this.ngControl?.control : this.internalCtrl;\n\n        return control.hasError('required');\n    }\n\n    /**\n     * Apply Validators.required on the internal form, based on ngControl or [required] attribute, giving priority to attribute.\n     */\n    private applyRequired(): void {\n        // Required status on parent validator\n        const outerRequiredStatus = this?.ngControl?.control?.validator?.({} as AbstractControl)?.required;\n\n        // Wanted required status, giving priority to template\n        const newRequiredStatus = typeof this._required !== 'undefined' ? this._required : outerRequiredStatus;\n\n        // Actual internal validation status\n        const currentRequiredStatus = this.internalCtrl?.validator?.({} as AbstractControl)?.required;\n\n        // If wanted status is similar to actual status, stop everything\n        if (currentRequiredStatus === newRequiredStatus) {\n            return;\n        }\n\n        // Apply only if changed\n        if (newRequiredStatus) {\n            this.internalCtrl.setValidators(Validators.required);\n        } else {\n            this.internalCtrl.clearValidators();\n        }\n\n        this.internalCtrl.updateValueAndValidity();\n    }\n}\n"]}
|
|
@@ -307,4 +307,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
|
|
|
307
307
|
}], disabled: [{
|
|
308
308
|
type: Input
|
|
309
309
|
}] } });
|
|
310
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/select/select/select.component.ts","../../../../../../../projects/natural/src/lib/modules/select/select/select.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEH,SAAS,EACT,YAAY,EACZ,UAAU,EACV,MAAM,EACN,KAAK,EAEL,WAAW,EACX,SAAS,GACZ,MAAM,eAAe,CAAC;AACvB,OAAO,EAAuB,WAAW,EAAE,mBAAmB,EAAC,MAAM,gBAAgB,CAAC;AACtF,OAAO,EAAC,qBAAqB,EAAE,sBAAsB,EAAC,MAAM,gCAAgC,CAAC;AAC7F,OAAO,EAAC,KAAK,EAAC,MAAM,WAAW,CAAC;AAEhC,OAAO,EAAC,YAAY,EAAE,oBAAoB,EAAE,QAAQ,EAAE,GAAG,EAAC,MAAM,gBAAgB,CAAC;AAEjF,OAAO,EAAC,4BAA4B,EAAiB,MAAM,yCAAyC,CAAC;AAGrG,OAAO,EAAC,cAAc,EAAC,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAC,UAAU,EAAC,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAC,gBAAgB,EAAC,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAC,eAAe,EAAC,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAC,wBAAwB,EAAC,MAAM,oCAAoC,CAAC;AAC5E,OAAO,EAAC,oBAAoB,EAAC,MAAM,2BAA2B,CAAC;AAC/D,OAAO,EAAC,aAAa,EAAC,MAAM,wBAAwB,CAAC;AACrD,OAAO,EAAC,cAAc,EAAC,MAAM,yBAAyB,CAAC;AACvD,OAAO,EAAC,kBAAkB,EAAC,MAAM,8BAA8B,CAAC;AAChE,OAAO,EAAC,eAAe,EAAC,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAC,YAAY,EAAC,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EAAC,kBAAkB,EAAC,MAAM,4BAA4B,CAAC;;;;;;;;;;;;AAI9D;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AAsBH,MAAM,OAAO,sBAcT,SAAQ,cAAwC;IAG/B,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;IACP,WAAW,CAA0B;IAC7C,YAAY,CAAoB;IAElE;;OAEG;IAC6B,OAAO,CAAY;IAEnD;;OAEG;IACa,cAAc,GAAG,IAAI,CAAC;IAEtC;;OAEG;IACH,6EAA6E;IAC7D,WAAW,GAAsB,QAAQ,CAAC;IAE1D;;OAEG;IACH,6EAA6E;IAC7D,cAAc,GAA6B,IAAI,CAAC;IAEhE;;;OAGG;IACK,cAAc,GAAuB,IAAI,CAAC;IAElD;;OAEG;IACH,IACW,MAAM,CAAC,MAA0D;QACxE,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,mBAAmB,EAAE,EAAC,MAAM,EAAE,MAAM,EAAC,CAAC,CAAC;IACrE,CAAC;IAED;;OAEG;IACI,KAAK,GAAsC,IAAI,CAAC;IAEvD;;OAEG;IACI,OAAO,GAAG,KAAK,CAAC;IAEvB;;;OAGG;IACI,YAAY,GAAG,KAAK,CAAC;IAErB,OAAO,GAAG,CAAC,CAAC;IAEnB;;OAEG;IACK,QAAQ,GAAG,EAAE,CAAC;IAEtB;;OAEG;IACc,gBAAgB,GAAG,IAAI,4BAA4B,EAAkB,CAAC;IAEvF;;OAEG;IACH,IACoB,QAAQ,CAAC,QAAiB;QAC1C,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC;IACxE,CAAC;IAEe,QAAQ;QACpB,KAAK,CAAC,QAAQ,EAAE,CAAC;QACjB,IAAI,CAAC,WAAW,EAAE,CAAC;IACvB,CAAC;IAEM,eAAe;QAClB,IAAI,CAAC,YAAY,CAAC,YAAY;aACzB,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,oBAAoB,EAAE,EAAE,YAAY,CAAC,GAAG,CAAC,CAAC;aACpF,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;IAC5C,CAAC;IAEM,oBAAoB;QACvB,4EAA4E;QAC5E,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;YACvB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QACjD,CAAC;IACL,CAAC;IAEe,MAAM;QAClB,IAAI,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;YAC1B,IAAI,CAAC,KAAK,EAAE,CAAC;QACjB,CAAC;QACD,KAAK,CAAC,MAAM,EAAE,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACI,KAAK;QACR,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAChD,IAAI,CAAC,YAAY,CAAC,cAAc,EAAE,CAAC;IACvC,CAAC;IAED;;;;OAIG;IACI,UAAU;QACb,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;YAC3B,IAAI,CAAC,KAAK,EAAE,CAAC;YACb,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC;QAClC,CAAC;IACL,CAAC;IAEe,UAAU,CAAC,KAAyB;QAChD,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QACxB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;IAClD,CAAC;IAEO,WAAW;QACf,+CAA+C;QAC/C,IAAI,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,KAAK,UAAU,EAAE,CAAC;YAC9C,MAAM,IAAI,SAAS,CAAC,qDAAqD,CAAC,CAAC;QAC/E,CAAC;QAED,MAAM,iBAAiB,GAAG;YACtB,UAAU,EAAE;gBACR,SAAS,EAAE,CAAC;gBACZ,QAAQ,EAAE,IAAI,CAAC,QAAQ;aAC1B;SACJ,CAAC;QAEF,MAAM,SAAS,GAAG,KAAK,CAAC,iBAAiB,EAAE,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC;QACvE,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;IACtD,CAAC;IAEM,WAAW;QACd,yBAAyB;QACzB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACb,OAAO;QACX,CAAC;QAED,6EAA6E;QAC7E,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,IAAI,CAC1D,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,EACnC,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC,EACtC,GAAG,CAAC,IAAI,CAAC,EAAE;YACP,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YACrB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC;YAC3B,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;YACtD,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,OAAO,GAAG,QAAQ,CAAC;YAE5C,OAAO,IAAI,CAAC,KAAK,CAAC;QACtB,CAAC,CAAC,CACL,CAAC;QAEF,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACa,cAAc,CAAC,KAAyB;QACpD,IAAI,CAAC,YAAY,CAAC,cAAc,EAAE,CAAC;QACnC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;QAC9C,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QAErB,8FAA8F;QAC9F,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;YACzC,KAAK,GAAG,EAAE,CAAC;QACf,CAAC;QAED,KAAK,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;IAChC,CAAC;IAED;;OAEG;IACI,YAAY;QACf,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,OAAO,IAAI,CAAC,WAAW,CAAC;QAC5B,CAAC;QAED,OAAO,CAAC,IAAS,EAAE,EAAE;YACjB,IAAI,CAAC,IAAI,EAAE,CAAC;gBACR,OAAO,EAAE,CAAC;YACd,CAAC;YAED,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;gBAC3B,OAAO,IAAI,CAAC;YAChB,CAAC;YAED,OAAO,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC,EAAE,CAAC;QACxF,CAAC,CAAC;IACN,CAAC;IAEe,KAAK;QACjB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAClB,KAAK,CAAC,KAAK,EAAE,CAAC;IAClB,CAAC;IAEM,MAAM,CAAC,IAAwB;QAClC,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;YAC5C,IAAI,IAAI,EAAE,CAAC;gBACP,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC;YAChC,CAAC;YAED,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,WAAW,EAAE,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC;QACzE,CAAC;IACL,CAAC;IAEe,eAAe;QAC3B,OAAO,IAAI,CAAC,YAAY,EAAE,OAAO,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;IACxF,CAAC;IAEO,eAAe,CAAC,IAAmB;QACvC,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,WAAW,KAAK,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QAClG,IAAI,IAAI,IAAI,cAAc,KAAK,MAAM,EAAE,CAAC;YACpC,IAAI,GAAG,GAAG,GAAG,IAAI,GAAG,GAAG,CAAC;QAC5B,CAAC;QAED,OAAO;YACH,MAAM,EAAE;gBACJ,MAAM,EAAE;oBACJ;wBACI,UAAU,EAAE;4BACR;gCACI,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,IAAI;oCACpB,CAAC,CAAC;wCACI,CAAC,cAAc,CAAC,EAAE,EAAC,KAAK,EAAE,IAAI,EAAC;qCAClC;oCACH,CAAC,CAAC,IAAI;6BACb;yBACJ;qBACJ;iBACJ;aACJ;SACJ,CAAC;IACN,CAAC;IAEM,oBAAoB;QACvB,OAAO,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,KAAK,CAAC;IACjD,CAAC;uGA9QQ,sBAAsB;2FAAtB,sBAAsB,4RAmBjB,WAAW,6FADd,sBAAsB,uEC1GrC,+wGAgGA,2ZDvBQ,qBAAqB,w1BACrB,YAAY,0PACZ,eAAe,8BACf,kBAAkB,iuBAClB,cAAc,0WACd,WAAW,sZACX,mBAAmB,iNACnB,aAAa,oLACb,oBAAoB,kGACpB,wBAAwB,kOACxB,eAAe,wUACf,gBAAgB,8TAChB,UAAU;;2FAGL,sBAAsB;kBArBlC,SAAS;+BACI,gBAAgB,cAGd,IAAI,WACP;wBACL,qBAAqB;wBACrB,YAAY;wBACZ,eAAe;wBACf,kBAAkB;wBAClB,cAAc;wBACd,WAAW;wBACX,mBAAmB;wBACnB,aAAa;wBACb,oBAAoB;wBACpB,wBAAwB;wBACxB,eAAe;wBACf,gBAAgB;wBAChB,UAAU;qBACb;8BAoByC,WAAW;sBAApD,SAAS;uBAAC,sBAAsB;gBACC,YAAY;sBAA7C,YAAY;uBAAC,WAAW;gBAKO,OAAO;sBAAtC,KAAK;uBAAC,EAAC,QAAQ,EAAE,IAAI,EAAC;gBAKP,cAAc;sBAA7B,KAAK;gBAMU,WAAW;sBAA1B,KAAK;gBAMU,cAAc;sBAA7B,KAAK;gBAYK,MAAM;sBADhB,KAAK;gBAqCc,QAAQ;sBAD3B,KAAK","sourcesContent":["import {\n    AfterViewInit,\n    Component,\n    ContentChild,\n    DestroyRef,\n    inject,\n    Input,\n    OnInit,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatAutocompleteModule, MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {merge} from 'lodash-es';\nimport {Observable} from 'rxjs';\nimport {debounceTime, distinctUntilChanged, finalize, map} from 'rxjs/operators';\nimport {PaginatedData} from '../../../classes/data-source';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {NaturalAbstractModelService} from '../../../services/abstract-model.service';\nimport {ExtractTallOne, ExtractVall, Literal} from '../../../types/types';\nimport {AbstractSelect} from '../abstract-select.component';\nimport {RouterLink} from '@angular/router';\nimport {MatTooltipModule} from '@angular/material/tooltip';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatOptionModule} from '@angular/material/core';\nimport {CommonModule} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\ntype V<TService> = string | ExtractTallOne<TService>;\n\n/**\n * Default usage:\n * ```html\n * <natural-select [service]=\"myServiceInstance\" [(model)]=\"myModel\" (modelChange)=myChangeFn($event) />\n * ```\n *\n * Custom template usage :\n * ```html\n * <natural-select [service]=\"svc\" [(ngModel)]=\"model\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-select>\n * ```\n *\n * `[(ngModel)]` and `(ngModelChange)` are optional.\n *\n * Placeholder :\n * ```html\n * <natural-select placeholder=\"my placeholder\" />\n * ```\n *\n * Search with like %xxx% on specified field `name` instead of custom filter on whole object\n * ```html\n * <natural-select searchField=\"name\" />\n * ```\n *\n * Allows to input free string without selecting an option from autocomplete suggestions\n * ```html\n * <natural-select [optionRequired]=\"false\" />\n * ```\n */\n@Component({\n    selector: 'natural-select',\n    templateUrl: './select.component.html',\n    styleUrl: './select.component.scss',\n    standalone: true,\n    imports: [\n        MatAutocompleteModule,\n        CommonModule,\n        MatOptionModule,\n        MatFormFieldModule,\n        MatInputModule,\n        FormsModule,\n        ReactiveFormsModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatProgressSpinnerModule,\n        MatButtonModule,\n        MatTooltipModule,\n        RouterLink,\n    ],\n})\nexport class NaturalSelectComponent<\n        TService extends NaturalAbstractModelService<\n            any,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            any,\n            any\n        >,\n    >\n    extends AbstractSelect<V<TService>, V<TService>>\n    implements OnInit, ControlValueAccessor, AfterViewInit\n{\n    private readonly destroyRef = inject(DestroyRef);\n    @ViewChild(MatAutocompleteTrigger) public autoTrigger!: MatAutocompleteTrigger;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    /**\n     * Service with watchAll function that accepts queryVariables.\n     */\n    @Input({required: true}) public service!: TService;\n\n    /**\n     * If false, allows to input free string without selecting an option from autocomplete suggestions\n     */\n    @Input() public optionRequired = true;\n\n    /**\n     * The field on which to search for, default to 'custom'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchField: 'custom' | string = 'custom';\n\n    /**\n     * The operator with which to search for, default to 'search' if `searchField` is 'custom', else 'like'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchOperator: 'search' | string | null = null;\n\n    /**\n     * Cache the committed value during search mode.\n     * It's used to be restored in case we cancel the selection\n     */\n    private lastValidValue: V<TService> | null = null;\n\n    /**\n     * Additional filter for query\n     */\n    @Input()\n    public set filter(filter: ExtractVall<TService>['filter'] | null | undefined) {\n        this.variablesManager.set('additional-filter', {filter: filter});\n    }\n\n    /**\n     * Items returned by server to show in listing\n     */\n    public items: null | Observable<readonly any[]> = null;\n\n    /**\n     * Whether we are searching something\n     */\n    public loading = false;\n\n    /**\n     * If some items are not shown in result list\n     * Shows a message after list if true\n     */\n    public hasMoreItems = false;\n\n    public nbTotal = 0;\n\n    /**\n     * Default page size\n     */\n    private pageSize = 10;\n\n    /**\n     * Init search options\n     */\n    private readonly variablesManager = new NaturalQueryVariablesManager<QueryVariables>();\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public override set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.initService();\n    }\n\n    public ngAfterViewInit(): void {\n        this.internalCtrl.valueChanges\n            .pipe(takeUntilDestroyed(this.destroyRef), distinctUntilChanged(), debounceTime(300))\n            .subscribe(val => this.search(val));\n    }\n\n    public onInternalFormChange(): void {\n        // If we allow free string typing, then we propagate it as it is being typed\n        if (!this.optionRequired) {\n            this.propagateValue(this.internalCtrl.value);\n        }\n    }\n\n    public override onBlur(): void {\n        if (this.internalCtrl.dirty) {\n            this.reset();\n        }\n        super.onBlur();\n    }\n\n    /**\n     * Reset form to it's initial value\n     * Discard searched text (in autocomplete use case)\n     * Doest not commit the change to the model (no change event is emitted)\n     */\n    public reset(): void {\n        this.internalCtrl.setValue(this.lastValidValue);\n        this.internalCtrl.markAsPristine();\n    }\n\n    /**\n     * Enter semantic means we want to validate something.\n     * If we hit ENTER while typing a text, the stroke is ignored because the value is invalid (it's accepted in free text mode)\n     * If we hit ENTER while the input field is empty, we validate the unselection (empty is a valid value)\n     */\n    public onKeyEnter(): void {\n        if (!this.internalCtrl.value) {\n            this.clear();\n            this.autoTrigger.closePanel();\n        }\n    }\n\n    public override writeValue(value: V<TService> | null): void {\n        super.writeValue(value);\n        this.lastValidValue = this.internalCtrl.value;\n    }\n\n    private initService(): void {\n        // Assert given service has a watchAll function\n        if (typeof this.service.watchAll !== 'function') {\n            throw new TypeError('Provided service does not contain watchAll function');\n        }\n\n        const defaultPagination = {\n            pagination: {\n                pageIndex: 0,\n                pageSize: this.pageSize,\n            },\n        };\n\n        const variables = merge(defaultPagination, this.getSearchFilter(null));\n        this.variablesManager.set('variables', variables);\n    }\n\n    public startSearch(): void {\n        // Start search only once\n        if (this.items) {\n            return;\n        }\n\n        // Init query, and when query results arrive, finish loading, and count items\n        this.items = this.service.watchAll(this.variablesManager).pipe(\n            takeUntilDestroyed(this.destroyRef),\n            finalize(() => (this.loading = false)),\n            map(data => {\n                this.loading = false;\n                this.nbTotal = data.length;\n                const nbListed = Math.min(data.length, this.pageSize);\n                this.hasMoreItems = this.nbTotal > nbListed;\n\n                return data.items;\n            }),\n        );\n\n        this.loading = true;\n        this.items.subscribe();\n    }\n\n    /**\n     * Commit the model change\n     * Set internal form as pristine to reflect that the visible value match the model\n     */\n    public override propagateValue(value: V<TService> | null): void {\n        this.internalCtrl.markAsPristine();\n        this.lastValidValue = this.internalCtrl.value;\n        this.loading = false;\n\n        // If we cleared value via button, but we allow free string typing, then force to empty string\n        if (!this.optionRequired && value === null) {\n            value = '';\n        }\n\n        super.propagateValue(value);\n    }\n\n    /**\n     * Very important to return something, above all if [select]='displayedValue' attribute value is used\n     */\n    public getDisplayFn(): (item: V<TService> | null) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return (item: any) => {\n            if (!item) {\n                return '';\n            }\n\n            if (typeof item === 'string') {\n                return item;\n            }\n\n            return item.fullName || item.name || item.iban || item[this.searchField] || item.id;\n        };\n    }\n\n    public override clear(): void {\n        this.search(null);\n        super.clear();\n    }\n\n    public search(term: V<TService> | null): void {\n        if (typeof term === 'string' || term === null) {\n            if (term) {\n                this.loading = !!this.items;\n            }\n\n            this.variablesManager.merge('variables', this.getSearchFilter(term));\n        }\n    }\n\n    public override showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    private getSearchFilter(term: string | null): QueryVariables {\n        const searchOperator = this.searchOperator ?? (this.searchField === 'custom' ? 'search' : 'like');\n        if (term && searchOperator === 'like') {\n            term = '%' + term + '%';\n        }\n\n        return {\n            filter: {\n                groups: [\n                    {\n                        conditions: [\n                            {\n                                [this.searchField]: term\n                                    ? {\n                                          [searchOperator]: {value: term},\n                                      }\n                                    : null,\n                            },\n                        ],\n                    },\n                ],\n            },\n        };\n    }\n\n    public getVariablesForDebug(): Readonly<QueryVariables> | undefined {\n        return this.variablesManager.variables.value;\n    }\n}\n","<!-- Autocomplete menu -->\n<mat-autocomplete\n    #ac=\"matAutocomplete\"\n    (optionSelected)=\"propagateValue($event.option.value)\"\n    [displayWith]=\"getDisplayFn()\"\n    panelWidth=\"auto !important\"\n>\n    @for (item of items | async; track $index) {\n        <mat-option [value]=\"item\">\n            <ng-template\n                [ngTemplateOutletContext]=\"{item: item}\"\n                [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n            />\n        </mat-option>\n    }\n    @if (hasMoreItems) {\n        <div class=\"mat-caption\" i18n style=\"padding: 5px 10px\">Saisir pour chercher parmi {{ nbTotal }} résultats</div>\n    }\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n    <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n    <mat-label>{{ placeholder }}</mat-label>\n\n    <input\n        (blur)=\"onBlur()\"\n        (change)=\"onInternalFormChange()\"\n        (click)=\"autoTrigger.openPanel()\"\n        (focus)=\"startSearch()\"\n        (keydown.esc)=\"reset()\"\n        (keydown.enter)=\"onKeyEnter()\"\n        [formControl]=\"internalCtrl\"\n        [matAutocomplete]=\"ac\"\n        aria-label=\"Recherche et sélection\"\n        i18n-aria-label\n        matInput\n        [errorStateMatcher]=\"matcher\"\n    />\n\n    @if (hint) {\n        <mat-hint>{{ hint }}</mat-hint>\n    }\n\n    <!-- Meta data -->\n    @if (!loading && showIcon) {\n        <mat-icon [naturalIcon]=\"icon\" matIconPrefix />\n    }\n\n    @if (loading) {\n        <div class=\"loading-wrapper\" matIconPrefix>\n            <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n        </div>\n    }\n\n    <!-- Clear button -->\n    <div matIconSuffix>\n        @if (internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel) {\n            <button (click)=\"clear()\" mat-icon-button i18n-matTooltip matTooltip=\"Désélectionner\">\n                <mat-icon naturalIcon=\"close\" />\n            </button>\n        }\n        @if (internalCtrl.dirty && internalCtrl.enabled && optionRequired) {\n            <button (click)=\"reset()\" mat-icon-button i18n-matTooltip matTooltip=\"Annuler la recherche\">\n                <mat-icon naturalIcon=\"undo\" />\n            </button>\n        }\n        @if (internalCtrl.pristine && internalCtrl.value && navigateTo) {\n            <button\n                [routerLink]=\"navigateTo\"\n                (click)=\"$event.stopPropagation()\"\n                mat-icon-button\n                i18n-matTooltip\n                matTooltip=\"Naviguer vers\"\n            >\n                <mat-icon naturalIcon=\"open_in_browser\" />\n            </button>\n        }\n    </div>\n\n    @if (hasRequiredError()) {\n        <mat-error i18n>Ce champ est requis</mat-error>\n    }\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n@if (showClearButton()) {\n    <div class=\"external-buttons\">\n        @if (showClearButton()) {\n            <button (click)=\"clear()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n        }\n    </div>\n}\n"]}
|
|
310
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/select/select/select.component.ts","../../../../../../../projects/natural/src/lib/modules/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    Component,\n    ContentChild,\n    DestroyRef,\n    inject,\n    Input,\n    OnInit,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatAutocompleteModule, MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {merge} from 'lodash-es';\nimport {Observable} from 'rxjs';\nimport {debounceTime, distinctUntilChanged, finalize, map} from 'rxjs/operators';\nimport {PaginatedData} from '../../../classes/data-source';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {NaturalAbstractModelService} from '../../../services/abstract-model.service';\nimport {ExtractTallOne, ExtractVall, Literal} from '../../../types/types';\nimport {AbstractSelect} from '../abstract-select.component';\nimport {RouterLink} from '@angular/router';\nimport {MatTooltipModule} from '@angular/material/tooltip';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatOptionModule} from '@angular/material/core';\nimport {CommonModule} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n// Return the type of TValue for the given TService\ntype ValueTypeFor<TService> = string | ExtractTallOne<TService>;\n\n/**\n * Default usage:\n * ```html\n * <natural-select [service]=\"myServiceInstance\" [(model)]=\"myModel\" (modelChange)=myChangeFn($event) />\n * ```\n *\n * Custom template usage :\n * ```html\n * <natural-select [service]=\"svc\" [(ngModel)]=\"model\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-select>\n * ```\n *\n * `[(ngModel)]` and `(ngModelChange)` are optional.\n *\n * Placeholder :\n * ```html\n * <natural-select placeholder=\"my placeholder\" />\n * ```\n *\n * Search with like %xxx% on specified field `name` instead of custom filter on whole object\n * ```html\n * <natural-select searchField=\"name\" />\n * ```\n *\n * Allows to input free string without selecting an option from autocomplete suggestions\n * ```html\n * <natural-select [optionRequired]=\"false\" />\n * ```\n */\n@Component({\n    selector: 'natural-select',\n    templateUrl: './select.component.html',\n    styleUrl: './select.component.scss',\n    standalone: true,\n    imports: [\n        MatAutocompleteModule,\n        CommonModule,\n        MatOptionModule,\n        MatFormFieldModule,\n        MatInputModule,\n        FormsModule,\n        ReactiveFormsModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatProgressSpinnerModule,\n        MatButtonModule,\n        MatTooltipModule,\n        RouterLink,\n    ],\n})\nexport class NaturalSelectComponent<\n        TService extends NaturalAbstractModelService<\n            any,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            any,\n            any\n        >,\n    >\n    extends AbstractSelect<ValueTypeFor<TService>, ValueTypeFor<TService>>\n    implements OnInit, ControlValueAccessor, AfterViewInit\n{\n    private readonly destroyRef = inject(DestroyRef);\n    @ViewChild(MatAutocompleteTrigger) public autoTrigger!: MatAutocompleteTrigger;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    /**\n     * Service with watchAll function that accepts queryVariables.\n     */\n    @Input({required: true}) public service!: TService;\n\n    /**\n     * If false, allows to input free string without selecting an option from autocomplete suggestions\n     */\n    @Input() public optionRequired = true;\n\n    /**\n     * The field on which to search for, default to 'custom'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchField: 'custom' | string = 'custom';\n\n    /**\n     * The operator with which to search for, default to 'search' if `searchField` is 'custom', else 'like'.\n     */\n    // eslint-disable-next-line @typescript-eslint/no-redundant-type-constituents\n    @Input() public searchOperator: 'search' | string | null = null;\n\n    /**\n     * Cache the committed value during search mode.\n     * It's used to be restored in case we cancel the selection\n     */\n    private lastValidValue: ValueTypeFor<TService> | null = null;\n\n    /**\n     * Additional filter for query\n     */\n    @Input()\n    public set filter(filter: ExtractVall<TService>['filter'] | null | undefined) {\n        this.variablesManager.set('additional-filter', {filter: filter});\n    }\n\n    /**\n     * Items returned by server to show in listing\n     */\n    public items: null | Observable<readonly any[]> = null;\n\n    /**\n     * Whether we are searching something\n     */\n    public loading = false;\n\n    /**\n     * If some items are not shown in result list\n     * Shows a message after list if true\n     */\n    public hasMoreItems = false;\n\n    public nbTotal = 0;\n\n    /**\n     * Default page size\n     */\n    private pageSize = 10;\n\n    /**\n     * Init search options\n     */\n    private readonly variablesManager = new NaturalQueryVariablesManager<QueryVariables>();\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public override set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.initService();\n    }\n\n    public ngAfterViewInit(): void {\n        this.internalCtrl.valueChanges\n            .pipe(takeUntilDestroyed(this.destroyRef), distinctUntilChanged(), debounceTime(300))\n            .subscribe(val => this.search(val));\n    }\n\n    public onInternalFormChange(): void {\n        // If we allow free string typing, then we propagate it as it is being typed\n        if (!this.optionRequired) {\n            this.propagateValue(this.internalCtrl.value);\n        }\n    }\n\n    public override onBlur(): void {\n        if (this.internalCtrl.dirty) {\n            this.reset();\n        }\n        super.onBlur();\n    }\n\n    /**\n     * Reset form to it's initial value\n     * Discard searched text (in autocomplete use case)\n     * Doest not commit the change to the model (no change event is emitted)\n     */\n    public reset(): void {\n        this.internalCtrl.setValue(this.lastValidValue);\n        this.internalCtrl.markAsPristine();\n    }\n\n    /**\n     * Enter semantic means we want to validate something.\n     * If we hit ENTER while typing a text, the stroke is ignored because the value is invalid (it's accepted in free text mode)\n     * If we hit ENTER while the input field is empty, we validate the unselection (empty is a valid value)\n     */\n    public onKeyEnter(): void {\n        if (!this.internalCtrl.value) {\n            this.clear();\n            this.autoTrigger.closePanel();\n        }\n    }\n\n    public override writeValue(value: ValueTypeFor<TService> | null): void {\n        super.writeValue(value);\n        this.lastValidValue = this.internalCtrl.value;\n    }\n\n    private initService(): void {\n        // Assert given service has a watchAll function\n        if (typeof this.service.watchAll !== 'function') {\n            throw new TypeError('Provided service does not contain watchAll function');\n        }\n\n        const defaultPagination = {\n            pagination: {\n                pageIndex: 0,\n                pageSize: this.pageSize,\n            },\n        };\n\n        const variables = merge(defaultPagination, this.getSearchFilter(null));\n        this.variablesManager.set('variables', variables);\n    }\n\n    public startSearch(): void {\n        // Start search only once\n        if (this.items) {\n            return;\n        }\n\n        // Init query, and when query results arrive, finish loading, and count items\n        this.items = this.service.watchAll(this.variablesManager).pipe(\n            takeUntilDestroyed(this.destroyRef),\n            finalize(() => (this.loading = false)),\n            map(data => {\n                this.loading = false;\n                this.nbTotal = data.length;\n                const nbListed = Math.min(data.length, this.pageSize);\n                this.hasMoreItems = this.nbTotal > nbListed;\n\n                return data.items;\n            }),\n        );\n\n        this.loading = true;\n        this.items.subscribe();\n    }\n\n    /**\n     * Commit the model change\n     * Set internal form as pristine to reflect that the visible value match the model\n     */\n    public override propagateValue(value: ValueTypeFor<TService> | null): void {\n        this.internalCtrl.markAsPristine();\n        this.lastValidValue = this.internalCtrl.value;\n        this.loading = false;\n\n        // If we cleared value via button, but we allow free string typing, then force to empty string\n        if (!this.optionRequired && value === null) {\n            value = '';\n        }\n\n        super.propagateValue(value);\n    }\n\n    /**\n     * Very important to return something, above all if [select]='displayedValue' attribute value is used\n     */\n    public getDisplayFn(): (item: ValueTypeFor<TService> | null) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return (item: any) => {\n            if (!item) {\n                return '';\n            }\n\n            if (typeof item === 'string') {\n                return item;\n            }\n\n            return item.fullName || item.name || item.iban || item[this.searchField] || item.id;\n        };\n    }\n\n    public override clear(): void {\n        this.search(null);\n        super.clear();\n    }\n\n    public search(term: ValueTypeFor<TService> | null): void {\n        if (typeof term === 'string' || term === null) {\n            if (term) {\n                this.loading = !!this.items;\n            }\n\n            this.variablesManager.merge('variables', this.getSearchFilter(term));\n        }\n    }\n\n    public override showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    private getSearchFilter(term: string | null): QueryVariables {\n        const searchOperator = this.searchOperator ?? (this.searchField === 'custom' ? 'search' : 'like');\n        if (term && searchOperator === 'like') {\n            term = '%' + term + '%';\n        }\n\n        return {\n            filter: {\n                groups: [\n                    {\n                        conditions: [\n                            {\n                                [this.searchField]: term\n                                    ? {\n                                          [searchOperator]: {value: term},\n                                      }\n                                    : null,\n                            },\n                        ],\n                    },\n                ],\n            },\n        };\n    }\n\n    public getVariablesForDebug(): Readonly<QueryVariables> | undefined {\n        return this.variablesManager.variables.value;\n    }\n}\n","<!-- Autocomplete menu -->\n<mat-autocomplete\n    #ac=\"matAutocomplete\"\n    (optionSelected)=\"propagateValue($event.option.value)\"\n    [displayWith]=\"getDisplayFn()\"\n    panelWidth=\"auto !important\"\n>\n    @for (item of items | async; track $index) {\n        <mat-option [value]=\"item\">\n            <ng-template\n                [ngTemplateOutletContext]=\"{item: item}\"\n                [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n            />\n        </mat-option>\n    }\n    @if (hasMoreItems) {\n        <div class=\"mat-caption\" i18n style=\"padding: 5px 10px\">Saisir pour chercher parmi {{ nbTotal }} résultats</div>\n    }\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n    <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n    <mat-label>{{ placeholder }}</mat-label>\n\n    <input\n        (blur)=\"onBlur()\"\n        (change)=\"onInternalFormChange()\"\n        (click)=\"autoTrigger.openPanel()\"\n        (focus)=\"startSearch()\"\n        (keydown.esc)=\"reset()\"\n        (keydown.enter)=\"onKeyEnter()\"\n        [formControl]=\"internalCtrl\"\n        [matAutocomplete]=\"ac\"\n        aria-label=\"Recherche et sélection\"\n        i18n-aria-label\n        matInput\n        [errorStateMatcher]=\"matcher\"\n    />\n\n    @if (hint) {\n        <mat-hint>{{ hint }}</mat-hint>\n    }\n\n    <!-- Meta data -->\n    @if (!loading && showIcon) {\n        <mat-icon [naturalIcon]=\"icon\" matIconPrefix />\n    }\n\n    @if (loading) {\n        <div class=\"loading-wrapper\" matIconPrefix>\n            <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\" />\n        </div>\n    }\n\n    <!-- Clear button -->\n    <div matIconSuffix>\n        @if (internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel) {\n            <button (click)=\"clear()\" mat-icon-button i18n-matTooltip matTooltip=\"Désélectionner\">\n                <mat-icon naturalIcon=\"close\" />\n            </button>\n        }\n        @if (internalCtrl.dirty && internalCtrl.enabled && optionRequired) {\n            <button (click)=\"reset()\" mat-icon-button i18n-matTooltip matTooltip=\"Annuler la recherche\">\n                <mat-icon naturalIcon=\"undo\" />\n            </button>\n        }\n        @if (internalCtrl.pristine && internalCtrl.value && navigateTo) {\n            <button\n                [routerLink]=\"navigateTo\"\n                (click)=\"$event.stopPropagation()\"\n                mat-icon-button\n                i18n-matTooltip\n                matTooltip=\"Naviguer vers\"\n            >\n                <mat-icon naturalIcon=\"open_in_browser\" />\n            </button>\n        }\n    </div>\n\n    @if (hasRequiredError()) {\n        <mat-error i18n>Ce champ est requis</mat-error>\n    }\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n@if (showClearButton()) {\n    <div class=\"external-buttons\">\n        @if (showClearButton()) {\n            <button (click)=\"clear()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n        }\n    </div>\n}\n"]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { Component,
|
|
2
|
-
import { FormsModule,
|
|
1
|
+
import { Component, inject, Input } from '@angular/core';
|
|
2
|
+
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
3
3
|
import { NaturalEnumService } from '../../../services/enum.service';
|
|
4
4
|
import { AbstractSelect } from '../abstract-select.component';
|
|
5
5
|
import { MatOptionModule } from '@angular/material/core';
|
|
@@ -31,10 +31,6 @@ export class NaturalSelectEnumComponent extends AbstractSelect {
|
|
|
31
31
|
*/
|
|
32
32
|
multiple = false;
|
|
33
33
|
items;
|
|
34
|
-
constructor() {
|
|
35
|
-
const ngControl = inject(NgControl, { optional: true, self: true });
|
|
36
|
-
super(ngControl);
|
|
37
|
-
}
|
|
38
34
|
ngOnInit() {
|
|
39
35
|
super.ngOnInit();
|
|
40
36
|
this.items = this.enumService.get(this.enumName);
|
|
@@ -42,13 +38,13 @@ export class NaturalSelectEnumComponent extends AbstractSelect {
|
|
|
42
38
|
getDisplayFn() {
|
|
43
39
|
throw new Error('This should never be called');
|
|
44
40
|
}
|
|
45
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSelectEnumComponent, deps:
|
|
41
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSelectEnumComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
46
42
|
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.3", type: NaturalSelectEnumComponent, isStandalone: true, selector: "natural-select-enum", inputs: { enumName: "enumName", nullLabel: "nullLabel", optionDisabled: "optionDisabled", multiple: "multiple" }, usesInheritance: true, ngImport: i0, template: "<mat-form-field>\n <mat-label>{{ placeholder }}</mat-label>\n <mat-select\n (selectionChange)=\"propagateValue($event.value)\"\n [formControl]=\"internalCtrl\"\n (blur)=\"onBlur()\"\n [errorStateMatcher]=\"matcher\"\n [multiple]=\"multiple\"\n >\n @if (nullLabel) {\n <mat-option [value]=\"null\">{{ nullLabel }}</mat-option>\n }\n @for (item of items | async; track item.value) {\n <mat-option [value]=\"item.value\" [disabled]=\"optionDisabled ? optionDisabled(item) : false\">\n {{ item.name }}\n </mat-option>\n }\n </mat-select>\n\n @if (hint) {\n <mat-hint>{{ hint }}</mat-hint>\n }\n\n @if (hasRequiredError()) {\n <mat-error i18n>Ce champ est requis</mat-error>\n }\n</mat-form-field>\n", styles: [":host{display:flex;flex-direction:column}\n"], dependencies: [{ kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i1.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i1.MatLabel, selector: "mat-label" }, { kind: "directive", type: i1.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "directive", type: i1.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "component", type: i2.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i3.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i5.AsyncPipe, name: "async" }, { kind: "ngmodule", type: MatOptionModule }] });
|
|
47
43
|
}
|
|
48
44
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImport: i0, type: NaturalSelectEnumComponent, decorators: [{
|
|
49
45
|
type: Component,
|
|
50
46
|
args: [{ selector: 'natural-select-enum', standalone: true, imports: [MatFormFieldModule, MatSelectModule, FormsModule, ReactiveFormsModule, CommonModule, MatOptionModule], template: "<mat-form-field>\n <mat-label>{{ placeholder }}</mat-label>\n <mat-select\n (selectionChange)=\"propagateValue($event.value)\"\n [formControl]=\"internalCtrl\"\n (blur)=\"onBlur()\"\n [errorStateMatcher]=\"matcher\"\n [multiple]=\"multiple\"\n >\n @if (nullLabel) {\n <mat-option [value]=\"null\">{{ nullLabel }}</mat-option>\n }\n @for (item of items | async; track item.value) {\n <mat-option [value]=\"item.value\" [disabled]=\"optionDisabled ? optionDisabled(item) : false\">\n {{ item.name }}\n </mat-option>\n }\n </mat-select>\n\n @if (hint) {\n <mat-hint>{{ hint }}</mat-hint>\n }\n\n @if (hasRequiredError()) {\n <mat-error i18n>Ce champ est requis</mat-error>\n }\n</mat-form-field>\n", styles: [":host{display:flex;flex-direction:column}\n"] }]
|
|
51
|
-
}],
|
|
47
|
+
}], propDecorators: { enumName: [{
|
|
52
48
|
type: Input,
|
|
53
49
|
args: [{ required: true }]
|
|
54
50
|
}], nullLabel: [{
|
|
@@ -58,4 +54,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.3", ngImpor
|
|
|
58
54
|
}], multiple: [{
|
|
59
55
|
type: Input
|
|
60
56
|
}] } });
|
|
61
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
57
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZWN0LWVudW0uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmF0dXJhbC9zcmMvbGliL21vZHVsZXMvc2VsZWN0L3NlbGVjdC1lbnVtL3NlbGVjdC1lbnVtLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL3NlbGVjdC9zZWxlY3QtZW51bS9zZWxlY3QtZW51bS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQVMsTUFBTSxlQUFlLENBQUM7QUFDL0QsT0FBTyxFQUF1QixXQUFXLEVBQUUsbUJBQW1CLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUV0RixPQUFPLEVBQVEsa0JBQWtCLEVBQUMsTUFBTSxnQ0FBZ0MsQ0FBQztBQUN6RSxPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sOEJBQThCLENBQUM7QUFDNUQsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQ3ZELE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsZUFBZSxFQUFDLE1BQU0sMEJBQTBCLENBQUM7QUFDekQsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0sOEJBQThCLENBQUM7Ozs7Ozs7QUFXaEUsTUFBTSxPQUFPLDBCQUEyQixTQUFRLGNBQTRCO0lBQ3ZELFdBQVcsR0FBRyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztJQUUxRDs7T0FFRztJQUM2QixRQUFRLENBQVU7SUFFbEQ7O09BRUc7SUFDYSxTQUFTLENBQVU7SUFFbkM7O09BRUc7SUFDYSxjQUFjLENBQTRCO0lBRTFEOztPQUVHO0lBQ2EsUUFBUSxHQUFHLEtBQUssQ0FBQztJQUUxQixLQUFLLENBQXVCO0lBRW5CLFFBQVE7UUFDcEIsS0FBSyxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3JELENBQUM7SUFFTSxZQUFZO1FBQ2YsTUFBTSxJQUFJLEtBQUssQ0FBQyw2QkFBNkIsQ0FBQyxDQUFDO0lBQ25ELENBQUM7dUdBaENRLDBCQUEwQjsyRkFBMUIsMEJBQTBCLHdOQ25CdkMsNjBCQTJCQSxvR0RWYyxrQkFBa0IsK2RBQUUsZUFBZSxtckJBQUUsV0FBVyxzSUFBRSxtQkFBbUIsaU5BQUUsWUFBWSxtRkFBRSxlQUFlOzsyRkFFckcsMEJBQTBCO2tCQVB0QyxTQUFTOytCQUNJLHFCQUFxQixjQUduQixJQUFJLFdBQ1AsQ0FBQyxrQkFBa0IsRUFBRSxlQUFlLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLFlBQVksRUFBRSxlQUFlLENBQUM7OEJBUS9FLFFBQVE7c0JBQXZDLEtBQUs7dUJBQUMsRUFBQyxRQUFRLEVBQUUsSUFBSSxFQUFDO2dCQUtQLFNBQVM7c0JBQXhCLEtBQUs7Z0JBS1UsY0FBYztzQkFBN0IsS0FBSztnQkFLVSxRQUFRO3NCQUF2QixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtDb21wb25lbnQsIGluamVjdCwgSW5wdXQsIE9uSW5pdH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0NvbnRyb2xWYWx1ZUFjY2Vzc29yLCBGb3Jtc01vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHtPYnNlcnZhYmxlfSBmcm9tICdyeGpzJztcbmltcG9ydCB7SUVudW0sIE5hdHVyYWxFbnVtU2VydmljZX0gZnJvbSAnLi4vLi4vLi4vc2VydmljZXMvZW51bS5zZXJ2aWNlJztcbmltcG9ydCB7QWJzdHJhY3RTZWxlY3R9IGZyb20gJy4uL2Fic3RyYWN0LXNlbGVjdC5jb21wb25lbnQnO1xuaW1wb3J0IHtNYXRPcHRpb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHtDb21tb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQge01hdFNlbGVjdE1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvc2VsZWN0JztcbmltcG9ydCB7TWF0Rm9ybUZpZWxkTW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9mb3JtLWZpZWxkJztcblxudHlwZSBWYWx1ZSA9IElFbnVtWyd2YWx1ZSddIHwgSUVudW1bJ3ZhbHVlJ11bXTtcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICduYXR1cmFsLXNlbGVjdC1lbnVtJyxcbiAgICB0ZW1wbGF0ZVVybDogJy4vc2VsZWN0LWVudW0uY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsOiAnLi9zZWxlY3QtZW51bS5jb21wb25lbnQuc2NzcycsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbTWF0Rm9ybUZpZWxkTW9kdWxlLCBNYXRTZWxlY3RNb2R1bGUsIEZvcm1zTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlLCBDb21tb25Nb2R1bGUsIE1hdE9wdGlvbk1vZHVsZV0sXG59KVxuZXhwb3J0IGNsYXNzIE5hdHVyYWxTZWxlY3RFbnVtQ29tcG9uZW50IGV4dGVuZHMgQWJzdHJhY3RTZWxlY3Q8VmFsdWUsIFZhbHVlPiBpbXBsZW1lbnRzIE9uSW5pdCwgQ29udHJvbFZhbHVlQWNjZXNzb3Ige1xuICAgIHByaXZhdGUgcmVhZG9ubHkgZW51bVNlcnZpY2UgPSBpbmplY3QoTmF0dXJhbEVudW1TZXJ2aWNlKTtcblxuICAgIC8qKlxuICAgICAqIFRoZSBuYW1lIG9mIHRoZSBlbnVtIHR5cGUsIGVnOiBgXCJBY3Rpb25TdGF0dXNcImBcbiAgICAgKi9cbiAgICBASW5wdXQoe3JlcXVpcmVkOiB0cnVlfSkgcHVibGljIGVudW1OYW1lITogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogSWYgZ2l2ZW4gYW4gZXh0cmEgb3B0aW9uIGlzIGFkZGVkIHRvIHNlbGVjdCBgbnVsbGAgd2l0aCBnaXZlbiBsYWJlbFxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBudWxsTGFiZWw/OiBzdHJpbmc7XG5cbiAgICAvKipcbiAgICAgKiBGdW5jdGlvbnMgdGhhdCByZWNlaXZlcyBhbiBlbnVtIHZhbHVlIGFuZCByZXR1cm5zIHdoZXRoZXIgdGhhdCB2YWx1ZSBpcyBkaXNhYmxlZFxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBvcHRpb25EaXNhYmxlZD86IChpdGVtOiBJRW51bSkgPT4gYm9vbGVhbjtcblxuICAgIC8qKlxuICAgICAqIFdoZXRoZXIgdGhlIHVzZXIgc2hvdWxkIGJlIGFsbG93ZWQgdG8gc2VsZWN0IG11bHRpcGxlIG9wdGlvbnNcbiAgICAgKi9cbiAgICBASW5wdXQoKSBwdWJsaWMgbXVsdGlwbGUgPSBmYWxzZTtcblxuICAgIHB1YmxpYyBpdGVtcz86IE9ic2VydmFibGU8SUVudW1bXT47XG5cbiAgICBwdWJsaWMgb3ZlcnJpZGUgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgICAgIHN1cGVyLm5nT25Jbml0KCk7XG4gICAgICAgIHRoaXMuaXRlbXMgPSB0aGlzLmVudW1TZXJ2aWNlLmdldCh0aGlzLmVudW1OYW1lKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0RGlzcGxheUZuKCk6IChpdGVtOiBWYWx1ZSB8IG51bGwpID0+IHN0cmluZyB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignVGhpcyBzaG91bGQgbmV2ZXIgYmUgY2FsbGVkJyk7XG4gICAgfVxufVxuIiwiPG1hdC1mb3JtLWZpZWxkPlxuICAgIDxtYXQtbGFiZWw+e3sgcGxhY2Vob2xkZXIgfX08L21hdC1sYWJlbD5cbiAgICA8bWF0LXNlbGVjdFxuICAgICAgICAoc2VsZWN0aW9uQ2hhbmdlKT1cInByb3BhZ2F0ZVZhbHVlKCRldmVudC52YWx1ZSlcIlxuICAgICAgICBbZm9ybUNvbnRyb2xdPVwiaW50ZXJuYWxDdHJsXCJcbiAgICAgICAgKGJsdXIpPVwib25CbHVyKClcIlxuICAgICAgICBbZXJyb3JTdGF0ZU1hdGNoZXJdPVwibWF0Y2hlclwiXG4gICAgICAgIFttdWx0aXBsZV09XCJtdWx0aXBsZVwiXG4gICAgPlxuICAgICAgICBAaWYgKG51bGxMYWJlbCkge1xuICAgICAgICAgICAgPG1hdC1vcHRpb24gW3ZhbHVlXT1cIm51bGxcIj57eyBudWxsTGFiZWwgfX08L21hdC1vcHRpb24+XG4gICAgICAgIH1cbiAgICAgICAgQGZvciAoaXRlbSBvZiBpdGVtcyB8IGFzeW5jOyB0cmFjayBpdGVtLnZhbHVlKSB7XG4gICAgICAgICAgICA8bWF0LW9wdGlvbiBbdmFsdWVdPVwiaXRlbS52YWx1ZVwiIFtkaXNhYmxlZF09XCJvcHRpb25EaXNhYmxlZCA/IG9wdGlvbkRpc2FibGVkKGl0ZW0pIDogZmFsc2VcIj5cbiAgICAgICAgICAgICAgICB7eyBpdGVtLm5hbWUgfX1cbiAgICAgICAgICAgIDwvbWF0LW9wdGlvbj5cbiAgICAgICAgfVxuICAgIDwvbWF0LXNlbGVjdD5cblxuICAgIEBpZiAoaGludCkge1xuICAgICAgICA8bWF0LWhpbnQ+e3sgaGludCB9fTwvbWF0LWhpbnQ+XG4gICAgfVxuXG4gICAgQGlmIChoYXNSZXF1aXJlZEVycm9yKCkpIHtcbiAgICAgICAgPG1hdC1lcnJvciBpMThuPkNlIGNoYW1wIGVzdCByZXF1aXM8L21hdC1lcnJvcj5cbiAgICB9XG48L21hdC1mb3JtLWZpZWxkPlxuIl19
|