cx 24.3.7 → 24.3.8
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/dist/data.js +46 -15
- package/dist/manifest.js +672 -672
- package/dist/ui.js +11 -9
- package/dist/widgets.js +10 -7
- package/package.json +1 -1
- package/src/data/Grouper.js +144 -120
- package/src/data/Grouper.spec.js +57 -42
- package/src/ui/adapter/GroupAdapter.js +141 -138
- package/src/widgets/grid/Grid.js +10 -6
package/dist/ui.js
CHANGED
|
@@ -4149,15 +4149,7 @@ var GroupAdapter = /*#__PURE__*/ (function (_ArrayAdapter) {
|
|
|
4149
4149
|
results.forEach(function (gr) {
|
|
4150
4150
|
var _data;
|
|
4151
4151
|
keys.push(gr.key);
|
|
4152
|
-
var key = keys
|
|
4153
|
-
.map(function (key) {
|
|
4154
|
-
return Object.keys(key)
|
|
4155
|
-
.map(function (k) {
|
|
4156
|
-
return key[k];
|
|
4157
|
-
})
|
|
4158
|
-
.join(":");
|
|
4159
|
-
})
|
|
4160
|
-
.join("|");
|
|
4152
|
+
var key = keys.map(serializeKey).join("|");
|
|
4161
4153
|
var $group = _extends({}, gr.key, gr.aggregates, {
|
|
4162
4154
|
$name: gr.name,
|
|
4163
4155
|
$level: inverseLevel,
|
|
@@ -4243,6 +4235,16 @@ var GroupAdapter = /*#__PURE__*/ (function (_ArrayAdapter) {
|
|
|
4243
4235
|
return GroupAdapter;
|
|
4244
4236
|
})(ArrayAdapter);
|
|
4245
4237
|
GroupAdapter.prototype.groupName = "$group";
|
|
4238
|
+
function serializeKey(data) {
|
|
4239
|
+
var _data$toString;
|
|
4240
|
+
if (isObject(data))
|
|
4241
|
+
return Object.keys(data)
|
|
4242
|
+
.map(function (k) {
|
|
4243
|
+
return serializeKey(data[k]);
|
|
4244
|
+
})
|
|
4245
|
+
.join(":");
|
|
4246
|
+
return (_data$toString = data == null ? void 0 : data.toString()) != null ? _data$toString : "";
|
|
4247
|
+
}
|
|
4246
4248
|
|
|
4247
4249
|
var TreeAdapter = /*#__PURE__*/ (function (_ArrayAdapter) {
|
|
4248
4250
|
_inheritsLoose(TreeAdapter, _ArrayAdapter);
|
package/dist/widgets.js
CHANGED
|
@@ -16190,9 +16190,9 @@ var Grid = /*#__PURE__*/ (function (_Container) {
|
|
|
16190
16190
|
colSpan++;
|
|
16191
16191
|
}
|
|
16192
16192
|
if (colSpan > 1) skip = colSpan - 1;
|
|
16193
|
-
} else if (c.aggregate && c.
|
|
16193
|
+
} else if (c.aggregate && c.aggregateAliasGetter && c.caption !== false) {
|
|
16194
16194
|
empty = false;
|
|
16195
|
-
v =
|
|
16195
|
+
v = c.aggregateAliasGetter(group);
|
|
16196
16196
|
if (isString(ci.data.format)) v = Format.value(v, ci.data.format);
|
|
16197
16197
|
}
|
|
16198
16198
|
if (cls) cls += " ";
|
|
@@ -16282,9 +16282,9 @@ var Grid = /*#__PURE__*/ (function (_Container) {
|
|
|
16282
16282
|
colSpan++;
|
|
16283
16283
|
}
|
|
16284
16284
|
if (colSpan > 1) skip = colSpan - 1;
|
|
16285
|
-
} else if (c.aggregate && c.
|
|
16285
|
+
} else if (c.aggregate && c.aggregateAliasGetter && c.footer !== false) {
|
|
16286
16286
|
empty = false;
|
|
16287
|
-
v =
|
|
16287
|
+
v = c.aggregateAliasGetter(group);
|
|
16288
16288
|
if (isString(ci.data.format)) v = Format.value(v, ci.data.format);
|
|
16289
16289
|
}
|
|
16290
16290
|
if (cls) cls += " ";
|
|
@@ -16921,20 +16921,22 @@ var GridComponent = /*#__PURE__*/ (function (_VDOM$Component) {
|
|
|
16921
16921
|
},
|
|
16922
16922
|
"dropzone",
|
|
16923
16923
|
);
|
|
16924
|
+
var dataRecordClass = CSS.element(baseClass, "data");
|
|
16924
16925
|
var isDataRecord = widget.buffered
|
|
16925
16926
|
? function (item) {
|
|
16926
16927
|
var _item$props;
|
|
16927
16928
|
return (
|
|
16928
|
-
(
|
|
16929
|
+
(item == null ||
|
|
16930
|
+
(_item$props = item.props) == null ||
|
|
16929
16931
|
(_item$props = _item$props.instance) == null ||
|
|
16930
16932
|
(_item$props = _item$props.data) == null
|
|
16931
16933
|
? void 0
|
|
16932
|
-
: _item$props["class"]) ==
|
|
16934
|
+
: _item$props["class"]) == dataRecordClass
|
|
16933
16935
|
);
|
|
16934
16936
|
}
|
|
16935
16937
|
: function (item) {
|
|
16936
16938
|
var _item$props2;
|
|
16937
|
-
return (_item$props2 = item.props) == null || (_item$props2 = _item$props2.record) == null
|
|
16939
|
+
return item == null || (_item$props2 = item.props) == null || (_item$props2 = _item$props2.record) == null
|
|
16938
16940
|
? void 0
|
|
16939
16941
|
: _item$props2.type;
|
|
16940
16942
|
};
|
|
@@ -18408,6 +18410,7 @@ var GridColumnHeader = /*#__PURE__*/ (function (_Widget) {
|
|
|
18408
18410
|
};
|
|
18409
18411
|
if (!this.aggregateField && this.field) this.aggregateField = this.field;
|
|
18410
18412
|
if (!this.aggregateAlias) this.aggregateAlias = this.aggregateField;
|
|
18413
|
+
if (this.aggregateAlias) this.aggregateAliasGetter = Binding.get(this.aggregateAlias).value;
|
|
18411
18414
|
if (this.footer && isSelector(this.footer))
|
|
18412
18415
|
this.footer = {
|
|
18413
18416
|
value: this.footer,
|
package/package.json
CHANGED
package/src/data/Grouper.js
CHANGED
|
@@ -1,120 +1,144 @@
|
|
|
1
|
-
import {getSelector} from
|
|
2
|
-
import {AggregateFunction} from
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
})
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
}
|
|
120
|
-
|
|
1
|
+
import { getSelector } from "./getSelector";
|
|
2
|
+
import { AggregateFunction } from "./AggregateFunction";
|
|
3
|
+
import { Binding } from "./Binding";
|
|
4
|
+
|
|
5
|
+
/*
|
|
6
|
+
'column': {
|
|
7
|
+
index: 0,
|
|
8
|
+
sort: 'asc',
|
|
9
|
+
group: true
|
|
10
|
+
aggregate: 'count'
|
|
11
|
+
}
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
export class Grouper {
|
|
15
|
+
constructor(key, aggregates, dataGetter, nameGetter) {
|
|
16
|
+
this.keys = Object.keys(key).map((keyField) => {
|
|
17
|
+
let isSimpleField = keyField.indexOf(".") === -1;
|
|
18
|
+
let keySetter;
|
|
19
|
+
if (isSimpleField) {
|
|
20
|
+
// use simple field setter wherever possible
|
|
21
|
+
keySetter = function set(result, value) {
|
|
22
|
+
result[keyField] = value;
|
|
23
|
+
return result;
|
|
24
|
+
};
|
|
25
|
+
} else {
|
|
26
|
+
// for complex paths, use deep setter
|
|
27
|
+
let binding = Binding.get(keyField);
|
|
28
|
+
keySetter = (result, value) => binding.set(result, value);
|
|
29
|
+
}
|
|
30
|
+
return {
|
|
31
|
+
name: keyField,
|
|
32
|
+
keySetter,
|
|
33
|
+
value: getSelector(key[keyField]),
|
|
34
|
+
};
|
|
35
|
+
});
|
|
36
|
+
if (nameGetter) this.nameGetter = getSelector(nameGetter);
|
|
37
|
+
this.dataGetter = dataGetter || ((x) => x);
|
|
38
|
+
this.aggregates =
|
|
39
|
+
aggregates &&
|
|
40
|
+
transformValues(aggregates, (prop) => {
|
|
41
|
+
if (!AggregateFunction[prop.type]) throw new Error(`Unknown aggregate function '${prop.type}'.`);
|
|
42
|
+
|
|
43
|
+
return {
|
|
44
|
+
value: getSelector(prop.value),
|
|
45
|
+
weight: getSelector(prop.weight ?? 1),
|
|
46
|
+
type: prop.type,
|
|
47
|
+
};
|
|
48
|
+
});
|
|
49
|
+
this.reset();
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
reset() {
|
|
53
|
+
this.groups = this.initGroup(this.keys.length == 0);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
initGroup(leaf) {
|
|
57
|
+
if (!leaf) return {};
|
|
58
|
+
|
|
59
|
+
return {
|
|
60
|
+
records: [],
|
|
61
|
+
indexes: [],
|
|
62
|
+
aggregates:
|
|
63
|
+
this.aggregates &&
|
|
64
|
+
transformValues(this.aggregates, (prop) => {
|
|
65
|
+
let f = AggregateFunction[prop.type];
|
|
66
|
+
return {
|
|
67
|
+
processor: f(),
|
|
68
|
+
value: prop.value,
|
|
69
|
+
weight: prop.weight,
|
|
70
|
+
};
|
|
71
|
+
}),
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
process(record, index) {
|
|
76
|
+
let data = this.dataGetter(record);
|
|
77
|
+
let key = this.keys.map((k) => k.value(data));
|
|
78
|
+
let g = this.groups;
|
|
79
|
+
for (let i = 0; i < key.length; i++) {
|
|
80
|
+
let sg = g[key[i]];
|
|
81
|
+
if (!sg) {
|
|
82
|
+
sg = g[key[i]] = this.initGroup(i + 1 == key.length);
|
|
83
|
+
if (this.nameGetter) sg.name = this.nameGetter(data);
|
|
84
|
+
}
|
|
85
|
+
g = sg;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
g.records.push(record);
|
|
89
|
+
g.indexes.push(index);
|
|
90
|
+
|
|
91
|
+
if (g.aggregates) {
|
|
92
|
+
for (let k in g.aggregates)
|
|
93
|
+
g.aggregates[k].processor.process(g.aggregates[k].value(data), g.aggregates[k].weight(data));
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
processAll(records, indexes) {
|
|
98
|
+
if (indexes) {
|
|
99
|
+
for (let i = 0; i < records.length; i++) this.process(records[i], indexes[i]);
|
|
100
|
+
} else records.forEach((r, i) => this.process(r, i));
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
report(g, path, level, results) {
|
|
104
|
+
if (level == this.keys.length) {
|
|
105
|
+
let key = {};
|
|
106
|
+
this.keys.forEach((k, i) => {
|
|
107
|
+
key = k.keySetter(key, path[i]);
|
|
108
|
+
});
|
|
109
|
+
results.push({
|
|
110
|
+
key: key,
|
|
111
|
+
name: g.name,
|
|
112
|
+
records: g.records,
|
|
113
|
+
indexes: g.indexes,
|
|
114
|
+
aggregates: resolveKeyPaths(transformValues(g.aggregates, (p) => p.processor.getResult())),
|
|
115
|
+
});
|
|
116
|
+
} else {
|
|
117
|
+
Object.keys(g).forEach((k) => this.report(g[k], [...path, k], level + 1, results));
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
getResults() {
|
|
122
|
+
let g = this.groups;
|
|
123
|
+
let results = [];
|
|
124
|
+
this.report(g, [], 0, results);
|
|
125
|
+
return results;
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// transform object values using a function
|
|
130
|
+
function transformValues(o, f) {
|
|
131
|
+
let res = {};
|
|
132
|
+
for (let k in o) res[k] = f(o[k], k);
|
|
133
|
+
return res;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// transform keys like 'a.b.c' into nested objects
|
|
137
|
+
function resolveKeyPaths(o) {
|
|
138
|
+
let res = {};
|
|
139
|
+
for (let k in o) {
|
|
140
|
+
if (k.indexOf(".") > 0) res = Binding.get(k).set(res, o[k]);
|
|
141
|
+
else res[k] = o[k];
|
|
142
|
+
}
|
|
143
|
+
return res;
|
|
144
|
+
}
|
package/src/data/Grouper.spec.js
CHANGED
|
@@ -1,42 +1,57 @@
|
|
|
1
|
-
|
|
2
|
-
import assert from "assert";
|
|
3
|
-
|
|
4
|
-
describe("Grouper", function () {
|
|
5
|
-
describe("single grouping", function () {
|
|
6
|
-
it("should work", function () {
|
|
7
|
-
|
|
8
|
-
{ name: "John", age: 12 },
|
|
9
|
-
{ name: "Jane", age: 12 },
|
|
10
|
-
];
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
grouper.processAll(data);
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
// console.log(results);
|
|
17
|
-
assert.equal(results.length, 2);
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
];
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
}
|
|
1
|
+
let Grouper = require("./Grouper").Grouper;
|
|
2
|
+
import assert from "assert";
|
|
3
|
+
|
|
4
|
+
describe("Grouper", function () {
|
|
5
|
+
describe("single grouping", function () {
|
|
6
|
+
it("should work", function () {
|
|
7
|
+
let data = [
|
|
8
|
+
{ name: "John", age: 12 },
|
|
9
|
+
{ name: "Jane", age: 12 },
|
|
10
|
+
];
|
|
11
|
+
|
|
12
|
+
let grouper = new Grouper({ name: { bind: "name" } });
|
|
13
|
+
grouper.processAll(data);
|
|
14
|
+
|
|
15
|
+
let results = grouper.getResults();
|
|
16
|
+
// console.log(results);
|
|
17
|
+
assert.equal(results.length, 2);
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it("keys can have nested properties", function () {
|
|
21
|
+
let data = [
|
|
22
|
+
{ name: "John", age: 12 },
|
|
23
|
+
{ name: "Jane", age: 12 },
|
|
24
|
+
];
|
|
25
|
+
|
|
26
|
+
let grouper = new Grouper({ "person.name": { bind: "name" } });
|
|
27
|
+
grouper.processAll(data);
|
|
28
|
+
|
|
29
|
+
let results = grouper.getResults();
|
|
30
|
+
assert.equal(results.length, 2);
|
|
31
|
+
assert.equal(results[0].key.person?.name, "John");
|
|
32
|
+
assert.equal(results[1].key.person?.name, "Jane");
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
describe("multi grouping", function () {
|
|
37
|
+
it("should work", function () {
|
|
38
|
+
let data = [
|
|
39
|
+
{ name: "John", age: 12 },
|
|
40
|
+
{ name: "John", age: 12 },
|
|
41
|
+
{ name: "John", age: 13 },
|
|
42
|
+
{ name: "John", age: 14 },
|
|
43
|
+
{ name: "Jane", age: 12 },
|
|
44
|
+
{ name: "Jane", age: 13 },
|
|
45
|
+
{ name: "Jane", age: 14 },
|
|
46
|
+
];
|
|
47
|
+
|
|
48
|
+
let grouper = new Grouper({ name: { bind: "name" }, age: { bind: "age" } });
|
|
49
|
+
grouper.processAll(data);
|
|
50
|
+
|
|
51
|
+
let results = grouper.getResults();
|
|
52
|
+
// console.log(results);
|
|
53
|
+
assert.equal(results.length, 6);
|
|
54
|
+
assert.equal(results[0].records.length, 2);
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
});
|