fix(editor): worker loading in webpack env (#10832)

### TL;DR

Created dedicated worker entry points to avoid dynamic imports.

### What changed?

- Painters are provided during worker initialization
- Removed `ParagraphPaintConfigExtension` and the associated configuration system
- Created dedicated worker entry points in both the integration test and frontend packages
- Modified `ViewportLayoutPainter` to accept painters in its constructor
- Updated the `TurboRendererConfig` interface to require a `painterWorkerEntry` function

### Why make this change?

Webpack support. Extension objects in main thread are not available to be passed into workers. Dynamic painter path import is hard to support in webpack environment. With the [webpack-ignore](https://webpack.js.org/api/module-methods/#webpackignore) rule, there are still build errors in webpack.
This commit is contained in:
doodlewind
2025-03-14 05:26:57 +00:00
parent 8af0526a22
commit be9f44fc4f
13 changed files with 100 additions and 99 deletions

View File

@@ -4,4 +4,4 @@ export * from './paragraph-block.js';
export * from './paragraph-service.js';
export * from './paragraph-spec.js';
export * from './turbo/paragraph-layout-provider.js';
export * from './turbo/paragraph-painter-config.js';
export * from './turbo/paragraph-painter.worker.js';

View File

@@ -1,16 +0,0 @@
import { BlockPainterConfigIdentifier } from '@blocksuite/affine-gfx-turbo-renderer';
import type { Container } from '@blocksuite/global/di';
import { Extension } from '@blocksuite/store';
export class ParagraphPaintConfigExtension extends Extension {
static override setup(di: Container) {
const config = {
type: 'affine:paragraph',
path: new URL(
'@blocksuite/affine-block-paragraph/turbo-painter',
import.meta.url
).href,
};
di.addImpl(BlockPainterConfigIdentifier, config);
}
}

View File

@@ -39,16 +39,23 @@ function isParagraphLayout(layout: BlockLayout): layout is ParagraphLayout {
return layout.type === 'affine:paragraph';
}
export default class ParagraphLayoutPainter implements BlockLayoutPainter {
static readonly font = new FontFace(
'Inter',
`url(https://fonts.gstatic.com/s/inter/v18/UcCo3FwrK3iLTcviYwYZ8UA3.woff2)`
);
export class ParagraphLayoutPainter implements BlockLayoutPainter {
private static readonly supportFontFace =
typeof FontFace !== 'undefined' &&
typeof self !== 'undefined' &&
'fonts' in self;
static fontLoaded = false;
static readonly font = ParagraphLayoutPainter.supportFontFace
? new FontFace(
'Inter',
`url(https://fonts.gstatic.com/s/inter/v18/UcCo3FwrK3iLTcviYwYZ8UA3.woff2)`
)
: null;
static fontLoaded = !ParagraphLayoutPainter.supportFontFace;
static {
if (typeof self !== 'undefined' && 'fonts' in self) {
if (ParagraphLayoutPainter.supportFontFace && ParagraphLayoutPainter.font) {
// @ts-expect-error worker fonts API
self.fonts.add(ParagraphLayoutPainter.font);

View File

@@ -1,4 +1,5 @@
export * from './layout/block-layout-provider';
export * from './painter/painter.worker';
export * from './text-utils';
export * from './turbo-renderer';
export * from './types';

View File

@@ -17,12 +17,19 @@ class BlockPainterRegistry {
}
}
class ViewportLayoutPainter {
export class ViewportLayoutPainter {
private readonly canvas: OffscreenCanvas = new OffscreenCanvas(0, 0);
private ctx: OffscreenCanvasRenderingContext2D | null = null;
private zoom = 1;
public readonly registry = new BlockPainterRegistry();
constructor(painters: Record<string, BlockLayoutPainter>) {
Object.entries(painters).forEach(([type, painter]) => {
this.registry.register(type, painter);
});
self.onmessage = this.handler;
}
setSize(layoutRectW: number, layoutRectH: number, dpr: number, zoom: number) {
const width = layoutRectW * dpr * zoom;
const height = layoutRectH * dpr * zoom;
@@ -66,26 +73,16 @@ class ViewportLayoutPainter {
};
self.postMessage(message, { transfer: [bitmap] });
}
handler = async (e: MessageEvent<HostToWorkerMessage>) => {
const { type, data } = e.data;
switch (type) {
case 'paintLayout': {
const { layout, width, height, dpr, zoom, version } = data;
this.setSize(width, height, dpr, zoom);
this.paint(layout, version);
break;
}
}
};
}
const painter = new ViewportLayoutPainter();
self.onmessage = async (e: MessageEvent<HostToWorkerMessage>) => {
const { type, data } = e.data;
switch (type) {
case 'paintLayout': {
const { layout, width, height, dpr, zoom, version } = data;
painter.setSize(width, height, dpr, zoom);
painter.paint(layout, version);
break;
}
case 'registerPainter': {
const { painterConfigs } = data;
painterConfigs.forEach(async ({ type, path }) => {
painter.registry.register(type, new (await import(path)).default());
});
break;
}
}
};

View File

@@ -6,7 +6,6 @@ import {
type GfxViewportElement,
} from '@blocksuite/block-std/gfx';
import type { Container } from '@blocksuite/global/di';
import { createIdentifier } from '@blocksuite/global/di';
import { DisposableGroup } from '@blocksuite/global/disposable';
import debounce from 'lodash-es/debounce';
@@ -18,9 +17,7 @@ import {
syncCanvasSize,
} from './renderer-utils';
import type {
BlockPainterConfig,
MessagePaint,
MessageRegisterPainter,
RendererOptions,
RenderingState,
TurboRendererConfig,
@@ -29,16 +26,12 @@ import type {
} from './types';
const debug = false; // Toggle for debug logs
const workerUrl = new URL('./painter/painter.worker.ts', import.meta.url);
const defaultOptions: RendererOptions = {
zoomThreshold: 1, // With high enough zoom, fallback to DOM rendering
debounceTime: 1000, // During this period, fallback to DOM
};
export const BlockPainterConfigIdentifier =
createIdentifier<BlockPainterConfig>('block-painter-config');
export const TurboRendererConfigFactory =
ConfigExtensionFactory<TurboRendererConfig>('viewport-turbo-renderer');
@@ -47,13 +40,24 @@ export class ViewportTurboRendererExtension extends GfxExtension {
public state: RenderingState = 'inactive';
public readonly canvas: HTMLCanvasElement = document.createElement('canvas');
private readonly worker: Worker = new Worker(workerUrl, { type: 'module' });
private readonly worker: Worker;
private readonly disposables = new DisposableGroup();
private layoutCacheData: ViewportLayout | null = null;
private layoutVersion = 0;
private bitmap: ImageBitmap | null = null;
private viewportElement: GfxViewportElement | null = null;
constructor(gfx: GfxController) {
super(gfx);
const id = TurboRendererConfigFactory.identifier;
const config = this.std.getOptional(id);
if (!config) {
throw new Error('TurboRendererConfig not found');
}
this.worker = config.painterWorkerEntry();
}
static override extendGfx(gfx: GfxController) {
Object.defineProperty(gfx, 'turboRenderer', {
get() {
@@ -75,13 +79,6 @@ export class ViewportTurboRendererExtension extends GfxExtension {
};
}
get painterConfigs() {
const painterConfigMap = this.std.provider.getAll(
BlockPainterConfigIdentifier
);
return Array.from(painterConfigMap.values());
}
override mounted() {
const mountPoint = document.querySelector('.affine-edgeless-viewport');
if (mountPoint) {
@@ -123,13 +120,6 @@ export class ViewportTurboRendererExtension extends GfxExtension {
this.disposables.add(
this.std.store.slots.blockUpdated.subscribe(() => this.invalidate())
);
const painterConfigs = this.painterConfigs;
const message: MessageRegisterPainter = {
type: 'registerPainter',
data: { painterConfigs },
};
this.worker.postMessage(message);
}
override unmounted() {

View File

@@ -83,20 +83,9 @@ export interface RendererOptions {
debounceTime: number;
}
export interface BlockPainterConfig {
type: string;
path: string;
}
export interface TurboRendererConfig {
options?: Partial<RendererOptions>;
painterWorkerEntry: () => Worker;
}
export type MessageRegisterPainter = {
type: 'registerPainter';
data: {
painterConfigs: BlockPainterConfig[];
};
};
export type HostToWorkerMessage = MessagePaint | MessageRegisterPainter;
export type HostToWorkerMessage = MessagePaint;

View File

@@ -1,13 +1,21 @@
import { ViewportTurboRendererExtension } from '@blocksuite/affine-gfx-turbo-renderer';
import { ParagraphLayoutHandlerExtension } from '@blocksuite/affine/blocks/paragraph';
import {
TurboRendererConfigFactory,
ViewportTurboRendererExtension,
} from '@blocksuite/affine-gfx-turbo-renderer';
import { beforeEach, describe, expect, test } from 'vitest';
import { wait } from '../utils/common.js';
import { addSampleNotes } from '../utils/doc-generator.js';
import { setupEditor } from '../utils/setup.js';
import { createPainterWorker, setupEditor } from '../utils/setup.js';
describe('viewport turbo renderer', () => {
beforeEach(async () => {
const cleanup = await setupEditor('edgeless', [
ParagraphLayoutHandlerExtension,
TurboRendererConfigFactory({
painterWorkerEntry: createPainterWorker,
}),
ViewportTurboRendererExtension,
]);
return cleanup;

View File

@@ -1,7 +1,4 @@
import {
ParagraphLayoutHandlerExtension,
ParagraphPaintConfigExtension,
} from '@blocksuite/affine/blocks/paragraph';
import { ParagraphLayoutHandlerExtension } from '@blocksuite/affine/blocks/paragraph';
import {
TurboRendererConfigFactory,
ViewportTurboRendererExtension,
@@ -9,17 +6,13 @@ import {
} from '@blocksuite/affine/gfx/turbo-renderer';
import { addSampleNotes } from './doc-generator.js';
import { setupEditor } from './setup.js';
import { createPainterWorker, setupEditor } from './setup.js';
async function init() {
setupEditor('edgeless', [
ParagraphLayoutHandlerExtension,
ParagraphPaintConfigExtension,
TurboRendererConfigFactory({
options: {
zoomThreshold: 1,
debounceTime: 1000,
},
painterWorkerEntry: createPainterWorker,
}),
ViewportTurboRendererExtension,
]);

View File

@@ -99,6 +99,16 @@ async function createEditor(
return app;
}
export function createPainterWorker() {
const worker = new Worker(
new URL('./turbo-painter-entry.worker.ts', import.meta.url),
{
type: 'module',
}
);
return worker;
}
export async function setupEditor(
mode: DocMode = 'page',
extensions: ExtensionType[] = []

View File

@@ -0,0 +1,6 @@
import { ParagraphLayoutPainter } from '@blocksuite/affine-block-paragraph/turbo-painter';
import { ViewportLayoutPainter } from '@blocksuite/affine-gfx-turbo-renderer/painter';
new ViewportLayoutPainter({
'affine:paragraph': new ParagraphLayoutPainter(),
});