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

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

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

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

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

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

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

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

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

		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();
325
		hide(this.eolElement);
326 327 328 329

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

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

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

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

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

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

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

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

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

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

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

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

423 424 425 426 427 428 429
		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 已提交
430 431
		}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

		this.updateState(info);
	}

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

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

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

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

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

		this.updateState(update);
	}

665 666
	private _promptedScreenReader: boolean = false;

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

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

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

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

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

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

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

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

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

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

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

				info.selections[0] = selectionClone;
			}
		}

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

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

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

		this.updateState(info);
	}

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

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

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

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

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

		this.updateState(info);
	}

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

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

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

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

805 806
export class ShowLanguageExtensionsAction extends Action {

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

export class ChangeEncodingAction extends Action {

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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