mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-02-10 18:22:34 -05:00
SQL Operations Studio Public Preview 1 (0.23) release source code
This commit is contained in:
759
src/vs/editor/common/viewLayout/viewLineRenderer.ts
Normal file
759
src/vs/editor/common/viewLayout/viewLineRenderer.ts
Normal file
@@ -0,0 +1,759 @@
|
||||
/*---------------------------------------------------------------------------------------------
|
||||
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||
*--------------------------------------------------------------------------------------------*/
|
||||
'use strict';
|
||||
|
||||
import { ViewLineToken } from 'vs/editor/common/core/viewLineToken';
|
||||
import { CharCode } from 'vs/base/common/charCode';
|
||||
import { LineDecoration, LineDecorationsNormalizer } from 'vs/editor/common/viewLayout/lineDecorations';
|
||||
import * as strings from 'vs/base/common/strings';
|
||||
import { IStringBuilder, createStringBuilder } from 'vs/editor/common/core/stringBuilder';
|
||||
|
||||
export const enum RenderWhitespace {
|
||||
None = 0,
|
||||
Boundary = 1,
|
||||
All = 2
|
||||
}
|
||||
|
||||
class LinePart {
|
||||
_linePartBrand: void;
|
||||
|
||||
/**
|
||||
* last char index of this token (not inclusive).
|
||||
*/
|
||||
public readonly endIndex: number;
|
||||
public readonly type: string;
|
||||
|
||||
constructor(endIndex: number, type: string) {
|
||||
this.endIndex = endIndex;
|
||||
this.type = type;
|
||||
}
|
||||
}
|
||||
|
||||
export class RenderLineInput {
|
||||
|
||||
public readonly useMonospaceOptimizations: boolean;
|
||||
public readonly lineContent: string;
|
||||
public readonly mightContainRTL: boolean;
|
||||
public readonly fauxIndentLength: number;
|
||||
public readonly lineTokens: ViewLineToken[];
|
||||
public readonly lineDecorations: LineDecoration[];
|
||||
public readonly tabSize: number;
|
||||
public readonly spaceWidth: number;
|
||||
public readonly stopRenderingLineAfter: number;
|
||||
public readonly renderWhitespace: RenderWhitespace;
|
||||
public readonly renderControlCharacters: boolean;
|
||||
public readonly fontLigatures: boolean;
|
||||
|
||||
constructor(
|
||||
useMonospaceOptimizations: boolean,
|
||||
lineContent: string,
|
||||
mightContainRTL: boolean,
|
||||
fauxIndentLength: number,
|
||||
lineTokens: ViewLineToken[],
|
||||
lineDecorations: LineDecoration[],
|
||||
tabSize: number,
|
||||
spaceWidth: number,
|
||||
stopRenderingLineAfter: number,
|
||||
renderWhitespace: 'none' | 'boundary' | 'all',
|
||||
renderControlCharacters: boolean,
|
||||
fontLigatures: boolean
|
||||
) {
|
||||
this.useMonospaceOptimizations = useMonospaceOptimizations;
|
||||
this.lineContent = lineContent;
|
||||
this.mightContainRTL = mightContainRTL;
|
||||
this.fauxIndentLength = fauxIndentLength;
|
||||
this.lineTokens = lineTokens;
|
||||
this.lineDecorations = lineDecorations;
|
||||
this.tabSize = tabSize;
|
||||
this.spaceWidth = spaceWidth;
|
||||
this.stopRenderingLineAfter = stopRenderingLineAfter;
|
||||
this.renderWhitespace = (
|
||||
renderWhitespace === 'all'
|
||||
? RenderWhitespace.All
|
||||
: renderWhitespace === 'boundary'
|
||||
? RenderWhitespace.Boundary
|
||||
: RenderWhitespace.None
|
||||
);
|
||||
this.renderControlCharacters = renderControlCharacters;
|
||||
this.fontLigatures = fontLigatures;
|
||||
}
|
||||
|
||||
public equals(other: RenderLineInput): boolean {
|
||||
return (
|
||||
this.useMonospaceOptimizations === other.useMonospaceOptimizations
|
||||
&& this.lineContent === other.lineContent
|
||||
&& this.mightContainRTL === other.mightContainRTL
|
||||
&& this.fauxIndentLength === other.fauxIndentLength
|
||||
&& this.tabSize === other.tabSize
|
||||
&& this.spaceWidth === other.spaceWidth
|
||||
&& this.stopRenderingLineAfter === other.stopRenderingLineAfter
|
||||
&& this.renderWhitespace === other.renderWhitespace
|
||||
&& this.renderControlCharacters === other.renderControlCharacters
|
||||
&& this.fontLigatures === other.fontLigatures
|
||||
&& LineDecoration.equalsArr(this.lineDecorations, other.lineDecorations)
|
||||
&& ViewLineToken.equalsArr(this.lineTokens, other.lineTokens)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
export const enum CharacterMappingConstants {
|
||||
PART_INDEX_MASK = 0b11111111111111110000000000000000,
|
||||
CHAR_INDEX_MASK = 0b00000000000000001111111111111111,
|
||||
|
||||
CHAR_INDEX_OFFSET = 0,
|
||||
PART_INDEX_OFFSET = 16
|
||||
}
|
||||
|
||||
/**
|
||||
* Provides a both direction mapping between a line's character and its rendered position.
|
||||
*/
|
||||
export class CharacterMapping {
|
||||
|
||||
public static getPartIndex(partData: number): number {
|
||||
return (partData & CharacterMappingConstants.PART_INDEX_MASK) >>> CharacterMappingConstants.PART_INDEX_OFFSET;
|
||||
}
|
||||
|
||||
public static getCharIndex(partData: number): number {
|
||||
return (partData & CharacterMappingConstants.CHAR_INDEX_MASK) >>> CharacterMappingConstants.CHAR_INDEX_OFFSET;
|
||||
}
|
||||
|
||||
public readonly length: number;
|
||||
private readonly _data: Uint32Array;
|
||||
private readonly _absoluteOffsets: Uint32Array;
|
||||
|
||||
constructor(length: number, partCount: number) {
|
||||
this.length = length;
|
||||
this._data = new Uint32Array(this.length);
|
||||
this._absoluteOffsets = new Uint32Array(this.length);
|
||||
}
|
||||
|
||||
public setPartData(charOffset: number, partIndex: number, charIndex: number, partAbsoluteOffset: number): void {
|
||||
let partData = (
|
||||
(partIndex << CharacterMappingConstants.PART_INDEX_OFFSET)
|
||||
| (charIndex << CharacterMappingConstants.CHAR_INDEX_OFFSET)
|
||||
) >>> 0;
|
||||
this._data[charOffset] = partData;
|
||||
this._absoluteOffsets[charOffset] = partAbsoluteOffset + charIndex;
|
||||
}
|
||||
|
||||
public getAbsoluteOffsets(): Uint32Array {
|
||||
return this._absoluteOffsets;
|
||||
}
|
||||
|
||||
public charOffsetToPartData(charOffset: number): number {
|
||||
if (this.length === 0) {
|
||||
return 0;
|
||||
}
|
||||
if (charOffset < 0) {
|
||||
return this._data[0];
|
||||
}
|
||||
if (charOffset >= this.length) {
|
||||
return this._data[this.length - 1];
|
||||
}
|
||||
return this._data[charOffset];
|
||||
}
|
||||
|
||||
public partDataToCharOffset(partIndex: number, partLength: number, charIndex: number): number {
|
||||
if (this.length === 0) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
let searchEntry = (
|
||||
(partIndex << CharacterMappingConstants.PART_INDEX_OFFSET)
|
||||
| (charIndex << CharacterMappingConstants.CHAR_INDEX_OFFSET)
|
||||
) >>> 0;
|
||||
|
||||
let min = 0;
|
||||
let max = this.length - 1;
|
||||
while (min + 1 < max) {
|
||||
let mid = ((min + max) >>> 1);
|
||||
let midEntry = this._data[mid];
|
||||
if (midEntry === searchEntry) {
|
||||
return mid;
|
||||
} else if (midEntry > searchEntry) {
|
||||
max = mid;
|
||||
} else {
|
||||
min = mid;
|
||||
}
|
||||
}
|
||||
|
||||
if (min === max) {
|
||||
return min;
|
||||
}
|
||||
|
||||
let minEntry = this._data[min];
|
||||
let maxEntry = this._data[max];
|
||||
|
||||
if (minEntry === searchEntry) {
|
||||
return min;
|
||||
}
|
||||
if (maxEntry === searchEntry) {
|
||||
return max;
|
||||
}
|
||||
|
||||
let minPartIndex = CharacterMapping.getPartIndex(minEntry);
|
||||
let minCharIndex = CharacterMapping.getCharIndex(minEntry);
|
||||
|
||||
let maxPartIndex = CharacterMapping.getPartIndex(maxEntry);
|
||||
let maxCharIndex: number;
|
||||
|
||||
if (minPartIndex !== maxPartIndex) {
|
||||
// sitting between parts
|
||||
maxCharIndex = partLength;
|
||||
} else {
|
||||
maxCharIndex = CharacterMapping.getCharIndex(maxEntry);
|
||||
}
|
||||
|
||||
let minEntryDistance = charIndex - minCharIndex;
|
||||
let maxEntryDistance = maxCharIndex - charIndex;
|
||||
|
||||
if (minEntryDistance <= maxEntryDistance) {
|
||||
return min;
|
||||
}
|
||||
return max;
|
||||
}
|
||||
}
|
||||
|
||||
export class RenderLineOutput {
|
||||
_renderLineOutputBrand: void;
|
||||
|
||||
readonly characterMapping: CharacterMapping;
|
||||
readonly containsRTL: boolean;
|
||||
readonly containsForeignElements: boolean;
|
||||
|
||||
constructor(characterMapping: CharacterMapping, containsRTL: boolean, containsForeignElements: boolean) {
|
||||
this.characterMapping = characterMapping;
|
||||
this.containsRTL = containsRTL;
|
||||
this.containsForeignElements = containsForeignElements;
|
||||
}
|
||||
}
|
||||
|
||||
export function renderViewLine(input: RenderLineInput, sb: IStringBuilder): RenderLineOutput {
|
||||
if (input.lineContent.length === 0) {
|
||||
|
||||
let containsForeignElements = false;
|
||||
|
||||
// This is basically for IE's hit test to work
|
||||
let content: string = '<span><span>\u00a0</span></span>';
|
||||
|
||||
if (input.lineDecorations.length > 0) {
|
||||
// This line is empty, but it contains inline decorations
|
||||
let classNames: string[] = [];
|
||||
for (let i = 0, len = input.lineDecorations.length; i < len; i++) {
|
||||
const lineDecoration = input.lineDecorations[i];
|
||||
if (lineDecoration.insertsBeforeOrAfter) {
|
||||
classNames[i] = input.lineDecorations[i].className;
|
||||
containsForeignElements = true;
|
||||
}
|
||||
}
|
||||
|
||||
if (containsForeignElements) {
|
||||
content = `<span><span class="${classNames.join(' ')}">\u00a0</span></span>`;
|
||||
}
|
||||
}
|
||||
|
||||
sb.appendASCIIString(content);
|
||||
return new RenderLineOutput(
|
||||
new CharacterMapping(0, 0),
|
||||
false,
|
||||
containsForeignElements
|
||||
);
|
||||
}
|
||||
|
||||
return _renderLine(resolveRenderLineInput(input), sb);
|
||||
}
|
||||
|
||||
export class RenderLineOutput2 {
|
||||
constructor(
|
||||
public readonly characterMapping: CharacterMapping,
|
||||
public readonly html: string,
|
||||
public readonly containsRTL: boolean,
|
||||
public readonly containsForeignElements: boolean
|
||||
) {
|
||||
}
|
||||
}
|
||||
|
||||
export function renderViewLine2(input: RenderLineInput): RenderLineOutput2 {
|
||||
let sb = createStringBuilder(10000);
|
||||
let out = renderViewLine(input, sb);
|
||||
return new RenderLineOutput2(out.characterMapping, sb.build(), out.containsRTL, out.containsForeignElements);
|
||||
}
|
||||
|
||||
class ResolvedRenderLineInput {
|
||||
constructor(
|
||||
public readonly fontIsMonospace: boolean,
|
||||
public readonly lineContent: string,
|
||||
public readonly len: number,
|
||||
public readonly isOverflowing: boolean,
|
||||
public readonly parts: LinePart[],
|
||||
public readonly containsForeignElements: boolean,
|
||||
public readonly tabSize: number,
|
||||
public readonly containsRTL: boolean,
|
||||
public readonly spaceWidth: number,
|
||||
public readonly renderWhitespace: RenderWhitespace,
|
||||
public readonly renderControlCharacters: boolean,
|
||||
) {
|
||||
//
|
||||
}
|
||||
}
|
||||
|
||||
function resolveRenderLineInput(input: RenderLineInput): ResolvedRenderLineInput {
|
||||
const useMonospaceOptimizations = input.useMonospaceOptimizations;
|
||||
const lineContent = input.lineContent;
|
||||
|
||||
let isOverflowing: boolean;
|
||||
let len: number;
|
||||
|
||||
if (input.stopRenderingLineAfter !== -1 && input.stopRenderingLineAfter < lineContent.length) {
|
||||
isOverflowing = true;
|
||||
len = input.stopRenderingLineAfter;
|
||||
} else {
|
||||
isOverflowing = false;
|
||||
len = lineContent.length;
|
||||
}
|
||||
|
||||
let tokens = transformAndRemoveOverflowing(input.lineTokens, input.fauxIndentLength, len);
|
||||
if (input.renderWhitespace === RenderWhitespace.All || input.renderWhitespace === RenderWhitespace.Boundary) {
|
||||
tokens = _applyRenderWhitespace(lineContent, len, tokens, input.fauxIndentLength, input.tabSize, useMonospaceOptimizations, input.renderWhitespace === RenderWhitespace.Boundary);
|
||||
}
|
||||
let containsForeignElements = false;
|
||||
if (input.lineDecorations.length > 0) {
|
||||
for (let i = 0, len = input.lineDecorations.length; i < len; i++) {
|
||||
const lineDecoration = input.lineDecorations[i];
|
||||
if (lineDecoration.insertsBeforeOrAfter) {
|
||||
containsForeignElements = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
tokens = _applyInlineDecorations(lineContent, len, tokens, input.lineDecorations);
|
||||
}
|
||||
let containsRTL = false;
|
||||
if (input.mightContainRTL) {
|
||||
containsRTL = strings.containsRTL(lineContent);
|
||||
}
|
||||
if (!containsRTL && !input.fontLigatures) {
|
||||
tokens = splitLargeTokens(lineContent, tokens);
|
||||
}
|
||||
|
||||
return new ResolvedRenderLineInput(
|
||||
useMonospaceOptimizations,
|
||||
lineContent,
|
||||
len,
|
||||
isOverflowing,
|
||||
tokens,
|
||||
containsForeignElements,
|
||||
input.tabSize,
|
||||
containsRTL,
|
||||
input.spaceWidth,
|
||||
input.renderWhitespace,
|
||||
input.renderControlCharacters
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* In the rendering phase, characters are always looped until token.endIndex.
|
||||
* Ensure that all tokens end before `len` and the last one ends precisely at `len`.
|
||||
*/
|
||||
function transformAndRemoveOverflowing(tokens: ViewLineToken[], fauxIndentLength: number, len: number): LinePart[] {
|
||||
let result: LinePart[] = [], resultLen = 0;
|
||||
|
||||
// The faux indent part of the line should have no token type
|
||||
if (fauxIndentLength > 0) {
|
||||
result[resultLen++] = new LinePart(fauxIndentLength, '');
|
||||
}
|
||||
|
||||
for (let tokenIndex = 0, tokensLen = tokens.length; tokenIndex < tokensLen; tokenIndex++) {
|
||||
const token = tokens[tokenIndex];
|
||||
const endIndex = token.endIndex;
|
||||
if (endIndex <= fauxIndentLength) {
|
||||
// The faux indent part of the line should have no token type
|
||||
continue;
|
||||
}
|
||||
const type = token.getType();
|
||||
if (endIndex >= len) {
|
||||
result[resultLen++] = new LinePart(len, type);
|
||||
break;
|
||||
}
|
||||
result[resultLen++] = new LinePart(endIndex, type);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* written as a const enum to get value inlining.
|
||||
*/
|
||||
const enum Constants {
|
||||
LongToken = 50
|
||||
}
|
||||
|
||||
/**
|
||||
* See https://github.com/Microsoft/vscode/issues/6885.
|
||||
* It appears that having very large spans causes very slow reading of character positions.
|
||||
* So here we try to avoid that.
|
||||
*/
|
||||
function splitLargeTokens(lineContent: string, tokens: LinePart[]): LinePart[] {
|
||||
let lastTokenEndIndex = 0;
|
||||
let result: LinePart[] = [], resultLen = 0;
|
||||
for (let i = 0, len = tokens.length; i < len; i++) {
|
||||
const token = tokens[i];
|
||||
const tokenEndIndex = token.endIndex;
|
||||
let diff = (tokenEndIndex - lastTokenEndIndex);
|
||||
if (diff > Constants.LongToken) {
|
||||
const tokenType = token.type;
|
||||
const piecesCount = Math.ceil(diff / Constants.LongToken);
|
||||
for (let j = 1; j < piecesCount; j++) {
|
||||
let pieceEndIndex = lastTokenEndIndex + (j * Constants.LongToken);
|
||||
let lastCharInPiece = lineContent.charCodeAt(pieceEndIndex - 1);
|
||||
if (strings.isHighSurrogate(lastCharInPiece)) {
|
||||
// Don't cut in the middle of a surrogate pair
|
||||
pieceEndIndex--;
|
||||
}
|
||||
result[resultLen++] = new LinePart(pieceEndIndex, tokenType);
|
||||
}
|
||||
result[resultLen++] = new LinePart(tokenEndIndex, tokenType);
|
||||
} else {
|
||||
result[resultLen++] = token;
|
||||
}
|
||||
lastTokenEndIndex = tokenEndIndex;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* Whitespace is rendered by "replacing" tokens with a special-purpose `vs-whitespace` type that is later recognized in the rendering phase.
|
||||
* Moreover, a token is created for every visual indent because on some fonts the glyphs used for rendering whitespace (→ or ·) do not have the same width as .
|
||||
* The rendering phase will generate `style="width:..."` for these tokens.
|
||||
*/
|
||||
function _applyRenderWhitespace(lineContent: string, len: number, tokens: LinePart[], fauxIndentLength: number, tabSize: number, useMonospaceOptimizations: boolean, onlyBoundary: boolean): LinePart[] {
|
||||
|
||||
let result: LinePart[] = [], resultLen = 0;
|
||||
let tokenIndex = 0;
|
||||
let tokenType = tokens[tokenIndex].type;
|
||||
let tokenEndIndex = tokens[tokenIndex].endIndex;
|
||||
|
||||
let firstNonWhitespaceIndex = strings.firstNonWhitespaceIndex(lineContent);
|
||||
let lastNonWhitespaceIndex: number;
|
||||
if (firstNonWhitespaceIndex === -1) {
|
||||
// The entire line is whitespace
|
||||
firstNonWhitespaceIndex = len;
|
||||
lastNonWhitespaceIndex = len;
|
||||
} else {
|
||||
lastNonWhitespaceIndex = strings.lastNonWhitespaceIndex(lineContent);
|
||||
}
|
||||
|
||||
let tmpIndent = 0;
|
||||
for (let charIndex = 0; charIndex < fauxIndentLength; charIndex++) {
|
||||
const chCode = lineContent.charCodeAt(charIndex);
|
||||
if (chCode === CharCode.Tab) {
|
||||
tmpIndent = tabSize;
|
||||
} else {
|
||||
tmpIndent++;
|
||||
}
|
||||
}
|
||||
tmpIndent = tmpIndent % tabSize;
|
||||
|
||||
let wasInWhitespace = false;
|
||||
for (let charIndex = fauxIndentLength; charIndex < len; charIndex++) {
|
||||
const chCode = lineContent.charCodeAt(charIndex);
|
||||
|
||||
let isInWhitespace: boolean;
|
||||
if (charIndex < firstNonWhitespaceIndex || charIndex > lastNonWhitespaceIndex) {
|
||||
// in leading or trailing whitespace
|
||||
isInWhitespace = true;
|
||||
} else if (chCode === CharCode.Tab) {
|
||||
// a tab character is rendered both in all and boundary cases
|
||||
isInWhitespace = true;
|
||||
} else if (chCode === CharCode.Space) {
|
||||
// hit a space character
|
||||
if (onlyBoundary) {
|
||||
// rendering only boundary whitespace
|
||||
if (wasInWhitespace) {
|
||||
isInWhitespace = true;
|
||||
} else {
|
||||
const nextChCode = (charIndex + 1 < len ? lineContent.charCodeAt(charIndex + 1) : CharCode.Null);
|
||||
isInWhitespace = (nextChCode === CharCode.Space || nextChCode === CharCode.Tab);
|
||||
}
|
||||
} else {
|
||||
isInWhitespace = true;
|
||||
}
|
||||
} else {
|
||||
isInWhitespace = false;
|
||||
}
|
||||
|
||||
if (wasInWhitespace) {
|
||||
// was in whitespace token
|
||||
if (!isInWhitespace || (!useMonospaceOptimizations && tmpIndent >= tabSize)) {
|
||||
// leaving whitespace token or entering a new indent
|
||||
result[resultLen++] = new LinePart(charIndex, 'vs-whitespace');
|
||||
tmpIndent = tmpIndent % tabSize;
|
||||
}
|
||||
} else {
|
||||
// was in regular token
|
||||
if (charIndex === tokenEndIndex || (isInWhitespace && charIndex > fauxIndentLength)) {
|
||||
result[resultLen++] = new LinePart(charIndex, tokenType);
|
||||
tmpIndent = tmpIndent % tabSize;
|
||||
}
|
||||
}
|
||||
|
||||
if (chCode === CharCode.Tab) {
|
||||
tmpIndent = tabSize;
|
||||
} else {
|
||||
tmpIndent++;
|
||||
}
|
||||
|
||||
wasInWhitespace = isInWhitespace;
|
||||
|
||||
if (charIndex === tokenEndIndex) {
|
||||
tokenIndex++;
|
||||
tokenType = tokens[tokenIndex].type;
|
||||
tokenEndIndex = tokens[tokenIndex].endIndex;
|
||||
}
|
||||
}
|
||||
|
||||
if (wasInWhitespace) {
|
||||
// was in whitespace token
|
||||
result[resultLen++] = new LinePart(len, 'vs-whitespace');
|
||||
} else {
|
||||
// was in regular token
|
||||
result[resultLen++] = new LinePart(len, tokenType);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* Inline decorations are "merged" on top of tokens.
|
||||
* Special care must be taken when multiple inline decorations are at play and they overlap.
|
||||
*/
|
||||
function _applyInlineDecorations(lineContent: string, len: number, tokens: LinePart[], _lineDecorations: LineDecoration[]): LinePart[] {
|
||||
_lineDecorations.sort(LineDecoration.compare);
|
||||
const lineDecorations = LineDecorationsNormalizer.normalize(_lineDecorations);
|
||||
const lineDecorationsLen = lineDecorations.length;
|
||||
|
||||
let lineDecorationIndex = 0;
|
||||
let result: LinePart[] = [], resultLen = 0, lastResultEndIndex = 0;
|
||||
for (let tokenIndex = 0, len = tokens.length; tokenIndex < len; tokenIndex++) {
|
||||
const token = tokens[tokenIndex];
|
||||
const tokenEndIndex = token.endIndex;
|
||||
const tokenType = token.type;
|
||||
|
||||
while (lineDecorationIndex < lineDecorationsLen && lineDecorations[lineDecorationIndex].startOffset < tokenEndIndex) {
|
||||
const lineDecoration = lineDecorations[lineDecorationIndex];
|
||||
|
||||
if (lineDecoration.startOffset > lastResultEndIndex) {
|
||||
lastResultEndIndex = lineDecoration.startOffset;
|
||||
result[resultLen++] = new LinePart(lastResultEndIndex, tokenType);
|
||||
}
|
||||
|
||||
if (lineDecoration.endOffset + 1 <= tokenEndIndex) {
|
||||
// This line decoration ends before this token ends
|
||||
lastResultEndIndex = lineDecoration.endOffset + 1;
|
||||
result[resultLen++] = new LinePart(lastResultEndIndex, tokenType + ' ' + lineDecoration.className);
|
||||
lineDecorationIndex++;
|
||||
} else {
|
||||
// This line decoration continues on to the next token
|
||||
lastResultEndIndex = tokenEndIndex;
|
||||
result[resultLen++] = new LinePart(lastResultEndIndex, tokenType + ' ' + lineDecoration.className);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (tokenEndIndex > lastResultEndIndex) {
|
||||
lastResultEndIndex = tokenEndIndex;
|
||||
result[resultLen++] = new LinePart(lastResultEndIndex, tokenType);
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* This function is on purpose not split up into multiple functions to allow runtime type inference (i.e. performance reasons).
|
||||
* Notice how all the needed data is fully resolved and passed in (i.e. no other calls).
|
||||
*/
|
||||
function _renderLine(input: ResolvedRenderLineInput, sb: IStringBuilder): RenderLineOutput {
|
||||
const fontIsMonospace = input.fontIsMonospace;
|
||||
const containsForeignElements = input.containsForeignElements;
|
||||
const lineContent = input.lineContent;
|
||||
const len = input.len;
|
||||
const isOverflowing = input.isOverflowing;
|
||||
const parts = input.parts;
|
||||
const tabSize = input.tabSize;
|
||||
const containsRTL = input.containsRTL;
|
||||
const spaceWidth = input.spaceWidth;
|
||||
const renderWhitespace = input.renderWhitespace;
|
||||
const renderControlCharacters = input.renderControlCharacters;
|
||||
|
||||
const characterMapping = new CharacterMapping(len + 1, parts.length);
|
||||
|
||||
let charIndex = 0;
|
||||
let tabsCharDelta = 0;
|
||||
let charOffsetInPart = 0;
|
||||
|
||||
let prevPartContentCnt = 0;
|
||||
let partAbsoluteOffset = 0;
|
||||
|
||||
sb.appendASCIIString('<span>');
|
||||
|
||||
for (let partIndex = 0, tokensLen = parts.length; partIndex < tokensLen; partIndex++) {
|
||||
partAbsoluteOffset += prevPartContentCnt;
|
||||
|
||||
const part = parts[partIndex];
|
||||
const partEndIndex = part.endIndex;
|
||||
const partType = part.type;
|
||||
const partRendersWhitespace = (renderWhitespace !== RenderWhitespace.None && (partType.indexOf('vs-whitespace') >= 0));
|
||||
charOffsetInPart = 0;
|
||||
|
||||
sb.appendASCIIString('<span class="');
|
||||
sb.appendASCIIString(partType);
|
||||
sb.appendASCII(CharCode.DoubleQuote);
|
||||
|
||||
if (partRendersWhitespace) {
|
||||
|
||||
let partContentCnt = 0;
|
||||
{
|
||||
let _charIndex = charIndex;
|
||||
let _tabsCharDelta = tabsCharDelta;
|
||||
let _charOffsetInPart = charOffsetInPart;
|
||||
|
||||
for (; _charIndex < partEndIndex; _charIndex++) {
|
||||
const charCode = lineContent.charCodeAt(_charIndex);
|
||||
|
||||
if (charCode === CharCode.Tab) {
|
||||
let insertSpacesCount = tabSize - (_charIndex + _tabsCharDelta) % tabSize;
|
||||
_tabsCharDelta += insertSpacesCount - 1;
|
||||
_charOffsetInPart += insertSpacesCount - 1;
|
||||
partContentCnt += insertSpacesCount;
|
||||
} else {
|
||||
partContentCnt++;
|
||||
}
|
||||
|
||||
_charOffsetInPart++;
|
||||
}
|
||||
}
|
||||
|
||||
if (!fontIsMonospace && !containsForeignElements) {
|
||||
sb.appendASCIIString(' style="width:');
|
||||
sb.appendASCIIString(String(spaceWidth * partContentCnt));
|
||||
sb.appendASCIIString('px"');
|
||||
}
|
||||
sb.appendASCII(CharCode.GreaterThan);
|
||||
|
||||
for (; charIndex < partEndIndex; charIndex++) {
|
||||
characterMapping.setPartData(charIndex, partIndex, charOffsetInPart, partAbsoluteOffset);
|
||||
const charCode = lineContent.charCodeAt(charIndex);
|
||||
|
||||
if (charCode === CharCode.Tab) {
|
||||
let insertSpacesCount = tabSize - (charIndex + tabsCharDelta) % tabSize;
|
||||
tabsCharDelta += insertSpacesCount - 1;
|
||||
charOffsetInPart += insertSpacesCount - 1;
|
||||
if (insertSpacesCount > 0) {
|
||||
sb.write1(0x2192); // →
|
||||
insertSpacesCount--;
|
||||
}
|
||||
while (insertSpacesCount > 0) {
|
||||
sb.write1(0xA0); //
|
||||
insertSpacesCount--;
|
||||
}
|
||||
} else {
|
||||
// must be CharCode.Space
|
||||
sb.write1(0xb7); // ·
|
||||
}
|
||||
|
||||
charOffsetInPart++;
|
||||
}
|
||||
|
||||
prevPartContentCnt = partContentCnt;
|
||||
|
||||
} else {
|
||||
|
||||
let partContentCnt = 0;
|
||||
|
||||
if (containsRTL) {
|
||||
sb.appendASCIIString(' dir="ltr"');
|
||||
}
|
||||
sb.appendASCII(CharCode.GreaterThan);
|
||||
|
||||
for (; charIndex < partEndIndex; charIndex++) {
|
||||
characterMapping.setPartData(charIndex, partIndex, charOffsetInPart, partAbsoluteOffset);
|
||||
const charCode = lineContent.charCodeAt(charIndex);
|
||||
|
||||
switch (charCode) {
|
||||
case CharCode.Tab:
|
||||
let insertSpacesCount = tabSize - (charIndex + tabsCharDelta) % tabSize;
|
||||
tabsCharDelta += insertSpacesCount - 1;
|
||||
charOffsetInPart += insertSpacesCount - 1;
|
||||
while (insertSpacesCount > 0) {
|
||||
sb.write1(0xA0); //
|
||||
partContentCnt++;
|
||||
insertSpacesCount--;
|
||||
}
|
||||
break;
|
||||
|
||||
case CharCode.Space:
|
||||
sb.write1(0xA0); //
|
||||
partContentCnt++;
|
||||
break;
|
||||
|
||||
case CharCode.LessThan:
|
||||
sb.appendASCIIString('<');
|
||||
partContentCnt++;
|
||||
break;
|
||||
|
||||
case CharCode.GreaterThan:
|
||||
sb.appendASCIIString('>');
|
||||
partContentCnt++;
|
||||
break;
|
||||
|
||||
case CharCode.Ampersand:
|
||||
sb.appendASCIIString('&');
|
||||
partContentCnt++;
|
||||
break;
|
||||
|
||||
case CharCode.Null:
|
||||
sb.appendASCIIString('�');
|
||||
partContentCnt++;
|
||||
break;
|
||||
|
||||
case CharCode.UTF8_BOM:
|
||||
case CharCode.LINE_SEPARATOR_2028:
|
||||
sb.write1(0xfffd);
|
||||
partContentCnt++;
|
||||
break;
|
||||
|
||||
default:
|
||||
if (renderControlCharacters && charCode < 32) {
|
||||
sb.write1(9216 + charCode);
|
||||
partContentCnt++;
|
||||
} else {
|
||||
sb.write1(charCode);
|
||||
partContentCnt++;
|
||||
}
|
||||
}
|
||||
|
||||
charOffsetInPart++;
|
||||
}
|
||||
|
||||
prevPartContentCnt = partContentCnt;
|
||||
}
|
||||
|
||||
sb.appendASCIIString('</span>');
|
||||
|
||||
}
|
||||
|
||||
// When getting client rects for the last character, we will position the
|
||||
// text range at the end of the span, insteaf of at the beginning of next span
|
||||
characterMapping.setPartData(len, parts.length - 1, charOffsetInPart, partAbsoluteOffset);
|
||||
|
||||
if (isOverflowing) {
|
||||
sb.appendASCIIString('<span>…</span>');
|
||||
}
|
||||
|
||||
sb.appendASCIIString('</span>');
|
||||
|
||||
return new RenderLineOutput(characterMapping, containsRTL, containsForeignElements);
|
||||
}
|
||||
Reference in New Issue
Block a user