mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-25 10:22:55 +08:00
refactor(editor): improve worker renderer code structure (#10011)
This commit is contained in:
@@ -2,11 +2,7 @@ import { GfxControllerIdentifier } from '@blocksuite/block-std/gfx';
|
|||||||
import type { AffineEditorContainer } from '@blocksuite/presets';
|
import type { AffineEditorContainer } from '@blocksuite/presets';
|
||||||
|
|
||||||
import { getSentenceRects, segmentSentences } from './text-utils.js';
|
import { getSentenceRects, segmentSentences } from './text-utils.js';
|
||||||
import {
|
import { type ParagraphLayout, type SectionLayout } from './types.js';
|
||||||
type ParagraphLayout,
|
|
||||||
type SectionLayout,
|
|
||||||
type ViewportState,
|
|
||||||
} from './types.js';
|
|
||||||
|
|
||||||
export class CanvasRenderer {
|
export class CanvasRenderer {
|
||||||
private readonly worker: Worker;
|
private readonly worker: Worker;
|
||||||
@@ -25,26 +21,13 @@ export class CanvasRenderer {
|
|||||||
this.editorContainer = editorContainer;
|
this.editorContainer = editorContainer;
|
||||||
this.targetContainer = targetContainer;
|
this.targetContainer = targetContainer;
|
||||||
|
|
||||||
this.worker = new Worker(new URL('./canvas.worker.ts', import.meta.url), {
|
this.worker = new Worker(new URL('./painter.worker.ts', import.meta.url), {
|
||||||
type: 'module',
|
type: 'module',
|
||||||
});
|
});
|
||||||
}
|
|
||||||
|
|
||||||
private initWorkerSize(width: number, height: number) {
|
if (!this.targetContainer.querySelector('canvas')) {
|
||||||
const dpr = window.devicePixelRatio;
|
this.targetContainer.append(this.canvas);
|
||||||
const viewport = this.editorContainer.std.get(
|
}
|
||||||
GfxControllerIdentifier
|
|
||||||
).viewport;
|
|
||||||
const viewportState: ViewportState = {
|
|
||||||
zoom: viewport.zoom,
|
|
||||||
viewScale: viewport.viewScale,
|
|
||||||
viewportX: viewport.viewportX,
|
|
||||||
viewportY: viewport.viewportY,
|
|
||||||
};
|
|
||||||
this.worker.postMessage({
|
|
||||||
type: 'init',
|
|
||||||
data: { width, height, dpr, viewport: viewportState },
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get viewport() {
|
get viewport() {
|
||||||
@@ -124,72 +107,74 @@ export class CanvasRenderer {
|
|||||||
return { section, hostRect };
|
return { section, hostRect };
|
||||||
}
|
}
|
||||||
|
|
||||||
public async render(): Promise<void> {
|
private initSectionRenderer(width: number, height: number) {
|
||||||
const hostLayout = this.getHostLayout();
|
const dpr = window.devicePixelRatio;
|
||||||
if (!hostLayout) return;
|
this.worker.postMessage({
|
||||||
|
type: 'initSection',
|
||||||
const { section } = hostLayout;
|
data: { width, height, dpr, zoom: this.viewport.zoom },
|
||||||
const currentZoom = this.viewport.zoom;
|
});
|
||||||
|
}
|
||||||
// Use bitmap cache
|
|
||||||
if (
|
|
||||||
this.lastZoom === currentZoom &&
|
|
||||||
this.lastSection &&
|
|
||||||
this.lastBitmap &&
|
|
||||||
this.lastMode === this.editorContainer.mode
|
|
||||||
) {
|
|
||||||
this.drawBitmap(this.lastBitmap, this.lastSection);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Need to re-render if zoom changed or no cached bitmap
|
|
||||||
this.initWorkerSize(section.rect.w, section.rect.h);
|
|
||||||
|
|
||||||
|
private async renderSection(section: SectionLayout): Promise<void> {
|
||||||
return new Promise(resolve => {
|
return new Promise(resolve => {
|
||||||
if (!this.worker) return;
|
if (!this.worker) return;
|
||||||
|
|
||||||
this.worker.postMessage({
|
this.worker.postMessage({
|
||||||
type: 'draw',
|
type: 'paintSection',
|
||||||
data: {
|
data: { section },
|
||||||
section,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
this.worker.onmessage = (e: MessageEvent) => {
|
this.worker.onmessage = (e: MessageEvent) => {
|
||||||
const { type, bitmap } = e.data;
|
if (e.data.type === 'bitmapPainted') {
|
||||||
if (type === 'render') {
|
this.handlePaintedBitmap(e.data.bitmap, section, resolve);
|
||||||
const hostRect = this.getHostRect();
|
|
||||||
this.canvas.style.width = hostRect.width + 'px';
|
|
||||||
this.canvas.style.height = hostRect.height + 'px';
|
|
||||||
this.canvas.width = hostRect.width * window.devicePixelRatio;
|
|
||||||
this.canvas.height = hostRect.height * window.devicePixelRatio;
|
|
||||||
|
|
||||||
if (!this.targetContainer.querySelector('canvas')) {
|
|
||||||
this.targetContainer.append(this.canvas);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a copy of bitmap for caching
|
|
||||||
const tempCanvas = new OffscreenCanvas(bitmap.width, bitmap.height);
|
|
||||||
const tempCtx = tempCanvas.getContext('2d')!;
|
|
||||||
tempCtx.drawImage(bitmap, 0, 0);
|
|
||||||
const bitmapCopy = tempCanvas.transferToImageBitmap();
|
|
||||||
|
|
||||||
// Cache the current state
|
|
||||||
this.lastZoom = currentZoom;
|
|
||||||
this.lastSection = section;
|
|
||||||
this.lastMode = this.editorContainer.mode;
|
|
||||||
if (this.lastBitmap) {
|
|
||||||
this.lastBitmap.close();
|
|
||||||
}
|
|
||||||
this.lastBitmap = bitmapCopy;
|
|
||||||
|
|
||||||
this.drawBitmap(bitmap, section);
|
|
||||||
resolve();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private handlePaintedBitmap(
|
||||||
|
bitmap: ImageBitmap,
|
||||||
|
section: SectionLayout,
|
||||||
|
resolve: () => void
|
||||||
|
) {
|
||||||
|
const tempCanvas = new OffscreenCanvas(bitmap.width, bitmap.height);
|
||||||
|
const tempCtx = tempCanvas.getContext('2d')!;
|
||||||
|
tempCtx.drawImage(bitmap, 0, 0);
|
||||||
|
const bitmapCopy = tempCanvas.transferToImageBitmap();
|
||||||
|
|
||||||
|
this.updateCacheState(section, bitmapCopy);
|
||||||
|
this.drawBitmap(bitmap, section);
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
|
||||||
|
private syncCanvasSize() {
|
||||||
|
const hostRect = this.getHostRect();
|
||||||
|
const dpr = window.devicePixelRatio;
|
||||||
|
this.canvas.style.width = `${hostRect.width}px`;
|
||||||
|
this.canvas.style.height = `${hostRect.height}px`;
|
||||||
|
this.canvas.width = hostRect.width * dpr;
|
||||||
|
this.canvas.height = hostRect.height * dpr;
|
||||||
|
}
|
||||||
|
|
||||||
|
private updateCacheState(section: SectionLayout, bitmapCopy: ImageBitmap) {
|
||||||
|
this.lastZoom = this.viewport.zoom;
|
||||||
|
this.lastSection = section;
|
||||||
|
this.lastMode = this.editorContainer.mode;
|
||||||
|
if (this.lastBitmap) {
|
||||||
|
this.lastBitmap.close();
|
||||||
|
}
|
||||||
|
this.lastBitmap = bitmapCopy;
|
||||||
|
}
|
||||||
|
|
||||||
|
private canUseCache(currentZoom: number): boolean {
|
||||||
|
return (
|
||||||
|
this.lastZoom === currentZoom &&
|
||||||
|
!!this.lastSection &&
|
||||||
|
!!this.lastBitmap &&
|
||||||
|
this.lastMode === this.editorContainer.mode
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private drawBitmap(bitmap: ImageBitmap, section: SectionLayout) {
|
private drawBitmap(bitmap: ImageBitmap, section: SectionLayout) {
|
||||||
const ctx = this.canvas.getContext('2d');
|
const ctx = this.canvas.getContext('2d');
|
||||||
if (!ctx) return;
|
if (!ctx) return;
|
||||||
@@ -223,6 +208,22 @@ export class CanvasRenderer {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public destroy() {
|
public destroy() {
|
||||||
if (this.lastBitmap) {
|
if (this.lastBitmap) {
|
||||||
this.lastBitmap.close();
|
this.lastBitmap.close();
|
||||||
|
|||||||
@@ -37,16 +37,15 @@
|
|||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
#to-canvas-button {
|
.top-bar {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 5px;
|
top: 0;
|
||||||
left: 5px;
|
left: 0;
|
||||||
}
|
padding: 5px;
|
||||||
|
display: flex;
|
||||||
#switch-mode-button {
|
gap: 5px;
|
||||||
position: absolute;
|
justify-content: flex-start;
|
||||||
top: 5px;
|
align-items: center;
|
||||||
left: 85px;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
@@ -54,8 +53,10 @@
|
|||||||
<div id="container">
|
<div id="container">
|
||||||
<div id="left-column"></div>
|
<div id="left-column"></div>
|
||||||
<div id="right-column">
|
<div id="right-column">
|
||||||
<button id="to-canvas-button">to canvas</button>
|
<div class="top-bar">
|
||||||
<button id="switch-mode-button">switch mode</button>
|
<button id="to-canvas-button">to canvas</button>
|
||||||
|
<button id="switch-mode-button">switch mode</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<script type="module" src="./main.ts"></script>
|
<script type="module" src="./main.ts"></script>
|
||||||
|
|||||||
@@ -1,23 +1,23 @@
|
|||||||
import { type SectionLayout, type ViewportState } from './types.js';
|
import { type SectionLayout } from './types.js';
|
||||||
|
|
||||||
type WorkerMessageInit = {
|
type WorkerMessageInit = {
|
||||||
type: 'init';
|
type: 'initSection';
|
||||||
data: {
|
data: {
|
||||||
width: number;
|
width: number;
|
||||||
height: number;
|
height: number;
|
||||||
dpr: number;
|
dpr: number;
|
||||||
viewport: ViewportState;
|
zoom: number;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
type WorkerMessageDraw = {
|
type WorkerMessagePaint = {
|
||||||
type: 'draw';
|
type: 'paintSection';
|
||||||
data: {
|
data: {
|
||||||
section: SectionLayout;
|
section: SectionLayout;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
type WorkerMessage = WorkerMessageInit | WorkerMessageDraw;
|
type WorkerMessage = WorkerMessageInit | WorkerMessagePaint;
|
||||||
|
|
||||||
const meta = {
|
const meta = {
|
||||||
emSize: 2048,
|
emSize: 2048,
|
||||||
@@ -45,33 +45,28 @@ function getBaseline() {
|
|||||||
return y;
|
return y;
|
||||||
}
|
}
|
||||||
|
|
||||||
class CanvasWorkerManager {
|
/** Section painter in worker */
|
||||||
|
class SectionPainter {
|
||||||
private canvas: OffscreenCanvas | null = null;
|
private canvas: OffscreenCanvas | null = null;
|
||||||
private ctx: OffscreenCanvasRenderingContext2D | null = null;
|
private ctx: OffscreenCanvasRenderingContext2D | null = null;
|
||||||
private viewport: ViewportState | null = null;
|
private zoom = 1;
|
||||||
|
|
||||||
init(
|
init(modelWidth: number, modelHeight: number, dpr: number, zoom: number) {
|
||||||
modelWidth: number,
|
const width = modelWidth * dpr * zoom;
|
||||||
modelHeight: number,
|
const height = modelHeight * dpr * zoom;
|
||||||
dpr: number,
|
|
||||||
viewport: ViewportState
|
|
||||||
) {
|
|
||||||
const width = modelWidth * dpr * viewport.zoom;
|
|
||||||
const height = modelHeight * dpr * viewport.zoom;
|
|
||||||
this.canvas = new OffscreenCanvas(width, height);
|
this.canvas = new OffscreenCanvas(width, height);
|
||||||
this.ctx = this.canvas.getContext('2d')!;
|
this.ctx = this.canvas.getContext('2d')!;
|
||||||
this.ctx.scale(dpr, dpr);
|
this.ctx.scale(dpr, dpr);
|
||||||
this.ctx.fillStyle = 'lightgrey';
|
this.ctx.fillStyle = 'lightgrey';
|
||||||
this.ctx.fillRect(0, 0, width, height);
|
this.ctx.fillRect(0, 0, width, height);
|
||||||
this.viewport = viewport;
|
this.zoom = zoom;
|
||||||
}
|
}
|
||||||
|
|
||||||
draw(section: SectionLayout) {
|
paint(section: SectionLayout) {
|
||||||
const { canvas, ctx } = this;
|
const { canvas, ctx } = this;
|
||||||
if (!canvas || !ctx) return;
|
if (!canvas || !ctx) return;
|
||||||
|
|
||||||
const zoom = this.viewport!.zoom;
|
ctx.scale(this.zoom, this.zoom);
|
||||||
ctx.scale(zoom, zoom);
|
|
||||||
|
|
||||||
// Track rendered positions to avoid duplicate rendering across all paragraphs and sentences
|
// Track rendered positions to avoid duplicate rendering across all paragraphs and sentences
|
||||||
const renderedPositions = new Set<string>();
|
const renderedPositions = new Set<string>();
|
||||||
@@ -101,24 +96,24 @@ class CanvasWorkerManager {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const bitmap = canvas.transferToImageBitmap();
|
const bitmap = canvas.transferToImageBitmap();
|
||||||
self.postMessage({ type: 'render', bitmap }, { transfer: [bitmap] });
|
self.postMessage({ type: 'bitmapPainted', bitmap }, { transfer: [bitmap] });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const manager = new CanvasWorkerManager();
|
const painter = new SectionPainter();
|
||||||
|
|
||||||
self.onmessage = async (e: MessageEvent<WorkerMessage>) => {
|
self.onmessage = async (e: MessageEvent<WorkerMessage>) => {
|
||||||
const { type, data } = e.data;
|
const { type, data } = e.data;
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'init': {
|
case 'initSection': {
|
||||||
const { width, height, dpr, viewport } = data;
|
const { width, height, dpr, zoom } = data;
|
||||||
manager.init(width, height, dpr, viewport);
|
painter.init(width, height, dpr, zoom);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 'draw': {
|
case 'paintSection': {
|
||||||
await font.load();
|
await font.load();
|
||||||
const { section } = data;
|
const { section } = data;
|
||||||
manager.draw(section);
|
painter.paint(section);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user