mainThreadTerminalService.ts 9.8 KB
Newer Older
D
Daniel Imms 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

J
Johannes Rieken 已提交
6
import { IDisposable, dispose } from 'vs/base/common/lifecycle';
7
import { ITerminalService, ITerminalInstance, IShellLaunchConfig, ITerminalProcessExtHostProxy, ITerminalProcessExtHostRequest, ITerminalDimensions, EXT_HOST_CREATION_DELAY } from 'vs/workbench/parts/terminal/common/terminal';
J
Johannes Rieken 已提交
8
import { TPromise } from 'vs/base/common/winjs.base';
A
Alex Dima 已提交
9
import { ExtHostContext, ExtHostTerminalServiceShape, MainThreadTerminalServiceShape, MainContext, IExtHostContext, ShellLaunchConfigDto } from 'vs/workbench/api/node/extHost.protocol';
10
import { extHostNamedCustomer } from 'vs/workbench/api/electron-browser/extHostCustomers';
D
Daniel Imms 已提交
11

12
@extHostNamedCustomer(MainContext.MainThreadTerminalService)
13
export class MainThreadTerminalService implements MainThreadTerminalServiceShape {
D
Daniel Imms 已提交
14

15
	private _proxy: ExtHostTerminalServiceShape;
16 17
	private _toDispose: IDisposable[] = [];
	private _terminalProcesses: { [id: number]: ITerminalProcessExtHostProxy } = {};
18 19
	private _terminalOnDidWriteDataListeners: { [id: number]: IDisposable } = {};
	private _terminalOnDidAcceptInputListeners: { [id: number]: IDisposable } = {};
20

D
Daniel Imms 已提交
21
	constructor(
22
		extHostContext: IExtHostContext,
23
		@ITerminalService private terminalService: ITerminalService
D
Daniel Imms 已提交
24
	) {
25
		this._proxy = extHostContext.getProxy(ExtHostContext.ExtHostTerminalService);
D
Daniel Imms 已提交
26
		this._toDispose.push(terminalService.onInstanceCreated((instance) => {
D
Daniel Imms 已提交
27
			// Delay this message so the TerminalInstance constructor has a chance to finish and
28 29
			// return the ID normally to the extension host. The ID that is passed here will be used
			// to register non-extension API terminals in the extension host.
30
			setTimeout(() => this._onTerminalOpened(instance), EXT_HOST_CREATION_DELAY);
31
		}));
D
Daniel Imms 已提交
32 33 34
		this._toDispose.push(terminalService.onInstanceDisposed(instance => this._onTerminalDisposed(instance)));
		this._toDispose.push(terminalService.onInstanceProcessIdReady(instance => this._onTerminalProcessIdReady(instance)));
		this._toDispose.push(terminalService.onInstanceDimensionsChanged(instance => this._onInstanceDimensionsChanged(instance)));
35
		this._toDispose.push(terminalService.onInstanceRequestExtHostProcess(request => this._onTerminalRequestExtHostProcess(request)));
36
		this._toDispose.push(terminalService.onActiveInstanceChanged(instance => this._onActiveTerminalChanged(instance ? instance.id : undefined)));
37
		this._toDispose.push(terminalService.onInstanceTitleChanged(instance => this._onTitleChanged(instance.id, instance.title)));
G
Gabriel Arjones 已提交
38
		this._toDispose.push(terminalService.onInstanceProcessLaunching(instance => this._onProcessLaunching(instance.id)));
D
Daniel Imms 已提交
39 40 41 42 43 44

		// Set initial ext host state
		this.terminalService.terminalInstances.forEach(t => {
			this._onTerminalOpened(t);
			t.processReady.then(() => this._onTerminalProcessIdReady(t));
		});
45 46 47 48
		const activeInstance = this.terminalService.getActiveInstance();
		if (activeInstance) {
			this._proxy.$acceptActiveTerminalChanged(activeInstance.id);
		}
49 50 51 52
	}

	public dispose(): void {
		this._toDispose = dispose(this._toDispose);
53 54 55

		// TODO@Daniel: Should all the previously created terminals be disposed
		// when the extension host process goes down ?
D
Daniel Imms 已提交
56
	}
D
Daniel Imms 已提交
57

A
Alex Dima 已提交
58
	public $createTerminal(name?: string, shellPath?: string, shellArgs?: string[], cwd?: string, env?: { [key: string]: string }, waitOnExit?: boolean): Thenable<number> {
59 60 61 62
		const shellLaunchConfig: IShellLaunchConfig = {
			name,
			executable: shellPath,
			args: shellArgs,
D
Daniel Imms 已提交
63
			cwd,
64
			waitOnExit,
65 66
			ignoreConfigurationCwd: true,
			env
67
		};
68
		return TPromise.as(this.terminalService.createTerminal(shellLaunchConfig).id);
D
Daniel Imms 已提交
69 70
	}

A
Alex Dima 已提交
71
	public $createTerminalRenderer(name: string): Thenable<number> {
D
Daniel Imms 已提交
72 73
		const instance = this.terminalService.createTerminalRenderer(name);
		return TPromise.as(instance.id);
74 75
	}

D
Daniel Imms 已提交
76
	public $show(terminalId: number, preserveFocus: boolean): void {
D
Daniel Imms 已提交
77
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
78 79 80 81
		if (terminalInstance) {
			this.terminalService.setActiveInstance(terminalInstance);
			this.terminalService.showPanel(!preserveFocus);
		}
D
Daniel Imms 已提交
82
	}
D
Daniel Imms 已提交
83

D
Daniel Imms 已提交
84
	public $hide(terminalId: number): void {
85
		if (this.terminalService.getActiveInstance().id === terminalId) {
D
Daniel Imms 已提交
86
			this.terminalService.hidePanel();
87
		}
D
Daniel Imms 已提交
88 89 90
	}

	public $dispose(terminalId: number): void {
D
Daniel Imms 已提交
91
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
92 93 94
		if (terminalInstance) {
			terminalInstance.dispose();
		}
D
Daniel Imms 已提交
95 96
	}

97
	public $terminalRendererWrite(terminalId: number, text: string): void {
D
Daniel Imms 已提交
98
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
99 100 101 102 103
		if (terminalInstance && terminalInstance.shellLaunchConfig.isRendererOnly) {
			terminalInstance.write(text);
		}
	}

104 105 106 107 108 109 110
	public $terminalRendererSetName(terminalId: number, name: string): void {
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
		if (terminalInstance && terminalInstance.shellLaunchConfig.isRendererOnly) {
			terminalInstance.setTitle(name, false);
		}
	}

D
Daniel Imms 已提交
111 112 113 114 115 116 117
	public $terminalRendererSetDimensions(terminalId: number, dimensions: ITerminalDimensions): void {
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
		if (terminalInstance && terminalInstance.shellLaunchConfig.isRendererOnly) {
			terminalInstance.setDimensions(dimensions);
		}
	}

D
Daniel Imms 已提交
118
	public $terminalRendererRegisterOnInputListener(terminalId: number): void {
119
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
120 121 122 123 124 125 126
		if (!terminalInstance) {
			return;
		}

		// Listener already registered
		if (this._terminalOnDidAcceptInputListeners.hasOwnProperty(terminalId)) {
			return;
127
		}
128 129 130 131

		// Register
		this._terminalOnDidAcceptInputListeners[terminalId] = terminalInstance.onRendererInput(data => this._onTerminalRendererInput(terminalId, data));
		terminalInstance.addDisposable(this._terminalOnDidAcceptInputListeners[terminalId]);
132 133
	}

D
Daniel Imms 已提交
134
	public $sendText(terminalId: number, text: string, addNewLine: boolean): void {
D
Daniel Imms 已提交
135
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
136 137 138
		if (terminalInstance) {
			terminalInstance.sendText(text, addNewLine);
		}
D
Daniel Imms 已提交
139
	}
140

D
Daniel Imms 已提交
141
	public $registerOnDataListener(terminalId: number): void {
D
Daniel Imms 已提交
142
		const terminalInstance = this.terminalService.getInstanceFromId(terminalId);
143 144
		if (!terminalInstance) {
			return;
D
Daniel Imms 已提交
145
		}
146 147 148 149 150 151 152 153 154 155 156

		// Listener already registered
		if (this._terminalOnDidWriteDataListeners[terminalId]) {
			return;
		}

		// Register
		this._terminalOnDidWriteDataListeners[terminalId] = terminalInstance.onData(data => {
			this._onTerminalData(terminalId, data);
		});
		terminalInstance.addDisposable(this._terminalOnDidWriteDataListeners[terminalId]);
D
Daniel Imms 已提交
157 158
	}

159 160 161 162
	private _onActiveTerminalChanged(terminalId: number | undefined): void {
		this._proxy.$acceptActiveTerminalChanged(terminalId);
	}

D
Daniel Imms 已提交
163 164 165 166
	private _onTerminalData(terminalId: number, data: string): void {
		this._proxy.$acceptTerminalProcessData(terminalId, data);
	}

167 168 169 170
	private _onTitleChanged(terminalId: number, name: string): void {
		this._proxy.$acceptTerminalTitleChange(terminalId, name);
	}

G
Gabriel Arjones 已提交
171 172 173 174
	private _onProcessLaunching(terminalId: number): void {
		this._proxy.$acceptTerminalProcessLaunching(terminalId);
	}

D
Daniel Imms 已提交
175 176
	private _onTerminalRendererInput(terminalId: number, data: string): void {
		this._proxy.$acceptTerminalRendererInput(terminalId, data);
177 178
	}

179
	private _onTerminalDisposed(terminalInstance: ITerminalInstance): void {
180 181
		this._proxy.$acceptTerminalClosed(terminalInstance.id);
	}
182

183
	private _onTerminalOpened(terminalInstance: ITerminalInstance): void {
184 185 186 187 188 189 190
		if (terminalInstance.title) {
			this._proxy.$acceptTerminalOpened(terminalInstance.id, terminalInstance.title);
		} else {
			terminalInstance.waitForTitle().then(title => {
				this._proxy.$acceptTerminalOpened(terminalInstance.id, title);
			});
		}
191 192
	}

193 194 195
	private _onTerminalProcessIdReady(terminalInstance: ITerminalInstance): void {
		this._proxy.$acceptTerminalProcessId(terminalInstance.id, terminalInstance.processId);
	}
196

D
Daniel Imms 已提交
197 198 199 200 201 202 203 204
	private _onInstanceDimensionsChanged(instance: ITerminalInstance): void {
		// Only send the dimensions if the terminal is a renderer only as there is no API to access
		// dimensions on a plain Terminal.
		if (instance.shellLaunchConfig.isRendererOnly) {
			this._proxy.$acceptTerminalRendererDimensions(instance.id, instance.cols, instance.rows);
		}
	}

205 206 207 208 209 210 211 212 213 214
	private _onTerminalRequestExtHostProcess(request: ITerminalProcessExtHostRequest): void {
		this._terminalProcesses[request.proxy.terminalId] = request.proxy;
		const shellLaunchConfigDto: ShellLaunchConfigDto = {
			name: request.shellLaunchConfig.name,
			executable: request.shellLaunchConfig.executable,
			args: request.shellLaunchConfig.args,
			cwd: request.shellLaunchConfig.cwd,
			env: request.shellLaunchConfig.env
		};
		this._proxy.$createProcess(request.proxy.terminalId, shellLaunchConfigDto, request.cols, request.rows);
215
		request.proxy.onInput(data => this._proxy.$acceptProcessInput(request.proxy.terminalId, data));
D
Daniel Imms 已提交
216
		request.proxy.onResize(dimensions => this._proxy.$acceptProcessResize(request.proxy.terminalId, dimensions.cols, dimensions.rows));
217
		request.proxy.onShutdown(immediate => this._proxy.$acceptProcessShutdown(request.proxy.terminalId, immediate));
218 219 220 221 222 223 224 225 226 227 228 229 230
	}

	public $sendProcessTitle(terminalId: number, title: string): void {
		this._terminalProcesses[terminalId].emitTitle(title);
	}

	public $sendProcessData(terminalId: number, data: string): void {
		this._terminalProcesses[terminalId].emitData(data);
	}

	public $sendProcessPid(terminalId: number, pid: number): void {
		this._terminalProcesses[terminalId].emitPid(pid);
	}
D
Daniel Imms 已提交
231 232 233

	public $sendProcessExit(terminalId: number, exitCode: number): void {
		this._terminalProcesses[terminalId].emitExit(exitCode);
D
Daniel Imms 已提交
234
		delete this._terminalProcesses[terminalId];
D
Daniel Imms 已提交
235
	}
D
Daniel Imms 已提交
236
}