refactor(editor): extract drag handle widget (#9415)

This commit is contained in:
Saul-Mirone
2024-12-29 06:51:48 +00:00
parent b96a03b283
commit cec4a4b2c0
47 changed files with 277 additions and 92 deletions

View File

@@ -6,6 +6,8 @@ export const ZOOM_WHEEL_STEP = 0.1;
export const GRID_SIZE = 3000; export const GRID_SIZE = 3000;
export const GRID_GAP_MIN = 10; export const GRID_GAP_MIN = 10;
export const GRID_GAP_MAX = 50; export const GRID_GAP_MAX = 50;
export const DEFAULT_NOTE_OFFSET_X = 30;
export const DEFAULT_NOTE_OFFSET_Y = 40;
// TODO: need to check the default central area ratio // TODO: need to check the default central area ratio
export const DEFAULT_CENTRAL_AREA_RATIO = 0.3; export const DEFAULT_CENTRAL_AREA_RATIO = 0.3;

View File

@@ -104,7 +104,12 @@ import {
tryMoveNode, tryMoveNode,
} from './utils/mindmap/utils'; } from './utils/mindmap/utils';
export * from './extensions'; export * from './extensions';
export { getLastPropsKey, getSurfaceBlock } from './utils'; export {
addNote,
addNoteAtPoint,
getLastPropsKey,
getSurfaceBlock,
} from './utils';
export type { Options } from './utils/rough/core'; export type { Options } from './utils/rough/core';
export { sortIndex } from './utils/sort'; export { sortIndex } from './utils/sort';
export { updateXYWH } from './utils/update-xywh.js'; export { updateXYWH } from './utils/update-xywh.js';

View File

@@ -1,4 +1,3 @@
import { EdgelessCRUDIdentifier } from '@blocksuite/affine-block-surface';
import { focusTextModel } from '@blocksuite/affine-components/rich-text'; import { focusTextModel } from '@blocksuite/affine-components/rich-text';
import { import {
DEFAULT_NOTE_HEIGHT, DEFAULT_NOTE_HEIGHT,
@@ -18,7 +17,8 @@ import {
serializeXYWH, serializeXYWH,
} from '@blocksuite/global/utils'; } from '@blocksuite/global/utils';
import { DEFAULT_NOTE_OFFSET_X, DEFAULT_NOTE_OFFSET_Y } from './consts.js'; import { DEFAULT_NOTE_OFFSET_X, DEFAULT_NOTE_OFFSET_Y } from '../consts';
import { EdgelessCRUDIdentifier } from '../extensions/crud-extension';
export function addNoteAtPoint( export function addNoteAtPoint(
std: BlockStdScope, std: BlockStdScope,
@@ -96,7 +96,7 @@ export function addNote(
const doc = std.doc; const doc = std.doc;
const blockId = doc.addBlock( const blockId = doc.addBlock(
options.childFlavour, options.childFlavour as BlockSuite.Flavour,
{ type: options.childType }, { type: options.childType },
noteId noteId
); );
@@ -107,7 +107,10 @@ export function addNote(
note.edgeless.collapsedHeight = height; note.edgeless.collapsedHeight = height;
}); });
} }
gfx.tool.setTool('default'); gfx.tool.setTool(
// @ts-expect-error FIXME: resolve after gfx tool refactor
'default'
);
// Wait for edgelessTool updated // Wait for edgelessTool updated
requestAnimationFrame(() => { requestAnimationFrame(() => {

View File

@@ -33,5 +33,6 @@ export function normalizeWheelDeltaY(delta: number, zoom = 1) {
return newZoom; return newZoom;
} }
export { addNote, addNoteAtPoint } from './add-note';
export { getLastPropsKey } from './get-last-props-key'; export { getLastPropsKey } from './get-last-props-key';
export { getSurfaceBlock } from './get-surface-block'; export { getSurfaceBlock } from './get-surface-block';

View File

@@ -1,13 +1,14 @@
export * from './doc-display-meta-service.js'; export * from './doc-display-meta-service';
export * from './doc-mode-service.js'; export * from './doc-mode-service';
export * from './drag-handle-config.js'; export * from './drag-handle-config';
export * from './edit-props-store.js'; export * from './edit-props-store';
export * from './editor-setting-service.js'; export * from './editor-setting-service';
export * from './embed-option-service.js'; export * from './embed-option-service';
export * from './font-loader/index.js'; export * from './font-loader';
export * from './generate-url-service.js'; export * from './generate-url-service';
export * from './notification-service.js'; export * from './notification-service';
export * from './parse-url-service.js'; export * from './page-viewport-service';
export * from './quick-search-service.js'; export * from './parse-url-service';
export * from './telemetry-service/index.js'; export * from './quick-search-service';
export * from './theme-service.js'; export * from './telemetry-service';
export * from './theme-service';

View File

@@ -0,0 +1,15 @@
import type { ExtensionType } from '@blocksuite/block-std';
import { createIdentifier } from '@blocksuite/global/di';
import { Slot } from '@blocksuite/store';
import type { Viewport } from '../types';
export const PageViewportService = createIdentifier<Slot<Viewport>>(
'PageViewportService'
);
export const PageViewportServiceExtension: ExtensionType = {
setup: di => {
di.addImpl(PageViewportService, () => new Slot<Viewport>());
},
};

View File

@@ -1,20 +1,21 @@
export * from './button-popper.js'; export * from './auto-scroll';
export * from './collapsed/index.js'; export * from './button-popper';
export * from './dnd/index.js'; export * from './collapsed';
export * from './dom/index.js'; export * from './dnd';
export * from './edgeless.js'; export * from './dom';
export * from './event.js'; export * from './edgeless';
export * from './file/index.js'; export * from './event';
export * from './insert.js'; export * from './file';
export * from './is-abort-error.js'; export * from './insert';
export * from './math.js'; export * from './is-abort-error';
export * from './model/index.js'; export * from './math';
export * from './print-to-pdf.js'; export * from './model';
export * from './reference.js'; export * from './print-to-pdf';
export * from './reordering.js'; export * from './reference';
export * from './signal.js'; export * from './reordering';
export * from './spec/index.js'; export * from './signal';
export * from './string.js'; export * from './spec';
export * from './title.js'; export * from './string';
export * from './url.js'; export * from './title';
export * from './zod-schema.js'; export * from './url';
export * from './zod-schema';

View File

@@ -0,0 +1,47 @@
{
"name": "@blocksuite/affine-widget-drag-handle",
"description": "Drag handle for BlockSuite.",
"type": "module",
"scripts": {
"build": "tsc",
"test:unit": "nx vite:test --run --passWithNoTests",
"test:unit:coverage": "nx vite:test --run --coverage",
"test:e2e": "playwright test"
},
"sideEffects": false,
"keywords": [],
"author": "toeverything",
"license": "MIT",
"dependencies": {
"@blocksuite/affine-block-list": "workspace:*",
"@blocksuite/affine-block-note": "workspace:*",
"@blocksuite/affine-block-paragraph": "workspace:*",
"@blocksuite/affine-block-surface": "workspace:*",
"@blocksuite/affine-components": "workspace:*",
"@blocksuite/affine-model": "workspace:*",
"@blocksuite/affine-shared": "workspace:*",
"@blocksuite/block-std": "workspace:*",
"@blocksuite/global": "workspace:*",
"@blocksuite/icons": "^2.1.75",
"@blocksuite/inline": "workspace:*",
"@blocksuite/store": "workspace:*",
"@floating-ui/dom": "^1.6.10",
"@lit/context": "^1.1.2",
"@preact/signals-core": "^1.8.0",
"@toeverything/theme": "^1.1.1",
"lit": "^3.2.0",
"minimatch": "^10.0.1",
"zod": "^3.23.8"
},
"exports": {
".": "./src/index.ts",
"./effects": "./src/effects.ts"
},
"files": [
"src",
"dist",
"!src/__tests__",
"!dist/__tests__"
],
"version": "0.19.0"
}

View File

@@ -2,6 +2,7 @@ import { EdgelessCRUDIdentifier } from '@blocksuite/affine-block-surface';
import type { RootBlockModel } from '@blocksuite/affine-model'; import type { RootBlockModel } from '@blocksuite/affine-model';
import { DocModeProvider } from '@blocksuite/affine-shared/services'; import { DocModeProvider } from '@blocksuite/affine-shared/services';
import { import {
autoScroll,
calcDropTarget, calcDropTarget,
type DroppingType, type DroppingType,
type DropResult, type DropResult,
@@ -24,7 +25,6 @@ import { html } from 'lit';
import { query, state } from 'lit/decorators.js'; import { query, state } from 'lit/decorators.js';
import { styleMap } from 'lit/directives/style-map.js'; import { styleMap } from 'lit/directives/style-map.js';
import { autoScroll } from '../../../root-block/text-selection/utils.js';
import type { DragPreview } from './components/drag-preview.js'; import type { DragPreview } from './components/drag-preview.js';
import type { DropIndicator } from './components/drop-indicator.js'; import type { DropIndicator } from './components/drop-indicator.js';
import type { AFFINE_DRAG_HANDLE_WIDGET } from './consts.js'; import type { AFFINE_DRAG_HANDLE_WIDGET } from './consts.js';
@@ -75,7 +75,9 @@ export class AffineDragHandleWidget extends WidgetComponent<RootBlockModel> {
const blockId = closestBlock.model.id; const blockId = closestBlock.model.id;
const model = closestBlock.model; const model = closestBlock.model;
const isDatabase = matchFlavours(model, ['affine:database']); const isDatabase = matchFlavours(model, [
'affine:database' as BlockSuite.Flavour,
]);
if (isDatabase) return null; if (isDatabase) return null;
// note block can only be dropped into another note block // note block can only be dropped into another note block

View File

@@ -0,0 +1,10 @@
import { DragPreview } from './components/drag-preview';
import { DropIndicator } from './components/drop-indicator';
import { AFFINE_DRAG_HANDLE_WIDGET } from './consts';
import { AffineDragHandleWidget } from './drag-handle';
export function effects() {
customElements.define('affine-drag-preview', DragPreview);
customElements.define('affine-drop-indicator', DropIndicator);
customElements.define(AFFINE_DRAG_HANDLE_WIDGET, AffineDragHandleWidget);
}

View File

@@ -0,0 +1,7 @@
import type * as SurfaceEffects from '@blocksuite/affine-block-surface/effects';
declare type _GLOBAL_ = typeof SurfaceEffects;
export * from './consts';
export * from './drag-handle';
export * from './utils';

View File

@@ -188,7 +188,9 @@ export const getDropResult = (
const model = closestBlock.model; const model = closestBlock.model;
const isDatabase = matchFlavours(model, ['affine:database']); const isDatabase = matchFlavours(model, [
'affine:database' as BlockSuite.Flavour,
]);
if (isDatabase) { if (isDatabase) {
return dropIndicator; return dropIndicator;
} }

View File

@@ -1,4 +1,8 @@
import { ParagraphBlockComponent } from '@blocksuite/affine-block-paragraph'; import { ParagraphBlockComponent } from '@blocksuite/affine-block-paragraph';
import {
addNoteAtPoint,
getSurfaceBlock,
} from '@blocksuite/affine-block-surface';
import type { EmbedCardStyle, NoteBlockModel } from '@blocksuite/affine-model'; import type { EmbedCardStyle, NoteBlockModel } from '@blocksuite/affine-model';
import { import {
EMBED_CARD_HEIGHT, EMBED_CARD_HEIGHT,
@@ -28,8 +32,6 @@ import { GfxControllerIdentifier } from '@blocksuite/block-std/gfx';
import { Bound, Point } from '@blocksuite/global/utils'; import { Bound, Point } from '@blocksuite/global/utils';
import { Job, Slice, type SliceSnapshot } from '@blocksuite/store'; import { Job, Slice, type SliceSnapshot } from '@blocksuite/store';
import type { EdgelessRootBlockComponent } from '../../../edgeless/index.js';
import { addNoteAtPoint } from '../../../edgeless/utils/common.js';
import { DropIndicator } from '../components/drop-indicator.js'; import { DropIndicator } from '../components/drop-indicator.js';
import { AFFINE_DRAG_HANDLE_WIDGET } from '../consts.js'; import { AFFINE_DRAG_HANDLE_WIDGET } from '../consts.js';
import type { AffineDragHandleWidget } from '../drag-handle.js'; import type { AffineDragHandleWidget } from '../drag-handle.js';
@@ -39,16 +41,19 @@ import { surfaceRefToEmbed } from '../middleware/surface-ref-to-embed.js';
import { containBlock, includeTextSelection } from '../utils.js'; import { containBlock, includeTextSelection } from '../utils.js';
export class DragEventWatcher { export class DragEventWatcher {
private get _gfx() {
return this.widget.std.get(GfxControllerIdentifier);
}
private readonly _computeEdgelessBound = ( private readonly _computeEdgelessBound = (
x: number, x: number,
y: number, y: number,
width: number, width: number,
height: number height: number
) => { ) => {
const controller = this._std.get(GfxControllerIdentifier);
const border = 2; const border = 2;
const noteScale = this.widget.noteScale.peek(); const noteScale = this.widget.noteScale.peek();
const { viewport } = controller; const { viewport } = this._gfx;
const { left: viewportLeft, top: viewportTop } = viewport; const { left: viewportLeft, top: viewportTop } = viewport;
const currentViewBound = new Bound( const currentViewBound = new Bound(
x - viewportLeft, x - viewportLeft,
@@ -335,10 +340,9 @@ export class DragEventWatcher {
const state = context.get('dndState'); const state = context.get('dndState');
// If drop a note, should do nothing // If drop a note, should do nothing
const snapshot = this._deserializeSnapshot(state); const snapshot = this._deserializeSnapshot(state);
const edgelessRoot = this.widget const surfaceBlockModel = getSurfaceBlock(this.widget.doc);
.rootComponent as EdgelessRootBlockComponent;
if (!snapshot) { if (!snapshot || !surfaceBlockModel) {
return; return;
} }
@@ -358,7 +362,7 @@ export class DragEventWatcher {
const std = this._std; const std = this._std;
const job = this._getJob(); const job = this._getJob();
job job
.snapshotToSlice(snapshot, std.doc, edgelessRoot.surfaceBlockModel.id) .snapshotToSlice(snapshot, std.doc, surfaceBlockModel.id)
.catch(console.error); .catch(console.error);
}; };
@@ -404,9 +408,9 @@ export class DragEventWatcher {
} }
} }
const { left: viewportLeft, top: viewportTop } = edgelessRoot.viewport; const { left: viewportLeft, top: viewportTop } = this._gfx.viewport;
const newNoteId = addNoteAtPoint( const newNoteId = addNoteAtPoint(
edgelessRoot.std, this._std,
new Point(state.raw.x - viewportLeft, state.raw.y - viewportTop), new Point(state.raw.x - viewportLeft, state.raw.y - viewportTop),
{ {
scale: this.widget.noteScale.peek(), scale: this.widget.noteScale.peek(),

View File

@@ -27,6 +27,7 @@ export class EdgelessWatcher {
private readonly _handleEdgelessToolUpdated = ( private readonly _handleEdgelessToolUpdated = (
newTool: GfxToolsFullOptionValue newTool: GfxToolsFullOptionValue
) => { ) => {
// @ts-expect-error FIXME: resolve after gfx tool refactor
if (newTool.type === 'default') { if (newTool.type === 'default') {
this.checkTopLevelBlockSelection(); this.checkTopLevelBlockSelection();
} else { } else {

View File

@@ -1,26 +1,25 @@
import { PageViewportService } from '@blocksuite/affine-shared/services';
import { getScrollContainer } from '@blocksuite/affine-shared/utils'; import { getScrollContainer } from '@blocksuite/affine-shared/utils';
import type { PageRootBlockComponent } from '../../../page/page-root-block.js';
import type { AffineDragHandleWidget } from '../drag-handle.js'; import type { AffineDragHandleWidget } from '../drag-handle.js';
export class PageWatcher { export class PageWatcher {
get pageRoot() { get pageViewportService() {
return this.widget.rootComponent as PageRootBlockComponent; return this.widget.std.get(PageViewportService);
} }
constructor(readonly widget: AffineDragHandleWidget) {} constructor(readonly widget: AffineDragHandleWidget) {}
watch() { watch() {
const { pageRoot } = this;
const { disposables } = this.widget; const { disposables } = this.widget;
const scrollContainer = getScrollContainer(pageRoot); const scrollContainer = getScrollContainer(this.widget.rootComponent);
disposables.add( disposables.add(
this.widget.doc.slots.blockUpdated.on(() => this.widget.hide()) this.widget.doc.slots.blockUpdated.on(() => this.widget.hide())
); );
disposables.add( disposables.add(
pageRoot.slots.viewportUpdated.on(() => { this.pageViewportService.on(() => {
this.widget.hide(); this.widget.hide();
if (this.widget.dropIndicator) { if (this.widget.dropIndicator) {
this.widget.dropIndicator.rect = null; this.widget.dropIndicator.rect = null;

View File

@@ -6,10 +6,10 @@ import {
type PointerEventState, type PointerEventState,
type UIEventHandler, type UIEventHandler,
} from '@blocksuite/block-std'; } from '@blocksuite/block-std';
import { GfxControllerIdentifier } from '@blocksuite/block-std/gfx';
import { Point, throttle } from '@blocksuite/global/utils'; import { Point, throttle } from '@blocksuite/global/utils';
import { computed } from '@preact/signals-core'; import { computed } from '@preact/signals-core';
import type { EdgelessRootBlockComponent } from '../../../edgeless/index.js';
import { import {
DRAG_HANDLE_CONTAINER_WIDTH, DRAG_HANDLE_CONTAINER_WIDTH,
DRAG_HANDLE_GRABBER_BORDER_RADIUS, DRAG_HANDLE_GRABBER_BORDER_RADIUS,
@@ -30,19 +30,19 @@ import {
} from '../utils.js'; } from '../utils.js';
export class PointerEventWatcher { export class PointerEventWatcher {
private get _gfx() {
return this.widget.std.get(GfxControllerIdentifier);
}
private readonly _canEditing = (noteBlock: BlockComponent) => { private readonly _canEditing = (noteBlock: BlockComponent) => {
if (noteBlock.doc.id !== this.widget.doc.id) return false; if (noteBlock.doc.id !== this.widget.doc.id) return false;
if (this.widget.mode === 'page') return true; if (this.widget.mode === 'page') return true;
const edgelessRoot = this.widget const selection = this._gfx.selection;
.rootComponent as EdgelessRootBlockComponent;
const noteBlockId = noteBlock.model.id; const noteBlockId = noteBlock.model.id;
return ( return selection.editing && selection.selectedIds[0] === noteBlockId;
edgelessRoot.service.selection.editing &&
edgelessRoot.service.selection.selectedIds[0] === noteBlockId
);
}; };
/** /**

View File

@@ -0,0 +1,44 @@
{
"extends": "../../tsconfig.json",
"compilerOptions": {
"rootDir": "./src/",
"outDir": "./dist/",
"noEmit": false
},
"include": ["./src"],
"references": [
{
"path": "../../framework/global"
},
{
"path": "../../framework/store"
},
{
"path": "../../framework/block-std"
},
{
"path": "../../framework/inline"
},
{
"path": "../model"
},
{
"path": "../components"
},
{
"path": "../shared"
},
{
"path": "../block-paragraph"
},
{
"path": "../block-surface"
},
{
"path": "../block-note"
},
{
"path": "../block-list"
}
]
}

View File

@@ -30,6 +30,7 @@
"@blocksuite/affine-components": "workspace:*", "@blocksuite/affine-components": "workspace:*",
"@blocksuite/affine-model": "workspace:*", "@blocksuite/affine-model": "workspace:*",
"@blocksuite/affine-shared": "workspace:*", "@blocksuite/affine-shared": "workspace:*",
"@blocksuite/affine-widget-drag-handle": "workspace:*",
"@blocksuite/affine-widget-remote-selection": "workspace:*", "@blocksuite/affine-widget-remote-selection": "workspace:*",
"@blocksuite/affine-widget-scroll-anchoring": "workspace:*", "@blocksuite/affine-widget-scroll-anchoring": "workspace:*",
"@blocksuite/block-std": "workspace:*", "@blocksuite/block-std": "workspace:*",

View File

@@ -16,6 +16,7 @@ import {
type TelemetryEventMap, type TelemetryEventMap,
TelemetryProvider, TelemetryProvider,
} from '@blocksuite/affine-shared/services'; } from '@blocksuite/affine-shared/services';
import { getDropResult } from '@blocksuite/affine-widget-drag-handle';
import { RANGE_SYNC_EXCLUDE_ATTR } from '@blocksuite/block-std'; import { RANGE_SYNC_EXCLUDE_ATTR } from '@blocksuite/block-std';
import { import {
createRecordDetail, createRecordDetail,
@@ -46,7 +47,6 @@ import { computed, signal } from '@preact/signals-core';
import { css, html, nothing, unsafeCSS } from 'lit'; import { css, html, nothing, unsafeCSS } from 'lit';
import { EdgelessRootBlockComponent } from '../root-block/index.js'; import { EdgelessRootBlockComponent } from '../root-block/index.js';
import { getDropResult } from '../root-block/widgets/drag-handle/utils.js';
import { popSideDetail } from './components/layout.js'; import { popSideDetail } from './components/layout.js';
import type { DatabaseOptionsConfig } from './config.js'; import type { DatabaseOptionsConfig } from './config.js';
import { HostContextKey } from './context/host-context.js'; import { HostContextKey } from './context/host-context.js';

View File

@@ -26,6 +26,7 @@ import { SmoothCorner } from '@blocksuite/affine-components/smooth-corner';
import { effects as componentToggleButtonEffects } from '@blocksuite/affine-components/toggle-button'; import { effects as componentToggleButtonEffects } from '@blocksuite/affine-components/toggle-button';
import { ToggleSwitch } from '@blocksuite/affine-components/toggle-switch'; import { ToggleSwitch } from '@blocksuite/affine-components/toggle-switch';
import { effects as componentToolbarEffects } from '@blocksuite/affine-components/toolbar'; import { effects as componentToolbarEffects } from '@blocksuite/affine-components/toolbar';
import { effects as widgetDragHandleEffects } from '@blocksuite/affine-widget-drag-handle/effects';
import { effects as widgetRemoteSelectionEffects } from '@blocksuite/affine-widget-remote-selection/effects'; import { effects as widgetRemoteSelectionEffects } from '@blocksuite/affine-widget-remote-selection/effects';
import { effects as widgetScrollAnchoringEffects } from '@blocksuite/affine-widget-scroll-anchoring/effects'; import { effects as widgetScrollAnchoringEffects } from '@blocksuite/affine-widget-scroll-anchoring/effects';
import type { BlockComponent } from '@blocksuite/block-std'; import type { BlockComponent } from '@blocksuite/block-std';
@@ -148,7 +149,6 @@ import {
AFFINE_EMBED_CARD_TOOLBAR_WIDGET, AFFINE_EMBED_CARD_TOOLBAR_WIDGET,
AFFINE_FORMAT_BAR_WIDGET, AFFINE_FORMAT_BAR_WIDGET,
AffineAIPanelWidget, AffineAIPanelWidget,
AffineDragHandleWidget,
AffineEdgelessZoomToolbarWidget, AffineEdgelessZoomToolbarWidget,
AffineFormatBarWidget, AffineFormatBarWidget,
AffineImageToolbarWidget, AffineImageToolbarWidget,
@@ -176,9 +176,6 @@ import {
AIPanelGenerating, AIPanelGenerating,
AIPanelInput, AIPanelInput,
} from './root-block/widgets/ai-panel/components/index.js'; } from './root-block/widgets/ai-panel/components/index.js';
import { DragPreview } from './root-block/widgets/drag-handle/components/drag-preview.js';
import { DropIndicator } from './root-block/widgets/drag-handle/components/drop-indicator.js';
import { AFFINE_DRAG_HANDLE_WIDGET } from './root-block/widgets/drag-handle/consts.js';
import { import {
AFFINE_EDGELESS_AUTO_CONNECT_WIDGET, AFFINE_EDGELESS_AUTO_CONNECT_WIDGET,
EdgelessAutoConnectWidget, EdgelessAutoConnectWidget,
@@ -259,6 +256,7 @@ export function effects() {
widgetFrameTitleEffects(); widgetFrameTitleEffects();
widgetEdgelessElementToolbarEffects(); widgetEdgelessElementToolbarEffects();
widgetRemoteSelectionEffects(); widgetRemoteSelectionEffects();
widgetDragHandleEffects();
dataViewEffects(); dataViewEffects();
customElements.define('affine-database-title', DatabaseTitle); customElements.define('affine-database-title', DatabaseTitle);
@@ -433,7 +431,6 @@ export function effects() {
'edgeless-group-title-editor', 'edgeless-group-title-editor',
EdgelessGroupTitleEditor EdgelessGroupTitleEditor
); );
customElements.define('affine-drag-preview', DragPreview);
customElements.define(EDGELESS_TOOLBAR_WIDGET, EdgelessToolbarWidget); customElements.define(EDGELESS_TOOLBAR_WIDGET, EdgelessToolbarWidget);
customElements.define('edgeless-shape-style-panel', EdgelessShapeStylePanel); customElements.define('edgeless-shape-style-panel', EdgelessShapeStylePanel);
customElements.define( customElements.define(
@@ -446,7 +443,6 @@ export function effects() {
); );
customElements.define('edgeless-text-editor', EdgelessTextEditor); customElements.define('edgeless-text-editor', EdgelessTextEditor);
customElements.define('affine-image-toolbar', AffineImageToolbar); customElements.define('affine-image-toolbar', AffineImageToolbar);
customElements.define('affine-drop-indicator', DropIndicator);
customElements.define('mini-mindmap-root-block', MindmapRootBlock); customElements.define('mini-mindmap-root-block', MindmapRootBlock);
customElements.define('affine-block-selection', BlockSelection); customElements.define('affine-block-selection', BlockSelection);
customElements.define('edgeless-slide-menu', EdgelessSlideMenu); customElements.define('edgeless-slide-menu', EdgelessSlideMenu);
@@ -463,7 +459,6 @@ export function effects() {
AFFINE_PAGE_DRAGGING_AREA_WIDGET, AFFINE_PAGE_DRAGGING_AREA_WIDGET,
AffinePageDraggingAreaWidget AffinePageDraggingAreaWidget
); );
customElements.define(AFFINE_DRAG_HANDLE_WIDGET, AffineDragHandleWidget);
customElements.define(AFFINE_EDGELESS_COPILOT_WIDGET, EdgelessCopilotWidget); customElements.define(AFFINE_EDGELESS_COPILOT_WIDGET, EdgelessCopilotWidget);
customElements.define(AFFINE_IMAGE_TOOLBAR_WIDGET, AffineImageToolbarWidget); customElements.define(AFFINE_IMAGE_TOOLBAR_WIDGET, AffineImageToolbarWidget);

View File

@@ -31,7 +31,6 @@ export {
export { CopilotTool } from './root-block/edgeless/gfx-tool/copilot-tool.js'; export { CopilotTool } from './root-block/edgeless/gfx-tool/copilot-tool.js';
export * from './root-block/edgeless/gfx-tool/index.js'; export * from './root-block/edgeless/gfx-tool/index.js';
export { EditPropsMiddlewareBuilder } from './root-block/edgeless/middlewares/base.js'; export { EditPropsMiddlewareBuilder } from './root-block/edgeless/middlewares/base.js';
export * from './root-block/edgeless/utils/common.js';
export { EdgelessSnapManager } from './root-block/edgeless/utils/snap-manager.js'; export { EdgelessSnapManager } from './root-block/edgeless/utils/snap-manager.js';
export * from './root-block/index.js'; export * from './root-block/index.js';
export * from './schemas.js'; export * from './schemas.js';

View File

@@ -4,8 +4,10 @@ import {
DocDisplayMetaService, DocDisplayMetaService,
DocModeService, DocModeService,
EmbedOptionService, EmbedOptionService,
PageViewportServiceExtension,
ThemeService, ThemeService,
} from '@blocksuite/affine-shared/services'; } from '@blocksuite/affine-shared/services';
import { AFFINE_DRAG_HANDLE_WIDGET } from '@blocksuite/affine-widget-drag-handle';
import { import {
AFFINE_DOC_REMOTE_SELECTION_WIDGET, AFFINE_DOC_REMOTE_SELECTION_WIDGET,
AFFINE_EDGELESS_REMOTE_SELECTION_WIDGET, AFFINE_EDGELESS_REMOTE_SELECTION_WIDGET,
@@ -25,7 +27,6 @@ import { literal, unsafeStatic } from 'lit/static-html.js';
import { ExportManagerExtension } from '../../_common/export-manager/export-manager.js'; import { ExportManagerExtension } from '../../_common/export-manager/export-manager.js';
import { RootBlockAdapterExtensions } from '../adapters/extension.js'; import { RootBlockAdapterExtensions } from '../adapters/extension.js';
import { commands } from '../commands/index.js'; import { commands } from '../commands/index.js';
import { AFFINE_DRAG_HANDLE_WIDGET } from '../widgets/drag-handle/consts.js';
import { AFFINE_EDGELESS_AUTO_CONNECT_WIDGET } from '../widgets/edgeless-auto-connect/edgeless-auto-connect.js'; import { AFFINE_EDGELESS_AUTO_CONNECT_WIDGET } from '../widgets/edgeless-auto-connect/edgeless-auto-connect.js';
import { AFFINE_EDGELESS_ZOOM_TOOLBAR_WIDGET } from '../widgets/edgeless-zoom-toolbar/index.js'; import { AFFINE_EDGELESS_ZOOM_TOOLBAR_WIDGET } from '../widgets/edgeless-zoom-toolbar/index.js';
import { EDGELESS_ELEMENT_TOOLBAR_WIDGET } from '../widgets/element-toolbar/index.js'; import { EDGELESS_ELEMENT_TOOLBAR_WIDGET } from '../widgets/element-toolbar/index.js';
@@ -97,6 +98,7 @@ const EdgelessCommonExtension: ExtensionType[] = [
ExportManagerExtension, ExportManagerExtension,
ToolController, ToolController,
DNDAPIExtension, DNDAPIExtension,
PageViewportServiceExtension,
DocDisplayMetaService, DocDisplayMetaService,
RootBlockAdapterExtensions, RootBlockAdapterExtensions,
FileDropExtension, FileDropExtension,

View File

@@ -1,4 +1,5 @@
import type { SurfaceBlockComponent } from '@blocksuite/affine-block-surface'; import type { SurfaceBlockComponent } from '@blocksuite/affine-block-surface';
import { addNote } from '@blocksuite/affine-block-surface';
import { import {
DEFAULT_NOTE_HEIGHT, DEFAULT_NOTE_HEIGHT,
DEFAULT_NOTE_WIDTH, DEFAULT_NOTE_WIDTH,
@@ -11,7 +12,6 @@ import { Point } from '@blocksuite/global/utils';
import { effect } from '@preact/signals-core'; import { effect } from '@preact/signals-core';
import { hasClassNameInList } from '../../../_common/utils/index.js'; import { hasClassNameInList } from '../../../_common/utils/index.js';
import { addNote } from '../utils/common.js';
import { EXCLUDING_MOUSE_OUT_CLASS_LIST } from '../utils/consts.js'; import { EXCLUDING_MOUSE_OUT_CLASS_LIST } from '../utils/consts.js';
import { DraggingNoteOverlay, NoteOverlay } from '../utils/tool-overlay.js'; import { DraggingNoteOverlay, NoteOverlay } from '../utils/tool-overlay.js';

View File

@@ -6,8 +6,6 @@ import {
StrokeStyle, StrokeStyle,
} from '@blocksuite/affine-model'; } from '@blocksuite/affine-model';
export const DEFAULT_NOTE_OFFSET_X = 30;
export const DEFAULT_NOTE_OFFSET_Y = 40;
export const NOTE_OVERLAY_OFFSET_X = 6; export const NOTE_OVERLAY_OFFSET_X = 6;
export const NOTE_OVERLAY_OFFSET_Y = 6; export const NOTE_OVERLAY_OFFSET_Y = 6;
export const NOTE_OVERLAY_WIDTH = 100; export const NOTE_OVERLAY_WIDTH = 100;

View File

@@ -1,6 +1,7 @@
import { focusTextModel } from '@blocksuite/affine-components/rich-text'; import { focusTextModel } from '@blocksuite/affine-components/rich-text';
import type { NoteBlockModel, RootBlockModel } from '@blocksuite/affine-model'; import type { NoteBlockModel, RootBlockModel } from '@blocksuite/affine-model';
import { NoteDisplayMode } from '@blocksuite/affine-model'; import { NoteDisplayMode } from '@blocksuite/affine-model';
import { PageViewportService } from '@blocksuite/affine-shared/services';
import type { Viewport } from '@blocksuite/affine-shared/types'; import type { Viewport } from '@blocksuite/affine-shared/types';
import { import {
focusTitle, focusTitle,
@@ -141,10 +142,6 @@ export class PageRootBlockComponent extends BlockComponent<
return getScrollContainer(this); return getScrollContainer(this);
} }
get slots() {
return this.service.slots;
}
get viewport(): Viewport | null { get viewport(): Viewport | null {
if (!this.viewportElement) { if (!this.viewportElement) {
return null; return null;
@@ -172,9 +169,9 @@ export class PageRootBlockComponent extends BlockComponent<
get viewportElement(): HTMLDivElement | null { get viewportElement(): HTMLDivElement | null {
if (this._viewportElement) return this._viewportElement; if (this._viewportElement) return this._viewportElement;
this._viewportElement = this.host.closest( this._viewportElement = this.host.closest<HTMLDivElement>(
'.affine-page-viewport' '.affine-page-viewport'
) as HTMLDivElement | null; );
return this._viewportElement; return this._viewportElement;
} }
@@ -198,12 +195,14 @@ export class PageRootBlockComponent extends BlockComponent<
if (!viewport || !viewportElement) { if (!viewport || !viewportElement) {
return; return;
} }
const viewportService = this.std.get(PageViewportService);
// when observe viewportElement resize, emit viewport update event // when observe viewportElement resize, emit viewport update event
const resizeObserver = new ResizeObserver( const resizeObserver = new ResizeObserver(
(entries: ResizeObserverEntry[]) => { (entries: ResizeObserverEntry[]) => {
for (const { target } of entries) { for (const { target } of entries) {
if (target === viewportElement) { if (target === viewportElement) {
this.slots.viewportUpdated.emit(viewport); viewportService.emit(viewport);
break; break;
} }
} }

View File

@@ -4,8 +4,10 @@ import {
DocDisplayMetaService, DocDisplayMetaService,
DocModeService, DocModeService,
EmbedOptionService, EmbedOptionService,
PageViewportServiceExtension,
ThemeService, ThemeService,
} from '@blocksuite/affine-shared/services'; } from '@blocksuite/affine-shared/services';
import { AFFINE_DRAG_HANDLE_WIDGET } from '@blocksuite/affine-widget-drag-handle';
import { AFFINE_DOC_REMOTE_SELECTION_WIDGET } from '@blocksuite/affine-widget-remote-selection'; import { AFFINE_DOC_REMOTE_SELECTION_WIDGET } from '@blocksuite/affine-widget-remote-selection';
import { AFFINE_SCROLL_ANCHORING_WIDGET } from '@blocksuite/affine-widget-scroll-anchoring'; import { AFFINE_SCROLL_ANCHORING_WIDGET } from '@blocksuite/affine-widget-scroll-anchoring';
import { import {
@@ -20,7 +22,6 @@ import { literal, unsafeStatic } from 'lit/static-html.js';
import { ExportManagerExtension } from '../../_common/export-manager/export-manager.js'; import { ExportManagerExtension } from '../../_common/export-manager/export-manager.js';
import { RootBlockAdapterExtensions } from '../adapters/extension.js'; import { RootBlockAdapterExtensions } from '../adapters/extension.js';
import { commands } from '../commands/index.js'; import { commands } from '../commands/index.js';
import { AFFINE_DRAG_HANDLE_WIDGET } from '../widgets/drag-handle/consts.js';
import { AFFINE_EMBED_CARD_TOOLBAR_WIDGET } from '../widgets/embed-card-toolbar/embed-card-toolbar.js'; import { AFFINE_EMBED_CARD_TOOLBAR_WIDGET } from '../widgets/embed-card-toolbar/embed-card-toolbar.js';
import { AFFINE_FORMAT_BAR_WIDGET } from '../widgets/format-bar/format-bar.js'; import { AFFINE_FORMAT_BAR_WIDGET } from '../widgets/format-bar/format-bar.js';
import { AFFINE_INNER_MODAL_WIDGET } from '../widgets/inner-modal/inner-modal.js'; import { AFFINE_INNER_MODAL_WIDGET } from '../widgets/inner-modal/inner-modal.js';
@@ -74,6 +75,7 @@ export const PageRootBlockSpec: ExtensionType[] = [
WidgetViewMapExtension('affine:page', pageRootWidgetViewMap), WidgetViewMapExtension('affine:page', pageRootWidgetViewMap),
ExportManagerExtension, ExportManagerExtension,
DNDAPIExtension, DNDAPIExtension,
PageViewportServiceExtension,
DocDisplayMetaService, DocDisplayMetaService,
RootBlockAdapterExtensions, RootBlockAdapterExtensions,
FileDropExtension, FileDropExtension,

View File

@@ -1,3 +1,4 @@
import type { AFFINE_DRAG_HANDLE_WIDGET } from '@blocksuite/affine-widget-drag-handle';
import type { import type {
AFFINE_DOC_REMOTE_SELECTION_WIDGET, AFFINE_DOC_REMOTE_SELECTION_WIDGET,
AFFINE_EDGELESS_REMOTE_SELECTION_WIDGET, AFFINE_EDGELESS_REMOTE_SELECTION_WIDGET,
@@ -5,7 +6,6 @@ import type {
import type { EdgelessRootBlockComponent } from './edgeless/edgeless-root-block.js'; import type { EdgelessRootBlockComponent } from './edgeless/edgeless-root-block.js';
import type { PageRootBlockComponent } from './page/page-root-block.js'; import type { PageRootBlockComponent } from './page/page-root-block.js';
import type { AFFINE_DRAG_HANDLE_WIDGET } from './widgets/drag-handle/consts.js';
import type { AFFINE_EDGELESS_ZOOM_TOOLBAR_WIDGET } from './widgets/edgeless-zoom-toolbar/index.js'; import type { AFFINE_EDGELESS_ZOOM_TOOLBAR_WIDGET } from './widgets/edgeless-zoom-toolbar/index.js';
import type { EDGELESS_ELEMENT_TOOLBAR_WIDGET } from './widgets/element-toolbar/index.js'; import type { EDGELESS_ELEMENT_TOOLBAR_WIDGET } from './widgets/element-toolbar/index.js';
import type { AFFINE_EMBED_CARD_TOOLBAR_WIDGET } from './widgets/embed-card-toolbar/embed-card-toolbar.js'; import type { AFFINE_EMBED_CARD_TOOLBAR_WIDGET } from './widgets/embed-card-toolbar/embed-card-toolbar.js';

View File

@@ -6,7 +6,6 @@ export {
type AffineAIPanelState, type AffineAIPanelState,
type AffineAIPanelWidgetConfig, type AffineAIPanelWidgetConfig,
} from './ai-panel/type.js'; } from './ai-panel/type.js';
export { AffineDragHandleWidget } from './drag-handle/drag-handle.js';
export { export {
AFFINE_EDGELESS_COPILOT_WIDGET, AFFINE_EDGELESS_COPILOT_WIDGET,
EdgelessCopilotWidget, EdgelessCopilotWidget,
@@ -53,7 +52,3 @@ export {
type AffineSlashSubMenu, type AffineSlashSubMenu,
} from './slash-menu/index.js'; } from './slash-menu/index.js';
export { AffineSurfaceRefToolbar } from './surface-ref-toolbar/surface-ref-toolbar.js'; export { AffineSurfaceRefToolbar } from './surface-ref-toolbar/surface-ref-toolbar.js';
export {
AffineDocRemoteSelectionWidget,
EdgelessRemoteSelectionWidget,
} from '@blocksuite/affine-widget-remote-selection';

View File

@@ -1,5 +1,6 @@
import type { RootBlockModel } from '@blocksuite/affine-model'; import type { RootBlockModel } from '@blocksuite/affine-model';
import { import {
autoScroll,
getScrollContainer, getScrollContainer,
matchFlavours, matchFlavours,
} from '@blocksuite/affine-shared/utils'; } from '@blocksuite/affine-shared/utils';
@@ -15,7 +16,6 @@ import { state } from 'lit/decorators.js';
import { styleMap } from 'lit/directives/style-map.js'; import { styleMap } from 'lit/directives/style-map.js';
import type { PageRootBlockComponent } from '../../index.js'; import type { PageRootBlockComponent } from '../../index.js';
import { autoScroll } from '../../text-selection/utils.js';
type Rect = { type Rect = {
left: number; left: number;

View File

@@ -75,6 +75,9 @@
}, },
{ {
"path": "../affine/widget-remote-selection" "path": "../affine/widget-remote-selection"
},
{
"path": "../affine/widget-drag-handle"
} }
] ]
} }

View File

@@ -234,6 +234,23 @@ export const PackageList = [
'blocksuite/framework/store', 'blocksuite/framework/store',
], ],
}, },
{
location: 'blocksuite/affine/widget-drag-handle',
name: '@blocksuite/affine-widget-drag-handle',
workspaceDependencies: [
'blocksuite/affine/block-list',
'blocksuite/affine/block-note',
'blocksuite/affine/block-paragraph',
'blocksuite/affine/block-surface',
'blocksuite/affine/components',
'blocksuite/affine/model',
'blocksuite/affine/shared',
'blocksuite/framework/block-std',
'blocksuite/framework/global',
'blocksuite/framework/inline',
'blocksuite/framework/store',
],
},
{ {
location: 'blocksuite/affine/widget-remote-selection', location: 'blocksuite/affine/widget-remote-selection',
name: '@blocksuite/affine-widget-remote-selection', name: '@blocksuite/affine-widget-remote-selection',
@@ -274,6 +291,7 @@ export const PackageList = [
'blocksuite/affine/components', 'blocksuite/affine/components',
'blocksuite/affine/model', 'blocksuite/affine/model',
'blocksuite/affine/shared', 'blocksuite/affine/shared',
'blocksuite/affine/widget-drag-handle',
'blocksuite/affine/widget-remote-selection', 'blocksuite/affine/widget-remote-selection',
'blocksuite/affine/widget-scroll-anchoring', 'blocksuite/affine/widget-scroll-anchoring',
'blocksuite/framework/block-std', 'blocksuite/framework/block-std',
@@ -624,6 +642,7 @@ export type PackageName =
| '@blocksuite/data-view' | '@blocksuite/data-view'
| '@blocksuite/affine-model' | '@blocksuite/affine-model'
| '@blocksuite/affine-shared' | '@blocksuite/affine-shared'
| '@blocksuite/affine-widget-drag-handle'
| '@blocksuite/affine-widget-remote-selection' | '@blocksuite/affine-widget-remote-selection'
| '@blocksuite/affine-widget-scroll-anchoring' | '@blocksuite/affine-widget-scroll-anchoring'
| '@blocksuite/blocks' | '@blocksuite/blocks'

View File

@@ -24,6 +24,7 @@
{ "path": "./blocksuite/affine/data-view" }, { "path": "./blocksuite/affine/data-view" },
{ "path": "./blocksuite/affine/model" }, { "path": "./blocksuite/affine/model" },
{ "path": "./blocksuite/affine/shared" }, { "path": "./blocksuite/affine/shared" },
{ "path": "./blocksuite/affine/widget-drag-handle" },
{ "path": "./blocksuite/affine/widget-remote-selection" }, { "path": "./blocksuite/affine/widget-remote-selection" },
{ "path": "./blocksuite/affine/widget-scroll-anchoring" }, { "path": "./blocksuite/affine/widget-scroll-anchoring" },
{ "path": "./blocksuite/blocks" }, { "path": "./blocksuite/blocks" },

View File

@@ -3613,6 +3613,32 @@ __metadata:
languageName: unknown languageName: unknown
linkType: soft linkType: soft
"@blocksuite/affine-widget-drag-handle@workspace:*, @blocksuite/affine-widget-drag-handle@workspace:blocksuite/affine/widget-drag-handle":
version: 0.0.0-use.local
resolution: "@blocksuite/affine-widget-drag-handle@workspace:blocksuite/affine/widget-drag-handle"
dependencies:
"@blocksuite/affine-block-list": "workspace:*"
"@blocksuite/affine-block-note": "workspace:*"
"@blocksuite/affine-block-paragraph": "workspace:*"
"@blocksuite/affine-block-surface": "workspace:*"
"@blocksuite/affine-components": "workspace:*"
"@blocksuite/affine-model": "workspace:*"
"@blocksuite/affine-shared": "workspace:*"
"@blocksuite/block-std": "workspace:*"
"@blocksuite/global": "workspace:*"
"@blocksuite/icons": "npm:^2.1.75"
"@blocksuite/inline": "workspace:*"
"@blocksuite/store": "workspace:*"
"@floating-ui/dom": "npm:^1.6.10"
"@lit/context": "npm:^1.1.2"
"@preact/signals-core": "npm:^1.8.0"
"@toeverything/theme": "npm:^1.1.1"
lit: "npm:^3.2.0"
minimatch: "npm:^10.0.1"
zod: "npm:^3.23.8"
languageName: unknown
linkType: soft
"@blocksuite/affine-widget-remote-selection@workspace:*, @blocksuite/affine-widget-remote-selection@workspace:blocksuite/affine/widget-remote-selection": "@blocksuite/affine-widget-remote-selection@workspace:*, @blocksuite/affine-widget-remote-selection@workspace:blocksuite/affine/widget-remote-selection":
version: 0.0.0-use.local version: 0.0.0-use.local
resolution: "@blocksuite/affine-widget-remote-selection@workspace:blocksuite/affine/widget-remote-selection" resolution: "@blocksuite/affine-widget-remote-selection@workspace:blocksuite/affine/widget-remote-selection"
@@ -3696,6 +3722,7 @@ __metadata:
"@blocksuite/affine-components": "workspace:*" "@blocksuite/affine-components": "workspace:*"
"@blocksuite/affine-model": "workspace:*" "@blocksuite/affine-model": "workspace:*"
"@blocksuite/affine-shared": "workspace:*" "@blocksuite/affine-shared": "workspace:*"
"@blocksuite/affine-widget-drag-handle": "workspace:*"
"@blocksuite/affine-widget-remote-selection": "workspace:*" "@blocksuite/affine-widget-remote-selection": "workspace:*"
"@blocksuite/affine-widget-scroll-anchoring": "workspace:*" "@blocksuite/affine-widget-scroll-anchoring": "workspace:*"
"@blocksuite/block-std": "workspace:*" "@blocksuite/block-std": "workspace:*"