mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-02-09 01:32:34 -05:00
* Merge from vscode 64980ea1f3f532c82bb6c28d27bba9ef2c5b4463 * fix config changes * fix strictnull checks
193 lines
6.6 KiB
TypeScript
193 lines
6.6 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import 'vs/css!./lineNumbers';
|
|
import * as platform from 'vs/base/common/platform';
|
|
import { DynamicViewOverlay } from 'vs/editor/browser/view/dynamicViewOverlay';
|
|
import { RenderLineNumbersType, EditorOption } from 'vs/editor/common/config/editorOptions';
|
|
import { Position } from 'vs/editor/common/core/position';
|
|
import { editorActiveLineNumber, editorLineNumbers } from 'vs/editor/common/view/editorColorRegistry';
|
|
import { RenderingContext } from 'vs/editor/common/view/renderingContext';
|
|
import { ViewContext } from 'vs/editor/common/view/viewContext';
|
|
import * as viewEvents from 'vs/editor/common/view/viewEvents';
|
|
import { registerThemingParticipant } from 'vs/platform/theme/common/themeService';
|
|
|
|
export class LineNumbersOverlay extends DynamicViewOverlay {
|
|
|
|
public static readonly CLASS_NAME = 'line-numbers';
|
|
|
|
private readonly _context: ViewContext;
|
|
|
|
private _lineHeight!: number;
|
|
private _renderLineNumbers!: RenderLineNumbersType;
|
|
private _renderCustomLineNumbers!: ((lineNumber: number) => string) | null;
|
|
private _renderFinalNewline!: boolean;
|
|
private _lineNumbersLeft!: number;
|
|
private _lineNumbersWidth!: number;
|
|
private _lastCursorModelPosition: Position;
|
|
private _renderResult: string[] | null;
|
|
|
|
constructor(context: ViewContext) {
|
|
super();
|
|
this._context = context;
|
|
|
|
this._readConfig();
|
|
|
|
this._lastCursorModelPosition = new Position(1, 1);
|
|
this._renderResult = null;
|
|
this._context.addEventHandler(this);
|
|
}
|
|
|
|
private _readConfig(): void {
|
|
const options = this._context.configuration.options;
|
|
this._lineHeight = options.get(EditorOption.lineHeight);
|
|
const lineNumbers = options.get(EditorOption.lineNumbers);
|
|
this._renderLineNumbers = lineNumbers.renderType;
|
|
this._renderCustomLineNumbers = lineNumbers.renderFn;
|
|
this._renderFinalNewline = options.get(EditorOption.renderFinalNewline);
|
|
const layoutInfo = options.get(EditorOption.layoutInfo);
|
|
this._lineNumbersLeft = layoutInfo.lineNumbersLeft;
|
|
this._lineNumbersWidth = layoutInfo.lineNumbersWidth;
|
|
}
|
|
|
|
public dispose(): void {
|
|
this._context.removeEventHandler(this);
|
|
this._renderResult = null;
|
|
super.dispose();
|
|
}
|
|
|
|
// --- begin event handlers
|
|
|
|
public onConfigurationChanged(e: viewEvents.ViewConfigurationChangedEvent): boolean {
|
|
this._readConfig();
|
|
return true;
|
|
}
|
|
public onCursorStateChanged(e: viewEvents.ViewCursorStateChangedEvent): boolean {
|
|
const primaryViewPosition = e.selections[0].getPosition();
|
|
this._lastCursorModelPosition = this._context.model.coordinatesConverter.convertViewPositionToModelPosition(primaryViewPosition);
|
|
|
|
if (this._renderLineNumbers === RenderLineNumbersType.Relative || this._renderLineNumbers === RenderLineNumbersType.Interval) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
public onFlushed(e: viewEvents.ViewFlushedEvent): boolean {
|
|
return true;
|
|
}
|
|
public onLinesChanged(e: viewEvents.ViewLinesChangedEvent): boolean {
|
|
return true;
|
|
}
|
|
public onLinesDeleted(e: viewEvents.ViewLinesDeletedEvent): boolean {
|
|
return true;
|
|
}
|
|
public onLinesInserted(e: viewEvents.ViewLinesInsertedEvent): boolean {
|
|
return true;
|
|
}
|
|
public onScrollChanged(e: viewEvents.ViewScrollChangedEvent): boolean {
|
|
return e.scrollTopChanged;
|
|
}
|
|
public onZonesChanged(e: viewEvents.ViewZonesChangedEvent): boolean {
|
|
return true;
|
|
}
|
|
|
|
// --- end event handlers
|
|
|
|
private _getLineRenderLineNumber(viewLineNumber: number): string {
|
|
const modelPosition = this._context.model.coordinatesConverter.convertViewPositionToModelPosition(new Position(viewLineNumber, 1));
|
|
if (modelPosition.column !== 1) {
|
|
return '';
|
|
}
|
|
const modelLineNumber = modelPosition.lineNumber;
|
|
|
|
if (!this._renderFinalNewline) {
|
|
const lineCount = this._context.model.getLineCount();
|
|
const lineContent = this._context.model.getLineContent(modelLineNumber);
|
|
|
|
if (modelLineNumber === lineCount && lineContent === '') {
|
|
return '';
|
|
}
|
|
}
|
|
|
|
if (this._renderCustomLineNumbers) {
|
|
return this._renderCustomLineNumbers(modelLineNumber);
|
|
}
|
|
|
|
if (this._renderLineNumbers === RenderLineNumbersType.Relative) {
|
|
const diff = Math.abs(this._lastCursorModelPosition.lineNumber - modelLineNumber);
|
|
if (diff === 0) {
|
|
return '<span class="relative-current-line-number">' + modelLineNumber + '</span>';
|
|
}
|
|
return String(diff);
|
|
}
|
|
|
|
if (this._renderLineNumbers === RenderLineNumbersType.Interval) {
|
|
if (this._lastCursorModelPosition.lineNumber === modelLineNumber) {
|
|
return String(modelLineNumber);
|
|
}
|
|
if (modelLineNumber % 10 === 0) {
|
|
return String(modelLineNumber);
|
|
}
|
|
return '';
|
|
}
|
|
|
|
return String(modelLineNumber);
|
|
}
|
|
|
|
public prepareRender(ctx: RenderingContext): void {
|
|
if (this._renderLineNumbers === RenderLineNumbersType.Off) {
|
|
this._renderResult = null;
|
|
return;
|
|
}
|
|
|
|
const lineHeightClassName = (platform.isLinux ? (this._lineHeight % 2 === 0 ? ' lh-even' : ' lh-odd') : '');
|
|
const visibleStartLineNumber = ctx.visibleRange.startLineNumber;
|
|
const visibleEndLineNumber = ctx.visibleRange.endLineNumber;
|
|
const common = '<div class="' + LineNumbersOverlay.CLASS_NAME + lineHeightClassName + '" style="left:' + this._lineNumbersLeft.toString() + 'px;width:' + this._lineNumbersWidth.toString() + 'px;">';
|
|
|
|
const output: string[] = [];
|
|
for (let lineNumber = visibleStartLineNumber; lineNumber <= visibleEndLineNumber; lineNumber++) {
|
|
const lineIndex = lineNumber - visibleStartLineNumber;
|
|
|
|
const renderLineNumber = this._getLineRenderLineNumber(lineNumber);
|
|
|
|
if (renderLineNumber) {
|
|
output[lineIndex] = (
|
|
common
|
|
+ renderLineNumber
|
|
+ '</div>'
|
|
);
|
|
} else {
|
|
output[lineIndex] = '';
|
|
}
|
|
}
|
|
|
|
this._renderResult = output;
|
|
}
|
|
|
|
public render(startLineNumber: number, lineNumber: number): string {
|
|
if (!this._renderResult) {
|
|
return '';
|
|
}
|
|
const lineIndex = lineNumber - startLineNumber;
|
|
if (lineIndex < 0 || lineIndex >= this._renderResult.length) {
|
|
return '';
|
|
}
|
|
return this._renderResult[lineIndex];
|
|
}
|
|
}
|
|
|
|
// theming
|
|
|
|
registerThemingParticipant((theme, collector) => {
|
|
const lineNumbers = theme.getColor(editorLineNumbers);
|
|
if (lineNumbers) {
|
|
collector.addRule(`.monaco-editor .line-numbers { color: ${lineNumbers}; }`);
|
|
}
|
|
const activeLineNumber = theme.getColor(editorActiveLineNumber);
|
|
if (activeLineNumber) {
|
|
collector.addRule(`.monaco-editor .current-line ~ .line-numbers { color: ${activeLineNumber}; }`);
|
|
}
|
|
});
|