@xh/hoist 73.0.0-SNAPSHOT.1747084971867 → 73.0.0-SNAPSHOT.1747087613907
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/CHANGELOG.md +4 -0
- package/build/types/cmp/grid/Types.d.ts +4 -1
- package/build/types/cmp/store/StoreFilterField.d.ts +2 -3
- package/build/types/desktop/cmp/grid/impl/filter/headerfilter/values/ValuesTabModel.d.ts +3 -0
- package/cmp/grid/Types.ts +4 -1
- package/cmp/grid/filter/GridFilterModel.ts +1 -1
- package/cmp/store/StoreFilterField.ts +2 -3
- package/desktop/cmp/grid/impl/filter/headerfilter/values/ValuesTab.scss +13 -0
- package/desktop/cmp/grid/impl/filter/headerfilter/values/ValuesTab.ts +36 -3
- package/desktop/cmp/grid/impl/filter/headerfilter/values/ValuesTabModel.ts +60 -15
- package/package.json +1 -1
- package/svc/WebSocketService.ts +1 -1
- package/tsconfig.tsbuildinfo +1 -1
package/CHANGELOG.md
CHANGED
|
@@ -35,6 +35,10 @@
|
|
|
35
35
|
allow the client to do a potentially interactive popup login during the session to re-establish
|
|
36
36
|
the login. This is especially useful to allow recovery from expired or invalidated refresh
|
|
37
37
|
tokens.
|
|
38
|
+
* Improvements to Grid columns `HeaderFilter` component:
|
|
39
|
+
* `GridFilterModel` `commitOnChage` now set to `false` by default
|
|
40
|
+
* Addition of ability to append terms to active filter **only** when `commitOnChage:false`
|
|
41
|
+
* Grid column header filtering functionality now similar to Excel on Windows
|
|
38
42
|
|
|
39
43
|
### 🐞 Bug Fixes
|
|
40
44
|
|
|
@@ -60,7 +60,10 @@ export interface GridFilterModelConfig {
|
|
|
60
60
|
* gridModel's store.
|
|
61
61
|
*/
|
|
62
62
|
bind?: Store | View;
|
|
63
|
-
/**
|
|
63
|
+
/**
|
|
64
|
+
* True to update filters immediately after each change made in the column-based filter UI.
|
|
65
|
+
* Defaults to False.
|
|
66
|
+
*/
|
|
64
67
|
commitOnChange?: boolean;
|
|
65
68
|
/**
|
|
66
69
|
* Specifies the fields this model supports for filtering. Should be configs for
|
|
@@ -11,9 +11,8 @@ export interface StoreFilterFieldProps extends DefaultHoistProps {
|
|
|
11
11
|
autoApply?: boolean;
|
|
12
12
|
/**
|
|
13
13
|
* Field on optional model to which this component should bind its raw (text) value to persist
|
|
14
|
-
* across renders. Specify this field to control the state of this component directly
|
|
15
|
-
*
|
|
16
|
-
* use-cases - this prop is typically left unset.
|
|
14
|
+
* across renders. Specify this field to control the state of this component directly from a model.
|
|
15
|
+
* These are both advanced use-cases - this prop is typically left unset.
|
|
17
16
|
*/
|
|
18
17
|
bind?: string;
|
|
19
18
|
/** Names of field(s) to exclude from search. Cannot be used with `includeFields`. */
|
|
@@ -11,6 +11,7 @@ export declare class ValuesTabModel extends HoistModel {
|
|
|
11
11
|
pendingValues: any[];
|
|
12
12
|
/** Bound search term for `StoreFilterField` */
|
|
13
13
|
filterText: string;
|
|
14
|
+
combineCurrentFilters: boolean;
|
|
14
15
|
/** FieldFilter output by this model. */
|
|
15
16
|
get filter(): FieldFilterSpec;
|
|
16
17
|
get allVisibleRecsChecked(): boolean;
|
|
@@ -26,6 +27,8 @@ export declare class ValuesTabModel extends HoistModel {
|
|
|
26
27
|
reset(): void;
|
|
27
28
|
setRecsChecked(isChecked: boolean, values: any[]): void;
|
|
28
29
|
toggleAllRecsChecked(): void;
|
|
30
|
+
filterTextReaction(): void;
|
|
31
|
+
combineCurrentFiltersToggleReaction(): void;
|
|
29
32
|
private getFilter;
|
|
30
33
|
private doSyncWithFilter;
|
|
31
34
|
private syncGrid;
|
package/cmp/grid/Types.ts
CHANGED
|
@@ -90,7 +90,10 @@ export interface GridFilterModelConfig {
|
|
|
90
90
|
*/
|
|
91
91
|
bind?: Store | View;
|
|
92
92
|
|
|
93
|
-
/**
|
|
93
|
+
/**
|
|
94
|
+
* True to update filters immediately after each change made in the column-based filter UI.
|
|
95
|
+
* Defaults to False.
|
|
96
|
+
*/
|
|
94
97
|
commitOnChange?: boolean;
|
|
95
98
|
|
|
96
99
|
/**
|
|
@@ -46,7 +46,7 @@ export class GridFilterModel extends HoistModel {
|
|
|
46
46
|
static BLANK_PLACEHOLDER = '[blank]';
|
|
47
47
|
|
|
48
48
|
constructor(
|
|
49
|
-
{bind, commitOnChange =
|
|
49
|
+
{bind, commitOnChange = false, fieldSpecs, fieldSpecDefaults}: GridFilterModelConfig,
|
|
50
50
|
gridModel: GridModel
|
|
51
51
|
) {
|
|
52
52
|
super();
|
|
@@ -21,9 +21,8 @@ export interface StoreFilterFieldProps extends DefaultHoistProps {
|
|
|
21
21
|
|
|
22
22
|
/**
|
|
23
23
|
* Field on optional model to which this component should bind its raw (text) value to persist
|
|
24
|
-
* across renders. Specify this field to control the state of this component directly
|
|
25
|
-
*
|
|
26
|
-
* use-cases - this prop is typically left unset.
|
|
24
|
+
* across renders. Specify this field to control the state of this component directly from a model.
|
|
25
|
+
* These are both advanced use-cases - this prop is typically left unset.
|
|
27
26
|
*/
|
|
28
27
|
bind?: string;
|
|
29
28
|
|
|
@@ -1,4 +1,17 @@
|
|
|
1
1
|
.xh-values-filter-tab {
|
|
2
|
+
.store-filter-header {
|
|
3
|
+
padding: 5px 7px;
|
|
4
|
+
border-bottom: 1px solid var(--xh-grid-header-border-color);
|
|
5
|
+
row-gap: 5px;
|
|
6
|
+
.bp5-control-indicator {
|
|
7
|
+
font-size: 1em;
|
|
8
|
+
}
|
|
9
|
+
span {
|
|
10
|
+
font-size: var(--xh-grid-compact-header-font-size-px);
|
|
11
|
+
color: var(--xh-grid-header-text-color);
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
|
|
2
15
|
&__hidden-values-message {
|
|
3
16
|
display: flex;
|
|
4
17
|
padding: var(--xh-pad-half-px);
|
|
@@ -5,10 +5,11 @@
|
|
|
5
5
|
* Copyright © 2025 Extremely Heavy Industries Inc.
|
|
6
6
|
*/
|
|
7
7
|
import {grid} from '@xh/hoist/cmp/grid';
|
|
8
|
-
import {div, placeholder, vframe} from '@xh/hoist/cmp/layout';
|
|
8
|
+
import {div, hframe, placeholder, span, vbox, vframe} from '@xh/hoist/cmp/layout';
|
|
9
9
|
import {storeFilterField} from '@xh/hoist/cmp/store';
|
|
10
10
|
import {hoistCmp, uses} from '@xh/hoist/core';
|
|
11
11
|
import {button} from '@xh/hoist/desktop/cmp/button';
|
|
12
|
+
import {checkbox} from '@xh/hoist/desktop/cmp/input';
|
|
12
13
|
import {panel} from '@xh/hoist/desktop/cmp/panel';
|
|
13
14
|
import {toolbar} from '@xh/hoist/desktop/cmp/toolbar';
|
|
14
15
|
import {Icon} from '@xh/hoist/icon';
|
|
@@ -39,7 +40,8 @@ const tbar = hoistCmp.factory(() => {
|
|
|
39
40
|
placeholder: 'Search...',
|
|
40
41
|
flex: 1,
|
|
41
42
|
autoFocus: true,
|
|
42
|
-
matchMode: 'any'
|
|
43
|
+
matchMode: 'any',
|
|
44
|
+
includeFields: ['value']
|
|
43
45
|
})
|
|
44
46
|
);
|
|
45
47
|
});
|
|
@@ -47,7 +49,38 @@ const tbar = hoistCmp.factory(() => {
|
|
|
47
49
|
const body = hoistCmp.factory<ValuesTabModel>(({model}) => {
|
|
48
50
|
const {isCustomFilter} = model.headerFilterModel;
|
|
49
51
|
if (isCustomFilter) return customFilterPlaceholder();
|
|
50
|
-
return vframe(grid(), hiddenValuesMessage());
|
|
52
|
+
return vframe(storeFilterSelect(), grid(), hiddenValuesMessage());
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
const storeFilterSelect = hoistCmp.factory<ValuesTabModel>(({model}) => {
|
|
56
|
+
const {gridModel, allVisibleRecsChecked, filterText, headerFilterModel} = model,
|
|
57
|
+
{store} = gridModel;
|
|
58
|
+
|
|
59
|
+
return vbox({
|
|
60
|
+
className: 'store-filter-header',
|
|
61
|
+
items: [
|
|
62
|
+
hframe(
|
|
63
|
+
checkbox({
|
|
64
|
+
disabled: store.empty,
|
|
65
|
+
displayUnsetState: true,
|
|
66
|
+
value: allVisibleRecsChecked,
|
|
67
|
+
onChange: () => model.toggleAllRecsChecked()
|
|
68
|
+
}),
|
|
69
|
+
span(`(Select All${filterText ? ' Search Results' : ''})`)
|
|
70
|
+
),
|
|
71
|
+
hframe({
|
|
72
|
+
omit:
|
|
73
|
+
!filterText ||
|
|
74
|
+
!model.columnFilters.length ||
|
|
75
|
+
store.empty ||
|
|
76
|
+
headerFilterModel.commitOnChange,
|
|
77
|
+
items: [
|
|
78
|
+
checkbox({bind: 'combineCurrentFilters'}),
|
|
79
|
+
span(`Add current selection to filter`)
|
|
80
|
+
]
|
|
81
|
+
})
|
|
82
|
+
]
|
|
83
|
+
});
|
|
51
84
|
});
|
|
52
85
|
|
|
53
86
|
const customFilterPlaceholder = hoistCmp.factory<ValuesTabModel>(({model}) => {
|
|
@@ -10,7 +10,7 @@ import {FieldFilterSpec} from '@xh/hoist/data';
|
|
|
10
10
|
import {HeaderFilterModel} from '../HeaderFilterModel';
|
|
11
11
|
import {checkbox} from '@xh/hoist/desktop/cmp/input';
|
|
12
12
|
import {action, bindable, computed, makeObservable, observable} from '@xh/hoist/mobx';
|
|
13
|
-
import {castArray, difference, isEmpty, partition, uniq, without} from 'lodash';
|
|
13
|
+
import {castArray, difference, flatten, isEmpty, map, partition, uniq, without} from 'lodash';
|
|
14
14
|
|
|
15
15
|
export class ValuesTabModel extends HoistModel {
|
|
16
16
|
override xhImpl = true;
|
|
@@ -26,6 +26,12 @@ export class ValuesTabModel extends HoistModel {
|
|
|
26
26
|
/** Bound search term for `StoreFilterField` */
|
|
27
27
|
@bindable filterText: string = null;
|
|
28
28
|
|
|
29
|
+
/*
|
|
30
|
+
* Available only when commit on change is false merge
|
|
31
|
+
* current filter with pendingValues on commit
|
|
32
|
+
*/
|
|
33
|
+
@bindable combineCurrentFilters: boolean = false;
|
|
34
|
+
|
|
29
35
|
/** FieldFilter output by this model. */
|
|
30
36
|
@computed.struct
|
|
31
37
|
get filter(): FieldFilterSpec {
|
|
@@ -81,11 +87,22 @@ export class ValuesTabModel extends HoistModel {
|
|
|
81
87
|
this.headerFilterModel = headerFilterModel;
|
|
82
88
|
this.gridModel = this.createGridModel();
|
|
83
89
|
|
|
84
|
-
this.addReaction(
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
90
|
+
this.addReaction(
|
|
91
|
+
{
|
|
92
|
+
track: () => this.pendingValues,
|
|
93
|
+
run: () => this.syncGrid(),
|
|
94
|
+
fireImmediately: true
|
|
95
|
+
},
|
|
96
|
+
{
|
|
97
|
+
track: () => this.filterText,
|
|
98
|
+
run: () => this.filterTextReaction(),
|
|
99
|
+
debounce: 300
|
|
100
|
+
},
|
|
101
|
+
{
|
|
102
|
+
track: () => this.combineCurrentFilters,
|
|
103
|
+
run: () => this.combineCurrentFiltersToggleReaction()
|
|
104
|
+
}
|
|
105
|
+
);
|
|
89
106
|
}
|
|
90
107
|
|
|
91
108
|
syncWithFilter() {
|
|
@@ -115,6 +132,42 @@ export class ValuesTabModel extends HoistModel {
|
|
|
115
132
|
//-------------------
|
|
116
133
|
// Implementation
|
|
117
134
|
//-------------------
|
|
135
|
+
@action
|
|
136
|
+
filterTextReaction() {
|
|
137
|
+
if (!this.filterText) {
|
|
138
|
+
this.combineCurrentFilters = false;
|
|
139
|
+
this.doSyncWithFilter();
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
const {records} = this.gridModel.store,
|
|
144
|
+
currentFilterValues = flatten(map(this.columnFilters, 'value')),
|
|
145
|
+
checkedRecs = records.filter(
|
|
146
|
+
it =>
|
|
147
|
+
this.headerFilterModel.commitOnChange ||
|
|
148
|
+
currentFilterValues.length ||
|
|
149
|
+
it.get('isChecked')
|
|
150
|
+
),
|
|
151
|
+
values = map(checkedRecs, it => it.get('value'));
|
|
152
|
+
|
|
153
|
+
this.pendingValues = uniq(
|
|
154
|
+
this.combineCurrentFilters ? [...currentFilterValues, ...values] : values
|
|
155
|
+
);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
@action combineCurrentFiltersToggleReaction() {
|
|
159
|
+
if (!this.filterText) return;
|
|
160
|
+
|
|
161
|
+
const {records} = this.gridModel.store,
|
|
162
|
+
currentFilterValues = flatten(map(this.columnFilters, 'value')),
|
|
163
|
+
checkedRecs = records.filter(it => it.get('isChecked')),
|
|
164
|
+
values = map(checkedRecs, it => it.get('value'));
|
|
165
|
+
|
|
166
|
+
this.pendingValues = uniq(
|
|
167
|
+
this.combineCurrentFilters ? [...currentFilterValues, ...values] : values
|
|
168
|
+
);
|
|
169
|
+
}
|
|
170
|
+
|
|
118
171
|
private getFilter() {
|
|
119
172
|
const {gridFilterModel, pendingValues, values, valueCount, field} = this,
|
|
120
173
|
included = pendingValues.map(it => gridFilterModel.fromDisplayValue(it)),
|
|
@@ -217,17 +270,10 @@ export class ValuesTabModel extends HoistModel {
|
|
|
217
270
|
onRowClicked: ({data: record}) => {
|
|
218
271
|
this.setRecsChecked(!record.get('isChecked'), record.get('value'));
|
|
219
272
|
},
|
|
273
|
+
hideHeaders: true,
|
|
220
274
|
columns: [
|
|
221
275
|
{
|
|
222
276
|
field: 'isChecked',
|
|
223
|
-
headerName: ({gridModel}) => {
|
|
224
|
-
return checkbox({
|
|
225
|
-
disabled: gridModel.store.empty,
|
|
226
|
-
displayUnsetState: true,
|
|
227
|
-
value: this.allVisibleRecsChecked,
|
|
228
|
-
onChange: () => this.toggleAllRecsChecked()
|
|
229
|
-
});
|
|
230
|
-
},
|
|
231
277
|
width: 28,
|
|
232
278
|
autosizable: false,
|
|
233
279
|
pinned: true,
|
|
@@ -245,7 +291,6 @@ export class ValuesTabModel extends HoistModel {
|
|
|
245
291
|
},
|
|
246
292
|
{
|
|
247
293
|
field: 'value',
|
|
248
|
-
displayName: '(Select All)',
|
|
249
294
|
align: 'left',
|
|
250
295
|
comparator: (v1, v2, sortDir, abs, {defaultComparator}) => {
|
|
251
296
|
const mul = sortDir === 'desc' ? -1 : 1;
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@xh/hoist",
|
|
3
|
-
"version": "73.0.0-SNAPSHOT.
|
|
3
|
+
"version": "73.0.0-SNAPSHOT.1747087613907",
|
|
4
4
|
"description": "Hoist add-on for building and deploying React Applications.",
|
|
5
5
|
"repository": "github:xh/hoist-react",
|
|
6
6
|
"homepage": "https://xh.io",
|
package/svc/WebSocketService.ts
CHANGED
|
@@ -220,7 +220,7 @@ export class WebSocketService extends HoistService {
|
|
|
220
220
|
}
|
|
221
221
|
|
|
222
222
|
// 3) Unhappy path -- attempt a (throttled) reconnect.
|
|
223
|
-
if (
|
|
223
|
+
if (olderThan(this._lastHeartbeatReconnectAttempt, this.HEARTBEAT_RECONNECT_INTERVAL)) {
|
|
224
224
|
this._lastHeartbeatReconnectAttempt = new Date();
|
|
225
225
|
this.logWarn('Heartbeat found websocket not connected - attempting to reconnect...');
|
|
226
226
|
this.noteTelemetryEvent('heartbeatReconnectAttempt');
|