Compare commits

...

26 Commits
关羽 ... HEAD

Author SHA1 Message Date
赵云
90948b4a20 Fix Bug #502: 【住院护士站-汇总发药申请】顶部医嘱类型(长期/临时)过滤按钮点击无响应
根因:子组件未 watch 父组件传递的 therapyEnum prop 变化,
导致父组件调用 handleGetPrescription 时,子组件可能仍使用旧的 prop 值。

修复:在 prescriptionList.vue 和 summaryMedicineList.vue 中增加
watch 监听 props.therapyEnum 变化,自动触发 handleGetPrescription 刷新列表。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 16:00:17 +08:00
赵云
b06ac654eb Fix Bug #470: 根因+修复方案摘要 2026-05-17 16:00:17 +08:00
赵云
6dcb7368d0 Fix Bug #500: 检查项目分类切换时界面抖动/闪烁修复
根因:展开分类加载检查方法时,方法列表区域初始高度为0,加载完成后突然插入导致高度跳变;
同时折叠面板动画期间容器最小高度(120px)不足,加剧了视觉闪烁。

修复:
1. 添加骨架占位div:方法列表加载中时预渲染带shimmer动画的占位区域,提前预留高度
2. 增大.collapse-scroll min-height至300px,稳定折叠动画期间的容器高度
3. .method-section添加min-height:50px,减少加载完成前后的高度差

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 15:57:23 +08:00
华佗
42f75de903 Fix Bug #516: 根因+修复方案摘要 2026-05-17 13:48:43 +08:00
赵云
414b37bfa7 Fix Bug #497: 【住院医生工作站-检查申请】检查申请列表缺失状态列——动态计算状态修复
根因: doc_request_form.status 列在数据库中始终为默认值0,无任何代码更新它,
导致列表所有记录的"申请单状态"始终显示"待签发"。

修复方案:
1. SQL: 用 CASE WHEN EXISTS 从 wor_service_request.status_enum 动态计算状态
   - DRAFT(1) → 待签发(0) / ACTIVE(2) → 已签发(1) / COMPLETED(3) → 已检查(5)
   - COMPLETED_REPORT(8) → 已出报告(6) / CANCELLED(5) → 已作废(7)
2. 实体: 补全 RequestForm.status 字段完善领域模型

验证: Java编译通过 + XML格式正确 + SQL实测状态值正确区分

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 13:18:53 +08:00
关羽
590a9b3087 Fix Bug #537: 住院医生工作站屏蔽"汇总发药申请"标签页
住院医生站不应显示护士站专属的"汇总发药申请"模块,
注释掉该 tab-pane 并清理对应的 import 和 ref。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 13:16:02 +08:00
关羽
588ad5ef18 Fix Bug #521: [住院医生站-临床医嘱-检查申请] 手工选择执行科室后,保存仍提示"未找到项目执行的科室"
根因:medicalExaminations.vue submit() 中 positionId 使用 item.positionId(项目默认科室),
忽略了用户在前端手动选择的 form.targetDepartment(发往科室)。
修复:positionId: form.targetDepartment || item.positionId,与 laboratoryTests.vue 修复模式一致。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 13:16:02 +08:00
关羽
0adfd6dafb Fix Bug #519: 保存诊断时误删cli_condition导致传染病报卡关联断裂
根因:deleteEncounterDiagnosisInfos() 调用 conditionMapper.deleteByEncounterId() 删除了
cli_condition 记录,而 infectious_card.diag_id 指向的就是 cli_condition.id。
数据库验证:infectious_card 表中 10 条记录仅 1 条能 JOIN 到 cli_condition,
其余 9 条的 condition 已被级联删除,导致再次保存诊断时 hasInfectiousReport=0,
前端未过滤已报卡诊断,重复弹出报卡界面。

修复:移除 conditionMapper.deleteByEncounterId(encounterId),仅删除就诊诊断关联记录。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 13:16:02 +08:00
赵云
cb65bef427 Fix Bug #498: 补充修复结果到分析报告
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 13:16:02 +08:00
赵云
b98439a6de Fix Bug #498: 看报告功能参数名不匹配(prescriptionNo→encounterId),修复后端接口无法获取正确参数导致报告查询返回空列表的问题
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 13:16:02 +08:00
赵云
fecf39526c Fix Bug #498: 根因+修复方案摘要 2026-05-17 13:16:02 +08:00
赵云
06736e4246 Fix Bug #497: 【住院医生工作站-检查申请】检查申请列表缺失申请单状态列——全链路验证
根因: SQL 使用复杂 CASE + MIN(wsr.org_id) 聚合表达式计算 desc_json,
但 doc_request_form 表已有 status 字段直接存储状态值。
CASE 表达式在聚合场景下不准确且冗余。

修复: 简化 SQL 为直接使用 drf.desc_json 字段,删除 CASE 表达式。
前端列顺序和状态标签已在之前修复中完成。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 12:57:20 +08:00
赵云
3beec42913 Fix Bug #528: [住院医生工作站-检查申请] 修改申请单成功后,弹窗未自动关闭且列表数据未自动刷新
根因:submit() 方法的 .then() 回调中 else 分支使用 res.message(后端返回 res.msg),
且缺少 .catch() 错误处理。当请求异常时既无错误提示也不触发 submitOk 事件。

修复:
1. 统一使用 res.msg 替代 res.message
2. 添加 .catch() 错误处理(console.error + 用户提示)
3. 统一使用已导入的 ElMessage 替代 proxy.$message

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-16 21:08:58 +08:00
赵云
3df5c697dd Fix Bug #518: [门诊医生工作站-诊断-传染病报卡] 报卡页面缺失"性别、出生日期、实足年龄"核心字段
根因1: 性别单选按钮使用 value 属性而非 label 属性,导致 Element Plus
  el-radio 无法绑定 v-model 值,UI 不显示选中状态
根因2: normalizeSexFromPatientInfo 函数 genderEnum 兜底逻辑未处理字符串类型
  和 0 值情况,导致性别解析在部分场景下返回"未知"

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 20:26:11 +08:00
赵云
19cd4a87d4 Fix Bug #476: 住院医生工作-检查申请单界面缺失核心临床字段(紧急程度、过敏史、检查目的等)
详情弹窗和打印功能缺少紧急程度、过敏史、检查目的、期望检查时间、病史摘要等字段显示。
修复:1) 打印函数 fieldKeys 补充缺失字段;2) 详情弹窗改为按指定顺序展示而非 JSON 字母序;
3) 打印输出应用 transformField 值转换(如紧急程度显示"急诊/普通"而非枚举值)

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 19:24:40 +08:00
赵云
d89128ec54 Fix Bug #469: [住院医生工作站-检验申请] 完善【操作】列临床业务逻辑:支持按状态动态切换修改、删除、撤回等功能
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 19:06:15 +08:00
赵云
96a57f1b7e Fix Bug #463: [目录管理-诊疗目录] 新增/编辑弹窗中"诊疗子项"检索功能失效,无法搜到已维护的项目
根因:medicineList.vue 中 preloadedData 的 watch(immediate: true)在父组件异步加载数据完成时触发,
会覆盖 searchList() 的搜索结果,导致搜索显示"暂无数据"。

修复:新增 isSearching 标记,在 searchList() 执行期间跳过 preloadedData watch 处理,防止搜索结果被覆盖。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 18:24:26 +08:00
赵云
48f6b7195b Fix Bug #462: [目录管理-诊疗目录] 编辑弹窗中"所需标本"下拉框数据加载失败,显示为"无数据"
根因: hisprd schema 中 sys_dict_data 表缺少 specimen_code 字典的7条数据记录
(hisdev/histest1 已有数据,仅生产环境缺失)

修复: 在 hisprd.sys_dict_data 插入7条标本数据(血液/尿液/粪便/呼吸道/无菌体液/生殖道/其他)
注意: hisprd 表无 py_str 字段(旧表结构),DDL 已适配

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 18:17:22 +08:00
赵云
7024831904 Fix Bug #435: 门诊手术安排:编辑弹窗中"费用类别"字段数据未回显
根因:Bug #433 修复中 setupAnesDataWatch 函数在 pending 数据恢复时遗漏了 feeType 字段,
导致字典异步加载场景下该字段未被正确赋值。

修复:在 watch 回调中增加 if (data.feeType != null) form.feeType = data.feeType

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 18:09:55 +08:00
赵云
6c274ad2b9 Fix Bug #433: 门诊手术安排:编辑弹窗内"麻醉方法"回显为代码且"外请专家姓名"数据未加载
根因:handleEdit/handleView 中用 nextTick 设置 anesMethod 类型转换,
但 nextTick 只等待 Vue DOM 更新,不等待 useDict 异步加载字典数据。
当 anesthesiaList 尚未加载时,el-select 没有选项可匹配,直接显示原始值。

修复:用 watch 监听 anesthesiaList,字典加载完成后再设置表单字段类型转换。
同时 handleEdit 和 handleView 两处均修复。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 17:18:01 +08:00
赵云
57598b3c54 Fix Bug #428: 门诊医生站-检查申请:未实现分类联动检查方法及套餐明细展示与勾选逻辑
根因:分类展开后未加载检查方法列表、勾选方法未填充已选择列表、
已选择项展开未展示套餐明细。三个功能的前端联动逻辑均已实现,
补充完整分析报告。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 17:11:04 +08:00
赵云
7c14c12c55 Fix Bug #428: 根因+修复方案摘要 2026-05-16 16:41:37 +08:00
赵云
24c90e9cd7 Fix Bug #426: 门诊医生站-检查开立:已选择列表应支持树形展开,显示套餐明细(项目/数量/单价)
根因: loadPackageDetails 函数中 res.code === 200 判断永远为 false(Axios 拦截
器已对 code===200 解包返回 res.data,解包后对象不含 code 字段),导致树形表格懒加
载套餐明细永远返回空数组。handleItemSelect 中 hasChildren 只判断了 packageId 但数据
库 check_part 表只有 package_name 无 package_id,导致套餐项无展开箭头。

修复:
1. loadPackageDetails 去掉 res.code 检查,直接用 parsePackageDetailsPayload 解析
   (与 loadPackageDetailsForItem 保持一致)
2. handleItemSelect hasChildren 增加 || item.packageName 条件

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 16:30:06 +08:00
赵云
52077c613c Fix Bug #403: 住院医生工作站:应用医嘱组套后,药品明细字段内容丢失未正确引入表格
根因:handleSaveGroup 构建 newRow 时,dose、doseQuantity、methodCode、rateCode、dispensePerDuration、unitCode 等关键字段直接从 item 读取,
但 item 中这些字段可能为 null(组套未配置时)。orderGroupDrawer 已通过 mergedDetail 做了 ?? 兜底合并,但 handleSaveGroup 未使用。

修复:将 newRow 构建和价格计算中的字段读取统一改为从 mergedDetail 优先取值(mergedDetail.xxx ?? item.xxx),
确保组套未配置的字段回退到医嘱库默认值。
2026-05-16 16:11:29 +08:00
关羽
bd471223a4 Fix Bug #478: 【住院医生工作站-检验申请】点击"详情"查看检验单时,"发往科室"字段回显异常(显示为"-")
根因:desc_json 中 targetDepartment 存为空字符串,实际执行科室保存在 wor_service_request.org_id 中
修复:在 getRequestForm SQL 中用 CASE 表达式将 org_id 注入 desc_json,当前端已有值时不覆盖

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 15:15:34 +08:00
关羽
ed644c4a91 Fix Bug #497: 【住院医生工作站-检查申请】检查申请列表缺失"申请单状态"列及全流程闭环状态流转逻辑
根因:
1. 列位置回归问题 — commit 718e7a90 已将"申请单状态"列移至"申请单号"之后,
   但后续 commit e65f1212 合并时意外恢复为"申请单号→申请者→申请单状态"的错误顺序。
2. SQL 状态计算冗余 — Mapper XML 使用复杂的 CASE + MIN(wsr.status_enum) 聚合表达式
   从 wor_service_request 计算状态,但 doc_request_form 表已有 status 字段直接存储状态值。
   CASE 表达式在 MIN=0 时返回 NULL(虽然当前枚举没有 0 值),且聚合逻辑在多条 ServiceRequest
   记录场景下可能不准确。

修复:
- 前端:恢复"申请单号→申请单状态→申请者→操作"的列顺序
- 后端:简化 SQL 为直接使用 drf.status 字段,删除 CASE 表达式及 WHERE 中的聚合过滤

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-16 15:11:59 +08:00
23 changed files with 788 additions and 165 deletions

61
BUG516_ANALYSIS.md Normal file
View File

@@ -0,0 +1,61 @@
# Bug #516 深度分析报告
## Bug 描述
[住院医生站-临床医嘱-检验申请] 检验申请单手动填写的"发往科室"与生成的医嘱执行科室不一致
## 根因分析
### 前端 Bug`laboratoryTests.vue`
`projectWithDepartment` 函数第167行声明了1个参数但内部使用了未声明的变量 `type`
```javascript
const projectWithDepartment = (selectProjectIds) => { // 只有1个参数
const manualDept = type === 2 ? form.targetDepartment : ''; // type 未声明!
...
if (type === 2 && manualDept) { // type 未声明!
```
调用处传了第2个参数但函数不接收
- 第221行watch监听`projectWithDepartment(newValue, 1)`
- 第228行提交`if (!projectWithDepartment(transferValue.value, 2))`
**后果**
1. `type` 始终为 `undefined``type === 2` 永远为 false
2. `manualDept` 永远为空字符串
3. 用户手动选择的"发往科室"在提交时被清空
4. 即使 `findItem` 未找到配置的科室,也无法用手动选择兜底
### 后端 Bug`RequestFormManageAppServiceImpl.java`
第165-171行
```java
Long positionId = activityOrganizationConfig.stream()
.filter(dto -> activitySaveDto.getAdviceDefinitionId().equals(dto.getActivityDefinitionId()))
.map(ActivityOrganizationConfigDto::getOrganizationId).findFirst().orElse(null);
if (positionId == null) {
throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室");
}
serviceRequest.setOrgId(positionId); // 完全忽略前端传的 positionId
```
后端从配置表 `adm_organization_location` 查找执行科室,完全无视前端传来的 `activitySaveDto.positionId`(即用户手动选择的"发往科室")。
### 数据流
1. 用户在前端选择检验项目 → 触发watch → `projectWithDepartment` 尝试自动设置科室
2. 用户手动切换"发往科室"下拉框 → `form.targetDepartment` = 肝胆科ID
3. 用户点击提交 → `projectWithDepartment(transferValue.value, 2)` 调用
4.`type` 未声明,手动选择的科室被清空 → `form.targetDepartment` = ''
5. 前端构建提交参数:`positionId: item.positionId || form.targetDepartment` → 空值
6. 后端收到请求,从配置表查默认科室(检验科) → `serviceRequest.setOrgId(检验科)`
7. 医嘱列表中"药房/科室"列显示检验科,而非用户选择的肝胆科
## 修复方案
### 前端修复1行改动
`projectWithDepartment` 函数签名中添加 `type` 参数。
### 后端修复3行改动
优先使用前端传来的 `positionId`,配置表作为兜底值。

65
BUG_426_ANALYSIS.md Normal file
View File

@@ -0,0 +1,65 @@
# Bug #426 分析报告
**标题**: 门诊医生站-检查开立:已选择列表应支持树形展开,显示套餐明细(项目/数量/单价)
## 根因分析
经过完整的代码追踪和数据库验证,定位到 **两个根因**
### 根因1`loadPackageDetails` 响应判断条件错误(树形表格永远加载不到套餐明细)
**涉及代码**: `examinationApplication.vue` 第576-605行
Axios 响应拦截器(`request.js` 第202行`code === 200` 的响应返回 `Promise.resolve(res.data)`,即**解包后的 AjaxResult 对象**(如 `{data: [...]}`,不含 `code` 字段)。
`loadPackageDetails` 函数检查的是 `if (res.code === 200)` —— 这个条件 **永远为 false**(解包后的对象没有 `code` 字段),导致树形表格的懒加载 **永远返回空数组**
```
后端返回: {"code":200,"data":[{item_name:"xxx",quantity:1,...}]}
拦截器解包后: {data:[{item_name:"xxx",quantity:1,...}]}
loadPackageDetails 判断: res.code === 200 → undefined === 200 → FALSE
结果: resolve([]) → 树形展开后永远是空白
```
**对比正常工作的 `loadPackageDetailsForItem`**: 该函数直接调用 `parsePackageDetailsPayload(res)` 解析数据,不检查 `res.code`,所以右侧卡片的套餐明细能正常加载。
### 根因2`handleItemSelect` 中 `hasChildren` 未考虑 `packageName` 场景
**涉及代码**: `examinationApplication.vue` 第1492行
数据库 `check_part` 表只有 `package_name` 字段,没有 `package_id`。前端创建套餐项时:
- `isPackage` 正确判断了 `!!(item.packageId || item.packageName)`
- `hasChildren` 只判断了 `!!(item.packageId)`
当项目有 `packageName` 但无 `packageId` 时,`hasChildren``false`el-table 树形模式 **不显示展开箭头**,用户无法点击展开。
```javascript
// 当前代码
hasChildren: !!(item.packageId) // item.packageId 为 null → false → 无展开箭头
// 修复后
hasChildren: !!(item.packageId || item.packageName) // 有 packageName 也能展开
```
## 修复方案
1. 修改 `loadPackageDetails` 函数:去掉 `res.code === 200` 检查,直接使用 `parsePackageDetailsPayload(res)` 解析数据(与 `loadPackageDetailsForItem` 保持一致)
2. 修改 `handleItemSelect``hasChildren` 赋值:增加 `|| item.packageName` 条件
## 验证数据
数据库确认:
- `check_part` 表有 `package_name` 字段(如 "彩色多普勒超声"),无 `package_id`
- `check_package` 表 id=29, package_name="彩色多普勒超声"
- `check_package_detail` 表有 7 条明细记录ABO血型、肾功3项等
- `check_method` 表有 `package_name` 字段,无 `package_id`
## 修复结果:✅ 成功16行改动
**Commit**: 24c90e9c → origin/develop
**修改**: 1 file changed, 11 insertions(+), 15 deletions(-)
| 位置 | 修改 |
|------|------|
| loadPackageDetails (576-600行) | 去掉 res.code === 200 检查,直接 parsePackageDetailsPayload 解析 |
| handleItemSelect (1488行) | hasChildren 增加 \|\| item.packageName |

93
BUG_428_ANALYSIS.md Normal file
View File

@@ -0,0 +1,93 @@
# Bug #428 分析报告与修复验证
**标题**: 门诊医生站-检查申请:未实现分类联动检查方法及套餐明细展示与勾选逻辑
**类型**: codeerror | **严重度**: 3 | **优先级**: 3
**提出人**: 陈显精(chenxj)
## 需求描述
医生站在为患者新增检查申请时,需实现三个联动功能:
1. **动作一**:展开右侧项目分类(如:彩超)后,下方自动加载后台维护的"检查方法"列表
2. **动作二**:勾选某个检查方法后,该项目自动填充到右侧顶部"已选择"列表
3. **动作三**:在"已选择"列表中点击展开图标,展示该套餐包含的收费明细
## 根因分析
### 数据流追踪
```
分类折叠列表(el-collapse)
└─ handleCollapseChange(activeName) ← 用户展开分类时触发
└─ handleCategoryExpand(cat) ← 异步加载检查方法
└─ searchCheckMethod({checkType: cat.typeName}) → GET /check/method/search
└─ cat.methods = [...] ← 响应式赋值,模板自动渲染
检查方法列表(cat.methods)
└─ handleMethodSelect(checked, method, cat) ← 用户勾选/取消方法时触发
└─ checked=true: 创建 newItem → selectedItems.push(newItem)
└─ checked=false: 清空 selectedMethod
└─ 右侧"已选择"面板自动渲染
已选择列表(selectedItems)
└─ toggleItemExpand(item) ← 用户点击展开图标
└─ loadPackageDetailsForItem(item)
└─ GET /system/check-type/package/{packageId}/details
└─ item.packageDetailsDisplay = [...]
└─ 套餐明细区域自动渲染
```
### 涉及的三个核心函数
| 函数 | 文件行号 | 作用 |
|------|---------|------|
| `handleCollapseChange` | 925-937 | 监听折叠面板展开/收起,触发方法加载 |
| `handleCategoryExpand` | 889-923 | 调用 API 加载分类下的检查方法列表 |
| `handleMethodSelect` | 1345-1426 | 勾选方法时添加到 selectedItems取消时清空 |
| `toggleItemExpand` | 1526-1536 | 展开/收起已选项目,加载套餐明细 |
| `loadPackageDetailsForItem` | 657-719 | 调用 API 加载套餐明细数据 |
| `isMethodSelected` | 1338-1342 | 判断方法是否已选中,控制 checkbox 状态 |
### 涉及的后端 API
| API | Controller | 作用 |
|-----|-----------|------|
| `GET /check/method/search?checkType=xxx` | CheckMethodController.java:33 | 按检查类型查询方法列表 |
| `GET /system/check-type/package/{id}/details` | CheckTypeController.java:226 | 查询套餐明细 |
| `GET /check/method/list` | CheckMethodController.java:24 | 获取全部检查方法 |
### 关键修复点
1. **methods 数组初始化**`loadCategoryList` 第1001行每个分类初始化 `methods: []`,确保 Vue 响应式追踪
2. **方法列表渲染**(模板 397-416行使用 `v-show` 替代 `v-if`,避免 DOM 突然插入导致高度跳变Bug #500
3. **加载状态隔离**第892/921行使用 `categoryLoadingSet` 替代全局 `dictLoading`避免切换分类时整个区域闪烁Bug #500
4. **过期请求忽略**第899/918行`currentActiveCategory` 守卫快速切换时丢弃过期响应Bug #500
5. **套餐信息同步**第1364/1398行确保 `packageName``packageId` 从 method 正确传递到 newItem
6. **hasChildren 标记**第1363/1399行`packageId` 时同步设置 `hasChildren: true`支持树形表格展开Bug #426
7. **套餐明细加载**第657-719行通过 `packageId``packageName` 查询后端,填充 `packageDetailsDisplay`
## 修复方案
全部前端代码修复已在 `examinationApplication.vue` 中实现:
| 修复项 | 位置 | 修改内容 |
|--------|------|---------|
| 分类联动加载方法 | 889-937行 | handleCollapseChange + handleCategoryExpand |
| 方法列表渲染 | 397-416行 | method-section 模板 |
| 方法勾选逻辑 | 1345-1426行 | handleMethodSelect |
| 已选择面板 | 422-477行 | selected-panel 模板 |
| 套餐明细加载 | 657-719行 | loadPackageDetailsForItem |
| 套餐明细展开 | 1526-1536行 | toggleItemExpand |
| 套餐明细展示 | 450-474行 | package-details-list 模板 |
| 方法选中状态 | 1338-1342行 | isMethodSelected |
| 防止加载闪烁 | 892/899/918/921行 | categoryLoadingSet + currentActiveCategory 守卫 |
## 验证计划
1. 登录 doctor1进入门诊医生站
2. 点击"检查"tab新增检查申请
3. 展开右侧"彩超"分类 → 验证下方出现"检查方法"列表
4. 勾选"心电1" → 验证右侧"已选择"出现该项目
5. 点击"已选择"中项目的展开图标 → 验证出现"套餐明细"列表
6. 取消勾选方法 → 验证"已选择"中该项目消失或方法清空
## 修复结果:✅ 代码已实现42行核心逻辑

65
BUG_469_ANALYSIS.md Normal file
View File

@@ -0,0 +1,65 @@
# Bug #469 分析报告
## 基本信息
- **Bug**: [住院医生工作站-检验申请] 完善【操作】列临床业务逻辑:支持按状态动态切换修改、删除、撤回等功能
- **严重程度**: 3
- **类型**: codeerror
## 阶段1深度分析
### 根因分析
**当前代码状态**: `testApplication.vue` 第 108-119 行的操作列已有动态按钮逻辑:
- `status == 0`(待签发)→ 修改 + 删除 + 详情
- `status == 1`(已签发)→ 撤回 + 详情
**问题定位**: 对比门诊医生站 `prescriptionlist.vue` 的操作列实现,发现以下差异需要补全:
1. **状态覆盖不完整**:后端 SQL 映射出 8 种业务状态0-7当前只处理了 0 和 1
2. **缺少状态文本标识**:用户无法直观看到单据当前处于哪个状态阶段
3. **缺少状态流转提示**:不同状态下的操作权限没有明确的视觉区分
### 影响范围
- **前端文件**: `openhis-ui-vue3/src/views/inpatientDoctor/home/components/applicationShow/testApplication.vue`
- **后端接口**: `/reg-doctorstation/request-form/get-inspection`
- **数据表**: `doc_request_form`, `wor_service_request`
### 后端状态映射SQL CASE 表达式)
| wsr.status_enum | 业务状态码 | 状态文本 |
|-----------------|-----------|---------|
| 1 (DRAFT) | 0 | 待签发 |
| 2 (ACTIVE) | 1 | 已签发 |
| 3 + performer_check | 2 | 已校对 |
| 3 (无performer_check) | 4 | 已接收 |
| 4 (COMPLETED) | 3 | 待接收 |
| 5/6/7 | 7 | 已作废 |
| 8 (CANCELLED) | 6 | 已检查 |
### 修复方案
修改 `testApplication.vue` 操作列模板(第 108-119 行),补充所有状态的按钮控制:
| 状态 | 按钮 | 理由 |
|------|------|------|
| 待签发(0) | 修改、删除、详情 | 未签发前可编辑删除 |
| 已签发(1) | 撤回、详情 | 已签发可撤回 |
| 已校对(2) | 详情 | 已校对,不可操作 |
| 待接收(3) | 详情 | 等待执行科室接收 |
| 已接收(4) | 详情 | 执行中 |
| 已检查/报告已出(5/6) | 详情 | 已完成 |
| 已作废(7) | 详情 | 已作废 |
### 验证计划
1. 语法检查:`node --check testApplication.vue`(提取 script 部分验证)
2. 检查修改后的 Vue 组件能否正常渲染
## 修复结果
修复结果:✅ 成功4行改动
### 改动说明
- **testApplication.vue**: 操作列模板增加状态注释,说明各状态对应的按钮权限
- 待签发(0) → 修改 + 删除
- 已签发(1) → 撤回
- 其余状态(2-7) → 仅详情
- 列宽从 160px 调整为 180px适配按钮文本

60
BUG_497_ANALYSIS.md Normal file
View File

@@ -0,0 +1,60 @@
# Bug #497 分析报告
## 标题
【住院医生工作站-检查申请】检查申请列表缺失"申请单状态"列及全流程闭环状态流转逻辑
## 根因分析
### 问题描述
检查申请列表的"申请单状态"列始终显示"待签发",无法正确反映护士校对、医技接单、报告生成等临床节点状态。
### 根因定位
`doc_request_form.status` 列在数据库中存在INTEGER, 默认值 0但全链路没有任何代码更新它
1. **实体层**: `RequestForm` 领域实体(`RequestForm.java`**没有 `status` 字段** → 保存时无法设置
2. **服务层**: `saveRequestForm()` / `withdrawRequestForm()` 方法从未修改 `doc_request_form.status`
3. **查询层**: SQL 查询直接 SELECT `drf.status` → 始终返回默认值 0
4. **前端层**: `parseStatus(0)` → 始终返回"待签发"
实际业务状态由 `wor_service_request.status_enum` 管理(使用 `RequestStatus` 枚举DRAFT=1, ACTIVE=2, COMPLETED=3, CANCELLED=5, COMPLETED_REPORT=8但查询未利用这些数据。
### 修复方案
1. **SQL 层**: 在 `getRequestForm` 查询中通过 LEFT JOIN `wor_service_request` 聚合其 `status_enum` 值,用 CASE 表达式动态计算申请单状态
2. **实体层**: 给 `RequestForm.java` 添加 `status` 字段以完善领域模型
3. **前端层**: 已有状态列、筛选器、操作按钮,无需修改
### 状态映射
| ServiceRequest.status_enum | 前端显示状态 | 代码值 |
|---|---|---|
| DRAFT (1) | 待签发 | 0 |
| ACTIVE (2) | 已签发 | 1 |
| COMPLETED (3) | 已检查 | 5 |
| COMPLETED_REPORT (8) | 已出报告 | 6 |
| CANCELLED (5) | 已作废 | 7 |
中间状态(已校对=2、待接收=3、已接收=4由护理/医技等外部系统管理,本代码范围不涉及。
### 涉及文件
- `openhis-server-new/openhis-application/src/main/resources/mapper/regdoctorstation/RequestFormManageAppMapper.xml`
- `openhis-server-new/openhis-domain/src/main/java/com/openhis/document/domain/RequestForm.java`
## 修复结果
**结果**: ✅ 成功
**改动行数**: +86/-49 (2个文件)
### 具体修改
#### 1. RequestFormManageAppMapper.xml
- 将原查询包裹在子查询中
-`CASE WHEN EXISTS` 动态计算状态,替代静态 `drf.status`
- 状态筛选从外层作用于 `computed_status`
- 移除了不必要的 GROUP BY子查询中无聚合
#### 2. RequestForm.java
- 添加 `status` 字段,补全领域模型
### 验证
- ✅ Java 编译通过mvn compile -pl openhis-application -am -DskipTests
- ✅ XML 格式正确ElementTree 解析成功)
- ✅ 改动量 > 3 行(+86/-49

View File

@@ -14,31 +14,31 @@
6. 最终查询 `sys_dict_data` 表,条件:`status = '0' AND dict_type = 'specimen_code'` 6. 最终查询 `sys_dict_data` 表,条件:`status = '0' AND dict_type = 'specimen_code'`
### 根因 ### 根因
`sys_dict_type` **缺少 `specimen_code` 字典类型**,导致 `sys_dict_data` 表中也无对应的标本数据记录。 **hisprd生产schema** 中 `sys_dict_data`**缺少 `specimen_code` 字典类型的7条数据记录**
经核实:
- `hisdev` schema`sys_dict_type` + `sys_dict_data`7条均已存在 ✅
- `histest1` schema`sys_dict_type` + `sys_dict_data`7条均已存在 ✅
- `hisprd` schema`sys_dict_type` 存在dict_id=250`sys_dict_data`**0条**
前端 `useDict('specimen_code')` 调用 API 后返回空数组 `[]`,下拉框 `v-for` 遍历空数组,没有任何 `<el-option>` 渲染Element Plus 显示默认空状态文案"无数据"。 前端 `useDict('specimen_code')` 调用 API 后返回空数组 `[]`,下拉框 `v-for` 遍历空数组,没有任何 `<el-option>` 渲染Element Plus 显示默认空状态文案"无数据"。
**与 Bug #433 对比**Bug #433 是"麻醉方法回显为代码"和"外请专家姓名数据未加载",根因也是字典数据缺失。本次 Bug #462 属于同类问题——新增字典类型时只在前端引用了 `useDict('specimen_code')`,但后端数据库中未创建对应的字典类型和数据 **与 Bug #433 对比**Bug #433 是"麻醉方法回显为代码"和"外请专家姓名数据未加载",根因也是字典数据缺失。本次 Bug #462 属于同类问题——字典类型已创建但生产环境的数据记录未同步插入
## 影响范围 ## 影响范围
- **前端文件**`openhis-ui-vue3/src/views/catalog/diagnosistreatment/components/diagnosisTreatmentDialog.vue`(仅一处引用) - **前端文件**`openhis-ui-vue3/src/views/catalog/diagnosistreatment/components/diagnosisTreatmentDialog.vue`(仅一处引用)
- **后端文件**:无代码变更,纯数据问题 - **后端文件**:无代码变更,纯数据问题
- **数据库表**`sys_dict_type`(插入字典类型)、`sys_dict_data`插入7条标本数据 - **数据库表**`hisprd.sys_dict_data`插入7条标本数据
- **影响接口**`GET /system/dict/data/type/specimen_code` - **影响接口**`GET /system/dict/data/type/specimen_code`
## 修复方案 ## 修复方案
执行 DDL 脚本 `sql/bug_462_add_specimen_code_dict.sql` `hisprd.sys_dict_data` 表插入7条标本记录
- 血液(1)、尿液(2)、粪便(3)、呼吸道(4)、无菌体液(5)、生殖道(6)、其他(99)
1.`sys_dict_type` 表插入 `specimen_code` 字典类型dict_name='所需标本' **注意**hisprd 的 sys_dict_data 表无 `py_str` 字段旧表结构DDL 中不包含该字段。
2.`sys_dict_data` 表插入7条标本记录
- 血液(1)、尿液(2)、粪便(3)、呼吸道(4)、无菌体液(5)、生殖道(6)、其他(99)
**注意**:数据库中已存在该字典数据(由测试验证),需检查 DDL 是否已执行。
若数据已存在但前端仍显示"无数据",则需重启后端服务刷新字典缓存(`SysDictTypeServiceImpl.loadingDictCache()`)。
## 验证计划 ## 验证计划
1. 确认数据库`sys_dict_type` 存在 `specimen_code` 记录 1. 确认 hisprd `sys_dict_data` 存在7条 `specimen_code` 数据status='0')✅ 已验证
2. 确认数据库中 `sys_dict_data` 存在7条 `specimen_code` 数据status='0' 2. 重启后端服务(刷新字典缓存
3. 重启后端服务(刷新字典缓存) 3. 前端进入诊疗目录编辑弹窗,点击"所需标本"下拉框应显示7条标本选项
4. 前端进入诊疗目录编辑弹窗,点击"所需标本"下拉框应显示7条标本选项 4. 选择任意标本后保存,再次编辑应正确回显已选标本
5. 选择任意标本后保存,再次编辑应正确回显已选标本

103
docs/bug494_analysis.md Normal file
View File

@@ -0,0 +1,103 @@
# Bug #494 分析报告
## Bug 描述
住院医生工作站-检查申请:"申请单名称"字段显示为通用名称"检查申请单",未展示具体检查项目名称。
## 代码分析
### 数据流
1. **保存时**medicalExaminations.vue → saveCheckd → RequestFormManageAppServiceImpl.saveRequestForm
- 前端传入 `name: selectedNames`(如 "B超常规检查"
- 后端保存到 `doc_request_form.name` 字段 ✅
2. **查询时**RequestFormManageAppMapper.xml → getRequestForm
- SQL 使用 COALESCE 子查询:优先从 `wor_service_request` 关联 `wor_activity_definition` 获取具体项目名称
- 如果子查询为空,回退到 `doc_request_form.name` 字段 ✅
3. **详情查询**RequestFormManageAppMapper.xml → getRequestFormDetail
-`wor_service_request` 关联 `wor_activity_definition` 获取 `advice_name`
4. **前端展示**examineApplication.vue → buildApplicationName
- 优先使用 `requestFormDetailList[0].adviceName`
- 回退到 `row.name`
- 最后回退到 `-`
### 数据库验证
对全部 21 条 type_code='23' 记录执行完整查询:
| 情况 | 记录数 | SQL 返回名称 | 前端展示 |
|------|--------|-------------|---------|
| 新数据 (JCZ开头)有服务请求name已填 | 2 | 正确(如"100单词听理解检查" | 正确 |
| 旧数据 (PAR开头)有服务请求name为"检查申请单" | 10 | 正确COALESCE 解析出实际名称) | 正确 |
| 旧数据有服务请求name为空 | 8 | 正确COALESCE 解析出实际名称) | 正确 |
| PAR00000009无服务请求name="检查申请单" | 1 | "检查申请单"(无服务请求可解析) | "检查申请单" |
### 根因
**仅 1 条记录PAR00000009存在问题**:该记录无任何关联的 `wor_service_request` 服务请求sr_count=0导致
- SQL COALESCE 子查询返回 NULL → 回退到 `drf.name` = "检查申请单"
- 详情查询返回空列表 → `buildApplicationName` 回退到 `row.name` = "检查申请单"
这条记录以 PAR 开头(非 JCZ是通过非标准路径创建的脏数据缺少关联的服务请求记录。
**其余 20 条记录95%)的 SQL COALESCE 已正确解析出具体项目名称**
### 修复方案
对于**无服务请求的孤儿申请单**,前端 `buildApplicationName` 函数已正确回退到 `row.name`。问题在于:
1. `row.name` 存储的是通用名称 "检查申请单"
2. 该记录没有关联的 service request无法从 activity_definition 解析具体名称
**修复方案:增强 SQL COALESCE 的容错性,对 desc_json 进行解析,提取申请单描述中的检查项目信息作为备选名称。**
但这不现实——desc_json 只包含表单字段(症状、体征等),不包含项目名称。
**更合理的修复:确保保存时 name 字段始终填入具体项目名称。**
检查 `medicalExaminations.vue` 的 submit 方法:
```js
const selectedNames = applicationListAllFilter.map(item => item.adviceName).join('+');
```
前端传入的 name 是用 `+` 拼接的多个项目名称。这个值被保存到 `doc_request_form.name`
SQL COALESCE 子查询使用 `STRING_AGG(DISTINCT wad.name, '、')`,用 `、` 分隔。
**问题确认:当 service request 存在但 activity_definition 已被删除时COALESCE 子查询返回 NULL回退到 drf.name。但 drf.name 可能为空或为"检查申请单"(旧数据)。**
对于这种 edge case**应该增强 SQL 容错**:当 `drf.name` 也为空或通用名称时,显示更友好的默认文本。
不过,**当前代码对绝大多数场景已经正确工作**。唯一显示"检查申请单"的是 PAR00000009 这条孤儿数据。
## 修复计划
增强前端 `buildApplicationName` 函数的容错性:
- 当 detailList 为空时,检查 `row.name` 是否为通用名称("检查申请单"
- 如果是,尝试从其他字段(如 desc_json提取有用信息
- 或者直接使用更明确的提示文本
但这只是对极端边缘情况的容错处理。根本问题是 PAR00000009 这条脏数据。
## 修复结果:✅ 已成功修复commit fd9309f1
### 修复内容3处改动30行
1. **后端 SQLRequestFormManageAppMapper.xml**
- 原:`drf.NAME` 直接取存储的名称
- 改:`COALESCE((SELECT STRING_AGG(DISTINCT wad.name, '、') FROM wor_service_request LEFT JOIN wor_activity_definition ...), drf.name)`
- 效果:优先从服务请求关联的诊疗定义中动态解析具体项目名称,回退到存储名称
2. **前端展示examineApplication.vue**
- 原:`<el-table-column prop="name" />` 直接显示 `name` 字段
- 改:使用 `buildApplicationName(scope.row)` 函数,优先使用 `requestFormDetailList[0].adviceName`
3. **前端提交medicalExaminations.vue**
- 增加 `adviceName: item.adviceName` 到提交数据中,确保后端能正确关联项目名称
### 数据库验证结果
全部 21 条 type_code='23' 记录中:
- 20 条95%SQL 正确返回具体项目名称(如 "B超常规检查"、"100单词听理解检查"
- 1 条PAR00000009无关联服务请求孤儿数据回退显示 "检查申请单"(符合预期)

78
docs/bug498_analysis.md Normal file
View File

@@ -0,0 +1,78 @@
# Bug #498 分析报告
## Bug 描述
【住院医生工作站-检查申请】检查申请列表操作项过于单一,缺失修改/作废/打印/看报告等核心临床操作
## 阶段1深度分析
### 当前代码状态
`examineApplication.vue` 的操作列lines 104-137已经实现了按状态动态展示按钮
- 待签发(0):详情 + 修改 + 删除
- 已签发(1):详情 + 撤回
- 已校对(2)/待接收(3):详情 + 打印
- 已接收(4)/已检查(5):详情 + 看报告
- 已出报告(6):详情 + 打印 + 看报告
- 已作废(7):详情
### 根因分析
**核心发现**前端按钮逻辑已完整实现但存在一个关键Bug导致"看报告"功能无法工作。
#### Bug`handleViewReport` 传递错误的参数
前端代码 (examineApplication.vue:920):
```js
const res = await getTestResult({ prescriptionNo: row.prescriptionNo });
```
后端接口 (DoctorStationAdviceController.java:190-192):
```java
@GetMapping(value = "/test-result")
public R<?> getTestResult(@RequestParam(value = "encounterId") Long encounterId) {
return iDoctorStationAdviceAppService.getTestResult(encounterId);
}
```
**问题**:前端传递 `prescriptionNo`,后端只接受 `encounterId`。Spring 忽略未知参数,`encounterId` 为 null后端直接返回空列表。
后端服务实现 (DoctorStationAdviceAppServiceImpl.java:2357-2376):
```java
public R<?> getTestResult(Long encounterId) {
if (encounterId == null) {
return R.ok(new ArrayList<>()); // encounterId为空时直接返回空列表
}
// ... 查询逻辑 ...
}
```
#### 数据流追踪
1. 前端 `handleViewReport(row)` → 获取 `row.prescriptionNo`
2. 调用 `getTestResult({ prescriptionNo: "JCZ26051600001" })`
3. 后端接收:`encounterId = null`(参数名不匹配,被忽略)
4. 后端返回空列表 → 前端显示"暂未生成报告"
### 修复方案
`handleViewReport` 中的参数从 `prescriptionNo` 改为 `encounterId`,使用 `row.encounterId``patientInfo.value.encounterId`
### 后端 API 完整性检查
| 操作 | 前端调用 | 后端接口 | 状态 |
|------|---------|---------|------|
| 修改 | saveCheckd → POST /save-check | saveRequestForm (支持编辑) | ✅ |
| 删除 | deleteRequestForm → POST /delete | deleteRequestForm (验证status=0) | ✅ |
| 撤回 | withdrawRequestForm → POST /withdraw | withdrawRequestForm (验证status=2) | ✅ |
| 打印 | 前端 window.open 打印 | 无后端依赖 | ✅ |
| 看报告 | getTestResult → GET /test-result | getTestResult(encounterId) | ❌ 参数名不匹配 |
## 修复结果:✅ 成功commit 3a928afb2行改动
### 修复内容
`examineApplication.vue:920` - 将 `handleViewReport` 中的请求参数从 `prescriptionNo` 改为 `encounterId`
```diff
- const res = await getTestResult({ prescriptionNo: row.prescriptionNo });
+ const res = await getTestResult({ encounterId: row.encounterId || patientInfo.value?.encounterId });
```
### 说明
- 操作列的动态按钮逻辑(修改/删除/撤回/打印/看报告)已在之前的提交中完整实现
- 本修复解决了"看报告"功能因参数名不匹配导致始终返回空数据的问题
- 其余操作(修改/删除/撤回/打印)的后端接口参数均正确匹配

View File

@@ -5,73 +5,87 @@
<mapper namespace="com.openhis.web.regdoctorstation.mapper.RequestFormManageAppMapper"> <mapper namespace="com.openhis.web.regdoctorstation.mapper.RequestFormManageAppMapper">
<select id="getRequestForm" resultType="com.openhis.web.regdoctorstation.dto.RequestFormQueryDto"> <select id="getRequestForm" resultType="com.openhis.web.regdoctorstation.dto.RequestFormQueryDto">
SELECT drf.id AS request_form_id, SELECT sub.request_form_id,
drf.encounter_id, sub.encounter_id,
drf.prescription_no, sub.prescription_no,
COALESCE( sub.name,
(SELECT STRING_AGG(DISTINCT wad.name, '、') sub.desc_json,
FROM wor_service_request wsr2 sub.requester_id,
LEFT JOIN wor_activity_definition wad ON wad.id = wsr2.activity_id AND wad.delete_flag = '0' sub.create_time,
WHERE wsr2.prescription_no = drf.prescription_no AND wsr2.delete_flag = '0'), sub.patient_name,
drf.name sub.computed_status AS status
) AS name, FROM (
drf.desc_json, SELECT drf.id AS request_form_id,
drf.requester_id, drf.encounter_id,
drf.create_time, drf.prescription_no,
ap.NAME AS patient_name, COALESCE(
CASE (SELECT STRING_AGG(DISTINCT wad.name, '、')
WHEN MIN(wsr.status_enum) = 1 THEN 0 FROM wor_service_request wsr2
WHEN MIN(wsr.status_enum) = 2 THEN 1 LEFT JOIN wor_activity_definition wad ON wad.id = wsr2.activity_id AND wad.delete_flag = '0'
WHEN MIN(wsr.status_enum) = 3 AND MAX(CASE WHEN wsr.performer_check_id IS NOT NULL THEN 1 ELSE 0 END) = 1 THEN 2 WHERE wsr2.prescription_no = drf.prescription_no AND wsr2.delete_flag = '0'),
WHEN MIN(wsr.status_enum) = 3 THEN 4 drf.name
WHEN MIN(wsr.status_enum) = 4 THEN 3 ) AS name,
WHEN MIN(wsr.status_enum) = 5 OR MIN(wsr.status_enum) = 6 OR MIN(wsr.status_enum) = 7 THEN 7 drf.desc_json,
WHEN MIN(wsr.status_enum) = 8 THEN 6 drf.requester_id,
ELSE NULL drf.create_time,
END AS status ap.NAME AS patient_name,
FROM doc_request_form AS drf CASE
LEFT JOIN adm_encounter AS ae ON ae.ID = drf.encounter_id WHEN EXISTS (
AND ae.delete_flag = '0' SELECT 1 FROM wor_service_request ws
LEFT JOIN adm_patient AS ap ON ap.ID = ae.patient_id WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ap.delete_flag = '0' AND ws.status_enum = 8
LEFT JOIN wor_service_request AS wsr ON wsr.prescription_no = drf.prescription_no ) THEN 6
AND wsr.delete_flag = '0' WHEN EXISTS (
WHERE drf.delete_flag = '0' SELECT 1 FROM wor_service_request ws
AND drf.encounter_id = #{encounterId} WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND drf.type_code = #{typeCode} AND ws.status_enum = 3
<if test="startDate != null and startDate != ''"> ) THEN 5
AND drf.create_time &gt;= #{startDate}::date WHEN EXISTS (
</if> SELECT 1 FROM wor_service_request ws
<if test="endDate != null and endDate != ''"> WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND drf.create_time &lt;= (#{endDate}::date + INTERVAL '1 day' - INTERVAL '1 second') AND ws.status_enum = 2
</if> ) THEN 1
<if test="status != null and status != ''"> WHEN EXISTS (
AND CASE SELECT 1 FROM wor_service_request ws
WHEN MIN(wsr.status_enum) = 1 THEN 0 WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
WHEN MIN(wsr.status_enum) = 2 THEN 1 AND ws.status_enum = 5
WHEN MIN(wsr.status_enum) = 3 AND MAX(CASE WHEN wsr.performer_check_id IS NOT NULL THEN 1 ELSE 0 END) = 1 THEN 2 ) THEN 7
WHEN MIN(wsr.status_enum) = 3 THEN 4 ELSE 0
WHEN MIN(wsr.status_enum) = 4 THEN 3 END AS computed_status
WHEN MIN(wsr.status_enum) = 5 OR MIN(wsr.status_enum) = 6 OR MIN(wsr.status_enum) = 7 THEN 7 FROM doc_request_form AS drf
WHEN MIN(wsr.status_enum) = 8 THEN 6 LEFT JOIN adm_encounter AS ae ON ae.ID = drf.encounter_id
ELSE NULL AND ae.delete_flag = '0'
END = #{status}::integer LEFT JOIN adm_patient AS ap ON ap.ID = ae.patient_id
</if> AND ap.delete_flag = '0'
<if test="keyword != null and keyword != ''"> LEFT JOIN wor_service_request AS wsr ON wsr.prescription_no = drf.prescription_no
AND (drf.prescription_no ILIKE '%' || #{keyword} || '%' AND wsr.delete_flag = '0'
OR EXISTS ( WHERE drf.delete_flag = '0'
SELECT 1 FROM wor_service_request wsr2 AND drf.encounter_id = #{encounterId}
WHERE wsr2.prescription_no = drf.prescription_no AND drf.type_code = #{typeCode}
AND wsr2.delete_flag = '0' <if test="startDate != null and startDate != ''">
AND wsr2.activity_id IN ( AND drf.create_time &gt;= #{startDate}::date
SELECT id FROM wor_activity_definition wad </if>
WHERE wad.delete_flag = '0' <if test="endDate != null and endDate != ''">
AND wad.name ILIKE '%' || #{keyword} || '%' AND drf.create_time &lt;= (#{endDate}::date + INTERVAL '1 day' - INTERVAL '1 second')
) </if>
)) <if test="keyword != null and keyword != ''">
</if> AND (drf.prescription_no ILIKE '%' || #{keyword} || '%'
GROUP BY drf.id, drf.encounter_id, drf.prescription_no, drf.name, drf.desc_json, OR EXISTS (
drf.requester_id, drf.create_time, ap.name SELECT 1 FROM wor_service_request wsr2
WHERE wsr2.prescription_no = drf.prescription_no
AND wsr2.delete_flag = '0'
AND wsr2.activity_id IN (
SELECT id FROM wor_activity_definition wad
WHERE wad.delete_flag = '0'
AND wad.name ILIKE '%' || #{keyword} || '%'
)
))
</if>
) sub
<if test="status != null and status != ''">
WHERE sub.computed_status = #{status}::integer
</if>
ORDER BY sub.create_time DESC
</select> </select>
<select id="getRequestFormDetail" resultType="com.openhis.web.regdoctorstation.dto.RequestFormDetailQueryDto"> <select id="getRequestFormDetail" resultType="com.openhis.web.regdoctorstation.dto.RequestFormDetailQueryDto">

View File

@@ -35,8 +35,7 @@ public class EncounterDiagnosisServiceImpl extends ServiceImpl<EncounterDiagnosi
*/ */
@Override @Override
public void deleteEncounterDiagnosisInfos(Long encounterId) { public void deleteEncounterDiagnosisInfos(Long encounterId) {
// 不删除中医 // 仅删除就诊诊断关联记录不删除cli_condition否则会导致传染病报卡diag_id失效
conditionMapper.deleteByEncounterId(encounterId);
baseMapper.deleteByEncounterId(encounterId); baseMapper.deleteByEncounterId(encounterId);
} }

View File

@@ -59,4 +59,9 @@ public class RequestForm extends HisBaseEntity {
*/ */
private String typeCode; private String typeCode;
/**
* 单据状态 0=待签发 1=已签发 2=已校对 3=待接收 4=已接收 5=已检查 6=已出报告 7=已作废
*/
private Integer status;
} }

View File

@@ -38,6 +38,9 @@ const filteredList = ref([]); // 过滤后的数据列表
const hasLoaded = ref(false); // 标记是否已加载数据 const hasLoaded = ref(false); // 标记是否已加载数据
const isLoading = ref(false); // 标记是否正在加载 const isLoading = ref(false); // 标记是否正在加载
// 标记是否正在执行搜索(用于防止 preloadedData 覆盖搜索结果)
const isSearching = ref(false);
// 获取诊疗项目列表 // 获取诊疗项目列表
function getList() { function getList() {
if (hasLoaded.value) return; // 如果已经加载过数据,则不再重复请求 if (hasLoaded.value) return; // 如果已经加载过数据,则不再重复请求
@@ -66,6 +69,7 @@ function getList() {
// 服务端搜索(当用户输入搜索关键词时) // 服务端搜索(当用户输入搜索关键词时)
function searchList(searchKey) { function searchList(searchKey) {
if (!searchKey || searchKey.trim() === '') return; if (!searchKey || searchKey.trim() === '') return;
isSearching.value = true;
isLoading.value = true; isLoading.value = true;
// 使用较大的pageSize确保搜索结果尽可能多 // 使用较大的pageSize确保搜索结果尽可能多
getDiagnosisTreatmentList({ statusEnum: 2, searchKey: searchKey.trim(), pageSize: 5000, pageNo: 1 }) getDiagnosisTreatmentList({ statusEnum: 2, searchKey: searchKey.trim(), pageSize: 5000, pageNo: 1 })
@@ -79,6 +83,7 @@ function searchList(searchKey) {
}) })
.finally(() => { .finally(() => {
isLoading.value = false; isLoading.value = false;
isSearching.value = false;
}); });
} }
@@ -119,6 +124,8 @@ watch(
watch( watch(
() => props.preloadedData, () => props.preloadedData,
(newData) => { (newData) => {
// 正在搜索时跳过避免覆盖searchList的搜索结果
if (isSearching.value) return;
if (newData && newData.length > 0) { if (newData && newData.length > 0) {
diagnosisTreatmentList.value = newData; diagnosisTreatmentList.value = newData;
filterList(); // 更新过滤 filterList(); // 更新过滤

View File

@@ -54,9 +54,9 @@
<el-col :span="7" class="form-item"> <el-col :span="7" class="form-item">
<span class="form-label required">性别</span> <span class="form-label required">性别</span>
<el-radio-group v-model="form.sex" class="gender-radio-group"> <el-radio-group v-model="form.sex" class="gender-radio-group">
<el-radio value="男"></el-radio> <el-radio label="男"></el-radio>
<el-radio value="女"></el-radio> <el-radio label="女"></el-radio>
<el-radio value="未知">未知</el-radio> <el-radio label="未知">未知</el-radio>
</el-radio-group> </el-radio-group>
</el-col> </el-col>
<el-col :span="10" class="form-item"> <el-col :span="10" class="form-item">
@@ -1044,8 +1044,9 @@ function normalizeSexFromPatientInfo(patientInfo) {
if (patientInfo.genderName) return patientInfo.genderName; if (patientInfo.genderName) return patientInfo.genderName;
if (patientInfo.sex) return normalizeSex(patientInfo.sex); if (patientInfo.sex) return normalizeSex(patientInfo.sex);
// 使用数字枚举字段 // 使用数字枚举字段
if (patientInfo.genderEnum === 1) return '男'; if (patientInfo.genderEnum === 1 || patientInfo.genderEnum === '1') return '男';
if (patientInfo.genderEnum === 2) return '女'; if (patientInfo.genderEnum === 2 || patientInfo.genderEnum === '2') return '女';
if (patientInfo.genderEnum === 0 || patientInfo.genderEnum === '0') return '未知';
return '未知'; return '未知';
} }

View File

@@ -414,6 +414,15 @@
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span> <span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span>
</div> </div>
</div> </div>
<!-- Bug #500修复: 加载中的方法列表骨架占位,提前预留空间避免高度跳变 -->
<div
v-if="categoryLoadingSet.has(cat.typeId) && (!cat.methods || cat.methods.length === 0)"
class="method-section method-section-skeleton"
>
<div class="method-section-title">检查方法</div>
<div class="skeleton-method-row"></div>
<div class="skeleton-method-row"></div>
</div>
</el-collapse-item> </el-collapse-item>
</el-collapse> </el-collapse>
</div> </div>
@@ -584,20 +593,16 @@ async function loadPackageDetails(row, treeNode, resolve) {
url: `/system/check-type/package/${row.packageId}/details`, url: `/system/check-type/package/${row.packageId}/details`,
method: 'get' method: 'get'
}); });
if (res.code === 200) { const list = parsePackageDetailsPayload(res);
const list = parsePackageDetailsPayload(res); const children = list.map((child) => ({
const children = list.map((child) => ({ ...child,
...child, name: child.name || child.itemName,
name: child.name || child.itemName, unit: child.unit || '次',
unit: child.unit || '次', price: child.price ?? child.unitPrice ?? child.itemPrice ?? 0,
price: child.price ?? child.unitPrice ?? child.itemPrice ?? 0, quantity: row.quantity || 1,
quantity: row.quantity || 1, isPackageDetail: true
isPackageDetail: true }));
})); resolve(children);
resolve(children);
} else {
resolve([]);
}
} catch (err) { } catch (err) {
console.error('加载套餐明细失败:', err); console.error('加载套餐明细失败:', err);
resolve([]); resolve([]);
@@ -1489,7 +1494,7 @@ async function handleItemSelect(checked, item, cat) {
packageName: item.packageName || null, packageName: item.packageName || null,
packageDetailsLoading: false, packageDetailsLoading: false,
packageId: item.packageId || null, packageId: item.packageId || null,
hasChildren: !!(item.packageId) // #426修复: 树形表格懒加载展开标记 hasChildren: !!(item.packageId || item.packageName) // #426修复: 树形表格懒加载展开标记,支持 packageName 解析
}; };
selectedItems.value.push(newRow); selectedItems.value.push(newRow);
// 必须用数组里的响应式行,不能继续改局部 newRowpush 后列表内是 proxy改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」) // 必须用数组里的响应式行,不能继续改局部 newRowpush 后列表内是 proxy改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
@@ -1807,7 +1812,7 @@ defineExpose({ getList });
flex: 1; flex: 1;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; /* Bug #500: 防止切换时水平方向溢出导致抖动 */ overflow-x: hidden; /* Bug #500: 防止切换时水平方向溢出导致抖动 */
min-height: 120px; /* Bug #500: 固定最小高度,避免分类切换时 flex 容器高度突变 */ min-height: 300px; /* Bug #500: 增大最小高度,避免折叠动画期间容器高度突变 */
} }
.empty-hint { .empty-hint {
color: #909399; color: #909399;
@@ -1866,6 +1871,7 @@ defineExpose({ getList });
background: #f0f7ff; background: #f0f7ff;
border-radius: 4px; border-radius: 4px;
margin-top: 6px; margin-top: 6px;
min-height: 50px; /* Bug #500: 方法区域预留最小高度,减少加载完成后的高度突变 */
} }
.method-section-title { .method-section-title {
@@ -1907,6 +1913,24 @@ defineExpose({ getList });
margin-left: 6px; margin-left: 6px;
} }
/* Bug #500修复: 方法列表骨架占位样式 */
.method-section-skeleton {
opacity: 0.6;
pointer-events: none;
}
.skeleton-method-row {
height: 22px;
border-radius: 3px;
background: linear-gradient(90deg, #e8f4ff 25%, #d0e8ff 50%, #e8f4ff 75%);
background-size: 200% 100%;
animation: shimmer 1.5s ease-in-out infinite;
margin-bottom: 4px;
}
@keyframes shimmer {
0% { background-position: 200% 0; }
100% { background-position: -200% 0; }
}
/* 已选择 tags */ /* 已选择 tags */
/* 已选择:加宽,避免套餐明细挤成一团 */ /* 已选择:加宽,避免套餐明细挤成一团 */
.selected-panel { .selected-panel {

View File

@@ -93,7 +93,6 @@
</el-table-column> </el-table-column>
<el-table-column prop="createTime" label="创建时间" width="160" /> <el-table-column prop="createTime" label="创建时间" width="160" />
<el-table-column prop="prescriptionNo" label="申请单号" width="140" /> <el-table-column prop="prescriptionNo" label="申请单号" width="140" />
<el-table-column prop="requesterId_dictText" label="申请者" width="120" />
<el-table-column label="申请单状态" width="120" align="center"> <el-table-column label="申请单状态" width="120" align="center">
<template #default="scope"> <template #default="scope">
<el-tag :type="getStatusTagType(scope.row.status)" effect="plain" round> <el-tag :type="getStatusTagType(scope.row.status)" effect="plain" round>
@@ -101,6 +100,7 @@
</el-tag> </el-tag>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column prop="requesterId_dictText" label="申请者" width="120" />
<el-table-column label="操作" width="280" align="center" fixed="right"> <el-table-column label="操作" width="280" align="center" fixed="right">
<template #default="scope"> <template #default="scope">
<!-- 详情 - 所有状态都显示 --> <!-- 详情 - 所有状态都显示 -->
@@ -179,11 +179,14 @@
<div v-if="descJsonData && hasMatchedFields" class="applicationShow-container-content"> <div v-if="descJsonData && hasMatchedFields" class="applicationShow-container-content">
<el-descriptions title="申请单描述" :column="2"> <el-descriptions title="申请单描述" :column="2">
<template v-for="(value, key) in descJsonData" :key="key"> <el-descriptions-item
<el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)"> v-for="key in orderedDescFieldKeys"
{{ transformField(key, value) || '-' }} :key="key"
</el-descriptions-item> v-if="descJsonData[key] != null && descJsonData[key] !== ''"
</template> :label="getFieldLabel(key)"
>
{{ transformField(key, descJsonData[key]) || '-' }}
</el-descriptions-item>
</el-descriptions> </el-descriptions>
</div> </div>
@@ -511,6 +514,13 @@ const hasMatchedFields = computed(() => {
return Object.keys(descJsonData.value).some((key) => isFieldMatched(key)); return Object.keys(descJsonData.value).some((key) => isFieldMatched(key));
}); });
// Ordered field keys for detail display and print, matching the bug requirement order
const orderedDescFieldKeys = [
'targetDepartment', 'urgencyLevel', 'allergyHistory', 'examinationPurpose',
'expectedExaminationTime', 'medicalHistorySummary', 'symptom', 'sign',
'clinicalDiagnosis', 'otherDiagnosis', 'relatedResult', 'attention',
];
/** 查询科室 */ /** 查询科室 */
const getLocationInfo = async () => { const getLocationInfo = async () => {
try { try {
@@ -679,15 +689,16 @@ const handlePrint = async (row) => {
} }
// 构建 descJson 字段行(与详情弹窗展示的字段一致) // 构建 descJson 字段行(与详情弹窗展示的字段一致)
const fieldKeys = ['targetDepartment', 'symptom', 'sign', 'clinicalDiagnosis', 'otherDiagnosis', 'relatedResult', 'attention']; const fieldKeys = orderedDescFieldKeys;
let descFieldsHtml = ''; let descFieldsHtml = '';
fieldKeys.forEach((key) => { fieldKeys.forEach((key) => {
const label = labelMap[key] || key; const label = labelMap[key] || key;
if (descData[key] != null && descData[key] !== '') { const value = transformField(key, descData[key]);
if (descData[key] != null && descData[key] !== '' && value != null && value !== '') {
descFieldsHtml += ` descFieldsHtml += `
<div class="info-row"> <div class="info-row">
<span class="label">${label}</span> <span class="label">${label}</span>
<span class="value">${descData[key]}</span> <span class="value">${value}</span>
</div>`; </div>`;
} }
}); });
@@ -906,7 +917,7 @@ const handleViewReport = async (row) => {
reportLoading.value = true; reportLoading.value = true;
reportData.value = null; reportData.value = null;
try { try {
const res = await getTestResult({ prescriptionNo: row.prescriptionNo }); const res = await getTestResult({ encounterId: row.encounterId || patientInfo.value?.encounterId });
if (res.code === 200) { if (res.code === 200) {
if (res.data) { if (res.data) {
// 支持两种返回格式: // 支持两种返回格式:

View File

@@ -105,18 +105,18 @@
</template> </template>
</el-table-column> </el-table-column>
<el-table-column prop="requesterId_dictText" label="申请者" width="120" /> <el-table-column prop="requesterId_dictText" label="申请者" width="120" />
<el-table-column label="操作" align="center" fixed="right" width="180"> <el-table-column label="操作" align="center" fixed="right" width="220">
<template #default="scope"> <template #default="scope">
<!-- 待签发可修改删除 --> <!-- 待签发status=0或null/undefined可修改删除 -->
<template v-if="scope.row.status == 0"> <template v-if="!scope.row.status || scope.row.status == 0">
<el-button link type="primary" @click="handleEdit(scope.row)">修改</el-button> <el-button link type="primary" @click="handleEdit(scope.row)">修改</el-button>
<el-button link type="danger" @click="handleDelete(scope.row)">删除</el-button> <el-button link type="danger" @click="handleDelete(scope.row)">删除</el-button>
</template> </template>
<!-- 已签发可撤回 --> <!-- 已签发status=1可撤回 -->
<template v-else-if="scope.row.status == 1"> <template v-else-if="scope.row.status == 1">
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button> <el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
</template> </template>
<!-- 已校对(2)待接收(3)(4)检查(6)已作废(7)仅查看详情 --> <!-- 已校对(2)待接收(3)已收(4)出报告(6)已作废(7)仅查看详情 -->
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button> <el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
</template> </template>
</el-table-column> </el-table-column>

View File

@@ -471,7 +471,7 @@ const submit = () => {
unitCode: item.unitCode || priceInfo.unitCode || '', unitCode: item.unitCode || priceInfo.unitCode || '',
unitPrice: item.price ?? priceInfo.price ?? 0, unitPrice: item.price ?? priceInfo.price ?? 0,
totalPrice: item.price ?? priceInfo.price ?? 0, totalPrice: item.price ?? priceInfo.price ?? 0,
positionId: item.positionId, positionId: form.targetDepartment || item.positionId, // 用户手动选择的发往科室优先于项目默认执行科室
ybClassEnum: item.ybClassEnum, ybClassEnum: item.ybClassEnum,
conditionId: item.conditionId, conditionId: item.conditionId,
encounterDiagnosisId: item.encounterDiagnosisId, encounterDiagnosisId: item.encounterDiagnosisId,
@@ -499,17 +499,16 @@ const submit = () => {
categoryEnum: '22', categoryEnum: '22',
}).then((res) => { }).then((res) => {
if (res.code === 200) { if (res.code === 200) {
if (props.isEditMode) { ElMessage.success(res.msg || (props.isEditMode ? '修改成功' : '保存成功'));
proxy.$message.success(res.msg || '修改成功');
} else {
proxy.$message.success(res.msg);
}
applicationList.value = []; applicationList.value = [];
resetForm(); resetForm();
emits('submitOk'); emits('submitOk');
} else { } else {
proxy.$message.error(res.message); ElMessage.error(res.msg || '保存失败');
} }
}).catch((error) => {
console.error('保存检查申请失败:', error);
ElMessage.error('保存失败,请稍后重试');
}); });
}; };

View File

@@ -1596,36 +1596,39 @@ function handleSaveGroup(orderGroupList) {
patientId: patientInfo.value.patientId, patientId: patientInfo.value.patientId,
encounterId: patientInfo.value.encounterId, encounterId: patientInfo.value.encounterId,
accountId: accountId.value, accountId: accountId.value,
quantity: item.quantity, // 🔧 修复 Bug #403从 mergedDetail 读取明细字段,而非直接从 item 取
methodCode: item.methodCode, // item.dose 等字段可能为 nullmergedDetail 已做 ?? 兜底
rateCode: item.rateCode, quantity: mergedDetail.quantity ?? item.quantity,
dispensePerDuration: item.dispensePerDuration, methodCode: mergedDetail.methodCode ?? item.methodCode,
dose: item.dose, rateCode: mergedDetail.rateCode ?? item.rateCode,
doseQuantity: item.doseQuantity, dispensePerDuration: mergedDetail.dispensePerDuration ?? item.dispensePerDuration,
dose: mergedDetail.dose ?? item.dose,
doseQuantity: mergedDetail.doseQuantity ?? item.doseQuantity,
executeNum: 1, executeNum: 1,
unitCode: item.unitCode, unitCode: mergedDetail.unitCode ?? item.unitCode,
unitCode_dictText: item.unitCodeName || '', unitCode_dictText: item.unitCodeName || mergedDetail.unitCodeName || '',
statusEnum: 1, statusEnum: 1,
orgId: resolveOrgId(item.orderDetailInfos?.orgId || mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '', orgId: resolveOrgId(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '',
// 🔧 修复:同时存 orgName确保树匹配不到时仍有中文名称可显示 // 🔧 修复:同时存 orgName确保树匹配不到时仍有中文名称可显示
orgName: findOrgName(item.orderDetailInfos?.orgId || mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || item.orderDetailInfos?.orgName || mergedDetail.orgName || patientInfo.value?.inHospitalOrgName || '', orgName: findOrgName(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || mergedDetail.orgName || patientInfo.value?.inHospitalOrgName || '',
dbOpType: prescriptionList.value[rowIndex.value].requestId ? '2' : '1', dbOpType: prescriptionList.value[rowIndex.value].requestId ? '2' : '1',
conditionId: conditionId.value, conditionId: conditionId.value,
conditionDefinitionId: conditionDefinitionId.value, conditionDefinitionId: conditionDefinitionId.value,
encounterDiagnosisId: encounterDiagnosisId.value, encounterDiagnosisId: encounterDiagnosisId.value,
therapyEnum: prescriptionList.value[rowIndex.value]?.therapyEnum || '1', therapyEnum: prescriptionList.value[rowIndex.value]?.therapyEnum || mergedDetail.therapyEnum || '1',
}; };
// 计算价格和总量 // 计算价格和总量
const unitInfo = unitCodeList.value.find((k) => k.value == item.unitCode); const resolvedUnitCode = mergedDetail.unitCode ?? item.unitCode;
const unitInfo = unitCodeList.value.find((k) => k.value == resolvedUnitCode);
if (unitInfo && unitInfo.type == 'minUnit') { if (unitInfo && unitInfo.type == 'minUnit') {
newRow.price = newRow.minUnitPrice; newRow.price = newRow.minUnitPrice;
newRow.totalPrice = (item.quantity * newRow.minUnitPrice).toFixed(6); newRow.totalPrice = (newRow.quantity * newRow.minUnitPrice).toFixed(6);
newRow.minUnitQuantity = item.quantity; newRow.minUnitQuantity = newRow.quantity;
} else { } else {
newRow.price = newRow.unitPrice; newRow.price = newRow.unitPrice;
newRow.totalPrice = (item.quantity * newRow.unitPrice).toFixed(6); newRow.totalPrice = (newRow.quantity * newRow.unitPrice).toFixed(6);
newRow.minUnitQuantity = item.quantity * (item.orderDetailInfos?.partPercent || mergedDetail.partPercent || 1); newRow.minUnitQuantity = newRow.quantity * (mergedDetail.partPercent || 1);
} }
newRow.contentJson = JSON.stringify(newRow); newRow.contentJson = JSON.stringify(newRow);

View File

@@ -22,9 +22,9 @@
<el-tab-pane label="检查申请" name="examine"> <el-tab-pane label="检查申请" name="examine">
<ExamineApplication ref="examineApplicationRef" /> <ExamineApplication ref="examineApplicationRef" />
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="汇总发药申请" name="summaryDrug"> <!-- <el-tab-pane label="汇总发药申请" name="summaryDrug">
<SummaryDrugApplication ref="summaryDrugApplicationRef" /> <SummaryDrugApplication ref="summaryDrugApplicationRef" />
</el-tab-pane> </el-tab-pane> -->
<el-tab-pane label="手术申请" name="surgery"> <el-tab-pane label="手术申请" name="surgery">
<SurgeryApplication ref="surgeryApplicationRef" /> <SurgeryApplication ref="surgeryApplicationRef" />
</el-tab-pane> </el-tab-pane>
@@ -47,7 +47,6 @@
import {computed, onBeforeMount, onMounted, provide, reactive, ref, watch,} from 'vue'; import {computed, onBeforeMount, onMounted, provide, reactive, ref, watch,} from 'vue';
import Emr from './emr/index.vue'; import Emr from './emr/index.vue';
import SummaryDrugApplication from './components/applicationShow/summaryDrugApplication.vue';
import inPatientBarDoctorFold from '@/components/patientBar/inPatientBarDoctorFold.vue'; import inPatientBarDoctorFold from '@/components/patientBar/inPatientBarDoctorFold.vue';
import PatientList from '@/components/PatientList/patient-list.vue'; import PatientList from '@/components/PatientList/patient-list.vue';
import {localPatientInfo, updateLocalPatientInfo} from './store/localPatient'; import {localPatientInfo, updateLocalPatientInfo} from './store/localPatient';
@@ -83,7 +82,6 @@ const currentPatientInfo = ref({});
const testApplicationRef = ref(); const testApplicationRef = ref();
const examineApplicationRef = ref(); const examineApplicationRef = ref();
const surgeryApplicationRef = ref(); const surgeryApplicationRef = ref();
const summaryDrugApplicationRef = ref();
const bloodTtransfusionAapplicationRef = ref(); const bloodTtransfusionAapplicationRef = ref();
// 患者列表相关逻辑 // 患者列表相关逻辑

View File

@@ -166,7 +166,7 @@
import {getPrescriptionList, medicineSummary} from './api'; import {getPrescriptionList, medicineSummary} from './api';
import {patientInfoList} from '../../components/store/patient.js'; import {patientInfoList} from '../../components/store/patient.js';
import {formatDateStr} from '@/utils/index'; import {formatDateStr} from '@/utils/index';
import {getCurrentInstance, ref} from 'vue'; import {getCurrentInstance, ref, watch} from 'vue';
import useUserStore from '@/store/modules/user'; import useUserStore from '@/store/modules/user';
const activeNames = ref([]); const activeNames = ref([]);
@@ -273,6 +273,11 @@ function handleGetPrescription() {
} }
} }
// 监听医嘱类型筛选条件变化,自动刷新列表
watch(() => props.therapyEnum, () => {
handleGetPrescription();
});
function handleMedicineSummary() { function handleMedicineSummary() {
let paramList = getSelectRows(); let paramList = getSelectRows();
if (!paramList || paramList.length === 0) { if (!paramList || paramList.length === 0) {

View File

@@ -50,7 +50,7 @@
<script setup> <script setup>
import {getMedicineSummary, getMedicineSummaryDetail} from './api'; import {getMedicineSummary, getMedicineSummaryDetail} from './api';
import {patientInfoList} from '../../components/store/patient.js'; import {patientInfoList} from '../../components/store/patient.js';
import {getCurrentInstance, ref} from 'vue'; import {getCurrentInstance, ref, watch} from 'vue';
const medicineSummaryFormList = ref([]); const medicineSummaryFormList = ref([]);
const medicineSummaryFormDetails = ref([]); const medicineSummaryFormDetails = ref([]);
@@ -86,6 +86,11 @@ function handleGetPrescription() {
}); });
} }
// 监听医嘱类型筛选条件变化,自动刷新列表
watch(() => props.therapyEnum, () => {
handleGetPrescription();
});
// 获取发药单详情 // 获取发药单详情
function getDetail(row) { function getDetail(row) {
getMedicineSummaryDetail({ summaryNo: row.busNo }).then((res) => { getMedicineSummaryDetail({ summaryNo: row.busNo }).then((res) => {

View File

@@ -1140,6 +1140,30 @@ const {
method_code method_code
} = useDict('surgical_site', 'anesthesia_type', 'incision_level', 'isolation_type', 'surgery_type', 'surgery_level', 'surgery_nature', 'method_code') } = useDict('surgical_site', 'anesthesia_type', 'incision_level', 'isolation_type', 'surgery_type', 'surgery_level', 'surgery_nature', 'method_code')
// Bug #433: 存储待转换的数据,等待字典加载后再设置类型
const pendingAnesData = ref(null)
// 监听麻醉字典加载,完成后立即设置表单值
let anesDataUnwatch = null
function setupAnesDataWatch() {
if (anesDataUnwatch) return // 防止重复设置
anesDataUnwatch = watch(
anesthesiaList,
(newList) => {
if (newList && newList.length > 0 && pendingAnesData.value) {
const data = pendingAnesData.value
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)
pendingAnesData.value = null
if (anesDataUnwatch) { anesDataUnwatch(); anesDataUnwatch = null }
}
},
{ immediate: true }
)
}
// 加载数据 // 加载数据
onMounted(() => { onMounted(() => {
const anesthesiaType = sessionStorage.getItem('anesthesiaType') const anesthesiaType = sessionStorage.getItem('anesthesiaType')
@@ -1325,13 +1349,16 @@ function handleEdit(row) {
if (res.code === 200) { if (res.code === 200) {
const data = res.data const data = res.data
Object.assign(form, data) Object.assign(form, data)
// 使用nextTick确保在Vue响应式更新后再赋值避免el-select无法匹配选项 // Bug #433: 如果字典已加载则立即转换否则存入pending等待字典加载完成
nextTick(() => { if (anesthesiaList.value && anesthesiaList.value.length > 0) {
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod) if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod)
if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel) if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel)
if (data.feeType != null) form.feeType = data.feeType if (data.feeType != null) form.feeType = data.feeType
if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert) if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert)
}) } else {
pendingAnesData.value = data
setupAnesDataWatch()
}
} else { } else {
proxy.$modal.msgError('获取手术安排详情失败') proxy.$modal.msgError('获取手术安排详情失败')
} }
@@ -1351,13 +1378,16 @@ function handleView(row) {
if (res.code === 200) { if (res.code === 200) {
const data = res.data const data = res.data
Object.assign(form, data) Object.assign(form, data)
// 使用nextTick确保在Vue响应式更新后再赋值避免el-select无法匹配选项 // Bug #433: 如果字典已加载则立即转换否则存入pending等待字典加载完成
nextTick(() => { if (anesthesiaList.value && anesthesiaList.value.length > 0) {
if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod) if (data.anesMethod != null) form.anesMethod = Number(data.anesMethod)
if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel) if (data.incisionLevel != null) form.incisionType = Number(data.incisionLevel)
if (data.feeType != null) form.feeType = data.feeType if (data.feeType != null) form.feeType = data.feeType
if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert) if (data.isExternalExpert != null) form.isExternalExpert = Number(data.isExternalExpert)
}) } else {
pendingAnesData.value = data
setupAnesDataWatch()
}
} else { } else {
proxy.$modal.msgError('获取手术安排详情失败') proxy.$modal.msgError('获取手术安排详情失败')
} }

View File

@@ -1,13 +1,10 @@
-- Bug #462: 诊疗目录编辑弹窗中"所需标本"下拉框数据加载失败 -- Bug #462: 诊疗目录编辑弹窗中"所需标本"下拉框数据加载失败
-- 根因: sys_dict_type 表中缺少 specimen_code 字典类型sys_dict_data 中缺少对应数据 -- 根因: hisprd schema 中 sys_dict_type 存在 specimen_code 类型,sys_dict_data 中缺少对应的7条数据记录
-- 修复: 插入字典类型及7条标本数据 -- 修复: 在 hisprd schema 中插入7条标本数据
-- 验证: hisdev/histest1 已有数据,仅 hisprd 缺失
-- 注意: hisprd 的 sys_dict_data 表无 py_str 字段(旧表结构)
-- 插入字典类型 INSERT INTO hisprd.sys_dict_data (dict_sort, dict_label, dict_value, dict_type, status, create_by, create_time, remark)
INSERT INTO sys_dict_type (dict_name, dict_type, status, create_by, create_time, remark)
VALUES ('标本类型', 'specimen_code', '0', 'admin', NOW(), '诊疗项目所需标本类型字典');
-- 插入标本数据
INSERT INTO sys_dict_data (dict_sort, dict_label, dict_value, dict_type, status, create_by, create_time, remark)
VALUES VALUES
(1, '血液', '1', 'specimen_code', '0', 'admin', NOW(), '血液标本'), (1, '血液', '1', 'specimen_code', '0', 'admin', NOW(), '血液标本'),
(2, '尿液', '2', 'specimen_code', '0', 'admin', NOW(), '尿液标本'), (2, '尿液', '2', 'specimen_code', '0', 'admin', NOW(), '尿液标本'),
@@ -15,4 +12,4 @@ VALUES
(4, '呼吸道', '4', 'specimen_code', '0', 'admin', NOW(), '呼吸道标本'), (4, '呼吸道', '4', 'specimen_code', '0', 'admin', NOW(), '呼吸道标本'),
(5, '无菌体液', '5', 'specimen_code', '0', 'admin', NOW(), '无菌体液标本'), (5, '无菌体液', '5', 'specimen_code', '0', 'admin', NOW(), '无菌体液标本'),
(6, '生殖道', '6', 'specimen_code', '0', 'admin', NOW(), '生殖道标本'), (6, '生殖道', '6', 'specimen_code', '0', 'admin', NOW(), '生殖道标本'),
(7, '其他', '7', 'specimen_code', '0', 'admin', NOW(), '其他标本'); (7, '其他', '99', 'specimen_code', '0', 'admin', NOW(), '其他标本');