viewImpl.ts 21.8 KB
Newer Older
E
Erich Gamma 已提交
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 dom from 'vs/base/browser/dom';
A
Alex Dima 已提交
7
import { FastDomNode, createFastDomNode } from 'vs/base/browser/fastDomNode';
A
Alex Dima 已提交
8 9 10 11
import { IMouseEvent } from 'vs/base/browser/mouseEvent';
import { onUnexpectedError } from 'vs/base/common/errors';
import { IDisposable } from 'vs/base/common/lifecycle';
import { IPointerHandlerHelper } from 'vs/editor/browser/controller/mouseHandler';
J
Johannes Rieken 已提交
12
import { PointerHandler } from 'vs/editor/browser/controller/pointerHandler';
A
Alex Dima 已提交
13
import { ITextAreaHandlerHelper, TextAreaHandler } from 'vs/editor/browser/controller/textAreaHandler';
A
Alex Dima 已提交
14
import * as editorBrowser from 'vs/editor/browser/editorBrowser';
A
Alex Dima 已提交
15 16
import { ICommandDelegate, ViewController } from 'vs/editor/browser/view/viewController';
import { ViewOutgoingEvents } from 'vs/editor/browser/view/viewOutgoingEvents';
J
Johannes Rieken 已提交
17
import { ContentViewOverlays, MarginViewOverlays } from 'vs/editor/browser/view/viewOverlays';
A
Alex Dima 已提交
18
import { PartFingerprint, PartFingerprints, ViewPart } from 'vs/editor/browser/view/viewPart';
J
Johannes Rieken 已提交
19 20
import { ViewContentWidgets } from 'vs/editor/browser/viewParts/contentWidgets/contentWidgets';
import { CurrentLineHighlightOverlay } from 'vs/editor/browser/viewParts/currentLineHighlight/currentLineHighlight';
21
import { CurrentLineMarginHighlightOverlay } from 'vs/editor/browser/viewParts/currentLineMarginHighlight/currentLineMarginHighlight';
J
Johannes Rieken 已提交
22
import { DecorationsOverlay } from 'vs/editor/browser/viewParts/decorations/decorations';
A
Alex Dima 已提交
23
import { EditorScrollbar } from 'vs/editor/browser/viewParts/editorScrollbar/editorScrollbar';
J
Johannes Rieken 已提交
24 25
import { GlyphMarginOverlay } from 'vs/editor/browser/viewParts/glyphMargin/glyphMargin';
import { IndentGuidesOverlay } from 'vs/editor/browser/viewParts/indentGuides/indentGuides';
A
Alex Dima 已提交
26
import { LineNumbersOverlay } from 'vs/editor/browser/viewParts/lineNumbers/lineNumbers';
J
Johannes Rieken 已提交
27 28
import { ViewLines } from 'vs/editor/browser/viewParts/lines/viewLines';
import { LinesDecorationsOverlay } from 'vs/editor/browser/viewParts/linesDecorations/linesDecorations';
A
Alex Dima 已提交
29
import { Margin } from 'vs/editor/browser/viewParts/margin/margin';
30
import { MarginViewLineDecorationsOverlay } from 'vs/editor/browser/viewParts/marginDecorations/marginDecorations';
A
Alex Dima 已提交
31
import { Minimap } from 'vs/editor/browser/viewParts/minimap/minimap';
J
Johannes Rieken 已提交
32 33 34 35 36 37 38 39
import { ViewOverlayWidgets } from 'vs/editor/browser/viewParts/overlayWidgets/overlayWidgets';
import { DecorationsOverviewRuler } from 'vs/editor/browser/viewParts/overviewRuler/decorationsOverviewRuler';
import { OverviewRuler } from 'vs/editor/browser/viewParts/overviewRuler/overviewRuler';
import { Rulers } from 'vs/editor/browser/viewParts/rulers/rulers';
import { ScrollDecorationViewPart } from 'vs/editor/browser/viewParts/scrollDecoration/scrollDecoration';
import { SelectionsOverlay } from 'vs/editor/browser/viewParts/selections/selections';
import { ViewCursors } from 'vs/editor/browser/viewParts/viewCursors/viewCursors';
import { ViewZones } from 'vs/editor/browser/viewParts/viewZones/viewZones';
A
Alex Dima 已提交
40 41 42
import { Cursor } from 'vs/editor/common/controller/cursor';
import { Position } from 'vs/editor/common/core/position';
import { IConfiguration } from 'vs/editor/common/editorCommon';
A
Alex Dima 已提交
43
import { RenderingContext } from 'vs/editor/common/view/renderingContext';
A
Alex Dima 已提交
44 45
import { ViewContext } from 'vs/editor/common/view/viewContext';
import { ViewEventDispatcher } from 'vs/editor/common/view/viewEventDispatcher';
46
import * as viewEvents from 'vs/editor/common/view/viewEvents';
A
Alex Dima 已提交
47 48 49
import { ViewportData } from 'vs/editor/common/viewLayout/viewLinesViewportData';
import { ViewEventHandler } from 'vs/editor/common/viewModel/viewEventHandler';
import { IViewModel } from 'vs/editor/common/viewModel/viewModel';
50
import { IThemeService, getThemeTypeSelector } from 'vs/platform/theme/common/themeService';
A
renames  
Alex Dima 已提交
51
import { EditorOption } from 'vs/editor/common/config/editorOptions';
52

E
Erich Gamma 已提交
53

A
Alex Dima 已提交
54 55
export interface IContentWidgetData {
	widget: editorBrowser.IContentWidget;
A
Alex Dima 已提交
56
	position: editorBrowser.IContentWidgetPosition | null;
A
Alex Dima 已提交
57 58 59 60
}

export interface IOverlayWidgetData {
	widget: editorBrowser.IOverlayWidget;
A
Alex Dima 已提交
61
	position: editorBrowser.IOverlayWidgetPosition | null;
A
Alex Dima 已提交
62 63
}

A
Alex Dima 已提交
64 65
const invalidFunc = () => { throw new Error(`Invalid change accessor`); };

A
Alex Dima 已提交
66
export class View extends ViewEventHandler {
E
Erich Gamma 已提交
67

68
	private readonly eventDispatcher: ViewEventDispatcher;
E
Erich Gamma 已提交
69

70
	private _scrollbar: EditorScrollbar;
71 72
	private readonly _context: ViewContext;
	private readonly _cursor: Cursor;
E
Erich Gamma 已提交
73 74 75 76 77 78 79 80

	// The view lines
	private viewLines: ViewLines;

	// These are parts, but we must do some API related calls on them, so we keep a reference
	private viewZones: ViewZones;
	private contentWidgets: ViewContentWidgets;
	private overlayWidgets: ViewOverlayWidgets;
81
	private viewCursors: ViewCursors;
A
Alex Dima 已提交
82
	private viewParts: ViewPart[];
E
Erich Gamma 已提交
83

84 85
	private readonly _textAreaHandler: TextAreaHandler;
	private readonly pointerHandler: PointerHandler;
E
Erich Gamma 已提交
86

87
	private readonly outgoingEvents: ViewOutgoingEvents;
E
Erich Gamma 已提交
88 89

	// Dom nodes
A
Alex Dima 已提交
90 91 92
	private linesContent: FastDomNode<HTMLElement>;
	public domNode: FastDomNode<HTMLElement>;
	private overflowGuardContainer: FastDomNode<HTMLElement>;
E
Erich Gamma 已提交
93 94

	// Actual mutable state
A
Alex Dima 已提交
95
	private _renderAnimationFrame: IDisposable | null;
E
Erich Gamma 已提交
96

A
Alex Dima 已提交
97
	constructor(
98
		commandDelegate: ICommandDelegate,
A
Alex Dima 已提交
99
		configuration: IConfiguration,
100
		themeService: IThemeService,
J
Johannes Rieken 已提交
101
		model: IViewModel,
102
		cursor: Cursor,
A
Alex Dima 已提交
103
		outgoingEvents: ViewOutgoingEvents
A
Alex Dima 已提交
104
	) {
E
Erich Gamma 已提交
105
		super();
106
		this._cursor = cursor;
E
Erich Gamma 已提交
107
		this._renderAnimationFrame = null;
A
Alex Dima 已提交
108
		this.outgoingEvents = outgoingEvents;
E
Erich Gamma 已提交
109

M
Matt Bierner 已提交
110
		const viewController = new ViewController(configuration, model, this.outgoingEvents, commandDelegate);
E
Erich Gamma 已提交
111 112

		// The event dispatcher will always go through _renderOnce before dispatching any events
J
Johannes Rieken 已提交
113
		this.eventDispatcher = new ViewEventDispatcher((callback: () => void) => this._renderOnce(callback));
E
Erich Gamma 已提交
114

115 116 117
		// Ensure the view is the first event handler in order to update the layout
		this.eventDispatcher.addEventHandler(this);

E
Erich Gamma 已提交
118
		// The view context is passed on to most classes (basically to reduce param. counts in ctors)
119 120 121
		this._context = new ViewContext(configuration, themeService.getTheme(), model, this.eventDispatcher);

		this._register(themeService.onThemeChange(theme => {
122
			this._context.theme = theme;
123
			this.eventDispatcher.emit(new viewEvents.ViewThemeChangedEvent());
124
			this.render(true, false);
125
		}));
E
Erich Gamma 已提交
126

127 128
		this.viewParts = [];

129 130
		// Keyboard handler
		this._textAreaHandler = new TextAreaHandler(this._context, viewController, this.createTextAreaHandlerHelper());
131
		this.viewParts.push(this._textAreaHandler);
132

A
Alex Dima 已提交
133
		// These two dom nodes must be constructed up front, since references are needed in the layout provider (scrolling & co.)
A
Alex Dima 已提交
134
		this.linesContent = createFastDomNode(document.createElement('div'));
A
Alex Dima 已提交
135
		this.linesContent.setClassName('lines-content' + ' monaco-editor-background');
A
Alex Dima 已提交
136
		this.linesContent.setPosition('absolute');
A
Alex Dima 已提交
137

A
Alex Dima 已提交
138
		this.domNode = createFastDomNode(document.createElement('div'));
139
		this.domNode.setClassName(this.getEditorClassName());
A
Alex Dima 已提交
140

A
Alex Dima 已提交
141
		this.overflowGuardContainer = createFastDomNode(document.createElement('div'));
A
Alex Dima 已提交
142
		PartFingerprints.write(this.overflowGuardContainer, PartFingerprint.OverflowGuard);
A
Alex Dima 已提交
143
		this.overflowGuardContainer.setClassName('overflow-guard');
A
Alex Dima 已提交
144

145
		this._scrollbar = new EditorScrollbar(this._context, this.linesContent, this.domNode, this.overflowGuardContainer);
A
Alex Dima 已提交
146 147
		this.viewParts.push(this._scrollbar);

E
Erich Gamma 已提交
148
		// View Lines
149
		this.viewLines = new ViewLines(this._context, this.linesContent);
E
Erich Gamma 已提交
150 151

		// View Zones
152
		this.viewZones = new ViewZones(this._context);
E
Erich Gamma 已提交
153 154 155
		this.viewParts.push(this.viewZones);

		// Decorations overview ruler
M
Matt Bierner 已提交
156
		const decorationsOverviewRuler = new DecorationsOverviewRuler(this._context);
E
Erich Gamma 已提交
157 158 159
		this.viewParts.push(decorationsOverviewRuler);


M
Matt Bierner 已提交
160
		const scrollDecoration = new ScrollDecorationViewPart(this._context);
E
Erich Gamma 已提交
161 162
		this.viewParts.push(scrollDecoration);

M
Matt Bierner 已提交
163
		const contentViewOverlays = new ContentViewOverlays(this._context);
E
Erich Gamma 已提交
164
		this.viewParts.push(contentViewOverlays);
A
Alex Dima 已提交
165
		contentViewOverlays.addDynamicOverlay(new CurrentLineHighlightOverlay(this._context));
166
		contentViewOverlays.addDynamicOverlay(new SelectionsOverlay(this._context));
167
		contentViewOverlays.addDynamicOverlay(new IndentGuidesOverlay(this._context));
168
		contentViewOverlays.addDynamicOverlay(new DecorationsOverlay(this._context));
E
Erich Gamma 已提交
169

M
Matt Bierner 已提交
170
		const marginViewOverlays = new MarginViewOverlays(this._context);
E
Erich Gamma 已提交
171
		this.viewParts.push(marginViewOverlays);
A
Alex Dima 已提交
172
		marginViewOverlays.addDynamicOverlay(new CurrentLineMarginHighlightOverlay(this._context));
173
		marginViewOverlays.addDynamicOverlay(new GlyphMarginOverlay(this._context));
174
		marginViewOverlays.addDynamicOverlay(new MarginViewLineDecorationsOverlay(this._context));
175 176
		marginViewOverlays.addDynamicOverlay(new LinesDecorationsOverlay(this._context));
		marginViewOverlays.addDynamicOverlay(new LineNumbersOverlay(this._context));
E
Erich Gamma 已提交
177

M
Matt Bierner 已提交
178
		const margin = new Margin(this._context);
A
Alex Dima 已提交
179
		margin.getDomNode().appendChild(this.viewZones.marginDomNode);
A
Alex Dima 已提交
180
		margin.getDomNode().appendChild(marginViewOverlays.getDomNode());
181
		this.viewParts.push(margin);
E
Erich Gamma 已提交
182 183

		// Content widgets
184
		this.contentWidgets = new ViewContentWidgets(this._context, this.domNode);
E
Erich Gamma 已提交
185 186
		this.viewParts.push(this.contentWidgets);

187 188
		this.viewCursors = new ViewCursors(this._context);
		this.viewParts.push(this.viewCursors);
E
Erich Gamma 已提交
189 190

		// Overlay widgets
191
		this.overlayWidgets = new ViewOverlayWidgets(this._context);
E
Erich Gamma 已提交
192 193
		this.viewParts.push(this.overlayWidgets);

M
Matt Bierner 已提交
194
		const rulers = new Rulers(this._context);
195 196
		this.viewParts.push(rulers);

M
Matt Bierner 已提交
197
		const minimap = new Minimap(this._context);
A
Alex Dima 已提交
198 199
		this.viewParts.push(minimap);

E
Erich Gamma 已提交
200 201 202
		// -------------- Wire dom nodes up

		if (decorationsOverviewRuler) {
M
Matt Bierner 已提交
203
			const overviewRulerData = this._scrollbar.getOverviewRulerLayoutInfo();
E
Erich Gamma 已提交
204 205 206
			overviewRulerData.parent.insertBefore(decorationsOverviewRuler.getDomNode(), overviewRulerData.insertBefore);
		}

A
Alex Dima 已提交
207 208 209 210 211 212 213 214 215
		this.linesContent.appendChild(contentViewOverlays.getDomNode());
		this.linesContent.appendChild(rulers.domNode);
		this.linesContent.appendChild(this.viewZones.domNode);
		this.linesContent.appendChild(this.viewLines.getDomNode());
		this.linesContent.appendChild(this.contentWidgets.domNode);
		this.linesContent.appendChild(this.viewCursors.getDomNode());
		this.overflowGuardContainer.appendChild(margin.getDomNode());
		this.overflowGuardContainer.appendChild(this._scrollbar.getDomNode());
		this.overflowGuardContainer.appendChild(scrollDecoration.getDomNode());
216 217
		this.overflowGuardContainer.appendChild(this._textAreaHandler.textArea);
		this.overflowGuardContainer.appendChild(this._textAreaHandler.textAreaCover);
218
		this.overflowGuardContainer.appendChild(this.overlayWidgets.getDomNode());
A
Alex Dima 已提交
219 220 221
		this.overflowGuardContainer.appendChild(minimap.getDomNode());
		this.domNode.appendChild(this.overflowGuardContainer);
		this.domNode.appendChild(this.contentWidgets.overflowingContentWidgetsDomNode);
A
Alex Dima 已提交
222

A
Alex Dima 已提交
223
		this._applyLayout();
A
Alex Dima 已提交
224 225 226 227 228 229 230 231 232 233 234

		// Pointer handler
		this.pointerHandler = this._register(new PointerHandler(this._context, viewController, this.createPointerHandlerHelper()));

		this._register(model.addEventListener((events: viewEvents.ViewEvent[]) => {
			this.eventDispatcher.emitMany(events);
		}));

		this._register(this._cursor.addEventListener((events: viewEvents.ViewEvent[]) => {
			this.eventDispatcher.emitMany(events);
		}));
E
Erich Gamma 已提交
235 236 237 238 239 240
	}

	private _flushAccumulatedAndRenderNow(): void {
		this._renderNow();
	}

241
	private createPointerHandlerHelper(): IPointerHandlerHelper {
E
Erich Gamma 已提交
242
		return {
A
Alex Dima 已提交
243 244
			viewDomNode: this.domNode.domNode,
			linesContentDomNode: this.linesContent.domNode,
E
Erich Gamma 已提交
245 246 247 248 249

			focusTextArea: () => {
				this.focus();
			},

250 251 252
			getLastViewCursorsRenderData: () => {
				return this.viewCursors.getLastRenderData() || [];
			},
A
Alex Dima 已提交
253
			shouldSuppressMouseDownOnViewZone: (viewZoneId: string) => {
E
Erich Gamma 已提交
254 255
				return this.viewZones.shouldSuppressMouseDownOnViewZone(viewZoneId);
			},
256 257 258
			shouldSuppressMouseDownOnWidget: (widgetId: string) => {
				return this.contentWidgets.shouldSuppressMouseDownOnWidget(widgetId);
			},
E
Erich Gamma 已提交
259 260 261 262 263 264 265
			getPositionFromDOMInfo: (spanNode: HTMLElement, offset: number) => {
				this._flushAccumulatedAndRenderNow();
				return this.viewLines.getPositionFromDOMInfo(spanNode, offset);
			},

			visibleRangeForPosition2: (lineNumber: number, column: number) => {
				this._flushAccumulatedAndRenderNow();
A
Alex Dima 已提交
266
				return this.viewLines.visibleRangeForPosition(new Position(lineNumber, column));
E
Erich Gamma 已提交
267 268 269 270 271 272 273 274 275
			},

			getLineWidth: (lineNumber: number) => {
				this._flushAccumulatedAndRenderNow();
				return this.viewLines.getLineWidth(lineNumber);
			}
		};
	}

276
	private createTextAreaHandlerHelper(): ITextAreaHandlerHelper {
E
Erich Gamma 已提交
277 278 279
		return {
			visibleRangeForPositionRelativeToEditor: (lineNumber: number, column: number) => {
				this._flushAccumulatedAndRenderNow();
A
Alex Dima 已提交
280
				return this.viewLines.visibleRangeForPosition(new Position(lineNumber, column));
E
Erich Gamma 已提交
281 282 283 284
			}
		};
	}

A
Alex Dima 已提交
285
	private _applyLayout(): void {
A
Alex Dima 已提交
286
		const options = this._context.configuration.options;
A
renames  
Alex Dima 已提交
287
		const layoutInfo = options.get(EditorOption.layoutInfo);
A
Alex Dima 已提交
288

A
Alex Dima 已提交
289 290
		this.domNode.setWidth(layoutInfo.width);
		this.domNode.setHeight(layoutInfo.height);
E
Erich Gamma 已提交
291

A
Alex Dima 已提交
292 293
		this.overflowGuardContainer.setWidth(layoutInfo.width);
		this.overflowGuardContainer.setHeight(layoutInfo.height);
E
Erich Gamma 已提交
294

A
Alex Dima 已提交
295 296
		this.linesContent.setWidth(1000000);
		this.linesContent.setHeight(1000000);
E
Erich Gamma 已提交
297
	}
298

299
	private getEditorClassName() {
M
Matt Bierner 已提交
300
		const focused = this._textAreaHandler.isFocused() ? ' focused' : '';
A
Alex Dima 已提交
301
		return this._context.configuration.options.get(EditorOption.editorClassName) + ' ' + getThemeTypeSelector(this._context.theme.type) + focused;
302 303
	}

304 305
	// --- begin event handlers

A
Alex Dima 已提交
306
	public onConfigurationChanged(e: viewEvents.ViewConfigurationChangedEvent): boolean {
A
Alex Dima 已提交
307 308
		this.domNode.setClassName(this.getEditorClassName());
		this._applyLayout();
309
		return false;
E
Erich Gamma 已提交
310
	}
A
Alex Dima 已提交
311
	public onFocusChanged(e: viewEvents.ViewFocusChangedEvent): boolean {
S
smoke:  
Sandeep Somavarapu 已提交
312
		this.domNode.setClassName(this.getEditorClassName());
313
		this._context.model.setHasFocus(e.isFocused);
A
Alex Dima 已提交
314
		if (e.isFocused) {
A
Alex Dima 已提交
315
			this.outgoingEvents.emitViewFocusGained();
E
Erich Gamma 已提交
316
		} else {
A
Alex Dima 已提交
317
			this.outgoingEvents.emitViewFocusLost();
E
Erich Gamma 已提交
318 319 320
		}
		return false;
	}
321 322 323 324
	public onScrollChanged(e: viewEvents.ViewScrollChangedEvent): boolean {
		this.outgoingEvents.emitScrollChanged(e);
		return false;
	}
325 326 327 328
	public onThemeChanged(e: viewEvents.ViewThemeChangedEvent): boolean {
		this.domNode.setClassName(this.getEditorClassName());
		return false;
	}
329

E
Erich Gamma 已提交
330 331 332 333 334 335 336 337 338
	// --- end event handlers

	public dispose(): void {
		if (this._renderAnimationFrame !== null) {
			this._renderAnimationFrame.dispose();
			this._renderAnimationFrame = null;
		}

		this.eventDispatcher.removeEventHandler(this);
A
Alex Dima 已提交
339
		this.outgoingEvents.dispose();
E
Erich Gamma 已提交
340 341 342

		this.viewLines.dispose();

A
Alex Dima 已提交
343
		// Destroy view parts
A
Alex Dima 已提交
344
		for (let i = 0, len = this.viewParts.length; i < len; i++) {
E
Erich Gamma 已提交
345 346 347 348
			this.viewParts[i].dispose();
		}
		this.viewParts = [];

A
Alex Dima 已提交
349
		super.dispose();
E
Erich Gamma 已提交
350 351
	}

A
Alex Dima 已提交
352
	private _renderOnce(callback: () => any): any {
M
Matt Bierner 已提交
353
		const r = safeInvokeNoArg(callback);
A
Alex Dima 已提交
354 355 356
		this._scheduleRender();
		return r;
	}
E
Erich Gamma 已提交
357

A
Alex Dima 已提交
358 359 360 361 362
	private _scheduleRender(): void {
		if (this._renderAnimationFrame === null) {
			this._renderAnimationFrame = dom.runAtThisOrScheduleAtNextAnimationFrame(this._onRenderScheduled.bind(this), 100);
		}
	}
A
Alex Dima 已提交
363

A
Alex Dima 已提交
364 365 366 367
	private _onRenderScheduled(): void {
		this._renderAnimationFrame = null;
		this._flushAccumulatedAndRenderNow();
	}
368

A
Alex Dima 已提交
369 370 371
	private _renderNow(): void {
		safeInvokeNoArg(() => this._actualRender());
	}
A
Alex Dima 已提交
372

A
Alex Dima 已提交
373
	private _getViewPartsToRender(): ViewPart[] {
A
Alex Dima 已提交
374
		let result: ViewPart[] = [], resultLen = 0;
A
Alex Dima 已提交
375
		for (let i = 0, len = this.viewParts.length; i < len; i++) {
M
Matt Bierner 已提交
376
			const viewPart = this.viewParts[i];
A
Alex Dima 已提交
377
			if (viewPart.shouldRender()) {
A
Alex Dima 已提交
378
				result[resultLen++] = viewPart;
A
Alex Dima 已提交
379 380 381 382
			}
		}
		return result;
	}
383

A
Alex Dima 已提交
384 385 386 387
	private _actualRender(): void {
		if (!dom.isInDOM(this.domNode.domNode)) {
			return;
		}
388

A
Alex Dima 已提交
389
		let viewPartsToRender = this._getViewPartsToRender();
A
Alex Dima 已提交
390

A
Alex Dima 已提交
391 392 393 394
		if (!this.viewLines.shouldRender() && viewPartsToRender.length === 0) {
			// Nothing to render
			return;
		}
A
Alex Dima 已提交
395

396
		const partialViewportData = this._context.viewLayout.getLinesViewportData();
A
Alex Dima 已提交
397
		this._context.model.setViewport(partialViewportData.startLineNumber, partialViewportData.endLineNumber, partialViewportData.centeredLineNumber);
398

M
Matt Bierner 已提交
399
		const viewportData = new ViewportData(
400 401 402 403 404
			this._cursor.getViewSelections(),
			partialViewportData,
			this._context.viewLayout.getWhitespaceViewportData(),
			this._context.model
		);
A
Alex Dima 已提交
405

406 407 408 409 410
		if (this.contentWidgets.shouldRender()) {
			// Give the content widgets a chance to set their max width before a possible synchronous layout
			this.contentWidgets.onBeforeRender(viewportData);
		}

A
Alex Dima 已提交
411
		if (this.viewLines.shouldRender()) {
412
			this.viewLines.renderText(viewportData);
A
Alex Dima 已提交
413
			this.viewLines.onDidRender();
A
Alex Dima 已提交
414

A
Alex Dima 已提交
415 416 417
			// Rendering of viewLines might cause scroll events to occur, so collect view parts to render again
			viewPartsToRender = this._getViewPartsToRender();
		}
A
Alex Dima 已提交
418

M
Matt Bierner 已提交
419
		const renderingContext = new RenderingContext(this._context.viewLayout, viewportData, this.viewLines);
A
Alex Dima 已提交
420

A
Alex Dima 已提交
421 422
		// Render the rest of the parts
		for (let i = 0, len = viewPartsToRender.length; i < len; i++) {
M
Matt Bierner 已提交
423
			const viewPart = viewPartsToRender[i];
A
Alex Dima 已提交
424 425
			viewPart.prepareRender(renderingContext);
		}
426

A
Alex Dima 已提交
427
		for (let i = 0, len = viewPartsToRender.length; i < len; i++) {
M
Matt Bierner 已提交
428
			const viewPart = viewPartsToRender[i];
A
Alex Dima 已提交
429 430
			viewPart.render(renderingContext);
			viewPart.onDidRender();
E
Erich Gamma 已提交
431 432 433
		}
	}

A
Alex Dima 已提交
434 435
	// --- BEGIN CodeEditor helpers

436
	public delegateVerticalScrollbarMouseDown(browserEvent: IMouseEvent): void {
A
Alex Dima 已提交
437 438 439
		this._scrollbar.delegateVerticalScrollbarMouseDown(browserEvent);
	}

440 441
	public restoreState(scrollPosition: { scrollLeft: number; scrollTop: number; }): void {
		this._context.viewLayout.setScrollPositionNow({ scrollTop: scrollPosition.scrollTop });
442
		this._context.model.tokenizeViewport();
443 444 445 446 447
		this._renderNow();
		this.viewLines.updateLineWidths();
		this._context.viewLayout.setScrollPositionNow({ scrollLeft: scrollPosition.scrollLeft });
	}

A
Alex Dima 已提交
448
	public getOffsetForColumn(modelLineNumber: number, modelColumn: number): number {
M
Matt Bierner 已提交
449
		const modelPosition = this._context.model.validateModelPosition({
A
Alex Dima 已提交
450 451 452
			lineNumber: modelLineNumber,
			column: modelColumn
		});
M
Matt Bierner 已提交
453
		const viewPosition = this._context.model.coordinatesConverter.convertModelPositionToViewPosition(modelPosition);
A
Alex Dima 已提交
454
		this._flushAccumulatedAndRenderNow();
A
Alex Dima 已提交
455 456
		const visibleRange = this.viewLines.visibleRangeForPosition(new Position(viewPosition.lineNumber, viewPosition.column));
		if (!visibleRange) {
A
Alex Dima 已提交
457
			return -1;
E
Erich Gamma 已提交
458
		}
A
Alex Dima 已提交
459
		return visibleRange.left;
A
Alex Dima 已提交
460 461
	}

A
Alex Dima 已提交
462
	public getTargetAtClientPoint(clientX: number, clientY: number): editorBrowser.IMouseTarget | null {
A
Alex Dima 已提交
463 464 465
		return this.pointerHandler.getTargetAtClientPoint(clientX, clientY);
	}

466 467
	public createOverviewRuler(cssClassName: string): OverviewRuler {
		return new OverviewRuler(this._context, cssClassName);
E
Erich Gamma 已提交
468 469
	}

A
Alex Dima 已提交
470
	public change(callback: (changeAccessor: editorBrowser.IViewZoneChangeAccessor) => any): boolean {
A
Alex Dima 已提交
471
		let zonesHaveChanged = false;
A
Alex Dima 已提交
472

E
Erich Gamma 已提交
473
		this._renderOnce(() => {
M
Matt Bierner 已提交
474
			const changeAccessor: editorBrowser.IViewZoneChangeAccessor = {
A
Alex Dima 已提交
475
				addZone: (zone: editorBrowser.IViewZone): string => {
E
Erich Gamma 已提交
476 477 478
					zonesHaveChanged = true;
					return this.viewZones.addZone(zone);
				},
A
Alex Dima 已提交
479
				removeZone: (id: string): void => {
480 481 482
					if (!id) {
						return;
					}
E
Erich Gamma 已提交
483 484
					zonesHaveChanged = this.viewZones.removeZone(id) || zonesHaveChanged;
				},
A
Alex Dima 已提交
485
				layoutZone: (id: string): void => {
486 487 488
					if (!id) {
						return;
					}
E
Erich Gamma 已提交
489 490 491 492
					zonesHaveChanged = this.viewZones.layoutZone(id) || zonesHaveChanged;
				}
			};

A
Alex Dima 已提交
493
			safeInvoke1Arg(callback, changeAccessor);
E
Erich Gamma 已提交
494 495

			// Invalidate changeAccessor
A
Alex Dima 已提交
496 497 498
			changeAccessor.addZone = invalidFunc;
			changeAccessor.removeZone = invalidFunc;
			changeAccessor.layoutZone = invalidFunc;
E
Erich Gamma 已提交
499 500

			if (zonesHaveChanged) {
501
				this._context.viewLayout.onHeightMaybeChanged();
502
				this._context.privateViewEventBus.emit(new viewEvents.ViewZonesChangedEvent());
E
Erich Gamma 已提交
503 504 505 506 507
			}
		});
		return zonesHaveChanged;
	}

A
Alex Dima 已提交
508 509 510 511 512
	public render(now: boolean, everything: boolean): void {
		if (everything) {
			// Force everything to render...
			this.viewLines.forceShouldRender();
			for (let i = 0, len = this.viewParts.length; i < len; i++) {
M
Matt Bierner 已提交
513
				const viewPart = this.viewParts[i];
A
Alex Dima 已提交
514 515 516 517 518 519 520 521 522 523 524
				viewPart.forceShouldRender();
			}
		}
		if (now) {
			this._flushAccumulatedAndRenderNow();
		} else {
			this._scheduleRender();
		}
	}

	public focus(): void {
525
		this._textAreaHandler.focusTextArea();
A
Alex Dima 已提交
526 527 528
	}

	public isFocused(): boolean {
529
		return this._textAreaHandler.isFocused();
A
Alex Dima 已提交
530 531
	}

A
Alex Dima 已提交
532
	public addContentWidget(widgetData: IContentWidgetData): void {
A
Alex Dima 已提交
533 534 535
		this.contentWidgets.addWidget(widgetData.widget);
		this.layoutContentWidget(widgetData);
		this._scheduleRender();
E
Erich Gamma 已提交
536 537
	}

A
Alex Dima 已提交
538
	public layoutContentWidget(widgetData: IContentWidgetData): void {
M
Matt Bierner 已提交
539
		const newPosition = widgetData.position ? widgetData.position.position : null;
A
Alex Dima 已提交
540
		const newRange = widgetData.position ? widgetData.position.range || null : null;
M
Matt Bierner 已提交
541
		const newPreference = widgetData.position ? widgetData.position.preference : null;
A
Alex Dima 已提交
542
		this.contentWidgets.setWidgetPosition(widgetData.widget, newPosition, newRange, newPreference);
A
Alex Dima 已提交
543
		this._scheduleRender();
E
Erich Gamma 已提交
544 545
	}

A
Alex Dima 已提交
546
	public removeContentWidget(widgetData: IContentWidgetData): void {
A
Alex Dima 已提交
547 548
		this.contentWidgets.removeWidget(widgetData.widget);
		this._scheduleRender();
E
Erich Gamma 已提交
549 550
	}

A
Alex Dima 已提交
551
	public addOverlayWidget(widgetData: IOverlayWidgetData): void {
A
Alex Dima 已提交
552 553 554
		this.overlayWidgets.addWidget(widgetData.widget);
		this.layoutOverlayWidget(widgetData);
		this._scheduleRender();
E
Erich Gamma 已提交
555 556
	}

A
Alex Dima 已提交
557
	public layoutOverlayWidget(widgetData: IOverlayWidgetData): void {
M
Matt Bierner 已提交
558 559
		const newPreference = widgetData.position ? widgetData.position.preference : null;
		const shouldRender = this.overlayWidgets.setWidgetPosition(widgetData.widget, newPreference);
560 561 562
		if (shouldRender) {
			this._scheduleRender();
		}
E
Erich Gamma 已提交
563 564
	}

A
Alex Dima 已提交
565
	public removeOverlayWidget(widgetData: IOverlayWidgetData): void {
A
Alex Dima 已提交
566 567
		this.overlayWidgets.removeWidget(widgetData.widget);
		this._scheduleRender();
E
Erich Gamma 已提交
568 569
	}

A
Alex Dima 已提交
570
	// --- END CodeEditor helpers
E
Erich Gamma 已提交
571 572 573

}

J
Johannes Rieken 已提交
574
function safeInvokeNoArg(func: Function): any {
A
Alex Dima 已提交
575 576
	try {
		return func();
J
Johannes Rieken 已提交
577
	} catch (e) {
A
Alex Dima 已提交
578 579 580 581
		onUnexpectedError(e);
	}
}

J
Johannes Rieken 已提交
582
function safeInvoke1Arg(func: Function, arg1: any): any {
A
Alex Dima 已提交
583 584
	try {
		return func(arg1);
J
Johannes Rieken 已提交
585
	} catch (e) {
A
Alex Dima 已提交
586 587 588
		onUnexpectedError(e);
	}
}