Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/src/vs/workbench/contrib/preferences/common/preferences.ts
3296 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
6
import { raceTimeout } from '../../../../base/common/async.js';
7
import { CancellationToken } from '../../../../base/common/cancellation.js';
8
import { IStringDictionary } from '../../../../base/common/collections.js';
9
import { IExtensionRecommendations } from '../../../../base/common/product.js';
10
import { localize } from '../../../../nls.js';
11
import { IContextKeyService, RawContextKey } from '../../../../platform/contextkey/common/contextkey.js';
12
import { IExtensionGalleryService, IGalleryExtension } from '../../../../platform/extensionManagement/common/extensionManagement.js';
13
import { createDecorator } from '../../../../platform/instantiation/common/instantiation.js';
14
import { IProductService } from '../../../../platform/product/common/productService.js';
15
import { ISearchResult, ISettingsEditorModel } from '../../../services/preferences/common/preferences.js';
16
17
export interface IWorkbenchSettingsConfiguration {
18
workbench: {
19
settings: {
20
openDefaultSettings: boolean;
21
naturalLanguageSearchEndpoint: string;
22
naturalLanguageSearchKey: string;
23
naturalLanguageSearchAutoIngestFeedback: boolean;
24
useNaturalLanguageSearchPost: boolean;
25
enableNaturalLanguageSearch: boolean;
26
enableNaturalLanguageSearchFeedback: boolean;
27
};
28
};
29
}
30
31
export interface IEndpointDetails {
32
urlBase: string;
33
key?: string;
34
}
35
36
export const IPreferencesSearchService = createDecorator<IPreferencesSearchService>('preferencesSearchService');
37
38
export interface IPreferencesSearchService {
39
readonly _serviceBrand: undefined;
40
41
getLocalSearchProvider(filter: string): ISearchProvider;
42
getRemoteSearchProvider(filter: string, newExtensionsOnly?: boolean): ISearchProvider | undefined;
43
getAiSearchProvider(filter: string): IAiSearchProvider | undefined;
44
}
45
46
export interface ISearchProvider {
47
searchModel(preferencesModel: ISettingsEditorModel, token: CancellationToken): Promise<ISearchResult | null>;
48
}
49
50
export interface IRemoteSearchProvider extends ISearchProvider {
51
setFilter(filter: string): void;
52
}
53
54
export interface IAiSearchProvider extends IRemoteSearchProvider {
55
getLLMRankedResults(token: CancellationToken): Promise<ISearchResult | null>;
56
}
57
58
export const PREFERENCES_EDITOR_COMMAND_OPEN = 'workbench.preferences.action.openPreferencesEditor';
59
export const CONTEXT_PREFERENCES_SEARCH_FOCUS = new RawContextKey<boolean>('inPreferencesSearch', false);
60
61
export const SETTINGS_EDITOR_COMMAND_CLEAR_SEARCH_RESULTS = 'settings.action.clearSearchResults';
62
export const SETTINGS_EDITOR_COMMAND_SHOW_AI_RESULTS = 'settings.action.showAIResults';
63
export const SETTINGS_EDITOR_COMMAND_TOGGLE_AI_SEARCH = 'settings.action.toggleAiSearch';
64
export const SETTINGS_EDITOR_COMMAND_SHOW_CONTEXT_MENU = 'settings.action.showContextMenu';
65
export const SETTINGS_EDITOR_COMMAND_SUGGEST_FILTERS = 'settings.action.suggestFilters';
66
67
export const CONTEXT_SETTINGS_EDITOR = new RawContextKey<boolean>('inSettingsEditor', false);
68
export const CONTEXT_SETTINGS_JSON_EDITOR = new RawContextKey<boolean>('inSettingsJSONEditor', false);
69
export const CONTEXT_SETTINGS_SEARCH_FOCUS = new RawContextKey<boolean>('inSettingsSearch', false);
70
export const CONTEXT_TOC_ROW_FOCUS = new RawContextKey<boolean>('settingsTocRowFocus', false);
71
export const CONTEXT_SETTINGS_ROW_FOCUS = new RawContextKey<boolean>('settingRowFocus', false);
72
export const CONTEXT_KEYBINDINGS_EDITOR = new RawContextKey<boolean>('inKeybindings', false);
73
export const CONTEXT_KEYBINDINGS_SEARCH_FOCUS = new RawContextKey<boolean>('inKeybindingsSearch', false);
74
export const CONTEXT_KEYBINDING_FOCUS = new RawContextKey<boolean>('keybindingFocus', false);
75
export const CONTEXT_WHEN_FOCUS = new RawContextKey<boolean>('whenFocus', false);
76
export const CONTEXT_AI_SETTING_RESULTS_AVAILABLE = new RawContextKey<boolean>('aiSettingResultsAvailable', false);
77
78
export const KEYBINDINGS_EDITOR_COMMAND_SEARCH = 'keybindings.editor.searchKeybindings';
79
export const KEYBINDINGS_EDITOR_COMMAND_CLEAR_SEARCH_RESULTS = 'keybindings.editor.clearSearchResults';
80
export const KEYBINDINGS_EDITOR_COMMAND_CLEAR_SEARCH_HISTORY = 'keybindings.editor.clearSearchHistory';
81
export const KEYBINDINGS_EDITOR_COMMAND_RECORD_SEARCH_KEYS = 'keybindings.editor.recordSearchKeys';
82
export const KEYBINDINGS_EDITOR_COMMAND_SORTBY_PRECEDENCE = 'keybindings.editor.toggleSortByPrecedence';
83
export const KEYBINDINGS_EDITOR_COMMAND_DEFINE = 'keybindings.editor.defineKeybinding';
84
export const KEYBINDINGS_EDITOR_COMMAND_ADD = 'keybindings.editor.addKeybinding';
85
export const KEYBINDINGS_EDITOR_COMMAND_DEFINE_WHEN = 'keybindings.editor.defineWhenExpression';
86
export const KEYBINDINGS_EDITOR_COMMAND_ACCEPT_WHEN = 'keybindings.editor.acceptWhenExpression';
87
export const KEYBINDINGS_EDITOR_COMMAND_REJECT_WHEN = 'keybindings.editor.rejectWhenExpression';
88
export const KEYBINDINGS_EDITOR_COMMAND_REMOVE = 'keybindings.editor.removeKeybinding';
89
export const KEYBINDINGS_EDITOR_COMMAND_RESET = 'keybindings.editor.resetKeybinding';
90
export const KEYBINDINGS_EDITOR_COMMAND_COPY = 'keybindings.editor.copyKeybindingEntry';
91
export const KEYBINDINGS_EDITOR_COMMAND_COPY_COMMAND = 'keybindings.editor.copyCommandKeybindingEntry';
92
export const KEYBINDINGS_EDITOR_COMMAND_COPY_COMMAND_TITLE = 'keybindings.editor.copyCommandTitle';
93
export const KEYBINDINGS_EDITOR_COMMAND_SHOW_SIMILAR = 'keybindings.editor.showConflicts';
94
export const KEYBINDINGS_EDITOR_COMMAND_FOCUS_KEYBINDINGS = 'keybindings.editor.focusKeybindings';
95
export const KEYBINDINGS_EDITOR_SHOW_DEFAULT_KEYBINDINGS = 'keybindings.editor.showDefaultKeybindings';
96
export const KEYBINDINGS_EDITOR_SHOW_USER_KEYBINDINGS = 'keybindings.editor.showUserKeybindings';
97
export const KEYBINDINGS_EDITOR_SHOW_EXTENSION_KEYBINDINGS = 'keybindings.editor.showExtensionKeybindings';
98
99
export const MODIFIED_SETTING_TAG = 'modified';
100
export const EXTENSION_SETTING_TAG = 'ext:';
101
export const FEATURE_SETTING_TAG = 'feature:';
102
export const ID_SETTING_TAG = 'id:';
103
export const LANGUAGE_SETTING_TAG = 'lang:';
104
export const GENERAL_TAG_SETTING_TAG = 'tag:';
105
export const POLICY_SETTING_TAG = 'hasPolicy';
106
export const WORKSPACE_TRUST_SETTING_TAG = 'workspaceTrust';
107
export const REQUIRE_TRUSTED_WORKSPACE_SETTING_TAG = 'requireTrustedWorkspace';
108
export const KEYBOARD_LAYOUT_OPEN_PICKER = 'workbench.action.openKeyboardLayoutPicker';
109
110
export const ENABLE_LANGUAGE_FILTER = true;
111
112
export const ENABLE_EXTENSION_TOGGLE_SETTINGS = true;
113
export const EXTENSION_FETCH_TIMEOUT_MS = 1000;
114
115
export const STRING_MATCH_SEARCH_PROVIDER_NAME = 'local';
116
export const TF_IDF_SEARCH_PROVIDER_NAME = 'tfIdf';
117
export const FILTER_MODEL_SEARCH_PROVIDER_NAME = 'filterModel';
118
export const EMBEDDINGS_ONLY_SEARCH_PROVIDER_NAME = 'embeddingsOnly';
119
export const EMBEDDINGS_SEARCH_PROVIDER_NAME = 'embeddingsFull';
120
export const LLM_RANKED_SEARCH_PROVIDER_NAME = 'llmRanked';
121
122
export enum WorkbenchSettingsEditorSettings {
123
ShowAISearchToggle = 'workbench.settings.showAISearchToggle',
124
EnableNaturalLanguageSearch = 'workbench.settings.enableNaturalLanguageSearch',
125
}
126
127
export type ExtensionToggleData = {
128
settingsEditorRecommendedExtensions: IStringDictionary<IExtensionRecommendations>;
129
recommendedExtensionsGalleryInfo: IStringDictionary<IGalleryExtension>;
130
commonlyUsed: string[];
131
};
132
133
let cachedExtensionToggleData: ExtensionToggleData | undefined;
134
135
export async function getExperimentalExtensionToggleData(
136
contextKeyService: IContextKeyService,
137
extensionGalleryService: IExtensionGalleryService,
138
productService: IProductService,
139
): Promise<ExtensionToggleData | undefined> {
140
if (!ENABLE_EXTENSION_TOGGLE_SETTINGS) {
141
return undefined;
142
}
143
144
if (!extensionGalleryService.isEnabled()) {
145
return undefined;
146
}
147
148
if (contextKeyService.getContextKeyValue<boolean>('chatSetupHidden')) {
149
return undefined;
150
}
151
152
if (cachedExtensionToggleData) {
153
return cachedExtensionToggleData;
154
}
155
156
if (productService.extensionRecommendations && productService.commonlyUsedSettings) {
157
const settingsEditorRecommendedExtensions: IStringDictionary<IExtensionRecommendations> = {};
158
Object.keys(productService.extensionRecommendations).forEach(extensionId => {
159
const extensionInfo = productService.extensionRecommendations![extensionId];
160
if (extensionInfo.onSettingsEditorOpen) {
161
settingsEditorRecommendedExtensions[extensionId] = extensionInfo;
162
}
163
});
164
165
const recommendedExtensionsGalleryInfo: IStringDictionary<IGalleryExtension> = {};
166
for (const key in settingsEditorRecommendedExtensions) {
167
const extensionId = key;
168
// Recommend prerelease if not on Stable.
169
const isStable = productService.quality === 'stable';
170
try {
171
const extensions = await raceTimeout(
172
extensionGalleryService.getExtensions([{ id: extensionId, preRelease: !isStable }], CancellationToken.None),
173
EXTENSION_FETCH_TIMEOUT_MS);
174
if (extensions?.length === 1) {
175
recommendedExtensionsGalleryInfo[key] = extensions[0];
176
} else {
177
// same as network connection fail. we do not want a blank settings page: https://github.com/microsoft/vscode/issues/195722
178
// so instead of returning partial data we return undefined here
179
return undefined;
180
}
181
} catch (e) {
182
// Network connection fail. Return nothing rather than partial data.
183
return undefined;
184
}
185
}
186
187
cachedExtensionToggleData = {
188
settingsEditorRecommendedExtensions,
189
recommendedExtensionsGalleryInfo,
190
commonlyUsed: productService.commonlyUsedSettings
191
};
192
return cachedExtensionToggleData;
193
}
194
return undefined;
195
}
196
197
/**
198
* Compares two nullable numbers such that null values always come after defined ones.
199
*/
200
export function compareTwoNullableNumbers(a: number | undefined, b: number | undefined): number {
201
const aOrMax = a ?? Number.MAX_SAFE_INTEGER;
202
const bOrMax = b ?? Number.MAX_SAFE_INTEGER;
203
if (aOrMax < bOrMax) {
204
return -1;
205
} else if (aOrMax > bOrMax) {
206
return 1;
207
} else {
208
return 0;
209
}
210
}
211
212
export const PREVIEW_INDICATOR_DESCRIPTION = localize('previewIndicatorDescription', "Preview setting: this setting controls a new feature that is still under refinement yet ready to use. Feedback is welcome.");
213
export const EXPERIMENTAL_INDICATOR_DESCRIPTION = localize('experimentalIndicatorDescription', "Experimental setting: this setting controls a new feature that is actively being developed and may be unstable. It is subject to change or removal.");
214
215
export const knownAcronyms = new Set<string>();
216
[
217
'css',
218
'html',
219
'scss',
220
'less',
221
'json',
222
'js',
223
'ts',
224
'ie',
225
'id',
226
'php',
227
'scm',
228
].forEach(str => knownAcronyms.add(str));
229
230
export const knownTermMappings = new Map<string, string>();
231
knownTermMappings.set('power shell', 'PowerShell');
232
knownTermMappings.set('powershell', 'PowerShell');
233
knownTermMappings.set('javascript', 'JavaScript');
234
knownTermMappings.set('typescript', 'TypeScript');
235
knownTermMappings.set('github', 'GitHub');
236
knownTermMappings.set('jet brains', 'JetBrains');
237
knownTermMappings.set('jetbrains', 'JetBrains');
238
knownTermMappings.set('re sharper', 'ReSharper');
239
knownTermMappings.set('resharper', 'ReSharper');
240
241
export function wordifyKey(key: string): string {
242
key = key
243
.replace(/\.([a-z0-9])/g, (_, p1) => ` \u203A ${p1.toUpperCase()}`) // Replace dot with spaced '>'
244
.replace(/([a-z0-9])([A-Z])/g, '$1 $2') // Camel case to spacing, fooBar => foo Bar
245
.replace(/([A-Z]{1,})([A-Z][a-z])/g, '$1 $2') // Split consecutive capitals letters, AISearch => AI Search
246
.replace(/^[a-z]/g, match => match.toUpperCase()) // Upper casing all first letters, foo => Foo
247
.replace(/\b\w+\b/g, match => { // Upper casing known acronyms
248
return knownAcronyms.has(match.toLowerCase()) ?
249
match.toUpperCase() :
250
match;
251
});
252
253
for (const [k, v] of knownTermMappings) {
254
key = key.replace(new RegExp(`\\b${k}\\b`, 'gi'), v);
255
}
256
257
return key;
258
}
259
260