viewImpl.ts 25.4 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';

J
Johannes Rieken 已提交
7
import { onUnexpectedError } from 'vs/base/common/errors';
A
Alex Dima 已提交
8
import { IDisposable } from 'vs/base/common/lifecycle';
A
Alex Dima 已提交
9 10
import * as browser from 'vs/base/browser/browser';
import * as dom from 'vs/base/browser/dom';
A
Alex Dima 已提交
11
import { FastDomNode, createFastDomNode } from 'vs/base/browser/fastDomNode';
J
Johannes Rieken 已提交
12 13
import { ICommandService } from 'vs/platform/commands/common/commands';
import { Range } from 'vs/editor/common/core/range';
A
Alex Dima 已提交
14
import * as editorCommon from 'vs/editor/common/editorCommon';
J
Johannes Rieken 已提交
15 16
import { ViewEventHandler } from 'vs/editor/common/viewModel/viewEventHandler';
import { Configuration } from 'vs/editor/browser/config/configuration';
17
import { TextAreaHandler, ITextAreaHandlerHelper } from 'vs/editor/browser/controller/textAreaHandler';
J
Johannes Rieken 已提交
18
import { PointerHandler } from 'vs/editor/browser/controller/pointerHandler';
A
Alex Dima 已提交
19
import * as editorBrowser from 'vs/editor/browser/editorBrowser';
20
import { ViewController, ExecCoreEditorCommandFunc } from 'vs/editor/browser/view/viewController';
21
import { ViewEventDispatcher } from 'vs/editor/common/view/viewEventDispatcher';
J
Johannes Rieken 已提交
22
import { ContentViewOverlays, MarginViewOverlays } from 'vs/editor/browser/view/viewOverlays';
23
import { ViewLayout } from 'vs/editor/common/viewLayout/viewLayout';
J
Johannes Rieken 已提交
24 25
import { ViewContentWidgets } from 'vs/editor/browser/viewParts/contentWidgets/contentWidgets';
import { CurrentLineHighlightOverlay } from 'vs/editor/browser/viewParts/currentLineHighlight/currentLineHighlight';
26
import { CurrentLineMarginHighlightOverlay } from 'vs/editor/browser/viewParts/currentLineMarginHighlight/currentLineMarginHighlight';
J
Johannes Rieken 已提交
27 28 29 30 31
import { DecorationsOverlay } from 'vs/editor/browser/viewParts/decorations/decorations';
import { GlyphMarginOverlay } from 'vs/editor/browser/viewParts/glyphMargin/glyphMargin';
import { LineNumbersOverlay } from 'vs/editor/browser/viewParts/lineNumbers/lineNumbers';
import { IndentGuidesOverlay } from 'vs/editor/browser/viewParts/indentGuides/indentGuides';
import { ViewLines } from 'vs/editor/browser/viewParts/lines/viewLines';
32
import { Margin } from 'vs/editor/browser/viewParts/margin/margin';
J
Johannes Rieken 已提交
33
import { LinesDecorationsOverlay } from 'vs/editor/browser/viewParts/linesDecorations/linesDecorations';
34
import { MarginViewLineDecorationsOverlay } from 'vs/editor/browser/viewParts/marginDecorations/marginDecorations';
J
Johannes Rieken 已提交
35 36 37 38 39 40 41 42
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 已提交
43
import { ViewPart, PartFingerprint, PartFingerprints } from 'vs/editor/browser/view/viewPart';
44
import { ViewContext } from 'vs/editor/common/view/viewContext';
J
Johannes Rieken 已提交
45
import { IViewModel } from 'vs/editor/common/viewModel/viewModel';
A
Alex Dima 已提交
46
import { RenderingContext } from 'vs/editor/common/view/renderingContext';
J
Johannes Rieken 已提交
47
import { IPointerHandlerHelper } from 'vs/editor/browser/controller/mouseHandler';
A
Alex Dima 已提交
48
import { ViewOutgoingEvents } from 'vs/editor/browser/view/viewOutgoingEvents';
49
import { ViewportData } from 'vs/editor/common/viewLayout/viewLinesViewportData';
A
Alex Dima 已提交
50
import { EditorScrollbar } from 'vs/editor/browser/viewParts/editorScrollbar/editorScrollbar';
A
Alex Dima 已提交
51
import { Minimap } from 'vs/editor/browser/viewParts/minimap/minimap';
52
import * as viewEvents from 'vs/editor/common/view/viewEvents';
53
import { IEditorWhitespace } from 'vs/editor/common/viewLayout/whitespaceComputer';
E
Erich Gamma 已提交
54

A
Alex Dima 已提交
55 56 57 58 59 60 61 62 63 64 65
export interface IContentWidgetData {
	widget: editorBrowser.IContentWidget;
	position: editorBrowser.IContentWidgetPosition;
}

export interface IOverlayWidgetData {
	widget: editorBrowser.IOverlayWidget;
	position: editorBrowser.IOverlayWidgetPosition;
}

export class View extends ViewEventHandler {
E
Erich Gamma 已提交
66

J
Johannes Rieken 已提交
67
	private eventDispatcher: ViewEventDispatcher;
E
Erich Gamma 已提交
68

69
	private layoutProvider: ViewLayout;
70
	private _scrollbar: EditorScrollbar;
71
	private _context: ViewContext;
E
Erich Gamma 已提交
72 73 74 75 76 77 78 79

	// 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;
80
	private viewCursors: ViewCursors;
A
Alex Dima 已提交
81
	private viewParts: ViewPart[];
E
Erich Gamma 已提交
82

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

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

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

	// Actual mutable state
	private _isDisposed: boolean;

A
Alex Dima 已提交
96
	private _renderAnimationFrame: IDisposable;
E
Erich Gamma 已提交
97

A
Alex Dima 已提交
98
	constructor(
99
		commandService: ICommandService,
J
Johannes Rieken 已提交
100 101
		configuration: Configuration,
		model: IViewModel,
102
		execCoreEditorCommandFunc: ExecCoreEditorCommandFunc
A
Alex Dima 已提交
103
	) {
E
Erich Gamma 已提交
104 105 106
		super();
		this._isDisposed = false;
		this._renderAnimationFrame = null;
A
Alex Dima 已提交
107
		this.outgoingEvents = new ViewOutgoingEvents(model);
E
Erich Gamma 已提交
108

109
		let viewController = new ViewController(model, execCoreEditorCommandFunc, this.outgoingEvents, commandService);
E
Erich Gamma 已提交
110 111

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

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

E
Erich Gamma 已提交
117 118 119 120
		// The layout provider has such responsibilities as:
		// - scrolling (i.e. viewport / full size) & co.
		// - whitespaces (a.k.a. view zones) management & co.
		// - line heights updating & co.
121
		this.layoutProvider = new ViewLayout(configuration, model.getLineCount(), this.eventDispatcher);
E
Erich Gamma 已提交
122 123

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

126 127
		this.viewParts = [];

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

E
Erich Gamma 已提交
132
		this.createViewParts();
A
Alex Dima 已提交
133
		this._setLayout();
E
Erich Gamma 已提交
134 135

		// Pointer handler
136
		this.pointerHandler = new PointerHandler(this._context, viewController, this.createPointerHandlerHelper());
E
Erich Gamma 已提交
137

A
Alex Dima 已提交
138 139 140
		this._register(model.addEventListener((events: viewEvents.ViewEvent[]) => {
			this.eventDispatcher.emitMany(events);
		}));
E
Erich Gamma 已提交
141 142 143
	}

	private createViewParts(): void {
A
Alex Dima 已提交
144
		// These two dom nodes must be constructed up front, since references are needed in the layout provider (scrolling & co.)
A
Alex Dima 已提交
145
		this.linesContent = createFastDomNode(document.createElement('div'));
A
Alex Dima 已提交
146
		this.linesContent.setClassName('lines-content' + ' monaco-editor-background');
A
Alex Dima 已提交
147
		this.linesContent.setPosition('absolute');
A
Alex Dima 已提交
148

A
Alex Dima 已提交
149 150
		this.domNode = createFastDomNode(document.createElement('div'));
		this.domNode.setClassName(this._context.configuration.editor.viewInfo.editorClassName);
A
Alex Dima 已提交
151

A
Alex Dima 已提交
152
		this.overflowGuardContainer = createFastDomNode(document.createElement('div'));
A
Alex Dima 已提交
153
		PartFingerprints.write(this.overflowGuardContainer, PartFingerprint.OverflowGuard);
A
Alex Dima 已提交
154
		this.overflowGuardContainer.setClassName('overflow-guard');
A
Alex Dima 已提交
155

A
Alex Dima 已提交
156 157 158
		this._scrollbar = new EditorScrollbar(this._context, this.layoutProvider.getScrollable(), this.linesContent, this.domNode, this.overflowGuardContainer);
		this.viewParts.push(this._scrollbar);

E
Erich Gamma 已提交
159
		// View Lines
A
Alex Dima 已提交
160
		this.viewLines = new ViewLines(this._context, this.linesContent, this.layoutProvider);
E
Erich Gamma 已提交
161 162

		// View Zones
163
		this.viewZones = new ViewZones(this._context, this.layoutProvider);
E
Erich Gamma 已提交
164 165 166
		this.viewParts.push(this.viewZones);

		// Decorations overview ruler
A
Alex Dima 已提交
167
		let decorationsOverviewRuler = new DecorationsOverviewRuler(
J
Johannes Rieken 已提交
168 169
			this._context, this.layoutProvider.getScrollHeight(),
			(lineNumber: number) => this.layoutProvider.getVerticalOffsetForLineNumber(lineNumber)
E
Erich Gamma 已提交
170 171 172 173
		);
		this.viewParts.push(decorationsOverviewRuler);


A
Alex Dima 已提交
174
		let scrollDecoration = new ScrollDecorationViewPart(this._context);
E
Erich Gamma 已提交
175 176
		this.viewParts.push(scrollDecoration);

A
Alex Dima 已提交
177
		let contentViewOverlays = new ContentViewOverlays(this._context);
E
Erich Gamma 已提交
178
		this.viewParts.push(contentViewOverlays);
A
Alex Dima 已提交
179
		contentViewOverlays.addDynamicOverlay(new CurrentLineHighlightOverlay(this._context));
180 181
		contentViewOverlays.addDynamicOverlay(new SelectionsOverlay(this._context));
		contentViewOverlays.addDynamicOverlay(new DecorationsOverlay(this._context));
182
		contentViewOverlays.addDynamicOverlay(new IndentGuidesOverlay(this._context));
E
Erich Gamma 已提交
183

A
Alex Dima 已提交
184
		let marginViewOverlays = new MarginViewOverlays(this._context);
E
Erich Gamma 已提交
185
		this.viewParts.push(marginViewOverlays);
A
Alex Dima 已提交
186
		marginViewOverlays.addDynamicOverlay(new CurrentLineMarginHighlightOverlay(this._context));
187
		marginViewOverlays.addDynamicOverlay(new GlyphMarginOverlay(this._context));
188
		marginViewOverlays.addDynamicOverlay(new MarginViewLineDecorationsOverlay(this._context));
189 190
		marginViewOverlays.addDynamicOverlay(new LinesDecorationsOverlay(this._context));
		marginViewOverlays.addDynamicOverlay(new LineNumbersOverlay(this._context));
E
Erich Gamma 已提交
191

A
Alex Dima 已提交
192
		let margin = new Margin(this._context);
A
Alex Dima 已提交
193
		margin.getDomNode().appendChild(this.viewZones.marginDomNode);
A
Alex Dima 已提交
194
		margin.getDomNode().appendChild(marginViewOverlays.getDomNode());
195
		this.viewParts.push(margin);
E
Erich Gamma 已提交
196 197

		// Content widgets
198
		this.contentWidgets = new ViewContentWidgets(this._context, this.domNode);
E
Erich Gamma 已提交
199 200
		this.viewParts.push(this.contentWidgets);

201 202
		this.viewCursors = new ViewCursors(this._context);
		this.viewParts.push(this.viewCursors);
E
Erich Gamma 已提交
203 204

		// Overlay widgets
205
		this.overlayWidgets = new ViewOverlayWidgets(this._context);
E
Erich Gamma 已提交
206 207
		this.viewParts.push(this.overlayWidgets);

A
Alex Dima 已提交
208
		let rulers = new Rulers(this._context);
209 210
		this.viewParts.push(rulers);

211
		let minimap = new Minimap(this._context, this.layoutProvider, this._scrollbar);
A
Alex Dima 已提交
212 213
		this.viewParts.push(minimap);

E
Erich Gamma 已提交
214 215 216
		// -------------- Wire dom nodes up

		if (decorationsOverviewRuler) {
217
			let overviewRulerData = this._scrollbar.getOverviewRulerLayoutInfo();
E
Erich Gamma 已提交
218 219 220
			overviewRulerData.parent.insertBefore(decorationsOverviewRuler.getDomNode(), overviewRulerData.insertBefore);
		}

A
Alex Dima 已提交
221 222 223 224 225 226 227 228 229 230
		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());
		this.overflowGuardContainer.appendChild(this.overlayWidgets.getDomNode());
231 232
		this.overflowGuardContainer.appendChild(this._textAreaHandler.textArea);
		this.overflowGuardContainer.appendChild(this._textAreaHandler.textAreaCover);
A
Alex Dima 已提交
233 234 235
		this.overflowGuardContainer.appendChild(minimap.getDomNode());
		this.domNode.appendChild(this.overflowGuardContainer);
		this.domNode.appendChild(this.contentWidgets.overflowingContentWidgetsDomNode);
E
Erich Gamma 已提交
236 237 238 239 240 241
	}

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

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

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

			getScrollLeft: () => {
				return this.layoutProvider.getScrollLeft();
			},
254 255 256 257
			getScrollTop: () => {
				return this.layoutProvider.getScrollTop();
			},

J
Johannes Rieken 已提交
258
			setScrollPosition: (position: editorCommon.INewScrollPosition) => {
259
				this.layoutProvider.setScrollPosition(position);
E
Erich Gamma 已提交
260 261 262 263 264 265 266 267 268 269 270 271 272 273
			},

			isAfterLines: (verticalOffset: number) => {
				return this.layoutProvider.isAfterLines(verticalOffset);
			},
			getLineNumberAtVerticalOffset: (verticalOffset: number) => {
				return this.layoutProvider.getLineNumberAtVerticalOffset(verticalOffset);
			},
			getVerticalOffsetForLineNumber: (lineNumber: number) => {
				return this.layoutProvider.getVerticalOffsetForLineNumber(lineNumber);
			},
			getWhitespaceAtVerticalOffset: (verticalOffset: number) => {
				return this.layoutProvider.getWhitespaceAtVerticalOffset(verticalOffset);
			},
274 275 276
			getLastViewCursorsRenderData: () => {
				return this.viewCursors.getLastRenderData() || [];
			},
E
Erich Gamma 已提交
277 278 279
			shouldSuppressMouseDownOnViewZone: (viewZoneId: number) => {
				return this.viewZones.shouldSuppressMouseDownOnViewZone(viewZoneId);
			},
280 281 282
			shouldSuppressMouseDownOnWidget: (widgetId: string) => {
				return this.contentWidgets.shouldSuppressMouseDownOnWidget(widgetId);
			},
E
Erich Gamma 已提交
283 284 285 286 287 288 289
			getPositionFromDOMInfo: (spanNode: HTMLElement, offset: number) => {
				this._flushAccumulatedAndRenderNow();
				return this.viewLines.getPositionFromDOMInfo(spanNode, offset);
			},

			visibleRangeForPosition2: (lineNumber: number, column: number) => {
				this._flushAccumulatedAndRenderNow();
290
				let visibleRanges = this.viewLines.visibleRangesForRange2(new Range(lineNumber, column, lineNumber, column));
E
Erich Gamma 已提交
291 292 293 294 295 296 297 298 299 300 301 302 303
				if (!visibleRanges) {
					return null;
				}
				return visibleRanges[0];
			},

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

304
	private createTextAreaHandlerHelper(): ITextAreaHandlerHelper {
E
Erich Gamma 已提交
305 306 307
		return {
			visibleRangeForPositionRelativeToEditor: (lineNumber: number, column: number) => {
				this._flushAccumulatedAndRenderNow();
308
				let visibleRanges = this.viewLines.visibleRangesForRange2(new Range(lineNumber, column, lineNumber, column));
E
Erich Gamma 已提交
309 310 311 312
				if (!visibleRanges) {
					return null;
				}
				return visibleRanges[0];
313
			},
314 315
			getVerticalOffsetForLineNumber: (lineNumber: number) => {
				return this.layoutProvider.getVerticalOffsetForLineNumber(lineNumber);
E
Erich Gamma 已提交
316 317 318 319
			}
		};
	}

A
Alex Dima 已提交
320 321
	private _setLayout(): void {
		const layoutInfo = this._context.configuration.editor.layoutInfo;
A
Alex Dima 已提交
322
		if (browser.isChrome) {
A
tslint  
Alex Dima 已提交
323
			/* tslint:disable:no-unused-variable */
E
Erich Gamma 已提交
324 325
			// Access overflowGuardContainer.clientWidth to prevent relayouting bug in Chrome
			// See Bug 19676: Editor misses a layout event
A
Alex Dima 已提交
326
			let clientWidth = this.overflowGuardContainer.domNode.clientWidth + 'px';
A
tslint  
Alex Dima 已提交
327
			/* tslint:enable:no-unused-variable */
E
Erich Gamma 已提交
328
		}
A
Alex Dima 已提交
329 330
		this.domNode.setWidth(layoutInfo.width);
		this.domNode.setHeight(layoutInfo.height);
E
Erich Gamma 已提交
331

A
Alex Dima 已提交
332 333
		this.overflowGuardContainer.setWidth(layoutInfo.width);
		this.overflowGuardContainer.setHeight(layoutInfo.height);
E
Erich Gamma 已提交
334

A
Alex Dima 已提交
335 336
		this.linesContent.setWidth(1000000);
		this.linesContent.setHeight(1000000);
E
Erich Gamma 已提交
337 338

	}
339 340 341

	// --- begin event handlers

A
Alex Dima 已提交
342
	public onConfigurationChanged(e: viewEvents.ViewConfigurationChangedEvent): boolean {
A
Alex Dima 已提交
343
		if (e.viewInfo) {
A
Alex Dima 已提交
344
			this.domNode.setClassName(this._context.configuration.editor.viewInfo.editorClassName);
E
Erich Gamma 已提交
345
		}
A
Alex Dima 已提交
346 347 348
		if (e.layoutInfo) {
			this._setLayout();
		}
A
Alex Dima 已提交
349
		this.layoutProvider.onConfigurationChanged(e);
E
Erich Gamma 已提交
350 351
		return false;
	}
352 353
	public onFlushed(e: viewEvents.ViewFlushedEvent): boolean {
		this.layoutProvider.onFlushed(this._context.model.getLineCount());
354
		return false;
E
Erich Gamma 已提交
355
	}
A
Alex Dima 已提交
356
	public onFocusChanged(e: viewEvents.ViewFocusChangedEvent): boolean {
A
Alex Dima 已提交
357
		this.domNode.toggleClassName('focused', e.isFocused);
A
Alex Dima 已提交
358
		if (e.isFocused) {
A
Alex Dima 已提交
359
			this.outgoingEvents.emitViewFocusGained();
E
Erich Gamma 已提交
360
		} else {
A
Alex Dima 已提交
361
			this.outgoingEvents.emitViewFocusLost();
E
Erich Gamma 已提交
362 363 364
		}
		return false;
	}
365 366 367 368 369 370 371 372 373 374 375 376
	public onLinesDeleted(e: viewEvents.ViewLinesDeletedEvent): boolean {
		this.layoutProvider.onLinesDeleted(e);
		return false;
	}
	public onLinesInserted(e: viewEvents.ViewLinesInsertedEvent): boolean {
		this.layoutProvider.onLinesInserted(e);
		return false;
	}
	public onScrollChanged(e: viewEvents.ViewScrollChangedEvent): boolean {
		this.outgoingEvents.emitScrollChanged(e);
		return false;
	}
377 378
	public onScrollRequest(e: viewEvents.ViewScrollRequestEvent): boolean {
		this.layoutProvider.setScrollPosition({
379
			scrollTop: e.desiredScrollTop
380
		});
381 382
		return false;
	}
383

E
Erich Gamma 已提交
384 385 386 387 388 389 390 391 392 393
	// --- end event handlers

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

		this.eventDispatcher.removeEventHandler(this);
A
Alex Dima 已提交
394
		this.outgoingEvents.dispose();
E
Erich Gamma 已提交
395 396 397 398 399

		this.pointerHandler.dispose();

		this.viewLines.dispose();

A
Alex Dima 已提交
400
		// Destroy view parts
A
Alex Dima 已提交
401
		for (let i = 0, len = this.viewParts.length; i < len; i++) {
E
Erich Gamma 已提交
402 403 404 405 406
			this.viewParts[i].dispose();
		}
		this.viewParts = [];

		this.layoutProvider.dispose();
A
Alex Dima 已提交
407 408

		super.dispose();
E
Erich Gamma 已提交
409 410
	}

A
Alex Dima 已提交
411 412 413 414 415
	private _renderOnce(callback: () => any): any {
		let r = safeInvokeNoArg(callback);
		this._scheduleRender();
		return r;
	}
E
Erich Gamma 已提交
416

A
Alex Dima 已提交
417 418 419 420 421
	private _scheduleRender(): void {
		if (this._renderAnimationFrame === null) {
			this._renderAnimationFrame = dom.runAtThisOrScheduleAtNextAnimationFrame(this._onRenderScheduled.bind(this), 100);
		}
	}
A
Alex Dima 已提交
422

A
Alex Dima 已提交
423 424 425 426
	private _onRenderScheduled(): void {
		this._renderAnimationFrame = null;
		this._flushAccumulatedAndRenderNow();
	}
427

A
Alex Dima 已提交
428 429 430
	private _renderNow(): void {
		safeInvokeNoArg(() => this._actualRender());
	}
A
Alex Dima 已提交
431

A
Alex Dima 已提交
432 433 434 435 436 437 438 439 440 441
	private _getViewPartsToRender(): ViewPart[] {
		let result: ViewPart[] = [];
		for (let i = 0, len = this.viewParts.length; i < len; i++) {
			let viewPart = this.viewParts[i];
			if (viewPart.shouldRender()) {
				result.push(viewPart);
			}
		}
		return result;
	}
442

A
Alex Dima 已提交
443 444 445 446
	private _actualRender(): void {
		if (!dom.isInDOM(this.domNode.domNode)) {
			return;
		}
447

A
Alex Dima 已提交
448
		let viewPartsToRender = this._getViewPartsToRender();
A
Alex Dima 已提交
449

A
Alex Dima 已提交
450 451
		if (!this.viewLines.shouldRender() && viewPartsToRender.length === 0) {
			// Nothing to render
452
			this._textAreaHandler.writeToTextArea();
A
Alex Dima 已提交
453 454
			return;
		}
A
Alex Dima 已提交
455

A
Alex Dima 已提交
456 457
		let partialViewportData = this.layoutProvider.getLinesViewportData();
		this._context.model.setViewport(partialViewportData.startLineNumber, partialViewportData.endLineNumber, partialViewportData.centeredLineNumber);
458

A
Alex Dima 已提交
459
		let viewportData = new ViewportData(partialViewportData, this._context.model);
A
Alex Dima 已提交
460

A
Alex Dima 已提交
461 462
		if (this.viewLines.shouldRender()) {
			this.viewLines.renderText(viewportData, () => {
463
				this._textAreaHandler.writeToTextArea();
A
Alex Dima 已提交
464 465
			});
			this.viewLines.onDidRender();
A
Alex Dima 已提交
466

A
Alex Dima 已提交
467 468 469
			// Rendering of viewLines might cause scroll events to occur, so collect view parts to render again
			viewPartsToRender = this._getViewPartsToRender();
		} else {
470
			this._textAreaHandler.writeToTextArea();
A
Alex Dima 已提交
471
		}
A
Alex Dima 已提交
472

A
Alex Dima 已提交
473
		let renderingContext = new RenderingContext(this.layoutProvider, viewportData, this.viewLines);
A
Alex Dima 已提交
474

A
Alex Dima 已提交
475 476 477 478 479
		// Render the rest of the parts
		for (let i = 0, len = viewPartsToRender.length; i < len; i++) {
			let viewPart = viewPartsToRender[i];
			viewPart.prepareRender(renderingContext);
		}
480

A
Alex Dima 已提交
481 482 483 484
		for (let i = 0, len = viewPartsToRender.length; i < len; i++) {
			let viewPart = viewPartsToRender[i];
			viewPart.render(renderingContext);
			viewPart.onDidRender();
E
Erich Gamma 已提交
485 486 487
		}
	}

A
Alex Dima 已提交
488 489 490 491
	// --- BEGIN CodeEditor helpers

	public getScrollWidth(): number {
		return this.layoutProvider.getScrollWidth();
E
Erich Gamma 已提交
492 493
	}

A
Alex Dima 已提交
494 495
	public getScrollLeft(): number {
		return this.layoutProvider.getScrollLeft();
E
Erich Gamma 已提交
496 497
	}

A
Alex Dima 已提交
498 499 500
	public getScrollHeight(): number {
		return this.layoutProvider.getScrollHeight();
	}
E
Erich Gamma 已提交
501

A
Alex Dima 已提交
502 503
	public getScrollTop(): number {
		return this.layoutProvider.getScrollTop();
E
Erich Gamma 已提交
504 505
	}

A
Alex Dima 已提交
506 507 508 509
	public setScrollPosition(scrollPosition: editorCommon.INewScrollPosition): void {
		this.layoutProvider.setScrollPosition(scrollPosition);
	}

510 511
	public getVerticalOffsetForViewLineNumber(viewLineNumber: number): number {
		return this.layoutProvider.getVerticalOffsetForLineNumber(viewLineNumber);
A
Alex Dima 已提交
512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527
	}

	public delegateVerticalScrollbarMouseDown(browserEvent: MouseEvent): void {
		this._scrollbar.delegateVerticalScrollbarMouseDown(browserEvent);
	}

	public getOffsetForColumn(modelLineNumber: number, modelColumn: number): number {
		let modelPosition = this._context.model.validateModelPosition({
			lineNumber: modelLineNumber,
			column: modelColumn
		});
		let viewPosition = this._context.model.coordinatesConverter.convertModelPositionToViewPosition(modelPosition);
		this._flushAccumulatedAndRenderNow();
		let visibleRanges = this.viewLines.visibleRangesForRange2(new Range(viewPosition.lineNumber, viewPosition.column, viewPosition.lineNumber, viewPosition.column));
		if (!visibleRanges) {
			return -1;
E
Erich Gamma 已提交
528
		}
A
Alex Dima 已提交
529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546
		return visibleRanges[0].left;
	}

	public getTargetAtClientPoint(clientX: number, clientY: number): editorBrowser.IMouseTarget {
		return this.pointerHandler.getTargetAtClientPoint(clientX, clientY);
	}

	public getCompletelyVisibleViewRange(): Range {
		const partialData = this.layoutProvider.getLinesViewportData();
		const startViewLineNumber = partialData.completelyVisibleStartLineNumber;
		const endViewLineNumber = partialData.completelyVisibleEndLineNumber;

		return new Range(
			startViewLineNumber, this._context.model.getLineMinColumn(startViewLineNumber),
			endViewLineNumber, this._context.model.getLineMaxColumn(endViewLineNumber)
		);
	}

547 548 549 550 551 552 553 554 555 556 557
	public getCompletelyVisibleViewRangeAtScrollTop(scrollTop: number): Range {
		const partialData = this.layoutProvider.getLinesViewportDataAtScrollTop(scrollTop);
		const startViewLineNumber = partialData.completelyVisibleStartLineNumber;
		const endViewLineNumber = partialData.completelyVisibleEndLineNumber;

		return new Range(
			startViewLineNumber, this._context.model.getLineMinColumn(startViewLineNumber),
			endViewLineNumber, this._context.model.getLineMaxColumn(endViewLineNumber)
		);
	}

A
Alex Dima 已提交
558 559
	public getInternalEventBus(): ViewOutgoingEvents {
		return this.outgoingEvents;
E
Erich Gamma 已提交
560 561 562 563
	}

	public createOverviewRuler(cssClassName: string, minimumHeight: number, maximumHeight: number): OverviewRuler {
		return new OverviewRuler(
J
Johannes Rieken 已提交
564 565
			this._context, cssClassName, this.layoutProvider.getScrollHeight(), minimumHeight, maximumHeight,
			(lineNumber: number) => this.layoutProvider.getVerticalOffsetForLineNumber(lineNumber)
E
Erich Gamma 已提交
566 567 568
		);
	}

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

E
Erich Gamma 已提交
572
		this._renderOnce(() => {
A
Alex Dima 已提交
573
			let changeAccessor: editorBrowser.IViewZoneChangeAccessor = {
J
Johannes Rieken 已提交
574
				addZone: (zone: editorBrowser.IViewZone): number => {
E
Erich Gamma 已提交
575 576 577
					zonesHaveChanged = true;
					return this.viewZones.addZone(zone);
				},
J
Johannes Rieken 已提交
578
				removeZone: (id: number): void => {
579 580 581
					if (!id) {
						return;
					}
E
Erich Gamma 已提交
582 583 584
					zonesHaveChanged = this.viewZones.removeZone(id) || zonesHaveChanged;
				},
				layoutZone: (id: number): void => {
585 586 587
					if (!id) {
						return;
					}
E
Erich Gamma 已提交
588 589 590 591
					zonesHaveChanged = this.viewZones.layoutZone(id) || zonesHaveChanged;
				}
			};

A
Alex Dima 已提交
592
			safeInvoke1Arg(callback, changeAccessor);
E
Erich Gamma 已提交
593 594 595 596 597 598

			// Invalidate changeAccessor
			changeAccessor.addZone = null;
			changeAccessor.removeZone = null;

			if (zonesHaveChanged) {
A
Alex Dima 已提交
599
				this.layoutProvider.onHeightMaybeChanged();
600
				this._context.privateViewEventBus.emit(new viewEvents.ViewZonesChangedEvent());
E
Erich Gamma 已提交
601 602 603 604 605
			}
		});
		return zonesHaveChanged;
	}

A
Alex Dima 已提交
606
	public getWhitespaces(): IEditorWhitespace[] {
E
Erich Gamma 已提交
607 608 609
		return this.layoutProvider.getWhitespaces();
	}

A
Alex Dima 已提交
610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626
	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++) {
				let viewPart = this.viewParts[i];
				viewPart.forceShouldRender();
			}
		}
		if (now) {
			this._flushAccumulatedAndRenderNow();
		} else {
			this._scheduleRender();
		}
	}

	public setAriaActiveDescendant(id: string): void {
627
		this._textAreaHandler.setAriaActiveDescendant(id);
A
Alex Dima 已提交
628 629 630 631 632 633 634 635 636 637 638
	}

	public saveState(): editorCommon.IViewState {
		return this.layoutProvider.saveState();
	}

	public restoreState(state: editorCommon.IViewState): void {
		return this.layoutProvider.restoreState(state);
	}

	public focus(): void {
639
		this._textAreaHandler.focusTextArea();
A
Alex Dima 已提交
640 641 642
	}

	public isFocused(): boolean {
643
		return this._textAreaHandler.isFocused();
A
Alex Dima 已提交
644 645
	}

A
Alex Dima 已提交
646
	public addContentWidget(widgetData: IContentWidgetData): void {
A
Alex Dima 已提交
647 648 649
		this.contentWidgets.addWidget(widgetData.widget);
		this.layoutContentWidget(widgetData);
		this._scheduleRender();
E
Erich Gamma 已提交
650 651
	}

A
Alex Dima 已提交
652
	public layoutContentWidget(widgetData: IContentWidgetData): void {
A
Alex Dima 已提交
653 654 655 656
		let newPosition = widgetData.position ? widgetData.position.position : null;
		let newPreference = widgetData.position ? widgetData.position.preference : null;
		this.contentWidgets.setWidgetPosition(widgetData.widget, newPosition, newPreference);
		this._scheduleRender();
E
Erich Gamma 已提交
657 658
	}

A
Alex Dima 已提交
659
	public removeContentWidget(widgetData: IContentWidgetData): void {
A
Alex Dima 已提交
660 661
		this.contentWidgets.removeWidget(widgetData.widget);
		this._scheduleRender();
E
Erich Gamma 已提交
662 663
	}

A
Alex Dima 已提交
664
	public addOverlayWidget(widgetData: IOverlayWidgetData): void {
A
Alex Dima 已提交
665 666 667
		this.overlayWidgets.addWidget(widgetData.widget);
		this.layoutOverlayWidget(widgetData);
		this._scheduleRender();
E
Erich Gamma 已提交
668 669
	}

A
Alex Dima 已提交
670
	public layoutOverlayWidget(widgetData: IOverlayWidgetData): void {
671 672 673 674 675
		let newPreference = widgetData.position ? widgetData.position.preference : null;
		let shouldRender = this.overlayWidgets.setWidgetPosition(widgetData.widget, newPreference);
		if (shouldRender) {
			this._scheduleRender();
		}
E
Erich Gamma 已提交
676 677
	}

A
Alex Dima 已提交
678
	public removeOverlayWidget(widgetData: IOverlayWidgetData): void {
A
Alex Dima 已提交
679 680
		this.overlayWidgets.removeWidget(widgetData.widget);
		this._scheduleRender();
E
Erich Gamma 已提交
681 682
	}

A
Alex Dima 已提交
683
	// --- END CodeEditor helpers
E
Erich Gamma 已提交
684 685 686

}

J
Johannes Rieken 已提交
687
function safeInvokeNoArg(func: Function): any {
A
Alex Dima 已提交
688 689
	try {
		return func();
J
Johannes Rieken 已提交
690
	} catch (e) {
A
Alex Dima 已提交
691 692 693 694
		onUnexpectedError(e);
	}
}

J
Johannes Rieken 已提交
695
function safeInvoke1Arg(func: Function, arg1: any): any {
A
Alex Dima 已提交
696 697
	try {
		return func(arg1);
J
Johannes Rieken 已提交
698
	} catch (e) {
A
Alex Dima 已提交
699 700 701
		onUnexpectedError(e);
	}
}