@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,349 +1,349 @@
|
|
|
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
|
-
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
18
|
-
import { postConstruct, injectable, inject } from '@theia/core/shared/inversify';
|
|
19
|
-
import throttle = require('@theia/core/shared/lodash.throttle');
|
|
20
|
-
import * as deepEqual from 'fast-deep-equal';
|
|
21
|
-
import {
|
|
22
|
-
PreferenceService,
|
|
23
|
-
CompositeTreeNode,
|
|
24
|
-
SelectableTreeNode,
|
|
25
|
-
StatefulWidget,
|
|
26
|
-
TopDownTreeIterator,
|
|
27
|
-
PreferenceChanges,
|
|
28
|
-
ExpandableTreeNode,
|
|
29
|
-
PreferenceSchemaProvider,
|
|
30
|
-
} from '@theia/core/lib/browser';
|
|
31
|
-
import { unreachable } from '@theia/core/lib/common';
|
|
32
|
-
import { BaseWidget, DEFAULT_SCROLL_OPTIONS } from '@theia/core/lib/browser/widgets/widget';
|
|
33
|
-
import { PreferenceTreeModel, PreferenceFilterChangeEvent, PreferenceFilterChangeSource } from '../preference-tree-model';
|
|
34
|
-
import { PreferenceNodeRendererFactory, GeneralPreferenceNodeRenderer } from './components/preference-node-renderer';
|
|
35
|
-
import { Preference } from '../util/preference-types';
|
|
36
|
-
import { COMMONLY_USED_SECTION_PREFIX } from '../util/preference-tree-generator';
|
|
37
|
-
import { PreferencesScopeTabBar } from './preference-scope-tabbar-widget';
|
|
38
|
-
import { PreferenceNodeRendererCreatorRegistry } from './components/preference-node-renderer-creator';
|
|
39
|
-
|
|
40
|
-
export interface PreferencesEditorState {
|
|
41
|
-
firstVisibleChildID: string,
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
@injectable()
|
|
45
|
-
export class PreferencesEditorWidget extends BaseWidget implements StatefulWidget {
|
|
46
|
-
static readonly ID = 'settings.editor';
|
|
47
|
-
static readonly LABEL = 'Settings Editor';
|
|
48
|
-
|
|
49
|
-
override scrollOptions = DEFAULT_SCROLL_OPTIONS;
|
|
50
|
-
|
|
51
|
-
protected scrollContainer: HTMLDivElement;
|
|
52
|
-
/**
|
|
53
|
-
* Guards against scroll events and selection events looping into each other. Set before this widget initiates a selection.
|
|
54
|
-
*/
|
|
55
|
-
protected currentModelSelectionId = '';
|
|
56
|
-
/**
|
|
57
|
-
* Permits the user to expand multiple nodes without each one being collapsed on a new selection.
|
|
58
|
-
*/
|
|
59
|
-
protected lastUserSelection = '';
|
|
60
|
-
protected isAtScrollTop = true;
|
|
61
|
-
protected firstVisibleChildID = '';
|
|
62
|
-
protected renderers = new Map<string, GeneralPreferenceNodeRenderer>();
|
|
63
|
-
protected preferenceDataKeys = new Map<string, string>();
|
|
64
|
-
// The commonly used section will duplicate preference ID's, so we'll keep a separate list of them.
|
|
65
|
-
protected commonlyUsedRenderers = new Map<string, GeneralPreferenceNodeRenderer>();
|
|
66
|
-
|
|
67
|
-
@inject(PreferenceService) protected readonly preferenceService: PreferenceService;
|
|
68
|
-
@inject(PreferenceTreeModel) protected readonly model: PreferenceTreeModel;
|
|
69
|
-
@inject(PreferenceNodeRendererFactory) protected readonly rendererFactory: PreferenceNodeRendererFactory;
|
|
70
|
-
@inject(PreferenceNodeRendererCreatorRegistry) protected readonly rendererRegistry: PreferenceNodeRendererCreatorRegistry;
|
|
71
|
-
@inject(PreferenceSchemaProvider) protected readonly schemaProvider: PreferenceSchemaProvider;
|
|
72
|
-
@inject(PreferencesScopeTabBar) protected readonly tabbar: PreferencesScopeTabBar;
|
|
73
|
-
|
|
74
|
-
@postConstruct()
|
|
75
|
-
protected init(): void {
|
|
76
|
-
this.doInit();
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
protected async doInit(): Promise<void> {
|
|
80
|
-
|
|
81
|
-
this.id = PreferencesEditorWidget.ID;
|
|
82
|
-
this.title.label = PreferencesEditorWidget.LABEL;
|
|
83
|
-
this.addClass('settings-main');
|
|
84
|
-
this.toDispose.pushAll([
|
|
85
|
-
this.preferenceService.onPreferencesChanged(e => this.handlePreferenceChanges(e)),
|
|
86
|
-
this.model.onFilterChanged(e => this.handleDisplayChange(e)),
|
|
87
|
-
this.model.onSelectionChanged(e => this.handleSelectionChange(e)),
|
|
88
|
-
]);
|
|
89
|
-
this.createContainers();
|
|
90
|
-
await this.preferenceService.ready;
|
|
91
|
-
this.handleDisplayChange({ source: PreferenceFilterChangeSource.Schema });
|
|
92
|
-
this.rendererRegistry.onDidChange(() => this.handleRegistryChange());
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
protected createContainers(): void {
|
|
96
|
-
const innerWrapper = document.createElement('div');
|
|
97
|
-
innerWrapper.classList.add('settings-main-scroll-container');
|
|
98
|
-
this.scrollContainer = innerWrapper;
|
|
99
|
-
innerWrapper.addEventListener('scroll', this.onScroll, { passive: true });
|
|
100
|
-
this.node.appendChild(innerWrapper);
|
|
101
|
-
const noLeavesMessage = document.createElement('div');
|
|
102
|
-
noLeavesMessage.classList.add('settings-no-results-announcement');
|
|
103
|
-
noLeavesMessage.textContent = 'That search query has returned no results.';
|
|
104
|
-
this.node.appendChild(noLeavesMessage);
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
protected handleDisplayChange(e: PreferenceFilterChangeEvent): void {
|
|
108
|
-
const { isFiltered } = this.model;
|
|
109
|
-
const currentFirstVisible = this.firstVisibleChildID;
|
|
110
|
-
const leavesAreVisible = this.areLeavesVisible();
|
|
111
|
-
if (e.source === PreferenceFilterChangeSource.Search) {
|
|
112
|
-
this.handleSearchChange(isFiltered, leavesAreVisible);
|
|
113
|
-
} else if (e.source === PreferenceFilterChangeSource.Scope) {
|
|
114
|
-
this.handleScopeChange(isFiltered);
|
|
115
|
-
} else if (e.source === PreferenceFilterChangeSource.Schema) {
|
|
116
|
-
this.handleSchemaChange(isFiltered);
|
|
117
|
-
} else {
|
|
118
|
-
unreachable(e.source, 'Not all PreferenceFilterChangeSource enum variants handled.');
|
|
119
|
-
}
|
|
120
|
-
this.resetScroll(currentFirstVisible, e.source === PreferenceFilterChangeSource.Search && !isFiltered);
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
protected handleRegistryChange(): void {
|
|
124
|
-
for (const [id, renderer, collection] of this.allRenderers()) {
|
|
125
|
-
renderer.dispose();
|
|
126
|
-
collection.delete(id);
|
|
127
|
-
}
|
|
128
|
-
this.handleDisplayChange({ source: PreferenceFilterChangeSource.Schema });
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
protected handleSchemaChange(isFiltered: boolean): void {
|
|
132
|
-
for (const [id, renderer, collection] of this.allRenderers()) {
|
|
133
|
-
const node = this.model.getNode(renderer.nodeId);
|
|
134
|
-
if (!node || (Preference.LeafNode.is(node) && this.hasSchemaChanged(renderer, node))) {
|
|
135
|
-
renderer.dispose();
|
|
136
|
-
collection.delete(id);
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
if (this.model.root) {
|
|
140
|
-
const nodeIterator = Array.from(this.scrollContainer.children)[Symbol.iterator]();
|
|
141
|
-
let nextNode: HTMLElement | undefined = nodeIterator.next().value;
|
|
142
|
-
for (const node of new TopDownTreeIterator(this.model.root)) {
|
|
143
|
-
if (Preference.TreeNode.is(node)) {
|
|
144
|
-
const { collection, id } = this.analyzeIDAndGetRendererGroup(node.id);
|
|
145
|
-
const renderer = collection.get(id) ?? this.rendererFactory(node);
|
|
146
|
-
if (!renderer.node.parentElement) { // If it hasn't been attached yet, it hasn't been checked for the current search.
|
|
147
|
-
this.hideIfFailsFilters(renderer, isFiltered);
|
|
148
|
-
collection.set(id, renderer);
|
|
149
|
-
}
|
|
150
|
-
if (nextNode !== renderer.node) {
|
|
151
|
-
if (nextNode) {
|
|
152
|
-
renderer.insertBefore(nextNode);
|
|
153
|
-
} else {
|
|
154
|
-
renderer.appendTo(this.scrollContainer);
|
|
155
|
-
}
|
|
156
|
-
} else {
|
|
157
|
-
nextNode = nodeIterator.next().value;
|
|
158
|
-
}
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
protected handleScopeChange(isFiltered: boolean = this.model.isFiltered): void {
|
|
165
|
-
for (const [, renderer] of this.allRenderers()) {
|
|
166
|
-
const isHidden = this.hideIfFailsFilters(renderer, isFiltered);
|
|
167
|
-
if (isFiltered || !isHidden) {
|
|
168
|
-
renderer.handleScopeChange?.(isFiltered);
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
protected hasSchemaChanged(renderer: GeneralPreferenceNodeRenderer, node: Preference.LeafNode): boolean {
|
|
174
|
-
return !deepEqual(renderer.schema, node.preference.data);
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
protected handleSearchChange(isFiltered: boolean, leavesAreVisible: boolean): void {
|
|
178
|
-
if (leavesAreVisible) {
|
|
179
|
-
for (const [, renderer] of this.allRenderers()) {
|
|
180
|
-
const isHidden = this.hideIfFailsFilters(renderer, isFiltered);
|
|
181
|
-
if (!isHidden) {
|
|
182
|
-
renderer.handleSearchChange?.(isFiltered);
|
|
183
|
-
}
|
|
184
|
-
}
|
|
185
|
-
}
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
protected areLeavesVisible(): boolean {
|
|
189
|
-
const leavesAreVisible = this.model.totalVisibleLeaves > 0;
|
|
190
|
-
this.node.classList.toggle('no-results', !leavesAreVisible);
|
|
191
|
-
this.scrollContainer.classList.toggle('hidden', !leavesAreVisible);
|
|
192
|
-
return leavesAreVisible;
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
protected *allRenderers(): IterableIterator<[string, GeneralPreferenceNodeRenderer, Map<string, GeneralPreferenceNodeRenderer>]> {
|
|
196
|
-
for (const [id, renderer] of this.commonlyUsedRenderers.entries()) {
|
|
197
|
-
yield [id, renderer, this.commonlyUsedRenderers];
|
|
198
|
-
}
|
|
199
|
-
for (const [id, renderer] of this.renderers.entries()) {
|
|
200
|
-
yield [id, renderer, this.renderers];
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
protected handlePreferenceChanges(e: PreferenceChanges): void {
|
|
205
|
-
for (const id of Object.keys(e)) {
|
|
206
|
-
this.commonlyUsedRenderers.get(id)?.handleValueChange?.();
|
|
207
|
-
this.renderers.get(id)?.handleValueChange?.();
|
|
208
|
-
}
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
/**
|
|
212
|
-
* @returns true if the renderer is hidden, false otherwise.
|
|
213
|
-
*/
|
|
214
|
-
protected hideIfFailsFilters(renderer: GeneralPreferenceNodeRenderer, isFiltered: boolean): boolean {
|
|
215
|
-
const row = this.model.currentRows.get(renderer.nodeId);
|
|
216
|
-
if (!row || (CompositeTreeNode.is(row.node) && (isFiltered || row.visibleChildren === 0))) {
|
|
217
|
-
renderer.hide();
|
|
218
|
-
return true;
|
|
219
|
-
} else {
|
|
220
|
-
renderer.show();
|
|
221
|
-
return false;
|
|
222
|
-
}
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
protected resetScroll(nodeIDToScrollTo?: string, filterWasCleared: boolean = false): void {
|
|
226
|
-
if (this.scrollBar) { // Absent on widget creation
|
|
227
|
-
this.doResetScroll(nodeIDToScrollTo, filterWasCleared);
|
|
228
|
-
} else {
|
|
229
|
-
const interval = setInterval(() => {
|
|
230
|
-
if (this.scrollBar) {
|
|
231
|
-
clearInterval(interval);
|
|
232
|
-
this.doResetScroll(nodeIDToScrollTo, filterWasCleared);
|
|
233
|
-
}
|
|
234
|
-
}, 500);
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
protected doResetScroll(nodeIDToScrollTo?: string, filterWasCleared: boolean = false): void {
|
|
239
|
-
requestAnimationFrame(() => {
|
|
240
|
-
this.scrollBar?.update();
|
|
241
|
-
if (!filterWasCleared && nodeIDToScrollTo) {
|
|
242
|
-
const { id, collection } = this.analyzeIDAndGetRendererGroup(nodeIDToScrollTo);
|
|
243
|
-
const renderer = collection.get(id);
|
|
244
|
-
if (renderer?.visible) {
|
|
245
|
-
renderer.node.scrollIntoView();
|
|
246
|
-
return;
|
|
247
|
-
}
|
|
248
|
-
}
|
|
249
|
-
this.scrollContainer.scrollTop = 0;
|
|
250
|
-
});
|
|
251
|
-
};
|
|
252
|
-
|
|
253
|
-
protected doOnScroll(): void {
|
|
254
|
-
const { scrollContainer } = this;
|
|
255
|
-
const firstVisibleChildID = this.findFirstVisibleChildID();
|
|
256
|
-
this.setFirstVisibleChildID(firstVisibleChildID);
|
|
257
|
-
if (this.isAtScrollTop && scrollContainer.scrollTop !== 0) {
|
|
258
|
-
this.isAtScrollTop = false;
|
|
259
|
-
this.tabbar.toggleShadow(true);
|
|
260
|
-
} else if (!this.isAtScrollTop && scrollContainer.scrollTop === 0) {
|
|
261
|
-
this.isAtScrollTop = true;
|
|
262
|
-
this.tabbar.toggleShadow(false);
|
|
263
|
-
}
|
|
264
|
-
};
|
|
265
|
-
|
|
266
|
-
onScroll = throttle(this.doOnScroll.bind(this), 50);
|
|
267
|
-
|
|
268
|
-
protected findFirstVisibleChildID(): string | undefined {
|
|
269
|
-
const { scrollTop } = this.scrollContainer;
|
|
270
|
-
for (const [, renderer] of this.allRenderers()) {
|
|
271
|
-
const { offsetTop, offsetHeight } = renderer.node;
|
|
272
|
-
if (Math.abs(offsetTop - scrollTop) <= offsetHeight / 2) {
|
|
273
|
-
return renderer.nodeId;
|
|
274
|
-
}
|
|
275
|
-
}
|
|
276
|
-
}
|
|
277
|
-
|
|
278
|
-
protected setFirstVisibleChildID(id?: string): void {
|
|
279
|
-
if (id && id !== this.firstVisibleChildID) {
|
|
280
|
-
this.firstVisibleChildID = id;
|
|
281
|
-
let currentNode = this.model.getNode(id);
|
|
282
|
-
let expansionAncestor;
|
|
283
|
-
let selectionAncestor;
|
|
284
|
-
while (currentNode && (!expansionAncestor || !selectionAncestor)) {
|
|
285
|
-
if (!selectionAncestor && SelectableTreeNode.is(currentNode)) {
|
|
286
|
-
selectionAncestor = currentNode;
|
|
287
|
-
}
|
|
288
|
-
if (!expansionAncestor && ExpandableTreeNode.is(currentNode)) {
|
|
289
|
-
expansionAncestor = currentNode;
|
|
290
|
-
}
|
|
291
|
-
currentNode = currentNode.parent;
|
|
292
|
-
}
|
|
293
|
-
if (selectionAncestor) {
|
|
294
|
-
this.currentModelSelectionId = selectionAncestor.id;
|
|
295
|
-
expansionAncestor = expansionAncestor ?? selectionAncestor;
|
|
296
|
-
this.model.selectIfNotSelected(selectionAncestor);
|
|
297
|
-
if (!this.model.isFiltered && id !== this.lastUserSelection) {
|
|
298
|
-
this.lastUserSelection = '';
|
|
299
|
-
this.model.collapseAllExcept(expansionAncestor);
|
|
300
|
-
}
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
protected handleSelectionChange(selectionEvent: readonly Readonly<SelectableTreeNode>[]): void {
|
|
306
|
-
const node = selectionEvent[0];
|
|
307
|
-
if (node && node.id !== this.currentModelSelectionId) {
|
|
308
|
-
this.currentModelSelectionId = node.id;
|
|
309
|
-
this.lastUserSelection = node.id;
|
|
310
|
-
if (this.model.isFiltered && CompositeTreeNode.is(node)) {
|
|
311
|
-
for (const candidate of new TopDownTreeIterator(node, { pruneSiblings: true })) {
|
|
312
|
-
const { id, collection } = this.analyzeIDAndGetRendererGroup(candidate.id);
|
|
313
|
-
const renderer = collection.get(id);
|
|
314
|
-
if (renderer?.visible) {
|
|
315
|
-
// When filtered, treat the first visible child as the selected node, since it will be the one scrolled to.
|
|
316
|
-
this.lastUserSelection = renderer.nodeId;
|
|
317
|
-
renderer.node.scrollIntoView();
|
|
318
|
-
return;
|
|
319
|
-
}
|
|
320
|
-
}
|
|
321
|
-
} else {
|
|
322
|
-
const { id, collection } = this.analyzeIDAndGetRendererGroup(node.id);
|
|
323
|
-
const renderer = collection.get(id);
|
|
324
|
-
renderer?.node.scrollIntoView();
|
|
325
|
-
}
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
|
|
329
|
-
protected analyzeIDAndGetRendererGroup(nodeID: string): { id: string, group: string, collection: Map<string, GeneralPreferenceNodeRenderer> } {
|
|
330
|
-
const { id, group } = Preference.TreeNode.getGroupAndIdFromNodeId(nodeID);
|
|
331
|
-
const collection = group === COMMONLY_USED_SECTION_PREFIX ? this.commonlyUsedRenderers : this.renderers;
|
|
332
|
-
return { id, group, collection };
|
|
333
|
-
}
|
|
334
|
-
|
|
335
|
-
protected override getScrollContainer(): HTMLElement {
|
|
336
|
-
return this.scrollContainer;
|
|
337
|
-
}
|
|
338
|
-
|
|
339
|
-
storeState(): PreferencesEditorState {
|
|
340
|
-
return {
|
|
341
|
-
firstVisibleChildID: this.firstVisibleChildID,
|
|
342
|
-
};
|
|
343
|
-
}
|
|
344
|
-
|
|
345
|
-
restoreState(oldState: PreferencesEditorState): void {
|
|
346
|
-
this.firstVisibleChildID = oldState.firstVisibleChildID;
|
|
347
|
-
this.resetScroll(this.firstVisibleChildID);
|
|
348
|
-
}
|
|
349
|
-
}
|
|
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
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
18
|
+
import { postConstruct, injectable, inject } from '@theia/core/shared/inversify';
|
|
19
|
+
import throttle = require('@theia/core/shared/lodash.throttle');
|
|
20
|
+
import * as deepEqual from 'fast-deep-equal';
|
|
21
|
+
import {
|
|
22
|
+
PreferenceService,
|
|
23
|
+
CompositeTreeNode,
|
|
24
|
+
SelectableTreeNode,
|
|
25
|
+
StatefulWidget,
|
|
26
|
+
TopDownTreeIterator,
|
|
27
|
+
PreferenceChanges,
|
|
28
|
+
ExpandableTreeNode,
|
|
29
|
+
PreferenceSchemaProvider,
|
|
30
|
+
} from '@theia/core/lib/browser';
|
|
31
|
+
import { unreachable } from '@theia/core/lib/common';
|
|
32
|
+
import { BaseWidget, DEFAULT_SCROLL_OPTIONS } from '@theia/core/lib/browser/widgets/widget';
|
|
33
|
+
import { PreferenceTreeModel, PreferenceFilterChangeEvent, PreferenceFilterChangeSource } from '../preference-tree-model';
|
|
34
|
+
import { PreferenceNodeRendererFactory, GeneralPreferenceNodeRenderer } from './components/preference-node-renderer';
|
|
35
|
+
import { Preference } from '../util/preference-types';
|
|
36
|
+
import { COMMONLY_USED_SECTION_PREFIX } from '../util/preference-tree-generator';
|
|
37
|
+
import { PreferencesScopeTabBar } from './preference-scope-tabbar-widget';
|
|
38
|
+
import { PreferenceNodeRendererCreatorRegistry } from './components/preference-node-renderer-creator';
|
|
39
|
+
|
|
40
|
+
export interface PreferencesEditorState {
|
|
41
|
+
firstVisibleChildID: string,
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
@injectable()
|
|
45
|
+
export class PreferencesEditorWidget extends BaseWidget implements StatefulWidget {
|
|
46
|
+
static readonly ID = 'settings.editor';
|
|
47
|
+
static readonly LABEL = 'Settings Editor';
|
|
48
|
+
|
|
49
|
+
override scrollOptions = DEFAULT_SCROLL_OPTIONS;
|
|
50
|
+
|
|
51
|
+
protected scrollContainer: HTMLDivElement;
|
|
52
|
+
/**
|
|
53
|
+
* Guards against scroll events and selection events looping into each other. Set before this widget initiates a selection.
|
|
54
|
+
*/
|
|
55
|
+
protected currentModelSelectionId = '';
|
|
56
|
+
/**
|
|
57
|
+
* Permits the user to expand multiple nodes without each one being collapsed on a new selection.
|
|
58
|
+
*/
|
|
59
|
+
protected lastUserSelection = '';
|
|
60
|
+
protected isAtScrollTop = true;
|
|
61
|
+
protected firstVisibleChildID = '';
|
|
62
|
+
protected renderers = new Map<string, GeneralPreferenceNodeRenderer>();
|
|
63
|
+
protected preferenceDataKeys = new Map<string, string>();
|
|
64
|
+
// The commonly used section will duplicate preference ID's, so we'll keep a separate list of them.
|
|
65
|
+
protected commonlyUsedRenderers = new Map<string, GeneralPreferenceNodeRenderer>();
|
|
66
|
+
|
|
67
|
+
@inject(PreferenceService) protected readonly preferenceService: PreferenceService;
|
|
68
|
+
@inject(PreferenceTreeModel) protected readonly model: PreferenceTreeModel;
|
|
69
|
+
@inject(PreferenceNodeRendererFactory) protected readonly rendererFactory: PreferenceNodeRendererFactory;
|
|
70
|
+
@inject(PreferenceNodeRendererCreatorRegistry) protected readonly rendererRegistry: PreferenceNodeRendererCreatorRegistry;
|
|
71
|
+
@inject(PreferenceSchemaProvider) protected readonly schemaProvider: PreferenceSchemaProvider;
|
|
72
|
+
@inject(PreferencesScopeTabBar) protected readonly tabbar: PreferencesScopeTabBar;
|
|
73
|
+
|
|
74
|
+
@postConstruct()
|
|
75
|
+
protected init(): void {
|
|
76
|
+
this.doInit();
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
protected async doInit(): Promise<void> {
|
|
80
|
+
|
|
81
|
+
this.id = PreferencesEditorWidget.ID;
|
|
82
|
+
this.title.label = PreferencesEditorWidget.LABEL;
|
|
83
|
+
this.addClass('settings-main');
|
|
84
|
+
this.toDispose.pushAll([
|
|
85
|
+
this.preferenceService.onPreferencesChanged(e => this.handlePreferenceChanges(e)),
|
|
86
|
+
this.model.onFilterChanged(e => this.handleDisplayChange(e)),
|
|
87
|
+
this.model.onSelectionChanged(e => this.handleSelectionChange(e)),
|
|
88
|
+
]);
|
|
89
|
+
this.createContainers();
|
|
90
|
+
await this.preferenceService.ready;
|
|
91
|
+
this.handleDisplayChange({ source: PreferenceFilterChangeSource.Schema });
|
|
92
|
+
this.rendererRegistry.onDidChange(() => this.handleRegistryChange());
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
protected createContainers(): void {
|
|
96
|
+
const innerWrapper = document.createElement('div');
|
|
97
|
+
innerWrapper.classList.add('settings-main-scroll-container');
|
|
98
|
+
this.scrollContainer = innerWrapper;
|
|
99
|
+
innerWrapper.addEventListener('scroll', this.onScroll, { passive: true });
|
|
100
|
+
this.node.appendChild(innerWrapper);
|
|
101
|
+
const noLeavesMessage = document.createElement('div');
|
|
102
|
+
noLeavesMessage.classList.add('settings-no-results-announcement');
|
|
103
|
+
noLeavesMessage.textContent = 'That search query has returned no results.';
|
|
104
|
+
this.node.appendChild(noLeavesMessage);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
protected handleDisplayChange(e: PreferenceFilterChangeEvent): void {
|
|
108
|
+
const { isFiltered } = this.model;
|
|
109
|
+
const currentFirstVisible = this.firstVisibleChildID;
|
|
110
|
+
const leavesAreVisible = this.areLeavesVisible();
|
|
111
|
+
if (e.source === PreferenceFilterChangeSource.Search) {
|
|
112
|
+
this.handleSearchChange(isFiltered, leavesAreVisible);
|
|
113
|
+
} else if (e.source === PreferenceFilterChangeSource.Scope) {
|
|
114
|
+
this.handleScopeChange(isFiltered);
|
|
115
|
+
} else if (e.source === PreferenceFilterChangeSource.Schema) {
|
|
116
|
+
this.handleSchemaChange(isFiltered);
|
|
117
|
+
} else {
|
|
118
|
+
unreachable(e.source, 'Not all PreferenceFilterChangeSource enum variants handled.');
|
|
119
|
+
}
|
|
120
|
+
this.resetScroll(currentFirstVisible, e.source === PreferenceFilterChangeSource.Search && !isFiltered);
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
protected handleRegistryChange(): void {
|
|
124
|
+
for (const [id, renderer, collection] of this.allRenderers()) {
|
|
125
|
+
renderer.dispose();
|
|
126
|
+
collection.delete(id);
|
|
127
|
+
}
|
|
128
|
+
this.handleDisplayChange({ source: PreferenceFilterChangeSource.Schema });
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
protected handleSchemaChange(isFiltered: boolean): void {
|
|
132
|
+
for (const [id, renderer, collection] of this.allRenderers()) {
|
|
133
|
+
const node = this.model.getNode(renderer.nodeId);
|
|
134
|
+
if (!node || (Preference.LeafNode.is(node) && this.hasSchemaChanged(renderer, node))) {
|
|
135
|
+
renderer.dispose();
|
|
136
|
+
collection.delete(id);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
if (this.model.root) {
|
|
140
|
+
const nodeIterator = Array.from(this.scrollContainer.children)[Symbol.iterator]();
|
|
141
|
+
let nextNode: HTMLElement | undefined = nodeIterator.next().value;
|
|
142
|
+
for (const node of new TopDownTreeIterator(this.model.root)) {
|
|
143
|
+
if (Preference.TreeNode.is(node)) {
|
|
144
|
+
const { collection, id } = this.analyzeIDAndGetRendererGroup(node.id);
|
|
145
|
+
const renderer = collection.get(id) ?? this.rendererFactory(node);
|
|
146
|
+
if (!renderer.node.parentElement) { // If it hasn't been attached yet, it hasn't been checked for the current search.
|
|
147
|
+
this.hideIfFailsFilters(renderer, isFiltered);
|
|
148
|
+
collection.set(id, renderer);
|
|
149
|
+
}
|
|
150
|
+
if (nextNode !== renderer.node) {
|
|
151
|
+
if (nextNode) {
|
|
152
|
+
renderer.insertBefore(nextNode);
|
|
153
|
+
} else {
|
|
154
|
+
renderer.appendTo(this.scrollContainer);
|
|
155
|
+
}
|
|
156
|
+
} else {
|
|
157
|
+
nextNode = nodeIterator.next().value;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
protected handleScopeChange(isFiltered: boolean = this.model.isFiltered): void {
|
|
165
|
+
for (const [, renderer] of this.allRenderers()) {
|
|
166
|
+
const isHidden = this.hideIfFailsFilters(renderer, isFiltered);
|
|
167
|
+
if (isFiltered || !isHidden) {
|
|
168
|
+
renderer.handleScopeChange?.(isFiltered);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
protected hasSchemaChanged(renderer: GeneralPreferenceNodeRenderer, node: Preference.LeafNode): boolean {
|
|
174
|
+
return !deepEqual(renderer.schema, node.preference.data);
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
protected handleSearchChange(isFiltered: boolean, leavesAreVisible: boolean): void {
|
|
178
|
+
if (leavesAreVisible) {
|
|
179
|
+
for (const [, renderer] of this.allRenderers()) {
|
|
180
|
+
const isHidden = this.hideIfFailsFilters(renderer, isFiltered);
|
|
181
|
+
if (!isHidden) {
|
|
182
|
+
renderer.handleSearchChange?.(isFiltered);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
protected areLeavesVisible(): boolean {
|
|
189
|
+
const leavesAreVisible = this.model.totalVisibleLeaves > 0;
|
|
190
|
+
this.node.classList.toggle('no-results', !leavesAreVisible);
|
|
191
|
+
this.scrollContainer.classList.toggle('hidden', !leavesAreVisible);
|
|
192
|
+
return leavesAreVisible;
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
protected *allRenderers(): IterableIterator<[string, GeneralPreferenceNodeRenderer, Map<string, GeneralPreferenceNodeRenderer>]> {
|
|
196
|
+
for (const [id, renderer] of this.commonlyUsedRenderers.entries()) {
|
|
197
|
+
yield [id, renderer, this.commonlyUsedRenderers];
|
|
198
|
+
}
|
|
199
|
+
for (const [id, renderer] of this.renderers.entries()) {
|
|
200
|
+
yield [id, renderer, this.renderers];
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
protected handlePreferenceChanges(e: PreferenceChanges): void {
|
|
205
|
+
for (const id of Object.keys(e)) {
|
|
206
|
+
this.commonlyUsedRenderers.get(id)?.handleValueChange?.();
|
|
207
|
+
this.renderers.get(id)?.handleValueChange?.();
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
/**
|
|
212
|
+
* @returns true if the renderer is hidden, false otherwise.
|
|
213
|
+
*/
|
|
214
|
+
protected hideIfFailsFilters(renderer: GeneralPreferenceNodeRenderer, isFiltered: boolean): boolean {
|
|
215
|
+
const row = this.model.currentRows.get(renderer.nodeId);
|
|
216
|
+
if (!row || (CompositeTreeNode.is(row.node) && (isFiltered || row.visibleChildren === 0))) {
|
|
217
|
+
renderer.hide();
|
|
218
|
+
return true;
|
|
219
|
+
} else {
|
|
220
|
+
renderer.show();
|
|
221
|
+
return false;
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
protected resetScroll(nodeIDToScrollTo?: string, filterWasCleared: boolean = false): void {
|
|
226
|
+
if (this.scrollBar) { // Absent on widget creation
|
|
227
|
+
this.doResetScroll(nodeIDToScrollTo, filterWasCleared);
|
|
228
|
+
} else {
|
|
229
|
+
const interval = setInterval(() => {
|
|
230
|
+
if (this.scrollBar) {
|
|
231
|
+
clearInterval(interval);
|
|
232
|
+
this.doResetScroll(nodeIDToScrollTo, filterWasCleared);
|
|
233
|
+
}
|
|
234
|
+
}, 500);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
protected doResetScroll(nodeIDToScrollTo?: string, filterWasCleared: boolean = false): void {
|
|
239
|
+
requestAnimationFrame(() => {
|
|
240
|
+
this.scrollBar?.update();
|
|
241
|
+
if (!filterWasCleared && nodeIDToScrollTo) {
|
|
242
|
+
const { id, collection } = this.analyzeIDAndGetRendererGroup(nodeIDToScrollTo);
|
|
243
|
+
const renderer = collection.get(id);
|
|
244
|
+
if (renderer?.visible) {
|
|
245
|
+
renderer.node.scrollIntoView();
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
this.scrollContainer.scrollTop = 0;
|
|
250
|
+
});
|
|
251
|
+
};
|
|
252
|
+
|
|
253
|
+
protected doOnScroll(): void {
|
|
254
|
+
const { scrollContainer } = this;
|
|
255
|
+
const firstVisibleChildID = this.findFirstVisibleChildID();
|
|
256
|
+
this.setFirstVisibleChildID(firstVisibleChildID);
|
|
257
|
+
if (this.isAtScrollTop && scrollContainer.scrollTop !== 0) {
|
|
258
|
+
this.isAtScrollTop = false;
|
|
259
|
+
this.tabbar.toggleShadow(true);
|
|
260
|
+
} else if (!this.isAtScrollTop && scrollContainer.scrollTop === 0) {
|
|
261
|
+
this.isAtScrollTop = true;
|
|
262
|
+
this.tabbar.toggleShadow(false);
|
|
263
|
+
}
|
|
264
|
+
};
|
|
265
|
+
|
|
266
|
+
onScroll = throttle(this.doOnScroll.bind(this), 50);
|
|
267
|
+
|
|
268
|
+
protected findFirstVisibleChildID(): string | undefined {
|
|
269
|
+
const { scrollTop } = this.scrollContainer;
|
|
270
|
+
for (const [, renderer] of this.allRenderers()) {
|
|
271
|
+
const { offsetTop, offsetHeight } = renderer.node;
|
|
272
|
+
if (Math.abs(offsetTop - scrollTop) <= offsetHeight / 2) {
|
|
273
|
+
return renderer.nodeId;
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
protected setFirstVisibleChildID(id?: string): void {
|
|
279
|
+
if (id && id !== this.firstVisibleChildID) {
|
|
280
|
+
this.firstVisibleChildID = id;
|
|
281
|
+
let currentNode = this.model.getNode(id);
|
|
282
|
+
let expansionAncestor;
|
|
283
|
+
let selectionAncestor;
|
|
284
|
+
while (currentNode && (!expansionAncestor || !selectionAncestor)) {
|
|
285
|
+
if (!selectionAncestor && SelectableTreeNode.is(currentNode)) {
|
|
286
|
+
selectionAncestor = currentNode;
|
|
287
|
+
}
|
|
288
|
+
if (!expansionAncestor && ExpandableTreeNode.is(currentNode)) {
|
|
289
|
+
expansionAncestor = currentNode;
|
|
290
|
+
}
|
|
291
|
+
currentNode = currentNode.parent;
|
|
292
|
+
}
|
|
293
|
+
if (selectionAncestor) {
|
|
294
|
+
this.currentModelSelectionId = selectionAncestor.id;
|
|
295
|
+
expansionAncestor = expansionAncestor ?? selectionAncestor;
|
|
296
|
+
this.model.selectIfNotSelected(selectionAncestor);
|
|
297
|
+
if (!this.model.isFiltered && id !== this.lastUserSelection) {
|
|
298
|
+
this.lastUserSelection = '';
|
|
299
|
+
this.model.collapseAllExcept(expansionAncestor);
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
protected handleSelectionChange(selectionEvent: readonly Readonly<SelectableTreeNode>[]): void {
|
|
306
|
+
const node = selectionEvent[0];
|
|
307
|
+
if (node && node.id !== this.currentModelSelectionId) {
|
|
308
|
+
this.currentModelSelectionId = node.id;
|
|
309
|
+
this.lastUserSelection = node.id;
|
|
310
|
+
if (this.model.isFiltered && CompositeTreeNode.is(node)) {
|
|
311
|
+
for (const candidate of new TopDownTreeIterator(node, { pruneSiblings: true })) {
|
|
312
|
+
const { id, collection } = this.analyzeIDAndGetRendererGroup(candidate.id);
|
|
313
|
+
const renderer = collection.get(id);
|
|
314
|
+
if (renderer?.visible) {
|
|
315
|
+
// When filtered, treat the first visible child as the selected node, since it will be the one scrolled to.
|
|
316
|
+
this.lastUserSelection = renderer.nodeId;
|
|
317
|
+
renderer.node.scrollIntoView();
|
|
318
|
+
return;
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
} else {
|
|
322
|
+
const { id, collection } = this.analyzeIDAndGetRendererGroup(node.id);
|
|
323
|
+
const renderer = collection.get(id);
|
|
324
|
+
renderer?.node.scrollIntoView();
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
protected analyzeIDAndGetRendererGroup(nodeID: string): { id: string, group: string, collection: Map<string, GeneralPreferenceNodeRenderer> } {
|
|
330
|
+
const { id, group } = Preference.TreeNode.getGroupAndIdFromNodeId(nodeID);
|
|
331
|
+
const collection = group === COMMONLY_USED_SECTION_PREFIX ? this.commonlyUsedRenderers : this.renderers;
|
|
332
|
+
return { id, group, collection };
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
protected override getScrollContainer(): HTMLElement {
|
|
336
|
+
return this.scrollContainer;
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
storeState(): PreferencesEditorState {
|
|
340
|
+
return {
|
|
341
|
+
firstVisibleChildID: this.firstVisibleChildID,
|
|
342
|
+
};
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
restoreState(oldState: PreferencesEditorState): void {
|
|
346
|
+
this.firstVisibleChildID = oldState.firstVisibleChildID;
|
|
347
|
+
this.resetScroll(this.firstVisibleChildID);
|
|
348
|
+
}
|
|
349
|
+
}
|