toggleWordWrap.ts 11.0 KB
Newer Older
I
ivanixgames 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

A
Alex Dima 已提交
6
import * as nls from 'vs/nls';
J
Johannes Rieken 已提交
7
import { KeyCode, KeyMod } from 'vs/base/common/keyCodes';
A
Alex Dima 已提交
8
import { Disposable } from 'vs/base/common/lifecycle';
9
import { URI } from 'vs/base/common/uri';
10
import { ICodeEditor } from 'vs/editor/browser/editorBrowser';
A
Alex Dima 已提交
11 12 13 14 15 16 17 18
import { EditorAction, ServicesAccessor, registerEditorAction, registerEditorContribution } from 'vs/editor/browser/editorExtensions';
import { ICodeEditorService } from 'vs/editor/browser/services/codeEditorService';
import { EDITOR_DEFAULTS, InternalEditorOptions } from 'vs/editor/common/config/editorOptions';
import { IEditorContribution } from 'vs/editor/common/editorCommon';
import { ITextModel } from 'vs/editor/common/model';
import { ITextResourceConfigurationService } from 'vs/editor/common/services/resourceConfiguration';
import { MenuId, MenuRegistry } from 'vs/platform/actions/common/actions';
import { ContextKeyExpr, IContextKeyService } from 'vs/platform/contextkey/common/contextkey';
19
import { KeybindingWeight } from 'vs/platform/keybinding/common/keybindingsRegistry';
A
Alex Dima 已提交
20
import { INotificationService } from 'vs/platform/notification/common/notification';
A
Alex Dima 已提交
21

22 23 24 25 26 27 28 29 30 31 32 33 34 35
const transientWordWrapState = 'transientWordWrapState';
const isWordWrapMinifiedKey = 'isWordWrapMinified';
const isDominatedByLongLinesKey = 'isDominatedByLongLines';
const inDiffEditorKey = 'inDiffEditor';

/**
 * State written/read by the toggle word wrap action and associated with a particular model.
 */
interface IWordWrapTransientState {
	readonly forceWordWrap: 'on' | 'off' | 'wordWrapColumn' | 'bounded';
	readonly forceWordWrapMinified: boolean;
}

interface IWordWrapState {
A
Alex Dima 已提交
36
	readonly configuredWordWrap: 'on' | 'off' | 'wordWrapColumn' | 'bounded' | undefined;
37
	readonly configuredWordWrapMinified: boolean;
A
Alex Dima 已提交
38
	readonly transientState: IWordWrapTransientState | null;
39 40 41 42 43
}

/**
 * Store (in memory) the word wrap state for a particular model.
 */
44
export function writeTransientState(model: ITextModel, state: IWordWrapTransientState | null, codeEditorService: ICodeEditorService): void {
45 46 47 48 49 50
	codeEditorService.setTransientModelProperty(model, transientWordWrapState, state);
}

/**
 * Read (in memory) the word wrap state for a particular model.
 */
A
Alex Dima 已提交
51
function readTransientState(model: ITextModel, codeEditorService: ICodeEditorService): IWordWrapTransientState {
52 53 54
	return codeEditorService.getTransientModelProperty(model, transientWordWrapState);
}

A
Alex Dima 已提交
55
function readWordWrapState(model: ITextModel, configurationService: ITextResourceConfigurationService, codeEditorService: ICodeEditorService): IWordWrapState {
56
	const editorConfig = configurationService.getValue(model.uri, 'editor') as { wordWrap: 'on' | 'off' | 'wordWrapColumn' | 'bounded'; wordWrapMinified: boolean };
R
Rob Lourens 已提交
57
	let _configuredWordWrap = editorConfig && (typeof editorConfig.wordWrap === 'string' || typeof editorConfig.wordWrap === 'boolean') ? editorConfig.wordWrap : undefined;
A
Alex Dima 已提交
58 59 60 61 62 63 64 65

	// Compatibility with old true or false values
	if (<any>_configuredWordWrap === true) {
		_configuredWordWrap = 'on';
	} else if (<any>_configuredWordWrap === false) {
		_configuredWordWrap = 'off';
	}

R
Rob Lourens 已提交
66
	const _configuredWordWrapMinified = editorConfig && typeof editorConfig.wordWrapMinified === 'boolean' ? editorConfig.wordWrapMinified : undefined;
67 68
	const _transientState = readTransientState(model, codeEditorService);
	return {
A
Alex Dima 已提交
69
		configuredWordWrap: _configuredWordWrap,
70
		configuredWordWrapMinified: (typeof _configuredWordWrapMinified === 'boolean' ? _configuredWordWrapMinified : EDITOR_DEFAULTS.wordWrapMinified),
71 72 73 74
		transientState: _transientState
	};
}

75
function toggleWordWrap(editor: ICodeEditor, state: IWordWrapState): IWordWrapState {
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115
	if (state.transientState) {
		// toggle off => go to null
		return {
			configuredWordWrap: state.configuredWordWrap,
			configuredWordWrapMinified: state.configuredWordWrapMinified,
			transientState: null
		};
	}

	const config = editor.getConfiguration();
	let transientState: IWordWrapTransientState;

	const actualWrappingInfo = config.wrappingInfo;
	if (actualWrappingInfo.isWordWrapMinified) {
		// => wrapping due to minified file
		transientState = {
			forceWordWrap: 'off',
			forceWordWrapMinified: false
		};
	} else if (state.configuredWordWrap !== 'off') {
		// => wrapping is configured to be on (or some variant)
		transientState = {
			forceWordWrap: 'off',
			forceWordWrapMinified: false
		};
	} else {
		// => wrapping is configured to be off
		transientState = {
			forceWordWrap: 'on',
			forceWordWrapMinified: state.configuredWordWrapMinified
		};
	}

	return {
		configuredWordWrap: state.configuredWordWrap,
		configuredWordWrapMinified: state.configuredWordWrapMinified,
		transientState: transientState
	};
}

116
const TOGGLE_WORD_WRAP_ID = 'editor.action.toggleWordWrap';
A
Alex Dima 已提交
117
class ToggleWordWrapAction extends EditorAction {
A
Alex Dima 已提交
118 119

	constructor() {
120
		super({
121
			id: TOGGLE_WORD_WRAP_ID,
122 123
			label: nls.localize('toggle.wordwrap', "View: Toggle Word Wrap"),
			alias: 'View: Toggle Word Wrap',
124
			precondition: undefined,
125
			kbOpts: {
126
				kbExpr: null,
A
Alex Dima 已提交
127
				primary: KeyMod.Alt | KeyCode.KEY_Z,
128
				weight: KeybindingWeight.EditorContrib
129 130
			}
		});
I
ivanixgames 已提交
131 132
	}

133
	public run(accessor: ServicesAccessor, editor: ICodeEditor): void {
A
Alex Dima 已提交
134 135 136
		if (!editor.hasModel()) {
			return;
		}
A
Alex Dima 已提交
137 138 139
		const editorConfiguration = editor.getConfiguration();
		if (editorConfiguration.wrappingInfo.inDiffEditor) {
			// Cannot change wrapping settings inside the diff editor
140 141
			const notificationService = accessor.get(INotificationService);
			notificationService.info(nls.localize('wordWrap.notInDiffEditor', "Cannot toggle word wrap in a diff editor."));
A
Alex Dima 已提交
142 143 144
			return;
		}

145
		const textResourceConfigurationService = accessor.get(ITextResourceConfigurationService);
146
		const codeEditorService = accessor.get(ICodeEditorService);
147 148
		const model = editor.getModel();

149 150 151 152
		if (!canToggleWordWrap(model.uri)) {
			return;
		}

153
		// Read the current state
154
		const currentState = readWordWrapState(model, textResourceConfigurationService, codeEditorService);
155 156 157
		// Compute the new state
		const newState = toggleWordWrap(editor, currentState);
		// Write the new state
A
Alex Dima 已提交
158
		// (this will cause an event and the controller will apply the state)
159
		writeTransientState(model, newState.transientState, codeEditorService);
I
ivanixgames 已提交
160 161
	}
}
162 163 164

class ToggleWordWrapController extends Disposable implements IEditorContribution {

165
	private static readonly _ID = 'editor.contrib.toggleWordWrapController';
166 167

	constructor(
168
		private readonly editor: ICodeEditor,
169
		@IContextKeyService readonly contextKeyService: IContextKeyService,
170
		@ITextResourceConfigurationService readonly configurationService: ITextResourceConfigurationService,
171
		@ICodeEditorService readonly codeEditorService: ICodeEditorService
172 173 174
	) {
		super();

A
Alex Dima 已提交
175
		const configuration = this.editor.getConfiguration();
176 177 178
		const isWordWrapMinified = this.contextKeyService.createKey(isWordWrapMinifiedKey, this._isWordWrapMinified(configuration));
		const isDominatedByLongLines = this.contextKeyService.createKey(isDominatedByLongLinesKey, this._isDominatedByLongLines(configuration));
		const inDiffEditor = this.contextKeyService.createKey(inDiffEditorKey, this._inDiffEditor(configuration));
A
Alex Dima 已提交
179
		let currentlyApplyingEditorConfig = false;
180 181 182 183 184

		this._register(editor.onDidChangeConfiguration((e) => {
			if (!e.wrappingInfo) {
				return;
			}
A
Alex Dima 已提交
185 186 187 188
			const configuration = this.editor.getConfiguration();
			isWordWrapMinified.set(this._isWordWrapMinified(configuration));
			isDominatedByLongLines.set(this._isDominatedByLongLines(configuration));
			inDiffEditor.set(this._inDiffEditor(configuration));
A
Alex Dima 已提交
189 190 191 192
			if (!currentlyApplyingEditorConfig) {
				// I am not the cause of the word wrap getting changed
				ensureWordWrapSettings();
			}
193
		}));
194 195

		this._register(editor.onDidChangeModel((e) => {
A
Alex Dima 已提交
196 197 198
			ensureWordWrapSettings();
		}));

A
Alex Dima 已提交
199 200 201 202
		this._register(codeEditorService.onDidChangeTransientModelProperty(() => {
			ensureWordWrapSettings();
		}));

A
Alex Dima 已提交
203
		const ensureWordWrapSettings = () => {
204 205 206 207 208 209 210 211 212 213 214
			// Ensure correct word wrap settings
			const newModel = this.editor.getModel();
			if (!newModel) {
				return;
			}

			const configuration = this.editor.getConfiguration();
			if (this._inDiffEditor(configuration)) {
				return;
			}

215 216 217 218
			if (!canToggleWordWrap(newModel.uri)) {
				return;
			}

219 220 221 222
			// Read current configured values and toggle state
			const desiredState = readWordWrapState(newModel, this.configurationService, this.codeEditorService);

			// Apply the state
A
Alex Dima 已提交
223 224
			try {
				currentlyApplyingEditorConfig = true;
A
Alex Dima 已提交
225
				this._applyWordWrapState(desiredState);
A
Alex Dima 已提交
226 227 228 229
			} finally {
				currentlyApplyingEditorConfig = false;
			}
		};
230 231
	}

A
Alex Dima 已提交
232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248
	private _applyWordWrapState(state: IWordWrapState): void {
		if (state.transientState) {
			// toggle is on
			this.editor.updateOptions({
				wordWrap: state.transientState.forceWordWrap,
				wordWrapMinified: state.transientState.forceWordWrapMinified
			});
			return;
		}

		// toggle is off
		this.editor.updateOptions({
			wordWrap: state.configuredWordWrap,
			wordWrapMinified: state.configuredWordWrapMinified
		});
	}

A
Alex Dima 已提交
249 250 251 252 253 254 255 256 257 258
	private _isWordWrapMinified(config: InternalEditorOptions): boolean {
		return config.wrappingInfo.isWordWrapMinified;
	}

	private _isDominatedByLongLines(config: InternalEditorOptions): boolean {
		return config.wrappingInfo.isDominatedByLongLines;
	}

	private _inDiffEditor(config: InternalEditorOptions): boolean {
		return config.wrappingInfo.inDiffEditor;
259 260 261 262 263 264 265
	}

	public getId(): string {
		return ToggleWordWrapController._ID;
	}
}

266 267 268 269 270 271 272
function canToggleWordWrap(uri: URI): boolean {
	if (!uri) {
		return false;
	}
	return (uri.scheme !== 'output' && uri.scheme !== 'vscode');
}

273

274
registerEditorContribution(ToggleWordWrapController);
275

276
registerEditorAction(ToggleWordWrapAction);
277

278 279
MenuRegistry.appendMenuItem(MenuId.EditorTitle, {
	command: {
280
		id: TOGGLE_WORD_WRAP_ID,
A
Alex Dima 已提交
281
		title: nls.localize('unwrapMinified', "Disable wrapping for this file"),
A
Alex Dima 已提交
282
		iconLocation: { dark: URI.parse(require.toUrl('vs/workbench/contrib/codeEditor/browser/WordWrap_16x.svg')) }
A
Alex Dima 已提交
283 284 285 286
	},
	group: 'navigation',
	order: 1,
	when: ContextKeyExpr.and(
287 288 289
		ContextKeyExpr.not(inDiffEditorKey),
		ContextKeyExpr.has(isDominatedByLongLinesKey),
		ContextKeyExpr.has(isWordWrapMinifiedKey)
A
Alex Dima 已提交
290 291 292 293
	)
});
MenuRegistry.appendMenuItem(MenuId.EditorTitle, {
	command: {
294
		id: TOGGLE_WORD_WRAP_ID,
A
Alex Dima 已提交
295
		title: nls.localize('wrapMinified', "Enable wrapping for this file"),
A
Alex Dima 已提交
296
		iconLocation: { dark: URI.parse(require.toUrl('vs/workbench/contrib/codeEditor/browser/WordWrap_16x.svg')) }
297 298 299
	},
	group: 'navigation',
	order: 1,
A
Alex Dima 已提交
300
	when: ContextKeyExpr.and(
301 302 303
		ContextKeyExpr.not(inDiffEditorKey),
		ContextKeyExpr.has(isDominatedByLongLinesKey),
		ContextKeyExpr.not(isWordWrapMinifiedKey)
A
Alex Dima 已提交
304
	)
305
});
306 307 308 309 310 311 312 313 314 315 316


// View menu
MenuRegistry.appendMenuItem(MenuId.MenubarViewMenu, {
	group: '5_editor',
	command: {
		id: TOGGLE_WORD_WRAP_ID,
		title: nls.localize({ key: 'miToggleWordWrap', comment: ['&& denotes a mnemonic'] }, "Toggle &&Word Wrap")
	},
	order: 1
});