# 实现方案 — 模板字段唯一性、删除交互与报告批量导出(2026-04-17-10-21-18) ## 一、修改文件清单 1. `src/pages/TemplateManage.tsx` — 字段唯一性校验 + 删除按钮 + 键盘删除增强 2. `src/utils/defaultContent.ts` — `smartField()` 增加删除按钮 HTML 3. `src/index.css` — 智能字段删除按钮样式 4. `src/pages/ReportManage.tsx` — 复选框、批量操作栏、导出功能 ## 二、详细改动 ### 2.1 `src/pages/TemplateManage.tsx` #### A. `insertSmartField` 增加唯一性校验 ```ts const insertSmartField = (field: FormField) => { editorRef.current?.focus(); if (editorRef.current?.querySelector(`[data-bind="${field.key}"]`)) { alert(`字段 "${field.label}" 已存在,请勿重复插入。`); return; } // ... 原有 insertHTML 逻辑,同时给 wrapper 增加 delete-btn }; ``` #### B. 给 `smart-field-wrapper` 增加删除按钮 插入的 HTML 改为: ```html × ``` #### C. 点击删除按钮事件(复用或扩展已有的 `handleEditorClick` capture 事件) 在已有的 `handleEditorClick` 中,除了处理 `.image-placeholder`,再增加对 `.delete-btn` 在 `.smart-field-wrapper` 内的判断: ```ts const wrapper = targetEl.closest('.smart-field-wrapper') as HTMLElement | null; if (targetEl.closest('.delete-btn') && wrapper) { e.preventDefault(); e.stopPropagation(); wrapper.remove(); // 同步保存模板内容到 localStorage return; } ``` #### D. 增强 `keydown` 删除逻辑 当前逻辑只处理 "光标在文本节点内且 offset 为 0/末尾" 的情况。需要额外处理: - 光标直接在字段后面(`range.startContainer` 是 `
`,`range.startOffset` 指向字段节点位置)时按 Backspace。
- 光标直接在字段前面时按 Delete。
- 选区 collapsed 且紧邻字段的各种边界情况。
实现策略:在 `keydown` 中统一写一个 `findAdjacentSmartField(range, direction)` 辅助函数,先尝试从文本节点 sibling 找,若找不到则从父级块节点的 childNodes 中按 offset 找。
### 2.2 `src/utils/defaultContent.ts`
修改 `smartField(key)` 辅助函数,使其输出包含 `×` 的单行 HTML。
### 2.3 `src/index.css`
新增 `.smart-field-wrapper .delete-btn` 的样式:
```css
.smart-field-wrapper .delete-btn {
display: inline-flex;
align-items: center;
justify-content: center;
width: 14px;
height: 14px;
margin-right: 2px;
background: #ef4444;
color: white;
border-radius: 50%;
font-size: 10px;
line-height: 1;
cursor: pointer;
user-select: none;
}
.smart-field-wrapper .delete-btn:hover {
background: #dc2626;
}
@media print {
.smart-field-wrapper .delete-btn {
display: none !important;
}
}
```
### 2.4 `src/pages/ReportManage.tsx`
#### A. 状态扩展
```ts
const [selectedIds, setSelectedIds] = useState` 增加 Checkbox,状态为 `selectedIds.length === filteredReports.length && filteredReports.length > 0`。
- 每行 ` ` 最左侧增加 Checkbox,checked 状态为 `selectedIds.includes(report.id)`,onChange 时切换选中状态。
- 当 `selectedIds.length > 0` 时,在搜索栏下方显示批量操作栏(flex row),包含:
- `已选择 N 项` 文本
- **批量删除** 按钮(红色)
- **批量导出 PDF** 按钮
- **批量导出 JSON** 按钮
- **取消选择** 按钮
#### C. 单报告导出
- **PDF**:调用 `printDocument(report.content)`。
- **JSON**:构建对象:
```ts
const exportData = {
meta: { id: report.id, title: report.title, createdAt: report.createdAt, updatedAt: report.updatedAt, author: report.author, authorName: report.authorName, status: report.status },
fields: { /* 所有 DEFAULT_FORM_FIELDS 的 key 对应的值 */ }
};
```
通过 `Blob` + `URL.createObjectURL` + `` 触发下载。
#### D. 批量导出
- **批量 PDF**:`const mergedHTML = selectedReports.map(r => r.content).join(''); printDocument(mergedHTML);`
- **批量 JSON**:`const exportData = selectedReports.map(r => ({ meta: ..., fields: ... }));` 同样通过 Blob 下载为 `reports_export_时间戳.json`。
#### E. 批量删除
```ts
const handleBulkDelete = () => {
if (!window.confirm(`确定要删除选中的 ${selectedIds.length} 份报告吗?`)) return;
const updated = reports.filter(r => !selectedIds.includes(r.id));
setReports(updated);
storage.set('reports', updated);
setSelectedIds([]);
};
```
## 三、风险与回滚
- **风险**:修改 `defaultContent.ts` 中的 `smartField` HTML 结构后,旧模板中已存在的 `smart-field-wrapper` 没有删除按钮。这属于正常行为(旧数据不 retroactive),新建报告时的新模板会带删除按钮。
- **风险**:`keydown` 删除逻辑改动较大,可能在不同浏览器下对边界 selection 的解析有差异,需要手工测试。
- **回滚**:如出现问题,可回退 `TemplateManage.tsx`、`defaultContent.ts`、`index.css`、`ReportManage.tsx` 的修改。