mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-18 23:07:02 +08:00
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:
@@ -143,7 +143,11 @@ export class AttachmentBlockComponent extends CaptionedBlockComponent<Attachment
|
|||||||
this.disposables.add(this.resourceController.subscribe());
|
this.disposables.add(this.resourceController.subscribe());
|
||||||
this.disposables.add(this.resourceController);
|
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) {
|
if (!this.model.props.style && !this.store.readonly) {
|
||||||
this.store.withoutTransact(() => {
|
this.store.withoutTransact(() => {
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import {
|
|||||||
} from '@blocksuite/affine-shared/commands';
|
} from '@blocksuite/affine-shared/commands';
|
||||||
import { ImageSelection } from '@blocksuite/affine-shared/selection';
|
import { ImageSelection } from '@blocksuite/affine-shared/selection';
|
||||||
import { unsafeCSSVarV2 } from '@blocksuite/affine-shared/theme';
|
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 type { BlockComponent, UIEventStateContext } from '@blocksuite/std';
|
||||||
import {
|
import {
|
||||||
BlockSelection,
|
BlockSelection,
|
||||||
@@ -15,8 +15,9 @@ import {
|
|||||||
TextSelection,
|
TextSelection,
|
||||||
} from '@blocksuite/std';
|
} from '@blocksuite/std';
|
||||||
import type { BaseSelection } from '@blocksuite/store';
|
import type { BaseSelection } from '@blocksuite/store';
|
||||||
|
import { computed } from '@preact/signals-core';
|
||||||
import { css, html, type PropertyValues } from 'lit';
|
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 { styleMap } from 'lit/directives/style-map.js';
|
||||||
import { when } from 'lit/directives/when.js';
|
import { when } from 'lit/directives/when.js';
|
||||||
|
|
||||||
@@ -25,7 +26,9 @@ import { ImageResizeManager } from '../image-resize-manager';
|
|||||||
import { shouldResizeImage } from '../utils';
|
import { shouldResizeImage } from '../utils';
|
||||||
import { ImageSelectedRect } from './image-selected-rect';
|
import { ImageSelectedRect } from './image-selected-rect';
|
||||||
|
|
||||||
export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
export class ImageBlockPageComponent extends SignalWatcher(
|
||||||
|
WithDisposable(ShadowlessElement)
|
||||||
|
) {
|
||||||
static override styles = css`
|
static override styles = css`
|
||||||
affine-page-image {
|
affine-page-image {
|
||||||
position: relative;
|
position: relative;
|
||||||
@@ -68,6 +71,8 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
|||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
resizeable$ = computed(() => this.block.resizeable$.value);
|
||||||
|
|
||||||
private _isDragging = false;
|
private _isDragging = false;
|
||||||
|
|
||||||
private get _doc() {
|
private get _doc() {
|
||||||
@@ -134,21 +139,21 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
|||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
Delete: ctx => {
|
Delete: ctx => {
|
||||||
if (this._host.store.readonly || !this._isSelected) return;
|
if (this._host.store.readonly || !this.resizeable$.peek()) return;
|
||||||
|
|
||||||
addParagraph(ctx);
|
addParagraph(ctx);
|
||||||
this._doc.deleteBlock(this._model);
|
this._doc.deleteBlock(this._model);
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
Backspace: ctx => {
|
Backspace: ctx => {
|
||||||
if (this._host.store.readonly || !this._isSelected) return;
|
if (this._host.store.readonly || !this.resizeable$.peek()) return;
|
||||||
|
|
||||||
addParagraph(ctx);
|
addParagraph(ctx);
|
||||||
this._doc.deleteBlock(this._model);
|
this._doc.deleteBlock(this._model);
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
Enter: ctx => {
|
Enter: ctx => {
|
||||||
if (this._host.store.readonly || !this._isSelected) return;
|
if (this._host.store.readonly || !this.resizeable$.peek()) return;
|
||||||
|
|
||||||
addParagraph(ctx);
|
addParagraph(ctx);
|
||||||
return true;
|
return true;
|
||||||
@@ -213,19 +218,6 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
|||||||
|
|
||||||
private _handleSelection() {
|
private _handleSelection() {
|
||||||
const selection = this._host.selection;
|
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._disposables.addFromEvent(
|
||||||
this.resizeImg,
|
this.resizeImg,
|
||||||
@@ -249,7 +241,7 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
|||||||
this.block.handleEvent(
|
this.block.handleEvent(
|
||||||
'click',
|
'click',
|
||||||
() => {
|
() => {
|
||||||
if (!this._isSelected) return;
|
if (!this.resizeable$.peek()) return;
|
||||||
|
|
||||||
selection.update(selList =>
|
selection.update(selList =>
|
||||||
selList.filter(
|
selList.filter(
|
||||||
@@ -356,7 +348,7 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
|||||||
override render() {
|
override render() {
|
||||||
const imageSize = this._normalizeImageSize();
|
const imageSize = this._normalizeImageSize();
|
||||||
|
|
||||||
const imageSelectedRect = this._isSelected
|
const imageSelectedRect = this.resizeable$.value
|
||||||
? ImageSelectedRect(this._doc.readonly)
|
? ImageSelectedRect(this._doc.readonly)
|
||||||
: null;
|
: null;
|
||||||
|
|
||||||
@@ -389,9 +381,6 @@ export class ImageBlockPageComponent extends WithDisposable(ShadowlessElement) {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@state()
|
|
||||||
accessor _isSelected = false;
|
|
||||||
|
|
||||||
@property({ attribute: false })
|
@property({ attribute: false })
|
||||||
accessor block!: ImageBlockComponent;
|
accessor block!: ImageBlockComponent;
|
||||||
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { getLoadingIconWith } from '@blocksuite/affine-components/icons';
|
|||||||
import { Peekable } from '@blocksuite/affine-components/peek';
|
import { Peekable } from '@blocksuite/affine-components/peek';
|
||||||
import { ResourceController } from '@blocksuite/affine-components/resource';
|
import { ResourceController } from '@blocksuite/affine-components/resource';
|
||||||
import type { ImageBlockModel } from '@blocksuite/affine-model';
|
import type { ImageBlockModel } from '@blocksuite/affine-model';
|
||||||
|
import { ImageSelection } from '@blocksuite/affine-shared/selection';
|
||||||
import {
|
import {
|
||||||
ThemeProvider,
|
ThemeProvider,
|
||||||
ToolbarRegistryIdentifier,
|
ToolbarRegistryIdentifier,
|
||||||
@@ -30,6 +31,13 @@ import {
|
|||||||
enableOn: () => !IS_MOBILE,
|
enableOn: () => !IS_MOBILE,
|
||||||
})
|
})
|
||||||
export class ImageBlockComponent extends CaptionedBlockComponent<ImageBlockModel> {
|
export class ImageBlockComponent extends CaptionedBlockComponent<ImageBlockModel> {
|
||||||
|
resizeable$ = computed(() =>
|
||||||
|
this.std.selection.value.some(
|
||||||
|
selection =>
|
||||||
|
selection.is(ImageSelection) && selection.blockId === this.blockId
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
resourceController = new ResourceController(
|
resourceController = new ResourceController(
|
||||||
computed(() => this.model.props.sourceId$.value),
|
computed(() => this.model.props.sourceId$.value),
|
||||||
'Image'
|
'Image'
|
||||||
@@ -104,7 +112,11 @@ export class ImageBlockComponent extends CaptionedBlockComponent<ImageBlockModel
|
|||||||
this.disposables.add(this.resourceController.subscribe());
|
this.disposables.add(this.resourceController.subscribe());
|
||||||
this.disposables.add(this.resourceController);
|
this.disposables.add(this.resourceController);
|
||||||
|
|
||||||
this.refreshData();
|
this.disposables.add(
|
||||||
|
this.model.props.sourceId$.subscribe(() => {
|
||||||
|
this.refreshData();
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override firstUpdated() {
|
override firstUpdated() {
|
||||||
|
|||||||
@@ -100,7 +100,11 @@ export class ImageEdgelessBlockComponent extends GfxBlockComponent<ImageBlockMod
|
|||||||
this.disposables.add(this.resourceController.subscribe());
|
this.disposables.add(this.resourceController.subscribe());
|
||||||
this.disposables.add(this.resourceController);
|
this.disposables.add(this.resourceController);
|
||||||
|
|
||||||
this.refreshData();
|
this.disposables.add(
|
||||||
|
this.model.props.sourceId$.subscribe(() => {
|
||||||
|
this.refreshData();
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override renderGfxBlock() {
|
override renderGfxBlock() {
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import {
|
|||||||
pasteMiddleware,
|
pasteMiddleware,
|
||||||
replaceIdMiddleware,
|
replaceIdMiddleware,
|
||||||
surfaceRefToEmbed,
|
surfaceRefToEmbed,
|
||||||
|
uploadMiddleware,
|
||||||
} from '@blocksuite/affine-shared/adapters';
|
} from '@blocksuite/affine-shared/adapters';
|
||||||
import {
|
import {
|
||||||
clearAndSelectFirstModelCommand,
|
clearAndSelectFirstModelCommand,
|
||||||
@@ -34,14 +35,17 @@ export class PageClipboard extends ReadOnlyClipboard {
|
|||||||
// When pastina a surface-ref block to another doc
|
// When pastina a surface-ref block to another doc
|
||||||
const surfaceRefToEmbedMiddleware = surfaceRefToEmbed(this.std);
|
const surfaceRefToEmbedMiddleware = surfaceRefToEmbed(this.std);
|
||||||
const replaceId = replaceIdMiddleware(this.std.store.workspace.idGenerator);
|
const replaceId = replaceIdMiddleware(this.std.store.workspace.idGenerator);
|
||||||
|
const upload = uploadMiddleware(this.std);
|
||||||
this.std.clipboard.use(paste);
|
this.std.clipboard.use(paste);
|
||||||
this.std.clipboard.use(surfaceRefToEmbedMiddleware);
|
this.std.clipboard.use(surfaceRefToEmbedMiddleware);
|
||||||
this.std.clipboard.use(replaceId);
|
this.std.clipboard.use(replaceId);
|
||||||
|
this.std.clipboard.use(upload);
|
||||||
this._disposables.add({
|
this._disposables.add({
|
||||||
dispose: () => {
|
dispose: () => {
|
||||||
this.std.clipboard.unuse(paste);
|
this.std.clipboard.unuse(paste);
|
||||||
this.std.clipboard.unuse(surfaceRefToEmbedMiddleware);
|
this.std.clipboard.unuse(surfaceRefToEmbedMiddleware);
|
||||||
this.std.clipboard.unuse(replaceId);
|
this.std.clipboard.unuse(replaceId);
|
||||||
|
this.std.clipboard.unuse(upload);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -35,6 +35,7 @@ export type ResolvedStateInfo = StateInfo & ResolvedStateInfoPart;
|
|||||||
export class ResourceController implements Disposable {
|
export class ResourceController implements Disposable {
|
||||||
readonly blobUrl$ = signal<string | null>(null);
|
readonly blobUrl$ = signal<string | null>(null);
|
||||||
|
|
||||||
|
// TODO(@fundon): default `loading` status.
|
||||||
readonly state$ = signal<Partial<BlobState>>({});
|
readonly state$ = signal<Partial<BlobState>>({});
|
||||||
|
|
||||||
readonly resolvedState$ = computed<ResolvedStateInfoPart>(() => {
|
readonly resolvedState$ = computed<ResolvedStateInfoPart>(() => {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
|
import { AttachmentBlockSchema } from '@blocksuite/affine-model';
|
||||||
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
|
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
|
||||||
import { sha } from '@blocksuite/global/utils';
|
|
||||||
import {
|
import {
|
||||||
type AssetsManager,
|
type AssetsManager,
|
||||||
BaseAdapter,
|
BaseAdapter,
|
||||||
@@ -88,40 +88,49 @@ export class AttachmentAdapter extends BaseAdapter<Attachment> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override async toSliceSnapshot(
|
override async toSliceSnapshot({
|
||||||
payload: AttachmentToSliceSnapshotPayload
|
assets,
|
||||||
): Promise<SliceSnapshot | null> {
|
file: files,
|
||||||
|
pageId,
|
||||||
|
workspaceId,
|
||||||
|
}: AttachmentToSliceSnapshotPayload): Promise<SliceSnapshot | null> {
|
||||||
|
if (files.length === 0) return null;
|
||||||
|
|
||||||
const content: SliceSnapshot['content'] = [];
|
const content: SliceSnapshot['content'] = [];
|
||||||
for (const item of payload.file) {
|
const flavour = AttachmentBlockSchema.model.flavour;
|
||||||
const blobId = await sha(await item.arrayBuffer());
|
|
||||||
payload.assets?.getAssets().set(blobId, item);
|
for (const blob of files) {
|
||||||
await payload.assets?.writeToBlob(blobId);
|
const id = nanoid();
|
||||||
|
const { name, size, type } = blob;
|
||||||
|
|
||||||
|
assets?.uploadingAssetsMap.set(id, {
|
||||||
|
blob,
|
||||||
|
mapInto: sourceId => ({ sourceId }),
|
||||||
|
});
|
||||||
|
|
||||||
content.push({
|
content.push({
|
||||||
type: 'block',
|
type: 'block',
|
||||||
flavour: 'affine:attachment',
|
flavour,
|
||||||
id: nanoid(),
|
id,
|
||||||
props: {
|
props: {
|
||||||
name: item.name,
|
name,
|
||||||
size: item.size,
|
size,
|
||||||
type: item.type,
|
type,
|
||||||
embed: false,
|
embed: false,
|
||||||
style: 'horizontalThin',
|
style: 'horizontalThin',
|
||||||
index: 'a0',
|
index: 'a0',
|
||||||
xywh: '[0,0,0,0]',
|
xywh: '[0,0,0,0]',
|
||||||
rotate: 0,
|
rotate: 0,
|
||||||
sourceId: blobId,
|
|
||||||
},
|
},
|
||||||
children: [],
|
children: [],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (content.length === 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return {
|
return {
|
||||||
type: 'slice',
|
type: 'slice',
|
||||||
content,
|
content,
|
||||||
workspaceId: payload.workspaceId,
|
pageId,
|
||||||
pageId: payload.pageId,
|
workspaceId,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
|
import { ImageBlockSchema } from '@blocksuite/affine-model';
|
||||||
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
|
import { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
|
||||||
import { sha } from '@blocksuite/global/utils';
|
|
||||||
import {
|
import {
|
||||||
type AssetsManager,
|
type AssetsManager,
|
||||||
BaseAdapter,
|
BaseAdapter,
|
||||||
@@ -88,33 +88,40 @@ export class ImageAdapter extends BaseAdapter<Image> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override async toSliceSnapshot(
|
override async toSliceSnapshot({
|
||||||
payload: ImageToSliceSnapshotPayload
|
assets,
|
||||||
): Promise<SliceSnapshot | null> {
|
file: files,
|
||||||
|
pageId,
|
||||||
|
workspaceId,
|
||||||
|
}: ImageToSliceSnapshotPayload): Promise<SliceSnapshot | null> {
|
||||||
|
if (files.length === 0) return null;
|
||||||
|
|
||||||
const content: SliceSnapshot['content'] = [];
|
const content: SliceSnapshot['content'] = [];
|
||||||
for (const item of payload.file) {
|
const flavour = ImageBlockSchema.model.flavour;
|
||||||
const blobId = await sha(await item.arrayBuffer());
|
|
||||||
payload.assets?.getAssets().set(blobId, item);
|
for (const blob of files) {
|
||||||
await payload.assets?.writeToBlob(blobId);
|
const id = nanoid();
|
||||||
|
const { size } = blob;
|
||||||
|
|
||||||
|
assets?.uploadingAssetsMap.set(id, {
|
||||||
|
blob,
|
||||||
|
mapInto: sourceId => ({ sourceId }),
|
||||||
|
});
|
||||||
|
|
||||||
content.push({
|
content.push({
|
||||||
type: 'block',
|
type: 'block',
|
||||||
flavour: 'affine:image',
|
flavour,
|
||||||
id: nanoid(),
|
id,
|
||||||
props: {
|
props: { size },
|
||||||
size: item.size,
|
|
||||||
sourceId: blobId,
|
|
||||||
},
|
|
||||||
children: [],
|
children: [],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (content.length === 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return {
|
return {
|
||||||
type: 'slice',
|
type: 'slice',
|
||||||
content,
|
content,
|
||||||
workspaceId: payload.workspaceId,
|
pageId,
|
||||||
pageId: payload.pageId,
|
workspaceId,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,3 +9,4 @@ export * from './proxy';
|
|||||||
export * from './replace-id';
|
export * from './replace-id';
|
||||||
export * from './surface-ref-to-embed';
|
export * from './surface-ref-to-embed';
|
||||||
export * from './title';
|
export * from './title';
|
||||||
|
export * from './upload';
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ import { matchModels } from '../../utils';
|
|||||||
|
|
||||||
export const replaceIdMiddleware =
|
export const replaceIdMiddleware =
|
||||||
(idGenerator: () => string): TransformerMiddleware =>
|
(idGenerator: () => string): TransformerMiddleware =>
|
||||||
({ slots, docCRUD }) => {
|
({ slots, docCRUD, assetsManager }) => {
|
||||||
const idMap = new Map<string, string>();
|
const idMap = new Map<string, string>();
|
||||||
|
|
||||||
// After Import
|
// After Import
|
||||||
@@ -169,6 +169,16 @@ export const replaceIdMiddleware =
|
|||||||
}
|
}
|
||||||
snapshot.id = newId;
|
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') {
|
if (snapshot.flavour === 'affine:surface') {
|
||||||
// Generate new IDs for images and frames in advance.
|
// Generate new IDs for images and frames in advance.
|
||||||
snapshot.children.forEach(child => {
|
snapshot.children.forEach(child => {
|
||||||
|
|||||||
114
blocksuite/affine/shared/src/adapters/middlewares/upload.ts
Normal file
114
blocksuite/affine/shared/src/adapters/middlewares/upload.ts
Normal file
@@ -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 { BlockSuiteError, ErrorCode } from '@blocksuite/global/exceptions';
|
||||||
|
|
||||||
|
import type { BlockProps } from '../model';
|
||||||
import type { BlobCRUD } from './type';
|
import type { BlobCRUD } from './type';
|
||||||
|
|
||||||
type AssetsManagerConfig = {
|
type AssetsManagerConfig = {
|
||||||
@@ -18,6 +19,16 @@ function makeNewNameWhenConflict(names: Set<string>, name: string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class AssetsManager {
|
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 _assetsMap = new Map<string, Blob>();
|
||||||
|
|
||||||
private readonly _blob: BlobCRUD;
|
private readonly _blob: BlobCRUD;
|
||||||
|
|||||||
@@ -245,8 +245,9 @@ export class Transformer {
|
|||||||
parent?: string,
|
parent?: string,
|
||||||
index?: number
|
index?: number
|
||||||
): Promise<Slice | undefined> => {
|
): Promise<Slice | undefined> => {
|
||||||
SliceSnapshotSchema.parse(snapshot);
|
|
||||||
try {
|
try {
|
||||||
|
SliceSnapshotSchema.parse(snapshot);
|
||||||
|
|
||||||
this._slots.beforeImport.next({
|
this._slots.beforeImport.next({
|
||||||
type: 'slice',
|
type: 'slice',
|
||||||
snapshot,
|
snapshot,
|
||||||
@@ -525,11 +526,11 @@ export class Transformer {
|
|||||||
for (let index = 0; index < nodes.length; index++) {
|
for (let index = 0; index < nodes.length; index++) {
|
||||||
const node = nodes[index];
|
const node = nodes[index];
|
||||||
const { draft } = node;
|
const { draft } = node;
|
||||||
const { id, flavour } = draft;
|
const { id, flavour, props } = draft;
|
||||||
|
|
||||||
const actualIndex =
|
const actualIndex =
|
||||||
startIndex !== undefined ? startIndex + index : undefined;
|
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;
|
const model = doc.getBlock(id)?.model;
|
||||||
if (!model) {
|
if (!model) {
|
||||||
|
|||||||
@@ -771,14 +771,11 @@ export async function pasteContent(
|
|||||||
|
|
||||||
export async function pasteTestImage(page: Page) {
|
export async function pasteTestImage(page: Page) {
|
||||||
await page.evaluate(async () => {
|
await page.evaluate(async () => {
|
||||||
const imageBlob = await fetch(`${location.origin}/test-card-1.png`).then(
|
const resp = await fetch(`${location.origin}/test-card-1.png`);
|
||||||
response => response.blob()
|
const blob = await resp.blob();
|
||||||
);
|
const file = new File([blob], 'test-card-1.png', {
|
||||||
|
|
||||||
const imageFile = new File([imageBlob], 'test-card-1.png', {
|
|
||||||
type: 'image/png',
|
type: 'image/png',
|
||||||
});
|
});
|
||||||
|
|
||||||
const e = new ClipboardEvent('paste', {
|
const e = new ClipboardEvent('paste', {
|
||||||
clipboardData: new DataTransfer(),
|
clipboardData: new DataTransfer(),
|
||||||
});
|
});
|
||||||
@@ -788,7 +785,7 @@ export async function pasteTestImage(page: Page) {
|
|||||||
value: document,
|
value: document,
|
||||||
});
|
});
|
||||||
|
|
||||||
e.clipboardData?.items.add(imageFile);
|
e.clipboardData?.items.add(file);
|
||||||
document.dispatchEvent(e);
|
document.dispatchEvent(e);
|
||||||
});
|
});
|
||||||
await waitNextFrame(page);
|
await waitNextFrame(page);
|
||||||
|
|||||||
Reference in New Issue
Block a user