refactor(editor): simplify worker renderer message and canvas transfer (#10199)

- Fixed frame delay on panning.
- Removed redundant worker message.
- Removed redundant offscreen bitmap transfer.
- Refactored logic using a clearer `refresh` method entry.
- Extracted plain utils.
This commit is contained in:
doodlewind
2025-02-17 02:35:28 +00:00
parent 04cb303535
commit 1476ca922b
4 changed files with 189 additions and 206 deletions

View File

@@ -0,0 +1,109 @@
import type { Viewport } from '@blocksuite/block-std/gfx';
import { Pane } from 'tweakpane';
import { getSentenceRects, segmentSentences } from './text-utils.js';
import type { ParagraphLayout, SectionLayout } from './types.js';
export function syncCanvasSize(canvas: HTMLCanvasElement, host: HTMLElement) {
const hostRect = host.getBoundingClientRect();
const dpr = window.devicePixelRatio;
canvas.style.position = 'absolute';
canvas.style.left = '0px';
canvas.style.top = '0px';
canvas.style.width = '100%';
canvas.style.height = '100%';
canvas.width = hostRect.width * dpr;
canvas.height = hostRect.height * dpr;
canvas.style.pointerEvents = 'none';
}
export function getSectionLayout(
host: HTMLElement,
viewport: Viewport
): SectionLayout {
const paragraphBlocks = host.querySelectorAll(
'.affine-paragraph-rich-text-wrapper [data-v-text="true"]'
);
const zoom = viewport.zoom;
let sectionMinX = Infinity;
let sectionMinY = Infinity;
let sectionMaxX = -Infinity;
let sectionMaxY = -Infinity;
const paragraphs: ParagraphLayout[] = Array.from(paragraphBlocks).map(p => {
const sentences = segmentSentences(p.textContent || '');
const sentenceLayouts = sentences.map(sentence => {
const rects = getSentenceRects(p, sentence);
rects.forEach(({ rect }) => {
sectionMinX = Math.min(sectionMinX, rect.x);
sectionMinY = Math.min(sectionMinY, rect.y);
sectionMaxX = Math.max(sectionMaxX, rect.x + rect.w);
sectionMaxY = Math.max(sectionMaxY, rect.y + rect.h);
});
return {
text: sentence,
rects: rects.map(rect => {
const [x, y] = viewport.toModelCoordFromClientCoord([
rect.rect.x,
rect.rect.y,
]);
return {
...rect,
rect: {
x,
y,
w: rect.rect.w / zoom / viewport.viewScale,
h: rect.rect.h / zoom / viewport.viewScale,
},
};
}),
};
});
return {
sentences: sentenceLayouts,
zoom,
};
});
const sectionModelCoord = viewport.toModelCoordFromClientCoord([
sectionMinX,
sectionMinY,
]);
const w = (sectionMaxX - sectionMinX) / zoom / viewport.viewScale;
const h = (sectionMaxY - sectionMinY) / zoom / viewport.viewScale;
const section: SectionLayout = {
paragraphs,
rect: {
x: sectionModelCoord[0],
y: sectionModelCoord[1],
w: Math.max(w, 0),
h: Math.max(h, 0),
},
};
return section;
}
export function initTweakpane(
viewportElement: HTMLElement,
onStateChange: (value: boolean) => void
) {
const debugPane = new Pane({ container: viewportElement });
const paneElement = debugPane.element;
paneElement.style.position = 'absolute';
paneElement.style.top = '10px';
paneElement.style.left = '10px';
paneElement.style.width = '250px';
debugPane.title = 'Viewport Turbo Renderer';
const params = { enabled: true };
debugPane
.addBinding(params, 'enabled', {
label: 'Enable',
})
.on('change', ({ value }) => {
onStateChange(value);
});
}

View File

@@ -1,8 +1,9 @@
import { type SectionLayout } from './types.js'; import { type SectionLayout } from './types.js';
type WorkerMessageInit = { type WorkerMessagePaint = {
type: 'initSection'; type: 'paintSection';
data: { data: {
section: SectionLayout;
width: number; width: number;
height: number; height: number;
dpr: number; dpr: number;
@@ -10,14 +11,7 @@ type WorkerMessageInit = {
}; };
}; };
type WorkerMessagePaint = { type WorkerMessage = WorkerMessagePaint;
type: 'paintSection';
data: {
section: SectionLayout;
};
};
type WorkerMessage = WorkerMessageInit | WorkerMessagePaint;
const meta = { const meta = {
emSize: 2048, emSize: 2048,
@@ -46,14 +40,21 @@ function getBaseline() {
/** Section painter in worker */ /** Section painter in worker */
class SectionPainter { class SectionPainter {
private canvas: OffscreenCanvas | null = null; private readonly canvas: OffscreenCanvas = new OffscreenCanvas(0, 0);
private ctx: OffscreenCanvasRenderingContext2D | null = null; private ctx: OffscreenCanvasRenderingContext2D | null = null;
private zoom = 1; private zoom = 1;
init(modelWidth: number, modelHeight: number, dpr: number, zoom: number) { setSize(
const width = modelWidth * dpr * zoom; sectionRectW: number,
const height = modelHeight * dpr * zoom; sectionRectH: number,
this.canvas = new OffscreenCanvas(width, height); dpr: number,
zoom: number
) {
const width = sectionRectW * dpr * zoom;
const height = sectionRectH * dpr * zoom;
this.canvas.width = width;
this.canvas.height = height;
this.ctx = this.canvas.getContext('2d')!; this.ctx = this.canvas.getContext('2d')!;
this.ctx.scale(dpr, dpr); this.ctx.scale(dpr, dpr);
this.zoom = zoom; this.zoom = zoom;
@@ -130,13 +131,9 @@ self.onmessage = async (e: MessageEvent<WorkerMessage>) => {
} }
switch (type) { switch (type) {
case 'initSection': {
const { width, height, dpr, zoom } = data;
painter.init(width, height, dpr, zoom);
break;
}
case 'paintSection': { case 'paintSection': {
const { section } = data; const { section, width, height, dpr, zoom } = data;
painter.setSize(width, height, dpr, zoom);
painter.paint(section); painter.paint(section);
break; break;
} }

View File

@@ -6,16 +6,27 @@ import {
} from '@blocksuite/block-std'; } from '@blocksuite/block-std';
import { GfxControllerIdentifier } from '@blocksuite/block-std/gfx'; import { GfxControllerIdentifier } from '@blocksuite/block-std/gfx';
import { type Container, type ServiceIdentifier } from '@blocksuite/global/di'; import { type Container, type ServiceIdentifier } from '@blocksuite/global/di';
import { Pane } from 'tweakpane'; import { nextTick } from '@blocksuite/global/utils';
import { type Pane } from 'tweakpane';
import { getSentenceRects, segmentSentences } from './text-utils.js'; import {
import { type ParagraphLayout, type SectionLayout } from './types.js'; getSectionLayout,
initTweakpane,
syncCanvasSize,
} from './dom-utils.js';
import { type SectionLayout } from './types.js';
export const ViewportTurboRendererIdentifier = LifeCycleWatcherIdentifier( export const ViewportTurboRendererIdentifier = LifeCycleWatcherIdentifier(
'ViewportTurboRenderer' 'ViewportTurboRenderer'
) as ServiceIdentifier<ViewportTurboRendererExtension>; ) as ServiceIdentifier<ViewportTurboRendererExtension>;
interface Tile {
bitmap: ImageBitmap;
}
export class ViewportTurboRendererExtension extends LifeCycleWatcher { export class ViewportTurboRendererExtension extends LifeCycleWatcher {
state: 'monitoring' | 'paused' = 'paused';
static override setup(di: Container) { static override setup(di: Container) {
di.addImpl(ViewportTurboRendererIdentifier, this, [StdIdentifier]); di.addImpl(ViewportTurboRendererIdentifier, this, [StdIdentifier]);
} }
@@ -24,7 +35,7 @@ export class ViewportTurboRendererExtension extends LifeCycleWatcher {
private readonly worker: Worker; private readonly worker: Worker;
private lastZoom: number | null = null; private lastZoom: number | null = null;
private lastSection: SectionLayout | null = null; private lastSection: SectionLayout | null = null;
private lastBitmap: ImageBitmap | null = null; private tile: Tile | null = null;
private debugPane: Pane | null = null; private debugPane: Pane | null = null;
constructor(std: BlockStdScope) { constructor(std: BlockStdScope) {
@@ -38,21 +49,26 @@ export class ViewportTurboRendererExtension extends LifeCycleWatcher {
const viewportElement = document.querySelector('.affine-edgeless-viewport'); const viewportElement = document.querySelector('.affine-edgeless-viewport');
if (viewportElement) { if (viewportElement) {
viewportElement.append(this.canvas); viewportElement.append(this.canvas);
this.debugPane = new Pane({ container: viewportElement as HTMLElement }); initTweakpane(viewportElement as HTMLElement, (value: boolean) => {
this.initTweakpane(); this.state = value ? 'monitoring' : 'paused';
this.canvas.style.display = value ? 'block' : 'none';
});
} }
this.viewport.viewportUpdated.on(async () => { syncCanvasSize(this.canvas, this.std.host);
await this.render(); this.viewport.viewportUpdated.on(() => {
this.refresh().catch(console.error);
}); });
document.fonts.load('15px Inter').then(async () => { document.fonts.load('15px Inter').then(() => {
await this.render(); this.state = 'monitoring';
this.refresh().catch(console.error);
}); });
} }
override unmounted() { override unmounted() {
if (this.lastBitmap) { if (this.tile) {
this.lastBitmap.close(); this.tile.bitmap.close();
this.tile = null;
} }
if (this.debugPane) { if (this.debugPane) {
this.debugPane.dispose(); this.debugPane.dispose();
@@ -62,126 +78,38 @@ export class ViewportTurboRendererExtension extends LifeCycleWatcher {
this.canvas.remove(); this.canvas.remove();
} }
private initTweakpane() {
if (!this.debugPane) return;
const paneElement = this.debugPane.element;
paneElement.style.position = 'absolute';
paneElement.style.top = '10px';
paneElement.style.left = '10px';
paneElement.style.width = '250px';
this.debugPane.title = 'Viewport Turbo Renderer';
const params = {
enabled: true,
};
this.debugPane
.addBinding(params, 'enabled', {
label: 'Enable',
})
.on('change', ({ value }) => {
this.canvas.style.display = value ? 'block' : 'none';
});
}
get viewport() { get viewport() {
return this.std.get(GfxControllerIdentifier).viewport; return this.std.get(GfxControllerIdentifier).viewport;
} }
getHostRect() { private async refresh() {
return this.std.host.getBoundingClientRect(); await nextTick(); // Improves stability during zooming
if (this.canUseCache()) {
this.drawCachedBitmap(this.lastSection!);
} else {
const section = getSectionLayout(this.std.host, this.viewport);
await this.paintSection(section);
this.lastSection = section;
this.lastZoom = this.viewport.zoom;
this.drawCachedBitmap(section);
}
} }
getHostLayout() { private async paintSection(section: SectionLayout): Promise<void> {
if (!document.fonts.check('15px Inter')) return null;
const paragraphBlocks = this.std.host.querySelectorAll(
'.affine-paragraph-rich-text-wrapper [data-v-text="true"]'
);
const { viewport } = this;
const zoom = this.viewport.zoom;
const hostRect = this.getHostRect();
let sectionMinX = Infinity;
let sectionMinY = Infinity;
let sectionMaxX = -Infinity;
let sectionMaxY = -Infinity;
const paragraphs: ParagraphLayout[] = Array.from(paragraphBlocks).map(p => {
const sentences = segmentSentences(p.textContent || '');
const sentenceLayouts = sentences.map(sentence => {
const rects = getSentenceRects(p, sentence);
rects.forEach(({ rect }) => {
sectionMinX = Math.min(sectionMinX, rect.x);
sectionMinY = Math.min(sectionMinY, rect.y);
sectionMaxX = Math.max(sectionMaxX, rect.x + rect.w);
sectionMaxY = Math.max(sectionMaxY, rect.y + rect.h);
});
return {
text: sentence,
rects: rects.map(rect => {
const [x, y] = viewport.toModelCoordFromClientCoord([
rect.rect.x,
rect.rect.y,
]);
return {
...rect,
rect: {
x,
y,
w: rect.rect.w / zoom / viewport.viewScale,
h: rect.rect.h / zoom / viewport.viewScale,
},
};
}),
};
});
return {
sentences: sentenceLayouts,
zoom,
};
});
if (paragraphs.length === 0) return null;
const sectionModelCoord = viewport.toModelCoordFromClientCoord([
sectionMinX,
sectionMinY,
]);
const w = (sectionMaxX - sectionMinX) / zoom / viewport.viewScale;
const h = (sectionMaxY - sectionMinY) / zoom / viewport.viewScale;
const section: SectionLayout = {
paragraphs,
rect: {
x: sectionModelCoord[0],
y: sectionModelCoord[1],
w: Math.max(w, 0),
h: Math.max(h, 0),
},
};
return { section, hostRect };
}
private initSectionRenderer(width: number, height: number) {
const dpr = window.devicePixelRatio;
this.worker.postMessage({
type: 'initSection',
data: { width, height, dpr, zoom: this.viewport.zoom },
});
}
private async renderSection(section: SectionLayout): Promise<void> {
return new Promise(resolve => { return new Promise(resolve => {
if (!this.worker) return; if (!this.worker) return;
const dpr = window.devicePixelRatio;
this.worker.postMessage({ this.worker.postMessage({
type: 'paintSection', type: 'paintSection',
data: { section }, data: {
section,
width: section.rect.w,
height: section.rect.h,
dpr,
zoom: this.viewport.zoom,
},
}); });
this.worker.onmessage = (e: MessageEvent) => { this.worker.onmessage = (e: MessageEvent) => {
@@ -197,90 +125,39 @@ export class ViewportTurboRendererExtension extends LifeCycleWatcher {
section: SectionLayout, section: SectionLayout,
resolve: () => void resolve: () => void
) { ) {
const tempCanvas = new OffscreenCanvas(bitmap.width, bitmap.height); if (this.tile) {
const tempCtx = tempCanvas.getContext('2d')!; this.tile.bitmap.close();
tempCtx.drawImage(bitmap, 0, 0); }
const bitmapCopy = tempCanvas.transferToImageBitmap(); this.tile = { bitmap };
this.drawCachedBitmap(section);
this.updateCacheState(section, bitmapCopy);
this.drawBitmap(bitmap, section);
resolve(); resolve();
} }
private syncCanvasSize() { private canUseCache(): boolean {
const hostRect = this.getHostRect();
const dpr = window.devicePixelRatio;
this.canvas.style.position = 'absolute';
this.canvas.style.left = '0px';
this.canvas.style.top = '0px';
this.canvas.style.width = '100%';
this.canvas.style.height = '100%';
this.canvas.width = hostRect.width * dpr;
this.canvas.height = hostRect.height * dpr;
this.canvas.style.pointerEvents = 'none';
}
private updateCacheState(section: SectionLayout, bitmapCopy: ImageBitmap) {
this.lastZoom = this.viewport.zoom;
this.lastSection = section;
if (this.lastBitmap) {
this.lastBitmap.close();
}
this.lastBitmap = bitmapCopy;
}
private canUseCache(currentZoom: number): boolean {
return ( return (
this.lastZoom === currentZoom && !!this.lastSection && !!this.lastBitmap !!this.lastSection && !!this.tile && this.viewport.zoom === this.lastZoom
); );
} }
private drawBitmap(bitmap: ImageBitmap, section: SectionLayout) { private drawCachedBitmap(section: SectionLayout) {
if (this.state === 'paused') return;
const bitmap = this.tile!.bitmap;
const ctx = this.canvas.getContext('2d'); const ctx = this.canvas.getContext('2d');
if (!ctx) return; if (!ctx) return;
ctx.clearRect(0, 0, this.canvas.width, this.canvas.height); ctx.clearRect(0, 0, this.canvas.width, this.canvas.height);
const bitmapCanvas = new OffscreenCanvas(
section.rect.w * window.devicePixelRatio * this.viewport.zoom,
section.rect.h * window.devicePixelRatio * this.viewport.zoom
);
const bitmapCtx = bitmapCanvas.getContext('bitmaprenderer');
if (!bitmapCtx) return;
const tempCanvas = new OffscreenCanvas(bitmap.width, bitmap.height);
const tempCtx = tempCanvas.getContext('2d')!;
tempCtx.drawImage(bitmap, 0, 0);
const bitmapCopy = tempCanvas.transferToImageBitmap();
bitmapCtx.transferFromImageBitmap(bitmapCopy);
const sectionViewCoord = this.viewport.toViewCoord( const sectionViewCoord = this.viewport.toViewCoord(
section.rect.x, section.rect.x,
section.rect.y section.rect.y
); );
ctx.drawImage( ctx.drawImage(
bitmapCanvas, bitmap,
sectionViewCoord[0] * window.devicePixelRatio, sectionViewCoord[0] * window.devicePixelRatio,
sectionViewCoord[1] * window.devicePixelRatio, sectionViewCoord[1] * window.devicePixelRatio,
section.rect.w * window.devicePixelRatio * this.viewport.zoom, section.rect.w * window.devicePixelRatio * this.viewport.zoom,
section.rect.h * window.devicePixelRatio * this.viewport.zoom section.rect.h * window.devicePixelRatio * this.viewport.zoom
); );
} }
public async render(): Promise<void> {
const hostLayout = this.getHostLayout();
if (!hostLayout) return;
const { section } = hostLayout;
const currentZoom = this.viewport.zoom;
if (this.canUseCache(currentZoom)) {
this.drawBitmap(this.lastBitmap!, this.lastSection!);
} else {
this.syncCanvasSize();
this.initSectionRenderer(section.rect.w, section.rect.h);
await this.renderSection(section);
}
}
} }

View File

@@ -5,7 +5,7 @@ import { setupEditor } from './setup.js';
async function init() { async function init() {
setupEditor('edgeless', [ViewportTurboRendererExtension]); setupEditor('edgeless', [ViewportTurboRendererExtension]);
addSampleNotes(doc, 6); addSampleNotes(doc, 1);
doc.load(); doc.load();
} }