editorStatus.ts 45.9 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
	}
E
Erich Gamma 已提交
122 123
}

124 125 126 127 128
interface StateDelta {
	selectionStatus?: string;
	mode?: string;
	encoding?: string;
	EOL?: string;
129
	indentation?: string;
130
	tabFocusMode?: boolean;
131
	screenReaderMode?: boolean;
132
	metadata?: string | null;
133 134 135
}

class State {
136 137
	private _selectionStatus: string | null | undefined;
	get selectionStatus(): string | null | undefined { return this._selectionStatus; }
138

139 140
	private _mode: string | null | undefined;
	get mode(): string | null | undefined { return this._mode; }
141

142 143
	private _encoding: string | null | undefined;
	get encoding(): string | null | undefined { return this._encoding; }
144

145 146
	private _EOL: string | null | undefined;
	get EOL(): string | null | undefined { return this._EOL; }
147

148 149
	private _indentation: string | null | undefined;
	get indentation(): string | null | undefined { return this._indentation; }
I
isidor 已提交
150

151 152
	private _tabFocusMode: boolean | null | undefined;
	get tabFocusMode(): boolean | null | undefined { return this._tabFocusMode; }
153

154 155
	private _screenReaderMode: boolean | null | undefined;
	get screenReaderMode(): boolean | null | undefined { return this._screenReaderMode; }
156

157 158
	private _metadata: string | null | undefined;
	get metadata(): string | null | undefined { return this._metadata; }
159

160 161 162 163 164 165
	constructor() {
		this._selectionStatus = null;
		this._mode = null;
		this._encoding = null;
		this._EOL = null;
		this._tabFocusMode = false;
166
		this._screenReaderMode = false;
167
		this._metadata = null;
168 169
	}

M
Matt Bierner 已提交
170
	update(update: StateDelta): StateChange | null {
B
Benjamin Pasero 已提交
171
		const e = new StateChange();
172 173
		let somethingChanged = false;

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

		if (somethingChanged) {
			return e;
		}
		return null;
	}
}

238 239 240 241 242 243
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 已提交
244
const nlsTabFocusMode = nls.localize('tabFocusModeEnabled', "Tab Moves Focus");
245
const nlsScreenReaderDetected = nls.localize('screenReaderDetected', "Screen Reader Optimized");
A
Alex Dima 已提交
246
const nlsScreenReaderDetectedTitle = nls.localize('screenReaderDetectedExtra', "If you are not using a Screen Reader, please change the setting `editor.accessibilitySupport` to \"off\".");
E
Erich Gamma 已提交
247

248
function setDisplay(el: HTMLElement, desiredValue: string): void {
A
Alex Dima 已提交
249 250 251 252
	if (el.style.display !== desiredValue) {
		el.style.display = desiredValue;
	}
}
253
function show(el: HTMLElement): void {
254
	setDisplay(el, '');
A
Alex Dima 已提交
255
}
256
function hide(el: HTMLElement): void {
257
	setDisplay(el, 'none');
A
Alex Dima 已提交
258 259
}

260
export class EditorStatus implements IStatusbarItem {
261
	private state: State;
262 263
	private element: HTMLElement;
	private tabFocusModeElement: HTMLElement;
264
	private screenRedearModeElement: HTMLElement;
I
isidor 已提交
265
	private indentationElement: HTMLElement;
266 267 268 269
	private selectionElement: HTMLElement;
	private encodingElement: HTMLElement;
	private eolElement: HTMLElement;
	private modeElement: HTMLElement;
270
	private metadataElement: HTMLElement;
E
Erich Gamma 已提交
271
	private toDispose: IDisposable[];
272
	private activeEditorListeners: IDisposable[];
M
Matt Bierner 已提交
273 274 275
	private delayedRender: IDisposable | null;
	private toRender: StateChange | null;
	private screenReaderNotification: INotificationHandle | null;
E
Erich Gamma 已提交
276

277
	constructor(
278 279 280 281 282 283
		@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,
284
		@IConfigurationService private readonly configurationService: IConfigurationService,
285 286
		@INotificationService private readonly notificationService: INotificationService,
		@IAccessibilityService private readonly accessibilityService: IAccessibilityService
287 288
	) {
		this.toDispose = [];
289
		this.activeEditorListeners = [];
290
		this.state = new State();
E
Erich Gamma 已提交
291 292
	}

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

296
		this.tabFocusModeElement = append(this.element, $('a.editor-status-tabfocusmode.status-bar-info'));
297 298 299
		this.tabFocusModeElement.title = nls.localize('disableTabMode', "Disable Accessibility Mode");
		this.tabFocusModeElement.onclick = () => this.onTabFocusModeClick();
		this.tabFocusModeElement.textContent = nlsTabFocusMode;
300
		hide(this.tabFocusModeElement);
301

302 303
		this.screenRedearModeElement = append(this.element, $('a.editor-status-screenreadermode.status-bar-info'));
		this.screenRedearModeElement.textContent = nlsScreenReaderDetected;
304
		this.screenRedearModeElement.title = nlsScreenReaderDetectedTitle;
305
		this.screenRedearModeElement.onclick = () => this.onScreenReaderModeClick();
306 307
		hide(this.screenRedearModeElement);

308 309 310
		this.selectionElement = append(this.element, $('a.editor-status-selection'));
		this.selectionElement.title = nls.localize('gotoLine', "Go to Line");
		this.selectionElement.onclick = () => this.onSelectionClick();
311
		hide(this.selectionElement);
312

313
		this.indentationElement = append(this.element, $('a.editor-status-indentation'));
314
		this.indentationElement.title = nls.localize('selectIndentation', "Select Indentation");
315 316 317
		this.indentationElement.onclick = () => this.onIndentationClick();
		hide(this.indentationElement);

318 319 320
		this.encodingElement = append(this.element, $('a.editor-status-encoding'));
		this.encodingElement.title = nls.localize('selectEncoding', "Select Encoding");
		this.encodingElement.onclick = () => this.onEncodingClick();
321
		hide(this.encodingElement);
322 323 324 325

		this.eolElement = append(this.element, $('a.editor-status-eol'));
		this.eolElement.title = nls.localize('selectEOL', "Select End of Line Sequence");
		this.eolElement.onclick = () => this.onEOLClick();
326
		hide(this.eolElement);
327 328 329 330

		this.modeElement = append(this.element, $('a.editor-status-mode'));
		this.modeElement.title = nls.localize('selectLanguageMode', "Select Language Mode");
		this.modeElement.onclick = () => this.onModeClick();
331
		hide(this.modeElement);
332

333 334 335 336
		this.metadataElement = append(this.element, $('span.editor-status-metadata'));
		this.metadataElement.title = nls.localize('fileInfo', "File Information");
		hide(this.metadataElement);

337 338 339
		this.delayedRender = null;
		this.toRender = null;

340
		this.toDispose.push(
341 342 343 344 345 346 347 348
			{
				dispose: () => {
					if (this.delayedRender) {
						this.delayedRender.dispose();
						this.delayedRender = null;
					}
				}
			},
349
			this.editorService.onDidActiveEditorChange(() => this.updateStatusBar()),
350 351
			this.untitledEditorService.onDidChangeEncoding(r => this.onResourceEncodingChange(r)),
			this.textFileService.models.onModelEncodingChanged(e => this.onResourceEncodingChange(e.resource)),
352
			TabFocus.onDidChangeTabFocus(e => this.onTabFocusModeChange()),
353
		);
E
Erich Gamma 已提交
354

J
Joao Moreno 已提交
355
		return combinedDisposable(this.toDispose);
356 357
	}

358
	private updateState(update: StateDelta): void {
B
Benjamin Pasero 已提交
359
		const changed = this.state.update(update);
360 361 362
		if (!changed) {
			// Nothing really changed
			return;
E
Erich Gamma 已提交
363 364
		}

365 366 367 368
		if (!this.toRender) {
			this.toRender = changed;
			this.delayedRender = runAtThisOrScheduleAtNextAnimationFrame(() => {
				this.delayedRender = null;
B
Benjamin Pasero 已提交
369
				const toRender = this.toRender;
370
				this.toRender = null;
M
Matt Bierner 已提交
371 372 373
				if (toRender) {
					this._renderNow(toRender);
				}
374 375 376 377 378 379
			});
		} else {
			this.toRender.combine(changed);
		}
	}

380
	private _renderNow(changed: StateChange): void {
381 382 383 384 385 386
		if (changed.tabFocusMode) {
			if (this.state.tabFocusMode && this.state.tabFocusMode === true) {
				show(this.tabFocusModeElement);
			} else {
				hide(this.tabFocusModeElement);
			}
E
Erich Gamma 已提交
387 388
		}

389 390 391 392 393 394 395 396
		if (changed.screenReaderMode) {
			if (this.state.screenReaderMode && this.state.screenReaderMode === true) {
				show(this.screenRedearModeElement);
			} else {
				hide(this.screenRedearModeElement);
			}
		}

I
isidor 已提交
397 398 399 400 401 402 403 404 405
		if (changed.indentation) {
			if (this.state.indentation) {
				this.indentationElement.textContent = this.state.indentation;
				show(this.indentationElement);
			} else {
				hide(this.indentationElement);
			}
		}

406
		if (changed.selectionStatus) {
407
			if (this.state.selectionStatus && !this.state.screenReaderMode) {
408 409 410 411 412
				this.selectionElement.textContent = this.state.selectionStatus;
				show(this.selectionElement);
			} else {
				hide(this.selectionElement);
			}
E
Erich Gamma 已提交
413 414
		}

415 416 417 418 419 420 421
		if (changed.encoding) {
			if (this.state.encoding) {
				this.encodingElement.textContent = this.state.encoding;
				show(this.encodingElement);
			} else {
				hide(this.encodingElement);
			}
E
Erich Gamma 已提交
422 423
		}

424 425 426 427 428 429 430
		if (changed.EOL) {
			if (this.state.EOL) {
				this.eolElement.textContent = this.state.EOL === '\r\n' ? nlsEOLCRLF : nlsEOLLF;
				show(this.eolElement);
			} else {
				hide(this.eolElement);
			}
E
Erich Gamma 已提交
431 432
		}

433 434 435 436 437 438 439 440
		if (changed.mode) {
			if (this.state.mode) {
				this.modeElement.textContent = this.state.mode;
				show(this.modeElement);
			} else {
				hide(this.modeElement);
			}
		}
441 442 443 444 445 446 447 448 449

		if (changed.metadata) {
			if (this.state.metadata) {
				this.metadataElement.textContent = this.state.metadata;
				show(this.metadataElement);
			} else {
				hide(this.metadataElement);
			}
		}
E
Erich Gamma 已提交
450 451
	}

M
Matt Bierner 已提交
452
	private getSelectionLabel(info: IEditorSelectionStatus): string | undefined {
E
Erich Gamma 已提交
453
		if (!info || !info.selections) {
M
Matt Bierner 已提交
454
			return undefined;
E
Erich Gamma 已提交
455 456 457 458
		}

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

			return strings.format(nlsSingleSelection, info.selections[0].positionLineNumber, info.selections[0].positionColumn);
E
Erich Gamma 已提交
463
		}
B
Benjamin Pasero 已提交
464 465 466 467 468 469 470 471 472

		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 已提交
473
		return undefined;
E
Erich Gamma 已提交
474 475
	}

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

479
		action.run();
480 481 482
		action.dispose();
	}

I
isidor 已提交
483 484
	private onIndentationClick(): void {
		const action = this.instantiationService.createInstance(ChangeIndentationAction, ChangeIndentationAction.ID, ChangeIndentationAction.LABEL);
485
		action.run();
I
isidor 已提交
486 487 488
		action.dispose();
	}

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

J
Joao Moreno 已提交
508
			Event.once(this.screenReaderNotification.onDidClose)(() => {
B
Benjamin Pasero 已提交
509 510
				this.screenReaderNotification = null;
			});
511
		}
512 513
	}

514 515 516 517 518
	private onSelectionClick(): void {
		this.quickOpenService.show(':'); // "Go to line"
	}

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

521
		action.run();
522 523 524 525
		action.dispose();
	}

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

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

	private onTabFocusModeClick(): void {
533
		TabFocus.setTabFocusMode(false);
534 535
	}

536 537
	private updateStatusBar(): void {
		const activeControl = this.editorService.activeControl;
M
Matt Bierner 已提交
538
		const activeCodeEditor = activeControl ? types.withNullAsUndefined(getCodeEditor(activeControl.getControl())) : undefined;
539 540

		// Update all states
541 542 543 544 545 546 547
		this.onScreenReaderModeChange(activeCodeEditor);
		this.onSelectionChange(activeCodeEditor);
		this.onModeChange(activeCodeEditor);
		this.onEOLChange(activeCodeEditor);
		this.onEncodingChange(activeControl);
		this.onIndentationChange(activeCodeEditor);
		this.onMetadataChange(activeControl);
E
Erich Gamma 已提交
548

549 550 551 552
		// Dispose old active editor listeners
		dispose(this.activeEditorListeners);

		// Attach new listeners to active editor
553
		if (activeCodeEditor) {
554

555
			// Hook Listener for Configuration changes
556
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeConfiguration((event: IConfigurationChangedEvent) => {
557
				if (event.accessibilitySupport) {
558
					this.onScreenReaderModeChange(activeCodeEditor);
559 560 561
				}
			}));

562
			// Hook Listener for Selection changes
563 564
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeCursorPosition((event: ICursorPositionChangedEvent) => {
				this.onSelectionChange(activeCodeEditor);
565 566 567
			}));

			// Hook Listener for mode changes
568 569
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeModelLanguage((event: IModelLanguageChangedEvent) => {
				this.onModeChange(activeCodeEditor);
570 571 572
			}));

			// Hook Listener for content changes
573 574
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeModelContent((e) => {
				this.onEOLChange(activeCodeEditor);
575

M
Matt Bierner 已提交
576
				const selections = activeCodeEditor.getSelections();
M
Matt Bierner 已提交
577 578 579 580 581 582
				if (selections) {
					for (const change of e.changes) {
						if (selections.some(selection => Range.areIntersecting(selection, change.range))) {
							this.onSelectionChange(activeCodeEditor);
							break;
						}
583 584
					}
				}
585 586 587
			}));

			// Hook Listener for content options changes
588 589
			this.activeEditorListeners.push(activeCodeEditor.onDidChangeModelOptions((event: IModelOptionsChangedEvent) => {
				this.onIndentationChange(activeCodeEditor);
590
			}));
E
Erich Gamma 已提交
591
		}
592 593

		// Handle binary editors
594
		else if (activeControl instanceof BaseBinaryResourceEditor || activeControl instanceof BinaryResourceDiffEditor) {
595
			const binaryEditors: BaseBinaryResourceEditor[] = [];
596 597
			if (activeControl instanceof BinaryResourceDiffEditor) {
				const details = activeControl.getDetailsEditor();
598 599 600 601
				if (details instanceof BaseBinaryResourceEditor) {
					binaryEditors.push(details);
				}

602
				const master = activeControl.getMasterEditor();
603 604 605
				if (master instanceof BaseBinaryResourceEditor) {
					binaryEditors.push(master);
				}
606
			} else {
607
				binaryEditors.push(activeControl);
608 609 610 611
			}

			binaryEditors.forEach(editor => {
				this.activeEditorListeners.push(editor.onMetadataChanged(metadata => {
612
					this.onMetadataChange(activeControl);
613
				}));
614 615 616 617

				this.activeEditorListeners.push(editor.onDidOpenInPlace(() => {
					this.updateStatusBar();
				}));
618
			});
619
		}
620
	}
E
Erich Gamma 已提交
621

M
Matt Bierner 已提交
622 623
	private onModeChange(editorWidget: ICodeEditor | undefined): void {
		let info: StateDelta = { mode: undefined };
E
Erich Gamma 已提交
624 625

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

		this.updateState(info);
	}

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

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

I
isidor 已提交
653 654 655
		this.updateState(update);
	}

M
Matt Bierner 已提交
656
	private onMetadataChange(editor: IBaseEditor | undefined): void {
M
Matt Bierner 已提交
657
		const update: StateDelta = { metadata: undefined };
658 659

		if (editor instanceof BaseBinaryResourceEditor || editor instanceof BinaryResourceDiffEditor) {
660
			update.metadata = editor.getMetadata();
661 662 663 664 665
		}

		this.updateState(update);
	}

666 667
	private _promptedScreenReader: boolean = false;

M
Matt Bierner 已提交
668
	private onScreenReaderModeChange(editorWidget: ICodeEditor | undefined): void {
669 670 671 672
		let screenReaderMode = false;

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

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

690 691
		if (screenReaderMode === false && this.screenReaderNotification) {
			this.screenReaderNotification.close();
692 693
		}

694 695 696
		this.updateState({ screenReaderMode: screenReaderMode });
	}

M
Matt Bierner 已提交
697
	private onSelectionChange(editorWidget: ICodeEditor | undefined): void {
B
Benjamin Pasero 已提交
698
		const info: IEditorSelectionStatus = {};
E
Erich Gamma 已提交
699 700

		// We only support text based editors
701
		if (editorWidget) {
E
Erich Gamma 已提交
702 703 704 705 706 707

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

			// Compute selection length
			info.charactersSelected = 0;
A
Alex Dima 已提交
708
			const textModel = editorWidget.getModel();
E
Erich Gamma 已提交
709
			if (textModel) {
710
				info.selections.forEach(selection => {
M
Matt Bierner 已提交
711
					info.charactersSelected! += textModel.getValueLengthInRange(selection);
E
Erich Gamma 已提交
712 713 714 715 716
				});
			}

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

719 720 721 722 723 724 725
				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 已提交
726 727 728 729 730

				info.selections[0] = selectionClone;
			}
		}

731
		this.updateState({ selectionStatus: this.getSelectionLabel(info) });
E
Erich Gamma 已提交
732 733
	}

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

A
Alex Dima 已提交
737 738
		if (editorWidget && !editorWidget.getConfiguration().readOnly) {
			const codeEditorModel = editorWidget.getModel();
I
isidor 已提交
739 740 741
			if (codeEditorModel) {
				info.EOL = codeEditorModel.getEOL();
			}
E
Erich Gamma 已提交
742 743 744 745 746
		}

		this.updateState(info);
	}

747
	private onEncodingChange(e?: IBaseEditor): void {
E
Erich Gamma 已提交
748 749 750 751
		if (e && !this.isActiveEditor(e)) {
			return;
		}

M
Matt Bierner 已提交
752
		const info: StateDelta = { encoding: undefined };
E
Erich Gamma 已提交
753 754

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

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

781
	private onTabFocusModeChange(): void {
B
Benjamin Pasero 已提交
782
		const info: StateDelta = { tabFocusMode: TabFocus.getTabFocusMode() };
E
Erich Gamma 已提交
783 784 785 786

		this.updateState(info);
	}

787 788
	private isActiveEditor(control: IBaseEditor): boolean {
		const activeControl = this.editorService.activeControl;
E
Erich Gamma 已提交
789

790
		return !!activeControl && activeControl === control;
E
Erich Gamma 已提交
791 792 793
	}
}

M
Matt Bierner 已提交
794
function isWritableCodeEditor(codeEditor: ICodeEditor | undefined): boolean {
A
Alex Dima 已提交
795
	if (!codeEditor) {
S
Sandeep Somavarapu 已提交
796 797
		return false;
	}
A
Alex Dima 已提交
798 799 800
	const config = codeEditor.getConfiguration();
	return (!config.readOnly);
}
S
Sandeep Somavarapu 已提交
801

A
Alex Dima 已提交
802
function isWritableBaseEditor(e: IBaseEditor): boolean {
M
Matt Bierner 已提交
803
	return e && isWritableCodeEditor(getCodeEditor(e.getControl()) || undefined);
E
Erich Gamma 已提交
804 805
}

806 807
export class ShowLanguageExtensionsAction extends Action {

808
	static readonly ID = 'workbench.action.showLanguageExtensions';
809 810

	constructor(
811
		private fileExtension: string,
812
		@ICommandService private readonly commandService: ICommandService,
813 814
		@IExtensionGalleryService galleryService: IExtensionGalleryService
	) {
815 816
		super(ShowLanguageExtensionsAction.ID, nls.localize('showLanguageExtensions', "Search Marketplace Extensions for '{0}'...", fileExtension));

817 818 819
		this.enabled = galleryService.isEnabled();
	}

J
Johannes Rieken 已提交
820
	run(): Promise<void> {
R
Rob Lourens 已提交
821
		return this.commandService.executeCommand('workbench.extensions.action.showExtensionsForLanguage', this.fileExtension).then(() => undefined);
822 823 824
	}
}

E
Erich Gamma 已提交
825 826
export class ChangeModeAction extends Action {

B
Benjamin Pasero 已提交
827 828
	static readonly ID = 'workbench.action.editor.changeLanguageMode';
	static readonly LABEL = nls.localize('changeMode', "Change Language Mode");
E
Erich Gamma 已提交
829 830 831 832

	constructor(
		actionId: string,
		actionLabel: string,
833 834 835
		@IModeService private readonly modeService: IModeService,
		@IModelService private readonly modelService: IModelService,
		@IEditorService private readonly editorService: IEditorService,
836
		@IConfigurationService private readonly configurationService: IConfigurationService,
837 838 839 840
		@IQuickInputService private readonly quickInputService: IQuickInputService,
		@IPreferencesService private readonly preferencesService: IPreferencesService,
		@IInstantiationService private readonly instantiationService: IInstantiationService,
		@IUntitledEditorService private readonly untitledEditorService: IUntitledEditorService
E
Erich Gamma 已提交
841 842 843 844
	) {
		super(actionId, actionLabel);
	}

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

851
		const textModel = activeTextEditorWidget.getModel();
M
Matt Bierner 已提交
852
		const resource = this.editorService.activeEditor ? toResource(this.editorService.activeEditor, { supportSideBySide: true }) : null;
B
Benjamin Pasero 已提交
853 854

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

		// Compute mode
M
Matt Bierner 已提交
860
		let currentModeId: string | undefined;
861
		let modeId: string;
862
		if (textModel) {
863
			modeId = textModel.getLanguageIdentifier().language;
M
Matt Bierner 已提交
864
			currentModeId = this.modeService.getLanguageName(modeId) || undefined;
E
Erich Gamma 已提交
865 866 867
		}

		// All languages are valid picks
868
		const languages = this.modeService.getRegisteredLanguageNames();
C
Christof Marti 已提交
869
		const picks: QuickPickInput[] = languages.sort().map((lang, index) => {
B
Benjamin Pasero 已提交
870 871 872 873 874 875
			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()));
			}
876

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

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

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

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

912 913
			configureModeSettings = { label: nls.localize('configureModeSettings', "Configure '{0}' language based settings...", currentModeId) };
			picks.unshift(configureModeSettings);
914
			configureModeAssociations = { label: nls.localize('configureAssociationsExt', "Configure File Association for '{0}'...", ext) };
915 916
			picks.unshift(configureModeAssociations);
		}
E
Erich Gamma 已提交
917

918
		// Offer to "Auto Detect"
C
Christof Marti 已提交
919
		const autoDetectMode: IQuickPickItem = {
920 921
			label: nls.localize('autoDetect', "Auto Detect")
		};
922

B
Benjamin Pasero 已提交
923
		if (hasLanguageSupport) {
924 925
			picks.unshift(autoDetectMode);
		}
926

C
Christof Marti 已提交
927
		return this.quickInputService.pick(picks, { placeHolder: nls.localize('pickLanguage', "Select Language Mode"), matchOnDescription: true }).then(pick => {
928 929 930
			if (!pick) {
				return;
			}
B
Benjamin Pasero 已提交
931

932 933 934 935
			if (pick === galleryAction) {
				galleryAction.run();
				return;
			}
B
Benjamin Pasero 已提交
936

937 938
			// User decided to permanently configure associations, return right after
			if (pick === configureModeAssociations) {
M
Matt Bierner 已提交
939 940 941
				if (resource) {
					this.configureFileAssociation(resource);
				}
942 943
				return;
			}
944

945 946
			// User decided to configure settings for current language
			if (pick === configureModeSettings) {
947
				this.preferencesService.configureSettingsForLanguage(modeId);
948 949 950
				return;
			}

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

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

			// Change mode
M
Matt Bierner 已提交
986 987 988 989 990
			if (typeof languageSelection !== 'undefined') {
				for (const textModel of models) {
					this.modelService.setMode(textModel, languageSelection);
				}
			}
E
Erich Gamma 已提交
991 992
		});
	}
993 994

	private configureFileAssociation(resource: uri): void {
B
Benjamin Pasero 已提交
995 996 997
		const extension = extname(resource);
		const base = basename(resource);
		const currentAssociation = this.modeService.getModeIdByFilepathOrFirstLine(base);
998

999
		const languages = this.modeService.getRegisteredLanguageNames();
C
Christof Marti 已提交
1000
		const picks: IQuickPickItem[] = languages.sort().map((lang, index) => {
1001 1002
			const id = this.modeService.getModeIdForLanguageName(lang.toLowerCase());

C
Christof Marti 已提交
1003
			return <IQuickPickItem>{
1004 1005
				id,
				label: lang,
R
Rob Lourens 已提交
1006
				description: (id === currentAssociation) ? nls.localize('currentAssociation', "Current Association") : undefined
1007 1008 1009
			};
		});

1010
		setTimeout(() => {
B
Benjamin Pasero 已提交
1011
			this.quickInputService.pick(picks, { placeHolder: nls.localize('pickLanguageToConfigure', "Select Language Mode to Associate with '{0}'", extension || base) }).then(language => {
1012
				if (language) {
1013
					const fileAssociationsConfig = this.configurationService.inspect(FILES_ASSOCIATIONS_CONFIG);
1014 1015

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

1022 1023 1024 1025 1026 1027 1028
					// 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 已提交
1029
					const currentAssociations = deepClone((target === ConfigurationTarget.WORKSPACE) ? fileAssociationsConfig.workspace : fileAssociationsConfig.user) || Object.create(null);
1030 1031
					currentAssociations[associationKey] = language.id;

1032
					this.configurationService.updateValue(FILES_ASSOCIATIONS_CONFIG, currentAssociations, target);
1033
				}
1034
			});
1035
		}, 50 /* quick open is sensitive to being opened so soon after another */);
1036
	}
E
Erich Gamma 已提交
1037 1038
}

C
Christof Marti 已提交
1039
export interface IChangeEOLEntry extends IQuickPickItem {
E
Erich Gamma 已提交
1040 1041 1042
	eol: EndOfLineSequence;
}

I
isidor 已提交
1043
class ChangeIndentationAction extends Action {
I
isidor 已提交
1044

B
Benjamin Pasero 已提交
1045 1046
	static readonly ID = 'workbench.action.editor.changeIndentation';
	static readonly LABEL = nls.localize('changeIndentation', "Change Indentation");
I
isidor 已提交
1047 1048 1049 1050

	constructor(
		actionId: string,
		actionLabel: string,
1051 1052
		@IEditorService private readonly editorService: IEditorService,
		@IQuickInputService private readonly quickInputService: IQuickInputService
I
isidor 已提交
1053 1054 1055 1056
	) {
		super(actionId, actionLabel);
	}

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

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

C
Christof Marti 已提交
1067
		const picks: QuickPickInput<IQuickPickItem & { run(): void }>[] = [
1068 1069 1070 1071 1072 1073
			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 已提交
1074
		].map((a: IEditorAction) => {
B
Benjamin Pasero 已提交
1075 1076
			return {
				id: a.id,
B
Benjamin Pasero 已提交
1077
				label: a.label,
M
Matt Bierner 已提交
1078
				detail: (language === LANGUAGE_DEFAULT) ? undefined : a.alias,
1079
				run: () => {
1080
					activeTextEditorWidget.focus();
1081 1082
					a.run();
				}
B
Benjamin Pasero 已提交
1083 1084 1085
			};
		});

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

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

E
Erich Gamma 已提交
1093 1094
export class ChangeEOLAction extends Action {

B
Benjamin Pasero 已提交
1095 1096
	static readonly ID = 'workbench.action.editor.changeEOL';
	static readonly LABEL = nls.localize('changeEndOfLine', "Change End of Line Sequence");
E
Erich Gamma 已提交
1097 1098 1099 1100

	constructor(
		actionId: string,
		actionLabel: string,
1101 1102
		@IEditorService private readonly editorService: IEditorService,
		@IQuickInputService private readonly quickInputService: IQuickInputService
E
Erich Gamma 已提交
1103 1104 1105 1106
	) {
		super(actionId, actionLabel);
	}

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

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

1117
		const textModel = activeTextEditorWidget.getModel();
E
Erich Gamma 已提交
1118

B
Benjamin Pasero 已提交
1119
		const EOLOptions: IChangeEOLEntry[] = [
1120 1121
			{ label: nlsEOLLF, eol: EndOfLineSequence.LF },
			{ label: nlsEOLCRLF, eol: EndOfLineSequence.CRLF },
E
Erich Gamma 已提交
1122 1123
		];

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

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

export class ChangeEncodingAction extends Action {

B
Benjamin Pasero 已提交
1140 1141
	static readonly ID = 'workbench.action.editor.changeEncoding';
	static readonly LABEL = nls.localize('changeEncoding', "Change File Encoding");
E
Erich Gamma 已提交
1142 1143 1144 1145

	constructor(
		actionId: string,
		actionLabel: string,
1146 1147 1148 1149
		@IEditorService private readonly editorService: IEditorService,
		@IQuickInputService private readonly quickInputService: IQuickInputService,
		@ITextResourceConfigurationService private readonly textResourceConfigurationService: ITextResourceConfigurationService,
		@IFileService private readonly fileService: IFileService
E
Erich Gamma 已提交
1150 1151 1152 1153
	) {
		super(actionId, actionLabel);
	}

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

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

C
Christof Marti 已提交
1168 1169
		let saveWithEncodingPick: IQuickPickItem;
		let reopenWithEncodingPick: IQuickPickItem;
B
Benjamin Pasero 已提交
1170 1171 1172 1173
		if (language === LANGUAGE_DEFAULT) {
			saveWithEncodingPick = { label: nls.localize('saveWithEncoding', "Save with Encoding") };
			reopenWithEncodingPick = { label: nls.localize('reopenWithEncoding', "Reopen with Encoding") };
		} else {
B
Benjamin Pasero 已提交
1174 1175
			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 已提交
1176
		}
E
Erich Gamma 已提交
1177

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

1187
		return pickActionPromise.then(action => {
E
Erich Gamma 已提交
1188
			if (!action) {
R
Rob Lourens 已提交
1189
				return undefined;
E
Erich Gamma 已提交
1190 1191
			}

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

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

1200 1201
					return this.fileService.resolveContent(resource, { autoGuessEncoding: true, acceptTextOnly: true }).then(content => content.encoding, err => null);
				})
B
Benjamin Pasero 已提交
1202
				.then((guessedEncoding: string) => {
1203
					const isReopenWithEncoding = (action === reopenWithEncodingPick);
1204

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

M
Matt Bierner 已提交
1207
					let directMatchIndex: number | undefined;
M
Matt Bierner 已提交
1208
					let aliasMatchIndex: number | undefined;
1209 1210

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

1226 1227 1228 1229 1230 1231 1232 1233 1234
							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;
							}

1235
							return { id: key, label: SUPPORTED_ENCODINGS[key].labelLong, description: key };
1236 1237
						});

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

B
Benjamin Pasero 已提交
1240 1241
					// 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 已提交
1242
						picks.unshift({ type: 'separator' });
B
Benjamin Pasero 已提交
1243
						picks.unshift({ id: guessedEncoding, label: SUPPORTED_ENCODINGS[guessedEncoding].labelLong, description: nls.localize('guessedEncoding', "Guessed from content") });
1244
					}
1245

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