新增 useImageCanvasAssetCanvasBridge 承载素材加入画布和画布 drop 桥接 新增 hook 单测覆盖素材建层、pointer drop 和删除素材清理图层 精简 ImageCanvasEditorView 中的素材到画布胶水 更新图片画布拆分计划和 TRACKING 浏览器回归记录
192 lines
5.3 KiB
TypeScript
192 lines
5.3 KiB
TypeScript
import {
|
|
type Dispatch,
|
|
type MutableRefObject,
|
|
type RefObject,
|
|
type SetStateAction,
|
|
useCallback,
|
|
useMemo,
|
|
} from 'react';
|
|
|
|
import {
|
|
createLayerFromAsset,
|
|
isLayerLinkedToAsset,
|
|
} from './ImageCanvasEditorModel';
|
|
import type {
|
|
AssetPointerDragState,
|
|
CanvasLayer,
|
|
CanvasViewport,
|
|
EditorAsset,
|
|
EditorAssetFolder,
|
|
} from './ImageCanvasEditorTypes';
|
|
import { useImageCanvasAssetPointerDragBridge } from './useImageCanvasAssetPointerDragBridge';
|
|
import { useImageCanvasCanvasDropWorkflow } from './useImageCanvasCanvasDropWorkflow';
|
|
|
|
type CanvasPoint = { x: number; y: number };
|
|
|
|
type UploadFilesToCanvasOptions = {
|
|
folderId?: string;
|
|
canvasPoint: CanvasPoint;
|
|
addToCanvas: true;
|
|
};
|
|
|
|
type UseImageCanvasAssetLayerCleanupOptions = {
|
|
layers: CanvasLayer[];
|
|
setLayers: Dispatch<SetStateAction<CanvasLayer[]>>;
|
|
setSelectedLayerId: Dispatch<SetStateAction<string | null>>;
|
|
setSelectedLayerIds: Dispatch<SetStateAction<string[]>>;
|
|
};
|
|
|
|
type UseImageCanvasAssetCanvasBridgeOptions = {
|
|
assetPointerDragRef: RefObject<AssetPointerDragState | null>;
|
|
suppressAssetClickRef: RefObject<boolean>;
|
|
assets: EditorAsset[];
|
|
assetFolders: EditorAssetFolder[];
|
|
layerCounterRef: MutableRefObject<number>;
|
|
viewport: CanvasViewport;
|
|
canvasSize: { width: number; height: number };
|
|
resolveAssetFolderId: (clientX: number, clientY: number) => string | null;
|
|
resolveCanvasPoint: (clientX: number, clientY: number) => CanvasPoint | null;
|
|
getCanvasDropPoint: (clientX: number, clientY: number) => CanvasPoint;
|
|
setAssetPointerDrag: Dispatch<SetStateAction<AssetPointerDragState | null>>;
|
|
setActiveUploadFolderId: Dispatch<SetStateAction<string>>;
|
|
setUploadDropTarget: Dispatch<SetStateAction<'canvas' | 'assets' | null>>;
|
|
setHoveredLayerId: Dispatch<SetStateAction<string | null>>;
|
|
updateAssetMoveDropFolder: (folderId: string | null) => void;
|
|
moveAssetToFolder: (assetId: string, folderId: string) => void;
|
|
captureCanvasHistory: () => void;
|
|
appendCanvasLayersWithResources: (nextLayers: CanvasLayer[]) => void;
|
|
selectSingleLayer: (layerId: string | null) => void;
|
|
addUploadedFiles: (
|
|
files: FileList | File[],
|
|
options: UploadFilesToCanvasOptions,
|
|
) => void;
|
|
};
|
|
|
|
export function useImageCanvasAssetLayerCleanup({
|
|
layers,
|
|
setLayers,
|
|
setSelectedLayerId,
|
|
setSelectedLayerIds,
|
|
}: UseImageCanvasAssetLayerCleanupOptions) {
|
|
return useCallback(
|
|
(deletedAssets: EditorAsset[]) => {
|
|
if (!deletedAssets.length) {
|
|
return;
|
|
}
|
|
setLayers((currentLayers) =>
|
|
currentLayers.filter(
|
|
(layer) =>
|
|
!deletedAssets.some((asset) => isLayerLinkedToAsset(layer, asset)),
|
|
),
|
|
);
|
|
setSelectedLayerIds((currentIds) =>
|
|
currentIds.filter((layerId) =>
|
|
layers.every(
|
|
(layer) =>
|
|
layer.id !== layerId ||
|
|
!deletedAssets.some((asset) =>
|
|
isLayerLinkedToAsset(layer, asset),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
setSelectedLayerId((currentId) => {
|
|
if (!currentId) {
|
|
return currentId;
|
|
}
|
|
const currentLayer = layers.find((layer) => layer.id === currentId);
|
|
return currentLayer &&
|
|
deletedAssets.some((asset) => isLayerLinkedToAsset(currentLayer, asset))
|
|
? null
|
|
: currentId;
|
|
});
|
|
},
|
|
[layers, setLayers, setSelectedLayerId, setSelectedLayerIds],
|
|
);
|
|
}
|
|
|
|
export function useImageCanvasAssetCanvasBridge({
|
|
assetPointerDragRef,
|
|
suppressAssetClickRef,
|
|
assets,
|
|
assetFolders,
|
|
layerCounterRef,
|
|
viewport,
|
|
canvasSize,
|
|
resolveAssetFolderId,
|
|
resolveCanvasPoint,
|
|
getCanvasDropPoint,
|
|
setAssetPointerDrag,
|
|
setActiveUploadFolderId,
|
|
setUploadDropTarget,
|
|
setHoveredLayerId,
|
|
updateAssetMoveDropFolder,
|
|
moveAssetToFolder,
|
|
captureCanvasHistory,
|
|
appendCanvasLayersWithResources,
|
|
selectSingleLayer,
|
|
addUploadedFiles,
|
|
}: UseImageCanvasAssetCanvasBridgeOptions) {
|
|
const addAssetLayer = useCallback(
|
|
(asset: EditorAsset, position?: CanvasPoint) => {
|
|
setActiveUploadFolderId(asset.folderId);
|
|
layerCounterRef.current += 1;
|
|
const nextLayer = createLayerFromAsset(
|
|
asset,
|
|
layerCounterRef.current,
|
|
viewport,
|
|
{
|
|
x: position?.x ?? canvasSize.width / 2,
|
|
y: position?.y ?? canvasSize.height / 2,
|
|
},
|
|
);
|
|
captureCanvasHistory();
|
|
appendCanvasLayersWithResources([nextLayer]);
|
|
selectSingleLayer(nextLayer.id);
|
|
setHoveredLayerId(null);
|
|
},
|
|
[
|
|
appendCanvasLayersWithResources,
|
|
canvasSize.height,
|
|
canvasSize.width,
|
|
captureCanvasHistory,
|
|
layerCounterRef,
|
|
selectSingleLayer,
|
|
setActiveUploadFolderId,
|
|
setHoveredLayerId,
|
|
viewport,
|
|
],
|
|
);
|
|
|
|
useImageCanvasAssetPointerDragBridge({
|
|
assetPointerDragRef,
|
|
suppressAssetClickRef,
|
|
assets,
|
|
resolveAssetFolderId,
|
|
resolveCanvasPoint,
|
|
setAssetPointerDrag,
|
|
setUploadDropTarget,
|
|
updateAssetMoveDropFolder,
|
|
moveAssetToFolder,
|
|
addAssetLayer,
|
|
});
|
|
|
|
const canvasDropWorkflow = useImageCanvasCanvasDropWorkflow({
|
|
assets,
|
|
assetFolders,
|
|
setUploadDropTarget,
|
|
updateAssetMoveDropFolder,
|
|
getCanvasDropPoint,
|
|
addAssetLayer,
|
|
addUploadedFiles,
|
|
});
|
|
|
|
return useMemo(
|
|
() => ({
|
|
addAssetLayer,
|
|
...canvasDropWorkflow,
|
|
}),
|
|
[addAssetLayer, canvasDropWorkflow],
|
|
);
|
|
}
|