@ecodev/natural 55.6.1 → 55.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/classes/utility.mjs +12 -2
- package/esm2022/lib/modules/dropdown-components/public-api.mjs +3 -2
- package/esm2022/lib/modules/dropdown-components/type-text/type-text.component.mjs +4 -1
- package/esm2022/lib/modules/dropdown-components/types.mjs +1 -1
- package/esm2022/lib/modules/select/abstract-select.component.mjs +8 -2
- package/esm2022/lib/modules/select/select/select.component.mjs +3 -3
- package/esm2022/lib/modules/select/select-enum/select-enum.component.mjs +3 -3
- package/esm2022/lib/modules/select/select-hierarchic/select-hierarchic.component.mjs +3 -3
- package/fesm2022/ecodev-natural.mjs +28 -9
- package/fesm2022/ecodev-natural.mjs.map +1 -1
- package/lib/classes/utility.d.ts +6 -0
- package/lib/modules/dropdown-components/abstract-association-select-component.directive.d.ts +1 -1
- package/lib/modules/dropdown-components/public-api.d.ts +2 -1
- package/lib/modules/dropdown-components/type-date/type-date.component.d.ts +1 -1
- package/lib/modules/dropdown-components/type-number/type-number.component.d.ts +1 -1
- package/lib/modules/dropdown-components/type-select/type-select.component.d.ts +1 -1
- package/lib/modules/dropdown-components/type-text/type-text.component.d.ts +3 -0
- package/lib/modules/dropdown-components/types.d.ts +2 -2
- package/lib/modules/select/abstract-select.component.d.ts +5 -1
- package/package.json +1 -1
|
@@ -86,10 +86,20 @@ export function cleanSameValues(source, modified) {
|
|
|
86
86
|
}
|
|
87
87
|
/**
|
|
88
88
|
* Returns the plural form of the given name
|
|
89
|
+
*
|
|
90
|
+
* This is **not** necessarily valid english grammar. Its only purpose is for internal usage, not for humans.
|
|
91
|
+
*
|
|
92
|
+
* This **MUST** be kept in sync with `\Ecodev\Felix\Api\Plural:make()`.
|
|
93
|
+
*
|
|
94
|
+
* This is a bit performance-sensitive, so we should keep it fast and only cover cases that we actually need.
|
|
89
95
|
*/
|
|
90
96
|
export function makePlural(name) {
|
|
97
|
+
// Words ending in a y preceded by a vowel form their plurals by adding -s:
|
|
98
|
+
if (name.match(/([aeiou])y$/)) {
|
|
99
|
+
return name + 's';
|
|
100
|
+
}
|
|
91
101
|
const plural = name + 's';
|
|
92
|
-
return plural.replace(/ys$/, 'ies').replace(/ss$/, 'ses')
|
|
102
|
+
return plural.replace(/ys$/, 'ies').replace(/ss$/, 'ses');
|
|
93
103
|
}
|
|
94
104
|
/**
|
|
95
105
|
* Returns the string with the first letter as capital
|
|
@@ -231,4 +241,4 @@ export function validateColumns(data) {
|
|
|
231
241
|
}
|
|
232
242
|
return data.split(',').filter(string => string);
|
|
233
243
|
}
|
|
234
|
-
//# 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, pickBy} from 'lodash-es';\nimport {Literal} from '../types/types';\nimport type {ReadonlyDeep} from 'type-fest';\nimport {PaginationInput, Sorting, SortingOrder} from './query-variable-manager';\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: null): null;\nexport function formatIsoDate(date: Date): string;\nexport function formatIsoDate(date: Date | null): string | null;\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 allows 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: unknown = object[key];\n\n        if (value === null || value === undefined) {\n            // noop\n        } else if (hasId(value)) {\n            value = value.id;\n        } else if (isArray(value)) {\n            value = value.map((i: unknown) => (hasId(i) ? i.id : i));\n        } else if (typeof value === 'object' && !(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\nfunction hasId(value: unknown): value is {id: unknown} {\n    return !!value && typeof value === 'object' && 'id' in value && !!value.id;\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\nexport function deepFreeze<T extends Literal>(o: T): ReadonlyDeep<T> {\n    Object.values(o).forEach(v => Object.isFrozen(v) || deepFreeze(v));\n\n    return Object.freeze(o) as ReadonlyDeep<T>;\n}\n\n/**\n * Return a valid PaginationInput from whatever is available from data. Invalid properties/types will be dropped.\n */\nexport function validatePagination(data: unknown): PaginationInput | null {\n    if (!data || typeof data !== 'object' || Array.isArray(data)) {\n        return null;\n    }\n\n    const pagination: PaginationInput = {};\n\n    if ('offset' in data && (data.offset === null || typeof data.offset === 'number')) {\n        pagination.offset = data.offset;\n    }\n\n    if ('pageIndex' in data && (data.pageIndex === null || typeof data.pageIndex === 'number')) {\n        pagination.pageIndex = data.pageIndex;\n    }\n\n    if ('pageSize' in data && (data.pageSize === null || typeof data.pageSize === 'number')) {\n        pagination.pageSize = data.pageSize;\n    }\n\n    return pagination;\n}\n\n/**\n * Return a valid Sortings from whatever is available from data. Invalid properties/types will be dropped.\n */\nexport function validateSorting(data: unknown): Array<Sorting> | null {\n    if (!Array.isArray(data)) {\n        return null;\n    }\n    const result: Array<Sorting> = [];\n    data.forEach(s => {\n        const r = validateOneSorting(s);\n        if (r) {\n            result.push(r);\n        }\n    });\n\n    return result;\n}\n\nfunction validateOneSorting(data: unknown): Sorting | null {\n    if (!data || typeof data !== 'object' || !('field' in data)) {\n        return null;\n    }\n\n    const sorting: Sorting = {field: data.field};\n\n    if (\n        'order' in data &&\n        (data.order === SortingOrder.ASC || data.order === SortingOrder.DESC || data.order === null)\n    ) {\n        sorting.order = data.order;\n    }\n\n    if ('nullAsHighest' in data && (data.nullAsHighest === null || typeof data.nullAsHighest === 'boolean')) {\n        sorting.nullAsHighest = data.nullAsHighest;\n    }\n\n    if (\n        'emptyStringAsHighest' in data &&\n        (data.emptyStringAsHighest === null || typeof data.emptyStringAsHighest === 'boolean')\n    ) {\n        sorting.emptyStringAsHighest = data.emptyStringAsHighest;\n    }\n\n    return sorting;\n}\n\n/**\n * Return valid columns from whatever is available from data. Invalid properties/types will be dropped.\n */\nexport function validateColumns(data: unknown): string[] | null {\n    if (typeof data !== 'string') {\n        return null;\n    }\n\n    return data.split(',').filter(string => string);\n}\n"]}
|
|
244
|
+
//# 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, pickBy} from 'lodash-es';\nimport {Literal} from '../types/types';\nimport type {ReadonlyDeep} from 'type-fest';\nimport {PaginationInput, Sorting, SortingOrder} from './query-variable-manager';\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: null): null;\nexport function formatIsoDate(date: Date): string;\nexport function formatIsoDate(date: Date | null): string | null;\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 allows 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: unknown = object[key];\n\n        if (value === null || value === undefined) {\n            // noop\n        } else if (hasId(value)) {\n            value = value.id;\n        } else if (isArray(value)) {\n            value = value.map((i: unknown) => (hasId(i) ? i.id : i));\n        } else if (typeof value === 'object' && !(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\nfunction hasId(value: unknown): value is {id: unknown} {\n    return !!value && typeof value === 'object' && 'id' in value && !!value.id;\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 *\n * This is **not** necessarily valid english grammar. Its only purpose is for internal usage, not for humans.\n *\n * This **MUST** be kept in sync with `\\Ecodev\\Felix\\Api\\Plural:make()`.\n *\n * This is a bit performance-sensitive, so we should keep it fast and only cover cases that we actually need.\n */\nexport function makePlural(name: string): string {\n    // Words ending in a y preceded by a vowel form their plurals by adding -s:\n    if (name.match(/([aeiou])y$/)) {\n        return name + 's';\n    }\n\n    const plural = name + 's';\n\n    return plural.replace(/ys$/, 'ies').replace(/ss$/, 'ses');\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\nexport function deepFreeze<T extends Literal>(o: T): ReadonlyDeep<T> {\n    Object.values(o).forEach(v => Object.isFrozen(v) || deepFreeze(v));\n\n    return Object.freeze(o) as ReadonlyDeep<T>;\n}\n\n/**\n * Return a valid PaginationInput from whatever is available from data. Invalid properties/types will be dropped.\n */\nexport function validatePagination(data: unknown): PaginationInput | null {\n    if (!data || typeof data !== 'object' || Array.isArray(data)) {\n        return null;\n    }\n\n    const pagination: PaginationInput = {};\n\n    if ('offset' in data && (data.offset === null || typeof data.offset === 'number')) {\n        pagination.offset = data.offset;\n    }\n\n    if ('pageIndex' in data && (data.pageIndex === null || typeof data.pageIndex === 'number')) {\n        pagination.pageIndex = data.pageIndex;\n    }\n\n    if ('pageSize' in data && (data.pageSize === null || typeof data.pageSize === 'number')) {\n        pagination.pageSize = data.pageSize;\n    }\n\n    return pagination;\n}\n\n/**\n * Return a valid Sortings from whatever is available from data. Invalid properties/types will be dropped.\n */\nexport function validateSorting(data: unknown): Array<Sorting> | null {\n    if (!Array.isArray(data)) {\n        return null;\n    }\n    const result: Array<Sorting> = [];\n    data.forEach(s => {\n        const r = validateOneSorting(s);\n        if (r) {\n            result.push(r);\n        }\n    });\n\n    return result;\n}\n\nfunction validateOneSorting(data: unknown): Sorting | null {\n    if (!data || typeof data !== 'object' || !('field' in data)) {\n        return null;\n    }\n\n    const sorting: Sorting = {field: data.field};\n\n    if (\n        'order' in data &&\n        (data.order === SortingOrder.ASC || data.order === SortingOrder.DESC || data.order === null)\n    ) {\n        sorting.order = data.order;\n    }\n\n    if ('nullAsHighest' in data && (data.nullAsHighest === null || typeof data.nullAsHighest === 'boolean')) {\n        sorting.nullAsHighest = data.nullAsHighest;\n    }\n\n    if (\n        'emptyStringAsHighest' in data &&\n        (data.emptyStringAsHighest === null || typeof data.emptyStringAsHighest === 'boolean')\n    ) {\n        sorting.emptyStringAsHighest = data.emptyStringAsHighest;\n    }\n\n    return sorting;\n}\n\n/**\n * Return valid columns from whatever is available from data. Invalid properties/types will be dropped.\n */\nexport function validateColumns(data: unknown): string[] | null {\n    if (typeof data !== 'string') {\n        return null;\n    }\n\n    return data.split(',').filter(string => string);\n}\n"]}
|
|
@@ -7,7 +7,8 @@ export { TypeNumberComponent } from './type-number/type-number.component';
|
|
|
7
7
|
export { TypeHierarchicSelectorComponent, } from './type-hierarchic-selector/type-hierarchic-selector.component';
|
|
8
8
|
export { TypeDateComponent } from './type-date/type-date.component';
|
|
9
9
|
export { TypeDateRangeComponent } from './type-date-range/type-date-range.component';
|
|
10
|
-
export { TypeTextComponent } from './type-text/type-text.component';
|
|
10
|
+
export { TypeTextComponent, InvalidWithValueStateMatcher } from './type-text/type-text.component';
|
|
11
11
|
export { TypeBooleanComponent } from './type-boolean/type-boolean.component';
|
|
12
12
|
export { TypeOptionsComponent } from './type-options/type-options.component';
|
|
13
|
-
|
|
13
|
+
export { possibleComparableOperators } from './types';
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2Ryb3Bkb3duLWNvbXBvbmVudHMvcHVibGljLWFwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUNILE9BQU8sRUFBMEIsbUJBQW1CLEVBQUMsTUFBTSxxQ0FBcUMsQ0FBQztBQUVqRyxPQUFPLEVBRUgsMEJBQTBCLEdBQzdCLE1BQU0scURBQXFELENBQUM7QUFFN0QsT0FBTyxFQUEwQixtQkFBbUIsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBRWpHLE9BQU8sRUFFSCwrQkFBK0IsR0FDbEMsTUFBTSwrREFBK0QsQ0FBQztBQUV2RSxPQUFPLEVBQXdCLGlCQUFpQixFQUFDLE1BQU0saUNBQWlDLENBQUM7QUFFekYsT0FBTyxFQUE2QixzQkFBc0IsRUFBQyxNQUFNLDZDQUE2QyxDQUFDO0FBRS9HLE9BQU8sRUFBQyxpQkFBaUIsRUFBRSw0QkFBNEIsRUFBQyxNQUFNLGlDQUFpQyxDQUFDO0FBRWhHLE9BQU8sRUFBMkIsb0JBQW9CLEVBQUMsTUFBTSx1Q0FBdUMsQ0FBQztBQUNyRyxPQUFPLEVBQXVDLG9CQUFvQixFQUFDLE1BQU0sdUNBQXVDLENBQUM7QUFDakgsT0FBTyxFQUFDLDJCQUEyQixFQUFnQyxNQUFNLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBQdWJsaWMgQVBJIFN1cmZhY2Ugb2YgbmF0dXJhbC1zZWFyY2ggZHJvcGRvd24gY29tcG9uZW50c1xuICovXG5leHBvcnQge1R5cGVTZWxlY3RDb25maWd1cmF0aW9uLCBUeXBlU2VsZWN0Q29tcG9uZW50fSBmcm9tICcuL3R5cGUtc2VsZWN0L3R5cGUtc2VsZWN0LmNvbXBvbmVudCc7XG5cbmV4cG9ydCB7XG4gICAgVHlwZVNlbGVjdE5hdHVyYWxDb25maWd1cmF0aW9uLFxuICAgIFR5cGVOYXR1cmFsU2VsZWN0Q29tcG9uZW50LFxufSBmcm9tICcuL3R5cGUtbmF0dXJhbC1zZWxlY3QvdHlwZS1uYXR1cmFsLXNlbGVjdC5jb21wb25lbnQnO1xuXG5leHBvcnQge1R5cGVOdW1iZXJDb25maWd1cmF0aW9uLCBUeXBlTnVtYmVyQ29tcG9uZW50fSBmcm9tICcuL3R5cGUtbnVtYmVyL3R5cGUtbnVtYmVyLmNvbXBvbmVudCc7XG5cbmV4cG9ydCB7XG4gICAgVHlwZUhpZXJhcmNoaWNTZWxlY3RvckNvbmZpZ3VyYXRpb24sXG4gICAgVHlwZUhpZXJhcmNoaWNTZWxlY3RvckNvbXBvbmVudCxcbn0gZnJvbSAnLi90eXBlLWhpZXJhcmNoaWMtc2VsZWN0b3IvdHlwZS1oaWVyYXJjaGljLXNlbGVjdG9yLmNvbXBvbmVudCc7XG5cbmV4cG9ydCB7VHlwZURhdGVDb25maWd1cmF0aW9uLCBUeXBlRGF0ZUNvbXBvbmVudH0gZnJvbSAnLi90eXBlLWRhdGUvdHlwZS1kYXRlLmNvbXBvbmVudCc7XG5cbmV4cG9ydCB7VHlwZURhdGVSYW5nZUNvbmZpZ3VyYXRpb24sIFR5cGVEYXRlUmFuZ2VDb21wb25lbnR9IGZyb20gJy4vdHlwZS1kYXRlLXJhbmdlL3R5cGUtZGF0ZS1yYW5nZS5jb21wb25lbnQnO1xuXG5leHBvcnQge1R5cGVUZXh0Q29tcG9uZW50LCBJbnZhbGlkV2l0aFZhbHVlU3RhdGVNYXRjaGVyfSBmcm9tICcuL3R5cGUtdGV4dC90eXBlLXRleHQuY29tcG9uZW50JztcblxuZXhwb3J0IHtUeXBlQm9vbGVhbkNvbmZpZ3VyYXRpb24sIFR5cGVCb29sZWFuQ29tcG9uZW50fSBmcm9tICcuL3R5cGUtYm9vbGVhbi90eXBlLWJvb2xlYW4uY29tcG9uZW50JztcbmV4cG9ydCB7VHlwZU9wdGlvbnNDb25maWd1cmF0aW9uLCBUeXBlT3B0aW9uLCBUeXBlT3B0aW9uc0NvbXBvbmVudH0gZnJvbSAnLi90eXBlLW9wdGlvbnMvdHlwZS1vcHRpb25zLmNvbXBvbmVudCc7XG5leHBvcnQge3Bvc3NpYmxlQ29tcGFyYWJsZU9wZXJhdG9ycywgUG9zc2libGVDb21wYXJhYmxlT3BlcnRvcktleXN9IGZyb20gJy4vdHlwZXMnO1xuIl19
|
|
@@ -11,6 +11,9 @@ import * as i2 from "@angular/material/form-field";
|
|
|
11
11
|
import * as i3 from "@angular/material/input";
|
|
12
12
|
import * as i4 from "@angular/forms";
|
|
13
13
|
import * as i5 from "@angular/common";
|
|
14
|
+
/**
|
|
15
|
+
* Show an error message if the control has a value and an error, even if control is not dirty and not touched.
|
|
16
|
+
*/
|
|
14
17
|
export class InvalidWithValueStateMatcher {
|
|
15
18
|
isErrorState(control) {
|
|
16
19
|
return control && control.invalid && control.value;
|
|
@@ -60,4 +63,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
|
|
|
60
63
|
type: Inject,
|
|
61
64
|
args: [NATURAL_DROPDOWN_DATA]
|
|
62
65
|
}] }, { type: i1.NaturalDropdownRef }]; } });
|
|
63
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
66
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZS10ZXh0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25hdHVyYWwvc3JjL2xpYi9tb2R1bGVzL2Ryb3Bkb3duLWNvbXBvbmVudHMvdHlwZS10ZXh0L3R5cGUtdGV4dC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9kcm9wZG93bi1jb21wb25lbnRzL3R5cGUtdGV4dC90eXBlLXRleHQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFNBQVMsRUFBRSxNQUFNLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDaEQsT0FBTyxFQUFDLFdBQVcsRUFBRSxVQUFVLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFFekYsT0FBTyxFQUFDLGVBQWUsRUFBQyxNQUFNLE1BQU0sQ0FBQztBQUdyQyxPQUFPLEVBQUMscUJBQXFCLEVBQXNCLE1BQU0sa0RBQWtELENBQUM7QUFFNUcsT0FBTyxFQUFDLFlBQVksRUFBQyxNQUFNLGlCQUFpQixDQUFDO0FBQzdDLE9BQU8sRUFBQyxjQUFjLEVBQUMsTUFBTSx5QkFBeUIsQ0FBQztBQUN2RCxPQUFPLEVBQUMsa0JBQWtCLEVBQUMsTUFBTSw4QkFBOEIsQ0FBQzs7Ozs7OztBQUVoRTs7R0FFRztBQUNILE1BQU0sT0FBTyw0QkFBNEI7SUFDOUIsWUFBWSxDQUFDLE9BQTJCO1FBQzNDLE9BQU8sT0FBTyxJQUFJLE9BQU8sQ0FBQyxPQUFPLElBQUksT0FBTyxDQUFDLEtBQUssQ0FBQztJQUN2RCxDQUFDO0NBQ0o7QUFRRCxNQUFNLE9BQU8saUJBQWlCO0lBSzFCLFlBQ21DLElBQWdDLEVBQ3JELFdBQStCO1FBQS9CLGdCQUFXLEdBQVgsV0FBVyxDQUFvQjtRQU43QixrQkFBYSxHQUFHLElBQUksZUFBZSxDQUFTLEVBQUUsQ0FBQyxDQUFDO1FBQ2hELGFBQVEsR0FBRyxJQUFJLFdBQVcsQ0FBQyxFQUFFLEVBQUUsRUFBQyxXQUFXLEVBQUUsSUFBSSxFQUFDLENBQUMsQ0FBQztRQUNwRCxZQUFPLEdBQUcsSUFBSSw0QkFBNEIsRUFBRSxDQUFDO1FBTXpELElBQUksQ0FBQyxRQUFRLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsRUFBRTtZQUN6QyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxLQUFLLEtBQUssSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQyxDQUFDO1FBQzVFLENBQUMsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFFBQVEsQ0FBQyxhQUFhLENBQUMsQ0FBQyxVQUFVLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQztRQUVuRCxJQUFJLElBQUksQ0FBQyxTQUFTLElBQUksSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUU7WUFDdkMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUMsRUFBRSxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1NBQzFEO0lBQ0wsQ0FBQztJQUVNLFlBQVk7UUFDZixJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLEVBQUU7WUFDdEIsT0FBTyxFQUFFLENBQUM7U0FDYjtRQUVELE9BQU8sRUFBQyxJQUFJLEVBQUUsRUFBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLEVBQUMsRUFBQyxDQUFDO0lBQ2hELENBQUM7SUFFTSxPQUFPO1FBQ1YsT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQztJQUMvQixDQUFDO0lBRU0sT0FBTztRQUNWLE9BQU8sSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUM7SUFDL0IsQ0FBQztJQUVNLEtBQUs7UUFDUixJQUFJLElBQUksQ0FBQyxPQUFPLEVBQUUsRUFBRTtZQUNoQixJQUFJLENBQUMsV0FBVyxDQUFDLEtBQUssQ0FBQyxFQUFDLFNBQVMsRUFBRSxJQUFJLENBQUMsWUFBWSxFQUFFLEVBQUMsQ0FBQyxDQUFDO1NBQzVEO2FBQU07WUFDSCxJQUFJLENBQUMsV0FBVyxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUMsK0VBQStFO1NBQzVHO0lBQ0wsQ0FBQzs4R0ExQ1EsaUJBQWlCLGtCQU1kLHFCQUFxQjtrR0FOeEIsaUJBQWlCLHdFQzNCOUIsaVdBWUEsMEtEYWMsa0JBQWtCLHVZQUFFLGNBQWMsMFdBQUUsV0FBVyxtbkJBQUUsbUJBQW1CLGlOQUFFLFlBQVk7OzJGQUVuRixpQkFBaUI7a0JBTjdCLFNBQVM7aUNBR00sSUFBSSxXQUNQLENBQUMsa0JBQWtCLEVBQUUsY0FBYyxFQUFFLFdBQVcsRUFBRSxtQkFBbUIsRUFBRSxZQUFZLENBQUM7OzBCQVF4RixNQUFNOzJCQUFDLHFCQUFxQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29tcG9uZW50LCBJbmplY3R9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtGb3JtQ29udHJvbCwgVmFsaWRhdG9ycywgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7RXJyb3JTdGF0ZU1hdGNoZXJ9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NvcmUnO1xuaW1wb3J0IHtCZWhhdmlvclN1YmplY3R9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHtGaWx0ZXJHcm91cENvbmRpdGlvbkZpZWxkfSBmcm9tICcuLi8uLi9zZWFyY2gvY2xhc3Nlcy9ncmFwaHFsLWRvY3RyaW5lLnR5cGVzJztcbmltcG9ydCB7TmF0dXJhbERyb3Bkb3duUmVmfSBmcm9tICcuLi8uLi9zZWFyY2gvZHJvcGRvd24tY29udGFpbmVyL2Ryb3Bkb3duLXJlZic7XG5pbXBvcnQge05BVFVSQUxfRFJPUERPV05fREFUQSwgTmF0dXJhbERyb3Bkb3duRGF0YX0gZnJvbSAnLi4vLi4vc2VhcmNoL2Ryb3Bkb3duLWNvbnRhaW5lci9kcm9wZG93bi5zZXJ2aWNlJztcbmltcG9ydCB7RHJvcGRvd25Db21wb25lbnR9IGZyb20gJy4uLy4uL3NlYXJjaC90eXBlcy9kcm9wZG93bi1jb21wb25lbnQnO1xuaW1wb3J0IHtDb21tb25Nb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQge01hdElucHV0TW9kdWxlfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9pbnB1dCc7XG5pbXBvcnQge01hdEZvcm1GaWVsZE1vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvZm9ybS1maWVsZCc7XG5cbi8qKlxuICogU2hvdyBhbiBlcnJvciBtZXNzYWdlIGlmIHRoZSBjb250cm9sIGhhcyBhIHZhbHVlIGFuZCBhbiBlcnJvciwgZXZlbiBpZiBjb250cm9sIGlzIG5vdCBkaXJ0eSBhbmQgbm90IHRvdWNoZWQuXG4gKi9cbmV4cG9ydCBjbGFzcyBJbnZhbGlkV2l0aFZhbHVlU3RhdGVNYXRjaGVyIGltcGxlbWVudHMgRXJyb3JTdGF0ZU1hdGNoZXIge1xuICAgIHB1YmxpYyBpc0Vycm9yU3RhdGUoY29udHJvbDogRm9ybUNvbnRyb2wgfCBudWxsKTogYm9vbGVhbiB7XG4gICAgICAgIHJldHVybiBjb250cm9sICYmIGNvbnRyb2wuaW52YWxpZCAmJiBjb250cm9sLnZhbHVlO1xuICAgIH1cbn1cblxuQENvbXBvbmVudCh7XG4gICAgdGVtcGxhdGVVcmw6ICcuL3R5cGUtdGV4dC5jb21wb25lbnQuaHRtbCcsXG4gICAgc3R5bGVVcmxzOiBbJy4vdHlwZS10ZXh0LmNvbXBvbmVudC5zY3NzJ10sXG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBpbXBvcnRzOiBbTWF0Rm9ybUZpZWxkTW9kdWxlLCBNYXRJbnB1dE1vZHVsZSwgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUsIENvbW1vbk1vZHVsZV0sXG59KVxuZXhwb3J0IGNsYXNzIFR5cGVUZXh0Q29tcG9uZW50IGltcGxlbWVudHMgRHJvcGRvd25Db21wb25lbnQge1xuICAgIHB1YmxpYyByZWFkb25seSByZW5kZXJlZFZhbHVlID0gbmV3IEJlaGF2aW9yU3ViamVjdDxzdHJpbmc+KCcnKTtcbiAgICBwdWJsaWMgcmVhZG9ubHkgZm9ybUN0cmwgPSBuZXcgRm9ybUNvbnRyb2woJycsIHtub25OdWxsYWJsZTogdHJ1ZX0pO1xuICAgIHB1YmxpYyByZWFkb25seSBtYXRjaGVyID0gbmV3IEludmFsaWRXaXRoVmFsdWVTdGF0ZU1hdGNoZXIoKTtcblxuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcihcbiAgICAgICAgQEluamVjdChOQVRVUkFMX0RST1BET1dOX0RBVEEpIGRhdGE6IE5hdHVyYWxEcm9wZG93bkRhdGE8bmV2ZXI+LFxuICAgICAgICBwcm90ZWN0ZWQgZHJvcGRvd25SZWY6IE5hdHVyYWxEcm9wZG93blJlZixcbiAgICApIHtcbiAgICAgICAgdGhpcy5mb3JtQ3RybC52YWx1ZUNoYW5nZXMuc3Vic2NyaWJlKHZhbHVlID0+IHtcbiAgICAgICAgICAgIHRoaXMucmVuZGVyZWRWYWx1ZS5uZXh0KHZhbHVlID09PSBudWxsID8gJycgOiB0aGlzLmZvcm1DdHJsLnZhbHVlICsgJycpO1xuICAgICAgICB9KTtcblxuICAgICAgICB0aGlzLmZvcm1DdHJsLnNldFZhbGlkYXRvcnMoW1ZhbGlkYXRvcnMucmVxdWlyZWRdKTtcblxuICAgICAgICBpZiAoZGF0YS5jb25kaXRpb24gJiYgZGF0YS5jb25kaXRpb24ubGlrZSkge1xuICAgICAgICAgICAgdGhpcy5mb3JtQ3RybC5zZXRWYWx1ZSgnJyArIGRhdGEuY29uZGl0aW9uLmxpa2UudmFsdWUpO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgcHVibGljIGdldENvbmRpdGlvbigpOiBGaWx0ZXJHcm91cENvbmRpdGlvbkZpZWxkIHtcbiAgICAgICAgaWYgKCF0aGlzLmZvcm1DdHJsLnZhbHVlKSB7XG4gICAgICAgICAgICByZXR1cm4ge307XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4ge2xpa2U6IHt2YWx1ZTogdGhpcy5mb3JtQ3RybC52YWx1ZX19O1xuICAgIH1cblxuICAgIHB1YmxpYyBpc1ZhbGlkKCk6IGJvb2xlYW4ge1xuICAgICAgICByZXR1cm4gdGhpcy5mb3JtQ3RybC52YWxpZDtcbiAgICB9XG5cbiAgICBwdWJsaWMgaXNEaXJ0eSgpOiBib29sZWFuIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZm9ybUN0cmwuZGlydHk7XG4gICAgfVxuXG4gICAgcHVibGljIGNsb3NlKCk6IHZvaWQge1xuICAgICAgICBpZiAodGhpcy5pc1ZhbGlkKCkpIHtcbiAgICAgICAgICAgIHRoaXMuZHJvcGRvd25SZWYuY2xvc2Uoe2NvbmRpdGlvbjogdGhpcy5nZXRDb25kaXRpb24oKX0pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgdGhpcy5kcm9wZG93blJlZi5jbG9zZSgpOyAvLyB1bmRlZmluZWQgdmFsdWUsIGRpc2NhcmQgY2hhbmdlcyAvIHByZXZlbnQgdG8gYWRkIGEgY29uZGl0aW9uIChvbiBuZXcgZmllbGRzXG4gICAgICAgIH1cbiAgICB9XG59XG4iLCI8bWF0LWZvcm0tZmllbGQ+XG4gICAgPG1hdC1sYWJlbCBpMThuPlZhbGV1cjwvbWF0LWxhYmVsPlxuICAgIDxpbnB1dFxuICAgICAgICAoa2V5ZG93bi5lbnRlcik9XCJjbG9zZSgpXCJcbiAgICAgICAgW2Vycm9yU3RhdGVNYXRjaGVyXT1cIm1hdGNoZXJcIlxuICAgICAgICBbZm9ybUNvbnRyb2xdPVwiZm9ybUN0cmxcIlxuICAgICAgICBbcmVxdWlyZWRdPVwidHJ1ZVwiXG4gICAgICAgIG1hdElucHV0XG4gICAgICAgIHR5cGU9XCJ0ZXh0XCJcbiAgICAvPlxuICAgIDxtYXQtZXJyb3IgKm5nSWY9XCJmb3JtQ3RybC5oYXNFcnJvcigncmVxdWlyZWQnKVwiPio8L21hdC1lcnJvcj5cbjwvbWF0LWZvcm0tZmllbGQ+XG4iXX0=
|
|
@@ -38,4 +38,4 @@ export const possibleDiscreteOperators = [
|
|
|
38
38
|
label: $localize `sans`,
|
|
39
39
|
},
|
|
40
40
|
];
|
|
41
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9kcm9wZG93bi1jb21wb25lbnRzL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVlBLE1BQU0sQ0FBQyxNQUFNLDJCQUEyQixHQUEyQztJQUMvRTtRQUNJLEdBQUcsRUFBRSxNQUFNO1FBQ1gsS0FBSyxFQUFFLEdBQUc7S0FDYjtJQUNEO1FBQ0ksR0FBRyxFQUFFLGFBQWE7UUFDbEIsS0FBSyxFQUFFLEdBQUc7S0FDYjtJQUNEO1FBQ0ksR0FBRyxFQUFFLE9BQU87UUFDWixLQUFLLEVBQUUsR0FBRztLQUNiO0lBQ0Q7UUFDSSxHQUFHLEVBQUUsZ0JBQWdCO1FBQ3JCLEtBQUssRUFBRSxHQUFHO0tBQ2I7SUFDRDtRQUNJLEdBQUcsRUFBRSxTQUFTO1FBQ2QsS0FBSyxFQUFFLEdBQUc7S0FDYjtDQUNLLENBQUM7QUFTWCxNQUFNLENBQUMsTUFBTSx5QkFBeUIsR0FBeUM7SUFDM0U7UUFDSSxHQUFHLEVBQUUsSUFBSTtRQUNULEtBQUssRUFBRSxTQUFTLENBQUEsS0FBSztLQUN4QjtJQUNEO1FBQ0ksR0FBRyxFQUFFLE9BQU87UUFDWixLQUFLLEVBQUUsU0FBUyxDQUFBLFdBQVc7S0FDOUI7SUFDRDtRQUNJLEdBQUcsRUFBRSxLQUFLO1FBQ1YsS0FBSyxFQUFFLFNBQVMsQ0FBQSxNQUFNO0tBQ3pCO0lBQ0Q7UUFDSSxHQUFHLEVBQUUsTUFBTTtRQUNYLEtBQUssRUFBRSxTQUFTLENBQUEsTUFBTTtLQUN6QjtDQUNLLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0ZpbHRlckdyb3VwQ29uZGl0aW9uRmllbGR9IGZyb20gJy4uL3NlYXJjaC9jbGFzc2VzL2dyYXBocWwtZG9jdHJpbmUudHlwZXMnO1xuXG5leHBvcnQgdHlwZSBQb3NzaWJsZUNvbXBhcmFibGVPcGVydG9yS2V5cyA9IGtleW9mIFBpY2s8XG4gICAgRmlsdGVyR3JvdXBDb25kaXRpb25GaWVsZCxcbiAgICAnbGVzcycgfCAnbGVzc09yRXF1YWwnIHwgJ2VxdWFsJyB8ICdncmVhdGVyT3JFcXVhbCcgfCAnZ3JlYXRlcidcbj47XG5cbmV4cG9ydCBpbnRlcmZhY2UgUG9zc2libGVDb21wYXJhYmxlT3BlcmF0b3Ige1xuICAgIGtleTogUG9zc2libGVDb21wYXJhYmxlT3BlcnRvcktleXM7XG4gICAgbGFiZWw6IHN0cmluZztcbn1cblxuZXhwb3J0IGNvbnN0IHBvc3NpYmxlQ29tcGFyYWJsZU9wZXJhdG9yczogUmVhZG9ubHk8UG9zc2libGVDb21wYXJhYmxlT3BlcmF0b3JbXT4gPSBbXG4gICAge1xuICAgICAgICBrZXk6ICdsZXNzJyxcbiAgICAgICAgbGFiZWw6ICc8JyxcbiAgICB9LFxuICAgIHtcbiAgICAgICAga2V5OiAnbGVzc09yRXF1YWwnLFxuICAgICAgICBsYWJlbDogJ+KJpCcsXG4gICAgfSxcbiAgICB7XG4gICAgICAgIGtleTogJ2VxdWFsJyxcbiAgICAgICAgbGFiZWw6ICc9JyxcbiAgICB9LFxuICAgIHtcbiAgICAgICAga2V5OiAnZ3JlYXRlck9yRXF1YWwnLFxuICAgICAgICBsYWJlbDogJ+KJpScsXG4gICAgfSxcbiAgICB7XG4gICAgICAgIGtleTogJ2dyZWF0ZXInLFxuICAgICAgICBsYWJlbDogJz4nLFxuICAgIH0sXG5dIGFzIGNvbnN0O1xuXG5leHBvcnQgdHlwZSBQb3NzaWJsZURpc2NyZXRlT3BlcmF0b3JLZXlzID0gJ2lzJyB8ICdpc25vdCcgfCAnYW55JyB8ICdub25lJztcblxuZXhwb3J0IGludGVyZmFjZSBQb3NzaWJsZURpc2NyZXRlT3BlcmF0b3Ige1xuICAgIGtleTogUG9zc2libGVEaXNjcmV0ZU9wZXJhdG9yS2V5cztcbiAgICBsYWJlbDogc3RyaW5nO1xufVxuXG5leHBvcnQgY29uc3QgcG9zc2libGVEaXNjcmV0ZU9wZXJhdG9yczogUmVhZG9ubHk8UG9zc2libGVEaXNjcmV0ZU9wZXJhdG9yW10+ID0gW1xuICAgIHtcbiAgICAgICAga2V5OiAnaXMnLFxuICAgICAgICBsYWJlbDogJGxvY2FsaXplYGVzdGAsXG4gICAgfSxcbiAgICB7XG4gICAgICAgIGtleTogJ2lzbm90JyxcbiAgICAgICAgbGFiZWw6ICRsb2NhbGl6ZWBuJ2VzdCBwYXNgLFxuICAgIH0sXG4gICAge1xuICAgICAgICBrZXk6ICdhbnknLFxuICAgICAgICBsYWJlbDogJGxvY2FsaXplYGF2ZWNgLFxuICAgIH0sXG4gICAge1xuICAgICAgICBrZXk6ICdub25lJyxcbiAgICAgICAgbGFiZWw6ICRsb2NhbGl6ZWBzYW5zYCxcbiAgICB9LFxuXSBhcyBjb25zdDtcbiJdfQ==
|
|
@@ -38,6 +38,10 @@ export class AbstractSelect extends NaturalAbstractController {
|
|
|
38
38
|
constructor(ngControl) {
|
|
39
39
|
super();
|
|
40
40
|
this.ngControl = ngControl;
|
|
41
|
+
/**
|
|
42
|
+
* Mat-hint
|
|
43
|
+
*/
|
|
44
|
+
this.hint = null;
|
|
41
45
|
/**
|
|
42
46
|
* Whether to show the search icon
|
|
43
47
|
*/
|
|
@@ -165,7 +169,7 @@ export class AbstractSelect extends NaturalAbstractController {
|
|
|
165
169
|
this.internalCtrl.updateValueAndValidity();
|
|
166
170
|
}
|
|
167
171
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: AbstractSelect, deps: [{ token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
168
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.2", type: AbstractSelect, inputs: { placeholder: "placeholder", required: "required", navigateTo: "navigateTo", clearLabel: "clearLabel", showIcon: "showIcon", icon: "icon", displayWith: "displayWith", disabled: "disabled" }, outputs: { selectionChange: "selectionChange", blur: "blur" }, usesInheritance: true, ngImport: i0 }); }
|
|
172
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.2", type: AbstractSelect, inputs: { placeholder: "placeholder", hint: "hint", required: "required", navigateTo: "navigateTo", clearLabel: "clearLabel", showIcon: "showIcon", icon: "icon", displayWith: "displayWith", disabled: "disabled" }, outputs: { selectionChange: "selectionChange", blur: "blur" }, usesInheritance: true, ngImport: i0 }); }
|
|
169
173
|
}
|
|
170
174
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: AbstractSelect, decorators: [{
|
|
171
175
|
type: Directive
|
|
@@ -175,6 +179,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
|
|
|
175
179
|
type: Self
|
|
176
180
|
}] }]; }, propDecorators: { placeholder: [{
|
|
177
181
|
type: Input
|
|
182
|
+
}], hint: [{
|
|
183
|
+
type: Input
|
|
178
184
|
}], required: [{
|
|
179
185
|
type: Input
|
|
180
186
|
}], navigateTo: [{
|
|
@@ -194,4 +200,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
|
|
|
194
200
|
}], disabled: [{
|
|
195
201
|
type: Input
|
|
196
202
|
}] } });
|
|
197
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-select.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/select/abstract-select.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {Directive, DoCheck, EventEmitter, Input, OnDestroy, OnInit, Optional, Output, Self} from '@angular/core';\nimport {\n    AbstractControl,\n    ControlValueAccessor,\n    FormControl,\n    FormControlDirective,\n    FormControlName,\n    NgControl,\n    Validators,\n} from '@angular/forms';\nimport {ErrorStateMatcher} from '@angular/material/core';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\n\n/**\n * This will completely ignore internal formControl and instead use the one from the component\n * which comes from outside of this component. This basically allows us to **not** depend on\n * touched status propagation between outside and inside world, and thus get rid of our legacy\n * custom FormControl class (\"NaturalFormControl\").\n */\nclass ExternalFormControlMatcher<T, I> extends ErrorStateMatcher {\n    public constructor(private readonly component: AbstractSelect<T, I>) {\n        super();\n    }\n\n    public override isErrorState(): boolean {\n        const externalCtrl = this.component.ngControl?.control || this.component.internalCtrl;\n        if (externalCtrl) {\n            return !!(externalCtrl.errors && (externalCtrl.touched || externalCtrl.dirty));\n        }\n\n        return false;\n    }\n}\n\n@Directive()\nexport abstract class AbstractSelect<V, I>\n    extends NaturalAbstractController\n    implements OnInit, OnDestroy, ControlValueAccessor, DoCheck\n{\n    @Input() public placeholder?: string;\n\n    /**\n     * If the field is required\n     */\n    @Input()\n    public set required(value: boolean) {\n        this._required = coerceBooleanProperty(value);\n        this.applyRequired();\n    }\n\n    public get required(): boolean {\n        return !!this._required;\n    }\n\n    private _required: boolean | undefined;\n\n    /**\n     * Add a suffix button that is a link to given destination\n     */\n    @Input() public navigateTo?: any[] | string | null;\n\n    /**\n     * If provided cause a new clear button to appear\n     */\n    @Input() public clearLabel?: string;\n\n    /**\n     * Whether to show the search icon\n     */\n    @Input() public showIcon = true;\n\n    /**\n     * Icon name\n     */\n    @Input() public icon = 'search';\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: V | null) => string;\n\n    /**\n     * Emit the selected value whenever it changes\n     */\n    @Output() public readonly selectionChange = new EventEmitter<V | null>();\n\n    /**\n     * Emits when internal input is blurred\n     */\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() public readonly blur = new EventEmitter<void>();\n\n    /**\n     * Contains internal representation for current selection AND searched text (for autocomplete)\n     *\n     * It is **not** necessarily `V | null`.\n     *\n     * - NaturalSelectComponent: `string | V | null`. We allow `string`\n     *   only when `optionRequired` is false, so most of the time it is `V | null`.\n     * - NaturalSelectHierarchicComponent: `string | null`.\n     * - NaturalSelectEnumComponent: `V | null`.\n     *\n     * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :\n     *  - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed\n     *  - Dirty status (changed value) means we are in search/autocomplete mode\n     */\n    public readonly internalCtrl = new FormControl<I | null>(null);\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onChange?: (item: V | null) => void;\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onTouched?: () => void;\n\n    public readonly matcher: ExternalFormControlMatcher<V, I>;\n\n    public constructor(@Optional() @Self() public readonly ngControl: NgControl | null) {\n        super();\n\n        if (this.ngControl) {\n            this.ngControl.valueAccessor = this;\n        }\n\n        this.matcher = new ExternalFormControlMatcher(this);\n    }\n\n    public ngDoCheck(): void {\n        if (this.ngControl) {\n            this.applyRequired();\n        }\n    }\n\n    public writeValue(value: I | null): void {\n        this.internalCtrl.setValue(value);\n    }\n\n    public ngOnInit(): void {\n        const isReactive = this.ngControl instanceof FormControlDirective || this.ngControl instanceof FormControlName;\n        if (isReactive && typeof this._required !== 'undefined') {\n            console.warn('<natural-select-*> should not be used as ReactiveForm and with the [required] attribute');\n        }\n    }\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public registerOnChange(fn: (item: V | null) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public abstract getDisplayFn(): (item: V | null) => string;\n\n    /**\n     * Commit the model to null\n     * Emit and event to update the model\n     */\n    public clear(): void {\n        this.internalCtrl.setValue(null);\n        this.propagateValue(null);\n    }\n\n    /**\n     * If input is dirty (search running) restore to model value\n     */\n    public onBlur(): void {\n        this.touch();\n        this.blur.emit();\n    }\n\n    /**\n     * Commit the model change\n     */\n    public propagateValue(value: V | null): void {\n        // before selectionChange to allow formControl to update before change is effectively emitted\n        if (this.onChange) {\n            this.onChange(value);\n        }\n\n        this.selectionChange.emit(value);\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    public touch(): void {\n        if (this.onTouched) {\n            this.onTouched();\n        }\n    }\n\n    public hasRequiredError(): boolean {\n        const control = this.ngControl?.control ? this.ngControl?.control : this.internalCtrl;\n\n        return control.hasError('required');\n    }\n\n    /**\n     * Apply Validators.required on the internal form, based on ngControl or [required] attribute, giving priority to attribute.\n     */\n    private applyRequired(): void {\n        // Required status on parent validator\n        const outerRequiredStatus = this?.ngControl?.control?.validator?.({} as AbstractControl)?.required;\n\n        // Wanted required status, giving priority to template\n        const newRequiredStatus = typeof this._required !== 'undefined' ? this._required : outerRequiredStatus;\n\n        // Actual internal validation status\n        const currentRequiredStatus = this.internalCtrl?.validator?.({} as AbstractControl)?.required;\n\n        // If wanted status is similar to actual status, stop everything\n        if (currentRequiredStatus === newRequiredStatus) {\n            return;\n        }\n\n        // Apply only if changed\n        if (newRequiredStatus) {\n            this.internalCtrl.setValidators(Validators.required);\n        } else {\n            this.internalCtrl.clearValidators();\n        }\n\n        this.internalCtrl.updateValueAndValidity();\n    }\n}\n"]}
|
|
203
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-select.component.js","sourceRoot":"","sources":["../../../../../../projects/natural/src/lib/modules/select/abstract-select.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {Directive, DoCheck, EventEmitter, Input, OnDestroy, OnInit, Optional, Output, Self} from '@angular/core';\nimport {\n    AbstractControl,\n    ControlValueAccessor,\n    FormControl,\n    FormControlDirective,\n    FormControlName,\n    NgControl,\n    Validators,\n} from '@angular/forms';\nimport {ErrorStateMatcher} from '@angular/material/core';\nimport {NaturalAbstractController} from '../../classes/abstract-controller';\n\n/**\n * This will completely ignore internal formControl and instead use the one from the component\n * which comes from outside of this component. This basically allows us to **not** depend on\n * touched status propagation between outside and inside world, and thus get rid of our legacy\n * custom FormControl class (\"NaturalFormControl\").\n */\nclass ExternalFormControlMatcher<T, I> extends ErrorStateMatcher {\n    public constructor(private readonly component: AbstractSelect<T, I>) {\n        super();\n    }\n\n    public override isErrorState(): boolean {\n        const externalCtrl = this.component.ngControl?.control || this.component.internalCtrl;\n        if (externalCtrl) {\n            return !!(externalCtrl.errors && (externalCtrl.touched || externalCtrl.dirty));\n        }\n\n        return false;\n    }\n}\n\n@Directive()\nexport abstract class AbstractSelect<V, I>\n    extends NaturalAbstractController\n    implements OnInit, OnDestroy, ControlValueAccessor, DoCheck\n{\n    @Input() public placeholder?: string;\n\n    /**\n     * Mat-hint\n     */\n    @Input() public hint: string | null = null;\n\n    /**\n     * If the field is required\n     */\n    @Input()\n    public set required(value: boolean) {\n        this._required = coerceBooleanProperty(value);\n        this.applyRequired();\n    }\n\n    public get required(): boolean {\n        return !!this._required;\n    }\n\n    private _required: boolean | undefined;\n\n    /**\n     * Add a suffix button that is a link to given destination\n     */\n    @Input() public navigateTo?: any[] | string | null;\n\n    /**\n     * If provided cause a new clear button to appear\n     */\n    @Input() public clearLabel?: string;\n\n    /**\n     * Whether to show the search icon\n     */\n    @Input() public showIcon = true;\n\n    /**\n     * Icon name\n     */\n    @Input() public icon = 'search';\n\n    /**\n     * Function to customize the rendering of the selected item as text in input\n     */\n    @Input() public displayWith?: (item: V | null) => string;\n\n    /**\n     * Emit the selected value whenever it changes\n     */\n    @Output() public readonly selectionChange = new EventEmitter<V | null>();\n\n    /**\n     * Emits when internal input is blurred\n     */\n    // eslint-disable-next-line @angular-eslint/no-output-native\n    @Output() public readonly blur = new EventEmitter<void>();\n\n    /**\n     * Contains internal representation for current selection AND searched text (for autocomplete)\n     *\n     * It is **not** necessarily `V | null`.\n     *\n     * - NaturalSelectComponent: `string | V | null`. We allow `string`\n     *   only when `optionRequired` is false, so most of the time it is `V | null`.\n     * - NaturalSelectHierarchicComponent: `string | null`.\n     * - NaturalSelectEnumComponent: `V | null`.\n     *\n     * In natural-select context, we use pristine and dirty to identify if the displayed value is search or committed model :\n     *  - Pristine status (unchanged value) means the model is displayed and propagated = the selection is committed\n     *  - Dirty status (changed value) means we are in search/autocomplete mode\n     */\n    public readonly internalCtrl = new FormControl<I | null>(null);\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onChange?: (item: V | null) => void;\n\n    /**\n     * Interface with ControlValueAccessor\n     * Notifies parent model / form controller\n     */\n    public onTouched?: () => void;\n\n    public readonly matcher: ExternalFormControlMatcher<V, I>;\n\n    public constructor(@Optional() @Self() public readonly ngControl: NgControl | null) {\n        super();\n\n        if (this.ngControl) {\n            this.ngControl.valueAccessor = this;\n        }\n\n        this.matcher = new ExternalFormControlMatcher(this);\n    }\n\n    public ngDoCheck(): void {\n        if (this.ngControl) {\n            this.applyRequired();\n        }\n    }\n\n    public writeValue(value: I | null): void {\n        this.internalCtrl.setValue(value);\n    }\n\n    public ngOnInit(): void {\n        const isReactive = this.ngControl instanceof FormControlDirective || this.ngControl instanceof FormControlName;\n        if (isReactive && typeof this._required !== 'undefined') {\n            console.warn('<natural-select-*> should not be used as ReactiveForm and with the [required] attribute');\n        }\n    }\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public registerOnChange(fn: (item: V | null) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public abstract getDisplayFn(): (item: V | null) => string;\n\n    /**\n     * Commit the model to null\n     * Emit and event to update the model\n     */\n    public clear(): void {\n        this.internalCtrl.setValue(null);\n        this.propagateValue(null);\n    }\n\n    /**\n     * If input is dirty (search running) restore to model value\n     */\n    public onBlur(): void {\n        this.touch();\n        this.blur.emit();\n    }\n\n    /**\n     * Commit the model change\n     */\n    public propagateValue(value: V | null): void {\n        // before selectionChange to allow formControl to update before change is effectively emitted\n        if (this.onChange) {\n            this.onChange(value);\n        }\n\n        this.selectionChange.emit(value);\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n    }\n\n    public showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    public touch(): void {\n        if (this.onTouched) {\n            this.onTouched();\n        }\n    }\n\n    public hasRequiredError(): boolean {\n        const control = this.ngControl?.control ? this.ngControl?.control : this.internalCtrl;\n\n        return control.hasError('required');\n    }\n\n    /**\n     * Apply Validators.required on the internal form, based on ngControl or [required] attribute, giving priority to attribute.\n     */\n    private applyRequired(): void {\n        // Required status on parent validator\n        const outerRequiredStatus = this?.ngControl?.control?.validator?.({} as AbstractControl)?.required;\n\n        // Wanted required status, giving priority to template\n        const newRequiredStatus = typeof this._required !== 'undefined' ? this._required : outerRequiredStatus;\n\n        // Actual internal validation status\n        const currentRequiredStatus = this.internalCtrl?.validator?.({} as AbstractControl)?.required;\n\n        // If wanted status is similar to actual status, stop everything\n        if (currentRequiredStatus === newRequiredStatus) {\n            return;\n        }\n\n        // Apply only if changed\n        if (newRequiredStatus) {\n            this.internalCtrl.setValidators(Validators.required);\n        } else {\n            this.internalCtrl.clearValidators();\n        }\n\n        this.internalCtrl.updateValueAndValidity();\n    }\n}\n"]}
|
|
@@ -260,7 +260,7 @@ export class NaturalSelectComponent extends AbstractSelect {
|
|
|
260
260
|
return this.variablesManager.variables.value;
|
|
261
261
|
}
|
|
262
262
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: NaturalSelectComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
263
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: NaturalSelectComponent, isStandalone: true, selector: "natural-select", inputs: { service: "service", optionRequired: "optionRequired", searchField: "searchField", searchOperator: "searchOperator", filter: "filter", disabled: "disabled" }, queries: [{ propertyName: "itemTemplate", first: true, predicate: TemplateRef, descendants: true }], viewQueries: [{ propertyName: "autoTrigger", first: true, predicate: MatAutocompleteTrigger, descendants: true }], usesInheritance: true, ngImport: i0, template: "<!-- Autocomplete menu -->\n<mat-autocomplete\n #ac=\"matAutocomplete\"\n (optionSelected)=\"propagateValue($event.option.value)\"\n [displayWith]=\"getDisplayFn()\"\n panelWidth=\"auto !important\"\n>\n <mat-option *ngFor=\"let item of items | async\" [value]=\"item\">\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n ></ng-template>\n </mat-option>\n <div *ngIf=\"moreNbItems > 0\" class=\"mat-caption\" i18n style=\"padding: 5px 10px\"\n >{{ moreNbItems }} \u00E9l\u00E9ment(s) suppl\u00E9mentaire(s)</div\n >\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n <mat-label>{{ placeholder }}</mat-label>\n\n <input\n (blur)=\"onBlur()\"\n (change)=\"onInternalFormChange()\"\n (click)=\"autoTrigger.openPanel()\"\n (focus)=\"startSearch()\"\n (keydown.esc)=\"reset()\"\n (keydown.enter)=\"onKeyEnter()\"\n [formControl]=\"internalCtrl\"\n [matAutocomplete]=\"ac\"\n aria-label=\"Recherche et s\u00E9lection\"\n i18n-aria-label\n matInput\n [errorStateMatcher]=\"matcher\"\n />\n\n <!-- Meta data -->\n <mat-icon *ngIf=\"!loading && showIcon\" [naturalIcon]=\"icon\" matIconPrefix></mat-icon>\n\n <div class=\"loading-wrapper\" matIconPrefix *ngIf=\"loading\">\n <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\"></mat-progress-spinner>\n </div>\n\n <!-- Clear button -->\n <div class=\"suffix-buttons\" matIconSuffix>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel\"\n (click)=\"clear()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"D\u00E9s\u00E9lectionner\"\n >\n <mat-icon naturalIcon=\"close\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.dirty && internalCtrl.enabled && optionRequired\"\n (click)=\"reset()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Annuler la recherche\"\n >\n <mat-icon naturalIcon=\"undo\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && navigateTo\"\n [routerLink]=\"navigateTo\"\n (click)=\"$event.stopPropagation()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Naviguer vers\"\n >\n <mat-icon naturalIcon=\"open_in_browser\"></mat-icon>\n </button>\n </div>\n\n <mat-error *ngIf=\"hasRequiredError()\" i18n>Ce champ est requis</mat-error>\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n<div *ngIf=\"showClearButton()\" class=\"external-buttons\">\n <button (click)=\"clear()\" *ngIf=\"showClearButton()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .suffix-buttons,:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"], dependencies: [{ kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i1.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple", "hideSingleSelectionIndicator"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i2.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i1.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i4.MatPrefix, selector: "[matPrefix], [matIconPrefix], [matTextPrefix]", inputs: ["matTextPrefix"] }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i5.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i6.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i8.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i9.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i9.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i10.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }] }); }
|
|
263
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: NaturalSelectComponent, isStandalone: true, selector: "natural-select", inputs: { service: "service", optionRequired: "optionRequired", searchField: "searchField", searchOperator: "searchOperator", filter: "filter", disabled: "disabled" }, queries: [{ propertyName: "itemTemplate", first: true, predicate: TemplateRef, descendants: true }], viewQueries: [{ propertyName: "autoTrigger", first: true, predicate: MatAutocompleteTrigger, descendants: true }], usesInheritance: true, ngImport: i0, template: "<!-- Autocomplete menu -->\n<mat-autocomplete\n #ac=\"matAutocomplete\"\n (optionSelected)=\"propagateValue($event.option.value)\"\n [displayWith]=\"getDisplayFn()\"\n panelWidth=\"auto !important\"\n>\n <mat-option *ngFor=\"let item of items | async\" [value]=\"item\">\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n ></ng-template>\n </mat-option>\n <div *ngIf=\"moreNbItems > 0\" class=\"mat-caption\" i18n style=\"padding: 5px 10px\"\n >{{ moreNbItems }} \u00E9l\u00E9ment(s) suppl\u00E9mentaire(s)</div\n >\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n <mat-label>{{ placeholder }}</mat-label>\n\n <input\n (blur)=\"onBlur()\"\n (change)=\"onInternalFormChange()\"\n (click)=\"autoTrigger.openPanel()\"\n (focus)=\"startSearch()\"\n (keydown.esc)=\"reset()\"\n (keydown.enter)=\"onKeyEnter()\"\n [formControl]=\"internalCtrl\"\n [matAutocomplete]=\"ac\"\n aria-label=\"Recherche et s\u00E9lection\"\n i18n-aria-label\n matInput\n [errorStateMatcher]=\"matcher\"\n />\n\n <mat-hint *ngIf=\"hint\">{{ hint }}</mat-hint>\n\n <!-- Meta data -->\n <mat-icon *ngIf=\"!loading && showIcon\" [naturalIcon]=\"icon\" matIconPrefix></mat-icon>\n\n <div class=\"loading-wrapper\" matIconPrefix *ngIf=\"loading\">\n <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\"></mat-progress-spinner>\n </div>\n\n <!-- Clear button -->\n <div class=\"suffix-buttons\" matIconSuffix>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel\"\n (click)=\"clear()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"D\u00E9s\u00E9lectionner\"\n >\n <mat-icon naturalIcon=\"close\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.dirty && internalCtrl.enabled && optionRequired\"\n (click)=\"reset()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Annuler la recherche\"\n >\n <mat-icon naturalIcon=\"undo\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && navigateTo\"\n [routerLink]=\"navigateTo\"\n (click)=\"$event.stopPropagation()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Naviguer vers\"\n >\n <mat-icon naturalIcon=\"open_in_browser\"></mat-icon>\n </button>\n </div>\n\n <mat-error *ngIf=\"hasRequiredError()\" i18n>Ce champ est requis</mat-error>\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n<div *ngIf=\"showClearButton()\" class=\"external-buttons\">\n <button (click)=\"clear()\" *ngIf=\"showClearButton()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .suffix-buttons,:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"], dependencies: [{ kind: "ngmodule", type: MatAutocompleteModule }, { kind: "component", type: i1.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple", "hideSingleSelectionIndicator"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i2.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i1.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "directive", type: i4.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "directive", type: i4.MatPrefix, selector: "[matPrefix], [matIconPrefix], [matTextPrefix]", inputs: ["matTextPrefix"] }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i5.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i6.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: NaturalIconDirective, selector: "mat-icon[naturalIcon]", inputs: ["naturalIcon", "size"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i8.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i9.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i9.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i10.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }] }); }
|
|
264
264
|
}
|
|
265
265
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: NaturalSelectComponent, decorators: [{
|
|
266
266
|
type: Component,
|
|
@@ -278,7 +278,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
|
|
|
278
278
|
MatButtonModule,
|
|
279
279
|
MatTooltipModule,
|
|
280
280
|
RouterLink,
|
|
281
|
-
], template: "<!-- Autocomplete menu -->\n<mat-autocomplete\n #ac=\"matAutocomplete\"\n (optionSelected)=\"propagateValue($event.option.value)\"\n [displayWith]=\"getDisplayFn()\"\n panelWidth=\"auto !important\"\n>\n <mat-option *ngFor=\"let item of items | async\" [value]=\"item\">\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n ></ng-template>\n </mat-option>\n <div *ngIf=\"moreNbItems > 0\" class=\"mat-caption\" i18n style=\"padding: 5px 10px\"\n >{{ moreNbItems }} \u00E9l\u00E9ment(s) suppl\u00E9mentaire(s)</div\n >\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n <mat-label>{{ placeholder }}</mat-label>\n\n <input\n (blur)=\"onBlur()\"\n (change)=\"onInternalFormChange()\"\n (click)=\"autoTrigger.openPanel()\"\n (focus)=\"startSearch()\"\n (keydown.esc)=\"reset()\"\n (keydown.enter)=\"onKeyEnter()\"\n [formControl]=\"internalCtrl\"\n [matAutocomplete]=\"ac\"\n aria-label=\"Recherche et s\u00E9lection\"\n i18n-aria-label\n matInput\n [errorStateMatcher]=\"matcher\"\n />\n\n <!-- Meta data -->\n <mat-icon *ngIf=\"!loading && showIcon\" [naturalIcon]=\"icon\" matIconPrefix></mat-icon>\n\n <div class=\"loading-wrapper\" matIconPrefix *ngIf=\"loading\">\n <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\"></mat-progress-spinner>\n </div>\n\n <!-- Clear button -->\n <div class=\"suffix-buttons\" matIconSuffix>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel\"\n (click)=\"clear()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"D\u00E9s\u00E9lectionner\"\n >\n <mat-icon naturalIcon=\"close\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.dirty && internalCtrl.enabled && optionRequired\"\n (click)=\"reset()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Annuler la recherche\"\n >\n <mat-icon naturalIcon=\"undo\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && navigateTo\"\n [routerLink]=\"navigateTo\"\n (click)=\"$event.stopPropagation()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Naviguer vers\"\n >\n <mat-icon naturalIcon=\"open_in_browser\"></mat-icon>\n </button>\n </div>\n\n <mat-error *ngIf=\"hasRequiredError()\" i18n>Ce champ est requis</mat-error>\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n<div *ngIf=\"showClearButton()\" class=\"external-buttons\">\n <button (click)=\"clear()\" *ngIf=\"showClearButton()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .suffix-buttons,:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"] }]
|
|
281
|
+
], template: "<!-- Autocomplete menu -->\n<mat-autocomplete\n #ac=\"matAutocomplete\"\n (optionSelected)=\"propagateValue($event.option.value)\"\n [displayWith]=\"getDisplayFn()\"\n panelWidth=\"auto !important\"\n>\n <mat-option *ngFor=\"let item of items | async\" [value]=\"item\">\n <ng-template\n [ngTemplateOutletContext]=\"{item: item}\"\n [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n ></ng-template>\n </mat-option>\n <div *ngIf=\"moreNbItems > 0\" class=\"mat-caption\" i18n style=\"padding: 5px 10px\"\n >{{ moreNbItems }} \u00E9l\u00E9ment(s) suppl\u00E9mentaire(s)</div\n >\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n <mat-label>{{ placeholder }}</mat-label>\n\n <input\n (blur)=\"onBlur()\"\n (change)=\"onInternalFormChange()\"\n (click)=\"autoTrigger.openPanel()\"\n (focus)=\"startSearch()\"\n (keydown.esc)=\"reset()\"\n (keydown.enter)=\"onKeyEnter()\"\n [formControl]=\"internalCtrl\"\n [matAutocomplete]=\"ac\"\n aria-label=\"Recherche et s\u00E9lection\"\n i18n-aria-label\n matInput\n [errorStateMatcher]=\"matcher\"\n />\n\n <mat-hint *ngIf=\"hint\">{{ hint }}</mat-hint>\n\n <!-- Meta data -->\n <mat-icon *ngIf=\"!loading && showIcon\" [naturalIcon]=\"icon\" matIconPrefix></mat-icon>\n\n <div class=\"loading-wrapper\" matIconPrefix *ngIf=\"loading\">\n <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\"></mat-progress-spinner>\n </div>\n\n <!-- Clear button -->\n <div class=\"suffix-buttons\" matIconSuffix>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel\"\n (click)=\"clear()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"D\u00E9s\u00E9lectionner\"\n >\n <mat-icon naturalIcon=\"close\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.dirty && internalCtrl.enabled && optionRequired\"\n (click)=\"reset()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Annuler la recherche\"\n >\n <mat-icon naturalIcon=\"undo\"></mat-icon>\n </button>\n <button\n *ngIf=\"internalCtrl.pristine && internalCtrl.value && navigateTo\"\n [routerLink]=\"navigateTo\"\n (click)=\"$event.stopPropagation()\"\n mat-icon-button\n i18n-matTooltip\n matTooltip=\"Naviguer vers\"\n >\n <mat-icon naturalIcon=\"open_in_browser\"></mat-icon>\n </button>\n </div>\n\n <mat-error *ngIf=\"hasRequiredError()\" i18n>Ce champ est requis</mat-error>\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n<div *ngIf=\"showClearButton()\" class=\"external-buttons\">\n <button (click)=\"clear()\" *ngIf=\"showClearButton()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n</div>\n", styles: [":host{display:flex;flex-direction:column}:host>*:not(:last-child){margin-bottom:20px}:host>mat-autocomplete{margin-bottom:0!important}:host .suffix-buttons,:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons{display:flex;flex-direction:row}:host .external-buttons>*:not(:last-child){margin-right:10px}:host .loading-wrapper{display:flex;justify-content:center;align-items:center;width:48px;height:48px}\n"] }]
|
|
282
282
|
}], propDecorators: { autoTrigger: [{
|
|
283
283
|
type: ViewChild,
|
|
284
284
|
args: [MatAutocompleteTrigger]
|
|
@@ -299,4 +299,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
|
|
|
299
299
|
}], disabled: [{
|
|
300
300
|
type: Input
|
|
301
301
|
}] } });
|
|
302
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/select/select/select.component.ts","../../../../../../../projects/natural/src/lib/modules/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Component, ContentChild, Input, OnDestroy, OnInit, TemplateRef, ViewChild} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatAutocompleteTrigger, MatAutocompleteModule} from '@angular/material/autocomplete';\nimport {merge} from 'lodash-es';\nimport {Observable} from 'rxjs';\nimport {debounceTime, distinctUntilChanged, finalize, map, takeUntil} from 'rxjs/operators';\nimport {PaginatedData} from '../../../classes/data-source';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {NaturalAbstractModelService} from '../../../services/abstract-model.service';\nimport {ExtractTallOne, ExtractVall, Literal} from '../../../types/types';\nimport {AbstractSelect} from '../abstract-select.component';\nimport {RouterLink} from '@angular/router';\nimport {MatTooltipModule} from '@angular/material/tooltip';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatOptionModule} from '@angular/material/core';\nimport {CommonModule} from '@angular/common';\n\ntype V<TService> = string | ExtractTallOne<TService>;\n\n/**\n * Default usage:\n * ```html\n * <natural-select [service]=\"myServiceInstance\" [(model)]=\"myModel\" (modelChange)=myChangeFn($event)></natural-select>\n * ```\n *\n * Custom template usage :\n * ```html\n * <natural-select [service]=\"svc\" [(ngModel)]=\"model\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-select>\n * ```\n *\n * `[(ngModel)]` and `(ngModelChange)` are optional.\n *\n * Placeholder :\n * ```html\n * <natural-select placeholder=\"my placeholder\"></natural-select>\n * ```\n *\n * Search with like %xxx% on specified field `name` instead of custom filter on whole object\n * ```html\n * <natural-select searchField=\"name\"></natural-select>\n * ```\n *\n * Allows to input free string without selecting an option from autocomplete suggestions\n * ```html\n * <natural-select [optionRequired]=\"false\"></natural-select>\n * ```\n */\n@Component({\n    selector: 'natural-select',\n    templateUrl: './select.component.html',\n    styleUrls: ['./select.component.scss'],\n    standalone: true,\n    imports: [\n        MatAutocompleteModule,\n        CommonModule,\n        MatOptionModule,\n        MatFormFieldModule,\n        MatInputModule,\n        FormsModule,\n        ReactiveFormsModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatProgressSpinnerModule,\n        MatButtonModule,\n        MatTooltipModule,\n        RouterLink,\n    ],\n})\nexport class NaturalSelectComponent<\n        TService extends NaturalAbstractModelService<\n            any,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            any,\n            any\n        >,\n    >\n    extends AbstractSelect<V<TService>, V<TService>>\n    implements OnInit, OnDestroy, ControlValueAccessor, AfterViewInit\n{\n    @ViewChild(MatAutocompleteTrigger) public autoTrigger!: MatAutocompleteTrigger;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    /**\n     * Service with watchAll function that accepts queryVariables.\n     */\n    @Input({required: true}) public service!: TService;\n\n    /**\n     * If false, allows to input free string without selecting an option from autocomplete suggestions\n     */\n    @Input() public optionRequired = true;\n\n    /**\n     * The field on which to search for, default to 'custom'.\n     */\n    @Input() public searchField: 'custom' | string = 'custom';\n\n    /**\n     * The operator with which to search for, default to 'search' if `searchField` is 'custom', else 'like'.\n     */\n    @Input() public searchOperator: 'search' | string | null = null;\n\n    /**\n     * Cache the committed value during search mode.\n     * It's used to be restored in case we cancel the selection\n     */\n    private lastValidValue: V<TService> | null = null;\n\n    /**\n     * Additional filter for query\n     */\n    @Input()\n    public set filter(filter: ExtractVall<TService>['filter'] | null | undefined) {\n        this.variablesManager.set('additional-filter', {filter: filter});\n    }\n\n    /**\n     * Items returned by server to show in listing\n     */\n    public items: null | Observable<readonly any[]> = null;\n\n    /**\n     * Whether we are searching something\n     */\n    public loading = false;\n\n    /**\n     * Number of items not shown in result list\n     * Shows a message after list if positive\n     */\n    public moreNbItems = 0;\n\n    /**\n     * Default page size\n     */\n    private pageSize = 10;\n\n    /**\n     * Init search options\n     */\n    private readonly variablesManager = new NaturalQueryVariablesManager<QueryVariables>();\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public override set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public ngAfterViewInit(): void {\n        this.internalCtrl.valueChanges\n            .pipe(takeUntil(this.ngUnsubscribe), distinctUntilChanged(), debounceTime(300))\n            .subscribe(val => this.search(val));\n    }\n\n    public onInternalFormChange(): void {\n        // If we allow free string typing, then we propagate it as it is being typed\n        if (!this.optionRequired) {\n            this.propagateValue(this.internalCtrl.value);\n        }\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.initService();\n    }\n\n    public override onBlur(): void {\n        if (this.internalCtrl.dirty) {\n            this.reset();\n        }\n        super.onBlur();\n    }\n\n    /**\n     * Reset form to it's initial value\n     * Discard searched text (in autocomplete use case)\n     * Doest not commit the change to the model (no change event is emitted)\n     */\n    public reset(): void {\n        this.internalCtrl.setValue(this.lastValidValue);\n        this.internalCtrl.markAsPristine();\n    }\n\n    /**\n     * Enter semantic means we want to validate something.\n     * If we hit ENTER while typing a text, the stroke is ignored because the value is invalid (it's accepted in free text mode)\n     * If we hit ENTER while the input field is empty, we validate the unselection (empty is a valid value)\n     */\n    public onKeyEnter(): void {\n        if (!this.internalCtrl.value) {\n            this.clear();\n            this.autoTrigger.closePanel();\n        }\n    }\n\n    public override writeValue(value: V<TService> | null): void {\n        super.writeValue(value);\n        this.lastValidValue = this.internalCtrl.value;\n    }\n\n    private initService(): void {\n        // Assert given service has a watchAll function\n        if (typeof this.service.watchAll !== 'function') {\n            throw new TypeError('Provided service does not contain watchAll function');\n        }\n\n        const defaultPagination = {\n            pagination: {\n                pageIndex: 0,\n                pageSize: this.pageSize,\n            },\n        };\n\n        const variables = merge(defaultPagination, this.getSearchFilter(null));\n        this.variablesManager.set('variables', variables);\n    }\n\n    public startSearch(): void {\n        // Start search only once\n        if (this.items) {\n            return;\n        }\n\n        // Init query, and when query results arrive, finish loading, and count items\n        this.items = this.service.watchAll(this.variablesManager).pipe(\n            takeUntil(this.ngUnsubscribe),\n            finalize(() => (this.loading = false)),\n            map(data => {\n                this.loading = false;\n                const nbTotal = data.length;\n                const nbListed = Math.min(data.length, this.pageSize);\n                this.moreNbItems = nbTotal - nbListed;\n\n                return data.items;\n            }),\n        );\n\n        this.loading = true;\n        this.items.subscribe();\n    }\n\n    /**\n     * Commit the model change\n     * Set internal form as pristine to reflect that the visible value match the model\n     */\n    public override propagateValue(value: V<TService> | null): void {\n        this.internalCtrl.markAsPristine();\n        this.lastValidValue = this.internalCtrl.value;\n        this.loading = false;\n\n        // If we cleared value via button, but we allow free string typing, then force to empty string\n        if (!this.optionRequired && value === null) {\n            value = '';\n        }\n\n        super.propagateValue(value);\n    }\n\n    /**\n     * Very important to return something, above all if [select]='displayedValue' attribute value is used\n     */\n    public getDisplayFn(): (item: V<TService> | null) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return (item: any) => {\n            if (!item) {\n                return '';\n            }\n\n            if (typeof item === 'string') {\n                return item;\n            }\n\n            return item.fullName || item.name || item.iban || item[this.searchField] || item.id;\n        };\n    }\n\n    public override clear(): void {\n        this.search(null);\n        super.clear();\n    }\n\n    public search(term: V<TService> | null): void {\n        if (typeof term === 'string' || term === null) {\n            if (term) {\n                this.loading = !!this.items;\n            }\n\n            this.variablesManager.merge('variables', this.getSearchFilter(term));\n        }\n    }\n\n    public override showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    private getSearchFilter(term: string | null): QueryVariables {\n        const searchOperator = this.searchOperator ?? (this.searchField === 'custom' ? 'search' : 'like');\n        if (term && searchOperator === 'like') {\n            term = '%' + term + '%';\n        }\n\n        return {\n            filter: {\n                groups: [\n                    {\n                        conditions: [\n                            {\n                                [this.searchField]: term\n                                    ? {\n                                          [searchOperator]: {value: term},\n                                      }\n                                    : null,\n                            },\n                        ],\n                    },\n                ],\n            },\n        };\n    }\n\n    public getVariablesForDebug(): Readonly<QueryVariables> | undefined {\n        return this.variablesManager.variables.value;\n    }\n}\n","<!-- Autocomplete menu -->\n<mat-autocomplete\n    #ac=\"matAutocomplete\"\n    (optionSelected)=\"propagateValue($event.option.value)\"\n    [displayWith]=\"getDisplayFn()\"\n    panelWidth=\"auto !important\"\n>\n    <mat-option *ngFor=\"let item of items | async\" [value]=\"item\">\n        <ng-template\n            [ngTemplateOutletContext]=\"{item: item}\"\n            [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n        ></ng-template>\n    </mat-option>\n    <div *ngIf=\"moreNbItems > 0\" class=\"mat-caption\" i18n style=\"padding: 5px 10px\"\n        >{{ moreNbItems }} élément(s) supplémentaire(s)</div\n    >\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n    <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n    <mat-label>{{ placeholder }}</mat-label>\n\n    <input\n        (blur)=\"onBlur()\"\n        (change)=\"onInternalFormChange()\"\n        (click)=\"autoTrigger.openPanel()\"\n        (focus)=\"startSearch()\"\n        (keydown.esc)=\"reset()\"\n        (keydown.enter)=\"onKeyEnter()\"\n        [formControl]=\"internalCtrl\"\n        [matAutocomplete]=\"ac\"\n        aria-label=\"Recherche et sélection\"\n        i18n-aria-label\n        matInput\n        [errorStateMatcher]=\"matcher\"\n    />\n\n    <!-- Meta data -->\n    <mat-icon *ngIf=\"!loading && showIcon\" [naturalIcon]=\"icon\" matIconPrefix></mat-icon>\n\n    <div class=\"loading-wrapper\" matIconPrefix *ngIf=\"loading\">\n        <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\"></mat-progress-spinner>\n    </div>\n\n    <!-- Clear button -->\n    <div class=\"suffix-buttons\" matIconSuffix>\n        <button\n            *ngIf=\"internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel\"\n            (click)=\"clear()\"\n            mat-icon-button\n            i18n-matTooltip\n            matTooltip=\"Désélectionner\"\n        >\n            <mat-icon naturalIcon=\"close\"></mat-icon>\n        </button>\n        <button\n            *ngIf=\"internalCtrl.dirty && internalCtrl.enabled && optionRequired\"\n            (click)=\"reset()\"\n            mat-icon-button\n            i18n-matTooltip\n            matTooltip=\"Annuler la recherche\"\n        >\n            <mat-icon naturalIcon=\"undo\"></mat-icon>\n        </button>\n        <button\n            *ngIf=\"internalCtrl.pristine && internalCtrl.value && navigateTo\"\n            [routerLink]=\"navigateTo\"\n            (click)=\"$event.stopPropagation()\"\n            mat-icon-button\n            i18n-matTooltip\n            matTooltip=\"Naviguer vers\"\n        >\n            <mat-icon naturalIcon=\"open_in_browser\"></mat-icon>\n        </button>\n    </div>\n\n    <mat-error *ngIf=\"hasRequiredError()\" i18n>Ce champ est requis</mat-error>\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n<div *ngIf=\"showClearButton()\" class=\"external-buttons\">\n    <button (click)=\"clear()\" *ngIf=\"showClearButton()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n</div>\n"]}
|
|
302
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/select/select/select.component.ts","../../../../../../../projects/natural/src/lib/modules/select/select/select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Component, ContentChild, Input, OnDestroy, OnInit, TemplateRef, ViewChild} from '@angular/core';\nimport {ControlValueAccessor, FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatAutocompleteTrigger, MatAutocompleteModule} from '@angular/material/autocomplete';\nimport {merge} from 'lodash-es';\nimport {Observable} from 'rxjs';\nimport {debounceTime, distinctUntilChanged, finalize, map, takeUntil} from 'rxjs/operators';\nimport {PaginatedData} from '../../../classes/data-source';\nimport {NaturalQueryVariablesManager, QueryVariables} from '../../../classes/query-variable-manager';\nimport {NaturalAbstractModelService} from '../../../services/abstract-model.service';\nimport {ExtractTallOne, ExtractVall, Literal} from '../../../types/types';\nimport {AbstractSelect} from '../abstract-select.component';\nimport {RouterLink} from '@angular/router';\nimport {MatTooltipModule} from '@angular/material/tooltip';\nimport {MatButtonModule} from '@angular/material/button';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {NaturalIconDirective} from '../../icon/icon.directive';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatOptionModule} from '@angular/material/core';\nimport {CommonModule} from '@angular/common';\n\ntype V<TService> = string | ExtractTallOne<TService>;\n\n/**\n * Default usage:\n * ```html\n * <natural-select [service]=\"myServiceInstance\" [(model)]=\"myModel\" (modelChange)=myChangeFn($event)></natural-select>\n * ```\n *\n * Custom template usage :\n * ```html\n * <natural-select [service]=\"svc\" [(ngModel)]=\"model\">\n *     <ng-template let-item=\"item\">\n *         <span>{{ item.xxx }}</span>\n *     </ng-template>\n * </natural-select>\n * ```\n *\n * `[(ngModel)]` and `(ngModelChange)` are optional.\n *\n * Placeholder :\n * ```html\n * <natural-select placeholder=\"my placeholder\"></natural-select>\n * ```\n *\n * Search with like %xxx% on specified field `name` instead of custom filter on whole object\n * ```html\n * <natural-select searchField=\"name\"></natural-select>\n * ```\n *\n * Allows to input free string without selecting an option from autocomplete suggestions\n * ```html\n * <natural-select [optionRequired]=\"false\"></natural-select>\n * ```\n */\n@Component({\n    selector: 'natural-select',\n    templateUrl: './select.component.html',\n    styleUrls: ['./select.component.scss'],\n    standalone: true,\n    imports: [\n        MatAutocompleteModule,\n        CommonModule,\n        MatOptionModule,\n        MatFormFieldModule,\n        MatInputModule,\n        FormsModule,\n        ReactiveFormsModule,\n        MatIconModule,\n        NaturalIconDirective,\n        MatProgressSpinnerModule,\n        MatButtonModule,\n        MatTooltipModule,\n        RouterLink,\n    ],\n})\nexport class NaturalSelectComponent<\n        TService extends NaturalAbstractModelService<\n            any,\n            any,\n            PaginatedData<Literal>,\n            QueryVariables,\n            any,\n            any,\n            any,\n            any,\n            any,\n            any\n        >,\n    >\n    extends AbstractSelect<V<TService>, V<TService>>\n    implements OnInit, OnDestroy, ControlValueAccessor, AfterViewInit\n{\n    @ViewChild(MatAutocompleteTrigger) public autoTrigger!: MatAutocompleteTrigger;\n    @ContentChild(TemplateRef) public itemTemplate?: TemplateRef<any>;\n\n    /**\n     * Service with watchAll function that accepts queryVariables.\n     */\n    @Input({required: true}) public service!: TService;\n\n    /**\n     * If false, allows to input free string without selecting an option from autocomplete suggestions\n     */\n    @Input() public optionRequired = true;\n\n    /**\n     * The field on which to search for, default to 'custom'.\n     */\n    @Input() public searchField: 'custom' | string = 'custom';\n\n    /**\n     * The operator with which to search for, default to 'search' if `searchField` is 'custom', else 'like'.\n     */\n    @Input() public searchOperator: 'search' | string | null = null;\n\n    /**\n     * Cache the committed value during search mode.\n     * It's used to be restored in case we cancel the selection\n     */\n    private lastValidValue: V<TService> | null = null;\n\n    /**\n     * Additional filter for query\n     */\n    @Input()\n    public set filter(filter: ExtractVall<TService>['filter'] | null | undefined) {\n        this.variablesManager.set('additional-filter', {filter: filter});\n    }\n\n    /**\n     * Items returned by server to show in listing\n     */\n    public items: null | Observable<readonly any[]> = null;\n\n    /**\n     * Whether we are searching something\n     */\n    public loading = false;\n\n    /**\n     * Number of items not shown in result list\n     * Shows a message after list if positive\n     */\n    public moreNbItems = 0;\n\n    /**\n     * Default page size\n     */\n    private pageSize = 10;\n\n    /**\n     * Init search options\n     */\n    private readonly variablesManager = new NaturalQueryVariablesManager<QueryVariables>();\n\n    /**\n     * Whether the value can be changed\n     */\n    @Input()\n    public override set disabled(disabled: boolean) {\n        disabled ? this.internalCtrl.disable() : this.internalCtrl.enable();\n    }\n\n    public ngAfterViewInit(): void {\n        this.internalCtrl.valueChanges\n            .pipe(takeUntil(this.ngUnsubscribe), distinctUntilChanged(), debounceTime(300))\n            .subscribe(val => this.search(val));\n    }\n\n    public onInternalFormChange(): void {\n        // If we allow free string typing, then we propagate it as it is being typed\n        if (!this.optionRequired) {\n            this.propagateValue(this.internalCtrl.value);\n        }\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.initService();\n    }\n\n    public override onBlur(): void {\n        if (this.internalCtrl.dirty) {\n            this.reset();\n        }\n        super.onBlur();\n    }\n\n    /**\n     * Reset form to it's initial value\n     * Discard searched text (in autocomplete use case)\n     * Doest not commit the change to the model (no change event is emitted)\n     */\n    public reset(): void {\n        this.internalCtrl.setValue(this.lastValidValue);\n        this.internalCtrl.markAsPristine();\n    }\n\n    /**\n     * Enter semantic means we want to validate something.\n     * If we hit ENTER while typing a text, the stroke is ignored because the value is invalid (it's accepted in free text mode)\n     * If we hit ENTER while the input field is empty, we validate the unselection (empty is a valid value)\n     */\n    public onKeyEnter(): void {\n        if (!this.internalCtrl.value) {\n            this.clear();\n            this.autoTrigger.closePanel();\n        }\n    }\n\n    public override writeValue(value: V<TService> | null): void {\n        super.writeValue(value);\n        this.lastValidValue = this.internalCtrl.value;\n    }\n\n    private initService(): void {\n        // Assert given service has a watchAll function\n        if (typeof this.service.watchAll !== 'function') {\n            throw new TypeError('Provided service does not contain watchAll function');\n        }\n\n        const defaultPagination = {\n            pagination: {\n                pageIndex: 0,\n                pageSize: this.pageSize,\n            },\n        };\n\n        const variables = merge(defaultPagination, this.getSearchFilter(null));\n        this.variablesManager.set('variables', variables);\n    }\n\n    public startSearch(): void {\n        // Start search only once\n        if (this.items) {\n            return;\n        }\n\n        // Init query, and when query results arrive, finish loading, and count items\n        this.items = this.service.watchAll(this.variablesManager).pipe(\n            takeUntil(this.ngUnsubscribe),\n            finalize(() => (this.loading = false)),\n            map(data => {\n                this.loading = false;\n                const nbTotal = data.length;\n                const nbListed = Math.min(data.length, this.pageSize);\n                this.moreNbItems = nbTotal - nbListed;\n\n                return data.items;\n            }),\n        );\n\n        this.loading = true;\n        this.items.subscribe();\n    }\n\n    /**\n     * Commit the model change\n     * Set internal form as pristine to reflect that the visible value match the model\n     */\n    public override propagateValue(value: V<TService> | null): void {\n        this.internalCtrl.markAsPristine();\n        this.lastValidValue = this.internalCtrl.value;\n        this.loading = false;\n\n        // If we cleared value via button, but we allow free string typing, then force to empty string\n        if (!this.optionRequired && value === null) {\n            value = '';\n        }\n\n        super.propagateValue(value);\n    }\n\n    /**\n     * Very important to return something, above all if [select]='displayedValue' attribute value is used\n     */\n    public getDisplayFn(): (item: V<TService> | null) => string {\n        if (this.displayWith) {\n            return this.displayWith;\n        }\n\n        return (item: any) => {\n            if (!item) {\n                return '';\n            }\n\n            if (typeof item === 'string') {\n                return item;\n            }\n\n            return item.fullName || item.name || item.iban || item[this.searchField] || item.id;\n        };\n    }\n\n    public override clear(): void {\n        this.search(null);\n        super.clear();\n    }\n\n    public search(term: V<TService> | null): void {\n        if (typeof term === 'string' || term === null) {\n            if (term) {\n                this.loading = !!this.items;\n            }\n\n            this.variablesManager.merge('variables', this.getSearchFilter(term));\n        }\n    }\n\n    public override showClearButton(): boolean {\n        return this.internalCtrl?.enabled && !!this.clearLabel && !!this.internalCtrl.value;\n    }\n\n    private getSearchFilter(term: string | null): QueryVariables {\n        const searchOperator = this.searchOperator ?? (this.searchField === 'custom' ? 'search' : 'like');\n        if (term && searchOperator === 'like') {\n            term = '%' + term + '%';\n        }\n\n        return {\n            filter: {\n                groups: [\n                    {\n                        conditions: [\n                            {\n                                [this.searchField]: term\n                                    ? {\n                                          [searchOperator]: {value: term},\n                                      }\n                                    : null,\n                            },\n                        ],\n                    },\n                ],\n            },\n        };\n    }\n\n    public getVariablesForDebug(): Readonly<QueryVariables> | undefined {\n        return this.variablesManager.variables.value;\n    }\n}\n","<!-- Autocomplete menu -->\n<mat-autocomplete\n    #ac=\"matAutocomplete\"\n    (optionSelected)=\"propagateValue($event.option.value)\"\n    [displayWith]=\"getDisplayFn()\"\n    panelWidth=\"auto !important\"\n>\n    <mat-option *ngFor=\"let item of items | async\" [value]=\"item\">\n        <ng-template\n            [ngTemplateOutletContext]=\"{item: item}\"\n            [ngTemplateOutlet]=\"itemTemplate ? itemTemplate : defaultACItem\"\n        ></ng-template>\n    </mat-option>\n    <div *ngIf=\"moreNbItems > 0\" class=\"mat-caption\" i18n style=\"padding: 5px 10px\"\n        >{{ moreNbItems }} élément(s) supplémentaire(s)</div\n    >\n</mat-autocomplete>\n\n<ng-template #defaultACItem let-item=\"item\">\n    <span>{{ getDisplayFn()(item) }}</span>\n</ng-template>\n\n<!-- Input for autocomplete -->\n<mat-form-field>\n    <mat-label>{{ placeholder }}</mat-label>\n\n    <input\n        (blur)=\"onBlur()\"\n        (change)=\"onInternalFormChange()\"\n        (click)=\"autoTrigger.openPanel()\"\n        (focus)=\"startSearch()\"\n        (keydown.esc)=\"reset()\"\n        (keydown.enter)=\"onKeyEnter()\"\n        [formControl]=\"internalCtrl\"\n        [matAutocomplete]=\"ac\"\n        aria-label=\"Recherche et sélection\"\n        i18n-aria-label\n        matInput\n        [errorStateMatcher]=\"matcher\"\n    />\n\n    <mat-hint *ngIf=\"hint\">{{ hint }}</mat-hint>\n\n    <!-- Meta data -->\n    <mat-icon *ngIf=\"!loading && showIcon\" [naturalIcon]=\"icon\" matIconPrefix></mat-icon>\n\n    <div class=\"loading-wrapper\" matIconPrefix *ngIf=\"loading\">\n        <mat-progress-spinner [diameter]=\"21\" [strokeWidth]=\"5\" mode=\"indeterminate\"></mat-progress-spinner>\n    </div>\n\n    <!-- Clear button -->\n    <div class=\"suffix-buttons\" matIconSuffix>\n        <button\n            *ngIf=\"internalCtrl.pristine && internalCtrl.value && internalCtrl.enabled && !clearLabel\"\n            (click)=\"clear()\"\n            mat-icon-button\n            i18n-matTooltip\n            matTooltip=\"Désélectionner\"\n        >\n            <mat-icon naturalIcon=\"close\"></mat-icon>\n        </button>\n        <button\n            *ngIf=\"internalCtrl.dirty && internalCtrl.enabled && optionRequired\"\n            (click)=\"reset()\"\n            mat-icon-button\n            i18n-matTooltip\n            matTooltip=\"Annuler la recherche\"\n        >\n            <mat-icon naturalIcon=\"undo\"></mat-icon>\n        </button>\n        <button\n            *ngIf=\"internalCtrl.pristine && internalCtrl.value && navigateTo\"\n            [routerLink]=\"navigateTo\"\n            (click)=\"$event.stopPropagation()\"\n            mat-icon-button\n            i18n-matTooltip\n            matTooltip=\"Naviguer vers\"\n        >\n            <mat-icon naturalIcon=\"open_in_browser\"></mat-icon>\n        </button>\n    </div>\n\n    <mat-error *ngIf=\"hasRequiredError()\" i18n>Ce champ est requis</mat-error>\n</mat-form-field>\n\n<!-- Additional (un)select/(un)link buttons for more visual cohesion with natural-relations --><!-- [clearLabel] and/or [selectLabel] has to be given as attribute input -->\n<div *ngIf=\"showClearButton()\" class=\"external-buttons\">\n    <button (click)=\"clear()\" *ngIf=\"showClearButton()\" color=\"warn\" mat-button>{{ clearLabel }}</button>\n</div>\n"]}
|