@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,244 +1,244 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2019 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
|
-
|
|
19
|
-
import { inject, injectable, postConstruct } from '@theia/core/shared/inversify';
|
|
20
|
-
import URI from '@theia/core/lib/common/uri';
|
|
21
|
-
import { PreferenceProvider, PreferenceResolveResult, PreferenceScope } from '@theia/core/lib/browser/preferences';
|
|
22
|
-
import { WorkspaceService } from '@theia/workspace/lib/browser/workspace-service';
|
|
23
|
-
import { PreferenceConfigurations } from '@theia/core/lib/browser/preferences/preference-configurations';
|
|
24
|
-
import { FolderPreferenceProvider, FolderPreferenceProviderFactory } from './folder-preference-provider';
|
|
25
|
-
import { FileStat } from '@theia/filesystem/lib/common/files';
|
|
26
|
-
|
|
27
|
-
@injectable()
|
|
28
|
-
export class FoldersPreferencesProvider extends PreferenceProvider {
|
|
29
|
-
|
|
30
|
-
@inject(WorkspaceService)
|
|
31
|
-
protected readonly workspaceService: WorkspaceService;
|
|
32
|
-
|
|
33
|
-
@inject(FolderPreferenceProviderFactory)
|
|
34
|
-
protected readonly folderPreferenceProviderFactory: FolderPreferenceProviderFactory;
|
|
35
|
-
|
|
36
|
-
@inject(PreferenceConfigurations)
|
|
37
|
-
protected readonly configurations: PreferenceConfigurations;
|
|
38
|
-
|
|
39
|
-
protected readonly providers = new Map<string, FolderPreferenceProvider>();
|
|
40
|
-
|
|
41
|
-
@postConstruct()
|
|
42
|
-
protected init(): void {
|
|
43
|
-
this.doInit();
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
protected async doInit(): Promise<void> {
|
|
47
|
-
await this.workspaceService.roots;
|
|
48
|
-
|
|
49
|
-
this.updateProviders();
|
|
50
|
-
this.workspaceService.onWorkspaceChanged(() => this.updateProviders());
|
|
51
|
-
|
|
52
|
-
const readyPromises: Promise<void>[] = [];
|
|
53
|
-
for (const provider of this.providers.values()) {
|
|
54
|
-
readyPromises.push(provider.ready.catch(e => console.error(e)));
|
|
55
|
-
}
|
|
56
|
-
Promise.all(readyPromises).then(() => this._ready.resolve());
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
protected updateProviders(): void {
|
|
60
|
-
const roots = this.workspaceService.tryGetRoots();
|
|
61
|
-
const toDelete = new Set(this.providers.keys());
|
|
62
|
-
for (const folder of roots) {
|
|
63
|
-
for (const configPath of this.configurations.getPaths()) {
|
|
64
|
-
for (const configName of [...this.configurations.getSectionNames(), this.configurations.getConfigName()]) {
|
|
65
|
-
const sectionUri = this.configurations.createUri(folder.resource, configPath, configName);
|
|
66
|
-
const sectionKey = sectionUri.toString();
|
|
67
|
-
toDelete.delete(sectionKey);
|
|
68
|
-
if (!this.providers.has(sectionKey)) {
|
|
69
|
-
const provider = this.createProvider(sectionUri, configName, folder);
|
|
70
|
-
this.providers.set(sectionKey, provider);
|
|
71
|
-
}
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
for (const key of toDelete) {
|
|
76
|
-
const provider = this.providers.get(key);
|
|
77
|
-
if (provider) {
|
|
78
|
-
this.providers.delete(key);
|
|
79
|
-
provider.dispose();
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
override getConfigUri(resourceUri?: string, sectionName: string = this.configurations.getConfigName()): URI | undefined {
|
|
85
|
-
for (const provider of this.getFolderProviders(resourceUri)) {
|
|
86
|
-
const configUri = provider.getConfigUri(resourceUri);
|
|
87
|
-
if (configUri && this.configurations.getName(configUri) === sectionName) {
|
|
88
|
-
return configUri;
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
return undefined;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
override getContainingConfigUri(resourceUri?: string, sectionName: string = this.configurations.getConfigName()): URI | undefined {
|
|
95
|
-
for (const provider of this.getFolderProviders(resourceUri)) {
|
|
96
|
-
const configUri = provider.getConfigUri();
|
|
97
|
-
if (provider.contains(resourceUri) && this.configurations.getName(configUri) === sectionName) {
|
|
98
|
-
return configUri;
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
return undefined;
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
override getDomain(): string[] {
|
|
105
|
-
return this.workspaceService.tryGetRoots().map(root => root.resource.toString());
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
override resolve<T>(preferenceName: string, resourceUri?: string): PreferenceResolveResult<T> {
|
|
109
|
-
const result: PreferenceResolveResult<T> = {};
|
|
110
|
-
const groups = this.groupProvidersByConfigName(resourceUri);
|
|
111
|
-
for (const group of groups.values()) {
|
|
112
|
-
for (const provider of group) {
|
|
113
|
-
const { value, configUri } = provider.resolve<T>(preferenceName, resourceUri);
|
|
114
|
-
if (configUri && value !== undefined) {
|
|
115
|
-
result.configUri = configUri;
|
|
116
|
-
result.value = PreferenceProvider.merge(result.value as any, value as any) as any;
|
|
117
|
-
break;
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
return result;
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
getPreferences(resourceUri?: string): { [p: string]: any } {
|
|
125
|
-
let result = {};
|
|
126
|
-
const groups = this.groupProvidersByConfigName(resourceUri);
|
|
127
|
-
for (const group of groups.values()) {
|
|
128
|
-
for (const provider of group) {
|
|
129
|
-
if (provider.getConfigUri(resourceUri)) {
|
|
130
|
-
const preferences = provider.getPreferences();
|
|
131
|
-
result = PreferenceProvider.merge(result, preferences) as any;
|
|
132
|
-
break;
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
}
|
|
136
|
-
return result;
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
async setPreference(preferenceName: string, value: any, resourceUri?: string): Promise<boolean> {
|
|
140
|
-
const firstPathFragment = preferenceName.split('.', 1)[0];
|
|
141
|
-
const defaultConfigName = this.configurations.getConfigName();
|
|
142
|
-
const configName = this.configurations.isSectionName(firstPathFragment) ? firstPathFragment : defaultConfigName;
|
|
143
|
-
|
|
144
|
-
const providers = this.getFolderProviders(resourceUri);
|
|
145
|
-
let configPath: string | undefined;
|
|
146
|
-
const candidates = providers.filter(provider => {
|
|
147
|
-
// Attempt to figure out the settings folder (.vscode or .theia) we're interested in.
|
|
148
|
-
const containingConfigUri = provider.getConfigUri(resourceUri);
|
|
149
|
-
if (configPath === undefined && containingConfigUri) {
|
|
150
|
-
configPath = this.configurations.getPath(containingConfigUri);
|
|
151
|
-
}
|
|
152
|
-
const providerName = this.configurations.getName(containingConfigUri ?? provider.getConfigUri());
|
|
153
|
-
return providerName === configName || providerName === defaultConfigName;
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
const configNameAndPathMatches = [];
|
|
157
|
-
const configNameOnlyMatches = [];
|
|
158
|
-
const configUriMatches = [];
|
|
159
|
-
const otherMatches = [];
|
|
160
|
-
|
|
161
|
-
for (const candidate of candidates) {
|
|
162
|
-
const domainMatches = candidate.getConfigUri(resourceUri);
|
|
163
|
-
const configUri = domainMatches ?? candidate.getConfigUri();
|
|
164
|
-
const nameMatches = this.configurations.getName(configUri) === configName;
|
|
165
|
-
const pathMatches = this.configurations.getPath(configUri) === configPath;
|
|
166
|
-
|
|
167
|
-
// Perfect match, run immediately in case we can bail out early.
|
|
168
|
-
if (nameMatches && domainMatches) {
|
|
169
|
-
if (await candidate.setPreference(preferenceName, value, resourceUri)) {
|
|
170
|
-
return true;
|
|
171
|
-
}
|
|
172
|
-
} else if (nameMatches && pathMatches) { // Right file in the right folder.
|
|
173
|
-
configNameAndPathMatches.push(candidate);
|
|
174
|
-
} else if (nameMatches) { // Right file.
|
|
175
|
-
configNameOnlyMatches.push(candidate);
|
|
176
|
-
} else if (domainMatches) { // Currently valid and governs target URI
|
|
177
|
-
configUriMatches.push(candidate);
|
|
178
|
-
} else {
|
|
179
|
-
otherMatches.push(candidate);
|
|
180
|
-
}
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
const candidateSets = [configNameAndPathMatches, configNameOnlyMatches, configUriMatches, otherMatches];
|
|
184
|
-
|
|
185
|
-
for (const candidateSet of candidateSets) {
|
|
186
|
-
for (const candidate of candidateSet) {
|
|
187
|
-
if (await candidate.setPreference(preferenceName, value, resourceUri)) {
|
|
188
|
-
return true;
|
|
189
|
-
}
|
|
190
|
-
}
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
return false;
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
override canHandleScope(scope: PreferenceScope): boolean {
|
|
197
|
-
return this.workspaceService.isMultiRootWorkspaceOpened && scope === PreferenceScope.Folder || scope === PreferenceScope.Workspace;
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
protected groupProvidersByConfigName(resourceUri?: string): Map<string, FolderPreferenceProvider[]> {
|
|
201
|
-
const groups = new Map<string, FolderPreferenceProvider[]>();
|
|
202
|
-
const providers = this.getFolderProviders(resourceUri);
|
|
203
|
-
for (const configName of [this.configurations.getConfigName(), ...this.configurations.getSectionNames()]) {
|
|
204
|
-
const group = [];
|
|
205
|
-
for (const provider of providers) {
|
|
206
|
-
if (this.configurations.getName(provider.getConfigUri()) === configName) {
|
|
207
|
-
group.push(provider);
|
|
208
|
-
}
|
|
209
|
-
}
|
|
210
|
-
groups.set(configName, group);
|
|
211
|
-
}
|
|
212
|
-
return groups;
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
protected getFolderProviders(resourceUri?: string): FolderPreferenceProvider[] {
|
|
216
|
-
if (!resourceUri) {
|
|
217
|
-
return [];
|
|
218
|
-
}
|
|
219
|
-
const resourcePath = new URI(resourceUri).path;
|
|
220
|
-
let folder: Readonly<{ relativity: number, uri?: string }> = { relativity: Number.MAX_SAFE_INTEGER };
|
|
221
|
-
const providers = new Map<string, FolderPreferenceProvider[]>();
|
|
222
|
-
for (const provider of this.providers.values()) {
|
|
223
|
-
const uri = provider.folderUri.toString();
|
|
224
|
-
const folderProviders = (providers.get(uri) || []);
|
|
225
|
-
folderProviders.push(provider);
|
|
226
|
-
providers.set(uri, folderProviders);
|
|
227
|
-
|
|
228
|
-
// in case we have nested folders mounted as workspace roots, select the innermost enclosing folder
|
|
229
|
-
const relativity = provider.folderUri.path.relativity(resourcePath);
|
|
230
|
-
if (relativity >= 0 && folder.relativity > relativity) {
|
|
231
|
-
folder = { relativity, uri };
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
return folder.uri && providers.get(folder.uri) || [];
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
protected createProvider(uri: URI, section: string, folder: FileStat): FolderPreferenceProvider {
|
|
238
|
-
const provider = this.folderPreferenceProviderFactory(uri, section, folder);
|
|
239
|
-
this.toDispose.push(provider);
|
|
240
|
-
this.toDispose.push(provider.onDidPreferencesChanged(change => this.onDidPreferencesChangedEmitter.fire(change)));
|
|
241
|
-
return provider;
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2019 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
|
+
|
|
19
|
+
import { inject, injectable, postConstruct } from '@theia/core/shared/inversify';
|
|
20
|
+
import URI from '@theia/core/lib/common/uri';
|
|
21
|
+
import { PreferenceProvider, PreferenceResolveResult, PreferenceScope } from '@theia/core/lib/browser/preferences';
|
|
22
|
+
import { WorkspaceService } from '@theia/workspace/lib/browser/workspace-service';
|
|
23
|
+
import { PreferenceConfigurations } from '@theia/core/lib/browser/preferences/preference-configurations';
|
|
24
|
+
import { FolderPreferenceProvider, FolderPreferenceProviderFactory } from './folder-preference-provider';
|
|
25
|
+
import { FileStat } from '@theia/filesystem/lib/common/files';
|
|
26
|
+
|
|
27
|
+
@injectable()
|
|
28
|
+
export class FoldersPreferencesProvider extends PreferenceProvider {
|
|
29
|
+
|
|
30
|
+
@inject(WorkspaceService)
|
|
31
|
+
protected readonly workspaceService: WorkspaceService;
|
|
32
|
+
|
|
33
|
+
@inject(FolderPreferenceProviderFactory)
|
|
34
|
+
protected readonly folderPreferenceProviderFactory: FolderPreferenceProviderFactory;
|
|
35
|
+
|
|
36
|
+
@inject(PreferenceConfigurations)
|
|
37
|
+
protected readonly configurations: PreferenceConfigurations;
|
|
38
|
+
|
|
39
|
+
protected readonly providers = new Map<string, FolderPreferenceProvider>();
|
|
40
|
+
|
|
41
|
+
@postConstruct()
|
|
42
|
+
protected init(): void {
|
|
43
|
+
this.doInit();
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
protected async doInit(): Promise<void> {
|
|
47
|
+
await this.workspaceService.roots;
|
|
48
|
+
|
|
49
|
+
this.updateProviders();
|
|
50
|
+
this.workspaceService.onWorkspaceChanged(() => this.updateProviders());
|
|
51
|
+
|
|
52
|
+
const readyPromises: Promise<void>[] = [];
|
|
53
|
+
for (const provider of this.providers.values()) {
|
|
54
|
+
readyPromises.push(provider.ready.catch(e => console.error(e)));
|
|
55
|
+
}
|
|
56
|
+
Promise.all(readyPromises).then(() => this._ready.resolve());
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
protected updateProviders(): void {
|
|
60
|
+
const roots = this.workspaceService.tryGetRoots();
|
|
61
|
+
const toDelete = new Set(this.providers.keys());
|
|
62
|
+
for (const folder of roots) {
|
|
63
|
+
for (const configPath of this.configurations.getPaths()) {
|
|
64
|
+
for (const configName of [...this.configurations.getSectionNames(), this.configurations.getConfigName()]) {
|
|
65
|
+
const sectionUri = this.configurations.createUri(folder.resource, configPath, configName);
|
|
66
|
+
const sectionKey = sectionUri.toString();
|
|
67
|
+
toDelete.delete(sectionKey);
|
|
68
|
+
if (!this.providers.has(sectionKey)) {
|
|
69
|
+
const provider = this.createProvider(sectionUri, configName, folder);
|
|
70
|
+
this.providers.set(sectionKey, provider);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
for (const key of toDelete) {
|
|
76
|
+
const provider = this.providers.get(key);
|
|
77
|
+
if (provider) {
|
|
78
|
+
this.providers.delete(key);
|
|
79
|
+
provider.dispose();
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
override getConfigUri(resourceUri?: string, sectionName: string = this.configurations.getConfigName()): URI | undefined {
|
|
85
|
+
for (const provider of this.getFolderProviders(resourceUri)) {
|
|
86
|
+
const configUri = provider.getConfigUri(resourceUri);
|
|
87
|
+
if (configUri && this.configurations.getName(configUri) === sectionName) {
|
|
88
|
+
return configUri;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
return undefined;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
override getContainingConfigUri(resourceUri?: string, sectionName: string = this.configurations.getConfigName()): URI | undefined {
|
|
95
|
+
for (const provider of this.getFolderProviders(resourceUri)) {
|
|
96
|
+
const configUri = provider.getConfigUri();
|
|
97
|
+
if (provider.contains(resourceUri) && this.configurations.getName(configUri) === sectionName) {
|
|
98
|
+
return configUri;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
return undefined;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
override getDomain(): string[] {
|
|
105
|
+
return this.workspaceService.tryGetRoots().map(root => root.resource.toString());
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
override resolve<T>(preferenceName: string, resourceUri?: string): PreferenceResolveResult<T> {
|
|
109
|
+
const result: PreferenceResolveResult<T> = {};
|
|
110
|
+
const groups = this.groupProvidersByConfigName(resourceUri);
|
|
111
|
+
for (const group of groups.values()) {
|
|
112
|
+
for (const provider of group) {
|
|
113
|
+
const { value, configUri } = provider.resolve<T>(preferenceName, resourceUri);
|
|
114
|
+
if (configUri && value !== undefined) {
|
|
115
|
+
result.configUri = configUri;
|
|
116
|
+
result.value = PreferenceProvider.merge(result.value as any, value as any) as any;
|
|
117
|
+
break;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
return result;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
getPreferences(resourceUri?: string): { [p: string]: any } {
|
|
125
|
+
let result = {};
|
|
126
|
+
const groups = this.groupProvidersByConfigName(resourceUri);
|
|
127
|
+
for (const group of groups.values()) {
|
|
128
|
+
for (const provider of group) {
|
|
129
|
+
if (provider.getConfigUri(resourceUri)) {
|
|
130
|
+
const preferences = provider.getPreferences();
|
|
131
|
+
result = PreferenceProvider.merge(result, preferences) as any;
|
|
132
|
+
break;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
return result;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
async setPreference(preferenceName: string, value: any, resourceUri?: string): Promise<boolean> {
|
|
140
|
+
const firstPathFragment = preferenceName.split('.', 1)[0];
|
|
141
|
+
const defaultConfigName = this.configurations.getConfigName();
|
|
142
|
+
const configName = this.configurations.isSectionName(firstPathFragment) ? firstPathFragment : defaultConfigName;
|
|
143
|
+
|
|
144
|
+
const providers = this.getFolderProviders(resourceUri);
|
|
145
|
+
let configPath: string | undefined;
|
|
146
|
+
const candidates = providers.filter(provider => {
|
|
147
|
+
// Attempt to figure out the settings folder (.vscode or .theia) we're interested in.
|
|
148
|
+
const containingConfigUri = provider.getConfigUri(resourceUri);
|
|
149
|
+
if (configPath === undefined && containingConfigUri) {
|
|
150
|
+
configPath = this.configurations.getPath(containingConfigUri);
|
|
151
|
+
}
|
|
152
|
+
const providerName = this.configurations.getName(containingConfigUri ?? provider.getConfigUri());
|
|
153
|
+
return providerName === configName || providerName === defaultConfigName;
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
const configNameAndPathMatches = [];
|
|
157
|
+
const configNameOnlyMatches = [];
|
|
158
|
+
const configUriMatches = [];
|
|
159
|
+
const otherMatches = [];
|
|
160
|
+
|
|
161
|
+
for (const candidate of candidates) {
|
|
162
|
+
const domainMatches = candidate.getConfigUri(resourceUri);
|
|
163
|
+
const configUri = domainMatches ?? candidate.getConfigUri();
|
|
164
|
+
const nameMatches = this.configurations.getName(configUri) === configName;
|
|
165
|
+
const pathMatches = this.configurations.getPath(configUri) === configPath;
|
|
166
|
+
|
|
167
|
+
// Perfect match, run immediately in case we can bail out early.
|
|
168
|
+
if (nameMatches && domainMatches) {
|
|
169
|
+
if (await candidate.setPreference(preferenceName, value, resourceUri)) {
|
|
170
|
+
return true;
|
|
171
|
+
}
|
|
172
|
+
} else if (nameMatches && pathMatches) { // Right file in the right folder.
|
|
173
|
+
configNameAndPathMatches.push(candidate);
|
|
174
|
+
} else if (nameMatches) { // Right file.
|
|
175
|
+
configNameOnlyMatches.push(candidate);
|
|
176
|
+
} else if (domainMatches) { // Currently valid and governs target URI
|
|
177
|
+
configUriMatches.push(candidate);
|
|
178
|
+
} else {
|
|
179
|
+
otherMatches.push(candidate);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
const candidateSets = [configNameAndPathMatches, configNameOnlyMatches, configUriMatches, otherMatches];
|
|
184
|
+
|
|
185
|
+
for (const candidateSet of candidateSets) {
|
|
186
|
+
for (const candidate of candidateSet) {
|
|
187
|
+
if (await candidate.setPreference(preferenceName, value, resourceUri)) {
|
|
188
|
+
return true;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
return false;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
override canHandleScope(scope: PreferenceScope): boolean {
|
|
197
|
+
return this.workspaceService.isMultiRootWorkspaceOpened && scope === PreferenceScope.Folder || scope === PreferenceScope.Workspace;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
protected groupProvidersByConfigName(resourceUri?: string): Map<string, FolderPreferenceProvider[]> {
|
|
201
|
+
const groups = new Map<string, FolderPreferenceProvider[]>();
|
|
202
|
+
const providers = this.getFolderProviders(resourceUri);
|
|
203
|
+
for (const configName of [this.configurations.getConfigName(), ...this.configurations.getSectionNames()]) {
|
|
204
|
+
const group = [];
|
|
205
|
+
for (const provider of providers) {
|
|
206
|
+
if (this.configurations.getName(provider.getConfigUri()) === configName) {
|
|
207
|
+
group.push(provider);
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
groups.set(configName, group);
|
|
211
|
+
}
|
|
212
|
+
return groups;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
protected getFolderProviders(resourceUri?: string): FolderPreferenceProvider[] {
|
|
216
|
+
if (!resourceUri) {
|
|
217
|
+
return [];
|
|
218
|
+
}
|
|
219
|
+
const resourcePath = new URI(resourceUri).path;
|
|
220
|
+
let folder: Readonly<{ relativity: number, uri?: string }> = { relativity: Number.MAX_SAFE_INTEGER };
|
|
221
|
+
const providers = new Map<string, FolderPreferenceProvider[]>();
|
|
222
|
+
for (const provider of this.providers.values()) {
|
|
223
|
+
const uri = provider.folderUri.toString();
|
|
224
|
+
const folderProviders = (providers.get(uri) || []);
|
|
225
|
+
folderProviders.push(provider);
|
|
226
|
+
providers.set(uri, folderProviders);
|
|
227
|
+
|
|
228
|
+
// in case we have nested folders mounted as workspace roots, select the innermost enclosing folder
|
|
229
|
+
const relativity = provider.folderUri.path.relativity(resourcePath);
|
|
230
|
+
if (relativity >= 0 && folder.relativity > relativity) {
|
|
231
|
+
folder = { relativity, uri };
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
return folder.uri && providers.get(folder.uri) || [];
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
protected createProvider(uri: URI, section: string, folder: FileStat): FolderPreferenceProvider {
|
|
238
|
+
const provider = this.folderPreferenceProviderFactory(uri, section, folder);
|
|
239
|
+
this.toDispose.push(provider);
|
|
240
|
+
this.toDispose.push(provider.onDidPreferencesChanged(change => this.onDidPreferencesChangedEmitter.fire(change)));
|
|
241
|
+
return provider;
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
}
|
package/src/browser/index.ts
CHANGED
|
@@ -1,23 +1,23 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2018 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
|
-
export * from '@theia/core/lib/browser/preferences';
|
|
18
|
-
export * from './abstract-resource-preference-provider';
|
|
19
|
-
export * from './user-preference-provider';
|
|
20
|
-
export * from './workspace-preference-provider';
|
|
21
|
-
export * from './folders-preferences-provider';
|
|
22
|
-
export * from './folder-preference-provider';
|
|
23
|
-
export * from './user-configs-preference-provider';
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2018 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
|
+
export * from '@theia/core/lib/browser/preferences';
|
|
18
|
+
export * from './abstract-resource-preference-provider';
|
|
19
|
+
export * from './user-preference-provider';
|
|
20
|
+
export * from './workspace-preference-provider';
|
|
21
|
+
export * from './folders-preferences-provider';
|
|
22
|
+
export * from './folder-preference-provider';
|
|
23
|
+
export * from './user-configs-preference-provider';
|