新增图片画布项目页

新增 /project 项目页和我的页项目入口

补齐图片画布工程列表、重命名和删除 API

支持 /editor/canvas 按 projectid 加载指定工程

更新图片画布文档、TRACKING 和对应测试
This commit is contained in:
2026-06-14 00:11:36 +08:00
parent b2122481ff
commit 85834a423d
32 changed files with 1800 additions and 20 deletions

View File

@@ -0,0 +1,125 @@
/* @vitest-environment jsdom */
import { render, screen, waitFor, within } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { afterEach, describe, expect, it, vi } from 'vitest';
import { ProjectGalleryView } from './ProjectGalleryView';
const listEditorProjectsMock = vi.hoisted(() => vi.fn());
const createEditorProjectMock = vi.hoisted(() => vi.fn());
const renameEditorProjectMock = vi.hoisted(() => vi.fn());
const deleteEditorProjectMock = vi.hoisted(() => vi.fn());
vi.mock('../../services/image-editor/editorProjectClient', () => ({
listEditorProjects: listEditorProjectsMock,
createEditorProject: createEditorProjectMock,
renameEditorProject: renameEditorProjectMock,
deleteEditorProject: deleteEditorProjectMock,
}));
const projectItems = [
{
projectId: 'editor-project-1',
title: '角色设定板',
viewport: { x: 0, y: 0, scale: 1 },
layers: [{ layerId: 'layer-1', resourceId: 'resource-1' }],
resources: [
{
resourceId: 'resource-1',
projectId: 'editor-project-1',
imageSrc: 'data:image/png;base64,one',
width: 1024,
height: 1024,
sourceType: 'uploaded',
},
],
updatedAt: '2026-06-12T08:00:00.000Z',
},
{
projectId: 'editor-project-2',
title: '场景草图',
viewport: { x: 0, y: 0, scale: 1 },
layers: [],
resources: [],
updatedAt: '2026-06-12T07:00:00.000Z',
},
];
describe('ProjectGalleryView', () => {
afterEach(() => {
listEditorProjectsMock.mockReset();
createEditorProjectMock.mockReset();
renameEditorProjectMock.mockReset();
deleteEditorProjectMock.mockReset();
});
it('opens a project from the gallery card', async () => {
const onOpenProject = vi.fn();
listEditorProjectsMock.mockResolvedValueOnce(projectItems);
const user = userEvent.setup();
render(<ProjectGalleryView onOpenProject={onOpenProject} />);
await screen.findByText('角色设定板');
await user.click(screen.getByRole('button', { name: '打开项目角色设定板' }));
expect(onOpenProject).toHaveBeenCalledWith('editor-project-1');
});
it('renames and deletes a project from the hover menu', async () => {
listEditorProjectsMock.mockResolvedValueOnce(projectItems);
renameEditorProjectMock.mockResolvedValueOnce({
...projectItems[0],
title: '新角色设定板',
});
deleteEditorProjectMock.mockResolvedValueOnce('editor-project-2');
const user = userEvent.setup();
render(<ProjectGalleryView onOpenProject={vi.fn()} />);
await screen.findByText('角色设定板');
await user.click(screen.getByRole('button', { name: '打开项目角色设定板菜单' }));
await user.click(screen.getByRole('menuitem', { name: //u }));
await user.clear(screen.getByLabelText('项目名称'));
await user.type(screen.getByLabelText('项目名称'), '新角色设定板');
await user.click(screen.getByRole('button', { name: '保存' }));
expect(renameEditorProjectMock).toHaveBeenCalledWith(
'editor-project-1',
'新角色设定板',
);
await screen.findByText('新角色设定板');
await user.click(screen.getByRole('button', { name: '打开项目场景草图菜单' }));
await user.click(screen.getByRole('menuitem', { name: //u }));
expect(deleteEditorProjectMock).toHaveBeenCalledWith('editor-project-2');
await waitFor(() => {
expect(screen.queryByText('场景草图')).toBeNull();
});
});
it('supports batch selection actions from the bottom toolbar', async () => {
listEditorProjectsMock.mockResolvedValueOnce(projectItems);
deleteEditorProjectMock.mockResolvedValue('deleted');
const user = userEvent.setup();
render(<ProjectGalleryView onOpenProject={vi.fn()} />);
await screen.findByText('角色设定板');
await user.click(screen.getByRole('button', { name: '选择' }));
const toolbar = screen.getByRole('toolbar', { name: '批量操作' });
await user.click(within(toolbar).getByRole('button', { name: '全选' }));
expect(
within(toolbar).getByRole('button', { name: '取消全选 · 已选 2' }),
).toBeTruthy();
await user.click(within(toolbar).getByRole('button', { name: '删除' }));
expect(deleteEditorProjectMock).toHaveBeenCalledWith('editor-project-1');
expect(deleteEditorProjectMock).toHaveBeenCalledWith('editor-project-2');
});
});

View File

@@ -0,0 +1,415 @@
import {
Check,
CheckSquare,
MoreHorizontal,
Pencil,
Plus,
Square,
Trash2,
X,
} from 'lucide-react';
import { useCallback, useEffect, useMemo, useState } from 'react';
import {
createEditorProject,
deleteEditorProject,
listEditorProjects,
renameEditorProject,
type EditorProjectSnapshot,
} from '../../services/image-editor/editorProjectClient';
import { PlatformActionButton } from '../common/PlatformActionButton';
import { PlatformEmptyState } from '../common/PlatformEmptyState';
import { PlatformIconButton } from '../common/PlatformIconButton';
type ProjectGalleryViewProps = {
onOpenProject: (projectId: string) => void;
};
type RenameDraft = {
projectId: string;
title: string;
};
function resolveProjectPreview(project: EditorProjectSnapshot) {
const layerResourceIds = new Set(
project.layers
.map((layer) => layer.resourceId)
.filter((resourceId) => resourceId.trim().length > 0),
);
return (
project.resources.find((resource) => layerResourceIds.has(resource.resourceId)) ??
project.resources[0] ??
null
);
}
function formatProjectUpdatedAt(value: string) {
const date = new Date(value);
if (Number.isNaN(date.getTime())) {
return '';
}
return new Intl.DateTimeFormat('zh-CN', {
month: '2-digit',
day: '2-digit',
hour: '2-digit',
minute: '2-digit',
}).format(date);
}
export function ProjectGalleryView({ onOpenProject }: ProjectGalleryViewProps) {
const [projects, setProjects] = useState<EditorProjectSnapshot[]>([]);
const [isLoading, setIsLoading] = useState(true);
const [errorMessage, setErrorMessage] = useState<string | null>(null);
const [activeMenuProjectId, setActiveMenuProjectId] = useState<string | null>(
null,
);
const [renameDraft, setRenameDraft] = useState<RenameDraft | null>(null);
const [selectedProjectIds, setSelectedProjectIds] = useState<Set<string>>(
() => new Set(),
);
const [isSelectionMode, setIsSelectionMode] = useState(false);
const selectedCount = selectedProjectIds.size;
const allSelected =
projects.length > 0 && selectedProjectIds.size === projects.length;
const refreshProjects = useCallback(() => {
setIsLoading(true);
setErrorMessage(null);
listEditorProjects()
.then((items) => {
setProjects(items);
setSelectedProjectIds((currentIds) => {
const availableIds = new Set(items.map((project) => project.projectId));
return new Set(
[...currentIds].filter((projectId) => availableIds.has(projectId)),
);
});
})
.catch((error: unknown) => {
setErrorMessage(
error instanceof Error ? error.message : '读取项目列表失败',
);
})
.finally(() => setIsLoading(false));
}, []);
useEffect(() => {
refreshProjects();
}, [refreshProjects]);
const createProject = useCallback(() => {
setErrorMessage(null);
createEditorProject()
.then((project) => onOpenProject(project.projectId))
.catch((error: unknown) => {
setErrorMessage(
error instanceof Error ? error.message : '创建项目失败',
);
});
}, [onOpenProject]);
const closeSelectionMode = useCallback(() => {
setIsSelectionMode(false);
setSelectedProjectIds(new Set());
}, []);
const toggleProjectSelection = useCallback((projectId: string) => {
setSelectedProjectIds((currentIds) => {
const nextIds = new Set(currentIds);
if (nextIds.has(projectId)) {
nextIds.delete(projectId);
} else {
nextIds.add(projectId);
}
return nextIds;
});
}, []);
const deleteProjects = useCallback(
(projectIds: string[]) => {
if (projectIds.length === 0) {
return;
}
setErrorMessage(null);
void Promise.all(projectIds.map((projectId) => deleteEditorProject(projectId)))
.then(() => {
setProjects((currentProjects) =>
currentProjects.filter(
(project) => !projectIds.includes(project.projectId),
),
);
setSelectedProjectIds((currentIds) => {
const nextIds = new Set(currentIds);
projectIds.forEach((projectId) => nextIds.delete(projectId));
return nextIds;
});
setActiveMenuProjectId(null);
})
.catch((error: unknown) => {
setErrorMessage(
error instanceof Error ? error.message : '删除项目失败',
);
});
},
[],
);
const submitRename = useCallback(() => {
if (!renameDraft) {
return;
}
const title = renameDraft.title.trim();
if (!title) {
return;
}
setErrorMessage(null);
renameEditorProject(renameDraft.projectId, title)
.then((project) => {
setProjects((currentProjects) =>
currentProjects.map((item) =>
item.projectId === project.projectId ? project : item,
),
);
setRenameDraft(null);
setActiveMenuProjectId(null);
})
.catch((error: unknown) => {
setErrorMessage(
error instanceof Error ? error.message : '重命名项目失败',
);
});
}, [renameDraft]);
const projectCards = useMemo(
() =>
projects.map((project) => {
const preview = resolveProjectPreview(project);
const selected = selectedProjectIds.has(project.projectId);
return (
<article
key={project.projectId}
className={[
'project-gallery__card',
selected ? 'project-gallery__card--selected' : '',
]
.filter(Boolean)
.join(' ')}
>
<button
type="button"
className="project-gallery__card-button"
onClick={() => {
if (isSelectionMode) {
toggleProjectSelection(project.projectId);
return;
}
onOpenProject(project.projectId);
}}
aria-label={`打开项目${project.title}`}
>
<span className="project-gallery__preview">
{preview ? (
<img src={preview.imageSrc} alt="" />
) : (
<span className="project-gallery__preview-empty" />
)}
{isSelectionMode ? (
<span className="project-gallery__checkbox">
{selected ? <Check className="h-4 w-4" /> : null}
</span>
) : null}
</span>
<span className="project-gallery__meta">
<span>{project.title}</span>
<span>{formatProjectUpdatedAt(project.updatedAt)}</span>
</span>
</button>
{!isSelectionMode ? (
<div className="project-gallery__card-menu-wrap">
<PlatformIconButton
label={`打开项目${project.title}菜单`}
icon={<MoreHorizontal className="h-4 w-4" />}
variant="surfaceFloating"
className="h-8 w-8"
onClick={(event) => {
event.stopPropagation();
setActiveMenuProjectId((currentProjectId) =>
currentProjectId === project.projectId
? null
: project.projectId,
);
}}
/>
{activeMenuProjectId === project.projectId ? (
<div className="project-gallery__card-menu" role="menu">
<button
type="button"
role="menuitem"
onClick={() =>
setRenameDraft({
projectId: project.projectId,
title: project.title,
})
}
>
<Pencil className="h-4 w-4" />
<span></span>
</button>
<button
type="button"
role="menuitem"
onClick={() => deleteProjects([project.projectId])}
>
<Trash2 className="h-4 w-4" />
<span></span>
</button>
</div>
) : null}
</div>
) : null}
</article>
);
}),
[
activeMenuProjectId,
deleteProjects,
isSelectionMode,
onOpenProject,
projects,
selectedProjectIds,
toggleProjectSelection,
],
);
return (
<main className="project-gallery" aria-label="项目">
<header className="project-gallery__header">
<div>
<h1></h1>
<span>{projects.length} </span>
</div>
<div className="project-gallery__header-actions">
<PlatformActionButton
tone="secondary"
size="sm"
onClick={() => setIsSelectionMode(true)}
disabled={projects.length === 0}
>
</PlatformActionButton>
<PlatformActionButton size="sm" onClick={createProject}>
<Plus className="h-4 w-4" />
</PlatformActionButton>
</div>
</header>
{errorMessage ? (
<div className="project-gallery__error" role="alert">
{errorMessage}
</div>
) : null}
{isLoading ? (
<PlatformEmptyState surface="subpanel" size="panel">
</PlatformEmptyState>
) : projects.length === 0 ? (
<button
type="button"
className="project-gallery__new-card"
onClick={createProject}
>
<Plus className="h-6 w-6" />
<span></span>
</button>
) : (
<section className="project-gallery__grid">{projectCards}</section>
)}
{renameDraft ? (
<div className="project-gallery__modal-backdrop" role="presentation">
<form
className="project-gallery__rename-dialog"
onSubmit={(event) => {
event.preventDefault();
submitRename();
}}
>
<div className="project-gallery__rename-header">
<h2></h2>
<PlatformIconButton
label="关闭重命名"
icon={<X className="h-4 w-4" />}
variant="surfaceFloating"
className="h-8 w-8"
onClick={() => setRenameDraft(null)}
/>
</div>
<input
aria-label="项目名称"
value={renameDraft.title}
onChange={(event) =>
setRenameDraft((currentDraft) =>
currentDraft
? { ...currentDraft, title: event.target.value }
: currentDraft,
)
}
autoFocus
/>
<div className="project-gallery__rename-actions">
<PlatformActionButton
type="button"
tone="secondary"
onClick={() => setRenameDraft(null)}
>
</PlatformActionButton>
<PlatformActionButton type="submit"></PlatformActionButton>
</div>
</form>
</div>
) : null}
{isSelectionMode ? (
<div className="project-gallery__batch-toolbar" role="toolbar" aria-label="批量操作">
<PlatformActionButton
tone="secondary"
size="sm"
onClick={() => {
setSelectedProjectIds(
allSelected
? new Set()
: new Set(projects.map((project) => project.projectId)),
);
}}
>
{allSelected ? (
<CheckSquare className="h-4 w-4" />
) : (
<Square className="h-4 w-4" />
)}
{selectedCount > 0
? `${allSelected ? '取消全选' : '全选'} · 已选 ${selectedCount}`
: '全选'}
</PlatformActionButton>
<PlatformActionButton
tone="warning"
size="sm"
disabled={selectedCount === 0}
onClick={() => deleteProjects([...selectedProjectIds])}
>
<Trash2 className="h-4 w-4" />
</PlatformActionButton>
<PlatformActionButton tone="secondary" size="sm" onClick={closeSelectionMode}>
</PlatformActionButton>
</div>
) : null}
</main>
);
}
export default ProjectGalleryView;