Files
AFFiNE-Mirror/blocksuite/affine/blocks/root/src/edgeless/edgeless-root-block.ts

543 lines
14 KiB
TypeScript

import { NoteConfigExtension } from '@blocksuite/affine-block-note';
import type {
SurfaceBlockComponent,
SurfaceBlockModel,
} from '@blocksuite/affine-block-surface';
import {
EdgelessLegacySlotIdentifier,
getBgGridGap,
normalizeWheelDeltaY,
} from '@blocksuite/affine-block-surface';
import { isSingleMindMapNode } from '@blocksuite/affine-gfx-mindmap';
import { mountShapeTextEditor } from '@blocksuite/affine-gfx-shape';
import {
NoteBlockModel,
type RootBlockModel,
type ShapeElementModel,
} from '@blocksuite/affine-model';
import {
EditorSettingProvider,
EditPropsStore,
FontLoaderService,
ThemeProvider,
ViewportElementProvider,
} from '@blocksuite/affine-shared/services';
import {
isTouchPadPinchEvent,
matchModels,
requestConnectedFrame,
requestThrottledConnectedFrame,
} from '@blocksuite/affine-shared/utils';
import { IS_WINDOWS } from '@blocksuite/global/env';
import { Bound, Point, Vec } from '@blocksuite/global/gfx';
import {
BlockComponent,
type GfxBlockComponent,
SurfaceSelection,
type UIEventHandler,
} from '@blocksuite/std';
import {
GfxControllerIdentifier,
type GfxViewportElement,
} from '@blocksuite/std/gfx';
import { effect } from '@preact/signals-core';
import { css, html } from 'lit';
import { query } from 'lit/decorators.js';
import { repeat } from 'lit/directives/repeat.js';
import type { EdgelessSelectedRectWidget } from './components/rects/edgeless-selected-rect.js';
import { EdgelessPageKeyboardManager } from './edgeless-keyboard.js';
import type { EdgelessRootService } from './edgeless-root-service.js';
import { isCanvasElement } from './utils/query.js';
export class EdgelessRootBlockComponent extends BlockComponent<
RootBlockModel,
EdgelessRootService
> {
static override styles = css`
affine-edgeless-root {
-webkit-user-select: none;
user-select: none;
display: block;
height: 100%;
touch-action: none;
}
.widgets-container {
position: absolute;
left: 0;
top: 0;
pointer-events: none;
contain: size layout;
height: 100%;
width: 100%;
}
.widgets-container > * {
pointer-events: auto;
}
.edgeless-background {
height: 100%;
background-color: var(--affine-background-primary-color);
background-image: radial-gradient(
var(--affine-edgeless-grid-color) 1px,
var(--affine-background-primary-color) 1px
);
}
.edgeless-container {
color: var(--affine-text-primary-color);
position: relative;
}
@media print {
.selected {
background-color: transparent !important;
}
}
`;
private readonly _refreshLayerViewport = requestThrottledConnectedFrame(
() => {
const { zoom, translateX, translateY } = this.gfx.viewport;
const gap = getBgGridGap(zoom);
if (this.backgroundElm) {
this.backgroundElm.style.setProperty(
'background-position',
`${translateX}px ${translateY}px`
);
this.backgroundElm.style.setProperty(
'background-size',
`${gap}px ${gap}px`
);
}
},
this
);
private _resizeObserver: ResizeObserver | null = null;
keyboardManager: EdgelessPageKeyboardManager | null = null;
get dispatcher() {
return this.std.event;
}
get fontLoader() {
return this.std.get(FontLoaderService);
}
get gfx() {
return this.std.get(GfxControllerIdentifier);
}
get selectedRectWidget() {
return this.host.view.getWidget(
'edgeless-selected-rect',
this.host.id
) as EdgelessSelectedRectWidget;
}
get slots() {
return this.std.get(EdgelessLegacySlotIdentifier);
}
get surfaceBlockModel() {
return this.model.children.find(
child => child.flavour === 'affine:surface'
) as SurfaceBlockModel;
}
get viewportElement(): HTMLElement {
return this.std.get(ViewportElementProvider).viewportElement;
}
private _initFontLoader() {
this.std
.get(FontLoaderService)
.ready.then(() => {
this.surface.refresh();
})
.catch(console.error);
}
private _initLayerUpdateEffect() {
const updateLayers = requestThrottledConnectedFrame(() => {
const blocks = Array.from(
this.gfxViewportElm.children as HTMLCollectionOf<GfxBlockComponent>
);
blocks.forEach((block: GfxBlockComponent) => {
block.updateZIndex?.();
});
});
this._disposables.add(
this.gfx.layer.slots.layerUpdated.subscribe(() => updateLayers())
);
}
private _initPanEvent() {
this.disposables.add(
this.dispatcher.add('pan', ctx => {
const { viewport } = this.gfx;
if (viewport.locked) return;
const multiPointersState = ctx.get('multiPointerState');
const [p1, p2] = multiPointersState.pointers;
const dx =
(0.25 * (p1.delta.x + p2.delta.x)) /
viewport.zoom /
viewport.viewScale;
const dy =
(0.25 * (p1.delta.y + p2.delta.y)) /
viewport.zoom /
viewport.viewScale;
// direction is opposite
viewport.applyDeltaCenter(-dx, -dy);
})
);
}
private _initPinchEvent() {
this.disposables.add(
this.dispatcher.add('pinch', ctx => {
const { viewport } = this.gfx;
if (viewport.locked) return;
const multiPointersState = ctx.get('multiPointerState');
const [p1, p2] = multiPointersState.pointers;
const currentCenter = new Point(
0.5 * (p1.x + p2.x),
0.5 * (p1.y + p2.y)
);
const lastDistance = Vec.dist(
[p1.x - p1.delta.x, p1.y - p1.delta.y],
[p2.x - p2.delta.x, p2.y - p2.delta.y]
);
const currentDistance = Vec.dist([p1.x, p1.y], [p2.x, p2.y]);
const zoom = (currentDistance / lastDistance) * viewport.zoom;
const [baseX, baseY] = viewport.toModelCoord(
currentCenter.x,
currentCenter.y
);
viewport.setZoom(zoom, new Point(baseX, baseY));
return false;
})
);
}
private _initPixelRatioChangeEffect() {
let media: MediaQueryList;
const onPixelRatioChange = () => {
if (media) {
this.gfx.viewport.onResize();
media.removeEventListener('change', onPixelRatioChange);
}
media = matchMedia(`(resolution: ${window.devicePixelRatio}dppx)`);
media.addEventListener('change', onPixelRatioChange);
};
onPixelRatioChange();
this._disposables.add(() => {
media?.removeEventListener('change', onPixelRatioChange);
});
}
private _initRemoteCursor() {
let rafId: number | null = null;
const setRemoteCursor = (pos: { x: number; y: number }) => {
if (rafId) cancelAnimationFrame(rafId);
rafId = requestConnectedFrame(() => {
if (!this.gfx.viewport) return;
const cursorPosition = this.gfx.viewport.toModelCoord(pos.x, pos.y);
this.gfx.selection.setCursor({
x: cursorPosition[0],
y: cursorPosition[1],
});
rafId = null;
}, this);
};
this.handleEvent('pointerMove', e => {
const pointerEvent = e.get('pointerState');
setRemoteCursor(pointerEvent);
});
}
private _initResizeEffect() {
const resizeObserver = new ResizeObserver((_: ResizeObserverEntry[]) => {
this.gfx.selection.set(this.gfx.selection.surfaceSelections);
this.gfx.viewport.onResize();
});
resizeObserver.observe(this.viewportElement);
this._resizeObserver = resizeObserver;
}
private _initSlotEffects() {
const { disposables } = this;
this.disposables.add(
this.std.get(ThemeProvider).theme$.subscribe(() => this.surface.refresh())
);
disposables.add(
effect(() => {
this.style.cursor = this.gfx.cursor$.value;
})
);
}
private _initViewport() {
const { std, gfx } = this;
const run = () => {
const animationFn = std.getOptional(
NoteConfigExtension.identifier
)?.pageBlockViewportFitAnimation;
if (animationFn) {
const pageBlock = this.model.children.find(
(child): child is NoteBlockModel =>
matchModels(child, [NoteBlockModel]) && child.isPageBlock()
);
if (pageBlock && animationFn({ std: this.std, note: pageBlock })) {
return;
}
}
const storedViewport = std.get(EditPropsStore).getStorage('viewport');
if (storedViewport) {
if ('xywh' in storedViewport) {
const bound = Bound.deserialize(storedViewport.xywh);
gfx.viewport.setViewportByBound(bound, storedViewport.padding);
} else {
const { zoom, centerX, centerY } = storedViewport;
gfx.viewport.setViewport(zoom, [centerX, centerY]);
}
return;
}
this.gfx.fitToScreen();
};
run();
this._disposables.add(() => {
std.get(EditPropsStore).setStorage('viewport', {
centerX: gfx.viewport.centerX,
centerY: gfx.viewport.centerY,
zoom: gfx.viewport.zoom,
});
});
}
private _initWheelEvent() {
this._disposables.add(
this.dispatcher.add('wheel', ctx => {
const config = this.std.getOptional(EditorSettingProvider)?.setting$;
const state = ctx.get('defaultState');
const e = state.event as WheelEvent;
const edgelessScrollZoom = config?.peek().edgelessScrollZoom ?? false;
e.preventDefault();
const { viewport } = this.gfx;
if (viewport.locked) return;
// zoom
if (isTouchPadPinchEvent(e) || edgelessScrollZoom) {
const rect = this.getBoundingClientRect();
// Perform zooming relative to the mouse position
const [baseX, baseY] = this.gfx.viewport.toModelCoord(
e.clientX - rect.x,
e.clientY - rect.y
);
const zoom = normalizeWheelDeltaY(e.deltaY, viewport.zoom);
viewport.setZoom(zoom, new Point(baseX, baseY), true);
e.stopPropagation();
}
// pan
else {
const simulateHorizontalScroll = IS_WINDOWS && e.shiftKey;
const dx = simulateHorizontalScroll
? e.deltaY / viewport.zoom
: e.deltaX / viewport.zoom;
const dy = simulateHorizontalScroll ? 0 : e.deltaY / viewport.zoom;
viewport.applyDeltaCenter(dx, dy);
viewport.viewportMoved.next([dx, dy]);
e.stopPropagation();
}
})
);
}
override bindHotKey(
keymap: Record<string, UIEventHandler>,
options?: { global?: boolean; flavour?: boolean }
): () => void {
const { gfx } = this;
const selection = gfx.selection;
Object.keys(keymap).forEach(key => {
if (key.length === 1 && key >= 'A' && key <= 'z') {
const handler = keymap[key];
keymap[key] = ctx => {
const elements = selection.selectedElements;
if (isSingleMindMapNode(elements) && !selection.editing) {
const target = gfx.getElementById(
elements[0].id
) as ShapeElementModel;
if (target.text) {
this.doc.transact(() => {
target.text!.delete(0, target.text!.length);
target.text!.insert(0, key);
});
}
mountShapeTextEditor(target, this);
} else {
handler(ctx);
}
};
}
});
return super.bindHotKey(keymap, options);
}
override connectedCallback() {
super.connectedCallback();
this._initViewport();
this.keyboardManager = new EdgelessPageKeyboardManager(this);
this.handleEvent('selectionChange', () => {
const surface = this.host.selection.value.find(
(sel): sel is SurfaceSelection => sel.is(SurfaceSelection)
);
if (!surface) return;
const el = this.gfx.getElementById(surface.elements[0]);
if (isCanvasElement(el)) {
return true;
}
return;
});
}
override disconnectedCallback() {
super.disconnectedCallback();
if (this._resizeObserver) {
this._resizeObserver.disconnect();
this._resizeObserver = null;
}
this.keyboardManager = null;
}
override firstUpdated() {
this._initSlotEffects();
this._initResizeEffect();
this._initPixelRatioChangeEffect();
this._initFontLoader();
this._initRemoteCursor();
this._initLayerUpdateEffect();
this._initWheelEvent();
this._initPanEvent();
this._initPinchEvent();
if (this.doc.readonly) {
this.gfx.tool.setTool('pan', { panning: true });
} else {
this.gfx.tool.setTool('default');
}
this.gfx.viewport.elementReady.next(this.gfxViewportElm);
requestConnectedFrame(() => {
this.requestUpdate();
}, this);
this._disposables.add(
this.gfx.viewport.viewportUpdated.subscribe(() => {
this._refreshLayerViewport();
})
);
this._refreshLayerViewport();
}
override renderBlock() {
const widgets = repeat(
Object.entries(this.widgets),
([id]) => id,
([_, widget]) => widget
);
return html`
<div class="edgeless-background edgeless-container">
<gfx-viewport
.maxConcurrentRenders=${6}
.viewport=${this.gfx.viewport}
.getModelsInViewport=${() => {
const blocks = this.gfx.grid.search(
this.gfx.viewport.viewportBounds,
{
useSet: true,
filter: ['block'],
}
);
return blocks;
}}
.host=${this.host}
>
${this.renderChildren(this.model)}
${this.renderChildren(this.surfaceBlockModel)}
</gfx-viewport>
</div>
<!--
Used to mount component before widgets
Eg., canvas text editor
-->
<div class="edgeless-mount-point"></div>
<div class="widgets-container">${widgets}</div>
`;
}
@query('.edgeless-background')
accessor backgroundElm: HTMLDivElement | null = null;
@query('gfx-viewport')
accessor gfxViewportElm!: GfxViewportElement;
@query('.edgeless-mount-point')
accessor mountElm: HTMLDivElement | null = null;
@query('affine-surface')
accessor surface!: SurfaceBlockComponent;
}