notebookBrowser.ts 15.8 KB
Newer Older
R
rebornix 已提交
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 { IMouseWheelEvent } from 'vs/base/browser/mouseEvent';
R
rebornix 已提交
7 8
import { IListEvent, IListMouseEvent } from 'vs/base/browser/ui/list/list';
import { IListOptions, IListStyles } from 'vs/base/browser/ui/list/listWidget';
9
import { ProgressBar } from 'vs/base/browser/ui/progressbar/progressbar';
10
import { ToolBar } from 'vs/base/browser/ui/toolbar/toolbar';
11
import { CancellationTokenSource } from 'vs/base/common/cancellation';
R
rebornix 已提交
12 13 14
import { Event } from 'vs/base/common/event';
import { DisposableStore, IDisposable } from 'vs/base/common/lifecycle';
import { ScrollEvent } from 'vs/base/common/scrollable';
R
rebornix 已提交
15
import { URI } from 'vs/base/common/uri';
16 17 18 19
import { CodeEditorWidget } from 'vs/editor/browser/widget/codeEditorWidget';
import { BareFontInfo } from 'vs/editor/common/config/fontInfo';
import { Range } from 'vs/editor/common/core/range';
import { FindMatch } from 'vs/editor/common/model';
20
import { RawContextKey, ContextKeyExpr } from 'vs/platform/contextkey/common/contextkey';
21
import { OutputRenderer } from 'vs/workbench/contrib/notebook/browser/view/output/outputRenderer';
22
import { CellViewModel, IModelDecorationsChangeAccessor, NotebookViewModel } from 'vs/workbench/contrib/notebook/browser/viewModel/notebookViewModel';
R
rebornix 已提交
23
import { NotebookCellTextModel } from 'vs/workbench/contrib/notebook/common/model/notebookCellTextModel';
R
rebornix 已提交
24
import { CellKind, IOutput, IRenderOutput, NotebookCellMetadata, NotebookDocumentMetadata } from 'vs/workbench/contrib/notebook/common/notebookCommon';
R
rebornix 已提交
25
import { Webview } from 'vs/workbench/contrib/webview/browser/webview';
26
import { CellLanguageStatusBarItem } from 'vs/workbench/contrib/notebook/browser/view/renderers/cellRenderer';
R
rebornix 已提交
27 28

export const KEYBINDING_CONTEXT_NOTEBOOK_FIND_WIDGET_FOCUSED = new RawContextKey<boolean>('notebookFindWidgetFocused', false);
R
rebornix 已提交
29

30 31 32
// Is Notebook
export const NOTEBOOK_IS_ACTIVE_EDITOR = ContextKeyExpr.equals('activeEditor', 'workbench.editor.notebook');

33
// Editor keys
34
export const NOTEBOOK_EDITOR_FOCUSED = new RawContextKey<boolean>('notebookEditorFocused', false);
35 36 37
export const NOTEBOOK_EDITOR_EDITABLE = new RawContextKey<boolean>('notebookEditable', true);
export const NOTEBOOK_EDITOR_RUNNABLE = new RawContextKey<boolean>('notebookRunnable', true);
export const NOTEBOOK_EDITOR_EXECUTING_NOTEBOOK = new RawContextKey<boolean>('notebookExecuting', false);
38

39 40 41 42 43 44 45
// Cell keys
export const NOTEBOOK_VIEW_TYPE = new RawContextKey<string>('notebookViewType', undefined);
export const NOTEBOOK_CELL_TYPE = new RawContextKey<string>('notebookCellType', undefined); // code, markdown
export const NOTEBOOK_CELL_EDITABLE = new RawContextKey<boolean>('notebookCellEditable', false); // bool
export const NOTEBOOK_CELL_RUNNABLE = new RawContextKey<boolean>('notebookCellRunnable', false); // bool
export const NOTEBOOK_CELL_MARKDOWN_EDIT_MODE = new RawContextKey<boolean>('notebookCellMarkdownEditMode', false); // bool
export const NOTEBOOK_CELL_RUN_STATE = new RawContextKey<string>('notebookCellRunState', undefined); // idle, running
R
rebornix 已提交
46
export const NOTEBOOK_CELL_HAS_OUTPUTS = new RawContextKey<boolean>('notebookCellHasOutputs', false); // bool
47

R
rebornix 已提交
48 49 50 51 52 53
export interface NotebookLayoutInfo {
	width: number;
	height: number;
	fontInfo: BareFontInfo;
}

R
rebornix 已提交
54 55 56 57 58 59 60 61 62 63 64
export interface NotebookLayoutChangeEvent {
	width?: boolean;
	height?: boolean;
	fontInfo?: boolean;
}

export interface CodeCellLayoutInfo {
	readonly fontInfo: BareFontInfo | null;
	readonly editorHeight: number;
	readonly editorWidth: number;
	readonly totalHeight: number;
65
	readonly outputContainerOffset: number;
R
rebornix 已提交
66 67
	readonly outputTotalHeight: number;
	readonly indicatorHeight: number;
R
rebornix 已提交
68
	readonly bottomToolbarOffset: number;
R
rebornix 已提交
69 70 71 72 73 74
}

export interface CodeCellLayoutChangeEvent {
	editorHeight?: boolean;
	outputHeight?: boolean;
	totalHeight?: boolean;
R
rebornix 已提交
75 76
	outerWidth?: number;
	font?: BareFontInfo;
R
rebornix 已提交
77 78 79 80 81
}

export interface MarkdownCellLayoutInfo {
	readonly fontInfo: BareFontInfo | null;
	readonly editorWidth: number;
R
rebornix 已提交
82
	readonly bottomToolbarOffset: number;
R
rebornix 已提交
83
	readonly totalHeight: number;
R
rebornix 已提交
84 85 86
}

export interface MarkdownCellLayoutChangeEvent {
R
rebornix 已提交
87 88
	font?: BareFontInfo;
	outerWidth?: number;
R
rebornix 已提交
89
	totalHeight?: number;
R
rebornix 已提交
90 91
}

R
rebornix 已提交
92
export interface ICellViewModel {
R
rebornix 已提交
93
	readonly model: NotebookCellTextModel;
R
rebornix 已提交
94
	readonly id: string;
95
	dragging: boolean;
R
rebornix 已提交
96 97
	handle: number;
	uri: URI;
98
	language: string;
R
rebornix 已提交
99
	cellKind: CellKind;
100
	editState: CellEditState;
101 102
	readonly runState: CellRunState;
	currentTokenSource: CancellationTokenSource | undefined;
R
rebornix 已提交
103 104
	focusMode: CellFocusMode;
	getText(): string;
105
	save(): void;
106
	metadata: NotebookCellMetadata | undefined;
R
rebornix 已提交
107
	getEvaluatedMetadata(documentMetadata: NotebookDocumentMetadata | undefined): NotebookCellMetadata;
R
rebornix 已提交
108 109
}

R
rebornix 已提交
110 111 112 113 114
export interface INotebookEditorMouseEvent {
	readonly event: MouseEvent;
	readonly target: CellViewModel;
}

R
rebornix 已提交
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
export interface INotebookEditorContribution {
	/**
	 * Dispose this contribution.
	 */
	dispose(): void;
	/**
	 * Store view state.
	 */
	saveViewState?(): any;
	/**
	 * Restore view state.
	 */
	restoreViewState?(state: any): void;
}

130
export interface INotebookEditor {
R
rebornix 已提交
131 132 133 134

	/**
	 * Notebook view model attached to the current editor
	 */
R
rebornix 已提交
135
	viewModel: NotebookViewModel | undefined;
R
rebornix 已提交
136

R
rebornix 已提交
137 138 139 140 141
	/**
	 * An event emitted when the model of this editor has changed.
	 * @event
	 */
	readonly onDidChangeModel: Event<void>;
142 143
	isNotebookEditor: boolean;

R
rebornix 已提交
144
	getDomNode(): HTMLElement;
R
rebornix 已提交
145 146
	getInnerWebview(): Webview | undefined;

R
rebornix 已提交
147 148 149
	/**
	 * Focus the notebook editor cell list
	 */
R
rebornix 已提交
150
	focus(): void;
R
rebornix 已提交
151

R
rebornix 已提交
152 153 154
	/**
	 * Select & focus cell
	 */
R
rebornix 已提交
155
	selectElement(cell: ICellViewModel): void;
R
rebornix 已提交
156

R
rebornix 已提交
157 158 159
	/**
	 * Layout info for the notebook editor
	 */
R
rebornix 已提交
160
	getLayoutInfo(): NotebookLayoutInfo;
R
rebornix 已提交
161 162 163
	/**
	 * Fetch the output renderers for notebook outputs.
	 */
R
rebornix 已提交
164
	getOutputRenderer(): OutputRenderer;
R
rebornix 已提交
165 166 167 168

	/**
	 * Insert a new cell around `cell`
	 */
R
rebornix 已提交
169
	insertNotebookCell(cell: ICellViewModel | undefined, type: CellKind, direction?: 'above' | 'below', initialText?: string, ui?: boolean): CellViewModel | null;
R
rebornix 已提交
170 171 172 173

	/**
	 * Delete a cell from the notebook
	 */
174
	deleteNotebookCell(cell: ICellViewModel): Promise<boolean>;
R
rebornix 已提交
175

176 177 178
	/**
	 * Move a cell up one spot
	 */
179
	moveCellUp(cell: ICellViewModel): Promise<boolean>;
180 181 182 183

	/**
	 * Move a cell down one spot
	 */
184
	moveCellDown(cell: ICellViewModel): Promise<boolean>;
185

R
Rob Lourens 已提交
186 187 188
	/**
	 * Move a cell above or below another cell
	 */
189
	moveCell(cell: ICellViewModel, relativeToCell: ICellViewModel, direction: 'above' | 'below'): Promise<boolean>;
R
Rob Lourens 已提交
190

R
rebornix 已提交
191 192 193 194 195 196
	/**
	 * Switch the cell into editing mode.
	 *
	 * For code cell, the monaco editor will be focused.
	 * For markdown cell, it will switch from preview mode to editing mode, which focuses the monaco editor.
	 */
R
rebornix 已提交
197
	editNotebookCell(cell: ICellViewModel): void;
R
rebornix 已提交
198 199 200 201

	/**
	 * Quit cell editing mode.
	 */
R
rebornix 已提交
202
	saveNotebookCell(cell: ICellViewModel): void;
R
rebornix 已提交
203 204 205 206

	/**
	 * Focus the container of a cell (the monaco editor inside is not focused).
	 */
R
rebornix 已提交
207
	focusNotebookCell(cell: ICellViewModel, focusEditor: boolean): void;
R
rebornix 已提交
208

209 210 211 212 213
	/**
	 * Execute the given notebook cell
	 */
	executeNotebookCell(cell: ICellViewModel): Promise<void>;

214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
	/**
	 * Cancel the cell execution
	 */
	cancelNotebookCellExecution(cell: ICellViewModel): void;

	/**
	 * Executes all notebook cells in order
	 */
	executeNotebook(): Promise<void>;

	/**
	 * Cancel the notebook execution
	 */
	cancelNotebookExecution(): void;

R
rebornix 已提交
229 230 231
	/**
	 * Get current active cell
	 */
R
rebornix 已提交
232
	getActiveCell(): ICellViewModel | undefined;
R
rebornix 已提交
233 234 235 236

	/**
	 * Layout the cell with a new height
	 */
237
	layoutNotebookCell(cell: ICellViewModel, height: number): Promise<void>;
R
rebornix 已提交
238 239 240 241

	/**
	 * Render the output in webview layer
	 */
R
rebornix 已提交
242
	createInset(cell: ICellViewModel, output: IOutput, shadowContent: string, offset: number): void;
R
rebornix 已提交
243 244 245 246

	/**
	 * Remove the output from the webview layer
	 */
R
rebornix 已提交
247
	removeInset(output: IOutput): void;
R
rebornix 已提交
248

249 250 251 252 253
	/**
	 * Send message to the webview for outputs.
	 */
	postMessage(message: any): void;

R
rebornix 已提交
254 255 256
	/**
	 * Trigger the editor to scroll from scroll event programmatically
	 */
257
	triggerScroll(event: IMouseWheelEvent): void;
R
rebornix 已提交
258 259 260 261

	/**
	 * Reveal cell into viewport.
	 */
R
rebornix 已提交
262
	revealInView(cell: ICellViewModel): void;
R
rebornix 已提交
263 264 265 266

	/**
	 * Reveal cell into viewport center.
	 */
R
rebornix 已提交
267
	revealInCenter(cell: ICellViewModel): void;
R
rebornix 已提交
268 269 270 271

	/**
	 * Reveal cell into viewport center if cell is currently out of the viewport.
	 */
R
rebornix 已提交
272
	revealInCenterIfOutsideViewport(cell: ICellViewModel): void;
R
rebornix 已提交
273 274 275 276

	/**
	 * Reveal a line in notebook cell into viewport with minimal scrolling.
	 */
R
rebornix 已提交
277
	revealLineInView(cell: ICellViewModel, line: number): void;
R
rebornix 已提交
278

R
rebornix 已提交
279 280 281
	/**
	 * Reveal a line in notebook cell into viewport center.
	 */
R
rebornix 已提交
282
	revealLineInCenter(cell: ICellViewModel, line: number): void;
R
rebornix 已提交
283 284 285 286

	/**
	 * Reveal a line in notebook cell into viewport center.
	 */
R
rebornix 已提交
287
	revealLineInCenterIfOutsideViewport(cell: ICellViewModel, line: number): void;
R
rebornix 已提交
288

R
rebornix 已提交
289 290 291
	/**
	 * Reveal a range in notebook cell into viewport with minimal scrolling.
	 */
R
rebornix 已提交
292
	revealRangeInView(cell: ICellViewModel, range: Range): void;
R
rebornix 已提交
293 294 295 296

	/**
	 * Reveal a range in notebook cell into viewport center.
	 */
R
rebornix 已提交
297
	revealRangeInCenter(cell: ICellViewModel, range: Range): void;
R
rebornix 已提交
298 299 300 301

	/**
	 * Reveal a range in notebook cell into viewport center.
	 */
R
rebornix 已提交
302
	revealRangeInCenterIfOutsideViewport(cell: ICellViewModel, range: Range): void;
R
rebornix 已提交
303

R
rebornix 已提交
304 305 306 307 308
	/**
	 * Set hidden areas on cell text models.
	 */
	setHiddenAreas(_ranges: ICellRange[]): boolean;

R
rebornix 已提交
309
	setCellSelection(cell: ICellViewModel, selection: Range): void;
R
rebornix 已提交
310

R
rebornix 已提交
311 312 313 314
	/**
	 * Change the decorations on cells.
	 * The notebook is virtualized and this method should be called to create/delete editor decorations safely.
	 */
R
rebornix 已提交
315
	changeDecorations(callback: (changeAccessor: IModelDecorationsChangeAccessor) => any): any;
R
rebornix 已提交
316

R
rebornix 已提交
317 318 319 320 321 322 323 324 325 326 327
	/**
	 * An event emitted on a "mouseup".
	 * @event
	 */
	onMouseUp(listener: (e: INotebookEditorMouseEvent) => void): IDisposable;

	/**
	 * An event emitted on a "mousedown".
	 * @event
	 */
	onMouseDown(listener: (e: INotebookEditorMouseEvent) => void): IDisposable;
R
rebornix 已提交
328 329 330 331 332 333 334

	/**
	 * Get a contribution of this editor.
	 * @id Unique identifier of the contribution.
	 * @return The contribution or null if contribution not found.
	 */
	getContribution<T extends INotebookEditorContribution>(id: string): T;
R
rebornix 已提交
335 336
}

R
rebornix 已提交
337 338 339 340 341 342 343 344 345 346
export interface INotebookCellList {
	onWillScroll: Event<ScrollEvent>;
	onDidChangeFocus: Event<IListEvent<ICellViewModel>>;
	onDidChangeContentHeight: Event<number>;
	scrollTop: number;
	scrollHeight: number;
	scrollLeft: number;
	length: number;
	rowsContainer: HTMLElement;
	readonly onDidRemoveOutput: Event<IOutput>;
347
	readonly onDidHideOutput: Event<IOutput>;
R
rebornix 已提交
348 349
	readonly onMouseUp: Event<IListMouseEvent<CellViewModel>>;
	readonly onMouseDown: Event<IListMouseEvent<CellViewModel>>;
R
rebornix 已提交
350 351 352
	detachViewModel(): void;
	attachViewModel(viewModel: NotebookViewModel): void;
	clear(): void;
353
	getViewIndex(cell: ICellViewModel): number | undefined;
R
rebornix 已提交
354 355 356 357 358 359 360 361 362 363 364 365
	focusElement(element: ICellViewModel): void;
	selectElement(element: ICellViewModel): void;
	getFocusedElements(): ICellViewModel[];
	revealElementInView(element: ICellViewModel): void;
	revealElementInCenterIfOutsideViewport(element: ICellViewModel): void;
	revealElementInCenter(element: ICellViewModel): void;
	revealElementLineInView(element: ICellViewModel, line: number): void;
	revealElementLineInCenter(element: ICellViewModel, line: number): void;
	revealElementLineInCenterIfOutsideViewport(element: ICellViewModel, line: number): void;
	revealElementRangeInView(element: ICellViewModel, range: Range): void;
	revealElementRangeInCenter(element: ICellViewModel, range: Range): void;
	revealElementRangeInCenterIfOutsideViewport(element: ICellViewModel, range: Range): void;
R
rebornix 已提交
366
	setHiddenAreas(_ranges: ICellRange[], triggerViewUpdate: boolean): boolean;
R
rebornix 已提交
367 368 369 370 371 372 373 374 375 376 377
	domElementOfElement(element: ICellViewModel): HTMLElement | null;
	focusView(): void;
	getAbsoluteTopOfElement(element: ICellViewModel): number;
	triggerScrollFromMouseWheelEvent(browserEvent: IMouseWheelEvent): void;
	updateElementHeight2(element: ICellViewModel, size: number): void;
	domFocus(): void;
	setCellSelection(element: ICellViewModel, range: Range): void;
	style(styles: IListStyles): void;
	updateOptions(options: IListOptions<ICellViewModel>): void;
	layout(height?: number, width?: number): void;
	dispose(): void;
378 379 380 381 382

	// TODO resolve differences between List<CellViewModel> and INotebookCellList<ICellViewModel>
	getFocus(): number[];
	setFocus(indexes: number[]): void;
	setSelection(indexes: number[]): void;
R
rebornix 已提交
383 384
}

385
export interface BaseCellRenderTemplate {
R
rebornix 已提交
386 387
	container: HTMLElement;
	cellContainer: HTMLElement;
388
	toolbar: ToolBar;
389
	focusIndicator: HTMLElement;
R
Rob Lourens 已提交
390
	insertionIndicatorTop: HTMLElement;
391
	disposables: DisposableStore;
392
	elementDisposables: DisposableStore;
R
rebornix 已提交
393
	bottomCellContainer: HTMLElement;
R
Rob Lourens 已提交
394
	currentRenderedCell?: ICellViewModel;
395 396
	statusBarContainer: HTMLElement;
	languageStatusBarItem: CellLanguageStatusBarItem;
R
rebornix 已提交
397
	toJSON: () => any;
398 399 400
}

export interface MarkdownCellRenderTemplate extends BaseCellRenderTemplate {
401 402
	editorPart: HTMLElement;
	editorContainer: HTMLElement;
R
rebornix 已提交
403
	foldingIndicator: HTMLElement;
404 405 406
}

export interface CodeCellRenderTemplate extends BaseCellRenderTemplate {
R
Rob Lourens 已提交
407 408
	cellRunStatusContainer: HTMLElement;
	cellStatusMessageContainer: HTMLElement;
409 410 411 412 413 414
	runToolbar: ToolBar;
	runButtonContainer: HTMLElement;
	executionOrderLabel: HTMLElement;
	outputContainer: HTMLElement;
	editor: CodeEditorWidget;
	progressBar: ProgressBar;
R
rebornix 已提交
415
}
R
rebornix 已提交
416 417 418 419 420 421 422 423 424

export interface IOutputTransformContribution {
	/**
	 * Dispose this contribution.
	 */
	dispose(): void;

	render(output: IOutput, container: HTMLElement, preferredMimeType: string | undefined): IRenderOutput;
}
R
rebornix 已提交
425 426 427 428 429 430

export interface CellFindMatch {
	cell: CellViewModel;
	matches: FindMatch[];
}

R
rebornix 已提交
431 432 433 434 435 436 437 438 439
export enum CellRevealType {
	Line,
	Range
}

export enum CellRevealPosition {
	Top,
	Center
}
R
rebornix 已提交
440

441 442 443 444 445 446
export enum CellRunState {
	Idle,
	Running
}

export enum CellEditState {
R
rebornix 已提交
447 448 449 450 451
	/**
	 * Default state.
	 * For markdown cell, it's Markdown preview.
	 * For code cell, the browser focus should be on the container instead of the editor
	 */
452
	Preview,
R
rebornix 已提交
453 454 455 456 457 458 459


	/**
	 * Eding mode. Source for markdown or code is rendered in editors and the state will be persistent.
	 */
	Editing
}
R
rebornix 已提交
460 461 462 463 464 465 466 467 468 469 470 471

export enum CellFocusMode {
	Container,
	Editor
}

export enum CursorAtBoundary {
	None,
	Top,
	Bottom,
	Both
}
R
rebornix 已提交
472

473 474 475 476 477 478 479
export interface CellViewModelStateChangeEvent {
	metadataChanged?: boolean;
	selectionChanged?: boolean;
	focusModeChanged?: boolean;
	runStateChanged?: boolean;
	editStateChanged?: boolean;
	languageChanged?: boolean;
R
rebornix 已提交
480 481
	foldingStateChanged?: boolean;
	contentChanged?: boolean;
482
	outputIsHoveredChanged?: boolean;
483 484
}

R
rebornix 已提交
485
/**
R
rebornix 已提交
486
 * [start, end]
R
rebornix 已提交
487 488
 */
export interface ICellRange {
R
rebornix 已提交
489 490 491
	/**
	 * zero based index
	 */
R
rebornix 已提交
492
	start: number;
R
rebornix 已提交
493

R
rebornix 已提交
494
	/**
R
rebornix 已提交
495
	 * zero based index
R
rebornix 已提交
496
	 */
R
rebornix 已提交
497
	end: number;
R
rebornix 已提交
498 499 500 501 502 503 504 505 506 507 508 509 510 511
}


/**
 * @param _ranges
 */
export function reduceCellRanges(_ranges: ICellRange[]): ICellRange[] {
	if (!_ranges.length) {
		return [];
	}

	let ranges = _ranges.sort((a, b) => a.start - b.start);
	let result: ICellRange[] = [];
	let currentRangeStart = ranges[0].start;
R
rebornix 已提交
512
	let currentRangeEnd = ranges[0].end + 1;
R
rebornix 已提交
513 514 515 516 517

	for (let i = 0, len = ranges.length; i < len; i++) {
		let range = ranges[i];

		if (range.start > currentRangeEnd) {
R
rebornix 已提交
518
			result.push({ start: currentRangeStart, end: currentRangeEnd - 1 });
R
rebornix 已提交
519
			currentRangeStart = range.start;
R
rebornix 已提交
520 521 522
			currentRangeEnd = range.end + 1;
		} else if (range.end + 1 > currentRangeEnd) {
			currentRangeEnd = range.end + 1;
R
rebornix 已提交
523 524 525
		}
	}

R
rebornix 已提交
526
	result.push({ start: currentRangeStart, end: currentRangeEnd - 1 });
R
rebornix 已提交
527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543
	return result;
}

export function getVisibleCells(cells: CellViewModel[], hiddenRanges: ICellRange[]) {
	if (!hiddenRanges.length) {
		return cells;
	}

	let start = 0;
	let hiddenRangeIndex = 0;
	let result: any[] = [];

	while (start < cells.length && hiddenRangeIndex < hiddenRanges.length) {
		if (start < hiddenRanges[hiddenRangeIndex].start) {
			result.push(...cells.slice(start, hiddenRanges[hiddenRangeIndex].start));
		}

R
rebornix 已提交
544
		start = hiddenRanges[hiddenRangeIndex].end + 1;
R
rebornix 已提交
545 546 547 548 549 550 551 552 553
		hiddenRangeIndex++;
	}

	if (start < cells.length) {
		result.push(...cells.slice(start));
	}

	return result;
}