release: v1.2.0 手术图文病历报告系统

This commit is contained in:
2026-04-16 21:41:21 +08:00
parent ebe13ba8af
commit c55a55a27b
42 changed files with 73 additions and 1922 deletions

View File

@@ -1,113 +0,0 @@
---
name: medical-report-dev-workflow
description: Enforce a strict five-step workflow for all code modification requests on the Medical Surgical Report System (Gemini-图文报告系统-V1.1). Use whenever the user asks to change, add, fix, refactor, or update any project code, feature, UI, logic, or configuration. Do NOT use for simple questions, explanations, or read-only research.
---
# 手术图文病历报告系统 —— 修改需求五步工作流
当用户提出任何程序修改需求时,必须严格按照以下顺序执行。禁止跳过任何步骤,禁止在方案未经过用户二次人工审核确认前进入下一步。
---
## Step 0: Gitea 备份 + 记录时间戳
1. 获取当前时间戳,格式:`{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec}`(例如 `2026-04-16-16-39-42`)。
2. 执行 Gitea 备份:
```bash
git add .
git commit -m "backup before modification at {时间戳}"
git remote set-url origin http://192.168.31.5:5002/admin/Mdeical_Sur_Report.git
git branch -M main
git push -u origin main
```
3. 若首次初始化仓库,执行 `git init` 和 `git checkout -b main` 后再提交。
4. 此步骤中所有 git 操作若遇到报错(如 nothing to commit通过调整文件内容或空提交确保能推送成功。
---
## Step 1: 工程分析目录检查
确保以下目录存在(如不存在则创建):
```
.\工程分析\
```
---
## Step 2: 需求分析文档
将用户提出的修改需求整理分析,写入文件:
```
.\工程分析\需求分析-{时间戳}.md
```
文档必须包含:
- 需求背景
- 功能目标(精确描述要实现什么)
- 涉及页面/模块/文件
- 验收标准
---
## Step 3: 实现方案文档(必须经过用户二次人工审核)
基于需求分析,编写详细实现方案,写入文件:
```
.\工程分析\实现方案-{时间戳}.md
```
文档必须包含:
- 技术思路与实现路径
- 需要修改/新增的文件清单
- 关键代码变更说明
- 潜在风险点及应对策略
**重要约束**:写完此文档后,必须立即停止并向用户汇报,等待用户二次人工审核确认。在得到明确的"确认"、"同意"、"可以执行"等回复前,禁止继续进入 Step 4 或开始任何代码修改。
---
## Step 4: 测试方案文档(必须经过用户二次人工审核)
实现方案经用户确认后,编写测试方案,写入文件:
```
.\工程分析\测试方案-{时间戳}.md
```
文档必须包含:
- 测试项清单
- 测试步骤(操作路径)
- 预期结果
- 回归验证范围(确保没有破坏现有功能)
**重要约束**:写完此文档后,必须立即停止并向用户汇报,等待用户二次人工审核确认。在得到明确的"确认"、"同意"、"可以执行"等回复前,禁止继续进入 Step 5。
---
## Step 5: 执行修改 + 经验记录
测试方案经用户确认后,开始执行:
1. 按实现方案修改代码。
2. 运行 `npm run lint` 进行 TypeScript 类型检查,确保零错误。
3. 运行 `npm run build` 验证生产构建是否通过。
4. 修改完成后,在以下文档中追加本次执行过程中遇到的关键问题及解决方案:
```
.\工程分析\经验记录.md
```
每条记录必须使用四段式格式:
- **A. 具体问题**
- **B. 产生问题原因**
- **C. 解决问题方案**
- **D. 后续如何避免问题**
5. 向用户汇报最终执行结果和验证情况。
---
## 快速参考
| 项目 | 值 |
|------|-----|
| Gitea 地址 | `http://192.168.31.5:5002/admin/Mdeical_Sur_Report.git` |
| 工程分析目录 | `.\工程分析\` |
| 类型检查 | `npm run lint` |
| 构建验证 | `npm run build` |
| 开发服务器 | `npm run dev` |

View File

@@ -6,7 +6,7 @@
## 1. 项目概述
**手术图文病历报告系统**Gemini-图文报告系统-V1.1是一款基于 **React 19 + TypeScript + Vite + Tailwind CSS 4** 开发的纯前端医疗图文报告管理应用。所有数据持久化在浏览器 `localStorage` 中,无需后端服务即可独立运行。
**手术图文病历报告系统**是一款基于 **React 19 + TypeScript + Vite + Tailwind CSS 4** 开发的纯前端医疗图文报告管理应用。所有数据持久化在浏览器 `localStorage` 中,无需后端服务即可独立运行。
### 核心功能
- **图文报告生成**:基于 `contentEditable` 的富文本编辑器,支持插入表格、图片占位符,可从本地或手术视频中截取关键帧插入报告。
@@ -34,6 +34,7 @@
- **图标**Lucide React
- **动画**Motion
- **语言**TypeScript 5.8`tsconfig.json``jsx: "react-jsx"``moduleResolution: "bundler"`
- **其他依赖**`@google/genai`(预留 AI 功能)
### 运行时架构
- **纯前端 SPA**:无后端 API所有业务逻辑在浏览器端执行。
@@ -46,8 +47,7 @@
```
.
├── docker-compose.yaml # Docker Compose 配置(端口 8080:80
├── docker-compose.qnap.yml # QNAP 专用 Docker Compose
├── docker-compose.yaml # Docker Compose 配置(端口 4002:80
├── Dockerfile # 多阶段构建node:20-alpine -> nginx:alpine
├── nginx.conf # Nginx SPA 回退配置try_files
├── package.json # 依赖与脚本
@@ -65,7 +65,7 @@
├── pages/
│ ├── Login.tsx # 登录页(初始化默认数据)
│ ├── Dashboard.tsx # 工作台概览(统计图表 + 快捷入口)
│ ├── ReportEditor.tsx # 图文报告编辑器(最复杂的页面)
│ ├── ReportEditor.tsx # 图文报告编辑器(最复杂的页面,约 1400+ 行
│ ├── ReportManage.tsx # 报告列表管理
│ ├── ReportView.tsx # 报告查看/打印
│ ├── TemplateManage.tsx # 模板管理
@@ -106,7 +106,7 @@ npm run dev
### Docker 部署
```bash
# 构建并启动(访问 http://localhost:8080
# 构建并启动(访问 http://localhost:4002
docker-compose up -d --build
# 停止
@@ -136,10 +136,13 @@ docker-compose down
- 各页面在 `useEffect` 中读取 `storage.get('currentUser')`,未登录则 `navigate('/')`
- `Sidebar.tsx``navItems``roles` 数组过滤菜单。
- `user` 角色只能查看/编辑自己创建的报告;`super`/`admin` 可查看全院报告。
- `admin` 只能管理同部门(`department`)的 `user` 角色用户,且一个部门只能有一个管理员。
- 用户对象包含 `visibleTemplates`(可视模板)和 `manageableTemplates`(可管理模板)数组,用于细粒度模板权限控制。
### 数据持久化约定
- **禁止直接调用 `localStorage`**,统一使用 `src/utils/storage.ts` 中的 `storage.get / storage.set / storage.remove`
- localStorage 中存储的 key 包括:`users``reports``templates``systemSettings``currentUser``multiSelectOptions``anesthesiaOptions``reportEditorDraft_{username}``restore_{reportId}`sessionStorage
- localStorage 中存储的 key 包括:`users``reports``templates``systemSettings``currentUser``multiSelectOptions``anesthesiaOptions``reportEditorDraft_{username}`
- sessionStorage 中存储的 key 包括:`restore_{reportId}`(用于历史版本恢复)。
- 报告编辑器会在 `beforeunload``visibilitychange` 时自动保存草稿到 `reportEditorDraft_{username}`
### 样式约定
@@ -149,19 +152,20 @@ docker-compose down
- A4 打印尺寸为 `210mm × 297mm`,打印时通过 `visibility` 控制只显示 `.print-content` 区域。
### 编辑器实现细节
- `ReportEditor.tsx` 使用原生 `contentEditable` + `document.execCommand` 实现富文本,而非第三方编辑器。
- `ReportEditor.tsx``TemplateManage.tsx` 使用原生 `contentEditable` + `document.execCommand` 实现富文本,而非第三方编辑器。
- 图片通过 `.image-placeholder` 占位符插入,支持两种填充方式:
1. 点击占位符上传本地图片Base64 存入 HTML
2. 从右侧“视频分析”面板拖拽自动抽帧的截图到占位符中。
- 视频中上传后会根据 `systemSettings.framePositions` 自动抽帧(通过 `<video>` + `<canvas>` 实现)。
- 自动抽帧支持“自动帧插入”功能:开启后,抽得的帧会按延迟顺序自动填入编辑器中的空图片占位符。
### TypeScript 类型
核心类型定义在 `src/types.ts`
- `User`:用户,角色为 `'super' | 'admin' | 'user'`
- `Report`:报告,状态为 `'draft' | 'completed'`,含 `content`HTML 字符串)
- `User`:用户,角色为 `'super' | 'admin' | 'user'`,含 `visibleTemplates``manageableTemplates`
- `Report`:报告,状态为 `'draft' | 'completed'`,含 `content`HTML 字符串)`videos``capturedFrames``history`
- `Template`:模板,结构与报告内容类似
- `SystemSettings`:系统设置,含 `frameCount``framePositions``apiEndpoint`
- `CapturedFrame`:视频抽帧结果
- `SystemSettings`:系统设置,含 `frameCount``framePositions``apiEndpoint``apiKey``defaultTemplate``frameMode``autoInsertFrames``autoInsertFrameIndices``autoInsertDelay`
- `CapturedFrame`:视频抽帧结果,含 `dataUrl``isManual`
### 路径别名
- `vite.config.ts``tsconfig.json` 均配置了 `@/` 指向项目根目录(`.`)。
@@ -201,3 +205,4 @@ docker-compose down
- **打印逻辑**已封装在 `src/utils/print.ts`,需要打印 A4 报告时直接调用 `printDocument(htmlContent)`
- **修改样式时优先检查 `src/index.css`**Tailwind v4 的主题变量和打印样式都在那里。
- **添加新页面后**,记得在 `src/App.tsx` 注册路由,并在 `src/components/Sidebar.tsx``navItems` 中配置菜单和可见角色。
- **修改 Docker 端口映射时**,同步更新 `docker-compose.yaml` 和本文件中的说明。

View File

@@ -1,49 +0,0 @@
version: "3.8"
services:
tuwen_system:
# 使用官方 Node 镜像,无需 Dockerfile 构建
image: node:20-alpine
container_name: tuwen_system
# 将 NAS 上的源码目录挂载到容器内
volumes:
- /share/Container/tuwen_system:/app
working_dir: /app
# 端口映射NAS 4002 端口 -> 容器 3000 端口
# (因为 package.json 中 dev 脚本固定为 --port=3000
ports:
- "4002:3000"
restart: unless-stopped
tty: true
stdin_open: true
# 启动命令:先检查 node_modules 是否存在,避免每次重启都重新安装
# 然后运行开发服务package.json 中已带 --host=0.0.0.0
command: >
sh -c "
if [ ! -d node_modules ]; then
echo 'Installing dependencies...' &&
npm install;
else
echo 'Dependencies already installed, skipping npm install...';
fi &&
echo 'Fixing binary permissions...' &&
chmod -R +x node_modules/.bin/ 2>/dev/null || true &&
echo 'Starting dev server...' &&
npm run dev
"
environment:
# 应用访问地址(用于系统内跳转链接等场景)
- APP_URL=http://192.168.31.5:4002
# 网络代理配置(国内环境下加速 npm install
# 如果不需要代理,可将下面 5 行注释掉
- HTTP_PROXY=http://192.168.31.7:7893
- HTTPS_PROXY=http://192.168.31.7:7893
- http_proxy=http://192.168.31.7:7893
- https_proxy=http://192.168.31.7:7893
- NO_PROXY=localhost,127.0.0.1,192.168.31.0/24

View File

@@ -1,11 +1,11 @@
version: "3.8"
services:
app:
tuwen_system:
build:
context: .
dockerfile: Dockerfile
container_name: medical-report-app
ports:
- "8080:80"
container_name: tuwen_system
restart: unless-stopped
ports:
- "4002:80"

View File

@@ -36,9 +36,11 @@ export default function SystemSettings() {
savedSettings.defaultTemplate = savedTemplates[0].id;
}
if (!savedSettings.frameMode) savedSettings.frameMode = 'uniform';
if (typeof savedSettings.autoInsertFrames !== 'boolean') savedSettings.autoInsertFrames = false;
if (typeof savedSettings.autoInsertDelay !== 'number') savedSettings.autoInsertDelay = 0;
setSettings(savedSettings);
} else if (savedTemplates.length > 0) {
setSettings(prev => ({ ...prev, defaultTemplate: savedTemplates[0].id, frameMode: prev.frameMode || 'uniform' }));
setSettings(prev => ({ ...prev, defaultTemplate: savedTemplates[0].id, frameMode: prev.frameMode || 'uniform', autoInsertFrames: typeof prev.autoInsertFrames === 'boolean' ? prev.autoInsertFrames : false, autoInsertDelay: typeof prev.autoInsertDelay === 'number' ? prev.autoInsertDelay : 0 }));
}
setTemplates(savedTemplates);
}, [navigate]);
@@ -172,6 +174,33 @@ export default function SystemSettings() {
</div>
</div>
<div className="flex items-center gap-2">
<input
type="checkbox"
id="autoInsertFrames"
checked={settings.autoInsertFrames || false}
onChange={(e) => setSettings({ ...settings, autoInsertFrames: e.target.checked })}
className="w-4 h-4 accent-accent cursor-pointer"
/>
<label htmlFor="autoInsertFrames" className="text-sm text-text-main cursor-pointer"></label>
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
<div className="space-y-1.5">
<label className="block text-xs font-bold text-text-main uppercase tracking-wider"> (s)</label>
<input
type="number"
min={0}
step={0.1}
value={settings.autoInsertDelay || 0}
onChange={(e) => setSettings({ ...settings, autoInsertDelay: Math.max(0, parseFloat(e.target.value) || 0) })}
className="input-minimal bg-white w-full"
/>
</div>
</div>
<p className="text-[11px] text-text-muted"></p>
<div className="space-y-1.5">
<label className="block text-xs font-bold text-text-main uppercase tracking-wider"> (%)</label>
<div className="grid grid-cols-2 sm:grid-cols-4 md:grid-cols-6 gap-3">
@@ -191,11 +220,30 @@ export default function SystemSettings() {
className="input-minimal w-full pr-6 text-center"
/>
<span className="absolute right-2 top-1/2 -translate-y-1/2 text-[10px] text-text-muted">%</span>
{settings.autoInsertFrames && (
<span
onClick={() => {
const current = settings.autoInsertFrameIndices || [];
const next = current.includes(idx)
? current.filter(i => i !== idx)
: [...current, idx].sort((a, b) => a - b);
setSettings({ ...settings, autoInsertFrameIndices: next });
}}
className={`absolute top-1 left-1 cursor-pointer transition-colors ${
(settings.autoInsertFrameIndices || []).includes(idx) ? 'text-green-500' : 'text-slate-300'
}`}
>
<Check size={12} />
</span>
)}
<button
type="button"
onClick={() => {
const newPos = settings.framePositions.filter((_, i) => i !== idx);
setSettings({ ...settings, framePositions: newPos, frameCount: newPos.length });
const newIndices = (settings.autoInsertFrameIndices || [])
.filter(i => i !== idx)
.map(i => i > idx ? i - 1 : i);
setSettings({ ...settings, framePositions: newPos, frameCount: newPos.length, autoInsertFrameIndices: newIndices });
}}
className="absolute -top-2 -right-2 w-5 h-5 bg-red-500 text-white rounded-full flex items-center justify-center text-[10px] opacity-0 group-hover:opacity-100 transition-all shadow-sm"
>

View File

@@ -75,4 +75,7 @@ export interface SystemSettings {
apiKey: string;
defaultTemplate?: string;
frameMode?: 'uniform' | 'keep';
autoInsertFrames?: boolean;
autoInsertFrameIndices?: number[];
autoInsertDelay?: number;
}

View File

@@ -1,97 +0,0 @@
# 代码编纂工作流
> 本工作流为项目修改类需求的标准执行流程。后续所有项目修改相关需求,均需严格按以下步骤执行。
---
## 前置约定
- 时间戳格式:`{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec}`
- 示例:`2026-04-16-18-35-00`
- 所有方案文档均存放于 `.\工程分析\` 目录下。
---
## 执行步骤
### Step 0. 记录开始时间
每次执行前,以当前时间生成时间戳,作为本次需求的唯一标识。
### Step 1. 创建/确认工程分析目录
确保 `.\工程分析\` 文件夹存在。如不存在,则自动创建。
### Step 2. 需求分析
将用户提出的需求整理、拆解、澄清后,写入文档:
```
.\工程分析\需求分析-{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec}.md
```
内容要求:
- 原始需求摘要
- 需求拆解(功能点 / 非功能点)
- 待确认问题(如有)
- 影响范围预估
### Step 3. 实现方案(需人工确认)
基于需求分析,撰写详细的实现方案,写入文档:
```
.\工程分析\实现方案-{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec}.md
```
内容要求:
- 实现思路与架构决策
- 涉及修改的文件清单
- 具体的代码变更说明
- 风险点与回滚策略
**⚠️ 此文档写完后,必须提交给用户进行二次人工审核确认,得到明确批准后方可进入下一步。**
### Step 4. 测试方案(需人工确认)
基于实现方案,撰写测试方案,写入文档:
```
.\工程分析\测试方案-{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec}.md
```
内容要求:
- 测试目标
- 测试用例设计
- 测试环境准备
- 验收标准
**⚠️ 此文档写完后,必须提交给用户进行二次人工审核确认,得到明确批准后方可进入下一步。**
### Step 5. 执行修改前准备
1. **阅读 `.\工程分析\经验记录.md`**,回顾历史问题,避免重复犯错。
2. 确认实现方案和测试方案均已获得用户批准。
### Step 6. 执行修改
按照已批准的实现方案和测试方案,执行具体的代码修改与测试验证。
### Step 7. 更新经验记录
修改完成后,将本次执行过程中遇到的关键问题及解决方案,以 **四段式** 追加写入 `.\工程分析\经验记录.md`
- **A. 具体问题**
- **B. 产生问题原因**
- **C. 解决问题方案**
- **D. 后续如何避免问题**
### Step 8. Gitea 备份 Commit
`.\工程分析\` 目录下的所有文档使用 Gitea 进行备份,提交 Commit。
Commit Message 格式:
```
{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec} - {本次修改的简要描述}
```
Commit 完成后,提醒用户备份已完成。
---
## 当前状态
- [x] 工作流文档建立
- [x] 工程分析目录创建
- [x] 经验记录初始文档创建

View File

@@ -1,69 +0,0 @@
# 实现方案 — 2026-04-16-16-51-00
## 技术思路
本次修复聚焦于 `ReportEditor.tsx` 中草稿draft恢复与默认模板加载的竞争条件问题。核心思路是
1. **拒绝加载空白草稿**:将空字符串(或仅空白字符)的 draft 视为无效,不拦截默认模板加载流程。
2. **草稿中携带模板信息**:在自动保存 draft 时追加 `loadedTemplateId`,恢复草稿时同步还原,确保模板选择器显示正确。
3. **统一初始化路径**`useEffect``useLayoutEffect` 中的草稿/默认模板判断逻辑保持一致,避免其中一个抢先设置空白内容。
---
## 修改文件清单
| 文件 | 变更类型 | 说明 |
|------|----------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 修复草稿恢复条件、保存/恢复 `loadedTemplateId`、统一空白判断逻辑 |
---
## 关键代码变更说明
### 1. 保存 draft 时追加 `loadedTemplateId`
`saveDraftToStorage` 回调中,将当前 `loadedTemplateId` 一并持久化:
```tsx
storage.set(key, {
content: contentRef.current,
loadedTemplateId, // 新增
draftReportId: reportId || null,
...stateRef.current
});
```
### 2. 恢复 draft 时同步恢复 `loadedTemplateId`useEffect 初始化逻辑)
**新建报告分支(无 `reportId`**
- 将条件 `typeof draft.content === 'string'` 改为 `typeof draft.content === 'string' && draft.content.trim().length > 0`
- 若草稿有效,除了回填内容,还要执行 `setLoadedTemplateId(draft.loadedTemplateId || '')`
**编辑报告分支(有 `reportId`**
- 同样使用 `draft.content.trim().length > 0` 判断,避免空白草稿覆盖已有报告内容。
- 恢复 `loadedTemplateId`(虽然在编辑模式下模板选择器通常不显示具体模板名,但保持一致性)。
### 3. useLayoutEffect 安全网逻辑同步修复
`useLayoutEffect`(第 611 行起)作为 editor ref 就绪后的二次安全网,其草稿判断条件也要同步修改:
- `typeof draft.content === 'string'``typeof draft.content === 'string' && draft.content.trim().length > 0`
- 恢复 `loadedTemplateId`
### 4. 默认模板加载时同步设置 `loadedTemplateId`
在加载 `settings.defaultTemplate` 对应模板内容时,当前代码已经设置了 `setLoadedTemplateId(tpl.id)`,无需改动。
在兜底使用 `defaultReportContent` 时,`loadedTemplateId` 保持为空字符串(显示"无"),这符合语义,因为兜底内容不是用户选中的模板。
---
## 风险点及应对策略
| 风险 | 影响 | 应对策略 |
|------|------|----------|
| 修改了多处 draft 判断逻辑,可能漏改 | 某些路由切换场景仍出现空白 | 在 `useEffect``useLayoutEffect` 两处共 **4 个草稿加载点** 统一替换判断条件 |
| `loadedTemplateId` 加入 draft 后,旧 draft 兼容性 | 旧 draft 没有该字段,恢复时值为 `undefined`,模板选择器短暂显示"无" | 使用 `draft.loadedTemplateId || ''` 兜底,不影响功能 |
| 用户之前确实清空了内容再离开 | 会被视为无效草稿而丢失空白状态 | 这是预期行为:空白内容等价于未开始编辑,应回退到默认模板 |
---
## 改动范围总结
- 仅修改 `src/pages/ReportEditor.tsx`,不触及路由、存储封装或其他页面逻辑。
- 不引入新依赖。

View File

@@ -1,102 +0,0 @@
# 实现方案 — 2026-04-16-17-07-04
## 技术思路
`ReportEditor.tsx` 中为关键帧卡片新增一个 **"插入"** 按钮,点击后自动将当前帧图片填充到编辑器中第一个空置的 `.image-placeholder` 中。
为保持代码整洁,将现有 `handleDrop` 中的占位符填充逻辑抽取为可复用的 `fillPlaceholder` 函数,供拖拽放下和按钮点击共同调用。
---
## 修改文件清单
| 文件 | 变更类型 | 说明 |
|------|----------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 新增 `fillPlaceholder` 函数、新增 `insertFrameToPlaceholder` 函数、修改关键帧卡片 JSX |
---
## 关键代码变更说明
### 1. 抽取公共填充函数 `fillPlaceholder`
`handleDrop` 中的图片填充逻辑抽离:
```tsx
const fillPlaceholder = (placeholder: HTMLElement, frame: CapturedFrame) => {
placeholder.innerHTML = `
<span class="delete-btn" contenteditable="false">×</span>
<img src="${frame.dataUrl}" style="max-width: 100%; height: auto; display: block; margin: 0 auto;" draggable="false">
`;
placeholder.classList.add('has-image');
if (editorRef.current) contentRef.current = editorRef.current.innerHTML;
saveDraftToStorage();
};
```
并同步简化 `handleDrop`
```tsx
const handleDrop = (e: React.DragEvent, placeholder: HTMLElement) => {
e.preventDefault();
const frameId = e.dataTransfer.getData('frameId');
const frame = capturedFrames.find(f => f.id.toString() === frameId);
if (frame) {
fillPlaceholder(placeholder, frame);
}
};
```
### 2. 新增一键插入函数 `insertFrameToPlaceholder`
```tsx
const insertFrameToPlaceholder = (frame: CapturedFrame) => {
if (!editorRef.current) {
alert('编辑器未准备好');
return;
}
const emptyPlaceholder = editorRef.current.querySelector('.image-placeholder:not(.has-image)') as HTMLElement | null;
if (!emptyPlaceholder) {
alert('没有可插入图片的空位');
return;
}
fillPlaceholder(emptyPlaceholder, frame);
};
```
### 3. 在关键帧卡片底部添加 "插入" 按钮
修改现有 JSX约第 1303 行附近),在 `timeFormatted` 与 "可拖拽" 之间插入按钮:
```tsx
<div className="text-[9px] font-bold text-text-muted mt-1.5 px-1 flex justify-between items-center">
<span>{frame.timeFormatted}</span>
<div className="flex items-center gap-2">
<button
onClick={(e) => { e.stopPropagation(); insertFrameToPlaceholder(frame); }}
className="text-accent opacity-0 group-hover:opacity-100 transition-opacity hover:underline"
>
</button>
<span className="text-accent opacity-0 group-hover:opacity-100 transition-opacity"></span>
</div>
</div>
```
按钮使用 `opacity-0 group-hover:opacity-100 transition-opacity`,与 "可拖拽" 显隐行为完全一致。`e.stopPropagation()` 避免触发卡片的 `onClick`(即跳转到视频位置)。
---
## 风险点及应对策略
| 风险 | 影响 | 应对策略 |
|------|------|----------|
| `editorRef.current` 为空时点击插入 | JS 报错 | 函数开头增加判空并 alert 提示 |
| 没有空占位符时点击插入 | 用户困惑 | 未找到 `.image-placeholder:not(.has-image)` 时弹出友好提示 |
| 按钮点击触发卡片 onClick | 视频意外跳转 | 使用 `e.stopPropagation()` 阻止冒泡 |
| `handleDrop` 抽离后功能回退 | 拖拽失效 | 保持 `handleDrop` 调用 `fillPlaceholder`,逻辑与原来一致 |
---
## 改动范围总结
- 仅修改 `src/pages/ReportEditor.tsx`,不触及其他文件。
- 不引入新依赖。

View File

@@ -1,81 +0,0 @@
# 实现方案 — 2026-04-16-17-15-37
## 技术思路
仅调整 `ReportEditor.tsx` 中关键帧卡片的 JSX 结构:将 "插入" 按钮从底部文字行移到图片层的相对定位容器中,并修改其 className 为实体按钮样式(绿色背景、白色文字、圆角、阴影)。
`insertFrameToPlaceholder` 等逻辑函数无需改动。
---
## 修改文件清单
| 文件 | 变更类型 | 说明 |
|------|----------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 移动 "插入" 按钮位置、更新样式 className |
---
## 关键代码变更说明
### 当前 JSX 片段(约第 1303~1318 行)
```tsx
<div className="relative">
<img src={frame.dataUrl} className="w-full aspect-video object-cover rounded-lg" />
{frame.isManual && <span className="manual-frame-badge"></span>}
</div>
<div className="text-[9px] font-bold text-text-muted mt-1.5 px-1 flex justify-between items-center">
<span>{frame.timeFormatted}</span>
<div className="flex items-center gap-2">
<button
onClick={(e) => { e.stopPropagation(); insertFrameToPlaceholder(frame); }}
className="text-accent opacity-0 group-hover:opacity-100 transition-opacity hover:underline"
>
</button>
<span className="text-accent opacity-0 group-hover:opacity-100 transition-opacity"></span>
</div>
</div>
```
### 修改后 JSX 片段
```tsx
<div className="relative">
<img src={frame.dataUrl} className="w-full aspect-video object-cover rounded-lg" />
{frame.isManual && <span className="manual-frame-badge"></span>}
<button
onClick={(e) => { e.stopPropagation(); insertFrameToPlaceholder(frame); }}
className="absolute inset-0 m-auto w-fit h-fit px-3 py-1.5 bg-emerald-500 text-white text-[10px] font-bold rounded-full shadow-md opacity-0 group-hover:opacity-100 transition-opacity hover:bg-emerald-600"
>
</button>
</div>
<div className="text-[9px] font-bold text-text-muted mt-1.5 px-1 flex justify-between items-center">
<span>{frame.timeFormatted}</span>
<span className="text-accent opacity-0 group-hover:opacity-100 transition-opacity"></span>
</div>
```
### 样式说明
- `absolute inset-0 m-auto w-fit h-fit`:使按钮在图片容器内水平和垂直居中。
- `px-3 py-1.5`:实体按钮的内边距。
- `bg-emerald-500 text-white`:翠绿背景 + 白色文字,与 "可拖拽" 的蓝色(`text-accent`)形成明显区分。
- `rounded-full shadow-md`:圆角胶囊形状 + 轻微阴影,呈现按钮质感。
- `opacity-0 group-hover:opacity-100 transition-opacity`:保持 hover 显示/隐藏行为与 "可拖拽" 一致。
---
## 风险点及应对策略
| 风险 | 影响 | 应对策略 |
|------|------|----------|
| 按钮遮挡 "手动" 徽章 | 视觉冲突 | "手动" 徽章位于左上角,按钮位于正中央,空间不重叠 |
| 按钮点击触发卡片 onClick | 视频跳转 | 保留 `e.stopPropagation()` |
| 绝对定位在相对容器中未居中 | 偏位 | 使用 `inset-0 m-auto w-fit h-fit` 确保 Flex/Grid 内的居中 |
---
## 改动范围总结
- 仅做 JSX 结构和样式的微调整,不修改任何逻辑函数。
- 不引入新依赖。

View File

@@ -1,77 +0,0 @@
# 实现方案 — 2026-04-16-17-21-58
## 技术思路
`ReportEditor.tsx` 中关键帧卡片的 "插入" 按钮从图片层的 absolute 覆盖层移回底部文字行,放置在 `timeFormatted` 与 "可拖拽" 之间。颜色恢复为与 "可拖拽" 一致的蓝色(`bg-accent` / `text-white`),但保留实体胶囊按钮样式。
---
## 修改文件清单
| 文件 | 变更类型 | 说明 |
|------|----------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 移动 "插入" 按钮位置、调整颜色为蓝色 |
---
## 关键代码变更说明
### 当前 JSX 片段(约第 1316~1331 行)
```tsx
<div className="relative">
<img src={frame.dataUrl} className="w-full aspect-video object-cover rounded-lg" />
{frame.isManual && <span className="manual-frame-badge"></span>}
<button
onClick={(e) => { e.stopPropagation(); insertFrameToPlaceholder(frame); }}
className="absolute inset-0 m-auto w-fit h-fit px-3 py-1.5 bg-emerald-500 text-white text-[10px] font-bold rounded-full shadow-md opacity-0 group-hover:opacity-100 transition-opacity hover:bg-emerald-600"
>
</button>
</div>
<div className="text-[9px] font-bold text-text-muted mt-1.5 px-1 flex justify-between items-center">
<span>{frame.timeFormatted}</span>
<span className="text-accent opacity-0 group-hover:opacity-100 transition-opacity"></span>
</div>
```
### 修改后 JSX 片段
```tsx
<div className="relative">
<img src={frame.dataUrl} className="w-full aspect-video object-cover rounded-lg" />
{frame.isManual && <span className="manual-frame-badge"></span>}
</div>
<div className="text-[9px] font-bold text-text-muted mt-1.5 px-1 flex justify-between items-center">
<span>{frame.timeFormatted}</span>
<div className="flex items-center gap-2">
<button
onClick={(e) => { e.stopPropagation(); insertFrameToPlaceholder(frame); }}
className="px-2 py-0.5 bg-accent text-white text-[9px] font-bold rounded-full shadow-sm opacity-0 group-hover:opacity-100 transition-opacity hover:bg-blue-700"
>
</button>
<span className="text-accent opacity-0 group-hover:opacity-100 transition-opacity"></span>
</div>
</div>
```
### 样式说明
- 按钮位于底部 `timeFormatted` 与 "可拖拽" 之间,不再覆盖图片。
- `px-2 py-0.5 bg-accent text-white rounded-full shadow-sm`:蓝色实体小胶囊按钮,与 "可拖拽" 蓝色一致。
- `hover:bg-blue-700`:加深蓝色反馈。
- `opacity-0 group-hover:opacity-100 transition-opacity`:保持 hover 显隐与 "可拖拽" 同步。
---
## 风险点及应对策略
| 风险 | 影响 | 应对策略 |
|------|------|----------|
| 按钮太小导致不易点击 | 体验下降 | 保留 `px-2 py-0.5` 的实体按钮,比纯文字链接更易点击 |
| 底部文字区域变宽 | 布局错乱 | 使用 `flex items-center gap-2` 控制间距,保持在一行内 |
---
## 改动范围总结
- 纯 JSX 结构和样式的微调整,不修改任何逻辑函数。
- 不引入新依赖。

View File

@@ -1,118 +0,0 @@
# 实现方案 — 2026-04-16-18-51-06
## 根因分析
`ReportEditor.tsx` 中使用了 `stateRef.current` 作为「草稿自动保存」的数据来源。组件卸载时,`saveDraftToStorage()` 会将 `stateRef.current` 写入 `localStorage`
但在页面初始化(`useEffect``useLayoutEffect`)从已保存报告或 draft 恢复数据时,仅通过 `setState` 更新了 React state**没有同步更新 `stateRef.current` 中的 `videos``capturedFrames` 字段**。
这导致:
1. 用户首次进入 `/report-editor` 时,数据从 localStorage 正确恢复;
2. 用户离开页面时,`stateRef.current` 仍保存着初始的空数组;
3. 组件卸载触发的 `saveDraftToStorage()` 用空数组覆盖了 draft
4. 用户再次返回 `/report-editor` 时,系统优先读取被覆盖后的 draft导致视频分析数据全部丢失。
## 修改文件清单
| 文件 | 修改类型 | 说明 |
|------|---------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 在 4 处数据恢复逻辑后追加 `stateRef.current` 同步赋值 |
## 具体代码变更
### 修改点 1初始化 useEffect — 从 draft 恢复已有报告(约第 128 行后)
在已有代码:
```tsx
setLoadedTemplateId(draft.loadedTemplateId || '');
stateRef.current = { ...stateRef.current, loadedTemplateId: draft.loadedTemplateId || '' };
```
**追加同步**
```tsx
stateRef.current = {
...stateRef.current,
reportData: draft.reportData,
videos: draft.videos,
capturedFrames: draft.capturedFrames,
loadedTemplateId: draft.loadedTemplateId || ''
};
```
### 修改点 2初始化 useEffect — 从已保存报告found恢复约第 146 行后)
在设置完 `contentLoadedRef.current = true;` 之后,**追加同步**
```tsx
stateRef.current = {
...stateRef.current,
reportData: found,
videos: found.videos || [],
capturedFrames: found.capturedFrames || []
};
```
### 修改点 3初始化 useEffect — 从 draft 恢复新建报告(约第 176 行后)
与修改点 1 类似,在 `setLoadedTemplateId(draft.loadedTemplateId || '');` 之后,**追加同步**
```tsx
stateRef.current = {
...stateRef.current,
reportData: draft.reportData,
videos: draft.videos,
capturedFrames: draft.capturedFrames,
loadedTemplateId: draft.loadedTemplateId || ''
};
```
### 修改点 4useLayoutEffect 安全网 — 从 draft 恢复已有报告(约第 677 行后)
`setLoadedTemplateId(draft.loadedTemplateId || '');` 之后,**追加同步**
```tsx
stateRef.current = {
...stateRef.current,
reportData: draft.reportData,
videos: draft.videos,
capturedFrames: draft.capturedFrames,
loadedTemplateId: draft.loadedTemplateId || ''
};
```
### 修改点 5useLayoutEffect 安全网 — 从已保存报告found恢复约第 692 行后)
`contentLoadedRef.current = true;` 之后,**追加同步**
```tsx
stateRef.current = {
...stateRef.current,
reportData: found,
videos: found.videos || [],
capturedFrames: found.capturedFrames || []
};
```
### 修改点 6useLayoutEffect 安全网 — 从 draft 恢复新建报告(约第 701 行后)
`setLoadedTemplateId(draft.loadedTemplateId || '');` 之后,**追加同步**
```tsx
stateRef.current = {
...stateRef.current,
reportData: draft.reportData,
videos: draft.videos,
capturedFrames: draft.capturedFrames,
loadedTemplateId: draft.loadedTemplateId || ''
};
```
## 风险点
| 风险 | 级别 | 应对措施 |
|------|------|---------|
| `stateRef` 仍可能在其他未覆盖路径中不同步 | 低 | 已检查所有数据恢复入口init effect + layout effect后续若新增恢复逻辑需保持同步习惯 |
| `found.videos` / `found.capturedFrames` 为 undefined | 低 | 代码中使用 `|| []` 做防御性处理 |
## 回滚策略
本次修改仅增加 `stateRef.current` 的同步赋值语句,不涉及删除或重构现有逻辑。如出现异常,可直接 `git revert` 回滚。
---
**⚠️ 请审核以上方案,确认无误后回复「确认」或提出修改意见,我将进入测试方案编写阶段。**

View File

@@ -1,132 +0,0 @@
# 实现方案 — 2026-04-16-19-06-18
## 根因分析
### 问题 1路由切换后所有内容丢失
`ReportEditor.tsx` 的初始化 `useEffect` 在从 draft 恢复数据时,将 `stateRef.current` 的同步代码放在了 `if (editorRef.current && draft.content.trim().length > 0)` 条件块的**内部**。
这导致当以下任一情况发生时,`stateRef.current` 不会被同步:
- `editorRef.current` 在 useEffect 执行时尚未挂载(组件首次渲染时常见);
- `draft.content` 为空字符串或仅包含空白(新建报告时常见)。
`stateRef.current` 保持为初始值(空的 `reportData`、空的 `videos`、空的 `capturedFrames`)时,用户离开页面触发的 `saveDraftToStorage()` 会**用这些空值覆盖 localStorage 中的 draft**。再次返回 `/report-editor` 时,系统读取了这个被清空的 draft导致所有内容全部丢失。
### 问题 2自动帧插入 UI 批量刷新
`autoCaptureFrames` 是一个 `async` 函数,内部通过 `for` 循环逐帧处理。循环中每次摘取到新帧后都会调用 `setCapturedFrames(accumulatedFrames)`,但由于 React 18 的**自动批处理**机制在异步函数中连续调用的状态更新会被合并DOM 重渲染被推迟到整个循环结束后才执行一次。因此用户看不到逐帧实时摘取的过程,只有等全部帧处理完后,关键帧列表和 placeholder 中的图片才会一次性批量出现。
## 修改文件清单
| 文件 | 修改类型 | 说明 |
|------|---------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 移动 stateRef 同步位置 + 引入 `flushSync` |
## 具体代码变更
### 变更 1useEffect — draft 恢复已有报告(约第 123 行区域)
**当前代码(有问题的结构):**
```tsx
if (editorRef.current && typeof draft.content === 'string' && draft.content.trim().length > 0) {
editorRef.current.innerHTML = draft.content;
// ...
stateRef.current = { ...stateRef.current, reportData: draft.reportData, ... };
}
```
**修改为:**
`stateRef.current` 的赋值提取到 `if (editorRef.current && ...)` 的**外部**,紧接在 `setCapturedFrames` 之后:
```tsx
if (draft.activeTab) setActiveTab(draft.activeTab);
stateRef.current = {
...stateRef.current,
reportData: draft.reportData,
videos: draft.videos,
capturedFrames: draft.capturedFrames,
loadedTemplateId: draft.loadedTemplateId || ''
};
if (editorRef.current && typeof draft.content === 'string' && draft.content.trim().length > 0) {
editorRef.current.innerHTML = draft.content;
contentRef.current = draft.content;
contentLoadedRef.current = true;
setLoadedTemplateId(draft.loadedTemplateId || '');
setTimeout(() => updatePageHeight(), 0);
}
```
### 变更 2useEffect — found 恢复已有报告(约第 141 行区域)
**修改为:**
`stateRef.current` 的赋值提取到 `if (editorRef.current)` 的**外部**,紧接在 `setReportData(found)` 之后:
```tsx
setReportData(found);
stateRef.current = {
...stateRef.current,
reportData: found,
videos: found.videos || [],
capturedFrames: found.capturedFrames || []
};
if (editorRef.current) {
// ... 恢复 editor content ...
}
```
同时顺手清理重复赋值 `contentRef.current = found.content;`(第 149-150 行重复)。
### 变更 3useEffect — draft 恢复新建报告(约第 184 行区域)
**修改为:**
与变更 1 类似,将 `stateRef.current` 的赋值提取到 `if (editorRef.current && ...)` 的**外部**
```tsx
if (draft.activeTab) setActiveTab(draft.activeTab);
stateRef.current = {
...stateRef.current,
reportData: draft.reportData,
videos: draft.videos,
capturedFrames: draft.capturedFrames,
loadedTemplateId: draft.loadedTemplateId || ''
};
if (editorRef.current && typeof draft.content === 'string' && draft.content.trim().length > 0) {
// ... 恢复 editor content ...
}
```
### 变更 4autoCaptureFrames 引入 flushSync约第 519-521 行区域)
在文件顶部增加导入:
```tsx
import { flushSync } from 'react-dom';
```
`autoCaptureFrames` 的 for 循环中,每次更新 `capturedFrames` 时使用 `flushSync` 强制同步渲染:
**当前代码:**
```tsx
setCapturedFrames(accumulatedFrames);
stateRef.current = { ...stateRef.current, capturedFrames: accumulatedFrames };
```
**修改为:**
```tsx
flushSync(() => {
setCapturedFrames(accumulatedFrames);
});
stateRef.current = { ...stateRef.current, capturedFrames: accumulatedFrames };
```
这样每一帧被摘取后都会立即触发 DOM 更新,用户可以在右侧「视频分析」面板中实时看到关键帧逐张出现,同时自动插入逻辑也能按配置延迟后逐张填充 placeholder。
## 风险点
| 风险 | 级别 | 应对措施 |
|------|------|---------|
| `flushSync` 在循环中可能导致轻微渲染卡顿 | 低 | 关键帧数量通常仅 10~20 张,现代浏览器完全可以承受;已在 async 函数中使用,不会阻塞视频 seek 事件 |
| 移动 `stateRef` 同步位置可能与其他逻辑产生时序影响 | 低 | 仅将同步从 `if` 内部移到外部,逻辑语义完全一致,只是确保在所有路径下都被执行 |
| `found` 分支中 `contentRef.current` 重复赋值 | 极低 | 顺手清理,不影响行为 |
## 回滚策略
本次修改仅调整代码执行顺序并引入一个 React 标准 API`flushSync`),未改变数据结构或接口。如出现异常,可直接 `git revert` 回滚。
---
**⚠️ 请审核以上方案,确认无误后回复「确认」或提出修改意见,我将进入测试方案编写阶段。**

View File

@@ -1,25 +0,0 @@
# 实现方案 — 2026-04-16-19-18-14
## 部署步骤
1. **构建生产包**:运行 `npm run build` 先本地验证构建是否通过(可选但推荐)。
2. **停止旧容器**`docker-compose down` 停止并移除当前运行的 `medical-report-app` 容器。
3. **重新构建镜像**`docker-compose build --no-cache` 基于最新代码重新构建 Docker 镜像。
4. **启动新容器**`docker-compose up -d` 后台启动新容器。
5. **验证部署**:检查容器状态 `docker ps`,并尝试访问 `http://localhost:8080` 确认应用正常。
## 修改文件清单
无需修改源代码,仅执行构建和容器操作。
## 风险点
| 风险 | 级别 | 应对措施 |
|------|------|---------|
| 构建失败 | 低 | 本地已执行 `tsc --noEmit` 通过,构建风险低 |
| 端口 8080 被占用 | 低 | `docker-compose down` 会先释放旧容器占用的端口 |
| Docker 未安装/未启动 | 中 | 如遇报错,根据错误信息处理 |
## 回滚策略
若部署后异常,可执行 `docker-compose down` 后回退到上一个可用的 Git commit 再重新构建。

View File

@@ -1,141 +0,0 @@
# 实现方案 — 2026-04-16-19-28-04
## 根因分析
当前 `ReportEditor.tsx` 的自动保存机制过度依赖两个 `useRef``stateRef``contentRef`)作为"数据快照"
1. **用户操作时**:各事件处理器先更新 React state再手动同步 `stateRef.current`,然后调用 `saveDraftToStorage()` 写入 localStorage。
2. **组件卸载时**`useEffect` 的 cleanup 调用 `save()`,同样读取 `stateRef.current``contentRef.current` 来保存 draft。
这个机制存在致命缺陷:
- **React 18 `StrictMode`** 在开发/预览环境下会"挂载 → 立即卸载 → 重新挂载"。在首次模拟卸载时,`stateRef.current` 还是组件创建时的初始空值(`videos: []``capturedFrames: []``reportData: 默认值`。cleanup 中的 `save()` 会用这个空值 **覆盖** localStorage 里已经存在的正确 draft。
- 即使不在 StrictMode 下,只要任何恢复路径或用户操作路径遗漏了 `stateRef.current` 的同步,卸载保存时就会丢失数据。
此前两次修复只把 `stateRef.current` 同步移到了更多恢复分支中,但**没有从根本上消除对 `stateRef` 的依赖**,因此问题依旧。
## 修改文件清单
| 文件 | 修改类型 | 说明 |
|------|---------|------|
| `src/pages/ReportEditor.tsx` | 修改 | 重构 `saveDraftToStorage` + 自动保存 effect + `useLayoutEffect` 依赖 |
## 具体代码变更
### 变更 1重构 `saveDraftToStorage`
**当前实现(依赖 ref**
```tsx
const saveDraftToStorage = React.useCallback(() => {
const user = storage.get<User | null>('currentUser', null);
const key = user ? `reportEditorDraft_${user.username}` : '';
if (key) {
storage.set(key, {
content: contentRef.current,
draftReportId: reportId || null,
...stateRef.current
});
}
}, [reportId]);
```
**修改为(直接从最新 state 和 DOM 读取):**
```tsx
const saveDraftToStorage = React.useCallback(() => {
const user = storage.get<User | null>('currentUser', null);
const key = user ? `reportEditorDraft_${user.username}` : '';
if (key) {
storage.set(key, {
content: editorRef.current?.innerHTML || '',
draftReportId: reportId || null,
reportData,
videos,
capturedFrames,
activeTab,
loadedTemplateId
});
}
}, [reportData, videos, capturedFrames, activeTab, loadedTemplateId, reportId]);
```
**效果**
- `saveDraftToStorage` 的闭包永远绑定当前渲染周期的最新 state不再需要 `stateRef` 作为中转。
- 编辑器内容直接从 `editorRef.current?.innerHTML` 读取,不再依赖可能滞后的 `contentRef`
- 用户操作中所有调用 `saveDraftToStorage()` 的地方(表单 onChange、上传视频、截图等会自动生效。
### 变更 2重构自动保存 effect
**当前实现:**
```tsx
useEffect(() => {
const save = () => { ... };
window.addEventListener('beforeunload', save);
document.addEventListener('visibilitychange', save);
return () => {
window.removeEventListener('beforeunload', save);
document.removeEventListener('visibilitychange', save);
save();
};
}, [reportId]);
```
**修改为:**
```tsx
useEffect(() => {
const handleBeforeUnload = () => saveDraftToStorage();
const handleVisibilityChange = () => {
if (document.visibilityState === 'hidden') saveDraftToStorage();
};
window.addEventListener('beforeunload', handleBeforeUnload);
document.addEventListener('visibilitychange', handleVisibilityChange);
return () => {
window.removeEventListener('beforeunload', handleBeforeUnload);
document.removeEventListener('visibilitychange', handleVisibilityChange);
saveDraftToStorage();
};
}, [saveDraftToStorage]);
```
**效果**
- 由于 `saveDraftToStorage` 的 dependency 包含了所有关键 state每次 state 变化后 effect 都会重新注册,但最重要的是 **cleanup 中调用的 `saveDraftToStorage` 永远指向最新的闭包**,不会因为 ref 滞后而用空值覆盖 draft。
### 变更 3给 `useLayoutEffect` 安全网添加 `[]` 依赖
**当前:**
```tsx
React.useLayoutEffect(() => {
if (contentLoadedRef.current || !editorRef.current) return;
// ... 恢复逻辑 ...
});
```
**修改为:**
```tsx
React.useLayoutEffect(() => {
if (contentLoadedRef.current || !editorRef.current) return;
// ... 恢复逻辑 ...
}, []);
```
**效果**
- 避免在每次渲染后重复执行安全网,防止潜在的意外覆盖或性能损耗。
- 保留其原有功能:仅在组件挂载时,如果 `useEffect` 初始化因 ref 未 ready 未能恢复内容,则作为兜底恢复编辑器 DOM。
### 变更 4可选但建议移除对 `contentRef` 的强依赖
`contentRef` 在旧代码中用于 draft 保存。修改 `saveDraftToStorage` 后直接读取 `editorRef.current?.innerHTML``contentRef` 仍可保留供其他旧代码使用,不影响功能。
## 风险点
| 风险 | 级别 | 应对措施 |
|------|------|---------|
| `saveDraftToStorage` dependency 数组较长,可能导致 effect 频繁重新注册 | 低 | 重新注册事件监听器的开销极小,远小于 localStorage 写入本身 |
| `editorRef.current?.innerHTML` 在卸载时读取可能拿到不完整 DOM | 极低 | `editorRef` 指向的 DOM 在 cleanup 执行时尚未被 React 移除,内容完整 |
| `useLayoutEffect` 添加 `[]` 后闭包值陈旧 | 低 | `useLayoutEffect` 内部仅读取 `reportId`(来自 URL在生命周期内不变和 localStorage不受影响 |
## 回滚策略
本次修改仅重构保存函数的实现方式,不改变数据结构或存储 key。如出现异常可直接 `git revert` 回滚。
---
**⚠️ 请审核以上方案,确认无误后回复「确认」或提出修改意见,我将进入测试方案编写阶段。**

View File

@@ -1,71 +0,0 @@
# 项目修改工作流指南
> 本工作流适用于所有项目修改相关需求。每次收到修改需求时,必须严格按照以下步骤执行。
---
## 工作流步骤
### Step 0: 备份与记录时间戳
每次执行前,必须先用 Gitea 进行代码备份,并记录问题开始时间:
- 时间戳格式:`{Year}-{Mon}-{Day}-{Hour}-{Min}-{Sec}`
- 备份命令:
```bash
git init
git checkout -b main
git add .
git commit -m "backup before modification at {时间戳}"
git remote add origin http://192.168.31.5:5002/admin/Mdeical_Sur_Report.git
git push -u origin main
```
- 若远程已存在,则使用 `git remote set-url origin http://192.168.31.5:5002/admin/Mdeical_Sur_Report.git`
### Step 1: 工程分析目录检查
确保 `..\工程分析` 文件夹存在(已创建)。
### Step 2: 需求分析文档
将用户提出的需求整理后写入:
```
.\工程分析\需求分析-{时间戳}.md
```
- 内容需包含:需求背景、功能目标、涉及页面/模块、验收标准
### Step 3: 实现方案文档
基于需求分析,编写详细实现方案并写入:
```
.\工程分析\实现方案-{时间戳}.md
```
- 内容需包含:技术思路、修改文件清单、关键代码变更说明、风险点
- **写完此文档后,必须暂停并交由用户进行二次人工审核确认。未经确认不得继续。**
### Step 4: 测试方案文档
实现方案确认后,编写测试方案并写入:
```
.\工程分析\测试方案-{时间戳}.md
```
- 内容需包含:测试项、测试步骤、预期结果、回归验证范围
- **写完此文档后,必须暂停并交由用户进行二次人工审核确认。未经确认不得继续。**
### Step 5: 执行修改与经验记录
测试方案确认后,开始执行代码修改:
1. 按方案实施修改
2. 执行 `npm run lint` 进行类型检查
3. 如有必要,执行 `npm run build` 验证构建
4. 修改完成后,在以下文档中追加本次执行过程中的关键问题及解决方案:
```
.\工程分析\经验记录.md
```
- 记录格式(四段式):
- **A. 具体问题**
- **B. 产生问题原因**
- **C. 解决问题方案**
- **D. 后续如何避免问题**
---
## 快捷入口
- **工程分析目录**: `C:\Users\Administrator\Downloads\Gemini-图文报告系统-V1.1\工程分析`
- **Gitea 远程地址**: `http://192.168.31.5:5002/admin/Mdeical_Sur_Report.git`
- **类型检查命令**: `npm run lint`
- **构建命令**: `npm run build`

View File

@@ -1,76 +0,0 @@
# 测试方案 — 2026-04-16-16-51-00
## 测试环境准备
1. 确保项目依赖已安装:`npm install` 已完成。
2. 使用测试账号 `admin / 123456`(超级管理员)登录系统。
3.**系统设置** 页面中,确认 **图文报告生成默认模板** 已设置为 **"腹腔镜胆囊切除术报告"**。
---
## 测试项清单
### 测试项 1新建报告时正确加载默认模板
**测试步骤**
1. 登录后,点击左侧菜单 **"图文报告生成"**`/report-editor`,无 `id` 参数)。
2. 观察页面顶部 **"当前模板(及重置模板):"** 下拉框显示内容。
3. 观察中间编辑区域(`editor-content-wrapper print-wrapper`)是否有模板内容。
**预期结果**
- 顶部模板选择器显示 **"腹腔镜胆囊切除术报告"**(或用户设置的默认模板名称),而非"无"。
- 编辑区域显示该模板的完整 HTML 内容(包含标题、表格、图片占位符等),不是纯白色空白。
---
### 测试项 2从其他页面返回后未编辑不显示空白
**测试步骤**
1.**工作台**`/dashboard`)页面停留。
2. 通过左侧菜单再次进入 **"图文报告生成"**。
3. 不要输入任何内容,直接再切回 **工作台**,然后再切回 **"图文报告生成"**。
**预期结果**
- 每次进入 `/report-editor`,编辑区域都应正确显示默认模板内容。
- 不会出现白色空白页面。
---
### 测试项 3编辑已有报告时不被空白草稿覆盖
**测试步骤**
1. 进入 **报告管理**打开一份已有内容的报告进行编辑URL 带有 `?id=xxx`)。
2. 不要做任何修改,直接刷新浏览器页面。
**预期结果**
- 页面加载后显示该报告原有的完整内容,不会被空白草稿覆盖为默认模板。
---
### 测试项 4有效草稿恢复后模板选择器显示正确
**测试步骤**
1. 进入 **"图文报告生成"**,确认已加载默认模板。
2. 在编辑器中随意输入几个字(确保内容非空)。
3. 切到 **工作台**,再切回 **"图文报告生成"**。
**预期结果**
- 编辑器恢复刚才输入的内容。
- 顶部模板选择器仍显示 **"腹腔镜胆囊切除术报告"**(因为草稿中已保存 `loadedTemplateId`)。
---
### 测试项 5构建与类型检查回归
**测试步骤**
1. 在项目根目录执行:
```bash
npm run lint
npm run build
```
**预期结果**
- `npm run lint` 无 TypeScript 编译错误。
- `npm run build` 构建成功,生成 `dist/` 目录。
---
## 回归验证范围
- [ ] `SystemSettings.tsx` 未被修改,默认模板设置功能保持正常。
- [ ] `storage.ts` 未被修改localStorage 读写保持正常。
- [ ] 报告保存(草稿/完成)功能未被破坏。
- [ ] 视频分析面板与编辑器交互保持正常。

View File

@@ -1,88 +0,0 @@
# 测试方案 — 2026-04-16-17-07-04
## 测试环境准备
1. 项目已构建通过,无类型错误。
2. 使用测试账号 `admin / 123456`(超级管理员)登录系统。
3. 进入 **图文报告生成** 页面,确认编辑器中已加载默认模板(包含若干 `image-placeholder` 图片占位符)。
4.**视频分析** 面板中上传至少一个手术视频,等待系统自动抽帧完成。
---
## 测试项清单
### 测试项 1"插入" 按钮显隐行为正确
**测试步骤**
1. 在视频分析面板中查看自动抽帧的关键帧列表。
2. 将鼠标移入某张关键帧截图卡片上。
3. 再将鼠标移出该卡片。
**预期结果**
- 鼠标移入时,截图卡片底部同时显示 **"插入"** 和 **"可拖拽"** 两个提示文字。
- 鼠标移出时,两者同时隐藏。
- "插入" 位于截图时间(如 `00:15`)和 "可拖拽" 之间。
---
### 测试项 2点击 "插入" 自动填充第一个空置占位符
**测试步骤**
1. 确认编辑器中存在至少一个尚未填充图片的 `.image-placeholder`(未显示图片的灰色虚线框)。
2. 在视频分析面板中,将鼠标悬停到第一张关键帧截图上,点击 **"插入"**。
3. 再次点击第二张关键帧截图的 **"插入"**。
**预期结果**
- 第一次点击后,编辑器中**第一个**空置占位符被填充为第一张关键帧图片。
- 第二次点击后,编辑器中**第二个**空置占位符被填充为第二张关键帧图片。
- 填充后的占位符变为正常图片显示,且带有删除按钮 `×`
---
### 测试项 3无空占位符时给出提示
**测试步骤**
1. 连续点击 "插入" 直到编辑器中所有 `image-placeholder` 都被填满。
2. 再次点击任意关键帧的 **"插入"**。
**预期结果**
- 浏览器弹出提示框:**"没有可插入图片的空位"**。
- 不会报错,现有已填充的图片不受影响。
---
### 测试项 4点击 "插入" 不触发视频跳转
**测试步骤**
1. 点击某张关键帧卡片的 **"插入"** 按钮。
2. 观察左侧视频播放器。
**预期结果**
- 视频播放器的当前时间**不发生变化**(不会因为点击了卡片本身而跳转到该帧位置)。
---
### 测试项 5原有拖拽功能保持正常
**测试步骤**
1. 手动拖拽视频分析面板中的某张关键帧截图到编辑器中的空置 `image-placeholder` 上。
**预期结果**
- 拖拽释放后,该占位符正确显示被拖拽的图片。
- 拖拽功能与 "插入" 按钮功能互不干扰。
---
### 测试项 6构建与类型检查回归
**测试步骤**
1. 在项目根目录执行:
```bash
npm run lint
npm run build
```
**预期结果**
- `npm run lint` 无 TypeScript 编译错误。
- `npm run build` 构建成功。
---
## 回归验证范围
- [ ] `SystemSettings.tsx` 未被修改,系统设置功能保持正常。
- [ ] `storage.ts` 未被修改,存储读写保持正常。
- [ ] 编辑器打印、保存草稿、完成报告功能保持正常。
- [ ] 关键帧的 "手动截取"、删除功能保持正常。

View File

@@ -1,73 +0,0 @@
# 测试方案 — 2026-04-16-17-15-37
## 测试环境准备
1. 项目已完成上一次构建,无类型错误。
2. 使用测试账号 `admin / 123456`(超级管理员)登录系统。
3. 进入 **图文报告生成** 页面并上传视频完成抽帧,确保视频分析面板有关键帧截图。
---
## 测试项清单
### 测试项 1"插入" 按钮位于图片中央且为实体按钮样式
**测试步骤**
1. 将鼠标悬停到任意关键帧截图卡片上。
2. 观察按钮的位置、颜色、形状。
**预期结果**
- 按钮显示在图片的**正中央**。
- 按钮是**实体胶囊按钮**(带背景色、圆角、阴影),不是纯文字链接。
- 按钮颜色为**绿色系**`emerald-500`),与底部 "可拖拽" 的蓝色明显不同。
---
### 测试项 2"插入" 按钮显隐行为正确
**测试步骤**
1. 鼠标移入关键帧卡片。
2. 鼠标移出关键帧卡片。
**预期结果**
- 移入时按钮与 "可拖拽" 文字同时淡入显示。
- 移出时按钮与 "可拖拽" 文字同时淡出隐藏。
- 未悬停时图片上无遮挡文字。
---
### 测试项 3点击按钮功能保持正常
**测试步骤**
1. 鼠标悬停到关键帧卡片,点击中央的 **"插入"** 按钮。
**预期结果**
- 编辑器中第一个空置的 `image-placeholder` 被填充为该帧图片。
- 视频播放器时间不发生变化(未触发卡片跳转)。
- 若无可插入空位,弹出 `没有可插入图片的空位` 提示。
---
### 测试项 4"手动" 徽章不被遮挡
**测试步骤**
1. 查看手动截取的关键帧截图。
**预期结果**
- 左上角的 **"手动"** 徽章清晰可见,未被中央按钮遮挡。
---
### 测试项 5构建与类型检查回归
**测试步骤**
1. 在项目根目录执行:
```bash
npm run lint
npm run build
```
**预期结果**
- `npm run lint` 无 TypeScript 编译错误。
- `npm run build` 构建成功。
---
## 回归验证范围
- [ ] 拖拽插入功能未被破坏。
- [ ] 关键帧卡片的删除、视频跳转功能正常。
- [ ] 编辑器保存、打印功能正常。

View File

@@ -1,72 +0,0 @@
# 测试方案 — 2026-04-16-17-21-58
## 测试环境准备
1. 项目已完成上一次构建,无类型错误。
2. 使用测试账号 `admin / 123456`(超级管理员)登录系统。
3. 进入 **图文报告生成** 页面并上传视频完成抽帧,确保视频分析面板有关键帧截图。
---
## 测试项清单
### 测试项 1"插入" 按钮位于底部正确位置且不遮盖图片
**测试步骤**
1. 将鼠标悬停到任意关键帧截图卡片上。
2. 观察 "插入" 按钮的位置。
**预期结果**
- 按钮位于卡片底部,**时间格式(如 `00:15`**和 **"可拖拽"** 提示之间。
- 按钮**不覆盖在图片上**,图片内容完全可见,无遮挡。
---
### 测试项 2"插入" 按钮为蓝色实体按钮样式
**测试步骤**
1. 鼠标悬停到关键帧卡片,观察 "插入" 按钮的颜色和形状。
**预期结果**
- 按钮为**蓝色实体胶囊按钮**`bg-accent` 背景 + 白色文字 + 圆角 + 轻微阴影)。
- 按钮颜色与底部 "可拖拽" 的蓝色一致,视觉上协调统一。
- 不是纯文字链接样式。
---
### 测试项 3显隐行为与 "可拖拽" 同步
**测试步骤**
1. 鼠标移入关键帧卡片。
2. 鼠标移出关键帧卡片。
**预期结果**
- 移入时,"插入" 按钮与 "可拖拽" 文字**同时淡入显示**。
- 移出时,"插入" 按钮与 "可拖拽" 文字**同时淡出隐藏**。
---
### 测试项 4点击功能与事件冒泡正常
**测试步骤**
1. 鼠标悬停到关键帧卡片,点击 **"插入"** 按钮。
**预期结果**
- 编辑器中第一个空置的 `image-placeholder` 被自动填充(或弹出 `没有可插入图片的空位` 提示)。
- 视频播放器时间不发生变化(未触发卡片跳转)。
---
### 测试项 5构建与类型检查回归
**测试步骤**
1. 在项目根目录执行:
```bash
npm run lint
npm run build
```
**预期结果**
- `npm run lint` 无 TypeScript 编译错误。
- `npm run build` 构建成功。
---
## 回归验证范围
- [ ] 拖拽插入功能未被破坏。
- [ ] 关键帧卡片的删除、手动徽章显示正常。
- [ ] 编辑器保存、打印功能正常。

View File

@@ -1,90 +0,0 @@
# 测试方案 — 2026-04-16-18-51-06
## 测试目标
验证在 `/report-editor` 页面离开并重新返回后,**视频分析相关的所有图片数据(自动关键帧、手动截图、拖拽到 placeholder 的截图)能够正确恢复**,且不影响报告基本信息和其他页面功能。
## 测试环境
- 浏览器Chrome / Edge推荐
- 前置条件已登录系统localStorage 中有当前用户信息
- 测试文件:准备一个时长超过 30 秒的 MP4 视频文件(用于自动关键帧摘取)
## 测试用例设计
### 用例 1新建报告 — 自动关键帧摘取后路由切换
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 1.1 | 进入 `/report-editor`(不带 `?id` | 页面正常加载,显示默认模板内容 |
| 1.2 | 填写患者姓名、住院号等基本信息 | 基本信息输入正常 |
| 1.3 | 切换到「视频分析」页签,上传测试视频 | 视频上传成功,视频列表中显示文件名 |
| 1.4 | 点击「自动关键帧摘取」 | 右侧生成多张自动关键帧缩略图 |
| 1.5 | 点击浏览器地址栏,手动跳转至 `/report-manage` | 页面跳转成功 |
| 1.6 | 再次在地址栏输入 `/report-editor` 返回 | **右侧「视频分析」中自动关键帧缩略图全部保留** |
| 1.7 | 点击「保存草稿」,再跳转离开并返回 | 自动关键帧缩略图仍然保留 |
### 用例 2新建报告 — 手动截图后路由切换
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 2.1 | 在新建报告页面上传视频并播放 | 视频正常播放 |
| 2.2 | 拖动进度条到某一时刻,点击「手动截图」 | 右侧生成一张手动截图缩略图 |
| 2.3 | 再次截取 2-3 张不同时间点的截图 | 所有手动截图均显示在右侧列表 |
| 2.4 | 跳转至 `/report-manage`,再返回 `/report-editor` | **所有手动截图缩略图全部保留** |
| 2.5 | 点击「保存草稿」后再次离开并返回 | 手动截图仍然保留 |
### 用例 3新建/编辑报告 — 拖拽截图到 image-placeholder 后路由切换
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 3.1 | 在编辑器中插入一个 `image-placeholder` | placeholder 正常显示在编辑器中 |
| 3.2 | 从右侧「视频分析」中拖拽一张自动关键帧到 placeholder | placeholder 中显示该图片,且带有删除按钮 |
| 3.3 | 再插入一个 placeholder拖拽一张手动截图到其中 | 第二张 placeholder 也正确显示图片 |
| 3.4 | 跳转至 `/report-manage`,再返回 `/report-editor` | **编辑器中两个 placeholder 内的图片均保留可见** |
| 3.5 | 查看右侧「视频分析」面板 | 被拖拽的原始帧/截图缩略图也仍然保留在列表中 |
| 3.6 | 点击「保存草稿」后再次离开并返回 | 编辑器和右侧面板的图片均保留 |
### 用例 4编辑已有报告 — 保存后数据完整恢复
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 4.1 | 对任意一份已有报告点击「编辑」,进入 `/report-editor?id=xxx` | 报告内容和基本信息正常加载 |
| 4.2 | 上传视频、自动摘取关键帧、手动截图、拖拽一张到 placeholder | 所有操作正常生效 |
| 4.3 | 点击「保存草稿」 | 提示保存成功 |
| 4.4 | 跳转至 `/report-manage`,找到该报告,再次点击「编辑」 | 进入 `/report-editor?id=xxx` |
| 4.5 | 检查编辑器、基本信息、视频分析面板 | **所有数据和图片完整恢复,无丢失** |
### 用例 5边界场景 — 多次快速路由切换
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 5.1 | 在 `/report-editor` 中完成视频上传、截图、拖拽 | 数据正常 |
| 5.2 | 快速连续切换:/report-editor → /report-manage → /report-editor → /report-manage → /report-editor | **最终返回时,所有视频分析数据仍然完整保留** |
| 5.3 | 检查 localStorage 中 `reportEditorDraft_{username}` | draft 中 `videos``capturedFrames` 均非空数组 |
### 用例 6回归测试 — 模板切换不污染数据
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 6.1 | 在 `/report-editor` 中上传视频并截取若干帧 | 数据正常 |
| 6.2 | 切换模板(顶部模板选择下拉框) | 编辑器内容按模板重置,报告基本信息清空 |
| 6.3 | 检查视频分析面板 | 根据现有逻辑,模板切换会清空 videos 和 capturedFrames此行为保持不变 |
| 6.4 | 若切换模板后不希望丢失视频数据,可后续作为优化项提出 | — |
## 验收标准
- [ ] 用例 1自动关键帧在路由切换后 100% 保留;
- [ ] 用例 2手动截图在路由切换后 100% 保留;
- [ ] 用例 3拖拽到 placeholder 的图片在路由切换后 100% 保留;
- [ ] 用例 4编辑已有报告保存后再次编辑数据完整无丢失
- [ ] 用例 5多次快速切换路由后数据不丢失、不异常
- [ ] 用例 6模板切换的现有行为未被意外改变。
## 测试方式
由于本项目目前无自动化测试框架,所有测试用例均通过 **手工浏览器验证** 执行。测试人员按上表逐步操作,观察实际结果是否与预期一致。
---
**⚠️ 请审核以上测试方案,确认无误后回复「确认」或提出修改意见,我将进入最终执行阶段(修改代码 + 更新经验记录 + Gitea 备份)。**

View File

@@ -1,89 +0,0 @@
# 测试方案 — 2026-04-16-19-06-18
## 测试目标
1. 验证路由切换后,报告编辑器内容、基本信息、视频分析数据(自动关键帧、手动截图、拖拽图片)均**完整保留不丢失**。
2. 验证开启「自动帧插入」后,关键帧在自动摘取过程中能够**实时逐张显示**,并在达到延迟时间后**逐张插入**到 `image-placeholder` 中,而不是全部处理完成后一次性批量出现。
## 测试环境
- 浏览器Chrome / Edge推荐
- 前置条件已登录系统localStorage 中有当前用户信息
- 测试文件:准备一个时长超过 60 秒的 MP4 视频文件
- 系统设置:在「系统设置」中开启「自动帧插入」,配置插入延迟为 1~2 秒,并勾选若干自动插入的帧索引(如第 0、3、5 帧)
## 测试用例设计
### 用例 1新建报告 — 路由切换后数据完整保留
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 1.1 | 进入 `/report-editor`(不带 `?id` | 页面正常加载默认模板 |
| 1.2 | 填写患者姓名、住院号、科室等基本信息 | 输入内容正常保留在表单中 |
| 1.3 | 在编辑器中输入一段自定义文字 | 编辑器内容正常显示 |
| 1.4 | 切换到「视频分析」页签,上传测试视频 | 视频出现在右侧列表 |
| 1.5 | 点击「自动关键帧摘取」 | 右侧出现多张自动关键帧缩略图 |
| 1.6 | 手动截取 2 张截图 | 手动截图出现在右侧列表 |
| 1.7 | 拖拽 1 张自动关键帧到编辑器的 `image-placeholder` | placeholder 正确显示图片 |
| 1.8 | 在地址栏手动跳转到 `/report-manage` | 页面正常跳转 |
| 1.9 | 再次输入 `/report-editor` 返回新建报告页面 | **编辑器内容、基本信息、自动关键帧、手动截图、placeholder 中的图片均完整保留** |
| 1.10 | 点击「保存草稿」,再次离开并返回 | 所有数据仍然完整 |
### 用例 2编辑已有报告 — 保存后重新编辑数据完整
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 2.1 | 进入 `/report-manage`,对已有报告点击「编辑」 | `/report-editor?id=xxx` 正常加载 |
| 2.2 | 修改患者姓名,上传视频,自动摘取关键帧 | 修改和视频分析数据正常显示 |
| 2.3 | 点击「保存草稿」 | 提示保存成功 |
| 2.4 | 跳转离开后再返回 `/report-editor?id=xxx` | **修改后的基本信息、报告内容、视频分析数据完整恢复** |
### 用例 3边界场景 — 多次快速路由切换
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 3.1 | 在 `/report-editor` 中完成用例 1 的全部操作 | 数据正常 |
| 3.2 | 快速连续切换:/report-editor → /report-manage → /report-editor → /report-manage → /report-editor | 最终返回时,**没有任何数据丢失或变空** |
| 3.3 | 检查 localStorage 中 `reportEditorDraft_{username}` | draft 中 `reportData``videos``capturedFrames` 均非空 |
### 用例 4自动帧插入 — 关键帧实时逐张显示
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 4.1 | 进入系统设置,开启「自动帧插入」,设置延迟 1 秒,勾选索引 0、2、4 为自动插入 | 设置保存成功 |
| 4.2 | 进入 `/report-editor`,上传视频,确保编辑器中有至少 3 个空的 `image-placeholder` | placeholder 就位 |
| 4.3 | 点击「自动关键帧摘取」 | **右侧关键帧列表中,每隔一小段时间(视频 seek 时间)就有新的一张缩略图出现**,而不是等全部结束后一次性全部出现 |
| 4.4 | 观察 placeholder | 当第 0 帧被摘取后,等待约 1 秒,第一个 placeholder 被填充;第 2 帧摘取后约 1 秒,第二个 placeholder 被填充;以此类推,**逐张插入** |
### 用例 5自动帧插入 — 无延迟配置的即时插入
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 5.1 | 在系统设置中将自动插入延迟设为 0 秒 | 设置保存成功 |
| 5.2 | 进入 `/report-editor`,上传视频,点击「自动关键帧摘取」 | 每摘取到一张指定索引的帧,**几乎立即**填充到下一个空的 placeholder 中 |
| 5.3 | 观察右侧关键帧列表 | 同样能看到帧逐张实时出现 |
### 用例 6回归测试 — 模板切换行为不变
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 6.1 | 在 `/report-editor` 中上传视频、截取若干帧 | 数据正常 |
| 6.2 | 切换模板 | 编辑器内容按模板重置,基本信息清空,视频分析根据现有逻辑可能被清空 |
| 6.3 | 离开并返回 | 模板切换后的状态保持一致,无异常崩溃 |
## 验收标准
- [ ] 用例 1新建报告路由切换后编辑器内容、基本信息、视频分析数据 100% 保留;
- [ ] 用例 2编辑已有报告保存后再次编辑数据完整无丢失
- [ ] 用例 3多次快速切换路由后数据不丢失、不异常变空
- [ ] 用例 4开启自动帧插入且有延迟时关键帧实时逐张显示、逐张插入 placeholder
- [ ] 用例 5延迟为 0 时,指定帧摘取后立即插入 placeholder
- [ ] 用例 6模板切换的现有行为未被意外改变页面无崩溃。
## 测试方式
由于本项目目前无自动化测试框架,所有测试用例均通过 **手工浏览器验证** 执行。测试人员按上表逐步操作,观察实际结果是否与预期一致。
---
**⚠️ 请审核以上测试方案,确认无误后回复「确认」或提出修改意见,我将进入最终执行阶段(修改代码 + 更新经验记录 + Gitea 备份)。**

View File

@@ -1,19 +0,0 @@
# 测试方案 — 2026-04-16-19-18-14
## 测试目标
验证应用重新部署后,服务正常启动且可通过端口访问。
## 测试用例
| 用例 | 操作 | 预期结果 |
|------|------|---------|
| 1 | 执行 `docker ps` | 看到 `medical-report-app` 容器在运行,端口映射为 `8080:80` |
| 2 | 浏览器访问 `http://localhost:8080` | 页面正常加载,显示登录或报告管理界面 |
| 3 | 登录后进入 `/report-editor` | 页面正常加载,无白屏或报错 |
## 验收标准
- [ ] 容器运行状态正常
- [ ] 浏览器可正常访问应用首页
- [ ] 核心页面 `/report-editor` 可正常加载

View File

@@ -1,80 +0,0 @@
# 测试方案 — 2026-04-16-19-28-04
## 测试目标
彻底验证路由切换后,报告编辑器内容、基本信息、视频列表、关键帧截图在任何场景下均不再丢失。
## 测试环境
- 浏览器Chrome / Edge
- 前置条件已登录系统localStorage 中有当前用户信息
- 测试文件:准备一个时长超过 30 秒的 MP4 视频文件
## 测试用例设计
### 用例 1新建报告 — 填写基本信息后切换路由
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 1.1 | 进入 `/report-editor`(不带 `?id` | 页面正常加载默认模板 |
| 1.2 | 填写患者姓名、住院号、科室 | 输入内容保留在表单中 |
| 1.3 | 在编辑器中输入文字 | 编辑器内容正常显示 |
| 1.4 | 跳转到 `/report-manage`,再返回 `/report-editor` | **基本信息和编辑器内容均完整保留** |
### 用例 2新建报告 — 上传视频 + 自动关键帧后切换路由
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 2.1 | 在 `/report-editor` 中上传视频 | 视频出现在右侧列表 |
| 2.2 | 点击「自动关键帧摘取」 | 右侧出现多张关键帧缩略图 |
| 2.3 | 手动截取 2 张截图 | 手动截图出现在右侧 |
| 2.4 | 跳转到 `/report-manage`,再返回 `/report-editor` | **视频列表、自动关键帧、手动截图全部保留** |
### 用例 3新建报告 — 拖拽图片到 placeholder 后切换路由
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 3.1 | 插入 `image-placeholder`,拖拽一张关键帧到其中 | placeholder 显示图片 |
| 3.2 | 跳转到 `/report-manage`,再返回 `/report-editor` | **placeholder 中的图片保留,右侧关键帧列表也保留** |
### 用例 4编辑已有报告 — 修改后保存并重新编辑
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 4.1 | 从 `/report-manage` 编辑已有报告 | `/report-editor?id=xxx` 正常加载 |
| 4.2 | 修改患者姓名,上传视频,自动摘取关键帧 | 所有数据正常显示 |
| 4.3 | 点击「保存草稿」 | 提示保存成功 |
| 4.4 | 离开并重新进入 `/report-editor?id=xxx` | **修改后的所有数据完整恢复** |
### 用例 5边界 — 多次快速路由切换
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 5.1 | 完成用例 2 和 3 的所有操作 | 数据正常 |
| 5.2 | 连续快速切换 /report-editor ↔ /report-manage 3 次以上 | **最终返回时没有任何数据丢失或变空** |
| 5.3 | 检查 localStorage `reportEditorDraft_{username}` | draft 中 `reportData``videos``capturedFrames``content` 均非空 |
### 用例 6回归 — 模板切换后行为正常
| 步骤 | 操作 | 预期结果 |
|------|------|---------|
| 6.1 | 在 `/report-editor` 中上传视频并截取若干帧 | 数据正常 |
| 6.2 | 切换模板 | 编辑器内容重置,基本信息清空(现有预期行为) |
| 6.3 | 离开并返回 | 页面无崩溃,状态与模板切换后一致 |
## 验收标准
- [ ] 用例 1基本信息和编辑器内容在路由切换后 100% 保留;
- [ ] 用例 2视频列表和关键帧在路由切换后 100% 保留;
- [ ] 用例 3拖拽到 placeholder 的图片在路由切换后 100% 保留;
- [ ] 用例 4编辑已有报告保存后重新编辑数据完整无丢失
- [ ] 用例 5多次快速切换路由后数据不丢失、不异常变空
- [ ] 用例 6模板切换行为未被意外改变页面无崩溃。
## 测试方式
手工浏览器验证。
---
**⚠️ 请审核以上测试方案,确认无误后回复「确认」或提出修改意见,我将进入最终执行阶段。**

View File

@@ -1,36 +0,0 @@
# 需求分析 — 2026-04-16-16-51-00
## 需求背景
用户在进入 **report-editor**(图文报告生成)页面时,期望编辑器能够自动加载在 **system-settings** 中配置的"图文报告生成默认模板"。但目前存在以下异常现象:
1. 超级管理员进入 report-editor 时,顶部模板选择器显示 **"无"**,编辑区域(`editor-content-wrapper print-wrapper`)为纯白色空白,没有加载任何模板内容。
2. 在 system-settings 中已正确设置默认模板为 **"腹腔镜胆囊切除术报告"**,但 report-editor 未按预期加载。
3. 用户从 `/dashboard` 等其他页面返回 `/report-editor` 后,若之前未进行过有效编辑,也可能出现白色空白模板的情况。
## 功能目标
修复 report-editor 页面初始化时的模板加载逻辑,确保:
- 新建报告(无 `reportId`)时,优先加载系统设置中的默认模板。
- 只有当用户确实在编辑器中有过有效编辑内容时才从本地草稿draft恢复。
- 模板选择器中的"当前模板(及重置模板):"应正确反映当前加载的模板名称,而不是显示"无"。
- 从其他页面返回或重新进入 report-editor 时,未编辑状态下不应出现空白模板。
## 涉及页面/模块
- `src/pages/ReportEditor.tsx` —— 核心问题所在,草稿恢复与默认模板加载逻辑
- `src/pages/SystemSettings.tsx` —— 默认模板设置页面(验证配置读取逻辑)
- `src/utils/storage.ts` —— localStorage 读写封装(辅助确认)
## 问题根因(预分析)
通过代码审阅,发现以下导致空白的根因:
1. **空字符串草稿被当作有效内容加载**`saveDraftToStorage` 在组件卸载时自动保存草稿。如果用户未在编辑器中输入任何内容,保存的 `content` 会是空字符串 `""`。在初始化 effect 中,判断条件 `typeof draft.content === 'string'` 对空字符串也返回 `true`,导致编辑器被填充为空白 HTML跳过了后续默认模板加载逻辑。
2. **草稿中未记录模板 ID**`loadedTemplateId` 没有被存入 draft。当从 draft 恢复时,即使内容非空,模板选择器也因缺少 `loadedTemplateId` 而显示"无"。
3. **默认模板加载逻辑被空白草稿截断**:由于空草稿提前将 `contentLoadedRef.current` 设为 `true`,真正的默认模板分支 (`settings.defaultTemplate`) 永远不会执行。
## 验收标准
- [ ] 超级管理员进入 report-editor 时,编辑区域正确显示 system-settings 中设置的默认模板内容。
- [ ] 模板选择器显示当前已加载的默认模板名称,而非"无"。
- [ ] 从 dashboard 等其他页面返回 report-editor未编辑情况下不显示空白模板。
- [ ] `npm run lint` 类型检查零错误。
- [ ] `npm run build` 构建通过。

View File

@@ -1,26 +0,0 @@
# 需求分析 — 2026-04-16-17-07-04
## 需求背景
`report-editor` 页面的 **视频分析 - 关键帧摘取** 区域,目前用户需要通过鼠标拖拽的方式将截图插入到报告编辑器的 `image-placeholder` 中。当关键帧数量较多或占位符位置较远时,拖拽操作不够便捷,用户体验有待提升。
## 功能目标
为每个关键帧截图增加一个 **"插入"** 按钮,实现一键自动插入到编辑器中第一个空置的 `image-placeholder` 中。
具体要求:
1. "插入" 按钮放置在每张截图底部的时间格式(如 `00:15`)与 "可拖拽" 提示文字之间。
2. "插入" 按钮的显隐行为与右侧的 "可拖拽" 文字保持一致:仅当鼠标聚焦/悬停到该截图卡片上时才显示(`opacity-0 group-hover:opacity-100`)。
3. 点击 "插入" 后,系统在当前编辑器内容中查找第一个尚未填充图片的 `.image-placeholder` 元素(即不含 `has-image` class 的占位符)。
4. 找到后,将该关键帧的图片数据自动填充到该占位符中(效果等价于拖拽放下)。
5. 填充后同步更新 `contentRef.current` 并保存草稿。
6. 若编辑器中不存在空置的 `image-placeholder`,则给出提示(如 `alert('没有可插入图片的空位')`)。
## 涉及页面/模块
- `src/pages/ReportEditor.tsx` —— 关键帧列表渲染区域 + 新增插入逻辑
## 验收标准
- [ ] 每个关键帧截图底部出现 "插入" 按钮,位置在时间文字和 "可拖拽" 之间。
- [ ] "插入" 按钮只在鼠标悬停到该截图卡片上时显示,移开即隐藏。
- [ ] 点击 "插入" 后,第一个空置的 `image-placeholder` 被自动填充为该帧图片。
- [ ] 若无可插入的空占位符,弹出友好提示。
- [ ] 插入后编辑器内容、草稿状态正确同步。
- [ ] `npm run lint` 零错误,`npm run build` 构建通过。

View File

@@ -1,25 +0,0 @@
# 需求分析 — 2026-04-16-17-15-37
## 需求背景
上一个需求已为关键帧截图添加了 "插入" 按钮,但用户对当前按钮的呈现方式提出了优化诉求:按钮位置在底部文字区域不够醒目,且纯文字链接样式不够直观,蓝色与 "可拖拽" 提示颜色过于接近,辨识度不足。
## 功能目标
调整关键帧截图卡片上 "插入" 按钮的位置和样式:
1. **位置调整**:将 "插入" 按钮从卡片底部的文字区域移到**图片中央**(覆盖在图片之上)。
2. **样式调整**:改为**实体按钮样式**(带背景色、圆角、内边距、阴影),而非纯文字链接。
3. **颜色调整**:不再使用蓝色(`text-accent`),改用与 "可拖拽" 蓝色提示有**明显区分度**的颜色(如绿色系 `emerald-500`hover 时才显示在图片上方。
4. 保留原有行为:
- 鼠标未悬停时按钮不可见。
- 点击后仍按从前到后顺序自动填充第一个空置 `image-placeholder`
- 点击不触发卡片跳转到视频位置。
## 涉及页面/模块
- `src/pages/ReportEditor.tsx` —— 关键帧卡片 JSX 结构调整
## 验收标准
- [ ] "插入" 按钮位于关键帧图片的正中央。
- [ ] 按钮为实体按钮样式(圆角、背景色、阴影)。
- [ ] 按钮颜色不是蓝色,与 "可拖拽" 有明显区分。
- [ ] 鼠标悬停到卡片上时按钮淡入显示,移开时淡出隐藏。
- [ ] 点击按钮后插入逻辑正常工作。
- [ ] `npm run lint` 零错误,`npm run build` 构建通过。

View File

@@ -1,22 +0,0 @@
# 需求分析 — 2026-04-16-17-21-58
## 需求背景
用户对上一个优化后的 "插入" 按钮位置提出了微调诉求:按钮覆盖在图片正中央会遮挡图片内容,不够理想。
## 功能目标
调整关键帧截图卡片上 "插入" 按钮的位置和颜色:
1. **位置调整**:将 "插入" 按钮从图片正中央移回卡片底部,位于**时间格式(如 `00:15`**和 **"可拖拽"** 提示文字之间。
2. **不遮盖图片**:按钮不再以 absolute 覆盖层形式存在于图片之上。
3. **颜色调整**:按钮恢复为与 "可拖拽" 一致的蓝色(`bg-accent` / `text-white`)。
4. **保留实体按钮样式**:保持 `px-3 py-1.5 rounded-full shadow-md` 的实体胶囊按钮外观,不再使用纯文字链接。
## 涉及页面/模块
- `src/pages/ReportEditor.tsx` —— 关键帧卡片 JSX 结构调整
## 验收标准
- [ ] "插入" 按钮位于关键帧卡片底部,时间文字与 "可拖拽" 之间。
- [ ] 按钮不覆盖在图片上,不遮挡图片内容。
- [ ] 按钮为蓝色实体胶囊按钮,与 "可拖拽" 蓝色一致。
- [ ] 悬停时按钮和 "可拖拽" 同时显示,移开时同时隐藏。
- [ ] 点击按钮后插入逻辑正常工作,不触发视频跳转。
- [ ] `npm run lint` 零错误,`npm run build` 构建通过。

View File

@@ -1,36 +0,0 @@
# 需求分析 — 2026-04-16-18-51-06
## 原始需求摘要
`/report-editor` 页面中进行操作后,离开该页面(例如进入 `/report-manage`),再返回 `/report-editor` 时,**视频分析相关数据丢失**,具体表现为:
1. 自动关键帧摘取的图片消失;
2. 自动/手动拖拽到报告 `image-placeholder` 上的视频截图消失;
3. 手动截取的视频截图消失。
报告的基本信息(患者姓名、住院号等)保存正常。
## 需求拆解
### 功能点
- 修复路由切换后 `capturedFrames`(关键帧/截图)数据丢失的问题;
- 修复路由切换后 `videos`(已上传视频列表)数据丢失的问题;
- 确保 `stateRef.current` 与 React state 在数据恢复后保持同步;
- 确保组件卸载时保存的 draft 包含完整的视频分析数据。
### 非功能点
- 保持现有 localStorage 存储机制不变;
- 最小化代码改动,避免引入新的状态管理库;
- 不破坏现有报告保存/打印/模板切换等功能。
## 影响范围预估
| 模块 | 影响程度 | 说明 |
|------|---------|------|
| `src/pages/ReportEditor.tsx` | 高 | 初始化逻辑、`useLayoutEffect` 安全网、`stateRef` 同步 |
| `src/utils/storage.ts` | 无 | 不涉及修改 |
| 其他页面 | 低 | 仅受 `/report-editor` 数据恢复正确性影响 |
## 待确认问题
无。问题现象明确,根因已定位。

View File

@@ -1,34 +0,0 @@
# 需求分析 — 2026-04-16-19-06-18
## 原始需求摘要
### 问题 1路由切换后所有内容丢失
`/report-editor` 页面操作后,切换到 `/report-manage` 等其他页面,再返回 `/report-editor` 时,**所有内容全部变空**
- 报告编辑器内容丢失;
- 基本信息(患者姓名、住院号等)丢失;
- 视频分析中的自动关键帧、手动截图、拖拽到 `image-placeholder` 的图片全部丢失。
### 问题 2自动帧插入的 UI 刷新时序异常
开启「自动帧插入」后,系统应在**每摘取到一张特定关键帧后、经过设定的延迟时间,立即将该帧插入到报告中**。但目前的实际表现是:用户需要等待所有关键帧全部摘取完成后,右侧关键帧列表和报告中的插入图片才会**一次性批量蹦出**,无法看到逐帧实时更新的效果。
## 需求拆解
### 功能点
- **问题 1**:修复路由切换后报告内容、基本信息、视频分析数据全部丢失的根因;
- **问题 2**:修复 `autoCaptureFrames` 中 React 状态更新被批处理导致的 UI 延迟刷新问题,使关键帧在摘取过程中实时可见、并按延迟配置逐帧插入 placeholder。
### 非功能点
- 保持现有 localStorage 草稿机制不变;
- 最小化改动范围,不引入新的状态管理库;
- 不破坏模板切换、保存草稿、打印等现有功能。
## 影响范围预估
| 模块 | 影响程度 | 说明 |
|------|---------|------|
| `src/pages/ReportEditor.tsx` | 高 | 初始化数据恢复逻辑 + autoCaptureFrames 的渲染同步 |
| 其他页面 | 低 | 仅涉及 `/report-editor` 的数据持久化和恢复 |
## 待确认问题
无。问题现象明确,根因已定位。

View File

@@ -1,30 +0,0 @@
# 需求分析 — 2026-04-16-19-18-14
## 原始需求摘要
用户要求**重新部署当前应用**,使最新的代码修改(已修复的路由切换数据丢失、自动帧插入实时刷新等问题)在生产环境中生效。
## 需求拆解
### 功能点
- 基于最新代码重新构建 Docker 镜像;
- 停止并移除旧运行的容器;
- 启动新容器并暴露服务;
- 验证部署后的应用可正常访问。
### 非功能点
- 尽量缩短服务中断时间;
- 保留旧镜像以便快速回滚(可选);
- 部署失败时需要有明确的错误信息。
## 影响范围预估
| 模块 | 影响程度 | 说明 |
|------|---------|------|
| Docker 容器 `medical-report-app` | 高 | 需要重建并重启 |
| 本地端口 `8080` | 中 | 重新绑定到新容器 |
| 源代码 / Git 仓库 | 无 | 仅读取最新代码进行构建 |
## 待确认问题
无。部署流程明确。

View File

@@ -1,34 +0,0 @@
# 需求分析 — 2026-04-16-19-28-04
## 原始需求摘要
`/report-editor` 页面进行操作(填写基本信息、上传视频、自动/手动截取关键帧、拖拽图片到 placeholder离开该页面进入 `/report-manage` 等其他界面后,再次返回 `/report-editor` 时,**所有内容全部丢失**
- 报告编辑器中的文本和图片丢失;
- 基本信息(患者姓名、住院号等)丢失;
- 视频分析面板中的视频列表和关键帧截图全部丢失。
此前两次修复尝试未能解决该问题。
## 需求拆解
### 功能点
- 彻底修复路由切换后报告内容、基本信息、视频分析数据全部丢失的问题;
- 确保自动保存机制(草稿保存)在任何情况下都不会用空值覆盖已有的有效 draft
- 确保组件卸载时保存的 draft 100% 反映用户最新的操作状态。
### 非功能点
- 最小化对现有 UI 和交互逻辑的侵入;
- 保持现有 localStorage 存储机制不变;
- 同时兼顾 React 18 `StrictMode` 在开发/预览环境下的双重挂载行为。
## 影响范围预估
| 模块 | 影响程度 | 说明 |
|------|---------|------|
| `src/pages/ReportEditor.tsx` | 高 | 自动保存逻辑 `saveDraftToStorage` 和自动保存 effect 需要重构 |
| `useLayoutEffect` 安全网 | 中 | 需要添加依赖数组,避免重复执行 |
| 其他组件 | 无 | 不涉及修改 |
## 待确认问题
无。根因已定位,修复方案明确。