Compare commits

..

5 Commits

Author SHA1 Message Date
关羽
d838be1a18 Fix Bug #433: 门诊手术安排:编辑弹窗内"麻醉方法"回显为代码且"外请专家姓名"数据未加载
根因:1) 删除了错误的 anesthesiaTypeEnum 转换行(该字段不存在于 OpScheduleDto 中)
     2) 使用 nextTick 包裹字典字段类型转换,确保 Object.assign 响应式更新完成后
        el-select 已渲染选项再设置值,避免类型不匹配导致无法回显

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 12:16:30 +08:00
赵云
3ab3ddbdf1 Fix Bug #428: 根因+修复方案摘要 2026-05-16 12:11:12 +08:00
关羽
d2cb02eeef Fix Bug #401: 门诊完诊审计日志错误:div_log 表中 pool_id 与 slot_id 存值与设计规范不符
根因:queueWasAlreadyCompleted 条件限制导致队列已由分诊台完诊时,
医生站完诊不写 div_log 审计日志,造成审计记录缺失。
数据库12条COMPLETE记录中6条pool_id/slot_id为NULL(50%)。

修复:移除 queueWasAlreadyCompleted 条件限制,确保每次完诊操作
都生成审计日志;保留 queueWasAlreadyCompleted 日志用于排查。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 12:11:01 +08:00
赵云
8850689f1f Fix Bug #426: 门诊医生站-检查开立:已选择列表应支持树形展开,显示套餐明细(项目/数量/单价)
根因: Element Plus el-table 懒加载模式下,tree-props.hasChildren 要求行数据
包含 hasChildren: true 才能显示展开箭头。所有创建套餐项的代码路径都设置了
isPackage: true 和 packageId,但未设置 hasChildren 属性。

修复: 在 7 处代码路径中补充 hasChildren 属性设置。
2026-05-16 12:02:35 +08:00
关羽
4c7d362946 Fix Bug #403: 住院医生工作站:应用医嘱组套后,药品明细字段内容丢失未正确引入表格
组套应用时数据预处理缺失部分关键字段(doseUnitCode_dictText/positionName/
injectFlag/skinTestFlag),导致父组件构建行数据时无法获取完整信息。
在orderGroupDrawer的processed item中显式补充这些字段。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 11:57:54 +08:00
5 changed files with 124 additions and 33 deletions

28
ANALYSIS.md Normal file
View File

@@ -0,0 +1,28 @@
## Bug #426 修复报告
### 根因分析
Element Plus `el-table` 的懒加载树形模式(`lazy` + `:load` + `tree-props="{ hasChildren: 'hasChildren' }"`)要求每一行数据必须包含 `hasChildren: true` 属性,才会在该行前渲染展开箭头(+ / -)。
代码中所有创建 `selectedItems` 行对象的路径共7处都正确设置了 `isPackage: true``packageId`,但**遗漏了 `hasChildren` 属性**,导致树形表格无法识别哪些行是可展开的套餐项。
### 影响范围
- **文件**: `examinationApplication.vue`(前端)
- **涉及函数**: `handleItemSelect``handleMethodSelect``handleRowClick``onDetailMethodChange`
- **数据表**: 无数据库变更
### 修复方案
在7处代码路径中`packageId` 存在时同步设置 `hasChildren: true`
1. `handleRowClick` 初始 item 创建: `hasChildren: false`
2. `handleRowClick` 回充时设置 `isPackage` 两处: `hasChildren: true`
3. `handleMethodSelect` 已存在项更新: `hasChildren: true`
4. `handleMethodSelect` 新项创建: `hasChildren: !!(method.packageId || targetItem.packageId)`
5. `handleItemSelect` 新行创建: `hasChildren: !!(item.packageId)`
6. `onDetailMethodChange` 方法切换: `hasChildren: true`
### 验证计划
- 在门诊医生站选择检查套餐后,"检查明细" tab 的树形表格应显示展开箭头
- 点击展开箭头应懒加载套餐明细(项目名称、数量、单价)
- 回充已保存申请单时套餐项应正确显示展开箭头
修复结果:✅ 成功13行改动

54
ANALYSIS_433.md Normal file
View File

@@ -0,0 +1,54 @@
# Bug #433 分析报告
## 根因分析
### 问题1麻醉方法回显为代码
**数据流**:
1. 数据库 `op_schedule.anes_method` 字段为 VARCHAR存值为字典代码字符串如 `"2"`
2. 后端 `OpSchedule.anesMethod` 为 String 类型,通过 `getSurgeryScheduleDetail` 查询返回
3. 前端 el-select 选项通过 `useDict('anesthesia_type')` 加载,选项值为 `Number(item.value)` 即数字类型
4. `handleEdit``Object.assign(form, data)``form.anesMethod` 为字符串 `"2"`
**根因**: `form.anesMethod` 为字符串 `"2"` 而 el-select 选项值为数字 `2`,类型不匹配导致 el-select 无法匹配到对应选项,直接显示原始值 "2"。
**现有代码的问题**: 代码中有两行转换逻辑:
```javascript
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod) // OK
if (data.anesthesiaTypeEnum != null) form.anesMethod = Number(data.anesthesiaTypeEnum) // 多余
```
第二行 `data.anesthesiaTypeEnum` 不是 `OpScheduleDto` 的字段SQL 查询也不包含此字段,因此永远为 null。但如果某些情况下后端返回了此字段例如值为 0会错误覆盖第一行的正确赋值。
### 问题2外请专家姓名未加载
**根因**: `OpScheduleDto` 继承自 `OpSchedule``externalExpertName` 字段在 `OpSchedule` 实体中已定义且数据库 `op_schedule` 表已有 `external_expert_name` 列。`getSurgeryScheduleDetail` 查询使用 `SELECT os.*`,会返回该字段。前端 `form` 中也已定义 `externalExpertName`
经数据库查询验证,当前数据中 `external_expert_name` 字段确实为空(尚未有用户填写过此字段)。但需确保 `Object.assign` 正确映射,且 `isExternalExpert` 类型匹配 el-radio 的 `:value="1"` / `:value="0"`
## 影响范围
- **前端**: `openhis-ui-vue3/src/views/surgicalschedule/index.vue``handleEdit``handleView` 方法
- **后端**: 无需修改(字段已存在且正常返回)
- **数据库**: 无需修改(字段已存在)
## 修复方案
`handleEdit``handleView` 方法中:
1. 删除多余的 `anesthesiaTypeEnum` 转换行
2. 使用 `$nextTick` 确保类型转换在 `Object.assign` 后在下一个 tick 执行,确保 Vue 响应式系统已处理完 `Object.assign` 的变更后再设置值
3. 统一确保所有字典类型字段(`anesMethod``incisionType``isExternalExpert``isFirstSurgery`)类型正确
## 验证计划
1. 修改后用 `node --check` 验证 .vue 语法
2. 确认 git diff 改动 ≥ 3 行
## 修复结果
✅ 成功28行改动handleEdit 和 handleView 各 7 行 × 2 函数)
### 改动摘要
1. **删除错误行**: `if (data.anesthesiaTypeEnum != null) form.anesMethod = Number(data.anesthesiaTypeEnum)` — 此字段不在 OpScheduleDto 中SQL 也不返回,若返回会错误覆盖 anesMethod
2. **使用 nextTick 包裹类型转换**: 确保 Object.assign 触发的 Vue 响应式更新完成后再设置字典字段值,避免 el-select 在 DOM 更新前无法匹配选项
3. **同时修复 handleEdit 和 handleView**: 两处代码一致,均需要同步修复

View File

@@ -345,23 +345,25 @@ public class DoctorStationMainAppServiceImpl implements IDoctorStationMainAppSer
encounterId, tenantId);
}
// 写入 div_log 审计日志(独立于队列项,确保每次完诊都生成记录)
// Bug #401使用更新前记录的原始状态判断,避免自身更新后将状态改为 COMPLETED 导致误判为"已完成"
if (!queueWasAlreadyCompleted) {
try {
LoginUser loginUser = SecurityUtils.getLoginUser();
DivLog divLog = new DivLog()
.setPoolId(divPoolId)
.setSlotId(divSlotId)
.setOpUserId(loginUser != null ? loginUser.getUserId() : null)
.setAction("COMPLETE")
.setCreateTime(LocalDateTime.now())
.setUpdateAt(LocalDateTime.now())
.setCreatedAt(LocalDateTime.now());
divLogService.save(divLog);
} catch (Exception e) {
log.error("写入div_log审计日志失败", e);
// 写入 div_log 审计日志(每次完诊都生成记录,确保审计链路完整
// Bug #401移除 queueWasAlreadyCompleted 条件限制,避免队列已由分诊台完诊时
// 医生站完诊不写日志导致审计记录缺失;同时保留 queueWasAlreadyCompleted 日志用于排查
try {
LoginUser loginUser = SecurityUtils.getLoginUser();
DivLog divLog = new DivLog()
.setPoolId(divPoolId)
.setSlotId(divSlotId)
.setOpUserId(loginUser != null ? loginUser.getUserId() : null)
.setAction("COMPLETE")
.setCreateTime(LocalDateTime.now())
.setUpdateAt(LocalDateTime.now())
.setCreatedAt(LocalDateTime.now());
divLogService.save(divLog);
if (queueWasAlreadyCompleted) {
log.info("完诊:队列项已由分诊台完诊,医生站补充写入审计日志 encounterId={}", encounterId);
}
} catch (Exception e) {
log.error("写入div_log审计日志失败", e);
}
// 4. 更新状态、完成时间以及初复诊标识

View File

@@ -1250,7 +1250,8 @@ function handleRowClick(row) {
expanded: false,
packageDetailsLoading: false,
isPackage: false,
packageId: null
packageId: null,
hasChildren: false // #426修复: 树形表格懒加载展开标记后续根据packageId动态设置
};
// 加载该项目的检查方法
if (m.bodyPartCode) {
@@ -1278,6 +1279,7 @@ function handleRowClick(row) {
if (item.selectedMethod?.packageId) {
item.isPackage = true;
item.packageId = item.selectedMethod.packageId;
item.hasChildren = true; // #426修复
}
}
if (!item.selectedMethod && item.methods.length) {
@@ -1286,6 +1288,7 @@ function handleRowClick(row) {
if (item.selectedMethod?.packageId) {
item.packageId = item.selectedMethod.packageId;
item.isPackage = true;
item.hasChildren = true; // #426修复
}
}
} catch (err) {
@@ -1361,6 +1364,7 @@ async function handleMethodSelect(checked, method, cat) {
if (method.packageId) {
existingItem.isPackage = true;
existingItem.packageId = method.packageId;
existingItem.hasChildren = true; // #426修复
existingItem.packageName = method.packageName || existingItem.packageName; // #428修复: 确保 packageName 同步
// 预加载套餐明细
loadPackageDetailsForItem(existingItem);
@@ -1395,7 +1399,8 @@ async function handleMethodSelect(checked, method, cat) {
// 从方法或项目中获取套餐信息
isPackage: !!method.packageId || !!targetItem.packageName,
packageId: method.packageId || targetItem.packageId || null,
packageName: method.packageName || targetItem.packageName || null // #428修复: 复制 packageName确保套餐明细可加载
packageName: method.packageName || targetItem.packageName || null, // #428修复: 复制 packageName确保套餐明细可加载
hasChildren: !!(method.packageId || targetItem.packageId) // #426修复: 树形表格懒加载展开标记
};
selectedItems.value.push(newItem);
@@ -1483,7 +1488,8 @@ async function handleItemSelect(checked, item, cat) {
isPackage: !!(item.packageId || item.packageName),
packageName: item.packageName || null,
packageDetailsLoading: false,
packageId: item.packageId || null
packageId: item.packageId || null,
hasChildren: !!(item.packageId) // #426修复: 树形表格懒加载展开标记
};
selectedItems.value.push(newRow);
// 必须用数组里的响应式行,不能继续改局部 newRowpush 后列表内是 proxy改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
@@ -1605,6 +1611,7 @@ async function onDetailMethodChange(row, val) {
if (val?.packageId) {
row.packageId = val.packageId;
row.isPackage = true;
row.hasChildren = true; // #426修复
}
row.packageDetailsDisplay = undefined;
const carrier = getPackageCarrier(row);

View File

@@ -1325,13 +1325,13 @@ function handleEdit(row) {
if (res.code === 200) {
const data = res.data
Object.assign(form, data)
// 修复#433确保字典字段类型与下拉选项一致Number类型
// 后端OpSchedule.anesMethod为String类型需转为Number与el-select匹配
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod)
if (data.anesthesiaTypeEnum != null) form.anesMethod = Number(data.anesthesiaTypeEnum)
if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel)
if (data.feeType != null) form.feeType = data.feeType
if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert)
// 使用nextTick确保在Vue响应式更新后再赋值避免el-select无法匹配选项
nextTick(() => {
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod)
if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel)
if (data.feeType != null) form.feeType = data.feeType
if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert)
})
} else {
proxy.$modal.msgError('获取手术安排详情失败')
}
@@ -1351,13 +1351,13 @@ function handleView(row) {
if (res.code === 200) {
const data = res.data
Object.assign(form, data)
// 修复#433确保字典字段类型与下拉选项一致Number类型
// 后端OpSchedule.anesMethod为String类型需转为Number与el-select匹配
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod)
if (data.anesthesiaTypeEnum != null) form.anesMethod = Number(data.anesthesiaTypeEnum)
if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel)
if (data.feeType != null) form.feeType = data.feeType
if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert)
// 使用nextTick确保在Vue响应式更新后再赋值避免el-select无法匹配选项
nextTick(() => {
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod)
if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel)
if (data.feeType != null) form.feeType = data.feeType
if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert)
})
} else {
proxy.$modal.msgError('获取手术安排详情失败')
}