@ngutil/data 0.0.10

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/README.md ADDED
@@ -0,0 +1,7 @@
1
+ # list
2
+
3
+ ## Example
4
+
5
+ ```html
6
+ <xyz-grid [nuList]="src" [nuListFilter]="filter" [nuListSorter]="sorter" [nuListTrackBy]="trackBy"> </xyz-grid>
7
+ ```
@@ -0,0 +1,3 @@
1
+ export { sortBy } from "./query/sorter";
2
+ export { filterBy } from "./query/filter";
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9kYXRhL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFZLE1BQU0sZ0JBQWdCLENBQUE7QUFDakQsT0FBTyxFQUFFLFFBQVEsRUFBWSxNQUFNLGdCQUFnQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgc29ydEJ5LCBTb3J0ZXJGbiB9IGZyb20gXCIuL3F1ZXJ5L3NvcnRlclwiXG5leHBvcnQgeyBmaWx0ZXJCeSwgRmlsdGVyRm4gfSBmcm9tIFwiLi9xdWVyeS9maWx0ZXJcIlxuIl19
@@ -0,0 +1,2 @@
1
+ export { ModelRefNorm, ModelMeta, ModelRefByIndex, ModelRefByKey, UnknownMeta } from "./meta";
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9kYXRhL3NyYy9tb2RlbC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBR0gsWUFBWSxFQUNaLFNBQVMsRUFJVCxlQUFlLEVBQ2YsYUFBYSxFQUNiLFdBQVcsRUFDZCxNQUFNLFFBQVEsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7XG4gICAgTW9kZWwsXG4gICAgTW9kZWxSZWYsXG4gICAgTW9kZWxSZWZOb3JtLFxuICAgIE1vZGVsTWV0YSxcbiAgICBNb2RlbE1ldGFQcm9wcyxcbiAgICBNb2RlbFJlZkZpbHRlcixcbiAgICBNb2RlbE1ldGFJbnB1dCxcbiAgICBNb2RlbFJlZkJ5SW5kZXgsXG4gICAgTW9kZWxSZWZCeUtleSxcbiAgICBVbmtub3duTWV0YVxufSBmcm9tIFwiLi9tZXRhXCJcbiJdfQ==
@@ -0,0 +1,125 @@
1
+ import { isEqual } from "lodash";
2
+ import { filterBy, pathGetterCompile } from "../query";
3
+ export class ModelMeta {
4
+ static coerce(value) {
5
+ if (value instanceof ModelMeta) {
6
+ return value;
7
+ }
8
+ else {
9
+ return new ModelMeta(value);
10
+ }
11
+ }
12
+ #getKey;
13
+ constructor(props) {
14
+ this.keys = props.keys;
15
+ if (this.keys.length > 0) {
16
+ const getters = this.keys.map(pathGetterCompile);
17
+ if (props.trackBy == null) {
18
+ this.trackBy = (index, item) => getters.map(p => p(item)).join("∀");
19
+ }
20
+ else {
21
+ this.trackBy = props.trackBy;
22
+ }
23
+ this.#getKey = (item) => getters.map(p => p(item)).join("∀");
24
+ }
25
+ else {
26
+ if (props.trackBy == null) {
27
+ throw new Error("Can't compile track by function without `keys` declaration");
28
+ }
29
+ else {
30
+ this.trackBy = props.trackBy;
31
+ }
32
+ }
33
+ }
34
+ isEqual(a, b) {
35
+ return isEqual(a, b);
36
+ }
37
+ isEqualByTrack(a, b) {
38
+ if (a == null || b == null) {
39
+ return a == null && b == null;
40
+ }
41
+ if (a.index == null || b.index == null) {
42
+ return a.index == null && b.index == null;
43
+ }
44
+ if (a.model == null || b.model == null) {
45
+ return a.model == null && b.model == null;
46
+ }
47
+ return this.trackBy(a.index, a.model) === this.trackBy(b.index, b.model);
48
+ }
49
+ isEqualByKey(a, b) {
50
+ if (this.#getKey != null) {
51
+ if (a == null || b == null) {
52
+ return a == null && b == null;
53
+ }
54
+ return this.#getKey(a) === this.#getKey(b);
55
+ }
56
+ console.warn("Primary keys is not defined for", a, b);
57
+ return false;
58
+ }
59
+ normalizeRef(ref) {
60
+ if (ref instanceof ModelRefNorm) {
61
+ return ref;
62
+ }
63
+ if (ref.key != null && ref.index != null) {
64
+ throw new Error("Only provide `pk` or `index` value not both");
65
+ }
66
+ if (ref.key != null) {
67
+ const keyValue = (Array.isArray(ref.key) ? ref.key : [ref.key]);
68
+ if (keyValue.length > 0) {
69
+ if (this.keys.length === 0) {
70
+ throw new Error("Can't normalize ref without `keys`");
71
+ }
72
+ if (keyValue.length !== this.keys.length) {
73
+ throw new Error(`Wrong number of \`key\` values for this keys: [${this.keys.join(",")}]`);
74
+ }
75
+ return new ModelRefByKey(keyValue, this.keys);
76
+ }
77
+ else {
78
+ console.warn("Empty key in ModelRef", ref);
79
+ }
80
+ }
81
+ if (ref.index != null) {
82
+ return new ModelRefByIndex(ref.index);
83
+ }
84
+ throw new Error("Missing `key` or `index` value");
85
+ }
86
+ }
87
+ export class ModelRefNorm {
88
+ #filter;
89
+ toFilter() {
90
+ if (this.#filter == null) {
91
+ return (this.#filter = this._asFilter());
92
+ }
93
+ return this.#filter;
94
+ }
95
+ }
96
+ export class ModelRefByKey extends ModelRefNorm {
97
+ #keys;
98
+ constructor(key, keys) {
99
+ super();
100
+ this.key = key;
101
+ this.#keys = keys;
102
+ }
103
+ _asFilter() {
104
+ const filter = {};
105
+ for (let i = 0; i < this.#keys.length; i++) {
106
+ filter[this.#keys[i]] = this.key[i];
107
+ }
108
+ return filterBy(filter);
109
+ }
110
+ }
111
+ export class ModelRefByIndex extends ModelRefNorm {
112
+ constructor(index) {
113
+ super();
114
+ this.index = index;
115
+ }
116
+ _asFilter() {
117
+ return (item, index) => this.index === index;
118
+ }
119
+ }
120
+ export class UnknownMeta extends ModelMeta {
121
+ constructor() {
122
+ super({ keys: [], trackBy: (index) => index });
123
+ }
124
+ }
125
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"meta.js","sourceRoot":"","sources":["../../../../../packages/data/src/model/meta.ts"],"names":[],"mappings":"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","sourcesContent":["import { TrackByFunction } from \"@angular/core\"\n\nimport { isEqual } from \"lodash\"\nimport { Primitive } from \"utility-types\"\n\nimport { filterBy, pathGetterCompile } from \"../query\"\n\nexport type Model = { [key: string]: any }\n\nexport type ModelRef = { key?: Primitive | Array<Primitive>; index?: number }\nexport type ModelRefFilter<T> = (item: T, index: number) => boolean\n\nexport interface ModelMetaProps<T extends Model> {\n    readonly keys: readonly string[]\n    readonly trackBy?: TrackByFunction<T>\n}\n\nexport type ModelMetaInput<T extends Model> = ModelMeta<T> | ModelMetaProps<T>\n\nexport type TrackedModel<T extends Model> = { index: number; model: T }\n\nexport class ModelMeta<T extends Model> implements ModelMetaProps<T> {\n    static coerce<M extends Model>(value: ModelMetaInput<M>): ModelMeta<M> {\n        if (value instanceof ModelMeta) {\n            return value\n        } else {\n            return new ModelMeta(value)\n        }\n    }\n\n    readonly keys: readonly string[]\n    readonly trackBy: TrackByFunction<T>\n    readonly #getKey?: (model: T) => string\n\n    constructor(props: ModelMetaProps<T>) {\n        this.keys = props.keys\n\n        if (this.keys.length > 0) {\n            const getters = this.keys.map(pathGetterCompile)\n            if (props.trackBy == null) {\n                this.trackBy = (index: number, item: T) => getters.map(p => p(item)).join(\"∀\")\n            } else {\n                this.trackBy = props.trackBy\n            }\n            this.#getKey = (item: T) => getters.map(p => p(item)).join(\"∀\")\n        } else {\n            if (props.trackBy == null) {\n                throw new Error(\"Can't compile track by function without `keys` declaration\")\n            } else {\n                this.trackBy = props.trackBy\n            }\n        }\n    }\n\n    isEqual(a: T, b: T): boolean {\n        return isEqual(a, b)\n    }\n\n    isEqualByTrack(a: TrackedModel<T>, b: TrackedModel<T>): boolean {\n        if (a == null || b == null) {\n            return a == null && b == null\n        }\n\n        if (a.index == null || b.index == null) {\n            return a.index == null && b.index == null\n        }\n\n        if (a.model == null || b.model == null) {\n            return a.model == null && b.model == null\n        }\n\n        return this.trackBy(a.index, a.model) === this.trackBy(b.index, b.model)\n    }\n\n    isEqualByKey(a: T, b: T): boolean {\n        if (this.#getKey != null) {\n            if (a == null || b == null) {\n                return a == null && b == null\n            }\n            return this.#getKey(a) === this.#getKey(b)\n        }\n        console.warn(\"Primary keys is not defined for\", a, b)\n        return false\n    }\n\n    normalizeRef(ref: ModelRef | ModelRefNorm): ModelRefNorm {\n        if (ref instanceof ModelRefNorm) {\n            return ref\n        }\n\n        if (ref.key != null && ref.index != null) {\n            throw new Error(\"Only provide `pk` or `index` value not both\")\n        }\n\n        if (ref.key != null) {\n            const keyValue = (Array.isArray(ref.key) ? ref.key : [ref.key]) as readonly Primitive[]\n\n            if (keyValue.length > 0) {\n                if (this.keys.length === 0) {\n                    throw new Error(\"Can't normalize ref without `keys`\")\n                }\n\n                if (keyValue.length !== this.keys.length) {\n                    throw new Error(`Wrong number of \\`key\\` values for this keys: [${this.keys.join(\",\")}]`)\n                }\n                return new ModelRefByKey(keyValue, this.keys)\n            } else {\n                console.warn(\"Empty key in ModelRef\", ref)\n            }\n        }\n\n        if (ref.index != null) {\n            return new ModelRefByIndex(ref.index)\n        }\n\n        throw new Error(\"Missing `key` or `index` value\")\n    }\n}\n\nexport abstract class ModelRefNorm {\n    readonly key?: readonly Primitive[]\n    readonly index?: number\n\n    #filter?: ModelRefFilter<any>\n    toFilter() {\n        if (this.#filter == null) {\n            return (this.#filter = this._asFilter())\n        }\n        return this.#filter\n    }\n\n    protected abstract _asFilter(): ModelRefFilter<any>\n}\n\nexport class ModelRefByKey extends ModelRefNorm {\n    override readonly key: readonly Primitive[]\n\n    #keys: readonly string[]\n    constructor(key: readonly Primitive[], keys: readonly string[]) {\n        super()\n        this.key = key\n        this.#keys = keys\n    }\n\n    protected _asFilter(): ModelRefFilter<any> {\n        const filter = {} as any\n        for (let i = 0; i < this.#keys.length; i++) {\n            filter[this.#keys[i]] = this.key[i]\n        }\n        return filterBy(filter)\n    }\n}\n\nexport class ModelRefByIndex extends ModelRefNorm {\n    override readonly index: number\n\n    constructor(index: number) {\n        super()\n        this.index = index\n    }\n\n    protected _asFilter(): ModelRefFilter<any> {\n        return (item: any, index: number) => this.index === index\n    }\n}\n\nexport class UnknownMeta<T extends Model> extends ModelMeta<T> {\n    constructor() {\n        super({ keys: [], trackBy: (index: number) => index })\n    }\n}\n"]}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './index';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmd1dGlsLWRhdGEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9kYXRhL3NyYy9uZ3V0aWwtZGF0YS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2luZGV4JztcbiJdfQ==
@@ -0,0 +1,76 @@
1
+ import { groupBy, isEqual, sortBy } from "lodash";
2
+ import { toSorted } from "@ngutil/common";
3
+ import { filterBy } from "./filter";
4
+ const INPUT = Symbol("INPUT");
5
+ export function queryExecutor(query, previous) {
6
+ const executors = {
7
+ filterFn: filterExecutor(query.filter, previous?.filterFn),
8
+ sorterFn: sorterExecutor(query.sorter, previous?.sorterFn),
9
+ grouperFn: grouperExecutor(query.grouper, previous?.grouperFn),
10
+ slicerFn: sliceExecutor(query.slice, previous?.slicerFn)
11
+ };
12
+ const changed = previous == null || Object.entries(executors).some(([k, v]) => previous[k] !== v);
13
+ if (!changed) {
14
+ return previous;
15
+ }
16
+ const executor = (items) => {
17
+ if (items == null || items.length === 0) {
18
+ return { items: [], total: 0 };
19
+ }
20
+ let result = items;
21
+ if (executors.filterFn) {
22
+ result = result.filter(executors.filterFn);
23
+ }
24
+ if (executors.sorterFn) {
25
+ result = toSorted(result, executors.sorterFn);
26
+ }
27
+ const total = result.length;
28
+ // TODO: grouper
29
+ if (executors.slicerFn) {
30
+ result = executors.slicerFn(result);
31
+ }
32
+ return { items: result, total };
33
+ };
34
+ for (const [k, v] of Object.entries(executors)) {
35
+ Object.defineProperty(executor, k, {
36
+ value: v,
37
+ enumerable: true,
38
+ writable: false
39
+ });
40
+ }
41
+ return executor;
42
+ }
43
+ function filterExecutor(filter, prev) {
44
+ return compileExec(filter, prev, filterBy);
45
+ }
46
+ function sorterExecutor(sorter, prev) {
47
+ return compileExec(sorter, prev, sortBy);
48
+ }
49
+ function grouperExecutor(sorter, prev) {
50
+ return compileExec(sorter, prev, groupBy);
51
+ }
52
+ function sliceExecutor(slice, prev) {
53
+ return compileExec(slice, prev, sliceBy);
54
+ }
55
+ function sliceBy(slice) {
56
+ return (items) => items.slice(slice.start, slice.end);
57
+ }
58
+ function compileExec(input, prev, compiler) {
59
+ if (input == null) {
60
+ return undefined;
61
+ }
62
+ if (prev != null) {
63
+ // thorically input is readonly
64
+ if (prev[INPUT] === input || isEqual(prev[INPUT], input)) {
65
+ return prev;
66
+ }
67
+ }
68
+ const exec = compiler(input);
69
+ Object.defineProperty(exec, INPUT, {
70
+ value: input,
71
+ enumerable: true,
72
+ writable: false
73
+ });
74
+ return exec;
75
+ }
76
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"executor.js","sourceRoot":"","sources":["../../../../../packages/data/src/query/executor.ts"],"names":[],"mappings":"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","sourcesContent":["import { groupBy, isEqual, sortBy } from \"lodash\"\nimport { Primitive } from \"utility-types\"\n\nimport { DeepReadonly, toSorted } from \"@ngutil/common\"\n\nimport { Model } from \"../model\"\nimport { Filter, filterBy } from \"./filter\"\nimport { Query, QueryResult } from \"./query\"\nimport { Slice } from \"./slice\"\nimport { Sorter } from \"./sorter\"\n\nconst INPUT = Symbol(\"INPUT\")\n\ninterface PartialExecutor {\n    [INPUT]: any\n}\n\nexport interface QueryExecutors<T extends Model> {\n    readonly filterFn?: FilterExecutor<T>\n    readonly sorterFn?: SorterExecutor<T>\n    readonly grouperFn?: GrouperExecutor<T>\n    readonly slicerFn?: SliceExecutor<T>\n}\n\nexport interface QueryExecutor<T extends Model> extends QueryExecutors<T> {\n    (items: readonly T[]): QueryResult<T>\n}\n\nexport function queryExecutor<T extends Model>(query: Query<T>, previous?: QueryExecutor<T>): QueryExecutor<T> {\n    const executors: QueryExecutors<T> = {\n        filterFn: filterExecutor(query.filter, previous?.filterFn),\n        sorterFn: sorterExecutor(query.sorter, previous?.sorterFn),\n        grouperFn: grouperExecutor(query.grouper, previous?.grouperFn),\n        slicerFn: sliceExecutor(query.slice, previous?.slicerFn)\n    }\n\n    const changed = previous == null || Object.entries(executors).some(([k, v]) => (previous as any)[k] !== v)\n    if (!changed) {\n        return previous\n    }\n\n    const executor = (items: readonly T[]): QueryResult<T> => {\n        if (items == null || items.length === 0) {\n            return { items: [], total: 0 }\n        }\n\n        let result: readonly T[] = items\n\n        if (executors.filterFn) {\n            result = result.filter(executors.filterFn)\n        }\n\n        if (executors.sorterFn) {\n            result = toSorted(result, executors.sorterFn)\n        }\n\n        const total = result.length\n\n        // TODO: grouper\n\n        if (executors.slicerFn) {\n            result = executors.slicerFn(result)\n        }\n\n        return { items: result, total }\n    }\n\n    for (const [k, v] of Object.entries(executors)) {\n        Object.defineProperty(executor, k, {\n            value: v,\n            enumerable: true,\n            writable: false\n        })\n    }\n\n    return executor\n}\n\ninterface FilterExecutor<T extends Model> extends PartialExecutor {\n    (item: T): boolean\n}\n\nfunction filterExecutor<T extends Model>(\n    filter?: DeepReadonly<Filter<T>>,\n    prev?: FilterExecutor<T>\n): FilterExecutor<T> | undefined {\n    return compileExec<T>(filter, prev, filterBy)\n}\n\ninterface SorterExecutor<T extends Model> extends PartialExecutor {\n    (a: T, b: T): number\n}\n\nfunction sorterExecutor<T extends Model>(\n    sorter?: DeepReadonly<Sorter<T>>,\n    prev?: SorterExecutor<T>\n): SorterExecutor<T> | undefined {\n    return compileExec<T>(sorter, prev, sortBy)\n}\n\ninterface GrouperExecutor<T extends Model> extends PartialExecutor {\n    (item: T): Primitive\n}\n\nfunction grouperExecutor<T extends Model>(\n    sorter?: DeepReadonly<Sorter<T>>,\n    prev?: GrouperExecutor<T>\n): GrouperExecutor<T> | undefined {\n    return compileExec<T>(sorter, prev, groupBy)\n}\n\ninterface SliceExecutor<T extends Model> extends PartialExecutor {\n    (items: readonly T[]): T[]\n}\n\nfunction sliceExecutor<T extends Model>(\n    slice?: DeepReadonly<Slice>,\n    prev?: SliceExecutor<T>\n): SliceExecutor<T> | undefined {\n    return compileExec<T>(slice, prev, sliceBy)\n}\n\nfunction sliceBy<T extends Model>(slice: Slice) {\n    return (items: T[]) => items.slice(slice.start, slice.end)\n}\n\nfunction compileExec<T extends Model>(input: any, prev: any, compiler: any): any {\n    if (input == null) {\n        return undefined\n    }\n\n    if (prev != null) {\n        // thorically input is readonly\n        if (prev[INPUT] === input || isEqual(prev[INPUT], input)) {\n            return prev\n        }\n    }\n\n    const exec = compiler(input) as any\n    Object.defineProperty(exec, INPUT, {\n        value: input,\n        enumerable: true,\n        writable: false\n    })\n    return exec\n}\n"]}
@@ -0,0 +1,288 @@
1
+ import { flattenDeep, intersection } from "lodash";
2
+ import { deepClone, isPlainObject } from "@ngutil/common";
3
+ import { pathGetterCompile } from "./path";
4
+ export const OPERATORS = [
5
+ "==" /* FilterOp.Eq */,
6
+ "===" /* FilterOp.EqStrict */,
7
+ "==*" /* FilterOp.EqInsesitive */,
8
+ "!=" /* FilterOp.Neq */,
9
+ "!==" /* FilterOp.NeqStrict */,
10
+ "!=*" /* FilterOp.NeqInsesitive */,
11
+ ">" /* FilterOp.Gt */,
12
+ ">*" /* FilterOp.GtInsesitive */,
13
+ ">=" /* FilterOp.Gte */,
14
+ ">=*" /* FilterOp.GteInsesitive */,
15
+ "<" /* FilterOp.Lt */,
16
+ "<*" /* FilterOp.LtInsesitive */,
17
+ "<=" /* FilterOp.Lte */,
18
+ "<=*" /* FilterOp.LteInsesitive */,
19
+ "%" /* FilterOp.Contains */,
20
+ "%*" /* FilterOp.ContainsInsesitive */,
21
+ "^" /* FilterOp.StartsWith */,
22
+ "^*" /* FilterOp.StartsWithInsesitive */,
23
+ "$" /* FilterOp.EndsWith */,
24
+ "$*" /* FilterOp.EndsWithInsesitive */,
25
+ "~" /* FilterOp.Regexp */,
26
+ "~*" /* FilterOp.RegexpInsesitive */,
27
+ "|" /* FilterOp.Or */,
28
+ "&" /* FilterOp.And */
29
+ ];
30
+ export function asOperators(value) {
31
+ const ops = intersection(Object.keys(value), OPERATORS);
32
+ if (ops.length > 0) {
33
+ return ops.map(op => {
34
+ return { op, value: value[op] };
35
+ });
36
+ }
37
+ else {
38
+ return [{ op: "==" /* FilterOp.Eq */, value }];
39
+ }
40
+ }
41
+ /**
42
+ * @example
43
+ * ```ts
44
+ * items.filter(filterBy({id: 42}))
45
+ * ```
46
+ */
47
+ export function filterBy(filters) {
48
+ return _filterCompile(filters);
49
+ }
50
+ // TODO: Normalize filter
51
+ // type _Filter = { path: string; op: FilterOp; value: any }
52
+ // type _Or = { "|": Array<_Filter> }
53
+ // type _And = { "&": Array<_Filter> }
54
+ // export type NormalizedFilter = _Or | _And
55
+ // /**
56
+ // * @example
57
+ // * ```ts
58
+ // * normalizeFilter({id: 2, name: {"=*": "AnyName"}}) -> {id: {"==": 2}, name: {"=*": "AnyName"}}}
59
+ // * normalizeFilter({id: {">": 0, "<": 10}}) -> {"&": [{id: {">": 0}}, {id: {"<": 10}}]}
60
+ // * ```
61
+ // */
62
+ // export function normalizeFilter<T extends Model>(filters: Filters<T>): Filters<T> {
63
+ // return _normalizeFilter(filters)
64
+ // }
65
+ // function _normalizeFilter(filters: any, path?: string): any {
66
+ // const result = {} as any
67
+ // for (const [path, v] of Object.entries(filters)) {
68
+ // switch (path) {
69
+ // case FilterOp.And:
70
+ // if (!Array.isArray(v)) {
71
+ // throw new Error("The '&' (AND) operator must have array type")
72
+ // }
73
+ // if (!result[FilterOp.And]) {
74
+ // result[FilterOp.And] = []
75
+ // }
76
+ // result[FilterOp.And] = result[FilterOp.And].concat(v.map(f => _normalizeFilter(f)))
77
+ // break
78
+ // case FilterOp.Or:
79
+ // if (!Array.isArray(v)) {
80
+ // throw new Error("The '|' (OR) operator must have array type")
81
+ // }
82
+ // if (!result[FilterOp.Or]) {
83
+ // result[FilterOp.Or] = []
84
+ // }
85
+ // result[FilterOp.Or] = result[FilterOp.Or].concat(v.map(f => _normalizeFilter(f)))
86
+ // break
87
+ // default:
88
+ // for (const entry of asOperators(v)) {
89
+ // switch (entry.op) {
90
+ // case FilterOp.And:
91
+ // if (!result[FilterOp.And]) {
92
+ // result[FilterOp.And] = []
93
+ // }
94
+ // result[FilterOp.And] = result[FilterOp.And].concat(
95
+ // entry.value.map((v: any) => _normalizeFilter(v, path))
96
+ // )
97
+ // break
98
+ // case FilterOp.Or:
99
+ // if (!result[FilterOp.Or]) {
100
+ // result[FilterOp.Or] = []
101
+ // }
102
+ // result[FilterOp.Or] = result[FilterOp.Or].concat(
103
+ // entry.value.map((v: any) => _normalizeFilter(v, path))
104
+ // )
105
+ // break
106
+ // default:
107
+ // if (!result[FilterOp.And]) {
108
+ // result[FilterOp.And] = []
109
+ // }
110
+ // result[FilterOp.And].push({ path, ...entry })
111
+ // }
112
+ // }
113
+ // }
114
+ // }
115
+ // return result
116
+ // }
117
+ function _filterCompile(filters) {
118
+ let getter;
119
+ const result = [];
120
+ for (const [pth, value] of Object.entries(filters)) {
121
+ switch (pth) {
122
+ case "&" /* FilterOp.And */:
123
+ if (!Array.isArray(value)) {
124
+ throw new Error("Root '&' (AND) operator must have array type");
125
+ }
126
+ result.splice(result.length, 0, ...value.map((_filterCompile)));
127
+ break;
128
+ case "|" /* FilterOp.Or */:
129
+ if (!Array.isArray(value)) {
130
+ throw new Error("Root '|' (OR) operator must have array type");
131
+ }
132
+ result.push(or_(value.map((_filterCompile))));
133
+ break;
134
+ default:
135
+ getter = pathGetterCompile(pth);
136
+ if (isPlainObject(value)) {
137
+ result.push(and_(Object.entries(value).map(([op, opv]) => filterCompileOp(getter, op, opv))));
138
+ }
139
+ else {
140
+ result.push(filterCompileOp(getter, "==" /* FilterOp.Eq */, value));
141
+ }
142
+ break;
143
+ }
144
+ }
145
+ return and_(result);
146
+ }
147
+ function filterCompileOp(getter, op, value) {
148
+ let lower;
149
+ let regex;
150
+ switch (op) {
151
+ case "==" /* FilterOp.Eq */:
152
+ // eslint-disable-next-line eqeqeq
153
+ return matcher(getter, v => v == value);
154
+ case "===" /* FilterOp.EqStrict */:
155
+ return matcher(getter, v => v === value);
156
+ case "==*" /* FilterOp.EqInsesitive */:
157
+ lower = String(value).toLocaleLowerCase();
158
+ return matcher(getter, v => String(v).toLocaleLowerCase() === lower);
159
+ case "!=" /* FilterOp.Neq */:
160
+ // eslint-disable-next-line eqeqeq
161
+ return matcher(getter, v => v != value);
162
+ case "!==" /* FilterOp.NeqStrict */:
163
+ return matcher(getter, v => v !== value);
164
+ case "!=*" /* FilterOp.NeqInsesitive */:
165
+ lower = String(value).toLocaleLowerCase();
166
+ return matcher(getter, v => String(v).toLocaleLowerCase() !== lower);
167
+ case ">" /* FilterOp.Gt */:
168
+ return matcher(getter, v => v > value);
169
+ case ">*" /* FilterOp.GtInsesitive */:
170
+ lower = String(value).toLocaleLowerCase();
171
+ return matcher(getter, v => String(v).toLocaleLowerCase() > lower);
172
+ case ">=" /* FilterOp.Gte */:
173
+ return matcher(getter, v => v >= value);
174
+ case ">=*" /* FilterOp.GteInsesitive */:
175
+ lower = String(value).toLocaleLowerCase();
176
+ return matcher(getter, v => String(v).toLocaleLowerCase() >= lower);
177
+ case "<" /* FilterOp.Lt */:
178
+ return matcher(getter, v => v < value);
179
+ case "<*" /* FilterOp.LtInsesitive */:
180
+ lower = String(value).toLocaleLowerCase();
181
+ return matcher(getter, v => String(v).toLocaleLowerCase() < lower);
182
+ case "<=" /* FilterOp.Lte */:
183
+ return matcher(getter, v => v <= value);
184
+ case "<=*" /* FilterOp.LteInsesitive */:
185
+ lower = String(value).toLocaleLowerCase();
186
+ return matcher(getter, v => String(v).toLocaleLowerCase() <= lower);
187
+ case "%" /* FilterOp.Contains */:
188
+ lower = String(value);
189
+ return matcher(getter, v => (Array.isArray(v) ? v.includes(value) : String(v).includes(lower)));
190
+ case "%*" /* FilterOp.ContainsInsesitive */:
191
+ lower = String(value).toLocaleLowerCase();
192
+ return matcher(getter, v => String(v).toLocaleLowerCase().includes(lower));
193
+ case "^" /* FilterOp.StartsWith */:
194
+ lower = String(value);
195
+ return matcher(getter, v => String(v).startsWith(lower));
196
+ case "^*" /* FilterOp.StartsWithInsesitive */:
197
+ lower = String(value).toLocaleLowerCase();
198
+ return matcher(getter, v => String(v).toLocaleLowerCase().startsWith(lower));
199
+ case "$" /* FilterOp.EndsWith */:
200
+ lower = String(value);
201
+ return matcher(getter, v => String(v).endsWith(lower));
202
+ case "$*" /* FilterOp.EndsWithInsesitive */:
203
+ lower = String(value).toLocaleLowerCase();
204
+ return matcher(getter, v => String(v).toLocaleLowerCase().endsWith(lower));
205
+ case "~" /* FilterOp.Regexp */:
206
+ regex = value instanceof RegExp ? value : new RegExp(value, "msv");
207
+ return matcher(getter, v => regex.test(v));
208
+ case "~*" /* FilterOp.RegexpInsesitive */:
209
+ regex = value instanceof RegExp ? value : new RegExp(value, "msvi");
210
+ return matcher(getter, v => regex.test(v));
211
+ case "&" /* FilterOp.And */:
212
+ if (!Array.isArray(value)) {
213
+ throw new Error("Root '&' (AND) operator must have array type");
214
+ }
215
+ return and_(flattenDeep(value.map(v => {
216
+ if (isPlainObject(v)) {
217
+ return Object.entries(v).map(([op, opv]) => filterCompileOp(getter, op, opv));
218
+ }
219
+ else {
220
+ return filterCompileOp(getter, "==" /* FilterOp.Eq */, v);
221
+ }
222
+ })));
223
+ case "|" /* FilterOp.Or */:
224
+ if (!Array.isArray(value)) {
225
+ throw new Error("Root '|' (OR) operator must have array type");
226
+ }
227
+ return or_(flattenDeep(value.map(v => {
228
+ if (isPlainObject(v)) {
229
+ return Object.entries(v).map(([op, opv]) => filterCompileOp(getter, op, opv));
230
+ }
231
+ else {
232
+ return filterCompileOp(getter, "==" /* FilterOp.Eq */, v);
233
+ }
234
+ })));
235
+ }
236
+ throw new Error(`Unexpected operator: ${op}`);
237
+ }
238
+ function matcher(getter, predict) {
239
+ return obj => getter(obj).some(predict);
240
+ }
241
+ function and_(fns) {
242
+ if (fns.length === 0) {
243
+ return _ => true;
244
+ }
245
+ return item => {
246
+ for (const fn of fns) {
247
+ if (!fn(item)) {
248
+ return false;
249
+ }
250
+ }
251
+ return true;
252
+ };
253
+ }
254
+ function or_(fns) {
255
+ if (fns.length === 0) {
256
+ return _ => true;
257
+ }
258
+ return item => {
259
+ for (const fn of fns) {
260
+ if (fn(item)) {
261
+ return true;
262
+ }
263
+ }
264
+ return false;
265
+ };
266
+ }
267
+ export function filterMerge(...filters) {
268
+ let result = undefined;
269
+ for (const filter of filters) {
270
+ if (filter == null) {
271
+ continue;
272
+ }
273
+ if (result == null) {
274
+ result = deepClone(filter);
275
+ }
276
+ else {
277
+ for (const [k, v] of Object.entries(filter)) {
278
+ if (v === undefined) {
279
+ delete result[k];
280
+ continue;
281
+ }
282
+ result[k] = deepClone(v);
283
+ }
284
+ }
285
+ }
286
+ return result;
287
+ }
288
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter.js","sourceRoot":"","sources":["../../../../../packages/data/src/query/filter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAA;AAElD,OAAO,EAAe,SAAS,EAAE,aAAa,EAAgB,MAAM,gBAAgB,CAAA;AAGpF,OAAO,EAAc,iBAAiB,EAAE,MAAM,QAAQ,CAAA;AA6BtD,MAAM,CAAC,MAAM,SAAS,GAAkB;;;;;;;;;;;;;;;;;;;;;;;;;CAyBvC,CAAA;AAED,MAAM,UAAU,WAAW,CAAC,KAAU;IAClC,MAAM,GAAG,GAAG,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,SAAS,CAAe,CAAA;IACrE,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACjB,OAAO,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE;YAChB,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,EAAE,CAAC,EAAE,CAAA;QACnC,CAAC,CAAC,CAAA;IACN,CAAC;SAAM,CAAC;QACJ,OAAO,CAAC,EAAE,EAAE,wBAAa,EAAE,KAAK,EAAE,CAAC,CAAA;IACvC,CAAC;AACL,CAAC;AAoED;;;;;GAKG;AACH,MAAM,UAAU,QAAQ,CAAkB,OAAkB;IACxD,OAAO,cAAc,CAAI,OAAO,CAAC,CAAA;AACrC,CAAC;AAED,yBAAyB;AACzB,4DAA4D;AAC5D,qCAAqC;AACrC,sCAAsC;AACtC,4CAA4C;AAE5C,MAAM;AACN,cAAc;AACd,WAAW;AACX,oGAAoG;AACpG,0FAA0F;AAC1F,SAAS;AACT,MAAM;AACN,sFAAsF;AACtF,uCAAuC;AACvC,IAAI;AAEJ,gEAAgE;AAChE,+BAA+B;AAE/B,yDAAyD;AACzD,0BAA0B;AAC1B,iCAAiC;AACjC,2CAA2C;AAC3C,qFAAqF;AACrF,oBAAoB;AAEpB,+CAA+C;AAC/C,gDAAgD;AAChD,oBAAoB;AAEpB,sGAAsG;AACtG,wBAAwB;AAExB,gCAAgC;AAChC,2CAA2C;AAC3C,oFAAoF;AACpF,oBAAoB;AAEpB,8CAA8C;AAC9C,+CAA+C;AAC/C,oBAAoB;AAEpB,oGAAoG;AACpG,wBAAwB;AAExB,uBAAuB;AACvB,wDAAwD;AACxD,0CAA0C;AAC1C,6CAA6C;AAC7C,2DAA2D;AAC3D,4DAA4D;AAC5D,gCAAgC;AAEhC,kFAAkF;AAClF,yFAAyF;AACzF,gCAAgC;AAChC,oCAAoC;AAEpC,4CAA4C;AAC5C,0DAA0D;AAC1D,2DAA2D;AAC3D,gCAAgC;AAEhC,gFAAgF;AAChF,yFAAyF;AACzF,gCAAgC;AAChC,oCAAoC;AAEpC,mCAAmC;AACnC,2DAA2D;AAC3D,4DAA4D;AAC5D,gCAAgC;AAEhC,4EAA4E;AAC5E,wBAAwB;AACxB,oBAAoB;AACpB,YAAY;AACZ,QAAQ;AAER,oBAAoB;AACpB,IAAI;AAEJ,SAAS,cAAc,CAAkB,OAAkB;IACvD,IAAI,MAAkB,CAAA;IACtB,MAAM,MAAM,GAAkB,EAAE,CAAA;IAChC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QACjD,QAAQ,GAAG,EAAE,CAAC;YACV;gBACI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;oBACxB,MAAM,IAAI,KAAK,CAAC,8CAA8C,CAAC,CAAA;gBACnE,CAAC;gBACD,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,GAAG,KAAK,CAAC,GAAG,CAAC,CAAA,cAAiB,CAAA,CAAC,CAAC,CAAA;gBAChE,MAAK;YAET;gBACI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;oBACxB,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAA;gBAClE,CAAC;gBACD,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA,cAAiB,CAAA,CAAC,CAAC,CAAC,CAAA;gBAC9C,MAAK;YAET;gBACI,MAAM,GAAG,iBAAiB,CAAC,GAAG,CAAC,CAAA;gBAC/B,IAAI,aAAa,CAAC,KAAK,CAAC,EAAE,CAAC;oBACvB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,MAAM,EAAE,EAAS,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAA;gBACxG,CAAC;qBAAM,CAAC;oBACJ,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,MAAM,0BAAe,KAAK,CAAC,CAAC,CAAA;gBAC5D,CAAC;gBAED,MAAK;QACb,CAAC;IACL,CAAC;IACD,OAAO,IAAI,CAAC,MAAM,CAAC,CAAA;AACvB,CAAC;AAED,SAAS,eAAe,CAAC,MAAkB,EAAE,EAAY,EAAE,KAAU;IACjE,IAAI,KAAa,CAAA;IACjB,IAAI,KAAa,CAAA;IACjB,QAAQ,EAAE,EAAE,CAAC;QACT;YACI,kCAAkC;YAClC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,KAAK,CAAC,CAAA;QAE3C;YACI,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,KAAK,CAAC,CAAA;QAE5C;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,KAAK,KAAK,CAAC,CAAA;QAExE;YACI,kCAAkC;YAClC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,KAAK,CAAC,CAAA;QAE3C;YACI,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,KAAK,CAAC,CAAA;QAE5C;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,KAAK,KAAK,CAAC,CAAA;QAExE;YACI,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,KAAK,CAAC,CAAA;QAE1C;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,GAAG,KAAK,CAAC,CAAA;QAEtE;YACI,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,KAAK,CAAC,CAAA;QAE3C;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,IAAI,KAAK,CAAC,CAAA;QAEvE;YACI,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,KAAK,CAAC,CAAA;QAE1C;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,GAAG,KAAK,CAAC,CAAA;QAEtE;YACI,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,KAAK,CAAC,CAAA;QAE3C;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,IAAI,KAAK,CAAC,CAAA;QAEvE;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAA;YACrB,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAA;QAEnG;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAA;QAE9E;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAA;YACrB,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAA;QAE5D;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAA;QAEhF;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAA;YACrB,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAA;QAE1D;YACI,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAA;YACzC,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,iBAAiB,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAA;QAE9E;YACI,KAAK,GAAG,KAAK,YAAY,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,KAAK,EAAE,KAAK,CAAC,CAAA;YAClE,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QAE9C;YACI,KAAK,GAAG,KAAK,YAAY,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,CAAA;YACnE,OAAO,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QAE9C;YACI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;gBACxB,MAAM,IAAI,KAAK,CAAC,8CAA8C,CAAC,CAAA;YACnE,CAAC;YACD,OAAO,IAAI,CACP,WAAW,CACP,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBACV,IAAI,aAAa,CAAC,CAAC,CAAC,EAAE,CAAC;oBACnB,OAAO,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,MAAM,EAAE,EAAS,EAAE,GAAG,CAAC,CAAC,CAAA;gBACxF,CAAC;qBAAM,CAAC;oBACJ,OAAO,eAAe,CAAC,MAAM,0BAAe,CAAC,CAAC,CAAA;gBAClD,CAAC;YACL,CAAC,CAAC,CACL,CACJ,CAAA;QAEL;YACI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;gBACxB,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAA;YAClE,CAAC;YACD,OAAO,GAAG,CACN,WAAW,CACP,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBACV,IAAI,aAAa,CAAC,CAAC,CAAC,EAAE,CAAC;oBACnB,OAAO,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,MAAM,EAAE,EAAS,EAAE,GAAG,CAAC,CAAC,CAAA;gBACxF,CAAC;qBAAM,CAAC;oBACJ,OAAO,eAAe,CAAC,MAAM,0BAAe,CAAC,CAAC,CAAA;gBAClD,CAAC;YACL,CAAC,CAAC,CACL,CACJ,CAAA;IACT,CAAC;IAED,MAAM,IAAI,KAAK,CAAC,wBAAwB,EAAE,EAAE,CAAC,CAAA;AACjD,CAAC;AAED,SAAS,OAAO,CAAC,MAAkB,EAAE,OAAgC;IACjE,OAAO,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;AAC3C,CAAC;AAED,SAAS,IAAI,CAAC,GAAe;IACzB,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACnB,OAAO,CAAC,CAAC,EAAE,CAAC,IAAI,CAAA;IACpB,CAAC;IAED,OAAO,IAAI,CAAC,EAAE;QACV,KAAK,MAAM,EAAE,IAAI,GAAG,EAAE,CAAC;YACnB,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC;gBACZ,OAAO,KAAK,CAAA;YAChB,CAAC;QACL,CAAC;QACD,OAAO,IAAI,CAAA;IACf,CAAC,CAAA;AACL,CAAC;AAED,SAAS,GAAG,CAAC,GAAe;IACxB,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACnB,OAAO,CAAC,CAAC,EAAE,CAAC,IAAI,CAAA;IACpB,CAAC;IAED,OAAO,IAAI,CAAC,EAAE;QACV,KAAK,MAAM,EAAE,IAAI,GAAG,EAAE,CAAC;YACnB,IAAI,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC;gBACX,OAAO,IAAI,CAAA;YACf,CAAC;QACL,CAAC;QACD,OAAO,KAAK,CAAA;IAChB,CAAC,CAAA;AACL,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,GAAG,OAAc;IACzC,IAAI,MAAM,GAAuC,SAAS,CAAA;IAE1D,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;QAC3B,IAAI,MAAM,IAAI,IAAI,EAAE,CAAC;YACjB,SAAQ;QACZ,CAAC;QACD,IAAI,MAAM,IAAI,IAAI,EAAE,CAAC;YACjB,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAA;QAC9B,CAAC;aAAM,CAAC;YACJ,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC1C,IAAI,CAAC,KAAK,SAAS,EAAE,CAAC;oBAClB,OAAO,MAAM,CAAC,CAAC,CAAC,CAAA;oBAChB,SAAQ;gBACZ,CAAC;gBAED,MAAM,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAA;YAC5B,CAAC;QACL,CAAC;IACL,CAAC;IAED,OAAO,MAAa,CAAA;AACxB,CAAC","sourcesContent":["import { flattenDeep, intersection } from \"lodash\"\n\nimport { AsPrimitive, deepClone, isPlainObject, MaxRecursion } from \"@ngutil/common\"\n\nimport { Model } from \"../model\"\nimport { PathGetter, pathGetterCompile } from \"./path\"\n\nexport const enum FilterOp {\n    Eq = \"==\",\n    EqStrict = \"===\",\n    EqInsesitive = \"==*\",\n    Neq = \"!=\",\n    NeqStrict = \"!==\",\n    NeqInsesitive = \"!=*\",\n    Gt = \">\",\n    GtInsesitive = \">*\",\n    Gte = \">=\",\n    GteInsesitive = \">=*\",\n    Lt = \"<\",\n    LtInsesitive = \"<*\",\n    Lte = \"<=\",\n    LteInsesitive = \"<=*\",\n    Contains = \"%\",\n    ContainsInsesitive = \"%*\",\n    StartsWith = \"^\",\n    StartsWithInsesitive = \"^*\",\n    EndsWith = \"$\",\n    EndsWithInsesitive = \"$*\",\n    Regexp = \"~\",\n    RegexpInsesitive = \"~*\",\n    Or = \"|\",\n    And = \"&\"\n}\n\nexport const OPERATORS: Array<string> = [\n    FilterOp.Eq,\n    FilterOp.EqStrict,\n    FilterOp.EqInsesitive,\n    FilterOp.Neq,\n    FilterOp.NeqStrict,\n    FilterOp.NeqInsesitive,\n    FilterOp.Gt,\n    FilterOp.GtInsesitive,\n    FilterOp.Gte,\n    FilterOp.GteInsesitive,\n    FilterOp.Lt,\n    FilterOp.LtInsesitive,\n    FilterOp.Lte,\n    FilterOp.LteInsesitive,\n    FilterOp.Contains,\n    FilterOp.ContainsInsesitive,\n    FilterOp.StartsWith,\n    FilterOp.StartsWithInsesitive,\n    FilterOp.EndsWith,\n    FilterOp.EndsWithInsesitive,\n    FilterOp.Regexp,\n    FilterOp.RegexpInsesitive,\n    FilterOp.Or,\n    FilterOp.And\n]\n\nexport function asOperators(value: any): Array<{ op: FilterOp; value: any }> {\n    const ops = intersection(Object.keys(value), OPERATORS) as FilterOp[]\n    if (ops.length > 0) {\n        return ops.map(op => {\n            return { op, value: value[op] }\n        })\n    } else {\n        return [{ op: FilterOp.Eq, value }]\n    }\n}\n\n// export type Filter<T extends Model> = _Filter<Flatten<T>, [], MaxRecursion>\n// TODO: fix recursion\nexport type Filter<T extends Model> = { [key: string]: any }\n\ntype _Filter<F, P extends number[], D extends number> = P[\"length\"] extends D ? never : _ObjectFilter<F, P, D>\n\ntype _ObjectFilter<F, P extends number[], D extends number> = P[\"length\"] extends D\n    ? never\n    : { [K in keyof F]?: Operators<F[K], P, D> }\n\ntype RootOr<F, P extends number[], D extends number> = P[\"length\"] extends D\n    ? never\n    : { [FilterOp.Or]: Array<_Filter<F, [...P, 0], D>> | undefined }\n\ntype RootAnd<F, P extends number[], D extends number> = P[\"length\"] extends D\n    ? never\n    : { [FilterOp.And]: Array<_Filter<F, [...P, 0], D>> | undefined }\n\nexport type Operators<T, P extends number[], D extends number = MaxRecursion> = P[\"length\"] extends D\n    ? never\n    : SimpleOperators<T>\n\ntype LocalOr<T, P extends number[], D extends number> = P[\"length\"] extends D\n    ? never\n    : { [FilterOp.Or]: Array<Operators<T, [...P, 0], D>> | undefined }\n\ntype LocalAnd<T, P extends number[], D extends number> = P[\"length\"] extends D\n    ? never\n    : { [FilterOp.And]: Array<Operators<T, [...P, 0], D>> | undefined }\n\ntype SimpleOperators<T> =\n    | EqStrict<T>\n    | NeqStrict<T>\n    | Eq<T>\n    | Neq<T>\n    | Gt<T>\n    | Gte<T>\n    | Lt<T>\n    | Lte<T>\n    | Contains<T>\n    | StartsWith<T>\n    | EndsWith<T>\n    | Regexp<T>\n    | T\n    | undefined\n    | null\n\ntype EqStrict<T> = Operator<T, T, FilterOp.EqStrict, never, null>\ntype NeqStrict<T> = Operator<T, T, FilterOp.NeqStrict, never, null>\ntype Eq<T> = Operator<T, T | AsPrimitive<T>, FilterOp.Eq, FilterOp.EqInsesitive, null>\ntype Neq<T> = Operator<T, T | AsPrimitive<T>, FilterOp.Neq, FilterOp.NeqInsesitive, null>\ntype Gt<T> = Operator<T, T | AsPrimitive<T>, FilterOp.Gt, FilterOp.GtInsesitive>\ntype Gte<T> = Operator<T, T | AsPrimitive<T>, FilterOp.Gte, FilterOp.GteInsesitive>\ntype Lt<T> = Operator<T, T | AsPrimitive<T>, FilterOp.Lt, FilterOp.LtInsesitive>\ntype Lte<T> = Operator<T, T | AsPrimitive<T>, FilterOp.Lte, FilterOp.LteInsesitive>\ntype Contains<T> = Operator<T, string, FilterOp.Contains, FilterOp.ContainsInsesitive>\ntype StartsWith<T> = Operator<T, string, FilterOp.StartsWith, FilterOp.StartsWithInsesitive>\ntype EndsWith<T> = Operator<T, string, FilterOp.EndsWith, FilterOp.EndsWithInsesitive>\ntype Regexp<T> = Operator<T, string, FilterOp.Regexp, FilterOp.RegexpInsesitive, RegExp>\n\ntype Operator<T, R, OP extends string, OPI extends string = never, ET = never> =\n    | (T extends R ? { [k in OP]: T | ET | R } : never)\n    | (T extends string ? (OPI extends never ? never : { [k in OPI]: string }) : never)\n\nexport type FilterFn<T = any> = (item: T) => boolean\n\n/**\n * @example\n * ```ts\n * items.filter(filterBy({id: 42}))\n * ```\n */\nexport function filterBy<T extends Model>(filters: Filter<T>): FilterFn<T> {\n    return _filterCompile<T>(filters)\n}\n\n// TODO: Normalize filter\n// type _Filter = { path: string; op: FilterOp; value: any }\n// type _Or = { \"|\": Array<_Filter> }\n// type _And = { \"&\": Array<_Filter> }\n// export type NormalizedFilter = _Or | _And\n\n// /**\n//  * @example\n//  * ```ts\n//  * normalizeFilter({id: 2, name: {\"=*\": \"AnyName\"}}) -> {id: {\"==\": 2}, name: {\"=*\": \"AnyName\"}}}\n//  * normalizeFilter({id: {\">\": 0, \"<\": 10}}) -> {\"&\": [{id: {\">\": 0}}, {id: {\"<\": 10}}]}\n//  * ```\n//  */\n// export function normalizeFilter<T extends Model>(filters: Filters<T>): Filters<T> {\n//     return _normalizeFilter(filters)\n// }\n\n// function _normalizeFilter(filters: any, path?: string): any {\n//     const result = {} as any\n\n//     for (const [path, v] of Object.entries(filters)) {\n//         switch (path) {\n//             case FilterOp.And:\n//                 if (!Array.isArray(v)) {\n//                     throw new Error(\"The '&' (AND) operator must have array type\")\n//                 }\n\n//                 if (!result[FilterOp.And]) {\n//                     result[FilterOp.And] = []\n//                 }\n\n//                 result[FilterOp.And] = result[FilterOp.And].concat(v.map(f => _normalizeFilter(f)))\n//                 break\n\n//             case FilterOp.Or:\n//                 if (!Array.isArray(v)) {\n//                     throw new Error(\"The '|' (OR) operator must have array type\")\n//                 }\n\n//                 if (!result[FilterOp.Or]) {\n//                     result[FilterOp.Or] = []\n//                 }\n\n//                 result[FilterOp.Or] = result[FilterOp.Or].concat(v.map(f => _normalizeFilter(f)))\n//                 break\n\n//             default:\n//                 for (const entry of asOperators(v)) {\n//                     switch (entry.op) {\n//                         case FilterOp.And:\n//                             if (!result[FilterOp.And]) {\n//                                 result[FilterOp.And] = []\n//                             }\n\n//                             result[FilterOp.And] = result[FilterOp.And].concat(\n//                                 entry.value.map((v: any) => _normalizeFilter(v, path))\n//                             )\n//                             break\n\n//                         case FilterOp.Or:\n//                             if (!result[FilterOp.Or]) {\n//                                 result[FilterOp.Or] = []\n//                             }\n\n//                             result[FilterOp.Or] = result[FilterOp.Or].concat(\n//                                 entry.value.map((v: any) => _normalizeFilter(v, path))\n//                             )\n//                             break\n\n//                         default:\n//                             if (!result[FilterOp.And]) {\n//                                 result[FilterOp.And] = []\n//                             }\n\n//                             result[FilterOp.And].push({ path, ...entry })\n//                     }\n//                 }\n//         }\n//     }\n\n//     return result\n// }\n\nfunction _filterCompile<T extends Model>(filters: Filter<T>): FilterFn<T> {\n    let getter: PathGetter\n    const result: FilterFn<T>[] = []\n    for (const [pth, value] of Object.entries(filters)) {\n        switch (pth) {\n            case FilterOp.And:\n                if (!Array.isArray(value)) {\n                    throw new Error(\"Root '&' (AND) operator must have array type\")\n                }\n                result.splice(result.length, 0, ...value.map(_filterCompile<T>))\n                break\n\n            case FilterOp.Or:\n                if (!Array.isArray(value)) {\n                    throw new Error(\"Root '|' (OR) operator must have array type\")\n                }\n                result.push(or_(value.map(_filterCompile<T>)))\n                break\n\n            default:\n                getter = pathGetterCompile(pth)\n                if (isPlainObject(value)) {\n                    result.push(and_(Object.entries(value).map(([op, opv]) => filterCompileOp(getter, op as any, opv))))\n                } else {\n                    result.push(filterCompileOp(getter, FilterOp.Eq, value))\n                }\n\n                break\n        }\n    }\n    return and_(result)\n}\n\nfunction filterCompileOp(getter: PathGetter, op: FilterOp, value: any): FilterFn {\n    let lower: string\n    let regex: RegExp\n    switch (op) {\n        case FilterOp.Eq:\n            // eslint-disable-next-line eqeqeq\n            return matcher(getter, v => v == value)\n\n        case FilterOp.EqStrict:\n            return matcher(getter, v => v === value)\n\n        case FilterOp.EqInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase() === lower)\n\n        case FilterOp.Neq:\n            // eslint-disable-next-line eqeqeq\n            return matcher(getter, v => v != value)\n\n        case FilterOp.NeqStrict:\n            return matcher(getter, v => v !== value)\n\n        case FilterOp.NeqInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase() !== lower)\n\n        case FilterOp.Gt:\n            return matcher(getter, v => v > value)\n\n        case FilterOp.GtInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase() > lower)\n\n        case FilterOp.Gte:\n            return matcher(getter, v => v >= value)\n\n        case FilterOp.GteInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase() >= lower)\n\n        case FilterOp.Lt:\n            return matcher(getter, v => v < value)\n\n        case FilterOp.LtInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase() < lower)\n\n        case FilterOp.Lte:\n            return matcher(getter, v => v <= value)\n\n        case FilterOp.LteInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase() <= lower)\n\n        case FilterOp.Contains:\n            lower = String(value)\n            return matcher(getter, v => (Array.isArray(v) ? v.includes(value) : String(v).includes(lower)))\n\n        case FilterOp.ContainsInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase().includes(lower))\n\n        case FilterOp.StartsWith:\n            lower = String(value)\n            return matcher(getter, v => String(v).startsWith(lower))\n\n        case FilterOp.StartsWithInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase().startsWith(lower))\n\n        case FilterOp.EndsWith:\n            lower = String(value)\n            return matcher(getter, v => String(v).endsWith(lower))\n\n        case FilterOp.EndsWithInsesitive:\n            lower = String(value).toLocaleLowerCase()\n            return matcher(getter, v => String(v).toLocaleLowerCase().endsWith(lower))\n\n        case FilterOp.Regexp:\n            regex = value instanceof RegExp ? value : new RegExp(value, \"msv\")\n            return matcher(getter, v => regex.test(v))\n\n        case FilterOp.RegexpInsesitive:\n            regex = value instanceof RegExp ? value : new RegExp(value, \"msvi\")\n            return matcher(getter, v => regex.test(v))\n\n        case FilterOp.And:\n            if (!Array.isArray(value)) {\n                throw new Error(\"Root '&' (AND) operator must have array type\")\n            }\n            return and_(\n                flattenDeep(\n                    value.map(v => {\n                        if (isPlainObject(v)) {\n                            return Object.entries(v).map(([op, opv]) => filterCompileOp(getter, op as any, opv))\n                        } else {\n                            return filterCompileOp(getter, FilterOp.Eq, v)\n                        }\n                    })\n                )\n            )\n\n        case FilterOp.Or:\n            if (!Array.isArray(value)) {\n                throw new Error(\"Root '|' (OR) operator must have array type\")\n            }\n            return or_(\n                flattenDeep(\n                    value.map(v => {\n                        if (isPlainObject(v)) {\n                            return Object.entries(v).map(([op, opv]) => filterCompileOp(getter, op as any, opv))\n                        } else {\n                            return filterCompileOp(getter, FilterOp.Eq, v)\n                        }\n                    })\n                )\n            )\n    }\n\n    throw new Error(`Unexpected operator: ${op}`)\n}\n\nfunction matcher(getter: PathGetter, predict: (value: any) => boolean): FilterFn {\n    return obj => getter(obj).some(predict)\n}\n\nfunction and_(fns: FilterFn[]): FilterFn {\n    if (fns.length === 0) {\n        return _ => true\n    }\n\n    return item => {\n        for (const fn of fns) {\n            if (!fn(item)) {\n                return false\n            }\n        }\n        return true\n    }\n}\n\nfunction or_(fns: FilterFn[]): FilterFn {\n    if (fns.length === 0) {\n        return _ => true\n    }\n\n    return item => {\n        for (const fn of fns) {\n            if (fn(item)) {\n                return true\n            }\n        }\n        return false\n    }\n}\n\nexport function filterMerge(...filters: any[]): any | undefined {\n    let result: { [key: string]: any } | undefined = undefined\n\n    for (const filter of filters) {\n        if (filter == null) {\n            continue\n        }\n        if (result == null) {\n            result = deepClone(filter)\n        } else {\n            for (const [k, v] of Object.entries(filter)) {\n                if (v === undefined) {\n                    delete result[k]\n                    continue\n                }\n\n                result[k] = deepClone(v)\n            }\n        }\n    }\n\n    return result as any\n}\n"]}