From 08e6426453d5b867cdd343061ef2e93bcbd734d0 Mon Sep 17 00:00:00 2001 From: Simon Lamon Date: Sun, 4 Sep 2022 11:08:53 +0000 Subject: [PATCH 1/2] Improve dotted line for canvas --- .../xterm-addon-canvas/src/BaseRenderLayer.ts | 25 ++-- .../xterm-addon-canvas/src/TextRenderLayer.ts | 132 +++++++++++++----- 2 files changed, 110 insertions(+), 47 deletions(-) diff --git a/addons/xterm-addon-canvas/src/BaseRenderLayer.ts b/addons/xterm-addon-canvas/src/BaseRenderLayer.ts index ae39601693..690b9e7f57 100644 --- a/addons/xterm-addon-canvas/src/BaseRenderLayer.ts +++ b/addons/xterm-addon-canvas/src/BaseRenderLayer.ts @@ -82,11 +82,11 @@ export abstract class BaseRenderLayer implements IRenderLayer { } } - public onOptionsChanged(): void {} - public onBlur(): void {} - public onFocus(): void {} - public onCursorMove(): void {} - public onGridChanged(startRow: number, endRow: number): void {} + public onOptionsChanged(): void { } + public onBlur(): void { } + public onFocus(): void { } + public onCursorMove(): void { } + public onGridChanged(startRow: number, endRow: number): void { } public onSelectionChanged(start: [number, number] | undefined, end: [number, number] | undefined, columnSelectMode: boolean = false): void { this._selectionStart = start; @@ -232,15 +232,12 @@ export abstract class BaseRenderLayer implements IRenderLayer { this._ctx.beginPath(); this._ctx.strokeStyle = this._ctx.fillStyle; this._ctx.lineWidth = window.devicePixelRatio; - this._ctx.setLineDash([window.devicePixelRatio * 2, window.devicePixelRatio]); + this._ctx.setLineDash([window.devicePixelRatio, window.devicePixelRatio]); const xLeft = x * this._scaledCellWidth; const yMid = (y + 1) * this._scaledCellHeight - window.devicePixelRatio - 1; this._ctx.moveTo(xLeft, yMid); - for (let xOffset = 0; xOffset < width; xOffset++) { - // const xLeft = x * this._scaledCellWidth; - const xRight = (x + width + xOffset) * this._scaledCellWidth; - this._ctx.lineTo(xRight, yMid); - } + const xRight = (x + width) * this._scaledCellWidth; + this._ctx.lineTo(xRight, yMid); this._ctx.stroke(); this._ctx.closePath(); this._ctx.restore(); @@ -639,9 +636,9 @@ export abstract class BaseRenderLayer implements IRenderLayer { x < end[0] && y < end[1]; } return (y > start[1] && y < end[1]) || - (start[1] === end[1] && y === start[1] && x >= start[0] && x < end[0]) || - (start[1] < end[1] && y === end[1] && x < end[0]) || - (start[1] < end[1] && y === start[1] && x >= start[0]); + (start[1] === end[1] && y === start[1] && x >= start[0] && x < end[0]) || + (start[1] < end[1] && y === end[1] && x < end[0]) || + (start[1] < end[1] && y === start[1] && x >= start[0]); } } diff --git a/addons/xterm-addon-canvas/src/TextRenderLayer.ts b/addons/xterm-addon-canvas/src/TextRenderLayer.ts index 86e21ad5c8..f422aba653 100644 --- a/addons/xterm-addon-canvas/src/TextRenderLayer.ts +++ b/addons/xterm-addon-canvas/src/TextRenderLayer.ts @@ -8,7 +8,7 @@ import { CharData, ICellData } from 'common/Types'; import { GridCache } from './GridCache'; import { BaseRenderLayer } from './BaseRenderLayer'; import { AttributeData } from 'common/buffer/AttributeData'; -import { NULL_CELL_CODE, Content, UnderlineStyle } from 'common/buffer/Constants'; +import { NULL_CELL_CODE, Content, UnderlineStyle, FgFlags } from 'common/buffer/Constants'; import { IColorSet } from 'browser/Types'; import { CellData } from 'common/buffer/CellData'; import { IOptionsService, IBufferService, IDecorationService } from 'common/services/Services'; @@ -230,79 +230,145 @@ export class TextRenderLayer extends BaseRenderLayer { ctx.restore(); } + /** + * Draws the foreground for a specified range of columns. Tries to batch adjacent + * cells of the same color together to reduce draw calls. + */ private _drawForeground(firstRow: number, lastRow: number): void { + const cols = this._bufferService.cols; + let startX: number = 0; + let startY: number = 0; + let prevFillStyle: string | null = null; + let prevStyle: FgFlags | undefined = undefined; + let prevUnderlineStyle: UnderlineStyle = UnderlineStyle.NONE; + this._forEachCell(firstRow, lastRow, (cell, x, y) => { + let nextFillStyle: string | null = null; + let nextStyle: FgFlags | undefined = undefined; + let nextUnderlineStyle: UnderlineStyle = UnderlineStyle.NONE; + if (cell.isInvisible()) { - return; + nextFillStyle = null; + nextStyle = undefined; + nextUnderlineStyle = UnderlineStyle.NONE; } - this._drawChars(cell, x, y); + else { + this._drawChars(cell, x, y); + } + if (cell.isUnderline() || cell.isStrikethrough()) { - this._ctx.save(); + // this._ctx.save(); if (cell.isInverse()) { if (cell.isBgDefault()) { - this._ctx.fillStyle = this._colors.background.css; + nextFillStyle = this._colors.background.css; } else if (cell.isBgRGB()) { - this._ctx.fillStyle = `rgb(${AttributeData.toColorRGB(cell.getBgColor()).join(',')})`; + nextFillStyle = `rgb(${AttributeData.toColorRGB(cell.getBgColor()).join(',')})`; } else { let bg = cell.getBgColor(); if (this._optionsService.rawOptions.drawBoldTextInBrightColors && cell.isBold() && bg < 8) { bg += 8; } - this._ctx.fillStyle = this._colors.ansi[bg].css; + nextFillStyle = this._colors.ansi[bg].css; } } else { if (cell.isFgDefault()) { - this._ctx.fillStyle = this._colors.foreground.css; + nextFillStyle = this._colors.foreground.css; } else if (cell.isFgRGB()) { - this._ctx.fillStyle = `rgb(${AttributeData.toColorRGB(cell.getFgColor()).join(',')})`; + nextFillStyle = `rgb(${AttributeData.toColorRGB(cell.getFgColor()).join(',')})`; } else { let fg = cell.getFgColor(); if (this._optionsService.rawOptions.drawBoldTextInBrightColors && cell.isBold() && fg < 8) { fg += 8; } - this._ctx.fillStyle = this._colors.ansi[fg].css; + nextFillStyle = this._colors.ansi[fg].css; } } if (cell.isStrikethrough()) { - this._fillMiddleLineAtCells(x, y, cell.getWidth()); + nextStyle = (nextStyle || 0) + FgFlags.STRIKETHROUGH; } if (cell.isUnderline()) { + nextStyle = (nextStyle || 0) + FgFlags.UNDERLINE; if (!cell.isUnderlineColorDefault()) { if (cell.isUnderlineColorRGB()) { - this._ctx.fillStyle = `rgb(${AttributeData.toColorRGB(cell.getUnderlineColor()).join(',')})`; + nextFillStyle = `rgb(${AttributeData.toColorRGB(cell.getUnderlineColor()).join(',')})`; } else { let fg = cell.getUnderlineColor(); if (this._optionsService.rawOptions.drawBoldTextInBrightColors && cell.isBold() && fg < 8) { fg += 8; } - this._ctx.fillStyle = this._colors.ansi[fg].css; + nextFillStyle = this._colors.ansi[fg].css; } } - switch (cell.extended.underlineStyle) { - case UnderlineStyle.DOUBLE: - this._fillBottomLineAtCells(x, y, cell.getWidth(), -window.devicePixelRatio); - this._fillBottomLineAtCells(x, y, cell.getWidth(), window.devicePixelRatio); - break; - case UnderlineStyle.CURLY: - this._curlyUnderlineAtCell(x, y, cell.getWidth()); - break; - case UnderlineStyle.DOTTED: - this._dottedUnderlineAtCell(x, y, cell.getWidth()); - break; - case UnderlineStyle.DASHED: - this._dashedUnderlineAtCell(x, y, cell.getWidth()); - break; - case UnderlineStyle.SINGLE: - default: - this._fillBottomLineAtCells(x, y, cell.getWidth()); - break; - } + nextUnderlineStyle = cell.extended.underlineStyle; } - this._ctx.restore(); } + + if (prevFillStyle === null) { + // This is either the first iteration, or the default foreground was set. Either way, we + // don't need to draw anything. + startX = x; + startY = y; + } + + if (y !== startY) { + // our row changed, draw the previous row + this._drawForegroundHelper(startX, startY, cols - startX, prevFillStyle, prevStyle!, prevUnderlineStyle); + startX = x; + startY = y; + } + else if (prevFillStyle !== nextFillStyle || prevStyle !== nextStyle || prevUnderlineStyle !== nextUnderlineStyle) { + // something changed in the middle of a line + this._drawForegroundHelper(startX, startY, x - startX, prevFillStyle, prevStyle!, prevUnderlineStyle); + startX = x; + startY = y; + } + + prevFillStyle = nextFillStyle; + prevStyle = nextStyle; + prevUnderlineStyle = nextUnderlineStyle; }); + + // flush the last style we encountered + if (prevFillStyle || prevStyle || prevUnderlineStyle) { + this._drawForegroundHelper(startX, startY, cols - startX, prevFillStyle, prevStyle, prevUnderlineStyle); + } + + // this._ctx.restore(); + } + + private _drawForegroundHelper(startX: number, startY: number, width: number, fillStyle: string | null, style: FgFlags | undefined, underlineStyle: UnderlineStyle): void { + if (fillStyle) { + this._ctx.fillStyle = fillStyle; + } + if (style !== undefined) { + + if (style & FgFlags.STRIKETHROUGH) { + this._fillMiddleLineAtCells(startX, startY, width); + } + if (style & FgFlags.UNDERLINE) { + switch (underlineStyle) { + case UnderlineStyle.DOUBLE: + this._fillBottomLineAtCells(startX, startY, width, -window.devicePixelRatio); + this._fillBottomLineAtCells(startX, startY, width, window.devicePixelRatio); + break; + case UnderlineStyle.CURLY: + this._curlyUnderlineAtCell(startX, startY, width); + break; + case UnderlineStyle.DOTTED: + this._dottedUnderlineAtCell(startX, startY, width); + break; + case UnderlineStyle.DASHED: + this._dashedUnderlineAtCell(startX, startY, width); + break; + case UnderlineStyle.SINGLE: + default: + this._fillBottomLineAtCells(startX, startY, width); + break; + } + } + } } public onGridChanged(firstRow: number, lastRow: number): void { From d5b3c726dc7d257bcf20a7b25d88c2de21911c73 Mon Sep 17 00:00:00 2001 From: Simon Lamon Date: Sun, 4 Sep 2022 11:14:59 +0000 Subject: [PATCH 2/2] clean up context save/restore --- addons/xterm-addon-canvas/src/TextRenderLayer.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/addons/xterm-addon-canvas/src/TextRenderLayer.ts b/addons/xterm-addon-canvas/src/TextRenderLayer.ts index f422aba653..4c20b207c8 100644 --- a/addons/xterm-addon-canvas/src/TextRenderLayer.ts +++ b/addons/xterm-addon-canvas/src/TextRenderLayer.ts @@ -242,6 +242,8 @@ export class TextRenderLayer extends BaseRenderLayer { let prevStyle: FgFlags | undefined = undefined; let prevUnderlineStyle: UnderlineStyle = UnderlineStyle.NONE; + this._ctx.save(); + this._forEachCell(firstRow, lastRow, (cell, x, y) => { let nextFillStyle: string | null = null; let nextStyle: FgFlags | undefined = undefined; @@ -257,8 +259,6 @@ export class TextRenderLayer extends BaseRenderLayer { } if (cell.isUnderline() || cell.isStrikethrough()) { - // this._ctx.save(); - if (cell.isInverse()) { if (cell.isBgDefault()) { nextFillStyle = this._colors.background.css; @@ -335,7 +335,7 @@ export class TextRenderLayer extends BaseRenderLayer { this._drawForegroundHelper(startX, startY, cols - startX, prevFillStyle, prevStyle, prevUnderlineStyle); } - // this._ctx.restore(); + this._ctx.restore(); } private _drawForegroundHelper(startX: number, startY: number, width: number, fillStyle: string | null, style: FgFlags | undefined, underlineStyle: UnderlineStyle): void {