@ecodev/natural 42.4.0 → 43.0.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/esm2020/lib/classes/query-variable-manager.mjs +2 -2
- package/esm2020/lib/classes/rxjs.mjs +2 -1
- package/esm2020/lib/modules/columns-picker/columns-picker-column.directive.mjs +6 -2
- package/esm2020/lib/modules/common/directives/linkable-tab.directive.mjs +2 -2
- package/esm2020/lib/modules/logger/error-handler.mjs +26 -11
- package/esm2020/lib/modules/logger/public-api.mjs +3 -3
- package/esm2020/lib/modules/relations/relations.component.mjs +9 -4
- package/esm2020/lib/modules/search/dropdown-container/dropdown-container.component.mjs +3 -3
- package/esm2020/lib/modules/search/dropdown-container/dropdown.service.mjs +1 -1
- package/esm2020/lib/modules/search/input/input.component.mjs +3 -1
- package/esm2020/lib/modules/sidenav/sidenav-container/sidenav-container.component.mjs +4 -5
- package/esm2020/lib/modules/table-button/table-button.component.mjs +21 -6
- package/esm2020/lib/services/abstract-model.service.mjs +45 -39
- package/fesm2015/ecodev-natural.mjs +112 -63
- package/fesm2015/ecodev-natural.mjs.map +1 -1
- package/fesm2020/ecodev-natural.mjs +109 -62
- package/fesm2020/ecodev-natural.mjs.map +1 -1
- package/lib/modules/columns-picker/columns-picker-column.directive.d.ts +1 -1
- package/lib/modules/logger/error-handler.d.ts +19 -3
- package/lib/modules/logger/public-api.d.ts +2 -2
- package/lib/modules/relations/relations.component.d.ts +1 -0
- package/lib/modules/search/dropdown-container/dropdown-container.component.d.ts +1 -1
- package/lib/modules/search/dropdown-container/dropdown.service.d.ts +1 -1
- package/lib/modules/search/input/input.component.d.ts +2 -2
- package/lib/modules/sidenav/sidenav-container/sidenav-container.component.d.ts +2 -3
- package/lib/modules/table-button/table-button.component.d.ts +9 -6
- package/lib/services/abstract-model.service.d.ts +2 -2
- package/package.json +1 -1
|
@@ -120,7 +120,7 @@ export class NaturalQueryVariablesManager {
|
|
|
120
120
|
*/
|
|
121
121
|
updateVariables() {
|
|
122
122
|
const merged = {};
|
|
123
|
-
this.channels.forEach(
|
|
123
|
+
this.channels.forEach(channelVariables => {
|
|
124
124
|
if (channelVariables.filter) {
|
|
125
125
|
// Merge filter's groups first
|
|
126
126
|
const groups = this.mergeGroupList(merged.filter && merged.filter.groups ? merged.filter.groups : [], channelVariables.filter.groups || []);
|
|
@@ -169,4 +169,4 @@ export class NaturalQueryVariablesManager {
|
|
|
169
169
|
return groups;
|
|
170
170
|
}
|
|
171
171
|
}
|
|
172
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-variable-manager.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/query-variable-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import {cloneDeep, defaultsDeep, isArray, mergeWith, omit} from 'lodash-es';\nimport {BehaviorSubject} from 'rxjs';\nimport {Literal} from '../types/types';\nimport {mergeOverrideArray} from './utility';\nimport {hasMixedGroupLogic} from './query-variable-manager-utils';\n\nexport interface QueryVariables {\n    filter?: any | null;\n    pagination?: PaginationInput | null;\n    sorting?: Array<Sorting> | null;\n}\n\nexport interface PaginationInput {\n    offset?: number | null;\n    pageIndex?: number | null;\n    pageSize?: number | null;\n}\n\nexport interface Sorting {\n    field: any;\n    order?: SortingOrder | null;\n    nullAsHighest?: boolean | null;\n}\n\nexport enum SortingOrder {\n    ASC = 'ASC',\n    DESC = 'DESC',\n}\n\n/**\n * During lodash merge, concat arrays\n */\nfunction mergeConcatArray(destValue: any, source: any): any {\n    if (isArray(source)) {\n        if (destValue) {\n            return destValue.concat(source);\n        } else {\n            return source;\n        }\n    }\n}\n\n/**\n * Filter manager stores a set of channels that contain a variable object and exposes an observable \"variables\" that updates with the result\n * of all channels merged together.\n *\n * A channel is supposed to be used by a given aspect of the GUI (pagination, sorting, search, others ?).\n *\n * ```ts\n * const fm = new QueryVariablesManager();\n * fm.merge('componentA-variables', {a : [1, 2, 3]});\n * ```\n *\n * Variables attributes is a BehaviorSubject. That mean it's not mandatory to subscribe, we can just call getValue or value attributes on\n * it :\n *\n * ```ts\n * console.log(fm.variables.value); // {a : [1, 2, 3]}\n * ```\n *\n * Set new variables for 'componentA-variables':\n *\n * ```ts\n * fm.merge('componentA-variables', {a : [1, 2]});\n * console.log(fm.variables.value); // {a : [1, 2, 3]}\n * ```\n *\n * Set new variables for new channel:\n *\n * ```ts\n * fm.merge('componentB-variables', {a : [3, 4]});\n * console.log(fm.variables.value); // {a : [1, 2, 3, 4]}\n * ```\n */\nexport class NaturalQueryVariablesManager<T extends QueryVariables = QueryVariables> {\n    public readonly variables: BehaviorSubject<T | undefined> = new BehaviorSubject<T | undefined>(undefined);\n    private readonly channels: Map<string, Partial<T>> = new Map<string, Partial<T>>();\n\n    public constructor(queryVariablesManager?: NaturalQueryVariablesManager<T>) {\n        if (queryVariablesManager) {\n            this.channels = queryVariablesManager.getChannelsCopy();\n            this.updateVariables();\n        }\n    }\n\n    /**\n     * Set or override all the variables that may exist in the given channel\n     */\n    public set(channelName: string, variables: Partial<T> | null | undefined): void {\n        // cloneDeep to change reference and prevent some interactions when merge\n        if (variables) {\n            this.channels.set(channelName, cloneDeep(variables));\n        } else {\n            this.channels.delete(channelName);\n        }\n        this.updateVariables();\n    }\n\n    /**\n     * Return a deep clone of the variables for the given channel name.\n     *\n     * Avoid returning the same reference to prevent an attribute change, then another channel update that would\n     * used this changed attribute without having explicitly asked QueryVariablesManager to update it.\n     */\n    public get(channelName: string): Partial<T> | undefined {\n        return cloneDeep(this.channels.get(channelName));\n    }\n\n    /**\n     * Merge variable into a channel, overriding arrays in same channel / key\n     */\n    public merge(channelName: string, newVariables: Partial<T>): void {\n        const variables = this.channels.get(channelName);\n        if (variables) {\n            mergeWith(variables, cloneDeep(newVariables), mergeOverrideArray); // merge preserves references, cloneDeep prevent that\n            this.updateVariables();\n        } else {\n            this.set(channelName, newVariables);\n        }\n    }\n\n    /**\n     * Apply default values to a channel\n     * Note : lodash defaults only defines values on destinations keys that are undefined\n     */\n    public defaults(channelName: string, newVariables: Partial<T>): void {\n        const variables = this.channels.get(channelName);\n        if (variables) {\n            defaultsDeep(variables, newVariables);\n            this.updateVariables();\n        } else {\n            this.set(channelName, newVariables);\n        }\n    }\n\n    private getChannelsCopy(): Map<string, Partial<T>> {\n        return new Map<string, Partial<T>>(this.channels);\n    }\n\n    /**\n     * Merge channels in a single object\n     * Arrays are concatenated\n     * Filter groups are combined smartly (see mergeGroupList)\n     */\n    private updateVariables(): void {\n        const merged: Literal = {};\n\n        this.channels.forEach((channelVariables: Literal) => {\n            if (channelVariables.filter) {\n                // Merge filter's groups first\n                const groups = this.mergeGroupList(\n                    merged.filter && merged.filter.groups ? merged.filter.groups : [],\n                    channelVariables.filter.groups || [],\n                );\n\n                // Merge filter key (that contain groups)\n                if (groups && groups.length) {\n                    if (merged.filter) {\n                        merged.filter.groups = groups;\n                    } else {\n                        merged.filter = {groups: groups};\n                    }\n                } else {\n                    mergeWith(merged, {filter: channelVariables.filter}, mergeConcatArray);\n                }\n            }\n\n            // Merge other attributes than filter\n            mergeWith(merged, omit(channelVariables, 'filter'), mergeConcatArray);\n        });\n\n        this.variables.next(merged as T);\n    }\n\n    /**\n     * Cross merge two filters\n     * Only accepts groups with same groupLogic (ignores the first one, because there is no groupLogic in this one)\n     * @throws In case two non-empty lists of groups are given and at one of them mix groupLogic value, throws an error\n     */\n    private mergeGroupList(groupsA: Literal[], groupsB: Literal[]): Literal {\n        if (groupsA.length === 0 && groupsB.length === 0) {\n            return []; // empty listings, return empty lists\n        }\n\n        if (groupsA.length === 0 && groupsB.length > 0) {\n            return groupsB; // One list is empty, return the one that is not\n        }\n\n        if (groupsB.length === 0 && groupsA.length > 0) {\n            return groupsA; // One list is empty, return the one that is not\n        }\n\n        const groups: Literal[] = [];\n\n        if (hasMixedGroupLogic(groupsA) || hasMixedGroupLogic(groupsB)) {\n            throw Error('QueryVariables groups contain mixed group logics');\n        }\n\n        groupsA.forEach(groupA => {\n            groupsB.forEach(groupB => {\n                groups.push(mergeWith(cloneDeep(groupA), groupB, mergeConcatArray));\n            });\n        });\n\n        return groups;\n    }\n}\n"]}
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-variable-manager.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/query-variable-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import {cloneDeep, defaultsDeep, isArray, mergeWith, omit} from 'lodash-es';\nimport {BehaviorSubject} from 'rxjs';\nimport {Literal} from '../types/types';\nimport {mergeOverrideArray} from './utility';\nimport {hasMixedGroupLogic} from './query-variable-manager-utils';\n\nexport interface QueryVariables {\n    filter?: any | null;\n    pagination?: PaginationInput | null;\n    sorting?: Array<Sorting> | null;\n}\n\nexport interface PaginationInput {\n    offset?: number | null;\n    pageIndex?: number | null;\n    pageSize?: number | null;\n}\n\nexport interface Sorting {\n    field: any;\n    order?: SortingOrder | null;\n    nullAsHighest?: boolean | null;\n}\n\nexport enum SortingOrder {\n    ASC = 'ASC',\n    DESC = 'DESC',\n}\n\n/**\n * During lodash merge, concat arrays\n */\nfunction mergeConcatArray(destValue: any, source: any): any {\n    if (isArray(source)) {\n        if (destValue) {\n            return destValue.concat(source);\n        } else {\n            return source;\n        }\n    }\n}\n\n/**\n * Filter manager stores a set of channels that contain a variable object and exposes an observable \"variables\" that updates with the result\n * of all channels merged together.\n *\n * A channel is supposed to be used by a given aspect of the GUI (pagination, sorting, search, others ?).\n *\n * ```ts\n * const fm = new QueryVariablesManager();\n * fm.merge('componentA-variables', {a : [1, 2, 3]});\n * ```\n *\n * Variables attributes is a BehaviorSubject. That mean it's not mandatory to subscribe, we can just call getValue or value attributes on\n * it :\n *\n * ```ts\n * console.log(fm.variables.value); // {a : [1, 2, 3]}\n * ```\n *\n * Set new variables for 'componentA-variables':\n *\n * ```ts\n * fm.merge('componentA-variables', {a : [1, 2]});\n * console.log(fm.variables.value); // {a : [1, 2, 3]}\n * ```\n *\n * Set new variables for new channel:\n *\n * ```ts\n * fm.merge('componentB-variables', {a : [3, 4]});\n * console.log(fm.variables.value); // {a : [1, 2, 3, 4]}\n * ```\n */\nexport class NaturalQueryVariablesManager<T extends QueryVariables = QueryVariables> {\n    public readonly variables: BehaviorSubject<T | undefined> = new BehaviorSubject<T | undefined>(undefined);\n    private readonly channels: Map<string, Partial<T>> = new Map<string, Partial<T>>();\n\n    public constructor(queryVariablesManager?: NaturalQueryVariablesManager<T>) {\n        if (queryVariablesManager) {\n            this.channels = queryVariablesManager.getChannelsCopy();\n            this.updateVariables();\n        }\n    }\n\n    /**\n     * Set or override all the variables that may exist in the given channel\n     */\n    public set(channelName: string, variables: Partial<T> | null | undefined): void {\n        // cloneDeep to change reference and prevent some interactions when merge\n        if (variables) {\n            this.channels.set(channelName, cloneDeep(variables));\n        } else {\n            this.channels.delete(channelName);\n        }\n        this.updateVariables();\n    }\n\n    /**\n     * Return a deep clone of the variables for the given channel name.\n     *\n     * Avoid returning the same reference to prevent an attribute change, then another channel update that would\n     * used this changed attribute without having explicitly asked QueryVariablesManager to update it.\n     */\n    public get(channelName: string): Partial<T> | undefined {\n        return cloneDeep(this.channels.get(channelName));\n    }\n\n    /**\n     * Merge variable into a channel, overriding arrays in same channel / key\n     */\n    public merge(channelName: string, newVariables: Partial<T>): void {\n        const variables = this.channels.get(channelName);\n        if (variables) {\n            mergeWith(variables, cloneDeep(newVariables), mergeOverrideArray); // merge preserves references, cloneDeep prevent that\n            this.updateVariables();\n        } else {\n            this.set(channelName, newVariables);\n        }\n    }\n\n    /**\n     * Apply default values to a channel\n     * Note : lodash defaults only defines values on destinations keys that are undefined\n     */\n    public defaults(channelName: string, newVariables: Partial<T>): void {\n        const variables = this.channels.get(channelName);\n        if (variables) {\n            defaultsDeep(variables, newVariables);\n            this.updateVariables();\n        } else {\n            this.set(channelName, newVariables);\n        }\n    }\n\n    private getChannelsCopy(): Map<string, Partial<T>> {\n        return new Map<string, Partial<T>>(this.channels);\n    }\n\n    /**\n     * Merge channels in a single object\n     * Arrays are concatenated\n     * Filter groups are combined smartly (see mergeGroupList)\n     */\n    private updateVariables(): void {\n        const merged: T = {} as T;\n\n        this.channels.forEach(channelVariables => {\n            if (channelVariables.filter) {\n                // Merge filter's groups first\n                const groups = this.mergeGroupList(\n                    merged.filter && merged.filter.groups ? merged.filter.groups : [],\n                    channelVariables.filter.groups || [],\n                );\n\n                // Merge filter key (that contain groups)\n                if (groups && groups.length) {\n                    if (merged.filter) {\n                        merged.filter.groups = groups;\n                    } else {\n                        merged.filter = {groups: groups};\n                    }\n                } else {\n                    mergeWith(merged, {filter: channelVariables.filter}, mergeConcatArray);\n                }\n            }\n\n            // Merge other attributes than filter\n            mergeWith(merged, omit(channelVariables, 'filter'), mergeConcatArray);\n        });\n\n        this.variables.next(merged);\n    }\n\n    /**\n     * Cross merge two filters\n     * Only accepts groups with same groupLogic (ignores the first one, because there is no groupLogic in this one)\n     * @throws In case two non-empty lists of groups are given and at one of them mix groupLogic value, throws an error\n     */\n    private mergeGroupList(groupsA: Literal[], groupsB: Literal[]): Literal {\n        if (groupsA.length === 0 && groupsB.length === 0) {\n            return []; // empty listings, return empty lists\n        }\n\n        if (groupsA.length === 0 && groupsB.length > 0) {\n            return groupsB; // One list is empty, return the one that is not\n        }\n\n        if (groupsB.length === 0 && groupsA.length > 0) {\n            return groupsA; // One list is empty, return the one that is not\n        }\n\n        const groups: Literal[] = [];\n\n        if (hasMixedGroupLogic(groupsA) || hasMixedGroupLogic(groupsB)) {\n            throw Error('QueryVariables groups contain mixed group logics');\n        }\n\n        groupsA.forEach(groupA => {\n            groupsB.forEach(groupB => {\n                groups.push(mergeWith(cloneDeep(groupA), groupB, mergeConcatArray));\n            });\n        });\n\n        return groups;\n    }\n}\n"]}
|
|
@@ -39,9 +39,10 @@ export function cancellableTimeout(canceller, milliSeconds = 0) {
|
|
|
39
39
|
*/
|
|
40
40
|
export function debug(debugName) {
|
|
41
41
|
return tap({
|
|
42
|
+
subscribe: () => console.log('SUBSCRIBE', debugName),
|
|
42
43
|
next: value => console.log('NEXT', debugName, value),
|
|
43
44
|
error: error => console.log('ERROR', debugName, error),
|
|
44
45
|
complete: () => console.log('COMPLETE', debugName),
|
|
45
46
|
});
|
|
46
47
|
}
|
|
47
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnhqcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9jbGFzc2VzL3J4anMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUF1QyxLQUFLLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDakUsT0FBTyxFQUFDLEdBQUcsRUFBRSxJQUFJLEVBQUUsU0FBUyxFQUFFLEdBQUcsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRXpEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0EyQkc7QUFDSCxNQUFNLFVBQVUsa0JBQWtCLENBQUMsU0FBOEIsRUFBRSxlQUF1QixDQUFDO0lBQ3ZGLE9BQU8sS0FBSyxDQUFDLFlBQVksQ0FBQyxDQUFDLElBQUksQ0FDM0IsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUNQLFNBQVMsQ0FBQyxTQUFTLENBQUMsRUFDcEIsR0FBRyxDQUFDLEdBQUcsRUFBRTtRQUNMLE9BQU87SUFDWCxDQUFDLENBQUMsQ0FDTCxDQUFDO0FBQ04sQ0FBQztBQUVEOzs7R0FHRztBQUNILE1BQU0sVUFBVSxLQUFLLENBQUksU0FBaUI7SUFDdEMsT0FBTyxHQUFHLENBQUk7UUFDVixTQUFTLEVBQUUsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxXQUFXLEVBQUUsU0FBUyxDQUFDO1FBQ3BELElBQUksRUFBRSxLQUFLLENBQUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxFQUFFLFNBQVMsRUFBRSxLQUFLLENBQUM7UUFDcEQsS0FBSyxFQUFFLEtBQUssQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssQ0FBQztRQUN0RCxRQUFRLEVBQUUsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLEVBQUUsU0FBUyxDQUFDO0tBQ3JELENBQUMsQ0FBQztBQUNQLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge01vbm9UeXBlT3BlcmF0b3JGdW5jdGlvbiwgT2JzZXJ2YWJsZSwgdGltZXJ9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHttYXAsIHRha2UsIHRha2VVbnRpbCwgdGFwfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbi8qKlxuICogQmVoYXZlIGxpa2Ugc2V0VGltZW91dCgpLCBidXQgd2l0aCBhIG1hbmRhdG9yeSBjYW5jZWwgbWVjaGFuaXNtLlxuICpcbiAqIFRoaXMgaXMgdHlwaWNhbGx5IHVzZWZ1bCB0byByZXBsYWNlIHNldFRpbWVvdXQoKSBpbiBjb21wb25lbnRzIHdoZXJlIHRoZSBjYWxsYmFja1xuICogd291bGQgY3Jhc2ggaWYgZXhlY3V0ZWQgYWZ0ZXIgdGhlIGNvbXBvbmVudCBkZXN0cnVjdGlvbi4gVGhhdCBjYW4gZWFzaWx5IGhhcHBlblxuICogd2hlbiB0aGUgdXNlciBuYXZpZ2F0ZSBxdWlja2x5IGJldHdlZW4gcGFnZXMuXG4gKlxuICogVHlwaWNhbCB1c2FnZSBpbiBhIGNvbXBvbmVudCB3b3VsZCBiZTpcbiAqXG4gKiBgYGB0c1xuICogY2FuY2VsbGFibGVUaW1lb3V0KHRoaXMubmdVbnN1YnNjcmliZSkuc3Vic2NyaWJlKG15Q2FsbGJhY2spO1xuICogYGBgXG4gKlxuICogSW5zdGVhZCBvZiB0aGUgbW9yZSBlcnJvciBwcm9uZTpcbiAqXG4gKiBgYGB0c1xuICogcHVibGljIGZvbygpOiB2b2lkIHtcbiAqICAgICB0aGlzLnRpbWVvdXQgPSBzZXRUaW1lb3V0KG15Q2FsbEJhY2spO1xuICogfVxuICpcbiAqIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAqICAgICBpZiAodGhpcy50aW1lb3V0KSB7XG4gKiAgICAgICAgIGNsZWFyVGltZW91dCh0aGlzLnRpbWVvdXQpO1xuICogICAgICAgICB0aGlzLnRpbWVvdXQgPSBudWxsO1xuICogICAgICB9XG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNhbmNlbGxhYmxlVGltZW91dChjYW5jZWxsZXI6IE9ic2VydmFibGU8dW5rbm93bj4sIG1pbGxpU2Vjb25kczogbnVtYmVyID0gMCk6IE9ic2VydmFibGU8dm9pZD4ge1xuICAgIHJldHVybiB0aW1lcihtaWxsaVNlY29uZHMpLnBpcGUoXG4gICAgICAgIHRha2UoMSksXG4gICAgICAgIHRha2VVbnRpbChjYW5jZWxsZXIpLFxuICAgICAgICBtYXAoKCkgPT4ge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9KSxcbiAgICApO1xufVxuXG4vKipcbiAqIEZvciBkZWJ1Z2dpbmcgcHVycG9zZSBvbmx5LCB3aWxsIGR1bXAgaW4gY29uc29sZSBldmVyeXRoaW5nIHRoYXQgaGFwcGVuIHRvXG4gKiB0aGUgb2JzZXJ2YWJsZVxuICovXG5leHBvcnQgZnVuY3Rpb24gZGVidWc8VD4oZGVidWdOYW1lOiBzdHJpbmcpOiBNb25vVHlwZU9wZXJhdG9yRnVuY3Rpb248VD4ge1xuICAgIHJldHVybiB0YXA8VD4oe1xuICAgICAgICBzdWJzY3JpYmU6ICgpID0+IGNvbnNvbGUubG9nKCdTVUJTQ1JJQkUnLCBkZWJ1Z05hbWUpLFxuICAgICAgICBuZXh0OiB2YWx1ZSA9PiBjb25zb2xlLmxvZygnTkVYVCcsIGRlYnVnTmFtZSwgdmFsdWUpLFxuICAgICAgICBlcnJvcjogZXJyb3IgPT4gY29uc29sZS5sb2coJ0VSUk9SJywgZGVidWdOYW1lLCBlcnJvciksXG4gICAgICAgIGNvbXBsZXRlOiAoKSA9PiBjb25zb2xlLmxvZygnQ09NUExFVEUnLCBkZWJ1Z05hbWUpLFxuICAgIH0pO1xufVxuIl19
|
|
@@ -11,6 +11,10 @@ export class NaturalColumnsPickerColumnDirective {
|
|
|
11
11
|
* Initial visibility state
|
|
12
12
|
*/
|
|
13
13
|
this.hidden = false;
|
|
14
|
+
/**
|
|
15
|
+
* Localized label of column, if absent default to key
|
|
16
|
+
*/
|
|
17
|
+
this.label = '';
|
|
14
18
|
}
|
|
15
19
|
/**
|
|
16
20
|
* This must be the column key as defined in matColumnDef
|
|
@@ -23,7 +27,7 @@ export class NaturalColumnsPickerColumnDirective {
|
|
|
23
27
|
this.label = this.key;
|
|
24
28
|
}
|
|
25
29
|
ngAfterViewInit() {
|
|
26
|
-
this.label = this.elementRef.nativeElement.textContent;
|
|
30
|
+
this.label = this.elementRef.nativeElement.textContent ?? '';
|
|
27
31
|
}
|
|
28
32
|
}
|
|
29
33
|
NaturalColumnsPickerColumnDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: NaturalColumnsPickerColumnDirective, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
@@ -40,4 +44,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImpor
|
|
|
40
44
|
}], naturalColumnsPickerColumn: [{
|
|
41
45
|
type: Input
|
|
42
46
|
}] } });
|
|
43
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
47
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sdW1ucy1waWNrZXItY29sdW1uLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2NvbHVtbnMtcGlja2VyL2NvbHVtbnMtcGlja2VyLWNvbHVtbi5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFnQixTQUFTLEVBQWMsS0FBSyxFQUFTLE1BQU0sZUFBZSxDQUFDOztBQUtsRixNQUFNLE9BQU8sbUNBQW1DO0lBa0I1QyxZQUFvQyxVQUFtQztRQUFuQyxlQUFVLEdBQVYsVUFBVSxDQUF5QjtRQWZ2RTs7V0FFRztRQUNhLFlBQU8sR0FBRyxJQUFJLENBQUM7UUFFL0I7O1dBRUc7UUFDYSxXQUFNLEdBQUcsS0FBSyxDQUFDO1FBRS9COztXQUVHO1FBQ0ksVUFBSyxHQUFXLEVBQUUsQ0FBQztJQUVnRCxDQUFDO0lBRTNFOztPQUVHO0lBQ0gsSUFDVywwQkFBMEIsQ0FBQyxLQUFhO1FBQy9DLElBQUksQ0FBQyxHQUFHLEdBQUcsS0FBSyxDQUFDO0lBQ3JCLENBQUM7SUFFTSxRQUFRO1FBQ1gsdURBQXVEO1FBQ3ZELElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQztJQUMxQixDQUFDO0lBRU0sZUFBZTtRQUNsQixJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLFdBQVcsSUFBSSxFQUFFLENBQUM7SUFDakUsQ0FBQzs7Z0lBbkNRLG1DQUFtQztvSEFBbkMsbUNBQW1DOzJGQUFuQyxtQ0FBbUM7a0JBSC9DLFNBQVM7bUJBQUM7b0JBQ1AsUUFBUSxFQUFFLDhCQUE4QjtpQkFDM0M7aUdBT21CLE9BQU87c0JBQXRCLEtBQUs7Z0JBS1UsTUFBTTtzQkFBckIsS0FBSztnQkFhSywwQkFBMEI7c0JBRHBDLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0FmdGVyVmlld0luaXQsIERpcmVjdGl2ZSwgRWxlbWVudFJlZiwgSW5wdXQsIE9uSW5pdH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBEaXJlY3RpdmUoe1xuICAgIHNlbGVjdG9yOiAnW25hdHVyYWxDb2x1bW5zUGlja2VyQ29sdW1uXScsXG59KVxuZXhwb3J0IGNsYXNzIE5hdHVyYWxDb2x1bW5zUGlja2VyQ29sdW1uRGlyZWN0aXZlIGltcGxlbWVudHMgQWZ0ZXJWaWV3SW5pdCwgT25Jbml0IHtcbiAgICBwdWJsaWMga2V5ITogc3RyaW5nO1xuXG4gICAgLyoqXG4gICAgICogSW5pdGlhbCBjaGVja2VkIHN0YXRlXG4gICAgICovXG4gICAgQElucHV0KCkgcHVibGljIGNoZWNrZWQgPSB0cnVlO1xuXG4gICAgLyoqXG4gICAgICogSW5pdGlhbCB2aXNpYmlsaXR5IHN0YXRlXG4gICAgICovXG4gICAgQElucHV0KCkgcHVibGljIGhpZGRlbiA9IGZhbHNlO1xuXG4gICAgLyoqXG4gICAgICogTG9jYWxpemVkIGxhYmVsIG9mIGNvbHVtbiwgaWYgYWJzZW50IGRlZmF1bHQgdG8ga2V5XG4gICAgICovXG4gICAgcHVibGljIGxhYmVsOiBzdHJpbmcgPSAnJztcblxuICAgIHB1YmxpYyBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IGVsZW1lbnRSZWY6IEVsZW1lbnRSZWY8SFRNTEVsZW1lbnQ+KSB7fVxuXG4gICAgLyoqXG4gICAgICogVGhpcyBtdXN0IGJlIHRoZSBjb2x1bW4ga2V5IGFzIGRlZmluZWQgaW4gbWF0Q29sdW1uRGVmXG4gICAgICovXG4gICAgQElucHV0KClcbiAgICBwdWJsaWMgc2V0IG5hdHVyYWxDb2x1bW5zUGlja2VyQ29sdW1uKHZhbHVlOiBzdHJpbmcpIHtcbiAgICAgICAgdGhpcy5rZXkgPSB2YWx1ZTtcbiAgICB9XG5cbiAgICBwdWJsaWMgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgICAgIC8vIERlZmF1bHQgbGFiZWwgdG8ga2V5IGJlZm9yZSByZWFsIGxhYmVsIGlzIGFjY2Vzc2libGVcbiAgICAgICAgdGhpcy5sYWJlbCA9IHRoaXMua2V5O1xuICAgIH1cblxuICAgIHB1YmxpYyBuZ0FmdGVyVmlld0luaXQoKTogdm9pZCB7XG4gICAgICAgIHRoaXMubGFiZWwgPSB0aGlzLmVsZW1lbnRSZWYubmF0aXZlRWxlbWVudC50ZXh0Q29udGVudCA/PyAnJztcbiAgICB9XG59XG4iXX0=
|
|
@@ -46,7 +46,7 @@ export class NaturalLinkableTabDirective extends NaturalAbstractController {
|
|
|
46
46
|
}
|
|
47
47
|
});
|
|
48
48
|
// When mat-tab-groups selected tab change, update url
|
|
49
|
-
this.component.selectedTabChange.pipe(takeUntil(this.ngUnsubscribe)).subscribe(
|
|
49
|
+
this.component.selectedTabChange.pipe(takeUntil(this.ngUnsubscribe)).subscribe(event => {
|
|
50
50
|
const activatedTabName = getTabId(event.tab);
|
|
51
51
|
const segments = this.route.snapshot.url;
|
|
52
52
|
if (!segments.length) {
|
|
@@ -76,4 +76,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImpor
|
|
|
76
76
|
}], ctorParameters: function () { return [{ type: i1.MatTabGroup }, { type: i2.ActivatedRoute }, { type: i2.Router }]; }, propDecorators: { naturalLinkableTab: [{
|
|
77
77
|
type: Input
|
|
78
78
|
}] } });
|
|
79
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlua2FibGUtdGFiLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2NvbW1vbi9kaXJlY3RpdmVzL2xpbmthYmxlLXRhYi5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFnQixTQUFTLEVBQUUsS0FBSyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRzlELE9BQU8sRUFBQyxLQUFLLEVBQUMsTUFBTSxXQUFXLENBQUM7QUFDaEMsT0FBTyxFQUFDLFNBQVMsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBQ3pDLE9BQU8sRUFBQyx5QkFBeUIsRUFBQyxNQUFNLHNDQUFzQyxDQUFDOzs7O0FBRS9FOztHQUVHO0FBQ0gsU0FBUyxRQUFRLENBQUMsR0FBVztJQUN6QixPQUFPLEdBQUcsQ0FBQyxPQUFPLEVBQUUsZ0JBQWdCLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxFQUFFLElBQUksRUFBRSxDQUFDO0FBQ3hFLENBQUM7QUFFRDs7Ozs7Ozs7R0FRRztBQUlILE1BQU0sT0FBTywyQkFBNEIsU0FBUSx5QkFBeUI7SUFNdEUsWUFDcUIsU0FBc0IsRUFDdEIsS0FBcUIsRUFDckIsTUFBYztRQUUvQixLQUFLLEVBQUUsQ0FBQztRQUpTLGNBQVMsR0FBVCxTQUFTLENBQWE7UUFDdEIsVUFBSyxHQUFMLEtBQUssQ0FBZ0I7UUFDckIsV0FBTSxHQUFOLE1BQU0sQ0FBUTtRQVJuQzs7V0FFRztRQUNhLHVCQUFrQixHQUFpQixJQUFJLENBQUM7SUFReEQsQ0FBQztJQUVNLGVBQWU7UUFDbEIsSUFBSSxJQUFJLENBQUMsa0JBQWtCLEtBQUssS0FBSyxFQUFFO1lBQ25DLE9BQU87U0FDVjtRQUVELGdFQUFnRTtRQUNoRSxJQUFJLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUMsRUFBRTtZQUN6RSw0Q0FBNEM7WUFDNUMsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxRQUFRLENBQUMsQ0FBQztZQUU1QyxvREFBb0Q7WUFDcEQsNkVBQTZFO1lBQzdFLElBQUksUUFBUSxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRTtnQkFDNUIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxhQUFhLEdBQUcsUUFBUSxDQUFDO2FBQzNDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7UUFFSCxzREFBc0Q7UUFDdEQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxpQkFBaUIsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUNuRixNQUFNLGdCQUFnQixHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDN0MsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDO1lBQ3pDLElBQUksQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFO2dCQUNsQix1R0FBdUc7Z0JBQ3ZHLE1BQU0sSUFBSSxLQUFLLENBQUMsd0RBQXdELENBQUMsQ0FBQzthQUM3RTtZQUVELHdGQUF3RjtZQUN4RixNQUFNLE1BQU0sR0FBRyxLQUFLLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLENBQUM7WUFFL0QsSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQyxHQUFHLEVBQUUsTUFBTSxDQUFDLEVBQUU7Z0JBQ2hDLFVBQVUsRUFBRSxJQUFJLENBQUMsS0FBSztnQkFDdEIsbUJBQW1CLEVBQUUsVUFBVTtnQkFDL0IsUUFBUSxFQUFFLGdCQUFnQixJQUFJLGdCQUFnQixDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLFNBQVM7YUFDdkYsQ0FBQyxDQUFDO1FBQ1AsQ0FBQyxDQUFDLENBQUM7SUFDUCxDQUFDO0lBRU8sV0FBVyxDQUFDLFFBQXVCO1FBQ3ZDLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsT0FBTyxFQUFFLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsUUFBUSxLQUFLLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ3ZGLENBQUM7O3dIQXJEUSwyQkFBMkI7NEdBQTNCLDJCQUEyQjsyRkFBM0IsMkJBQTJCO2tCQUh2QyxTQUFTO21CQUFDO29CQUNQLFFBQVEsRUFBRSxtQ0FBbUM7aUJBQ2hEO29KQUttQixrQkFBa0I7c0JBQWpDLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0FmdGVyVmlld0luaXQsIERpcmVjdGl2ZSwgSW5wdXR9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtNYXRUYWIsIE1hdFRhYkdyb3VwfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC90YWJzJztcbmltcG9ydCB7QWN0aXZhdGVkUm91dGUsIFJvdXRlcn0gZnJvbSAnQGFuZ3VsYXIvcm91dGVyJztcbmltcG9ydCB7Y2xvbmV9IGZyb20gJ2xvZGFzaC1lcyc7XG5pbXBvcnQge3Rha2VVbnRpbH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHtOYXR1cmFsQWJzdHJhY3RDb250cm9sbGVyfSBmcm9tICcuLi8uLi8uLi9jbGFzc2VzL2Fic3RyYWN0LWNvbnRyb2xsZXInO1xuXG4vKipcbiAqIFJldHVybnMgYW4gaWRlbnRpZmllciBmb3IgdGhlIHRhYlxuICovXG5mdW5jdGlvbiBnZXRUYWJJZCh0YWI6IE1hdFRhYik6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRhYi5jb250ZW50Py52aWV3Q29udGFpbmVyUmVmLmVsZW1lbnQubmF0aXZlRWxlbWVudC5pZCA/PyAnJztcbn1cblxuLyoqXG4gKiBVc2FnZSA6XG4gKlxuICogPG1hdC10YWItZ3JvdXAgW25hdHVyYWxMaW5rYWJsZVRhYl09XCIhaXNQYW5lbFwiPlxuICogICAgIDxtYXQtdGFiIGxhYmVsPVwiVGhpcmQgMVwiPnRoaXJkIDE8L21hdC10YWI+IC8vIEZpcnN0IHRhYiBkb2Vzbid0IG5lZWQgaWQuIFRoaXMga2VlcHMgdXJsIGNsZWFuIG9uIGRlZmF1bHQgb25lXG4gKiAgICAgPG1hdC10YWIgbGFiZWw9XCJUaGlyZCAyXCIgaWQ9XCJ0aGlyZDJcIj5UaGlyZCAyPC9tYXQtdGFiPlxuICogICAgIC4uLlxuICogPC9tYXQtdGFiLWdyb3VwPlxuICovXG5ARGlyZWN0aXZlKHtcbiAgICBzZWxlY3RvcjogJ21hdC10YWItZ3JvdXBbbmF0dXJhbExpbmthYmxlVGFiXScsXG59KVxuZXhwb3J0IGNsYXNzIE5hdHVyYWxMaW5rYWJsZVRhYkRpcmVjdGl2ZSBleHRlbmRzIE5hdHVyYWxBYnN0cmFjdENvbnRyb2xsZXIgaW1wbGVtZW50cyBBZnRlclZpZXdJbml0IHtcbiAgICAvKipcbiAgICAgKiBJZiBmYWxzZSwgZGlzYWJsZXMgdGhlIHBlcnNpc3RlbnQgbmF2aWdhdGlvblxuICAgICAqL1xuICAgIEBJbnB1dCgpIHB1YmxpYyBuYXR1cmFsTGlua2FibGVUYWI6IGJvb2xlYW4gfCAnJyA9IHRydWU7XG5cbiAgICBwdWJsaWMgY29uc3RydWN0b3IoXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgY29tcG9uZW50OiBNYXRUYWJHcm91cCxcbiAgICAgICAgcHJpdmF0ZSByZWFkb25seSByb3V0ZTogQWN0aXZhdGVkUm91dGUsXG4gICAgICAgIHByaXZhdGUgcmVhZG9ubHkgcm91dGVyOiBSb3V0ZXIsXG4gICAgKSB7XG4gICAgICAgIHN1cGVyKCk7XG4gICAgfVxuXG4gICAgcHVibGljIG5nQWZ0ZXJWaWV3SW5pdCgpOiB2b2lkIHtcbiAgICAgICAgaWYgKHRoaXMubmF0dXJhbExpbmthYmxlVGFiID09PSBmYWxzZSkge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gV2hlbiB1cmwgcGFyYW1zIGNoYW5nZSwgdXBkYXRlIHRoZSBtYXQtdGFiLWdyb3VwIHNlbGVjdGVkIHRhYlxuICAgICAgICB0aGlzLnJvdXRlLmZyYWdtZW50LnBpcGUodGFrZVVudGlsKHRoaXMubmdVbnN1YnNjcmliZSkpLnN1YnNjcmliZShmcmFnbWVudCA9PiB7XG4gICAgICAgICAgICAvLyBHZXQgaW5kZXggb2YgdGFiIHRoYXQgbWF0Y2hlcyB3YW50ZWQgbmFtZVxuICAgICAgICAgICAgY29uc3QgdGFiSW5kZXggPSB0aGlzLmdldFRhYkluZGV4KGZyYWdtZW50KTtcblxuICAgICAgICAgICAgLy8gSWYgdGFiIGluZGV4IGlzIHZhbGlkICg+PSAwKSBnbyB0byBnaXZlbiBmcmFnbWVudFxuICAgICAgICAgICAgLy8gSWYgdGhlcmUgaXMgbm8gZnJhZ21lbnQgYXQgYWxsLCBnbyB0byBmaXJzdCB0YWIgKGluZGV4IGlzIC0xIGluIHRoaXMgY2FzZSlcbiAgICAgICAgICAgIGlmICh0YWJJbmRleCA+PSAwIHx8ICFmcmFnbWVudCkge1xuICAgICAgICAgICAgICAgIHRoaXMuY29tcG9uZW50LnNlbGVjdGVkSW5kZXggPSB0YWJJbmRleDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gV2hlbiBtYXQtdGFiLWdyb3VwcyBzZWxlY3RlZCB0YWIgY2hhbmdlLCB1cGRhdGUgdXJsXG4gICAgICAgIHRoaXMuY29tcG9uZW50LnNlbGVjdGVkVGFiQ2hhbmdlLnBpcGUodGFrZVVudGlsKHRoaXMubmdVbnN1YnNjcmliZSkpLnN1YnNjcmliZShldmVudCA9PiB7XG4gICAgICAgICAgICBjb25zdCBhY3RpdmF0ZWRUYWJOYW1lID0gZ2V0VGFiSWQoZXZlbnQudGFiKTtcbiAgICAgICAgICAgIGNvbnN0IHNlZ21lbnRzID0gdGhpcy5yb3V0ZS5zbmFwc2hvdC51cmw7XG4gICAgICAgICAgICBpZiAoIXNlZ21lbnRzLmxlbmd0aCkge1xuICAgICAgICAgICAgICAgIC8vIFRoaXMgc2hvdWxkIG5ldmVyIGhhcHBlbiBpbiBub3JtYWwgdXNhZ2UsIGJlY2F1c2UgaXQgd291bGQgbWVhbnMgdGhlcmUgaXMgbm8gcm91dGUgYXQgYWxsIGluIHRoZSBhcHBcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ0Nhbm5vdCB1cGRhdGUgVVJMIGZvciB0YWJzIHdpdGhvdXQgYW55IHNlZ21lbnRzIGluIFVSTCcpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAvLyBHZXQgdXJsIG1hdHJpeCBwYXJhbXMgKC9zZWdtZW50O21hdHJpeD1wYXJhbSkgb25seSB3aXRob3V0IHJvdXRlIHBhcmFtcyAoc2VnbWVudC86aWQpXG4gICAgICAgICAgICBjb25zdCBwYXJhbXMgPSBjbG9uZShzZWdtZW50c1tzZWdtZW50cy5sZW5ndGggLSAxXS5wYXJhbWV0ZXJzKTtcblxuICAgICAgICAgICAgdGhpcy5yb3V0ZXIubmF2aWdhdGUoWycuJywgcGFyYW1zXSwge1xuICAgICAgICAgICAgICAgIHJlbGF0aXZlVG86IHRoaXMucm91dGUsXG4gICAgICAgICAgICAgICAgcXVlcnlQYXJhbXNIYW5kbGluZzogJ3ByZXNlcnZlJyxcbiAgICAgICAgICAgICAgICBmcmFnbWVudDogYWN0aXZhdGVkVGFiTmFtZSAmJiBhY3RpdmF0ZWRUYWJOYW1lLmxlbmd0aCA/IGFjdGl2YXRlZFRhYk5hbWUgOiB1bmRlZmluZWQsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSk7XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBnZXRUYWJJbmRleChmcmFnbWVudDogc3RyaW5nIHwgbnVsbCk6IG51bWJlciB7XG4gICAgICAgIHJldHVybiB0aGlzLmNvbXBvbmVudC5fdGFicy50b0FycmF5KCkuZmluZEluZGV4KHRhYiA9PiBmcmFnbWVudCA9PT0gZ2V0VGFiSWQodGFiKSk7XG4gICAgfVxufVxuIl19
|
|
@@ -1,11 +1,20 @@
|
|
|
1
1
|
import { DOCUMENT } from '@angular/common';
|
|
2
2
|
import { HttpHeaders } from '@angular/common/http';
|
|
3
3
|
import { ErrorHandler, Inject, Injectable, InjectionToken, Optional } from '@angular/core';
|
|
4
|
-
import { catchError, EMPTY } from 'rxjs';
|
|
4
|
+
import { catchError, EMPTY, first, of } from 'rxjs';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
import * as i1 from "@angular/common/http";
|
|
7
|
-
export const NaturalLoggerConfigUrl = new InjectionToken('
|
|
8
|
-
export const NaturalLoggerConfigExtra = new InjectionToken('
|
|
7
|
+
export const NaturalLoggerConfigUrl = new InjectionToken('Absolute URL of the log server');
|
|
8
|
+
export const NaturalLoggerConfigExtra = new InjectionToken('Class that may provide extra data to log');
|
|
9
|
+
/**
|
|
10
|
+
* Replace Angular's error handler to also send the log to a remote server via HTTP POST.
|
|
11
|
+
*
|
|
12
|
+
* Usage is automatic as soon we import the module via:
|
|
13
|
+
*
|
|
14
|
+
* ```ts
|
|
15
|
+
* NaturalErrorModule.forRoot('http://example.com', ExtraService),
|
|
16
|
+
* ```
|
|
17
|
+
*/
|
|
9
18
|
export class NaturalErrorHandler extends ErrorHandler {
|
|
10
19
|
constructor(http, document, url, loggerExtra) {
|
|
11
20
|
super();
|
|
@@ -17,18 +26,13 @@ export class NaturalErrorHandler extends ErrorHandler {
|
|
|
17
26
|
handleError(error) {
|
|
18
27
|
console.error(error);
|
|
19
28
|
const params = {
|
|
29
|
+
message: this.toMessage(error),
|
|
20
30
|
href: this.document.defaultView?.window.location.href,
|
|
21
31
|
host: this.document.defaultView?.window.location.hostname,
|
|
22
32
|
path: this.document.defaultView?.window.location.pathname,
|
|
23
33
|
agent: this.document.defaultView?.window.navigator.userAgent,
|
|
24
34
|
level: 'error',
|
|
25
35
|
};
|
|
26
|
-
if (error?.message) {
|
|
27
|
-
params.message = error.message;
|
|
28
|
-
}
|
|
29
|
-
else {
|
|
30
|
-
params.message = error;
|
|
31
|
-
}
|
|
32
36
|
if (error?.stack) {
|
|
33
37
|
params.stacktrace = error.stack;
|
|
34
38
|
}
|
|
@@ -42,7 +46,10 @@ export class NaturalErrorHandler extends ErrorHandler {
|
|
|
42
46
|
params.url = error.url;
|
|
43
47
|
}
|
|
44
48
|
if (this.loggerExtra) {
|
|
45
|
-
this.loggerExtra
|
|
49
|
+
this.loggerExtra
|
|
50
|
+
.getExtras(error)
|
|
51
|
+
.pipe(catchError(e => of({ getExtrasErrorMessage: this.toMessage(e) })), first())
|
|
52
|
+
.subscribe(result => {
|
|
46
53
|
this.postLog(Object.assign(params, result));
|
|
47
54
|
});
|
|
48
55
|
}
|
|
@@ -50,6 +57,14 @@ export class NaturalErrorHandler extends ErrorHandler {
|
|
|
50
57
|
this.postLog(params);
|
|
51
58
|
}
|
|
52
59
|
}
|
|
60
|
+
toMessage(error) {
|
|
61
|
+
if (error && typeof error === 'object' && 'message' in error) {
|
|
62
|
+
return '' + error.message;
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
return '' + error;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
53
68
|
/**
|
|
54
69
|
* Send parameters to remote log
|
|
55
70
|
*/
|
|
@@ -85,4 +100,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImpor
|
|
|
85
100
|
type: Inject,
|
|
86
101
|
args: [NaturalLoggerConfigExtra]
|
|
87
102
|
}] }]; } });
|
|
88
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"error-handler.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/logger/error-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import {DOCUMENT} from '@angular/common';\nimport {HttpClient, HttpHeaders} from '@angular/common/http';\nimport {ErrorHandler, Inject, Injectable, InjectionToken, Optional} from '@angular/core';\nimport {catchError, EMPTY, first, Observable, of} from 'rxjs';\n\nexport interface NaturalLoggerType {\n    message: string;\n    stacktrace?: string;\n    href?: string;\n    host?: string;\n    path?: string;\n    agent?: string;\n    status?: number;\n    referrer?: string;\n    url?: string;\n    userId?: string;\n    user?: string;\n\n    [key: string]: any;\n}\n\nexport interface NaturalLoggerExtra {\n    /**\n     * Return an observable of extra data that will be logged. Those data will be merged into\n     * the original data, and so it can override things.\n     *\n     * Only the first emitted value will be used.\n     */\n    getExtras(error: unknown): Observable<Partial<NaturalLoggerType>>;\n}\n\nexport const NaturalLoggerConfigUrl = new InjectionToken<string>('Absolute URL of the log server');\nexport const NaturalLoggerConfigExtra = new InjectionToken<NaturalLoggerExtra>(\n    'Class that may provide extra data to log',\n);\n\n/**\n * Replace Angular's error handler to also send the log to a remote server via HTTP POST.\n *\n * Usage is automatic as soon we import the module via:\n *\n * ```ts\n * NaturalErrorModule.forRoot('http://example.com', ExtraService),\n * ```\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalErrorHandler extends ErrorHandler {\n    public constructor(\n        private readonly http: HttpClient,\n        @Inject(DOCUMENT) private readonly document: Document,\n        @Optional() @Inject(NaturalLoggerConfigUrl) private readonly url: string,\n        @Optional() @Inject(NaturalLoggerConfigExtra) private readonly loggerExtra?: NaturalLoggerExtra,\n    ) {\n        super();\n    }\n\n    public handleError(error: any): void {\n        console.error(error);\n\n        const params: NaturalLoggerType = {\n            message: this.toMessage(error),\n            href: this.document.defaultView?.window.location.href,\n            host: this.document.defaultView?.window.location.hostname,\n            path: this.document.defaultView?.window.location.pathname,\n            agent: this.document.defaultView?.window.navigator.userAgent,\n            level: 'error',\n        };\n\n        if (error?.stack) {\n            params.stacktrace = error.stack;\n        }\n\n        if (typeof error?.status !== 'undefined') {\n            params.status = error.status;\n        }\n\n        if (error?.referrer) {\n            params.referrer = error.referrer;\n        }\n\n        if (error?.url) {\n            params.url = error.url;\n        }\n\n        if (this.loggerExtra) {\n            this.loggerExtra\n                .getExtras(error)\n                .pipe(\n                    catchError(e => of({getExtrasErrorMessage: this.toMessage(e)})),\n                    first(),\n                )\n                .subscribe(result => {\n                    this.postLog(Object.assign(params, result));\n                });\n        } else {\n            this.postLog(params);\n        }\n    }\n\n    private toMessage(error: any): string {\n        if (error && typeof error === 'object' && 'message' in error) {\n            return '' + error.message;\n        } else {\n            return '' + error;\n        }\n    }\n\n    /**\n     * Send parameters to remote log\n     */\n    private postLog(params: NaturalLoggerType): void {\n        if (this.url) {\n            this.http\n                .post(this.url, params, {headers: new HttpHeaders().set('content-type', 'application/json')})\n                .pipe(\n                    catchError(() => {\n                        return EMPTY;\n                    }),\n                )\n                .subscribe();\n        }\n    }\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/*
|
|
2
2
|
* Public API Surface of natural
|
|
3
3
|
*/
|
|
4
|
-
export
|
|
5
|
-
export
|
|
6
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
4
|
+
export { NaturalErrorModule } from './error.module';
|
|
5
|
+
export { NaturalErrorHandler, NaturalLoggerConfigExtra, NaturalLoggerConfigUrl, } from './error-handler';
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2xvZ2dlci9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFFbEQsT0FBTyxFQUNILG1CQUFtQixFQUNuQix3QkFBd0IsRUFDeEIsc0JBQXNCLEdBR3pCLE1BQU0saUJBQWlCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIG5hdHVyYWxcbiAqL1xuXG5leHBvcnQge05hdHVyYWxFcnJvck1vZHVsZX0gZnJvbSAnLi9lcnJvci5tb2R1bGUnO1xuXG5leHBvcnQge1xuICAgIE5hdHVyYWxFcnJvckhhbmRsZXIsXG4gICAgTmF0dXJhbExvZ2dlckNvbmZpZ0V4dHJhLFxuICAgIE5hdHVyYWxMb2dnZXJDb25maWdVcmwsXG4gICAgTmF0dXJhbExvZ2dlckV4dHJhLFxuICAgIE5hdHVyYWxMb2dnZXJUeXBlLFxufSBmcm9tICcuL2Vycm9yLWhhbmRsZXInO1xuIl19
|
|
@@ -52,6 +52,7 @@ export class NaturalRelationsComponent extends NaturalAbstractController {
|
|
|
52
52
|
* Observable variables/options for listing service usage and apollo watchQuery
|
|
53
53
|
*/
|
|
54
54
|
this.variablesManager = new NaturalQueryVariablesManager();
|
|
55
|
+
this.removing = new Set();
|
|
55
56
|
}
|
|
56
57
|
/**
|
|
57
58
|
* The filter used to filter relations
|
|
@@ -85,7 +86,11 @@ export class NaturalRelationsComponent extends NaturalAbstractController {
|
|
|
85
86
|
* Refetch result to display it in table
|
|
86
87
|
*/
|
|
87
88
|
removeRelation(relation) {
|
|
88
|
-
this.
|
|
89
|
+
this.removing.add(relation);
|
|
90
|
+
this.linkMutationService
|
|
91
|
+
.unlink(this.main, relation, this.otherName)
|
|
92
|
+
.pipe(finalize(() => this.removing.delete(relation)))
|
|
93
|
+
.subscribe(() => this.dataSource?.remove(relation));
|
|
89
94
|
}
|
|
90
95
|
/**
|
|
91
96
|
* Link action
|
|
@@ -162,10 +167,10 @@ export class NaturalRelationsComponent extends NaturalAbstractController {
|
|
|
162
167
|
}
|
|
163
168
|
}
|
|
164
169
|
NaturalRelationsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: NaturalRelationsComponent, deps: [{ token: i1.NaturalLinkMutationService }, { token: i2.NaturalHierarchicSelectorDialogService }], target: i0.ɵɵFactoryTarget.Component });
|
|
165
|
-
NaturalRelationsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.0", type: NaturalRelationsComponent, selector: "natural-relations", inputs: { service: "service", placeholder: "placeholder", autocompleteSelectorFilter: "autocompleteSelectorFilter", displayWith: "displayWith", disabled: "disabled", main: "main", hierarchicSelectorFilters: "hierarchicSelectorFilters", hierarchicSelectorConfig: "hierarchicSelectorConfig", otherName: "otherName", filter: "filter" }, outputs: { selectionChange: "selectionChange" }, queries: [{ propertyName: "itemTemplate", first: true, predicate: TemplateRef, descendants: true }], viewQueries: [{ propertyName: "select", first: true, predicate: NaturalSelectComponent, descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"body\">\n <ng-template #defaultNameCell let-item=\"item\">\n {{ getDisplayFn()(item) }}\n </ng-template>\n\n <table *ngIf=\"dataSource\" [dataSource]=\"dataSource\" class=\"natural-row-click\" mat-table>\n <tr *matHeaderRowDef=\"displayedColumns\" mat-header-row style=\"display: none\"></tr>\n <tr *matRowDef=\"let row; columns: displayedColumns\" mat-row></tr>\n\n <ng-container matColumnDef=\"name\">\n <th *matHeaderCellDef i18n mat-header-cell>Titre</th>\n <td *matCellDef=\"let item\" mat-cell>\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultNameCell\"\n ></ng-template>\n </td>\n </ng-container>\n\n <ng-container matColumnDef=\"unlink\">\n <th *matHeaderCellDef mat-header-cell></th>\n <td *matCellDef=\"let element\" mat-cell>\n <button\n *ngIf=\"!disabled\"\n (click)=\"removeRelation(element)\"\n color=\"warn\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Dissocier\"\n >\n <natural-icon name=\"link_off\"></natural-icon>\n </button>\n </td>\n </ng-container>\n </table>\n\n <mat-paginator\n (page)=\"pagination($event)\"\n *ngIf=\"dataSource?.data && (dataSource?.data?.length || 0) > (dataSource?.data?.pageSize || 0)\"\n [length]=\"dataSource?.data?.length || 0\"\n [pageIndex]=\"dataSource?.data?.pageIndex || 0\"\n [pageSizeOptions]=\"pageSizeOptions\"\n [pageSize]=\"dataSource?.data?.pageSize || 0\"\n ></mat-paginator>\n\n <div *ngIf=\"!loading && dataSource?.data?.length === 0\" class=\"margin-v mat-body\">\n <span i18n>Aucun r\u00E9sultat</span>\n </div>\n\n <mat-progress-spinner *ngIf=\"loading\" [diameter]=\"40\" class=\"loading\" mode=\"indeterminate\"></mat-progress-spinner>\n</div>\n\n<natural-select\n #select\n (selectionChange)=\"addRelations([$any($event)])\"\n *ngIf=\"!hierarchicSelectorConfig && service && !disabled\"\n [displayWith]=\"getDisplayFn()\"\n [filter]=\"autocompleteSelectorFilter\"\n [placeholder]=\"placeholder\"\n [service]=\"service\"\n [showIcon]=\"false\"\n></natural-select>\n\n<div *ngIf=\"hierarchicSelectorConfig && !disabled\">\n <button (click)=\"openNaturalHierarchicSelector()\" color=\"primary\" mat-flat-button>{{ placeholder }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host .body{display:flex;flex-direction:column}:host .loading{margin:20px auto}:host .mat-column-unlink{width:2.5em}\n"], components: [{ type: i3.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { type: i3.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { type: i3.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i5.NaturalIconComponent, selector: "natural-icon", inputs: ["label", "labelColor", "labelPosition", "name", "size"] }, { type: i6.MatPaginator, selector: "mat-paginator", inputs: ["disabled"], exportAs: ["matPaginator"] }, { type: i7.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }, { type: i8.NaturalSelectComponent, selector: "natural-select", inputs: ["service", "optionRequired", "searchField", "filter", "disabled"] }], directives: [{ type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { type: i3.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { type: i3.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { type: i3.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { type: i3.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { type: i3.MatCellDef, selector: "[matCellDef]" }, { type: i3.MatCell, selector: "mat-cell, td[mat-cell]" }, { type: i9.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i10.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }] });
|
|
170
|
+
NaturalRelationsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.0", type: NaturalRelationsComponent, selector: "natural-relations", inputs: { service: "service", placeholder: "placeholder", autocompleteSelectorFilter: "autocompleteSelectorFilter", displayWith: "displayWith", disabled: "disabled", main: "main", hierarchicSelectorFilters: "hierarchicSelectorFilters", hierarchicSelectorConfig: "hierarchicSelectorConfig", otherName: "otherName", filter: "filter" }, outputs: { selectionChange: "selectionChange" }, queries: [{ propertyName: "itemTemplate", first: true, predicate: TemplateRef, descendants: true }], viewQueries: [{ propertyName: "select", first: true, predicate: NaturalSelectComponent, descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"body\">\n <ng-template #defaultNameCell let-item=\"item\">\n {{ getDisplayFn()(item) }}\n </ng-template>\n\n <table *ngIf=\"dataSource\" [dataSource]=\"dataSource\" class=\"natural-row-click\" mat-table>\n <tr *matHeaderRowDef=\"displayedColumns\" mat-header-row style=\"display: none\"></tr>\n <tr *matRowDef=\"let row; columns: displayedColumns\" mat-row></tr>\n\n <ng-container matColumnDef=\"name\">\n <th *matHeaderCellDef i18n mat-header-cell>Titre</th>\n <td *matCellDef=\"let item\" mat-cell>\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultNameCell\"\n ></ng-template>\n </td>\n </ng-container>\n\n <ng-container matColumnDef=\"unlink\">\n <th *matHeaderCellDef mat-header-cell></th>\n <td *matCellDef=\"let element\" mat-cell>\n <button\n *ngIf=\"!disabled\"\n (click)=\"removeRelation(element)\"\n [disabled]=\"removing.has(element)\"\n color=\"warn\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Dissocier\"\n >\n <natural-icon name=\"link_off\"></natural-icon>\n </button>\n </td>\n </ng-container>\n </table>\n\n <mat-paginator\n (page)=\"pagination($event)\"\n *ngIf=\"dataSource?.data && (dataSource?.data?.length || 0) > (dataSource?.data?.pageSize || 0)\"\n [length]=\"dataSource?.data?.length || 0\"\n [pageIndex]=\"dataSource?.data?.pageIndex || 0\"\n [pageSizeOptions]=\"pageSizeOptions\"\n [pageSize]=\"dataSource?.data?.pageSize || 0\"\n ></mat-paginator>\n\n <div *ngIf=\"!loading && dataSource?.data?.length === 0\" class=\"margin-v mat-body\">\n <span i18n>Aucun r\u00E9sultat</span>\n </div>\n\n <mat-progress-spinner *ngIf=\"loading\" [diameter]=\"40\" class=\"loading\" mode=\"indeterminate\"></mat-progress-spinner>\n</div>\n\n<natural-select\n #select\n (selectionChange)=\"addRelations([$any($event)])\"\n *ngIf=\"!hierarchicSelectorConfig && service && !disabled\"\n [displayWith]=\"getDisplayFn()\"\n [filter]=\"autocompleteSelectorFilter\"\n [placeholder]=\"placeholder\"\n [service]=\"service\"\n [showIcon]=\"false\"\n></natural-select>\n\n<div *ngIf=\"hierarchicSelectorConfig && !disabled\">\n <button (click)=\"openNaturalHierarchicSelector()\" color=\"primary\" mat-flat-button>{{ placeholder }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host .body{display:flex;flex-direction:column}:host .loading{margin:20px auto}:host .mat-column-unlink{width:2.5em}\n"], components: [{ type: i3.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { type: i3.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { type: i3.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i5.NaturalIconComponent, selector: "natural-icon", inputs: ["label", "labelColor", "labelPosition", "name", "size"] }, { type: i6.MatPaginator, selector: "mat-paginator", inputs: ["disabled"], exportAs: ["matPaginator"] }, { type: i7.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }, { type: i8.NaturalSelectComponent, selector: "natural-select", inputs: ["service", "optionRequired", "searchField", "filter", "disabled"] }], directives: [{ type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { type: i3.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { type: i3.MatColumnDef, selector: "[matColumnDef]", inputs: ["sticky", "matColumnDef"] }, { type: i3.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { type: i3.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { type: i3.MatCellDef, selector: "[matCellDef]" }, { type: i3.MatCell, selector: "mat-cell, td[mat-cell]" }, { type: i9.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i10.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }] });
|
|
166
171
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: NaturalRelationsComponent, decorators: [{
|
|
167
172
|
type: Component,
|
|
168
|
-
args: [{ selector: 'natural-relations', template: "<div class=\"body\">\n <ng-template #defaultNameCell let-item=\"item\">\n {{ getDisplayFn()(item) }}\n </ng-template>\n\n <table *ngIf=\"dataSource\" [dataSource]=\"dataSource\" class=\"natural-row-click\" mat-table>\n <tr *matHeaderRowDef=\"displayedColumns\" mat-header-row style=\"display: none\"></tr>\n <tr *matRowDef=\"let row; columns: displayedColumns\" mat-row></tr>\n\n <ng-container matColumnDef=\"name\">\n <th *matHeaderCellDef i18n mat-header-cell>Titre</th>\n <td *matCellDef=\"let item\" mat-cell>\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultNameCell\"\n ></ng-template>\n </td>\n </ng-container>\n\n <ng-container matColumnDef=\"unlink\">\n <th *matHeaderCellDef mat-header-cell></th>\n <td *matCellDef=\"let element\" mat-cell>\n <button\n *ngIf=\"!disabled\"\n (click)=\"removeRelation(element)\"\n color=\"warn\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Dissocier\"\n >\n <natural-icon name=\"link_off\"></natural-icon>\n </button>\n </td>\n </ng-container>\n </table>\n\n <mat-paginator\n (page)=\"pagination($event)\"\n *ngIf=\"dataSource?.data && (dataSource?.data?.length || 0) > (dataSource?.data?.pageSize || 0)\"\n [length]=\"dataSource?.data?.length || 0\"\n [pageIndex]=\"dataSource?.data?.pageIndex || 0\"\n [pageSizeOptions]=\"pageSizeOptions\"\n [pageSize]=\"dataSource?.data?.pageSize || 0\"\n ></mat-paginator>\n\n <div *ngIf=\"!loading && dataSource?.data?.length === 0\" class=\"margin-v mat-body\">\n <span i18n>Aucun r\u00E9sultat</span>\n </div>\n\n <mat-progress-spinner *ngIf=\"loading\" [diameter]=\"40\" class=\"loading\" mode=\"indeterminate\"></mat-progress-spinner>\n</div>\n\n<natural-select\n #select\n (selectionChange)=\"addRelations([$any($event)])\"\n *ngIf=\"!hierarchicSelectorConfig && service && !disabled\"\n [displayWith]=\"getDisplayFn()\"\n [filter]=\"autocompleteSelectorFilter\"\n [placeholder]=\"placeholder\"\n [service]=\"service\"\n [showIcon]=\"false\"\n></natural-select>\n\n<div *ngIf=\"hierarchicSelectorConfig && !disabled\">\n <button (click)=\"openNaturalHierarchicSelector()\" color=\"primary\" mat-flat-button>{{ placeholder }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host .body{display:flex;flex-direction:column}:host .loading{margin:20px auto}:host .mat-column-unlink{width:2.5em}\n"] }]
|
|
173
|
+
args: [{ selector: 'natural-relations', template: "<div class=\"body\">\n <ng-template #defaultNameCell let-item=\"item\">\n {{ getDisplayFn()(item) }}\n </ng-template>\n\n <table *ngIf=\"dataSource\" [dataSource]=\"dataSource\" class=\"natural-row-click\" mat-table>\n <tr *matHeaderRowDef=\"displayedColumns\" mat-header-row style=\"display: none\"></tr>\n <tr *matRowDef=\"let row; columns: displayedColumns\" mat-row></tr>\n\n <ng-container matColumnDef=\"name\">\n <th *matHeaderCellDef i18n mat-header-cell>Titre</th>\n <td *matCellDef=\"let item\" mat-cell>\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultNameCell\"\n ></ng-template>\n </td>\n </ng-container>\n\n <ng-container matColumnDef=\"unlink\">\n <th *matHeaderCellDef mat-header-cell></th>\n <td *matCellDef=\"let element\" mat-cell>\n <button\n *ngIf=\"!disabled\"\n (click)=\"removeRelation(element)\"\n [disabled]=\"removing.has(element)\"\n color=\"warn\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Dissocier\"\n >\n <natural-icon name=\"link_off\"></natural-icon>\n </button>\n </td>\n </ng-container>\n </table>\n\n <mat-paginator\n (page)=\"pagination($event)\"\n *ngIf=\"dataSource?.data && (dataSource?.data?.length || 0) > (dataSource?.data?.pageSize || 0)\"\n [length]=\"dataSource?.data?.length || 0\"\n [pageIndex]=\"dataSource?.data?.pageIndex || 0\"\n [pageSizeOptions]=\"pageSizeOptions\"\n [pageSize]=\"dataSource?.data?.pageSize || 0\"\n ></mat-paginator>\n\n <div *ngIf=\"!loading && dataSource?.data?.length === 0\" class=\"margin-v mat-body\">\n <span i18n>Aucun r\u00E9sultat</span>\n </div>\n\n <mat-progress-spinner *ngIf=\"loading\" [diameter]=\"40\" class=\"loading\" mode=\"indeterminate\"></mat-progress-spinner>\n</div>\n\n<natural-select\n #select\n (selectionChange)=\"addRelations([$any($event)])\"\n *ngIf=\"!hierarchicSelectorConfig && service && !disabled\"\n [displayWith]=\"getDisplayFn()\"\n [filter]=\"autocompleteSelectorFilter\"\n [placeholder]=\"placeholder\"\n [service]=\"service\"\n [showIcon]=\"false\"\n></natural-select>\n\n<div *ngIf=\"hierarchicSelectorConfig && !disabled\">\n <button (click)=\"openNaturalHierarchicSelector()\" color=\"primary\" mat-flat-button>{{ placeholder }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host .body{display:flex;flex-direction:column}:host .loading{margin:20px auto}:host .mat-column-unlink{width:2.5em}\n"] }]
|
|
169
174
|
}], ctorParameters: function () { return [{ type: i1.NaturalLinkMutationService }, { type: i2.NaturalHierarchicSelectorDialogService }]; }, propDecorators: { select: [{
|
|
170
175
|
type: ViewChild,
|
|
171
176
|
args: [NaturalSelectComponent]
|
|
@@ -195,4 +200,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImpor
|
|
|
195
200
|
}], filter: [{
|
|
196
201
|
type: Input
|
|
197
202
|
}] } });
|
|
198
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"relations.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/relations/relations.component.ts","../../../../../../projects/natural/src/lib/modules/relations/relations.component.html"],"names":[],"mappings":"AAAA,OAAO,EACH,SAAS,EACT,YAAY,EACZ,YAAY,EACZ,KAAK,EAIL,MAAM,EAEN,WAAW,EACX,SAAS,GACZ,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAC,QAAQ,EAAC,MAAM,MAAM,CAAC;AAC9B,OAAO,EAAC,yBAAyB,EAAC,MAAM,mCAAmC,CAAC;AAC5E,OAAO,EAAC,iBAAiB,EAAgB,MAAM,2BAA2B,CAAC;AAC3E,OAAO,EAAC,4BAA4B,EAAkC,MAAM,sCAAsC,CAAC;AAOnH,OAAO,EAAC,sBAAsB,EAAC,MAAM,mCAAmC,CAAC;AACzE,OAAO,EAAC,QAAQ,EAAE,SAAS,EAAC,MAAM,gBAAgB,CAAC;;;;;;;;;;;;AAInD;;;;;;;GAOG;AAOH,MAAM,OAAO,yBAcT,SAAQ,yBAAyB;IAwEjC,YACqB,mBAA+C,EAC/C,wBAAgE;QAEjF,KAAK,EAAE,CAAC;QAHS,wBAAmB,GAAnB,mBAAmB,CAA4B;QAC/C,6BAAwB,GAAxB,wBAAwB,CAAwC;QAlErF;;WAEG;QACa,gBAAW,GAAG,EAAE,CAAC;QAYjC;;WAEG;QACa,aAAQ,GAAG,KAAK,CAAC;QAOP,oBAAe,GAAG,IAAI,YAAY,EAAQ,CAAC;QAqB9D,YAAO,GAAG,KAAK,CAAC;QAEvB;;WAEG;QACI,qBAAgB,GAAG,CAAC,MAAM,CAAC,CAAC;QAE5B,oBAAe,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE,GAAG,CAAC,CAAC;QAChC,sBAAiB,GAAG;YAC1B,SAAS,EAAE,CAAC;YACZ,QAAQ,EAAE,EAAE;SACf,CAAC;QAEF;;WAEG;QACK,qBAAgB,GAAiD,IAAI,4BAA4B,EAAE,CAAC;IAO5G,CAAC;IAED;;;;;OAKG;IACH,IAAoB,MAAM,CAAC,MAAc;QACrC,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,kBAAkB,EAAE,EAAC,MAAM,EAAE,MAAM,EAAC,CAAC,CAAC;IACpE,CAAC;IAEM,QAAQ;QACX,IAAI,CAAC,UAAU,EAAE,CAAC;QAElB,yCAAyC;QACzC,IAAI,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;YACpC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC;SAClE;IACL,CAAC;IAEM,WAAW,CAAC,OAAsB;QACrC,IAAI,IAAI,CAAC,OAAO,EAAE;YACd,IAAI,CAAC,UAAU,EAAE,CAAC;SACrB;QAED,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE;YAC/D,IAAI,CAAC,gBAAgB,CAAC,GAAG,EAAE,CAAC;SAC/B;aAAM,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE;YACzE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SACxC;IACL,CAAC;IAED;;;OAGG;IACI,cAAc,CAAC,QAAwB;QAC1C,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,SAAS,EAAE,CAAC;IACrF,CAAC;IAED;;;;OAIG;IACI,YAAY,CAAC,SAA2B;QAC3C,MAAM,WAAW,GAAG,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CACzC,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,CACrE,CAAC;QAEF,QAAQ,CAAC,WAAW,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE;YACjC,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC;YAC5B,IAAI,IAAI,CAAC,MAAM,EAAE;gBACb,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;aAC5B;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEM,UAAU,CAAC,KAAiB;QAC/B,IAAI,UAAU,GAA2B,IAAI,CAAC;QAC9C,IACI,KAAK;YACL,CAAC,KAAK,CAAC,SAAS,KAAK,IAAI,CAAC,iBAAiB,CAAC,SAAS,IAAI,KAAK,CAAC,QAAQ,KAAK,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAC9G;YACE,UAAU,GAAG;gBACT,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,QAAQ,EAAE,KAAK,CAAC,QAAQ;aAC3B,CAAC;SACL;QAED,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,YAAY,EAAE,EAAC,UAAU,EAAE,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,EAAC,CAAC,CAAC;IAC5G,CAAC;IAEM,YAAY;QACf,IAAI,IAAI,CAAC,WAAW,EAAE;YAClB,OAAO,IAAI,CAAC,WAAW,CAAC;SAC3B;QAED,OAAO,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAC5D,CAAC;IAEM,6BAA6B;QAChC,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QAExC,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,wBAAwB,EAAE;YAChD,OAAO;SACV;QAED,MAAM,QAAQ,GAAG,EAAE,CAAC;QAEpB,MAAM,gBAAgB,GAA2B;YAC7C,gBAAgB,EAAE,IAAI,CAAC,wBAAwB;YAC/C,mBAAmB,EAAE,QAAQ;YAC7B,iBAAiB,EAAE,IAAI,CAAC,yBAAyB;YACjD,QAAQ,EAAE,IAAI;SACjB,CAAC;QAEF,IAAI,CAAC,wBAAwB;aACxB,IAAI,CAAC,gBAAgB,CAAC;aACtB,WAAW,EAAE;aACb,SAAS,CAAC,MAAM,CAAC,EAAE;YAChB,IAAI,MAAM,IAAI,MAAM,CAAC,mBAAmB,KAAK,SAAS,EAAE;gBACpD,MAAM,SAAS,GAAG,MAAM,CAAC,mBAAmB,CAAC,WAAW,CAAC,CAAC;gBAC1D,IAAI,SAAS,CAAC,MAAM,EAAE;oBAClB,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;iBAChC;aACJ;QACL,CAAC,CAAC,CAAC;IACX,CAAC;IAED;;OAEG;IACK,UAAU;QACd,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACf,OAAO;SACV;QAED,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;QAClG,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC,CAAC;QAC9F,IAAI,CAAC,UAAU,GAAG,IAAI,iBAAiB,CAAC,QAAQ,CAAC,CAAC;IACtD,CAAC;IAEO,YAAY;QAChB,IAAI,CAAC,IAAI,CAAC,wBAAwB,EAAE;YAChC,OAAO;SACV;QAED,OAAO,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC;IAC7F,CAAC;;sHA9NQ,yBAAyB;0GAAzB,yBAAyB,keAkBpB,WAAW,wFADd,sBAAsB,4FC5DrC,inFAkEA;2FDvBa,yBAAyB;kBALrC,SAAS;+BACI,mBAAmB;sKAqBc,MAAM;sBAAhD,SAAS;uBAAC,sBAAsB;gBACC,YAAY;sBAA7C,YAAY;uBAAC,WAAW;gBAET,OAAO;sBAAtB,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAKU,0BAA0B;sBAAzC,KAAK;gBAKU,WAAW;sBAA1B,KAAK;gBAKU,QAAQ;sBAAvB,KAAK;gBAKU,IAAI;sBAAnB,KAAK;gBAEoB,eAAe;sBAAxC,MAAM;gBAKS,yBAAyB;sBAAxC,KAAK;gBAKU,wBAAwB;sBAAvC,KAAK;gBAKU,SAAS;sBAAxB,KAAK;gBAqCc,MAAM;sBAAzB,KAAK","sourcesContent":["import {\n    Component,\n    ContentChild,\n    EventEmitter,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    SimpleChanges,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {PageEvent} from '@angular/material/paginator';\nimport {forkJoin} from 'rxjs';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\nimport {NaturalDataSource, PaginatedData} from '../../classes/data-source';\nimport {NaturalQueryVariablesManager, PaginationInput, QueryVariables} from '../../classes/query-variable-manager';\nimport {HierarchicFiltersConfiguration} from '../../modules/hierarchic-selector/classes/hierarchic-filters-configuration';\nimport {LinkableObject, NaturalLinkMutationService} from '../../services/link-mutation.service';\nimport {NaturalHierarchicConfiguration} from '../hierarchic-selector/classes/hierarchic-configuration';\nimport {HierarchicDialogConfig} from '../hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component';\nimport {NaturalHierarchicSelectorDialogService} from '../hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service';\nimport {Filter} from '../search/classes/graphql-doctrine.types';\nimport {NaturalSelectComponent} from '../select/select/select.component';\nimport {finalize, takeUntil} from 'rxjs/operators';\nimport {NaturalAbstractModelService} from '../../services/abstract-model.service';\nimport {Literal} from '../../types/types';\n\n/**\n * Custom template usage :\n * <natural-relations [main]=\"owner\" [service]=\"svc\" [filter]=\"{}\" placeholder=\"Select an item\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-relations>\n */\n\n@Component({\n    selector: 'natural-relations',\n    templateUrl: './relations.component.html',\n    styleUrls: ['./relations.component.scss'],\n})\nexport class NaturalRelationsComponent<\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 NaturalAbstractController\n    implements OnInit, OnChanges, OnDestroy\n{\n    @ViewChild(NaturalSelectComponent) private select?: NaturalSelectComponent<TService>;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    @Input() public service?: TService;\n\n    /**\n     * The placeholder used in the button to add a new relation\n     */\n    @Input() public placeholder = '';\n\n    /**\n     * Filter for autocomplete selector\n     */\n    @Input() public autocompleteSelectorFilter?: Filter;\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: any) => string;\n\n    /**\n     * Whether the relations can be changed\n     */\n    @Input() public disabled = false;\n\n    /**\n     * The main object to which all relations belong to\n     */\n    @Input() public main!: LinkableObject & {permissions?: {update: boolean}};\n\n    @Output() public readonly selectionChange = new EventEmitter<void>();\n\n    /**\n     * Filters for hierarchic selector\n     */\n    @Input() public hierarchicSelectorFilters?: HierarchicFiltersConfiguration | null;\n\n    /**\n     * Configuration in case we prefer hierarchic selection over autocomplete selection\n     */\n    @Input() public hierarchicSelectorConfig?: NaturalHierarchicConfiguration[];\n\n    /**\n     * Link mutation semantic\n     */\n    @Input() public otherName?: string | null;\n\n    /**\n     * Listing service instance\n     */\n    public dataSource?: NaturalDataSource;\n    public loading = false;\n\n    /**\n     * Table columns\n     */\n    public displayedColumns = ['name'];\n\n    public pageSizeOptions = [5, 10, 50, 100];\n    protected defaultPagination = {\n        pageIndex: 0,\n        pageSize: 25,\n    };\n\n    /**\n     * Observable variables/options for listing service usage and apollo watchQuery\n     */\n    private variablesManager: NaturalQueryVariablesManager<QueryVariables> = new NaturalQueryVariablesManager();\n\n    public constructor(\n        private readonly linkMutationService: NaturalLinkMutationService,\n        private readonly hierarchicSelectorDialog: NaturalHierarchicSelectorDialogService,\n    ) {\n        super();\n    }\n\n    /**\n     * The filter used to filter relations\n     *\n     * So if the relations are from one action -> to many objectives, then the filter must filter\n     * the objectives that have indeed a relation to the particular action.\n     */\n    @Input() public set filter(filter: Filter) {\n        this.variablesManager.set('relations-filter', {filter: filter});\n    }\n\n    public ngOnInit(): void {\n        this.pagination();\n\n        // Force disabled if cannot update object\n        if (this.main && this.main.permissions) {\n            this.disabled = this.disabled || !this.main.permissions.update;\n        }\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (this.service) {\n            this.queryItems();\n        }\n\n        if (this.disabled && this.displayedColumns.indexOf('unlink') > -1) {\n            this.displayedColumns.pop();\n        } else if (!this.disabled && this.displayedColumns.indexOf('unlink') === -1) {\n            this.displayedColumns.push('unlink');\n        }\n    }\n\n    /**\n     * Unlink action\n     * Refetch result to display it in table\n     */\n    public removeRelation(relation: LinkableObject): void {\n        this.linkMutationService.unlink(this.main, relation, this.otherName).subscribe();\n    }\n\n    /**\n     * Link action\n     * Refetch result to display it in table\n     * TODO : could maybe use \"update\" attribute of apollo.mutate function to update table faster (but hard to do it here)\n     */\n    public addRelations(relations: LinkableObject[]): void {\n        const observables = relations.map(relation =>\n            this.linkMutationService.link(this.main, relation, this.otherName),\n        );\n\n        forkJoin(observables).subscribe(() => {\n            this.selectionChange.emit();\n            if (this.select) {\n                this.select.clear(false);\n            }\n        });\n    }\n\n    public pagination(event?: PageEvent): void {\n        let pagination: PaginationInput | null = null;\n        if (\n            event &&\n            (event.pageIndex !== this.defaultPagination.pageIndex || event.pageSize !== this.defaultPagination.pageSize)\n        ) {\n            pagination = {\n                pageIndex: event.pageIndex,\n                pageSize: event.pageSize,\n            };\n        }\n\n        this.variablesManager.set('pagination', {pagination: pagination ? pagination : this.defaultPagination});\n    }\n\n    public getDisplayFn(): (item: any) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return item => (item ? item.fullName || item.name : '');\n    }\n\n    public openNaturalHierarchicSelector(): void {\n        const selectAtKey = this.getSelectKey();\n\n        if (!selectAtKey || !this.hierarchicSelectorConfig) {\n            return;\n        }\n\n        const selected = {};\n\n        const hierarchicConfig: HierarchicDialogConfig = {\n            hierarchicConfig: this.hierarchicSelectorConfig,\n            hierarchicSelection: selected,\n            hierarchicFilters: this.hierarchicSelectorFilters,\n            multiple: true,\n        };\n\n        this.hierarchicSelectorDialog\n            .open(hierarchicConfig)\n            .afterClosed()\n            .subscribe(result => {\n                if (result && result.hierarchicSelection !== undefined) {\n                    const selection = result.hierarchicSelection[selectAtKey];\n                    if (selection.length) {\n                        this.addRelations(selection);\n                    }\n                }\n            });\n    }\n\n    /**\n     * Get list from database\n     */\n    private queryItems(): void {\n        if (!this.service) {\n            return;\n        }\n\n        this.loading = true;\n        const queryRef = this.service.watchAll(this.variablesManager).pipe(takeUntil(this.ngUnsubscribe));\n        queryRef.pipe(finalize(() => (this.loading = false))).subscribe(() => (this.loading = false));\n        this.dataSource = new NaturalDataSource(queryRef);\n    }\n\n    private getSelectKey(): string | undefined {\n        if (!this.hierarchicSelectorConfig) {\n            return;\n        }\n\n        return this.hierarchicSelectorConfig.filter(c => !!c.selectableAtKey)[0].selectableAtKey;\n    }\n}\n","<div class=\"body\">\n    <ng-template #defaultNameCell let-item=\"item\">\n        {{ getDisplayFn()(item) }}\n    </ng-template>\n\n    <table *ngIf=\"dataSource\" [dataSource]=\"dataSource\" class=\"natural-row-click\" mat-table>\n        <tr *matHeaderRowDef=\"displayedColumns\" mat-header-row style=\"display: none\"></tr>\n        <tr *matRowDef=\"let row; columns: displayedColumns\" mat-row></tr>\n\n        <ng-container matColumnDef=\"name\">\n            <th *matHeaderCellDef i18n mat-header-cell>Titre</th>\n            <td *matCellDef=\"let item\" mat-cell>\n                <ng-template\n                    [ngTemplateOutletContext]=\"{item: item}\"\n                    [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultNameCell\"\n                ></ng-template>\n            </td>\n        </ng-container>\n\n        <ng-container matColumnDef=\"unlink\">\n            <th *matHeaderCellDef mat-header-cell></th>\n            <td *matCellDef=\"let element\" mat-cell>\n                <button\n                    *ngIf=\"!disabled\"\n                    (click)=\"removeRelation(element)\"\n                    color=\"warn\"\n                    mat-icon-button\n                    i18n-matTooltip\n                    matTooltip=\"Dissocier\"\n                >\n                    <natural-icon name=\"link_off\"></natural-icon>\n                </button>\n            </td>\n        </ng-container>\n    </table>\n\n    <mat-paginator\n        (page)=\"pagination($event)\"\n        *ngIf=\"dataSource?.data && (dataSource?.data?.length || 0) > (dataSource?.data?.pageSize || 0)\"\n        [length]=\"dataSource?.data?.length || 0\"\n        [pageIndex]=\"dataSource?.data?.pageIndex || 0\"\n        [pageSizeOptions]=\"pageSizeOptions\"\n        [pageSize]=\"dataSource?.data?.pageSize || 0\"\n    ></mat-paginator>\n\n    <div *ngIf=\"!loading && dataSource?.data?.length === 0\" class=\"margin-v mat-body\">\n        <span i18n>Aucun résultat</span>\n    </div>\n\n    <mat-progress-spinner *ngIf=\"loading\" [diameter]=\"40\" class=\"loading\" mode=\"indeterminate\"></mat-progress-spinner>\n</div>\n\n<natural-select\n    #select\n    (selectionChange)=\"addRelations([$any($event)])\"\n    *ngIf=\"!hierarchicSelectorConfig && service && !disabled\"\n    [displayWith]=\"getDisplayFn()\"\n    [filter]=\"autocompleteSelectorFilter\"\n    [placeholder]=\"placeholder\"\n    [service]=\"service\"\n    [showIcon]=\"false\"\n></natural-select>\n\n<div *ngIf=\"hierarchicSelectorConfig && !disabled\">\n    <button (click)=\"openNaturalHierarchicSelector()\" color=\"primary\" mat-flat-button>{{ placeholder }}</button>\n</div>\n"]}
|
|
203
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"relations.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/relations/relations.component.ts","../../../../../../projects/natural/src/lib/modules/relations/relations.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ContentChild,\n    EventEmitter,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    SimpleChanges,\n    TemplateRef,\n    ViewChild,\n} from '@angular/core';\nimport {PageEvent} from '@angular/material/paginator';\nimport {forkJoin} from 'rxjs';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\nimport {NaturalDataSource, PaginatedData} from '../../classes/data-source';\nimport {NaturalQueryVariablesManager, PaginationInput, QueryVariables} from '../../classes/query-variable-manager';\nimport {HierarchicFiltersConfiguration} from '../../modules/hierarchic-selector/classes/hierarchic-filters-configuration';\nimport {LinkableObject, NaturalLinkMutationService} from '../../services/link-mutation.service';\nimport {NaturalHierarchicConfiguration} from '../hierarchic-selector/classes/hierarchic-configuration';\nimport {HierarchicDialogConfig} from '../hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.component';\nimport {NaturalHierarchicSelectorDialogService} from '../hierarchic-selector/hierarchic-selector-dialog/hierarchic-selector-dialog.service';\nimport {Filter} from '../search/classes/graphql-doctrine.types';\nimport {NaturalSelectComponent} from '../select/select/select.component';\nimport {finalize, takeUntil} from 'rxjs/operators';\nimport {NaturalAbstractModelService} from '../../services/abstract-model.service';\nimport {Literal} from '../../types/types';\n\n/**\n * Custom template usage :\n * <natural-relations [main]=\"owner\" [service]=\"svc\" [filter]=\"{}\" placeholder=\"Select an item\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-relations>\n */\n\n@Component({\n    selector: 'natural-relations',\n    templateUrl: './relations.component.html',\n    styleUrls: ['./relations.component.scss'],\n})\nexport class NaturalRelationsComponent<\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 NaturalAbstractController\n    implements OnInit, OnChanges, OnDestroy\n{\n    @ViewChild(NaturalSelectComponent) private select?: NaturalSelectComponent<TService>;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    @Input() public service?: TService;\n\n    /**\n     * The placeholder used in the button to add a new relation\n     */\n    @Input() public placeholder = '';\n\n    /**\n     * Filter for autocomplete selector\n     */\n    @Input() public autocompleteSelectorFilter?: Filter;\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: any) => string;\n\n    /**\n     * Whether the relations can be changed\n     */\n    @Input() public disabled = false;\n\n    /**\n     * The main object to which all relations belong to\n     */\n    @Input() public main!: LinkableObject & {permissions?: {update: boolean}};\n\n    @Output() public readonly selectionChange = new EventEmitter<void>();\n\n    /**\n     * Filters for hierarchic selector\n     */\n    @Input() public hierarchicSelectorFilters?: HierarchicFiltersConfiguration | null;\n\n    /**\n     * Configuration in case we prefer hierarchic selection over autocomplete selection\n     */\n    @Input() public hierarchicSelectorConfig?: NaturalHierarchicConfiguration[];\n\n    /**\n     * Link mutation semantic\n     */\n    @Input() public otherName?: string | null;\n\n    /**\n     * Listing service instance\n     */\n    public dataSource?: NaturalDataSource;\n    public loading = false;\n\n    /**\n     * Table columns\n     */\n    public displayedColumns = ['name'];\n\n    public pageSizeOptions = [5, 10, 50, 100];\n    protected defaultPagination = {\n        pageIndex: 0,\n        pageSize: 25,\n    };\n\n    /**\n     * Observable variables/options for listing service usage and apollo watchQuery\n     */\n    private variablesManager: NaturalQueryVariablesManager<QueryVariables> = new NaturalQueryVariablesManager();\n\n    public readonly removing = new Set<LinkableObject>();\n\n    public constructor(\n        private readonly linkMutationService: NaturalLinkMutationService,\n        private readonly hierarchicSelectorDialog: NaturalHierarchicSelectorDialogService,\n    ) {\n        super();\n    }\n\n    /**\n     * The filter used to filter relations\n     *\n     * So if the relations are from one action -> to many objectives, then the filter must filter\n     * the objectives that have indeed a relation to the particular action.\n     */\n    @Input() public set filter(filter: Filter) {\n        this.variablesManager.set('relations-filter', {filter: filter});\n    }\n\n    public ngOnInit(): void {\n        this.pagination();\n\n        // Force disabled if cannot update object\n        if (this.main && this.main.permissions) {\n            this.disabled = this.disabled || !this.main.permissions.update;\n        }\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (this.service) {\n            this.queryItems();\n        }\n\n        if (this.disabled && this.displayedColumns.indexOf('unlink') > -1) {\n            this.displayedColumns.pop();\n        } else if (!this.disabled && this.displayedColumns.indexOf('unlink') === -1) {\n            this.displayedColumns.push('unlink');\n        }\n    }\n\n    /**\n     * Unlink action\n     * Refetch result to display it in table\n     */\n    public removeRelation(relation: LinkableObject): void {\n        this.removing.add(relation);\n\n        this.linkMutationService\n            .unlink(this.main, relation, this.otherName)\n            .pipe(finalize(() => this.removing.delete(relation)))\n            .subscribe(() => this.dataSource?.remove(relation));\n    }\n\n    /**\n     * Link action\n     * Refetch result to display it in table\n     * TODO : could maybe use \"update\" attribute of apollo.mutate function to update table faster (but hard to do it here)\n     */\n    public addRelations(relations: LinkableObject[]): void {\n        const observables = relations.map(relation =>\n            this.linkMutationService.link(this.main, relation, this.otherName),\n        );\n\n        forkJoin(observables).subscribe(() => {\n            this.selectionChange.emit();\n            if (this.select) {\n                this.select.clear(false);\n            }\n        });\n    }\n\n    public pagination(event?: PageEvent): void {\n        let pagination: PaginationInput | null = null;\n        if (\n            event &&\n            (event.pageIndex !== this.defaultPagination.pageIndex || event.pageSize !== this.defaultPagination.pageSize)\n        ) {\n            pagination = {\n                pageIndex: event.pageIndex,\n                pageSize: event.pageSize,\n            };\n        }\n\n        this.variablesManager.set('pagination', {pagination: pagination ? pagination : this.defaultPagination});\n    }\n\n    public getDisplayFn(): (item: any) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return item => (item ? item.fullName || item.name : '');\n    }\n\n    public openNaturalHierarchicSelector(): void {\n        const selectAtKey = this.getSelectKey();\n\n        if (!selectAtKey || !this.hierarchicSelectorConfig) {\n            return;\n        }\n\n        const selected = {};\n\n        const hierarchicConfig: HierarchicDialogConfig = {\n            hierarchicConfig: this.hierarchicSelectorConfig,\n            hierarchicSelection: selected,\n            hierarchicFilters: this.hierarchicSelectorFilters,\n            multiple: true,\n        };\n\n        this.hierarchicSelectorDialog\n            .open(hierarchicConfig)\n            .afterClosed()\n            .subscribe(result => {\n                if (result && result.hierarchicSelection !== undefined) {\n                    const selection = result.hierarchicSelection[selectAtKey];\n                    if (selection.length) {\n                        this.addRelations(selection);\n                    }\n                }\n            });\n    }\n\n    /**\n     * Get list from database\n     */\n    private queryItems(): void {\n        if (!this.service) {\n            return;\n        }\n\n        this.loading = true;\n        const queryRef = this.service.watchAll(this.variablesManager).pipe(takeUntil(this.ngUnsubscribe));\n        queryRef.pipe(finalize(() => (this.loading = false))).subscribe(() => (this.loading = false));\n        this.dataSource = new NaturalDataSource(queryRef);\n    }\n\n    private getSelectKey(): string | undefined {\n        if (!this.hierarchicSelectorConfig) {\n            return;\n        }\n\n        return this.hierarchicSelectorConfig.filter(c => !!c.selectableAtKey)[0].selectableAtKey;\n    }\n}\n","<div class=\"body\">\n    <ng-template #defaultNameCell let-item=\"item\">\n        {{ getDisplayFn()(item) }}\n    </ng-template>\n\n    <table *ngIf=\"dataSource\" [dataSource]=\"dataSource\" class=\"natural-row-click\" mat-table>\n        <tr *matHeaderRowDef=\"displayedColumns\" mat-header-row style=\"display: none\"></tr>\n        <tr *matRowDef=\"let row; columns: displayedColumns\" mat-row></tr>\n\n        <ng-container matColumnDef=\"name\">\n            <th *matHeaderCellDef i18n mat-header-cell>Titre</th>\n            <td *matCellDef=\"let item\" mat-cell>\n                <ng-template\n                    [ngTemplateOutletContext]=\"{item: item}\"\n                    [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultNameCell\"\n                ></ng-template>\n            </td>\n        </ng-container>\n\n        <ng-container matColumnDef=\"unlink\">\n            <th *matHeaderCellDef mat-header-cell></th>\n            <td *matCellDef=\"let element\" mat-cell>\n                <button\n                    *ngIf=\"!disabled\"\n                    (click)=\"removeRelation(element)\"\n                    [disabled]=\"removing.has(element)\"\n                    color=\"warn\"\n                    mat-icon-button\n                    i18n-matTooltip\n                    matTooltip=\"Dissocier\"\n                >\n                    <natural-icon name=\"link_off\"></natural-icon>\n                </button>\n            </td>\n        </ng-container>\n    </table>\n\n    <mat-paginator\n        (page)=\"pagination($event)\"\n        *ngIf=\"dataSource?.data && (dataSource?.data?.length || 0) > (dataSource?.data?.pageSize || 0)\"\n        [length]=\"dataSource?.data?.length || 0\"\n        [pageIndex]=\"dataSource?.data?.pageIndex || 0\"\n        [pageSizeOptions]=\"pageSizeOptions\"\n        [pageSize]=\"dataSource?.data?.pageSize || 0\"\n    ></mat-paginator>\n\n    <div *ngIf=\"!loading && dataSource?.data?.length === 0\" class=\"margin-v mat-body\">\n        <span i18n>Aucun résultat</span>\n    </div>\n\n    <mat-progress-spinner *ngIf=\"loading\" [diameter]=\"40\" class=\"loading\" mode=\"indeterminate\"></mat-progress-spinner>\n</div>\n\n<natural-select\n    #select\n    (selectionChange)=\"addRelations([$any($event)])\"\n    *ngIf=\"!hierarchicSelectorConfig && service && !disabled\"\n    [displayWith]=\"getDisplayFn()\"\n    [filter]=\"autocompleteSelectorFilter\"\n    [placeholder]=\"placeholder\"\n    [service]=\"service\"\n    [showIcon]=\"false\"\n></natural-select>\n\n<div *ngIf=\"hierarchicSelectorConfig && !disabled\">\n    <button (click)=\"openNaturalHierarchicSelector()\" color=\"primary\" mat-flat-button>{{ placeholder }}</button>\n</div>\n"]}
|
|
@@ -71,10 +71,10 @@ export class NaturalDropdownContainerComponent extends BasePortalOutlet {
|
|
|
71
71
|
}
|
|
72
72
|
}
|
|
73
73
|
NaturalDropdownContainerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: NaturalDropdownContainerComponent, deps: [{ token: i0.ElementRef }, { token: i1.ConfigurableFocusTrapFactory }, { token: NATURAL_DROPDOWN_CONTAINER_DATA }], target: i0.ɵɵFactoryTarget.Component });
|
|
74
|
-
NaturalDropdownContainerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.0", type: NaturalDropdownContainerComponent, selector: "ng-component", viewQueries: [{ propertyName: "portalOutlet", first: true, predicate: CdkPortalOutlet, descendants: true, static: true }, { propertyName: "templateRef", first: true, predicate: TemplateRef, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div\n (@transformMenu.done)=\"onAnimationDone($event)\"\n [@transformMenu]=\"panelAnimationState\"\n class=\"natural-dropdown-container mat-elevation-z2\"\n role=\"menu\"\n tabindex=\"-1\"\n>\n <div class=\"natural-dropdown-container-content\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div *ngIf=\"data.showValidateButton\" class=\"natural-dropdown-validate-button\">\n <button (click)=\"close()\" color=\"primary\" mat-raised-button>Valider</button>\n </div>\n</div>\n", styles: [".natural-dropdown-container{display:flex;flex-direction:column;border-radius:2px;height:100%}.natural-dropdown-container-content{flex:1;padding:5px;overflow:auto}.natural-dropdown-container .natural-dropdown-validate-button{flex:none;display:flex;flex-direction:row;justify-content:flex-end;margin:5px}\n"], components: [{ type: i2.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], animations: [naturalDropdownAnimations.transformMenu, naturalDropdownAnimations.fadeInItems], encapsulation: i0.ViewEncapsulation.None });
|
|
74
|
+
NaturalDropdownContainerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.0", type: NaturalDropdownContainerComponent, selector: "ng-component", viewQueries: [{ propertyName: "portalOutlet", first: true, predicate: CdkPortalOutlet, descendants: true, static: true }, { propertyName: "templateRef", first: true, predicate: TemplateRef, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div\n (@transformMenu.done)=\"onAnimationDone($event)\"\n [@transformMenu]=\"panelAnimationState\"\n class=\"natural-dropdown-container mat-elevation-z2\"\n role=\"menu\"\n tabindex=\"-1\"\n>\n <div class=\"natural-dropdown-container-content\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div *ngIf=\"data.showValidateButton\" class=\"natural-dropdown-validate-button\">\n <button (click)=\"close()\" color=\"primary\" mat-raised-button i18n>Valider</button>\n </div>\n</div>\n", styles: [".natural-dropdown-container{display:flex;flex-direction:column;border-radius:2px;height:100%}.natural-dropdown-container-content{flex:1;padding:5px;overflow:auto}.natural-dropdown-container .natural-dropdown-validate-button{flex:none;display:flex;flex-direction:row;justify-content:flex-end;margin:5px}\n"], components: [{ type: i2.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], animations: [naturalDropdownAnimations.transformMenu, naturalDropdownAnimations.fadeInItems], encapsulation: i0.ViewEncapsulation.None });
|
|
75
75
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImport: i0, type: NaturalDropdownContainerComponent, decorators: [{
|
|
76
76
|
type: Component,
|
|
77
|
-
args: [{ encapsulation: ViewEncapsulation.None, preserveWhitespaces: false, animations: [naturalDropdownAnimations.transformMenu, naturalDropdownAnimations.fadeInItems], template: "<div\n (@transformMenu.done)=\"onAnimationDone($event)\"\n [@transformMenu]=\"panelAnimationState\"\n class=\"natural-dropdown-container mat-elevation-z2\"\n role=\"menu\"\n tabindex=\"-1\"\n>\n <div class=\"natural-dropdown-container-content\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div *ngIf=\"data.showValidateButton\" class=\"natural-dropdown-validate-button\">\n <button (click)=\"close()\" color=\"primary\" mat-raised-button>Valider</button>\n </div>\n</div>\n", styles: [".natural-dropdown-container{display:flex;flex-direction:column;border-radius:2px;height:100%}.natural-dropdown-container-content{flex:1;padding:5px;overflow:auto}.natural-dropdown-container .natural-dropdown-validate-button{flex:none;display:flex;flex-direction:row;justify-content:flex-end;margin:5px}\n"] }]
|
|
77
|
+
args: [{ encapsulation: ViewEncapsulation.None, preserveWhitespaces: false, animations: [naturalDropdownAnimations.transformMenu, naturalDropdownAnimations.fadeInItems], template: "<div\n (@transformMenu.done)=\"onAnimationDone($event)\"\n [@transformMenu]=\"panelAnimationState\"\n class=\"natural-dropdown-container mat-elevation-z2\"\n role=\"menu\"\n tabindex=\"-1\"\n>\n <div class=\"natural-dropdown-container-content\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div *ngIf=\"data.showValidateButton\" class=\"natural-dropdown-validate-button\">\n <button (click)=\"close()\" color=\"primary\" mat-raised-button i18n>Valider</button>\n </div>\n</div>\n", styles: [".natural-dropdown-container{display:flex;flex-direction:column;border-radius:2px;height:100%}.natural-dropdown-container-content{flex:1;padding:5px;overflow:auto}.natural-dropdown-container .natural-dropdown-validate-button{flex:none;display:flex;flex-direction:row;justify-content:flex-end;margin:5px}\n"] }]
|
|
78
78
|
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.ConfigurableFocusTrapFactory }, { type: undefined, decorators: [{
|
|
79
79
|
type: Inject,
|
|
80
80
|
args: [NATURAL_DROPDOWN_CONTAINER_DATA]
|
|
@@ -85,4 +85,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.0", ngImpor
|
|
|
85
85
|
type: ViewChild,
|
|
86
86
|
args: [TemplateRef, { static: true }]
|
|
87
87
|
}] } });
|
|
88
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
88
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dropdown-container.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/search/dropdown-container/dropdown-container.component.ts","../../../../../../../projects/natural/src/lib/modules/search/dropdown-container/dropdown-container.component.html"],"names":[],"mappings":"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","sourcesContent":["import {AnimationEvent} from '@angular/animations';\nimport {ConfigurableFocusTrapFactory, FocusTrap} from '@angular/cdk/a11y';\nimport {BasePortalOutlet, CdkPortalOutlet, ComponentPortal, TemplatePortal} from '@angular/cdk/portal';\nimport {\n    Component,\n    ComponentRef,\n    ElementRef,\n    EmbeddedViewRef,\n    Inject,\n    InjectionToken,\n    OnDestroy,\n    TemplateRef,\n    ViewChild,\n    ViewEncapsulation,\n} from '@angular/core';\nimport {Subject} from 'rxjs';\nimport {naturalDropdownAnimations} from './dropdown-container-animations';\n\nexport function throwMatDialogContentAlreadyAttachedError(): void {\n    throw Error('Attempting to attach dialog content after content is already attached');\n}\n\nexport interface NaturalDropdownContainerData {\n    showValidateButton: boolean;\n}\n\nexport const NATURAL_DROPDOWN_CONTAINER_DATA = new InjectionToken<NaturalDropdownContainerData>(\n    'NaturalDropdownContainerData',\n);\n\n@Component({\n    templateUrl: './dropdown-container.component.html',\n    styleUrls: ['./dropdown-container.component.scss'],\n    encapsulation: ViewEncapsulation.None,\n    preserveWhitespaces: false,\n    animations: [naturalDropdownAnimations.transformMenu, naturalDropdownAnimations.fadeInItems],\n})\nexport class NaturalDropdownContainerComponent extends BasePortalOutlet implements OnDestroy {\n    @ViewChild(CdkPortalOutlet, {static: true}) public portalOutlet!: CdkPortalOutlet;\n    @ViewChild(TemplateRef, {static: true}) public templateRef!: TemplateRef<any>;\n\n    public readonly closed = new Subject<void>();\n\n    /** Current state of the panel animation. */\n    public panelAnimationState: 'void' | 'enter' = 'void';\n\n    /** Emits whenever an animation on the menu completes. */\n    private animationDone = new Subject<void>();\n\n    private focusTrap: FocusTrap | null = null;\n    private elementFocusedBeforeDialogWasOpened: HTMLElement | null = null;\n\n    public constructor(\n        private readonly elementRef: ElementRef<HTMLElement>,\n        private readonly focusTrapFactory: ConfigurableFocusTrapFactory,\n        @Inject(NATURAL_DROPDOWN_CONTAINER_DATA) public data: NaturalDropdownContainerData,\n    ) {\n        super();\n    }\n\n    public ngOnDestroy(): void {\n        this.closed.complete();\n    }\n\n    public close(): void {\n        this.closed.next();\n    }\n\n    public attachTemplatePortal<C>(portal: TemplatePortal<C>): EmbeddedViewRef<C> {\n        return this.portalOutlet.attachTemplatePortal(portal);\n    }\n\n    public attachComponentPortal<T>(portal: ComponentPortal<T>): ComponentRef<T> {\n        if (this.portalOutlet.hasAttached()) {\n            throwMatDialogContentAlreadyAttachedError();\n        }\n\n        return this.portalOutlet.attachComponentPortal(portal);\n    }\n\n    public startAnimation(): void {\n        this.panelAnimationState = 'enter';\n    }\n\n    public onAnimationDone(event: AnimationEvent): void {\n        if (event.toState === 'enter') {\n            this.trapFocus();\n        } else if (event.toState === 'exit') {\n            this.restoreFocus();\n        }\n\n        this.animationDone.next();\n    }\n\n    private trapFocus(): void {\n        if (!this.focusTrap) {\n            this.focusTrap = this.focusTrapFactory.create(this.elementRef.nativeElement);\n        }\n\n        this.focusTrap.focusInitialElementWhenReady();\n    }\n\n    /** Restores focus to the element that was focused before the dialog opened. */\n    private restoreFocus(): void {\n        const toFocus = this.elementFocusedBeforeDialogWasOpened;\n\n        // We need the extra check, because IE can set the `activeElement` to null in some cases.\n        if (toFocus && typeof toFocus.focus === 'function') {\n            toFocus.focus();\n        }\n\n        if (this.focusTrap) {\n            this.focusTrap.destroy();\n        }\n    }\n}\n","<div\n    (@transformMenu.done)=\"onAnimationDone($event)\"\n    [@transformMenu]=\"panelAnimationState\"\n    class=\"natural-dropdown-container mat-elevation-z2\"\n    role=\"menu\"\n    tabindex=\"-1\"\n>\n    <div class=\"natural-dropdown-container-content\">\n        <ng-template cdkPortalOutlet></ng-template>\n    </div>\n\n    <div *ngIf=\"data.showValidateButton\" class=\"natural-dropdown-validate-button\">\n        <button (click)=\"close()\" color=\"primary\" mat-raised-button i18n>Valider</button>\n    </div>\n</div>\n"]}
|