tunnelView.ts 43.9 KB
Newer Older
A
Alex Ross 已提交
1 2 3 4 5 6 7 8
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

import 'vs/css!./media/tunnelView';
import * as nls from 'vs/nls';
import * as dom from 'vs/base/browser/dom';
9
import { IViewDescriptor, IEditableData, IViewsService, IViewDescriptorService } from 'vs/workbench/common/views';
10
import { WorkbenchAsyncDataTree } from 'vs/platform/list/browser/listService';
A
Alex Ross 已提交
11
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
12
import { IContextMenuService, IContextViewService } from 'vs/platform/contextview/browser/contextView';
A
Alex Ross 已提交
13 14 15 16
import { IContextKeyService, IContextKey, RawContextKey, ContextKeyExpr } from 'vs/platform/contextkey/common/contextkey';
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
import { IOpenerService } from 'vs/platform/opener/common/opener';
A
Alex Ross 已提交
17
import { IQuickInputService, IQuickPickItem, QuickPickInput } from 'vs/platform/quickinput/common/quickInput';
A
Alex Ross 已提交
18 19 20
import { ICommandService, ICommandHandler, CommandsRegistry } from 'vs/platform/commands/common/commands';
import { Event, Emitter } from 'vs/base/common/event';
import { IListVirtualDelegate } from 'vs/base/browser/ui/list/list';
21
import { ITreeRenderer, ITreeNode, IAsyncDataSource, ITreeContextMenuEvent, ITreeMouseEvent } from 'vs/base/browser/ui/tree/tree';
A
Alex Ross 已提交
22
import { IWorkbenchEnvironmentService } from 'vs/workbench/services/environment/common/environmentService';
23
import { Disposable, IDisposable, toDisposable, MutableDisposable, dispose, DisposableStore } from 'vs/base/common/lifecycle';
24
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
A
Alex Ross 已提交
25
import { IconLabel } from 'vs/base/browser/ui/iconLabel/iconLabel';
J
João Moreno 已提交
26
import { ActionRunner, IAction } from 'vs/base/common/actions';
27 28
import { IMenuService, MenuId, IMenu, MenuRegistry, MenuItemAction, ILocalizedString, SubmenuItemAction } from 'vs/platform/actions/common/actions';
import { createAndFillInContextMenuActions, createAndFillInActionBarActions, MenuEntryActionViewItem, SubmenuEntryActionViewItem } from 'vs/platform/actions/browser/menuEntryActionViewItem';
A
Alex Ross 已提交
29
import { IRemoteExplorerService, TunnelModel, MakeAddress, TunnelType, ITunnelItem, Tunnel, mapHasTunnelLocalhostOrAllInterfaces, TUNNEL_VIEW_ID } from 'vs/workbench/services/remote/common/remoteExplorerService';
A
Alex Ross 已提交
30
import { IClipboardService } from 'vs/platform/clipboard/common/clipboardService';
31
import { INotificationService, Severity } from 'vs/platform/notification/common/notification';
32 33 34
import { InputBox, MessageType } from 'vs/base/browser/ui/inputbox/inputBox';
import { attachInputBoxStyler } from 'vs/platform/theme/common/styler';
import { once } from 'vs/base/common/functional';
35
import { KeyCode, KeyMod } from 'vs/base/common/keyCodes';
36 37
import { IThemeService } from 'vs/platform/theme/common/themeService';
import { IKeyboardEvent } from 'vs/base/browser/keyboardEvent';
S
SteVen Batten 已提交
38
import { ViewPane, IViewPaneOptions } from 'vs/workbench/browser/parts/views/viewPaneContainer';
A
Alex Ross 已提交
39
import { URI } from 'vs/base/common/uri';
A
Alex Ross 已提交
40
import { RemoteTunnel } from 'vs/platform/remote/common/tunnel';
S
Sandeep Somavarapu 已提交
41
import { SyncDescriptor } from 'vs/platform/instantiation/common/descriptors';
42
import { KeybindingsRegistry, KeybindingWeight } from 'vs/platform/keybinding/common/keybindingsRegistry';
43
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
44
import { ActionViewItem } from 'vs/base/browser/ui/actionbar/actionViewItems';
A
Alex Ross 已提交
45 46

export const forwardedPortsViewEnabled = new RawContextKey<boolean>('forwardedPortsViewEnabled', false);
A
Alex Ross 已提交
47 48 49 50 51 52 53 54 55 56 57 58 59 60

class TunnelTreeVirtualDelegate implements IListVirtualDelegate<ITunnelItem> {
	getHeight(element: ITunnelItem): number {
		return 22;
	}

	getTemplateId(element: ITunnelItem): string {
		return 'tunnelItemTemplate';
	}
}

export interface ITunnelViewModel {
	onForwardedPortsChanged: Event<void>;
	readonly forwarded: TunnelItem[];
A
Alex Ross 已提交
61
	readonly detected: TunnelItem[];
62
	readonly candidates: TunnelItem[];
63
	readonly input: TunnelItem;
A
Alex Ross 已提交
64
	groups(): Promise<ITunnelGroup[]>;
A
Alex Ross 已提交
65 66 67 68 69 70
}

export class TunnelViewModel extends Disposable implements ITunnelViewModel {
	private _onForwardedPortsChanged: Emitter<void> = new Emitter();
	public onForwardedPortsChanged: Event<void> = this._onForwardedPortsChanged.event;
	private model: TunnelModel;
71
	private _input: TunnelItem;
72
	private _candidates: Map<string, { host: string, port: number, detail: string }> = new Map();
A
Alex Ross 已提交
73 74

	constructor(
75
		@IRemoteExplorerService private readonly remoteExplorerService: IRemoteExplorerService) {
A
Alex Ross 已提交
76 77 78 79 80
		super();
		this.model = remoteExplorerService.tunnelModel;
		this._register(this.model.onForwardPort(() => this._onForwardedPortsChanged.fire()));
		this._register(this.model.onClosePort(() => this._onForwardedPortsChanged.fire()));
		this._register(this.model.onPortName(() => this._onForwardedPortsChanged.fire()));
81
		this._register(this.model.onCandidatesChanged(() => this._onForwardedPortsChanged.fire()));
82 83 84 85 86 87 88
		this._input = {
			label: nls.localize('remote.tunnelsView.add', "Forward a Port..."),
			tunnelType: TunnelType.Add,
			remoteHost: 'localhost',
			remotePort: 0,
			description: ''
		};
A
Alex Ross 已提交
89 90
	}

A
Alex Ross 已提交
91
	async groups(): Promise<ITunnelGroup[]> {
A
Alex Ross 已提交
92
		const groups: ITunnelGroup[] = [];
93 94 95 96
		this._candidates = new Map();
		(await this.model.candidates).forEach(candidate => {
			this._candidates.set(MakeAddress(candidate.host, candidate.port), candidate);
		});
97
		if ((this.model.forwarded.size > 0) || this.remoteExplorerService.getEditableData(undefined)) {
A
Alex Ross 已提交
98 99 100 101 102 103
			groups.push({
				label: nls.localize('remote.tunnelsView.forwarded', "Forwarded"),
				tunnelType: TunnelType.Forwarded,
				items: this.forwarded
			});
		}
A
Alex Ross 已提交
104
		if (this.model.detected.size > 0) {
A
Alex Ross 已提交
105
			groups.push({
106
				label: nls.localize('remote.tunnelsView.detected', "Existing Tunnels"),
A
Alex Ross 已提交
107 108
				tunnelType: TunnelType.Detected,
				items: this.detected
A
Alex Ross 已提交
109 110
			});
		}
A
Alex Ross 已提交
111 112
		const candidates = await this.candidates;
		if (candidates.length > 0) {
A
Alex Ross 已提交
113
			groups.push({
114
				label: nls.localize('remote.tunnelsView.candidates', "Not Forwarded"),
A
Alex Ross 已提交
115 116 117 118
				tunnelType: TunnelType.Candidate,
				items: candidates
			});
		}
119 120
		if (groups.length === 0) {
			groups.push(this._input);
121
		}
A
Alex Ross 已提交
122 123 124
		return groups;
	}

125 126 127 128 129 130 131
	private addProcessInfoFromCandidate(tunnelItem: ITunnelItem) {
		const key = MakeAddress(tunnelItem.remoteHost, tunnelItem.remotePort);
		if (this._candidates.has(key)) {
			tunnelItem.description = this._candidates.get(key)!.detail;
		}
	}

A
Alex Ross 已提交
132
	get forwarded(): TunnelItem[] {
133
		const forwarded = Array.from(this.model.forwarded.values()).map(tunnel => {
134 135 136
			const tunnelItem = TunnelItem.createFromTunnel(tunnel);
			this.addProcessInfoFromCandidate(tunnelItem);
			return tunnelItem;
A
Alex Ross 已提交
137 138 139 140 141 142
		}).sort((a: TunnelItem, b: TunnelItem) => {
			if (a.remotePort === b.remotePort) {
				return a.remoteHost < b.remoteHost ? -1 : 1;
			} else {
				return a.remotePort < b.remotePort ? -1 : 1;
			}
A
Alex Ross 已提交
143
		});
144 145 146 147
		if (this.remoteExplorerService.getEditableData(undefined)) {
			forwarded.push(this._input);
		}
		return forwarded;
A
Alex Ross 已提交
148 149
	}

A
Alex Ross 已提交
150 151
	get detected(): TunnelItem[] {
		return Array.from(this.model.detected.values()).map(tunnel => {
152 153 154
			const tunnelItem = TunnelItem.createFromTunnel(tunnel, TunnelType.Detected, false);
			this.addProcessInfoFromCandidate(tunnelItem);
			return tunnelItem;
A
Alex Ross 已提交
155 156 157
		});
	}

158 159 160
	get candidates(): TunnelItem[] {
		const candidates: TunnelItem[] = [];
		this._candidates.forEach(value => {
161 162
			if (!mapHasTunnelLocalhostOrAllInterfaces(this.model.forwarded, value.host, value.port) &&
				!mapHasTunnelLocalhostOrAllInterfaces(this.model.detected, value.host, value.port)) {
163
				candidates.push(new TunnelItem(TunnelType.Candidate, value.host, value.port, undefined, undefined, false, undefined, value.detail));
164
			}
A
Alex Ross 已提交
165
		});
166
		return candidates;
A
Alex Ross 已提交
167 168
	}

169
	get input(): TunnelItem {
170 171 172
		return this._input;
	}

A
Alex Ross 已提交
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
	dispose() {
		super.dispose();
	}
}

interface ITunnelTemplateData {
	elementDisposable: IDisposable;
	container: HTMLElement;
	iconLabel: IconLabel;
	actionBar: ActionBar;
}

class TunnelTreeRenderer extends Disposable implements ITreeRenderer<ITunnelGroup | ITunnelItem, ITunnelItem, ITunnelTemplateData> {
	static readonly ITEM_HEIGHT = 22;
	static readonly TREE_TEMPLATE_ID = 'tunnelItemTemplate';

A
Alex Ross 已提交
189
	private inputDone?: (success: boolean, finishEditing: boolean) => void;
A
Alex Ross 已提交
190 191 192 193 194 195
	private _actionRunner: ActionRunner | undefined;

	constructor(
		private readonly viewId: string,
		@IMenuService private readonly menuService: IMenuService,
		@IContextKeyService private readonly contextKeyService: IContextKeyService,
196 197 198 199
		@IInstantiationService private readonly instantiationService: IInstantiationService,
		@IContextViewService private readonly contextViewService: IContextViewService,
		@IThemeService private readonly themeService: IThemeService,
		@IRemoteExplorerService private readonly remoteExplorerService: IRemoteExplorerService
A
Alex Ross 已提交
200 201 202 203 204 205 206 207 208 209 210 211 212
	) {
		super();
	}

	set actionRunner(actionRunner: ActionRunner) {
		this._actionRunner = actionRunner;
	}

	get templateId(): string {
		return TunnelTreeRenderer.TREE_TEMPLATE_ID;
	}

	renderTemplate(container: HTMLElement): ITunnelTemplateData {
213
		container.classList.add('custom-view-tree-node-item');
A
Alex Ross 已提交
214 215 216 217 218 219
		const iconLabel = new IconLabel(container, { supportHighlights: true });
		// dom.addClass(iconLabel.element, 'tunnel-view-label');
		const actionsContainer = dom.append(iconLabel.element, dom.$('.actions'));
		const actionBar = new ActionBar(actionsContainer, {
			actionViewItemProvider: (action: IAction) => {
				if (action instanceof MenuItemAction) {
220
					return this.instantiationService.createInstance(MenuEntryActionViewItem, action);
221 222
				} else if (action instanceof SubmenuItemAction) {
					return this.instantiationService.createInstance(SubmenuEntryActionViewItem, action);
A
Alex Ross 已提交
223 224 225 226 227 228 229 230 231 232
				}

				return undefined;
			}
		});

		return { iconLabel, actionBar, container, elementDisposable: Disposable.None };
	}

	private isTunnelItem(item: ITunnelGroup | ITunnelItem): item is ITunnelItem {
233
		return !!((<ITunnelItem>item).remotePort);
A
Alex Ross 已提交
234 235 236 237 238
	}

	renderElement(element: ITreeNode<ITunnelGroup | ITunnelItem, ITunnelGroup | ITunnelItem>, index: number, templateData: ITunnelTemplateData): void {
		templateData.elementDisposable.dispose();
		const node = element.element;
239

A
Alex Ross 已提交
240 241
		// reset
		templateData.actionBar.clear();
A
Alex Ross 已提交
242
		let editableData: IEditableData | undefined;
A
Alex Ross 已提交
243
		if (this.isTunnelItem(node)) {
244
			editableData = this.remoteExplorerService.getEditableData(node);
245 246
			if (editableData) {
				templateData.iconLabel.element.style.display = 'none';
A
Alex Ross 已提交
247
				this.renderInputBox(templateData.container, editableData);
248 249 250
			} else {
				templateData.iconLabel.element.style.display = 'flex';
				this.renderTunnel(node, templateData);
A
Alex Ross 已提交
251
			}
252
		} else if ((node.tunnelType === TunnelType.Add) && (editableData = this.remoteExplorerService.getEditableData(undefined))) {
A
Alex Ross 已提交
253 254
			templateData.iconLabel.element.style.display = 'none';
			this.renderInputBox(templateData.container, editableData);
A
Alex Ross 已提交
255
		} else {
A
Alex Ross 已提交
256
			templateData.iconLabel.element.style.display = 'flex';
A
Alex Ross 已提交
257 258 259 260
			templateData.iconLabel.setLabel(node.label);
		}
	}

261
	private renderTunnel(node: ITunnelItem, templateData: ITunnelTemplateData) {
262 263
		const label = node.label + (node.description ? (' - ' + node.description) : '');
		templateData.iconLabel.setLabel(node.label, node.description, { title: label, extraClasses: ['tunnel-view-label'] });
264
		templateData.actionBar.context = node;
265
		const contextKeyService = this._register(this.contextKeyService.createScoped());
266 267 268
		contextKeyService.createKey('view', this.viewId);
		contextKeyService.createKey('tunnelType', node.tunnelType);
		contextKeyService.createKey('tunnelCloseable', node.closeable);
269 270 271
		const disposableStore = new DisposableStore();
		templateData.elementDisposable = disposableStore;
		const menu = disposableStore.add(this.menuService.createMenu(MenuId.TunnelInline, contextKeyService));
272
		const actions: IAction[] = [];
273
		disposableStore.add(createAndFillInActionBarActions(menu, { shouldForwardArgs: true }, actions));
274 275 276 277 278 279 280 281
		if (actions) {
			templateData.actionBar.push(actions, { icon: true, label: false });
			if (this._actionRunner) {
				templateData.actionBar.actionRunner = this._actionRunner;
			}
		}
	}

A
Alex Ross 已提交
282
	private renderInputBox(container: HTMLElement, editableData: IEditableData): IDisposable {
A
Alex Ross 已提交
283 284 285 286 287
		// Required for FireFox. The blur event doesn't fire on FireFox when you just mash the "+" button to forward a port.
		if (this.inputDone) {
			this.inputDone(false, false);
			this.inputDone = undefined;
		}
288 289 290 291 292
		const value = editableData.startingValue || '';
		const inputBox = new InputBox(container, this.contextViewService, {
			ariaLabel: nls.localize('remote.tunnelsView.input', "Press Enter to confirm or Escape to cancel."),
			validationOptions: {
				validation: (value) => {
293 294
					const message = editableData.validationMessage(value);
					if (!message || message.severity !== Severity.Error) {
295 296 297 298
						return null;
					}

					return {
299
						content: message.content,
300 301 302 303 304 305 306 307 308 309 310
						formatContent: true,
						type: MessageType.ERROR
					};
				}
			},
			placeholder: editableData.placeholder || ''
		});
		const styler = attachInputBoxStyler(inputBox, this.themeService);

		inputBox.value = value;
		inputBox.focus();
311
		inputBox.select({ start: 0, end: editableData.startingValue ? editableData.startingValue.length : 0 });
312 313

		const done = once((success: boolean, finishEditing: boolean) => {
A
Alex Ross 已提交
314 315 316
			if (this.inputDone) {
				this.inputDone = undefined;
			}
317
			inputBox.element.style.display = 'none';
A
Alex Ross 已提交
318
			const inputValue = inputBox.value;
319 320
			dispose(toDispose);
			if (finishEditing) {
A
Alex Ross 已提交
321
				editableData.onFinish(inputValue, success);
322 323
			}
		});
A
Alex Ross 已提交
324
		this.inputDone = done;
325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347

		const toDispose = [
			inputBox,
			dom.addStandardDisposableListener(inputBox.inputElement, dom.EventType.KEY_DOWN, (e: IKeyboardEvent) => {
				if (e.equals(KeyCode.Enter)) {
					if (inputBox.validate()) {
						done(true, true);
					}
				} else if (e.equals(KeyCode.Escape)) {
					done(false, true);
				}
			}),
			dom.addDisposableListener(inputBox.inputElement, dom.EventType.BLUR, () => {
				done(inputBox.isInputValid(), true);
			}),
			styler
		];

		return toDisposable(() => {
			done(false, false);
		});
	}

A
Alex Ross 已提交
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371
	disposeElement(resource: ITreeNode<ITunnelGroup | ITunnelItem, ITunnelGroup | ITunnelItem>, index: number, templateData: ITunnelTemplateData): void {
		templateData.elementDisposable.dispose();
	}

	disposeTemplate(templateData: ITunnelTemplateData): void {
		templateData.actionBar.dispose();
		templateData.elementDisposable.dispose();
	}
}

class TunnelDataSource implements IAsyncDataSource<ITunnelViewModel, ITunnelItem | ITunnelGroup> {
	hasChildren(element: ITunnelViewModel | ITunnelItem | ITunnelGroup) {
		if (element instanceof TunnelViewModel) {
			return true;
		} else if (element instanceof TunnelItem) {
			return false;
		} else if ((<ITunnelGroup>element).items) {
			return true;
		}
		return false;
	}

	getChildren(element: ITunnelViewModel | ITunnelItem | ITunnelGroup) {
		if (element instanceof TunnelViewModel) {
A
Alex Ross 已提交
372
			return element.groups();
A
Alex Ross 已提交
373 374 375 376 377 378 379 380 381 382 383 384
		} else if (element instanceof TunnelItem) {
			return [];
		} else if ((<ITunnelGroup>element).items) {
			return (<ITunnelGroup>element).items!;
		}
		return [];
	}
}

interface ITunnelGroup {
	tunnelType: TunnelType;
	label: string;
A
Alex Ross 已提交
385
	items?: ITunnelItem[] | Promise<ITunnelItem[]>;
A
Alex Ross 已提交
386 387 388
}

class TunnelItem implements ITunnelItem {
A
Alex Ross 已提交
389
	static createFromTunnel(tunnel: Tunnel, type: TunnelType = TunnelType.Forwarded, closeable?: boolean) {
390
		return new TunnelItem(type, tunnel.remoteHost, tunnel.remotePort, tunnel.localAddress, tunnel.localPort, closeable === undefined ? tunnel.closeable : closeable, tunnel.name, tunnel.description);
A
Alex Ross 已提交
391 392
	}

A
Alex Ross 已提交
393 394
	constructor(
		public tunnelType: TunnelType,
395 396
		public remoteHost: string,
		public remotePort: number,
A
Alex Ross 已提交
397
		public localAddress?: string,
398
		public localPort?: number,
A
Alex Ross 已提交
399 400 401 402 403
		public closeable?: boolean,
		public name?: string,
		private _description?: string,
	) { }
	get label(): string {
A
Alex Ross 已提交
404
		if (this.name) {
A
Alex Ross 已提交
405
			return nls.localize('remote.tunnelsView.forwardedPortLabel0', "{0}", this.name);
A
Alex Ross 已提交
406
		} else if (this.localAddress) {
A
Alex Ross 已提交
407
			return nls.localize('remote.tunnelsView.forwardedPortLabel1', "{0} \u2192 {1}", this.remotePort, TunnelItem.compactLongAddress(this.localAddress));
A
Alex Ross 已提交
408
		} else {
A
Alex Ross 已提交
409
			return nls.localize('remote.tunnelsView.forwardedPortLabel2', "{0}", this.remotePort);
A
Alex Ross 已提交
410 411 412
		}
	}

A
Alex Ross 已提交
413 414 415 416 417 418 419
	private static compactLongAddress(address: string): string {
		if (address.length < 15) {
			return address;
		}
		return new URL(address).host;
	}

420 421 422 423
	set description(description: string | undefined) {
		this._description = description;
	}

A
Alex Ross 已提交
424
	get description(): string | undefined {
A
Alex Ross 已提交
425
		if (this._description) {
A
Alex Ross 已提交
426
			return this._description;
A
Alex Ross 已提交
427
		} else if (this.name) {
A
Alex Ross 已提交
428
			return nls.localize('remote.tunnelsView.forwardedPortDescription0', "{0} \u2192 {1}", this.remotePort, this.localAddress);
A
Alex Ross 已提交
429
		}
A
Alex Ross 已提交
430
		return undefined;
A
Alex Ross 已提交
431 432 433 434 435
	}
}

export const TunnelTypeContextKey = new RawContextKey<TunnelType>('tunnelType', TunnelType.Add);
export const TunnelCloseableContextKey = new RawContextKey<boolean>('tunnelCloseable', false);
436 437 438
const TunnelViewFocusContextKey = new RawContextKey<boolean>('tunnelViewFocus', false);
const TunnelViewSelectionKeyName = 'tunnelViewSelection';
const TunnelViewSelectionContextKey = new RawContextKey<ITunnelItem | undefined>(TunnelViewSelectionKeyName, undefined);
439
const PortChangableContextKey = new RawContextKey<boolean>('portChangable', false);
A
Alex Ross 已提交
440

M
Matt Bierner 已提交
441 442
class TunnelDataTree extends WorkbenchAsyncDataTree<any, any, any> { }

S
SteVen Batten 已提交
443
export class TunnelPanel extends ViewPane {
A
Alex Ross 已提交
444
	static readonly ID = TUNNEL_VIEW_ID;
445
	static readonly TITLE = nls.localize('remote.tunnel', "Forwarded Ports");
M
Matt Bierner 已提交
446
	private tree!: TunnelDataTree;
A
Alex Ross 已提交
447 448
	private tunnelTypeContext: IContextKey<TunnelType>;
	private tunnelCloseableContext: IContextKey<boolean>;
449 450
	private tunnelViewFocusContext: IContextKey<boolean>;
	private tunnelViewSelectionContext: IContextKey<ITunnelItem | undefined>;
451
	private portChangableContextKey: IContextKey<boolean>;
A
Alex Ross 已提交
452
	private isEditing: boolean = false;
A
Alex Ross 已提交
453 454 455 456 457
	private titleActions: IAction[] = [];
	private readonly titleActionsDisposable = this._register(new MutableDisposable());

	constructor(
		protected viewModel: ITunnelViewModel,
S
SteVen Batten 已提交
458
		options: IViewPaneOptions,
A
Alex Ross 已提交
459 460 461 462 463
		@IKeybindingService protected keybindingService: IKeybindingService,
		@IContextMenuService protected contextMenuService: IContextMenuService,
		@IContextKeyService protected contextKeyService: IContextKeyService,
		@IConfigurationService protected configurationService: IConfigurationService,
		@IInstantiationService protected readonly instantiationService: IInstantiationService,
464
		@IViewDescriptorService viewDescriptorService: IViewDescriptorService,
465
		@IOpenerService openerService: IOpenerService,
A
Alex Ross 已提交
466 467 468
		@IQuickInputService protected quickInputService: IQuickInputService,
		@ICommandService protected commandService: ICommandService,
		@IMenuService private readonly menuService: IMenuService,
469
		@IContextViewService private readonly contextViewService: IContextViewService,
470
		@IThemeService themeService: IThemeService,
471 472
		@IRemoteExplorerService private readonly remoteExplorerService: IRemoteExplorerService,
		@ITelemetryService telemetryService: ITelemetryService,
A
Alex Ross 已提交
473
	) {
474
		super(options, keybindingService, contextMenuService, configurationService, contextKeyService, viewDescriptorService, instantiationService, openerService, themeService, telemetryService);
A
Alex Ross 已提交
475 476
		this.tunnelTypeContext = TunnelTypeContextKey.bindTo(contextKeyService);
		this.tunnelCloseableContext = TunnelCloseableContextKey.bindTo(contextKeyService);
477 478
		this.tunnelViewFocusContext = TunnelViewFocusContextKey.bindTo(contextKeyService);
		this.tunnelViewSelectionContext = TunnelViewSelectionContextKey.bindTo(contextKeyService);
479
		this.portChangableContextKey = PortChangableContextKey.bindTo(contextKeyService);
A
Alex Ross 已提交
480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499

		const scopedContextKeyService = this._register(this.contextKeyService.createScoped());
		scopedContextKeyService.createKey('view', TunnelPanel.ID);

		const titleMenu = this._register(this.menuService.createMenu(MenuId.TunnelTitle, scopedContextKeyService));
		const updateActions = () => {
			this.titleActions = [];
			this.titleActionsDisposable.value = createAndFillInActionBarActions(titleMenu, undefined, this.titleActions);
			this.updateActions();
		};

		this._register(titleMenu.onDidChange(updateActions));
		updateActions();

		this._register(toDisposable(() => {
			this.titleActions = [];
		}));
	}

	protected renderBody(container: HTMLElement): void {
J
Joao Moreno 已提交
500 501
		super.renderBody(container);

A
Alex Ross 已提交
502 503
		const panelContainer = dom.append(container, dom.$('.tree-explorer-viewlet-tree-view'));
		const treeContainer = dom.append(panelContainer, dom.$('.customview-tree'));
504
		treeContainer.classList.add('file-icon-themable-tree', 'show-file-icons');
A
Alex Ross 已提交
505

506
		const renderer = new TunnelTreeRenderer(TunnelPanel.ID, this.menuService, this.contextKeyService, this.instantiationService, this.contextViewService, this.themeService, this.remoteExplorerService);
M
Matt Bierner 已提交
507
		this.tree = this.instantiationService.createInstance(TunnelDataTree,
A
Alex Ross 已提交
508 509 510 511 512 513 514 515 516 517 518 519 520 521
			'RemoteTunnels',
			treeContainer,
			new TunnelTreeVirtualDelegate(),
			[renderer],
			new TunnelDataSource(),
			{
				collapseByDefault: (e: ITunnelItem | ITunnelGroup): boolean => {
					return false;
				},
				keyboardNavigationLabelProvider: {
					getKeyboardNavigationLabel: (item: ITunnelItem | ITunnelGroup) => {
						return item.label;
					}
				},
522 523 524 525 526 527 528 529 530 531 532 533
				multipleSelectionSupport: false,
				accessibilityProvider: {
					getAriaLabel: (item: ITunnelItem | ITunnelGroup) => {
						if (item instanceof TunnelItem) {
							if (item.localAddress) {
								return nls.localize('remote.tunnel.ariaLabelForwarded', "Remote port {0}:{1} forwarded to local address {2}", item.remoteHost, item.remotePort, item.localAddress);
							} else {
								return nls.localize('remote.tunnel.ariaLabelCandidate', "Remote port {0}:{1} not forwarded", item.remoteHost, item.remotePort);
							}
						} else {
							return item.label;
						}
I
isidor 已提交
534 535
					},
					getWidgetAriaLabel: () => nls.localize('tunnelView', "Tunnel View")
J
João Moreno 已提交
536
				}
A
Alex Ross 已提交
537 538 539 540 541 542
			}
		);
		const actionRunner: ActionRunner = new ActionRunner();
		renderer.actionRunner = actionRunner;

		this._register(this.tree.onContextMenu(e => this.onContextMenu(e, actionRunner)));
543
		this._register(this.tree.onMouseDblClick(e => this.onMouseDblClick(e)));
544 545 546
		this._register(this.tree.onDidChangeFocus(e => this.onFocusChanged(e.elements)));
		this._register(this.tree.onDidFocus(() => this.tunnelViewFocusContext.set(true)));
		this._register(this.tree.onDidBlur(() => this.tunnelViewFocusContext.set(false)));
A
Alex Ross 已提交
547 548 549

		this.tree.setInput(this.viewModel);
		this._register(this.viewModel.onForwardedPortsChanged(() => {
A
Alex Ross 已提交
550
			this._onDidChangeViewWelcomeState.fire();
A
Alex Ross 已提交
551 552 553
			this.tree.updateChildren(undefined, true);
		}));

554
		this._register(Event.debounce(this.tree.onDidOpen, (last, event) => event, 75, true)(e => {
555
			if (e.element && (e.element.tunnelType === TunnelType.Add)) {
556
				this.commandService.executeCommand(ForwardPortAction.INLINE_ID);
A
Alex Ross 已提交
557 558
			}
		}));
559 560

		this._register(this.remoteExplorerService.onDidChangeEditable(async e => {
A
Alex Ross 已提交
561 562
			this.isEditing = !!this.remoteExplorerService.getEditableData(e);
			this._onDidChangeViewWelcomeState.fire();
563

A
Alex Ross 已提交
564
			if (!this.isEditing) {
565
				treeContainer.classList.remove('highlight');
566 567 568 569
			}

			await this.tree.updateChildren(undefined, false);

A
Alex Ross 已提交
570
			if (this.isEditing) {
571
				treeContainer.classList.add('highlight');
A
Alex Ross 已提交
572 573 574 575
				if (!e) {
					// When we are in editing mode for a new forward, rather than updating an existing one we need to reveal the input box since it might be out of view.
					this.tree.reveal(this.viewModel.input);
				}
576 577 578 579
			} else {
				this.tree.domFocus();
			}
		}));
A
Alex Ross 已提交
580 581 582
	}

	private get contributedContextMenu(): IMenu {
583
		const contributedContextMenu = this._register(this.menuService.createMenu(MenuId.TunnelContext, this.tree.contextKeyService));
A
Alex Ross 已提交
584 585 586 587 588 589 590
		return contributedContextMenu;
	}

	getActions(): IAction[] {
		return this.titleActions;
	}

A
Alex Ross 已提交
591 592 593 594
	shouldShowWelcome(): boolean {
		return this.viewModel.forwarded.length === 0 && this.viewModel.candidates.length === 0 && !this.isEditing;
	}

A
Alex Ross 已提交
595 596 597 598 599
	focus(): void {
		super.focus();
		this.tree.domFocus();
	}

600 601 602 603
	private onFocusChanged(elements: ITunnelItem[]) {
		const item = elements && elements.length ? elements[0] : undefined;
		if (item) {
			this.tunnelViewSelectionContext.set(item);
604
			this.tunnelTypeContext.set(item.tunnelType);
605
			this.tunnelCloseableContext.set(!!item.closeable);
606
			this.portChangableContextKey.set(!!item.localPort);
607
		} else {
608
			this.tunnelTypeContext.reset();
609 610
			this.tunnelViewSelectionContext.reset();
			this.tunnelCloseableContext.reset();
611
			this.portChangableContextKey.reset();
612 613 614
		}
	}

A
Alex Ross 已提交
615
	private onContextMenu(treeEvent: ITreeContextMenuEvent<ITunnelItem | ITunnelGroup>, actionRunner: ActionRunner): void {
616
		if ((treeEvent.element !== null) && !(treeEvent.element instanceof TunnelItem)) {
A
Alex Ross 已提交
617 618 619 620 621 622 623 624
			return;
		}
		const node: ITunnelItem | null = treeEvent.element;
		const event: UIEvent = treeEvent.browserEvent;

		event.preventDefault();
		event.stopPropagation();

625 626 627 628
		if (node) {
			this.tree!.setFocus([node]);
			this.tunnelTypeContext.set(node.tunnelType);
			this.tunnelCloseableContext.set(!!node.closeable);
629
			this.portChangableContextKey.set(!!node.localPort);
630 631
		} else {
			this.tunnelTypeContext.set(TunnelType.Add);
632
			this.tunnelCloseableContext.set(false);
633
			this.portChangableContextKey.set(false);
634
		}
A
Alex Ross 已提交
635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658

		const actions: IAction[] = [];
		this._register(createAndFillInContextMenuActions(this.contributedContextMenu, { shouldForwardArgs: true }, actions, this.contextMenuService));

		this.contextMenuService.showContextMenu({
			getAnchor: () => treeEvent.anchor,
			getActions: () => actions,
			getActionViewItem: (action) => {
				const keybinding = this.keybindingService.lookupKeybinding(action.id);
				if (keybinding) {
					return new ActionViewItem(action, action, { label: true, keybinding: keybinding.getLabel() });
				}
				return undefined;
			},
			onHide: (wasCancelled?: boolean) => {
				if (wasCancelled) {
					this.tree!.domFocus();
				}
			},
			getActionsContext: () => node,
			actionRunner
		});
	}

659 660 661 662 663 664
	private onMouseDblClick(e: ITreeMouseEvent<ITunnelGroup | ITunnelItem | null>): void {
		if (!e.element) {
			this.commandService.executeCommand(ForwardPortAction.INLINE_ID);
		}
	}

A
Alex Ross 已提交
665
	protected layoutBody(height: number, width: number): void {
J
João Moreno 已提交
666
		super.layoutBody(height, width);
A
Alex Ross 已提交
667 668 669 670 671 672 673
		this.tree.layout(height, width);
	}
}

export class TunnelPanelDescriptor implements IViewDescriptor {
	readonly id = TunnelPanel.ID;
	readonly name = TunnelPanel.TITLE;
S
Sandeep Somavarapu 已提交
674
	readonly ctorDescriptor: SyncDescriptor<TunnelPanel>;
A
Alex Ross 已提交
675 676 677 678 679
	readonly canToggleVisibility = true;
	readonly hideByDefault = false;
	readonly workspace = true;
	readonly group = 'details@0';
	readonly remoteAuthority?: string | string[];
A
Alex Ross 已提交
680
	readonly canMoveView = true;
A
Alex Ross 已提交
681 682

	constructor(viewModel: ITunnelViewModel, environmentService: IWorkbenchEnvironmentService) {
S
Sandeep Somavarapu 已提交
683
		this.ctorDescriptor = new SyncDescriptor(TunnelPanel, [viewModel]);
684
		this.remoteAuthority = environmentService.remoteAuthority ? environmentService.remoteAuthority.split('+')[0] : undefined;
A
Alex Ross 已提交
685 686 687
	}
}

J
jeanp413 已提交
688 689 690 691 692 693 694 695 696 697 698
function validationMessage(validationString: string | null): { content: string, severity: Severity } | null {
	if (!validationString) {
		return null;
	}

	return {
		content: validationString,
		severity: Severity.Error
	};
}

699 700 701
namespace LabelTunnelAction {
	export const ID = 'remote.tunnel.label';
	export const LABEL = nls.localize('remote.tunnel.label', "Set Label");
A
Alex Ross 已提交
702 703 704

	export function handler(): ICommandHandler {
		return async (accessor, arg) => {
705 706
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
			if (context instanceof TunnelItem) {
A
Alex Ross 已提交
707
				const remoteExplorerService = accessor.get(IRemoteExplorerService);
708
				remoteExplorerService.setEditable(context, {
709
					onFinish: async (value, success) => {
710
						if (success) {
711
							remoteExplorerService.tunnelModel.name(context.remoteHost, context.remotePort, value);
712
						}
713
						remoteExplorerService.setEditable(context, null);
714 715
					},
					validationMessage: () => null,
716
					placeholder: nls.localize('remote.tunnelsView.labelPlaceholder', "Port label"),
717
					startingValue: context.name
718
				});
A
Alex Ross 已提交
719 720 721 722 723 724
			}
			return;
		};
	}
}

725 726 727
const invalidPortString: string = nls.localize('remote.tunnelsView.portNumberValid', "Forwarded port is invalid.");
const maxPortNumber: number = 65536;
const invalidPortNumberString: string = nls.localize('remote.tunnelsView.portNumberToHigh', "Port number must be \u2265 0 and < {0}.", maxPortNumber);
728

A
Alex Ross 已提交
729
export namespace ForwardPortAction {
730 731
	export const INLINE_ID = 'remote.tunnel.forwardInline';
	export const COMMANDPALETTE_ID = 'remote.tunnel.forwardCommandPalette';
732
	export const LABEL: ILocalizedString = { value: nls.localize('remote.tunnel.forward', "Forward a Port"), original: 'Forward a Port' };
733
	export const TREEITEM_LABEL = nls.localize('remote.tunnel.forwardItem', "Forward Port");
A
Alex Ross 已提交
734
	const forwardPrompt = nls.localize('remote.tunnel.forwardPrompt', "Port number or address (eg. 3000 or 10.10.10.10:2000).");
A
Alex Ross 已提交
735

736 737 738 739 740 741 742 743
	function parseInput(value: string): { host: string, port: number } | undefined {
		const matches = value.match(/^([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+\:|localhost:)?([0-9]+)$/);
		if (!matches) {
			return undefined;
		}
		return { host: matches[1]?.substring(0, matches[1].length - 1) || 'localhost', port: Number(matches[2]) };
	}

A
Alex Ross 已提交
744
	function validateInput(value: string): string | null {
745 746
		const parsed = parseInput(value);
		if (!parsed) {
747
			return invalidPortString;
748 749
		} else if (parsed.port >= maxPortNumber) {
			return invalidPortNumberString;
A
Alex Ross 已提交
750 751 752 753
		}
		return null;
	}

A
Alex Ross 已提交
754 755
	function error(notificationService: INotificationService, tunnel: RemoteTunnel | void, host: string, port: number) {
		if (!tunnel) {
756
			notificationService.warn(nls.localize('remote.tunnel.forwardError', "Unable to forward {0}:{1}. The host may not be available or that remote port may already be forwarded", host, port));
A
Alex Ross 已提交
757 758 759
		}
	}

760
	export function inlineHandler(): ICommandHandler {
A
Alex Ross 已提交
761 762
		return async (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
A
Alex Ross 已提交
763
			const notificationService = accessor.get(INotificationService);
A
Alex Ross 已提交
764
			if (arg instanceof TunnelItem) {
A
Alex Ross 已提交
765
				remoteExplorerService.forward({ host: arg.remoteHost, port: arg.remotePort }).then(tunnel => error(notificationService, tunnel, arg.remoteHost, arg.remotePort));
766 767
			} else {
				remoteExplorerService.setEditable(undefined, {
768
					onFinish: async (value, success) => {
769 770
						let parsed: { host: string, port: number } | undefined;
						if (success && (parsed = parseInput(value))) {
A
Alex Ross 已提交
771
							remoteExplorerService.forward({ host: parsed.host, port: parsed.port }).then(tunnel => error(notificationService, tunnel, parsed!.host, parsed!.port));
A
Alex Ross 已提交
772
						}
773
						remoteExplorerService.setEditable(undefined, null);
A
Alex Ross 已提交
774
					},
J
jeanp413 已提交
775
					validationMessage: (value) => validationMessage(validateInput(value)),
A
Alex Ross 已提交
776
					placeholder: forwardPrompt
A
Alex Ross 已提交
777
				});
778 779 780 781 782 783 784
			}
		};
	}

	export function commandPaletteHandler(): ICommandHandler {
		return async (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
A
Alex Ross 已提交
785
			const notificationService = accessor.get(INotificationService);
786 787 788 789 790 791 792 793 794
			const viewsService = accessor.get(IViewsService);
			const quickInputService = accessor.get(IQuickInputService);
			await viewsService.openView(TunnelPanel.ID, true);
			const value = await quickInputService.input({
				prompt: forwardPrompt,
				validateInput: (value) => Promise.resolve(validateInput(value))
			});
			let parsed: { host: string, port: number } | undefined;
			if (value && (parsed = parseInput(value))) {
A
Alex Ross 已提交
795
				remoteExplorerService.forward({ host: parsed.host, port: parsed.port }).then(tunnel => error(notificationService, tunnel, parsed!.host, parsed!.port));
A
Alex Ross 已提交
796 797 798 799 800
			}
		};
	}
}

A
Alex Ross 已提交
801
interface QuickPickTunnel extends IQuickPickItem {
A
Alex Ross 已提交
802
	tunnel?: ITunnelItem
A
Alex Ross 已提交
803 804
}

A
Alex Ross 已提交
805 806 807 808 809 810 811 812
function makeTunnelPicks(tunnels: Tunnel[]): QuickPickInput<QuickPickTunnel>[] {
	const picks: QuickPickInput<QuickPickTunnel>[] = tunnels.map(forwarded => {
		const item = TunnelItem.createFromTunnel(forwarded);
		return {
			label: item.label,
			description: item.description,
			tunnel: item
		};
A
Alex Ross 已提交
813 814 815
	});
	if (picks.length === 0) {
		picks.push({
816
			label: nls.localize('remote.tunnel.closeNoPorts', "No ports currently forwarded. Try running the {0} command", ForwardPortAction.LABEL.value)
A
Alex Ross 已提交
817 818 819 820 821
		});
	}
	return picks;
}

A
Alex Ross 已提交
822
namespace ClosePortAction {
A
Alex Ross 已提交
823 824
	export const INLINE_ID = 'remote.tunnel.closeInline';
	export const COMMANDPALETTE_ID = 'remote.tunnel.closeCommandPalette';
825
	export const LABEL: ILocalizedString = { value: nls.localize('remote.tunnel.close', "Stop Forwarding Port"), original: 'Stop Forwarding Port' };
A
Alex Ross 已提交
826

A
Alex Ross 已提交
827
	export function inlineHandler(): ICommandHandler {
A
Alex Ross 已提交
828
		return async (accessor, arg) => {
829
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
830
			if (context instanceof TunnelItem) {
A
Alex Ross 已提交
831
				const remoteExplorerService = accessor.get(IRemoteExplorerService);
832
				await remoteExplorerService.close({ host: context.remoteHost, port: context.remotePort });
A
Alex Ross 已提交
833 834 835
			}
		};
	}
A
Alex Ross 已提交
836 837 838 839 840 841 842

	export function commandPaletteHandler(): ICommandHandler {
		return async (accessor) => {
			const quickInputService = accessor.get(IQuickInputService);
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
			const commandService = accessor.get(ICommandService);

A
Alex Ross 已提交
843
			const picks: QuickPickInput<QuickPickTunnel>[] = makeTunnelPicks(Array.from(remoteExplorerService.tunnelModel.forwarded.values()).filter(tunnel => tunnel.closeable));
A
Alex Ross 已提交
844 845 846 847 848 849 850 851
			const result = await quickInputService.pick(picks, { placeHolder: nls.localize('remote.tunnel.closePlaceholder', "Choose a port to stop forwarding") });
			if (result && result.tunnel) {
				await remoteExplorerService.close({ host: result.tunnel.remoteHost, port: result.tunnel.remotePort });
			} else if (result) {
				await commandService.executeCommand(ForwardPortAction.COMMANDPALETTE_ID);
			}
		};
	}
A
Alex Ross 已提交
852 853
}

854
export namespace OpenPortInBrowserAction {
A
Alex Ross 已提交
855 856 857 858 859 860 861 862
	export const ID = 'remote.tunnel.open';
	export const LABEL = nls.localize('remote.tunnel.open', "Open in Browser");

	export function handler(): ICommandHandler {
		return async (accessor, arg) => {
			if (arg instanceof TunnelItem) {
				const model = accessor.get(IRemoteExplorerService).tunnelModel;
				const openerService = accessor.get(IOpenerService);
863
				const key = MakeAddress(arg.remoteHost, arg.remotePort);
864
				return run(model, openerService, key);
A
Alex Ross 已提交
865 866 867
			}
		};
	}
868 869 870 871 872 873 874 875 876

	export function run(model: TunnelModel, openerService: IOpenerService, key: string) {
		const tunnel = model.forwarded.get(key) || model.detected.get(key);
		let address: string | undefined;
		if (tunnel && tunnel.localAddress && (address = model.address(tunnel.remoteHost, tunnel.remotePort))) {
			return openerService.open(URI.parse('http://' + address));
		}
		return Promise.resolve();
	}
A
Alex Ross 已提交
877 878 879
}

namespace CopyAddressAction {
A
Alex Ross 已提交
880 881 882 883 884 885 886 887 888 889 890
	export const INLINE_ID = 'remote.tunnel.copyAddressInline';
	export const COMMANDPALETTE_ID = 'remote.tunnel.copyAddressCommandPalette';
	export const INLINE_LABEL = nls.localize('remote.tunnel.copyAddressInline', "Copy Address");
	export const COMMANDPALETTE_LABEL = nls.localize('remote.tunnel.copyAddressCommandPalette', "Copy Forwarded Port Address");

	async function copyAddress(remoteExplorerService: IRemoteExplorerService, clipboardService: IClipboardService, tunnelItem: ITunnelItem) {
		const address = remoteExplorerService.tunnelModel.address(tunnelItem.remoteHost, tunnelItem.remotePort);
		if (address) {
			await clipboardService.writeText(address.toString());
		}
	}
A
Alex Ross 已提交
891

A
Alex Ross 已提交
892
	export function inlineHandler(): ICommandHandler {
A
Alex Ross 已提交
893
		return async (accessor, arg) => {
894 895 896
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
			if (context instanceof TunnelItem) {
				return copyAddress(accessor.get(IRemoteExplorerService), accessor.get(IClipboardService), context);
A
Alex Ross 已提交
897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913
			}
		};
	}

	export function commandPaletteHandler(): ICommandHandler {
		return async (accessor, arg) => {
			const quickInputService = accessor.get(IQuickInputService);
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
			const commandService = accessor.get(ICommandService);
			const clipboardService = accessor.get(IClipboardService);

			const tunnels = Array.from(remoteExplorerService.tunnelModel.forwarded.values()).concat(Array.from(remoteExplorerService.tunnelModel.detected.values()));
			const result = await quickInputService.pick(makeTunnelPicks(tunnels), { placeHolder: nls.localize('remote.tunnel.copyAddressPlaceholdter', "Choose a forwarded port") });
			if (result && result.tunnel) {
				await copyAddress(remoteExplorerService, clipboardService, result.tunnel);
			} else if (result) {
				await commandService.executeCommand(ForwardPortAction.COMMANDPALETTE_ID);
A
Alex Ross 已提交
914 915 916 917 918
			}
		};
	}
}

919 920 921 922 923 924 925 926 927 928 929 930
namespace RefreshTunnelViewAction {
	export const ID = 'remote.tunnel.refresh';
	export const LABEL = nls.localize('remote.tunnel.refreshView', "Refresh");

	export function handler(): ICommandHandler {
		return (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
			return remoteExplorerService.refresh();
		};
	}
}

931 932 933 934 935 936 937
namespace ChangeLocalPortAction {
	export const ID = 'remote.tunnel.changeLocalPort';
	export const LABEL = nls.localize('remote.tunnel.changeLocalPort', "Change Local Port");

	function validateInput(value: string): string | null {
		if (!value.match(/^[0-9]+$/)) {
			return invalidPortString;
938 939
		} else if (Number(value) >= maxPortNumber) {
			return invalidPortNumberString;
940 941 942 943 944 945 946
		}
		return null;
	}

	export function handler(): ICommandHandler {
		return async (accessor, arg) => {
			const remoteExplorerService = accessor.get(IRemoteExplorerService);
947
			const notificationService = accessor.get(INotificationService);
948 949 950 951 952 953 954
			const context = (arg !== undefined || arg instanceof TunnelItem) ? arg : accessor.get(IContextKeyService).getContextKeyValue(TunnelViewSelectionKeyName);
			if (context instanceof TunnelItem) {
				remoteExplorerService.setEditable(context, {
					onFinish: async (value, success) => {
						remoteExplorerService.setEditable(context, null);
						if (success) {
							await remoteExplorerService.close({ host: context.remoteHost, port: context.remotePort });
955
							const numberValue = Number(value);
956
							const newForward = await remoteExplorerService.forward({ host: context.remoteHost, port: context.remotePort }, numberValue, context.name);
957 958 959
							if (newForward && newForward.tunnelLocalPort !== numberValue) {
								notificationService.warn(nls.localize('remote.tunnel.changeLocalPortNumber', "The local port {0} is not available. Port number {1} has been used instead", value, newForward.tunnelLocalPort));
							}
960 961
						}
					},
J
jeanp413 已提交
962
					validationMessage: (value) => validationMessage(validateInput(value)),
963 964 965 966 967 968 969
					placeholder: nls.localize('remote.tunnelsView.changePort', "New local port")
				});
			}
		};
	}
}

970 971
const tunnelViewCommandsWeightBonus = 10; // give our commands a little bit more weight over other default list/tree commands

972 973
KeybindingsRegistry.registerCommandAndKeybindingRule({
	id: LabelTunnelAction.ID,
974
	weight: KeybindingWeight.WorkbenchContrib + tunnelViewCommandsWeightBonus,
975 976 977 978 979 980 981
	when: ContextKeyExpr.and(TunnelViewFocusContextKey, TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded)),
	primary: KeyCode.F2,
	mac: {
		primary: KeyCode.Enter
	},
	handler: LabelTunnelAction.handler()
});
982 983
CommandsRegistry.registerCommand(ForwardPortAction.INLINE_ID, ForwardPortAction.inlineHandler());
CommandsRegistry.registerCommand(ForwardPortAction.COMMANDPALETTE_ID, ForwardPortAction.commandPaletteHandler());
984 985
KeybindingsRegistry.registerCommandAndKeybindingRule({
	id: ClosePortAction.INLINE_ID,
986
	weight: KeybindingWeight.WorkbenchContrib + tunnelViewCommandsWeightBonus,
987 988 989 990 991 992 993 994
	when: ContextKeyExpr.and(TunnelCloseableContextKey, TunnelViewFocusContextKey),
	primary: KeyCode.Delete,
	mac: {
		primary: KeyMod.CtrlCmd | KeyCode.Backspace
	},
	handler: ClosePortAction.inlineHandler()
});

A
Alex Ross 已提交
995
CommandsRegistry.registerCommand(ClosePortAction.COMMANDPALETTE_ID, ClosePortAction.commandPaletteHandler());
A
Alex Ross 已提交
996
CommandsRegistry.registerCommand(OpenPortInBrowserAction.ID, OpenPortInBrowserAction.handler());
997 998
KeybindingsRegistry.registerCommandAndKeybindingRule({
	id: CopyAddressAction.INLINE_ID,
999
	weight: KeybindingWeight.WorkbenchContrib + tunnelViewCommandsWeightBonus,
1000 1001 1002 1003
	when: ContextKeyExpr.or(ContextKeyExpr.and(TunnelViewFocusContextKey, TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded)), ContextKeyExpr.and(TunnelViewFocusContextKey, TunnelTypeContextKey.isEqualTo(TunnelType.Detected))),
	primary: KeyMod.CtrlCmd | KeyCode.KEY_C,
	handler: CopyAddressAction.inlineHandler()
});
A
Alex Ross 已提交
1004
CommandsRegistry.registerCommand(CopyAddressAction.COMMANDPALETTE_ID, CopyAddressAction.commandPaletteHandler());
1005
CommandsRegistry.registerCommand(RefreshTunnelViewAction.ID, RefreshTunnelViewAction.handler());
1006
CommandsRegistry.registerCommand(ChangeLocalPortAction.ID, ChangeLocalPortAction.handler());
A
Alex Ross 已提交
1007

A
Alex Ross 已提交
1008 1009 1010 1011 1012 1013 1014
MenuRegistry.appendMenuItem(MenuId.CommandPalette, ({
	command: {
		id: ClosePortAction.COMMANDPALETTE_ID,
		title: ClosePortAction.LABEL
	},
	when: forwardedPortsViewEnabled
}));
1015 1016
MenuRegistry.appendMenuItem(MenuId.CommandPalette, ({
	command: {
1017
		id: ForwardPortAction.COMMANDPALETTE_ID,
1018 1019 1020 1021
		title: ForwardPortAction.LABEL
	},
	when: forwardedPortsViewEnabled
}));
A
Alex Ross 已提交
1022 1023 1024 1025 1026 1027 1028
MenuRegistry.appendMenuItem(MenuId.CommandPalette, ({
	command: {
		id: CopyAddressAction.COMMANDPALETTE_ID,
		title: CopyAddressAction.COMMANDPALETTE_LABEL
	},
	when: forwardedPortsViewEnabled
}));
A
Alex Ross 已提交
1029 1030 1031 1032
MenuRegistry.appendMenuItem(MenuId.TunnelTitle, ({
	group: 'navigation',
	order: 0,
	command: {
A
Alex Ross 已提交
1033
		id: ForwardPortAction.INLINE_ID,
A
Alex Ross 已提交
1034
		title: ForwardPortAction.LABEL,
1035
		icon: { id: 'codicon/plus' }
A
Alex Ross 已提交
1036 1037
	}
}));
1038 1039 1040 1041 1042 1043 1044 1045 1046
MenuRegistry.appendMenuItem(MenuId.TunnelTitle, ({
	group: 'navigation',
	order: 1,
	command: {
		id: RefreshTunnelViewAction.ID,
		title: RefreshTunnelViewAction.LABEL,
		icon: { id: 'codicon/refresh' }
	}
}));
A
Alex Ross 已提交
1047 1048 1049 1050
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 0,
	command: {
A
Alex Ross 已提交
1051 1052
		id: CopyAddressAction.INLINE_ID,
		title: CopyAddressAction.INLINE_LABEL,
A
Alex Ross 已提交
1053
	},
A
Alex Ross 已提交
1054
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), TunnelTypeContextKey.isEqualTo(TunnelType.Detected))
A
Alex Ross 已提交
1055 1056 1057 1058 1059 1060 1061 1062
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 1,
	command: {
		id: OpenPortInBrowserAction.ID,
		title: OpenPortInBrowserAction.LABEL,
	},
A
Alex Ross 已提交
1063
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), TunnelTypeContextKey.isEqualTo(TunnelType.Detected))
A
Alex Ross 已提交
1064 1065 1066 1067 1068
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 2,
	command: {
1069 1070
		id: LabelTunnelAction.ID,
		title: LabelTunnelAction.LABEL,
A
Alex Ross 已提交
1071 1072 1073
	},
	when: TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded)
}));
1074 1075 1076 1077 1078 1079 1080
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '1_manage',
	order: 0,
	command: {
		id: ChangeLocalPortAction.ID,
		title: ChangeLocalPortAction.LABEL,
	},
1081
	when: ContextKeyExpr.and(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), PortChangableContextKey)
1082
}));
A
Alex Ross 已提交
1083 1084 1085 1086
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
	group: '0_manage',
	order: 1,
	command: {
1087
		id: ForwardPortAction.INLINE_ID,
1088
		title: ForwardPortAction.TREEITEM_LABEL,
A
Alex Ross 已提交
1089
	},
1090
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Candidate), TunnelTypeContextKey.isEqualTo(TunnelType.Add))
A
Alex Ross 已提交
1091 1092
}));
MenuRegistry.appendMenuItem(MenuId.TunnelContext, ({
1093 1094
	group: '1_manage',
	order: 1,
A
Alex Ross 已提交
1095
	command: {
A
Alex Ross 已提交
1096
		id: ClosePortAction.INLINE_ID,
A
Alex Ross 已提交
1097 1098 1099 1100 1101 1102 1103 1104 1105 1106
		title: ClosePortAction.LABEL,
	},
	when: TunnelCloseableContextKey
}));

MenuRegistry.appendMenuItem(MenuId.TunnelInline, ({
	order: 0,
	command: {
		id: OpenPortInBrowserAction.ID,
		title: OpenPortInBrowserAction.LABEL,
1107
		icon: { id: 'codicon/globe' }
A
Alex Ross 已提交
1108
	},
A
Alex Ross 已提交
1109
	when: ContextKeyExpr.or(TunnelTypeContextKey.isEqualTo(TunnelType.Forwarded), TunnelTypeContextKey.isEqualTo(TunnelType.Detected))
A
Alex Ross 已提交
1110 1111 1112 1113
}));
MenuRegistry.appendMenuItem(MenuId.TunnelInline, ({
	order: 0,
	command: {
1114
		id: ForwardPortAction.INLINE_ID,
1115
		title: ForwardPortAction.TREEITEM_LABEL,
1116
		icon: { id: 'codicon/plus' }
A
Alex Ross 已提交
1117
	},
A
Alex Ross 已提交
1118
	when: TunnelTypeContextKey.isEqualTo(TunnelType.Candidate)
A
Alex Ross 已提交
1119 1120 1121 1122
}));
MenuRegistry.appendMenuItem(MenuId.TunnelInline, ({
	order: 2,
	command: {
A
Alex Ross 已提交
1123
		id: ClosePortAction.INLINE_ID,
A
Alex Ross 已提交
1124
		title: ClosePortAction.LABEL,
1125
		icon: { id: 'codicon/x' }
A
Alex Ross 已提交
1126 1127 1128
	},
	when: TunnelCloseableContextKey
}));