extHostCommands.ts 6.4 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5 6
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/
'use strict';

J
Johannes Rieken 已提交
7 8 9
import { validateConstraint } from 'vs/base/common/types';
import { ICommandHandlerDescription } from 'vs/platform/commands/common/commands';
import { TPromise } from 'vs/base/common/winjs.base';
J
Johannes Rieken 已提交
10 11
import * as extHostTypes from 'vs/workbench/api/node/extHostTypes';
import * as extHostTypeConverter from 'vs/workbench/api/node/extHostTypeConverters';
J
Johannes Rieken 已提交
12
import { cloneAndChange } from 'vs/base/common/objects';
13
import { MainContext, MainThreadCommandsShape, ExtHostCommandsShape, ObjectIdentifier, IMainContext } from './extHost.protocol';
14 15 16 17
import { ExtHostHeapService } from 'vs/workbench/api/node/extHostHeapService';
import { isFalsyOrEmpty } from 'vs/base/common/arrays';
import * as modes from 'vs/editor/common/modes';
import * as vscode from 'vscode';
E
Erich Gamma 已提交
18

19 20 21 22 23 24
interface CommandHandler {
	callback: Function;
	thisArg: any;
	description: ICommandHandlerDescription;
}

J
Joao Moreno 已提交
25 26 27 28
export interface ArgumentProcessor {
	processArgument(arg: any): any;
}

A
Alex Dima 已提交
29
export class ExtHostCommands extends ExtHostCommandsShape {
E
Erich Gamma 已提交
30

J
Johannes Rieken 已提交
31
	private _commands = new Map<string, CommandHandler>();
32
	private _proxy: MainThreadCommandsShape;
33
	private _converter: CommandsConverter;
J
Joao Moreno 已提交
34
	private _argumentProcessors: ArgumentProcessor[] = [];
E
Erich Gamma 已提交
35

36
	constructor(
37
		mainContext: IMainContext,
38
		heapService: ExtHostHeapService
39
	) {
A
Alex Dima 已提交
40
		super();
41
		this._proxy = mainContext.get(MainContext.MainThreadCommands);
42 43 44 45 46
		this._converter = new CommandsConverter(this, heapService);
	}

	get converter(): CommandsConverter {
		return this._converter;
E
Erich Gamma 已提交
47 48
	}

J
Joao Moreno 已提交
49 50 51 52
	registerArgumentProcessor(processor: ArgumentProcessor): void {
		this._argumentProcessors.push(processor);
	}

53
	registerCommand(id: string, callback: <T>(...args: any[]) => T | Thenable<T>, thisArg?: any, description?: ICommandHandlerDescription): extHostTypes.Disposable {
E
Erich Gamma 已提交
54 55 56 57 58

		if (!id.trim().length) {
			throw new Error('invalid id');
		}

J
Johannes Rieken 已提交
59
		if (this._commands.has(id)) {
E
Erich Gamma 已提交
60 61 62
			throw new Error('command with id already exists');
		}

J
Johannes Rieken 已提交
63
		this._commands.set(id, { callback, thisArg, description });
64
		this._proxy.$registerCommand(id);
E
Erich Gamma 已提交
65

J
Johannes Rieken 已提交
66
		return new extHostTypes.Disposable(() => {
J
Johannes Rieken 已提交
67
			if (this._commands.delete(id)) {
J
Johannes Rieken 已提交
68 69 70
				this._proxy.$unregisterCommand(id);
			}
		});
E
Erich Gamma 已提交
71 72 73 74
	}

	executeCommand<T>(id: string, ...args: any[]): Thenable<T> {

J
Johannes Rieken 已提交
75
		if (this._commands.has(id)) {
E
Erich Gamma 已提交
76 77
			// we stay inside the extension host and support
			// to pass any kind of parameters around
78
			return this.$executeContributedCommand<T>(id, ...args);
E
Erich Gamma 已提交
79 80

		} else {
81 82
			// automagically convert some argument types

J
Johannes Rieken 已提交
83
			args = cloneAndChange(args, function (value) {
84 85 86 87 88 89 90 91 92 93 94 95 96
				if (value instanceof extHostTypes.Position) {
					return extHostTypeConverter.fromPosition(value);
				}
				if (value instanceof extHostTypes.Range) {
					return extHostTypeConverter.fromRange(value);
				}
				if (value instanceof extHostTypes.Location) {
					return extHostTypeConverter.location.from(value);
				}
				if (!Array.isArray(value)) {
					return value;
				}
			});
E
Erich Gamma 已提交
97

98
			return this._proxy.$executeCommand<T>(id, args);
E
Erich Gamma 已提交
99 100 101 102
		}

	}

103
	$executeContributedCommand<T>(id: string, ...args: any[]): Thenable<T> {
J
Johannes Rieken 已提交
104
		let command = this._commands.get(id);
E
Erich Gamma 已提交
105
		if (!command) {
106
			return TPromise.wrapError<T>(new Error(`Contributed command '${id}' does not exist.`));
E
Erich Gamma 已提交
107
		}
108

J
Johannes Rieken 已提交
109
		let { callback, thisArg, description } = command;
110 111 112 113

		if (description) {
			for (let i = 0; i < description.args.length; i++) {
				try {
J
Johannes Rieken 已提交
114
					validateConstraint(args[i], description.args[i].constraint);
115
				} catch (err) {
116
					return TPromise.wrapError<T>(new Error(`Running the contributed command:'${id}' failed. Illegal argument '${description.args[i].name}' - ${description.args[i].description}`));
117 118
				}
			}
119 120
		}

J
Joao Moreno 已提交
121 122
		args = args.map(arg => this._argumentProcessors.reduce((r, p) => p.processArgument(r), arg));

123
		try {
124
			let result = callback.apply(thisArg, args);
125
			return TPromise.as(result);
E
Erich Gamma 已提交
126
		} catch (err) {
127
			// console.log(err);
128 129 130 131 132
			// try {
			// 	console.log(toErrorMessage(err));
			// } catch (err) {
			// 	//
			// }
133
			return TPromise.wrapError<T>(new Error(`Running the contributed command:'${id}' failed.`));
E
Erich Gamma 已提交
134 135 136
		}
	}

137
	getCommands(filterUnderscoreCommands: boolean = false): Thenable<string[]> {
138
		return this._proxy.$getCommands().then(result => {
139 140 141 142 143
			if (filterUnderscoreCommands) {
				result = result.filter(command => command[0] !== '_');
			}
			return result;
		});
E
Erich Gamma 已提交
144
	}
145 146 147

	$getContributedCommandHandlerDescriptions(): TPromise<{ [id: string]: string | ICommandHandlerDescription }> {
		const result: { [id: string]: string | ICommandHandlerDescription } = Object.create(null);
J
Johannes Rieken 已提交
148
		this._commands.forEach((command, id) => {
J
Johannes Rieken 已提交
149
			let { description } = command;
150 151 152
			if (description) {
				result[id] = description;
			}
J
Johannes Rieken 已提交
153
		});
154 155
		return TPromise.as(result);
	}
E
Erich Gamma 已提交
156
}
157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174


export class CommandsConverter {

	private _commands: ExtHostCommands;
	private _heap: ExtHostHeapService;

	// --- conversion between internal and api commands
	constructor(commands: ExtHostCommands, heap: ExtHostHeapService) {

		this._commands = commands;
		this._heap = heap;
		this._commands.registerCommand('_internal_command_delegation', this._executeConvertedCommand, this);
	}

	toInternal(command: vscode.Command): modes.Command {

		if (!command) {
M
Matt Bierner 已提交
175
			return undefined;
176 177 178 179 180 181 182
		}

		const result: modes.Command = {
			id: command.command,
			title: command.title
		};

183
		if (command.command && !isFalsyOrEmpty(command.arguments)) {
184 185 186 187 188 189 190 191 192 193
			// we have a contributed command with arguments. that
			// means we don't want to send the arguments around

			const id = this._heap.keep(command);
			ObjectIdentifier.mixin(result, id);

			result.id = '_internal_command_delegation';
			result.arguments = [id];
		}

194 195 196 197
		if (command.tooltip) {
			result.tooltip = command.tooltip;
		}

198 199 200 201 202 203
		return result;
	}

	fromInternal(command: modes.Command): vscode.Command {

		if (!command) {
M
Matt Bierner 已提交
204
			return undefined;
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
		}

		const id = ObjectIdentifier.of(command);
		if (typeof id === 'number') {
			return this._heap.get<vscode.Command>(id);

		} else {
			return {
				command: command.id,
				title: command.title,
				arguments: command.arguments
			};
		}
	}

220
	private _executeConvertedCommand<R>(...args: any[]): Thenable<R> {
J
Johannes Rieken 已提交
221
		const actualCmd = this._heap.get<vscode.Command>(args[0]);
222 223 224
		return this._commands.executeCommand(actualCmd.command, ...actualCmd.arguments);
	}

J
Johannes Rieken 已提交
225
}