refactor(editor): optimize pasting process of attachments and images (#12276)

Related to: [BS-3146](https://linear.app/affine-design/issue/BS-3146/import-paste-接口改进优化)
This commit is contained in:
fundon
2025-05-18 01:57:42 +00:00
parent f3693a91c3
commit 8726b0e462
14 changed files with 240 additions and 76 deletions
@@ -143,7 +143,11 @@ export class AttachmentBlockComponent extends CaptionedBlockComponent<Attachment
this.disposables.add(this.resourceController.subscribe());
this.disposables.add(this.resourceController);
this.refreshData();
this.disposables.add(
this.model.props.sourceId$.subscribe(() => {
this.refreshData();
})
);
if (!this.model.props.style && !this.store.readonly) {
this.store.withoutTransact(() => {
@@ -7,7 +7,7 @@ import {
} from '@blocksuite/affine-shared/commands';
import { ImageSelection } from '@blocksuite/affine-shared/selection';
import { unsafeCSSVarV2 } from '@blocksuite/affine-shared/theme';
import { WithDisposable } from '@blocksuite/global/lit';
import { SignalWatcher, WithDisposable } from '@blocksuite/global/lit';
import type { BlockComponent, UIEventStateContext } from '@blocksuite/std';
import {
BlockSelection,
@@ -15,8 +15,9 @@ import {
TextSelection,
} from '@blocksuite/std';
import type { BaseSelection } from '@blocksuite/store';
import { computed } from '@preact/signals-core';
import { css, html, type PropertyValues } from 'lit';
import { property, query, state } from 'lit/decorators.js';
import { property, query } from 'lit/decorators.js';
import { styleMap } from 'lit/directives/style-map.js';
import { when } from 'lit/directives/when.js';
@@ -25,7 +26,9 @@ import { ImageResizeManager } from '../image-resize-manager';
import { shouldResizeImage } from '../utils';
import { ImageSelectedRect } from './image-selected-rect';
export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
export class ImageBlockPageComponent extends SignalWatcher(
WithDisposable(ShadowlessElement)
) {
static override styles = css`
affine-page-image {
position: relative;
@@ -68,6 +71,8 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
}
`;
resizeable$ = computed(() => this.block.resizeable$.value);
private _isDragging = false;
private get _doc() {
@@ -134,21 +139,21 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
return true;
},
Delete: ctx => {
if (this._host.store.readonly || !this._isSelected) return;
if (this._host.store.readonly || !this.resizeable$.peek()) return;
addParagraph(ctx);
this._doc.deleteBlock(this._model);
return true;
},
Backspace: ctx => {
if (this._host.store.readonly || !this._isSelected) return;
if (this._host.store.readonly || !this.resizeable$.peek()) return;
addParagraph(ctx);
this._doc.deleteBlock(this._model);
return true;
},
Enter: ctx => {
if (this._host.store.readonly || !this._isSelected) return;
if (this._host.store.readonly || !this.resizeable$.peek()) return;
addParagraph(ctx);
return true;
@@ -213,19 +218,6 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
private _handleSelection() {
const selection = this._host.selection;
this._disposables.add(
selection.slots.changed.subscribe(selList => {
this._isSelected = selList.some(
sel => sel.blockId === this.block.blockId && sel.is(ImageSelection)
);
})
);
this._disposables.add(
this._model.propsUpdated.subscribe(() => {
this.requestUpdate();
})
);
this._disposables.addFromEvent(
this.resizeImg,
@@ -249,7 +241,7 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
this.block.handleEvent(
'click',
() => {
if (!this._isSelected) return;
if (!this.resizeable$.peek()) return;
selection.update(selList =>
selList.filter(
@@ -356,7 +348,7 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
override render() {
const imageSize = this._normalizeImageSize();
const imageSelectedRect = this._isSelected
const imageSelectedRect = this.resizeable$.value
? ImageSelectedRect(this._doc.readonly)
: null;
@@ -389,9 +381,6 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
`;
}
@state()
accessor _isSelected = false;
@property({ attribute: false })
accessor block!: ImageBlockComponent;
@@ -4,6 +4,7 @@ import { getLoadingIconWith } from '@blocksuite/affine-components/icons';
import { Peekable } from '@blocksuite/affine-components/peek';
import { ResourceController } from '@blocksuite/affine-components/resource';
import type { ImageBlockModel } from '@blocksuite/affine-model';
import { ImageSelection } from '@blocksuite/affine-shared/selection';
import {
ThemeProvider,
ToolbarRegistryIdentifier,
@@ -30,6 +31,13 @@ import {
enableOn: () => !IS_MOBILE,
})
export class ImageBlockComponent extends CaptionedBlockComponent<ImageBlockModel> {
resizeable$ = computed(() =>
this.std.selection.value.some(
selection =>
selection.is(ImageSelection) && selection.blockId === this.blockId
)
);
resourceController = new ResourceController(
computed(() => this.model.props.sourceId$.value),
'Image'
@@ -104,7 +112,11 @@ export class ImageBlockComponent extends CaptionedBlockComponent<ImageBlockModel
this.disposables.add(this.resourceController.subscribe());
this.disposables.add(this.resourceController);
this.refreshData();
this.disposables.add(
this.model.props.sourceId$.subscribe(() => {
this.refreshData();
})
);
}
override firstUpdated() {
@@ -100,7 +100,11 @@ export class ImageEdgelessBlockComponent extends GfxBlockComponent<ImageBlockMod
this.disposables.add(this.resourceController.subscribe());
this.disposables.add(this.resourceController);
this.refreshData();
this.disposables.add(
this.model.props.sourceId$.subscribe(() => {
this.refreshData();
})
);
}
override renderGfxBlock() {
@@ -3,6 +3,7 @@ import {
pasteMiddleware,
replaceIdMiddleware,
surfaceRefToEmbed,
uploadMiddleware,
} from '@blocksuite/affine-shared/adapters';
import {
clearAndSelectFirstModelCommand,
@@ -34,14 +35,17 @@ export class PageClipboard extends ReadOnlyClipboard {
// When pastina a surface-ref block to another doc
const surfaceRefToEmbedMiddleware = surfaceRefToEmbed(this.std);
const replaceId = replaceIdMiddleware(this.std.store.workspace.idGenerator);
const upload = uploadMiddleware(this.std);
this.std.clipboard.use(paste);
this.std.clipboard.use(surfaceRefToEmbedMiddleware);
this.std.clipboard.use(replaceId);
this.std.clipboard.use(upload);
this._disposables.add({
dispose: () => {
this.std.clipboard.unuse(paste);
this.std.clipboard.unuse(surfaceRefToEmbedMiddleware);
this.std.clipboard.unuse(replaceId);
this.std.clipboard.unuse(upload);
},
});
};
@@ -35,6 +35,7 @@ export type ResolvedStateInfo = StateInfo & ResolvedStateInfoPart;
export class ResourceController implements Disposable {
readonly blobUrl$ = signal<string | null>(null);
// TODO(@fundon): default `loading` status.
readonly state$ = signal<Partial<BlobState>>({});
readonly resolvedState$ = computed<ResolvedStateInfoPart>(() => {
@@ -1,5 +1,5 @@
import { AttachmentBlockSchema } from '@blocksuite/affine-model';
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
import { sha } from '@blocksuite/global/utils';
import {
type AssetsManager,
BaseAdapter,
@@ -88,40 +88,49 @@ export class AttachmentAdapter extends BaseAdapter<Attachment> {
);
}
override async toSliceSnapshot(
payload: AttachmentToSliceSnapshotPayload
): Promise<SliceSnapshot | null> {
override async toSliceSnapshot({
assets,
file: files,
pageId,
workspaceId,
}: AttachmentToSliceSnapshotPayload): Promise<SliceSnapshot | null> {
if (files.length === 0) return null;
const content: SliceSnapshot['content'] = [];
for (const item of payload.file) {
const blobId = await sha(await item.arrayBuffer());
payload.assets?.getAssets().set(blobId, item);
await payload.assets?.writeToBlob(blobId);
const flavour = AttachmentBlockSchema.model.flavour;
for (const blob of files) {
const id = nanoid();
const { name, size, type } = blob;
assets?.uploadingAssetsMap.set(id, {
blob,
mapInto: sourceId => ({ sourceId }),
});
content.push({
type: 'block',
flavour: 'affine:attachment',
id: nanoid(),
flavour,
id,
props: {
name: item.name,
size: item.size,
type: item.type,
name,
size,
type,
embed: false,
style: 'horizontalThin',
index: 'a0',
xywh: '[0,0,0,0]',
rotate: 0,
sourceId: blobId,
},
children: [],
});
}
if (content.length === 0) {
return null;
}
return {
type: 'slice',
content,
workspaceId: payload.workspaceId,
pageId: payload.pageId,
pageId,
workspaceId,
};
}
}
+26 -19
View File
@@ -1,5 +1,5 @@
import { ImageBlockSchema } from '@blocksuite/affine-model';
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
import { sha } from '@blocksuite/global/utils';
import {
type AssetsManager,
BaseAdapter,
@@ -88,33 +88,40 @@ export class ImageAdapter extends BaseAdapter<Image> {
);
}
override async toSliceSnapshot(
payload: ImageToSliceSnapshotPayload
): Promise<SliceSnapshot | null> {
override async toSliceSnapshot({
assets,
file: files,
pageId,
workspaceId,
}: ImageToSliceSnapshotPayload): Promise<SliceSnapshot | null> {
if (files.length === 0) return null;
const content: SliceSnapshot['content'] = [];
for (const item of payload.file) {
const blobId = await sha(await item.arrayBuffer());
payload.assets?.getAssets().set(blobId, item);
await payload.assets?.writeToBlob(blobId);
const flavour = ImageBlockSchema.model.flavour;
for (const blob of files) {
const id = nanoid();
const { size } = blob;
assets?.uploadingAssetsMap.set(id, {
blob,
mapInto: sourceId => ({ sourceId }),
});
content.push({
type: 'block',
flavour: 'affine:image',
id: nanoid(),
props: {
size: item.size,
sourceId: blobId,
},
flavour,
id,
props: { size },
children: [],
});
}
if (content.length === 0) {
return null;
}
return {
type: 'slice',
content,
workspaceId: payload.workspaceId,
pageId: payload.pageId,
pageId,
workspaceId,
};
}
}
@@ -9,3 +9,4 @@ export * from './proxy';
export * from './replace-id';
export * from './surface-ref-to-embed';
export * from './title';
export * from './upload';
@@ -19,7 +19,7 @@ import { matchModels } from '../../utils';
export const replaceIdMiddleware =
(idGenerator: () => string): TransformerMiddleware =>
({ slots, docCRUD }) => {
({ slots, docCRUD, assetsManager }) => {
const idMap = new Map<string, string>();
// After Import
@@ -169,6 +169,16 @@ export const replaceIdMiddleware =
}
snapshot.id = newId;
// Should be re-paired.
if (['affine:attachment', 'affine:image'].includes(snapshot.flavour)) {
if (!assetsManager.uploadingAssetsMap.has(original)) return;
const data = assetsManager.uploadingAssetsMap.get(original)!;
assetsManager.uploadingAssetsMap.set(newId, data);
assetsManager.uploadingAssetsMap.delete(original);
return;
}
if (snapshot.flavour === 'affine:surface') {
// Generate new IDs for images and frames in advance.
snapshot.children.forEach(child => {
@@ -0,0 +1,114 @@
import { sha } from '@blocksuite/global/utils';
import type { BlockStdScope } from '@blocksuite/std';
import type {
BlockModel,
BlockProps,
TransformerMiddleware,
} from '@blocksuite/store';
import { filter, from, map, mergeMap } from 'rxjs';
const ALLOWED_FLAVOURS = new Set(['affine:attachment', 'affine:image']);
export const uploadMiddleware = (
std: BlockStdScope,
concurrent = 5
): TransformerMiddleware => {
const blockView$ = std.view.viewUpdated.pipe(
filter(payload => payload.type === 'block'),
filter(payload => ALLOWED_FLAVOURS.has(payload.view.model.flavour))
);
return ({ assetsManager }) => {
async function upload(
model: BlockModel,
{
blob,
mapInto,
abortController,
}: {
blob: Blob;
mapInto: (blobId: string) => Partial<BlockProps>;
abortController?: AbortController;
}
) {
if (!abortController) return null;
const signal = abortController.signal;
if (signal.aborted) return null;
// Double check
if (!model.store.hasBlock(model.id)) return null;
try {
signal.throwIfAborted();
const blobId = await Promise.race([
(async function processUpload() {
const blobId = await sha(await blob.arrayBuffer());
assetsManager.getAssets().set(blobId, blob);
await assetsManager.writeToBlob(blobId);
return await new Promise<string | null>(resolve => {
model.store.withoutTransact(() => {
if (signal.aborted) return resolve(null);
model.store.updateBlock(model, mapInto(blobId));
resolve(blobId);
});
});
})(),
// If the signal is not aborted, it will be in the pending state.
new Promise<null>(resolve => {
signal.addEventListener('abort', () => resolve(null), {
once: true,
});
if (signal.aborted) {
resolve(null);
}
}),
]);
return blobId;
} catch (err) {
console.error(err);
return null;
}
}
blockView$
.pipe(
map(payload => {
if (assetsManager.uploadingAssetsMap.size === 0) return null;
const model = payload.view.model;
if (!assetsManager.uploadingAssetsMap.has(model.id)) return null;
const state = assetsManager.uploadingAssetsMap.get(model.id)!;
if (payload.method === 'add') {
state.abortController = new AbortController();
return { model, state };
} else {
state.abortController?.abort();
assetsManager.uploadingAssetsMap.delete(model.id);
return null;
}
}),
filter(Boolean),
mergeMap(
({ model, state }) =>
from(
upload(model, state).then(() => {
assetsManager.uploadingAssetsMap.delete(model.id);
})
),
concurrent
)
)
.subscribe();
};
};
@@ -1,5 +1,6 @@
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
import type { BlockProps } from '../model';
import type { BlobCRUD } from './type';
type AssetsManagerConfig = {
@@ -18,6 +19,16 @@ function makeNewNameWhenConflict(names: Set<string>, name: string) {
}
export class AssetsManager {
// `blockId` is the key.
readonly uploadingAssetsMap = new Map<
string,
{
blob: Blob;
abortController?: AbortController;
mapInto: (blobId: string) => Partial<BlockProps>;
}
>();
private readonly _assetsMap = new Map<string, Blob>();
private readonly _blob: BlobCRUD;
@@ -245,8 +245,9 @@ export class Transformer {
parent?: string,
index?: number
): Promise<Slice | undefined> => {
SliceSnapshotSchema.parse(snapshot);
try {
SliceSnapshotSchema.parse(snapshot);
this._slots.beforeImport.next({
type: 'slice',
snapshot,
@@ -525,11 +526,11 @@ export class Transformer {
for (let index = 0; index < nodes.length; index++) {
const node = nodes[index];
const { draft } = node;
const { id, flavour } = draft;
const { id, flavour, props } = draft;
const actualIndex =
startIndex !== undefined ? startIndex + index : undefined;
doc.addBlock(flavour, { id, ...draft.props }, parentId, actualIndex);
doc.addBlock(flavour, { id, ...props }, parentId, actualIndex);
const model = doc.getBlock(id)?.model;
if (!model) {
+4 -7
View File
@@ -771,14 +771,11 @@ export async function pasteContent(
export async function pasteTestImage(page: Page) {
await page.evaluate(async () => {
const imageBlob = await fetch(`${location.origin}/test-card-1.png`).then(
response => response.blob()
);
const imageFile = new File([imageBlob], 'test-card-1.png', {
const resp = await fetch(`${location.origin}/test-card-1.png`);
const blob = await resp.blob();
const file = new File([blob], 'test-card-1.png', {
type: 'image/png',
});
const e = new ClipboardEvent('paste', {
clipboardData: new DataTransfer(),
});
@@ -788,7 +785,7 @@ export async function pasteTestImage(page: Page) {
value: document,
});
e.clipboardData?.items.add(imageFile);
e.clipboardData?.items.add(file);
document.dispatchEvent(e);
});
await waitNextFrame(page);