extHostCommands.ts 6.5 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 10
import { IThreadService } from 'vs/workbench/services/thread/common/threadService';
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 已提交
11 12
import * as extHostTypes from 'vs/workbench/api/node/extHostTypes';
import * as extHostTypeConverter from 'vs/workbench/api/node/extHostTypeConverters';
J
Johannes Rieken 已提交
13
import { cloneAndChange } from 'vs/base/common/objects';
14 15 16 17 18
import { MainContext, MainThreadCommandsShape, ExtHostCommandsShape, ObjectIdentifier } from './extHost.protocol';
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 已提交
19

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

J
Johannes Rieken 已提交
84
			args = cloneAndChange(args, function (value) {
85 86 87 88 89 90 91 92 93 94 95 96 97
				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 已提交
98

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

	}

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

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

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

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

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

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

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


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 已提交
176
			return undefined;
177 178 179 180 181 182 183
		}

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

184
		if (command.command && !isFalsyOrEmpty(command.arguments)) {
185 186 187 188 189 190 191 192 193 194
			// 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];
		}

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

199 200 201 202 203 204
		return result;
	}

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

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

		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
			};
		}
	}

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

J
Johannes Rieken 已提交
226
}