htmlMain.ts 7.1 KB
Newer Older
1 2 3 4
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/
5

6
import * as path from 'path';
M
Martin Aeschlimann 已提交
7
import * as fs from 'fs';
D
Dirk Baeumer 已提交
8 9
import * as nls from 'vscode-nls';
const localize = nls.loadMessageBundle();
10

11
import { languages, ExtensionContext, IndentAction, Position, TextDocument, Range, CompletionItem, CompletionItemKind, SnippetString, workspace } from 'vscode';
12
import { LanguageClient, LanguageClientOptions, ServerOptions, TransportKind, RequestType, TextDocumentPositionParams } from 'vscode-languageclient';
J
Johannes Rieken 已提交
13
import { EMPTY_ELEMENTS } from './htmlEmptyTagsShared';
14
import { activateTagClosing } from './tagClosing';
15
import TelemetryReporter from 'vscode-extension-telemetry';
P
Pine Wu 已提交
16
import { getCustomDataPathsInAllWorkspaces } from './customData';
17

18 19
namespace TagCloseRequest {
	export const type: RequestType<TextDocumentPositionParams, string, any, any> = new RequestType('html/tag');
20 21
}

22 23 24 25 26 27
interface IPackageInfo {
	name: string;
	version: string;
	aiKey: string;
}

28 29
let telemetryReporter: TelemetryReporter | null;

30

31
export function activate(context: ExtensionContext) {
32
	let toDispose = context.subscriptions;
33

34
	let packageInfo = getPackageInfo(context);
35
	telemetryReporter = packageInfo && new TelemetryReporter(packageInfo.name, packageInfo.version, packageInfo.aiKey);
36

M
Martin Aeschlimann 已提交
37 38 39
	let serverMain = readJSONFile(context.asAbsolutePath('./server/package.json')).main;
	let serverModule = context.asAbsolutePath(path.join('server', serverMain));

40
	// The debug options for the server
41
	let debugOptions = { execArgv: ['--nolazy', '--inspect=6045'] };
42 43 44 45 46 47 48 49

	// If the extension is launch in debug mode the debug server options are use
	// Otherwise the run options are used
	let serverOptions: ServerOptions = {
		run: { module: serverModule, transport: TransportKind.ipc },
		debug: { module: serverModule, transport: TransportKind.ipc, options: debugOptions }
	};

50
	let documentSelector = ['html', 'handlebars', 'razor'];
51
	let embeddedLanguages = { css: true, javascript: true };
52

P
Pine Wu 已提交
53
	let { tagPaths, attributePaths } = getCustomDataPathsInAllWorkspaces(workspace.workspaceFolders);
54

55 56
	// Options to control the language client
	let clientOptions: LanguageClientOptions = {
57
		documentSelector,
58
		synchronize: {
59
			configurationSection: ['html', 'css', 'javascript'], // the settings to synchronize
60 61
		},
		initializationOptions: {
62 63 64
			embeddedLanguages,
			tagPaths,
			attributePaths
65 66 67 68
		}
	};

	// Create the language client and start the client.
69
	let client = new LanguageClient('html', localize('htmlserver.name', 'HTML Language Server'), serverOptions, clientOptions);
70
	client.registerProposedFeatures();
71

72
	let disposable = client.start();
73
	toDispose.push(disposable);
74
	client.onReady().then(() => {
75 76 77 78
		let tagRequestor = (document: TextDocument, position: Position) => {
			let param = client.code2ProtocolConverter.asTextDocumentPositionParams(document, position);
			return client.sendRequest(TagCloseRequest.type, param);
		};
79
		disposable = activateTagClosing(tagRequestor, { html: true, handlebars: true, razor: true }, 'html.autoClosingTags');
80 81 82
		toDispose.push(disposable);

		disposable = client.onTelemetry(e => {
83 84 85 86
			if (telemetryReporter) {
				telemetryReporter.sendTelemetryEvent(e.key, e.data);
			}
		});
87
		toDispose.push(disposable);
88
	});
89

90
	languages.setLanguageConfiguration('html', {
91 92 93 94
		indentationRules: {
			increaseIndentPattern: /<(?!\?|(?:area|base|br|col|frame|hr|html|img|input|link|meta|param)\b|[^>]*\/>)([-_\.A-Za-z0-9]+)(?=\s|>)\b[^>]*>(?!.*<\/\1>)|<!--(?!.*-->)|\{[^}"']*$/,
			decreaseIndentPattern: /^\s*(<\/(?!html)[-_\.A-Za-z0-9]+\b[^>]*>|-->|\})/
		},
95
		wordPattern: /(-?\d*\.\d\w*)|([^\`\~\!\@\$\^\&\*\(\)\=\+\[\{\]\}\\\|\;\:\'\"\,\.\<\>\/\s]+)/g,
J
Johannes Rieken 已提交
96
		onEnterRules: [
97 98
			{
				beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))([_:\\w][_:\\w-.\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
99
				afterText: /^<\/([_:\w][_:\w-.\d]*)\s*>/i,
100 101 102 103 104 105 106 107
				action: { indentAction: IndentAction.IndentOutdent }
			},
			{
				beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))(\\w[\\w\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
				action: { indentAction: IndentAction.Indent }
			}
		],
	});
M
Martin Aeschlimann 已提交
108 109

	languages.setLanguageConfiguration('handlebars', {
110
		wordPattern: /(-?\d*\.\d\w*)|([^\`\~\!\@\$\^\&\*\(\)\=\+\[\{\]\}\\\|\;\:\'\"\,\.\<\>\/\s]+)/g,
J
Johannes Rieken 已提交
111
		onEnterRules: [
M
Martin Aeschlimann 已提交
112 113
			{
				beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))([_:\\w][_:\\w-.\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
114
				afterText: /^<\/([_:\w][_:\w-.\d]*)\s*>/i,
M
Martin Aeschlimann 已提交
115 116 117 118 119 120 121 122
				action: { indentAction: IndentAction.IndentOutdent }
			},
			{
				beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))(\\w[\\w\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
				action: { indentAction: IndentAction.Indent }
			}
		],
	});
E
Erich Gamma 已提交
123 124

	languages.setLanguageConfiguration('razor', {
125
		wordPattern: /(-?\d*\.\d\w*)|([^\`\~\!\@\$\^\&\*\(\)\-\=\+\[\{\]\}\\\|\;\:\'\"\,\.\<\>\/\s]+)/g,
J
Johannes Rieken 已提交
126
		onEnterRules: [
E
Erich Gamma 已提交
127 128
			{
				beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))([_:\\w][_:\\w-.\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
129
				afterText: /^<\/([_:\w][_:\w-.\d]*)\s*>/i,
E
Erich Gamma 已提交
130 131 132 133 134 135 136 137
				action: { indentAction: IndentAction.IndentOutdent }
			},
			{
				beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))(\\w[\\w\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
				action: { indentAction: IndentAction.Indent }
			}
		],
	});
138

139
	const regionCompletionRegExpr = /^(\s*)(<(!(-(-\s*(#\w*)?)?)?)?)?$/;
140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162
	languages.registerCompletionItemProvider(documentSelector, {
		provideCompletionItems(doc, pos) {
			let lineUntilPos = doc.getText(new Range(new Position(pos.line, 0), pos));
			let match = lineUntilPos.match(regionCompletionRegExpr);
			if (match) {
				let range = new Range(new Position(pos.line, match[1].length), pos);
				let beginProposal = new CompletionItem('#region', CompletionItemKind.Snippet);
				beginProposal.range = range;
				beginProposal.insertText = new SnippetString('<!-- #region $1-->');
				beginProposal.documentation = localize('folding.start', 'Folding Region Start');
				beginProposal.filterText = match[2];
				beginProposal.sortText = 'za';
				let endProposal = new CompletionItem('#endregion', CompletionItemKind.Snippet);
				endProposal.range = range;
				endProposal.insertText = new SnippetString('<!-- #endregion -->');
				endProposal.documentation = localize('folding.end', 'Folding Region End');
				endProposal.filterText = match[2];
				endProposal.sortText = 'zb';
				return [beginProposal, endProposal];
			}
			return null;
		}
	});
163 164
}

165
function getPackageInfo(context: ExtensionContext): IPackageInfo | null {
M
Martin Aeschlimann 已提交
166
	let extensionPackage = readJSONFile(context.asAbsolutePath('./package.json'));
167 168 169 170 171 172 173 174
	if (extensionPackage) {
		return {
			name: extensionPackage.name,
			version: extensionPackage.version,
			aiKey: extensionPackage.aiKey
		};
	}
	return null;
175
}
176

M
Martin Aeschlimann 已提交
177 178 179 180 181 182 183 184 185 186

function readJSONFile(location: string) {
	try {
		return JSON.parse(fs.readFileSync(location).toString());
	} catch (e) {
		console.log(`Problems reading ${location}: ${e}`);
		return {};
	}
}

187 188 189
export function deactivate(): Promise<any> {
	return telemetryReporter ? telemetryReporter.dispose() : Promise.resolve(null);
}