@ecodev/natural 41.0.0 → 41.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/ecodev-natural.umd.js +34 -5
- package/bundles/ecodev-natural.umd.js.map +1 -1
- package/ecodev-natural.metadata.json +1 -1
- package/esm2015/lib/classes/query-variable-manager.js +7 -3
- package/esm2015/lib/classes/utility.js +13 -1
- package/esm2015/lib/modules/file/abstract-file.js +14 -4
- package/esm2015/lib/modules/file/file-drop.directive.js +4 -2
- package/esm2015/lib/modules/relations/relations.component.js +2 -2
- package/fesm2015/ecodev-natural.js +34 -6
- package/fesm2015/ecodev-natural.js.map +1 -1
- package/lib/classes/query-variable-manager.d.ts +7 -1
- package/lib/classes/utility.d.ts +5 -0
- package/lib/modules/file/abstract-file.d.ts +8 -1
- package/package.json +1 -1
|
@@ -63,9 +63,13 @@ export class NaturalQueryVariablesManager {
|
|
|
63
63
|
}
|
|
64
64
|
this.updateVariables();
|
|
65
65
|
}
|
|
66
|
+
/**
|
|
67
|
+
* Return a deep clone of the variables for the given channel name.
|
|
68
|
+
*
|
|
69
|
+
* Avoid returning the same reference to prevent an attribute change, then another channel update that would
|
|
70
|
+
* used this changed attribute without having explicitly asked QueryVariablesManager to update it.
|
|
71
|
+
*/
|
|
66
72
|
get(channelName) {
|
|
67
|
-
// Avoid to return the same reference to prevent an attribute change, then another channel update that would used this changed
|
|
68
|
-
// attribute without having explicitly asked QueryVariablesManager to update it.
|
|
69
73
|
return cloneDeep(this.channels.get(channelName));
|
|
70
74
|
}
|
|
71
75
|
/**
|
|
@@ -154,4 +158,4 @@ export class NaturalQueryVariablesManager {
|
|
|
154
158
|
return groups;
|
|
155
159
|
}
|
|
156
160
|
}
|
|
157
|
-
//# 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 an aspect of the GUI (pagination, sorting, search, others ?).\n * Each\n *\n * const fm = new QueryVariablesManager();\n * fm.merge('componentA-variables', {a : [1, 2, 3]});\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 : console.log(fm.variables.value); //  {a : [1, 2, 3]}\n *\n * Set new variables for 'componentA-variables'\n * fm.merge('componentA-variables', {a : [1, 2]);\n * console.log(fm.variables.value); //  {a : [1, 2, 3]}\n *\n * Set new variables for new channel\n * fm.merge('componentB-variables', {a : [3, 4]);\n * console.log(fm.variables.value); // {a : [1, 2, 3, 4]}\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    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    public get(channelName: string): Partial<T> | undefined {\n        // Avoid to return the same reference to prevent an attribute change, then another channel update that would used this changed\n        // attribute without having explicitly asked QueryVariablesManager to update it.\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    public 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"]}
|
|
161
|
+
//# 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 an aspect of the GUI (pagination, sorting, search, others ?).\n * Each\n *\n * const fm = new QueryVariablesManager();\n * fm.merge('componentA-variables', {a : [1, 2, 3]});\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 : console.log(fm.variables.value); //  {a : [1, 2, 3]}\n *\n * Set new variables for 'componentA-variables'\n * fm.merge('componentA-variables', {a : [1, 2]);\n * console.log(fm.variables.value); //  {a : [1, 2, 3]}\n *\n * Set new variables for new channel\n * fm.merge('componentB-variables', {a : [3, 4]);\n * console.log(fm.variables.value); // {a : [1, 2, 3, 4]}\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    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"]}
|
|
@@ -151,4 +151,16 @@ export function mergeOverrideArray(destValue, source) {
|
|
|
151
151
|
return source;
|
|
152
152
|
}
|
|
153
153
|
}
|
|
154
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utility.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/utility.ts"],"names":[],"mappings":"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","sourcesContent":["import {isArray, isEmpty, isObject, pickBy} from 'lodash-es';\nimport {Literal} from '../types/types';\n\n/**\n * Very basic formatting to get only date, without time and ignoring entirely the timezone\n *\n * So something like: \"2021-09-23\"\n */\nexport function formatIsoDate(date: Date | null): string | null {\n    if (!date) {\n        return null;\n    }\n\n    const y = date.getFullYear();\n    const m = date.getMonth() + 1;\n    const d = date.getDate();\n\n    return y + '-' + (m < 10 ? '0' : '') + m + '-' + (d < 10 ? '0' : '') + d;\n}\n\n/**\n * Format a date and time in a way that will preserve the local time zone.\n * This allow the server side to know the day (without time) that was selected on client side.\n *\n * So something like: \"2021-09-23T17:57:16+09:00\"\n */\nexport function formatIsoDateTime(date: Date): string {\n    const timezoneOffsetInMinutes = date.getTimezoneOffset();\n    const timezoneOffsetInHours = -Math.trunc(timezoneOffsetInMinutes / 60); // UTC minus local time\n    const sign = timezoneOffsetInHours >= 0 ? '+' : '-';\n    const hoursLeadingZero = Math.abs(timezoneOffsetInHours) < 10 ? '0' : '';\n    const remainderMinutes = -(timezoneOffsetInMinutes % 60);\n    const minutesLeadingZero = Math.abs(remainderMinutes) < 10 ? '0' : '';\n\n    // It's a bit unfortunate that we need to construct a new Date instance,\n    // but we don't want the original Date instance to be modified\n    const correctedDate = new Date(\n        date.getFullYear(),\n        date.getMonth(),\n        date.getDate(),\n        date.getHours(),\n        date.getMinutes(),\n        date.getSeconds(),\n        date.getMilliseconds(),\n    );\n    correctedDate.setHours(date.getHours() + timezoneOffsetInHours);\n\n    const iso = correctedDate\n        .toISOString()\n        .replace(/\\.\\d{3}Z/, '')\n        .replace('Z', '');\n\n    return (\n        iso +\n        sign +\n        hoursLeadingZero +\n        Math.abs(timezoneOffsetInHours).toString() +\n        ':' +\n        minutesLeadingZero +\n        remainderMinutes\n    );\n}\n\n/**\n * Relations to full objects are converted to their IDs only.\n *\n * So {user: {id: 123}} becomes {user: 123}\n */\nexport function relationsToIds(object: Literal): Literal {\n    const newObj: Literal = {};\n    Object.keys(object).forEach(key => {\n        let value: string | Literal = object[key];\n        if (isObject(value) && value.id) {\n            value = value.id;\n        } else if (isArray(value)) {\n            value = value.map((i: string | Literal) => (isObject(i) && i.id ? i.id : i));\n        } else if (isObject(value) && !(value instanceof File) && !(value instanceof Date)) {\n            value = pickBy(value, (v, k) => k !== '__typename'); // omit(value, ['__typename']) ?\n        }\n\n        newObj[key] = value;\n    });\n\n    return newObj;\n}\n\n/**\n * Remove from source object the attributes with same value as modified\n * Does not consider arrays\n */\nexport function cleanSameValues(source: Literal, modified: Literal): Literal {\n    Object.keys(source).forEach(key => {\n        if (source[key] instanceof Object) {\n            cleanSameValues(source[key], modified[key]);\n            if (isEmpty(source[key])) {\n                delete source[key];\n            }\n        } else if (modified && source[key] === modified[key]) {\n            delete source[key];\n        }\n    });\n\n    return source;\n}\n\n/**\n * Returns the plural form of the given name\n */\nexport function makePlural(name: string): string {\n    const plural = name + 's';\n\n    return plural.replace(/ys$/, 'ies').replace(/ss$/, 'ses').replace(/xs$/, 'xes');\n}\n\n/**\n * Returns the string with the first letter as capital\n */\nexport function upperCaseFirstLetter(term: string): string {\n    return term.charAt(0).toUpperCase() + term.slice(1);\n}\n\n/**\n * Returns the string with the first letter as lower case\n */\nexport function lowerCaseFirstLetter(term: string): string {\n    return term.charAt(0).toLowerCase() + term.slice(1);\n}\n\n/**\n * Replace all attributes of first object with the ones provided by the second, but keeps the reference\n */\nexport function replaceObjectKeepingReference(obj: Literal | null, newObj: Literal | null): void {\n    if (!obj || !newObj) {\n        return;\n    }\n\n    Object.keys(obj).forEach(key => {\n        delete obj[key];\n    });\n\n    Object.keys(newObj).forEach(key => {\n        obj[key] = newObj[key];\n    });\n}\n\n/**\n * Get contrasted color for text in the slider thumb\n * @param hexBgColor string in hexadecimals representing the background color\n */\nexport function getForegroundColor(hexBgColor: string): 'black' | 'white' {\n    const rgb = hexToRgb(hexBgColor.slice(0, 7)); // splice remove alpha and consider only \"visible\" color at 100% alpha\n    const o = Math.round((rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000);\n\n    return o > 125 ? 'black' : 'white';\n}\n\nfunction hexToRgb(hex: string): {r: number; g: number; b: number} {\n    // Expand shorthand form (e.g. \"03F\") to full form (e.g. \"0033FF\")\n    const shorthandRegex = /^#?([a-f\\d])([a-f\\d])([a-f\\d])$/i;\n    hex = hex.replace(shorthandRegex, (m, r, g, b) => {\n        return r + r + g + g + b + b;\n    });\n\n    const result = /^#?([a-f\\d]{2})([a-f\\d]{2})([a-f\\d]{2})$/i.exec(hex);\n    return result\n        ? {\n              r: parseInt(result[1], 16),\n              g: parseInt(result[2], 16),\n              b: parseInt(result[3], 16),\n          }\n        : {\n              r: 0,\n              g: 0,\n              b: 0,\n          };\n}\n\n/**\n * During lodash.mergeWith, overrides arrays\n */\nexport function mergeOverrideArray(destValue: any, source: any): any {\n    if (isArray(source)) {\n        return source;\n    }\n}\n"]}
|
|
154
|
+
/**
|
|
155
|
+
* Copy text to clipboard.
|
|
156
|
+
* Accepts line breaks `\n` as textarea do.
|
|
157
|
+
*/
|
|
158
|
+
export function copyToClipboard(document, text) {
|
|
159
|
+
const input = document.createElement('textarea');
|
|
160
|
+
document.body.append(input);
|
|
161
|
+
input.value = text;
|
|
162
|
+
input.select();
|
|
163
|
+
document.execCommand('copy');
|
|
164
|
+
document.body.removeChild(input);
|
|
165
|
+
}
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utility.js","sourceRoot":"","sources":["../../../../../projects/natural/src/lib/classes/utility.ts"],"names":[],"mappings":"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","sourcesContent":["import {isArray, isEmpty, isObject, pickBy} from 'lodash-es';\nimport {Literal} from '../types/types';\n\n/**\n * Very basic formatting to get only date, without time and ignoring entirely the timezone\n *\n * So something like: \"2021-09-23\"\n */\nexport function formatIsoDate(date: Date | null): string | null {\n    if (!date) {\n        return null;\n    }\n\n    const y = date.getFullYear();\n    const m = date.getMonth() + 1;\n    const d = date.getDate();\n\n    return y + '-' + (m < 10 ? '0' : '') + m + '-' + (d < 10 ? '0' : '') + d;\n}\n\n/**\n * Format a date and time in a way that will preserve the local time zone.\n * This allow the server side to know the day (without time) that was selected on client side.\n *\n * So something like: \"2021-09-23T17:57:16+09:00\"\n */\nexport function formatIsoDateTime(date: Date): string {\n    const timezoneOffsetInMinutes = date.getTimezoneOffset();\n    const timezoneOffsetInHours = -Math.trunc(timezoneOffsetInMinutes / 60); // UTC minus local time\n    const sign = timezoneOffsetInHours >= 0 ? '+' : '-';\n    const hoursLeadingZero = Math.abs(timezoneOffsetInHours) < 10 ? '0' : '';\n    const remainderMinutes = -(timezoneOffsetInMinutes % 60);\n    const minutesLeadingZero = Math.abs(remainderMinutes) < 10 ? '0' : '';\n\n    // It's a bit unfortunate that we need to construct a new Date instance,\n    // but we don't want the original Date instance to be modified\n    const correctedDate = new Date(\n        date.getFullYear(),\n        date.getMonth(),\n        date.getDate(),\n        date.getHours(),\n        date.getMinutes(),\n        date.getSeconds(),\n        date.getMilliseconds(),\n    );\n    correctedDate.setHours(date.getHours() + timezoneOffsetInHours);\n\n    const iso = correctedDate\n        .toISOString()\n        .replace(/\\.\\d{3}Z/, '')\n        .replace('Z', '');\n\n    return (\n        iso +\n        sign +\n        hoursLeadingZero +\n        Math.abs(timezoneOffsetInHours).toString() +\n        ':' +\n        minutesLeadingZero +\n        remainderMinutes\n    );\n}\n\n/**\n * Relations to full objects are converted to their IDs only.\n *\n * So {user: {id: 123}} becomes {user: 123}\n */\nexport function relationsToIds(object: Literal): Literal {\n    const newObj: Literal = {};\n    Object.keys(object).forEach(key => {\n        let value: string | Literal = object[key];\n        if (isObject(value) && value.id) {\n            value = value.id;\n        } else if (isArray(value)) {\n            value = value.map((i: string | Literal) => (isObject(i) && i.id ? i.id : i));\n        } else if (isObject(value) && !(value instanceof File) && !(value instanceof Date)) {\n            value = pickBy(value, (v, k) => k !== '__typename'); // omit(value, ['__typename']) ?\n        }\n\n        newObj[key] = value;\n    });\n\n    return newObj;\n}\n\n/**\n * Remove from source object the attributes with same value as modified\n * Does not consider arrays\n */\nexport function cleanSameValues(source: Literal, modified: Literal): Literal {\n    Object.keys(source).forEach(key => {\n        if (source[key] instanceof Object) {\n            cleanSameValues(source[key], modified[key]);\n            if (isEmpty(source[key])) {\n                delete source[key];\n            }\n        } else if (modified && source[key] === modified[key]) {\n            delete source[key];\n        }\n    });\n\n    return source;\n}\n\n/**\n * Returns the plural form of the given name\n */\nexport function makePlural(name: string): string {\n    const plural = name + 's';\n\n    return plural.replace(/ys$/, 'ies').replace(/ss$/, 'ses').replace(/xs$/, 'xes');\n}\n\n/**\n * Returns the string with the first letter as capital\n */\nexport function upperCaseFirstLetter(term: string): string {\n    return term.charAt(0).toUpperCase() + term.slice(1);\n}\n\n/**\n * Returns the string with the first letter as lower case\n */\nexport function lowerCaseFirstLetter(term: string): string {\n    return term.charAt(0).toLowerCase() + term.slice(1);\n}\n\n/**\n * Replace all attributes of first object with the ones provided by the second, but keeps the reference\n */\nexport function replaceObjectKeepingReference(obj: Literal | null, newObj: Literal | null): void {\n    if (!obj || !newObj) {\n        return;\n    }\n\n    Object.keys(obj).forEach(key => {\n        delete obj[key];\n    });\n\n    Object.keys(newObj).forEach(key => {\n        obj[key] = newObj[key];\n    });\n}\n\n/**\n * Get contrasted color for text in the slider thumb\n * @param hexBgColor string in hexadecimals representing the background color\n */\nexport function getForegroundColor(hexBgColor: string): 'black' | 'white' {\n    const rgb = hexToRgb(hexBgColor.slice(0, 7)); // splice remove alpha and consider only \"visible\" color at 100% alpha\n    const o = Math.round((rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000);\n\n    return o > 125 ? 'black' : 'white';\n}\n\nfunction hexToRgb(hex: string): {r: number; g: number; b: number} {\n    // Expand shorthand form (e.g. \"03F\") to full form (e.g. \"0033FF\")\n    const shorthandRegex = /^#?([a-f\\d])([a-f\\d])([a-f\\d])$/i;\n    hex = hex.replace(shorthandRegex, (m, r, g, b) => {\n        return r + r + g + g + b + b;\n    });\n\n    const result = /^#?([a-f\\d]{2})([a-f\\d]{2})([a-f\\d]{2})$/i.exec(hex);\n    return result\n        ? {\n              r: parseInt(result[1], 16),\n              g: parseInt(result[2], 16),\n              b: parseInt(result[3], 16),\n          }\n        : {\n              r: 0,\n              g: 0,\n              b: 0,\n          };\n}\n\n/**\n * During lodash.mergeWith, overrides arrays\n */\nexport function mergeOverrideArray(destValue: any, source: any): any {\n    if (isArray(source)) {\n        return source;\n    }\n}\n\n/**\n * Copy text to clipboard.\n * Accepts line breaks `\\n` as textarea do.\n */\nexport function copyToClipboard(document: Document, text: string): void {\n    const input = document.createElement('textarea');\n    document.body.append(input);\n    input.value = text;\n    input.select();\n    document.execCommand('copy');\n    document.body.removeChild(input);\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
// tslint:disable:directive-class-suffix
|
|
2
|
-
import { Directive,
|
|
3
|
-
import { acceptType, createInvisibleFileInputWrap,
|
|
2
|
+
import { Directive, ElementRef, EventEmitter, HostListener, Inject, Input, Output, } from '@angular/core';
|
|
3
|
+
import { acceptType, createInvisibleFileInputWrap, detectSwipe, eventToFiles, fileListToArray, isFileInput, stopEvent, } from './utils';
|
|
4
4
|
import { NaturalFileService } from './file.service';
|
|
5
5
|
import { NaturalAbstractController } from '../../classes/abstract-controller';
|
|
6
6
|
import { DOCUMENT } from '@angular/common';
|
|
@@ -49,6 +49,13 @@ export class NaturalAbstractFile extends NaturalAbstractController {
|
|
|
49
49
|
* no effect.
|
|
50
50
|
*/
|
|
51
51
|
this.selectable = false;
|
|
52
|
+
/**
|
|
53
|
+
* If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.
|
|
54
|
+
*
|
|
55
|
+
* It is useful to set this to false if there is two upload on a page with different purpose
|
|
56
|
+
* and the second upload should not be confused with the first one.
|
|
57
|
+
*/
|
|
58
|
+
this.broadcast = true;
|
|
52
59
|
/**
|
|
53
60
|
* The single valid file that has been selected.
|
|
54
61
|
*
|
|
@@ -129,7 +136,9 @@ export class NaturalAbstractFile extends NaturalAbstractController {
|
|
|
129
136
|
}
|
|
130
137
|
if (selection.valid.length || selection.invalid.length) {
|
|
131
138
|
this.filesChange.emit(selection);
|
|
132
|
-
this.
|
|
139
|
+
if (this.broadcast) {
|
|
140
|
+
this.naturalFileService.filesChanged.next(selection);
|
|
141
|
+
}
|
|
133
142
|
}
|
|
134
143
|
this.getFileElement().value = '';
|
|
135
144
|
}
|
|
@@ -206,8 +215,9 @@ NaturalAbstractFile.propDecorators = {
|
|
|
206
215
|
maxSize: [{ type: Input }],
|
|
207
216
|
fileSelectionDisabled: [{ type: Input }],
|
|
208
217
|
selectable: [{ type: Input }],
|
|
218
|
+
broadcast: [{ type: Input }],
|
|
209
219
|
fileChange: [{ type: Output }],
|
|
210
220
|
filesChange: [{ type: Output }],
|
|
211
221
|
onChange: [{ type: HostListener, args: ['change', ['$event'],] }]
|
|
212
222
|
};
|
|
213
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-file.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/abstract-file.ts"],"names":[],"mappings":"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","sourcesContent":["// tslint:disable:directive-class-suffix\nimport {\n    Directive,\n    EventEmitter,\n    ElementRef,\n    Input,\n    Output,\n    HostListener,\n    OnInit,\n    OnDestroy,\n    OnChanges,\n    SimpleChanges,\n    Inject,\n} from '@angular/core';\nimport {\n    acceptType,\n    createInvisibleFileInputWrap,\n    isFileInput,\n    detectSwipe,\n    fileListToArray,\n    eventToFiles,\n    stopEvent,\n} from './utils';\nimport {NaturalFileService} from './file.service';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\nimport {DOCUMENT} from '@angular/common';\n\nexport interface InvalidFile {\n    file: File;\n    error: string;\n}\n\nexport interface FileSelection {\n    /**\n     * The list of files that have been selected.\n     */\n    valid: File[];\n\n    /**\n     * The list of files that have been selected but are invalid according to validators.\n     */\n    invalid: InvalidFile[];\n}\n\n/**\n * A master base set of logic intended to support file select/drag/drop operations\n *\n * In most cases you probably want click-to-select and drag-to-select, so you should use:\n *\n *     <div naturalFileDrop [selectable]=\"true\"></div>\n *\n * @dynamic\n */\n@Directive()\nexport abstract class NaturalAbstractFile extends NaturalAbstractController implements OnInit, OnDestroy, OnChanges {\n    private fileElement?: HTMLInputElement;\n    private readonly validators = [\n        {name: 'accept', fn: this.acceptValidator},\n        {name: 'fileSize', fn: this.fileSizeValidator},\n    ];\n\n    /**\n     * Whether we should accept a single file or multiple files\n     */\n    @Input() public multiple = false;\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element\n     * it can be a mixed of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept = '';\n\n    /**\n     * Maximum file size in bytes. 0 means no validation at all.\n     */\n    @Input() public maxSize = 0;\n\n    /**\n     * Disable the file selection entirely\n     */\n    @Input() public fileSelectionDisabled = false;\n\n    /**\n     * Whether the user can click on the element to select something\n     *\n     * This has only effect during initialization. Subsequent changes will have\n     * no effect.\n     */\n    @Input() public selectable = false;\n\n    /**\n     * The single valid file that has been selected.\n     *\n     * It is for convenience of use, and will only emit if there is at least one\n     * valid file. See `filesChange` for a more complete output.\n     */\n    @Output() public readonly fileChange = new EventEmitter<File>();\n\n    /**\n     * The list of files that have been selected.\n     */\n    @Output() public readonly filesChange = new EventEmitter<FileSelection>();\n\n    constructor(\n        private readonly element: ElementRef<HTMLElement>,\n        protected readonly naturalFileService: NaturalFileService,\n        @Inject(DOCUMENT) private readonly document: Document,\n    ) {\n        super();\n    }\n\n    public ngOnDestroy(): void {\n        super.ngOnDestroy();\n        delete this.fileElement; // faster memory release of dom element\n    }\n\n    public ngOnInit(): void {\n        if (this.selectable) {\n            this.enableSelecting();\n        }\n\n        this.getFileElement().multiple = this.multiple;\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.accept) {\n            this.getFileElement().setAttribute('accept', changes.accept.currentValue || '*');\n        }\n    }\n\n    private getFileElement(): HTMLInputElement {\n        if (this.fileElement) {\n            return this.fileElement;\n        }\n\n        // elm is a file input\n        if (isFileInput(this.element.nativeElement)) {\n            this.fileElement = this.element.nativeElement;\n\n            return this.fileElement;\n        }\n\n        // create foo file input\n        const label = createInvisibleFileInputWrap(this.document);\n        this.fileElement = label.getElementsByTagName('input')[0];\n        this.fileElement.addEventListener('change', this.changeFn.bind(this));\n        this.element.nativeElement.appendChild(label);\n\n        return this.fileElement;\n    }\n\n    private enableSelecting(): void {\n        const elm = this.element.nativeElement;\n\n        if (isFileInput(elm)) {\n            const bindedHandler = () => this.beforeSelect();\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            return;\n        } else {\n            const bindedHandler = (event: Event) => this.clickHandler(event);\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            elm.addEventListener('touchend', bindedHandler);\n        }\n    }\n\n    protected handleFiles(files: File[]): void {\n        const selection: FileSelection = {\n            valid: [],\n            invalid: [],\n        };\n\n        for (const file of files) {\n            const error = this.validate(file);\n            if (error) {\n                selection.invalid.push({\n                    file: file,\n                    error: error,\n                });\n            } else {\n                selection.valid.push(file);\n            }\n        }\n\n        if (selection.valid.length) {\n            this.fileChange.emit(selection.valid[0]);\n        }\n\n        if (selection.valid.length || selection.invalid.length) {\n            this.filesChange.emit(selection);\n            this.naturalFileService.filesChanged.next(selection);\n        }\n\n        this.getFileElement().value = '';\n    }\n\n    /**\n     * Called when input has files\n     */\n    private changeFn(event: Event): void {\n        if (!(event.target instanceof HTMLInputElement)) {\n            return;\n        }\n\n        const fileList = event.target.files;\n\n        if (!fileList) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(fileListToArray(fileList));\n    }\n\n    private clickHandler(event: Event): boolean {\n        const elm = this.element.nativeElement;\n        if (elm.getAttribute('disabled') || this.fileSelectionDisabled) {\n            return false;\n        }\n\n        // prevent the click if it is a swipe\n        if (detectSwipe(event)) {\n            return true;\n        }\n\n        const fileElm = this.getFileElement();\n        fileElm.click();\n        this.beforeSelect();\n\n        return false;\n    }\n\n    private beforeSelect(): void {\n        if (!this.fileElement) {\n            return;\n        }\n\n        // if no files in array, be sure browser doesnt prevent reselect of same file (see github issue 27)\n        this.fileElement.value = '';\n    }\n\n    @HostListener('change', ['$event'])\n    public onChange(event: Event): void {\n        const fileList = this.getFileElement().files;\n        const files: File[] = fileList ? fileListToArray(fileList) : eventToFiles(event);\n\n        if (!files.length) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(files);\n    }\n\n    private validate(file: File): string | null {\n        for (const validator of this.validators) {\n            if (!validator.fn.call(this, file)) {\n                return validator.name;\n            }\n        }\n\n        return null;\n    }\n\n    private acceptValidator(item: File): boolean {\n        return acceptType(this.accept, item.type, item.name);\n    }\n\n    private fileSizeValidator(item: File): boolean {\n        return !(this.maxSize && item.size > this.maxSize);\n    }\n}\n"]}
|
|
223
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-file.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/file/abstract-file.ts"],"names":[],"mappings":"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","sourcesContent":["// tslint:disable:directive-class-suffix\nimport {\n    Directive,\n    ElementRef,\n    EventEmitter,\n    HostListener,\n    Inject,\n    Input,\n    OnChanges,\n    OnDestroy,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport {\n    acceptType,\n    createInvisibleFileInputWrap,\n    detectSwipe,\n    eventToFiles,\n    fileListToArray,\n    isFileInput,\n    stopEvent,\n} from './utils';\nimport {NaturalFileService} from './file.service';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\nimport {DOCUMENT} from '@angular/common';\n\nexport interface InvalidFile {\n    file: File;\n    error: string;\n}\n\nexport interface FileSelection {\n    /**\n     * The list of files that have been selected.\n     */\n    valid: File[];\n\n    /**\n     * The list of files that have been selected but are invalid according to validators.\n     */\n    invalid: InvalidFile[];\n}\n\n/**\n * A master base set of logic intended to support file select/drag/drop operations\n *\n * In most cases you probably want click-to-select and drag-to-select, so you should use:\n *\n *     <div naturalFileDrop [selectable]=\"true\"></div>\n *\n * @dynamic\n */\n@Directive()\nexport abstract class NaturalAbstractFile extends NaturalAbstractController implements OnInit, OnDestroy, OnChanges {\n    private fileElement?: HTMLInputElement;\n    private readonly validators = [\n        {name: 'accept', fn: this.acceptValidator},\n        {name: 'fileSize', fn: this.fileSizeValidator},\n    ];\n\n    /**\n     * Whether we should accept a single file or multiple files\n     */\n    @Input() public multiple = false;\n\n    /**\n     * Comma-separated list of unique file type specifiers. Like the native element\n     * it can be a mixed of mime-type and file extensions.\n     *\n     * See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file#accept\n     */\n    @Input() public accept = '';\n\n    /**\n     * Maximum file size in bytes. 0 means no validation at all.\n     */\n    @Input() public maxSize = 0;\n\n    /**\n     * Disable the file selection entirely\n     */\n    @Input() public fileSelectionDisabled = false;\n\n    /**\n     * Whether the user can click on the element to select something\n     *\n     * This has only effect during initialization. Subsequent changes will have\n     * no effect.\n     */\n    @Input() public selectable = false;\n\n    /**\n     * If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.\n     *\n     * It is useful to set this to false if there is two upload on a page with different purpose\n     * and the second upload should not be confused with the first one.\n     */\n    @Input() public broadcast = true;\n\n    /**\n     * The single valid file that has been selected.\n     *\n     * It is for convenience of use, and will only emit if there is at least one\n     * valid file. See `filesChange` for a more complete output.\n     */\n    @Output() public readonly fileChange = new EventEmitter<File>();\n\n    /**\n     * The list of files that have been selected.\n     */\n    @Output() public readonly filesChange = new EventEmitter<FileSelection>();\n\n    constructor(\n        private readonly element: ElementRef<HTMLElement>,\n        protected readonly naturalFileService: NaturalFileService,\n        @Inject(DOCUMENT) private readonly document: Document,\n    ) {\n        super();\n    }\n\n    public ngOnDestroy(): void {\n        super.ngOnDestroy();\n        delete this.fileElement; // faster memory release of dom element\n    }\n\n    public ngOnInit(): void {\n        if (this.selectable) {\n            this.enableSelecting();\n        }\n\n        this.getFileElement().multiple = this.multiple;\n    }\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        if (changes.accept) {\n            this.getFileElement().setAttribute('accept', changes.accept.currentValue || '*');\n        }\n    }\n\n    private getFileElement(): HTMLInputElement {\n        if (this.fileElement) {\n            return this.fileElement;\n        }\n\n        // elm is a file input\n        if (isFileInput(this.element.nativeElement)) {\n            this.fileElement = this.element.nativeElement;\n\n            return this.fileElement;\n        }\n\n        // create foo file input\n        const label = createInvisibleFileInputWrap(this.document);\n        this.fileElement = label.getElementsByTagName('input')[0];\n        this.fileElement.addEventListener('change', this.changeFn.bind(this));\n        this.element.nativeElement.appendChild(label);\n\n        return this.fileElement;\n    }\n\n    private enableSelecting(): void {\n        const elm = this.element.nativeElement;\n\n        if (isFileInput(elm)) {\n            const bindedHandler = () => this.beforeSelect();\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            return;\n        } else {\n            const bindedHandler = (event: Event) => this.clickHandler(event);\n            elm.addEventListener('click', bindedHandler);\n            elm.addEventListener('touchstart', bindedHandler);\n            elm.addEventListener('touchend', bindedHandler);\n        }\n    }\n\n    protected handleFiles(files: File[]): void {\n        const selection: FileSelection = {\n            valid: [],\n            invalid: [],\n        };\n\n        for (const file of files) {\n            const error = this.validate(file);\n            if (error) {\n                selection.invalid.push({\n                    file: file,\n                    error: error,\n                });\n            } else {\n                selection.valid.push(file);\n            }\n        }\n\n        if (selection.valid.length) {\n            this.fileChange.emit(selection.valid[0]);\n        }\n\n        if (selection.valid.length || selection.invalid.length) {\n            this.filesChange.emit(selection);\n\n            if (this.broadcast) {\n                this.naturalFileService.filesChanged.next(selection);\n            }\n        }\n\n        this.getFileElement().value = '';\n    }\n\n    /**\n     * Called when input has files\n     */\n    private changeFn(event: Event): void {\n        if (!(event.target instanceof HTMLInputElement)) {\n            return;\n        }\n\n        const fileList = event.target.files;\n\n        if (!fileList) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(fileListToArray(fileList));\n    }\n\n    private clickHandler(event: Event): boolean {\n        const elm = this.element.nativeElement;\n        if (elm.getAttribute('disabled') || this.fileSelectionDisabled) {\n            return false;\n        }\n\n        // prevent the click if it is a swipe\n        if (detectSwipe(event)) {\n            return true;\n        }\n\n        const fileElm = this.getFileElement();\n        fileElm.click();\n        this.beforeSelect();\n\n        return false;\n    }\n\n    private beforeSelect(): void {\n        if (!this.fileElement) {\n            return;\n        }\n\n        // if no files in array, be sure browser doesnt prevent reselect of same file (see github issue 27)\n        this.fileElement.value = '';\n    }\n\n    @HostListener('change', ['$event'])\n    public onChange(event: Event): void {\n        const fileList = this.getFileElement().files;\n        const files: File[] = fileList ? fileListToArray(fileList) : eventToFiles(event);\n\n        if (!files.length) {\n            return;\n        }\n\n        stopEvent(event);\n        this.handleFiles(files);\n    }\n\n    private validate(file: File): string | null {\n        for (const validator of this.validators) {\n            if (!validator.fn.call(this, file)) {\n                return validator.name;\n            }\n        }\n\n        return null;\n    }\n\n    private acceptValidator(item: File): boolean {\n        return acceptType(this.accept, item.type, item.name);\n    }\n\n    private fileSizeValidator(item: File): boolean {\n        return !(this.maxSize && item.size > this.maxSize);\n    }\n}\n"]}
|
|
@@ -79,7 +79,9 @@ export class NaturalFileDropDirective extends NaturalAbstractFile {
|
|
|
79
79
|
this.closeDrags();
|
|
80
80
|
}
|
|
81
81
|
hasObservers() {
|
|
82
|
-
return this.fileChange.observed ||
|
|
82
|
+
return (this.fileChange.observed ||
|
|
83
|
+
this.filesChange.observed ||
|
|
84
|
+
(this.broadcast && this.naturalFileService.filesChanged.observed));
|
|
83
85
|
}
|
|
84
86
|
}
|
|
85
87
|
NaturalFileDropDirective.decorators = [
|
|
@@ -94,4 +96,4 @@ NaturalFileDropDirective.propDecorators = {
|
|
|
94
96
|
onDragOver: [{ type: HostListener, args: ['dragover', ['$event'],] }],
|
|
95
97
|
onDragLeave: [{ type: HostListener, args: ['dragleave', ['$event'],] }]
|
|
96
98
|
};
|
|
97
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
99
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1kcm9wLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2ZpbGUvZmlsZS1kcm9wLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFFLFlBQVksRUFBRSxXQUFXLEVBQUUsWUFBWSxFQUFVLE1BQU0sRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUNqRyxPQUFPLEVBQUMsbUJBQW1CLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztBQUNwRCxPQUFPLEVBQUMsWUFBWSxFQUFFLFNBQVMsRUFBQyxNQUFNLFNBQVMsQ0FBQztBQUNoRCxPQUFPLEVBQUMsY0FBYyxFQUFFLE9BQU8sRUFBQyxNQUFNLE1BQU0sQ0FBQztBQUM3QyxPQUFPLEVBQUMsU0FBUyxFQUFFLFlBQVksRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRXZEOzs7Ozs7Ozs7OztHQVdHO0FBSUgsTUFBTSxPQUFPLHdCQUF5QixTQUFRLG1CQUFtQjtJQUhqRTs7UUFJbUQsa0JBQWEsR0FBRyxLQUFLLENBQUM7UUFFckU7O1dBRUc7UUFDdUIsYUFBUSxHQUFHLElBQUksWUFBWSxFQUFXLENBQUM7UUFFaEQsZ0JBQVcsR0FBRyxJQUFJLE9BQU8sRUFBVyxDQUFDO0lBa0YxRCxDQUFDO0lBaEZVLFFBQVE7UUFDWCxLQUFLLENBQUMsUUFBUSxFQUFFLENBQUM7UUFFakIsb0VBQW9FO1FBQ3BFLDRFQUE0RTtRQUM1RSwwRUFBMEU7UUFDMUUscUVBQXFFO1FBQ3JFLElBQUksQ0FBQyxXQUFXO2FBQ1gsSUFBSSxDQUNELFNBQVMsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLEVBQzdCLFlBQVksQ0FBQyxHQUFHLEVBQUUsY0FBYyxFQUFFO1lBQzlCLE9BQU8sRUFBRSxJQUFJO1lBQ2IsUUFBUSxFQUFFLElBQUk7U0FDakIsQ0FBQyxDQUNMO2FBQ0EsU0FBUyxDQUFDLFFBQVEsQ0FBQyxFQUFFO1lBQ2xCLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQzdCLElBQUksQ0FBQyxhQUFhLEdBQUcsUUFBUSxDQUFDO1FBQ2xDLENBQUMsQ0FBQyxDQUFDO0lBQ1gsQ0FBQztJQUdNLE1BQU0sQ0FBQyxLQUFnQjtRQUMxQixJQUFJLElBQUksQ0FBQyxxQkFBcUIsRUFBRTtZQUM1QixPQUFPO1NBQ1Y7UUFFRCxJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7UUFFbEIsTUFBTSxLQUFLLEdBQUcsWUFBWSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ2xDLElBQUksQ0FBQyxLQUFLLENBQUMsTUFBTSxFQUFFO1lBQ2YsT0FBTztTQUNWO1FBRUQsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ2pCLElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDNUIsQ0FBQztJQUdNLFVBQVUsQ0FBQyxLQUFnQjtRQUM5QixJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksRUFBRSxFQUFFO1lBQ3RCLE9BQU87U0FDVjtRQUVELElBQUksSUFBSSxDQUFDLHFCQUFxQixFQUFFO1lBQzVCLE9BQU87U0FDVjtRQUVELFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUVqQixnQkFBZ0I7UUFDaEIsTUFBTSxRQUFRLEdBQUcsS0FBSyxDQUFDLFlBQVksQ0FBQztRQUNwQyxJQUFJLFFBQVEsRUFBRTtZQUNWLFFBQVEsQ0FBQyxVQUFVLEdBQUcsTUFBTSxDQUFDO1NBQ2hDO1FBRUQsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDaEMsQ0FBQztJQUVPLFVBQVU7UUFDZCxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBR00sV0FBVyxDQUFDLEtBQWdCO1FBQy9CLElBQUksSUFBSSxDQUFDLHFCQUFxQixFQUFFO1lBQzVCLE9BQU87U0FDVjtRQUVELFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNqQixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDdEIsQ0FBQztJQUVPLFlBQVk7UUFDaEIsT0FBTyxDQUNILElBQUksQ0FBQyxVQUFVLENBQUMsUUFBUTtZQUN4QixJQUFJLENBQUMsV0FBVyxDQUFDLFFBQVE7WUFDekIsQ0FBQyxJQUFJLENBQUMsU0FBUyxJQUFJLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLENBQ3BFLENBQUM7SUFDTixDQUFDOzs7WUE1RkosU0FBUyxTQUFDO2dCQUNQLFFBQVEsRUFBRSw0Q0FBNEM7YUFDekQ7Ozs0QkFFSSxXQUFXLFNBQUMseUJBQXlCO3VCQUtyQyxNQUFNO3FCQXlCTixZQUFZLFNBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDO3lCQWlCL0IsWUFBWSxTQUFDLFVBQVUsRUFBRSxDQUFDLFFBQVEsQ0FBQzswQkF5Qm5DLFlBQVksU0FBQyxXQUFXLEVBQUUsQ0FBQyxRQUFRLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0RpcmVjdGl2ZSwgRXZlbnRFbWl0dGVyLCBIb3N0QmluZGluZywgSG9zdExpc3RlbmVyLCBPbkluaXQsIE91dHB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge05hdHVyYWxBYnN0cmFjdEZpbGV9IGZyb20gJy4vYWJzdHJhY3QtZmlsZSc7XG5pbXBvcnQge2V2ZW50VG9GaWxlcywgc3RvcEV2ZW50fSBmcm9tICcuL3V0aWxzJztcbmltcG9ydCB7YXN5bmNTY2hlZHVsZXIsIFN1YmplY3R9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHt0YWtlVW50aWwsIHRocm90dGxlVGltZX0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuXG4vKipcbiAqIFRoaXMgZGlyZWN0aXZlIGhhcyBhbGwgb3B0aW9ucyB0byBzZWxlY3QgZmlsZXMsIGFuZCBhZGRzIHN1cHBvcnQgZm9yIGRyYWcnbidkcm9wLlxuICpcbiAqIEl0IHdpbGwgYWRkIHRoZSBDU1MgY2xhc3MgYG5hdHVyYWwtZmlsZS1vdmVyYCBvbiB0aGUgY29tcG9uZW50IHdoZW4gYSBmaWxlIGlzXG4gKiBkcmFnZ2VkIG92ZXIuIEl0IGlzIHVwIHRvIHRoZSBjb21wb25lbnQgdG8gaGF2ZSBzb21lIHNwZWNpZmljIHN0eWxpbmcgYnkgdXNpbmdcbiAqIHRoaXMgY2xhc3MuXG4gKlxuICogSW4gbW9zdCBjYXNlcyB5b3UgcHJvYmFibHkgYWxzbyB3YW50IGNsaWNrLXRvLXNlbGVjdCwgc28geW91IHNob3VsZCB1c2U6XG4gKlxuICogICAgIDxkaXYgbmF0dXJhbEZpbGVEcm9wIFtzZWxlY3RhYmxlXT1cInRydWVcIj48L2Rpdj5cbiAqXG4gKi9cbkBEaXJlY3RpdmUoe1xuICAgIHNlbGVjdG9yOiAnOm5vdChbbmF0dXJhbEZpbGVTZWxlY3RdKVtuYXR1cmFsRmlsZURyb3BdJyxcbn0pXG5leHBvcnQgY2xhc3MgTmF0dXJhbEZpbGVEcm9wRGlyZWN0aXZlIGV4dGVuZHMgTmF0dXJhbEFic3RyYWN0RmlsZSBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gICAgQEhvc3RCaW5kaW5nKCdjbGFzcy5uYXR1cmFsLWZpbGUtb3ZlcicpIHB1YmxpYyBmaWxlT3ZlckNsYXNzID0gZmFsc2U7XG5cbiAgICAvKipcbiAgICAgKiBFbWl0cyB3aGVuZXZlciBmaWxlcyBhcmUgYmVpbmcgZHJhZ2dlZCBvdmVyXG4gICAgICovXG4gICAgQE91dHB1dCgpIHB1YmxpYyByZWFkb25seSBmaWxlT3ZlciA9IG5ldyBFdmVudEVtaXR0ZXI8Ym9vbGVhbj4oKTtcblxuICAgIHByaXZhdGUgcmVhZG9ubHkgcmF3RmlsZU92ZXIgPSBuZXcgU3ViamVjdDxib29sZWFuPigpO1xuXG4gICAgcHVibGljIG5nT25Jbml0KCk6IHZvaWQge1xuICAgICAgICBzdXBlci5uZ09uSW5pdCgpO1xuXG4gICAgICAgIC8vIEF1dG9tYXRpY2FsbHkgY2hhbmdlIHRoZSBjbGFzcywgYnV0IG5vdCB0b28gb2Z0ZW4gdG8gYXZvaWQgdmlzdWFsXG4gICAgICAgIC8vIGZsaWNrZXJpbmcgaW4gQ2hyb21lIHdoZW4gaG92ZXJpbmcgYWNyb3NzIGNoaWxkIEhUTUwgZWxlbWVudCBvZiBvdXIgaG9zdC5cbiAgICAgICAgLy8gSXQncyBub3QgYWJzb2x1dGVseSBwZXJmZWN0IGFuZCBpZiBkcmFnZ2luZyBzbG93bHkgYW5kIHByZWNpc2VseSB3ZSBjYW5cbiAgICAgICAgLy8gc3RpbGwgc2VlIGZsaWNrZXIsIGJ1dCBpdCBzaG91bGQgYmUgYmV0dGVyIGZvciBtb3N0IG5vcm1hbCB1c2FnZXMuXG4gICAgICAgIHRoaXMucmF3RmlsZU92ZXJcbiAgICAgICAgICAgIC5waXBlKFxuICAgICAgICAgICAgICAgIHRha2VVbnRpbCh0aGlzLm5nVW5zdWJzY3JpYmUpLFxuICAgICAgICAgICAgICAgIHRocm90dGxlVGltZSgyMDAsIGFzeW5jU2NoZWR1bGVyLCB7XG4gICAgICAgICAgICAgICAgICAgIGxlYWRpbmc6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgIHRyYWlsaW5nOiB0cnVlLFxuICAgICAgICAgICAgICAgIH0pLFxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgLnN1YnNjcmliZShmaWxlT3ZlciA9PiB7XG4gICAgICAgICAgICAgICAgdGhpcy5maWxlT3Zlci5lbWl0KGZpbGVPdmVyKTtcbiAgICAgICAgICAgICAgICB0aGlzLmZpbGVPdmVyQ2xhc3MgPSBmaWxlT3ZlcjtcbiAgICAgICAgICAgIH0pO1xuICAgIH1cblxuICAgIEBIb3N0TGlzdGVuZXIoJ2Ryb3AnLCBbJyRldmVudCddKVxuICAgIHB1YmxpYyBvbkRyb3AoZXZlbnQ6IERyYWdFdmVudCk6IHZvaWQge1xuICAgICAgICBpZiAodGhpcy5maWxlU2VsZWN0aW9uRGlzYWJsZWQpIHtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuY2xvc2VEcmFncygpO1xuXG4gICAgICAgIGNvbnN0IGZpbGVzID0gZXZlbnRUb0ZpbGVzKGV2ZW50KTtcbiAgICAgICAgaWYgKCFmaWxlcy5sZW5ndGgpIHtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIHN0b3BFdmVudChldmVudCk7XG4gICAgICAgIHRoaXMuaGFuZGxlRmlsZXMoZmlsZXMpO1xuICAgIH1cblxuICAgIEBIb3N0TGlzdGVuZXIoJ2RyYWdvdmVyJywgWyckZXZlbnQnXSlcbiAgICBwdWJsaWMgb25EcmFnT3ZlcihldmVudDogRHJhZ0V2ZW50KTogdm9pZCB7XG4gICAgICAgIGlmICghdGhpcy5oYXNPYnNlcnZlcnMoKSkge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHRoaXMuZmlsZVNlbGVjdGlvbkRpc2FibGVkKSB7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBzdG9wRXZlbnQoZXZlbnQpO1xuXG4gICAgICAgIC8vIENoYW5nZSBjdXJzb3JcbiAgICAgICAgY29uc3QgdHJhbnNmZXIgPSBldmVudC5kYXRhVHJhbnNmZXI7XG4gICAgICAgIGlmICh0cmFuc2Zlcikge1xuICAgICAgICAgICAgdHJhbnNmZXIuZHJvcEVmZmVjdCA9ICdjb3B5JztcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMucmF3RmlsZU92ZXIubmV4dCh0cnVlKTtcbiAgICB9XG5cbiAgICBwcml2YXRlIGNsb3NlRHJhZ3MoKTogdm9pZCB7XG4gICAgICAgIHRoaXMucmF3RmlsZU92ZXIubmV4dChmYWxzZSk7XG4gICAgfVxuXG4gICAgQEhvc3RMaXN0ZW5lcignZHJhZ2xlYXZlJywgWyckZXZlbnQnXSlcbiAgICBwdWJsaWMgb25EcmFnTGVhdmUoZXZlbnQ6IERyYWdFdmVudCk6IHZvaWQge1xuICAgICAgICBpZiAodGhpcy5maWxlU2VsZWN0aW9uRGlzYWJsZWQpIHtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIHN0b3BFdmVudChldmVudCk7XG4gICAgICAgIHRoaXMuY2xvc2VEcmFncygpO1xuICAgIH1cblxuICAgIHByaXZhdGUgaGFzT2JzZXJ2ZXJzKCk6IGJvb2xlYW4ge1xuICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgdGhpcy5maWxlQ2hhbmdlLm9ic2VydmVkIHx8XG4gICAgICAgICAgICB0aGlzLmZpbGVzQ2hhbmdlLm9ic2VydmVkIHx8XG4gICAgICAgICAgICAodGhpcy5icm9hZGNhc3QgJiYgdGhpcy5uYXR1cmFsRmlsZVNlcnZpY2UuZmlsZXNDaGFuZ2VkLm9ic2VydmVkKVxuICAgICAgICApO1xuICAgIH1cbn1cbiJdfQ==
|
|
@@ -155,7 +155,7 @@ export class NaturalRelationsComponent extends NaturalAbstractController {
|
|
|
155
155
|
NaturalRelationsComponent.decorators = [
|
|
156
156
|
{ type: Component, args: [{
|
|
157
157
|
selector: 'natural-relations',
|
|
158
|
-
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 (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",
|
|
158
|
+
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",
|
|
159
159
|
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"]
|
|
160
160
|
},] }
|
|
161
161
|
];
|
|
@@ -178,4 +178,4 @@ NaturalRelationsComponent.propDecorators = {
|
|
|
178
178
|
otherName: [{ type: Input }],
|
|
179
179
|
filter: [{ type: Input }]
|
|
180
180
|
};
|
|
181
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"relations.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/relations/relations.component.ts"],"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    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() 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"]}
|
|
181
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"relations.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/relations/relations.component.ts"],"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    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() 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"]}
|
|
@@ -345,6 +345,18 @@ function mergeOverrideArray(destValue, source) {
|
|
|
345
345
|
return source;
|
|
346
346
|
}
|
|
347
347
|
}
|
|
348
|
+
/**
|
|
349
|
+
* Copy text to clipboard.
|
|
350
|
+
* Accepts line breaks `\n` as textarea do.
|
|
351
|
+
*/
|
|
352
|
+
function copyToClipboard(document, text) {
|
|
353
|
+
const input = document.createElement('textarea');
|
|
354
|
+
document.body.append(input);
|
|
355
|
+
input.value = text;
|
|
356
|
+
input.select();
|
|
357
|
+
document.execCommand('copy');
|
|
358
|
+
document.body.removeChild(input);
|
|
359
|
+
}
|
|
348
360
|
|
|
349
361
|
// Basic; loosely typed structure for graphql-doctrine filters
|
|
350
362
|
// Logical operator to be used in conditions
|
|
@@ -433,9 +445,13 @@ class NaturalQueryVariablesManager {
|
|
|
433
445
|
}
|
|
434
446
|
this.updateVariables();
|
|
435
447
|
}
|
|
448
|
+
/**
|
|
449
|
+
* Return a deep clone of the variables for the given channel name.
|
|
450
|
+
*
|
|
451
|
+
* Avoid returning the same reference to prevent an attribute change, then another channel update that would
|
|
452
|
+
* used this changed attribute without having explicitly asked QueryVariablesManager to update it.
|
|
453
|
+
*/
|
|
436
454
|
get(channelName) {
|
|
437
|
-
// Avoid to return the same reference to prevent an attribute change, then another channel update that would used this changed
|
|
438
|
-
// attribute without having explicitly asked QueryVariablesManager to update it.
|
|
439
455
|
return cloneDeep(this.channels.get(channelName));
|
|
440
456
|
}
|
|
441
457
|
/**
|
|
@@ -7962,6 +7978,13 @@ class NaturalAbstractFile extends NaturalAbstractController {
|
|
|
7962
7978
|
* no effect.
|
|
7963
7979
|
*/
|
|
7964
7980
|
this.selectable = false;
|
|
7981
|
+
/**
|
|
7982
|
+
* If true, the file selection will be broadcast through `NaturalFileService.filesChanged`.
|
|
7983
|
+
*
|
|
7984
|
+
* It is useful to set this to false if there is two upload on a page with different purpose
|
|
7985
|
+
* and the second upload should not be confused with the first one.
|
|
7986
|
+
*/
|
|
7987
|
+
this.broadcast = true;
|
|
7965
7988
|
/**
|
|
7966
7989
|
* The single valid file that has been selected.
|
|
7967
7990
|
*
|
|
@@ -8042,7 +8065,9 @@ class NaturalAbstractFile extends NaturalAbstractController {
|
|
|
8042
8065
|
}
|
|
8043
8066
|
if (selection.valid.length || selection.invalid.length) {
|
|
8044
8067
|
this.filesChange.emit(selection);
|
|
8045
|
-
this.
|
|
8068
|
+
if (this.broadcast) {
|
|
8069
|
+
this.naturalFileService.filesChanged.next(selection);
|
|
8070
|
+
}
|
|
8046
8071
|
}
|
|
8047
8072
|
this.getFileElement().value = '';
|
|
8048
8073
|
}
|
|
@@ -8119,6 +8144,7 @@ NaturalAbstractFile.propDecorators = {
|
|
|
8119
8144
|
maxSize: [{ type: Input }],
|
|
8120
8145
|
fileSelectionDisabled: [{ type: Input }],
|
|
8121
8146
|
selectable: [{ type: Input }],
|
|
8147
|
+
broadcast: [{ type: Input }],
|
|
8122
8148
|
fileChange: [{ type: Output }],
|
|
8123
8149
|
filesChange: [{ type: Output }],
|
|
8124
8150
|
onChange: [{ type: HostListener, args: ['change', ['$event'],] }]
|
|
@@ -8200,7 +8226,9 @@ class NaturalFileDropDirective extends NaturalAbstractFile {
|
|
|
8200
8226
|
this.closeDrags();
|
|
8201
8227
|
}
|
|
8202
8228
|
hasObservers() {
|
|
8203
|
-
return this.fileChange.observed ||
|
|
8229
|
+
return (this.fileChange.observed ||
|
|
8230
|
+
this.filesChange.observed ||
|
|
8231
|
+
(this.broadcast && this.naturalFileService.filesChanged.observed));
|
|
8204
8232
|
}
|
|
8205
8233
|
}
|
|
8206
8234
|
NaturalFileDropDirective.decorators = [
|
|
@@ -9073,7 +9101,7 @@ class NaturalRelationsComponent extends NaturalAbstractController {
|
|
|
9073
9101
|
NaturalRelationsComponent.decorators = [
|
|
9074
9102
|
{ type: Component, args: [{
|
|
9075
9103
|
selector: 'natural-relations',
|
|
9076
|
-
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 (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",
|
|
9104
|
+
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",
|
|
9077
9105
|
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"]
|
|
9078
9106
|
},] }
|
|
9079
9107
|
];
|
|
@@ -10116,5 +10144,5 @@ NaturalMatomoService.ctorParameters = () => [
|
|
|
10116
10144
|
* Generated bundle index. Do not edit.
|
|
10117
10145
|
*/
|
|
10118
10146
|
|
|
10119
|
-
export { AvatarComponent, AvatarService, FileComponent, IconsConfigService, LOCAL_STORAGE, NATURAL_DROPDOWN_DATA, NATURAL_SEO_CONFIG, NaturalAbstractController, NaturalAbstractDetail, NaturalAbstractEditableList, NaturalAbstractList, NaturalAbstractModelService, NaturalAbstractNavigableList, NaturalAbstractPanel, NaturalAlertModule, NaturalAlertService, NaturalAvatarModule, NaturalCapitalizePipe, NaturalColumnsPickerColumnDirective, NaturalColumnsPickerComponent, NaturalColumnsPickerModule, NaturalCommonModule, NaturalConfirmComponent, NaturalDataSource, NaturalDetailHeaderComponent, NaturalDetailHeaderModule, NaturalDialogTriggerComponent, NaturalDialogTriggerModule, NaturalDropdownComponentsModule, NaturalDropdownRef, NaturalEllipsisPipe, NaturalEnumPipe, NaturalEnumService, NaturalFileDropDirective, NaturalFileModule, NaturalFileSelectDirective, NaturalFileService, NaturalFixedButtonComponent, NaturalFixedButtonDetailComponent, NaturalFixedButtonDetailModule, NaturalFixedButtonModule, NaturalHierarchicSelectorComponent, NaturalHierarchicSelectorDialogComponent, NaturalHierarchicSelectorDialogService, NaturalHierarchicSelectorModule, NaturalHierarchicSelectorService, NaturalHttpPrefixDirective, NaturalIconComponent, NaturalIconModule, NaturalLinkMutationService, NaturalLinkableTabDirective, NaturalMatomoModule, NaturalMatomoService, NaturalMemoryStorage, NaturalPanelsComponent, NaturalPanelsModule, NaturalPanelsService, NaturalPersistenceService, NaturalQueryVariablesManager, NaturalRelationsComponent, NaturalRelationsModule, NaturalSearchComponent, NaturalSearchModule, NaturalSelectComponent, NaturalSelectEnumComponent, NaturalSelectHierarchicComponent, NaturalSelectModule, NaturalSeoService, NaturalSidenavComponent, NaturalSidenavContainerComponent, NaturalSidenavContentComponent, NaturalSidenavModule, NaturalSidenavService, NaturalSidenavStackService, NaturalSrcDensityDirective, NaturalStampComponent, NaturalStampModule, NaturalSwissDatePipe, NaturalSwissParsingDateAdapter, NaturalTableButtonComponent, NaturalTableButtonModule, PanelsHooksConfig, ReactiveAsteriskDirective, SESSION_STORAGE, SortingOrder, TypeDateComponent, TypeDateRangeComponent, TypeHierarchicSelectorComponent, TypeNaturalSelectComponent, TypeNumberComponent, TypeSelectComponent, TypeTextComponent, available, cancellableTimeout, cleanSameValues, collectErrors, debug, decimal, deliverableEmail, ensureHttpPrefix, fallbackIfNoOpenedPanels, formatIsoDate, formatIsoDateTime, fromUrl, getForegroundColor, hasFilesAndProcessDate, ifValid, integer, localStorageFactory, localStorageProvider, lowerCaseFirstLetter, makePlural, memoryLocalStorageProvider, memorySessionStorageProvider, mergeOverrideArray, money, naturalPanelsUrlMatcher, relationsToIds, replaceObjectKeepingReference, replaceOperatorByField, replaceOperatorByName, sessionStorageFactory, sessionStorageProvider, toGraphQLDoctrineFilter, toNavigationParameters, toUrl, unique, upperCaseFirstLetter, urlValidator, validTlds, validateAllFormControls, wrapLike, NaturalAbstractFile as ɵa, NaturalDropdownService as ɵb, AbstractAssociationSelectComponent as ɵc, NATURAL_DROPDOWN_CONTAINER_DATA as ɵd, NaturalDropdownContainerComponent as ɵe, naturalDropdownAnimations as ɵf, AbstractSelect as ɵg, NaturalGroupComponent as ɵh, NaturalInputComponent as ɵi, FacetSelectorComponent as ɵj };
|
|
10147
|
+
export { AvatarComponent, AvatarService, FileComponent, IconsConfigService, LOCAL_STORAGE, NATURAL_DROPDOWN_DATA, NATURAL_SEO_CONFIG, NaturalAbstractController, NaturalAbstractDetail, NaturalAbstractEditableList, NaturalAbstractList, NaturalAbstractModelService, NaturalAbstractNavigableList, NaturalAbstractPanel, NaturalAlertModule, NaturalAlertService, NaturalAvatarModule, NaturalCapitalizePipe, NaturalColumnsPickerColumnDirective, NaturalColumnsPickerComponent, NaturalColumnsPickerModule, NaturalCommonModule, NaturalConfirmComponent, NaturalDataSource, NaturalDetailHeaderComponent, NaturalDetailHeaderModule, NaturalDialogTriggerComponent, NaturalDialogTriggerModule, NaturalDropdownComponentsModule, NaturalDropdownRef, NaturalEllipsisPipe, NaturalEnumPipe, NaturalEnumService, NaturalFileDropDirective, NaturalFileModule, NaturalFileSelectDirective, NaturalFileService, NaturalFixedButtonComponent, NaturalFixedButtonDetailComponent, NaturalFixedButtonDetailModule, NaturalFixedButtonModule, NaturalHierarchicSelectorComponent, NaturalHierarchicSelectorDialogComponent, NaturalHierarchicSelectorDialogService, NaturalHierarchicSelectorModule, NaturalHierarchicSelectorService, NaturalHttpPrefixDirective, NaturalIconComponent, NaturalIconModule, NaturalLinkMutationService, NaturalLinkableTabDirective, NaturalMatomoModule, NaturalMatomoService, NaturalMemoryStorage, NaturalPanelsComponent, NaturalPanelsModule, NaturalPanelsService, NaturalPersistenceService, NaturalQueryVariablesManager, NaturalRelationsComponent, NaturalRelationsModule, NaturalSearchComponent, NaturalSearchModule, NaturalSelectComponent, NaturalSelectEnumComponent, NaturalSelectHierarchicComponent, NaturalSelectModule, NaturalSeoService, NaturalSidenavComponent, NaturalSidenavContainerComponent, NaturalSidenavContentComponent, NaturalSidenavModule, NaturalSidenavService, NaturalSidenavStackService, NaturalSrcDensityDirective, NaturalStampComponent, NaturalStampModule, NaturalSwissDatePipe, NaturalSwissParsingDateAdapter, NaturalTableButtonComponent, NaturalTableButtonModule, PanelsHooksConfig, ReactiveAsteriskDirective, SESSION_STORAGE, SortingOrder, TypeDateComponent, TypeDateRangeComponent, TypeHierarchicSelectorComponent, TypeNaturalSelectComponent, TypeNumberComponent, TypeSelectComponent, TypeTextComponent, available, cancellableTimeout, cleanSameValues, collectErrors, copyToClipboard, debug, decimal, deliverableEmail, ensureHttpPrefix, fallbackIfNoOpenedPanels, formatIsoDate, formatIsoDateTime, fromUrl, getForegroundColor, hasFilesAndProcessDate, ifValid, integer, localStorageFactory, localStorageProvider, lowerCaseFirstLetter, makePlural, memoryLocalStorageProvider, memorySessionStorageProvider, mergeOverrideArray, money, naturalPanelsUrlMatcher, relationsToIds, replaceObjectKeepingReference, replaceOperatorByField, replaceOperatorByName, sessionStorageFactory, sessionStorageProvider, toGraphQLDoctrineFilter, toNavigationParameters, toUrl, unique, upperCaseFirstLetter, urlValidator, validTlds, validateAllFormControls, wrapLike, NaturalAbstractFile as ɵa, NaturalDropdownService as ɵb, AbstractAssociationSelectComponent as ɵc, NATURAL_DROPDOWN_CONTAINER_DATA as ɵd, NaturalDropdownContainerComponent as ɵe, naturalDropdownAnimations as ɵf, AbstractSelect as ɵg, NaturalGroupComponent as ɵh, NaturalInputComponent as ɵi, FacetSelectorComponent as ɵj };
|
|
10120
10148
|
//# sourceMappingURL=ecodev-natural.js.map
|