editorStatus.ts 45.8 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.
 *--------------------------------------------------------------------------------------------*/

import 'vs/css!./media/editorstatus';
7
import * as nls from 'vs/nls';
8
import { $, append, runAtThisOrScheduleAtNextAnimationFrame } from 'vs/base/browser/dom';
9
import * as strings from 'vs/base/common/strings';
B
Benjamin Pasero 已提交
10
import { extname, basename } from 'vs/base/common/resources';
11
import * as types from 'vs/base/common/types';
12
import { URI as uri } from 'vs/base/common/uri';
J
Johannes Rieken 已提交
13 14
import { IStatusbarItem } from 'vs/workbench/browser/parts/statusbar/statusbar';
import { Action } from 'vs/base/common/actions';
15
import { language, LANGUAGE_DEFAULT } from 'vs/base/common/platform';
J
Johannes Rieken 已提交
16
import { UntitledEditorInput } from 'vs/workbench/common/editor/untitledEditorInput';
17
import { IFileEditorInput, EncodingMode, IEncodingSupport, toResource, SideBySideEditorInput, IEditor as IBaseEditor, IEditorInput } from 'vs/workbench/common/editor';
J
Johannes Rieken 已提交
18 19
import { IDisposable, combinedDisposable, dispose } from 'vs/base/common/lifecycle';
import { IUntitledEditorService } from 'vs/workbench/services/untitled/common/untitledEditorService';
20
import { IEditorAction } from 'vs/editor/common/editorCommon';
A
Alex Dima 已提交
21
import { EndOfLineSequence, ITextModel } from 'vs/editor/common/model';
22
import { IModelLanguageChangedEvent, IModelOptionsChangedEvent } from 'vs/editor/common/model/textModelEvents';
23 24
import { TrimTrailingWhitespaceAction } from 'vs/editor/contrib/linesOperations/linesOperations';
import { IndentUsingSpaces, IndentUsingTabs, DetectIndentation, IndentationToSpacesAction, IndentationToTabsAction } from 'vs/editor/contrib/indentation/indentation';
J
Johannes Rieken 已提交
25 26
import { BaseBinaryResourceEditor } from 'vs/workbench/browser/parts/editor/binaryEditor';
import { BinaryResourceDiffEditor } from 'vs/workbench/browser/parts/editor/binaryDiffEditor';
27
import { IEditorService } from 'vs/workbench/services/editor/common/editorService';
C
Christof Marti 已提交
28
import { IQuickOpenService } from 'vs/platform/quickOpen/common/quickOpen';
29
import { SUPPORTED_ENCODINGS, IFileService, FILES_ASSOCIATIONS_CONFIG } from 'vs/platform/files/common/files';
J
Johannes Rieken 已提交
30
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
A
Alex Dima 已提交
31
import { IModeService, ILanguageSelection } from 'vs/editor/common/services/modeService';
J
Johannes Rieken 已提交
32
import { IModelService } from 'vs/editor/common/services/modelService';
33
import { Range } from 'vs/editor/common/core/range';
J
Johannes Rieken 已提交
34 35
import { Selection } from 'vs/editor/common/core/selection';
import { TabFocus } from 'vs/editor/common/config/commonEditorConfig';
36
import { ICommandService } from 'vs/platform/commands/common/commands';
37
import { IExtensionGalleryService } from 'vs/platform/extensionManagement/common/extensionManagement';
38
import { ITextFileService } from 'vs/workbench/services/textfile/common/textfiles';
39
import { ICursorPositionChangedEvent } from 'vs/editor/common/controller/cursorEvents';
40
import { IConfigurationChangedEvent, IEditorOptions } from 'vs/editor/common/config/editorOptions';
41
import { ITextResourceConfigurationService } from 'vs/editor/common/services/resourceConfiguration';
42
import { ConfigurationTarget, IConfigurationService } from 'vs/platform/configuration/common/configuration';
43
import { deepClone } from 'vs/base/common/objects';
B
Benjamin Pasero 已提交
44
import { ICodeEditor, isCodeEditor, isDiffEditor, getCodeEditor } from 'vs/editor/browser/editorBrowser';
45
import { Schemas } from 'vs/base/common/network';
S
Sandeep Somavarapu 已提交
46
import { IPreferencesService } from 'vs/workbench/services/preferences/common/preferences';
C
Christof Marti 已提交
47
import { IQuickInputService, IQuickPickItem, QuickPickInput } from 'vs/platform/quickinput/common/quickInput';
48
import { getIconClasses } from 'vs/editor/common/services/getIconClasses';
49
import { timeout } from 'vs/base/common/async';
50
import { INotificationHandle, INotificationService, Severity } from 'vs/platform/notification/common/notification';
J
Joao Moreno 已提交
51
import { Event } from 'vs/base/common/event';
52
import { IAccessibilityService, AccessibilitySupport } from 'vs/platform/accessibility/common/accessibility';
B
Benjamin Pasero 已提交
53

B
Benjamin Pasero 已提交
54 55 56
class SideBySideEditorEncodingSupport implements IEncodingSupport {
	constructor(private master: IEncodingSupport, private details: IEncodingSupport) { }

B
Benjamin Pasero 已提交
57
	getEncoding(): string {
B
Benjamin Pasero 已提交
58
		return this.master.getEncoding(); // always report from modified (right hand) side
59 60
	}

B
Benjamin Pasero 已提交
61
	setEncoding(encoding: string, mode: EncodingMode): void {
B
Benjamin Pasero 已提交
62 63 64 65
		[this.master, this.details].forEach(s => s.setEncoding(encoding, mode));
	}
}

M
Matt Bierner 已提交
66
function toEditorWithEncodingSupport(input: IEditorInput): IEncodingSupport | null {
B
Benjamin Pasero 已提交
67 68

	// Untitled Editor
69 70 71 72
	if (input instanceof UntitledEditorInput) {
		return input;
	}

B
Benjamin Pasero 已提交
73 74 75 76 77 78 79 80 81 82 83 84 85
	// Side by Side (diff) Editor
	if (input instanceof SideBySideEditorInput) {
		const masterEncodingSupport = toEditorWithEncodingSupport(input.master);
		const detailsEncodingSupport = toEditorWithEncodingSupport(input.details);

		if (masterEncodingSupport && detailsEncodingSupport) {
			return new SideBySideEditorEncodingSupport(masterEncodingSupport, detailsEncodingSupport);
		}

		return masterEncodingSupport;
	}

	// File or Resource Editor
86 87 88 89 90
	let encodingSupport = input as IFileEditorInput;
	if (types.areFunctions(encodingSupport.setEncoding, encodingSupport.getEncoding)) {
		return encodingSupport;
	}

B
Benjamin Pasero 已提交
91
	// Unsupported for any other editor
92
	return null;
93 94
}

E
Erich Gamma 已提交
95
interface IEditorSelectionStatus {
96
	selections?: Selection[];
E
Erich Gamma 已提交
97 98 99
	charactersSelected?: number;
}

100
class StateChange {
A
Alex Dima 已提交
101
	_stateChangeBrand: void;
102

M
Matt Bierner 已提交
103 104 105 106 107 108 109 110
	indentation: boolean = false;
	selectionStatus: boolean = false;
	mode: boolean = false;
	encoding: boolean = false;
	EOL: boolean = false;
	tabFocusMode: boolean = false;
	screenReaderMode: boolean = false;
	metadata: boolean = false;
111

B
Benjamin Pasero 已提交
112
	combine(other: StateChange) {
113 114 115 116 117 118
		this.indentation = this.indentation || other.indentation;
		this.selectionStatus = this.selectionStatus || other.selectionStatus;
		this.mode = this.mode || other.mode;
		this.encoding = this.encoding || other.encoding;
		this.EOL = this.EOL || other.EOL;
		this.tabFocusMode = this.tabFocusMode || other.tabFocusMode;
119
		this.screenReaderMode = this.screenReaderMode || other.screenReaderMode;
120
		this.metadata = this.metadata || other.metadata;
121
	}
122 123 124 125 126 127 128 129 130 131 132

	public hasChanges(): boolean {
		return this.indentation
			|| this.selectionStatus
			|| this.mode
			|| this.encoding
			|| this.EOL
			|| this.tabFocusMode
			|| this.screenReaderMode
			|| this.metadata;
	}
E
Erich Gamma 已提交
133 134
}

135 136 137 138 139
interface StateDelta {
	selectionStatus?: string;
	mode?: string;
	encoding?: string;
	EOL?: string;
140
	indentation?: string;
141
	tabFocusMode?: boolean;
142
	screenReaderMode?: boolean;
143
	metadata?: string | null;
144 145 146
}

class State {
147 148
	private _selectionStatus: string | null | undefined;
	get selectionStatus(): string | null | undefined { return this._selectionStatus; }
149

150 151
	private _mode: string | null | undefined;
	get mode(): string | null | undefined { return this._mode; }
152

153 154
	private _encoding: string | null | undefined;
	get encoding(): string | null | undefined { return this._encoding; }
155

156 157
	private _EOL: string | null | undefined;
	get EOL(): string | null | undefined { return this._EOL; }
158

159 160
	private _indentation: string | null | undefined;
	get indentation(): string | null | undefined { return this._indentation; }
I
isidor 已提交
161

162 163
	private _tabFocusMode: boolean | null | undefined;
	get tabFocusMode(): boolean | null | undefined { return this._tabFocusMode; }
164

165 166
	private _screenReaderMode: boolean | null | undefined;
	get screenReaderMode(): boolean | null | undefined { return this._screenReaderMode; }
167

168 169
	private _metadata: string | null | undefined;
	get metadata(): string | null | undefined { return this._metadata; }
170

171 172 173 174 175 176
	constructor() {
		this._selectionStatus = null;
		this._mode = null;
		this._encoding = null;
		this._EOL = null;
		this._tabFocusMode = false;
177
		this._screenReaderMode = false;
178
		this._metadata = null;
179 180
	}

181 182
	update(update: StateDelta): StateChange {
		const change = new StateChange();
183

184
		if ('selectionStatus' in update) {
185 186
			if (this._selectionStatus !== update.selectionStatus) {
				this._selectionStatus = update.selectionStatus;
187
				change.selectionStatus = true;
188 189
			}
		}
190
		if ('indentation' in update) {
191 192
			if (this._indentation !== update.indentation) {
				this._indentation = update.indentation;
193
				change.indentation = true;
I
isidor 已提交
194 195
			}
		}
196
		if ('mode' in update) {
197 198
			if (this._mode !== update.mode) {
				this._mode = update.mode;
199
				change.mode = true;
200 201
			}
		}
202
		if ('encoding' in update) {
203 204
			if (this._encoding !== update.encoding) {
				this._encoding = update.encoding;
205
				change.encoding = true;
206 207
			}
		}
208
		if ('EOL' in update) {
209 210
			if (this._EOL !== update.EOL) {
				this._EOL = update.EOL;
211
				change.EOL = true;
212 213
			}
		}
214
		if ('tabFocusMode' in update) {
215 216
			if (this._tabFocusMode !== update.tabFocusMode) {
				this._tabFocusMode = update.tabFocusMode;
217
				change.tabFocusMode = true;
218 219
			}
		}
220
		if ('screenReaderMode' in update) {
221 222
			if (this._screenReaderMode !== update.screenReaderMode) {
				this._screenReaderMode = update.screenReaderMode;
223
				change.screenReaderMode = true;
224 225
			}
		}
226
		if ('metadata' in update) {
227 228
			if (this._metadata !== update.metadata) {
				this._metadata = update.metadata;
229
				change.metadata = true;
230 231
			}
		}
232

233
		return change;
234 235 236
	}
}

237 238 239 240 241 242
const nlsSingleSelectionRange = nls.localize('singleSelectionRange', "Ln {0}, Col {1} ({2} selected)");
const nlsSingleSelection = nls.localize('singleSelection', "Ln {0}, Col {1}");
const nlsMultiSelectionRange = nls.localize('multiSelectionRange', "{0} selections ({1} characters selected)");
const nlsMultiSelection = nls.localize('multiSelection', "{0} selections");
const nlsEOLLF = nls.localize('endOfLineLineFeed', "LF");
const nlsEOLCRLF = nls.localize('endOfLineCarriageReturnLineFeed', "CRLF");
J
Jens Hausdorf 已提交
243
const nlsTabFocusMode = nls.localize('tabFocusModeEnabled', "Tab Moves Focus");
244
const nlsScreenReaderDetected = nls.localize('screenReaderDetected', "Screen Reader Optimized");
A
Alex Dima 已提交
245
const nlsScreenReaderDetectedTitle = nls.localize('screenReaderDetectedExtra', "If you are not using a Screen Reader, please change the setting `editor.accessibilitySupport` to \"off\".");
E
Erich Gamma 已提交
246

247

248
class StatusBarItem {
249 250 251
	private _showing = true;

	constructor(
M
Matt Bierner 已提交
252 253
		private readonly element: HTMLElement,
		title: string,
254 255
	) {
		this.setVisible(false);
M
Matt Bierner 已提交
256
		this.element.title = title;
257
	}
258

259 260 261
	public set textContent(value: string) {
		this.element.textContent = value;
	}
262

263 264 265 266 267
	public set onclick(value: () => void) {
		this.element.onclick = value;
	}

	public setVisible(shouldShow: boolean): void {
268 269 270 271
		if (shouldShow !== this._showing) {
			this._showing = shouldShow;
			this.element.style.display = shouldShow ? '' : 'none';
		}
A
Alex Dima 已提交
272 273
	}
}
274

A
Alex Dima 已提交
275

276
export class EditorStatus implements IStatusbarItem {
277
	private state: State;
278
	private element: HTMLElement;
279 280 281 282 283 284 285 286
	private tabFocusModeElement: StatusBarItem;
	private screenRedearModeElement: StatusBarItem;
	private indentationElement: StatusBarItem;
	private selectionElement: StatusBarItem;
	private encodingElement: StatusBarItem;
	private eolElement: StatusBarItem;
	private modeElement: StatusBarItem;
	private metadataElement: StatusBarItem;
E
Erich Gamma 已提交
287
	private toDispose: IDisposable[];
288
	private activeEditorListeners: IDisposable[];
M
Matt Bierner 已提交
289 290 291
	private delayedRender: IDisposable | null;
	private toRender: StateChange | null;
	private screenReaderNotification: INotificationHandle | null;
E
Erich Gamma 已提交
292

293
	constructor(
294 295 296 297 298 299
		@IEditorService private readonly editorService: IEditorService,
		@IQuickOpenService private readonly quickOpenService: IQuickOpenService,
		@IInstantiationService private readonly instantiationService: IInstantiationService,
		@IUntitledEditorService private readonly untitledEditorService: IUntitledEditorService,
		@IModeService private readonly modeService: IModeService,
		@ITextFileService private readonly textFileService: ITextFileService,
300
		@IConfigurationService private readonly configurationService: IConfigurationService,
301 302
		@INotificationService private readonly notificationService: INotificationService,
		@IAccessibilityService private readonly accessibilityService: IAccessibilityService
303 304
	) {
		this.toDispose = [];
305
		this.activeEditorListeners = [];
306
		this.state = new State();
E
Erich Gamma 已提交
307 308
	}

B
Benjamin Pasero 已提交
309
	render(container: HTMLElement): IDisposable {
310 311
		this.element = append(container, $('.editor-statusbar-item'));

M
Matt Bierner 已提交
312 313 314
		this.tabFocusModeElement = new StatusBarItem(
			append(this.element, $('a.editor-status-tabfocusmode.status-bar-info')),
			nls.localize('disableTabMode', "Disable Accessibility Mode"));
315 316 317
		this.tabFocusModeElement.onclick = () => this.onTabFocusModeClick();
		this.tabFocusModeElement.textContent = nlsTabFocusMode;

M
Matt Bierner 已提交
318 319 320
		this.screenRedearModeElement = new StatusBarItem(
			append(this.element, $('a.editor-status-screenreadermode.status-bar-info')),
			nlsScreenReaderDetectedTitle);
321 322 323
		this.screenRedearModeElement.textContent = nlsScreenReaderDetected;
		this.screenRedearModeElement.onclick = () => this.onScreenReaderModeClick();

M
Matt Bierner 已提交
324 325 326
		this.selectionElement = new StatusBarItem(
			append(this.element, $('a.editor-status-selection')),
			nls.localize('gotoLine', "Go to Line"));
327 328
		this.selectionElement.onclick = () => this.onSelectionClick();

M
Matt Bierner 已提交
329 330 331
		this.indentationElement = new StatusBarItem(
			append(this.element, $('a.editor-status-indentation')),
			nls.localize('selectIndentation', "Select Indentation"));
332 333
		this.indentationElement.onclick = () => this.onIndentationClick();

M
Matt Bierner 已提交
334 335 336
		this.encodingElement = new StatusBarItem(
			append(this.element, $('a.editor-status-encoding')),
			nls.localize('selectEncoding', "Select Encoding"));
337 338
		this.encodingElement.onclick = () => this.onEncodingClick();

M
Matt Bierner 已提交
339 340 341
		this.eolElement = new StatusBarItem(
			append(this.element, $('a.editor-status-eol')),
			nls.localize('selectEOL', "Select End of Line Sequence"));
342 343
		this.eolElement.onclick = () => this.onEOLClick();

M
Matt Bierner 已提交
344 345 346
		this.modeElement = new StatusBarItem(
			append(this.element, $('a.editor-status-mode')),
			nls.localize('selectLanguageMode', "Select Language Mode"));
347 348
		this.modeElement.onclick = () => this.onModeClick();

M
Matt Bierner 已提交
349 350 351
		this.metadataElement = new StatusBarItem(
			append(this.element, $('span.editor-status-metadata')),
			nls.localize('fileInfo', "File Information"));
352

353 354 355
		this.delayedRender = null;
		this.toRender = null;

356
		this.toDispose.push(
357 358 359 360 361 362 363 364
			{
				dispose: () => {
					if (this.delayedRender) {
						this.delayedRender.dispose();
						this.delayedRender = null;
					}
				}
			},
365
			this.editorService.onDidActiveEditorChange(() => this.updateStatusBar()),
366 367
			this.untitledEditorService.onDidChangeEncoding(r => this.onResourceEncodingChange(r)),
			this.textFileService.models.onModelEncodingChanged(e => this.onResourceEncodingChange(e.resource)),
368
			TabFocus.onDidChangeTabFocus(e => this.onTabFocusModeChange()),
369
		);
E
Erich Gamma 已提交
370

J
Joao Moreno 已提交
371
		return combinedDisposable(this.toDispose);
372 373
	}

374
	private updateState(update: StateDelta): void {
B
Benjamin Pasero 已提交
375
		const changed = this.state.update(update);
376
		if (!changed.hasChanges()) {
377 378
			// Nothing really changed
			return;
E
Erich Gamma 已提交
379 380
		}

381 382 383 384
		if (!this.toRender) {
			this.toRender = changed;
			this.delayedRender = runAtThisOrScheduleAtNextAnimationFrame(() => {
				this.delayedRender = null;
B
Benjamin Pasero 已提交
385
				const toRender = this.toRender;
386
				this.toRender = null;
M
Matt Bierner 已提交
387 388 389
				if (toRender) {
					this._renderNow(toRender);
				}
390 391 392 393 394 395
			});
		} else {
			this.toRender.combine(changed);
		}
	}

396
	private _renderNow(changed: StateChange): void {
397
		if (changed.tabFocusMode) {
398
			this.tabFocusModeElement.setVisible(!!this.state.tabFocusMode);
E
Erich Gamma 已提交
399 400
		}

401
		if (changed.screenReaderMode) {
402
			this.screenRedearModeElement.setVisible(!!this.state.screenReaderMode);
403 404
		}

I
isidor 已提交
405 406
		if (changed.indentation) {
			if (this.state.indentation) {
407 408
				this.indentationElement.textContent = this.state.indentation;
				this.indentationElement.setVisible(true);
I
isidor 已提交
409
			} else {
410
				this.indentationElement.setVisible(false);
I
isidor 已提交
411 412 413
			}
		}

414
		if (changed.selectionStatus) {
415
			if (this.state.selectionStatus && !this.state.screenReaderMode) {
416 417
				this.selectionElement.textContent = this.state.selectionStatus;
				this.selectionElement.setVisible(true);
418
			} else {
419
				this.selectionElement.setVisible(false);
420
			}
E
Erich Gamma 已提交
421 422
		}

423 424
		if (changed.encoding) {
			if (this.state.encoding) {
425 426
				this.encodingElement.textContent = this.state.encoding;
				this.encodingElement.setVisible(true);
427
			} else {
428
				this.encodingElement.setVisible(false);
429
			}
E
Erich Gamma 已提交
430 431
		}

432 433
		if (changed.EOL) {
			if (this.state.EOL) {
434 435
				this.eolElement.textContent = this.state.EOL === '\r\n' ? nlsEOLCRLF : nlsEOLLF;
				this.eolElement.setVisible(true);
436
			} else {
437
				this.eolElement.setVisible(false);
438
			}
E
Erich Gamma 已提交
439 440
		}

441 442
		if (changed.mode) {
			if (this.state.mode) {
443 444
				this.modeElement.textContent = this.state.mode;
				this.modeElement.setVisible(true);
445
			} else {
446
				this.modeElement.setVisible(false);
447 448
			}
		}
449 450 451

		if (changed.metadata) {
			if (this.state.metadata) {
452 453
				this.metadataElement.textContent = this.state.metadata;
				this.metadataElement.setVisible(true);
454
			} else {
455
				this.metadataElement.setVisible(false);
456 457
			}
		}
E
Erich Gamma 已提交
458 459
	}

M
Matt Bierner 已提交
460
	private getSelectionLabel(info: IEditorSelectionStatus): string | undefined {
E
Erich Gamma 已提交
461
		if (!info || !info.selections) {
M
Matt Bierner 已提交
462
			return undefined;
E
Erich Gamma 已提交
463 464 465 466
		}

		if (info.selections.length === 1) {
			if (info.charactersSelected) {
467
				return strings.format(nlsSingleSelectionRange, info.selections[0].positionLineNumber, info.selections[0].positionColumn, info.charactersSelected);
E
Erich Gamma 已提交
468
			}
B
Benjamin Pasero 已提交
469 470

			return strings.format(nlsSingleSelection, info.selections[0].positionLineNumber, info.selections[0].positionColumn);
E
Erich Gamma 已提交
471
		}
B
Benjamin Pasero 已提交
472 473 474 475 476 477 478 479 480

		if (info.charactersSelected) {
			return strings.format(nlsMultiSelectionRange, info.selections.length, info.charactersSelected);
		}

		if (info.selections.length > 0) {
			return strings.format(nlsMultiSelection, info.selections.length);
		}

M
Matt Bierner 已提交
481
		return undefined;
E
Erich Gamma 已提交
482 483
	}

484
	private onModeClick(): void {
B
Benjamin Pasero 已提交
485
		const action = this.instantiationService.createInstance(ChangeModeAction, ChangeModeAction.ID, ChangeModeAction.LABEL);
486

487
		action.run();
488 489 490
		action.dispose();
	}

I
isidor 已提交
491 492
	private onIndentationClick(): void {
		const action = this.instantiationService.createInstance(ChangeIndentationAction, ChangeIndentationAction.ID, ChangeIndentationAction.LABEL);
493
		action.run();
I
isidor 已提交
494 495 496
		action.dispose();
	}

497
	private onScreenReaderModeClick(): void {
498 499 500
		if (!this.screenReaderNotification) {
			this.screenReaderNotification = this.notificationService.prompt(
				Severity.Info,
A
Alex Dima 已提交
501
				nls.localize('screenReaderDetectedExplanation.question', "Are you using a screen reader to operate VS Code? (Certain features like folding, minimap or word wrap are disabled when using a screen reader)"),
502 503 504 505 506 507 508 509 510 511
				[{
					label: nls.localize('screenReaderDetectedExplanation.answerYes', "Yes"),
					run: () => {
						this.configurationService.updateValue('editor.accessibilitySupport', 'on', ConfigurationTarget.USER);
					}
				}, {
					label: nls.localize('screenReaderDetectedExplanation.answerNo', "No"),
					run: () => {
						this.configurationService.updateValue('editor.accessibilitySupport', 'off', ConfigurationTarget.USER);
					}
B
Benjamin Pasero 已提交
512 513
				}],
				{ sticky: true }
514
			);
B
Benjamin Pasero 已提交
515

J
Joao Moreno 已提交
516
			Event.once(this.screenReaderNotification.onDidClose)(() => {
B
Benjamin Pasero 已提交
517 518
				this.screenReaderNotification = null;
			});
519
		}
520 521
	}

522 523 524 525 526
	private onSelectionClick(): void {
		this.quickOpenService.show(':'); // "Go to line"
	}

	private onEOLClick(): void {
B
Benjamin Pasero 已提交
527
		const action = this.instantiationService.createInstance(ChangeEOLAction, ChangeEOLAction.ID, ChangeEOLAction.LABEL);
528

529
		action.run();
530 531 532 533
		action.dispose();
	}

	private onEncodingClick(): void {
B
Benjamin Pasero 已提交
534
		const action = this.instantiationService.createInstance(ChangeEncodingAction, ChangeEncodingAction.ID, ChangeEncodingAction.LABEL);
535

536
		action.run();
537 538 539 540
		action.dispose();
	}

	private onTabFocusModeClick(): void {
541
		TabFocus.setTabFocusMode(false);
542 543
	}

544 545
	private updateStatusBar(): void {
		const activeControl = this.editorService.activeControl;
M
Matt Bierner 已提交
546
		const activeCodeEditor = activeControl ? types.withNullAsUndefined(getCodeEditor(activeControl.getControl())) : undefined;
547 548

		// Update all states
549 550 551 552 553 554 555
		this.onScreenReaderModeChange(activeCodeEditor);
		this.onSelectionChange(activeCodeEditor);
		this.onModeChange(activeCodeEditor);
		this.onEOLChange(activeCodeEditor);
		this.onEncodingChange(activeControl);
		this.onIndentationChange(activeCodeEditor);
		this.onMetadataChange(activeControl);
E
Erich Gamma 已提交
556

557 558 559 560
		// Dispose old active editor listeners
		dispose(this.activeEditorListeners);

		// Attach new listeners to active editor
561
		if (activeCodeEditor) {
562

563
			// Hook Listener for Configuration changes
564
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeConfiguration((event: IConfigurationChangedEvent) => {
565
				if (event.accessibilitySupport) {
566
					this.onScreenReaderModeChange(activeCodeEditor);
567 568 569
				}
			}));

570
			// Hook Listener for Selection changes
571 572
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeCursorPosition((event: ICursorPositionChangedEvent) => {
				this.onSelectionChange(activeCodeEditor);
573 574 575
			}));

			// Hook Listener for mode changes
576 577
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeModelLanguage((event: IModelLanguageChangedEvent) => {
				this.onModeChange(activeCodeEditor);
578 579 580
			}));

			// Hook Listener for content changes
581 582
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeModelContent((e) => {
				this.onEOLChange(activeCodeEditor);
583

M
Matt Bierner 已提交
584
				const selections = activeCodeEditor.getSelections();
M
Matt Bierner 已提交
585 586 587 588 589 590
				if (selections) {
					for (const change of e.changes) {
						if (selections.some(selection => Range.areIntersecting(selection, change.range))) {
							this.onSelectionChange(activeCodeEditor);
							break;
						}
591 592
					}
				}
593 594 595
			}));

			// Hook Listener for content options changes
596 597
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeModelOptions((event: IModelOptionsChangedEvent) => {
				this.onIndentationChange(activeCodeEditor);
598
			}));
E
Erich Gamma 已提交
599
		}
600 601

		// Handle binary editors
602
		else if (activeControl instanceof BaseBinaryResourceEditor || activeControl instanceof BinaryResourceDiffEditor) {
603
			const binaryEditors: BaseBinaryResourceEditor[] = [];
604 605
			if (activeControl instanceof BinaryResourceDiffEditor) {
				const details = activeControl.getDetailsEditor();
606 607 608 609
				if (details instanceof BaseBinaryResourceEditor) {
					binaryEditors.push(details);
				}

610
				const master = activeControl.getMasterEditor();
611 612 613
				if (master instanceof BaseBinaryResourceEditor) {
					binaryEditors.push(master);
				}
614
			} else {
615
				binaryEditors.push(activeControl);
616 617 618 619
			}

			binaryEditors.forEach(editor => {
				this.activeEditorListeners.push(editor.onMetadataChanged(metadata => {
620
					this.onMetadataChange(activeControl);
621
				}));
622 623 624 625

				this.activeEditorListeners.push(editor.onDidOpenInPlace(() => {
					this.updateStatusBar();
				}));
626
			});
627
		}
628
	}
E
Erich Gamma 已提交
629

M
Matt Bierner 已提交
630 631
	private onModeChange(editorWidget: ICodeEditor | undefined): void {
		let info: StateDelta = { mode: undefined };
E
Erich Gamma 已提交
632 633

		// We only support text based editors
634
		if (editorWidget) {
A
Alex Dima 已提交
635
			const textModel = editorWidget.getModel();
E
Erich Gamma 已提交
636 637
			if (textModel) {
				// Compute mode
A
Alex Dima 已提交
638
				const modeId = textModel.getLanguageIdentifier().language;
M
Matt Bierner 已提交
639
				info = { mode: this.modeService.getLanguageName(modeId) || undefined };
E
Erich Gamma 已提交
640 641 642 643 644 645
			}
		}

		this.updateState(info);
	}

M
Matt Bierner 已提交
646 647
	private onIndentationChange(editorWidget: ICodeEditor | undefined): void {
		const update: StateDelta = { indentation: undefined };
648

649
		if (editorWidget) {
A
Alex Dima 已提交
650
			const model = editorWidget.getModel();
651 652 653 654
			if (model) {
				const modelOpts = model.getOptions();
				update.indentation = (
					modelOpts.insertSpaces
D
David Lechner 已提交
655
						? nls.localize('spacesSize', "Spaces: {0}", modelOpts.indentSize)
656 657
						: nls.localize({ key: 'tabSize', comment: ['Tab corresponds to the tab key'] }, "Tab Size: {0}", modelOpts.tabSize)
				);
658
			}
I
isidor 已提交
659
		}
660

I
isidor 已提交
661 662 663
		this.updateState(update);
	}

M
Matt Bierner 已提交
664
	private onMetadataChange(editor: IBaseEditor | undefined): void {
M
Matt Bierner 已提交
665
		const update: StateDelta = { metadata: undefined };
666 667

		if (editor instanceof BaseBinaryResourceEditor || editor instanceof BinaryResourceDiffEditor) {
668
			update.metadata = editor.getMetadata();
669 670 671 672 673
		}

		this.updateState(update);
	}

674 675
	private _promptedScreenReader: boolean = false;

M
Matt Bierner 已提交
676
	private onScreenReaderModeChange(editorWidget: ICodeEditor | undefined): void {
677 678 679 680
		let screenReaderMode = false;

		// We only support text based editors
		if (editorWidget) {
681
			const screenReaderDetected = (this.accessibilityService.getAccessibilitySupport() === AccessibilitySupport.Enabled);
682
			if (screenReaderDetected) {
683
				const screenReaderConfiguration = this.configurationService.getValue<IEditorOptions>('editor').accessibilitySupport;
684 685 686 687 688 689 690 691 692 693
				if (screenReaderConfiguration === 'auto') {
					// show explanation
					if (!this._promptedScreenReader) {
						this._promptedScreenReader = true;
						setTimeout(() => {
							this.onScreenReaderModeClick();
						}, 100);
					}
				}
			}
694 695 696 697

			screenReaderMode = (editorWidget.getConfiguration().accessibilitySupport === AccessibilitySupport.Enabled);
		}

698 699
		if (screenReaderMode === false && this.screenReaderNotification) {
			this.screenReaderNotification.close();
700 701
		}

702 703 704
		this.updateState({ screenReaderMode: screenReaderMode });
	}

M
Matt Bierner 已提交
705
	private onSelectionChange(editorWidget: ICodeEditor | undefined): void {
B
Benjamin Pasero 已提交
706
		const info: IEditorSelectionStatus = {};
E
Erich Gamma 已提交
707 708

		// We only support text based editors
709
		if (editorWidget) {
E
Erich Gamma 已提交
710 711 712 713 714 715

			// Compute selection(s)
			info.selections = editorWidget.getSelections() || [];

			// Compute selection length
			info.charactersSelected = 0;
A
Alex Dima 已提交
716
			const textModel = editorWidget.getModel();
E
Erich Gamma 已提交
717
			if (textModel) {
718
				info.selections.forEach(selection => {
M
Matt Bierner 已提交
719
					info.charactersSelected! += textModel.getValueLengthInRange(selection);
E
Erich Gamma 已提交
720 721 722 723 724
				});
			}

			// Compute the visible column for one selection. This will properly handle tabs and their configured widths
			if (info.selections.length === 1) {
M
Matt Bierner 已提交
725
				const visibleColumn = editorWidget.getVisibleColumnFromPosition(editorWidget.getPosition()!);
E
Erich Gamma 已提交
726

727 728 729 730 731 732 733
				let selectionClone = info.selections[0].clone(); // do not modify the original position we got from the editor
				selectionClone = new Selection(
					selectionClone.selectionStartLineNumber,
					selectionClone.selectionStartColumn,
					selectionClone.positionLineNumber,
					visibleColumn
				);
E
Erich Gamma 已提交
734 735 736 737 738

				info.selections[0] = selectionClone;
			}
		}

739
		this.updateState({ selectionStatus: this.getSelectionLabel(info) });
E
Erich Gamma 已提交
740 741
	}

M
Matt Bierner 已提交
742 743
	private onEOLChange(editorWidget: ICodeEditor | undefined): void {
		const info: StateDelta = { EOL: undefined };
A
Alex Dima 已提交
744

A
Alex Dima 已提交
745 746
		if (editorWidget && !editorWidget.getConfiguration().readOnly) {
			const codeEditorModel = editorWidget.getModel();
I
isidor 已提交
747 748 749
			if (codeEditorModel) {
				info.EOL = codeEditorModel.getEOL();
			}
E
Erich Gamma 已提交
750 751 752 753 754
		}

		this.updateState(info);
	}

755
	private onEncodingChange(e?: IBaseEditor): void {
E
Erich Gamma 已提交
756 757 758 759
		if (e && !this.isActiveEditor(e)) {
			return;
		}

M
Matt Bierner 已提交
760
		const info: StateDelta = { encoding: undefined };
E
Erich Gamma 已提交
761 762

		// We only support text based editors
763
		if (e && (isCodeEditor(e.getControl()) || isDiffEditor(e.getControl()))) {
M
Matt Bierner 已提交
764
			const encodingSupport: IEncodingSupport | null = e.input ? toEditorWithEncodingSupport(e.input) : null;
765
			if (encodingSupport) {
B
Benjamin Pasero 已提交
766 767
				const rawEncoding = encodingSupport.getEncoding();
				const encodingInfo = SUPPORTED_ENCODINGS[rawEncoding];
E
Erich Gamma 已提交
768 769 770 771 772 773 774 775 776 777 778
				if (encodingInfo) {
					info.encoding = encodingInfo.labelShort; // if we have a label, take it from there
				} else {
					info.encoding = rawEncoding; // otherwise use it raw
				}
			}
		}

		this.updateState(info);
	}

779
	private onResourceEncodingChange(resource: uri): void {
780 781 782
		const activeControl = this.editorService.activeControl;
		if (activeControl) {
			const activeResource = toResource(activeControl.input, { supportSideBySide: true });
B
Benjamin Pasero 已提交
783
			if (activeResource && activeResource.toString() === resource.toString()) {
784
				return this.onEncodingChange(<IBaseEditor>activeControl); // only update if the encoding changed for the active resource
785 786
			}
		}
E
Erich Gamma 已提交
787 788
	}

789
	private onTabFocusModeChange(): void {
B
Benjamin Pasero 已提交
790
		const info: StateDelta = { tabFocusMode: TabFocus.getTabFocusMode() };
E
Erich Gamma 已提交
791 792 793 794

		this.updateState(info);
	}

795 796
	private isActiveEditor(control: IBaseEditor): boolean {
		const activeControl = this.editorService.activeControl;
E
Erich Gamma 已提交
797

798
		return !!activeControl && activeControl === control;
E
Erich Gamma 已提交
799 800 801
	}
}

M
Matt Bierner 已提交
802
function isWritableCodeEditor(codeEditor: ICodeEditor | undefined): boolean {
A
Alex Dima 已提交
803
	if (!codeEditor) {
S
Sandeep Somavarapu 已提交
804 805
		return false;
	}
A
Alex Dima 已提交
806 807 808
	const config = codeEditor.getConfiguration();
	return (!config.readOnly);
}
S
Sandeep Somavarapu 已提交
809

A
Alex Dima 已提交
810
function isWritableBaseEditor(e: IBaseEditor): boolean {
M
Matt Bierner 已提交
811
	return e && isWritableCodeEditor(getCodeEditor(e.getControl()) || undefined);
E
Erich Gamma 已提交
812 813
}

814 815
export class ShowLanguageExtensionsAction extends Action {

816
	static readonly ID = 'workbench.action.showLanguageExtensions';
817 818

	constructor(
819
		private fileExtension: string,
820
		@ICommandService private readonly commandService: ICommandService,
821 822
		@IExtensionGalleryService galleryService: IExtensionGalleryService
	) {
823 824
		super(ShowLanguageExtensionsAction.ID, nls.localize('showLanguageExtensions', "Search Marketplace Extensions for '{0}'...", fileExtension));

825 826 827
		this.enabled = galleryService.isEnabled();
	}

J
Johannes Rieken 已提交
828
	run(): Promise<void> {
R
Rob Lourens 已提交
829
		return this.commandService.executeCommand('workbench.extensions.action.showExtensionsForLanguage', this.fileExtension).then(() => undefined);
830 831 832
	}
}

E
Erich Gamma 已提交
833 834
export class ChangeModeAction extends Action {

B
Benjamin Pasero 已提交
835 836
	static readonly ID = 'workbench.action.editor.changeLanguageMode';
	static readonly LABEL = nls.localize('changeMode', "Change Language Mode");
E
Erich Gamma 已提交
837 838 839 840

	constructor(
		actionId: string,
		actionLabel: string,
841 842 843
		@IModeService private readonly modeService: IModeService,
		@IModelService private readonly modelService: IModelService,
		@IEditorService private readonly editorService: IEditorService,
844
		@IConfigurationService private readonly configurationService: IConfigurationService,
845 846 847 848
		@IQuickInputService private readonly quickInputService: IQuickInputService,
		@IPreferencesService private readonly preferencesService: IPreferencesService,
		@IInstantiationService private readonly instantiationService: IInstantiationService,
		@IUntitledEditorService private readonly untitledEditorService: IUntitledEditorService
E
Erich Gamma 已提交
849 850 851 852
	) {
		super(actionId, actionLabel);
	}

J
Johannes Rieken 已提交
853
	run(): Promise<any> {
854 855
		const activeTextEditorWidget = getCodeEditor(this.editorService.activeTextEditorWidget);
		if (!activeTextEditorWidget) {
C
Christof Marti 已提交
856
			return this.quickInputService.pick([{ label: nls.localize('noEditor', "No text editor active at this time") }]);
E
Erich Gamma 已提交
857 858
		}

859
		const textModel = activeTextEditorWidget.getModel();
M
Matt Bierner 已提交
860
		const resource = this.editorService.activeEditor ? toResource(this.editorService.activeEditor, { supportSideBySide: true }) : null;
B
Benjamin Pasero 已提交
861 862

		let hasLanguageSupport = !!resource;
M
Matt Bierner 已提交
863
		if (resource && resource.scheme === Schemas.untitled && !this.untitledEditorService.hasAssociatedFilePath(resource)) {
B
Benjamin Pasero 已提交
864
			hasLanguageSupport = false; // no configuration for untitled resources (e.g. "Untitled-1")
865
		}
E
Erich Gamma 已提交
866 867

		// Compute mode
M
Matt Bierner 已提交
868
		let currentModeId: string | undefined;
869
		let modeId: string;
870
		if (textModel) {
871
			modeId = textModel.getLanguageIdentifier().language;
M
Matt Bierner 已提交
872
			currentModeId = this.modeService.getLanguageName(modeId) || undefined;
E
Erich Gamma 已提交
873 874 875
		}

		// All languages are valid picks
876
		const languages = this.modeService.getRegisteredLanguageNames();
C
Christof Marti 已提交
877
		const picks: QuickPickInput[] = languages.sort().map((lang, index) => {
B
Benjamin Pasero 已提交
878 879 880 881 882 883
			let description: string;
			if (currentModeId === lang) {
				description = nls.localize('languageDescription', "({0}) - Configured Language", this.modeService.getModeIdForLanguageName(lang.toLowerCase()));
			} else {
				description = nls.localize('languageDescriptionConfigured', "({0})", this.modeService.getModeIdForLanguageName(lang.toLowerCase()));
			}
884

885
			// construct a fake resource to be able to show nice icons if any
M
Matt Bierner 已提交
886
			let fakeResource: uri | undefined;
887 888 889 890 891 892 893 894 895 896
			const extensions = this.modeService.getExtensions(lang);
			if (extensions && extensions.length) {
				fakeResource = uri.file(extensions[0]);
			} else {
				const filenames = this.modeService.getFilenames(lang);
				if (filenames && filenames.length) {
					fakeResource = uri.file(filenames[0]);
				}
			}

C
Christof Marti 已提交
897
			return <IQuickPickItem>{
898
				label: lang,
C
Christof Marti 已提交
899
				iconClasses: getIconClasses(this.modelService, this.modeService, fakeResource),
B
Benjamin Pasero 已提交
900
				description
E
Erich Gamma 已提交
901 902
			};
		});
903

B
Benjamin Pasero 已提交
904
		if (hasLanguageSupport) {
C
Christof Marti 已提交
905
			picks.unshift({ type: 'separator', label: nls.localize('languagesPicks', "languages (identifier)") });
906
		}
E
Erich Gamma 已提交
907

908
		// Offer action to configure via settings
C
Christof Marti 已提交
909 910
		let configureModeAssociations: IQuickPickItem;
		let configureModeSettings: IQuickPickItem;
911
		let galleryAction: Action;
M
Matt Bierner 已提交
912
		if (hasLanguageSupport && resource) {
B
Benjamin Pasero 已提交
913
			const ext = extname(resource) || basename(resource);
S
Sandeep Somavarapu 已提交
914 915 916 917 918 919

			galleryAction = this.instantiationService.createInstance(ShowLanguageExtensionsAction, ext);
			if (galleryAction.enabled) {
				picks.unshift(galleryAction);
			}

920 921
			configureModeSettings = { label: nls.localize('configureModeSettings', "Configure '{0}' language based settings...", currentModeId) };
			picks.unshift(configureModeSettings);
922
			configureModeAssociations = { label: nls.localize('configureAssociationsExt', "Configure File Association for '{0}'...", ext) };
923 924
			picks.unshift(configureModeAssociations);
		}
E
Erich Gamma 已提交
925

926
		// Offer to "Auto Detect"
C
Christof Marti 已提交
927
		const autoDetectMode: IQuickPickItem = {
928 929
			label: nls.localize('autoDetect', "Auto Detect")
		};
930

B
Benjamin Pasero 已提交
931
		if (hasLanguageSupport) {
932 933
			picks.unshift(autoDetectMode);
		}
934

C
Christof Marti 已提交
935
		return this.quickInputService.pick(picks, { placeHolder: nls.localize('pickLanguage', "Select Language Mode"), matchOnDescription: true }).then(pick => {
936 937 938
			if (!pick) {
				return;
			}
B
Benjamin Pasero 已提交
939

940 941 942 943
			if (pick === galleryAction) {
				galleryAction.run();
				return;
			}
B
Benjamin Pasero 已提交
944

945 946
			// User decided to permanently configure associations, return right after
			if (pick === configureModeAssociations) {
M
Matt Bierner 已提交
947 948 949
				if (resource) {
					this.configureFileAssociation(resource);
				}
950 951
				return;
			}
952

953 954
			// User decided to configure settings for current language
			if (pick === configureModeSettings) {
955
				this.preferencesService.configureSettingsForLanguage(modeId);
956 957 958
				return;
			}

959
			// Change mode for active editor
B
Benjamin Pasero 已提交
960
			const activeEditor = this.editorService.activeEditor;
961
			const activeTextEditorWidget = this.editorService.activeTextEditorWidget;
A
Alex Dima 已提交
962
			const models: ITextModel[] = [];
963 964
			if (isCodeEditor(activeTextEditorWidget)) {
				const codeEditorModel = activeTextEditorWidget.getModel();
965 966
				if (codeEditorModel) {
					models.push(codeEditorModel);
967
				}
968 969
			} else if (isDiffEditor(activeTextEditorWidget)) {
				const diffEditorModel = activeTextEditorWidget.getModel();
970 971 972 973 974 975 976
				if (diffEditorModel) {
					if (diffEditorModel.original) {
						models.push(diffEditorModel.original);
					}
					if (diffEditorModel.modified) {
						models.push(diffEditorModel.modified);
					}
E
Erich Gamma 已提交
977
				}
978
			}
979

980
			// Find mode
M
Matt Bierner 已提交
981
			let languageSelection: ILanguageSelection | undefined;
982
			if (pick === autoDetectMode) {
M
Matt Bierner 已提交
983 984 985 986 987 988
				if (textModel) {
					const resource = toResource(activeEditor, { supportSideBySide: true });
					if (resource) {
						languageSelection = this.modeService.createByFilepathOrFirstLine(resource.fsPath, textModel.getLineContent(1));
					}
				}
989
			} else {
A
Alex Dima 已提交
990
				languageSelection = this.modeService.createByLanguageName(pick.label);
E
Erich Gamma 已提交
991
			}
992 993

			// Change mode
M
Matt Bierner 已提交
994 995 996 997 998
			if (typeof languageSelection !== 'undefined') {
				for (const textModel of models) {
					this.modelService.setMode(textModel, languageSelection);
				}
			}
E
Erich Gamma 已提交
999 1000
		});
	}
1001 1002

	private configureFileAssociation(resource: uri): void {
B
Benjamin Pasero 已提交
1003 1004 1005
		const extension = extname(resource);
		const base = basename(resource);
		const currentAssociation = this.modeService.getModeIdByFilepathOrFirstLine(base);
1006

1007
		const languages = this.modeService.getRegisteredLanguageNames();
C
Christof Marti 已提交
1008
		const picks: IQuickPickItem[] = languages.sort().map((lang, index) => {
1009 1010
			const id = this.modeService.getModeIdForLanguageName(lang.toLowerCase());

C
Christof Marti 已提交
1011
			return <IQuickPickItem>{
1012 1013
				id,
				label: lang,
R
Rob Lourens 已提交
1014
				description: (id === currentAssociation) ? nls.localize('currentAssociation', "Current Association") : undefined
1015 1016 1017
			};
		});

1018
		setTimeout(() => {
B
Benjamin Pasero 已提交
1019
			this.quickInputService.pick(picks, { placeHolder: nls.localize('pickLanguageToConfigure', "Select Language Mode to Associate with '{0}'", extension || base) }).then(language => {
1020
				if (language) {
1021
					const fileAssociationsConfig = this.configurationService.inspect(FILES_ASSOCIATIONS_CONFIG);
1022 1023

					let associationKey: string;
B
Benjamin Pasero 已提交
1024
					if (extension && base[0] !== '.') {
1025 1026
						associationKey = `*${extension}`; // only use "*.ext" if the file path is in the form of <name>.<ext>
					} else {
B
Benjamin Pasero 已提交
1027
						associationKey = base; // otherwise use the basename (e.g. .gitignore, Dockerfile)
1028 1029
					}

1030 1031 1032 1033 1034 1035 1036
					// If the association is already being made in the workspace, make sure to target workspace settings
					let target = ConfigurationTarget.USER;
					if (fileAssociationsConfig.workspace && !!fileAssociationsConfig.workspace[associationKey]) {
						target = ConfigurationTarget.WORKSPACE;
					}

					// Make sure to write into the value of the target and not the merged value from USER and WORKSPACE config
M
Matt Bierner 已提交
1037
					const currentAssociations = deepClone((target === ConfigurationTarget.WORKSPACE) ? fileAssociationsConfig.workspace : fileAssociationsConfig.user) || Object.create(null);
1038 1039
					currentAssociations[associationKey] = language.id;

1040
					this.configurationService.updateValue(FILES_ASSOCIATIONS_CONFIG, currentAssociations, target);
1041
				}
1042
			});
1043
		}, 50 /* quick open is sensitive to being opened so soon after another */);
1044
	}
E
Erich Gamma 已提交
1045 1046
}

C
Christof Marti 已提交
1047
export interface IChangeEOLEntry extends IQuickPickItem {
E
Erich Gamma 已提交
1048 1049 1050
	eol: EndOfLineSequence;
}

I
isidor 已提交
1051
class ChangeIndentationAction extends Action {
I
isidor 已提交
1052

B
Benjamin Pasero 已提交
1053 1054
	static readonly ID = 'workbench.action.editor.changeIndentation';
	static readonly LABEL = nls.localize('changeIndentation', "Change Indentation");
I
isidor 已提交
1055 1056 1057 1058

	constructor(
		actionId: string,
		actionLabel: string,
1059 1060
		@IEditorService private readonly editorService: IEditorService,
		@IQuickInputService private readonly quickInputService: IQuickInputService
I
isidor 已提交
1061 1062 1063 1064
	) {
		super(actionId, actionLabel);
	}

J
Johannes Rieken 已提交
1065
	run(): Promise<any> {
1066 1067
		const activeTextEditorWidget = getCodeEditor(this.editorService.activeTextEditorWidget);
		if (!activeTextEditorWidget) {
C
Christof Marti 已提交
1068
			return this.quickInputService.pick([{ label: nls.localize('noEditor', "No text editor active at this time") }]);
I
isidor 已提交
1069
		}
B
Benjamin Pasero 已提交
1070

1071
		if (!isWritableCodeEditor(activeTextEditorWidget)) {
C
Christof Marti 已提交
1072
			return this.quickInputService.pick([{ label: nls.localize('noWritableCodeEditor', "The active code editor is read-only.") }]);
I
isidor 已提交
1073
		}
1074

C
Christof Marti 已提交
1075
		const picks: QuickPickInput<IQuickPickItem & { run(): void }>[] = [
1076 1077 1078 1079 1080 1081
			activeTextEditorWidget.getAction(IndentUsingSpaces.ID),
			activeTextEditorWidget.getAction(IndentUsingTabs.ID),
			activeTextEditorWidget.getAction(DetectIndentation.ID),
			activeTextEditorWidget.getAction(IndentationToSpacesAction.ID),
			activeTextEditorWidget.getAction(IndentationToTabsAction.ID),
			activeTextEditorWidget.getAction(TrimTrailingWhitespaceAction.ID)
A
Alex Dima 已提交
1082
		].map((a: IEditorAction) => {
B
Benjamin Pasero 已提交
1083 1084
			return {
				id: a.id,
B
Benjamin Pasero 已提交
1085
				label: a.label,
M
Matt Bierner 已提交
1086
				detail: (language === LANGUAGE_DEFAULT) ? undefined : a.alias,
1087
				run: () => {
1088
					activeTextEditorWidget.focus();
1089 1090
					a.run();
				}
B
Benjamin Pasero 已提交
1091 1092 1093
			};
		});

C
Christof Marti 已提交
1094
		picks.splice(3, 0, { type: 'separator', label: nls.localize('indentConvert', "convert file") });
C
Christof Marti 已提交
1095
		picks.unshift({ type: 'separator', label: nls.localize('indentView', "change view") });
I
isidor 已提交
1096

C
Christof Marti 已提交
1097
		return this.quickInputService.pick(picks, { placeHolder: nls.localize('pickAction', "Select Action"), matchOnDetail: true }).then(action => action && action.run());
I
isidor 已提交
1098 1099 1100
	}
}

E
Erich Gamma 已提交
1101 1102
export class ChangeEOLAction extends Action {

B
Benjamin Pasero 已提交
1103 1104
	static readonly ID = 'workbench.action.editor.changeEOL';
	static readonly LABEL = nls.localize('changeEndOfLine', "Change End of Line Sequence");
E
Erich Gamma 已提交
1105 1106 1107 1108

	constructor(
		actionId: string,
		actionLabel: string,
1109 1110
		@IEditorService private readonly editorService: IEditorService,
		@IQuickInputService private readonly quickInputService: IQuickInputService
E
Erich Gamma 已提交
1111 1112 1113 1114
	) {
		super(actionId, actionLabel);
	}

J
Johannes Rieken 已提交
1115
	run(): Promise<any> {
1116 1117
		const activeTextEditorWidget = getCodeEditor(this.editorService.activeTextEditorWidget);
		if (!activeTextEditorWidget) {
C
Christof Marti 已提交
1118
			return this.quickInputService.pick([{ label: nls.localize('noEditor', "No text editor active at this time") }]);
E
Erich Gamma 已提交
1119 1120
		}

1121
		if (!isWritableCodeEditor(activeTextEditorWidget)) {
C
Christof Marti 已提交
1122
			return this.quickInputService.pick([{ label: nls.localize('noWritableCodeEditor', "The active code editor is read-only.") }]);
E
Erich Gamma 已提交
1123 1124
		}

1125
		const textModel = activeTextEditorWidget.getModel();
E
Erich Gamma 已提交
1126

B
Benjamin Pasero 已提交
1127
		const EOLOptions: IChangeEOLEntry[] = [
1128 1129
			{ label: nlsEOLLF, eol: EndOfLineSequence.LF },
			{ label: nlsEOLCRLF, eol: EndOfLineSequence.CRLF },
E
Erich Gamma 已提交
1130 1131
		];

1132
		const selectedIndex = (textModel && textModel.getEOL() === '\n') ? 0 : 1;
E
Erich Gamma 已提交
1133

C
Christof Marti 已提交
1134
		return this.quickInputService.pick(EOLOptions, { placeHolder: nls.localize('pickEndOfLine', "Select End of Line Sequence"), activeItem: EOLOptions[selectedIndex] }).then(eol => {
E
Erich Gamma 已提交
1135
			if (eol) {
1136
				const activeCodeEditor = getCodeEditor(this.editorService.activeTextEditorWidget);
1137
				if (activeCodeEditor && activeCodeEditor.hasModel() && isWritableCodeEditor(activeCodeEditor)) {
1138
					const textModel = activeCodeEditor.getModel();
A
Alex Dima 已提交
1139
					textModel.pushEOL(eol.eol);
E
Erich Gamma 已提交
1140 1141 1142 1143 1144 1145 1146 1147
				}
			}
		});
	}
}

export class ChangeEncodingAction extends Action {

B
Benjamin Pasero 已提交
1148 1149
	static readonly ID = 'workbench.action.editor.changeEncoding';
	static readonly LABEL = nls.localize('changeEncoding', "Change File Encoding");
E
Erich Gamma 已提交
1150 1151 1152 1153

	constructor(
		actionId: string,
		actionLabel: string,
1154 1155 1156 1157
		@IEditorService private readonly editorService: IEditorService,
		@IQuickInputService private readonly quickInputService: IQuickInputService,
		@ITextResourceConfigurationService private readonly textResourceConfigurationService: ITextResourceConfigurationService,
		@IFileService private readonly fileService: IFileService
E
Erich Gamma 已提交
1158 1159 1160 1161
	) {
		super(actionId, actionLabel);
	}

J
Johannes Rieken 已提交
1162
	run(): Promise<any> {
1163
		if (!getCodeEditor(this.editorService.activeTextEditorWidget)) {
C
Christof Marti 已提交
1164
			return this.quickInputService.pick([{ label: nls.localize('noEditor', "No text editor active at this time") }]);
E
Erich Gamma 已提交
1165 1166
		}

M
Matt Bierner 已提交
1167
		const activeControl = this.editorService.activeControl;
M
Matt Bierner 已提交
1168 1169 1170
		if (!activeControl) {
			return this.quickInputService.pick([{ label: nls.localize('noEditor', "No text editor active at this time") }]);
		}
M
Matt Bierner 已提交
1171
		const encodingSupport: IEncodingSupport | null = toEditorWithEncodingSupport(activeControl.input);
1172
		if (!encodingSupport) {
C
Christof Marti 已提交
1173
			return this.quickInputService.pick([{ label: nls.localize('noFileEditor', "No file active at this time") }]);
E
Erich Gamma 已提交
1174 1175
		}

C
Christof Marti 已提交
1176 1177
		let saveWithEncodingPick: IQuickPickItem;
		let reopenWithEncodingPick: IQuickPickItem;
B
Benjamin Pasero 已提交
1178 1179 1180 1181
		if (language === LANGUAGE_DEFAULT) {
			saveWithEncodingPick = { label: nls.localize('saveWithEncoding', "Save with Encoding") };
			reopenWithEncodingPick = { label: nls.localize('reopenWithEncoding', "Reopen with Encoding") };
		} else {
B
Benjamin Pasero 已提交
1182 1183
			saveWithEncodingPick = { label: nls.localize('saveWithEncoding', "Save with Encoding"), detail: 'Save with Encoding', };
			reopenWithEncodingPick = { label: nls.localize('reopenWithEncoding', "Reopen with Encoding"), detail: 'Reopen with Encoding' };
B
Benjamin Pasero 已提交
1184
		}
E
Erich Gamma 已提交
1185

B
Benjamin Pasero 已提交
1186
		let pickActionPromise: Promise<IQuickPickItem>;
1187
		if (encodingSupport instanceof UntitledEditorInput) {
B
Benjamin Pasero 已提交
1188
			pickActionPromise = Promise.resolve(saveWithEncodingPick);
1189
		} else if (!isWritableBaseEditor(activeControl)) {
B
Benjamin Pasero 已提交
1190
			pickActionPromise = Promise.resolve(reopenWithEncodingPick);
E
Erich Gamma 已提交
1191
		} else {
C
Christof Marti 已提交
1192
			pickActionPromise = this.quickInputService.pick([reopenWithEncodingPick, saveWithEncodingPick], { placeHolder: nls.localize('pickAction', "Select Action"), matchOnDetail: true });
E
Erich Gamma 已提交
1193 1194
		}

1195
		return pickActionPromise.then(action => {
E
Erich Gamma 已提交
1196
			if (!action) {
R
Rob Lourens 已提交
1197
				return undefined;
E
Erich Gamma 已提交
1198 1199
			}

M
Matt Bierner 已提交
1200
			const resource = toResource(activeControl!.input, { supportSideBySide: true });
1201

1202
			return timeout(50 /* quick open is sensitive to being opened so soon after another */)
1203
				.then(() => {
1204
					if (!resource || !this.fileService.canHandleResource(resource)) {
B
Benjamin Pasero 已提交
1205
						return Promise.resolve(null); // encoding detection only possible for resources the file service can handle
1206 1207
					}

1208 1209
					return this.fileService.resolveContent(resource, { autoGuessEncoding: true, acceptTextOnly: true }).then(content => content.encoding, err => null);
				})
B
Benjamin Pasero 已提交
1210
				.then((guessedEncoding: string) => {
1211
					const isReopenWithEncoding = (action === reopenWithEncodingPick);
1212

M
Matt Bierner 已提交
1213
					const configuredEncoding = this.textResourceConfigurationService.getValue(types.withNullAsUndefined(resource), 'files.encoding');
1214

M
Matt Bierner 已提交
1215
					let directMatchIndex: number | undefined;
M
Matt Bierner 已提交
1216
					let aliasMatchIndex: number | undefined;
1217 1218

					// All encodings are valid picks
C
Christof Marti 已提交
1219
					const picks: QuickPickInput[] = Object.keys(SUPPORTED_ENCODINGS)
1220 1221 1222 1223 1224 1225 1226 1227 1228 1229
						.sort((k1, k2) => {
							if (k1 === configuredEncoding) {
								return -1;
							} else if (k2 === configuredEncoding) {
								return 1;
							}

							return SUPPORTED_ENCODINGS[k1].order - SUPPORTED_ENCODINGS[k2].order;
						})
						.filter(k => {
B
Benjamin Pasero 已提交
1230 1231 1232 1233
							if (k === guessedEncoding && guessedEncoding !== configuredEncoding) {
								return false; // do not show encoding if it is the guessed encoding that does not match the configured
							}

1234 1235 1236 1237 1238 1239 1240 1241 1242
							return !isReopenWithEncoding || !SUPPORTED_ENCODINGS[k].encodeOnly; // hide those that can only be used for encoding if we are about to decode
						})
						.map((key, index) => {
							if (key === encodingSupport.getEncoding()) {
								directMatchIndex = index;
							} else if (SUPPORTED_ENCODINGS[key].alias === encodingSupport.getEncoding()) {
								aliasMatchIndex = index;
							}

1243
							return { id: key, label: SUPPORTED_ENCODINGS[key].labelLong, description: key };
1244 1245
						});

C
Christof Marti 已提交
1246 1247
					const items = picks.slice() as IQuickPickItem[];

B
Benjamin Pasero 已提交
1248 1249
					// If we have a guessed encoding, show it first unless it matches the configured encoding
					if (guessedEncoding && configuredEncoding !== guessedEncoding && SUPPORTED_ENCODINGS[guessedEncoding]) {
C
Christof Marti 已提交
1250
						picks.unshift({ type: 'separator' });
B
Benjamin Pasero 已提交
1251
						picks.unshift({ id: guessedEncoding, label: SUPPORTED_ENCODINGS[guessedEncoding].labelLong, description: nls.localize('guessedEncoding', "Guessed from content") });
1252
					}
1253

C
Christof Marti 已提交
1254
					return this.quickInputService.pick(picks, {
1255
						placeHolder: isReopenWithEncoding ? nls.localize('pickEncodingForReopen', "Select File Encoding to Reopen File") : nls.localize('pickEncodingForSave', "Select File Encoding to Save with"),
C
Christof Marti 已提交
1256
						activeItem: items[typeof directMatchIndex === 'number' ? directMatchIndex : typeof aliasMatchIndex === 'number' ? aliasMatchIndex : -1]
1257
					}).then(encoding => {
M
Matt Bierner 已提交
1258 1259 1260 1261 1262 1263 1264 1265 1266 1267
						if (!encoding) {
							return;
						}
						const activeControl = this.editorService.activeControl;
						if (!activeControl) {
							return;
						}
						const encodingSupport = toEditorWithEncodingSupport(activeControl.input);
						if (typeof encoding.id !== 'undefined' && encodingSupport && encodingSupport.getEncoding() !== encoding.id) {
							encodingSupport.setEncoding(encoding.id, isReopenWithEncoding ? EncodingMode.Decode : EncodingMode.Encode); // Set new encoding
1268
						}
1269
					});
E
Erich Gamma 已提交
1270 1271 1272
				});
		});
	}
J
Johannes Rieken 已提交
1273
}