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 } 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';
M
Matt Bierner 已提交
18
import { IDisposable, combinedDisposable, dispose, toDisposable } from 'vs/base/common/lifecycle';
J
Johannes Rieken 已提交
19
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(
M
Matt Bierner 已提交
357 358 359 360
			toDisposable(() => {
				if (this.delayedRender) {
					this.delayedRender.dispose();
					this.delayedRender = null;
361
				}
M
Matt Bierner 已提交
362
			}),
363
			this.editorService.onDidActiveEditorChange(() => this.updateStatusBar()),
364 365
			this.untitledEditorService.onDidChangeEncoding(r => this.onResourceEncodingChange(r)),
			this.textFileService.models.onModelEncodingChanged(e => this.onResourceEncodingChange(e.resource)),
366
			TabFocus.onDidChangeTabFocus(e => this.onTabFocusModeChange()),
367
		);
E
Erich Gamma 已提交
368

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

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

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

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

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

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

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

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

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

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

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

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

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

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

		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 已提交
479
		return undefined;
E
Erich Gamma 已提交
480 481
	}

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

485
		action.run();
486 487 488
		action.dispose();
	}

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

495
	private onScreenReaderModeClick(): void {
496 497 498
		if (!this.screenReaderNotification) {
			this.screenReaderNotification = this.notificationService.prompt(
				Severity.Info,
A
Alex Dima 已提交
499
				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)"),
500 501 502 503 504 505 506 507 508 509
				[{
					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 已提交
510 511
				}],
				{ sticky: true }
512
			);
B
Benjamin Pasero 已提交
513

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

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

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

527
		action.run();
528 529 530 531
		action.dispose();
	}

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

534
		action.run();
535 536 537 538
		action.dispose();
	}

	private onTabFocusModeClick(): void {
539
		TabFocus.setTabFocusMode(false);
540 541
	}

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

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

555 556 557 558
		// Dispose old active editor listeners
		dispose(this.activeEditorListeners);

		// Attach new listeners to active editor
559
		if (activeCodeEditor) {
560

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

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

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

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

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

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

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

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

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

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

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

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

		this.updateState(info);
	}

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

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

I
isidor 已提交
659 660 661
		this.updateState(update);
	}

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

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

		this.updateState(update);
	}

672 673
	private _promptedScreenReader: boolean = false;

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

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

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

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

700 701 702
		this.updateState({ screenReaderMode: screenReaderMode });
	}

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

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

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

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

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

725 726 727 728 729 730 731
				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 已提交
732 733 734 735 736

				info.selections[0] = selectionClone;
			}
		}

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

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

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

		this.updateState(info);
	}

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

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

		// We only support text based editors
761
		if (e && (isCodeEditor(e.getControl()) || isDiffEditor(e.getControl()))) {
M
Matt Bierner 已提交
762
			const encodingSupport: IEncodingSupport | null = e.input ? toEditorWithEncodingSupport(e.input) : null;
763
			if (encodingSupport) {
B
Benjamin Pasero 已提交
764 765
				const rawEncoding = encodingSupport.getEncoding();
				const encodingInfo = SUPPORTED_ENCODINGS[rawEncoding];
E
Erich Gamma 已提交
766 767 768 769 770 771 772 773 774 775 776
				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);
	}

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

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

		this.updateState(info);
	}

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

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

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

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

812 813
export class ShowLanguageExtensionsAction extends Action {

814
	static readonly ID = 'workbench.action.showLanguageExtensions';
815 816

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

823 824 825
		this.enabled = galleryService.isEnabled();
	}

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

E
Erich Gamma 已提交
831 832
export class ChangeModeAction extends Action {

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

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

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

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

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

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

		// All languages are valid picks
874
		const languages = this.modeService.getRegisteredLanguageNames();
C
Christof Marti 已提交
875
		const picks: QuickPickInput[] = languages.sort().map((lang, index) => {
B
Benjamin Pasero 已提交
876 877 878 879 880 881
			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()));
			}
882

883
			// construct a fake resource to be able to show nice icons if any
M
Matt Bierner 已提交
884
			let fakeResource: uri | undefined;
885 886 887 888 889 890 891 892 893 894
			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 已提交
895
			return <IQuickPickItem>{
896
				label: lang,
C
Christof Marti 已提交
897
				iconClasses: getIconClasses(this.modelService, this.modeService, fakeResource),
B
Benjamin Pasero 已提交
898
				description
E
Erich Gamma 已提交
899 900
			};
		});
901

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

1028 1029 1030 1031 1032 1033 1034
					// 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 已提交
1035
					const currentAssociations = deepClone((target === ConfigurationTarget.WORKSPACE) ? fileAssociationsConfig.workspace : fileAssociationsConfig.user) || Object.create(null);
1036 1037
					currentAssociations[associationKey] = language.id;

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

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

I
isidor 已提交
1049
class ChangeIndentationAction extends Action {
I
isidor 已提交
1050

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

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

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

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

C
Christof Marti 已提交
1073
		const picks: QuickPickInput<IQuickPickItem & { run(): void }>[] = [
1074 1075 1076 1077 1078 1079
			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 已提交
1080
		].map((a: IEditorAction) => {
B
Benjamin Pasero 已提交
1081 1082
			return {
				id: a.id,
B
Benjamin Pasero 已提交
1083
				label: a.label,
1084
				detail: Language.isDefaultVariant() ? undefined : a.alias,
1085
				run: () => {
1086
					activeTextEditorWidget.focus();
1087 1088
					a.run();
				}
B
Benjamin Pasero 已提交
1089 1090 1091
			};
		});

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

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

E
Erich Gamma 已提交
1099 1100
export class ChangeEOLAction extends Action {

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

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

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

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

1123
		const textModel = activeTextEditorWidget.getModel();
E
Erich Gamma 已提交
1124

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

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

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

export class ChangeEncodingAction extends Action {

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

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

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

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

C
Christof Marti 已提交
1174 1175
		let saveWithEncodingPick: IQuickPickItem;
		let reopenWithEncodingPick: IQuickPickItem;
1176
		if (Language.isDefaultVariant()) {
B
Benjamin Pasero 已提交
1177 1178 1179
			saveWithEncodingPick = { label: nls.localize('saveWithEncoding', "Save with Encoding") };
			reopenWithEncodingPick = { label: nls.localize('reopenWithEncoding', "Reopen with Encoding") };
		} else {
B
Benjamin Pasero 已提交
1180 1181
			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 已提交
1182
		}
E
Erich Gamma 已提交
1183

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

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

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

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

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

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

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

					// All encodings are valid picks
C
Christof Marti 已提交
1217
					const picks: QuickPickInput[] = Object.keys(SUPPORTED_ENCODINGS)
1218 1219 1220 1221 1222 1223 1224 1225 1226 1227
						.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 已提交
1228 1229 1230 1231
							if (k === guessedEncoding && guessedEncoding !== configuredEncoding) {
								return false; // do not show encoding if it is the guessed encoding that does not match the configured
							}

1232 1233 1234 1235 1236 1237 1238 1239 1240
							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;
							}

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

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

B
Benjamin Pasero 已提交
1246 1247
					// 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 已提交
1248
						picks.unshift({ type: 'separator' });
B
Benjamin Pasero 已提交
1249
						picks.unshift({ id: guessedEncoding, label: SUPPORTED_ENCODINGS[guessedEncoding].labelLong, description: nls.localize('guessedEncoding', "Guessed from content") });
1250
					}
1251

C
Christof Marti 已提交
1252
					return this.quickInputService.pick(picks, {
1253
						placeHolder: isReopenWithEncoding ? nls.localize('pickEncodingForReopen', "Select File Encoding to Reopen File") : nls.localize('pickEncodingForSave', "Select File Encoding to Save with"),
C
Christof Marti 已提交
1254
						activeItem: items[typeof directMatchIndex === 'number' ? directMatchIndex : typeof aliasMatchIndex === 'number' ? aliasMatchIndex : -1]
1255
					}).then(encoding => {
M
Matt Bierner 已提交
1256 1257 1258 1259 1260 1261 1262 1263 1264 1265
						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
1266
						}
1267
					});
E
Erich Gamma 已提交
1268 1269 1270
				});
		});
	}
J
Johannes Rieken 已提交
1271
}