editorCommands.ts 22.1 KB
Newer Older
S
Sandeep Somavarapu 已提交
1 2 3 4 5 6 7
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

import * as nls from 'vs/nls';
import * as types from 'vs/base/common/types';
J
Johannes Rieken 已提交
8 9 10
import { ServicesAccessor } from 'vs/platform/instantiation/common/instantiation';
import { KeybindingsRegistry } from 'vs/platform/keybinding/common/keybindingsRegistry';
import { IEditorGroupService } from 'vs/workbench/services/group/common/groupService';
11
import { ActiveEditorMoveArguments, ActiveEditorMovePositioning, ActiveEditorMovePositioningBy, EditorCommands, TextCompareEditorVisibleContext, EditorInput, IEditorIdentifier, IEditorCommandsContext, ActiveEditorGroupEmptyContext, MultipleEditorGroupsContext } from 'vs/workbench/common/editor';
J
Johannes Rieken 已提交
12
import { IWorkbenchEditorService } from 'vs/workbench/services/editor/common/editorService';
13
import { IEditor, Position, POSITIONS, Direction, IEditorInput } from 'vs/platform/editor/common/editor';
14
import { EditorContextKeys } from 'vs/editor/common/editorContextKeys';
15
import { TextDiffEditor } from 'vs/workbench/browser/parts/editor/textDiffEditor';
16
import { EditorStacksModel, EditorGroup } from 'vs/workbench/common/editor/editorStacksModel';
I
isidor 已提交
17
import { KeyMod, KeyCode, KeyChord } from 'vs/base/common/keyCodes';
18
import { TPromise } from 'vs/base/common/winjs.base';
19
import URI from 'vs/base/common/uri';
B
Benjamin Pasero 已提交
20
import { IQuickOpenService } from 'vs/platform/quickOpen/common/quickOpen';
21
import { IDiffEditorOptions } from 'vs/editor/common/config/editorOptions';
I
isidor 已提交
22 23
import { IListService } from 'vs/platform/list/browser/listService';
import { List } from 'vs/base/browser/ui/list/listWidget';
I
isidor 已提交
24
import { distinct } from 'vs/base/common/arrays';
B
Benjamin Pasero 已提交
25
import { INextEditorGroupsService } from 'vs/workbench/services/group/common/nextEditorGroupsService';
26
import { ContextKeyExpr } from 'vs/platform/contextkey/common/contextkey';
27

28
export const CLOSE_SAVED_EDITORS_COMMAND_ID = 'workbench.action.closeUnmodifiedEditors';
I
isidor 已提交
29
export const CLOSE_EDITORS_IN_GROUP_COMMAND_ID = 'workbench.action.closeEditorsInGroup';
30
export const CLOSE_EDITORS_TO_THE_RIGHT_COMMAND_ID = 'workbench.action.closeEditorsToTheRight';
I
isidor 已提交
31
export const CLOSE_EDITOR_COMMAND_ID = 'workbench.action.closeActiveEditor';
32
export const CLOSE_EDITOR_GROUP_COMMAND_ID = 'workbench.action.closeActiveEditorGroup';
I
isidor 已提交
33
export const CLOSE_OTHER_EDITORS_IN_GROUP_COMMAND_ID = 'workbench.action.closeOtherEditors';
34
export const KEEP_EDITOR_COMMAND_ID = 'workbench.action.keepEditor';
B
Benjamin Pasero 已提交
35
export const SHOW_EDITORS_IN_GROUP = 'workbench.action.showEditorsInGroup';
36
export const TOGGLE_DIFF_INLINE_MODE = 'toggle.diff.editorMode';
B
Benjamin Pasero 已提交
37 38 39 40 41

export const NAVIGATE_IN_GROUP_ONE_PREFIX = 'edt one ';
export const NAVIGATE_IN_GROUP_TWO_PREFIX = 'edt two ';
export const NAVIGATE_IN_GROUP_THREE_PREFIX = 'edt three ';
export const NAVIGATE_ALL_EDITORS_GROUP_PREFIX = 'edt ';
42

43
export function setup(): void {
44 45
	registerActiveEditorMoveCommand();
	registerDiffEditorCommands();
46
	registerOpenEditorAtIndexCommands();
47
	registerEditorCommands();
S
Sandeep Somavarapu 已提交
48 49
}

B
Benjamin Pasero 已提交
50
const isActiveEditorMoveArg = function (arg: ActiveEditorMoveArguments): boolean {
S
Sandeep Somavarapu 已提交
51 52 53 54
	if (!types.isObject(arg)) {
		return false;
	}

55
	const activeEditorMoveArg: ActiveEditorMoveArguments = arg;
S
Sandeep Somavarapu 已提交
56 57 58 59 60 61 62 63 64

	if (!types.isString(activeEditorMoveArg.to)) {
		return false;
	}

	if (!types.isUndefined(activeEditorMoveArg.by) && !types.isString(activeEditorMoveArg.by)) {
		return false;
	}

65
	if (!types.isUndefined(activeEditorMoveArg.value) && !types.isNumber(activeEditorMoveArg.value)) {
S
Sandeep Somavarapu 已提交
66 67 68 69 70 71
		return false;
	}

	return true;
};

72
function registerActiveEditorMoveCommand(): void {
A
Alex Dima 已提交
73
	KeybindingsRegistry.registerCommandAndKeybindingRule({
74
		id: EditorCommands.MoveActiveEditor,
S
Sandeep Somavarapu 已提交
75
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
76
		when: EditorContextKeys.editorTextFocus,
S
Sandeep Somavarapu 已提交
77
		primary: null,
78
		handler: (accessor, args: any) => moveActiveEditor(args, accessor),
S
Sandeep Somavarapu 已提交
79
		description: {
80
			description: nls.localize('editorCommand.activeEditorMove.description', "Move the active editor by tabs or groups"),
S
Sandeep Somavarapu 已提交
81 82 83
			args: [
				{
					name: nls.localize('editorCommand.activeEditorMove.arg.name', "Active editor move argument"),
J
Joao Moreno 已提交
84
					description: nls.localize('editorCommand.activeEditorMove.arg.description', "Argument Properties:\n\t* 'to': String value providing where to move.\n\t* 'by': String value providing the unit for move. By tab or by group.\n\t* 'value': Number value providing how many positions or an absolute position to move."),
S
Sandeep Somavarapu 已提交
85 86 87 88 89 90 91
					constraint: isActiveEditorMoveArg
				}
			]
		}
	});
}

92
function moveActiveEditor(args: ActiveEditorMoveArguments = {}, accessor: ServicesAccessor): void {
S
Sandeep Somavarapu 已提交
93
	args.to = args.to || ActiveEditorMovePositioning.RIGHT;
94
	args.by = args.by || ActiveEditorMovePositioningBy.TAB;
95
	args.value = types.isUndefined(args.value) ? 1 : args.value;
S
Sandeep Somavarapu 已提交
96

97
	const activeEditor = accessor.get(IWorkbenchEditorService).getActiveEditor();
98 99 100 101 102 103 104
	if (activeEditor) {
		switch (args.by) {
			case ActiveEditorMovePositioningBy.TAB:
				return moveActiveTab(args, activeEditor, accessor);
			case ActiveEditorMovePositioningBy.GROUP:
				return moveActiveEditorToGroup(args, activeEditor, accessor);
		}
S
Sandeep Somavarapu 已提交
105 106 107
	}
}

108
function moveActiveTab(args: ActiveEditorMoveArguments, activeEditor: IEditor, accessor: ServicesAccessor): void {
109
	const editorGroupsService: IEditorGroupService = accessor.get(IEditorGroupService);
S
Sandeep Somavarapu 已提交
110
	const editorGroup = editorGroupsService.getStacksModel().groupAt(activeEditor.position);
111
	let index = editorGroup.indexOf(activeEditor.input);
S
Sandeep Somavarapu 已提交
112 113 114 115 116 117 118 119
	switch (args.to) {
		case ActiveEditorMovePositioning.FIRST:
			index = 0;
			break;
		case ActiveEditorMovePositioning.LAST:
			index = editorGroup.count - 1;
			break;
		case ActiveEditorMovePositioning.LEFT:
120
			index = index - args.value;
S
Sandeep Somavarapu 已提交
121 122
			break;
		case ActiveEditorMovePositioning.RIGHT:
123
			index = index + args.value;
S
Sandeep Somavarapu 已提交
124 125
			break;
		case ActiveEditorMovePositioning.CENTER:
126
			index = Math.round(editorGroup.count / 2) - 1;
S
Sandeep Somavarapu 已提交
127 128
			break;
		case ActiveEditorMovePositioning.POSITION:
129
			index = args.value - 1;
S
Sandeep Somavarapu 已提交
130 131
			break;
	}
132

S
Sandeep Somavarapu 已提交
133
	index = index < 0 ? 0 : index >= editorGroup.count ? editorGroup.count - 1 : index;
134
	editorGroupsService.moveEditor(activeEditor.input, editorGroup, editorGroup, { index });
S
Sandeep Somavarapu 已提交
135 136
}

137
function moveActiveEditorToGroup(args: ActiveEditorMoveArguments, activeEditor: IEditor, accessor: ServicesAccessor): void {
138
	let newPosition = activeEditor.position;
S
Sandeep Somavarapu 已提交
139 140
	switch (args.to) {
		case ActiveEditorMovePositioning.LEFT:
S
Sandeep Somavarapu 已提交
141 142 143 144 145 146
			newPosition = newPosition - 1;
			break;
		case ActiveEditorMovePositioning.RIGHT:
			newPosition = newPosition + 1;
			break;
		case ActiveEditorMovePositioning.FIRST:
147
			newPosition = Position.ONE;
S
Sandeep Somavarapu 已提交
148 149
			break;
		case ActiveEditorMovePositioning.LAST:
150
			newPosition = Position.THREE;
S
Sandeep Somavarapu 已提交
151 152
			break;
		case ActiveEditorMovePositioning.CENTER:
153
			newPosition = Position.TWO;
S
Sandeep Somavarapu 已提交
154 155
			break;
		case ActiveEditorMovePositioning.POSITION:
156
			newPosition = args.value - 1;
S
Sandeep Somavarapu 已提交
157 158
			break;
	}
159

S
Sandeep Somavarapu 已提交
160 161
	newPosition = POSITIONS.indexOf(newPosition) !== -1 ? newPosition : activeEditor.position;
	accessor.get(IEditorGroupService).moveEditor(activeEditor.input, activeEditor.position, newPosition);
162 163 164 165 166 167
}

function registerDiffEditorCommands(): void {
	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: 'workbench.action.compareEditor.nextChange',
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
168
		when: TextCompareEditorVisibleContext,
169 170 171 172 173 174 175
		primary: null,
		handler: accessor => navigateInDiffEditor(accessor, true)
	});

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: 'workbench.action.compareEditor.previousChange',
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
176
		when: TextCompareEditorVisibleContext,
177 178 179 180 181 182 183 184 185 186 187 188 189 190
		primary: null,
		handler: accessor => navigateInDiffEditor(accessor, false)
	});

	function navigateInDiffEditor(accessor: ServicesAccessor, next: boolean): void {
		let editorService = accessor.get(IWorkbenchEditorService);
		const candidates = [editorService.getActiveEditor(), ...editorService.getVisibleEditors()].filter(e => e instanceof TextDiffEditor);

		if (candidates.length > 0) {
			next ? (<TextDiffEditor>candidates[0]).getDiffNavigator().next() : (<TextDiffEditor>candidates[0]).getDiffNavigator().previous();
		}
	}

	KeybindingsRegistry.registerCommandAndKeybindingRule({
191 192
		id: TOGGLE_DIFF_INLINE_MODE,
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
193
		when: void 0,
194
		primary: void 0,
B
Benjamin Pasero 已提交
195
		handler: (accessor, resource, context: IEditorCommandsContext) => {
196
			const editorService = accessor.get(IWorkbenchEditorService);
B
Benjamin Pasero 已提交
197 198 199 200 201 202 203 204 205
			const editorGroupService = accessor.get(IEditorGroupService);

			let editor: IEditor;
			if (context) {
				const position = positionAndInput(editorGroupService, editorService, context).position;
				editor = editorService.getVisibleEditors()[position];
			} else {
				editor = editorService.getActiveEditor();
			}
206

207 208 209 210 211 212 213 214
			if (editor instanceof TextDiffEditor) {
				const control = editor.getControl();
				const isInlineMode = !control.renderSideBySide;
				control.updateOptions(<IDiffEditorOptions>{
					renderSideBySide: isInlineMode
				});
			}
		}
215
	});
216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
}

function registerOpenEditorAtIndexCommands(): void {

	// Keybindings to focus a specific index in the tab folder if tabs are enabled
	for (let i = 0; i < 9; i++) {
		const editorIndex = i;
		const visibleIndex = i + 1;

		KeybindingsRegistry.registerCommandAndKeybindingRule({
			id: 'workbench.action.openEditorAtIndex' + visibleIndex,
			weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
			when: void 0,
			primary: KeyMod.Alt | toKeyCode(visibleIndex),
			mac: { primary: KeyMod.WinCtrl | toKeyCode(visibleIndex) },
			handler: accessor => {
				const editorService = accessor.get(IWorkbenchEditorService);
				const editorGroupService = accessor.get(IEditorGroupService);

				const active = editorService.getActiveEditor();
				if (active) {
					const group = editorGroupService.getStacksModel().groupAt(active.position);
					const editor = group.getEditor(editorIndex);

					if (editor) {
B
Benjamin Pasero 已提交
241
						return editorService.openEditor(editor).then(() => void 0);
242 243
					}
				}
244 245

				return void 0;
246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
			}
		});
	}

	function toKeyCode(index: number): KeyCode {
		switch (index) {
			case 0: return KeyCode.KEY_0;
			case 1: return KeyCode.KEY_1;
			case 2: return KeyCode.KEY_2;
			case 3: return KeyCode.KEY_3;
			case 4: return KeyCode.KEY_4;
			case 5: return KeyCode.KEY_5;
			case 6: return KeyCode.KEY_6;
			case 7: return KeyCode.KEY_7;
			case 8: return KeyCode.KEY_8;
			case 9: return KeyCode.KEY_9;
		}
263 264

		return void 0;
265
	}
266 267
}

268
function registerEditorCommands() {
269

I
isidor 已提交
270
	KeybindingsRegistry.registerCommandAndKeybindingRule({
271
		id: CLOSE_SAVED_EDITORS_COMMAND_ID,
I
isidor 已提交
272
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
273
		when: void 0,
I
isidor 已提交
274
		primary: KeyChord(KeyMod.CtrlCmd | KeyCode.KEY_K, KeyCode.KEY_U),
I
isidor 已提交
275
		handler: (accessor, resource: URI | object, context: IEditorCommandsContext) => {
276
			const editorGroupService = accessor.get(IEditorGroupService);
I
isidor 已提交
277
			const model = editorGroupService.getStacksModel();
278
			const editorService = accessor.get(IWorkbenchEditorService);
279
			const contexts = getMultiSelectedEditorContexts(context, accessor.get(IListService));
I
isidor 已提交
280 281 282 283
			if (contexts.length === 0 && model.activeGroup) {
				// If command is triggered from the command palette use the active group
				contexts.push({ groupId: model.activeGroup.id });
			}
284

285 286 287
			let positionOne: { savedOnly: boolean } = void 0;
			let positionTwo: { savedOnly: boolean } = void 0;
			let positionThree: { savedOnly: boolean } = void 0;
I
isidor 已提交
288
			contexts.forEach(c => {
289
				switch (model.positionOfGroup(model.getGroup(c.groupId))) {
290 291 292
					case Position.ONE: positionOne = { savedOnly: true }; break;
					case Position.TWO: positionTwo = { savedOnly: true }; break;
					case Position.THREE: positionThree = { savedOnly: true }; break;
I
isidor 已提交
293 294 295 296
				}
			});

			return editorService.closeEditors({ positionOne, positionTwo, positionThree });
297 298 299
		}
	});

I
isidor 已提交
300
	KeybindingsRegistry.registerCommandAndKeybindingRule({
301
		id: CLOSE_EDITORS_IN_GROUP_COMMAND_ID,
I
isidor 已提交
302
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
303
		when: void 0,
I
isidor 已提交
304
		primary: KeyChord(KeyMod.CtrlCmd | KeyCode.KEY_K, KeyCode.KEY_W),
I
isidor 已提交
305
		handler: (accessor, resource: URI | object, context: IEditorCommandsContext) => {
306 307
			const editorGroupService = accessor.get(IEditorGroupService);
			const editorService = accessor.get(IWorkbenchEditorService);
308 309 310
			const contexts = getMultiSelectedEditorContexts(context, accessor.get(IListService));
			const distinctGroupIds = distinct(contexts.map(c => c.groupId));
			const model = editorGroupService.getStacksModel();
311

312 313
			if (distinctGroupIds.length) {
				return editorService.closeEditors(distinctGroupIds.map(gid => model.positionOfGroup(model.getGroup(gid))));
314 315 316 317
			}
			const activeEditor = editorService.getActiveEditor();
			if (activeEditor) {
				return editorService.closeEditors(activeEditor.position);
318 319 320 321 322 323
			}

			return TPromise.as(false);
		}
	});

I
isidor 已提交
324
	KeybindingsRegistry.registerCommandAndKeybindingRule({
I
isidor 已提交
325
		id: CLOSE_EDITOR_COMMAND_ID,
I
isidor 已提交
326
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
327
		when: void 0,
I
isidor 已提交
328 329
		primary: KeyMod.CtrlCmd | KeyCode.KEY_W,
		win: { primary: KeyMod.CtrlCmd | KeyCode.F4, secondary: [KeyMod.CtrlCmd | KeyCode.KEY_W] },
I
isidor 已提交
330
		handler: (accessor, resource: URI | object, context: IEditorCommandsContext) => {
I
isidor 已提交
331
			const editorGroupService = accessor.get(IEditorGroupService);
332
			const editorService = accessor.get(IWorkbenchEditorService);
333
			const nextEditorGroupService = accessor.get(INextEditorGroupsService);
I
isidor 已提交
334

335 336 337
			const contexts = getMultiSelectedEditorContexts(context, accessor.get(IListService));
			const groupIds = distinct(contexts.map(context => context.groupId));
			const model = editorGroupService.getStacksModel();
338 339

			const editorsToClose = new Map<Position, IEditorInput[]>();
I
isidor 已提交
340

341 342 343
			groupIds.forEach(groupId => {
				const group = model.getGroup(groupId);
				const position = model.positionOfGroup(group);
344
				if (position >= 0) {
I
isidor 已提交
345 346 347
					const inputs = contexts.map(c => {
						if (c && groupId === c.groupId && types.isNumber(c.editorIndex)) {
							return group.getEditor(c.editorIndex);
348
						}
I
isidor 已提交
349

I
isidor 已提交
350
						return group.activeEditor;
I
isidor 已提交
351 352 353 354 355
					}).filter(input => !!input);

					if (inputs.length) {
						editorsToClose.set(position, inputs);
					}
356
				}
357
			});
I
isidor 已提交
358

359
			if (editorsToClose.size === 0) {
360
				return nextEditorGroupService.activeGroup.closeEditor();
361 362
			}

363 364 365 366 367
			return editorService.closeEditors({
				positionOne: editorsToClose.get(Position.ONE),
				positionTwo: editorsToClose.get(Position.TWO),
				positionThree: editorsToClose.get(Position.THREE)
			});
368
		}
369 370 371 372 373 374 375 376 377 378 379 380 381 382
	});

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: CLOSE_EDITOR_GROUP_COMMAND_ID,
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
		when: ContextKeyExpr.and(ActiveEditorGroupEmptyContext, MultipleEditorGroupsContext),
		primary: KeyMod.CtrlCmd | KeyCode.KEY_W,
		win: { primary: KeyMod.CtrlCmd | KeyCode.F4, secondary: [KeyMod.CtrlCmd | KeyCode.KEY_W] },
		handler: (accessor, resource: URI | object, context: IEditorCommandsContext) => {
			const nextEditorGroupService = accessor.get(INextEditorGroupsService);

			// TODO@grid handle more cases from the related CLOSE_EDITOR_COMMAND_ID and also revisit command ID
			nextEditorGroupService.removeGroup(nextEditorGroupService.activeGroup);
		}
383 384
	});

I
isidor 已提交
385
	KeybindingsRegistry.registerCommandAndKeybindingRule({
I
isidor 已提交
386
		id: CLOSE_OTHER_EDITORS_IN_GROUP_COMMAND_ID,
I
isidor 已提交
387
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
388 389
		when: void 0,
		primary: void 0,
I
isidor 已提交
390
		mac: { primary: KeyMod.CtrlCmd | KeyMod.Alt | KeyCode.KEY_T },
I
isidor 已提交
391
		handler: (accessor, resource: URI | object, context: IEditorCommandsContext) => {
I
isidor 已提交
392
			const editorGroupService = accessor.get(IEditorGroupService);
393
			const editorService = accessor.get(IWorkbenchEditorService);
394 395
			const contexts = getMultiSelectedEditorContexts(context, accessor.get(IListService));
			const model = editorGroupService.getStacksModel();
396

397 398 399 400 401 402 403 404 405 406 407
			if (contexts.length === 0) {
				// Cover the case when run from command palette
				const activeGroup = model.activeGroup;
				const activeEditor = editorService.getActiveEditorInput();
				if (activeGroup && activeEditor) {
					contexts.push({ groupId: activeGroup.id, editorIndex: activeGroup.indexOf(activeEditor) });
				}
			}

			const groupIds = distinct(contexts.map(context => context.groupId));
			const editorsToClose = new Map<Position, IEditorInput[]>();
408 409
			groupIds.forEach(groupId => {
				const group = model.getGroup(groupId);
410
				const inputsToSkip = contexts.map(c => {
I
isidor 已提交
411
					if (c.groupId === groupId && types.isNumber(c.editorIndex)) {
412
						return group.getEditor(c.editorIndex);
413
					}
414

B
Benjamin Pasero 已提交
415
					return void 0;
416
				}).filter(input => !!input);
417

418
				const toClose = group.getEditors().filter(input => inputsToSkip.indexOf(input) === -1);
419
				editorsToClose.set(model.positionOfGroup(group), toClose);
420 421 422 423 424 425 426
			});

			return editorService.closeEditors({
				positionOne: editorsToClose.get(Position.ONE),
				positionTwo: editorsToClose.get(Position.TWO),
				positionThree: editorsToClose.get(Position.THREE)
			});
427 428 429 430 431 432 433 434
		}
	});

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: CLOSE_EDITORS_TO_THE_RIGHT_COMMAND_ID,
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
		when: void 0,
		primary: void 0,
435
		handler: (accessor, resource: URI, context: IEditorCommandsContext) => {
436 437
			const editorGroupService = accessor.get(IEditorGroupService);
			const editorService = accessor.get(IWorkbenchEditorService);
I
isidor 已提交
438

B
Benjamin Pasero 已提交
439
			const { position, input } = positionAndInput(editorGroupService, editorService, context);
440 441 442

			if (typeof position === 'number' && input) {
				return editorService.closeEditors(position, { except: input, direction: Direction.RIGHT });
443 444
			}

445 446 447 448 449 450 451 452 453
			return TPromise.as(false);
		}
	});

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: KEEP_EDITOR_COMMAND_ID,
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
		when: void 0,
		primary: KeyChord(KeyMod.CtrlCmd | KeyCode.KEY_K, KeyCode.Enter),
454
		handler: (accessor, resource: URI, context: IEditorCommandsContext) => {
455 456 457
			const editorGroupService = accessor.get(IEditorGroupService);
			const editorService = accessor.get(IWorkbenchEditorService);

B
Benjamin Pasero 已提交
458
			const { position, input } = positionAndInput(editorGroupService, editorService, context);
459

I
isidor 已提交
460
			if (typeof position === 'number' && input) {
461
				return editorGroupService.pinEditor(position, input);
462 463
			}

I
isidor 已提交
464
			return TPromise.as(false);
465 466
		}
	});
B
Benjamin Pasero 已提交
467 468 469 470 471 472

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: SHOW_EDITORS_IN_GROUP,
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(),
		when: void 0,
		primary: void 0,
B
Benjamin Pasero 已提交
473
		handler: (accessor, resource: URI, context: IEditorCommandsContext) => {
B
Benjamin Pasero 已提交
474 475 476 477 478 479
			const editorGroupService = accessor.get(IEditorGroupService);
			const editorService = accessor.get(IWorkbenchEditorService);
			const quickOpenService = accessor.get(IQuickOpenService);

			const stacks = editorGroupService.getStacksModel();
			const groupCount = stacks.groups.length;
480
			if (groupCount <= 1) {
B
Benjamin Pasero 已提交
481 482 483
				return quickOpenService.show(NAVIGATE_ALL_EDITORS_GROUP_PREFIX);
			}

B
Benjamin Pasero 已提交
484
			const { position } = positionAndInput(editorGroupService, editorService, context);
B
Benjamin Pasero 已提交
485 486 487 488 489 490 491 492 493 494 495

			switch (position) {
				case Position.TWO:
					return quickOpenService.show(NAVIGATE_IN_GROUP_TWO_PREFIX);
				case Position.THREE:
					return quickOpenService.show(NAVIGATE_IN_GROUP_THREE_PREFIX);
			}

			return quickOpenService.show(NAVIGATE_IN_GROUP_ONE_PREFIX);
		}
	});
496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: '_workbench.printStacksModel',
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(0),
		handler(accessor: ServicesAccessor) {
			console.log(`${accessor.get(IEditorGroupService).getStacksModel().toString()}\n\n`);
		},
		when: void 0,
		primary: void 0
	});

	KeybindingsRegistry.registerCommandAndKeybindingRule({
		id: '_workbench.validateStacksModel',
		weight: KeybindingsRegistry.WEIGHT.workbenchContrib(0),
		handler(accessor: ServicesAccessor) {
			(<EditorStacksModel>accessor.get(IEditorGroupService).getStacksModel()).validate();
		},
		when: void 0,
		primary: void 0
	});
516
}
517

518
function positionAndInput(editorGroupService: IEditorGroupService, editorService: IWorkbenchEditorService, context?: IEditorCommandsContext): { position: Position, input: IEditorInput } {
B
Benjamin Pasero 已提交
519 520

	// Resolve from context
521 522 523
	const model = editorGroupService.getStacksModel();
	const group = context ? model.getGroup(context.groupId) : undefined;
	let position = group ? model.positionOfGroup(group) : undefined;
I
isidor 已提交
524
	let input = group && types.isNumber(context.editorIndex) ? group.getEditor(context.editorIndex) : undefined;
525 526 527 528 529 530 531 532 533 534

	// If position or input are not passed in take the position and input of the active editor.
	const active = editorService.getActiveEditor();
	if (active) {
		position = typeof position === 'number' ? position : active.position;
		input = input ? input : <EditorInput>active.input;
	}

	return { position, input };
}
I
isidor 已提交
535

536
export function getMultiSelectedEditorContexts(editorContext: IEditorCommandsContext, listService: IListService): IEditorCommandsContext[] {
B
Benjamin Pasero 已提交
537 538
	// First check for a focused list to return the selected items from
	const list = listService.lastFocusedList;
I
isidor 已提交
539
	if (list instanceof List && list.isDOMFocused()) {
I
isidor 已提交
540 541
		const elementToContext = (element: IEditorIdentifier | EditorGroup) =>
			element instanceof EditorGroup ? { groupId: element.id, editorIndex: undefined } : { groupId: element.group.id, editorIndex: element.group.indexOf(element.editor) };
542
		const onlyEditorGroupAndEditor = (e: IEditorIdentifier | EditorGroup) => e instanceof EditorGroup || ('editor' in e && 'group' in e);
I
isidor 已提交
543 544

		const focusedElements: (IEditorIdentifier | EditorGroup)[] = list.getFocusedElements().filter(onlyEditorGroupAndEditor);
545 546 547 548
		// need to take into account when editor context is { group: group }
		const focus = editorContext ? editorContext : focusedElements.length ? focusedElements.map(elementToContext)[0] : undefined;

		if (focus) {
I
isidor 已提交
549
			const selection: (IEditorIdentifier | EditorGroup)[] = list.getSelectedElements().filter(onlyEditorGroupAndEditor);
550 551 552 553
			// Only respect selection if it contains focused element
			if (selection && selection.some(s => s instanceof EditorGroup ? s.id === focus.groupId : s.group.id === focus.groupId && s.group.indexOf(s.editor) === focus.editorIndex)) {
				return selection.map(elementToContext);
			}
I
isidor 已提交
554

555
			return [focus];
I
isidor 已提交
556 557 558
		}
	}

B
Benjamin Pasero 已提交
559
	// Otherwise go with passed in context
I
isidor 已提交
560 561
	return !!editorContext ? [editorContext] : [];
}