@theia/preferences 1.45.1 → 1.46.0-next.72
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 +81 -81
- package/lib/browser/abstract-resource-preference-provider.d.ts +47 -47
- package/lib/browser/abstract-resource-preference-provider.js +240 -240
- package/lib/browser/abstract-resource-preference-provider.spec.d.ts +1 -1
- package/lib/browser/abstract-resource-preference-provider.spec.js +83 -83
- package/lib/browser/folder-preference-provider.d.ts +20 -20
- package/lib/browser/folder-preference-provider.js +59 -59
- package/lib/browser/folders-preferences-provider.d.ts +27 -27
- package/lib/browser/folders-preferences-provider.js +245 -245
- package/lib/browser/index.d.ts +7 -7
- package/lib/browser/index.js +34 -34
- package/lib/browser/monaco-jsonc-editor.d.ts +9 -9
- package/lib/browser/monaco-jsonc-editor.js +80 -80
- package/lib/browser/package.spec.js +25 -25
- package/lib/browser/preference-bindings.d.ts +4 -4
- package/lib/browser/preference-bindings.js +63 -63
- package/lib/browser/preference-frontend-module.d.ts +6 -6
- package/lib/browser/preference-frontend-module.js +52 -52
- package/lib/browser/preference-open-handler.d.ts +9 -9
- package/lib/browser/preference-open-handler.js +64 -64
- package/lib/browser/preference-transaction-manager.d.ts +100 -100
- package/lib/browser/preference-transaction-manager.js +293 -293
- package/lib/browser/preference-tree-model.d.ts +60 -60
- package/lib/browser/preference-tree-model.js +243 -243
- package/lib/browser/preferences-contribution.d.ts +37 -37
- package/lib/browser/preferences-contribution.js +280 -280
- package/lib/browser/preferences-json-schema-contribution.d.ts +17 -17
- package/lib/browser/preferences-json-schema-contribution.js +91 -91
- package/lib/browser/preferences-monaco-contribution.d.ts +1 -1
- package/lib/browser/preferences-monaco-contribution.js +27 -27
- package/lib/browser/section-preference-provider.d.ts +21 -21
- package/lib/browser/section-preference-provider.js +96 -96
- package/lib/browser/user-configs-preference-provider.d.ts +22 -22
- package/lib/browser/user-configs-preference-provider.js +137 -137
- package/lib/browser/user-preference-provider.d.ts +13 -13
- package/lib/browser/user-preference-provider.js +41 -41
- package/lib/browser/util/preference-scope-command-manager.d.ts +17 -17
- package/lib/browser/util/preference-scope-command-manager.js +87 -87
- package/lib/browser/util/preference-tree-generator.d.ts +31 -31
- package/lib/browser/util/preference-tree-generator.js +237 -237
- package/lib/browser/util/preference-tree-label-provider.d.ts +11 -11
- package/lib/browser/util/preference-tree-label-provider.js +77 -77
- package/lib/browser/util/preference-tree-label-provider.spec.d.ts +1 -1
- package/lib/browser/util/preference-tree-label-provider.spec.js +87 -87
- package/lib/browser/util/preference-types.d.ts +62 -62
- package/lib/browser/util/preference-types.js +128 -128
- package/lib/browser/views/components/preference-array-input.d.ts +28 -28
- package/lib/browser/views/components/preference-array-input.js +180 -180
- package/lib/browser/views/components/preference-boolean-input.d.ts +17 -17
- package/lib/browser/views/components/preference-boolean-input.js +79 -79
- package/lib/browser/views/components/preference-file-input.d.ts +29 -29
- package/lib/browser/views/components/preference-file-input.js +110 -110
- package/lib/browser/views/components/preference-json-input.d.ts +19 -19
- package/lib/browser/views/components/preference-json-input.js +93 -93
- package/lib/browser/views/components/preference-markdown-renderer.d.ts +12 -12
- package/lib/browser/views/components/preference-markdown-renderer.js +81 -81
- package/lib/browser/views/components/preference-node-renderer-creator.d.ts +48 -48
- package/lib/browser/views/components/preference-node-renderer-creator.js +132 -132
- package/lib/browser/views/components/preference-node-renderer.d.ts +112 -112
- package/lib/browser/views/components/preference-node-renderer.js +441 -441
- package/lib/browser/views/components/preference-number-input.d.ts +34 -34
- package/lib/browser/views/components/preference-number-input.js +142 -142
- package/lib/browser/views/components/preference-select-input.d.ts +28 -28
- package/lib/browser/views/components/preference-select-input.js +138 -138
- package/lib/browser/views/components/preference-string-input.d.ts +17 -17
- package/lib/browser/views/components/preference-string-input.js +89 -89
- package/lib/browser/views/preference-editor-widget.d.ts +67 -67
- package/lib/browser/views/preference-editor-widget.js +363 -363
- package/lib/browser/views/preference-scope-tabbar-widget.d.ts +54 -54
- package/lib/browser/views/preference-scope-tabbar-widget.js +343 -343
- package/lib/browser/views/preference-searchbar-widget.d.ts +53 -53
- package/lib/browser/views/preference-searchbar-widget.js +173 -173
- package/lib/browser/views/preference-tree-widget.d.ts +17 -17
- package/lib/browser/views/preference-tree-widget.js +104 -104
- package/lib/browser/views/preference-widget-bindings.d.ts +3 -3
- package/lib/browser/views/preference-widget-bindings.js +87 -87
- package/lib/browser/views/preference-widget.d.ts +36 -36
- package/lib/browser/views/preference-widget.js +126 -126
- package/lib/browser/workspace-file-preference-provider.d.ts +23 -23
- package/lib/browser/workspace-file-preference-provider.js +110 -110
- package/lib/browser/workspace-preference-provider.d.ts +28 -28
- package/lib/browser/workspace-preference-provider.js +142 -142
- package/package.json +10 -10
- package/src/browser/abstract-resource-preference-provider.spec.ts +95 -95
- package/src/browser/abstract-resource-preference-provider.ts +232 -232
- package/src/browser/folder-preference-provider.ts +58 -58
- package/src/browser/folders-preferences-provider.ts +244 -244
- package/src/browser/index.ts +23 -23
- package/src/browser/monaco-jsonc-editor.ts +67 -67
- package/src/browser/package.spec.ts +28 -28
- package/src/browser/preference-bindings.ts +65 -65
- package/src/browser/preference-frontend-module.ts +57 -57
- package/src/browser/preference-open-handler.ts +53 -53
- package/src/browser/preference-transaction-manager.ts +287 -287
- package/src/browser/preference-tree-model.ts +250 -250
- package/src/browser/preferences-contribution.ts +263 -263
- package/src/browser/preferences-json-schema-contribution.ts +86 -86
- package/src/browser/preferences-monaco-contribution.ts +27 -27
- package/src/browser/section-preference-provider.ts +83 -83
- package/src/browser/style/index.css +506 -506
- package/src/browser/style/preference-array.css +94 -94
- package/src/browser/style/preference-context-menu.css +74 -74
- package/src/browser/style/preference-file.css +31 -31
- package/src/browser/style/preference-object.css +49 -49
- package/src/browser/style/search-input.css +66 -66
- package/src/browser/user-configs-preference-provider.ts +127 -127
- package/src/browser/user-preference-provider.ts +35 -35
- package/src/browser/util/preference-scope-command-manager.ts +75 -75
- package/src/browser/util/preference-tree-generator.ts +226 -226
- package/src/browser/util/preference-tree-label-provider.spec.ts +108 -108
- package/src/browser/util/preference-tree-label-provider.ts +64 -64
- package/src/browser/util/preference-types.ts +169 -169
- package/src/browser/views/components/preference-array-input.ts +174 -174
- package/src/browser/views/components/preference-boolean-input.ts +69 -69
- package/src/browser/views/components/preference-file-input.ts +104 -104
- package/src/browser/views/components/preference-json-input.ts +78 -78
- package/src/browser/views/components/preference-markdown-renderer.ts +68 -68
- package/src/browser/views/components/preference-node-renderer-creator.ts +141 -141
- package/src/browser/views/components/preference-node-renderer.ts +477 -477
- package/src/browser/views/components/preference-number-input.ts +147 -147
- package/src/browser/views/components/preference-select-input.ts +131 -131
- package/src/browser/views/components/preference-string-input.ts +76 -76
- package/src/browser/views/preference-editor-widget.ts +349 -349
- package/src/browser/views/preference-scope-tabbar-widget.tsx +344 -344
- package/src/browser/views/preference-searchbar-widget.tsx +183 -183
- package/src/browser/views/preference-tree-widget.tsx +93 -93
- package/src/browser/views/preference-widget-bindings.ts +102 -102
- package/src/browser/views/preference-widget.tsx +117 -117
- package/src/browser/workspace-file-preference-provider.ts +100 -100
- package/src/browser/workspace-preference-provider.ts +134 -134
|
@@ -1,250 +1,250 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2020 Ericsson and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { inject, injectable, postConstruct } from '@theia/core/shared/inversify';
|
|
18
|
-
import {
|
|
19
|
-
TreeModelImpl,
|
|
20
|
-
TreeWidget,
|
|
21
|
-
CompositeTreeNode,
|
|
22
|
-
TopDownTreeIterator,
|
|
23
|
-
TreeNode,
|
|
24
|
-
PreferenceSchemaProvider,
|
|
25
|
-
PreferenceDataProperty,
|
|
26
|
-
NodeProps,
|
|
27
|
-
ExpandableTreeNode,
|
|
28
|
-
SelectableTreeNode,
|
|
29
|
-
PreferenceService,
|
|
30
|
-
} from '@theia/core/lib/browser';
|
|
31
|
-
import { Emitter } from '@theia/core';
|
|
32
|
-
import { PreferencesSearchbarWidget } from './views/preference-searchbar-widget';
|
|
33
|
-
import { PreferenceTreeGenerator, COMMONLY_USED_SECTION_PREFIX } from './util/preference-tree-generator';
|
|
34
|
-
import * as fuzzy from '@theia/core/shared/fuzzy';
|
|
35
|
-
import { PreferencesScopeTabBar } from './views/preference-scope-tabbar-widget';
|
|
36
|
-
import { Preference } from './util/preference-types';
|
|
37
|
-
import { Event } from '@theia/core/lib/common';
|
|
38
|
-
|
|
39
|
-
export interface PreferenceTreeNodeProps extends NodeProps {
|
|
40
|
-
visibleChildren: number;
|
|
41
|
-
isExpansible?: boolean;
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
export interface PreferenceTreeNodeRow extends Readonly<TreeWidget.NodeRow>, PreferenceTreeNodeProps {
|
|
45
|
-
node: Preference.TreeNode;
|
|
46
|
-
}
|
|
47
|
-
export enum PreferenceFilterChangeSource {
|
|
48
|
-
Schema,
|
|
49
|
-
Search,
|
|
50
|
-
Scope,
|
|
51
|
-
}
|
|
52
|
-
export interface PreferenceFilterChangeEvent {
|
|
53
|
-
source: PreferenceFilterChangeSource
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
@injectable()
|
|
57
|
-
export class PreferenceTreeModel extends TreeModelImpl {
|
|
58
|
-
|
|
59
|
-
@inject(PreferenceSchemaProvider) protected readonly schemaProvider: PreferenceSchemaProvider;
|
|
60
|
-
@inject(PreferencesSearchbarWidget) protected readonly filterInput: PreferencesSearchbarWidget;
|
|
61
|
-
@inject(PreferenceTreeGenerator) protected readonly treeGenerator: PreferenceTreeGenerator;
|
|
62
|
-
@inject(PreferencesScopeTabBar) protected readonly scopeTracker: PreferencesScopeTabBar;
|
|
63
|
-
@inject(PreferenceService) protected readonly preferenceService: PreferenceService;
|
|
64
|
-
|
|
65
|
-
protected readonly onTreeFilterChangedEmitter = new Emitter<PreferenceFilterChangeEvent>();
|
|
66
|
-
readonly onFilterChanged = this.onTreeFilterChangedEmitter.event;
|
|
67
|
-
|
|
68
|
-
protected lastSearchedFuzzy: string = '';
|
|
69
|
-
protected lastSearchedLiteral: string = '';
|
|
70
|
-
protected _currentScope: number = Number(Preference.DEFAULT_SCOPE.scope);
|
|
71
|
-
protected _isFiltered: boolean = false;
|
|
72
|
-
protected _currentRows: Map<string, PreferenceTreeNodeRow> = new Map();
|
|
73
|
-
protected _totalVisibleLeaves = 0;
|
|
74
|
-
|
|
75
|
-
get currentRows(): Readonly<Map<string, PreferenceTreeNodeRow>> {
|
|
76
|
-
return this._currentRows;
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
get totalVisibleLeaves(): number {
|
|
80
|
-
return this._totalVisibleLeaves;
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
get isFiltered(): boolean {
|
|
84
|
-
return this._isFiltered;
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
get propertyList(): { [key: string]: PreferenceDataProperty; } {
|
|
88
|
-
return this.schemaProvider.getCombinedSchema().properties;
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
get currentScope(): Preference.SelectedScopeDetails {
|
|
92
|
-
return this.scopeTracker.currentScope;
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
get onSchemaChanged(): Event<CompositeTreeNode> {
|
|
96
|
-
return this.treeGenerator.onSchemaChanged;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
@postConstruct()
|
|
100
|
-
protected override init(): void {
|
|
101
|
-
this.doInit();
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
protected async doInit(): Promise<void> {
|
|
105
|
-
super.init();
|
|
106
|
-
this.toDispose.pushAll([
|
|
107
|
-
this.treeGenerator.onSchemaChanged(newTree => this.handleNewSchema(newTree)),
|
|
108
|
-
this.scopeTracker.onScopeChanged(scopeDetails => {
|
|
109
|
-
this._currentScope = scopeDetails.scope;
|
|
110
|
-
this.updateFilteredRows(PreferenceFilterChangeSource.Scope);
|
|
111
|
-
}),
|
|
112
|
-
this.filterInput.onFilterChanged(newSearchTerm => {
|
|
113
|
-
this.lastSearchedLiteral = newSearchTerm;
|
|
114
|
-
this.lastSearchedFuzzy = newSearchTerm.replace(/\s/g, '');
|
|
115
|
-
this._isFiltered = newSearchTerm.length > 2;
|
|
116
|
-
if (this.isFiltered) {
|
|
117
|
-
this.expandAll();
|
|
118
|
-
} else if (CompositeTreeNode.is(this.root)) {
|
|
119
|
-
this.collapseAll(this.root);
|
|
120
|
-
}
|
|
121
|
-
this.updateFilteredRows(PreferenceFilterChangeSource.Search);
|
|
122
|
-
}),
|
|
123
|
-
this.onFilterChanged(() => {
|
|
124
|
-
this.filterInput.updateResultsCount(this._totalVisibleLeaves);
|
|
125
|
-
}),
|
|
126
|
-
this.onTreeFilterChangedEmitter,
|
|
127
|
-
]);
|
|
128
|
-
await this.preferenceService.ready;
|
|
129
|
-
this.handleNewSchema(this.treeGenerator.root);
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
private handleNewSchema(newRoot: CompositeTreeNode): void {
|
|
133
|
-
this.root = newRoot;
|
|
134
|
-
if (this.isFiltered) {
|
|
135
|
-
this.expandAll();
|
|
136
|
-
}
|
|
137
|
-
this.updateFilteredRows(PreferenceFilterChangeSource.Schema);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
protected updateRows(): void {
|
|
141
|
-
const root = this.root;
|
|
142
|
-
this._currentRows = new Map();
|
|
143
|
-
if (root) {
|
|
144
|
-
this._totalVisibleLeaves = 0;
|
|
145
|
-
let index = 0;
|
|
146
|
-
|
|
147
|
-
for (const node of new TopDownTreeIterator(root, {
|
|
148
|
-
pruneCollapsed: false,
|
|
149
|
-
pruneSiblings: true
|
|
150
|
-
})) {
|
|
151
|
-
if (TreeNode.isVisible(node) && Preference.TreeNode.is(node)) {
|
|
152
|
-
const { id } = Preference.TreeNode.getGroupAndIdFromNodeId(node.id);
|
|
153
|
-
if (CompositeTreeNode.is(node) || this.passesCurrentFilters(node, id)) {
|
|
154
|
-
this.updateVisibleChildren(node);
|
|
155
|
-
|
|
156
|
-
this._currentRows.set(node.id, {
|
|
157
|
-
index: index++,
|
|
158
|
-
node,
|
|
159
|
-
depth: node.depth,
|
|
160
|
-
visibleChildren: 0,
|
|
161
|
-
});
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
protected updateFilteredRows(source: PreferenceFilterChangeSource): void {
|
|
169
|
-
this.updateRows();
|
|
170
|
-
this.onTreeFilterChangedEmitter.fire({ source });
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
protected passesCurrentFilters(node: Preference.LeafNode, prefID: string): boolean {
|
|
174
|
-
if (!this.schemaProvider.isValidInScope(prefID, this._currentScope)) {
|
|
175
|
-
return false;
|
|
176
|
-
}
|
|
177
|
-
if (!this._isFiltered) {
|
|
178
|
-
return true;
|
|
179
|
-
}
|
|
180
|
-
// When filtering, VSCode will render an item that is present in the commonly used section only once but render both its possible parents in the left-hand tree.
|
|
181
|
-
// E.g. searching for editor.renderWhitespace will show one item in the main panel, but both 'Commonly Used' and 'Text Editor' in the left tree.
|
|
182
|
-
// That seems counterintuitive and introduces a number of special cases, so I prefer to remove the commonly used section entirely when the user searches.
|
|
183
|
-
if (node.id.startsWith(COMMONLY_USED_SECTION_PREFIX)) {
|
|
184
|
-
return false;
|
|
185
|
-
}
|
|
186
|
-
return fuzzy.test(this.lastSearchedFuzzy, prefID) // search matches preference name.
|
|
187
|
-
// search matches description. Fuzzy isn't ideal here because the score depends on the order of discovery.
|
|
188
|
-
|| (node.preference.data.description ?? '').includes(this.lastSearchedLiteral);
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
protected override isVisibleSelectableNode(node: TreeNode): node is SelectableTreeNode {
|
|
192
|
-
return CompositeTreeNode.is(node) && !!this._currentRows.get(node.id)?.visibleChildren;
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
protected updateVisibleChildren(node: TreeNode): void {
|
|
196
|
-
if (!CompositeTreeNode.is(node)) {
|
|
197
|
-
this._totalVisibleLeaves++;
|
|
198
|
-
let nextParent = node.parent?.id && this._currentRows.get(node.parent?.id);
|
|
199
|
-
while (nextParent && nextParent.node !== this.root) {
|
|
200
|
-
if (nextParent) {
|
|
201
|
-
nextParent.visibleChildren += 1;
|
|
202
|
-
}
|
|
203
|
-
nextParent = nextParent.node.parent?.id && this._currentRows.get(nextParent.node.parent?.id);
|
|
204
|
-
if (nextParent) {
|
|
205
|
-
nextParent.isExpansible = true;
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
}
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
collapseAllExcept(openNode: TreeNode | undefined): void {
|
|
212
|
-
if (ExpandableTreeNode.is(openNode)) {
|
|
213
|
-
this.expandNode(openNode);
|
|
214
|
-
}
|
|
215
|
-
if (CompositeTreeNode.is(this.root)) {
|
|
216
|
-
this.root.children.forEach(child => {
|
|
217
|
-
if (child !== openNode && ExpandableTreeNode.is(child)) {
|
|
218
|
-
this.collapseNode(child);
|
|
219
|
-
}
|
|
220
|
-
});
|
|
221
|
-
}
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
protected expandAll(): void {
|
|
225
|
-
if (CompositeTreeNode.is(this.root)) {
|
|
226
|
-
this.root.children.forEach(child => {
|
|
227
|
-
if (ExpandableTreeNode.is(child)) {
|
|
228
|
-
this.expandNode(child);
|
|
229
|
-
}
|
|
230
|
-
});
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
getNodeFromPreferenceId(id: string): Preference.TreeNode | undefined {
|
|
235
|
-
const node = this.getNode(this.treeGenerator.getNodeId(id));
|
|
236
|
-
return node && Preference.TreeNode.is(node) ? node : undefined;
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
/**
|
|
240
|
-
* @returns true if selection changed, false otherwise
|
|
241
|
-
*/
|
|
242
|
-
selectIfNotSelected(node: SelectableTreeNode): boolean {
|
|
243
|
-
const currentlySelected = this.selectedNodes[0];
|
|
244
|
-
if (node !== currentlySelected) {
|
|
245
|
-
this.selectNode(node);
|
|
246
|
-
return true;
|
|
247
|
-
}
|
|
248
|
-
return false;
|
|
249
|
-
}
|
|
250
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2020 Ericsson and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { inject, injectable, postConstruct } from '@theia/core/shared/inversify';
|
|
18
|
+
import {
|
|
19
|
+
TreeModelImpl,
|
|
20
|
+
TreeWidget,
|
|
21
|
+
CompositeTreeNode,
|
|
22
|
+
TopDownTreeIterator,
|
|
23
|
+
TreeNode,
|
|
24
|
+
PreferenceSchemaProvider,
|
|
25
|
+
PreferenceDataProperty,
|
|
26
|
+
NodeProps,
|
|
27
|
+
ExpandableTreeNode,
|
|
28
|
+
SelectableTreeNode,
|
|
29
|
+
PreferenceService,
|
|
30
|
+
} from '@theia/core/lib/browser';
|
|
31
|
+
import { Emitter } from '@theia/core';
|
|
32
|
+
import { PreferencesSearchbarWidget } from './views/preference-searchbar-widget';
|
|
33
|
+
import { PreferenceTreeGenerator, COMMONLY_USED_SECTION_PREFIX } from './util/preference-tree-generator';
|
|
34
|
+
import * as fuzzy from '@theia/core/shared/fuzzy';
|
|
35
|
+
import { PreferencesScopeTabBar } from './views/preference-scope-tabbar-widget';
|
|
36
|
+
import { Preference } from './util/preference-types';
|
|
37
|
+
import { Event } from '@theia/core/lib/common';
|
|
38
|
+
|
|
39
|
+
export interface PreferenceTreeNodeProps extends NodeProps {
|
|
40
|
+
visibleChildren: number;
|
|
41
|
+
isExpansible?: boolean;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export interface PreferenceTreeNodeRow extends Readonly<TreeWidget.NodeRow>, PreferenceTreeNodeProps {
|
|
45
|
+
node: Preference.TreeNode;
|
|
46
|
+
}
|
|
47
|
+
export enum PreferenceFilterChangeSource {
|
|
48
|
+
Schema,
|
|
49
|
+
Search,
|
|
50
|
+
Scope,
|
|
51
|
+
}
|
|
52
|
+
export interface PreferenceFilterChangeEvent {
|
|
53
|
+
source: PreferenceFilterChangeSource
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
@injectable()
|
|
57
|
+
export class PreferenceTreeModel extends TreeModelImpl {
|
|
58
|
+
|
|
59
|
+
@inject(PreferenceSchemaProvider) protected readonly schemaProvider: PreferenceSchemaProvider;
|
|
60
|
+
@inject(PreferencesSearchbarWidget) protected readonly filterInput: PreferencesSearchbarWidget;
|
|
61
|
+
@inject(PreferenceTreeGenerator) protected readonly treeGenerator: PreferenceTreeGenerator;
|
|
62
|
+
@inject(PreferencesScopeTabBar) protected readonly scopeTracker: PreferencesScopeTabBar;
|
|
63
|
+
@inject(PreferenceService) protected readonly preferenceService: PreferenceService;
|
|
64
|
+
|
|
65
|
+
protected readonly onTreeFilterChangedEmitter = new Emitter<PreferenceFilterChangeEvent>();
|
|
66
|
+
readonly onFilterChanged = this.onTreeFilterChangedEmitter.event;
|
|
67
|
+
|
|
68
|
+
protected lastSearchedFuzzy: string = '';
|
|
69
|
+
protected lastSearchedLiteral: string = '';
|
|
70
|
+
protected _currentScope: number = Number(Preference.DEFAULT_SCOPE.scope);
|
|
71
|
+
protected _isFiltered: boolean = false;
|
|
72
|
+
protected _currentRows: Map<string, PreferenceTreeNodeRow> = new Map();
|
|
73
|
+
protected _totalVisibleLeaves = 0;
|
|
74
|
+
|
|
75
|
+
get currentRows(): Readonly<Map<string, PreferenceTreeNodeRow>> {
|
|
76
|
+
return this._currentRows;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
get totalVisibleLeaves(): number {
|
|
80
|
+
return this._totalVisibleLeaves;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
get isFiltered(): boolean {
|
|
84
|
+
return this._isFiltered;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
get propertyList(): { [key: string]: PreferenceDataProperty; } {
|
|
88
|
+
return this.schemaProvider.getCombinedSchema().properties;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
get currentScope(): Preference.SelectedScopeDetails {
|
|
92
|
+
return this.scopeTracker.currentScope;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
get onSchemaChanged(): Event<CompositeTreeNode> {
|
|
96
|
+
return this.treeGenerator.onSchemaChanged;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
@postConstruct()
|
|
100
|
+
protected override init(): void {
|
|
101
|
+
this.doInit();
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
protected async doInit(): Promise<void> {
|
|
105
|
+
super.init();
|
|
106
|
+
this.toDispose.pushAll([
|
|
107
|
+
this.treeGenerator.onSchemaChanged(newTree => this.handleNewSchema(newTree)),
|
|
108
|
+
this.scopeTracker.onScopeChanged(scopeDetails => {
|
|
109
|
+
this._currentScope = scopeDetails.scope;
|
|
110
|
+
this.updateFilteredRows(PreferenceFilterChangeSource.Scope);
|
|
111
|
+
}),
|
|
112
|
+
this.filterInput.onFilterChanged(newSearchTerm => {
|
|
113
|
+
this.lastSearchedLiteral = newSearchTerm;
|
|
114
|
+
this.lastSearchedFuzzy = newSearchTerm.replace(/\s/g, '');
|
|
115
|
+
this._isFiltered = newSearchTerm.length > 2;
|
|
116
|
+
if (this.isFiltered) {
|
|
117
|
+
this.expandAll();
|
|
118
|
+
} else if (CompositeTreeNode.is(this.root)) {
|
|
119
|
+
this.collapseAll(this.root);
|
|
120
|
+
}
|
|
121
|
+
this.updateFilteredRows(PreferenceFilterChangeSource.Search);
|
|
122
|
+
}),
|
|
123
|
+
this.onFilterChanged(() => {
|
|
124
|
+
this.filterInput.updateResultsCount(this._totalVisibleLeaves);
|
|
125
|
+
}),
|
|
126
|
+
this.onTreeFilterChangedEmitter,
|
|
127
|
+
]);
|
|
128
|
+
await this.preferenceService.ready;
|
|
129
|
+
this.handleNewSchema(this.treeGenerator.root);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
private handleNewSchema(newRoot: CompositeTreeNode): void {
|
|
133
|
+
this.root = newRoot;
|
|
134
|
+
if (this.isFiltered) {
|
|
135
|
+
this.expandAll();
|
|
136
|
+
}
|
|
137
|
+
this.updateFilteredRows(PreferenceFilterChangeSource.Schema);
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
protected updateRows(): void {
|
|
141
|
+
const root = this.root;
|
|
142
|
+
this._currentRows = new Map();
|
|
143
|
+
if (root) {
|
|
144
|
+
this._totalVisibleLeaves = 0;
|
|
145
|
+
let index = 0;
|
|
146
|
+
|
|
147
|
+
for (const node of new TopDownTreeIterator(root, {
|
|
148
|
+
pruneCollapsed: false,
|
|
149
|
+
pruneSiblings: true
|
|
150
|
+
})) {
|
|
151
|
+
if (TreeNode.isVisible(node) && Preference.TreeNode.is(node)) {
|
|
152
|
+
const { id } = Preference.TreeNode.getGroupAndIdFromNodeId(node.id);
|
|
153
|
+
if (CompositeTreeNode.is(node) || this.passesCurrentFilters(node, id)) {
|
|
154
|
+
this.updateVisibleChildren(node);
|
|
155
|
+
|
|
156
|
+
this._currentRows.set(node.id, {
|
|
157
|
+
index: index++,
|
|
158
|
+
node,
|
|
159
|
+
depth: node.depth,
|
|
160
|
+
visibleChildren: 0,
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
protected updateFilteredRows(source: PreferenceFilterChangeSource): void {
|
|
169
|
+
this.updateRows();
|
|
170
|
+
this.onTreeFilterChangedEmitter.fire({ source });
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
protected passesCurrentFilters(node: Preference.LeafNode, prefID: string): boolean {
|
|
174
|
+
if (!this.schemaProvider.isValidInScope(prefID, this._currentScope)) {
|
|
175
|
+
return false;
|
|
176
|
+
}
|
|
177
|
+
if (!this._isFiltered) {
|
|
178
|
+
return true;
|
|
179
|
+
}
|
|
180
|
+
// When filtering, VSCode will render an item that is present in the commonly used section only once but render both its possible parents in the left-hand tree.
|
|
181
|
+
// E.g. searching for editor.renderWhitespace will show one item in the main panel, but both 'Commonly Used' and 'Text Editor' in the left tree.
|
|
182
|
+
// That seems counterintuitive and introduces a number of special cases, so I prefer to remove the commonly used section entirely when the user searches.
|
|
183
|
+
if (node.id.startsWith(COMMONLY_USED_SECTION_PREFIX)) {
|
|
184
|
+
return false;
|
|
185
|
+
}
|
|
186
|
+
return fuzzy.test(this.lastSearchedFuzzy, prefID) // search matches preference name.
|
|
187
|
+
// search matches description. Fuzzy isn't ideal here because the score depends on the order of discovery.
|
|
188
|
+
|| (node.preference.data.description ?? '').includes(this.lastSearchedLiteral);
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
protected override isVisibleSelectableNode(node: TreeNode): node is SelectableTreeNode {
|
|
192
|
+
return CompositeTreeNode.is(node) && !!this._currentRows.get(node.id)?.visibleChildren;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
protected updateVisibleChildren(node: TreeNode): void {
|
|
196
|
+
if (!CompositeTreeNode.is(node)) {
|
|
197
|
+
this._totalVisibleLeaves++;
|
|
198
|
+
let nextParent = node.parent?.id && this._currentRows.get(node.parent?.id);
|
|
199
|
+
while (nextParent && nextParent.node !== this.root) {
|
|
200
|
+
if (nextParent) {
|
|
201
|
+
nextParent.visibleChildren += 1;
|
|
202
|
+
}
|
|
203
|
+
nextParent = nextParent.node.parent?.id && this._currentRows.get(nextParent.node.parent?.id);
|
|
204
|
+
if (nextParent) {
|
|
205
|
+
nextParent.isExpansible = true;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
collapseAllExcept(openNode: TreeNode | undefined): void {
|
|
212
|
+
if (ExpandableTreeNode.is(openNode)) {
|
|
213
|
+
this.expandNode(openNode);
|
|
214
|
+
}
|
|
215
|
+
if (CompositeTreeNode.is(this.root)) {
|
|
216
|
+
this.root.children.forEach(child => {
|
|
217
|
+
if (child !== openNode && ExpandableTreeNode.is(child)) {
|
|
218
|
+
this.collapseNode(child);
|
|
219
|
+
}
|
|
220
|
+
});
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
protected expandAll(): void {
|
|
225
|
+
if (CompositeTreeNode.is(this.root)) {
|
|
226
|
+
this.root.children.forEach(child => {
|
|
227
|
+
if (ExpandableTreeNode.is(child)) {
|
|
228
|
+
this.expandNode(child);
|
|
229
|
+
}
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
getNodeFromPreferenceId(id: string): Preference.TreeNode | undefined {
|
|
235
|
+
const node = this.getNode(this.treeGenerator.getNodeId(id));
|
|
236
|
+
return node && Preference.TreeNode.is(node) ? node : undefined;
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
/**
|
|
240
|
+
* @returns true if selection changed, false otherwise
|
|
241
|
+
*/
|
|
242
|
+
selectIfNotSelected(node: SelectableTreeNode): boolean {
|
|
243
|
+
const currentlySelected = this.selectedNodes[0];
|
|
244
|
+
if (node !== currentlySelected) {
|
|
245
|
+
this.selectNode(node);
|
|
246
|
+
return true;
|
|
247
|
+
}
|
|
248
|
+
return false;
|
|
249
|
+
}
|
|
250
|
+
}
|