From 015716945e86d72fcd31dca52bbd7b2935929d05 Mon Sep 17 00:00:00 2001 From: kdletters Date: Wed, 17 Jun 2026 13:14:36 +0800 Subject: [PATCH 1/3] =?UTF-8?q?=E6=8B=86=E5=88=86=E5=9B=BE=E7=89=87?= =?UTF-8?q?=E7=94=BB=E5=B8=83=E7=B4=A0=E6=9D=90=E5=85=A5=E7=94=BB=E5=B8=83?= =?UTF-8?q?=E6=A1=A5=E6=8E=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 新增 useImageCanvasAssetCanvasBridge 承载素材加入画布和画布 drop 桥接 新增 hook 单测覆盖素材建层、pointer drop 和删除素材清理图层 精简 ImageCanvasEditorView 中的素材到画布胶水 更新图片画布拆分计划和 TRACKING 浏览器回归记录 --- TRACKING.md | 1 + ...构】图片画布编辑器前端拆分计划-2026-06-17.md | 10 +- .../image-editor/ImageCanvasEditorView.tsx | 101 ++----- .../useImageCanvasAssetCanvasBridge.test.tsx | 252 ++++++++++++++++++ .../useImageCanvasAssetCanvasBridge.ts | 191 +++++++++++++ 5 files changed, 480 insertions(+), 75 deletions(-) create mode 100644 src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx create mode 100644 src/components/image-editor/useImageCanvasAssetCanvasBridge.ts diff --git a/TRACKING.md b/TRACKING.md index 485aa786..b6fe00a2 100644 --- a/TRACKING.md +++ b/TRACKING.md @@ -139,3 +139,4 @@ - 2026-06-17 前端拆分第二十一阶段:新增 `useImageCanvasKeyboardShortcuts`,把 Ctrl / Cmd + Z 撤销、Ctrl / Cmd + Shift + Z 重做、Shift 状态、Backspace / Delete 删除、Escape 关闭临时面板和 Space 临时抓手从主视图抽出;主视图继续注入图层删除、生成对话框、快速编辑和 chrome 面板 setter。新增 hook 单测覆盖输入框忽略快捷键、删除选中图层、删除生成占位、Escape 保留生成中面板、Space 和 Shift;主视图从 1337 行降至 1250 行。验证命令:`npm run test -- src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10003/editor/canvas` 未登录直接弹出 `账号入口` 且未抢跑 `/api/editor/*`,登录后 `/api/editor/assets/library` 和 `/api/editor/projects/recent` 为 200,`AI画布工具栏` 与 `画布面板入口` 可见,viewport 背景为 `rgb(248, 250, 252)` 且 `background-image: none`;按住 Space 从 `文字工具` 临时切到 `抓手工具`,松开恢复 `文字工具`;`画布背景设置` 点击 `暖灰` 后背景变为 `rgb(243, 240, 234)`;点击 `生成工具` 后 `Image Generator` 占位框、`生成图片` 对话框和 `AI画布工具栏` 同时可见,登录后控制台无前端 error。 - 2026-06-17 前端拆分第二十二阶段:新增 `useImageCanvasAssetPointerDragBridge`,把素材卡片 pointer 拖拽到画布 / 文件夹的全局监听、拖拽激活、画布 drop 提示、文件夹高亮、移动素材、加入画布和点击抑制从主视图抽出;主视图继续负责素材库事实、画布建层、历史和工程资源持久化。同步恢复 `账号入口` 认证弹窗 portal 渲染,避免全屏画布遮住登录弹窗;`画布背景设置` 调整为独立白色浮层,包含当前色、色域、色相、自定义颜色、预设网格、HEX 输入和恢复默认。验证命令:`npm run test -- src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 未登录打开直接显示 `账号入口`;登录临时开发账号后 `画布背景设置` 显示当前色、色域、色相、自定义颜色、预设、HEX 和恢复默认,点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`;上传图片进入 `项目素材`,真实 pointer 拖拽素材到画布后图层数从 0 到 1 且 `AI画布工具栏` 保持可见;新建 `SmokeFolder` 后真实 pointer 拖拽素材到文件夹,`项目素材` 变 0、`SmokeFolder` 变 1,素材执行移动而非拷贝。控制台仅有未登录 refresh 401,登录后编辑器 API 均为 200。 - 2026-06-17 前端拆分第二十三阶段:新增 `ImageCanvasOverlayModel`,把生成输入框锚定、图标素材生成面板宽度、选中图片工具栏边界、快速编辑面板和角色动画面板定位从主视图抽出为纯模型;主视图继续保留生成 / quick edit / 角色动画状态机和舞台编排。新增模型单测覆盖锚定优先级、生成中隐藏、icon 宽度、工具栏 clamp、quick edit / 角色动画边界和生成 dialog 模式识别;主视图从 1182 行降至 1133 行。验证命令:`npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run test -- src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasGenerationWorkflow.test.tsx`、`npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasGenerationWorkflow.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 未登录打开显示 `账号入口`,登录后 `画布背景设置` 点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,点击 `生成工具` 后 `Image Generator`、`生成图片` 对话框和 `AI画布工具栏` 均可见;素材库已有素材真实 pointer 拖入画布后图层数从 0 到 1,工具栏保持可见,截图留存于 `output/playwright/editor-overlay-model-regression-20260617.png`。 +- 2026-06-17 前端拆分第二十四阶段:新增 `useImageCanvasAssetCanvasBridge`,把删除素材清理画布图层、素材加入画布、素材 pointer 拖入画布 / 文件夹、画布 drag/drop 分流和文件拖入画布参数组装从主视图抽成素材到画布桥接 hook;主视图继续保留素材库事实、上传读取、工程资源持久化和历史触发。新增 hook 单测覆盖素材建层、pointer drop 入画布和删除素材清理关联图层 / 选中态;主视图从 1133 行降至 1086 行。验证命令:`npm run test -- src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 清空会话后未登录直接显示 `账号入口`;登录临时开发账号后 `画布背景设置` 是完整设置面板,包含当前色、色域、色相、自定义颜色、预设、HEX 和恢复默认,点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,HEX 输入 `#ffffff` 后变为白色,按 Escape 关闭面板;登录后上传图片请求 `/api/editor/assets` 返回 200,素材库出现上传素材,`AI画布工具栏` 保持可见。 diff --git a/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md b/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md index 2ca1b736..ca76110d 100644 --- a/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md +++ b/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md @@ -199,6 +199,14 @@ - 该模块用独立单测覆盖生成结果优先锚定、生成中 / 手动关闭隐藏输入框、icon 描述项宽度、图片工具栏 clamp、quick edit 和角色动画面板边界,以及画布生成 dialog 模式识别。 - 本阶段主视图从 1182 行降至 1133 行;下一步若继续拆分,可在该模型基础上抽更大的 `useImageCanvasStageController`,承接舞台派生状态、右键菜单和工具切换胶水。 +## 第二十四阶段模块 + +- `useImageCanvasAssetCanvasBridge.ts` + - 承载素材库与画布之间的桥接工作流:删除素材时清理关联画布图层、素材加入画布创建图层、素材 pointer 拖入画布 / 文件夹、画布区域 drag over / leave / drop 分流,以及拖拽文件上传到画布的参数组装。 + - 主视图继续掌握素材库事实、上传文件读取、工程资源持久化、历史捕获触发时机和实际 API 副作用;该 hook 只负责把已有素材 / 文件 drop 转成画布动作,不反向读取路由、登录态或项目数据。 + - 该 hook 用独立单测覆盖素材建层、pointer drop 入画布和删除素材清理关联图层 / 选中态;原有 pointer drag 和 canvas drop hook 单测继续保留,主视图 DOM 测试继续覆盖真实素材库拖入画布路径。 + - 本阶段主视图从 1133 行降至 1086 行;下一步可继续抽 `useImageCanvasStageController` 或顶栏视图,但应优先选择能收敛舞台派生状态和右键菜单胶水的深边界。 + ## 后续阶段 - 后续可继续选择更高内聚的交互 workflow 或持久化边界,不再把生成链路继续拆成浅层 wrapper。 @@ -206,7 +214,7 @@ ## 验证计划 -- `npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx` +- `npm run test -- src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx` - `npm run typecheck` - `npm run check:encoding` - `git diff --check` diff --git a/src/components/image-editor/ImageCanvasEditorView.tsx b/src/components/image-editor/ImageCanvasEditorView.tsx index b534a315..004c0739 100644 --- a/src/components/image-editor/ImageCanvasEditorView.tsx +++ b/src/components/image-editor/ImageCanvasEditorView.tsx @@ -21,9 +21,7 @@ import { import { ImageCanvasSidebarView } from './ImageCanvasSidebarView'; import { ImageCanvasStageView } from './ImageCanvasStageView'; import { - createLayerFromAsset, isGeneratedLayer, - isLayerLinkedToAsset, resolveContextMenuPosition, } from './ImageCanvasEditorModel'; import { @@ -46,15 +44,16 @@ import type { CanvasLayer, CanvasTool, CanvasViewport, - EditorAsset, ImageContextMenuState, } from './ImageCanvasEditorTypes'; import { useCanvasHistory } from './useCanvasHistory'; import { useCanvasGenerationDialogs } from './useCanvasGenerationDialogs'; import { useImageCanvasAssetLibrary } from './useImageCanvasAssetLibrary'; +import { + useImageCanvasAssetCanvasBridge, + useImageCanvasAssetLayerCleanup, +} from './useImageCanvasAssetCanvasBridge'; import { useImageCanvasAssetExportWorkflow } from './useImageCanvasAssetExportWorkflow'; -import { useImageCanvasAssetPointerDragBridge } from './useImageCanvasAssetPointerDragBridge'; -import { useImageCanvasCanvasDropWorkflow } from './useImageCanvasCanvasDropWorkflow'; import { useImageCanvasEditorChrome } from './useImageCanvasEditorChrome'; import { useImageCanvasGenerationWorkflow } from './useImageCanvasGenerationWorkflow'; import { useImageCanvasKeyboardShortcuts } from './useImageCanvasKeyboardShortcuts'; @@ -178,41 +177,12 @@ export function ImageCanvasEditorView() { toggleBackgroundSettings, toggleMinimap, } = useImageCanvasEditorChrome({ openEditorLoginModal }); - const removeCanvasLayersLinkedToAssets = 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], - ); + const removeCanvasLayersLinkedToAssets = useImageCanvasAssetLayerCleanup({ + layers, + setLayers, + setSelectedLayerId, + setSelectedLayerIds, + }); const { assetFolders, setAssetFolders, @@ -664,53 +634,36 @@ export function ImageCanvasEditorView() { }; }, []); - const addAssetLayer = ( - asset: EditorAsset, - position?: { x: number; y: number }, - ) => { - 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); - }; - - useImageCanvasAssetPointerDragBridge({ + const { + addAssetLayer, + handleCanvasDragOver, + handleCanvasDragLeave, + handleCanvasDrop, + } = useImageCanvasAssetCanvasBridge({ assetPointerDragRef, suppressAssetClickRef, assets, + assetFolders, + layerCounterRef, + viewport, + canvasSize, resolveAssetFolderId, resolveCanvasPoint, + getCanvasDropPoint, setAssetPointerDrag, + setActiveUploadFolderId, setUploadDropTarget, + setHoveredLayerId, updateAssetMoveDropFolder, moveAssetToFolder, - addAssetLayer, + captureCanvasHistory, + appendCanvasLayersWithResources, + selectSingleLayer, + addUploadedFiles, }); deleteLayerByIdRef.current = deleteLayerById; - const { handleCanvasDragOver, handleCanvasDragLeave, handleCanvasDrop } = - useImageCanvasCanvasDropWorkflow({ - assets, - assetFolders, - setUploadDropTarget, - updateAssetMoveDropFolder, - getCanvasDropPoint, - addAssetLayer, - addUploadedFiles, - }); - const handleCanvasContextMenu = (event: ReactMouseEvent) => { event.preventDefault(); event.stopPropagation(); diff --git a/src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx b/src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx new file mode 100644 index 00000000..6b38dcfe --- /dev/null +++ b/src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx @@ -0,0 +1,252 @@ +/* @vitest-environment jsdom */ + +import { act, render, screen } from '@testing-library/react'; +import { useRef, useState } from 'react'; +import { describe, expect, it, vi } from 'vitest'; + +import type { + AssetPointerDragState, + CanvasLayer, + EditorAsset, + EditorAssetFolder, +} from './ImageCanvasEditorTypes'; +import { + useImageCanvasAssetCanvasBridge, + useImageCanvasAssetLayerCleanup, +} from './useImageCanvasAssetCanvasBridge'; + +const defaultAsset: EditorAsset = { + id: 'asset-1', + label: '素材一', + src: 'data:image/png;base64,asset-1', + width: 320, + height: 240, + folderId: 'project', + sourceKind: 'uploaded', + sourceType: 'uploaded', + persisted: true, + assetObjectId: 'asset-object-1', +}; + +const defaultFolder: EditorAssetFolder = { + id: 'project', + label: '项目素材', + collapsed: false, + systemDefault: true, + persisted: true, +}; + +function createLayer(overrides: Partial = {}): CanvasLayer { + return { + id: 'layer-asset-1', + resourceId: 'asset-object-1', + title: '素材一', + src: 'data:image/png;base64,asset-1', + x: 10, + y: 20, + width: 320, + height: 240, + originalWidth: 320, + originalHeight: 240, + zIndex: 1, + sourceType: 'uploaded', + sourceAssetId: 'asset-1', + ...overrides, + }; +} + +function dispatchPointerEvent( + type: string, + init: MouseEventInit & { pointerId: number }, +) { + const event = new MouseEvent(type, { + bubbles: true, + cancelable: true, + ...init, + }); + Object.defineProperty(event, 'pointerId', { value: init.pointerId }); + window.dispatchEvent(event); +} + +function AssetCanvasBridgeHarness({ + asset = defaultAsset, + resolveCanvasPoint = vi.fn(() => null), + appendCanvasLayersWithResources = vi.fn(), + captureCanvasHistory = vi.fn(), + selectSingleLayer = vi.fn(), +}: { + asset?: EditorAsset; + resolveCanvasPoint?: (clientX: number, clientY: number) => { + x: number; + y: number; + } | null; + appendCanvasLayersWithResources?: (nextLayers: CanvasLayer[]) => void; + captureCanvasHistory?: () => void; + selectSingleLayer?: (layerId: string | null) => void; +}) { + const assetPointerDragRef = useRef({ + assetId: asset.id, + pointerId: 7, + startClientX: 10, + startClientY: 10, + currentClientX: 40, + currentClientY: 40, + active: true, + dropFolderId: null, + }); + const suppressAssetClickRef = useRef(false); + const layerCounterRef = useRef(0); + const [activeUploadFolderId, setActiveUploadFolderId] = useState('project'); + const [hoveredLayerId, setHoveredLayerId] = useState('hovered'); + const [assetPointerDrag, setAssetPointerDrag] = + useState(assetPointerDragRef.current); + const [uploadDropTarget, setUploadDropTarget] = useState< + 'canvas' | 'assets' | null + >(null); + + const bridge = useImageCanvasAssetCanvasBridge({ + assetPointerDragRef, + suppressAssetClickRef, + assets: [asset], + assetFolders: [defaultFolder], + layerCounterRef, + viewport: { x: 10, y: 20, scale: 2 }, + canvasSize: { width: 900, height: 640 }, + resolveAssetFolderId: () => null, + resolveCanvasPoint, + getCanvasDropPoint: (clientX, clientY) => ({ + x: clientX - 10, + y: clientY - 20, + }), + setAssetPointerDrag, + setActiveUploadFolderId, + setUploadDropTarget, + setHoveredLayerId, + updateAssetMoveDropFolder: vi.fn(), + moveAssetToFolder: vi.fn(), + captureCanvasHistory, + appendCanvasLayersWithResources, + selectSingleLayer, + addUploadedFiles: vi.fn(), + }); + + return ( +
+ + {activeUploadFolderId} + {hoveredLayerId ?? '-'} + {assetPointerDrag ? 'dragging' : 'none'} + {uploadDropTarget ?? '-'} +
+ ); +} + +function AssetCleanupHarness({ + deletedAssets = [], +}: { + deletedAssets?: EditorAsset[]; +}) { + const [layers, setLayers] = useState([ + createLayer({ id: 'linked', sourceAssetId: 'asset-1' }), + createLayer({ + id: 'kept', + resourceId: 'asset-object-2', + src: 'data:image/png;base64,asset-2', + sourceAssetId: 'asset-2', + }), + ]); + const [selectedLayerId, setSelectedLayerId] = useState('linked'); + const [selectedLayerIds, setSelectedLayerIds] = useState(['linked', 'kept']); + const cleanup = useImageCanvasAssetLayerCleanup({ + layers, + setLayers, + setSelectedLayerId, + setSelectedLayerIds, + }); + + return ( +
+ + {layers.map((layer) => layer.id).join(',')} + {selectedLayerId ?? '-'} + {selectedLayerIds.join(',')} +
+ ); +} + +describe('useImageCanvasAssetCanvasBridge', () => { + it('creates a canvas layer from an asset and clears hover state', () => { + const appendCanvasLayersWithResources = vi.fn(); + const captureCanvasHistory = vi.fn(); + const selectSingleLayer = vi.fn(); + render( + , + ); + + act(() => { + screen.getByRole('button', { name: '添加到画布' }).click(); + }); + + expect(captureCanvasHistory).toHaveBeenCalledTimes(1); + expect(appendCanvasLayersWithResources).toHaveBeenCalledWith([ + expect.objectContaining({ + id: 'layer-asset-1-1', + sourceAssetId: 'asset-1', + x: 94, + y: 64, + width: 320, + height: 240, + }), + ]); + expect(selectSingleLayer).toHaveBeenCalledWith('layer-asset-1-1'); + expect(screen.getByTestId('folder').textContent).toBe('project'); + expect(screen.getByTestId('hover').textContent).toBe('-'); + }); + + it('delegates active pointer drops over the canvas into the layer path', () => { + const appendCanvasLayersWithResources = vi.fn(); + render( + ({ x: 480, y: 640 })} + appendCanvasLayersWithResources={appendCanvasLayersWithResources} + />, + ); + + act(() => { + dispatchPointerEvent('pointerup', { + pointerId: 7, + clientX: 48, + clientY: 64, + }); + }); + + expect(appendCanvasLayersWithResources).toHaveBeenCalledWith([ + expect.objectContaining({ sourceAssetId: 'asset-1' }), + ]); + expect(screen.getByTestId('drag').textContent).toBe('none'); + expect(screen.getByTestId('drop-target').textContent).toBe('-'); + }); + + it('removes canvas layers linked to deleted assets and keeps unrelated selection', () => { + render(); + + act(() => { + screen.getByRole('button', { name: '清理素材' }).click(); + }); + + expect(screen.getByTestId('layers').textContent).toBe('kept'); + expect(screen.getByTestId('selected').textContent).toBe('-'); + expect(screen.getByTestId('selected-many').textContent).toBe('kept'); + }); +}); diff --git a/src/components/image-editor/useImageCanvasAssetCanvasBridge.ts b/src/components/image-editor/useImageCanvasAssetCanvasBridge.ts new file mode 100644 index 00000000..fd93a600 --- /dev/null +++ b/src/components/image-editor/useImageCanvasAssetCanvasBridge.ts @@ -0,0 +1,191 @@ +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>; + setSelectedLayerId: Dispatch>; + setSelectedLayerIds: Dispatch>; +}; + +type UseImageCanvasAssetCanvasBridgeOptions = { + assetPointerDragRef: RefObject; + suppressAssetClickRef: RefObject; + assets: EditorAsset[]; + assetFolders: EditorAssetFolder[]; + layerCounterRef: MutableRefObject; + 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>; + setActiveUploadFolderId: Dispatch>; + setUploadDropTarget: Dispatch>; + setHoveredLayerId: Dispatch>; + 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], + ); +} From 1c92db19c17a5ae88ae523f6b46ba44220738ccd Mon Sep 17 00:00:00 2001 From: kdletters Date: Wed, 17 Jun 2026 13:39:48 +0800 Subject: [PATCH 2/3] =?UTF-8?q?=E6=8B=86=E5=88=86=E5=9B=BE=E7=89=87?= =?UTF-8?q?=E7=94=BB=E5=B8=83=E8=88=9E=E5=8F=B0=E6=8E=A7=E5=88=B6=E5=B1=82?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 新增 ImageCanvasStageControllerModel 承载舞台派生状态和右键菜单模型 新增 useImageCanvasStageController 收口清空焦点和右键菜单处理 精简 ImageCanvasEditorView 的舞台控制胶水 更新图片画布拆分计划和 TRACKING 验证记录 --- TRACKING.md | 1 + ...构】图片画布编辑器前端拆分计划-2026-06-17.md | 18 +- .../image-editor/ImageCanvasEditorView.tsx | 149 ++++------------- .../ImageCanvasStageControllerModel.test.ts | 155 ++++++++++++++++++ .../ImageCanvasStageControllerModel.ts | 148 +++++++++++++++++ .../useImageCanvasStageController.test.tsx | 152 +++++++++++++++++ .../useImageCanvasStageController.ts | 139 ++++++++++++++++ 7 files changed, 638 insertions(+), 124 deletions(-) create mode 100644 src/components/image-editor/ImageCanvasStageControllerModel.test.ts create mode 100644 src/components/image-editor/ImageCanvasStageControllerModel.ts create mode 100644 src/components/image-editor/useImageCanvasStageController.test.tsx create mode 100644 src/components/image-editor/useImageCanvasStageController.ts diff --git a/TRACKING.md b/TRACKING.md index b6fe00a2..c2c56134 100644 --- a/TRACKING.md +++ b/TRACKING.md @@ -140,3 +140,4 @@ - 2026-06-17 前端拆分第二十二阶段:新增 `useImageCanvasAssetPointerDragBridge`,把素材卡片 pointer 拖拽到画布 / 文件夹的全局监听、拖拽激活、画布 drop 提示、文件夹高亮、移动素材、加入画布和点击抑制从主视图抽出;主视图继续负责素材库事实、画布建层、历史和工程资源持久化。同步恢复 `账号入口` 认证弹窗 portal 渲染,避免全屏画布遮住登录弹窗;`画布背景设置` 调整为独立白色浮层,包含当前色、色域、色相、自定义颜色、预设网格、HEX 输入和恢复默认。验证命令:`npm run test -- src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 未登录打开直接显示 `账号入口`;登录临时开发账号后 `画布背景设置` 显示当前色、色域、色相、自定义颜色、预设、HEX 和恢复默认,点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`;上传图片进入 `项目素材`,真实 pointer 拖拽素材到画布后图层数从 0 到 1 且 `AI画布工具栏` 保持可见;新建 `SmokeFolder` 后真实 pointer 拖拽素材到文件夹,`项目素材` 变 0、`SmokeFolder` 变 1,素材执行移动而非拷贝。控制台仅有未登录 refresh 401,登录后编辑器 API 均为 200。 - 2026-06-17 前端拆分第二十三阶段:新增 `ImageCanvasOverlayModel`,把生成输入框锚定、图标素材生成面板宽度、选中图片工具栏边界、快速编辑面板和角色动画面板定位从主视图抽出为纯模型;主视图继续保留生成 / quick edit / 角色动画状态机和舞台编排。新增模型单测覆盖锚定优先级、生成中隐藏、icon 宽度、工具栏 clamp、quick edit / 角色动画边界和生成 dialog 模式识别;主视图从 1182 行降至 1133 行。验证命令:`npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run test -- src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasGenerationWorkflow.test.tsx`、`npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasGenerationWorkflow.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 未登录打开显示 `账号入口`,登录后 `画布背景设置` 点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,点击 `生成工具` 后 `Image Generator`、`生成图片` 对话框和 `AI画布工具栏` 均可见;素材库已有素材真实 pointer 拖入画布后图层数从 0 到 1,工具栏保持可见,截图留存于 `output/playwright/editor-overlay-model-regression-20260617.png`。 - 2026-06-17 前端拆分第二十四阶段:新增 `useImageCanvasAssetCanvasBridge`,把删除素材清理画布图层、素材加入画布、素材 pointer 拖入画布 / 文件夹、画布 drag/drop 分流和文件拖入画布参数组装从主视图抽成素材到画布桥接 hook;主视图继续保留素材库事实、上传读取、工程资源持久化和历史触发。新增 hook 单测覆盖素材建层、pointer drop 入画布和删除素材清理关联图层 / 选中态;主视图从 1133 行降至 1086 行。验证命令:`npm run test -- src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 清空会话后未登录直接显示 `账号入口`;登录临时开发账号后 `画布背景设置` 是完整设置面板,包含当前色、色域、色相、自定义颜色、预设、HEX 和恢复默认,点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,HEX 输入 `#ffffff` 后变为白色,按 Escape 关闭面板;登录后上传图片请求 `/api/editor/assets` 返回 200,素材库出现上传素材,`AI画布工具栏` 保持可见。 +- 2026-06-17 前端拆分第二十五阶段:新增 `ImageCanvasStageControllerModel` 和 `useImageCanvasStageController`,把舞台派生状态、生成 / 选中浮层位置、右键菜单目标、空白画布右键、图层右键和清空画布焦点从主视图抽出;主视图继续保留工具切换、上传 / 生成入口、图层命令、项目持久化和舞台 pointer 状态机。新增模型和 hook 单测覆盖选中 / 生成锚定、右键菜单位置、显示 / 解锁判断、清空焦点和空白 / 图层右键菜单;主视图从 1086 行降至 993 行。验证命令:`npm run test -- src/components/image-editor/ImageCanvasStageControllerModel.test.ts src/components/image-editor/useImageCanvasStageController.test.tsx`、`npm run test -- src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasUploadWorkflow.test.tsx src/routing/appPageRoutes.test.ts`、`npm run typecheck`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 清空会话后未登录直接显示 `账号入口`,关闭登录后 `画布背景色` 打开完整 `画布背景设置` dialog,包含色相、自定义颜色、预设、HEX 和恢复默认;点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,`AI画布工具栏` 保持可见,截图留存于 `output/playwright/editor-stage-controller-smoke-20260617.png`。 diff --git a/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md b/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md index ca76110d..c65c0432 100644 --- a/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md +++ b/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md @@ -205,16 +205,28 @@ - 承载素材库与画布之间的桥接工作流:删除素材时清理关联画布图层、素材加入画布创建图层、素材 pointer 拖入画布 / 文件夹、画布区域 drag over / leave / drop 分流,以及拖拽文件上传到画布的参数组装。 - 主视图继续掌握素材库事实、上传文件读取、工程资源持久化、历史捕获触发时机和实际 API 副作用;该 hook 只负责把已有素材 / 文件 drop 转成画布动作,不反向读取路由、登录态或项目数据。 - 该 hook 用独立单测覆盖素材建层、pointer drop 入画布和删除素材清理关联图层 / 选中态;原有 pointer drag 和 canvas drop hook 单测继续保留,主视图 DOM 测试继续覆盖真实素材库拖入画布路径。 - - 本阶段主视图从 1133 行降至 1086 行;下一步可继续抽 `useImageCanvasStageController` 或顶栏视图,但应优先选择能收敛舞台派生状态和右键菜单胶水的深边界。 + - 本阶段主视图从 1133 行降至 1086 行;下一步可继续抽顶栏视图或更高内聚的舞台控制层,但应优先选择能收敛真实状态规则的深边界。 + +## 第二十五阶段模块 + +- `ImageCanvasStageControllerModel.ts` + - 承载舞台派生状态和右键菜单模型:选中图层、生成对象锚点、生成输入框位置、选中浮动工具栏位置、图片菜单图层、右键菜单目标图层,以及显示 / 解锁菜单文案判断。 + - 该模型复用既有图层命令模型与浮层定位模型,不重新实现坐标公式,避免拆分后出现第二套右键目标和浮层定位规则。 + - 新增单测覆盖生成锚定、选中工具栏位置、右键目标集合、显示 / 解锁判断和菜单位置限制。 + +- `useImageCanvasStageController.ts` + - 承载舞台控制胶水:清空画布焦点、空白画布右键菜单和图层右键菜单处理。 + - 主视图继续保留工具切换、上传 / 生成入口、图层命令、项目持久化和舞台 pointer 状态机,避免把跨工作流动作塞进单个 hook。 + - 本阶段主视图从 1086 行降至 993 行;后续若继续拆分,应优先考虑顶栏 / 项目标题区域或把侧栏图层右键入口并入同一舞台菜单控制层。 ## 后续阶段 - 后续可继续选择更高内聚的交互 workflow 或持久化边界,不再把生成链路继续拆成浅层 wrapper。 -- 右键菜单定位、工程资源持久化、舞台派生状态和历史捕获仍在主视图编排,拆分前需要先确认不会破坏多生成对象同时存在、完成时读取最新占位框、素材拖拽上传位置和角色动画优先传 `objectKey` 的历史保护规则。 +- 工程资源持久化、工具切换和历史捕获仍在主视图编排,拆分前需要先确认不会破坏多生成对象同时存在、完成时读取最新占位框、素材拖拽上传位置和角色动画优先传 `objectKey` 的历史保护规则。 ## 验证计划 -- `npm run test -- src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx` +- `npm run test -- src/components/image-editor/ImageCanvasStageControllerModel.test.ts src/components/image-editor/useImageCanvasStageController.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasUploadWorkflow.test.tsx src/routing/appPageRoutes.test.ts` - `npm run typecheck` - `npm run check:encoding` - `git diff --check` diff --git a/src/components/image-editor/ImageCanvasEditorView.tsx b/src/components/image-editor/ImageCanvasEditorView.tsx index 004c0739..ed9a430a 100644 --- a/src/components/image-editor/ImageCanvasEditorView.tsx +++ b/src/components/image-editor/ImageCanvasEditorView.tsx @@ -1,12 +1,5 @@ import { Check, ChevronLeft, Download, Pencil, X } from 'lucide-react'; -import { - type MouseEvent as ReactMouseEvent, - useCallback, - useEffect, - useMemo, - useRef, - useState, -} from 'react'; +import { useCallback, useEffect, useMemo, useRef, useState } from 'react'; import { PlatformStatusMessage } from '../common/PlatformStatusMessage'; import { PlatformTextField } from '../common/PlatformTextField'; @@ -14,29 +7,14 @@ import { useAuthUi } from '../auth/AuthUiContext'; import { EditorIconButton } from './ImageCanvasEditorPrimitives'; import { ImageCanvasGenerationComposerView } from './ImageCanvasGenerationComposerView'; import { ImageCanvasMetadataModalView } from './ImageCanvasMetadataModalView'; -import { - getCanvasLayersByIds, - resolveContextTargetLayerIds, -} from './ImageCanvasLayerCommandModel'; import { ImageCanvasSidebarView } from './ImageCanvasSidebarView'; import { ImageCanvasStageView } from './ImageCanvasStageView'; -import { - isGeneratedLayer, - resolveContextMenuPosition, -} from './ImageCanvasEditorModel'; -import { - getGenerationFrameAriaLabel, - getGenerationFrameLabel, - getLayerKindLabel, -} from './ImageCanvasGenerationModel'; +import { resolveContextMenuPosition } from './ImageCanvasEditorModel'; import { isCanvasGenerationComposerVisible, resolveCharacterAnimationPanelStyle, - resolveGenerationAnchor, - resolveGenerationComposerStyle, resolveIconComposerStyle, resolveQuickEditPanelStyle, - resolveSelectedToolbarStyle, } from './ImageCanvasOverlayModel'; import type { AssetPointerDragState, @@ -59,6 +37,7 @@ import { useImageCanvasGenerationWorkflow } from './useImageCanvasGenerationWork import { useImageCanvasKeyboardShortcuts } from './useImageCanvasKeyboardShortcuts'; import { useImageCanvasLayerCommands } from './useImageCanvasLayerCommands'; import { useImageCanvasProjectPersistence } from './useImageCanvasProjectPersistence'; +import { useImageCanvasStageController } from './useImageCanvasStageController'; import { useImageCanvasStageInteractions } from './useImageCanvasStageInteractions'; import { useImageCanvasUploadWorkflow } from './useImageCanvasUploadWorkflow'; import { @@ -257,54 +236,6 @@ export function ImageCanvasEditorView() { } = useCanvasGenerationDialogs({ onActivate: handleActivateCanvasGenerationDialog, }); - const selectedLayer = useMemo( - () => layers.find((layer) => layer.id === selectedLayerId) ?? null, - [layers, selectedLayerId], - ); - const selectedLayerCount = selectedLayerIds.length; - const hasMultipleSelectedLayers = selectedLayerCount > 1; - const activeGenerationLayer = useMemo( - () => - activeCanvasGenerationDialog?.generatedLayerId - ? (layers.find( - (layer) => - layer.id === activeCanvasGenerationDialog.generatedLayerId, - ) ?? null) - : null, - [activeCanvasGenerationDialog, layers], - ); - const generationAnchor = activeCanvasGenerationDialog - ? resolveGenerationAnchor({ - dialog: activeCanvasGenerationDialog, - generatedLayer: activeGenerationLayer, - }) - : null; - const generationComposerStyle = resolveGenerationComposerStyle({ - dialog: activeCanvasGenerationDialog, - anchor: generationAnchor, - viewport, - }); - const selectedToolbarStyle = resolveSelectedToolbarStyle({ - selectedLayer, - viewport, - canvasSize, - }); - const imageContextMenuLayer = imageContextMenu - ? (layers.find((layer) => layer.id === imageContextMenu.layerId) ?? null) - : null; - const getContextTargetLayerIds = useCallback( - (menu: CanvasContextMenuState | null = contextMenu) => - resolveContextTargetLayerIds(menu, selectedLayerIdsRef.current), - [contextMenu], - ); - const contextTargetIds = getContextTargetLayerIds(contextMenu); - const contextTargetLayers = getCanvasLayersByIds(layers, contextTargetIds); - const contextShouldShowLayer = contextTargetLayers.some( - (layer) => layer.hidden, - ); - const contextShouldUnlockLayer = contextTargetLayers.some( - (layer) => layer.locked, - ); const canvasHistoryRefs = useMemo( () => ({ layersRef, @@ -463,6 +394,31 @@ export function ImageCanvasEditorView() { closeGenerateComposer, clearDeletedLayerGenerationState, } = generationWorkflow; + const { + selectedLayer, + generationComposerStyle, + selectedToolbarStyle, + imageContextMenuLayer, + contextShouldShowLayer, + contextShouldUnlockLayer, + clearCanvasFocus, + handleCanvasContextMenu, + handleLayerContextMenu, + } = useImageCanvasStageController({ + layers, + selectedLayerId, + selectedLayerIds, + activeCanvasGenerationDialog, + imageContextMenu, + setImageContextMenu, + contextMenu, + setContextMenu, + viewport, + canvasSize, + selectSingleLayer, + hideGeneratedLayerPanelAfterBlur, + getCanvasPointFromClient, + }); const iconComposerStyle = resolveIconComposerStyle({ dialog: activeCanvasGenerationDialog, composerStyle: generationComposerStyle, @@ -540,13 +496,6 @@ export function ImageCanvasEditorView() { appendCanvasLayersWithResources, selectSingleLayer, }); - - const clearCanvasFocus = useCallback(() => { - selectSingleLayer(null); - hideGeneratedLayerPanelAfterBlur(); - setImageContextMenu(null); - setContextMenu(null); - }, [hideGeneratedLayerPanelAfterBlur, selectSingleLayer]); const { canvasMarquee, isPanning, @@ -664,48 +613,6 @@ export function ImageCanvasEditorView() { deleteLayerByIdRef.current = deleteLayerById; - const handleCanvasContextMenu = (event: ReactMouseEvent) => { - event.preventDefault(); - event.stopPropagation(); - const position = resolveContextMenuPosition( - event.clientX, - event.clientY, - 'blank', - ); - setImageContextMenu(null); - setContextMenu({ - kind: 'blank', - ...position, - canvasPoint: getCanvasPointFromClient(event.clientX, event.clientY), - }); - }; - - const handleLayerContextMenu = ( - event: ReactMouseEvent, - layer: CanvasLayer, - ) => { - event.preventDefault(); - event.stopPropagation(); - if (!selectedLayerIds.includes(layer.id)) { - selectSingleLayer(layer.id); - } - const position = resolveContextMenuPosition( - event.clientX, - event.clientY, - 'layer', - ); - setContextMenu({ - kind: 'layer', - layerId: layer.id, - ...position, - canvasPoint: getCanvasPointFromClient(event.clientX, event.clientY), - }); - setImageContextMenu({ - layerId: layer.id, - ...position, - }); - }; - const switchTool = (tool: CanvasTool) => { clearActiveInteraction(); if (tool === 'upload') { diff --git a/src/components/image-editor/ImageCanvasStageControllerModel.test.ts b/src/components/image-editor/ImageCanvasStageControllerModel.test.ts new file mode 100644 index 00000000..ab134119 --- /dev/null +++ b/src/components/image-editor/ImageCanvasStageControllerModel.test.ts @@ -0,0 +1,155 @@ +/* @vitest-environment jsdom */ + +import { describe, expect, it } from 'vitest'; + +import type { + CanvasContextMenuState, + CanvasGenerationDialogState, + CanvasLayer, +} from './ImageCanvasEditorTypes'; +import { + createBlankCanvasContextMenu, + createLayerCanvasContextMenus, + resolveImageCanvasStageControllerModel, +} from './ImageCanvasStageControllerModel'; + +function createLayer(overrides: Partial = {}): CanvasLayer { + const id = overrides.id ?? 'layer-a'; + return { + id, + resourceId: `resource-${id}`, + title: id, + src: `data:image/png;base64,${id}`, + x: 100, + y: 80, + width: 240, + height: 120, + originalWidth: 240, + originalHeight: 120, + zIndex: 1, + sourceType: 'uploaded', + ...overrides, + }; +} + +function createDialog( + overrides: Partial = {}, +): CanvasGenerationDialogState { + return { + id: 'dialog-a', + mode: 'generate', + prompt: '', + status: 'idle', + placeholder: { + x: 300, + y: 200, + width: 360, + height: 260, + originalWidth: 1024, + originalHeight: 1024, + }, + ...overrides, + }; +} + +describe('ImageCanvasStageControllerModel', () => { + it('derives selected layer, generation anchor, and overlay positions', () => { + const selectedLayer = createLayer({ id: 'selected', x: 40, y: 60 }); + const generatedLayer = createLayer({ + id: 'generated', + x: 200, + y: 160, + width: 320, + height: 180, + }); + const dialog = createDialog({ generatedLayerId: generatedLayer.id }); + + const model = resolveImageCanvasStageControllerModel({ + layers: [selectedLayer, generatedLayer], + selectedLayerId: selectedLayer.id, + selectedLayerIds: [selectedLayer.id, generatedLayer.id], + activeCanvasGenerationDialog: dialog, + imageContextMenu: { layerId: generatedLayer.id, x: 0, y: 0 }, + contextMenu: null, + viewport: { x: 10, y: 20, scale: 2 }, + canvasSize: { width: 900, height: 640 }, + }); + + expect(model.selectedLayer).toBe(selectedLayer); + expect(model.selectedLayerCount).toBe(2); + expect(model.hasMultipleSelectedLayers).toBe(true); + expect(model.activeGenerationLayer).toBe(generatedLayer); + expect(model.generationAnchor).toBe(generatedLayer); + expect(model.generationComposerStyle).toEqual({ left: 730, top: 710 }); + expect(model.selectedToolbarStyle).toEqual({ left: 330, top: 128 }); + expect(model.imageContextMenuLayer).toBe(generatedLayer); + }); + + it('derives context target state from a layer menu and current selection', () => { + const visibleLayer = createLayer({ id: 'visible', locked: true }); + const hiddenLayer = createLayer({ id: 'hidden', hidden: true }); + const contextMenu: CanvasContextMenuState = { + kind: 'layer', + layerId: hiddenLayer.id, + x: 20, + y: 30, + canvasPoint: { x: 40, y: 50 }, + }; + + const model = resolveImageCanvasStageControllerModel({ + layers: [visibleLayer, hiddenLayer], + selectedLayerId: visibleLayer.id, + selectedLayerIds: [visibleLayer.id, hiddenLayer.id], + activeCanvasGenerationDialog: null, + imageContextMenu: null, + contextMenu, + viewport: { x: 0, y: 0, scale: 1 }, + canvasSize: { width: 900, height: 640 }, + }); + + expect(model.contextTargetIds).toEqual([visibleLayer.id, hiddenLayer.id]); + expect(model.contextTargetLayers).toEqual([visibleLayer, hiddenLayer]); + expect(model.contextShouldShowLayer).toBe(true); + expect(model.contextShouldUnlockLayer).toBe(true); + }); + + it('creates clamped blank and layer context menus with canvas points', () => { + window.innerWidth = 320; + window.innerHeight = 240; + + expect( + createBlankCanvasContextMenu({ + clientX: 1000, + clientY: 1000, + canvasPoint: { x: 12, y: 16 }, + }), + ).toEqual({ + kind: 'blank', + x: 124, + y: 56, + canvasPoint: { x: 12, y: 16 }, + }); + + expect( + createLayerCanvasContextMenus({ + clientX: 1000, + clientY: 1000, + layerId: 'layer-a', + canvasPoint: { x: 20, y: 30 }, + }), + ).toEqual({ + contextMenu: { + kind: 'layer', + layerId: 'layer-a', + x: 124, + y: 8, + canvasPoint: { x: 20, y: 30 }, + }, + imageContextMenu: { + layerId: 'layer-a', + x: 124, + y: 8, + }, + }); + }); +}); diff --git a/src/components/image-editor/ImageCanvasStageControllerModel.ts b/src/components/image-editor/ImageCanvasStageControllerModel.ts new file mode 100644 index 00000000..edbd1d96 --- /dev/null +++ b/src/components/image-editor/ImageCanvasStageControllerModel.ts @@ -0,0 +1,148 @@ +import { + getCanvasLayersByIds, + resolveContextTargetLayerIds, +} from './ImageCanvasLayerCommandModel'; +import { + resolveContextMenuPosition, +} from './ImageCanvasEditorModel'; +import type { + CanvasContextMenuState, + CanvasGenerationDialogState, + CanvasLayer, + CanvasViewport, + ImageContextMenuState, +} from './ImageCanvasEditorTypes'; +import { + resolveGenerationAnchor, + resolveGenerationComposerStyle, + resolveSelectedToolbarStyle, +} from './ImageCanvasOverlayModel'; + +export type ImageCanvasStageControllerModel = { + selectedLayer: CanvasLayer | null; + selectedLayerCount: number; + hasMultipleSelectedLayers: boolean; + activeGenerationLayer: CanvasLayer | null; + generationAnchor: CanvasLayer | CanvasGenerationDialogState['placeholder'] | null; + generationComposerStyle: ReturnType; + selectedToolbarStyle: ReturnType; + imageContextMenuLayer: CanvasLayer | null; + contextTargetIds: string[]; + contextTargetLayers: CanvasLayer[]; + contextShouldShowLayer: boolean; + contextShouldUnlockLayer: boolean; +}; + +type ResolveImageCanvasStageControllerModelOptions = { + layers: CanvasLayer[]; + selectedLayerId: string | null; + selectedLayerIds: string[]; + activeCanvasGenerationDialog: CanvasGenerationDialogState | null; + imageContextMenu: ImageContextMenuState | null; + contextMenu: CanvasContextMenuState | null; + viewport: CanvasViewport; + canvasSize: { width: number; height: number }; +}; + +export function resolveImageCanvasStageControllerModel({ + layers, + selectedLayerId, + selectedLayerIds, + activeCanvasGenerationDialog, + imageContextMenu, + contextMenu, + viewport, + canvasSize, +}: ResolveImageCanvasStageControllerModelOptions): ImageCanvasStageControllerModel { + const selectedLayer = + layers.find((layer) => layer.id === selectedLayerId) ?? null; + const selectedLayerCount = selectedLayerIds.length; + const activeGenerationLayer = + activeCanvasGenerationDialog?.generatedLayerId + ? (layers.find( + (layer) => layer.id === activeCanvasGenerationDialog.generatedLayerId, + ) ?? null) + : null; + const generationAnchor = activeCanvasGenerationDialog + ? resolveGenerationAnchor({ + dialog: activeCanvasGenerationDialog, + generatedLayer: activeGenerationLayer, + }) + : null; + const imageContextMenuLayer = imageContextMenu + ? (layers.find((layer) => layer.id === imageContextMenu.layerId) ?? null) + : null; + const contextTargetIds = resolveContextTargetLayerIds( + contextMenu, + selectedLayerIds, + ); + const contextTargetLayers = getCanvasLayersByIds(layers, contextTargetIds); + + return { + selectedLayer, + selectedLayerCount, + hasMultipleSelectedLayers: selectedLayerCount > 1, + activeGenerationLayer, + generationAnchor, + generationComposerStyle: resolveGenerationComposerStyle({ + dialog: activeCanvasGenerationDialog, + anchor: generationAnchor, + viewport, + }), + selectedToolbarStyle: resolveSelectedToolbarStyle({ + selectedLayer, + viewport, + canvasSize, + }), + imageContextMenuLayer, + contextTargetIds, + contextTargetLayers, + contextShouldShowLayer: contextTargetLayers.some((layer) => layer.hidden), + contextShouldUnlockLayer: contextTargetLayers.some((layer) => layer.locked), + }; +} + +export function createBlankCanvasContextMenu({ + clientX, + clientY, + canvasPoint, +}: { + clientX: number; + clientY: number; + canvasPoint: { x: number; y: number }; +}): CanvasContextMenuState { + return { + kind: 'blank', + ...resolveContextMenuPosition(clientX, clientY, 'blank'), + canvasPoint, + }; +} + +export function createLayerCanvasContextMenus({ + clientX, + clientY, + layerId, + canvasPoint, +}: { + clientX: number; + clientY: number; + layerId: string; + canvasPoint: { x: number; y: number }; +}): { + contextMenu: CanvasContextMenuState; + imageContextMenu: ImageContextMenuState; +} { + const position = resolveContextMenuPosition(clientX, clientY, 'layer'); + return { + contextMenu: { + kind: 'layer', + layerId, + ...position, + canvasPoint, + }, + imageContextMenu: { + layerId, + ...position, + }, + }; +} diff --git a/src/components/image-editor/useImageCanvasStageController.test.tsx b/src/components/image-editor/useImageCanvasStageController.test.tsx new file mode 100644 index 00000000..a70f3231 --- /dev/null +++ b/src/components/image-editor/useImageCanvasStageController.test.tsx @@ -0,0 +1,152 @@ +/* @vitest-environment jsdom */ + +import { fireEvent, render, screen } from '@testing-library/react'; +import { useState } from 'react'; +import { describe, expect, it, vi } from 'vitest'; + +import type { + CanvasContextMenuState, + CanvasLayer, + ImageContextMenuState, +} from './ImageCanvasEditorTypes'; +import { useImageCanvasStageController } from './useImageCanvasStageController'; + +function createLayer(overrides: Partial = {}): CanvasLayer { + const id = overrides.id ?? 'layer-a'; + return { + id, + resourceId: `resource-${id}`, + title: id, + src: `data:image/png;base64,${id}`, + x: 100, + y: 80, + width: 240, + height: 120, + originalWidth: 240, + originalHeight: 120, + zIndex: 1, + sourceType: 'uploaded', + ...overrides, + }; +} + +function StageControllerHarness({ + hideGeneratedLayerPanelAfterBlur = vi.fn(), +}: { + hideGeneratedLayerPanelAfterBlur?: () => void; +}) { + const layers = [ + createLayer({ id: 'first', hidden: true }), + createLayer({ id: 'second', locked: true }), + ]; + const [selectedLayerId, setSelectedLayerId] = useState('first'); + const [selectedLayerIds, setSelectedLayerIds] = useState(['first', 'second']); + const [contextMenu, setContextMenu] = + useState(null); + const [imageContextMenu, setImageContextMenu] = + useState({ layerId: 'first', x: 1, y: 2 }); + const selectSingleLayer = (layerId: string | null) => { + setSelectedLayerId(layerId); + setSelectedLayerIds(layerId ? [layerId] : []); + }; + const controller = useImageCanvasStageController({ + layers, + selectedLayerId, + selectedLayerIds, + activeCanvasGenerationDialog: null, + imageContextMenu, + setImageContextMenu, + contextMenu, + setContextMenu, + viewport: { x: 0, y: 0, scale: 1 }, + canvasSize: { width: 900, height: 640 }, + selectSingleLayer, + hideGeneratedLayerPanelAfterBlur, + getCanvasPointFromClient: (clientX, clientY) => ({ + x: clientX + 1, + y: clientY + 2, + }), + }); + + return ( +
+ + {selectedLayerId ?? '-'}:{selectedLayerIds.join(',')} + + + {contextMenu + ? `${contextMenu.kind}:${contextMenu.x}:${contextMenu.y}:${contextMenu.canvasPoint.x}:${contextMenu.canvasPoint.y}` + : '-'} + + + {imageContextMenu + ? `${imageContextMenu.layerId}:${imageContextMenu.x}:${imageContextMenu.y}` + : '-'} + + + {String(controller.contextShouldShowLayer)} + + + {String(controller.contextShouldUnlockLayer)} + + +
+ +
+ ); +} + +describe('useImageCanvasStageController', () => { + it('clears canvas focus and closes generated layer panels', () => { + const hideGeneratedLayerPanelAfterBlur = vi.fn(); + render( + , + ); + + fireEvent.click(screen.getByRole('button', { name: '清空焦点' })); + + expect(screen.getByTestId('selected').textContent).toBe('-:'); + expect(screen.getByTestId('image-context').textContent).toBe('-'); + expect(screen.getByTestId('context').textContent).toBe('-'); + expect(hideGeneratedLayerPanelAfterBlur).toHaveBeenCalledTimes(1); + }); + + it('creates blank and layer context menus and preserves multi-target state', () => { + render(); + + fireEvent.contextMenu(screen.getByTestId('canvas'), { + clientX: 20, + clientY: 30, + }); + expect(screen.getByTestId('context').textContent).toBe('blank:20:30:21:32'); + expect(screen.getByTestId('image-context').textContent).toBe('-'); + + fireEvent.contextMenu( + screen.getByRole('button', { name: '图层右键目标' }), + { + clientX: 40, + clientY: 50, + }, + ); + expect(screen.getByTestId('context').textContent).toBe('layer:40:50:41:52'); + expect(screen.getByTestId('image-context').textContent).toBe('second:40:50'); + expect(screen.getByTestId('show-layer').textContent).toBe('true'); + expect(screen.getByTestId('unlock-layer').textContent).toBe('true'); + }); + +}); diff --git a/src/components/image-editor/useImageCanvasStageController.ts b/src/components/image-editor/useImageCanvasStageController.ts new file mode 100644 index 00000000..be61b3e5 --- /dev/null +++ b/src/components/image-editor/useImageCanvasStageController.ts @@ -0,0 +1,139 @@ +import { + type Dispatch, + type MouseEvent as ReactMouseEvent, + type SetStateAction, + useCallback, + useMemo, +} from 'react'; + +import type { + CanvasContextMenuState, + CanvasGenerationDialogState, + CanvasLayer, + CanvasViewport, + ImageContextMenuState, +} from './ImageCanvasEditorTypes'; +import { + createBlankCanvasContextMenu, + createLayerCanvasContextMenus, + resolveImageCanvasStageControllerModel, +} from './ImageCanvasStageControllerModel'; + +type UseImageCanvasStageControllerOptions = { + layers: CanvasLayer[]; + selectedLayerId: string | null; + selectedLayerIds: string[]; + activeCanvasGenerationDialog: CanvasGenerationDialogState | null; + imageContextMenu: ImageContextMenuState | null; + setImageContextMenu: Dispatch>; + contextMenu: CanvasContextMenuState | null; + setContextMenu: Dispatch>; + viewport: CanvasViewport; + canvasSize: { width: number; height: number }; + selectSingleLayer: (layerId: string | null) => void; + hideGeneratedLayerPanelAfterBlur: () => void; + getCanvasPointFromClient: ( + clientX: number, + clientY: number, + ) => { x: number; y: number }; +}; + +export function useImageCanvasStageController({ + layers, + selectedLayerId, + selectedLayerIds, + activeCanvasGenerationDialog, + imageContextMenu, + setImageContextMenu, + contextMenu, + setContextMenu, + viewport, + canvasSize, + selectSingleLayer, + hideGeneratedLayerPanelAfterBlur, + getCanvasPointFromClient, +}: UseImageCanvasStageControllerOptions) { + const model = useMemo( + () => + resolveImageCanvasStageControllerModel({ + layers, + selectedLayerId, + selectedLayerIds, + activeCanvasGenerationDialog, + imageContextMenu, + contextMenu, + viewport, + canvasSize, + }), + [ + activeCanvasGenerationDialog, + canvasSize, + contextMenu, + imageContextMenu, + layers, + selectedLayerId, + selectedLayerIds, + viewport, + ], + ); + + const clearCanvasFocus = useCallback(() => { + selectSingleLayer(null); + hideGeneratedLayerPanelAfterBlur(); + setImageContextMenu(null); + setContextMenu(null); + }, [ + hideGeneratedLayerPanelAfterBlur, + selectSingleLayer, + setContextMenu, + setImageContextMenu, + ]); + + const handleCanvasContextMenu = useCallback( + (event: ReactMouseEvent) => { + event.preventDefault(); + event.stopPropagation(); + setImageContextMenu(null); + setContextMenu( + createBlankCanvasContextMenu({ + clientX: event.clientX, + clientY: event.clientY, + canvasPoint: getCanvasPointFromClient(event.clientX, event.clientY), + }), + ); + }, + [getCanvasPointFromClient, setContextMenu, setImageContextMenu], + ); + + const handleLayerContextMenu = useCallback( + (event: ReactMouseEvent, layer: CanvasLayer) => { + event.preventDefault(); + event.stopPropagation(); + if (!selectedLayerIds.includes(layer.id)) { + selectSingleLayer(layer.id); + } + const nextMenus = createLayerCanvasContextMenus({ + clientX: event.clientX, + clientY: event.clientY, + layerId: layer.id, + canvasPoint: getCanvasPointFromClient(event.clientX, event.clientY), + }); + setContextMenu(nextMenus.contextMenu); + setImageContextMenu(nextMenus.imageContextMenu); + }, + [ + getCanvasPointFromClient, + selectSingleLayer, + selectedLayerIds, + setContextMenu, + setImageContextMenu, + ], + ); + + return { + ...model, + clearCanvasFocus, + handleCanvasContextMenu, + handleLayerContextMenu, + }; +} From a319d2ae0d8c1d9e4acb30043f07479c5e033a1f Mon Sep 17 00:00:00 2001 From: kdletters Date: Wed, 17 Jun 2026 13:48:48 +0800 Subject: [PATCH 3/3] =?UTF-8?q?=E6=8B=86=E5=88=86=E5=9B=BE=E7=89=87?= =?UTF-8?q?=E7=94=BB=E5=B8=83=E9=A1=B6=E9=83=A8=E6=A0=8F=E8=A7=86=E5=9B=BE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 新增 ImageCanvasTopbarView 承载返回项目入口和项目标题区域 新增顶栏视图测试覆盖重命名和导出入口 精简 ImageCanvasEditorView 的顶部栏 JSX 更新图片画布拆分计划和 TRACKING 验证记录 --- TRACKING.md | 1 + ...构】图片画布编辑器前端拆分计划-2026-06-17.md | 10 +- .../image-editor/ImageCanvasEditorView.tsx | 124 +++------------ .../ImageCanvasTopbarView.test.tsx | 147 +++++++++++++++++ .../image-editor/ImageCanvasTopbarView.tsx | 149 ++++++++++++++++++ .../useImageCanvasAssetExportWorkflow.ts | 2 +- 6 files changed, 325 insertions(+), 108 deletions(-) create mode 100644 src/components/image-editor/ImageCanvasTopbarView.test.tsx create mode 100644 src/components/image-editor/ImageCanvasTopbarView.tsx diff --git a/TRACKING.md b/TRACKING.md index c2c56134..ef3b15f8 100644 --- a/TRACKING.md +++ b/TRACKING.md @@ -141,3 +141,4 @@ - 2026-06-17 前端拆分第二十三阶段:新增 `ImageCanvasOverlayModel`,把生成输入框锚定、图标素材生成面板宽度、选中图片工具栏边界、快速编辑面板和角色动画面板定位从主视图抽出为纯模型;主视图继续保留生成 / quick edit / 角色动画状态机和舞台编排。新增模型单测覆盖锚定优先级、生成中隐藏、icon 宽度、工具栏 clamp、quick edit / 角色动画边界和生成 dialog 模式识别;主视图从 1182 行降至 1133 行。验证命令:`npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run test -- src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasGenerationWorkflow.test.tsx`、`npm run test -- src/components/image-editor/ImageCanvasOverlayModel.test.ts src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasKeyboardShortcuts.test.tsx src/components/image-editor/ImageCanvasMetadataModalView.test.tsx src/components/image-editor/useImageCanvasAssetLibrary.test.tsx src/components/image-editor/useImageCanvasProjectPersistence.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/useImageCanvasStageInteractions.test.tsx src/components/image-editor/useImageCanvasViewportControls.test.tsx src/components/image-editor/ImageCanvasInteractionModel.test.ts src/components/image-editor/useCanvasHistory.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasGenerationWorkflow.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/auth/PlatformAuthModalShell.test.tsx src/components/auth/AuthGate.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 未登录打开显示 `账号入口`,登录后 `画布背景设置` 点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,点击 `生成工具` 后 `Image Generator`、`生成图片` 对话框和 `AI画布工具栏` 均可见;素材库已有素材真实 pointer 拖入画布后图层数从 0 到 1,工具栏保持可见,截图留存于 `output/playwright/editor-overlay-model-regression-20260617.png`。 - 2026-06-17 前端拆分第二十四阶段:新增 `useImageCanvasAssetCanvasBridge`,把删除素材清理画布图层、素材加入画布、素材 pointer 拖入画布 / 文件夹、画布 drag/drop 分流和文件拖入画布参数组装从主视图抽成素材到画布桥接 hook;主视图继续保留素材库事实、上传读取、工程资源持久化和历史触发。新增 hook 单测覆盖素材建层、pointer drop 入画布和删除素材清理关联图层 / 选中态;主视图从 1133 行降至 1086 行。验证命令:`npm run test -- src/components/image-editor/useImageCanvasAssetCanvasBridge.test.tsx src/components/image-editor/useImageCanvasAssetPointerDragBridge.test.tsx src/components/image-editor/useImageCanvasCanvasDropWorkflow.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 清空会话后未登录直接显示 `账号入口`;登录临时开发账号后 `画布背景设置` 是完整设置面板,包含当前色、色域、色相、自定义颜色、预设、HEX 和恢复默认,点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,HEX 输入 `#ffffff` 后变为白色,按 Escape 关闭面板;登录后上传图片请求 `/api/editor/assets` 返回 200,素材库出现上传素材,`AI画布工具栏` 保持可见。 - 2026-06-17 前端拆分第二十五阶段:新增 `ImageCanvasStageControllerModel` 和 `useImageCanvasStageController`,把舞台派生状态、生成 / 选中浮层位置、右键菜单目标、空白画布右键、图层右键和清空画布焦点从主视图抽出;主视图继续保留工具切换、上传 / 生成入口、图层命令、项目持久化和舞台 pointer 状态机。新增模型和 hook 单测覆盖选中 / 生成锚定、右键菜单位置、显示 / 解锁判断、清空焦点和空白 / 图层右键菜单;主视图从 1086 行降至 993 行。验证命令:`npm run test -- src/components/image-editor/ImageCanvasStageControllerModel.test.ts src/components/image-editor/useImageCanvasStageController.test.tsx`、`npm run test -- src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasUploadWorkflow.test.tsx src/routing/appPageRoutes.test.ts`、`npm run typecheck`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 清空会话后未登录直接显示 `账号入口`,关闭登录后 `画布背景色` 打开完整 `画布背景设置` dialog,包含色相、自定义颜色、预设、HEX 和恢复默认;点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,`AI画布工具栏` 保持可见,截图留存于 `output/playwright/editor-stage-controller-smoke-20260617.png`。 +- 2026-06-17 前端拆分第二十六阶段:新增 `ImageCanvasTopbarView`,把返回项目入口、项目标题展示 / 重命名表单、下载画布素材按钮和导出状态提示从主视图抽出;主视图继续保留 chrome hook、项目持久化、导出工作流和实际导出副作用。新增组件单测覆盖返回入口、标题编辑入口、重命名提交 / 取消、导出按钮禁用 / 启用和导出状态提示;主视图从 993 行降至 905 行。验证命令:`npm run test -- src/components/image-editor/ImageCanvasTopbarView.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/image-editor/useImageCanvasAssetExportWorkflow.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx`、`npm run typecheck`、`npm run check:encoding`、`git diff --check`;浏览器回归:`http://127.0.0.1:10007/editor/canvas` 未登录直接显示 `账号入口`,顶栏返回项目入口、项目名、`画布` 标签和下载按钮均可见;关闭登录后打开 `画布背景设置`,点击 `暖灰` 后 viewport 背景为 `rgb(243, 240, 234)` 且 `background-image: none`,`AI画布工具栏` 保持可见,截图留存于 `output/playwright/editor-topbar-smoke-20260617.png`。 diff --git a/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md b/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md index c65c0432..3644b970 100644 --- a/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md +++ b/docs/technical/【前端架构】图片画布编辑器前端拆分计划-2026-06-17.md @@ -219,6 +219,14 @@ - 主视图继续保留工具切换、上传 / 生成入口、图层命令、项目持久化和舞台 pointer 状态机,避免把跨工作流动作塞进单个 hook。 - 本阶段主视图从 1086 行降至 993 行;后续若继续拆分,应优先考虑顶栏 / 项目标题区域或把侧栏图层右键入口并入同一舞台菜单控制层。 +## 第二十六阶段模块 + +- `ImageCanvasTopbarView.tsx` + - 承载编辑器顶部栏视觉结构:返回项目入口、项目标题展示、项目标题重命名表单、下载画布素材按钮和导出状态提示。 + - 主视图继续保留项目标题 / 重命名状态所属的 chrome hook、项目持久化、导出工作流和实际导出副作用;顶栏只负责表单事件和显示,不接管业务状态。 + - 新增组件单测覆盖返回入口、标题编辑入口、重命名提交 / 取消、导出按钮禁用 / 启用和导出状态提示。 + - 本阶段主视图从 993 行降至 905 行;后续可继续评估隐藏上传 input / 侧栏拖拽预览这类根级浮层是否值得抽为编辑器 shell 视图。 + ## 后续阶段 - 后续可继续选择更高内聚的交互 workflow 或持久化边界,不再把生成链路继续拆成浅层 wrapper。 @@ -226,7 +234,7 @@ ## 验证计划 -- `npm run test -- src/components/image-editor/ImageCanvasStageControllerModel.test.ts src/components/image-editor/useImageCanvasStageController.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx src/components/image-editor/useImageCanvasUploadWorkflow.test.tsx src/routing/appPageRoutes.test.ts` +- `npm run test -- src/components/image-editor/ImageCanvasTopbarView.test.tsx src/components/image-editor/ImageCanvasEditorView.test.tsx src/components/image-editor/useImageCanvasAssetExportWorkflow.test.tsx src/components/image-editor/useImageCanvasEditorChrome.test.tsx` - `npm run typecheck` - `npm run check:encoding` - `git diff --check` diff --git a/src/components/image-editor/ImageCanvasEditorView.tsx b/src/components/image-editor/ImageCanvasEditorView.tsx index ed9a430a..b4f4fa69 100644 --- a/src/components/image-editor/ImageCanvasEditorView.tsx +++ b/src/components/image-editor/ImageCanvasEditorView.tsx @@ -1,14 +1,11 @@ -import { Check, ChevronLeft, Download, Pencil, X } from 'lucide-react'; import { useCallback, useEffect, useMemo, useRef, useState } from 'react'; -import { PlatformStatusMessage } from '../common/PlatformStatusMessage'; -import { PlatformTextField } from '../common/PlatformTextField'; import { useAuthUi } from '../auth/AuthUiContext'; -import { EditorIconButton } from './ImageCanvasEditorPrimitives'; import { ImageCanvasGenerationComposerView } from './ImageCanvasGenerationComposerView'; import { ImageCanvasMetadataModalView } from './ImageCanvasMetadataModalView'; import { ImageCanvasSidebarView } from './ImageCanvasSidebarView'; import { ImageCanvasStageView } from './ImageCanvasStageView'; +import { ImageCanvasTopbarView } from './ImageCanvasTopbarView'; import { resolveContextMenuPosition } from './ImageCanvasEditorModel'; import { isCanvasGenerationComposerVisible, @@ -731,108 +728,23 @@ export function ImageCanvasEditorView() { />
-
- - -
- {isRenamingProject ? ( -
{ - event.preventDefault(); - submitProjectRename(projectId); - }} - > - { - setProjectRenameValue(event.target.value); - resetProjectRenameError(); - }} - onKeyDown={(event) => { - if (event.key === 'Escape') { - event.preventDefault(); - cancelProjectRename(); - } - }} - /> - - - {projectRenameError ? ( - - {projectRenameError} - - ) : null} - - ) : ( -
- - -
- )} - 画布 -
-
- layer.src.trim().length > 0) - } - onClick={() => void exportCanvasAssets()} - /> - {assetExportStatus ? ( - - {assetExportStatus.message} - - ) : null} -
-
+ = {}): CanvasLayer { + const id = overrides.id ?? 'layer-a'; + return { + id, + resourceId: `resource-${id}`, + title: id, + src: `data:image/png;base64,${id}`, + x: 0, + y: 0, + width: 320, + height: 180, + originalWidth: 320, + originalHeight: 180, + zIndex: 1, + sourceType: 'uploaded', + ...overrides, + }; +} + +function renderTopbar( + overrides: Partial< + Parameters[0] + > = {}, +) { + const props: Parameters[0] = { + projectId: 'project-a', + projectTitle: '默认项目', + projectRenameValue: '默认项目', + isRenamingProject: false, + isProjectRenameSaving: false, + projectRenameError: null, + layers: [], + assetExportStatus: null, + isExportingAssets: false, + setProjectRenameValue: vi.fn(), + startProjectRename: vi.fn(), + cancelProjectRename: vi.fn(), + submitProjectRename: vi.fn(), + resetProjectRenameError: vi.fn(), + exportCanvasAssets: vi.fn(), + ...overrides, + }; + + render(); + + return props; +} + +describe('ImageCanvasTopbarView', () => { + it('shows the project title and project gallery link', () => { + const props = renderTopbar(); + + expect(screen.getByRole('heading', { name: '默认项目' })).toBeTruthy(); + expect( + screen.getByRole('link', { name: '返回项目页面' }).getAttribute('href'), + ).toBe('/project'); + + fireEvent.click(screen.getByRole('button', { name: '编辑项目名称' })); + + expect(props.startProjectRename).toHaveBeenCalledTimes(1); + }); + + it('submits, resets, and cancels project rename edits', () => { + const props = renderTopbar({ + isRenamingProject: true, + projectRenameValue: '草稿项目', + projectRenameError: '项目名称不能为空', + }); + + const input = screen.getByLabelText('项目名称'); + fireEvent.change(input, { target: { value: '新项目' } }); + fireEvent.click(screen.getByRole('button', { name: '保存项目名称' })); + + expect(props.setProjectRenameValue).toHaveBeenCalledWith('新项目'); + expect(props.resetProjectRenameError).toHaveBeenCalledTimes(1); + expect(props.submitProjectRename).toHaveBeenCalledWith('project-a'); + expect(screen.getByRole('alert').textContent).toBe('项目名称不能为空'); + + fireEvent.keyDown(input, { key: 'Escape' }); + + expect(props.cancelProjectRename).toHaveBeenCalledTimes(1); + }); + + it('exports only when canvas has exportable layers and shows export status', () => { + const exportCanvasAssets = vi.fn(); + const { rerender } = render( + , + ); + + expect( + (screen.getByRole('button', { name: '下载画布素材' }) as HTMLButtonElement) + .disabled, + ).toBe(true); + + rerender( + , + ); + + fireEvent.click(screen.getByRole('button', { name: '下载画布素材' })); + + expect(exportCanvasAssets).toHaveBeenCalledTimes(1); + expect(screen.getByRole('status').textContent).toBe('画布素材已导出'); + }); +}); diff --git a/src/components/image-editor/ImageCanvasTopbarView.tsx b/src/components/image-editor/ImageCanvasTopbarView.tsx new file mode 100644 index 00000000..9eb9edc8 --- /dev/null +++ b/src/components/image-editor/ImageCanvasTopbarView.tsx @@ -0,0 +1,149 @@ +import { Check, ChevronLeft, Download, Pencil, X } from 'lucide-react'; + +import { PlatformStatusMessage } from '../common/PlatformStatusMessage'; +import { PlatformTextField } from '../common/PlatformTextField'; +import { EditorIconButton } from './ImageCanvasEditorPrimitives'; +import type { CanvasLayer } from './ImageCanvasEditorTypes'; +import type { AssetExportStatus } from './useImageCanvasAssetExportWorkflow'; + +type ImageCanvasTopbarViewProps = { + projectId: string | null; + projectTitle: string; + projectRenameValue: string; + isRenamingProject: boolean; + isProjectRenameSaving: boolean; + projectRenameError: string | null; + layers: CanvasLayer[]; + assetExportStatus: AssetExportStatus | null; + isExportingAssets: boolean; + setProjectRenameValue: (value: string) => void; + startProjectRename: () => void; + cancelProjectRename: () => void; + submitProjectRename: (projectId: string | null) => void; + resetProjectRenameError: () => void; + exportCanvasAssets: () => void | Promise; +}; + +export function ImageCanvasTopbarView({ + projectId, + projectTitle, + projectRenameValue, + isRenamingProject, + isProjectRenameSaving, + projectRenameError, + layers, + assetExportStatus, + isExportingAssets, + setProjectRenameValue, + startProjectRename, + cancelProjectRename, + submitProjectRename, + resetProjectRenameError, + exportCanvasAssets, +}: ImageCanvasTopbarViewProps) { + const hasExportableLayer = layers.some( + (layer) => layer.src.trim().length > 0, + ); + + return ( +
+ + +
+ {isRenamingProject ? ( +
{ + event.preventDefault(); + submitProjectRename(projectId); + }} + > + { + setProjectRenameValue(event.target.value); + resetProjectRenameError(); + }} + onKeyDown={(event) => { + if (event.key === 'Escape') { + event.preventDefault(); + cancelProjectRename(); + } + }} + /> + + + {projectRenameError ? ( + + {projectRenameError} + + ) : null} + + ) : ( +
+ + +
+ )} + 画布 +
+
+ void exportCanvasAssets()} + /> + {assetExportStatus ? ( + + {assetExportStatus.message} + + ) : null} +
+
+ ); +} diff --git a/src/components/image-editor/useImageCanvasAssetExportWorkflow.ts b/src/components/image-editor/useImageCanvasAssetExportWorkflow.ts index 1717e5ac..89af0ddd 100644 --- a/src/components/image-editor/useImageCanvasAssetExportWorkflow.ts +++ b/src/components/image-editor/useImageCanvasAssetExportWorkflow.ts @@ -16,7 +16,7 @@ import { sanitizeExportFilePart, } from './ImageCanvasExportModel'; -type AssetExportStatus = { +export type AssetExportStatus = { tone: 'info' | 'success' | 'error'; message: string; };