-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathclient.ts
214 lines (198 loc) · 8.56 KB
/
client.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
import { TelemetryEvent } from '@redhat-developer/vscode-redhat-telemetry/lib';
import { commands, ConfigurationTarget, ExtensionContext, window, workspace } from 'vscode';
import { DidChangeConfigurationNotification, LanguageClientOptions } from 'vscode-languageclient';
import { LanguageClient } from 'vscode-languageclient/node';
import { JavaExtensionAPI } from '../../extension';
import { sendTelemetry } from '../../utils/telemetryUtils';
import { QuteClientCommandConstants } from '../commands/commandConstants';
import { registerQuteLSDependentCommands, registerVSCodeQuteCommands, synchronizeQuteValidationButton } from '../commands/registerCommands';
import { prepareExecutable } from './quteServerStarter';
import { resolveRequirements } from './requirements';
import { QuteSettings } from './settings';
export async function connectToQuteLS(context: ExtensionContext, api: JavaExtensionAPI): Promise<LanguageClient> {
registerVSCodeQuteCommands(context);
const requirements = await resolveRequirements(api);
const clientOptions: LanguageClientOptions = {
documentSelector: [
{ scheme: 'file', language: 'qute-html' },
{ scheme: 'file', language: 'qute-json' },
{ scheme: 'file', language: 'qute-yaml' },
{ scheme: 'file', language: 'qute-txt' },
{ scheme: 'untitled', language: 'qute-html' },
{ scheme: 'vscode-notebook-cell', language: 'qute-html' },
{ scheme: 'file', language: 'java' }
],
// wrap with key 'settings' so it can be handled same a DidChangeConfiguration
initializationOptions: {
settings: getQuteSettings(),
extendedClientCapabilities: {
commands: {
commandsKind: {
valueSet: [
QuteClientCommandConstants.OPEN_URI,
QuteClientCommandConstants.JAVA_DEFINTION,
QuteClientCommandConstants.COMMAND_CONFIGURATION_UPDATE,
QuteClientCommandConstants.COMMAND_SHOW_REFERENCES
]
}
}
},
},
synchronize: {
// preferences starting with these will trigger didChangeConfiguration
configurationSection: ['qute']
},
middleware: {
workspace: {
didChangeConfiguration: async () => {
// A settings.json is updated:
// 1. send the new Quet settings to the Qute language server
const result = quteLanguageClient.sendNotification(DidChangeConfigurationNotification.type, { settings: getQuteSettings() });
// 2. synchronize the Qute toggle button for validation
await synchronizeQuteValidationButton(window.activeTextEditor);
return result;
}
}
},
};
function bindQuteRequest(request: string) {
quteLanguageClient.onRequest(request, async (params: any) => {
await api.serverReady();
return commands.executeCommand("java.execute.workspaceCommand", request, params);
}
);
}
function bindQuteNotification(notification: string) {
context.subscriptions.push(commands.registerCommand(notification, (event: any) => {
quteLanguageClient.sendNotification(notification, event);
}));
}
const serverOptions = prepareExecutable(requirements);
const quteLanguageClient = new LanguageClient('qute', 'Qute Support', serverOptions, clientOptions);
await quteLanguageClient.start();
commands.executeCommand('setContext', 'QuteLSReady', true);
bindQuteRequest('qute/template/projects');
bindQuteNotification('qute/template/project/added');
bindQuteNotification('qute/template/project/removed');
bindQuteRequest('qute/template/project');
bindQuteRequest('qute/template/projectDataModel');
bindQuteRequest('qute/template/userTags');
bindQuteRequest('qute/template/javaTypes');
bindQuteRequest('qute/template/resolvedJavaType');
bindQuteRequest('qute/template/javaDefinition');
bindQuteRequest('qute/template/javadoc');
bindQuteRequest('qute/template/generateMissingJavaMember');
bindQuteRequest('qute/java/codeLens');
bindQuteRequest('qute/java/diagnostics');
bindQuteRequest('qute/java/documentLink');
bindQuteNotification('qute/dataModelChanged');
registerQuteLSDependentCommands(context, quteLanguageClient);
// Refresh the Qute context when editor tab has the focus
context.subscriptions.push(
window.onDidChangeActiveTextEditor(async editor => {
await synchronizeQuteValidationButton(editor);
})
);
// Refresh the Qute context when the language id changed (HTML -> Qute HTML or Qute HTML -> HTML)
context.subscriptions.push(
workspace.onDidOpenTextDocument(async (document) => {
// when settings.json is updated, onDidOpenTextDocument is called,
// the Qute context must be refreshed only for the activate text editor.
if (window.activeTextEditor?.document === document) {
await synchronizeQuteValidationButton(window.activeTextEditor);
}
// Display the experimental Qute validation pop-up if it hasn't been displayed and a Qute file is open
if (!hasShownQuteValidationPopUp(context) && document.languageId.includes('qute')) {
showQuteValidationPopUp(context);
}
})
);
// Display the experimental Qute validation pop-up if it hasn't been displayed and a Qute file is open
if (!hasShownQuteValidationPopUp(context)) {
for (const textDocument of workspace.textDocuments) {
if (textDocument.languageId.includes('qute')) {
showQuteValidationPopUp(context);
}
}
}
quteLanguageClient.onTelemetry(async (e: TelemetryEvent) => {
sendTelemetry(e.name, e.properties);
});
await setQuteValidationEnabledContext();
await synchronizeQuteValidationButton(window.activeTextEditor);
return quteLanguageClient;
}
/**
* Returns a json object with key 'qute' and a json object value that
* holds all qute. settings.
*
* Returns: {
* 'qute': {...}
* }
*/
function getQuteSettings(): any {
if (workspace.workspaceFolders && workspace.workspaceFolders.length > 1) {
// There are several workspace folders, returns the JSON qute settings per workspace folder:
//
/**
* "qute": {
"workspaceFolders": {
"file:///c%3A/Users/azerr/git/quarkus-ls/qute.jdt/com.redhat.qute.jdt.test/projects/maven/qute-quickstart": {
"validation": {
"enabled": true
}
},
"file:///c%3A/Users/azerr/git/quarkus-ls/qute.jdt/com.redhat.qute.jdt.test/projects/maven/qute-java17": {
"validation": {
"enabled": true
}
}
}
}
*/
const foldersSettings = {};
workspace.workspaceFolders.forEach(folder => {
const folderConfigQute = workspace.getConfiguration(undefined, folder).get('qute');
foldersSettings[folder.uri.toString()] = toJSONObject(folderConfigQute);
});
const qute = { qute: { workspaceFolders: foldersSettings } };
return qute;
}
// One workspace folder or none folder, return a single settings:
//
/**
* "qute": {
"validation": {
"enabled": true
}
}
*/
const configQute = workspace.getConfiguration().get('qute');
const qute = { qute: toJSONObject(configQute) };
return qute;
}
function toJSONObject(configQute: unknown): any {
const x = JSON.stringify(configQute); // configQute is not a JSON type
return JSON.parse(x);
}
function hasShownQuteValidationPopUp(context: ExtensionContext): boolean {
return context.globalState.get(QuteSettings.EXPERIMENTAL_QUTE_VALIDATION_FLAG, false);
}
async function showQuteValidationPopUp(context: ExtensionContext) {
const EXPERIMENTAL_QUTE_VALIDATION_ADVERTISEMENT = `Enable experimental validation for Qute files?
(You may change this setting, \`${QuteSettings.QUTE_VALIDATION_ENABLED}}\`, later)`;
const ENABLE_MESSAGE = `Enable`;
const DONT_SHOW_AGAIN_MESSAGE = "Don't show this again";
const input = await window.showInformationMessage(EXPERIMENTAL_QUTE_VALIDATION_ADVERTISEMENT, ENABLE_MESSAGE, DONT_SHOW_AGAIN_MESSAGE);
if (input === ENABLE_MESSAGE) {
workspace.getConfiguration().update(QuteSettings.QUTE_VALIDATION_ENABLED, true, ConfigurationTarget.Global);
}
context.globalState.update(QuteSettings.EXPERIMENTAL_QUTE_VALIDATION_FLAG, 'true');
}
/**
* Sets the `editorQuteValidationEnabled` context to `true` if the `qute.validation.enabled`
* setting is set to true. Sets to `false` otherwise.
*/
export async function setQuteValidationEnabledContext() {
await commands.executeCommand('setContext', 'editorQuteValidationEnabled', workspace.getConfiguration().get(QuteSettings.QUTE_VALIDATION_ENABLED));
}