mainThreadEditors.ts 12.0 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.
 *--------------------------------------------------------------------------------------------*/
'use strict';

A
Alex Dima 已提交
7
import URI, { UriComponents } from 'vs/base/common/uri';
J
Johannes Rieken 已提交
8
import { IDisposable, dispose } from 'vs/base/common/lifecycle';
9
import { disposed } from 'vs/base/common/errors';
J
Johannes Rieken 已提交
10
import { TPromise } from 'vs/base/common/winjs.base';
11
import { IDecorationRenderOptions, IDecorationOptions, ILineChange } from 'vs/editor/common/editorCommon';
A
Alex Dima 已提交
12
import { ISingleEditOperation } from 'vs/editor/common/model';
13
import { ICodeEditorService } from 'vs/editor/browser/services/codeEditorService';
J
Johannes Rieken 已提交
14 15
import { IWorkbenchEditorService } from 'vs/workbench/services/editor/common/editorService';
import { IEditorGroupService } from 'vs/workbench/services/group/common/groupService';
16
import { Position as EditorPosition, ITextEditorOptions } from 'vs/platform/editor/common/editor';
17
import { MainThreadTextEditor } from './mainThreadEditor';
18
import { ITextEditorConfigurationUpdate, TextEditorRevealType, IApplyEditsOptions, IUndoStopOptions, IResourceFileEdit } from 'vs/workbench/api/node/extHost.protocol';
J
Johannes Rieken 已提交
19
import { MainThreadDocumentsAndEditors } from './mainThreadDocumentsAndEditors';
J
Johannes Rieken 已提交
20
import { equals as objectEquals } from 'vs/base/common/objects';
21
import { ExtHostContext, MainThreadEditorsShape, ExtHostEditorsShape, ITextDocumentShowOptions, ITextEditorPositionData, IExtHostContext, IWorkspaceResourceEdit } from '../node/extHost.protocol';
22 23
import { IRange } from 'vs/editor/common/core/range';
import { ISelection } from 'vs/editor/common/core/selection';
24 25
import { ITextModelService } from 'vs/editor/common/services/resolverService';
import { IFileService } from 'vs/platform/files/common/files';
26
import { bulkEdit, IResourceEdit } from 'vs/editor/browser/services/bulkEdit';
27
import { IModelService } from 'vs/editor/common/services/modelService';
28
import { isCodeEditor, ICodeEditor } from 'vs/editor/browser/editorBrowser';
29

30
export class MainThreadEditors implements MainThreadEditorsShape {
31

32
	private _proxy: ExtHostEditorsShape;
J
Johannes Rieken 已提交
33
	private _documentsAndEditors: MainThreadDocumentsAndEditors;
34 35 36 37
	private _workbenchEditorService: IWorkbenchEditorService;
	private _toDispose: IDisposable[];
	private _textEditorsListenersMap: { [editorId: string]: IDisposable[]; };
	private _editorPositionData: ITextEditorPositionData;
38
	private _registeredDecorationTypes: { [decorationType: string]: boolean; };
E
Erich Gamma 已提交
39

40
	constructor(
J
Johannes Rieken 已提交
41
		documentsAndEditors: MainThreadDocumentsAndEditors,
42
		extHostContext: IExtHostContext,
J
Johannes Rieken 已提交
43
		@ICodeEditorService private _codeEditorService: ICodeEditorService,
44 45
		@IWorkbenchEditorService workbenchEditorService: IWorkbenchEditorService,
		@IEditorGroupService editorGroupService: IEditorGroupService,
46 47 48
		@ITextModelService private readonly _textModelResolverService: ITextModelService,
		@IFileService private readonly _fileService: IFileService,
		@IModelService private readonly _modelService: IModelService,
49
	) {
50
		this._proxy = extHostContext.getProxy(ExtHostContext.ExtHostEditors);
J
Johannes Rieken 已提交
51
		this._documentsAndEditors = documentsAndEditors;
52 53 54 55
		this._workbenchEditorService = workbenchEditorService;
		this._toDispose = [];
		this._textEditorsListenersMap = Object.create(null);
		this._editorPositionData = null;
E
Erich Gamma 已提交
56

J
Johannes Rieken 已提交
57 58
		this._toDispose.push(documentsAndEditors.onTextEditorAdd(editors => editors.forEach(this._onTextEditorAdd, this)));
		this._toDispose.push(documentsAndEditors.onTextEditorRemove(editors => editors.forEach(this._onTextEditorRemove, this)));
E
Erich Gamma 已提交
59

60
		this._toDispose.push(editorGroupService.onEditorsChanged(() => this._updateActiveAndVisibleTextEditors()));
61
		this._toDispose.push(editorGroupService.onEditorGroupMoved(() => this._updateActiveAndVisibleTextEditors()));
62 63

		this._registeredDecorationTypes = Object.create(null);
E
Erich Gamma 已提交
64 65
	}

66 67 68 69 70 71
	public dispose(): void {
		Object.keys(this._textEditorsListenersMap).forEach((editorId) => {
			dispose(this._textEditorsListenersMap[editorId]);
		});
		this._textEditorsListenersMap = Object.create(null);
		this._toDispose = dispose(this._toDispose);
72 73 74 75
		for (let decorationType in this._registeredDecorationTypes) {
			this._codeEditorService.removeDecorationType(decorationType);
		}
		this._registeredDecorationTypes = Object.create(null);
E
Erich Gamma 已提交
76 77
	}

78 79 80 81
	private _onTextEditorAdd(textEditor: MainThreadTextEditor): void {
		let id = textEditor.getId();
		let toDispose: IDisposable[] = [];
		toDispose.push(textEditor.onConfigurationChanged((opts) => {
82
			this._proxy.$acceptOptionsChanged(id, opts);
83
		}));
84 85
		toDispose.push(textEditor.onSelectionChanged((event) => {
			this._proxy.$acceptSelectionsChanged(id, event);
86
		}));
E
Erich Gamma 已提交
87

88
		this._textEditorsListenersMap[id] = toDispose;
E
Erich Gamma 已提交
89 90
	}

J
Johannes Rieken 已提交
91
	private _onTextEditorRemove(id: string): void {
92 93
		dispose(this._textEditorsListenersMap[id]);
		delete this._textEditorsListenersMap[id];
E
Erich Gamma 已提交
94 95
	}

96
	private _updateActiveAndVisibleTextEditors(): void {
E
Erich Gamma 已提交
97

98 99 100 101
		// editor columns
		let editorPositionData = this._getTextEditorPositionData();
		if (!objectEquals(this._editorPositionData, editorPositionData)) {
			this._editorPositionData = editorPositionData;
102
			this._proxy.$acceptEditorPositionData(this._editorPositionData);
E
Erich Gamma 已提交
103 104 105
		}
	}

106 107 108
	private _getTextEditorPositionData(): ITextEditorPositionData {
		let result: ITextEditorPositionData = Object.create(null);
		for (let workbenchEditor of this._workbenchEditorService.getVisibleEditors()) {
J
Johannes Rieken 已提交
109 110 111
			const id = this._documentsAndEditors.findTextEditorIdFor(workbenchEditor);
			if (id) {
				result[id] = workbenchEditor.position;
112
			}
E
Erich Gamma 已提交
113
		}
114
		return result;
E
Erich Gamma 已提交
115 116
	}

117
	// --- from extension host process
E
Erich Gamma 已提交
118

A
Alex Dima 已提交
119 120 121
	$tryShowTextDocument(resource: UriComponents, options: ITextDocumentShowOptions): TPromise<string> {
		const uri = URI.revive(resource);

122
		const editorOptions: ITextEditorOptions = {
123
			preserveFocus: options.preserveFocus,
124 125
			pinned: options.pinned,
			selection: options.selection
126
		};
E
Erich Gamma 已提交
127

128
		const input = {
A
Alex Dima 已提交
129
			resource: uri,
130
			options: editorOptions
E
Erich Gamma 已提交
131 132
		};

133
		return this._workbenchEditorService.openEditor(input, options.position).then(editor => {
134
			if (!editor) {
M
Matt Bierner 已提交
135
				return undefined;
E
Erich Gamma 已提交
136
			}
J
Johannes Rieken 已提交
137
			return this._documentsAndEditors.findTextEditorIdFor(editor);
E
Erich Gamma 已提交
138 139 140
		});
	}

141
	$tryShowEditor(id: string, position: EditorPosition): TPromise<void> {
J
Johannes Rieken 已提交
142
		let mainThreadEditor = this._documentsAndEditors.getEditor(id);
143 144 145 146 147 148
		if (mainThreadEditor) {
			let model = mainThreadEditor.getModel();
			return this._workbenchEditorService.openEditor({
				resource: model.uri,
				options: { preserveFocus: false }
			}, position).then(() => { return; });
E
Erich Gamma 已提交
149
		}
M
Matt Bierner 已提交
150
		return undefined;
E
Erich Gamma 已提交
151 152
	}

153
	$tryHideEditor(id: string): TPromise<void> {
J
Johannes Rieken 已提交
154
		let mainThreadEditor = this._documentsAndEditors.getEditor(id);
155 156 157 158 159
		if (mainThreadEditor) {
			let editors = this._workbenchEditorService.getVisibleEditors();
			for (let editor of editors) {
				if (mainThreadEditor.matches(editor)) {
					return this._workbenchEditorService.closeEditor(editor.position, editor.input).then(() => { return; });
E
Erich Gamma 已提交
160 161 162
				}
			}
		}
M
Matt Bierner 已提交
163
		return undefined;
E
Erich Gamma 已提交
164 165
	}

166
	$trySetSelections(id: string, selections: ISelection[]): TPromise<void> {
J
Johannes Rieken 已提交
167
		if (!this._documentsAndEditors.getEditor(id)) {
168
			return TPromise.wrapError(disposed(`TextEditor(${id})`));
E
Erich Gamma 已提交
169
		}
J
Johannes Rieken 已提交
170
		this._documentsAndEditors.getEditor(id).setSelections(selections);
171
		return TPromise.as(null);
E
Erich Gamma 已提交
172 173
	}

174
	$trySetDecorations(id: string, key: string, ranges: IDecorationOptions[]): TPromise<void> {
J
Johannes Rieken 已提交
175
		if (!this._documentsAndEditors.getEditor(id)) {
176
			return TPromise.wrapError(disposed(`TextEditor(${id})`));
177
		}
J
Johannes Rieken 已提交
178
		this._documentsAndEditors.getEditor(id).setDecorations(key, ranges);
179
		return TPromise.as(null);
E
Erich Gamma 已提交
180 181
	}

182
	$trySetDecorationsFast(id: string, key: string, ranges: number[]): TPromise<void> {
183 184 185
		if (!this._documentsAndEditors.getEditor(id)) {
			return TPromise.wrapError(disposed(`TextEditor(${id})`));
		}
186
		this._documentsAndEditors.getEditor(id).setDecorationsFast(key, ranges);
187 188 189
		return TPromise.as(null);
	}

190
	$tryRevealRange(id: string, range: IRange, revealType: TextEditorRevealType): TPromise<void> {
J
Johannes Rieken 已提交
191
		if (!this._documentsAndEditors.getEditor(id)) {
192
			return TPromise.wrapError(disposed(`TextEditor(${id})`));
193
		}
J
Johannes Rieken 已提交
194
		this._documentsAndEditors.getEditor(id).revealRange(range, revealType);
M
Matt Bierner 已提交
195
		return undefined;
E
Erich Gamma 已提交
196 197
	}

198
	$trySetOptions(id: string, options: ITextEditorConfigurationUpdate): TPromise<void> {
J
Johannes Rieken 已提交
199
		if (!this._documentsAndEditors.getEditor(id)) {
200
			return TPromise.wrapError(disposed(`TextEditor(${id})`));
E
Erich Gamma 已提交
201
		}
J
Johannes Rieken 已提交
202
		this._documentsAndEditors.getEditor(id).setConfiguration(options);
203
		return TPromise.as(null);
E
Erich Gamma 已提交
204 205
	}

J
Johannes Rieken 已提交
206
	$tryApplyEdits(id: string, modelVersionId: number, edits: ISingleEditOperation[], opts: IApplyEditsOptions): TPromise<boolean> {
J
Johannes Rieken 已提交
207
		if (!this._documentsAndEditors.getEditor(id)) {
208
			return TPromise.wrapError<boolean>(disposed(`TextEditor(${id})`));
209
		}
J
Johannes Rieken 已提交
210
		return TPromise.as(this._documentsAndEditors.getEditor(id).applyEdits(modelVersionId, edits, opts));
E
Erich Gamma 已提交
211 212
	}

213
	$tryApplyWorkspaceEdit(workspaceResourceEdits: IWorkspaceResourceEdit[], resourceFileEdits?: IResourceFileEdit): TPromise<boolean> {
214 215 216 217 218

		// First check if loaded models were not changed in the meantime
		for (let i = 0, len = workspaceResourceEdits.length; i < len; i++) {
			const workspaceResourceEdit = workspaceResourceEdits[i];
			if (workspaceResourceEdit.modelVersionId) {
219 220
				const uri = URI.revive(workspaceResourceEdit.resource);
				let model = this._modelService.getModel(uri);
221 222 223 224 225 226 227 228 229 230 231
				if (model && model.getVersionId() !== workspaceResourceEdit.modelVersionId) {
					// model changed in the meantime
					return TPromise.as(false);
				}
			}
		}

		// Convert to shape expected by bulkEdit below
		let resourceEdits: IResourceEdit[] = [];
		for (let i = 0, len = workspaceResourceEdits.length; i < len; i++) {
			const workspaceResourceEdit = workspaceResourceEdits[i];
232
			const uri = URI.revive(workspaceResourceEdit.resource);
233 234 235 236 237 238
			const edits = workspaceResourceEdit.edits;

			for (let j = 0, lenJ = edits.length; j < lenJ; j++) {
				const edit = edits[j];

				resourceEdits.push({
J
Johannes Rieken 已提交
239
					resource: uri,
240 241 242 243 244 245 246
					newText: edit.newText,
					newEol: edit.newEol,
					range: edit.range
				});
			}
		}

247
		let codeEditor: ICodeEditor;
248 249 250
		let editor = this._workbenchEditorService.getActiveEditor();
		if (editor) {
			let candidate = editor.getControl();
251
			if (isCodeEditor(candidate)) {
252 253 254 255
				codeEditor = candidate;
			}
		}

256 257 258 259 260 261 262 263 264 265 266
		return bulkEdit(
			this._textModelResolverService,
			codeEditor,
			resourceEdits,
			this._fileService,
			resourceFileEdits ? {
				renamedResources: resourceFileEdits.renamedResources.map(entry => ({ from: URI.revive(entry.from), to: URI.revive(entry.to) })),
				createdResources: resourceFileEdits.createdResources.map(entry => ({ uri: URI.revive(entry.uri), contents: entry.contents })),
				deletedResources: resourceFileEdits.deletedResources.map(URI.revive)
			} : undefined
		).then(() => true);
267 268
	}

269
	$tryInsertSnippet(id: string, template: string, ranges: IRange[], opts: IUndoStopOptions): TPromise<boolean> {
J
Johannes Rieken 已提交
270
		if (!this._documentsAndEditors.getEditor(id)) {
271
			return TPromise.wrapError<boolean>(disposed(`TextEditor(${id})`));
272
		}
J
Johannes Rieken 已提交
273
		return TPromise.as(this._documentsAndEditors.getEditor(id).insertSnippet(template, ranges, opts));
274 275
	}

276
	$registerTextEditorDecorationType(key: string, options: IDecorationRenderOptions): void {
277
		this._registeredDecorationTypes[key] = true;
J
Johannes Rieken 已提交
278
		this._codeEditorService.registerDecorationType(key, options);
E
Erich Gamma 已提交
279 280
	}

281
	$removeTextEditorDecorationType(key: string): void {
282
		delete this._registeredDecorationTypes[key];
J
Johannes Rieken 已提交
283
		this._codeEditorService.removeDecorationType(key);
284 285 286 287 288 289
	}

	$getDiffInformation(id: string): TPromise<ILineChange[]> {
		const editor = this._documentsAndEditors.getEditor(id);

		if (!editor) {
290
			return TPromise.wrapError<ILineChange[]>(new Error('No such TextEditor'));
291 292 293 294 295 296 297 298 299 300
		}

		const codeEditor = editor.getCodeEditor();
		const codeEditorId = codeEditor.getId();
		const diffEditors = this._codeEditorService.listDiffEditors();
		const [diffEditor] = diffEditors.filter(d => d.getOriginalEditor().getId() === codeEditorId || d.getModifiedEditor().getId() === codeEditorId);

		if (!diffEditor) {
			return TPromise.as([]);
		}
J
Johannes Rieken 已提交
301

302
		return TPromise.as(diffEditor.getLineChanges());
E
Erich Gamma 已提交
303
	}
A
Alex Dima 已提交
304
}