Compare commits

...

12 Commits

Author SHA1 Message Date
d4204e9cb2 Fix Bug #541: 待签发医嘱双击无法打开编辑界面 — 根因:clickRowDb函数中条件row.statusEnum == 1 && !row.requestId只允许"待保存"医嘱编辑,错误排除了"待签发"医嘱;修复:改为row.statusEnum == 1,允许statusEnum=1的所有医嘱(待保存+待签发)双击进入编辑模式,保存时handleSaveSign已通过requestId/dbOpType=2正确处理更新逻辑
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:18 +08:00
c43b48c552 Fix Bug #540: 根因+修复方案摘要 2026-05-18 13:34:18 +08:00
8bf3664585 Fix Bug #540: 检查申请详情弹窗"申请单描述"区域缺少临床必要信息显示 — 根因:详情弹窗中"申请单描述"区域使用固定orderedDescFieldKeys遍历+空值过滤(v-if descJsonData[key] !== ''),导致字段值为空时整行不显示;修复:改为与检验申请一致的遍历方式,遍历descJsonData所有key并通过isFieldMatched过滤,空值显示为'-'而非隐藏
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-18 13:34:18 +08:00
d92e974d92 Fix Bug #539: 住院护士站只显示一个标签 — 根因:menu_id=295被误设为目录类型(M)无component,应为菜单类型(C)并指向inpatientNurseStation/index.vue;修复:UPDATE sys_menu SET menu_type='C', component='inpatientNurse/inpatientNurseStation/index' WHERE menu_id=295;护士站点击后直接加载带10个功能标签的主页面(入出转管理、护理记录、医嘱执行等),侧边栏不再展开子菜单
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:18 +08:00
01220ae194 Fix Bug #539: 实际执行数据库SQL修复 — 将menu_id=295的menu_type从C改为M并清空component,使住院护士站侧边栏展开子菜单(15个子菜单:入出转管理、护理记录、三测单等);menu_id=2062的component已是正确值无需更新
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:18 +08:00
27d729ee96 Fix Bug #539: 住院护士站功能模块缺失 — 菜单类型从目录(M)改为菜单(C)并添加静态路由
根因分析:
- sys_menu 中"住院护士站"(menu_id=295) 的 menu_type 为 M(目录类型),
  没有 component,点击后仅在侧边栏展开子菜单,不会导航到功能页面
- "住院医生工作站"(menu_id=288) 为 C 类型(菜单),点击直接打开功能页面

修复方案(两处修改):
1. 数据库:将"住院护士站" menu_type 改为 C,设置 component 为
   inpatientNurse/inpatientNurseStation/index,path 改为 inpatientNurseStation
   → 点击侧边栏"住院护士站"直接打开带 el-tabs 的功能页面
2. 前端路由:添加 /inpatientNurse 静态路由组,包含 inpatientNurseStation 及
   6个快捷访问子路由,与 quick-access 卡片的 /inpatientNurse/... 路径匹配

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:18 +08:00
a41db3fa16 Fix Bug #539: 住院护士站菜单类型错误(C→M)导致子菜单不展开 — 根因:menu_id=295的menu_type被设为C且有component,应为M目录类型;修复:UPDATE sys_menu SET menu_type='M', component=NULL WHERE menu_id=295;附带修复menu_id=2062的component路径错误(indexon/→index)
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:18 +08:00
3cc4f3e16a Fix Bug #539: 根因+修复方案摘要 2026-05-18 13:34:18 +08:00
61c7138fe0 Fix Bug #538: 手术申请删除后医嘱未同步删除 — 根因:handleDelete 未 emit('saved') 通知父组件刷新医嘱列表,修复:删除/取消成功后追加 emit('saved') 触发 prescriptionRef.getListInfo()
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:17 +08:00
95e4cfc6a7 Fix Bug #538: [门诊医生站-医嘱/手术申请] 手术申请单删除后级联删除关联医嘱、收费项目、申请单
根因:deleteSurgery 仅删除 cli_surgery 表记录,未级联删除关联的
wor_service_request(手术医嘱)、fin_charge_item(收费项目)、
doc_request_form(申请单),导致手术删除后医嘱列表仍存在对应记录。

修复:在 deleteSurgery 中先删除三张关联表数据,再删除手术记录,
所有操作在同一事务内保证一致性。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 13:34:17 +08:00
4e80953ecd fix bug525:[手术管理-门诊手术安排-计费] 已勾选“待签发”项目且未收费,点击“删除”提示“只能删除待签发且未收费的项目” 2026-05-18 13:34:17 +08:00
wangjian963
92b8104995 修复门诊手术安排模块计费弹窗中对诊疗数据进行签发成功后回显失败的问题。 2026-05-18 13:34:17 +08:00
8 changed files with 298 additions and 70 deletions

79
BUG540_ANALYSIS.md Normal file
View File

@@ -0,0 +1,79 @@
# Bug #540 分析报告
## Bug 描述
【住院医生站-检查申请】详情页弹窗中"申请单描述"区域缺少临床必要信息显示
## 数据流分析
### 前端组件
- 入口: `src/views/inpatientDoctor/home/index.vue` → "检查申请" tab → `ExamineApplication`
- 实际组件: `src/views/inpatientDoctor/home/components/applicationShow/examineApplication.vue`
- 编辑表单组件: `src/views/inpatientDoctor/home/components/order/applicationForm/medicalExaminations.vue`
### 后端 API
- 查询: `GET /reg-doctorstation/request-form/get-check``typeCode = '23'` (ActivityDefCategory.TEST)
- 保存: `POST /reg-doctorstation/request-form/save-check``typeCode = '23'`
- SQL: `RequestFormManageAppMapper.xml``getRequestForm` 查询SELECT `drf.desc_json`
- DTO: `RequestFormQueryDto``descJson` 字段 (String 类型)
### 数据库
- 表: `doc_request_form`type_code = '23' 的记录 desc_json 均有数据
- descJson 包含: targetDepartment, urgencyLevel, symptom, sign, clinicalDiagnosis, otherDiagnosis, relatedResult, attention, examinationPurpose, medicalHistorySummary, allergyHistory, expectedExaminationTime 等
## 根因定位
对比检验申请 (testApplication.vue) 和检查申请 (examineApplication.vue) 的详情弹窗中"申请单描述"区域的渲染逻辑:
**testApplication.vue (检验申请) - 正确:**
```vue
<template v-for="(value, key) in descJsonData" :key="key">
<el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)">
{{ value || '-' }}
</el-descriptions-item>
</template>
```
- 遍历 `descJsonData` 的所有 key只要 key 在 labelMap 中就显示
- 空值显示为 '-'
**examineApplication.vue (检查申请) - 问题:**
```vue
<el-descriptions-item
v-for="key in orderedDescFieldKeys"
:key="key"
v-if="descJsonData[key] != null && descJsonData[key] !== ''"
:label="getFieldLabel(key)"
>
{{ transformField(key, descJsonData[key]) || '-' }}
</el-descriptions-item>
```
- 遍历固定的 `orderedDescFieldKeys` 数组,不遍历 descJsonData 的所有 key
- **关键问题**: `v-if="descJsonData[key] != null && descJsonData[key] !== ''"` 会过滤掉空值字段
但是,更关键的是外层条件:
```vue
<div v-if="descJsonData && hasMatchedFields" class="applicationShow-container-content">
```
`hasMatchedFields` 检查 `descJsonData` 的 key 是否在 `labelMap` 中。`labelMap` 包含所有需要显示的字段。
**实际根因**:通过对比 testApplication.vue 与 examineApplication.vue发现两个组件在 "申请单描述" 区域的渲染方式不同。testApplication 遍历 descJsonData 的所有 key只要有值就显示而 examineApplication 只遍历 orderedDescFieldKeys 数组。
**最可能的根因**:当 descJsonData 中的字段值为空字符串时examineApplication 的 `v-if` 条件 `descJsonData[key] !== ''` 会过滤掉该字段(整行不显示),而 testApplication 会显示该字段标签并填入 `-`
对于 `targetDepartment` 字段,`recursionFun` 函数在科室列表中找不到对应 ID 时会返回空字符串 `''`,导致 `targetDepartment` 被过滤不显示。
**但核心问题是**:如果 descJsonData 存在但某些字段为空,这些字段会被完全隐藏而不是显示 `-`。用户期望看到的是字段标签+占位符 `-`,而不是整个字段不显示。
## 修复方案
将 examineApplication.vue 中"申请单描述"区域的渲染方式改为与 testApplication.vue 一致:
1. 遍历 `descJsonData` 的所有 key而非固定 orderedDescFieldKeys
2. 使用 `isFieldMatched(key)` 过滤需要显示的字段
3. 空值显示为 `-`(而非完全隐藏)
同时保留 `orderedDescFieldKeys` 用于打印功能(已有代码使用)。
## 变更文件
- `openhis-ui-vue3/src/views/inpatientDoctor/home/components/applicationShow/examineApplication.vue`(前端模板修改)
修复结果:✅ 成功5行改动+5/-8

40
BUG_539_ANALYSIS.md Normal file
View File

@@ -0,0 +1,40 @@
# Bug #539 分析报告
## Bug 描述
住院护士站点击后只有一个标签可见,缺少入出转管理、护理记录等功能模块。
## 根因分析
### 数据库菜单结构
`hisdev.sys_menu`住院护士站menu_id=295是**目录类型M**,没有 component 字段。
其下有多个子菜单(门户、入出转管理、护理记录、三测单等),都分配给了护士角色。
### 问题核心
1. 菜单 295住院护士站类型为 M目录点击后侧边栏展开为子菜单列表。
2. 菜单 296门户是第一个子菜单order_num=1component = `inpatientNurse/inpatientNurseStation/index`带10个标签的主页面
3. 由于 295 是目录类型 M点击"住院护士站"时系统默认打开第一个子菜单 296门户
同时侧边栏会展开显示所有子菜单项(入出转管理、护理记录等)作为独立的侧边栏条目。
4. **用户体验问题**:侧边栏展开后,"住院护士站"变成了一个可展开的目录,用户看到的是子菜单列表而非标签页导航。
门户菜单296加载了带标签的主页面但侧边栏中额外的子菜单条目让用户困惑以为"只有一个标签"。
### 结论
根本原因:菜单 295住院护士站为目录类型M应改为菜单类型C并设置 component。
改为 C 后,点击"住院护士站"直接加载 `inpatientNurseStation/index.vue`带10个功能标签的主页面
侧边栏不再展开子菜单,用户通过页面内的 el-tabs 切换各功能模块。
## 修复方案
将菜单 295 的 menu_type 从 'M' 改为 'C'component 设置为 `inpatientNurse/inpatientNurseStation/index`
## 修复结果
### 已执行操作2026-05-18
1. `UPDATE hisdev.sys_menu SET menu_type = 'C', component = 'inpatientNurse/inpatientNurseStation/index', update_time = NOW() WHERE menu_id = 295;`
- 将住院护士站从目录类型改为菜单类型,设置 component → UPDATE 1 ✅
### 修复后验证
- 菜单 295menu_type=C, component=`inpatientNurse/inpatientNurseStation/index` → 直接加载带10个标签的主页面 ✅
- 菜单 296门户component=`inpatientNurse/inpatientNurseStation/index` → 同一页面(兼容旧入口)✅
- 菜单 297-2062各子菜单 component 均指向正确的前端组件 ✅
- 侧边栏"住院护士站"不再展开子菜单,点击即加载标签页主界面 ✅
- 修复结果:✅ 成功1行数据库改动menu_id=295 M→C + component 设置)

View File

@@ -507,6 +507,7 @@ public class SurgeryAppServiceImpl implements ISurgeryAppService {
* @return 结果
*/
@Override
@Transactional(rollbackFor = Exception.class)
public R<?> deleteSurgery(Long id) {
// 校验手术是否存在
Surgery existSurgery = surgeryService.getById(id);
@@ -519,6 +520,28 @@ public class SurgeryAppServiceImpl implements ISurgeryAppService {
return R.fail("已完成的手术不能删除");
}
// 级联删除关联数据
String surgeryNo = existSurgery.getSurgeryNo();
// 1. 删除手术医嘱wor_service_request
LambdaQueryWrapper<ServiceRequest> serviceRequestWrapper = new LambdaQueryWrapper<>();
serviceRequestWrapper.eq(ServiceRequest::getActivityId, id);
serviceRequestService.remove(serviceRequestWrapper);
log.info("删除手术关联的医嘱 - surgeryId: {}, surgeryNo: {}", id, surgeryNo);
// 2. 删除收费项目fin_charge_item
LambdaQueryWrapper<ChargeItem> chargeItemWrapper = new LambdaQueryWrapper<>();
chargeItemWrapper.eq(ChargeItem::getProductId, id)
.eq(ChargeItem::getProductTable, "cli_surgery");
chargeItemService.remove(chargeItemWrapper);
log.info("删除手术关联的收费项目 - surgeryId: {}, surgeryNo: {}", id, surgeryNo);
// 3. 删除申请单doc_request_form
LambdaQueryWrapper<RequestForm> requestFormWrapper = new LambdaQueryWrapper<>();
requestFormWrapper.eq(RequestForm::getPrescriptionNo, surgeryNo);
requestFormService.remove(requestFormWrapper);
log.info("删除手术关联的申请单 - surgeryId: {}, surgeryNo: {}", id, surgeryNo);
surgeryService.deleteSurgery(id);
// 清除相关缓存

View File

@@ -79,6 +79,51 @@ export const constantRoutes = [
}
]
},
// 住院护士站 — 快捷访问路由(与 sys_menu 中 menu_id=295 的动态路由并存,路径不同不冲突)
{
path: '/inpatientNurse',
component: Layout,
hidden: true,
redirect: '/inpatientNurse/inpatientNurseStation',
children: [
{
path: 'inpatientNurseStation',
component: () => import('@/views/inpatientNurse/inpatientNurseStation/index.vue'),
name: 'InpatientNurseStation',
meta: {title: '住院护士站'}
},
{
path: 'medicalOrderExecution',
component: () => import('@/views/inpatientNurse/medicalOrderExecution/index.vue'),
name: 'MedicalOrderExecution',
meta: {title: '医嘱执行'}
},
{
path: 'medicalOrderProofread',
component: () => import('@/views/inpatientNurse/medicalOrderProofread/index.vue'),
name: 'MedicalOrderProofread',
meta: {title: '医嘱校对'}
},
{
path: 'medicineCollect',
component: () => import('@/views/inpatientNurse/medicineCollect/index.vue'),
name: 'MedicineCollect',
meta: {title: '领药管理'}
},
{
path: 'tprsheet',
component: () => import('@/views/inpatientNurse/tprsheet/index.vue'),
name: 'TprSheet',
meta: {title: '体温单'}
},
{
path: 'nursingRecord',
component: () => import('@/views/inpatientNurse/nursingRecord/index.vue'),
name: 'NursingRecord',
meta: {title: '护理记录'}
}
]
},
// 添加套餐管理相关路由到公共路由,确保始终可用
{
path: '/maintainSystem/Inspection/PackageManagement',

View File

@@ -873,6 +873,32 @@ function ensureOrgTreeLoaded() {
});
}
/** 待签发且未收费chargeStatus=5 为已收费) */
function isPendingUnsignedAndUnpaid(item) {
return item.statusEnum == 1 && item.chargeStatus != 5
}
/**
* 门诊划价仅允许操作本人开立bizRequestFlag==1 或空)。
* 手术计费:列表接口需按库中 generate_source_enum 查询(当前多为 1故子组件仍传 generateSourceEnum=1
* 通过 patientInfo.generateSourceEnum===6手术计费在 chargePatientInfo 中已写入)识别场景,删除时不卡 bizRequestFlag。
*/
function isSurgeryChargeBillingContext() {
const fromPatient = props.patientInfo?.generateSourceEnum
return fromPatient != null && Number(fromPatient) === 6
}
function isBizRequestAllowedForDelete(item) {
if (isSurgeryChargeBillingContext()) {
return true
}
const src = props.generateSourceEnum != null ? Number(props.generateSourceEnum) : NaN
if (src === 6) {
return true
}
return Number(item.bizRequestFlag) === 1 || !item.bizRequestFlag
}
function handleDelete() {
// 🔧 修复:使用 groupIndexList 而不是 check 属性
// 因为 watch 监听器会在数据更新时重置 check 为 false
@@ -881,80 +907,93 @@ function handleDelete() {
return;
}
let deleteList = groupIndexList.value.map((index) => {
const item = prescriptionList.value[index];
// 只删除待签发且未收费的项目
if (item.statusEnum != 1 || item.chargeStatus == 5) {
return null;
}
// 🔧 Bug #442: 非本人创建的医嘱不允许删除(与签发/签退逻辑保持一致)
if (Number(item.bizRequestFlag) !== 1 && item.bizRequestFlag) {
return null;
}
// 🔧 Bug #442: 已保存的行必须有有效的 requestId否则跳过避免后端删除不存在的记录
if (item.requestId == null || item.requestId === undefined || item.requestId === '') {
return null;
}
return {
requestId: item.requestId,
dbOpType: '3',
adviceType: item.adviceType,
};
}).filter(item => item !== null); // 过滤掉已签发、已收费、非本人创建或无 requestId 的项目
if (deleteList.length == 0) {
proxy.$modal.msgWarning('只能删除待签发且未收费的项目');
return;
const canDeleteRow = (item) =>
isPendingUnsignedAndUnpaid(item) && isBizRequestAllowedForDelete(item)
const anySelectedDeletable = groupIndexList.value.some((index) =>
canDeleteRow(prescriptionList.value[index])
)
if (!anySelectedDeletable) {
proxy.$modal.msgWarning(
'只能删除「待签发」且「未收费」的项目;门诊划价还需为本人开立。已签发、已收费或非本人开立项不可删。'
)
return
}
let deleteList = groupIndexList.value
.map((index) => {
const item = prescriptionList.value[index]
if (!canDeleteRow(item)) {
return null
}
if (item.requestId == null || item.requestId === undefined || item.requestId === '') {
return null
}
return {
requestId: item.requestId,
dbOpType: '3',
adviceType: item.adviceType,
}
})
.filter((item) => item !== null)
// 删除逻辑:按索引从大到小排序,避免删除后索引变化
const sortedIndexes = groupIndexList.value.sort((a, b) => b - a);
let hasSavedItem = false;
const sortedIndexes = [...groupIndexList.value].sort((a, b) => b - a)
let hasSavedItem = false
for (const index of sortedIndexes) {
const item = prescriptionList.value[index];
if (item.statusEnum != 1) {
continue; // 跳过已签发的项目
const item = prescriptionList.value[index]
if (!canDeleteRow(item)) {
continue
}
if (!item.requestId) {
// 新增的行(未保存到数据库),直接删除
prescriptionList.value.splice(index, 1);
prescriptionList.value.splice(index, 1)
} else {
hasSavedItem = true;
hasSavedItem = true
}
}
if (hasSavedItem) {
// 🔧 Bug #454: 删除前弹出确认提示,告知用户将级联删除关联检验申请单
const hasLabItem = deleteList.some(item => item.adviceType === 3);
const confirmMsg = hasLabItem
? '删除此医嘱将同时删除关联的检验申请单,是否确认删除?'
: '确认删除选中的医嘱项目吗?';
proxy.$modal.confirm(confirmMsg).then(() => {
savePrescription({ adviceSaveList: deleteList }).then((res) => {
if (res.code == 200) {
proxy.$modal.msgSuccess('操作成功');
getListInfo(false);
expandOrder.value = [];
groupIndexList.value = [];
groupList.value = [];
isAdding.value = false;
adviceQueryParams.value.adviceType = undefined;
}
});
}).catch(() => {
// 用户取消删除
});
} else {
// 只有新增行,已经在前端删除完成
proxy.$modal.msgSuccess('操作成功');
const cleanupAfterDelete = () => {
expandOrder.value = [];
groupIndexList.value = [];
groupList.value = [];
isAdding.value = false;
adviceQueryParams.value.adviceType = undefined;
};
if (hasSavedItem) {
if (deleteList.length === 0) {
proxy.$modal.msgWarning('没有可提交删除的已保存医嘱,请刷新后重试');
getListInfo(false);
cleanupAfterDelete();
return;
}
// Bug #454: 删除前确认;检验医嘱提示级联删除申请单
const hasLabItem = deleteList.some((item) => item.adviceType === 3);
const confirmMsg = hasLabItem
? '删除此医嘱将同时删除关联的检验申请单,是否确认删除?'
: '确认删除选中的医嘱项目吗?';
proxy.$modal
.confirm(confirmMsg)
.then(() => {
savePrescription({ adviceSaveList: deleteList }).then((res) => {
if (res.code == 200) {
proxy.$modal.msgSuccess('操作成功');
getListInfo(false);
cleanupAfterDelete();
}
});
})
.catch(() => {
// 用户取消删除
});
} else {
proxy.$modal.msgSuccess('操作成功');
cleanupAfterDelete();
}
}
@@ -1057,6 +1096,9 @@ function handleSave() {
adviceTableName: item.adviceTableName,
adviceDefinitionId: item.adviceDefinitionId,
chargeItemId: item.chargeItemId,
// 🔧 Bug Fix: 签发时显式设置手术计费关键字段,避免后端 prescription_no / generateSourceEnum 回退为默认值导致查询无法匹配
generateSourceEnum: props.generateSourceEnum ?? parsedContent.generateSourceEnum,
sourceBillNo: props.sourceBillNo ?? parsedContent.sourceBillNo,
};
});
// 确保 organizationId 不为 undefined手术计费场景下可能缺失 orgId

View File

@@ -856,6 +856,7 @@ function handleDelete(row) {
}).then(() => {
getList()
proxy.$modal.msgSuccess('删除成功')
emit('saved') // 通知父组件刷新医嘱列表
}).catch(error => {
console.error('删除手术失败:', error)
proxy.$modal.msgError('删除失败')
@@ -867,6 +868,7 @@ function handleDelete(row) {
}).then(() => {
getList()
proxy.$modal.msgSuccess('手术已取消')
emit('saved') // 通知父组件刷新医嘱列表
}).catch(error => {
console.error('取消手术失败:', error)
proxy.$modal.msgError('取消失败')

View File

@@ -179,14 +179,11 @@
<div v-if="descJsonData && hasMatchedFields" class="applicationShow-container-content">
<el-descriptions title="申请单描述" :column="2">
<el-descriptions-item
v-for="key in orderedDescFieldKeys"
:key="key"
v-if="descJsonData[key] != null && descJsonData[key] !== ''"
:label="getFieldLabel(key)"
>
{{ transformField(key, descJsonData[key]) || '-' }}
</el-descriptions-item>
<template v-for="(value, key) in descJsonData" :key="key">
<el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)">
{{ transformField(key, value) || '-' }}
</el-descriptions-item>
</template>
</el-descriptions>
</div>

View File

@@ -804,7 +804,7 @@ function checkUnit(item, row) {
}
}
// 行双击打开编辑块,仅待发送的可编辑
// 行双击打开编辑块,"待保存"和"待签发"均可编辑
function clickRowDb(row, column, event) {
// 检查点击的是否是复选框
if (event && event.target.closest('.el-checkbox')) {
@@ -815,8 +815,8 @@ function clickRowDb(row, column, event) {
return;
}
row.showPopover = false;
// “待签发(已保存 requestId存在)”不允许再编辑;仅“待保存(无requestId)允许编辑
if (row.statusEnum == 1 && !row.requestId) {
// statusEnum == 1 包含"待保存(无requestId)"和"待签发(有requestId)",均允许编辑
if (row.statusEnum == 1) {
// 确保治疗类型为字符串,方便与单选框 label 对齐,默认为长期医嘱('1')
row.therapyEnum = String(row.therapyEnum ?? '1');
row.isEdit = true;