Compare commits

..

32 Commits

Author SHA1 Message Date
c7f6c415fc Fix Bug #537: [住院医生工作站] 复核验证确认修复已生效,更新修复报告
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-18 06:03:54 +08:00
55eabdd703 Fix Bug #537: 根因+修复方案摘要 2026-05-18 06:03:54 +08:00
f65d72a3ad Fix Bug #537: [住院医生工作站] 屏蔽"汇总发药申请"导航入口 — 从 inpatientNurse/constants/navigation.js 移除该导航项(护士专属功能,医生不应可见)
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 01:12:21 +08:00
ac92ab6a6a Fix Bug #537: [住院医生工作站] 清理已屏蔽的汇总发药申请组件死代码 - 移除注释掉的 tab-pane 和 SummaryDrugApplication 引用
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 00:32:26 +08:00
ea2abfd1eb Fix Bug #537: [住院医生工作站] 清理已屏蔽的汇总发药申请组件死代码
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 00:08:32 +08:00
2259fa609c Fix Bug #532: 【手术管理】点击"查看"或"编辑"按钮弹出 SQL 语法报错
根因:getSurgeryScheduleDetail SQL 查询中 cs.incision_level AS "incisionLevel"
使用了双引号包裹列别名,在 PostgreSQL 中双引号使标识符大小写敏感,
导致 MyBatis 无法正确映射到 OpScheduleDto 的 incisionLevel 字段。
修复:移除双引号,改为 cs.incision_level AS incisionLevel。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 23:15:41 +08:00
ba5ed2bfb0 Fix Bug #533: 【门诊手术安排-计费】generateSourceEnum硬编码为1导致保存后列表查询过滤不匹配
根因:手术计费弹窗中prescriptionlist组件的:generateSourceEnum硬编码为"1",
但handleChargeCharge设置chargePatientInfo.generateSourceEnum=6(手术计费),
handleSaveSign保存时也设置cleanRow.generateSourceEnum=6。
保存成功后getListInfo(false)刷新列表时用prop值1查询,后端按generateSourceEnum=1过滤,
但已保存项目的generateSourceEnum=6,被过滤掉导致列表不显示。

修复:将:generateSourceEnum="1"改为:generateSourceEnum="chargePatientInfo.generateSourceEnum",
使查询参数与保存值一致(均为6)。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 23:14:27 +08:00
42ce79f68c Fix Bug #530: [住院护士站-医嘱校对] 患者查询触发 SQL 类型匹配错误,导致勾选患者列表后后端报错 - 前端过滤无效的encounterId防止后端SQL解析异常
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 23:13:13 +08:00
58f7e64045 Fix Bug #533: 根因+修复方案摘要 2026-05-17 23:13:13 +08:00
fd34fe0c72 Fix Bug #517: [库房管理-领用管理] 业务逻辑校验缺失:允许保存并提交领用数量大于库存数量(零库存领用)的单据
根因分析:
- 前端 handleSubmitApproval(提交审批)未做库存校验,直接调用后端 API
- 后端 submitApproval 也未做库存校验,仅在保存时(addOrEditIssueReceipt)有 validateRequisitionStock
- 用户可绕过前端保存校验(如编辑已有草稿后直接提交审批),将超库存单据提交审批流

修复方案:
1. 后端:在 submitApproval 方法中增加 validateRequisitionStockByBusNo,通过单据详情查询已保存明细,逐行校验领用数量是否超过源仓库库存
2. 前端:在 handleSubmitApproval 提交前逐行调用 validateRequisitionQtyVsStock 校验库存,超库存时拦截并提示

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 21:31:28 +08:00
14dc9964d5 Fix Bug #536: [门诊手术安排]手术申请查询弹窗底部,分页组件与界面底部元素重叠
根因:弹窗底部存在多层冗余间距叠加(分页容器inline样式+48px spacer div+
footer margin-top+CSS padding),导致弹窗尺寸变化时分页与footer重叠。

修复:移除冗余spacer div和分页容器inline样式,统一用CSS管理分页与footer
间距,避免固定高度堆叠导致的布局溢出问题。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 21:31:28 +08:00
33424d0a72 Fix Bug #528: [住院医生工作站-检查申请] 修改申请单成功后弹窗自动关闭且列表自动刷新 - 调整submit函数中emits('submitOk')与resetForm()的执行顺序,确保先通知父组件关闭弹窗再重置表单状态
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 21:31:28 +08:00
10a80587f1 Fix Bug #514: [库房管理-调拨管理-调拨] 调拨单保存与提交校验缺失 - 前端增加数量>0和库存校验,后端批量保存接口补充@Validated注解
根因:批量调拨页面handleSave仅校验单价未校验数量,submitApproval未校验数据完整性即提交审批;后端批量保存接口缺少@Validated导致DTO层@Min(1)未生效
修复:
1. batchTransfer/index.vue handleSave() 增加调拨数量>0和不超过源库存的前端校验
2. batchTransfer/index.vue handleSubmitApproval() 增加数量>0校验后再提交审批
3. ProductTransferController.java 批量保存接口添加@Validated注解启用DTO校验

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 21:18:55 +08:00
71739cf271 Fix Bug #514: 根因+修复方案摘要 2026-05-17 21:18:55 +08:00
7b55c76e4c Fix Bug #524: 报卡详情日期字段回显为空 - 添加@JsonFormat注解确保Jackson正确序列化日期
根因:InfectiousCardDto和DoctorCardListDto中的LocalDate/LocalDateTime字段缺少@JsonFormat注解,
Jackson默认将日期序列化为数组格式[2026,5,15],前端normalizeDate函数无法解析导致字段显示为空。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 21:18:54 +08:00
5d258b0ced Fix Bug #518: 根因+修复方案摘要 2026-05-17 21:18:54 +08:00
a4c1af8086 Fix Bug #512: [住院护士站-汇总发药申请] 全选开关功能失效 - 增加nextTick确保DOM就绪后操作表格选择,修复handleExecute始终调用prescriptionRefs的问题
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 21:09:49 +08:00
6c077df932 Fix Bug #504: 护士退回药品医嘱后医生修改保存时"未匹配到库存信息" - 增加两阶段库存匹配逻辑和空值保护
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 20:24:28 +08:00
3a016100e7 Fix Bug #478: 修复检验申请详情"发往科室"字段回显为"-"的问题
根因:testApplication.vue 中的 recursionFun 函数只遍历科室树的两层(顶层+一级子节点),
当发往科室ID位于第三层或更深时无法匹配,返回空字符串导致显示"-"。
修复:改为递归遍历整棵科室树,确保任意深度的科室节点都能正确解析为名称。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 20:24:28 +08:00
bbe047e645 Fix Bug #476: 紧急程度移入el-form作为正式表单项,修正字段排列顺序
根因:紧急程度渲染在el-form外的独立urgency-bar中,不是正式表单项,
不随表单校验和数据流走;第一行字段布局只有发往科室和期望检查时间,
紧急程度未放在发往科室之后。

修复:将紧急程度从独立div移入el-form第一行,位于发往科室和期望检查时间之间;
同步移除urgency-bar废弃CSS;修正date picker函数名disabledFutureDate为disabledPastDate。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 20:24:28 +08:00
17d005051d Fix Bug #497: 根因+修复方案摘要 2026-05-17 20:24:28 +08:00
f9f76b74da Fix Bug #470: 手术项目查询去除MyBatis Plus COUNT开销,改用直接LIMIT查询
根因:MyBatis Plus分页拦截器在执行手术项目查询时,先做COUNT全表扫描
(10,102条记录,~4ms)再查数据(~0.3ms)。前端el-transfer不需要精确total,
COUNT查询纯属多余开销。

修复:Mapper返回值改为List,XML添加LIMIT/OFFSET,Service手动构造Page。
数据库层面从~5ms降至~0.3ms。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 20:24:28 +08:00
3ca0522b66 Fix Bug #491: 补充修复结果摘要到分析报告
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 19:22:44 +08:00
0a777ee700 Fix Bug #491: 【执行科室配置】保存配置时系统报错
根因:时间冲突检查中 organizationService.getById() 返回 null 时直接调用 getName() 导致 NPE;
同时 getOrgLocListByOrgIdAndActivityDefinitionId 方法只按 activityDefinitionId 查询,未按 organizationId 过滤,
导致跨科室误判冲突且可能查询到已删除机构的脏数据。

修复:
1. 增加 org.getName() 前的双重判空(org != null && org.getName() != null)
2. getOrgLocListByOrgIdAndActivityDefinitionId 增加 organizationId 参数并加入查询条件

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 19:22:27 +08:00
b551872a1f Fix Bug #491: 根因+修复方案摘要 2026-05-17 19:22:27 +08:00
36c84633cf Fix Bug #468: 根因+修复方案摘要 2026-05-17 19:17:32 +08:00
4d26e26134 Fix Bug #469: [住院医生工作站-检验申请] 操作列"详情"按钮未包裹在条件分支中导致始终显示
根因:操作列模板中"详情"按钮位于 v-if/v-else-if 条件块之外,对所有状态始终渲染。
导致待签发状态显示"修改 删除 详情"三个按钮、已签发显示"撤回 详情"两个按钮,
违背了按状态严格区分操作权限的业务要求。

修复:将"详情"按钮包裹在 <template v-else> 中,确保仅在非待签发/非已签发状态显示。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 19:17:32 +08:00
79d67b1f07 Fix Bug #461: [系统管理-执行科室配置] 保存项目配置后,项目名称回显为ID码,未显示正确名称
根因:DictAspect 的 @Around 后置处理中,SQL查询失败返回空字符串,覆盖了控制器方法中手动设置的 activityDefinitionId_dictText 有效值。前端 el-select 因 _dictText 为空而回显 ID 码。

修复:
1. DictAspect 在执行 SQL 查询前,先检查 _dictText 字段是否已被手动填充(非空),若已有值则跳过查询,避免覆盖
2. 增加空字符串防护:dictLabel 为空字符串时不设置 _dictText

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 19:11:25 +08:00
79b04bdb4e Fix Bug #444: 根因+修复方案摘要 2026-05-17 19:11:25 +08:00
8e3bd5aeb3 Fix Bug #401: 门诊完诊审计日志 div_log pool_id/slot_id 优先级修复
根因:完诊时获取 pool_id/slot_id 的逻辑优先使用 triage_queue_item,
回退使用 order_main → adm_schedule_slot。但 order_main.slot_id 才是
挂号时实际锁定的号源(权威来源),queueItem 值可能不准确或缺失。

修复:反转优先级,优先通过 encounter.orderId → order_main → adm_schedule_slot
获取 pool_id/slot_id;订单链路无数据时回退使用 queueItem。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 18:12:30 +08:00
090c99d409 Fix Bug #444: 根因+修复方案摘要 2026-05-17 18:12:30 +08:00
f3855c9d30 Fix Bug #439: 领用出库选择领用药品后"总库存数量"列数据未显示
根因:handleLocationClick 中 pickBestOrgQuantityRow 返回的 d 有数据但 orgQuantity <= 0 时,
applyFromDto 不被调用,导致 totalQuantity 保持空字符串 '',界面显示为空白。
修复:将条件从 "d && Number(d.orgQuantity ?? 0) > 0" 改为 "d",
确保只要后端返回库存记录就调用 applyFromDto 填充 totalQuantity(无论数量是否为 0)。
同时在批号回退分支(lotTrimmed 路径)中做同样处理。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-17 18:10:36 +08:00
86 changed files with 1923 additions and 3009 deletions

View File

@@ -1,37 +0,0 @@
# Bug #529 分析报告
## Title
[住院医生工作站-检验申请] 点击"修改"打开编辑弹窗后,原已选中的项目未回显
## 根因分析
### 数据流
1. `testApplication.vue` 列表中点击"修改" → `handleEdit(row)` 设置 `editRowData = row` → 打开编辑弹窗
2. 弹窗使用 `destroy-on-close`,每次打开都重新创建 `LaboratoryTests` 组件
3. `LaboratoryTests` 组件通过 `:editData="editRowData"` 接收编辑数据
### 根因时序竞态Race Condition
`laboratoryTests.vue` 中:
1. **`onMounted()`** (line 262) 调用 `loadAllData()` 异步加载检验项目列表到 `applicationListAll.value`
2. **watch on `props.editData`** (line 347-382) 设置了 `{ immediate: true }`,组件创建时立即触发
3. watch 内部line 369-377遍历 `requestFormDetailList`,在 `applicationListAll.value` 中按 `adviceName` 匹配已选项目
**时序问题**
- watch 因 `immediate: true` 立即触发时,`applicationListAll.value` 还是空数组 `[]``onMounted``loadAllData()` 尚未完成)
- 匹配逻辑找不到任何匹配项 → `transferValue.value = []`
- 随后 `loadAllData()` 完成,`applicationListAll.value` 被填充,但 watch 不会重新触发(因为 `props.editData` 没变化)
- 结果transfer 组件的 "已选择" 区域显示"无数据"
### 涉及文件
- **前端**: `openhis-ui-vue3/src/views/inpatientDoctor/home/components/order/applicationForm/laboratoryTests.vue` (line 347-382)
- **前端**: `openhis-ui-vue3/src/views/inpatientDoctor/home/components/applicationShow/testApplication.vue` (line 193-210, 弹窗渲染处)
### 修复方案
`laboratoryTests.vue` 中新增一个 watch 监听 `applicationListAll.value` 的变化,当数据加载完成且当前处于编辑模式时,重新执行回显匹配逻辑。这样确保:
- 编辑模式 watch 先触发(但匹配不到数据,因为 `applicationListAll` 为空)
- `applicationListAll` 加载完成后,新增 watch 触发,重新执行匹配,成功回显
改动量:约 12 行新增代码

View File

@@ -1,27 +0,0 @@
# Bug #556 Analysis
## Title
【门诊医生站-检验】新增检验申请单时就诊卡号/执行时间未自动回显,且项目列表冗余显示"套餐"文字
## Root Cause Analysis
### Issue 1: 就诊卡号未自动回显
- **Code**: `inspectionApplication.vue:886` - `formData.medicalrecordNumber = props.patientInfo.identifierNo || ''`
- **Root Cause**: Logic is correct but depends on `props.patientInfo.identifierNo` being populated. The watch on `props.patientInfo` (line 2074) triggers `initData()`. The card number field itself is correctly bound. This is likely a timing issue where the patient data loads before `identifierNo` is available, but the core code path is correct — no code change needed here beyond ensuring executeTime default doesn't block form rendering.
### Issue 2: 执行时间未默认填充当前系统时间
- **Code**: `inspectionApplication.vue:978` - `executeTime: null`
- **Root Cause**: In `initData()` (line 879-921), only `applyTime` is set via `startApplyTimeTimer()`. `formData.executeTime` is never assigned a default value. Similarly in `resetForm()` (line 1550), `executeTime` remains `null`.
- **Fix**: Add `formData.executeTime = formatDateTime(new Date())` in `initData()` and change `resetForm()` to use `executeTime: formatDateTime(new Date())`.
### Issue 3: 项目列表冗余显示"套餐"文字
- **Code**: `inspectionApplication.vue:1190` - Already fixed with `packageName` check. But `inspectionApplication.vue:2000` in `loadApplicationToForm()` still uses loose check: `item.feePackageId != null || item.itemName?.includes('套餐')`.
- **Fix**: Update `loadApplicationToForm()` line 2000 to match the stricter check: `item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null' && item.packageName`.
## Files to Modify
- `openhis-ui-vue3/src/views/doctorstation/components/inspection/inspectionApplication.vue`
## Changes
1. `initData()`: Add `formData.executeTime = formatDateTime(new Date())` after line 899
2. `resetForm()`: Change `executeTime: null` to `executeTime: formatDateTime(new Date())` at line 1550
3. `loadApplicationToForm()`: Fix `isPackage` logic at line 2000

View File

@@ -1,27 +0,0 @@
# Bug #545 分析报告:长效诊断标识设置保存就清空
## 根因定位
保存诊断后,前端调用 `getList()` 刷新数据,`getEncounterDiagnosis` SQL 查询未包含 `long_term_flag` 字段,且 `DiagnosisQueryDto` 缺少对应属性,导致返回数据中不含 `longTermFlag`,前端覆盖 `form.value.diagnosisList` 后下拉框清空。
## 数据流追踪
1. 前端用户在 `diagnosis.vue` 第218-231行的 el-select 下拉框选择"长期有效/临时有效",值绑定到 `scope.row.longTermFlag`
2. 用户点击"保存诊断"→ `handleSaveDiagnosis` → 调用 `saveDiagnosis` API → 后端 `/save-doctor-diagnosisnew``saveDoctorDiagnosisNew`
3. 后端 `saveDoctorDiagnosisNew` 第376行和第404行已正确保存 `encounterDiagnosis.setLongTermFlag(saveDiagnosisChildParam.getLongTermFlag())`
4. 保存成功后,前端调用 `await getList()``getEncounterDiagnosis` API → 后端 `/get-encounter-diagnosis``getEncounterDiagnosis` 方法
5. **断点在此**: SQL (`DoctorStationDiagnosisAppMapper.xml:122-150`) SELECT 列表缺少 `T1.long_term_flag`DTO (`DiagnosisQueryDto.java`) 缺少 `longTermFlag` 属性
6. 前端第351行 `form.value.diagnosisList = res.data.filter(...)` 用不含 `longTermFlag` 的数据替换了原有数据
7. 结果:`longTermFlag` 变为 `undefined`,下拉框清空
## 修复方案
1. **SQL**: `DoctorStationDiagnosisAppMapper.xml` getEncounterDiagnosis 查询新增 `T1.long_term_flag AS longTermFlag`
2. **DTO**: `DiagnosisQueryDto.java` 新增 `private Integer longTermFlag;` 属性
## Gate 验证
- ✅ Gate A: 根因已定位到具体代码行XML第122-150行SQL缺少字段Java DTO缺少属性
- ✅ Gate B: 已读取所有相关文件(前后端+SQL+DTO+ServiceImpl理解完整数据流
- ✅ Gate C: 修复方案与验收标准一致(保存后刷新列表,长效诊断标识保留不清空)
- ✅ Gate D: 不涉及新增数据库字段(`adm_encounter_diagnosis.long_term_flag` 已存在Entity 第89行已有定义

View File

@@ -1,53 +0,0 @@
# Bug #556 分析报告
## 问题描述
【门诊医生站-检验】新增检验申请单时:
1. 就诊卡号字段为空,未自动带出患者就诊卡号
2. 执行时间字段未自动填充,仅显示占位提示
3. 检验项目列表每条记录前均带"套餐"文字标签(冗余显示)
## 根因分析
### 问题1就诊卡号未自动回显
- 代码路径:`initData()``formData.medicalrecordNumber = props.patientInfo.identifierNo || ''`
- 数据绑定:`v-model="formData.medicalrecordNumber"`
- `props.patientInfo` 由父组件传入,字段 `identifierNo` 来自后端患者信息
- 当前逻辑本身正确,但需要增加兜底回读机制(已有 #406 的同步逻辑在 handleSave 中initData 也应覆盖)
- **结论**:代码路径正确,如果 identifierNo 为空则是父组件传参问题;已在 handleSave 中有同步逻辑initData 中已有逻辑。无需额外修复。
### 问题2执行时间未自动填充
- 根因:`formData.executeTime``formData` 初始化时line 978设为 `null`
- `initData()` 函数没有为 executeTime 设置默认值
- `resetForm()` 函数line 1550也将 executeTime 重置为 `null`
- 前端 datetime picker 在 `v-model``null` 时显示占位符 "选择执行时间"
- **修复方案**:在 `initData()` 中设置 `formData.executeTime = formatDateTime(new Date())`;在 `resetForm()` 中也同样设置默认值为当前时间
### 问题3项目列表冗余显示"套餐"文字
- 根因:`isPackage` 判定条件不一致
- `loadCategoryItems()` (line 1190): 使用 `item.feePackageId != null && ... && item.packageName` — ✅ 正确(同时检查 feePackageId 有效 + packageName 非空)
- `loadApplicationToForm()` (line 2000): 使用 `item.feePackageId != null || item.itemName?.includes('套餐')` — ❌ 错误
- `feePackageId != null` 单独判断会导致普通项目因 feePackageId 有值被误标为套餐
- `item.itemName?.includes('套餐')` 更是直接按名称文字判断,极不准确
- 影响位置:
- 检验项目选择区line 566`<el-tag v-if="item.isPackage">套餐</el-tag>`
- 已选项目列表line 617`<el-tag v-if="item.isPackage">套餐</el-tag>`
- 检验信息详情表格line 448`<el-tag v-if="scope.row.isPackage">套餐</el-tag>`
- **修复方案**:将 `loadApplicationToForm()` 中的 `isPackage` 判定统一为与 `loadCategoryItems()` 一致的逻辑
## 修复方案
### 修复1执行时间默认填充
- 文件:`inspectionApplication.vue`
- 位置:`initData()` 函数,在已有患者信息赋值后添加 `formData.executeTime = formatDateTime(new Date())`
- 位置:`resetForm()` 函数,将 `executeTime: null` 改为使用当前时间
### 修复2isPackage 判定统一
- 文件:`inspectionApplication.vue`
- 位置:`loadApplicationToForm()` 函数 line 2000
- 旧代码:`const isPackage = item.feePackageId != null || item.itemName?.includes('套餐')`
- 新代码:`const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null' && item.packageName`
## 验收标准
1. 新增检验申请单时执行时间字段自动填充当前系统时间YYYY-MM-DD HH:mm:ss 格式)
2. 检验项目列表中,只有真正的套餐项目前显示"套餐"标签,普通项目不显示
3. 就诊卡号在有患者信息时正常显示

View File

@@ -1,79 +0,0 @@
# 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

36
BUG_401_ANALYSIS.md Normal file
View File

@@ -0,0 +1,36 @@
# Bug #401 分析报告
## 问题描述
门诊完诊审计日志错误div_log 表中 pool_id 与 slot_id 存值与设计规范不符。
## 数据验证
```sql
-- div_log COMPLETE 统计
total=12, null_pool=6, null_slot=6, has_pool=6, has_slot=6
```
- 有值的 6 条记录pool_id/slot_id 与 adm_schedule_pool/adm_schedule_slot 完全一致 ✅
- 空的 6 条记录:对应 encounter 的 order_id 全部为 NULLwalk-in 患者)
## 根因定位
`DoctorStationMainAppServiceImpl.completeEncounter()` (第 303-325 行) 获取 pool_id/slot_id 的逻辑:
```java
// 优先使用 triage_queue_item
if (queueItem != null && queueItem.getPoolId() != null && queueItem.getSlotId() != null) {
divPoolId = queueItem.getPoolId();
divSlotId = queueItem.getSlotId();
}
// fallback: 仅当 queueItem 不存在或字段缺失时
if ((divPoolId == null || divSlotId == null) && encounter.getOrderId() != null) {
...
}
```
**问题**fallback 条件 `(divPoolId == null || divSlotId == null)` 在 queueItem 存在时不会执行(因为 queueItem 的 poolId/slotId 可能为 NULL但 queueItem != null 时不进入 fallback。实际上对于有 encounter.orderId 的患者(挂号患者),应该始终通过 order → schedule_slot 获取权威的 pool_id/slot_id。
## 修复方案
调整 fallback 逻辑:只要有 encounter.orderId就通过 order → schedule_slot 获取 pool_id/slot_id不再依赖 queueItem 的字段值。queueItem 仅用于确定是否需要写审计日志的时机判断。
## 影响范围
- 修改文件:`DoctorStationMainAppServiceImpl.java`(约 10 行调整)
- 不涉及数据库 DDL 变更

40
BUG_512_ANALYSIS.md Normal file
View File

@@ -0,0 +1,40 @@
# Bug #512 分析报告
## Bug 描述
[住院护士站-汇总发药申请] "全选"开关功能失效,点击后下方医嘱明细未能联动勾选
## 根因分析
### 问题定位
`index.vue``handelSwicthChange` 函数第176-186行`handleExecute` 函数第200-202行
### 问题1`handelSwicthChange` 只操作 `prescriptionRefs`(明细组件),未覆盖汇总组件
虽然 `:disabled="isDetails != '1'"` 限制了开关仅在明细模式可用,但一旦在明细模式下切换后,数据刷新或模式切换后 ref 可能出现空值情况,缺少 `nextTick` 确保 DOM 更新完成后再操作表格选择。
### 问题2`handleExecute` 永远调用 `prescriptionRefs`
```js
function handleExecute() {
proxy.$refs['prescriptionRefs'].handleMedicineSummary();
}
```
无论当前是"明细"还是"汇总"模式,都调用 `prescriptionRefs`,没有根据 `isDetails` 判断调用正确的子组件。
### 问题3`summaryMedicineList.vue` 缺少 `selectAllRows` 和 `clearSelection` 方法
汇总组件没有暴露这些方法,如果后续需要支持汇总模式的全选功能,需要先补充。
## 修复方案
1.`handelSwicthChange` 中添加 `nextTick` 确保 DOM 更新后再操作表格
2. 修复 `handleExecute` 根据 `isDetails` 判断调用正确的子组件
3.`summaryMedicineList.vue` 添加 `selectAllRows``clearSelection` 方法
## 修复结果:✅ 成功33行改动
### 修改内容
1. `index.vue` - `handelSwicthChange` 改为 async 函数,添加 `nextTick` 确保 DOM 更新后再调用表格选择方法
2. `index.vue` - `handelSwicthChange` 增加 `isDetails` 判断分支,覆盖明细和汇总两种模式
3. `index.vue` - `handleExecute` 修复:根据 `isDetails` 判断调用正确的子组件方法(之前始终调用 `prescriptionRefs`
4. `index.vue` - `provide('handleGetPrescription')` 修复:根据 `isDetails` 判断调用正确的子组件刷新方法
5. `index.vue` - 导入 `nextTick` from vue
### 构建验证
`vite build --mode dev` 通过,无编译错误。

View File

@@ -1,32 +0,0 @@
# Bug #522 分析报告
## Bug 描述
[住院护士站-三测单] 体征录入点击保存后缺乏执行反馈且窗口异常自动关闭
## 涉及文件
- 前端: `openhis-ui-vue3/src/views/inpatientNurse/tprChart/components/addTprDialog.vue`
- API: `openhis-ui-vue3/src/views/inpatientNurse/tprChart/components/api.js`
- 父组件: `openhis-ui-vue3/src/views/inpatientNurse/tprChart/index.vue`
## 根因分析
### 问题1弹窗异常自动关闭 — 根因
`addTprDialog.vue` 模板中,保存按钮使用了 `:disabled="buttonDisabled"`第50行和第108行**`buttonDisabled` 变量在整个 script setup 中从未声明**。
在 Vue 3 `<script setup>` + Composition API 中,模板引用的变量必须在 script 中声明。未声明的变量会触发 `ReferenceError`,导致组件渲染失败或运行时异常。这个错误会破坏组件的响应式系统,使得 `dialogVisible` 的响应式绑定失效,从而导致弹窗在保存操作后异常关闭。
### 问题2缺乏保存成功反馈 — 连带结果
虽然 `confirmCharge()` 函数在第1087行已有 `proxy.$modal.msgSuccess('保存成功')` 的调用,但由于 `buttonDisabled` 未声明引发的异常导致代码执行路径被破坏success 回调中的提示逻辑可能未能正常执行。
## 修复方案
1. **在 `addTprDialog.vue` 的 script setup 中新增 `buttonDisabled` ref 声明**,初始值为 `false`
2. **在保存操作中添加 loading 状态**点击保存后将按钮禁用API 返回后恢复,防止重复提交的同时也保证了响应式状态的一致性
## 验收标准
- [ ] 点击保存后弹窗保持开启状态
- [ ] 保存成功后弹出"保存成功"提示
- [ ] 左侧体征历史记录列表自动刷新
- [ ] 录入区域表单被清空,方便继续录入下一条

View File

@@ -1,40 +0,0 @@
# 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

@@ -1,42 +0,0 @@
# 分析报告 — Bug #469
## 问题描述
检验申请列表的【操作】列仅显示固定的"打印"和"删除"按钮,未根据申请单状态动态切换操作权限。
## 根因分析
文件 `openhis-ui-vue3/src/views/doctorstation/components/inspection/inspectionApplication.vue` 第97-104行
- 操作列模板中固定渲染"打印"和"删除"按钮,没有任何状态判断逻辑
- 缺少"修改"和"撤回"按钮
## 状态机设计
| 状态 | 条件 | 允许的操作 |
|------|------|-----------|
| 待开立 | applyStatus == 0 | 修改、删除 |
| 已开立 | applyStatus == 1 && needExecute != true | 撤回 |
| 已执行 | applyStatus == 1 && needExecute == true | 无(仅打印) |
## 修复方案
1. **前端 Vue**: 操作列改为 `v-if` 条件渲染按钮(修改/删除/撤回/打印)
2. **前端 API**: 新增撤回接口 `withdrawInspectionApplication(applyNo)`
3. **后端 Controller**: 新增 `POST /withdraw/{applyNo}` 端点
4. **后端 Service**: 新增 `withdrawInspectionLabApply` 方法,将 applyStatus 置回 0needRefund/needExecute 置回 false
## 修复结果
✅ 成功共14行改动2个commit完成
### 修复详情
1. **commit c643a78b** - 初始修复:将操作列从静态"打印/删除"改为基于状态的动态按钮(修改/删除/撤回/详情10行改动
2. **commit f369ea41** - 跟进修复:将"详情"按钮包裹在 `<template v-else>`避免对所有状态始终渲染4行改动
### 状态机实现
| 状态 | 条件 | 显示按钮 |
|------|------|---------|
| 待签发 | billStatus == '0' | 修改 + 删除 |
| 已签发 | billStatus == '1' | 撤回 |
| 其他状态 | 已采证/已送检/报告已出/已作废 | 详情 |
### 涉及文件
- `openhis-ui-vue3/src/views/inpatientDoctor/home/components/applicationShow/testApplication.vue` - 前端操作列动态按钮
- `openhis-ui-vue3/src/views/inpatientDoctor/home/components/applicationShow/api.js` - 前端APIdeleteRequestForm, withdrawRequestForm
- `openhis-server-new/openhis-application/src/main/java/com/openhis/web/regdoctorstation/controller/RequestFormManageController.java` - 后端Controller/delete, /withdraw 端点)
- `openhis-server-new/openhis-application/src/main/java/com/openhis/web/regdoctorstation/appservice/impl/RequestFormManageAppServiceImpl.java` - 后端Service实现

85
bug468_analysis.md Normal file
View File

@@ -0,0 +1,85 @@
# Bug #468 分析报告
## Bug 描述
[住院医生工作站-检验申请] 列表页缺失【单据状态】列,无法闭环管理检验医嘱执行进度
## 阶段1深度分析
### 数据流追踪
1. **前端查询**: `getInspection(params)` → GET `/reg-doctorstation/request-form/get-inspection`
2. **后端控制器**: `RequestFormManageController.getInspectionRequestForm()` → 调用 `iRequestFormManageAppService.getRequestForm()`
3. **后端服务**: `RequestFormManageAppServiceImpl.getRequestForm()` → 调用 `requestFormManageAppMapper.getRequestForm()`
4. **SQL查询**: `RequestFormManageAppMapper.xml` 中的 `getRequestForm` 语句
5. **状态计算**: SQL 使用 CASE WHEN 根据 `wor_service_request.status_enum` 计算 `computed_status`
6. **前端渲染**: `parseBillStatus(scope.row.billStatus ?? scope.row.status)` 显示状态文本
### 状态映射关系
**后端 ServiceRequest.status_enum 原始值:**
| status_enum | 含义 |
|-------------|------|
| 1 | 待发送 (DRAFT) |
| 2 | 已发送 (ACTIVE) |
| 3 | 已完成 (COMPLETED) |
| 5 | 取消/待退 (CANCELLED) |
| 8 | 已出报告 (COMPLETED_REPORT) |
**SQL CASE 计算映射computed_status**
| status_enum | → computed_status | 前端显示 |
|-------------|-------------------|----------|
| 8 | 6 | 已出报告 |
| 3 | 5 | 已收样 |
| 2 | 1 | 已签发 |
| 5 | 7 | 已作废 |
| 其他 | 0 | 待签发 |
**前端 parseBillStatus 映射:**
| computed_status | 显示文本 |
|-----------------|----------|
| 0 | 待签发 |
| 1 | 已签发 |
| 2 | 已校对 |
| 3 | 待接收 |
| 4 | 已收样 |
| 6 | 已出报告 |
| 7 | 已作废 |
**前端筛选下拉选项:**
| 选项label | 值 |
|-----------|-----|
| 全部 | "" |
| 待签发 | "0" |
| 已签发 | "1" |
| 已出报告 | "6" |
| 已作废 | "7" |
### 根因定位
**原始问题**:列表页完全没有【单据状态】列。
**已有修复**(已在 develop 分支合并):
1. 新增 `el-table-column` 单据状态列(位于申请单号之后)
2. 新增 `parseBillStatus()` 函数用于状态码→文本转换
3. 新增筛选表单中的单据状态下拉选择
4. 后端 SQL 新增 `computed_status` 动态计算逻辑
5. 前端使用 `scope.row.billStatus ?? scope.row.status` 兼容字段名
## 修复结果
✅ 成功Bug #468 已在 develop 分支修复并合并。
当前 guanyu 分支与 develop 分支代码完全一致diff 为空),无需额外代码改动。
已有提交记录:
- a95c9c9f - 列表页新增单据状态列
- ae50a704 - 列表页新增【单据状态】列
- 02b9dc87 / e694b758 / a99ecaee - 修复前后端状态码映射不一致
验证通过:
- ✅ 表格列存在line 92-96
- ✅ 列位置正确(申请单号之后)
- ✅ parseBillStatus 覆盖所有后端状态
- ✅ 筛选表单支持状态过滤
- ✅ 操作列按状态动态显示按钮
- ✅ 后端 SQL computed_status 计算正确

56
bug491_analysis.md Normal file
View File

@@ -0,0 +1,56 @@
# Bug #491 分析报告
## Bug 信息
- **标题**: 【执行科室配置】保存配置时系统报错
- **报错信息**: `Cannot invoke "com.openhis.administration.domain.Organization.getName()" because the return value of "com.openhis.administration.service..." is null`
- **严重程度**: 3 | **优先级**: 3 | **类型**: codeerror
## 复现步骤
1. 登录 HIS 系统 → 【系统管理】→【业务规则配置】→【执行科室配置】
2. 左侧选择科室(如"超声诊断科"
3. 新增或修改某行的时间区间
4. 点击【保存】按钮
5. 顶部弹出红色错误提示NPE
## 根因分析
### 文件定位
- `openhis-server-new/.../appservice/impl/OrganizationLocationAppServiceImpl.java`第161-175行
### 根本原因
`addOrEditOrgLoc` 方法中,保存时会检查时间冲突。当发现冲突时,代码需要获取冲突记录的科室名称用于错误提示:
```java
// 第171-172行
Organization org = organizationService.getById(organizationLocation.getOrganizationId());
String organizationName = org.getName(); // NPE 这里!
```
**问题**`organizationService.getById()` 可能返回 `null`(当冲突记录的 `organizationId` 指向已被删除的机构时),直接调用 `.getName()` 导致 NPE。
### 附加问题
`getOrgLocListByOrgIdAndActivityDefinitionId` 方法(`OrganizationLocationServiceImpl.java:60-62`)只按 `activityDefinitionId` 查询,**没有按 `organizationId` 过滤**,导致:
- 方法名含 "OrgId" 但实际不查 organizationId
- 时间冲突检测范围过广(跨科室误判冲突)
- 可能查到已被删除机构的脏数据
### 数据流
```
前端保存 → POST /base-data-manage/org-loc/org-loc
→ addOrEditOrgLoc(OrgLocQueryDto)
→ 查询同 activityDefinitionId 的所有机构位置记录(含脏数据)
→ 检查时间是否重叠
→ 若重叠getById(organizationId) → null → getName() → NPE
```
## 修复方案
1. `OrganizationLocationAppServiceImpl.java` 第172行增加 `org != null` 判空,回退为 `"未知科室"`
2. `IOrganizationLocationService.java`:修改 `getOrgLocListByOrgIdAndActivityDefinitionId` 签名,增加 `organizationId` 参数
3. `OrganizationLocationServiceImpl.java`:查询条件增加 `.eq(OrganizationLocation::getOrganizationId, organizationId)`
4. `OrganizationLocationAppServiceImpl.java` 第162行调用时传入 `orgLoc.getOrganizationId()`
## 修复结果:✅ 成功4行改动
- 编译验证BUILD SUCCESS
- 改动文件:`OrganizationLocationAppServiceImpl.java``IOrganizationLocationService.java``OrganizationLocationServiceImpl.java`
- 已提交并推送到远程分支 guanyu

View File

@@ -55,26 +55,3 @@
- `inpatientNurse/` 目录导航配置: 无残留引用 - `inpatientNurse/` 目录导航配置: 无残留引用
**结论**: 修复已生效代码层面无残留。Bug在禅道中仍为active状态需手动标记为resolvedAPI脚本的resolve_bug功能未实现 **结论**: 修复已生效代码层面无残留。Bug在禅道中仍为active状态需手动标记为resolvedAPI脚本的resolve_bug功能未实现
## 2026-05-18 最终复核
经再次验证确认:
- `inpatientDoctor/home/index.vue` 标签页列表: 仅8个正常标签页无"汇总发药申请"
- `inpatientNurse/constants/navigation.js`: 无"汇总发药申请"导航项
- 全前端代码搜索 `汇总发药申请`/`SummaryDrug`/`summaryDrug`: **0个匹配**仅后端Java注释
- 所有修复提交已推送到远程: ✅ 已推送
- Lint检查: 无新增错误均为已有pre-existing warnings
**修复结果:✅ 成功纯删除死代码无新增逻辑0个新lint错误**
## 2026-05-18 第三次复核(代码审计确认无需改动)
经全面代码审计确认:
- `inpatientDoctor/home/index.vue` 标签页列表: 仅8个正常标签页住院病历、诊断录入、临床医嘱、检验申请、检查申请、手术申请、输血申请、报告查询无"汇总发药申请"
- `inpatientNurse/constants/navigation.js`: 6个护士导航项无"汇总发药申请"
- `openhis-ui-vue3` 全目录搜索 `汇总发药申请`: 仅1处API注释`drug/inpatientMedicationDispensing/components/api.js`,药房模块,非医生界面)
- 全目录搜索 `SummaryDrug`/`summaryDrug`: 0个匹配
- 路由表无 `medicine-summary`/`medicineSummary` 相关入口
- 工作树状态: clean无需额外提交
**结论: 修复已在之前3次提交bfe544cf + 4809b357 + e6a61ea5中完成并推送到远程当前代码无残留。无需任何额外改动。**

View File

@@ -1,41 +0,0 @@
# Bug #547 分析报告
## Bug 描述
在"系统管理-执行科室配置"页面,选择科室(如检验科)后添加新项目并保存,显示"与未知科室时间冲突"错误。
## 根因定位
**核心问题在 `OrganizationLocationAppServiceImpl.java:161-174`**
时间冲突检测的查询逻辑存在两个缺陷:
### 缺陷1查询范围过窄
```java
// 只查同一科室 + 同一诊疗的记录
getOrgLocListByOrgIdAndActivityDefinitionId(orgLoc.getOrganizationId(), orgLoc.getActivityDefinitionId());
```
只查询**同一科室**的记录。如果同一诊疗项目在其他科室已有配置且时间重叠,不会被当前查询检测到。但系统本应阻止同一诊疗在多个科室同时段执行。
### 缺陷2"未知科室"错误提示
当冲突记录关联的科室被软删除(`delete_flag='1'`)时,`organizationService.getById()``@TableLogic` 注解影响查不到该科室,返回 null错误提示变成"与未知科室时间冲突"。
数据库验证发现确实存在软删除科室的组织位置记录内科门诊、上海学校医院、信息科等共9条
### 数据流
1. 前端选择科室 → 点击"添加新项目" → 填写诊疗和时间 → 点击"保存"
2. 后端 `addOrEditOrgLoc()` 接收请求
3. 查询现有冲突记录(**当前只查同科室**
4. 对冲突记录检查时间重叠
5. 查找冲突科室名称 → 若科室被软删除则返回 null → "未知科室"
## 修复方案
1. **修改冲突检测范围**:查询同一 `activityDefinitionId` 的所有记录(跨科室检测),而非仅限当前科室
2. **优雅处理"未知科室"**:当 `getById` 返回 null 时,使用 "已删除科室( ID )" 替代 "未知科室",提供更有用的信息
3. **新增 Service 方法**`getOrgLocListByActivityDefinitionId(Long activityDefinitionId)` 用于按诊疗定义查询所有记录
## 涉及文件
- `openhis-server-new/openhis-application/src/main/java/com/openhis/web/basedatamanage/appservice/impl/OrganizationLocationAppServiceImpl.java`
- `openhis-server-new/openhis-domain/src/main/java/com/openhis/administration/service/IOrganizationLocationService.java`
- `openhis-server-new/openhis-domain/src/main/java/com/openhis/administration/service/impl/OrganizationLocationServiceImpl.java`

Submodule his-repo updated: 5de8a22418...414c204578

View File

@@ -0,0 +1,53 @@
## Bug #530 分析报告
**标题**: [住院护士站-医嘱校对] 患者查询触发 SQL 类型匹配错误,导致勾选患者列表后后端报错
### 数据流追踪
1. `patientList.vue` → 树节点勾选触发 `handleCheckChange`
2. `handleCheckChange``updatePatientInfoList(checkedPatients)` 存储选中的患者节点
3. `handleGetPrescription()` 被触发 → `prescriptionList.vue``handleGetPrescription`
4. 前端构造 encounterIds: `patientInfoList.value.map((i) => i.encounterId).join(',')`
5. 后端解析: `Arrays.stream(encounterIds.split(",")).map(Long::parseLong).toList()`
6. SQL 执行: `SELECT ... FROM ... WHERE ii.encounter_id IN (?, ?, ...)`
### 根因定位
**`patientList.vue` 第122行**:患者数据格式化时
```javascript
const patients = records.map((item) => ({
id: item.id || item.encounterId, // 问题行
name: item.patientName || '',
leaf: true,
...item,
}));
```
而后端 `AdmissionPatientPageDto` 中**没有 `id` 字段**(只有 `encounterId``patientId` 等),所以 `item.id``undefined`,此时 `item.id || item.encounterId` 回退到 `item.encounterId`
但在 `prescriptionList.vue` 第186行提取 encounterId 时:
```javascript
let encounterIds = patientInfoList.value.map((i) => i.encounterId).join(',');
```
如果 `patientInfoList` 中的某个对象其 `encounterId` 因任何原因为 `undefined``null` 或空字符串,`join(',')` 会产生类似 `"123,,456"` 的字符串。
后端解析时:
```java
List<Long> encounterIdList = Arrays.stream(encounterIds.split(",")).map(Long::parseLong).toList();
```
`Long.parseLong("")` 会抛出 `NumberFormatException`,导致后端报错。
### 修复方案
`prescriptionList.vue``handleGetPrescription` 函数中,过滤掉无效的 encounterId 值:
- 过滤 `undefined``null`、空字符串
- 如果过滤后无有效 encounterId提示用户并阻止请求
### 验证门禁
- [x] Gate A根因已定位到 `prescriptionList.vue` 第186行未过滤无效 encounterId
- [x] Gate B已读取前后端所有相关文件理解完整数据流
- [x] Gate C修复方案与验收标准一致前端防御性处理 + 正常查询)
- [x] Gate D不涉及数据库字段变更

View File

@@ -1,30 +0,0 @@
# Bug #444 分析报告
## Bug 描述
生成临时医嘱界面,"已引用计费药品"列表未正常显示药品详细名称信息。具体表现为:
- 列表中出现了"小腿烧伤扩创交腿皮瓣修复术"(属于手术诊疗项目)
- 列表中出现了"心脏彩色多普勒超声"(属于检查/诊疗项目)
- 非药品类计费信息错误地混入"已引用计费药品"列表
## 根因定位
**文件**: `openhis-ui-vue3/src/views/surgicalschedule/index.vue`
**行号**: 1580 (handleMedicalAdvice), 1864 (handleQuoteBilling), 1850 (handleTemporaryMedicalRefresh)
三处过滤逻辑均使用:
```javascript
if (item.adviceType !== 1) return false;
```
**问题1主因**: `adviceType` 字段命名兼容不完整。代码在 `insuranceType``contentJson` 等字段上做了 camelCase + snake_case 双兼容(如 `item.insuranceType || item.insurance_type`),但 `adviceType` 只检查了 camelCase。若后端返回 snake_case 数据(`advice_type``item.adviceType``undefined``undefined !== 1``true`,导致所有非药品项目全部放行。
**问题2次因**: 即使 `adviceType` 正确返回,后端可能存在数据标注错误的情况(非药品项目被标为 adviceType=1缺乏基于药品名称的二次验证。
## 修复方案
1. `adviceType` 检查增加 snake_case 回退:`const at = item.adviceType ?? item.advice_type; if (at !== 1) return false;`
2. 增加药品名称关键字二次过滤:排除名称中包含"术"、"检查"、"超声"、"多普勒"等关键词的非药品项目
## 验收标准
1. "已引用计费药品"列表中只显示药品类项目
2. 不显示手术诊疗项目(如"小腿烧伤扩创交腿皮瓣修复术"
3. 不显示检查项目(如"心脏彩色多普勒超声"
4. 药品名称正常显示

View File

@@ -132,22 +132,7 @@ temporaryAdvices.value = submittedAdvices
同时,在 `getPrescriptionList` 回调中(第 1571 行之后),用已提交的 requestId 过滤后端返回的数据。 同时,在 `getPrescriptionList` 回调中(第 1571 行之后),用已提交的 requestId 过滤后端返回的数据。
## 修复结果 ## 总结
### 实际根因 - **根因**`handleMedicalAdvice` 每次打开都清空 `temporaryAdvices`,然后从后端重新拉取数据。但后端返回的新创建医嘱项可能没有 `requestId`,导致无法过滤。
`handleQuoteBilling` 函数中: - **修复**:保留已提交(有 requestId的医嘱数据不清空同时用这些 requestId 过滤后端返回的新数据。
1. **第1856行**:在调用 `getPrescriptionList` 之前先清空了 `temporaryAdvices.value = []`
2. **第1997-2019行旧代码**ID 匹配过滤逻辑依赖已被清空的 `temporaryAdvices.value`,因此过滤形同虚设
3. 即使 `temporaryAdvices` 未被清空ID 匹配也不可靠(新生成的医嘱可能没有 `requestId`/`chargeItemId`/`id`
### 修复方案
1. 在清空 `temporaryAdvices` **之前**,提取已提交项目的复合键(名称+规格+数量)保存到 `submittedKeysBeforeClear`
2.`submittedKeysBeforeClear` 替换原有的 ID 匹配过滤逻辑,确保即使后端未返回 `requestId` 也能正确过滤
3. 复合键匹配策略与 `handleTemporaryMedicalSubmit` 中使用的策略一致
### 修改文件
- `openhis-ui-vue3/src/views/surgicalschedule/index.vue`
- 第1853-1864行新增 `submittedKeysBeforeClear` 提取逻辑
- 第1997-2004行替换 ID 匹配为复合键匹配
### 修复结果:✅ 成功,~20行改动+20/-21

View File

@@ -1,7 +1,6 @@
package com.core.framework.config; package com.core.framework.config;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer; import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
import org.mybatis.spring.annotation.MapperScan; import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.autoconfigure.jackson.Jackson2ObjectMapperBuilderCustomizer; import org.springframework.boot.autoconfigure.jackson.Jackson2ObjectMapperBuilderCustomizer;
@@ -35,9 +34,7 @@ public class ApplicationConfig {
// 设置日期格式为 yyyy/M/d HH:mm:ss支持多种格式反序列化 // 设置日期格式为 yyyy/M/d HH:mm:ss支持多种格式反序列化
builder.simpleDateFormat("yyyy/M/d HH:mm:ss"); builder.simpleDateFormat("yyyy/M/d HH:mm:ss");
// 添加JavaTimeModule支持用于LocalDateTime // 添加JavaTimeModule支持用于LocalDateTime
JavaTimeModule javaTimeModule = new JavaTimeModule(); builder.modules(new JavaTimeModule());
javaTimeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
builder.modules(javaTimeModule);
builder.serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern("yyyy/M/d HH:mm:ss"))); builder.serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern("yyyy/M/d HH:mm:ss")));
}; };
} }

View File

@@ -4,7 +4,7 @@ import cn.hutool.core.util.ObjectUtil;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.core.common.core.domain.R; import com.core.common.core.domain.R;
import com.core.common.utils.SecurityUtils; import com.core.common.utils.SecurityUtils;
import com.openhis.common.enums.SlotStatus; import com.openhis.common.constant.CommonConstants;
import com.openhis.appointmentmanage.domain.DoctorSchedule; import com.openhis.appointmentmanage.domain.DoctorSchedule;
import com.openhis.appointmentmanage.domain.DoctorScheduleWithDateDto; import com.openhis.appointmentmanage.domain.DoctorScheduleWithDateDto;
import com.openhis.appointmentmanage.domain.SchedulePool; import com.openhis.appointmentmanage.domain.SchedulePool;
@@ -502,8 +502,8 @@ public class DoctorScheduleAppServiceImpl implements IDoctorScheduleAppService {
// 该排班下存在有效患者预约(号源槽:已预约/已锁定/已取号)则禁止删除;已退号、仅可用/已取消槽位不计入 // 该排班下存在有效患者预约(号源槽:已预约/已锁定/已取号)则禁止删除;已退号、仅可用/已取消槽位不计入
long appointmentCount = scheduleSlotService.count(new QueryWrapper<ScheduleSlot>() long appointmentCount = scheduleSlotService.count(new QueryWrapper<ScheduleSlot>()
.in("pool_id", poolIds) .in("pool_id", poolIds)
.in("status", SlotStatus.BOOKED.getValue(), SlotStatus.LOCKED.getValue(), .in("status", CommonConstants.SlotStatus.BOOKED, CommonConstants.SlotStatus.LOCKED,
SlotStatus.CHECKED_IN.getValue())); CommonConstants.SlotStatus.CHECKED_IN));
if (appointmentCount > 0) { if (appointmentCount > 0) {
return R.fail("该排班已有患者预约,禁止删除!如需取消请先处理患者退预约或使用'停诊'功能。"); return R.fail("该排班已有患者预约,禁止删除!如需取消请先处理患者退预约或使用'停诊'功能。");
} }

View File

@@ -9,7 +9,7 @@ import com.openhis.clinical.domain.Ticket;
import com.openhis.clinical.service.ITicketService; import com.openhis.clinical.service.ITicketService;
import com.openhis.web.appointmentmanage.appservice.ITicketAppService; import com.openhis.web.appointmentmanage.appservice.ITicketAppService;
import com.openhis.web.appointmentmanage.dto.TicketDto; import com.openhis.web.appointmentmanage.dto.TicketDto;
import com.openhis.common.enums.SlotStatus; import com.openhis.common.constant.CommonConstants.SlotStatus;
import com.openhis.common.enums.OrderStatus; import com.openhis.common.enums.OrderStatus;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
@@ -193,24 +193,25 @@ public class TicketAppServiceImpl implements ITicketAppService {
if (Boolean.TRUE.equals(raw.getIsStopped())) { if (Boolean.TRUE.equals(raw.getIsStopped())) {
dto.setStatus("已停诊"); dto.setStatus("已停诊");
} else { } else {
SlotStatus status = SlotStatus.getByValue(raw.getSlotStatus()); Integer slotStatus = raw.getSlotStatus();
if (status != null) { if (slotStatus != null) {
if (status == SlotStatus.LOCKED) { if (SlotStatus.CHECKED_IN.equals(slotStatus)) {
dto.setStatus("已取号");
} else if (SlotStatus.BOOKED.equals(slotStatus)) {
// order_main.status: 0=患者取消(已退号) 2=系统取消 其余=已预约
if (OrderStatus.PATIENT_CANCELLED.getValue().equals(raw.getOrderStatus())) { if (OrderStatus.PATIENT_CANCELLED.getValue().equals(raw.getOrderStatus())) {
dto.setStatus("已退号"); dto.setStatus("已退号");
} else if (OrderStatus.SYSTEM_CANCELLED.getValue().equals(raw.getOrderStatus())) {
dto.setStatus("系统取消");
} else { } else {
dto.setStatus("锁定"); dto.setStatus("预约");
} }
} else if (status == SlotStatus.BOOKED) { } else if (SlotStatus.RETURNED.equals(slotStatus)) {
if (OrderStatus.PATIENT_CANCELLED.getValue().equals(raw.getOrderStatus())) {
dto.setStatus("已退号");
} else {
dto.setStatus("已取号");
}
} else if (status == SlotStatus.CANCELLED) {
dto.setStatus("已停诊");
} else if (status == SlotStatus.RETURNED) {
dto.setStatus("已退号"); dto.setStatus("已退号");
} else if (SlotStatus.CANCELLED.equals(slotStatus)) {
dto.setStatus("已停诊");
} else if (SlotStatus.LOCKED.equals(slotStatus)) {
dto.setStatus("已锁定");
} else { } else {
dto.setStatus("未预约"); dto.setStatus("未预约");
} }
@@ -236,10 +237,6 @@ public class TicketAppServiceImpl implements ITicketAppService {
/** /**
* 统一状态入参,避免前端状态值大小写/中文/数字差异导致 SQL 条件失效后回全量数据 * 统一状态入参,避免前端状态值大小写/中文/数字差异导致 SQL 条件失效后回全量数据
*/ */
/**
* 规范前端传入的状态查询参数,映射到 SQL 的 slotStatusNormExpr 值。
* 数值映射: 0=待约 1=已约(签到后) 2=锁定(预约后) 3=已签到 4=已停诊 5=已退号
*/
private void normalizeQueryStatus(com.openhis.appointmentmanage.dto.TicketQueryDTO query) { private void normalizeQueryStatus(com.openhis.appointmentmanage.dto.TicketQueryDTO query) {
String rawStatus = query.getStatus(); String rawStatus = query.getStatus();
if (rawStatus == null) { if (rawStatus == null) {
@@ -266,31 +263,28 @@ public class TicketAppServiceImpl implements ITicketAppService {
case "已预约": case "已预约":
query.setStatus("booked"); query.setStatus("booked");
break; break;
case "locked":
case "2":
case "已锁定":
query.setStatus("locked");
break;
case "checked": case "checked":
case "checkin": case "checkin":
case "checkedin": case "checkedin":
case "3": case "2":
case "已取号": case "已取号":
query.setStatus("checked"); query.setStatus("checked");
break; break;
case "cancelled": case "cancelled":
case "canceled": case "canceled":
case "4": case "3":
case "已停诊": case "已停诊":
case "已取消": case "已取消":
query.setStatus("cancelled"); query.setStatus("cancelled");
break; break;
case "returned": case "returned":
case "4":
case "5": case "5":
case "已退号": case "已退号":
query.setStatus("returned"); query.setStatus("returned");
break; break;
default: default:
// 设置为 impossible 值,配合 mapper 的 otherwise 分支直接返回空
query.setStatus("__invalid__"); query.setStatus("__invalid__");
break; break;
} }
@@ -373,25 +367,26 @@ public class TicketAppServiceImpl implements ITicketAppService {
if (Boolean.TRUE.equals(raw.getIsStopped())) { if (Boolean.TRUE.equals(raw.getIsStopped())) {
dto.setStatus("已停诊"); dto.setStatus("已停诊");
} else { } else {
// 第二关:看独立的细分槽位状态 (0: 可用, 1: 已预约, 2: 已锁定...) // 第二关:看独立的细分槽位状态 (0: 可用, 1: 已预约, 2: 已取消...)
SlotStatus status = SlotStatus.getByValue(raw.getSlotStatus()); Integer slotStatus = raw.getSlotStatus();
if (status != null) { if (slotStatus != null) {
if (status == SlotStatus.LOCKED) { if (SlotStatus.CHECKED_IN.equals(slotStatus)) {
dto.setStatus("已取号");
} else if (SlotStatus.BOOKED.equals(slotStatus)) {
// order_main.status: 0=患者取消(已退号) 2=系统取消 其余=已预约
if (OrderStatus.PATIENT_CANCELLED.getValue().equals(raw.getOrderStatus())) { if (OrderStatus.PATIENT_CANCELLED.getValue().equals(raw.getOrderStatus())) {
dto.setStatus("已退号"); dto.setStatus("已退号");
} else if (OrderStatus.SYSTEM_CANCELLED.getValue().equals(raw.getOrderStatus())) {
dto.setStatus("系统取消");
} else { } else {
dto.setStatus("锁定"); dto.setStatus("预约");
} }
} else if (status == SlotStatus.BOOKED) { } else if (SlotStatus.RETURNED.equals(slotStatus)) {
if (OrderStatus.PATIENT_CANCELLED.getValue().equals(raw.getOrderStatus())) {
dto.setStatus("已退号");
} else {
dto.setStatus("已取号");
}
} else if (status == SlotStatus.CANCELLED) {
dto.setStatus("已停诊");
} else if (status == SlotStatus.RETURNED) {
dto.setStatus("已退号"); dto.setStatus("已退号");
} else if (SlotStatus.CANCELLED.equals(slotStatus)) {
dto.setStatus("已停诊");
} else if (SlotStatus.LOCKED.equals(slotStatus)) {
dto.setStatus("已锁定");
} else { } else {
dto.setStatus("未预约"); dto.setStatus("未预约");
} }

View File

@@ -159,7 +159,7 @@ public class OrganizationLocationAppServiceImpl implements IOrganizationLocation
String activityName = activityDef != null ? activityDef.getName() : ""; String activityName = activityDef != null ? activityDef.getName() : "";
List<OrganizationLocation> organizationLocationList = List<OrganizationLocation> organizationLocationList =
organizationLocationService.getOrgLocListByActivityDefinitionId(orgLoc.getActivityDefinitionId()); organizationLocationService.getOrgLocListByOrgIdAndActivityDefinitionId(orgLoc.getOrganizationId(), orgLoc.getActivityDefinitionId());
organizationLocationList = (orgLoc.getId() != null) organizationLocationList = (orgLoc.getId() != null)
? organizationLocationList.stream().filter(item -> !orgLoc.getId().equals(item.getId())).toList() ? organizationLocationList.stream().filter(item -> !orgLoc.getId().equals(item.getId())).toList()
: organizationLocationList; : organizationLocationList;
@@ -169,7 +169,7 @@ public class OrganizationLocationAppServiceImpl implements IOrganizationLocation
if (DateTimeUtils.isOverlap(organizationLocation.getStartTime(), organizationLocation.getEndTime(), if (DateTimeUtils.isOverlap(organizationLocation.getStartTime(), organizationLocation.getEndTime(),
orgLoc.getStartTime(), orgLoc.getEndTime())) { orgLoc.getStartTime(), orgLoc.getEndTime())) {
Organization org = organizationService.getById(organizationLocation.getOrganizationId()); Organization org = organizationService.getById(organizationLocation.getOrganizationId());
String organizationName = org != null ? org.getName() : ("科室[" + organizationLocation.getOrganizationId() + "]已删除"); String organizationName = org != null && org.getName() != null ? org.getName() : "未知科室";
return R.fail("当前诊疗:" + activityName + CommonConstants.Common.DASH + orgLoc.getStartTime() return R.fail("当前诊疗:" + activityName + CommonConstants.Common.DASH + orgLoc.getStartTime()
+ CommonConstants.Common.DASH + orgLoc.getEndTime() + "" + organizationName + "时间冲突"); + CommonConstants.Common.DASH + orgLoc.getEndTime() + "" + organizationName + "时间冲突");
} }

View File

@@ -18,7 +18,6 @@ import com.openhis.administration.mapper.PatientMapper;
import com.openhis.administration.service.*; import com.openhis.administration.service.*;
import com.openhis.common.constant.CommonConstants; import com.openhis.common.constant.CommonConstants;
import com.openhis.common.constant.PromptMsgConstant; import com.openhis.common.constant.PromptMsgConstant;
import com.openhis.common.enums.SlotStatus;
import com.openhis.common.enums.*; import com.openhis.common.enums.*;
import com.openhis.common.enums.ybenums.YbPayment; import com.openhis.common.enums.ybenums.YbPayment;
import com.openhis.common.utils.EnumUtils; import com.openhis.common.utils.EnumUtils;
@@ -644,7 +643,8 @@ public class OutpatientRegistrationAppServiceImpl implements IOutpatientRegistra
.set(Order::getStatus, OrderStatus.PATIENT_CANCELLED.getValue()) .set(Order::getStatus, OrderStatus.PATIENT_CANCELLED.getValue())
.set(Order::getPayStatus, PaymentStatus.REFUND_ALL.getValue()) .set(Order::getPayStatus, PaymentStatus.REFUND_ALL.getValue())
.set(Order::getCancelTime, new Date()) .set(Order::getCancelTime, new Date())
.set(Order::getCancelReason, "诊前退号") .set(Order::getCancelReason,
StringUtils.isNotEmpty(reason) ? reason : "诊前退号")
.set(Order::getUpdateTime, new Date()) .set(Order::getUpdateTime, new Date())
.setSql("version = version + 1") .setSql("version = version + 1")
.eq(Order::getId, appointmentOrder.getId()) .eq(Order::getId, appointmentOrder.getId())
@@ -660,27 +660,17 @@ public class OutpatientRegistrationAppServiceImpl implements IOutpatientRegistra
return appointmentOrder.getId(); return appointmentOrder.getId();
} }
// 只有已预约(1)的号源才能退号,对应签到后的 BOOKED 状态 int slotRows = scheduleSlotMapper.updateSlotStatus(slotId, CommonConstants.SlotStatus.AVAILABLE);
ScheduleSlot slot = scheduleSlotMapper.selectById(slotId); if (slotRows > 0) {
if (slot == null || !SlotStatus.BOOKED.getValue().equals(slot.getStatus())) { Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId);
log.warn("退号跳过:槽位非已预约状态, slotId={}, status={}", slotId, if (poolId != null) {
slot != null ? slot.getStatus() : null); schedulePoolMapper.refreshPoolStats(poolId);
return appointmentOrder.getId(); schedulePoolMapper.update(null,
} new LambdaUpdateWrapper<SchedulePool>()
.setSql("version = version + 1")
int slotRows = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.AVAILABLE.getValue()); .set(SchedulePool::getUpdateTime, new Date())
if (slotRows == 0) { .eq(SchedulePool::getId, poolId));
log.warn("退号时更新槽位状态未影响任何行, slotId={}", slotId); }
return appointmentOrder.getId();
}
Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId);
if (poolId != null) {
schedulePoolMapper.update(null,
new LambdaUpdateWrapper<SchedulePool>()
.setSql("booked_num = booked_num - 1, version = version + 1")
.set(SchedulePool::getUpdateTime, new Date())
.eq(SchedulePool::getId, poolId));
} }
return appointmentOrder.getId(); return appointmentOrder.getId();
} catch (Exception e) { } catch (Exception e) {

View File

@@ -507,7 +507,6 @@ public class SurgeryAppServiceImpl implements ISurgeryAppService {
* @return 结果 * @return 结果
*/ */
@Override @Override
@Transactional(rollbackFor = Exception.class)
public R<?> deleteSurgery(Long id) { public R<?> deleteSurgery(Long id) {
// 校验手术是否存在 // 校验手术是否存在
Surgery existSurgery = surgeryService.getById(id); Surgery existSurgery = surgeryService.getById(id);
@@ -520,28 +519,6 @@ public class SurgeryAppServiceImpl implements ISurgeryAppService {
return R.fail("已完成的手术不能删除"); 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); surgeryService.deleteSurgery(id);
// 清除相关缓存 // 清除相关缓存

View File

@@ -215,10 +215,7 @@ public class SurgicalScheduleAppServiceImpl implements ISurgicalScheduleAppServi
if (surgery != null) { if (surgery != null) {
surgery.setStatusEnum(1); // 1 = 已排期 surgery.setStatusEnum(1); // 1 = 已排期
surgery.setUpdateTime(new Date()); surgery.setUpdateTime(new Date());
// Bug #558: 手术安排时同步写入手术室确认时间和确认人
surgery.setOperatingRoomConfirmTime(new Date());
surgery.setOperatingRoomConfirmUser(loginUser.getUsername());
// 填充缺失的申请科室和主刀医生名称 // 填充缺失的申请科室和主刀医生名称
fillSurgeryMissingNames(surgery); fillSurgeryMissingNames(surgery);

View File

@@ -147,6 +147,6 @@ public interface IDoctorStationAdviceAppService {
*/ */
IPage<SurgeryItemDto> getSurgeryPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey); IPage<SurgeryItemDto> getSurgeryPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey);
IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey, String categoryCode); IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey);
} }

View File

@@ -63,21 +63,17 @@ public interface IDoctorStationEmrAppService {
* 获取待写病历列表 * 获取待写病历列表
* *
* @param doctorId 医生ID * @param doctorId 医生ID
* @param pageNo 当前页码 * @return 待写病历列表
* @param pageSize 每页条数
* @param patientName 患者姓名(可选)
* @return 待写病历分页数据
*/ */
R<?> getPendingEmrList(Long doctorId, Integer pageNo, Integer pageSize, String patientName); R<?> getPendingEmrList(Long doctorId);
/** /**
* 获取待写病历数量 * 获取待写病历数量
* *
* @param doctorId 医生ID * @param doctorId 医生ID
* @param patientName 患者姓名(可选)
* @return 待写病历数量 * @return 待写病历数量
*/ */
R<?> getPendingEmrCount(Long doctorId, String patientName); R<?> getPendingEmrCount(Long doctorId);
/** /**
* 检查患者是否需要写病历 * 检查患者是否需要写病历

View File

@@ -35,9 +35,6 @@ import com.openhis.medication.service.IMedicationDispenseService;
import com.openhis.medication.service.IMedicationRequestService; import com.openhis.medication.service.IMedicationRequestService;
import com.openhis.web.chargemanage.mapper.OutpatientRegistrationAppMapper; import com.openhis.web.chargemanage.mapper.OutpatientRegistrationAppMapper;
import com.openhis.web.doctorstation.appservice.IDoctorStationAdviceAppService; import com.openhis.web.doctorstation.appservice.IDoctorStationAdviceAppService;
import com.openhis.document.service.IRequestFormService;
import com.openhis.clinical.service.ISurgeryService;
import com.openhis.clinical.domain.Surgery;
import com.openhis.web.doctorstation.appservice.IDoctorStationInspectionLabApplyService; import com.openhis.web.doctorstation.appservice.IDoctorStationInspectionLabApplyService;
import com.openhis.web.doctorstation.dto.*; import com.openhis.web.doctorstation.dto.*;
import com.openhis.web.doctorstation.mapper.DoctorStationAdviceAppMapper; import com.openhis.web.doctorstation.mapper.DoctorStationAdviceAppMapper;
@@ -54,7 +51,6 @@ import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Lazy; import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
import com.openhis.document.domain.RequestForm;
import javax.annotation.Resource; import javax.annotation.Resource;
import java.math.BigDecimal; import java.math.BigDecimal;
@@ -73,7 +69,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
private static final Pattern INSPECTION_APPLY_NO_JSON = private static final Pattern INSPECTION_APPLY_NO_JSON =
Pattern.compile("\"applyNo\"\\s*:\\s*\"([^\"]+)\""); Pattern.compile("\"applyNo\"\\s*:\\s*\"([^\"]+)\"");
@Resource @Resource
AssignSeqUtil assignSeqUtil; AssignSeqUtil assignSeqUtil;
@@ -137,20 +132,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
@Lazy @Lazy
private IDoctorStationInspectionLabApplyService iDoctorStationInspectionLabApplyService; private IDoctorStationInspectionLabApplyService iDoctorStationInspectionLabApplyService;
/**
* 与RequestFormManageAppServiceImpl存在循环依赖需延迟注入删除手术医嘱时级联作废手术申请单。
*/
@Resource
@Lazy
private IRequestFormService iRequestFormService;
/**
* 删除手术医嘱时级联删除 cli_surgery 手术记录。
*/
@Resource
@Lazy
private ISurgeryService iSurgeryService;
// 缓存 key 前缀 // 缓存 key 前缀
private static final String ADVICE_BASE_INFO_CACHE_PREFIX = "advice:base:info:"; private static final String ADVICE_BASE_INFO_CACHE_PREFIX = "advice:base:info:";
// 缓存过期时间(小时) // 缓存过期时间(小时)
@@ -578,11 +559,9 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
if (adviceSaveList != null && !adviceSaveList.isEmpty()) { if (adviceSaveList != null && !adviceSaveList.isEmpty()) {
for (int i = 0; i < adviceSaveList.size(); i++) { for (int i = 0; i < adviceSaveList.size(); i++) {
AdviceSaveDto dto = adviceSaveList.get(i); AdviceSaveDto dto = adviceSaveList.get(i);
log.info("Request[{}]: requestId={}, dbOpType={}, adviceType={}, encounterId={}, patientId={}, categoryEnum={}, categoryEnum.class={}, categoryCode={}, categoryCode.class={}", log.info("Request[{}]: requestId={}, dbOpType={}, adviceType={}, encounterId={}, patientId={}",
i, dto.getRequestId(), dto.getDbOpType(), dto.getAdviceType(), i, dto.getRequestId(), dto.getDbOpType(), dto.getAdviceType(),
dto.getEncounterId(), dto.getPatientId(), dto.getEncounterId(), dto.getPatientId());
dto.getCategoryEnum(), dto.getCategoryEnum() != null ? dto.getCategoryEnum().getClass().getName() : "NULL",
dto.getCategoryCode(), dto.getCategoryCode() != null ? dto.getCategoryCode().getClass().getName() : "NULL");
} }
} }
@@ -1583,7 +1562,7 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
// 🔧 BugFix #498: categoryEnum=22(检查) 走 ServiceRequest不走 DeviceRequest // 🔧 BugFix #498: categoryEnum=22(检查) 走 ServiceRequest不走 DeviceRequest
// 检查申请单的诊疗定义ID存在 activityId不在 adviceDefinitionId // 检查申请单的诊疗定义ID存在 activityId不在 adviceDefinitionId
// deviceDefId 对应耗材定义ID不能用诊疗定义ID填充 // deviceDefId 对应耗材定义ID不能用诊疗定义ID填充
if (Integer.valueOf(22).equals(adviceSaveDto.getCategoryEnum())) { if (adviceSaveDto.getCategoryEnum() == 22) {
log.info("handDevice skip - 检查申请单(categoryEnum=22) 走 ServiceRequest 路径,跳过 DeviceRequest 保存"); log.info("handDevice skip - 检查申请单(categoryEnum=22) 走 ServiceRequest 路径,跳过 DeviceRequest 保存");
continue; // 跳过本次循环,不走耗材请求路径 continue; // 跳过本次循环,不走耗材请求路径
} else if (adviceSaveDto.getAdviceDefinitionId() != null) { } else if (adviceSaveDto.getAdviceDefinitionId() != null) {
@@ -1769,7 +1748,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
return StringUtils.isBlank(applyNo) ? null : applyNo; return StringUtils.isBlank(applyNo) ? null : applyNo;
} }
/** /**
* 处理诊疗 * 处理诊疗
*/ */
@@ -1818,50 +1796,31 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
} }
} }
} }
// 🔧 级联作废:在删除 ServiceRequest 之前,先读取所有待删除记录的级联信息 // 检验申请单在医嘱 contentJson 中写入 applyNo从医嘱删除时需先级联作废检验单避免检验页签仍显示孤儿申请
// 检验申请单contentJson 中写入 applyNo手术申请单categoryEnum=24 + prescriptionNo
Map<String, List<Long>> labApplyNoToRequestIds = new LinkedHashMap<>(); Map<String, List<Long>> labApplyNoToRequestIds = new LinkedHashMap<>();
Map<String, List<Long>> surgeryPrescriptionNoToRequestIds = new LinkedHashMap<>();
// 收集待删除的 ServiceRequest先查询再删除避免级联逻辑因记录已删除而失效
Map<Long, ServiceRequest> serviceRequestCache = new LinkedHashMap<>();
for (AdviceSaveDto adviceSaveDto : deleteList) { for (AdviceSaveDto adviceSaveDto : deleteList) {
Long requestId = adviceSaveDto.getRequestId(); Long requestId = adviceSaveDto.getRequestId();
// 🔧 Bug #442: 跳过 requestId 为 null 的记录 // 🔧 Bug #442: 跳过 requestId 为 null 的记录,避免删除不存在的诊疗请求
if (requestId == null) { if (requestId == null) {
log.warn("BugFix#442: handService - 跳过 requestId 为 null 的删除请求"); log.warn("BugFix#442: handService - 跳过 requestId 为 null 的删除请求");
continue; continue;
} }
ServiceRequest existing = iServiceRequestService.getById(requestId); iServiceRequestService.removeById(requestId);// 删除诊疗
ServiceRequest existing = iServiceRequestService.getById(adviceSaveDto.getRequestId());
if (existing == null) { if (existing == null) {
continue; continue;
} }
serviceRequestCache.put(requestId, existing);
log.info("【调试】handService 待删除医嘱: requestId={}, categoryEnum={}, prescriptionNo={}",
requestId, existing.getCategoryEnum(), existing.getPrescriptionNo());
// 检验申请单级联
String applyNo = extractInspectionApplyNoFromContentJson(existing.getContentJson()); String applyNo = extractInspectionApplyNoFromContentJson(existing.getContentJson());
if (StringUtils.isNotBlank(applyNo)) { if (StringUtils.isNotBlank(applyNo)) {
labApplyNoToRequestIds.computeIfAbsent(applyNo, k -> new ArrayList<>()) labApplyNoToRequestIds.computeIfAbsent(applyNo, k -> new ArrayList<>())
.add(requestId); .add(adviceSaveDto.getRequestId());
}
// 手术申请单级联categoryEnum=24
log.info("【调试】handService 判断手术条件: categoryEnum={}, prescriptionNo={}, isSurgery={}",
existing.getCategoryEnum(), existing.getPrescriptionNo(),
existing.getCategoryEnum() != null && existing.getCategoryEnum() == 24 && StringUtils.isNotBlank(existing.getPrescriptionNo()));
if (existing.getCategoryEnum() != null
&& existing.getCategoryEnum() == 24
&& StringUtils.isNotBlank(existing.getPrescriptionNo())) {
surgeryPrescriptionNoToRequestIds.computeIfAbsent(existing.getPrescriptionNo(), k -> new ArrayList<>())
.add(requestId);
log.info("【调试】handService 加入手术级联列表: prescriptionNo={}", existing.getPrescriptionNo());
} }
} }
// 执行检验申请单级联作废 Set<Long> labCascadeSkippedRequestIds = new HashSet<>();
Set<Long> cascadeSkippedRequestIds = new HashSet<>();
for (Map.Entry<String, List<Long>> e : labApplyNoToRequestIds.entrySet()) { for (Map.Entry<String, List<Long>> e : labApplyNoToRequestIds.entrySet()) {
R<?> delLab = iDoctorStationInspectionLabApplyService.deleteInspectionLabApply(e.getKey()); R<?> delLab = iDoctorStationInspectionLabApplyService.deleteInspectionLabApply(e.getKey());
if (delLab != null && R.isSuccess(delLab)) { if (delLab != null && R.isSuccess(delLab)) {
cascadeSkippedRequestIds.addAll(e.getValue()); labCascadeSkippedRequestIds.addAll(e.getValue());
log.info("handService - 级联作废检验申请单 applyNo={},已跳过重复删除的医嘱 requestIds={}", log.info("handService - 级联作废检验申请单 applyNo={},已跳过重复删除的医嘱 requestIds={}",
e.getKey(), e.getValue()); e.getKey(), e.getValue());
} else { } else {
@@ -1870,41 +1829,8 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
e.getKey(), msg); e.getKey(), msg);
} }
} }
// 🔧 手术申请单级联作废:删除手术医嘱时同步作废关联的手术申请单
for (Map.Entry<String, List<Long>> e : surgeryPrescriptionNoToRequestIds.entrySet()) {
String prescriptionNo = e.getKey();
try {
List<RequestForm> requestForms = iRequestFormService.list(
new LambdaQueryWrapper<RequestForm>()
.eq(RequestForm::getPrescriptionNo, prescriptionNo)
.in(RequestForm::getTypeCode, ActivityDefCategory.PROCEDURE.getCode().toString(), "SURGERY")
.and(w -> w.isNull(RequestForm::getDeleteFlag).or().eq(RequestForm::getDeleteFlag, "0")));
log.info("【调试】handService 查询手术申请单: prescriptionNo={}, 查到{}条", prescriptionNo, requestForms != null ? requestForms.size() : 0);
if (requestForms != null && !requestForms.isEmpty()) {
for (RequestForm requestForm : requestForms) {
iRequestFormService.removeById(requestForm.getId());
}
// 同步删除 cli_surgery 手术记录prescriptionNo = surgeryNo
Surgery surgery = iSurgeryService.getOne(
new LambdaQueryWrapper<Surgery>()
.eq(Surgery::getSurgeryNo, prescriptionNo)
.and(w -> w.isNull(Surgery::getDeleteFlag).or().eq(Surgery::getDeleteFlag, "0")));
if (surgery != null) {
iSurgeryService.removeById(surgery.getId());
log.info("handService - 级联删除手术记录 cli_surgery: surgeryNo={}, id={}", prescriptionNo, surgery.getId());
}
cascadeSkippedRequestIds.addAll(e.getValue());
log.info("handService - 级联作废手术申请单 prescriptionNo={}", prescriptionNo);
} else {
log.info("handService - 未找到手术申请单 prescriptionNo={}", prescriptionNo);
}
} catch (Exception ex) {
log.warn("handService - 级联作废手术申请单失败 prescriptionNo={} msg={}", prescriptionNo, ex.getMessage());
}
}
// 级联作废完成后,统一删除 ServiceRequest 及其子项、费用项
for (AdviceSaveDto adviceSaveDto : deleteList) { for (AdviceSaveDto adviceSaveDto : deleteList) {
if (cascadeSkippedRequestIds.contains(adviceSaveDto.getRequestId())) { if (labCascadeSkippedRequestIds.contains(adviceSaveDto.getRequestId())) {
continue; continue;
} }
Long requestId = adviceSaveDto.getRequestId(); Long requestId = adviceSaveDto.getRequestId();
@@ -1914,6 +1840,7 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
requestId));// 删除诊疗套餐对应的子项 requestId));// 删除诊疗套餐对应的子项
// 🔧 Bug Fix #219: 删除费用项 // 🔧 Bug Fix #219: 删除费用项
String serviceTable = CommonConstants.TableName.WOR_SERVICE_REQUEST; String serviceTable = CommonConstants.TableName.WOR_SERVICE_REQUEST;
// 直接删除费用项
iChargeItemService.deleteByServiceTableAndId(serviceTable, requestId); iChargeItemService.deleteByServiceTableAndId(serviceTable, requestId);
log.info("BugFix#219: 诊疗医嘱删除完成, requestId={}, serviceTable={}", requestId, serviceTable); log.info("BugFix#219: 诊疗医嘱删除完成, requestId={}, serviceTable={}", requestId, serviceTable);
} }
@@ -2192,6 +2119,11 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
CommonConstants.TableName.MED_MEDICATION_REQUEST, CommonConstants.TableName.WOR_DEVICE_REQUEST, CommonConstants.TableName.MED_MEDICATION_REQUEST, CommonConstants.TableName.WOR_DEVICE_REQUEST,
CommonConstants.TableName.WOR_SERVICE_REQUEST, practitionerId, Whether.NO.getCode(), CommonConstants.TableName.WOR_SERVICE_REQUEST, practitionerId, Whether.NO.getCode(),
sourceEnum, sourceBillNo); sourceEnum, sourceBillNo);
// 手术计费场景sourceBillNo 不为空时过滤掉药品1保留耗材2和诊疗3/6
if (sourceBillNo != null && !sourceBillNo.isEmpty()) {
requestBaseInfo.removeIf(dto -> dto.getAdviceType() != null
&& dto.getAdviceType() == 1);
}
for (RequestBaseDto requestBaseDto : requestBaseInfo) { for (RequestBaseDto requestBaseDto : requestBaseInfo) {
// 请求状态 // 请求状态
requestBaseDto requestBaseDto
@@ -2205,10 +2137,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
// 收费状态 // 收费状态
requestBaseDto.setChargeStatus_enumText( requestBaseDto.setChargeStatus_enumText(
EnumUtils.getInfoByValue(ChargeItemStatus.class, requestBaseDto.getChargeStatus())); EnumUtils.getInfoByValue(ChargeItemStatus.class, requestBaseDto.getChargeStatus()));
// 单位字典翻译失败时回退使用原始值(如手术申请硬编码了中文单位名)
if (StringUtils.isNotBlank(requestBaseDto.getUnitCode()) && StringUtils.isBlank(requestBaseDto.getUnitCode_dictText())) {
requestBaseDto.setUnitCode_dictText(requestBaseDto.getUnitCode());
}
} }
return R.ok(requestBaseInfo); return R.ok(requestBaseInfo);
} }
@@ -2551,17 +2479,21 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
} }
} }
// 使用 MyBatis Plus 分页查询 // 使用直接 LIMIT 查询,不触发 MyBatis Plus 的 COUNT 开销
IPage<SurgeryItemDto> result = doctorStationAdviceAppMapper.getSurgeryPage( List<SurgeryItemDto> records = doctorStationAdviceAppMapper.getSurgeryPage(
new Page<>(pageNo, pageSize), new Page<>(pageNo, pageSize),
PublicationStatus.ACTIVE.getValue(), PublicationStatus.ACTIVE.getValue(),
organizationId, organizationId,
searchKey); searchKey);
// 手动构造 Page 对象total 设为 records.size()(前端 el-transfer 不需要精确的 total 总数)
IPage<SurgeryItemDto> result = new com.baomidou.mybatisplus.extension.plugins.pagination.Page<>(pageNo, pageSize);
result.setRecords(records);
result.setTotal(records.size());
log.info("getSurgeryPage 完成: {}ms, total={}, records={}", System.currentTimeMillis() - start, result.getTotal(), result.getRecords().size()); log.info("getSurgeryPage 完成: {}ms, total={}, records={}", System.currentTimeMillis() - start, result.getTotal(), result.getRecords().size());
// 无搜索时将结果写入缓存 // 无搜索时将结果写入缓存
if (useCache && result instanceof com.baomidou.mybatisplus.extension.plugins.pagination.Page) { if (useCache) {
redisCache.setCacheObject(cacheKey, result, (int) CACHE_EXPIRE_HOURS, java.util.concurrent.TimeUnit.HOURS); redisCache.setCacheObject(cacheKey, result, (int) CACHE_EXPIRE_HOURS, java.util.concurrent.TimeUnit.HOURS);
log.info("缓存手术项目, key: {}, 过期时间: {} 小时", cacheKey, CACHE_EXPIRE_HOURS); log.info("缓存手术项目, key: {}, 过期时间: {} 小时", cacheKey, CACHE_EXPIRE_HOURS);
} }
@@ -2570,13 +2502,12 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
} }
@Override @Override
public IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey, String categoryCode) { public IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey) {
IPage<SurgeryItemDto> result = doctorStationAdviceAppMapper.getExaminationPage( IPage<SurgeryItemDto> result = doctorStationAdviceAppMapper.getExaminationPage(
new Page<>(pageNo, pageSize), new Page<>(pageNo, pageSize),
PublicationStatus.ACTIVE.getValue(), PublicationStatus.ACTIVE.getValue(),
organizationId, organizationId,
searchKey, searchKey);
categoryCode);
return result; return result;
} }

View File

@@ -29,7 +29,6 @@ import com.openhis.document.service.IEmrTemplateService;
import com.openhis.web.doctorstation.appservice.IDoctorStationEmrAppService; import com.openhis.web.doctorstation.appservice.IDoctorStationEmrAppService;
import com.openhis.web.doctorstation.dto.EmrTemplateDto; import com.openhis.web.doctorstation.dto.EmrTemplateDto;
import com.openhis.web.doctorstation.dto.PatientEmrDto; import com.openhis.web.doctorstation.dto.PatientEmrDto;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeanUtils; import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
@@ -42,7 +41,6 @@ import java.util.stream.Collectors;
/** /**
* 医生站-电子病历 应用实现类 * 医生站-电子病历 应用实现类
*/ */
@Slf4j
@Service @Service
public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppService { public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppService {
@@ -62,7 +60,13 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
IDocRecordService docRecordService; IDocRecordService docRecordService;
@Resource @Resource
private com.openhis.web.doctorstation.mapper.DoctorStationEmrAppMapper doctorStationEmrAppMapper; private EncounterMapper encounterMapper;
@Resource
private PatientMapper patientMapper;
@Resource
private com.openhis.administration.mapper.EncounterParticipantMapper encounterParticipantMapper;
/** /**
* 添加病人病历信息 * 添加病人病历信息
@@ -219,35 +223,52 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
* @return 待写病历列表 * @return 待写病历列表
*/ */
@Override @Override
public R<?> getPendingEmrList(Long doctorId, Integer pageNo, Integer pageSize, String patientName) { public R<?> getPendingEmrList(Long doctorId) {
List<Map<String, Object>> allRows = doctorStationEmrAppMapper.getPendingEmrList(doctorId, patientName); // 由于Encounter实体中没有jzPractitionerUserId字段我们需要通过关联查询来获取相关信息
int total = allRows.size(); // 使用医生工作站的mapper来查询相关数据
// 这里我们直接使用医生工作站的查询逻辑
// 分页截取 // 查询当前医生负责的、状态为"就诊中"但还没有写病历的患者
int fromIndex = (pageNo - 1) * pageSize; // 需要通过EncounterParticipant表来关联医生信息
int toIndex = Math.min(fromIndex + pageSize, total); List<Encounter> encounters = encounterMapper.selectList(
List<Map<String, Object>> pageRows; new LambdaQueryWrapper<Encounter>()
if (fromIndex >= total) { .eq(Encounter::getStatusEnum, EncounterStatus.IN_PROGRESS.getValue())
pageRows = new ArrayList<>(); );
} else {
pageRows = allRows.subList(fromIndex, toIndex);
}
// 计算年龄列 // 过滤出由指定医生负责且还没有写病历的就诊记录
for (Map<String, Object> row : pageRows) { List<Map<String, Object>> pendingEmrs = new ArrayList<>();
Object birthDate = row.get("birthDate"); for (Encounter encounter : encounters) {
if (birthDate instanceof Date) { // 检查该就诊记录是否已经有病历
row.put("age", calculateAge((Date) birthDate)); Emr existingEmr = emrService.getOne(
} else { new LambdaQueryWrapper<Emr>().eq(Emr::getEncounterId, encounter.getId())
row.put("age", null); );
// 检查该就诊是否由指定医生负责
boolean isAssignedToDoctor = isEncounterAssignedToDoctor(encounter.getId(), doctorId);
if (existingEmr == null && isAssignedToDoctor) {
// 如果没有病历且由该医生负责,则添加到待写病历列表
Map<String, Object> pendingEmr = new java.util.HashMap<>();
// 获取患者信息
Patient patient = patientMapper.selectById(encounter.getPatientId());
pendingEmr.put("encounterId", encounter.getId());
pendingEmr.put("patientId", encounter.getPatientId());
pendingEmr.put("patientName", patient != null ? patient.getName() : "未知");
pendingEmr.put("gender", patient != null ? patient.getGenderEnum() : null);
// 使用出生日期计算年龄
pendingEmr.put("age", patient != null && patient.getBirthDate() != null ?
calculateAge(patient.getBirthDate()) : null);
// 使用创建时间作为挂号时间
pendingEmr.put("registerTime", encounter.getCreateTime());
pendingEmr.put("busNo", encounter.getBusNo()); // 病历号
pendingEmrs.add(pendingEmr);
} }
row.remove("birthDate");
} }
Map<String, Object> result = new java.util.HashMap<>(); return R.ok(pendingEmrs);
result.put("rows", pageRows);
result.put("total", total);
return R.ok(result);
} }
/** /**
@@ -257,9 +278,14 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
* @return 待写病历数量 * @return 待写病历数量
*/ */
@Override @Override
public R<?> getPendingEmrCount(Long doctorId, String patientName) { public R<?> getPendingEmrCount(Long doctorId) {
Long count = doctorStationEmrAppMapper.getPendingEmrCount(doctorId, patientName); // 获取待写病历列表,然后返回数量
return R.ok(count != null ? count.intValue() : 0); R<?> result = getPendingEmrList(doctorId);
if (result.getCode() == 200) {
List<?> pendingEmrs = (List<?>) result.getData();
return R.ok(pendingEmrs.size());
}
return R.ok(0);
} }
/** /**
@@ -280,6 +306,24 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
return R.ok(needWrite); return R.ok(needWrite);
} }
/**
* 检查就诊是否分配给指定医生
*
* @param encounterId 就诊ID
* @param doctorId 医生ID
* @return 是否分配给指定医生
*/
private boolean isEncounterAssignedToDoctor(Long encounterId, Long doctorId) {
// 查询就诊参与者表,检查是否有指定医生的接诊记录
com.openhis.administration.domain.EncounterParticipant participant =
encounterParticipantMapper.selectOne(
new com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper<com.openhis.administration.domain.EncounterParticipant>()
.eq(com.openhis.administration.domain.EncounterParticipant::getEncounterId, encounterId)
.eq(com.openhis.administration.domain.EncounterParticipant::getPractitionerId, doctorId)
);
return participant != null;
}
/** /**
* 根据出生日期计算年龄 * 根据出生日期计算年龄

View File

@@ -226,9 +226,8 @@ public class DoctorStationAdviceController {
@RequestParam(value = "organizationId", required = false) Long organizationId, @RequestParam(value = "organizationId", required = false) Long organizationId,
@RequestParam(value = "pageNo", defaultValue = "1") Integer pageNo, @RequestParam(value = "pageNo", defaultValue = "1") Integer pageNo,
@RequestParam(value = "pageSize", defaultValue = "500") Integer pageSize, @RequestParam(value = "pageSize", defaultValue = "500") Integer pageSize,
@RequestParam(value = "searchKey", defaultValue = "") String searchKey, @RequestParam(value = "searchKey", defaultValue = "") String searchKey) {
@RequestParam(value = "categoryCode", defaultValue = "23") String categoryCode) { return R.ok(iDoctorStationAdviceAppService.getExaminationPage(organizationId, pageNo, pageSize, searchKey));
return R.ok(iDoctorStationAdviceAppService.getExaminationPage(organizationId, pageNo, pageSize, searchKey, categoryCode));
} }
} }

View File

@@ -26,36 +26,34 @@ public class PendingEmrController {
* 获取待写病历列表 * 获取待写病历列表
* *
* @param doctorId 医生ID * @param doctorId 医生ID
* @param pageNo 当前页码 * @return 待写病历列表
* @param pageSize 每页条数
* @param patientName 患者姓名(可选)
* @return 待写病历分页数据
*/ */
@GetMapping("/pending-list") @GetMapping("/pending-list")
public R<?> getPendingEmrList(@RequestParam(required = false) Long doctorId, public R<?> getPendingEmrList(@RequestParam(required = false) Long doctorId) {
@RequestParam(defaultValue = "1") Integer pageNum, // 如果没有传递医生ID则使用当前登录用户ID
@RequestParam(defaultValue = "10") Integer pageSize,
@RequestParam(required = false) String patientName) {
if (doctorId == null) { if (doctorId == null) {
doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getPractitionerId(); doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getUserId();
} }
return iDoctorStationEmrAppService.getPendingEmrList(doctorId, pageNum, pageSize, patientName);
// 调用服务获取待写病历列表
return iDoctorStationEmrAppService.getPendingEmrList(doctorId);
} }
/** /**
* 获取待写病历数量 * 获取待写病历数量
* *
* @param doctorId 医生ID * @param doctorId 医生ID
* @param patientName 患者姓名(可选)
* @return 待写病历数量 * @return 待写病历数量
*/ */
@GetMapping("/pending-count") @GetMapping("/pending-count")
public R<?> getPendingEmrCount(@RequestParam(required = false) Long doctorId, public R<?> getPendingEmrCount(@RequestParam(required = false) Long doctorId) {
@RequestParam(required = false) String patientName) { // 如果没有传递医生ID则使用当前登录用户ID
if (doctorId == null) { if (doctorId == null) {
doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getPractitionerId(); doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getUserId();
} }
return iDoctorStationEmrAppService.getPendingEmrCount(doctorId, patientName);
// 调用服务获取待写病历数量
return iDoctorStationEmrAppService.getPendingEmrCount(doctorId);
} }
/** /**

View File

@@ -96,11 +96,6 @@ public class DiagnosisQueryDto {
*/ */
private String diagnosisDoctor; private String diagnosisDoctor;
/**
* 长效诊断标识
*/
private Integer longTermFlag;
/** /**
* 是否已有传染病报卡0-无1-有) * 是否已有传染病报卡0-无1-有)
*/ */

View File

@@ -23,9 +23,6 @@ public class SurgeryItemDto {
@JsonSerialize(using = ToStringSerializer.class) @JsonSerialize(using = ToStringSerializer.class)
private Long orgId; private Long orgId;
/** 所属科室名称 */
private String orgName;
/** 执行科室ID */ /** 执行科室ID */
@JsonSerialize(using = ToStringSerializer.class) @JsonSerialize(using = ToStringSerializer.class)
private Long positionId; private Long positionId;

View File

@@ -195,7 +195,7 @@ public interface DoctorStationAdviceAppMapper {
* @param searchKey 模糊查询关键字(可选) * @param searchKey 模糊查询关键字(可选)
* @return 手术项目分页数据 * @return 手术项目分页数据
*/ */
IPage<SurgeryItemDto> getSurgeryPage(@Param("page") Page<SurgeryItemDto> page, List<SurgeryItemDto> getSurgeryPage(@Param("page") Page<SurgeryItemDto> page,
@Param("statusEnum") Integer statusEnum, @Param("statusEnum") Integer statusEnum,
@Param("organizationId") Long organizationId, @Param("organizationId") Long organizationId,
@Param("searchKey") String searchKey); @Param("searchKey") String searchKey);
@@ -203,7 +203,6 @@ public interface DoctorStationAdviceAppMapper {
IPage<SurgeryItemDto> getExaminationPage(@Param("page") Page<SurgeryItemDto> page, IPage<SurgeryItemDto> getExaminationPage(@Param("page") Page<SurgeryItemDto> page,
@Param("statusEnum") Integer statusEnum, @Param("statusEnum") Integer statusEnum,
@Param("organizationId") Long organizationId, @Param("organizationId") Long organizationId,
@Param("searchKey") String searchKey, @Param("searchKey") String searchKey);
@Param("categoryCode") String categoryCode);
} }

View File

@@ -1,20 +1,11 @@
package com.openhis.web.doctorstation.mapper; package com.openhis.web.doctorstation.mapper;
import org.apache.ibatis.annotations.Param;
import org.springframework.stereotype.Repository; import org.springframework.stereotype.Repository;
import java.util.List;
import java.util.Map;
/** /**
* 医生站-电子病历 应用Mapper * 医生站-电子病历 应用Mapper
*/ */
@Repository @Repository
public interface DoctorStationEmrAppMapper { public interface DoctorStationEmrAppMapper {
List<Map<String, Object>> getPendingEmrList(@Param("doctorId") Long doctorId,
@Param("patientName") String patientName);
Long getPendingEmrCount(@Param("doctorId") Long doctorId,
@Param("patientName") String patientName);
} }

View File

@@ -359,24 +359,6 @@ public class AdviceProcessAppServiceImpl implements IAdviceProcessAppService {
medRequestList.add(item); medRequestList.add(item);
} }
} }
// 校验医嘱是否已执行,已执行的医嘱需要先取消执行后才能退回
List<Long> allRequestIds = performInfoList.stream().map(PerformInfoDto::getRequestId).toList();
List<Procedure> allProcedures = procedureService.list(
new LambdaQueryWrapper<Procedure>()
.in(Procedure::getRequestId, allRequestIds)
.eq(Procedure::getDeleteFlag, "0"));
Set<Long> executedIds = allProcedures.stream()
.filter(p -> EventStatus.COMPLETED.getValue().equals(p.getStatusEnum()))
.map(Procedure::getId)
.collect(Collectors.toSet());
Set<Long> cancelledRefundIds = allProcedures.stream()
.filter(p -> EventStatus.CANCEL.getValue().equals(p.getStatusEnum()) && p.getRefundId() != null)
.map(Procedure::getRefundId)
.collect(Collectors.toSet());
executedIds.removeAll(cancelledRefundIds);
if (!executedIds.isEmpty()) {
return R.fail("该医嘱已执行,请先取消执行后再退回");
}
// 校验药品医嘱是否已发药,已发药的医嘱不允许退回 // 校验药品医嘱是否已发药,已发药的医嘱不允许退回
if (!medRequestList.isEmpty()) { if (!medRequestList.isEmpty()) {
List<Long> medReqIds = medRequestList.stream().map(PerformInfoDto::getRequestId).toList(); List<Long> medReqIds = medRequestList.stream().map(PerformInfoDto::getRequestId).toList();

View File

@@ -32,7 +32,6 @@ import org.springframework.stereotype.Service;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpServletResponse;
import java.math.BigDecimal;
import java.util.*; import java.util.*;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import java.util.stream.Stream; import java.util.stream.Stream;
@@ -203,70 +202,62 @@ public class ProductTransferAppServiceImpl implements IProductTransferAppService
@Override @Override
public R<?> addOrEditBatchTransferReceipt(List<ProductTransferDto> productTransferDtoList, Boolean flag) { public R<?> addOrEditBatchTransferReceipt(List<ProductTransferDto> productTransferDtoList, Boolean flag) {
// 校验调拨数量:必须 > 0 且不超过源库存数量(从数据库查实时库存)
Integer tenantId = SecurityUtils.getLoginUser().getTenantId(); Integer tenantId = SecurityUtils.getLoginUser().getTenantId();
Date now = DateUtils.getNowDate(); for (ProductTransferDto dto : productTransferDtoList) {
if (dto.getItemQuantity() == null || dto.getItemQuantity().compareTo(java.math.BigDecimal.ZERO) <= 0) {
return R.fail("调拨数量必须大于0");
}
// 查询该药品在源仓库的实时库存总量
List<InventoryItem> inventoryList = inventoryItemService.selectInventoryByItemId(
dto.getItemId(), dto.getLotNumber(), dto.getSourceLocationId(), tenantId);
java.math.BigDecimal actualStock = inventoryList.stream()
.map(InventoryItem::getQuantity)
.reduce(java.math.BigDecimal.ZERO, java.math.BigDecimal::add);
if (dto.getItemQuantity().compareTo(actualStock) > 0) {
return R.fail("调拨数量不可超出源库存数量(当前库存:" + actualStock + "");
}
}
List<String> idList = new ArrayList<>(); List<String> idList = new ArrayList<>();
if (flag) { if (flag) {
// 批量保存按钮 // 批量保存按钮
// 单据号取得
List<String> busNoList = productTransferDtoList.stream().map(ProductTransferDto::getBusNo).toList(); List<String> busNoList = productTransferDtoList.stream().map(ProductTransferDto::getBusNo).toList();
// 保存前:获取旧记录用于恢复预划扣库存 // 请求id取得
List<SupplyRequest> oldRequestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0)); List<SupplyRequest> requestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
if (!oldRequestList.isEmpty()) { if (!requestList.isEmpty()) {
// 恢复旧记录已预划扣的库存 List<Long> requestIdList = requestList.stream().map(SupplyRequest::getId).collect(Collectors.toList());
for (SupplyRequest oldReq : oldRequestList) { // 单据信息删除
if (oldReq.getItemId() != null && oldReq.getLotNumber() != null supplyRequestService.removeByIds(requestIdList);
&& oldReq.getSourceLocationId() != null && oldReq.getItemQuantity() != null) {
List<InventoryItem> invList = inventoryItemService.selectInventoryByItemId(
oldReq.getItemId(), oldReq.getLotNumber(), oldReq.getSourceLocationId(), tenantId);
if (!invList.isEmpty()) {
inventoryItemService.updateInventoryQuantity(
invList.get(0).getId(),
invList.get(0).getQuantity().add(oldReq.getItemQuantity()), now);
}
}
}
List<Long> oldIdList = oldRequestList.stream().map(SupplyRequest::getId).collect(Collectors.toList());
supplyRequestService.removeByIds(oldIdList);
}
// 校验 + 预划扣新记录
for (ProductTransferDto dto : productTransferDtoList) {
if (dto.getItemQuantity() == null || dto.getItemQuantity().compareTo(BigDecimal.ZERO) <= 0) {
return R.fail("调拨数量必须大于0");
}
List<InventoryItem> inventoryList = inventoryItemService.selectInventoryByItemId(
dto.getItemId(), dto.getLotNumber(), dto.getSourceLocationId(), tenantId);
BigDecimal actualStock = inventoryList.stream()
.map(InventoryItem::getQuantity)
.reduce(BigDecimal.ZERO, BigDecimal::add);
if (dto.getItemQuantity().compareTo(actualStock) > 0) {
return R.fail("调拨数量不可超出源库存数量(当前库存:" + actualStock + "");
}
// 预划扣源仓库库存
if (!inventoryList.isEmpty()) {
InventoryItem inv = inventoryList.get(0);
inventoryItemService.updateInventoryQuantity(
inv.getId(), inv.getQuantity().subtract(dto.getItemQuantity()), now);
}
} }
// 生成批量调拨单据 // 生成批量调拨单据
List<SupplyRequest> supplyRequestList = new ArrayList<>(); List<SupplyRequest> supplyRequestList = new ArrayList<>();
for (ProductTransferDto productTransferDto : productTransferDtoList) { for (ProductTransferDto productTransferDto : productTransferDtoList) {
// 初始化单据信息
SupplyRequest supplyRequest = new SupplyRequest(); SupplyRequest supplyRequest = new SupplyRequest();
BeanUtils.copyProperties(productTransferDto, supplyRequest); BeanUtils.copyProperties(productTransferDto, supplyRequest);
// 生成商品批量调拨单据
supplyRequest supplyRequest
// id
.setId(null) .setId(null)
// 单据分类:库存供应
.setCategoryEnum(SupplyCategory.STOCK_SUPPLY.getValue()) .setCategoryEnum(SupplyCategory.STOCK_SUPPLY.getValue())
// 单据类型:商品批量调拨
.setTypeEnum(SupplyType.PRODUCT_BATCH_TRANSFER.getValue()) .setTypeEnum(SupplyType.PRODUCT_BATCH_TRANSFER.getValue())
// 制单人
.setApplicantId(SecurityUtils.getLoginUser().getPractitionerId()) .setApplicantId(SecurityUtils.getLoginUser().getPractitionerId())
// 申请时间
.setApplyTime(DateUtils.getNowDate()) .setApplyTime(DateUtils.getNowDate())
// 源库存数量
.setTotalQuantity(productTransferDto.getTotalSourceQuantity()); .setTotalQuantity(productTransferDto.getTotalSourceQuantity());
supplyRequestList.add(supplyRequest); supplyRequestList.add(supplyRequest);
} }
supplyRequestService.saveOrUpdateBatch(supplyRequestList); supplyRequestService.saveOrUpdateBatch(supplyRequestList);
// 请求id取得
List<SupplyRequest> supplyRequestIdList = supplyRequestService.getSupplyByBusNo(busNoList.get(0)); List<SupplyRequest> supplyRequestIdList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
// 返回请求id列表
List<Long> requestIdList = supplyRequestIdList.stream().map(SupplyRequest::getId).toList(); List<Long> requestIdList = supplyRequestIdList.stream().map(SupplyRequest::getId).toList();
for (Long list : requestIdList) { for (Long list : requestIdList) {
idList.add(list.toString()); idList.add(list.toString());
@@ -274,58 +265,33 @@ public class ProductTransferAppServiceImpl implements IProductTransferAppService
} else { } else {
// 单独保存按钮 // 单独保存按钮
for (ProductTransferDto productTransferDto : productTransferDtoList) { for (ProductTransferDto productTransferDto : productTransferDtoList) {
// 更新已有记录:先恢复旧预划扣,再扣新的 // 初始化单据信息
if (productTransferDto.getId() != null) {
SupplyRequest oldReq = supplyRequestService.getById(productTransferDto.getId());
if (oldReq != null && oldReq.getItemId() != null && oldReq.getLotNumber() != null
&& oldReq.getSourceLocationId() != null && oldReq.getItemQuantity() != null) {
List<InventoryItem> invList = inventoryItemService.selectInventoryByItemId(
oldReq.getItemId(), oldReq.getLotNumber(), oldReq.getSourceLocationId(), tenantId);
if (!invList.isEmpty()) {
inventoryItemService.updateInventoryQuantity(
invList.get(0).getId(),
invList.get(0).getQuantity().add(oldReq.getItemQuantity()), now);
}
}
}
// 校验 + 预划扣
if (productTransferDto.getItemQuantity() == null
|| productTransferDto.getItemQuantity().compareTo(BigDecimal.ZERO) <= 0) {
return R.fail("调拨数量必须大于0");
}
List<InventoryItem> inventoryList = inventoryItemService.selectInventoryByItemId(
productTransferDto.getItemId(), productTransferDto.getLotNumber(),
productTransferDto.getSourceLocationId(), tenantId);
BigDecimal actualStock = inventoryList.stream()
.map(InventoryItem::getQuantity)
.reduce(BigDecimal.ZERO, BigDecimal::add);
if (productTransferDto.getItemQuantity().compareTo(actualStock) > 0) {
return R.fail("调拨数量不可超出源库存数量(当前库存:" + actualStock + "");
}
if (!inventoryList.isEmpty()) {
InventoryItem inv = inventoryList.get(0);
inventoryItemService.updateInventoryQuantity(
inv.getId(), inv.getQuantity().subtract(productTransferDto.getItemQuantity()), now);
}
SupplyRequest supplyRequest = new SupplyRequest(); SupplyRequest supplyRequest = new SupplyRequest();
BeanUtils.copyProperties(productTransferDto, supplyRequest); BeanUtils.copyProperties(productTransferDto, supplyRequest);
supplyRequest.setTotalQuantity(productTransferDto.getTotalSourceQuantity()); supplyRequest.setTotalQuantity(productTransferDto.getTotalSourceQuantity());
if (productTransferDto.getId() != null) { if (productTransferDto.getId() != null) {
// 更新单据信息
supplyRequestService.updateById(supplyRequest); supplyRequestService.updateById(supplyRequest);
} else { } else {
// 生成商品批量调拨单据
supplyRequest supplyRequest
// 单据分类:库存供应
.setCategoryEnum(SupplyCategory.STOCK_SUPPLY.getValue()) .setCategoryEnum(SupplyCategory.STOCK_SUPPLY.getValue())
// 单据类型:商品批量调拨
.setTypeEnum(SupplyType.PRODUCT_BATCH_TRANSFER.getValue()) .setTypeEnum(SupplyType.PRODUCT_BATCH_TRANSFER.getValue())
// 制单人
.setApplicantId(SecurityUtils.getLoginUser().getPractitionerId()) .setApplicantId(SecurityUtils.getLoginUser().getPractitionerId())
// 申请时间
.setApplyTime(DateUtils.getNowDate()); .setApplyTime(DateUtils.getNowDate());
supplyRequestService.save(supplyRequest); supplyRequestService.save(supplyRequest);
} }
// 返回单据id
return R.ok(supplyRequest.getId().toString(), null); return R.ok(supplyRequest.getId().toString(), null);
} }
} }
// 返回单据id
return R.ok(idList, null); return R.ok(idList, null);
} }
@@ -366,63 +332,33 @@ public class ProductTransferAppServiceImpl implements IProductTransferAppService
@Override @Override
public R<?> addOrEditTransferReceipt(List<ProductTransferDto> productTransferDtoList) { public R<?> addOrEditTransferReceipt(List<ProductTransferDto> productTransferDtoList) {
// 校验调拨数量:必须 > 0 且不超过源库存数量(从数据库查实时库存)
Integer tenantId = SecurityUtils.getLoginUser().getTenantId(); Integer tenantId = SecurityUtils.getLoginUser().getTenantId();
Date now = DateUtils.getNowDate(); for (ProductTransferDto dto : productTransferDtoList) {
if (dto.getItemQuantity() == null || dto.getItemQuantity().compareTo(java.math.BigDecimal.ZERO) <= 0) {
return R.fail("调拨数量必须大于0");
}
List<InventoryItem> inventoryList = inventoryItemService.selectInventoryByItemId(
dto.getItemId(), dto.getLotNumber(), dto.getSourceLocationId(), tenantId);
java.math.BigDecimal actualStock = inventoryList.stream()
.map(InventoryItem::getQuantity)
.reduce(java.math.BigDecimal.ZERO, java.math.BigDecimal::add);
if (dto.getItemQuantity().compareTo(actualStock) > 0) {
return R.fail("调拨数量不可超出源库存数量(当前库存:" + actualStock + "");
}
}
List<String> idList = new ArrayList<>(); List<String> idList = new ArrayList<>();
// 单据号取得 // 单据号取得
List<String> busNoList = productTransferDtoList.stream().map(ProductTransferDto::getBusNo).toList(); List<String> busNoList = productTransferDtoList.stream().map(ProductTransferDto::getBusNo).toList();
// 保存前:获取旧记录用于恢复预划扣库存 // 请求数据取得
List<SupplyRequest> oldRequestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0)); List<SupplyRequest> requestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
Map<String, BigDecimal> oldDeductionMap = new HashMap<>(); if (!requestList.isEmpty()) {
if (!oldRequestList.isEmpty()) { // 请求id取得
for (SupplyRequest oldReq : oldRequestList) { List<Long> requestIdList = requestList.stream().map(SupplyRequest::getId).collect(Collectors.toList());
if (oldReq.getItemId() != null && oldReq.getLotNumber() != null && oldReq.getSourceLocationId() != null // 单据信息删除
&& oldReq.getItemQuantity() != null) { supplyRequestService.removeByIds(requestIdList);
String key = oldReq.getItemId() + "_" + oldReq.getLotNumber() + "_" + oldReq.getSourceLocationId();
oldDeductionMap.merge(key, oldReq.getItemQuantity(), BigDecimal::add);
}
}
// 恢复旧记录已预划扣的库存
for (Map.Entry<String, BigDecimal> entry : oldDeductionMap.entrySet()) {
String[] parts = entry.getKey().split("_");
Long itemId = Long.parseLong(parts[0]);
String lotNumber = parts[1];
Long sourceLocationId = Long.parseLong(parts[2]);
BigDecimal restoreQty = entry.getValue();
List<InventoryItem> invList = inventoryItemService.selectInventoryByItemId(
itemId, lotNumber, sourceLocationId, tenantId);
if (!invList.isEmpty()) {
inventoryItemService.updateInventoryQuantity(
invList.get(0).getId(), invList.get(0).getQuantity().add(restoreQty), now);
}
}
// 删除旧记录
List<Long> oldIdList = oldRequestList.stream().map(SupplyRequest::getId).collect(Collectors.toList());
supplyRequestService.removeByIds(oldIdList);
}
// 校验 + 预划扣新记录
Map<String, BigDecimal> newDeductionMap = new HashMap<>();
for (ProductTransferDto dto : productTransferDtoList) {
if (dto.getItemQuantity() == null || dto.getItemQuantity().compareTo(BigDecimal.ZERO) <= 0) {
return R.fail("调拨数量必须大于0");
}
List<InventoryItem> inventoryList = inventoryItemService.selectInventoryByItemId(
dto.getItemId(), dto.getLotNumber(), dto.getSourceLocationId(), tenantId);
BigDecimal actualStock = inventoryList.stream()
.map(InventoryItem::getQuantity)
.reduce(BigDecimal.ZERO, BigDecimal::add);
if (dto.getItemQuantity().compareTo(actualStock) > 0) {
return R.fail("调拨数量不可超出源库存数量(当前库存:" + actualStock + "");
}
// 预划扣:扣减源仓库库存
if (!inventoryList.isEmpty()) {
InventoryItem inv = inventoryList.get(0);
inventoryItemService.updateInventoryQuantity(
inv.getId(), inv.getQuantity().subtract(dto.getItemQuantity()), now);
}
} }
List<SupplyRequest> supplyRequestList = new ArrayList<>(); List<SupplyRequest> supplyRequestList = new ArrayList<>();
@@ -469,22 +405,6 @@ public class ProductTransferAppServiceImpl implements IProductTransferAppService
*/ */
@Override @Override
public R<?> deleteReceipt(List<Long> supplyRequestIds) { public R<?> deleteReceipt(List<Long> supplyRequestIds) {
// 删除前恢复预划扣的库存
Integer tenantId = SecurityUtils.getLoginUser().getTenantId();
Date now = DateUtils.getNowDate();
for (Long reqId : supplyRequestIds) {
SupplyRequest sr = supplyRequestService.getById(reqId);
if (sr != null && sr.getItemId() != null && sr.getSourceLocationId() != null
&& sr.getItemQuantity() != null && sr.getLotNumber() != null) {
List<InventoryItem> invList = inventoryItemService.selectInventoryByItemId(
sr.getItemId(), sr.getLotNumber(), sr.getSourceLocationId(), tenantId);
if (!invList.isEmpty()) {
InventoryItem inv = invList.get(0);
inventoryItemService.updateInventoryQuantity(
inv.getId(), inv.getQuantity().add(sr.getItemQuantity()), now);
}
}
}
// 删除单据 // 删除单据
boolean result = supplyRequestService.removeByIds(supplyRequestIds); boolean result = supplyRequestService.removeByIds(supplyRequestIds);
return result ? R.ok(null, MessageUtils.createMessage(PromptMsgConstant.Common.M00004, null)) return result ? R.ok(null, MessageUtils.createMessage(PromptMsgConstant.Common.M00004, null))

View File

@@ -519,8 +519,58 @@ public class ReceiptApprovalAppServiceImpl implements IReceiptApprovalAppService
// 暂时先取出全部的库存,循环查库存同一会有问题,后续优化 // 暂时先取出全部的库存,循环查库存同一会有问题,后续优化
List<InventoryItem> inventoryItems = inventoryItemService.selectAllInventory(); List<InventoryItem> inventoryItems = inventoryItemService.selectAllInventory();
for (SupplyItemDetailDto supplyItemDetailDto : supplyItemDetailList) { for (SupplyItemDetailDto supplyItemDetailDto : supplyItemDetailList) {
// 🔧 源仓库库存已在保存时预划扣,审批通过时不再重复扣减 // 根据项目id,产品批号源仓库id 查询源仓库库存表信息
outList.add(supplyItemDetailDto); // List<InventoryItem> inventoryItemSourceList = inventoryItemService.selectInventoryByItemId(
// supplyItemDetailDto.getItemId(), supplyItemDetailDto.getLotNumber(),
// supplyItemDetailDto.getSourceLocationId(), SecurityUtils.getLoginUser().getTenantId());
List<InventoryItem> filteredInventoryItems = inventoryItems.stream()
.filter(item -> item.getItemId().equals(supplyItemDetailDto.getItemId())
&& item.getLotNumber().equals(supplyItemDetailDto.getLotNumber())
&& item.getLocationId().equals(supplyItemDetailDto.getSourceLocationId()))
.collect(Collectors.toList());
InventoryItem inventoryItemSource = new InventoryItem();
if (!filteredInventoryItems.isEmpty()) {
inventoryItemSource = filteredInventoryItems.get(0);
// 最小数量(最小单位库存数量)
BigDecimal minQuantity = inventoryItemSource.getQuantity();
// // 计算调拨后库存数量,结果取小单位
// // 供应申请的物品计量单位与包装单位相同
// if (supplyItemDetailDto.getItemUnit().equals(supplyItemDetailDto.getUnitCode())) {
// if
// (minQuantity.compareTo(supplyItemDetailDto.getItemQuantity().multiply(supplyItemDetailDto.getPartPercent()))
// < 0) {
// // 库存数量不足
// throw new ServiceException("操作失败,库存数量不足");
// } else {
// // 源仓库库存-(调拨数量*拆零比)
// minQuantity = minQuantity.subtract(
// supplyItemDetailDto.getPartPercent().multiply(supplyItemDetailDto.getItemQuantity()));
// }
// } else if (supplyItemDetailDto.getItemUnit().equals(supplyItemDetailDto.getMinUnitCode())) {
// 直接扣减库存
if (minQuantity.compareTo(supplyItemDetailDto.getItemQuantity()) < 0) {
// 库存数量不足
throw new ServiceException("操作失败,库存数量不足");
} else {
// 供应申请的物品计量单位与最小单位相同
// 源仓库库存-调拨数量
minQuantity = minQuantity.subtract(supplyItemDetailDto.getItemQuantity());
}
// }
// 更新源仓库库存数量
Boolean aBoolean
= inventoryItemService.updateInventoryQuantity(inventoryItemSource.getId(), minQuantity, now);
if (!aBoolean) {
throw new ServiceException("系统异常,请稍后重试");
}
// 添加到出库列表
outList.add(supplyItemDetailDto);
} else {
return R.fail(MessageUtils.createMessage(PromptMsgConstant.Common.M00007, null));
}
// 根据项目id,产品批号目的仓库id 查询目的仓库库存表信息 // 根据项目id,产品批号目的仓库id 查询目的仓库库存表信息
List<InventoryItem> inventoryItemPurposeList = inventoryItemService.selectInventoryByItemId( List<InventoryItem> inventoryItemPurposeList = inventoryItemService.selectInventoryByItemId(

View File

@@ -133,13 +133,47 @@ public class PatientInformationServiceImpl implements IPatientInformationService
@Override @Override
public IPage<PatientBaseInfoDto> getPatientInfo(PatientBaseInfoDto patientBaseInfoDto, String searchKey, public IPage<PatientBaseInfoDto> getPatientInfo(PatientBaseInfoDto patientBaseInfoDto, String searchKey,
Integer pageNo, Integer pageSize, HttpServletRequest request) { Integer pageNo, Integer pageSize, HttpServletRequest request) {
// 构建基础查询条件 // 获取登录者信息
LoginUser loginUser = SecurityUtils.getLoginUser(); LoginUser loginUser = SecurityUtils.getLoginUser();
Long userId = loginUser.getUserId();
Integer tenantId = loginUser.getTenantId().intValue();
// 先构建基础查询条件
QueryWrapper<PatientBaseInfoDto> queryWrapper = HisQueryUtils.buildQueryWrapper( QueryWrapper<PatientBaseInfoDto> queryWrapper = HisQueryUtils.buildQueryWrapper(
patientBaseInfoDto, searchKey, new HashSet<>(Arrays.asList(CommonConstants.FieldName.Name, patientBaseInfoDto, searchKey, new HashSet<>(Arrays.asList(CommonConstants.FieldName.Name,
CommonConstants.FieldName.BusNo, CommonConstants.FieldName.PyStr, CommonConstants.FieldName.WbStr)), CommonConstants.FieldName.BusNo, CommonConstants.FieldName.PyStr, CommonConstants.FieldName.WbStr)),
request); request);
// 检查是否是精确ID查询从门诊挂号页面跳转时使用
boolean hasExactIdQuery = (patientBaseInfoDto.getId() != null);
// 只有非精确ID查询时才添加医生患者过滤条件
if (!hasExactIdQuery) {
// 查询当前用户对应的医生信息
LambdaQueryWrapper<com.openhis.administration.domain.Practitioner> practitionerQuery = new LambdaQueryWrapper<>();
practitionerQuery.eq(com.openhis.administration.domain.Practitioner::getUserId, userId);
// 使用list()避免TooManyResultsException异常然后取第一个记录
List<com.openhis.administration.domain.Practitioner> practitionerList = practitionerService.list(practitionerQuery);
com.openhis.administration.domain.Practitioner practitioner = practitionerList != null && !practitionerList.isEmpty() ? practitionerList.get(0) : null;
// 如果当前用户是医生,添加医生患者过滤条件
if (practitioner != null) {
// 查询该医生作为接诊医生ADMITTER, code="1"和挂号医生REGISTRATION_DOCTOR, code="12"的所有就诊记录的患者ID
List<Long> doctorPatientIds = patientManageMapper.getPatientIdsByPractitionerId(
practitioner.getId(),
Arrays.asList(ParticipantType.ADMITTER.getCode(), ParticipantType.REGISTRATION_DOCTOR.getCode()),
tenantId);
if (doctorPatientIds != null && !doctorPatientIds.isEmpty()) {
// 添加患者ID过滤条件 - 注意:这里使用列名而不是表别名
queryWrapper.in("id", doctorPatientIds);
} else {
// 如果没有相关患者,返回空结果
queryWrapper.eq("id", -1); // 设置一个不存在的ID
}
}
// 如果不是医生,查询所有患者
}
IPage<PatientBaseInfoDto> patientInformationPage IPage<PatientBaseInfoDto> patientInformationPage
= patientManageMapper.getPatientPage(new Page<>(pageNo, pageSize), queryWrapper); = patientManageMapper.getPatientPage(new Page<>(pageNo, pageSize), queryWrapper);
@@ -235,7 +269,7 @@ public class PatientInformationServiceImpl implements IPatientInformationService
// log.debug("添加病人信息,patientInfoDto:{}", patientBaseInfoDto); // log.debug("添加病人信息,patientInfoDto:{}", patientBaseInfoDto);
// 如果患者没有输入身份证号则根据年龄自动生成 // 如果患者没有输入身份证号则根据年龄自动生成
String idCard = patientBaseInfoDto.getIdCard(); String idCard = patientBaseInfoDto.getIdCard();
if (idCard == null || idCard.length() < 6 || CommonConstants.Common.AREA_CODE.equals(idCard.substring(0, 6))) { if (idCard == null || CommonConstants.Common.AREA_CODE.equals(idCard.substring(0, 6))) {
if (patientBaseInfoDto.getAge() != null) { if (patientBaseInfoDto.getAge() != null) {
idCard = IdCardUtil.generateIdByAge(patientBaseInfoDto.getAge()); idCard = IdCardUtil.generateIdByAge(patientBaseInfoDto.getAge());
patientBaseInfoDto.setIdCard(idCard); patientBaseInfoDto.setIdCard(idCard);

View File

@@ -36,8 +36,6 @@ import com.openhis.workflow.domain.ActivityDefinition;
import com.openhis.workflow.service.IDeviceDispenseService; import com.openhis.workflow.service.IDeviceDispenseService;
import com.openhis.workflow.service.IDeviceRequestService; import com.openhis.workflow.service.IDeviceRequestService;
import com.openhis.workflow.service.IServiceRequestService; import com.openhis.workflow.service.IServiceRequestService;
import com.openhis.document.domain.RequestForm;
import com.openhis.document.service.IRequestFormService;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
@@ -87,9 +85,6 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
@Resource @Resource
IDeviceDispenseService iDeviceDispenseService; IDeviceDispenseService iDeviceDispenseService;
@Resource
IRequestFormService iRequestFormService;
/** /**
* 查询住院患者信息 * 查询住院患者信息
* *
@@ -271,38 +266,6 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
log.info("开始处理删除操作,共 {} 条记录", deleteList.size()); log.info("开始处理删除操作,共 {} 条记录", deleteList.size());
// 🔧 手术申请单级联作废:删除手术医嘱(categoryEnum=24)时同步作废关联的手术申请单
Map<String, List<Long>> surgeryPrescriptionNoToRequestIds = new LinkedHashMap<>();
for (RegAdviceSaveDto adviceDto : deleteList) {
if (adviceDto.getRequestId() == null) continue;
ServiceRequest existing = iServiceRequestService.getById(adviceDto.getRequestId());
if (existing == null) continue;
if (existing.getCategoryEnum() != null
&& existing.getCategoryEnum() == 24
&& existing.getPrescriptionNo() != null && !existing.getPrescriptionNo().isEmpty()) {
surgeryPrescriptionNoToRequestIds.computeIfAbsent(existing.getPrescriptionNo(), k -> new ArrayList<>())
.add(adviceDto.getRequestId());
}
}
for (Map.Entry<String, List<Long>> e : surgeryPrescriptionNoToRequestIds.entrySet()) {
String prescriptionNo = e.getKey();
try {
List<RequestForm> requestForms = iRequestFormService.list(
new LambdaQueryWrapper<RequestForm>()
.eq(RequestForm::getPrescriptionNo, prescriptionNo)
.eq(RequestForm::getTypeCode, ActivityDefCategory.PROCEDURE.getCode())
.and(w -> w.isNull(RequestForm::getDeleteFlag).or().eq(RequestForm::getDeleteFlag, "0")));
for (RequestForm requestForm : requestForms) {
iRequestFormService.removeById(requestForm.getId());
}
if (!requestForms.isEmpty()) {
log.info("级联作废手术申请单 prescriptionNo={}, 共{}条", prescriptionNo, requestForms.size());
}
} catch (Exception ex) {
log.warn("级联作废手术申请单失败 prescriptionNo={}", prescriptionNo, ex);
}
}
for (RegAdviceSaveDto adviceDto : deleteList) { for (RegAdviceSaveDto adviceDto : deleteList) {
Integer adviceType = adviceDto.getAdviceType(); Integer adviceType = adviceDto.getAdviceType();
Long requestId = adviceDto.getRequestId(); Long requestId = adviceDto.getRequestId();

View File

@@ -5,7 +5,6 @@ import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage; import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.core.common.core.domain.R; import com.core.common.core.domain.R;
import com.core.common.enums.DelFlag;
import com.core.common.exception.ServiceException; import com.core.common.exception.ServiceException;
import com.core.common.utils.AssignSeqUtil; import com.core.common.utils.AssignSeqUtil;
import com.core.common.utils.MessageUtils; import com.core.common.utils.MessageUtils;
@@ -18,8 +17,6 @@ import com.openhis.common.constant.PromptMsgConstant;
import com.openhis.common.enums.*; import com.openhis.common.enums.*;
import com.openhis.document.domain.RequestForm; import com.openhis.document.domain.RequestForm;
import com.openhis.document.service.IRequestFormService; import com.openhis.document.service.IRequestFormService;
import com.openhis.lab.domain.Specimen;
import com.openhis.lab.service.ISpecimenService;
import com.openhis.web.doctorstation.dto.ActivityChildrenJsonParams; import com.openhis.web.doctorstation.dto.ActivityChildrenJsonParams;
import com.openhis.web.doctorstation.utils.AdviceUtils; import com.openhis.web.doctorstation.utils.AdviceUtils;
import com.openhis.web.regdoctorstation.appservice.IRequestFormManageAppService; import com.openhis.web.regdoctorstation.appservice.IRequestFormManageAppService;
@@ -70,39 +67,6 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
@Resource @Resource
IActivityDefinitionService iActivityDefinitionService; IActivityDefinitionService iActivityDefinitionService;
@Resource
ISpecimenService iSpecimenService;
/**
* 校验当前用户是否有权操作该申请单(申请者本人或管理员)
*/
private R<?> validateRequestFormPermission(RequestForm requestForm) {
if (SecurityUtils.isAdmin(SecurityUtils.getUserId())) {
return null;
}
Long currentPractitionerId = SecurityUtils.getLoginUser().getPractitionerId();
Long requesterId = requestForm.getRequesterId();
if (currentPractitionerId == null || requesterId == null
|| !currentPractitionerId.equals(requesterId)) {
return R.fail("无操作权限,仅申请开立者或管理员可操作");
}
return null;
}
/**
* 校验关联医嘱是否已采证(存在已采集/已接收标本则不可撤回)
*/
private boolean hasCollectedSpecimen(List<Long> serviceRequestIds) {
if (serviceRequestIds == null || serviceRequestIds.isEmpty()) {
return false;
}
long count = iSpecimenService.count(
new LambdaQueryWrapper<Specimen>()
.in(Specimen::getServiceId, serviceRequestIds)
.ge(Specimen::getCollectionStatusEnum, SpecCollectStatus.COLLECTED.getValue()));
return count > 0;
}
/** /**
* 保存申请单 * 保存申请单
* *
@@ -117,7 +81,16 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
Long requestFormId = requestFormSaveDto.getRequestFormId(); Long requestFormId = requestFormSaveDto.getRequestFormId();
boolean isEdit = requestFormId != null && requestFormId != 0L; boolean isEdit = requestFormId != null && requestFormId != 0L;
// 🔧 手术/检查申请单优先使用前端传入的positionId用户手动选择的发往科室跳过执行科室配置校验 // 诊疗执行科室配置校验(必须在任何数据库操作之前)
List<ActivityOrganizationConfigDto> activityOrganizationConfig =
requestFormManageAppMapper.getActivityOrganizationConfig(typeCode);
if (activityOrganizationConfig.isEmpty()) {
throw new ServiceException("请先配置当前时间段的执行科室");
}
// 逐个校验activityList中的项目是否都配置了执行科室并收集positionId供后续使用
// 必须在任何数据库操作之前完成全部校验,避免部分保存后异常导致脏数据
// 🔧 Bug #516: 优先使用前端传入的positionId用户手动选择的发往科室仅在未选择时使用配置的执行科室
List<ActivitySaveDto> activityList = requestFormSaveDto.getActivityList(); List<ActivitySaveDto> activityList = requestFormSaveDto.getActivityList();
// 缓存校验结果,避免主循环中重复查询和可能出现的数据不一致 // 缓存校验结果,避免主循环中重复查询和可能出现的数据不一致
java.util.Map<Long, Long> activityIdToPositionIdMap = new java.util.HashMap<>(); java.util.Map<Long, Long> activityIdToPositionIdMap = new java.util.HashMap<>();
@@ -129,15 +102,14 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), frontendPositionId); activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), frontendPositionId);
continue; continue;
} }
// 前端未传入时,查询配置的执行科室(暂不校验,仅用于兼容无前端传入的场景) // 前端未传入时,使用配置的执行科室
List<ActivityOrganizationConfigDto> activityOrganizationConfig =
requestFormManageAppMapper.getActivityOrganizationConfig(typeCode);
Long configPositionId = activityOrganizationConfig.stream() Long configPositionId = activityOrganizationConfig.stream()
.filter(dto -> activitySaveDto.getAdviceDefinitionId().equals(dto.getActivityDefinitionId())) .filter(dto -> activitySaveDto.getAdviceDefinitionId().equals(dto.getActivityDefinitionId()))
.map(ActivityOrganizationConfigDto::getOrganizationId).findFirst().orElse(null); .map(ActivityOrganizationConfigDto::getOrganizationId).findFirst().orElse(null);
if (configPositionId != null) { if (configPositionId == null) {
activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), configPositionId); throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室");
} }
activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), configPositionId);
} }
} }
@@ -204,77 +176,73 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
ChargeItem chargeItem; ChargeItem chargeItem;
log.info("保存申请单typeCode={}, activityListSize={}, encounterId={}", typeCode, activityList != null ? activityList.size() : 0, encounterId); log.info("保存申请单typeCode={}, activityListSize={}, encounterId={}", typeCode, activityList != null ? activityList.size() : 0, encounterId);
// 🔧 手术申请单:跳过普通医嘱生成,只由 isProcedure 块生成手术医嘱,避免重复 for (ActivitySaveDto activitySaveDto : activityList) {
boolean isSurgeryRequest = ActivityDefCategory.PROCEDURE.getCode().equals(typeCode); serviceRequest = new ServiceRequest();
if (!isSurgeryRequest) { serviceRequest.setStatusEnum(RequestStatus.DRAFT.getValue());
for (ActivitySaveDto activitySaveDto : activityList) { serviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.SERVICE_RES_NO.getPrefix(), 4));
serviceRequest = new ServiceRequest(); serviceRequest.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源
serviceRequest.setStatusEnum(RequestStatus.DRAFT.getValue()); serviceRequest.setPrescriptionNo(prescriptionNo);
serviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.SERVICE_RES_NO.getPrefix(), 4)); serviceRequest.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue());// 治疗类型
serviceRequest.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源 serviceRequest.setQuantity(activitySaveDto.getQuantity()); // 请求数量
serviceRequest.setPrescriptionNo(prescriptionNo); serviceRequest.setUnitCode(activitySaveDto.getUnitCode()); // 请求单位编码
serviceRequest.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue());// 治疗类型 serviceRequest.setCategoryEnum(categoryEnum); // 请求类型
serviceRequest.setQuantity(activitySaveDto.getQuantity()); // 请求数量 serviceRequest.setActivityId(activitySaveDto.getAdviceDefinitionId());// 诊疗定义id
serviceRequest.setUnitCode(activitySaveDto.getUnitCode()); // 请求单位编码 serviceRequest.setPatientId(patientId); // 患者
serviceRequest.setCategoryEnum(categoryEnum); // 请求类型 serviceRequest.setRequesterId(practitionerId); // 开方医生
serviceRequest.setActivityId(activitySaveDto.getAdviceDefinitionId());// 诊疗定义id serviceRequest.setEncounterId(encounterId); // 诊id
serviceRequest.setPatientId(patientId); // 患者 serviceRequest.setAuthoredTime(curDate); // 请求签发时间
serviceRequest.setRequesterId(practitionerId); // 开方医生
serviceRequest.setEncounterId(encounterId); // 就诊id
serviceRequest.setAuthoredTime(curDate); // 请求签发时间
Long positionId = activityIdToPositionIdMap.get(activitySaveDto.getAdviceDefinitionId()); Long positionId = activityIdToPositionIdMap.get(activitySaveDto.getAdviceDefinitionId());
if (positionId == null) { if (positionId == null) {
throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室"); throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室");
} }
serviceRequest.setOrgId(positionId); // 执行科室 serviceRequest.setOrgId(positionId); // 执行科室
serviceRequest.setYbClassEnum(activitySaveDto.getYbClassEnum());// 类别医保编码 serviceRequest.setYbClassEnum(activitySaveDto.getYbClassEnum());// 类别医保编码
serviceRequest.setConditionId(activitySaveDto.getConditionId()); // 诊断id serviceRequest.setConditionId(activitySaveDto.getConditionId()); // 诊断id
serviceRequest.setEncounterDiagnosisId(activitySaveDto.getEncounterDiagnosisId()); // 就诊诊断id serviceRequest.setEncounterDiagnosisId(activitySaveDto.getEncounterDiagnosisId()); // 就诊诊断id
iServiceRequestService.save(serviceRequest); iServiceRequestService.save(serviceRequest);
chargeItem = new ChargeItem(); chargeItem = new ChargeItem();
chargeItem.setStatusEnum(ChargeItemStatus.DRAFT.getValue()); // 收费状态 chargeItem.setStatusEnum(ChargeItemStatus.DRAFT.getValue()); // 收费状态
chargeItem.setBusNo(AssignSeqEnum.CHARGE_ITEM_NO.getPrefix().concat(serviceRequest.getBusNo())); chargeItem.setBusNo(AssignSeqEnum.CHARGE_ITEM_NO.getPrefix().concat(serviceRequest.getBusNo()));
chargeItem.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源 chargeItem.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源
chargeItem.setPatientId(patientId); // 患者 chargeItem.setPatientId(patientId); // 患者
chargeItem.setContextEnum(activitySaveDto.getAdviceType()); // 类型 chargeItem.setContextEnum(activitySaveDto.getAdviceType()); // 类型
chargeItem.setEncounterId(encounterId); // 就诊id chargeItem.setEncounterId(encounterId); // 就诊id
chargeItem.setDefinitionId(activitySaveDto.getDefinitionId()); // 费用定价ID chargeItem.setDefinitionId(activitySaveDto.getDefinitionId()); // 费用定价ID
chargeItem.setDefDetailId(activitySaveDto.getDefinitionDetailId()); // 定价子表主键 chargeItem.setDefDetailId(activitySaveDto.getDefinitionDetailId()); // 定价子表主键
chargeItem.setEntererId(practitionerId);// 开立人ID chargeItem.setEntererId(practitionerId);// 开立人ID
chargeItem.setEnteredDate(curDate); // 开立时间 chargeItem.setEnteredDate(curDate); // 开立时间
chargeItem.setServiceTable(CommonConstants.TableName.WOR_SERVICE_REQUEST);// 医疗服务类型 chargeItem.setServiceTable(CommonConstants.TableName.WOR_SERVICE_REQUEST);// 医疗服务类型
chargeItem.setServiceId(serviceRequest.getId()); // 医疗服务ID chargeItem.setServiceId(serviceRequest.getId()); // 医疗服务ID
chargeItem.setProductTable(CommonConstants.TableName.WOR_ACTIVITY_DEFINITION);// 产品所在表 chargeItem.setProductTable(CommonConstants.TableName.WOR_ACTIVITY_DEFINITION);// 产品所在表
chargeItem.setProductId(activitySaveDto.getAdviceDefinitionId());// 收费项id chargeItem.setProductId(activitySaveDto.getAdviceDefinitionId());// 收费项id
chargeItem.setAccountId(activitySaveDto.getAccountId());// 关联账户ID chargeItem.setAccountId(activitySaveDto.getAccountId());// 关联账户ID
chargeItem.setRequestingOrgId(orgId); // 开立科室 chargeItem.setRequestingOrgId(orgId); // 开立科室
chargeItem.setConditionId(activitySaveDto.getConditionId()); // 诊断id chargeItem.setConditionId(activitySaveDto.getConditionId()); // 诊断id
chargeItem.setEncounterDiagnosisId(activitySaveDto.getEncounterDiagnosisId()); // 就诊诊断id chargeItem.setEncounterDiagnosisId(activitySaveDto.getEncounterDiagnosisId()); // 就诊诊断id
chargeItem.setQuantityValue(activitySaveDto.getQuantity()); // 数量 chargeItem.setQuantityValue(activitySaveDto.getQuantity()); // 数量
chargeItem.setQuantityUnit(activitySaveDto.getUnitCode()); // 单位 chargeItem.setQuantityUnit(activitySaveDto.getUnitCode()); // 单位
chargeItem.setUnitPrice(activitySaveDto.getUnitPrice()); // 单价 chargeItem.setUnitPrice(activitySaveDto.getUnitPrice()); // 单价
chargeItem.setTotalPrice(activitySaveDto.getTotalPrice()); // 总价 chargeItem.setTotalPrice(activitySaveDto.getTotalPrice()); // 总价
iChargeItemService.save(chargeItem); iChargeItemService.save(chargeItem);
// 处理诊疗套餐的子项信息 // 处理诊疗套餐的子项信息
ActivityDefinition activityDefinition = ActivityDefinition activityDefinition =
iActivityDefinitionService.getById(activitySaveDto.getAdviceDefinitionId()); iActivityDefinitionService.getById(activitySaveDto.getAdviceDefinitionId());
String childrenJson = activityDefinition.getChildrenJson(); String childrenJson = activityDefinition.getChildrenJson();
if (childrenJson != null) { if (childrenJson != null) {
// 诊疗子项参数类 // 诊疗子项参数类
ActivityChildrenJsonParams activityChildrenJsonParams = new ActivityChildrenJsonParams(); ActivityChildrenJsonParams activityChildrenJsonParams = new ActivityChildrenJsonParams();
activityChildrenJsonParams.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue()); // 治疗类型 activityChildrenJsonParams.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue()); // 治疗类型
activityChildrenJsonParams.setPatientId(serviceRequest.getPatientId()); // 患者 activityChildrenJsonParams.setPatientId(serviceRequest.getPatientId()); // 患者
activityChildrenJsonParams.setEncounterId(serviceRequest.getEncounterId()); // 就诊id activityChildrenJsonParams.setEncounterId(serviceRequest.getEncounterId()); // 就诊id
activityChildrenJsonParams.setAccountId(chargeItem.getAccountId()); // 账户id activityChildrenJsonParams.setAccountId(chargeItem.getAccountId()); // 账户id
activityChildrenJsonParams.setChargeItemId(chargeItem.getId()); // 费用项id activityChildrenJsonParams.setChargeItemId(chargeItem.getId()); // 费用项id
activityChildrenJsonParams.setParentId(serviceRequest.getId());// 子项诊疗的父id activityChildrenJsonParams.setParentId(serviceRequest.getId());// 子项诊疗的父id
activityChildrenJsonParams.setEncounterDiagnosisId(serviceRequest.getEncounterDiagnosisId()); activityChildrenJsonParams.setEncounterDiagnosisId(serviceRequest.getEncounterDiagnosisId());
adviceUtils.handleActivityChild(childrenJson, organizationId, activityChildrenJsonParams); adviceUtils.handleActivityChild(childrenJson, organizationId, activityChildrenJsonParams);
}
} }
} }
@@ -358,13 +326,6 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
} else if (surgeryName != null && !surgeryName.isEmpty()) { } else if (surgeryName != null && !surgeryName.isEmpty()) {
contentMap.put("surgeryName", surgeryName); contentMap.put("surgeryName", surgeryName);
} }
// 🔧 手术申请单级联删除contentJson 中记录 requestFormId 和 prescriptionNo删除医嘱时可定位并作废申请单
if (requestForm.getId() != null) {
contentMap.put("requestFormId", String.valueOf(requestForm.getId()));
}
if (prescriptionNo != null && !prescriptionNo.isEmpty()) {
contentMap.put("prescriptionNo", prescriptionNo);
}
if (surgeryCode != null && !surgeryCode.isEmpty()) { if (surgeryCode != null && !surgeryCode.isEmpty()) {
contentMap.put("surgeryCode", surgeryCode); contentMap.put("surgeryCode", surgeryCode);
} }
@@ -407,10 +368,9 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
surgeryChargeItem.setServiceTable(CommonConstants.TableName.WOR_SERVICE_REQUEST); surgeryChargeItem.setServiceTable(CommonConstants.TableName.WOR_SERVICE_REQUEST);
surgeryChargeItem.setServiceId(surgeryServiceRequest.getId()); surgeryChargeItem.setServiceId(surgeryServiceRequest.getId());
surgeryChargeItem.setProductTable(CommonConstants.TableName.WOR_ACTIVITY_DEFINITION); surgeryChargeItem.setProductTable(CommonConstants.TableName.WOR_ACTIVITY_DEFINITION);
// 优先从 activityList 获取 productId 和 definitionId // 优先从 activityList 获取 productId
if (activityList != null && !activityList.isEmpty()) { if (activityList != null && !activityList.isEmpty()) {
surgeryChargeItem.setProductId(activityList.get(0).getAdviceDefinitionId()); surgeryChargeItem.setProductId(activityList.get(0).getAdviceDefinitionId());
surgeryChargeItem.setDefinitionId(activityList.get(0).getDefinitionId());
surgeryChargeItem.setAccountId(activityList.get(0).getAccountId()); surgeryChargeItem.setAccountId(activityList.get(0).getAccountId());
} }
surgeryChargeItem.setRequestingOrgId(orgId); surgeryChargeItem.setRequestingOrgId(orgId);
@@ -449,10 +409,6 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
anesthesiaChargeItem.setServiceTable(CommonConstants.TableName.WOR_SERVICE_REQUEST); anesthesiaChargeItem.setServiceTable(CommonConstants.TableName.WOR_SERVICE_REQUEST);
anesthesiaChargeItem.setServiceId(surgeryServiceRequest.getId()); anesthesiaChargeItem.setServiceId(surgeryServiceRequest.getId());
anesthesiaChargeItem.setProductTable(CommonConstants.TableName.WOR_ACTIVITY_DEFINITION); anesthesiaChargeItem.setProductTable(CommonConstants.TableName.WOR_ACTIVITY_DEFINITION);
// 从 activityList 获取 definitionId
if (activityList != null && !activityList.isEmpty()) {
anesthesiaChargeItem.setDefinitionId(activityList.get(0).getDefinitionId());
}
anesthesiaChargeItem.setRequestingOrgId(orgId); anesthesiaChargeItem.setRequestingOrgId(orgId);
anesthesiaChargeItem.setQuantityValue(BigDecimal.valueOf(1)); anesthesiaChargeItem.setQuantityValue(BigDecimal.valueOf(1));
anesthesiaChargeItem.setQuantityUnit(""); anesthesiaChargeItem.setQuantityUnit("");
@@ -469,18 +425,6 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
log.error("生成手术医嘱过程中发生异常", e); log.error("生成手术医嘱过程中发生异常", e);
throw e; throw e;
} }
// 将手术项目名称写入申请单name字段确保医嘱删除后申请单仍保留正确名称
if (activityList != null && !activityList.isEmpty()) {
String surgeryDisplayName = activityList.stream()
.map(ActivitySaveDto::getAdviceDefinitionName)
.filter(name -> name != null && !name.isEmpty())
.distinct()
.collect(Collectors.joining(""));
if (!surgeryDisplayName.isEmpty()) {
requestForm.setName(surgeryDisplayName);
iRequestFormService.updateById(requestForm);
}
}
} else { } else {
log.info("不是手术申请单跳过手术医嘱生成typeCode={}", typeCode); log.info("不是手术申请单跳过手术医嘱生成typeCode={}", typeCode);
} }
@@ -564,17 +508,12 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
if (requestForm == null) { if (requestForm == null) {
return R.fail("申请单不存在"); return R.fail("申请单不存在");
} }
R<?> permissionResult = validateRequestFormPermission(requestForm);
if (permissionResult != null) {
return permissionResult;
}
String prescriptionNo = requestForm.getPrescriptionNo(); String prescriptionNo = requestForm.getPrescriptionNo();
// 查询该申请单下所有 ServiceRequest含子项 // 查询该申请单下所有 ServiceRequest含子项
List<ServiceRequest> serviceRequests = iServiceRequestService.list( List<ServiceRequest> serviceRequests = iServiceRequestService.list(
new LambdaQueryWrapper<ServiceRequest>() new LambdaQueryWrapper<ServiceRequest>()
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo) .eq(ServiceRequest::getPrescriptionNo, prescriptionNo));
.eq(ServiceRequest::getDeleteFlag, DelFlag.NO.getCode()));
if (serviceRequests == null || serviceRequests.isEmpty()) { if (serviceRequests == null || serviceRequests.isEmpty()) {
return R.fail("未找到关联的诊疗医嘱"); return R.fail("未找到关联的诊疗医嘱");
} }
@@ -604,7 +543,7 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
// 4. 删除申请单 // 4. 删除申请单
iRequestFormService.removeById(requestFormId); iRequestFormService.removeById(requestFormId);
log.info("申请单删除成功requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo); log.info("申请单删除成功requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
return R.ok("删除成功"); return R.ok("删除成功");
} }
@@ -617,43 +556,32 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
if (requestForm == null) { if (requestForm == null) {
return R.fail("申请单不存在"); return R.fail("申请单不存在");
} }
R<?> permissionResult = validateRequestFormPermission(requestForm);
if (permissionResult != null) {
return permissionResult;
}
String prescriptionNo = requestForm.getPrescriptionNo(); String prescriptionNo = requestForm.getPrescriptionNo();
// 查询该申请单下所有 ServiceRequest // 查询该申请单下所有 ServiceRequest
List<ServiceRequest> serviceRequests = iServiceRequestService.list( List<ServiceRequest> serviceRequests = iServiceRequestService.list(
new LambdaQueryWrapper<ServiceRequest>() new LambdaQueryWrapper<ServiceRequest>()
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo) .eq(ServiceRequest::getPrescriptionNo, prescriptionNo));
.eq(ServiceRequest::getDeleteFlag, DelFlag.NO.getCode()));
if (serviceRequests == null || serviceRequests.isEmpty()) { if (serviceRequests == null || serviceRequests.isEmpty()) {
return R.fail("未找到关联的诊疗医嘱"); return R.fail("未找到关联的诊疗医嘱");
} }
List<Long> serviceRequestIds = serviceRequests.stream()
.map(ServiceRequest::getId).collect(Collectors.toList());
// 校验:标本已采集则不可撤回
if (hasCollectedSpecimen(serviceRequestIds)) {
return R.fail("标本已采集,无法撤回");
}
// 校验:只有已签发(status=2)的申请单可撤回 // 校验:只有已签发(status=2)的申请单可撤回
boolean allActive = serviceRequests.stream() boolean allActive = serviceRequests.stream()
.allMatch(sr -> RequestStatus.ACTIVE.getValue().equals(sr.getStatusEnum())); .allMatch(sr -> RequestStatus.ACTIVE.getValue().equals(sr.getStatusEnum()));
if (!allActive) { if (!allActive) {
return R.fail("只有已签发且未采证的申请单可撤回"); return R.fail("只有已签发状态的申请单可撤回");
} }
// 将所有 ServiceRequest 状态改回待签发,与申请单展示状态同步 // 将所有 ServiceRequest 状态改回待签发(DRAFT=0)
List<Long> serviceRequestIds = serviceRequests.stream()
.map(ServiceRequest::getId).collect(Collectors.toList());
iServiceRequestService.update( iServiceRequestService.update(
new ServiceRequest().setStatusEnum(RequestStatus.DRAFT.getValue()), new ServiceRequest().setStatusEnum(RequestStatus.DRAFT.getValue()),
new LambdaUpdateWrapper<ServiceRequest>() new LambdaUpdateWrapper<ServiceRequest>()
.in(ServiceRequest::getId, serviceRequestIds)); .in(ServiceRequest::getId, serviceRequestIds));
log.info("申请单撤回成功requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo); log.info("申请单撤回成功requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
return R.ok("撤回成功"); return R.ok("撤回成功");
} }

View File

@@ -13,11 +13,6 @@ import java.math.BigDecimal;
@Accessors(chain = true) @Accessors(chain = true)
public class RequestFormDetailQueryDto { public class RequestFormDetailQueryDto {
/**
* 诊疗活动定义IDwor_service_request.activity_id与开立检验时项目字典的 id / adviceDefinitionId 一致,用于编辑回显)
*/
private Long activityId;
/** 医嘱名称 */ /** 医嘱名称 */
private String adviceName; private String adviceName;

View File

@@ -31,8 +31,8 @@ public class HomeController {
HomeStatisticsDto statisticsDto = homeStatisticsService.getHomeStatistics(); HomeStatisticsDto statisticsDto = homeStatisticsService.getHomeStatistics();
// 获取待写病历数量 // 获取待写病历数量
Long practitionerId = SecurityUtils.getLoginUser().getPractitionerId(); Long userId = SecurityUtils.getLoginUser().getUserId();
R<?> pendingEmrCount = doctorStationEmrAppService.getPendingEmrCount(practitionerId, null); R<?> pendingEmrCount = doctorStationEmrAppService.getPendingEmrCount(userId);
// 将待写病历数量添加到统计数据中 // 将待写病历数量添加到统计数据中
statisticsDto.setPendingEmr((Integer) pendingEmrCount.getData()); statisticsDto.setPendingEmr((Integer) pendingEmrCount.getData());

View File

@@ -74,6 +74,7 @@ public class TriageQueueAppServiceImpl implements TriageQueueAppService {
.eq(TriageQueueItem::getTenantId, tenantId) .eq(TriageQueueItem::getTenantId, tenantId)
.eq(TriageQueueItem::getQueueDate, qd) .eq(TriageQueueItem::getQueueDate, qd)
.eq(TriageQueueItem::getDeleteFlag, "0") .eq(TriageQueueItem::getDeleteFlag, "0")
.ne(TriageQueueItem::getStatus, TriageQueueStatus.COMPLETED.getValue())
.orderByAsc(TriageQueueItem::getQueueOrder); .orderByAsc(TriageQueueItem::getQueueOrder);
// 如果指定了科室,按科室过滤;否则查询所有科室(全科模式) // 如果指定了科室,按科室过滤;否则查询所有科室(全科模式)
@@ -91,6 +92,14 @@ public class TriageQueueAppServiceImpl implements TriageQueueAppService {
} }
}); });
} }
// 双重保险:再次过滤掉 COMPLETED 状态的患者(防止数据库中有异常数据)
if (list != null && !list.isEmpty()) {
int beforeSize = list.size();
list = list.stream()
.filter(item -> !TriageQueueStatus.COMPLETED.getValue().equals(item.getStatus()))
.collect(java.util.stream.Collectors.toList());
}
return R.ok(list); return R.ok(list);
} }

View File

@@ -117,7 +117,7 @@
) )
</if> </if>
<if test="searchKey != null and searchKey != ''"> <if test="searchKey != null and searchKey != ''">
AND (t1.name ILIKE '%' || '${searchKey}' || '%' OR t1.py_str ILIKE '%' || '${searchKey}' || '%') AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
</if> </if>
<if test="adviceDefinitionIdParamList != null and !adviceDefinitionIdParamList.isEmpty()"> <if test="adviceDefinitionIdParamList != null and !adviceDefinitionIdParamList.isEmpty()">
AND t1.id IN AND t1.id IN
@@ -181,7 +181,7 @@
WHERE t1.delete_flag = '0' WHERE t1.delete_flag = '0'
AND t1.status_enum = #{statusEnum} AND t1.status_enum = #{statusEnum}
<if test="searchKey != null and searchKey != ''"> <if test="searchKey != null and searchKey != ''">
AND (t1.name ILIKE '%' || '${searchKey}' || '%' OR t1.py_str ILIKE '%' || '${searchKey}' || '%') AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
</if> </if>
<if test="categoryCode != null and categoryCode != ''"> <if test="categoryCode != null and categoryCode != ''">
AND t1.category_code = #{categoryCode} AND t1.category_code = #{categoryCode}
@@ -278,7 +278,7 @@
AND T1.category_code != '手术' AND T1.category_code != '24' AND T1.category_code != '手术' AND T1.category_code != '24'
</if> </if>
<if test="searchKey != null and searchKey != ''"> <if test="searchKey != null and searchKey != ''">
AND (t1.name ILIKE '%' || '${searchKey}' || '%' OR t1.py_str ILIKE '%' || '${searchKey}' || '%') AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
</if> </if>
<if test="categoryCode != null and categoryCode != ''"> <if test="categoryCode != null and categoryCode != ''">
AND t1.category_code = #{categoryCode} AND t1.category_code = #{categoryCode}
@@ -871,7 +871,6 @@
</select> </select>
<!-- 手术项目专用分页查询:仅查手术 + 定价,无库存/草稿库存/取药科室等无关逻辑 --> <!-- 手术项目专用分页查询:仅查手术 + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
<!-- 使用 LIMIT/OFFSET 直接查询,避免 MyBatis Plus 分页插件的 COUNT 开销 -->
<select id="getSurgeryPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto"> <select id="getSurgeryPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto">
SELECT DISTINCT ON (t1.ID) SELECT DISTINCT ON (t1.ID)
t1.ID AS advice_definition_id, t1.ID AS advice_definition_id,
@@ -894,16 +893,15 @@
AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%') AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
</if> </if>
ORDER BY t1.ID, t1.name ASC, t2.ID ASC ORDER BY t1.ID, t1.name ASC, t2.ID ASC
LIMIT #{limit} OFFSET #{offset} LIMIT #{page.size} OFFSET ${(page.current - 1) * page.size}
</select> </select>
<!-- 检查/检验项目专用分页查询:仅查指定 category_code + 定价,无库存/草稿库存/取药科室等无关逻辑 --> <!-- 检查项目专用分页查询:仅查检查(23) + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
<select id="getExaminationPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto"> <select id="getExaminationPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto">
SELECT DISTINCT ON (t1.ID) SELECT DISTINCT ON (t1.ID)
t1.ID AS advice_definition_id, t1.ID AS advice_definition_id,
t1.NAME AS advice_name, t1.NAME AS advice_name,
t1.org_id AS org_id, t1.org_id AS org_id,
t3.name AS org_name,
t1.org_id AS position_id, t1.org_id AS position_id,
t2.ID AS charge_item_definition_id, t2.ID AS charge_item_definition_id,
t2.price AS price, t2.price AS price,
@@ -915,11 +913,8 @@
AND t2.delete_flag = '0' AND t2.delete_flag = '0'
AND t2.status_enum = #{statusEnum} AND t2.status_enum = #{statusEnum}
AND t2.instance_table = 'wor_activity_definition' AND t2.instance_table = 'wor_activity_definition'
LEFT JOIN adm_organization t3
ON t3.id = t1.org_id
AND t3.delete_flag = '0'
WHERE t1.delete_flag = '0' WHERE t1.delete_flag = '0'
AND t1.category_code = #{categoryCode} AND t1.category_code = '23'
<if test="searchKey != null and searchKey != ''"> <if test="searchKey != null and searchKey != ''">
AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%') AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
</if> </if>

View File

@@ -135,7 +135,6 @@
T1.onset_date AS onsetDate, T1.onset_date AS onsetDate,
T1.diagnosis_time AS diagnosisTime, T1.diagnosis_time AS diagnosisTime,
T1.doctor AS diagnosisDoctor, T1.doctor AS diagnosisDoctor,
T1.long_term_flag AS longTermFlag,
CASE WHEN EXISTS ( CASE WHEN EXISTS (
SELECT 1 FROM infectious_card T4 SELECT 1 FROM infectious_card T4
WHERE T4.diag_id = T2.id AND T4.delete_flag = '0' AND T4.status >= 1 WHERE T4.diag_id = T2.id AND T4.delete_flag = '0' AND T4.status >= 1

View File

@@ -4,38 +4,4 @@
"http://mybatis.org/dtd/mybatis-3-mapper.dtd"> "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.openhis.web.doctorstation.mapper.DoctorStationEmrAppMapper"> <mapper namespace="com.openhis.web.doctorstation.mapper.DoctorStationEmrAppMapper">
<select id="getPendingEmrList" resultType="java.util.HashMap"> </mapper>
SELECT e.id AS "encounterId",
e.patient_id AS "patientId",
p.name AS "patientName",
p.gender_enum AS "gender",
p.birth_date AS "birthDate",
e.create_time AS "registerTime",
e.bus_no AS "busNo"
FROM adm_encounter e
INNER JOIN adm_encounter_participant ep ON e.id = ep.encounter_id AND ep.practitioner_id = #{doctorId}
LEFT JOIN adm_patient p ON e.patient_id = p.id
LEFT JOIN doc_emr emr ON e.id = emr.encounter_id
WHERE e.status_enum = 2
AND emr.id IS NULL
<if test="patientName != null and patientName != ''">
AND p.name LIKE CONCAT('%', #{patientName}, '%')
</if>
ORDER BY e.create_time DESC
</select>
<select id="getPendingEmrCount" resultType="java.lang.Long">
SELECT COUNT(*)
FROM adm_encounter e
INNER JOIN adm_encounter_participant ep ON e.id = ep.encounter_id AND ep.practitioner_id = #{doctorId}
LEFT JOIN doc_emr emr ON e.id = emr.encounter_id
WHERE e.status_enum = 2
AND emr.id IS NULL
<if test="patientName != null and patientName != ''">
AND e.patient_id IN (
SELECT id FROM adm_patient WHERE name LIKE CONCAT('%', #{patientName}, '%')
)
</if>
</select>
</mapper>

View File

@@ -35,27 +35,21 @@
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0' WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ws.status_enum = 8 AND ws.status_enum = 8
) THEN 6 ) THEN 6
WHEN EXISTS (
SELECT 1 FROM wor_service_request ws
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ws.status_enum = 5
) THEN 7
WHEN EXISTS ( WHEN EXISTS (
SELECT 1 FROM wor_service_request ws SELECT 1 FROM wor_service_request ws
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0' WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ws.status_enum = 3 AND ws.status_enum = 3
) THEN 5 ) THEN 5
WHEN EXISTS (
SELECT 1 FROM wor_service_request ws
INNER JOIN lab_specimen ls ON ls.service_id = ws.id
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ls.collection_status_enum >= 1
) THEN 4
WHEN EXISTS ( WHEN EXISTS (
SELECT 1 FROM wor_service_request ws SELECT 1 FROM wor_service_request ws
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0' WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ws.status_enum = 2 AND ws.status_enum = 2
) THEN 1 ) THEN 1
WHEN EXISTS (
SELECT 1 FROM wor_service_request ws
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
AND ws.status_enum = 5
) THEN 7
ELSE 0 ELSE 0
END AS computed_status END AS computed_status
FROM doc_request_form AS drf FROM doc_request_form AS drf
@@ -63,6 +57,8 @@
AND ae.delete_flag = '0' AND ae.delete_flag = '0'
LEFT JOIN adm_patient AS ap ON ap.ID = ae.patient_id LEFT JOIN adm_patient AS ap ON ap.ID = ae.patient_id
AND ap.delete_flag = '0' AND ap.delete_flag = '0'
LEFT JOIN wor_service_request AS wsr ON wsr.prescription_no = drf.prescription_no
AND wsr.delete_flag = '0'
WHERE drf.delete_flag = '0' WHERE drf.delete_flag = '0'
AND drf.encounter_id = #{encounterId} AND drf.encounter_id = #{encounterId}
AND drf.type_code = #{typeCode} AND drf.type_code = #{typeCode}

View File

@@ -768,4 +768,36 @@ public class CommonConstants {
Integer ACCOUNT_DEVICE_TYPE = 6; Integer ACCOUNT_DEVICE_TYPE = 6;
} }
/**
* 号源槽位状态 (adm_schedule_slot.status)
*/
public interface SlotStatus {
/** 可用 / 待预约 */
Integer AVAILABLE = 0;
/** 已预约 */
Integer BOOKED = 1;
/** 已取消 / 已停诊 */
Integer CANCELLED = 2;
/** 已签到 / 已取号 */
Integer CHECKED_IN = 3;
/** 已锁定 */
Integer LOCKED = 4;
/** 已退号 */
Integer RETURNED = 5;
}
/**
* 预约订单状态 (order_main.status)
*/
public interface AppointmentOrderStatus {
/** 已预约 (待就诊) */
Integer BOOKED = 1;
/** 已取号 (已就诊) */
Integer CHECKED_IN = 2;
/** 已取消 */
Integer CANCELLED = 3;
/** 已退号 */
Integer RETURNED = 4;
}
} }

View File

@@ -1,57 +0,0 @@
package com.openhis.common.enums;
import lombok.AllArgsConstructor;
import lombok.Getter;
/**
* 号源槽位状态 (adm_schedule_slot.status)
*
* <pre>
* 状态流转:
* 预约 → 0→2 (锁定), locked_num+1
* 取消预约 → 2→0 (释放), locked_num-1
* 签到 → 2→1 (已约), locked_num-1, booked_num+1
* 退号 → 1→0 (释放), booked_num-1
* 停诊 → 任意→4 (已取消)
* </pre>
*
* @author system
*/
@Getter
@AllArgsConstructor
public enum SlotStatus implements HisEnumInterface {
/** 可用 / 待预约 */
AVAILABLE(0, "available", "可用"),
/** 已预约 */
BOOKED(1, "booked", "已预约"),
/** 已锁定 (约而不付:预约后锁定号源) */
LOCKED(2, "locked", "已锁定"),
/** 已签到 / 已取号 */
CHECKED_IN(3, "checked_in", "已签到"),
/** 已取消 / 已停诊 */
CANCELLED(4, "cancelled", "已取消"),
/** 已退号 */
RETURNED(5, "returned", "已退号");
private final Integer value;
private final String code;
private final String info;
public static SlotStatus getByValue(Integer value) {
if (value == null) {
return null;
}
for (SlotStatus val : values()) {
if (val.getValue().equals(value)) {
return val;
}
}
return null;
}
}

View File

@@ -38,12 +38,4 @@ public interface IOrganizationLocationService extends IService<OrganizationLocat
*/ */
List<OrganizationLocation> getOrgLocListByOrgIdAndActivityDefinitionId(Long organizationId, Long activityDefinitionId); List<OrganizationLocation> getOrgLocListByOrgIdAndActivityDefinitionId(Long organizationId, Long activityDefinitionId);
/**
* 根据诊疗定义id查询所有执行科室列表跨科室
*
* @param activityDefinitionId 诊疗定义id
* @return 执行科室列表
*/
List<OrganizationLocation> getOrgLocListByActivityDefinitionId(Long activityDefinitionId);
} }

View File

@@ -64,16 +64,4 @@ public class OrganizationLocationServiceImpl extends ServiceImpl<OrganizationLoc
.eq(OrganizationLocation::getActivityDefinitionId, activityDefinitionId)); .eq(OrganizationLocation::getActivityDefinitionId, activityDefinitionId));
} }
/**
* 根据诊疗定义id查询所有执行科室列表跨科室
*
* @param activityDefinitionId 诊疗定义id
* @return 执行科室列表
*/
@Override
public List<OrganizationLocation> getOrgLocListByActivityDefinitionId(Long activityDefinitionId) {
return baseMapper.selectList(new LambdaQueryWrapper<OrganizationLocation>()
.eq(OrganizationLocation::getActivityDefinitionId, activityDefinitionId));
}
} }

View File

@@ -10,11 +10,10 @@ import org.springframework.stereotype.Repository;
public interface SchedulePoolMapper extends BaseMapper<SchedulePool> { public interface SchedulePoolMapper extends BaseMapper<SchedulePool> {
/** /**
* 按号源池实时重算统计值。 * 按号源池实时重算统计值,避免并发场景下计数漂移
* *
* @param poolId 号源池ID * 说明available_num 在当前项目中可能为数据库生成列,因此这里仅维护
* @param bookedStatus 已约状态值,由 SlotStatus.BOOKED.getValue() 传入 * booked_num / locked_num剩余号由数据库或查询逻辑计算。
* @param lockedStatus 锁定状态值,由 SlotStatus.LOCKED.getValue() 传入
*/ */
@Update(""" @Update("""
UPDATE adm_schedule_pool p UPDATE adm_schedule_pool p
@@ -24,22 +23,20 @@ public interface SchedulePoolMapper extends BaseMapper<SchedulePool> {
FROM adm_schedule_slot s FROM adm_schedule_slot s
WHERE s.pool_id = p.id WHERE s.pool_id = p.id
AND s.delete_flag = '0' AND s.delete_flag = '0'
AND s.status = #{bookedStatus} AND s.status = 1
), 0), ), 0),
locked_num = COALESCE(( locked_num = COALESCE((
SELECT COUNT(1) SELECT COUNT(1)
FROM adm_schedule_slot s FROM adm_schedule_slot s
WHERE s.pool_id = p.id WHERE s.pool_id = p.id
AND s.delete_flag = '0' AND s.delete_flag = '0'
AND s.status = #{lockedStatus} AND s.status = 3
), 0), ), 0),
update_time = now() update_time = now()
WHERE p.id = #{poolId} WHERE p.id = #{poolId}
AND p.delete_flag = '0' AND p.delete_flag = '0'
""") """)
int refreshPoolStats(@Param("poolId") Long poolId, int refreshPoolStats(@Param("poolId") Long poolId);
@Param("bookedStatus") Integer bookedStatus,
@Param("lockedStatus") Integer lockedStatus);
/** /**
* 签到时更新号源池统计:锁定数-1已预约数+1 * 签到时更新号源池统计:锁定数-1已预约数+1

View File

@@ -22,12 +22,9 @@ public interface ScheduleSlotMapper extends BaseMapper<ScheduleSlot> {
TicketSlotDTO selectTicketSlotById(@Param("id") Long id); TicketSlotDTO selectTicketSlotById(@Param("id") Long id);
/** /**
* 原子抢占槽位:仅当当前状态=0(待约)时,更新为目标锁定状态 * 原子抢占槽位:仅当当前状态=0(可用)时,更新为1(已预约)
*
* @param slotId 槽位ID
* @param lockedStatus 锁定状态值,由 SlotStatus.LOCKED.getValue() 传入
*/ */
int lockSlotForBooking(@Param("slotId") Long slotId, @Param("lockedStatus") Integer lockedStatus); int lockSlotForBooking(@Param("slotId") Long slotId);
/** /**
* 按主键更新槽位状态。 * 按主键更新槽位状态。
@@ -37,16 +34,12 @@ public interface ScheduleSlotMapper extends BaseMapper<ScheduleSlot> {
/** /**
* 更新槽位状态并记录签到时间 * 更新槽位状态并记录签到时间
* *
* @param slotId 槽位ID * @param slotId 槽位ID
* @param status 目标状态,由 SlotStatus.BOOKED.getValue() 传入 * @param status 状态
* @param checkInTime 签到时间 * @param checkInTime 签到时间
* @param requiredStatus 前置状态,由 SlotStatus.LOCKED.getValue() 传入
* @return 结果 * @return 结果
*/ */
int updateSlotStatusAndCheckInTime(@Param("slotId") Long slotId, int updateSlotStatusAndCheckInTime(@Param("slotId") Long slotId, @Param("status") Integer status, @Param("checkInTime") Date checkInTime);
@Param("status") Integer status,
@Param("checkInTime") Date checkInTime,
@Param("requiredStatus") Integer requiredStatus);
/** /**
* 根据槽位ID查询所属号源池ID。 * 根据槽位ID查询所属号源池ID。

View File

@@ -1,12 +1,10 @@
package com.openhis.clinical.service.impl; package com.openhis.clinical.service.impl;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.openhis.appointmentmanage.domain.AppointmentConfig; import com.openhis.appointmentmanage.domain.AppointmentConfig;
import com.openhis.appointmentmanage.service.IAppointmentConfigService; import com.openhis.appointmentmanage.service.IAppointmentConfigService;
import com.openhis.appointmentmanage.domain.TicketSlotDTO; import com.openhis.appointmentmanage.domain.TicketSlotDTO;
import com.openhis.appointmentmanage.domain.SchedulePool;
import com.openhis.appointmentmanage.domain.ScheduleSlot; import com.openhis.appointmentmanage.domain.ScheduleSlot;
import com.openhis.appointmentmanage.mapper.SchedulePoolMapper; import com.openhis.appointmentmanage.mapper.SchedulePoolMapper;
import com.openhis.appointmentmanage.mapper.ScheduleSlotMapper; import com.openhis.appointmentmanage.mapper.ScheduleSlotMapper;
@@ -15,7 +13,7 @@ import com.openhis.clinical.domain.Ticket;
import com.openhis.clinical.mapper.TicketMapper; import com.openhis.clinical.mapper.TicketMapper;
import com.openhis.clinical.service.IOrderService; import com.openhis.clinical.service.IOrderService;
import com.openhis.clinical.service.ITicketService; import com.openhis.clinical.service.ITicketService;
import com.openhis.common.enums.SlotStatus; import com.openhis.common.constant.CommonConstants.SlotStatus;
import com.openhis.common.enums.OrderStatus; import com.openhis.common.enums.OrderStatus;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@@ -179,7 +177,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
logger.error("安全拦截号源底库核对失败slotId: {}", slotId); logger.error("安全拦截号源底库核对失败slotId: {}", slotId);
throw new RuntimeException("号源数据不存在"); throw new RuntimeException("号源数据不存在");
} }
if (slot.getSlotStatus() != null && SlotStatus.getByValue(slot.getSlotStatus()) != SlotStatus.AVAILABLE) { if (slot.getSlotStatus() != null && !SlotStatus.AVAILABLE.equals(slot.getSlotStatus())) {
throw new RuntimeException("手慢了!该号源已刚刚被他人抢占"); throw new RuntimeException("手慢了!该号源已刚刚被他人抢占");
} }
if (Boolean.TRUE.equals(slot.getIsStopped())) { if (Boolean.TRUE.equals(slot.getIsStopped())) {
@@ -207,7 +205,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
} }
// 原子抢占:避免并发下同一槽位被重复预约 // 原子抢占:避免并发下同一槽位被重复预约
int lockRows = scheduleSlotMapper.lockSlotForBooking(slotId, SlotStatus.LOCKED.getValue()); int lockRows = scheduleSlotMapper.lockSlotForBooking(slotId);
if (lockRows <= 0) { if (lockRows <= 0) {
throw new RuntimeException("手慢了!该号源已刚刚被他人抢占"); throw new RuntimeException("手慢了!该号源已刚刚被他人抢占");
} }
@@ -262,15 +260,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
throw new RuntimeException("预约成功但号源回填订单失败,请重试"); throw new RuntimeException("预约成功但号源回填订单失败,请重试");
} }
// 6. 预约成功后 locked_num+1原子递增替代全量 recount避免并发计数漂移 refreshPoolStatsBySlotId(slotId);
Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId);
if (poolId != null) {
schedulePoolMapper.update(null,
new LambdaUpdateWrapper<SchedulePool>()
.setSql("locked_num = locked_num + 1, version = version + 1")
.set(SchedulePool::getUpdateTime, new Date())
.eq(SchedulePool::getId, poolId));
}
return 1; return 1;
} }
@@ -287,8 +277,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
if (slot == null) { if (slot == null) {
throw new RuntimeException("号源槽位不存在"); throw new RuntimeException("号源槽位不存在");
} }
// 只有锁定态(2)的号源可以取消预约 if (slot.getSlotStatus() == null || !SlotStatus.BOOKED.equals(slot.getSlotStatus())) {
if (slot.getSlotStatus() == null || SlotStatus.getByValue(slot.getSlotStatus()) != SlotStatus.LOCKED) {
throw new RuntimeException("号源不可取消预约"); throw new RuntimeException("号源不可取消预约");
} }
@@ -303,7 +292,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
orderService.cancelAppointmentOrder(order.getId(), "患者取消预约"); orderService.cancelAppointmentOrder(order.getId(), "患者取消预约");
} }
int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.AVAILABLE.getValue()); int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.AVAILABLE);
if (updated > 0) { if (updated > 0) {
refreshPoolStatsBySlotId(slotId); refreshPoolStatsBySlotId(slotId);
} }
@@ -329,14 +318,11 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
orderService.updateOrderStatusById(latestOrder.getId(), OrderStatus.ACTIVE.getValue()); orderService.updateOrderStatusById(latestOrder.getId(), OrderStatus.ACTIVE.getValue());
orderMapper.updatePayStatus(latestOrder.getId(), 1, new Date()); orderMapper.updatePayStatus(latestOrder.getId(), 1, new Date());
// 2. 只有锁定态(2)的号源才能签到,签到时 2→1(LOCKED→BOOKED) // 2. 查询号源槽位信息
ScheduleSlot slot = scheduleSlotMapper.selectById(slotId); ScheduleSlot slot = scheduleSlotMapper.selectById(slotId);
if (slot == null || !SlotStatus.LOCKED.getValue().equals(slot.getStatus())) {
throw new RuntimeException("号源状态异常,无法签到");
}
// 3. 更新号源槽位状态 2→1LOCKED→BOOKED已预约=已签到) // 3. 更新号源槽位状态为已签到,记录签到时间
scheduleSlotMapper.updateSlotStatusAndCheckInTime(slotId, SlotStatus.BOOKED.getValue(), new Date(), SlotStatus.LOCKED.getValue()); scheduleSlotMapper.updateSlotStatusAndCheckInTime(slotId, SlotStatus.CHECKED_IN, new Date());
// 4. 更新号源池统计:锁定数-1已预约数+1 // 4. 更新号源池统计:锁定数-1已预约数+1
if (slot != null && slot.getPoolId() != null) { if (slot != null && slot.getPoolId() != null) {
@@ -365,7 +351,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
orderService.cancelAppointmentOrder(order.getId(), "医生停诊"); orderService.cancelAppointmentOrder(order.getId(), "医生停诊");
} }
int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.CANCELLED.getValue()); int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.CANCELLED);
if (updated > 0) { if (updated > 0) {
refreshPoolStatsBySlotId(slotId); refreshPoolStatsBySlotId(slotId);
} }
@@ -378,7 +364,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
private void refreshPoolStatsBySlotId(Long slotId) { private void refreshPoolStatsBySlotId(Long slotId) {
Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId); Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId);
if (poolId != null) { if (poolId != null) {
schedulePoolMapper.refreshPoolStats(poolId, SlotStatus.BOOKED.getValue(), SlotStatus.LOCKED.getValue()); schedulePoolMapper.refreshPoolStats(poolId);
} }
} }

View File

@@ -79,13 +79,11 @@ public class OpSchedule extends HisBaseEntity {
private String surgerySite; private String surgerySite;
/** 入院时间 */ /** 入院时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime admissionTime; private LocalDateTime admissionTime;
/** 入手术室时间 */ /** 入手术室时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime entryTime; private LocalDateTime entryTime;
/** 手术室编码 */ /** 手术室编码 */
@@ -144,23 +142,19 @@ public class OpSchedule extends HisBaseEntity {
private String assistant3Code; private String assistant3Code;
/** 手术开始时间 */ /** 手术开始时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime startTime; private LocalDateTime startTime;
/** 手术结束时间 */ /** 手术结束时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime endTime; private LocalDateTime endTime;
/** 麻醉开始时间 */ /** 麻醉开始时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime anesStart; private LocalDateTime anesStart;
/** 麻醉结束时间 */ /** 麻醉结束时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss") @JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime anesEnd; private LocalDateTime anesEnd;
/** 手术状态 */ /** 手术状态 */

View File

@@ -4,17 +4,14 @@
"http://mybatis.org/dtd/mybatis-3-mapper.dtd"> "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.openhis.appointmentmanage.mapper.ScheduleSlotMapper"> <mapper namespace="com.openhis.appointmentmanage.mapper.ScheduleSlotMapper">
<!-- <!-- 统一状态值(兼容数字/英文字符串存储),输出 Integer避免 resultType 映射 NumberFormatException -->
统一状态值映射: DB 数值 → 规范化输出
0=待约 1=已约(签到后) 2=锁定(预约后) 3=已签到 4=已停诊 5=已退号
-->
<sql id="slotStatusNormExpr"> <sql id="slotStatusNormExpr">
CASE CASE
WHEN LOWER(CONCAT('', s.status)) IN ('0', 'unbooked', 'available') THEN 0 WHEN LOWER(CONCAT('', s.status)) IN ('0', 'unbooked', 'available') THEN 0
WHEN LOWER(CONCAT('', s.status)) IN ('1', 'booked') THEN 1 WHEN LOWER(CONCAT('', s.status)) IN ('1', 'booked') THEN 1
WHEN LOWER(CONCAT('', s.status)) IN ('2', 'locked') THEN 2 WHEN LOWER(CONCAT('', s.status)) IN ('2', 'cancelled', 'canceled', 'stopped') THEN 2
WHEN LOWER(CONCAT('', s.status)) IN ('3', 'checked', 'checked_in', 'checkin') THEN 3 WHEN LOWER(CONCAT('', s.status)) IN ('3', 'checked', 'checked_in', 'checkin') THEN 3
WHEN LOWER(CONCAT('', s.status)) IN ('4', 'cancelled', 'canceled', 'stopped') THEN 4 WHEN LOWER(CONCAT('', s.status)) IN ('4', 'locked') THEN 4
WHEN LOWER(CONCAT('', s.status)) IN ('5', 'returned') THEN 5 WHEN LOWER(CONCAT('', s.status)) IN ('5', 'returned') THEN 5
ELSE NULL ELSE NULL
END END
@@ -34,9 +31,9 @@
CASE CASE
WHEN LOWER(CONCAT('', p.status)) IN ('0', 'unbooked', 'available') THEN 0 WHEN LOWER(CONCAT('', p.status)) IN ('0', 'unbooked', 'available') THEN 0
WHEN LOWER(CONCAT('', p.status)) IN ('1', 'booked') THEN 1 WHEN LOWER(CONCAT('', p.status)) IN ('1', 'booked') THEN 1
WHEN LOWER(CONCAT('', p.status)) IN ('2', 'locked') THEN 2 WHEN LOWER(CONCAT('', p.status)) IN ('2', 'cancelled', 'canceled', 'stopped') THEN 2
WHEN LOWER(CONCAT('', p.status)) IN ('3', 'checked', 'checked_in', 'checkin') THEN 3 WHEN LOWER(CONCAT('', p.status)) IN ('3', 'checked', 'checked_in', 'checkin') THEN 3
WHEN LOWER(CONCAT('', p.status)) IN ('4', 'cancelled', 'canceled', 'stopped') THEN 4 WHEN LOWER(CONCAT('', p.status)) IN ('4', 'locked') THEN 4
WHEN LOWER(CONCAT('', p.status)) IN ('5', 'returned') THEN 5 WHEN LOWER(CONCAT('', p.status)) IN ('5', 'returned') THEN 5
ELSE NULL ELSE NULL
END END
@@ -152,11 +149,10 @@
s.id = #{id} s.id = #{id}
</select> </select>
<!-- 预约锁定: 0→#{lockedStatus} (AVAILABLE→LOCKED),由枚举传入 -->
<update id="lockSlotForBooking"> <update id="lockSlotForBooking">
UPDATE adm_schedule_slot UPDATE adm_schedule_slot
SET SET
status = #{lockedStatus}, status = 1,
update_time = now() update_time = now()
WHERE WHERE
id = #{slotId} id = #{slotId}
@@ -178,7 +174,6 @@
AND delete_flag = '0' AND delete_flag = '0'
</update> </update>
<!-- 签到: #{requiredStatus}→#{status} (LOCKED→BOOKED),前置条件由枚举传入 -->
<update id="updateSlotStatusAndCheckInTime"> <update id="updateSlotStatusAndCheckInTime">
UPDATE adm_schedule_slot UPDATE adm_schedule_slot
SET SET
@@ -187,7 +182,6 @@
update_time = NOW() update_time = NOW()
WHERE WHERE
id = #{slotId} id = #{slotId}
AND status = #{requiredStatus}
AND delete_flag = '0' AND delete_flag = '0'
</update> </update>
@@ -208,7 +202,7 @@
update_time = now() update_time = now()
WHERE WHERE
id = #{slotId} id = #{slotId}
AND status = 2 AND status = 1
AND delete_flag = '0' AND delete_flag = '0'
</update> </update>
@@ -305,16 +299,15 @@
<if test="query.phone != null and query.phone != ''"> <if test="query.phone != null and query.phone != ''">
AND o.phone LIKE CONCAT('%', #{query.phone}, '%') AND o.phone LIKE CONCAT('%', #{query.phone}, '%')
</if> </if>
<!-- 5. 时间过滤: 仅待约(0)受时间限制,已锁定(2)/已约(1)/已签到(3)/已退号(5)不受影响 --> <!-- 5. 按系统时间过滤Bug #398 #399 修复:仅未预约受时间过滤,已预约/已取号/已退号不受影响 -->
AND ( AND (
(<include refid="slotStatusNormExpr" /> = 0 AND (p.schedule_date > CURRENT_DATE OR (p.schedule_date = CURRENT_DATE AND (CAST(p.schedule_date AS TIMESTAMP) + CAST(s.expect_time AS TIME)) >= NOW()))) (<include refid="slotStatusNormExpr" /> = 0 AND (p.schedule_date > CURRENT_DATE OR (p.schedule_date = CURRENT_DATE AND (CAST(p.schedule_date AS TIMESTAMP) + CAST(s.expect_time AS TIME)) >= NOW())))
OR <include refid="slotStatusNormExpr" /> = 1 OR <include refid="slotStatusNormExpr" /> = 1
OR <include refid="slotStatusNormExpr" /> = 2
OR <include refid="slotStatusNormExpr" /> = 3 OR <include refid="slotStatusNormExpr" /> = 3
OR <include refid="slotStatusNormExpr" /> = 5 OR <include refid="slotStatusNormExpr" /> = 5
OR <include refid="orderStatusNormExpr" /> = 4 OR <include refid="orderStatusNormExpr" /> = 4
) )
<!-- 6. 状态筛选: unbooked(0) locked(2) booked(2) checked(1) cancelled(4) returned(5) --> <!-- 6. 状态过滤 -->
<if test="query.status != null and query.status != '' and query.status != 'all'"> <if test="query.status != null and query.status != '' and query.status != 'all'">
<choose> <choose>
<when test="'unbooked'.equals(query.status) or '未预约'.equals(query.status)"> <when test="'unbooked'.equals(query.status) or '未预约'.equals(query.status)">
@@ -325,15 +318,7 @@
) )
</when> </when>
<when test="'booked'.equals(query.status) or '已预约'.equals(query.status)"> <when test="'booked'.equals(query.status) or '已预约'.equals(query.status)">
AND <include refid="slotStatusNormExpr" /> = 2 AND <include refid="slotStatusNormExpr" /> = 1
AND <include refid="orderStatusNormExpr" /> = 1
AND (
d.is_stopped IS NULL
OR d.is_stopped = FALSE
)
</when>
<when test="'locked'.equals(query.status) or '已锁定'.equals(query.status)">
AND <include refid="slotStatusNormExpr" /> = 2
AND <include refid="orderStatusNormExpr" /> = 1 AND <include refid="orderStatusNormExpr" /> = 1
AND ( AND (
d.is_stopped IS NULL d.is_stopped IS NULL
@@ -341,7 +326,13 @@
) )
</when> </when>
<when test="'checked'.equals(query.status) or '已取号'.equals(query.status)"> <when test="'checked'.equals(query.status) or '已取号'.equals(query.status)">
AND <include refid="slotStatusNormExpr" /> = 1 AND (
<include refid="slotStatusNormExpr" /> = 3
OR (
<include refid="slotStatusNormExpr" /> = 1
AND <include refid="orderStatusNormExpr" /> = 2
)
)
AND ( AND (
d.is_stopped IS NULL d.is_stopped IS NULL
OR d.is_stopped = FALSE OR d.is_stopped = FALSE
@@ -349,7 +340,7 @@
</when> </when>
<when test="'cancelled'.equals(query.status) or '已停诊'.equals(query.status) or '已取消'.equals(query.status)"> <when test="'cancelled'.equals(query.status) or '已停诊'.equals(query.status) or '已取消'.equals(query.status)">
AND ( AND (
<include refid="slotStatusNormExpr" /> = 4 <include refid="slotStatusNormExpr" /> = 2
OR d.is_stopped = TRUE OR d.is_stopped = TRUE
) )
</when> </when>

View File

@@ -79,51 +79,6 @@ 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', path: '/maintainSystem/Inspection/PackageManagement',

View File

@@ -172,12 +172,12 @@ export const SlotStatus = {
AVAILABLE: 0, AVAILABLE: 0,
/** 已预约 */ /** 已预约 */
BOOKED: 1, BOOKED: 1,
/** 已锁定 */ /** 已取消 / 已停诊 */
LOCKED: 2, CANCELLED: 2,
/** 已签到 / 已取号 */ /** 已签到 / 已取号 */
CHECKED_IN: 3, CHECKED_IN: 3,
/** 已取消 / 已停诊 */ /** 已锁定 */
CANCELLED: 4, LOCKED: 4,
}; };
/** /**
@@ -185,10 +185,10 @@ export const SlotStatus = {
*/ */
export const SlotStatusDescriptions = { export const SlotStatusDescriptions = {
0: '未预约', 0: '未预约',
1: '已取号', 1: '已预约',
2: '已锁定', 2: '已停诊',
3: '已取号', 3: '已取号',
4: '已停诊', 4: '已锁定',
}; };
/** /**
@@ -220,18 +220,3 @@ export function getSlotStatusDescription(value) {
export function getSlotStatusClass(status) { export function getSlotStatusClass(status) {
return SlotStatusClassMap[status] || 'status-unbooked'; return SlotStatusClassMap[status] || 'status-unbooked';
} }
/**
* 诊疗项目分类代码(对应后端 ActivityDefCategory 枚举)
* wor_activity_definition.category_code 字段
*/
export const ActivityCategory = {
/** 治疗 */
TREATMENT: '21',
/** 检验 */
PROOF: '22',
/** 检查 */
TEST: '23',
/** 手术 */
PROCEDURE: '24',
};

View File

@@ -34,7 +34,6 @@
<select id="status-select" class="search-select" v-model="selectedStatus" @change="onSearch"> <select id="status-select" class="search-select" v-model="selectedStatus" @change="onSearch">
<option value="all">全部</option> <option value="all">全部</option>
<option value="unbooked">未预约</option> <option value="unbooked">未预约</option>
<option value="locked">已锁定</option>
<option value="booked">已预约</option> <option value="booked">已预约</option>
<option value="checked">已取号</option> <option value="checked">已取号</option>
<option value="cancelled">已停诊</option> <option value="cancelled">已停诊</option>
@@ -254,7 +253,6 @@ import useUserStore from '@/store/modules/user';
const STATUS_CLASS_MAP = { const STATUS_CLASS_MAP = {
'未预约': 'status-unbooked', '未预约': 'status-unbooked',
'已锁定': 'status-locked',
'已预约': 'status-booked', '已预约': 'status-booked',
'已取号': 'status-checked', '已取号': 'status-checked',
'已退号': 'status-returned', '已退号': 'status-returned',
@@ -776,7 +774,6 @@ export default {
// 🔧 BugFix#399: 确保已取号状态正确匹配 // 🔧 BugFix#399: 确保已取号状态正确匹配
const statusMap = { const statusMap = {
unbooked: ['未预约'], unbooked: ['未预约'],
locked: ['已锁定'],
booked: ['已预约'], booked: ['已预约'],
checked: ['已取号', '已签到'], checked: ['已取号', '已签到'],
cancelled: ['已停诊', '已取消'], cancelled: ['已停诊', '已取消'],

View File

@@ -1685,7 +1685,7 @@ function loadCheckInPatientList() {
const today = formatDateStr(new Date(), 'YYYY-MM-DD'); const today = formatDateStr(new Date(), 'YYYY-MM-DD');
listTicket({ listTicket({
date: today, date: today,
status: 'locked', status: 'booked',
name: checkInSearchKey.value, // 支持姓名等模糊查询,后端需适配 name: checkInSearchKey.value, // 支持姓名等模糊查询,后端需适配
page: checkInPage.value, page: checkInPage.value,
limit: checkInLimit.value limit: checkInLimit.value

View File

@@ -461,10 +461,6 @@ watch(
console.log(prescriptionList.value,"prescriptionList.value") console.log(prescriptionList.value,"prescriptionList.value")
if(newValue&&newValue.length>0){ if(newValue&&newValue.length>0){
let saveList = prescriptionList.value.filter((item) => { let saveList = prescriptionList.value.filter((item) => {
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag
if (isSurgeryChargeBillingContext()) {
return item.check && item.statusEnum == 1
}
return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag) return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
}) })
console.log(saveList,"prescriptionList.value") console.log(saveList,"prescriptionList.value")
@@ -877,32 +873,6 @@ 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() { function handleDelete() {
// 🔧 修复:使用 groupIndexList 而不是 check 属性 // 🔧 修复:使用 groupIndexList 而不是 check 属性
// 因为 watch 监听器会在数据更新时重置 check 为 false // 因为 watch 监听器会在数据更新时重置 check 为 false
@@ -911,93 +881,80 @@ function handleDelete() {
return; return;
} }
const canDeleteRow = (item) => let deleteList = groupIndexList.value.map((index) => {
isPendingUnsignedAndUnpaid(item) && isBizRequestAllowedForDelete(item) const item = prescriptionList.value[index];
// 只删除待签发且未收费的项目
const anySelectedDeletable = groupIndexList.value.some((index) => if (item.statusEnum != 1 || item.chargeStatus == 5) {
canDeleteRow(prescriptionList.value[index]) return null;
)
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
for (const index of sortedIndexes) {
const item = prescriptionList.value[index]
if (!canDeleteRow(item)) {
continue
} }
// 🔧 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 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; // 跳过已签发的项目
}
if (!item.requestId) { if (!item.requestId) {
// 新增的行(未保存到数据库),直接删除 // 新增的行(未保存到数据库),直接删除
prescriptionList.value.splice(index, 1) prescriptionList.value.splice(index, 1);
} else { } else {
hasSavedItem = true hasSavedItem = true;
} }
} }
if (hasSavedItem) {
// 🔧 Bug #454: 删除前弹出确认提示,告知用户将级联删除关联检验申请单
const hasLabItem = deleteList.some(item => item.adviceType === 3);
const confirmMsg = hasLabItem
? '删除此医嘱将同时删除关联的检验申请单,是否确认删除?'
: '确认删除选中的医嘱项目吗?';
const cleanupAfterDelete = () => { 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('操作成功');
expandOrder.value = []; expandOrder.value = [];
groupIndexList.value = []; groupIndexList.value = [];
groupList.value = []; groupList.value = [];
isAdding.value = false; isAdding.value = false;
adviceQueryParams.value.adviceType = undefined; 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();
} }
} }
@@ -1029,9 +986,7 @@ function changeCheck(value,index,row){
groupList.value.map(k=>{ groupList.value.map(k=>{
if(k.check){ if(k.check){
if(k.statusEnum == 1){//待签发 if(k.statusEnum == 1){//待签发
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag if(Number(k.bizRequestFlag)==1||!k.bizRequestFlag){
const bizAllowed = isSurgeryChargeBillingContext() || Number(k.bizRequestFlag)==1||!k.bizRequestFlag
if(bizAllowed){
if(handleSaveDisabled.value&&!handleSingOutDisabled.value&&groupList.value.length>1){ if(handleSaveDisabled.value&&!handleSingOutDisabled.value&&groupList.value.length>1){
proxy.$modal.msgWarning('请选择相同的状态的项目进行操作') proxy.$modal.msgWarning('请选择相同的状态的项目进行操作')
return return
@@ -1046,9 +1001,7 @@ function changeCheck(value,index,row){
} }
} }
if(k.statusEnum == 2){ //已签发 if(k.statusEnum == 2){ //已签发
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag if(Number(k.bizRequestFlag)==1||!k.bizRequestFlag){
const bizAllowed = isSurgeryChargeBillingContext() || Number(k.bizRequestFlag)==1||!k.bizRequestFlag
if(bizAllowed){
if(!handleSaveDisabled.value&&handleSingOutDisabled.value&&groupList.value.length>1){ if(!handleSaveDisabled.value&&handleSingOutDisabled.value&&groupList.value.length>1){
proxy.$modal.msgWarning('请选择相同的状态的项目进行操作') proxy.$modal.msgWarning('请选择相同的状态的项目进行操作')
return return
@@ -1075,28 +1028,28 @@ function handleSave() {
return; return;
} }
let saveList = prescriptionList.value.filter((item) => { let saveList = prescriptionList.value.filter((item) => {
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag允许任何授权用户签发
// 门诊划价场景保留 bizRequestFlag 限制,只能操作本人开立的医嘱
if (isSurgeryChargeBillingContext()) {
return item.check && item.statusEnum == 1
}
return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag) return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
}); });
// 无可签发项目时提前返回,避免后端报"医嘱列表为空" // let saveList = prescriptionList.value
if (saveList.length == 0) { // .filter((item) => {
proxy.$modal.msgWarning('当前无可签发处方'); // return item.check;
return; // }).filter((item) => {
} // return item.statusEnum == 1&&item.bizRequestFlag==1
// })
// if (saveList.length == 0) {
// proxy.$modal.msgWarning('当前无可签发处方');
// return;
// }
// 此处签发处方和单行保存处方传参相同后台已经将传参存为JSON字符串此处直接转换为JSON即可 // 此处签发处方和单行保存处方传参相同后台已经将传参存为JSON字符串此处直接转换为JSON即可
let list = saveList.map((item) => { let list = saveList.map((item) => {
const parsedContent = item.contentJson ? JSON.parse(item.contentJson) : {}; const parsedContent = item.contentJson ? JSON.parse(item.contentJson) : {};
return { return {
...parsedContent, ...parsedContent,
requestId: item.requestId, requestId: item.requestId,
// 已有 requestId 的记录走 UPDATE 路径,新记录走 INSERT 路径 dbOpType: '1',
dbOpType: item.requestId ? '2' : '1',
groupId: item.groupId, groupId: item.groupId,
// 补充顶层关键字段(这些可能不在 contentJson 中,需从 API 响应顶层提取) // 🔧 Bug #443: 补充顶层关键字段(这些不在 contentJson 中,需从 API 响应顶层提取)
encounterId: item.encounterId, encounterId: item.encounterId,
patientId: item.patientId, patientId: item.patientId,
locationId: item.positionId, locationId: item.positionId,
@@ -1104,14 +1057,6 @@ function handleSave() {
adviceTableName: item.adviceTableName, adviceTableName: item.adviceTableName,
adviceDefinitionId: item.adviceDefinitionId, adviceDefinitionId: item.adviceDefinitionId,
chargeItemId: item.chargeItemId, chargeItemId: item.chargeItemId,
// 补充数量、单位、批号等字段(后端 handDevice 需要这些字段)
quantity: item.quantity,
unitCode: item.unitCode,
lotNumber: item.lotNumber,
categoryEnum: item.categoryEnum,
// 签发时显式设置手术计费关键字段,后端 generateSourceEnum 回退为默认值导致查询无法匹配
generateSourceEnum: props.generateSourceEnum ?? parsedContent.generateSourceEnum ?? item.generateSourceEnum,
sourceBillNo: props.sourceBillNo ?? parsedContent.sourceBillNo ?? item.sourceBillNo,
}; };
}); });
// 确保 organizationId 不为 undefined手术计费场景下可能缺失 orgId // 确保 organizationId 不为 undefined手术计费场景下可能缺失 orgId
@@ -1173,9 +1118,8 @@ function handleSaveSign(row, index) {
cleanRow.generateSourceEnum = 6; // 手术计费 cleanRow.generateSourceEnum = 6; // 手术计费
cleanRow.sourceBillNo = props.patientInfo.sourceBillNo; cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
} }
// 🔧 门诊计费场景:保存为草稿,让药品出现在临时医嘱弹窗"已引用计费药品(待生成医嘱)"中 console.log('cleanRow', cleanRow)
const adviceOpType = props.patientInfo.sourceBillNo ? '0' : '1' savePrescription({ adviceSaveList: [cleanRow] }, '1').then((res) => {
savePrescription({ adviceSaveList: [cleanRow] }, adviceOpType).then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$modal.msgSuccess('保存成功'); proxy.$modal.msgSuccess('保存成功');
getListInfo(false); getListInfo(false);
@@ -1199,10 +1143,6 @@ function handleSingOut() {
return item.check; return item.check;
}) })
.filter((item) => { .filter((item) => {
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag
if (isSurgeryChargeBillingContext()) {
return item.statusEnum == 2 && item.chargeStatus != 5
}
return item.statusEnum == 2 && item.chargeStatus != 5 && (Number(item.bizRequestFlag)==1||!item.bizRequestFlag) return item.statusEnum == 2 && item.chargeStatus != 5 && (Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
}) })
.map((item) => { .map((item) => {

View File

@@ -354,11 +354,6 @@ async function getList() {
if (!item.classification) { if (!item.classification) {
item.classification = '西医'; item.classification = '西医';
} }
// 转换 longTermFlag 为字符串,以匹配 useDict 返回的字典值类型(字符串)
// 避免 el-select 因类型不匹配(整数 1 vs 字符串 "1")导致下拉框清空
if (item.longTermFlag != null) {
item.longTermFlag = String(item.longTermFlag);
}
// 如果ybNo诊断编码符合传染病编码格式添加到selectedDiseases // 如果ybNo诊断编码符合传染病编码格式添加到selectedDiseases
if (item.ybNo && /^(01|02|03)/.test(item.ybNo)) { if (item.ybNo && /^(01|02|03)/.test(item.ybNo)) {
item.selectedDiseases = [item.ybNo]; item.selectedDiseases = [item.ybNo];
@@ -749,26 +744,22 @@ function handleInfectiousDiseaseReport() {
'手足口病': '0311', '手足口病': '0311',
}; };
// 获取所有命中传染病映射的诊断,但跳过已有已提交报卡的诊断 // 获取所有诊断名称对应的报卡编码,但跳过已有已提交报卡的诊断
const infectiousDiagnoses = form.value.diagnosisList const allSelectedDiseases = form.value.diagnosisList
.map(d => ({ .filter(d => d.name && d.hasInfectiousReport !== 1)
diagnosis: d, .map(d => diseaseNameToCode[d.name] || null)
diseaseCode: d.name && d.hasInfectiousReport !== 1 ? diseaseNameToCode[d.name] : null .filter(code => code);
}))
.filter(item => item.diseaseCode);
const allSelectedDiseases = infectiousDiagnoses.map(item => item.diseaseCode);
if (allSelectedDiseases.length === 0) { if (allSelectedDiseases.length === 0) {
return; return;
} }
// 优先使用命中传染病映射的主诊断,否则使用第一条命中的传染病诊断 // 优先使用主诊断(同样跳过已有报卡的)
const mainInfectiousDiagnosis = infectiousDiagnoses.find(item => item.diagnosis.maindiseFlag === 1)?.diagnosis; const mainDiagnosis = form.value.diagnosisList.find(d => d.maindiseFlag === 1 && d.hasInfectiousReport !== 1);
const firstInfectiousDiagnosis = infectiousDiagnoses[0].diagnosis; const firstDiagnosis = form.value.diagnosisList.find(d => d.hasInfectiousReport !== 1) || form.value.diagnosisList[0];
const diagnosisToShow = { const diagnosisToShow = {
...(mainInfectiousDiagnosis || firstInfectiousDiagnosis), ...(mainDiagnosis || firstDiagnosis),
selectedDiseases: allSelectedDiseases selectedDiseases: allSelectedDiseases
}; };

View File

@@ -548,8 +548,6 @@ const dialogReadOnly = ref(false);
const formRef = ref(null); const formRef = ref(null);
// 保存按钮加载状态,防止重复提交 // 保存按钮加载状态,防止重复提交
const submitLoading = ref(false); const submitLoading = ref(false);
// 数据加载中标志,防止 showReport 加载已有数据时 watch 清空分型字段
const loadingData = ref(false);
const props = defineProps({ const props = defineProps({
title: { title: {
@@ -942,8 +940,7 @@ const showSubtypeSelect = computed(() => {
// 监听疾病选择变化,自动清空分型选择 // 监听疾病选择变化,自动清空分型选择
watch(() => [form.value.selectedClassA, form.value.selectedClassB, form.value.selectedClassC], (newVal, oldVal) => { watch(() => [form.value.selectedClassA, form.value.selectedClassB, form.value.selectedClassC], (newVal, oldVal) => {
// 如果疾病选择发生变化,清空分型选择 // 如果疾病选择发生变化,清空分型选择
// 数据加载中时不清空,避免 showReport 加载已有数据时被错误清空 if (JSON.stringify(newVal) !== JSON.stringify(oldVal)) {
if (JSON.stringify(newVal) !== JSON.stringify(oldVal) && !loadingData.value) {
form.value.diseaseType = ''; form.value.diseaseType = '';
} }
}, { deep: true }); }, { deep: true });
@@ -1095,9 +1092,6 @@ function showReport(reportData = {}, readOnly = true) {
dialogVisible.value = true; dialogVisible.value = true;
dialogReadOnly.value = readOnly; dialogReadOnly.value = readOnly;
// 标记数据加载中,防止 watch 清空 diseaseType 分型字段
loadingData.value = true;
resetAddressSelector(); resetAddressSelector();
initProvinceOptions(); initProvinceOptions();
@@ -1155,9 +1149,6 @@ function showReport(reportData = {}, readOnly = true) {
form.value.addressCounty, form.value.addressCounty,
form.value.addressTown form.value.addressTown
); );
// 数据加载完成,恢复 watch 监听
loadingData.value = false;
} }
/** /**
@@ -1442,7 +1433,7 @@ async function buildSubmitData() {
const submitData = { const submitData = {
cardNo: formData.cardNo, cardNo: formData.cardNo,
visitId: props.patientInfo?.encounterId || formData.encounterId || null, visitId: props.patientInfo?.encounterId || formData.encounterId || null,
diagId: formData.diagnosisId || null, diagId: formData.diagnosisId ? Number(formData.diagnosisId) : null,
patId: formData.patientId || null, patId: formData.patientId || null,
idType: 1, // 默认身份证 idType: 1, // 默认身份证
idNo: formData.idNo, idNo: formData.idNo,

View File

@@ -316,13 +316,13 @@
<!-- Bug #384修复: 单价显示套餐价格(如果选中)或部位价格 --> <!-- Bug #384修复: 单价显示套餐价格(如果选中)或部位价格 -->
<el-table-column label="单价" width="75" align="right"> <el-table-column label="单价" width="75" align="right">
<template #default="scope"> <template #default="scope">
{{ formatDetailAmount(getSelectedItemAmount(scope.row)) }} {{ scope.row.selectedMethod?.packagePrice || scope.row.price }}
</template> </template>
</el-table-column> </el-table-column>
<!-- Bug #384修复: 金额使用有效价格计算 --> <!-- Bug #384修复: 金额使用有效价格计算 -->
<el-table-column label="金额" width="80" align="right"> <el-table-column label="金额" width="80" align="right">
<template #default="scope"> <template #default="scope">
{{ formatDetailAmount(getSelectedItemAmount(scope.row) * (scope.row.quantity || 1)) }} {{ ((scope.row.selectedMethod?.packagePrice || scope.row.price || 0) * (scope.row.quantity || 1)).toFixed(2) }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="类型" prop="checkType" width="70" align="center" /> <el-table-column label="类型" prop="checkType" width="70" align="center" />
@@ -392,6 +392,37 @@
<div v-if="categoryLoadingSet.has(cat.typeId)" class="category-loading-hint"> <div v-if="categoryLoadingSet.has(cat.typeId)" class="category-loading-hint">
加载中... 加载中...
</div> </div>
<!-- Bug #428修复: 渲染分类联动加载的检查方法列表 -->
<!-- Bug #500修复: v-if 改为 v-show避免方法列表加载时 DOM 突然插入导致高度跳变 -->
<div
v-show="cat.methods && cat.methods.length > 0"
class="method-section"
>
<div class="method-section-title">检查方法</div>
<div
v-for="method in cat.methods"
:key="method.id"
class="method-row"
>
<el-checkbox
:model-value="isMethodSelected(method, cat)"
@change="(val) => handleMethodSelect(val, method, cat)"
class="method-checkbox"
>
{{ method.name }}
</el-checkbox>
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span>
</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>
@@ -409,103 +440,46 @@
class="selected-item-card" class="selected-item-card"
:class="{ 'is-expanded': item.expanded }" :class="{ 'is-expanded': item.expanded }"
> >
<!-- 项目卡片头部:项目和检查方法解耦,点击展开查看方法/明细 --> <!-- Bug #384修复 + #426修复: 项目卡片头部,可展开/收起 -->
<div class="card-header" @click="toggleItemExpand(item)"> <div class="card-header" @click="toggleItemExpand(item)">
<el-tooltip :content="getDisplayItemName(item)" placement="top" :show-after="400"> <el-tag v-if="item.isPackage || item.packageName" size="small" type="warning" style="margin-right: 4px; flex-shrink: 0;">套餐</el-tag>
<span class="card-name">{{ getDisplayItemName(item) }}</span> <el-tooltip :content="item.name" placement="top" :show-after="400">
<span class="card-name">{{ item.name }}</span>
</el-tooltip> </el-tooltip>
<span class="card-price">¥{{ formatDetailAmount(getSelectedItemAmount(item)) }}</span> <span class="card-price">¥{{ formatDetailAmount(item.price) }}</span>
<el-icon :class="['expand-icon', { expanded: item.expanded }]"> <el-icon :class="['expand-icon', { expanded: item.expanded }]">
<ArrowDown /> <ArrowDown v-if="!item.expanded" />
<ArrowUp v-if="item.expanded" />
</el-icon> </el-icon>
<!-- 删除按钮 --> <!-- 删除按钮 -->
<el-button link type="danger" size="small" @click.stop="handleRemoveItem(idx, item)"> <el-button link type="danger" size="small" @click.stop="handleRemoveItem(idx, item)">
<el-icon><Close /></el-icon> <el-icon><Close /></el-icon>
</el-button> </el-button>
</div> </div>
<div v-if="item.expanded" class="selected-card-body"> <!-- Bug #428: 有套餐 ID 时默认展开;加载中/空/明细均在本区域展示 -->
<div v-if="shouldShowItemPackageBody(item)"> <div v-if="item.expanded && shouldShowPackageBody(item)" class="selected-card-body">
<div class="package-toggle" @click.stop="toggleItemPackageExpand(item)"> <div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div>
<span>项目套餐明细</span> <template v-else>
<el-icon :class="['expand-icon', { expanded: item.itemPackageExpanded }]"> <div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty">
<ArrowDown /> 暂无套餐明细
</el-icon>
</div> </div>
<div v-show="item.itemPackageExpanded"> <div v-else class="package-details-list">
<div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div> <div class="package-details-head">套餐明细</div>
<template v-else> <div
<div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty"> v-for="(detail, dIdx) in getPackageDetailsList(item)"
暂无套餐明细 :key="detail.id ?? detail.itemCode ?? `d-${dIdx}`"
</div> class="detail-row"
<div v-else class="package-details-list">
<div
v-for="(detail, dIdx) in getPackageDetailsList(item)"
:key="detail.id ?? detail.itemCode ?? `d-${dIdx}`"
class="detail-row"
>
<el-tooltip :content="detail.name" placement="top" :show-after="500">
<span class="detail-name">{{ detail.name }}</span>
</el-tooltip>
<div class="detail-meta">
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
</div>
</div>
</div>
</template>
</div>
</div>
<div class="selected-card-section">
<div class="selected-section-title">检查方法</div>
<div v-if="!item.methods || item.methods.length === 0" class="selected-method-empty">
暂无检查方法
</div>
<div
v-for="method in item.methods"
:key="method.id"
class="selected-method-option"
>
<el-checkbox
:model-value="item.selectedMethod?.id === method.id"
@change="(val) => selectMethodCheckbox(val, item, method)"
class="method-checkbox"
> >
{{ method.name }} <el-tooltip :content="detail.name" placement="top" :show-after="500">
</el-checkbox> <span class="detail-name">{{ detail.name }}</span>
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span> </el-tooltip>
</div> <div class="detail-meta">
</div> <span class="detail-qty">×{{ detail.quantity || 1 }}</span>
<div v-if="shouldShowMethodPackageBody(item)"> <span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
<div class="package-toggle" @click.stop="toggleMethodPackageExpand(item)">
<span>检查方法套餐明细</span>
<el-icon :class="['expand-icon', { expanded: item.methodPackageExpanded }]">
<ArrowDown />
</el-icon>
</div>
<div v-show="item.methodPackageExpanded">
<div v-if="item.methodPackageLoading" class="package-details-loading">加载中...</div>
<template v-else>
<div v-if="getMethodPackageDetailsList(item).length === 0" class="package-details-empty">
暂无检查方法套餐明细
</div>
<div v-else class="package-details-list method-package-list">
<div
v-for="(detail, dIdx) in getMethodPackageDetailsList(item)"
:key="detail.id ?? detail.itemCode ?? `md-${dIdx}`"
class="detail-row"
>
<el-tooltip :content="detail.name" placement="top" :show-after="500">
<span class="detail-name">{{ detail.name }}</span>
</el-tooltip>
<div class="detail-meta">
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
</div>
</div> </div>
</div> </div>
</template>
</div> </div>
</div> </template>
</div> </div>
</div> </div>
</div> </div>
@@ -519,7 +493,7 @@
<script setup> <script setup>
import { ref, reactive, computed, watch, onMounted, nextTick } from 'vue'; import { ref, reactive, computed, watch, onMounted, nextTick } from 'vue';
import { ElMessage, ElMessageBox } from 'element-plus'; import { ElMessage, ElMessageBox } from 'element-plus';
import { Printer, Delete, ArrowDown, Close } from '@element-plus/icons-vue'; import { Printer, Delete, ArrowDown, ArrowUp, Close } from '@element-plus/icons-vue';
import useUserStore from '@/store/modules/user'; import useUserStore from '@/store/modules/user';
import request from '@/utils/request'; import request from '@/utils/request';
import { listCheckMethod, searchCheckMethod, listCheckPackage } from '@/api/system/checkType'; import { listCheckMethod, searchCheckMethod, listCheckPackage } from '@/api/system/checkType';
@@ -650,48 +624,24 @@ async function loadPackageDetails(row, treeNode, resolve) {
} }
} }
// #428修复 + #426修复: 为已选择项目加载套餐明细通过packageId或packageName查询
/** 套餐明细挂在「部位」或已选的「检查方法」上(方法可带 packageId */
function getPackageCarrier(item) {
return item?.selectedMethod?.packageId ? item.selectedMethod : item;
}
function getPackageDetailsList(item) { function getPackageDetailsList(item) {
// 明细挂在行对象上,避免仅写入 methods 内嵌对象时首帧不触发视图更新(体感需点两次才展开) // 明细挂在行对象上,避免仅写入 methods 内嵌对象时首帧不触发视图更新(体感需点两次才展开)
if (Array.isArray(item?.packageDetailsDisplay)) { if (Array.isArray(item?.packageDetailsDisplay)) {
return item.packageDetailsDisplay; return item.packageDetailsDisplay;
} }
return Array.isArray(item?.packageDetails) ? item.packageDetails : []; const carrier = getPackageCarrier(item);
} return Array.isArray(carrier?.packageDetails) ? carrier.packageDetails : [];
function getMethodPackageDetailsList(item) {
if (Array.isArray(item?.methodPackageDetails)) {
return item.methodPackageDetails;
}
return Array.isArray(item?.selectedMethod?.packageDetails) ? item.selectedMethod.packageDetails : [];
} }
/** 有套餐 ID 或 packageName 的已选行才展示右侧套餐区(加载中 / 空 / 明细列表) */ /** 有套餐 ID 或 packageName 的已选行才展示右侧套餐区(加载中 / 空 / 明细列表) */
function shouldShowPackageBody(item) { function shouldShowPackageBody(item) {
return shouldShowItemPackageBody(item) || shouldShowMethodPackageBody(item); return !!(getPackageCarrier(item)?.packageId || item.packageName || item.packageId);
}
function hasItemPackage(item) {
return !!(item?.packageId || item?.packageName);
}
function hasMethodPackage(item) {
return !!(item?.selectedMethod?.packageId || item?.selectedMethod?.packageName);
}
function isSamePackage(item) {
if (!hasItemPackage(item) || !hasMethodPackage(item)) return false;
if (item.packageId && item.selectedMethod?.packageId) {
return String(item.packageId) === String(item.selectedMethod.packageId);
}
return String(item.packageName || '') === String(item.selectedMethod?.packageName || '');
}
function shouldShowItemPackageBody(item) {
return hasItemPackage(item);
}
function shouldShowMethodPackageBody(item) {
return hasMethodPackage(item) && !isSamePackage(item);
} }
/** 金额展示:统一两位小数 */ /** 金额展示:统一两位小数 */
@@ -700,18 +650,20 @@ function formatDetailAmount(value) {
return Number.isFinite(n) ? n.toFixed(2) : '0.00'; return Number.isFinite(n) ? n.toFixed(2) : '0.00';
} }
/** 已选卡片名称:去掉 UI 上冗余的“套餐”前缀,完整名称通过 tooltip 展示 */ /** 默认检查方法:优先与部位 packageId 一致的方法,否则取首个带套餐的方法,否则取第一个 */
function getDisplayItemName(item) { function pickDefaultMethod(methods, partItem) {
return String(item?.name || '').replace(/^套餐[:\-\s]*/, ''); if (!methods?.length) return null;
} if (methods.length === 1) return methods[0];
const pid = partItem?.packageId ?? null;
function getSelectedItemAmount(item) { if (pid != null && pid !== '') {
const itemPrice = Number(item?.price || 0); const matched = methods.find(
const methodPrice = Number(item?.selectedMethod?.packagePrice || 0); (x) => x.packageId != null && String(x.packageId) === String(pid)
if (!hasMethodPackage(item) || isSamePackage(item)) { );
return itemPrice; if (matched) return matched;
} }
return itemPrice + methodPrice; const withPkg = methods.find((x) => x.packageId != null);
if (withPkg) return withPkg;
return methods[0];
} }
function parsePackageDetailsPayload(res) { function parsePackageDetailsPayload(res) {
@@ -727,15 +679,15 @@ function parsePackageDetailsPayload(res) {
// #428: 为已选择项目加载套餐明细后端CheckTypeController /system/check-type/package/{id}/details // #428: 为已选择项目加载套餐明细后端CheckTypeController /system/check-type/package/{id}/details
async function loadPackageDetailsForItem(item) { async function loadPackageDetailsForItem(item) {
let packageId = item.packageId; const carrier = getPackageCarrier(item);
const packageName = item.packageName; let packageId = item.packageId || carrier?.packageId;
if (!packageId && !packageName) { if (!packageId && !item.packageName) {
return; return;
} }
item.packageDetailsLoading = true; item.packageDetailsLoading = true;
try { try {
if (!packageId && packageName) { if (!packageId && item.packageName) {
const pkgRes = await listCheckPackage({ packageName }); const pkgRes = await listCheckPackage({ packageName: item.packageName });
let packages = pkgRes?.data || []; let packages = pkgRes?.data || [];
if (!Array.isArray(packages)) { if (!Array.isArray(packages)) {
packages = packages.records || packages.data || []; packages = packages.records || packages.data || [];
@@ -747,7 +699,6 @@ async function loadPackageDetailsForItem(item) {
} }
packageId = packages[0].id; packageId = packages[0].id;
item.packageId = packageId; item.packageId = packageId;
item.packageName = item.packageName || packageName;
} }
if (!packageId) { if (!packageId) {
item.packageDetails = []; item.packageDetails = [];
@@ -767,6 +718,7 @@ async function loadPackageDetailsForItem(item) {
quantity: detail.quantity || 1 quantity: detail.quantity || 1
})); }));
item.packageDetailsDisplay = mapped; item.packageDetailsDisplay = mapped;
carrier.packageDetails = mapped;
if (res.code === 200 && res.data) { if (res.code === 200 && res.data) {
item.packageDetails = Array.isArray(res.data) item.packageDetails = Array.isArray(res.data)
? res.data.map((detail) => ({ ? res.data.map((detail) => ({
@@ -783,6 +735,7 @@ async function loadPackageDetailsForItem(item) {
} catch (err) { } catch (err) {
console.error('加载套餐明细失败:', err); console.error('加载套餐明细失败:', err);
item.packageDetailsDisplay = []; item.packageDetailsDisplay = [];
carrier.packageDetails = [];
item.packageDetails = []; item.packageDetails = [];
} finally { } finally {
item.packageDetailsLoading = false; item.packageDetailsLoading = false;
@@ -1105,10 +1058,7 @@ async function loadCategoryList() {
// 默认展开第一个 // 默认展开第一个
if (categoryList.value.length > 0) { if (categoryList.value.length > 0) {
const firstCat = categoryList.value[0]; activeNames.value = categoryList.value[0].typeId;
activeNames.value = firstCat.typeId;
await nextTick();
await handleCategoryExpand(firstCat);
} }
} catch (err) { } catch (err) {
console.error('加载检查项目分类失败', err); console.error('加载检查项目分类失败', err);
@@ -1129,9 +1079,10 @@ const filteredCategoryList = computed(() => {
}); });
// ====== 合计 ====== // ====== 合计 ======
// Bug #384修复: 如果选中了检查方法,使用套餐价格;否则使用部位价格
const totalAmountCalc = computed(() => { const totalAmountCalc = computed(() => {
const total = selectedItems.value.reduce((sum, item) => { const total = selectedItems.value.reduce((sum, item) => {
const effectivePrice = getSelectedItemAmount(item); const effectivePrice = item.selectedMethod?.packagePrice || item.price;
return sum + (effectivePrice * (item.quantity || 1)); return sum + (effectivePrice * (item.quantity || 1));
}, 0); }, 0);
return total.toFixed(2); return total.toFixed(2);
@@ -1264,7 +1215,8 @@ function handleSave() {
itemCode: String(item.id), itemCode: String(item.id),
itemName: item.name, itemName: item.name,
bodyPartCode: item.checkType || 'unknown', bodyPartCode: item.checkType || 'unknown',
itemFee: getSelectedItemAmount(item), // Bug #384修复: 如果选中了检查方法且有套餐价格,使用套餐价格;否则使用部位价格
itemFee: item.selectedMethod?.packagePrice || item.price,
performDeptCode: form.performDeptCode || '', performDeptCode: form.performDeptCode || '',
itemStatus: 0, itemStatus: 0,
itemSeq: index + 1, itemSeq: index + 1,
@@ -1317,8 +1269,6 @@ function handleRowClick(row) {
methods: [], methods: [],
selectedMethod: null, selectedMethod: null,
expanded: false, expanded: false,
itemPackageExpanded: true,
methodPackageExpanded: true,
packageDetailsLoading: false, packageDetailsLoading: false,
isPackage: false, isPackage: false,
packageId: null, packageId: null,
@@ -1348,10 +1298,17 @@ function handleRowClick(row) {
if (m.checkMethodId) { if (m.checkMethodId) {
item.selectedMethod = item.methods.find(md => String(md.id) === String(m.checkMethodId)) || null; item.selectedMethod = item.methods.find(md => String(md.id) === String(m.checkMethodId)) || null;
if (item.selectedMethod?.packageId) { if (item.selectedMethod?.packageId) {
item.isPackage = true;
item.packageId = item.selectedMethod.packageId;
item.hasChildren = true; // #426修复 item.hasChildren = true; // #426修复
} }
} }
if (!item.selectedMethod && item.methods.length) {
item.selectedMethod = pickDefaultMethod(item.methods, { packageId: item.packageId });
}
if (item.selectedMethod?.packageId) { if (item.selectedMethod?.packageId) {
item.packageId = item.selectedMethod.packageId;
item.isPackage = true;
item.hasChildren = true; // #426修复 item.hasChildren = true; // #426修复
} }
} }
@@ -1365,21 +1322,14 @@ function handleRowClick(row) {
selectedItems.value = itemsWithMethods; selectedItems.value = itemsWithMethods;
// 加载套餐明细(单个失败不影响其他项目和明细显示) // 加载套餐明细(单个失败不影响其他项目和明细显示)
for (const it of selectedItems.value) { for (const it of selectedItems.value) {
if (hasItemPackage(it)) { if (getPackageCarrier(it)?.packageId) {
try { try {
await loadPackageDetailsForItem(it); await loadPackageDetailsForItem(it);
} catch (e) { } catch (e) {
console.error('加载套餐明细失败:', it.name, e); console.error('加载套餐明细失败:', it.name, e);
} }
} }
if (hasMethodPackage(it) && !isSamePackage(it)) { it.expanded = !!getPackageCarrier(it)?.packageId;
try {
await loadMethodPackageDetails(it, it.selectedMethod);
} catch (e) {
console.error('加载检查方法套餐明细失败:', it.name, e);
}
}
it.expanded = shouldShowPackageBody(it);
} }
syncCategoryChecked(); syncCategoryChecked();
// Bug #384修复: 回充后更新检查方法显示 // Bug #384修复: 回充后更新检查方法显示
@@ -1409,6 +1359,97 @@ function handleDelete(row) {
}); });
} }
// Bug #428修复: 判断某个检查方法是否已被选中(任意项目关联了该方法)
function isMethodSelected(method, cat) {
return selectedItems.value.some(item =>
item.selectedMethod?.id === method.id && item.checkType === cat.typeName
);
}
// Bug #428修复: 勾选检查方法
async function handleMethodSelect(checked, method, cat) {
if (checked) {
// 找到该方法所属的第一个检查项目
const targetItem = cat.items[0];
if (!targetItem) {
// 如果分类下没有项目,尝试从其他分类找同名项目或创建
console.warn('分类下没有检查项目,无法关联方法');
return;
}
// 如果该项目已存在,只更新 selectedMethod
const existingItem = selectedItems.value.find(s => s.id === targetItem.id);
if (existingItem) {
existingItem.selectedMethod = method;
// 从方法中获取套餐信息(支持 packageId 或 packageName 解析)
if (method.packageId || method.packageName) {
existingItem.isPackage = true;
existingItem.packageId = method.packageId || existingItem.packageId;
existingItem.hasChildren = true; // #426修复
existingItem.packageName = method.packageName || existingItem.packageName; // #428修复: 确保 packageName 同步
// 预加载套餐明细
loadPackageDetailsForItem(existingItem);
}
updateMethodDisplay();
return;
}
// 如果该项目不存在,创建一个并关联方法
if (selectedItems.value.length > 0) {
const currentCategory = selectedItems.value[0].checkType;
// Bug #428修复: 使用 cat.typeName 进行比较(与 newItem.checkType 保持一致)
const newCategory = cat.typeName || '';
if (currentCategory !== newCategory) {
ElMessage.warning('一个检查单不能同时选择多个项目类型的检查项目');
return;
}
}
const newItem = {
id: targetItem.id, name: targetItem.name,
price: targetItem.price, quantity: 1,
serviceFee: targetItem.serviceFee || 0,
unit: targetItem.unit || '次',
applyPart: targetItem.name,
checkType: cat.typeName,
nationalCode: targetItem.nationalCode || '',
checked: true,
methods: cat.methods || [method], // #428修复: 复制分类下全部方法,允许用户切换
selectedMethod: method,
expanded: false,
// 从方法或项目中获取套餐信息
isPackage: !!method.packageId || !!targetItem.packageName,
packageId: method.packageId || targetItem.packageId || null,
packageName: method.packageName || targetItem.packageName || null, // #428修复: 复制 packageName确保套餐明细可加载
hasChildren: !!(method.packageId || method.packageName || targetItem.packageId || targetItem.packageName) // #426修复: 树形表格懒加载展开标记,支持 packageName 解析
};
selectedItems.value.push(newItem);
// 如果是套餐,预加载套餐明细
if (newItem.isPackage && (newItem.packageId || newItem.packageName)) {
loadPackageDetailsForItem(newItem);
}
// 自动回填执行科室
if (selectedItems.value.length === 1 && cat?.performDeptName) {
form.performDeptCode = cat.performDeptName;
}
// 同时勾选左侧项目的 checkbox
targetItem.checked = true;
} else {
// 取消选择方法:将 selectedItems 中关联该方法的项的 selectedMethod 清空
const itemsWithMethod = selectedItems.value.filter(
item => item.selectedMethod?.id === method.id
);
for (const item of itemsWithMethod) {
item.selectedMethod = null;
}
}
updateMethodDisplay();
}
// ====== 勾选逻辑 ====== // ====== 勾选逻辑 ======
async function handleItemSelect(checked, item, cat) { async function handleItemSelect(checked, item, cat) {
if (checked) { if (checked) {
@@ -1465,8 +1506,6 @@ async function handleItemSelect(checked, item, cat) {
methods: methods, methods: methods,
selectedMethod: null, selectedMethod: null,
expanded: false, expanded: false,
itemPackageExpanded: true,
methodPackageExpanded: true,
isPackage: !!(item.packageId || item.packageName), isPackage: !!(item.packageId || item.packageName),
packageName: item.packageName || null, packageName: item.packageName || null,
packageDetailsLoading: false, packageDetailsLoading: false,
@@ -1477,15 +1516,15 @@ async function handleItemSelect(checked, item, cat) {
// 必须用数组里的响应式行,不能继续改局部 newRowpush 后列表内是 proxy改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」) // 必须用数组里的响应式行,不能继续改局部 newRowpush 后列表内是 proxy改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
const row = selectedItems.value[selectedItems.value.length - 1]; const row = selectedItems.value[selectedItems.value.length - 1];
// 勾选项目只加入项目列表,检查方法由用户在“检查方法”区域手动选择 // 右侧不再展示「检查方法」列表:自动选默认方法(保存、计价仍依赖 selectedMethod
row.selectedMethod = null; if (methods.length >= 1) {
row.selectedMethod = pickDefaultMethod(methods, item);
}
updateMethodDisplay(); updateMethodDisplay();
// 新勾选项目后默认展开,直接展示检查方法状态和套餐明细 // 有套餐 ID 时默认展开(先显示加载区,明细写入行对象 packageDetailsDisplay
row.expanded = true; row.expanded = !!getPackageCarrier(row)?.packageId;
row.itemPackageExpanded = true; if (getPackageCarrier(row)?.packageId) {
row.methodPackageExpanded = true;
if (hasItemPackage(row)) {
await loadPackageDetailsForItem(row); await loadPackageDetailsForItem(row);
} }
@@ -1511,35 +1550,13 @@ async function handleItemSelect(checked, item, cat) {
// Bug #384修复 + #426修复: 展开/收起项目卡片 // Bug #384修复 + #426修复: 展开/收起项目卡片
async function toggleItemExpand(item) { async function toggleItemExpand(item) {
item.expanded = !item.expanded; item.expanded = !item.expanded;
if (item.expanded && hasItemPackage(item) && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) { if (item.expanded && (item.isPackage || item.packageName) && (!item.packageDetails || item.packageDetails.length === 0) && !item.packageDetailsLoading) {
await loadPackageDetailsForItem(item); await loadPackageDetailsForItem(item);
} }
if ( if (item.expanded && shouldShowPackageBody(item)) {
item.expanded && if (getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
shouldShowMethodPackageBody(item) && await loadPackageDetailsForItem(item);
getMethodPackageDetailsList(item).length === 0 && }
!item.methodPackageLoading
) {
await loadMethodPackageDetails(item, item.selectedMethod);
}
}
async function toggleItemPackageExpand(item) {
item.itemPackageExpanded = !item.itemPackageExpanded;
if (item.itemPackageExpanded && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
await loadPackageDetailsForItem(item);
}
}
async function toggleMethodPackageExpand(item) {
item.methodPackageExpanded = !item.methodPackageExpanded;
if (
item.methodPackageExpanded &&
item.selectedMethod &&
getMethodPackageDetailsList(item).length === 0 &&
!item.methodPackageLoading
) {
await loadMethodPackageDetails(item, item.selectedMethod);
} }
} }
@@ -1547,8 +1564,9 @@ async function toggleMethodPackageExpand(item) {
async function selectMethodCheckbox(checked, item, method) { async function selectMethodCheckbox(checked, item, method) {
if (checked) { if (checked) {
item.selectedMethod = method; item.selectedMethod = method;
item.expanded = true; if (item.expanded && (method.packageId || method.packageName)) {
item.methodPackageExpanded = true; loadPackageDetailsForItem(item);
}
// 动态加载该方法对应的套餐明细 // 动态加载该方法对应的套餐明细
await loadMethodPackageDetails(item, method); await loadMethodPackageDetails(item, method);
} else { } else {
@@ -1569,43 +1587,36 @@ async function loadMethodPackageDetails(item, method) {
item.methodPackageLoading = true; item.methodPackageLoading = true;
item.methodPackageDetails = []; item.methodPackageDetails = [];
try { try {
let packageId = method.packageId; if (!method.packageName) {
if (!packageId && !method.packageName) {
item.methodPackageLoading = false; item.methodPackageLoading = false;
return; return;
} }
// 通过packageName查询套餐获取packageId // 通过packageName查询套餐获取packageId
if (!packageId && method.packageName) { const pkgRes = await listCheckPackage({ packageName: method.packageName });
const pkgRes = await listCheckPackage({ packageName: method.packageName }); let packages = pkgRes?.data || [];
let packages = pkgRes?.data || []; if (!Array.isArray(packages)) {
if (!Array.isArray(packages)) { packages = packages.records || packages.data || [];
packages = packages.records || packages.data || [];
}
if (packages.length === 0) {
item.methodPackageLoading = false;
return;
}
packageId = packages[0].id;
method.packageId = packageId;
} }
if (packages.length === 0) {
item.methodPackageLoading = false;
return;
}
const packageId = packages[0].id;
// 查询套餐明细 // 查询套餐明细
const detailRes = await request({ const detailRes = await request({
url: `/system/check-type/package/${packageId}/details`, url: `/system/package/${packageId}/details`,
method: 'get' method: 'get'
}); });
const detailList = parsePackageDetailsPayload(detailRes); if (detailRes.code === 200 && detailRes.data) {
if (detailList.length > 0) { item.methodPackageDetails = detailRes.data.map(d => ({
const mapped = detailList.map(d => ({
id: d.id, id: d.id,
name: d.name || d.itemName, name: d.itemName || d.name,
quantity: d.quantity || 1, quantity: d.quantity || 1,
unit: d.unit || '次', unit: d.unit || '次',
price: d.price ?? d.unitPrice ?? d.itemPrice ?? 0, price: d.unitPrice || d.price || 0,
amount: d.amount || d.total || 0, amount: d.amount || d.total || 0,
checked: true // 默认勾选 checked: true // 默认勾选
})); }));
item.methodPackageDetails = mapped;
method.packageDetails = mapped;
} }
} catch (err) { } catch (err) {
console.error('加载方法套餐明细失败:', err); console.error('加载方法套餐明细失败:', err);
@@ -1619,18 +1630,20 @@ async function loadMethodPackageDetails(item, method) {
async function onDetailMethodChange(row, val) { async function onDetailMethodChange(row, val) {
row.selectedMethod = val || null; row.selectedMethod = val || null;
if (val?.packageId || val?.packageName) { if (val?.packageId || val?.packageName) {
row.packageId = val.packageId || row.packageId;
row.packageName = val.packageName || row.packageName;
row.isPackage = true;
row.hasChildren = true; // #426修复 row.hasChildren = true; // #426修复
} }
row.methodPackageDetails = []; row.packageDetailsDisplay = undefined;
updateMethodDisplay(); const carrier = getPackageCarrier(row);
row.expanded = shouldShowPackageBody(row); if (carrier) {
row.itemPackageExpanded = true; carrier.packageDetails = undefined;
row.methodPackageExpanded = true;
if (hasItemPackage(row)) {
await loadPackageDetailsForItem(row);
} }
if (val?.packageId || val?.packageName) { updateMethodDisplay();
await loadMethodPackageDetails(row, val); row.expanded = !!getPackageCarrier(row)?.packageId;
if (getPackageCarrier(row)?.packageId) {
await loadPackageDetailsForItem(row);
} }
nextTick(() => { nextTick(() => {
form.totalAmount = totalAmountCalc.value; form.totalAmount = totalAmountCalc.value;
@@ -1781,7 +1794,7 @@ defineExpose({ getList });
/* 右:分类面板 */ /* 右:分类面板 */
.category-panel { .category-panel {
width: 560px; width: 420px;
flex-shrink: 0; flex-shrink: 0;
background: #fff; background: #fff;
border-radius: 4px; border-radius: 4px;
@@ -1938,9 +1951,9 @@ defineExpose({ getList });
/* 已选择 tags */ /* 已选择 tags */
/* 已选择:加宽,避免套餐明细挤成一团 */ /* 已选择:加宽,避免套餐明细挤成一团 */
.selected-panel { .selected-panel {
width: 260px; width: 220px;
min-width: 240px; min-width: 200px;
max-width: 320px; max-width: 280px;
flex-shrink: 0; flex-shrink: 0;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -2000,8 +2013,9 @@ defineExpose({ getList });
font-size: 13px; font-size: 13px;
font-weight: 500; font-weight: 500;
color: #303133; color: #303133;
line-height: 1.4; overflow: hidden;
word-break: break-word; text-overflow: ellipsis;
white-space: nowrap;
} }
.card-price { .card-price {
@@ -2016,11 +2030,12 @@ defineExpose({ getList });
color: #909399; color: #909399;
transition: transform 0.2s ease; transition: transform 0.2s ease;
flex-shrink: 0; flex-shrink: 0;
transform: rotate(-90deg); transition: transform 0.2s;
transform: rotate(0deg);
} }
.expand-icon.expanded { .expand-icon.expanded {
transform: rotate(0deg); transform: rotate(90deg);
} }
/* Bug #428修复: 套餐明细列表样式 */ /* Bug #428修复: 套餐明细列表样式 */
@@ -2063,55 +2078,6 @@ defineExpose({ getList });
background: #fafbfc; background: #fafbfc;
} }
.selected-card-section {
padding: 10px;
border-bottom: 1px solid #ebeef5;
}
.selected-section-title {
font-size: 12px;
font-weight: 600;
color: #409eff;
margin-bottom: 8px;
padding-bottom: 6px;
border-bottom: 1px dashed #d9ecff;
}
.selected-method-option {
display: flex;
align-items: center;
justify-content: space-between;
gap: 8px;
padding: 4px 0;
}
.selected-method-option .method-checkbox {
flex: 1;
min-width: 0;
}
.selected-method-empty {
color: #c0c4cc;
font-size: 12px;
}
.package-toggle {
display: flex;
align-items: center;
justify-content: space-between;
padding: 10px;
font-size: 12px;
font-weight: 600;
color: #909399;
cursor: pointer;
border-bottom: 1px dashed #dcdfe6;
background: #fffbe6;
}
.package-toggle:hover {
color: #409eff;
}
.package-details-loading, .package-details-loading,
.package-details-empty { .package-details-empty {
padding: 12px 10px; padding: 12px 10px;

View File

@@ -898,9 +898,6 @@ const initData = async () => {
// 申请日期实时更新(启动定时器) // 申请日期实时更新(启动定时器)
startApplyTimeTimer() startApplyTimeTimer()
// 执行时间默认填充当前系统时间
formData.executeTime = formatDateTime(new Date())
// 获取主诊断信息 // 获取主诊断信息
try { try {
const res = await getEncounterDiagnosis(props.patientInfo.encounterId) const res = await getEncounterDiagnosis(props.patientInfo.encounterId)
@@ -1188,9 +1185,9 @@ const loadCategoryItems = async (categoryKey, loadMore = false) => {
// 映射数据格式(从检验项目维护页面的数据结构映射) // 映射数据格式(从检验项目维护页面的数据结构映射)
const mappedItems = records.map(item => { const mappedItems = records.map(item => {
// 套餐项目处理:需同时满足 feePackageId 有效且 packageName 非空 // 套餐项目处理:套餐项目使用套餐金额,普通项目使用零售价
// BugFix#556: 增加 packageName 联合判断,避免普通项目因 feePackageId 有值被误标为套餐 // BugFix#404: 增加对空字符串的判断避免空字符串被误认为有效套餐ID
const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null' && item.packageName const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null'
const itemPrice = isPackage const itemPrice = isPackage
? (parseFloat(item.packageAmount || 0) || parseFloat(item.retailPrice || 0) || parseFloat(item.price || 0)) ? (parseFloat(item.packageAmount || 0) || parseFloat(item.retailPrice || 0) || parseFloat(item.price || 0))
: (parseFloat(item.retailPrice || 0) || parseFloat(item.price || 0)) : (parseFloat(item.retailPrice || 0) || parseFloat(item.price || 0))

View File

@@ -89,14 +89,8 @@ const getList = async () => {
const response = await listPendingEmr(queryParams) const response = await listPendingEmr(queryParams)
// 根据后端返回的数据结构调整 // 根据后端返回的数据结构调整
if (response.code === 200) { if (response.code === 200) {
const data = response.data emrList.value = response.data || []
if (data && data.rows !== undefined) { total.value = Array.isArray(response.data) ? response.data.length : 0
emrList.value = data.rows || []
total.value = data.total || 0
} else {
emrList.value = Array.isArray(data) ? data : []
total.value = emrList.value.length
}
} else { } else {
ElMessage.error(response.msg || '获取待写病历列表失败') ElMessage.error(response.msg || '获取待写病历列表失败')
emrList.value = [] emrList.value = []

View File

@@ -3801,8 +3801,6 @@ function handleSaveHistory(value) {
uniqueKey: undefined, uniqueKey: undefined,
dbOpType: value.requestId ? '2' : '1', dbOpType: value.requestId ? '2' : '1',
minUnitQuantity: value.quantity * value.partPercent, minUnitQuantity: value.quantity * value.partPercent,
// 🔧 修复:确保 categoryEnum 被传递(耗材必填字段),避免后端 NPE
categoryEnum: value.categoryEnum || value.categoryCode,
conditionId: conditionId.value, conditionId: conditionId.value,
conditionDefinitionId: conditionDefinitionId.value, conditionDefinitionId: conditionDefinitionId.value,
encounterDiagnosisId: encounterDiagnosisId.value, encounterDiagnosisId: encounterDiagnosisId.value,

View File

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

View File

@@ -113,17 +113,10 @@ const getList = async () => {
loading.value = true loading.value = true
try { try {
const response = await listPendingEmr(queryParams) const response = await listPendingEmr(queryParams)
// 根据后端返回的数据结构调整
if (response.code === 200) { if (response.code === 200) {
const data = response.data emrList.value = response.data || []
if (data && data.rows !== undefined) { total.value = Array.isArray(response.data) ? response.data.length : 0
// 新分页格式 {rows, total}
emrList.value = data.rows || []
total.value = data.total || 0
} else {
// 兼容旧格式(数组)
emrList.value = Array.isArray(data) ? data : []
total.value = emrList.value.length
}
} else { } else {
ElMessage.error(response.msg || '获取待写病历列表失败') ElMessage.error(response.msg || '获取待写病历列表失败')
emrList.value = [] emrList.value = []

View File

@@ -86,7 +86,7 @@
</template> </template>
<el-table-column type="index" label="序号" width="60" align="center" /> <el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="patientName" label="患者姓名" width="120" /> <el-table-column prop="patientName" label="患者姓名" width="120" />
<el-table-column label="申请单名称" min-width="140"> <el-table-column label="申请单名称" width="140">
<template #default="scope"> <template #default="scope">
<span>{{ buildApplicationName(scope.row) }}</span> <span>{{ buildApplicationName(scope.row) }}</span>
</template> </template>
@@ -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>
@@ -444,9 +447,11 @@ const buildApplicationName = (row) => {
if (!details || details.length === 0) { if (!details || details.length === 0) {
return row.name || '-'; return row.name || '-';
} }
const names = details.map(d => d.adviceName).filter(Boolean); if (details.length === 1) {
if (names.length === 0) return row.name || '-'; return details[0].adviceName || row.name || '-';
return names.join(' + '); }
const first = details[0];
return `${first.adviceName || ''}${details.length}`;
}; };
/** /**

View File

@@ -116,7 +116,7 @@ import {computed, getCurrentInstance, ref, watch} from 'vue';
import {Refresh} from '@element-plus/icons-vue'; import {Refresh} from '@element-plus/icons-vue';
import {patientInfo} from '../../store/patient.js'; import {patientInfo} from '../../store/patient.js';
import {getSurgery} from './api'; import {getSurgery} from './api';
import {getDepartmentList} from '@/api/public.js'; import {getOrgList} from '@/views/doctorstation/components/api.js';
const { proxy } = getCurrentInstance(); const { proxy } = getCurrentInstance();
@@ -182,32 +182,25 @@ const hasMatchedFields = computed(() => {
/** 查询科室 */ /** 查询科室 */
const getLocationInfo = async () => { const getLocationInfo = async () => {
const res = await getDepartmentList(); const res = await getOrgList();
orgOptions.value = res.data || []; orgOptions.value = res.data.records;
}; };
const recursionFun = (targetDepartment) => { const recursionFun = (targetDepartment) => {
if (!targetDepartment || !orgOptions.value || orgOptions.value.length === 0) {
return '';
}
let name = ''; let name = '';
// 统一处理:扁平列表和树形结构都适用 for (let index = 0; index < orgOptions.value.length; index++) {
const findInList = (list) => { const obj = orgOptions.value[index];
for (const node of list) { if (obj.id == targetDepartment) {
if (String(node.id) === String(targetDepartment)) { name = obj.name;
name = node.name; }
return true; const subObjArray = obj['children'];
} for (let index = 0; index < subObjArray.length; index++) {
// 树形结构:递归查找 children const item = subObjArray[index];
if (node.children && node.children.length > 0) { if (item.id == targetDepartment) {
if (findInList(node.children)) { name = item.name;
return true;
}
} }
} }
return false; }
};
findInList(orgOptions.value);
return name; return name;
}; };

View File

@@ -41,9 +41,7 @@
<el-option label="全部" value="" /> <el-option label="全部" value="" />
<el-option label="待签发" value="0" /> <el-option label="待签发" value="0" />
<el-option label="已签发" value="1" /> <el-option label="已签发" value="1" />
<el-option label="已采证" value="4" /> <el-option label="已出报告" value="6" />
<el-option label="已送检" value="5" />
<el-option label="报告已出" value="6" />
<el-option label="已作废" value="7" /> <el-option label="已作废" value="7" />
</el-select> </el-select>
</el-form-item> </el-form-item>
@@ -93,15 +91,7 @@
<el-table-column prop="prescriptionNo" label="申请单号" width="140" /> <el-table-column prop="prescriptionNo" label="申请单号" width="140" />
<el-table-column label="单据状态" width="100" align="center"> <el-table-column label="单据状态" width="100" align="center">
<template #default="scope"> <template #default="scope">
<el-tag <span>{{ parseBillStatus(scope.row.billStatus ?? scope.row.status) }}</span>
:type="getBillStatusTagType(scope.row)"
effect="plain"
round
:class="{ 'report-status-tag': isReportStatus(scope.row) }"
@click="handleStatusClick(scope.row)"
>
{{ parseBillStatus(getBillStatus(scope.row)) }}
</el-tag>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="申请类型" width="100" align="center"> <el-table-column label="申请类型" width="100" align="center">
@@ -115,20 +105,20 @@
</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="280"> <el-table-column label="操作" align="center" fixed="right" width="220">
<template #default="scope"> <template #default="scope">
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button> <!-- 待签发status=0或null/undefined可修改删除 -->
<template v-if="canManageRow(scope.row)"> <template v-if="!scope.row.status || scope.row.status == 0">
<template v-if="isPendingStatus(scope.row)"> <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 v-else-if="isWithdrawableStatus(scope.row)">
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
</template>
</template> </template>
<template v-if="isReportStatus(scope.row)"> <!-- 已签发status=1可撤回 -->
<el-button link type="success" @click="handleViewReport(scope.row)">查看报告</el-button> <template v-else-if="scope.row.status == 1">
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
</template>
<!-- 已校对(2)待接收(3)已收样(4)已出报告(6)已作废(7)仅查看详情 -->
<template v-else>
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
</template> </template>
</template> </template>
</el-table-column> </el-table-column>
@@ -222,16 +212,13 @@
</template> </template>
<script setup> <script setup>
import {computed, getCurrentInstance, nextTick, ref, watch} from 'vue'; import {computed, getCurrentInstance, ref, watch} from 'vue';
import {Refresh, Search} from '@element-plus/icons-vue'; import {Refresh, Search} from '@element-plus/icons-vue';
import {patientInfo} from '../../store/patient.js'; import {patientInfo} from '../../store/patient.js';
import {getInspection, deleteRequestForm, withdrawRequestForm, getProofResult} from './api'; import {getInspection, deleteRequestForm, withdrawRequestForm} from './api';
import {getDepartmentList} from '@/api/public.js'; import {getDepartmentList} from '@/api/public.js';
import LaboratoryTests from '../order/applicationForm/laboratoryTests.vue'; import LaboratoryTests from '../order/applicationForm/laboratoryTests.vue';
import useUserStore from '@/store/modules/user'; import {saveInspection} from '../order/applicationForm/api.js';
import auth from '@/plugins/auth';
const userStore = useUserStore();
const { proxy } = getCurrentInstance(); const { proxy } = getCurrentInstance();
@@ -283,7 +270,7 @@ const fetchData = async () => {
if (res.code === 200 && res.data) { if (res.code === 200 && res.data) {
const raw = res.data?.records || res.data; const raw = res.data?.records || res.data;
const list = Array.isArray(raw) ? raw : [raw]; const list = Array.isArray(raw) ? raw : [raw];
tableData.value = list.filter(Boolean).sort(sortByCreateTimeDesc); tableData.value = list.filter(Boolean);
} else { } else {
tableData.value = []; tableData.value = [];
} }
@@ -342,110 +329,19 @@ const labelMap = {
* @param {string|number} status - 状态码 * @param {string|number} status - 状态码
* @returns {string} 状态文本 * @returns {string} 状态文本
*/ */
const getBillStatus = (row) => {
return row?.billStatus ?? row?.status ?? row?.statusEnum ?? row?.applyStatus;
};
const parseBillStatus = (status) => { const parseBillStatus = (status) => {
const statusMap = { const statusMap = {
'0': '待签发', '0': '待签发',
'1': '已签发', '1': '已签发',
'2': '已采证', '2': '已校对',
'3': '已送检', '3': '待接收',
'4': '已采证', '4': '已收样',
'5': '已送检', '6': '已出报告',
'6': '报告已出',
'8': '报告已出',
'7': '已作废', '7': '已作废',
}; };
return statusMap[String(status)] || '-'; return statusMap[String(status)] || '-';
}; };
const getBillStatusTagType = (row) => {
const typeMap = {
'0': 'info',
'1': 'primary',
'2': 'primary',
'3': 'warning',
'4': 'primary',
'5': 'warning',
'6': 'success',
'7': 'danger',
'8': 'success',
};
return typeMap[String(getBillStatus(row))] || 'info';
};
const isPendingStatus = (row) => {
const status = getBillStatus(row);
return status === undefined || status === null || status === '' || String(status) === '0';
};
const isWithdrawableStatus = (row) => String(getBillStatus(row)) === '1';
const isReportStatus = (row) => ['6', '8'].includes(String(getBillStatus(row)));
/**
* 是否可管理该申请单:申请者本人或管理员
*/
const canManageRow = (row) => {
if (auth.hasRole('admin')) {
return true;
}
const currentPractitionerId = userStore.practitionerId;
const requesterId = row?.requesterId;
if (!currentPractitionerId || !requesterId) {
return false;
}
return String(currentPractitionerId) === String(requesterId);
};
const sortByCreateTimeDesc = (a, b) => {
const aTime = a?.createTime ? new Date(a.createTime).getTime() : 0;
const bTime = b?.createTime ? new Date(b.createTime).getTime() : 0;
return bTime - aTime;
};
const handleStatusClick = (row) => {
if (isReportStatus(row)) {
handleViewReport(row);
}
};
const pickReportUrl = (data, row) => {
if (!data) return '';
if (typeof data === 'string') return data;
const raw = data.records || data;
const list = Array.isArray(raw) ? raw : [raw];
const matched =
list.find((item) => {
const reportNo = item.busNo || item.reportNo || item.applyNo || item.prescriptionNo;
return reportNo && row.prescriptionNo && String(reportNo) === String(row.prescriptionNo);
}) || list[0];
return matched?.requestUrl || matched?.pdfUrl || matched?.reportUrl || matched?.url || '';
};
const handleViewReport = async (row) => {
try {
const res = await getProofResult({
encounterId: row.encounterId || patientInfo.value?.encounterId,
prescriptionNo: row.prescriptionNo,
});
if (res?.code === 200) {
const url = pickReportUrl(res.data, row);
if (url) {
window.open(url, '_blank');
return;
}
}
proxy.$modal?.msgWarning?.('暂未获取到检验报告链接');
} catch (e) {
proxy.$modal?.msgError?.(e.message || '获取检验报告失败');
}
};
/** /**
* 解析申请类型(优先级代码) * 解析申请类型(优先级代码)
* @param {string} descJson - JSON字符串 * @param {string} descJson - JSON字符串
@@ -513,13 +409,8 @@ const hasMatchedFields = computed(() => {
/** 查询科室 */ /** 查询科室 */
const getLocationInfo = async () => { const getLocationInfo = async () => {
try { const res = await getDepartmentList();
const res = await getDepartmentList(); orgOptions.value = res.data || [];
orgOptions.value = Array.isArray(res.data) ? res.data : [];
} catch (e) {
console.warn('科室列表加载失败:', e.message);
orgOptions.value = [];
}
}; };
const recursionFun = (targetDepartment) => { const recursionFun = (targetDepartment) => {
@@ -547,13 +438,7 @@ const handleViewDetail = async (row) => {
if (row.descJson) { if (row.descJson) {
try { try {
const obj = JSON.parse(row.descJson); const obj = JSON.parse(row.descJson);
// 将发往科室 ID 转换为名称 obj.targetDepartment = recursionFun(obj.targetDepartment);
if (obj.targetDepartment) {
const deptName = recursionFun(obj.targetDepartment);
if (deptName) {
obj.targetDepartment = deptName;
}
}
// 转换申请类型编码为可读文本 // 转换申请类型编码为可读文本
if (obj.applicationType === 0) obj.applicationType = '普通'; if (obj.applicationType === 0) obj.applicationType = '普通';
else if (obj.applicationType === 1) obj.applicationType = '急诊'; else if (obj.applicationType === 1) obj.applicationType = '急诊';
@@ -572,12 +457,12 @@ const handleViewDetail = async (row) => {
* 修改检验申请单(待签发状态) * 修改检验申请单(待签发状态)
*/ */
const handleEdit = async (row) => { const handleEdit = async (row) => {
// 确保科室数据已加载
if (!orgOptions.value || orgOptions.value.length === 0) {
await getLocationInfo();
}
editRowData.value = row; editRowData.value = row;
editDialogVisible.value = true; editDialogVisible.value = true;
await nextTick();
editFormRef.value?.getList?.();
editFormRef.value?.getLocationInfo?.();
editFormRef.value?.getDiagnosisList?.();
}; };
/** /**
@@ -604,9 +489,9 @@ const submitEditForm = () => {
*/ */
const handleDelete = async (row) => { const handleDelete = async (row) => {
try { try {
await proxy.$modal?.confirm?.('确认作废该申请单吗?作废后不可撤销'); await proxy.$modal?.confirm?.(`确定要删除申请单 "${row.prescriptionNo}" 吗?此操作不可恢复。`);
} catch { } catch {
return; return; // 用户取消
} }
try { try {
@@ -623,15 +508,13 @@ const handleDelete = async (row) => {
}; };
/** /**
* 撤回检验申请单(已签发且未采证状态撤回) * 撤回检验申请单(已签发状态撤回至待签发
*/ */
const handleWithdraw = async (row) => { const handleWithdraw = async (row) => {
try { try {
await proxy.$modal?.confirm?.( await proxy.$modal?.confirm?.(`确定要撤回申请单 "${row.prescriptionNo}" 吗?撤回后将恢复为待签发状态。`);
'确认撤回该申请单吗?撤回后申请单及关联医嘱将恢复为待签发状态,护士站将同步更新。'
);
} catch { } catch {
return; return; // 用户取消
} }
try { try {
@@ -758,14 +641,6 @@ defineExpose({
animation: rotating 2s linear infinite; animation: rotating 2s linear infinite;
} }
.report-status-tag {
cursor: pointer;
background-color: #f0f9eb !important;
border-color: #67c23a !important;
color: #529b2e !important;
font-weight: 600;
}
@keyframes rotating { @keyframes rotating {
0% { 0% {
transform: rotate(0deg); transform: rotate(0deg);

View File

@@ -63,7 +63,7 @@
<span class="medicine-info"> 诊断{{ config.diagnosisName }} </span> <span class="medicine-info"> 诊断{{ config.diagnosisName }} </span>
<span class="medicine-info"> 皮试{{ row.skinTestFlag_enumText }} </span> <span class="medicine-info"> 皮试{{ row.skinTestFlag_enumText }} </span>
<span class="medicine-info"> 注射药品{{ row.injectFlag_enumText }} </span> <span class="medicine-info"> 注射药品{{ row.injectFlag_enumText }} </span>
<span class="total-amount"> <span class="total-amount" v-if="row.therapyEnum == '2'">
总金额{{ row.totalPrice ? Number(row.totalPrice).toFixed(2) + ' ' : '0.00 ' }} 总金额{{ row.totalPrice ? Number(row.totalPrice).toFixed(2) + ' ' : '0.00 ' }}
</span> </span>
</div> </div>
@@ -83,7 +83,7 @@
:controls="false" :controls="false"
style="width: 70px" style="width: 70px"
:ref="(el) => setInputRef('doseQuantity', el)" :ref="(el) => setInputRef('doseQuantity', el)"
@input="() => { convertValues(); calculateTotalAmount(); }" @input="convertValues"
@keyup.enter.prevent="handleEnter('doseQuantity')" @keyup.enter.prevent="handleEnter('doseQuantity')"
/> />
</el-form-item> </el-form-item>
@@ -110,7 +110,7 @@
:controls="false" :controls="false"
style="width: 70px; margin-left: 32px" style="width: 70px; margin-left: 32px"
:ref="(el) => setInputRef('dose', el)" :ref="(el) => setInputRef('dose', el)"
@input="() => { convertDoseValues(); calculateTotalAmount(); }" @input="convertDoseValues"
@keyup.enter.prevent="handleEnter('dose')" @keyup.enter.prevent="handleEnter('dose')"
/> />
</el-form-item> </el-form-item>
@@ -119,7 +119,7 @@
v-model="row.doseUnitCode" v-model="row.doseUnitCode"
style="width: 70px" style="width: 70px"
placeholder=" " placeholder=" "
@change="() => { convertValues(); calculateTotalAmount(); }" @change="convertValues"
> >
<el-option <el-option
v-for="item in row.unitCodeList" v-for="item in row.unitCodeList"
@@ -271,17 +271,13 @@
controls-position="right" controls-position="right"
:controls="false" :controls="false"
:ref="(el) => setInputRef('firstDose', el)" :ref="(el) => setInputRef('firstDose', el)"
@input="calculateTotalAmount"
@keyup.enter.prevent="handleEnter('firstDose')" @keyup.enter.prevent="handleEnter('firstDose')"
/> />
</el-form-item> </el-form-item>
<el-select v-model="row.doseUnitCode" style="width: 70px" placeholder=" "> <el-select v-model="row.unitCode" style="width: 70px" placeholder=" ">
<el-option <template v-for="item in row.unitCodeList" :key="item.value">
v-for="item in row.unitCodeList" <el-option v-if="checkUnit(item)" :value="item.value" :label="item.label" />
:value="item.value" </template>
:label="item.label"
:key="item.value"
/>
</el-select> </el-select>
</template> </template>
</div> </div>
@@ -430,7 +426,6 @@
<script setup lang="ts"> <script setup lang="ts">
import {computed, getCurrentInstance, nextTick, onMounted, ref, watch} from 'vue'; import {computed, getCurrentInstance, nextTick, onMounted, ref, watch} from 'vue';
import Decimal from 'decimal.js';
interface Config { interface Config {
diagnosisName: string; // 仅用于显示 diagnosisName: string; // 仅用于显示
@@ -628,18 +623,7 @@ const orgFallbackOption = (value: any) => {
const convertValues = () => props.handlers.convertValue('doseQuantity', props.row, props.index); const convertValues = () => props.handlers.convertValue('doseQuantity', props.row, props.index);
const convertDoseValues = () => props.handlers.convertValue('dose', props.row, props.index); const convertDoseValues = () => props.handlers.convertValue('dose', props.row, props.index);
const calculateTotalPrice = () => props.handlers.calculateTotal('price', props.row, props.index); const calculateTotalPrice = () => props.handlers.calculateTotal('price', props.row, props.index);
// 直接用 row 计算总金额:数量 * 单价,避免父组件索引不匹配的问题 const calculateTotalAmount = () => props.handlers.calculateTotal('amount', props.row, props.index);
const calculateTotalAmount = () => {
nextTick(() => {
const row = props.row;
const qty = new Decimal(row.doseQuantity || 0);
const isMinUnit = row.unitCode == row.minUnitCode;
const price = isMinUnit ? row.minUnitPrice : row.unitPrice;
// 四舍五入到2位再算与页面显示的单价一致
const roundedPrice = new Decimal(price || 0).toDecimalPlaces(2, Decimal.ROUND_HALF_UP);
row.totalPrice = qty.mul(roundedPrice).toFixed(6);
});
};
const setInputRef = props.handlers.setInputRef; const setInputRef = props.handlers.setInputRef;
defineExpose({ defineExpose({

View File

@@ -17,14 +17,17 @@
style="width: 300px; margin-bottom: 10px" style="width: 300px; margin-bottom: 10px"
> >
<template #append> <template #append>
<el-button @click="handleSearch" :loading="loading">搜索</el-button> <el-button @click="handleSearch">搜索</el-button>
</template> </template>
</el-input> </el-input>
<span class="total-count"> {{ totalCount }} </span> <span v-if="!searchKey" class="total-count"> {{ totalCount }} </span>
<span v-else class="total-count">搜索到 {{ filteredCount }} / {{ totalCount }} </span>
</div> </div>
<el-transfer <el-transfer
v-model="transferValue" v-model="transferValue"
:data="transferData" :data="transferData"
filter-placeholder="项目代码/名称"
filterable
:titles="['未选择', '已选择']" :titles="['未选择', '已选择']"
/> />
</div> </div>
@@ -131,11 +134,10 @@
</div> </div>
</template> </template>
<script setup name="LaboratoryTests"> <script setup name="LaboratoryTests">
import {getCurrentInstance, nextTick, onMounted, reactive, ref, watch, computed} from 'vue'; import {getCurrentInstance, onMounted, reactive, ref, watch, computed} from 'vue';
import {patientInfo} from '../../../store/patient.js'; import {patientInfo} from '../../../store/patient.js';
import {getExaminationPage, saveInspection} from './api'; import {getApplicationList, saveInspection} from './api';
import {ActivityCategory} from '@/utils/medicalConstants'; import {getOrgList} from '@/views/doctorstation/components/api.js';
import {getDepartmentList} from '@/api/public.js';
import {getEncounterDiagnosis} from '../../api.js'; import {getEncounterDiagnosis} from '../../api.js';
import {ElMessage} from 'element-plus'; import {ElMessage} from 'element-plus';
@@ -166,13 +168,13 @@ const loading = ref(false);
const orgOptions = ref([]); const orgOptions = ref([]);
const searchKey = ref(''); const searchKey = ref('');
const totalCount = ref(0); const totalCount = ref(0);
const skipDeptAutoFill = ref(false);
// 将已加载的全部数据转为 transfer 组件所需的格式 // 将已加载的全部数据转为 transfer 组件所需的格式
const buildTransferData = (records) => { const buildTransferData = (records) => {
return records.map((item) => { return records.map((item) => {
const price = item.price != null ? Number(item.price).toFixed(2) : '0.00'; const priceInfo = item.priceList?.[0] || {};
const unit = item.unitCodeDictText || item.unitCode || ''; const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
const unit = item.unitCode_dictText || item.unitCode || '';
return { return {
adviceDefinitionId: item.adviceDefinitionId, adviceDefinitionId: item.adviceDefinitionId,
orgId: item.orgId, orgId: item.orgId,
@@ -182,8 +184,7 @@ const buildTransferData = (records) => {
}); });
}; };
const selectedItemsCache = ref(new Map()); // 加载全部数据(不分页,一次性拉取)
const loadAllData = async () => { const loadAllData = async () => {
if (!patientInfo.value?.inHospitalOrgId) { if (!patientInfo.value?.inHospitalOrgId) {
applicationListAll.value = []; applicationListAll.value = [];
@@ -191,12 +192,13 @@ const loadAllData = async () => {
} }
loading.value = true; loading.value = true;
try { try {
const res = await getExaminationPage({ // 使用大 pageSize 一次性拉取所有启用状态的检验类诊疗项目
pageSize: 100, const res = await getApplicationList({
pageSize: 9999,
pageNo: 1, pageNo: 1,
categoryCode: ActivityCategory.PROOF, categoryCode: '22',
organizationId: patientInfo.value.inHospitalOrgId, organizationId: patientInfo.value.inHospitalOrgId,
searchKey: searchKey.value, adviceTypes: [3], // 1 药品 2 耗材 3 诊疗
}); });
if (res.code !== 200) { if (res.code !== 200) {
proxy.$message.error(res.message); proxy.$message.error(res.message);
@@ -205,9 +207,6 @@ const loadAllData = async () => {
} }
applicationListAll.value = res.data?.records || []; applicationListAll.value = res.data?.records || [];
totalCount.value = res.data?.total || 0; totalCount.value = res.data?.total || 0;
if (!searchKey.value) {
applyEditTransferSelection();
}
} catch (e) { } catch (e) {
proxy.$message.error('获取检验项目列表失败'); proxy.$message.error('获取检验项目列表失败');
applicationListAll.value = []; applicationListAll.value = [];
@@ -216,21 +215,33 @@ const loadAllData = async () => {
} }
}; };
const transferData = computed(() => buildTransferData(applicationListAll.value)); // 根据搜索关键词过滤数据
const filterData = (key) => {
if (!key || key.trim() === '') {
return applicationListAll.value;
}
const lowerKey = key.toLowerCase().trim();
return applicationListAll.value.filter((item) => {
return (
item.adviceName?.toLowerCase().includes(lowerKey) ||
item.pyStr?.toLowerCase().includes(lowerKey) ||
item.adviceBusNo?.toLowerCase().includes(lowerKey)
);
});
};
// transfer 组件实际显示的数据(受搜索词影响)
const transferData = computed(() => buildTransferData(filterData(searchKey.value)));
// 当前显示的条数
const filteredCount = computed(() => filterData(searchKey.value).length);
const getList = async () => { const getList = async () => {
await loadAllData(); await loadAllData();
}; };
let searchTimer = null;
const handleSearch = () => { const handleSearch = () => {
clearTimeout(searchTimer); // 搜索时保持已选中的项目不受影响
searchTimer = setTimeout(() => {
loadAllData();
}, 300);
}; };
// 编辑初始化标志:避免 applyEditTransferSelection 设置 transferValue 时触发 projectWithDepartment 覆盖 descJson 中的科室值
const isInitializing = ref(false);
const transferValue = ref([]); const transferValue = ref([]);
const form = reactive({ const form = reactive({
// categoryType: '', // 项目类别 // categoryType: '', // 项目类别
@@ -248,31 +259,7 @@ const form = reactive({
otherDiagnosisList: [], //其他断目录 otherDiagnosisList: [], //其他断目录
}); });
const rules = reactive({}); const rules = reactive({});
const normalizeOrgTreeIds = (nodes) => {
if (!Array.isArray(nodes)) return [];
return nodes.map((node) => ({
...node,
id: node.id != null ? String(node.id) : node.id,
children: node.children?.length ? normalizeOrgTreeIds(node.children) : undefined,
}));
};
const resolveTargetDepartmentId = (rawId) => {
if (rawId == null || rawId === '') return '';
const node = findTreeItem(orgOptions.value, rawId);
return node ? String(node.id) : String(rawId);
};
const applyTargetDepartmentEcho = () => {
if (form.targetDepartment) {
form.targetDepartment = resolveTargetDepartmentId(form.targetDepartment);
}
};
onMounted(() => { onMounted(() => {
getLocationInfo();
getDiagnosisList();
getList(); getList();
}); });
/** /**
@@ -286,17 +273,13 @@ const projectWithDepartment = (selectProjectIds, type) => {
const arr = []; const arr = [];
// 根据选中的项目id查找对应的项目从全部原始数据中查找 // 根据选中的项目id查找对应的项目从全部原始数据中查找
selectProjectIds.forEach((element) => { selectProjectIds.forEach((element) => {
let searchData = applicationListAll.value.find((item) => { const searchData = applicationListAll.value.find((item) => {
return element == item.adviceDefinitionId; return element == item.adviceDefinitionId;
}); });
if (!searchData) {
searchData = selectedItemsCache.value.get(element);
}
if (searchData) { if (searchData) {
const priceInfo = searchData.priceList?.[0] || {}; const priceInfo = searchData.priceList?.[0] || {};
const price = searchData.price != null ? Number(searchData.price).toFixed(2) const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
: priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00'; const unit = searchData.unitCode_dictText || searchData.unitCode || '';
const unit = searchData.unitCodeDictText || searchData.unitCode_dictText || searchData.unitCode || '';
arr.push({ arr.push({
adviceDefinitionId: searchData.adviceDefinitionId, adviceDefinitionId: searchData.adviceDefinitionId,
orgId: searchData.orgId, orgId: searchData.orgId,
@@ -305,9 +288,8 @@ const projectWithDepartment = (selectProjectIds, type) => {
}); });
} }
}); });
// 保存用户手动选择/回显的发往科室(提交、编辑回显时需要保留) // 保存用户手动选择的发往科室(提交时需要保留)
const manualDept = const manualDept = type === 2 ? form.targetDepartment : '';
type === 2 || (isEditMode.value && form.targetDepartment) ? form.targetDepartment : '';
// 清空科室 // 清空科室
form.targetDepartment = ''; form.targetDepartment = '';
if (arr.length > 0) { if (arr.length > 0) {
@@ -327,8 +309,8 @@ const projectWithDepartment = (selectProjectIds, type) => {
const findItem = findTreeItem(orgOptions.value, obj.orgId); const findItem = findTreeItem(orgOptions.value, obj.orgId);
if (!findItem) { if (!findItem) {
// type=2(提交)时,若用户已手动选择发往科室,则允许提交 // type=2(提交)时,若用户已手动选择发往科室,则允许提交
if ((type === 2 || isEditMode.value) && manualDept) { if (type === 2 && manualDept) {
form.targetDepartment = resolveTargetDepartmentId(manualDept); form.targetDepartment = manualDept;
isRelease = true; isRelease = true;
} else if (type === 2 && !manualDept) { } else if (type === 2 && !manualDept) {
// 提交时用户未手动选择科室,才提示错误 // 提交时用户未手动选择科室,才提示错误
@@ -344,10 +326,10 @@ const projectWithDepartment = (selectProjectIds, type) => {
} }
if (findItem && isRelease) { if (findItem && isRelease) {
// 提交时若用户已选「发往科室」,不得用项目默认执行科室覆盖 // 提交时若用户已选「发往科室」,不得用项目默认执行科室覆盖
if ((type === 2 || isEditMode.value) && manualDept) { if (type === 2 && manualDept) {
form.targetDepartment = resolveTargetDepartmentId(manualDept); form.targetDepartment = manualDept;
} else { } else {
form.targetDepartment = String(findItem.id); form.targetDepartment = findItem.id;
} }
} }
} }
@@ -357,123 +339,47 @@ const projectWithDepartment = (selectProjectIds, type) => {
watch( watch(
() => transferValue.value, () => transferValue.value,
(newValue) => { (newValue) => {
if (skipDeptAutoFill.value) return;
if (isInitializing.value) return;
newValue.forEach((id) => {
if (!selectedItemsCache.value.has(id)) {
const item = applicationListAll.value.find((i) => i.adviceDefinitionId == id);
if (item) selectedItemsCache.value.set(id, item);
}
});
projectWithDepartment(newValue, 1); projectWithDepartment(newValue, 1);
} }
); );
/** 编辑弹窗:根据申请单明细把右侧「已选择」与 transferValue 对齐(依赖 applicationListAll 已加载) */ // 编辑模式下,回显已有数据
const applyEditTransferSelection = () => {
const newData = props.editData
if (!newData?.requestFormId || !newData.requestFormDetailList?.length) {
return
}
if (!applicationListAll.value.length) {
return
}
const selectedIds = []
for (const detail of newData.requestFormDetailList) {
const idFromDetail = detail.activityId ?? detail.adviceDefinitionId
let matched = null
if (idFromDetail != null && idFromDetail !== '') {
matched = applicationListAll.value.find(
(item) => String(item.adviceDefinitionId) === String(idFromDetail)
)
}
if (!matched && detail.adviceName) {
matched = applicationListAll.value.find((item) => item.adviceName === detail.adviceName)
}
if (!matched && detail.adviceName) {
const norm = (s) => String(s || '').trim()
matched = applicationListAll.value.find(
(item) => norm(item.adviceName) === norm(detail.adviceName)
)
}
if (matched) {
selectedIds.push(matched.adviceDefinitionId)
}
}
const uniq = [...new Set(selectedIds)]
// 设置初始化标志,防止 transferValue 变化触发 projectWithDepartment 覆盖 descJson 中的科室值
isInitializing.value = true
skipDeptAutoFill.value = true
transferValue.value = uniq
nextTick(() => {
skipDeptAutoFill.value = false
})
isInitializing.value = false
if (newData.requestFormDetailList.length && uniq.length === 0) {
console.warn(
'[LaboratoryTests] 申请单明细未能在项目字典中匹配到项,请核对 activityId / 项目名称',
newData.requestFormDetailList
)
}
}
// 编辑模式下,回显已有数据(表单来自 descJson项目选择在字典加载后由 applyEditTransferSelection 完成)
watch( watch(
() => props.editData, () => props.editData,
(newData) => { (newData) => {
if (!newData || !newData.requestFormId) return if (!newData || !newData.requestFormId) return;
// 解析 descJson 回填表单
if (newData.descJson) { if (newData.descJson) {
try { try {
const obj = JSON.parse(newData.descJson) const obj = JSON.parse(newData.descJson);
Object.keys(form).forEach((key) => { Object.keys(form).forEach((key) => {
if (obj[key] !== undefined) { if (obj[key] !== undefined) {
form[key] = obj[key] form[key] = obj[key];
} }
}) });
applyTargetDepartmentEcho()
} catch (e) { } catch (e) {
console.error('解析 descJson 失败:', e) console.error('解析 descJson 失败:', e);
} }
} }
applyEditTransferSelection() // 回填已选项目
if (newData.requestFormDetailList && newData.requestFormDetailList.length > 0) {
// 从全部数据中匹配已选项目
const selectedIds = [];
newData.requestFormDetailList.forEach((detail) => {
const matched = applicationListAll.value.find(
(item) => item.adviceName === detail.adviceName
);
if (matched) {
selectedIds.push(matched.adviceDefinitionId);
}
});
transferValue.value = selectedIds;
}
}, },
{ immediate: true, deep: true } { immediate: true }
)
watch(
() => orgOptions.value,
() => {
applyTargetDepartmentEcho()
}
)
// 编辑模式下,项目字典首次加载完成后回显已选项目(搜索刷新不重置)
watch(
() => applicationListAll.value,
() => {
if (!props.editData?.requestFormId) return;
if (!props.editData.requestFormDetailList?.length) return;
if (!applicationListAll.value.length) return;
if (searchKey.value) return;
const selectedIds = [];
props.editData.requestFormDetailList.forEach((detail) => {
const matched = applicationListAll.value.find(
(item) => item.adviceName === detail.adviceName
);
if (matched) {
selectedIds.push(matched.adviceDefinitionId);
}
});
isInitializing.value = true;
transferValue.value = selectedIds;
isInitializing.value = false;
applyEditTransferSelection();
}
); );
const submit = () => { const submit = () => {
if (transferValue.value.length == 0) { if (transferValue.value.length == 0) {
return proxy.$message.error('请选择申请单'); return proxy.$message.error('请选择申请单');
@@ -481,29 +387,26 @@ const submit = () => {
if (!projectWithDepartment(transferValue.value, 2)) { if (!projectWithDepartment(transferValue.value, 2)) {
return; return;
} }
let applicationListAllFilter = transferValue.value.map((id) => { let applicationListAllFilter = applicationListAll.value.filter((item) => {
let item = applicationListAll.value.find((i) => i.adviceDefinitionId == id); return transferValue.value.includes(item.adviceDefinitionId);
if (!item) { });
item = selectedItemsCache.value.get(id); applicationListAllFilter = applicationListAllFilter.map((item) => {
}
if (!item) return null;
const priceInfo = item.priceList?.[0] || {};
return { return {
adviceDefinitionId: item.adviceDefinitionId /** 诊疗定义id */, adviceDefinitionId: item.adviceDefinitionId /** 诊疗定义id */,
quantity: 1, // /** 请求数量 */ quantity: 1, // /** 请求数量 */
unitCode: item.unitCode || priceInfo.unitCode || '' /** 请求单位编码 */, unitCode: item.priceList[0].unitCode /** 请求单位编码 */,
unitPrice: item.price ?? priceInfo.price ?? 0 /** 单价 */, unitPrice: item.priceList[0].price /** 单价 */,
totalPrice: item.price ?? priceInfo.price ?? 0 /** 总价 */, totalPrice: item.priceList[0].price /** 总价 */,
positionId: form.targetDepartment || item.positionId, // 用户指定发往科室优先于项目默认执行科室 positionId: form.targetDepartment || item.positionId, // 用户指定发往科室优先于项目默认执行科室
ybClassEnum: item.ybClassEnum || '', //类别医保编码 ybClassEnum: item.ybClassEnum, //类别医保编码
conditionId: item.conditionId || '', //诊断ID conditionId: item.conditionId, //诊断ID
encounterDiagnosisId: item.encounterDiagnosisId || '', //就诊诊断id encounterDiagnosisId: item.encounterDiagnosisId, //就诊诊断id
adviceType: item.adviceType || 3, ///** 医嘱类型 */ adviceType: item.adviceType, ///** 医嘱类型 */
definitionId: item.chargeItemDefinitionId || priceInfo.definitionId || '', //费用定价主表ID */ definitionId: item.priceList[0].definitionId, //费用定价主表ID */
definitionDetailId: item.definitionDetailId || priceInfo.definitionDetailId || '', //费用定价子表ID */ definitionDetailId: item.definitionDetailId, //费用定价子表ID */
accountId: patientInfo.value.accountId, // // 账户id accountId: patientInfo.value.accountId, // // 账户id
}; };
}).filter(Boolean); });
const params = { const params = {
activityList: applicationListAllFilter, activityList: applicationListAllFilter,
patientId: patientInfo.value.patientId, //患者ID patientId: patientInfo.value.patientId, //患者ID
@@ -518,7 +421,6 @@ const submit = () => {
if (res.code === 200) { if (res.code === 200) {
proxy.$message.success(isEditMode.value ? '修改成功' : res.msg); proxy.$message.success(isEditMode.value ? '修改成功' : res.msg);
transferValue.value = []; transferValue.value = [];
selectedItemsCache.value.clear();
emits('submitOk'); emits('submitOk');
} else { } else {
proxy.$message.error(res.message); proxy.$message.error(res.message);
@@ -527,9 +429,9 @@ const submit = () => {
}; };
/** 查询科室 */ /** 查询科室 */
const getLocationInfo = () => { const getLocationInfo = () => {
return getDepartmentList().then((res) => { getOrgList().then((res) => {
orgOptions.value = normalizeOrgTreeIds(res.data || []); orgOptions.value = res.data.records;
applyTargetDepartmentEcho(); console.log('科室========>', JSON.stringify(orgOptions.value));
}); });
}; };
// 获取诊断目录 // 获取诊断目录

View File

@@ -207,7 +207,6 @@ import {patientInfo} from '../../../store/patient.js';
import {getDepartmentList} from '@/api/public.js'; import {getDepartmentList} from '@/api/public.js';
import {getEncounterDiagnosis} from '../../api.js'; import {getEncounterDiagnosis} from '../../api.js';
import {getExaminationPage, saveCheckd} from './api'; import {getExaminationPage, saveCheckd} from './api';
import {ActivityCategory} from '@/utils/medicalConstants';
import {ElMessage, ElMessageBox} from 'element-plus'; import {ElMessage, ElMessageBox} from 'element-plus';
import {WarningFilled, Warning, Refresh, Files, Document, EditPen, Aim, DocumentCopy} from '@element-plus/icons-vue'; import {WarningFilled, Warning, Refresh, Files, Document, EditPen, Aim, DocumentCopy} from '@element-plus/icons-vue';
@@ -277,7 +276,6 @@ const getList = () => {
pageNo: 1, pageNo: 1,
pageSize: 5000, pageSize: 5000,
searchKey: '', searchKey: '',
categoryCode: ActivityCategory.TEST,
}) })
.then((res) => { .then((res) => {
if (res.code === 200 && res.data?.records) { if (res.code === 200 && res.data?.records) {
@@ -430,52 +428,11 @@ const loadEditData = () => {
const projectWithDepartment = (selectProjectIds) => { const projectWithDepartment = (selectProjectIds) => {
if (!selectProjectIds || selectProjectIds.length === 0) { if (!selectProjectIds || selectProjectIds.length === 0) {
form.targetDepartment = ''; form.targetDepartment = '';
return;
} }
// 获取第一个选中项目的发往科室orgId
// 优先使用配置的发往科室,如果没有则保留手动选择
const selectedProject = applicationListAll.value?.find(
item => selectProjectIds.includes(item.adviceDefinitionId)
);
if (selectedProject && selectedProject.orgId) {
// 项目配置了发往科室,自动填充
const orgId = selectedProject.orgId;
const orgName = selectedProject.orgName;
// 查找树中对应的节点,获取正确的 id 类型
const findNode = (nodes, targetId) => {
if (!nodes) return null;
for (const node of nodes) {
if (String(node.id) === String(targetId)) {
return node;
}
if (node.children && node.children.length > 0) {
const found = findNode(node.children, targetId);
if (found) return found;
}
}
return null;
};
const treeNode = findNode(orgOptions.value, orgId);
if (treeNode) {
// 使用树节点的原始 id 值(确保类型匹配)
form.targetDepartment = treeNode.id;
} else {
// 科室不在列表中(可能已删除),留空让用户手动选择
form.targetDepartment = '';
}
}
// 如果没有配置发往科室,保留手动选择(不修改 form.targetDepartment
}; };
watch(() => transferValue.value, (newValue) => { watch(() => transferValue.value, (newValue) => {
// 使用 nextTick 确保 DOM 更新完成后再设置值 projectWithDepartment(newValue);
nextTick(() => {
projectWithDepartment(newValue);
});
}); });
const getPriorityCode = () => { const getPriorityCode = () => {

View File

@@ -232,21 +232,64 @@ onMounted(() => {
* type(1watch监听类型 2:点击保存类型) * type(1watch监听类型 2:点击保存类型)
* selectProjectIds(选中项目的id数组) * selectProjectIds(选中项目的id数组)
* */ * */
const projectWithDepartment = (selectProjectIds) => { const projectWithDepartment = (selectProjectIds, type) => {
if (!selectProjectIds || selectProjectIds.length === 0) { //1.获取选中的项目 2.判断项目的执行科室是否相同 3.判断执行科室是否配置 4.将项目的执行科室复值到执行科室下拉选位置
form.targetDepartment = ''; let isRelease = true;
// 选中项目的数组
const arr = [];
// 根据选中的项目id查找对应的项目
selectProjectIds.forEach((element) => {
const searchData = applicationList.value.find((item) => {
return element == item.adviceDefinitionId;
});
arr.push(searchData);
});
// 清空科室
form.targetDepartment = '';
if (arr.length > 0) {
const obj = arr[0];
// 判断科室是否相同
const isCompare = arr.every((item) => {
return item.orgId == obj.orgId;
});
if (!isCompare) {
ElMessage({
type: 'error',
message: '执行科室不同',
});
isRelease = false;
}
// 选中项目中的执行科室id与全部科室数据做匹配
const findItem = findTreeItem(orgOptions.value, obj.orgId);
if (!findItem) {
isRelease = false;
ElMessage({
type: 'error',
message: '未找到项目执行的科室',
});
}
if (type == 1) {
if (isRelease) {
form.targetDepartment = findItem.id;
}
}
} }
return isRelease;
}; };
// 监听选择项目变化 // 监听选择项目变化
watch(() => transferValue.value, (newValue) => { watch(
projectWithDepartment(newValue); () => transferValue.value,
}); (newValue) => {
projectWithDepartment(newValue, 1);
}
);
const submit = () => { const submit = () => {
if (transferValue.value.length == 0) { if (transferValue.value.length == 0) {
return proxy.$message.error('请选择手术项目'); return proxy.$message.error('请选择申请单');
} }
if (!form.targetDepartment) { if (!projectWithDepartment(transferValue.value, 2)) {
return proxy.$message.error('请选择发往科室'); return;
} }
let applicationListAllFilter = applicationListAll.value.filter((item) => { let applicationListAllFilter = applicationListAll.value.filter((item) => {
return transferValue.value.includes(item.adviceDefinitionId); return transferValue.value.includes(item.adviceDefinitionId);
@@ -259,7 +302,7 @@ const submit = () => {
unitCode: item.unitCode, unitCode: item.unitCode,
unitPrice: item.price, unitPrice: item.price,
totalPrice: item.price, totalPrice: item.price,
positionId: form.targetDepartment || item.positionId, // 用户手动选择的发往科室优先于项目默认执行科室 positionId: item.positionId,
definitionId: item.chargeItemDefinitionId, definitionId: item.chargeItemDefinitionId,
accountId: patientInfo.value.accountId, accountId: patientInfo.value.accountId,
}; };

View File

@@ -198,7 +198,7 @@
v-model="scope.row.adviceName" v-model="scope.row.adviceName"
placeholder="请选择项目" placeholder="请选择项目"
@input="handleChange" @input="handleChange"
@focus="handleFocus(scope.row, scope.$index)" @click="handleFocus(scope.row, scope.$index)"
@keyup.enter.stop="handleFocus(scope.row, scope.$index)" @keyup.enter.stop="handleFocus(scope.row, scope.$index)"
@keydown=" @keydown="
(e) => { (e) => {
@@ -429,8 +429,6 @@ const props = defineProps({
}); });
const isAdding = ref(false); const isAdding = ref(false);
const isSaving = ref(false); const isSaving = ref(false);
// 标记双击编辑的是否为已有数据的行(用于保存后是否自动添加下一行)
const wasDoubleClickEdit = ref(false);
const prescriptionRef = ref(); const prescriptionRef = ref();
const expandOrder = ref([]); //目前的展开行 const expandOrder = ref([]); //目前的展开行
const stockList = ref([]); const stockList = ref([]);
@@ -607,26 +605,13 @@ function getListInfo(addNewRow) {
prescriptionList.value = res.data prescriptionList.value = res.data
.map((item) => { .map((item) => {
const parsedContent = JSON.parse(item.contentJson); const parsedContent = JSON.parse(item.contentJson);
// 构造 unitCodeList确保编辑时下拉框有正确的选项
console.log('【DEBUG】unitCode:', parsedContent?.unitCode, typeof parsedContent?.unitCode, 'unitCodeList:', JSON.stringify(parsedContent?.unitCodeList));
const unitCodeListData = parsedContent?.unitCodeList || [
{ value: String(parsedContent?.unitCode ?? item.unitCode ?? ''), label: parsedContent?.unitCode_dictText ?? item.unitCode_dictText ?? '', type: 'unit' },
{ value: String(parsedContent?.doseUnitCode ?? ''), label: parsedContent?.doseUnitCode_dictText ?? '', type: 'dose' },
{ value: String(parsedContent?.minUnitCode ?? ''), label: parsedContent?.minUnitCode_dictText ?? '', type: 'minUnit' },
];
return { return {
...parsedContent, ...parsedContent,
...item, ...item,
// 🔧 修复contentJson 中的 totalPrice 优先于 charge_item 表的 totalPrice
// charge_item.totalPrice 可能为 0 或 null新建医嘱时导致总金额显示为 "-"
totalPrice: parsedContent?.totalPrice || item.totalPrice,
isEdit: false, isEdit: false,
showPopover: false, showPopover: false,
doseQuantity: parsedContent?.doseQuantity, doseQuantity: parsedContent?.doseQuantity,
doseUnitCode_dictText: parsedContent?.doseUnitCode_dictText, doseUnitCode_dictText: parsedContent?.doseUnitCode_dictText,
// 确保 unitCode 为字符串类型,与 unitCodeList 的 option value 类型一致
unitCode: String(parsedContent?.unitCode ?? item.unitCode ?? ''),
unitCodeList: unitCodeListData,
// 确保 therapyEnum 被正确设置,优先使用 contentJson 中的值 // 确保 therapyEnum 被正确设置,优先使用 contentJson 中的值
therapyEnum: String(parsedContent?.therapyEnum ?? item.therapyEnum ?? '1'), therapyEnum: String(parsedContent?.therapyEnum ?? item.therapyEnum ?? '1'),
// 🔧 修复:确保 orgId 为 String 类型,与 organization 树的 id 类型一致 // 🔧 修复:确保 orgId 为 String 类型,与 organization 树的 id 类型一致
@@ -640,10 +625,6 @@ function getListInfo(addNewRow) {
}; };
}) })
.sort((a, b) => { .sort((a, b) => {
// 没有 requestTime 的项(新增/组套添加)排在最前面
if (!a.requestTime && !b.requestTime) return 0;
if (!a.requestTime) return -1;
if (!b.requestTime) return 1;
return new Date(b.requestTime) - new Date(a.requestTime); return new Date(b.requestTime) - new Date(a.requestTime);
}); });
getGroupMarkers(); // 更新标记 getGroupMarkers(); // 更新标记
@@ -810,7 +791,7 @@ function checkUnit(item, row) {
} }
} }
// 行双击打开编辑块:待保存、待签发医嘱均可编辑;已签发/已完成/停止不允许编辑 // 行双击打开编辑块,仅待发送的可编辑
function clickRowDb(row, column, event) { function clickRowDb(row, column, event) {
// 检查点击的是否是复选框 // 检查点击的是否是复选框
if (event && event.target.closest('.el-checkbox')) { if (event && event.target.closest('.el-checkbox')) {
@@ -821,18 +802,14 @@ function clickRowDb(row, column, event) {
return; return;
} }
row.showPopover = false; row.showPopover = false;
if (row.statusEnum == 1) { // “待签发(已保存 requestId存在)”不允许再编辑;仅“待保存(无requestId)”允许编辑
if (row.statusEnum == 1 && !row.requestId) {
// 确保治疗类型为字符串,方便与单选框 label 对齐,默认为长期医嘱('1') // 确保治疗类型为字符串,方便与单选框 label 对齐,默认为长期医嘱('1')
row.therapyEnum = String(row.therapyEnum ?? '1'); row.therapyEnum = String(row.therapyEnum ?? '1');
row.isEdit = true; row.isEdit = true;
const index = prescriptionList.value.findIndex((item) => item.uniqueKey === row.uniqueKey); const index = prescriptionList.value.findIndex((item) => item.uniqueKey === row.uniqueKey);
rowIndex.value = index; prescriptionList.value[index] = row;
if (index !== -1) {
prescriptionList.value[index] = row;
}
expandOrder.value = [row.uniqueKey]; expandOrder.value = [row.uniqueKey];
} else {
proxy.$modal.msgWarning('仅待保存或待签发医嘱允许编辑');
} }
} }
@@ -900,16 +877,31 @@ function handleDiagnosisChange(item) {
function handleFocus(row, index) { function handleFocus(row, index) {
rowIndex.value = index; rowIndex.value = index;
row.showPopover = true; row.showPopover = true;
// Bug #555: handleFocus 只负责开 popover 和初始化查询参数,搜索由 handleChange 统一处理
// 避免异步 refresh 用旧闭包 searchKey 覆盖 handleChange 的搜索结果
const adviceType = row.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType; const adviceType = row.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
let categoryCode = ''; // 用 adviceType + categoryCode 组合查找匹配的选项
if (row.adviceType !== undefined) { const selectValue = (adviceType == 1 && row.categoryCode) ? '1-' + row.categoryCode : adviceType;
const selectValue = (adviceType == 1 && row.categoryCode) ? '1-' + row.categoryCode : adviceType; const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType);
const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType); // If the row has an explicit adviceType (saved/existing row), use its own categoryCode.
categoryCode = selectedItem ? selectedItem.categoryCode : (row.categoryCode || ''); // If no type is selected (new row), use empty string for global search across all categories.
} const categoryCode = selectedItem ? selectedItem.categoryCode : (row.adviceType != null ? (row.categoryCode || '') : '');
adviceQueryParams.value = { adviceType, categoryCode, searchKey: '' }; const searchKey = row.adviceName || '';
nextTick(() => {
nextTick(() => {
const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
if (tableRef && tableRef.refresh) {
tableRef.refresh(adviceType, categoryCode, searchKey);
} else {
// fallback: 如果双重 nextTick 仍未挂载,延迟 100ms 再试
setTimeout(() => {
const tableRef2 = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
if (tableRef2 && tableRef2.refresh) {
tableRef2.refresh(adviceType, categoryCode, searchKey);
}
}, 100);
}
});
});
} }
function handleBlur(row) { function handleBlur(row) {
@@ -918,24 +910,20 @@ function handleBlur(row) {
function handleChange(value) { function handleChange(value) {
adviceQueryParams.value.searchKey = value; adviceQueryParams.value.searchKey = value;
// @focus 已先于 @input 执行rowIndex 必定有效 // 搜索词变化时,调用当前行子组件的 refresh 方法
const currentIndex = rowIndex.value; const index = rowIndex.value;
if (currentIndex < 0) return; if (index >= 0) {
const row = filterPrescriptionList.value[currentIndex]; const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
// popover 被 blur 关闭后,用户继续输入时自行打开 if (tableRef && tableRef.refresh) {
if (!row.showPopover) { const row = filterPrescriptionList.value[index];
row.showPopover = true; const adviceType = row?.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
} // 用 adviceType + categoryCode 组合查找匹配的选项
const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[currentIndex] : adviceTableRef.value;
if (tableRef && tableRef.refresh) {
const adviceType = row?.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
let categoryCode = '';
if (row?.adviceType !== undefined) {
const selectValue = (adviceType == 1 && row?.categoryCode) ? '1-' + row.categoryCode : adviceType; const selectValue = (adviceType == 1 && row?.categoryCode) ? '1-' + row.categoryCode : adviceType;
const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType); const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType);
categoryCode = selectedItem ? selectedItem.categoryCode : (adviceQueryParams.value.categoryCode || ''); // 修复Bug #486当行没有显式选择医嘱类型时不传categoryCode让搜索在全药库中进行
const categoryCode = selectedItem ? selectedItem.categoryCode : (row?.adviceType !== undefined ? (adviceQueryParams.value.categoryCode || '') : '');
tableRef.refresh(adviceType, categoryCode, value);
} }
tableRef.refresh(adviceType, categoryCode, value);
} }
} }
@@ -1194,7 +1182,7 @@ function handleSave() {
// 此处签发处方和单行保存处方传参相同后台已经将传参存为JSON字符串此处直接转换为JSON即可 // 此处签发处方和单行保存处方传参相同后台已经将传参存为JSON字符串此处直接转换为JSON即可
loading.value = true; loading.value = true;
let list = saveList.map((item) => { let list = saveList.map((item) => {
const parsedContent = item.contentJson ? JSON.parse(item.contentJson) : {}; const parsedContent = JSON.parse(item.contentJson);
return { return {
...parsedContent, ...parsedContent,
adviceType: item.adviceType, adviceType: item.adviceType,
@@ -1392,9 +1380,7 @@ function handleSaveSign(row, index) {
} }
}); });
} else { } else {
// 仅通过【新增】按钮创建的医嘱保存后才自动添加下一行空医嘱 if (prescriptionList.value[0].adviceName) {
// 双击编辑已有"待保存"医嘱保存时,不应自动添加空行
if (isAdding.value && prescriptionList.value[0].adviceName) {
handleAddPrescription(); handleAddPrescription();
} }
} }
@@ -1455,14 +1441,14 @@ function handleSaveBatch() {
function setValue(row) { function setValue(row) {
// 构造单位列表 // 构造单位列表
unitCodeList.value = [ unitCodeList.value = [
{ value: String(row.unitCode), label: row.unitCode_dictText, type: 'unit' }, { value: row.unitCode, label: row.unitCode_dictText, type: 'unit' },
{ {
value: String(row.doseUnitCode), value: row.doseUnitCode,
label: row.doseUnitCode_dictText, label: row.doseUnitCode_dictText,
type: 'dose', type: 'dose',
}, },
{ {
value: String(row.minUnitCode), value: row.minUnitCode,
label: row.minUnitCode_dictText, label: row.minUnitCode_dictText,
type: 'minUnit', type: 'minUnit',
}, },
@@ -1528,8 +1514,8 @@ function setValue(row) {
// dose: undefined, Removed to preserve dose value from group package // dose: undefined, Removed to preserve dose value from group package
unitCodeList: unitCodeList.value, unitCodeList: unitCodeList.value,
doseUnitCode: row.doseUnitCode, doseUnitCode: row.doseUnitCode,
minUnitCode: String(row.minUnitCode), minUnitCode: row.minUnitCode,
unitCode: row.partAttributeEnum == 1 ? String(row.minUnitCode) : String(row.unitCode), unitCode: row.partAttributeEnum == 1 ? row.minUnitCode : row.unitCode,
categoryEnum: row.categoryCode, categoryEnum: row.categoryCode,
definitionId: row.chargeItemDefinitionId, definitionId: row.chargeItemDefinitionId,
executeNum: 1, executeNum: 1,
@@ -1545,13 +1531,6 @@ function setValue(row) {
? new Decimal(selectedStock.price).div(row.partPercent).toFixed(6) ? new Decimal(selectedStock.price).div(row.partPercent).toFixed(6)
: prevRow.minUnitPrice, : prevRow.minUnitPrice,
positionName: selectedStock?.locationName, positionName: selectedStock?.locationName,
totalPrice: row.quantity
? (String(row.unitCode) == String(row.minUnitCode)
? (row.quantity * (selectedStock
? new Decimal(selectedStock.price).div(row.partPercent).toFixed(6)
: prevRow.minUnitPrice)).toFixed(6)
: (row.quantity * (selectedStock?.price ?? 0)).toFixed(6))
: undefined,
} }
: { : {
quantity: 1, quantity: 1,
@@ -1572,24 +1551,11 @@ function handleSaveGroup(orderGroupList) {
let successCount = 0; let successCount = 0;
// 收集所有要添加的新行,最后统一 unshift 到数组开头(置顶显示)
const newRows = [];
// 记录循环前的数组长度,用于清理循环中创建的临时行
const originalLength = prescriptionList.value.length;
orderGroupList.forEach((item) => { orderGroupList.forEach((item) => {
// 使用临时索引,先追加到末尾用于 setValue 填充 rowIndex.value = prescriptionList.value.length;
const tempIndex = prescriptionList.value.length;
prescriptionList.value[tempIndex] = {
uniqueKey: nextId.value++,
isEdit: false,
statusEnum: 1,
};
if (!item) { if (!item) {
console.warn('组套中的项目为空'); console.warn('组套中的项目为空');
prescriptionList.value.splice(tempIndex, 1);
return; return;
} }
@@ -1615,12 +1581,18 @@ function handleSaveGroup(orderGroupList) {
therapyEnum: item.orderDetailInfos?.therapyEnum || '1', therapyEnum: item.orderDetailInfos?.therapyEnum || '1',
}; };
rowIndex.value = tempIndex; // 预初始化空行(组套项带预填值,设为 false 让明细字段在表格中直接展示)
prescriptionList.value[rowIndex.value] = {
uniqueKey: nextId.value++,
isEdit: false,
statusEnum: 1,
};
setValue(mergedDetail); setValue(mergedDetail);
// 创建新的处方项目 // 创建新的处方项目
const newRow = { const newRow = {
...prescriptionList.value[tempIndex], ...prescriptionList.value[rowIndex.value],
patientId: patientInfo.value.patientId, patientId: patientInfo.value.patientId,
encounterId: patientInfo.value.encounterId, encounterId: patientInfo.value.encounterId,
accountId: accountId.value, accountId: accountId.value,
@@ -1639,14 +1611,11 @@ function handleSaveGroup(orderGroupList) {
orgId: resolveOrgId(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '', orgId: resolveOrgId(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '',
// 🔧 修复:同时存储 orgName确保树匹配不到时仍有中文名称可显示 // 🔧 修复:同时存储 orgName确保树匹配不到时仍有中文名称可显示
orgName: findOrgName(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || mergedDetail.orgName || patientInfo.value?.inHospitalOrgName || '', orgName: findOrgName(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || mergedDetail.orgName || patientInfo.value?.inHospitalOrgName || '',
dbOpType: prescriptionList.value[tempIndex].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,
diagnosisName: diagnosisName.value, therapyEnum: prescriptionList.value[rowIndex.value]?.therapyEnum || mergedDetail.therapyEnum || '1',
therapyEnum: prescriptionList.value[tempIndex]?.therapyEnum || mergedDetail.therapyEnum || '1',
// 🔧 修复:确保组套医嘱的 categoryEnum 被正确映射,防止后端 NPE
categoryEnum: mergedDetail?.categoryEnum || mergedDetail?.categoryCode || item?.categoryCode,
}; };
// 计算价格和总量 // 计算价格和总量
@@ -1663,14 +1632,11 @@ function handleSaveGroup(orderGroupList) {
} }
newRow.contentJson = JSON.stringify(newRow); newRow.contentJson = JSON.stringify(newRow);
newRows.push(newRow); prescriptionList.value[rowIndex.value] = newRow;
successCount++; successCount++;
}); });
// 清理循环中创建的临时行,统一添加到数组开头(置顶显示) if (successCount > 0) {
if (newRows.length > 0) {
prescriptionList.value.splice(originalLength); // 移除循环中追加到末尾的临时行
prescriptionList.value.unshift(...newRows);
proxy.$modal.msgSuccess(`成功添加 ${successCount} 个医嘱项`); proxy.$modal.msgSuccess(`成功添加 ${successCount} 个医嘱项`);
} }
} }
@@ -1690,8 +1656,6 @@ function handleSaveHistory(value) {
encounterDiagnosisId: encounterDiagnosisId.value, encounterDiagnosisId: encounterDiagnosisId.value,
// 确保 therapyEnum 被正确传递,默认为长期医嘱('1') // 确保 therapyEnum 被正确传递,默认为长期医嘱('1')
therapyEnum: value.therapyEnum || '1', therapyEnum: value.therapyEnum || '1',
// 🔧 修复:历史医嘱的 categoryCode(String) 映射为后端 categoryEnum(Integer),防止 NPE
categoryEnum: value.categoryEnum || value.categoryCode,
contentJson: JSON.stringify({ contentJson: JSON.stringify({
...value, ...value,
therapyEnum: value.therapyEnum || '1', therapyEnum: value.therapyEnum || '1',

View File

@@ -348,14 +348,13 @@ const adviceTypeList = computed(() => {
return val === 3 || val === 4; return val === 3 || val === 4;
}).map(item => ({ }).map(item => ({
label: item.label, label: item.label,
// drord_doctor_type 中耗材是 4但 /advice-base-info 后端耗材类型是 2 value: parseInt(item.value)
value: parseInt(item.value) === 4 ? 2 : parseInt(item.value)
})); }));
return [...filtered, { label: '全部', value: '' }]; return [...filtered, { label: '全部', value: '' }];
} }
// 默认值 // 默认值
return [ return [
{ label: '耗材', value: 2 }, { label: '耗材', value: 4 },
{ label: '诊疗', value: 3 }, { label: '诊疗', value: 3 },
{ label: '全部', value: '' }, { label: '全部', value: '' },
]; ];
@@ -374,8 +373,7 @@ const filterKeywords = ref({});
const queryParams = ref({ const queryParams = ref({
pageSize: 100, pageSize: 100,
pageNum: 1, pageNum: 1,
// 默认加载全部类型药品1+耗材2+诊疗3 adviceTypes: '2,3',
adviceTypes: [1, 2, 3],
}); });
/** /**
* 医嘱提交数据模型 * 医嘱提交数据模型
@@ -484,9 +482,8 @@ watch(
(visible) => { (visible) => {
if (visible) { if (visible) {
executeTime.value = formatDateStr(new Date(), 'YYYY-MM-DD HH:mm:ss'); executeTime.value = formatDateStr(new Date(), 'YYYY-MM-DD HH:mm:ss');
// 弹窗打开时按当前患者科室重新加载,避免复用上一次患者/登录科室的结果 // 弹窗打开时重新加载科室和位置选项,确保数据最新
loadDepartmentOptions(); loadDepartmentOptions();
getAdviceBaseInfos();
getDiseaseInitLoc(16); getDiseaseInitLoc(16);
} else { } else {
resetData(); resetData();
@@ -558,28 +555,12 @@ function loadDepartmentOptions() {
function getAdviceBaseInfos() { function getAdviceBaseInfos() {
adviceLoading.value = true; adviceLoading.value = true;
queryParams.value.searchKey = searchText.value; queryParams.value.searchKey = searchText.value;
// 字典值(3=诊疗,4=耗材)映射为后端adviceType(2=耗材,3=诊疗) queryParams.value.adviceType = adviceType.value;
if (adviceType.value === 4) {
queryParams.value.adviceTypes = [2];
} else if (adviceType.value === 3) {
queryParams.value.adviceTypes = [3];
} else {
queryParams.value.adviceTypes = [1, 2, 3];
}
queryParams.value.organizationId = orgId.value; queryParams.value.organizationId = orgId.value;
queryParams.value.adviceTypes = normalizeAdviceTypesForQuery(adviceType.value);
queryParams.value.organizationId = props.patientInfo.organizationId || orgId.value;
queryParams.value.pricingFlag = 1; // 划价标记 queryParams.value.pricingFlag = 1; // 划价标记
getAdviceBaseInfo(queryParams.value) getAdviceBaseInfo(queryParams.value)
.then((res) => { .then((res) => {
const list = res.data?.records || []; AdviceBaseInfoList.value = res.data?.records || [];
// 药品(1)和耗材(2)必须有库存才能展示,诊疗(3)无库存概念不过滤
AdviceBaseInfoList.value = list.filter(item => {
if (item.adviceType === 1 || item.adviceType === 2) {
return item.inventoryList && item.inventoryList.length > 0;
}
return true;
});
}) })
.finally(() => { .finally(() => {
adviceLoading.value = false; adviceLoading.value = false;
@@ -624,12 +605,6 @@ function getItemType_Text(type) {
const map = { 2: '耗材', 3: '诊疗' }; const map = { 2: '耗材', 3: '诊疗' };
return map[type] || '其他'; return map[type] || '其他';
} }
function normalizeAdviceTypesForQuery(type) {
if (type === '' || type === undefined || type === null) {
return '2,3';
}
return Number(type) === 4 ? 2 : type;
}
function getUnitCodeOptions(row) { function getUnitCodeOptions(row) {
const unitCodes = []; const unitCodes = [];
// 大单位:优先用 codecode 缺失时用字典文本兜底 // 大单位:优先用 codecode 缺失时用字典文本兜底

View File

@@ -262,7 +262,7 @@
</template> </template>
<script setup> <script setup>
import {nextTick, onMounted, ref} from 'vue'; import {computed, nextTick, onMounted, ref, watch} from 'vue';
import {ElMessage, ElMessageBox} from 'element-plus'; import {ElMessage, ElMessageBox} from 'element-plus';
// Element Plus 图标导入 // Element Plus 图标导入
import {User} from '@element-plus/icons-vue'; import {User} from '@element-plus/icons-vue';
@@ -366,9 +366,9 @@ const rawPrescriptionList = ref([]); // 原始未分组数据
const groupedPrescriptionList = ref([]); // 按encounterId分组后的数据 const groupedPrescriptionList = ref([]); // 按encounterId分组后的数据
const activeCollapseNames = ref([]); // Collapse激活状态 const activeCollapseNames = ref([]); // Collapse激活状态
const selectedRows = ref({}); // 选中的行数据 const selectedRows = ref({}); // 选中的行数据
const totalItemsCount = ref(0); // 总医嘱项数
const totalAmount = ref(0); // 总金额保留4位小数
const dialogVisible = ref(false); const dialogVisible = ref(false);
/** Tab 切换同步日期时跳过 date-picker change避免与 v-model 循环触发 */
const syncingDateFromTab = ref(false);
const selectedFeeItems = ref([]); const selectedFeeItems = ref([]);
const currentPatientInfo = ref(null); const currentPatientInfo = ref(null);
const queryParams = ref({ const queryParams = ref({
@@ -381,6 +381,24 @@ const userStore = useUserStore();
const userId = ref(safeGet(userStore, 'id', '')); const userId = ref(safeGet(userStore, 'id', ''));
const orgId = ref(safeGet(userStore, 'orgId', '')); const orgId = ref(safeGet(userStore, 'orgId', ''));
// ========== 计算属性 ==========
// 计算总统计信息(总项数、总金额)
const calculateTotalStats = computed(() => {
let itemsCount = 0;
let amount = 0;
safeArray(groupedPrescriptionList.value).forEach((patientGroup) => {
safeArray(patientGroup).forEach((item) => {
itemsCount++;
// 累加单价保留4位小数精度
amount = Math.round((amount + Number(safeGet(item, 'unitPrice', 0))) * 10000) / 10000;
});
});
totalItemsCount.value = itemsCount;
totalAmount.value = amount;
});
// ========== 方法 ========== // ========== 方法 ==========
/** /**
* 计算单个患者的总金额保留4位小数 * 计算单个患者的总金额保留4位小数
@@ -429,19 +447,16 @@ const handleTableSelectionChange = (index, val) => {
}; };
/** /**
* 按 Tab 同步日期范围(避免 date-picker @change 与 Tab v-model 互相覆盖) * 日期Tab切换
* @param {string} rangeType - today | yesterday | custom * @param {Object} tab - 标签页
*/ */
const applyDateRangeByTab = (rangeType) => { const handleDateTabClick = (tab) => {
const today = new Date(); const today = new Date();
const yesterday = new Date(today); const yesterday = new Date(today);
yesterday.setDate(today.getDate() - 1); yesterday.setDate(today.getDate() - 1);
const format = (date) => formatDateStr(date, 'YYYY-MM-DD'); const format = (date) => formatDateStr(date, 'YYYY-MM-DD');
syncingDateFromTab.value = true; switch (safeGet(tab, 'paneName')) {
dateRange.value = rangeType;
switch (rangeType) {
case 'today': case 'today':
dateRangeValue.value = [format(today), format(today)]; dateRangeValue.value = [format(today), format(today)];
break; break;
@@ -449,54 +464,27 @@ const applyDateRangeByTab = (rangeType) => {
dateRangeValue.value = [format(yesterday), format(yesterday)]; dateRangeValue.value = [format(yesterday), format(yesterday)];
break; break;
case 'custom': case 'custom':
if (safeArray(dateRangeValue.value).length < 2) { if (!dateRangeValue.value.length) {
dateRangeValue.value = [format(today), format(today)]; dateRangeValue.value = [format(today), format(today)];
} }
break; break;
default:
break;
} }
nextTick(() => {
syncingDateFromTab.value = false;
});
}; };
/** /**
* 日期Tab切换 * 日期选择器变化
* @param {Object} tab - 标签页
*/
const handleDateTabClick = (tab) => {
const rangeType = tab?.paneName ?? tab?.props?.name;
if (!rangeType) return;
applyDateRangeByTab(rangeType);
handleQuery();
};
/**
* 日期选择器变化(仅用户手动改日期时切到「自定义」)
* @param {Array} val - 选中日期 * @param {Array} val - 选中日期
*/ */
const handleDatePickerChange = (val) => { const handleDatePickerChange = (val) => {
if (syncingDateFromTab.value) return;
const dateVal = safeArray(val); const dateVal = safeArray(val);
if (dateVal.length !== 2) return; if (dateVal.length === 2) {
const start = new Date(dateVal[0]);
const end = new Date(dateVal[1]);
if (start > end) {
ElMessage.warning('开始日期不能晚于结束日期');
syncingDateFromTab.value = true;
dateRangeValue.value = [dateVal[1], dateVal[0]];
nextTick(() => {
syncingDateFromTab.value = false;
});
return;
}
if (dateRange.value !== 'custom') {
dateRange.value = 'custom'; dateRange.value = 'custom';
const start = new Date(dateVal[0]);
const end = new Date(dateVal[1]);
if (start > end) {
ElMessage.warning('开始日期不能晚于结束日期');
dateRangeValue.value = [dateVal[1], dateVal[0]];
}
} }
}; };
@@ -726,7 +714,24 @@ const handleSingleDelete = (row) => {
}; };
// ========== 初始化 ========== // ========== 初始化 ==========
onMounted(() => { onMounted(() => {
applyDateRangeByTab('today'); // 设置默认日期
const today = new Date();
const defaultDate = formatDateStr(today, 'YYYY-MM-DD');
dateRangeValue.value = [defaultDate, defaultDate];
// 监听日期变化自动查询
watch(
[dateRange, dateRangeValue],
([newRange, newVal], [oldRange, oldVal]) => {
if (oldRange !== undefined && safeArray(newVal).length === 2) {
handleQuery();
}
},
{ deep: true }
);
// 初始化统计信息
calculateTotalStats.value;
}); });
</script> </script>

View File

@@ -142,13 +142,6 @@
</template> </template>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="医嘱状态" prop="requestStatus_enumText" width="100">
<template #default="scope">
<el-tag :type="getStatusType(scope.row.requestStatus)" size="small">
{{ scope.row.requestStatus_enumText }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="执行科室" prop="positionName" width="230" /> <el-table-column label="执行科室" prop="positionName" width="230" />
<el-table-column label="签发时间" prop="requestTime" width="230" /> <el-table-column label="签发时间" prop="requestTime" width="230" />
</el-table> </el-table>
@@ -159,7 +152,7 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import {ref, computed, getCurrentInstance} from 'vue'; import {ref, computed} from 'vue';
import {adviceVerify, cancel, getPrescriptionList} from './api'; import {adviceVerify, cancel, getPrescriptionList} 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';
@@ -172,19 +165,6 @@ const { proxy } = getCurrentInstance();
const loading = ref(false); const loading = ref(false);
const chooseAll = ref(false); const chooseAll = ref(false);
const selectionTrigger = ref(0); const selectionTrigger = ref(0);
const getStatusType = (status) => {
const map = {
1: 'info', // 待发送
2: 'primary', // 已发送
3: 'success', // 已完成
4: 'warning', // 暂停
5: 'danger', // 取消/待退
6: 'danger', // 停嘱
7: 'info' // 不执行
}
return map[status] || 'info'
}
const hasDispensedSelected = computed(() => { const hasDispensedSelected = computed(() => {
selectionTrigger.value; selectionTrigger.value;
return getSelectRows().some(item => item.dispenseStatus === 4); return getSelectRows().some(item => item.dispenseStatus === 4);

View File

@@ -777,9 +777,6 @@ const InputOptions = ref([
// 日期范围 - 体征信息搜索时间 // 日期范围 - 体征信息搜索时间
const receptionTime = ref(null); const receptionTime = ref(null);
// 按钮禁用状态(保存按钮依赖)
const buttonDisabled = ref(false);
// 表单数据 - 体征录入 // 表单数据 - 体征录入
const formData = ref({ const formData = ref({
recordingDate: '', recordingDate: '',
@@ -850,10 +847,26 @@ function getPatientList() {
patientId: props.patientId patientId: props.patientId
} }
}).then((res) => { }).then((res) => {
if (res.code === 200 && res.data) {
patientList.value = Array.isArray(res.data) ? res.data : (res.data.data || []); // 判断返回的数据结构
let data = res.data;
if (res.data && res.data.data && typeof res.data.data === 'object') {
// 如果是嵌套结构 {data: {data: Array(3)}}
data = res.data.data;
} else if (res.data && typeof res.data === 'object' && res.data.data !== undefined) {
// 如果是 {code: 200, msg: '操作成功', data: Array(3)}
data = res.data.data;
}
console.log('=== data 长度 ===', data?.length);
if (res.code === 200 && data) {
console.log('=== 准备赋值 patientList.value ===');
patientList.value = data;
console.log('=== patientList.value 赋值后 ===', patientList.value);
console.log('=== patientList.value 长度 ===', patientList.value?.length);
} else { } else {
patientList.value = []; console.error('=== 查询失败或无数据 ===');
} }
}).catch(err => { }).catch(err => {
console.error('=== 查询报错 ===', err); console.error('=== 查询报错 ===', err);
@@ -872,22 +885,8 @@ function getPatientDetial() {
queryParams.value.patientId = props.patientId; queryParams.value.patientId = props.patientId;
// 默认查询今天的数据 // 默认查询今天的数据
const today = moment().format('YYYY-MM-DD'); const today = moment().format('YYYY-MM-DD');
const now = moment();
receptionTime.value = [today, today]; receptionTime.value = [today, today];
formData.value.recordingDate = today; formData.value.recordingDate = today;
// 自动填充最近的整点时间点2/6/10/14/18/22点
const hour = now.hour();
const timePoints = [2, 6, 10, 14, 18, 22];
let nearestHour = timePoints[0];
let minDiff = Math.abs(hour - nearestHour);
for (const tp of timePoints) {
const diff = Math.abs(hour - tp);
if (diff < minDiff) {
minDiff = diff;
nearestHour = tp;
}
}
formData.value.timePoint = String(nearestHour).padStart(2, '0') + '00';
// 自动加载数据 // 自动加载数据
getPatientList(); getPatientList();
listPatient(queryParams.value).then((res) => { listPatient(queryParams.value).then((res) => {
@@ -968,29 +967,6 @@ function confirmCharge() {
encounterId: props.patientInfo.encounterId, encounterId: props.patientInfo.encounterId,
}; };
// 自动获取当前日期和时间
const now = moment();
if (!params.recordingDate) {
params.recordingDate = now.format('YYYY-MM-DD');
formData.value.recordingDate = params.recordingDate;
}
if (!params.timePoint) {
// 取最近的整点时间点2/6/10/14/18/22点
const hour = now.hour();
const timePoints = [2, 6, 10, 14, 18, 22];
let nearestHour = timePoints[0];
let minDiff = Math.abs(hour - nearestHour);
for (const tp of timePoints) {
const diff = Math.abs(hour - tp);
if (diff < minDiff) {
minDiff = diff;
nearestHour = tp;
}
}
params.timePoint = String(nearestHour).padStart(2, '0') + '00';
formData.value.timePoint = params.timePoint;
}
// 收集所有录入的体征数据 // 收集所有录入的体征数据
const vitalSignsCode = []; const vitalSignsCode = [];
const vitalSignsValues = []; const vitalSignsValues = [];
@@ -1076,18 +1052,13 @@ function confirmCharge() {
vitalSignsValues.push(params.stoolVolume); vitalSignsValues.push(params.stoolVolume);
} }
// 校验:没有录入任何体征数据时提示用户
if (vitalSignsCode.length === 0) {
proxy.$modal.msgWarning('请录入患者体征信息');
return;
}
params.vitalSignsCode = vitalSignsCode; params.vitalSignsCode = vitalSignsCode;
params.vitalSignsValues = vitalSignsValues; params.vitalSignsValues = vitalSignsValues;
params.recordingDate = formData.value.recordingDate || moment(new Date()).format('YYYY-MM-DD');
addVitalSigns(params).then(res => { addVitalSigns(params).then(res => {
if (res.code === 200) { if (res.code === 200) {
proxy.$modal.msgSuccess('保存成功'); proxy.msgSuccess('保存成功');
// 保存成功后刷新列表 // 保存成功后刷新列表
getPatientList(); getPatientList();
// 清空表单 // 清空表单
@@ -1117,9 +1088,6 @@ function confirmCharge() {
stoolVolume: '', stoolVolume: '',
}; };
} }
}).catch(err => {
console.error('保存体征数据失败:', err);
proxy.$modal.msgError('保存失败,请重试');
}); });
} }
/** 重置操作表单 */ /** 重置操作表单 */

View File

@@ -386,7 +386,7 @@
:disabled="viewStatus == 'view'" :disabled="viewStatus == 'view'"
v-model="scope.row.itemQuantityDisplay" v-model="scope.row.itemQuantityDisplay"
placeholder="" placeholder=""
@input="(value) => handleItemQuantityChange(scope.row, value)" @change="(value) => handleItemQuantityChange(scope.row, value)"
:class="{ 'error-border': scope.row.error }" :class="{ 'error-border': scope.row.error }"
/> />
</div> </div>
@@ -971,10 +971,6 @@ function handleItemQuantityChange(row, value) {
quantityTemp = value; quantityTemp = value;
} }
row.totalPrice = ((row.price * quantityTemp) / row.partPercent).toFixed(2); row.totalPrice = ((row.price * quantityTemp) / row.partPercent).toFixed(2);
// 数量变更后重置保存标记,允许重新提交
row.isSave = false;
// 同步更新底部合计金额
handleTotalAmount();
} }
function handelApply() { function handelApply() {
@@ -1266,108 +1262,128 @@ function editBatchTransfer(index) {
} }
function handleSave(row, index) { function handleSave(row, index) {
// 过滤出未保存的行,已保存的行不重复提交 rowList.value = [];
const listToCheck = route.query.supplyBusNo if (route.query.supplyBusNo) {
? forms.purchaseinventoryList // 编辑
: form.purchaseinventoryList; forms.purchaseinventoryList.map((row, index) => {
const unsavedList = listToCheck.filter(item => !item.isSave); if (row) {
if (unsavedList.length === 0) { proxy.$refs['receiptHeaderRef'].validate((valid) => {
proxy.$modal.msgWarning('所有行均已保存,无需重复提交'); if (valid) {
return; proxy.$refs['formRef'].validate((valid) => {
if (valid) {
if (row.unitCode == row.unitList.minUnitCode) {
row.itemQuantity = forms.purchaseinventoryList[index].olditemQuantity
? forms.purchaseinventoryList[index].olditemQuantity
: forms.purchaseinventoryList[index].itemQuantity;
} else {
row.itemQuantity = forms.purchaseinventoryList[index].itemMaxQuantity
? forms.purchaseinventoryList[index].itemMaxQuantity
: forms.purchaseinventoryList[index].itemQuantity;
}
// let rows = JSON.parse(JSON.stringify(row))
// delete rows.itemMaxQuantity
if (row.unitCode == row.unitCode_dictText) {
if (row.unitCode_dictText == row.unitList.minUnitCode_dictText) {
row.unitCode = row.unitList.minUnitCode;
} else {
row.unitCode = row.unitList.unitCode;
row.unitCode_dictText = row.unitList.unitCode_dictText;
}
}
if (row.unitCode == row.unitList.unitCode) {
row.unitCode_dictText = row.unitList.unitCode_dictText;
} else if (row.unitCode == row.unitList.minUnitCode) {
row.unitCode_dictText = row.unitList.minUnitCode_dictText;
}
if (!forms.purchaseinventoryList[index].price || forms.purchaseinventoryList[index].price <= 0) {
proxy.$message.warning('调拨单价不能为空或为0请检查');
return;
}
forms.purchaseinventoryList[index].totalPrice =
forms.purchaseinventoryList[index].price * forms.purchaseinventoryList[index].itemQuantity;
rowList.value.push(JSON.parse(JSON.stringify(row)));
if (
rowList._rawValue &&
rowList._rawValue.length == forms.purchaseinventoryList.length
) {
addTransferProducts(rowList._rawValue);
}
}
});
}
});
}
});
} else {
//新增
form.purchaseinventoryList.map((row, index) => {
if (row) {
proxy.$refs['receiptHeaderRef'].validate((valid) => {
if (valid) {
proxy.$refs['formRef'].validate((valid) => {
if (valid) {
let rows = JSON.parse(JSON.stringify(row));
delete rows.itemMaxQuantity;
if (rows.unitCode == rows.unitList.minUnitCode) {
rows.itemQuantity = form.purchaseinventoryList[index].olditemQuantity
? form.purchaseinventoryList[index].olditemQuantity
: form.purchaseinventoryList[index].itemQuantity;
} else {
rows.itemQuantity = form.purchaseinventoryList[index].itemMaxQuantity
? form.purchaseinventoryList[index].itemMaxQuantity
: form.purchaseinventoryList[index].itemQuantity;
}
if (rows.unitCode == rows.unitCode_dictText) {
if (rows.unitCode_dictText == rows.unitList.minUnitCode_dictText) {
rows.unitCode = rows.unitList.minUnitCode;
} else {
rows.unitCode = rows.unitList.unitCode;
rows.unitCode_dictText = rows.unitList.unitCode_dictText;
}
}
if (rows.unitCode == rows.unitList.unitCode) {
rows.unitCode_dictText = rows.unitList.unitCode_dictText;
} else if (rows.unitCode == rows.unitList.minUnitCode) {
rows.unitCode_dictText = rows.unitList.minUnitCode_dictText;
}
if (!form.purchaseinventoryList[index].price || form.purchaseinventoryList[index].price <= 0) {
proxy.$message.warning('调拨单价不能为空或为0请检查');
return;
}
form.purchaseinventoryList[index].totalPrice =
form.purchaseinventoryList[index].price * form.purchaseinventoryList[index].itemQuantity;
rowList.value.push(JSON.parse(JSON.stringify(rows)));
if (
rowList._rawValue &&
rowList._rawValue.length == form.purchaseinventoryList.length
) {
addTransferProducts(rowList._rawValue);
}
}
});
}
});
}
});
} }
// 先校验表头
proxy.$refs['receiptHeaderRef'].validate((headerValid) => {
if (!headerValid) return;
// 逐行校验(避免异步回调导致重复提交)
const rowsToSave = [];
for (let i = 0; i < form.purchaseinventoryList.length; i++) {
const r = form.purchaseinventoryList[i];
if (!r) continue;
// 跳过已保存的行,避免重复提交导致预扣减库存叠加
if (r.isSave) continue;
// 校验当前行的必填字段
let rowValid = true;
for (const prop of ['name', 'unitCode']) {
const formRef = proxy.$refs['formRef'];
if (formRef && formRef.validateField) {
formRef.validateField(`purchaseinventoryList.${i}.${prop}`, (valid) => {
if (valid) rowValid = false;
});
}
}
if (!rowValid) {
proxy.$modal.msgWarning('第' + (i + 1) + '行数据不完整,请检查');
return;
}
// 单价校验
if (!r.price || r.price <= 0) {
proxy.$modal.msgWarning('第' + (i + 1) + '行调拨单价不能为空或为0');
return;
}
// 单位处理
const rowData = route.query.supplyBusNo
? JSON.parse(JSON.stringify(forms.purchaseinventoryList[i]))
: JSON.parse(JSON.stringify(r));
delete rowData.itemMaxQuantity;
if (rowData.unitCode == rowData.unitList?.minUnitCode) {
rowData.itemQuantity = r.olditemQuantity || r.itemQuantity;
} else {
rowData.itemQuantity = r.itemMaxQuantity || r.itemQuantity;
}
if (rowData.unitCode == rowData.unitCode_dictText) {
if (rowData.unitCode_dictText == rowData.unitList?.minUnitCode_dictText) {
rowData.unitCode = rowData.unitList.minUnitCode;
} else {
rowData.unitCode = rowData.unitList.unitCode;
rowData.unitCode_dictText = rowData.unitList.unitCode_dictText;
}
}
if (rowData.unitCode == rowData.unitList?.unitCode) {
rowData.unitCode_dictText = rowData.unitList.unitCode_dictText;
} else if (rowData.unitCode == rowData.unitList?.minUnitCode) {
rowData.unitCode_dictText = rowData.unitList.minUnitCode_dictText;
}
// 计算总价
r.totalPrice = r.price * rowData.itemQuantity;
rowsToSave.push(rowData);
}
// 所有行校验通过,一次性提交
if (rowsToSave.length > 0) {
addTransferProducts(rowsToSave);
}
});
} }
function addTransferProducts(rowList) { function addTransferProducts(rowList) {
addTransferProduct(JSON.parse(JSON.stringify(rowList))).then((res) => { addTransferProduct(JSON.parse(JSON.stringify(rowList))).then((res) => {
// 当前行没有id视为首次新增
// if (!row.id) {
// data.isAdding = false; // 允许新增下一行
// }
if (res.data) { if (res.data) {
proxy.$message.success('保存成功!'); proxy.$message.success('保存成功!');
let newIdIndex = 0;
form.purchaseinventoryList.map((row, index) => { form.purchaseinventoryList.map((row, index) => {
// 只有未保存的行才会拿到新 id和提交顺序一致 form.purchaseinventoryList[index].id = res.data[index];
if (!row.isSave && res.data[newIdIndex]) {
form.purchaseinventoryList[index].id = res.data[newIdIndex];
newIdIndex++;
}
form.purchaseinventoryList[index].isSave = true; form.purchaseinventoryList[index].isSave = true;
}); });
if (route.query.supplyBusNo) { if (route.query.supplyBusNo) {
// 编辑 // 编辑
let newIdIdx = 0;
forms.purchaseinventoryList.map((row, index) => { forms.purchaseinventoryList.map((row, index) => {
if (!row.isSave && res.data[newIdIdx]) { forms.purchaseinventoryList[index].id = res.data[index];
forms.purchaseinventoryList[index].id = res.data[newIdIdx];
newIdIdx++;
}
forms.purchaseinventoryList[index].isSave = true; forms.purchaseinventoryList[index].isSave = true;
}); });
} }

View File

@@ -740,59 +740,58 @@
</el-form-item> </el-form-item>
</el-form> </el-form>
<!-- 结果表格卡片 --> <!-- 结果表格 -->
<el-card shadow="never" class="apply-card"> <el-table
<el-table ref="applyTableRef"
ref="applyTableRef" v-loading="applyLoading"
v-loading="applyLoading" :data="applyList"
:data="applyList" row-key="surgeryNo"
row-key="surgeryNo" @row-click="handleApplyRowClick"
@row-click="handleApplyRowClick" :row-class-name="tableRowClassName"
:row-class-name="tableRowClassName" style="width: 100%"
style="width: 100%" max-height="340"
max-height="320" :scroll="{ y: 340 }"
> >
<el-table-column type="selection" width="55" :selectable="handleSelectable" /> <el-table-column type="selection" width="55" :selectable="handleSelectable" />
<el-table-column label="ID" align="center" width="80" fixed> <el-table-column label="ID" align="center" width="80" fixed>
<template #default="{ $index }"> <template #default="{ $index }">
{{ (applyQueryParams.pageNo - 1) * applyQueryParams.pageSize + $index + 1 }} {{ (applyQueryParams.pageNo - 1) * applyQueryParams.pageSize + $index + 1 }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="姓名" align="center" prop="name" width="100" /> <el-table-column label="姓名" align="center" prop="name" width="100" />
<el-table-column label="手术单号" align="center" prop="surgeryNo" width="120" /> <el-table-column label="手术单号" align="center" prop="surgeryNo" width="120" />
<el-table-column label="手术名称" align="center" prop="descJson.surgeryName" min-width="140" show-overflow-tooltip /> <el-table-column label="手术名称" align="center" prop="descJson.surgeryName" min-width="140" show-overflow-tooltip />
<el-table-column label="申请科室" align="center" width="100" prop="applyDeptName" /> <el-table-column label="申请科室" align="center" width="100" prop="applyDeptName" />
<el-table-column label="手术类型" align="center" width="90"> <el-table-column label="手术类型" align="center" width="90">
<template #default="scope"> <template #default="scope">
{{ getSurgeryTypeName(scope.row.surgeryType) }} {{ getSurgeryTypeName(scope.row.surgeryType) }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="手术等级" align="center" width="90"> <el-table-column label="手术等级" align="center" width="90">
<template #default="scope"> <template #default="scope">
{{ getSurgeryLevelName(scope.row.surgeryLevel || scope.row.descJson?.surgeryLevel) }} {{ getSurgeryLevelName(scope.row.surgeryLevel || scope.row.descJson?.surgeryLevel) }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="麻醉方式" align="center" width="90"> <el-table-column label="麻醉方式" align="center" width="90">
<template #default="scope"> <template #default="scope">
{{ getAnesthesiaName(scope.row.anesthesiaTypeEnum) }} {{ getAnesthesiaName(scope.row.anesthesiaTypeEnum) }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="主刀医生" align="center" width="100" prop="mainSurgeonName" /> <el-table-column label="主刀医生" align="center" width="100" prop="mainSurgeonName" />
</el-table> </el-table>
<!-- 分页在卡片内部 -->
<div class="apply-pagination"> <!-- 底部分页区 -->
<pagination <div class="pagination-container apply-pagination">
v-show="applyTotal > 0" <pagination
:total="applyTotal" v-show="applyTotal > 0"
:page="applyQueryParams.pageNo" :total="applyTotal"
:limit="applyQueryParams.pageSize" :page="applyQueryParams.pageNo"
layout="total, sizes, prev, pager, next" :limit="applyQueryParams.pageSize"
@update:page="val => applyQueryParams.pageNo = val" @update:page="val => applyQueryParams.pageNo = val"
@update:limit="val => applyQueryParams.pageSize = val" @update:limit="val => applyQueryParams.pageSize = val"
@pagination="getSurgicalScheduleList" @pagination="getSurgicalScheduleList"
/> />
</div> </div>
</el-card>
<!-- 底部操作区 --> <!-- 底部操作区 -->
<template #footer> <template #footer>
<div class="dialog-footer" style="padding-top: 12px; border-top: 1px solid #ebeef5"> <div class="dialog-footer" style="padding-top: 12px; border-top: 1px solid #ebeef5">
@@ -1068,6 +1067,15 @@ const temporaryPatientInfo = ref({})
const temporaryBillingMedicines = ref([]) const temporaryBillingMedicines = ref([])
const temporaryAdvices = ref([]) const temporaryAdvices = ref([])
// 🔧 新增:监听 temporaryAdvices 的变化,用于调试
watch(temporaryAdvices, (newVal, oldVal) => {
console.log('=== temporaryAdvices 变化 ===')
console.log('=== 新值 ===', newVal)
console.log('=== 新值[1]?.dosage ===', newVal[1]?.dosage)
console.log('=== 旧值 ===', oldVal)
console.log('=== 旧值[1]?.dosage ===', oldVal[1]?.dosage)
}, { deep: true })
const temporaryMedicalLoading = ref(false) // 🔧 新增:临时医嘱加载状态 const temporaryMedicalLoading = ref(false) // 🔧 新增:临时医嘱加载状态
const temporarySigned = ref(false) // 🔧 新增:签名状态,用于保持按钮名称一致性 const temporarySigned = ref(false) // 🔧 新增:签名状态,用于保持按钮名称一致性
@@ -1491,6 +1499,9 @@ async function closeChargeDialog() {
chargeSurgeryInfo.value = {} chargeSurgeryInfo.value = {}
} }
// 🔧 新增:标志位,用于区分是"打开"还是"刷新"
const isRefreshAction = ref(false)
// 处理医嘱按钮点击事件 // 处理医嘱按钮点击事件
function handleMedicalAdvice(row) { function handleMedicalAdvice(row) {
// 如果没有传入行数据,使用选中的行 // 如果没有传入行数据,使用选中的行
@@ -1518,7 +1529,31 @@ function handleMedicalAdvice(row) {
applyId: row.applyId // 手术申请单ID用于过滤关联医嘱 applyId: row.applyId // 手术申请单ID用于过滤关联医嘱
} }
// 🔧 每次打开临时医嘱都重新拉取最新数据,确保计费弹窗签发后数据自动更新 // 🔧 关键修复:如果已有提交的医嘱数据,并且是同一个患者的就诊,则使用保存的数据
// 这样可以保留 requestId避免重复创建医嘱记录
console.log('=== 检查是否使用已保存的医嘱数据 ===')
console.log('=== temporaryAdvices.value.length ===', temporaryAdvices.value.length)
console.log('=== temporaryAdvices.value[0]?.originalMedicine?.encounterId ===', temporaryAdvices.value[0]?.originalMedicine?.encounterId)
console.log('=== row.visitId ===', row.visitId)
console.log('=== isRefreshAction.value ===', isRefreshAction.value)
const isSameEncounter = temporaryAdvices.value.length > 0 &&
temporaryAdvices.value[0]?.originalMedicine?.encounterId === row.visitId &&
!isRefreshAction.value
console.log('=== isSameEncounter ===', isSameEncounter)
if (isSameEncounter) {
console.log('=== 使用已保存的医嘱数据,避免重复创建 ===')
console.log('=== temporaryAdvices.value[0]?.originalMedicine?.requestId ===', temporaryAdvices.value[0]?.originalMedicine?.requestId)
// 直接打开弹窗,使用已保存的数据
showTemporaryMedical.value = true
temporaryMedicalLoading.value = false
isRefreshAction.value = false // 重置标志位
return
}
// 🔧 修复:每次打开临时医嘱时都重新加载数据,避免使用缓存数据导致数据重复
// 先清空旧数据 // 先清空旧数据
temporaryBillingMedicines.value = [] temporaryBillingMedicines.value = []
temporaryAdvices.value = [] temporaryAdvices.value = []
@@ -1530,40 +1565,41 @@ function handleMedicalAdvice(row) {
temporaryMedicalLoading.value = true // 🔧 新增:开始加载 temporaryMedicalLoading.value = true // 🔧 新增:开始加载
// 调用计费接口获取数据 // 调用计费接口获取数据
getPrescriptionList(row.visitId, 6, row.operCode).then((res) => { getPrescriptionList(row.visitId).then((res) => {
console.log('=== 拉取计费数据返回结果 ===', res)
if (res.code === 200 && res.data) { if (res.code === 200 && res.data) {
// 🔧 修复:显示所有药品请求数据,不管有没有计费项目
// 根据用户需求:已引用计费药品(待生成医嘱)和临时医嘱预览(已生成)显示的数据应该相同
// 在提交医嘱之前状态应该是"待签发",提交之后变为"已签发"
// 再次打开医嘱界面的时候能看到这两个状态的药品
const seenIds = new Set(); const seenIds = new Set();
const filteredItems = res.data.filter(item => { const filteredItems = res.data.filter(item => {
// 匹配 encounterId // 匹配 encounterId
if (item.encounterId !== row.visitId) return false; if (item.encounterId !== row.visitId) return false;
// 只保留药品(1)和耗材(2),屏蔽诊疗(3)和手术(6) // 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3)
const at = Number(item.adviceType ?? item.advice_type); if (item.adviceType !== 1) return false;
if (at !== 1 && at !== 2) return false;
// 过滤掉名称为空的项目 // 过滤掉名称为空的项目
const medicineName = item.adviceName || item.advice_name; const medicineName = item.adviceName || item.advice_name;
if (!medicineName || medicineName.trim() === '') return false; if (!medicineName || medicineName.trim() === '') return false;
// 排除名称中包含手术/检查/诊疗关键词的非药品项目 // 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目(已有 requestId 的不应出现在"待生成"列表)
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影']; if (item.requestId) return false;
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
// 根据药品请求ID去重避免重复显示 // 根据药品请求ID去重避免重复显示
const itemId = item.requestId || item.id; const itemId = item.requestId || item.id;
if (itemId && seenIds.has(itemId)) return false; if (itemId && seenIds.has(itemId)) return false;
if (itemId) seenIds.add(itemId); if (itemId) seenIds.add(itemId);
return true; return true;
}) })
// 按 statusEnum 区分1=草稿(待生成)2=已签发(已生成)
const draftItems = filteredItems.filter(item => item.statusEnum === 1)
const activeItems = filteredItems.filter(item => item.statusEnum === 2)
// 🔧 修复限制返回数量最多显示前100条避免数据过多导致页面卡死 // 🔧 修复限制返回数量最多显示前100条避免数据过多导致页面卡死
const maxItems = 100 const maxItems = 100
if (draftItems.length > maxItems) { if (filteredItems.length > maxItems) {
ElMessage.warning(`待签发医嘱数量过多(${draftItems.length}条),仅显示前${maxItems}`) ElMessage.warning(`待签发医嘱数量过多(${filteredItems.length}条),仅显示前${maxItems}`)
draftItems.length = maxItems filteredItems.length = maxItems
} }
// === 待生成列表statusEnum=1 草稿状态的项目 === // 将过滤后的数据转换为临时医嘱需要的格式 - 兼容驼峰和下划线命名
temporaryBillingMedicines.value = draftItems.map(item => { // 对于从 adm_charge_item计费项目表查询来的项目特殊处理
temporaryBillingMedicines.value = filteredItems.map(item => {
try { try {
// 从 contentJson 或 content_json 中解析详细数据 - 兼容下划线和驼峰命名 // 从 contentJson 或 content_json 中解析详细数据 - 兼容下划线和驼峰命名
const jsonContent = item.contentJson || item.content_json; const jsonContent = item.contentJson || item.content_json;
@@ -1610,65 +1646,69 @@ function handleMedicalAdvice(row) {
}; };
} }
}); });
} else {
// 如果没有数据或接口调用失败,初始化空列表
temporaryBillingMedicines.value = []
}
// 将计费药品转换为临时医嘱数据
temporaryAdvices.value = temporaryBillingMedicines.value.map((medicine, index) => {
// 解析规格中的数值和单位
const specMatch = medicine.specification ? medicine.specification.match(/(\d+)(\D+)/) : null
const specValue = specMatch ? parseInt(specMatch[1]) : 1
const specUnit = specMatch ? specMatch[2] : 'ml'
// === 已生成列表statusEnum=2 已签发状态的项目,直接转为医嘱格式 === // 计算剂量 = 规格数值 × 数量
temporaryAdvices.value = activeItems.map((item, index) => { const dosage = specValue * (medicine.quantity || 1)
try {
const jsonContent = item.contentJson || item.content_json;
const contentData = jsonContent ? JSON.parse(jsonContent) : {};
const medicineName = contentData.adviceName || contentData.advice_name || item.adviceName || item.advice_name || '';
const spec = contentData.volume || contentData.specification || item.volume || item.specification || '';
const specMatch = spec.match(/(\d+)(\D+)/)
const specValue = specMatch ? parseInt(specMatch[1]) : 1
const specUnit = specMatch ? specMatch[2] : 'ml'
const dosage = specValue * (contentData.quantity || item.quantity || 1)
let usageCode = contentData.methodCode || 'iv' // 🔧 修复:优先从 contentJson 中读取已有的用法,如果没有则根据药品名称判断
let usageLabel = getUsageLabel(usageCode) let usageCode = 'iv' // 默认静脉注射编码
if (usageCode === 'iv') { let usageLabel = '静脉注射' // 默认显示名称
if (medicineName.includes('注射液')) { usageCode = 'iv'; usageLabel = '静脉注射' }
} else if (usageCode === 'po') {
if (medicineName.includes('片') || medicineName.includes('胶囊')) { usageCode = 'po'; usageLabel = '口服' }
}
return { // 尝试从 contentJson 中读取用法
id: index + 1, try {
adviceName: medicineName, const jsonContent = medicine.contentJson || medicine.content_json;
dosage, if (jsonContent) {
unit: specUnit, const contentData = JSON.parse(jsonContent);
usage: usageCode, if (contentData.methodCode) {
usageLabel, usageCode = contentData.methodCode;
frequency: '临时', usageLabel = getUsageLabel(contentData.methodCode);
executeTime: new Date().toLocaleString('zh-CN'),
originalMedicine: {
...item,
medicineName: medicineName,
specification: spec,
quantity: contentData.quantity || item.quantity || 1,
encounterId: row.visitId
}
}
} catch (e) {
return {
id: index + 1,
adviceName: item.adviceName || item.advice_name || '',
dosage: 1, unit: 'ml', usage: 'iv', usageLabel: '静脉注射',
frequency: '临时',
executeTime: new Date().toLocaleString('zh-CN'),
originalMedicine: {
...item,
medicineName: item.adviceName || item.advice_name || '',
specification: item.volume || item.specification || '',
quantity: item.quantity || 1,
encounterId: row.visitId
}
} }
} }
}) } catch (e) {
} else { // 解析失败,继续使用默认值
temporaryBillingMedicines.value = [] }
temporaryAdvices.value = []
} // 如果没有从 contentJson 中读取到用法,根据药品名称判断
if (!usageCode || usageCode === 'iv') {
if (medicine.medicineName && medicine.medicineName.includes('注射液')) {
usageCode = 'iv'
usageLabel = '静脉注射'
} else if (medicine.medicineName && medicine.medicineName.includes('片')) {
usageCode = 'po'
usageLabel = '口服'
} else if (medicine.medicineName && medicine.medicineName.includes('胶囊')) {
usageCode = 'po'
usageLabel = '口服'
}
}
return {
id: index + 1,
adviceName: medicine.medicineName || '',
dosage: dosage,
unit: specUnit,
usage: usageCode, // 🔧 修复:保存的是编码
usageLabel: usageLabel, // 🔧 新增:保存显示名称
frequency: '临时',
executeTime: new Date().toLocaleString('zh-CN'),
// 🔧 关键修复:确保 originalMedicine 中包含 encounterId以便后续判断是否为同一患者
originalMedicine: {
...medicine,
encounterId: row.visitId // 添加 encounterId 字段
}
}
})
// 打开临时医嘱弹窗 // 打开临时医嘱弹窗
showTemporaryMedical.value = true showTemporaryMedical.value = true
@@ -1694,6 +1734,11 @@ function closeTemporaryMedical() {
// 处理临时医嘱提交 // 处理临时医嘱提交
// 🔧 修复:提交成功后,更新 temporaryAdvices 中的 requestId以便下次提交时执行更新操作 // 🔧 修复:提交成功后,更新 temporaryAdvices 中的 requestId以便下次提交时执行更新操作
function handleTemporaryMedicalSubmit(data) { function handleTemporaryMedicalSubmit(data) {
console.log('=== handleTemporaryMedicalSubmit 被调用 ===')
console.log('=== data ===', data)
console.log('=== data.temporaryAdvices ===', data.temporaryAdvices)
console.log('=== data.temporaryAdvices[1]?.dosage ===', data.temporaryAdvices[1]?.dosage)
// 🔧 修复:使用用户修改后的数据,而不是重新加载数据 // 🔧 修复:使用用户修改后的数据,而不是重新加载数据
// 这样可以确保用户修改的内容(如剂量)在保存后仍然正确显示 // 这样可以确保用户修改的内容(如剂量)在保存后仍然正确显示
if (data.temporaryAdvices && data.temporaryAdvices.length > 0) { if (data.temporaryAdvices && data.temporaryAdvices.length > 0) {
@@ -1748,7 +1793,9 @@ function handleTemporaryMedicalSubmit(data) {
// 如果没有任何匹配键,清空待生成列表(所有项目都已提交) // 如果没有任何匹配键,清空待生成列表(所有项目都已提交)
temporaryBillingMedicines.value = [] temporaryBillingMedicines.value = []
} }
console.log('=== 使用用户修改后的临时医嘱数据 ===', temporaryAdvices.value)
console.log('=== temporaryAdvices.value[1]?.dosage ===', temporaryAdvices.value[1]?.dosage)
} else { } else {
// 如果没有传递数据,则清空 // 如果没有传递数据,则清空
temporaryAdvices.value = [] temporaryAdvices.value = []
@@ -1796,21 +1843,6 @@ function handleTemporaryMedicalRefresh() {
function handleQuoteBilling() { function handleQuoteBilling() {
// 重新拉取计费药品数据 // 重新拉取计费药品数据
if (temporaryPatientInfo.value.visitId) { if (temporaryPatientInfo.value.visitId) {
// 🔧 修复 Bug #445: 在清空之前提取已提交项目的复合匹配键
// 原因:后续的 ID 匹配过滤依赖 temporaryAdvices但 temporaryAdvices 会被先清空
// 新医嘱没有 requestId/chargeItemId需用名称+规格+数量的复合键匹配
const submittedKeys = new Set(
(temporaryAdvices.value || [])
.map(a => {
const om = a.originalMedicine || {}
const name = om.medicineName || om.adviceName || a.adviceName || ''
const spec = om.specification || om.volume || ''
const qty = om.quantity ?? 0
return `${name}|||${spec}|||${qty}`
})
.filter(k => k !== '|||0')
)
temporaryMedicalLoading.value = true // 🔧 新增:开始加载 temporaryMedicalLoading.value = true // 🔧 新增:开始加载
getPrescriptionList(temporaryPatientInfo.value.visitId, 6, temporaryPatientInfo.value.operCode).then((res) => { getPrescriptionList(temporaryPatientInfo.value.visitId, 6, temporaryPatientInfo.value.operCode).then((res) => {
if (res.code === 200 && res.data) { if (res.code === 200 && res.data) {
@@ -1818,72 +1850,18 @@ function handleQuoteBilling() {
temporaryBillingMedicines.value = [] temporaryBillingMedicines.value = []
temporaryAdvices.value = [] temporaryAdvices.value = []
// 🔧 修复 Bug #445: 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3/6) // 🔧 修复 Bug #445: 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3)
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中) // 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
// 先提取已签发项目(statusEnum=2)填充已生成列表
const activeItems = res.data.filter(item => {
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
const at = Number(item.adviceType ?? item.advice_type);
if (at !== 1 && at !== 2) return false;
if (item.statusEnum !== 2) return false;
const medicineName = item.adviceName || item.advice_name;
if (!medicineName || medicineName.trim() === '') return false;
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
return true;
})
temporaryAdvices.value = activeItems.map((item, index) => {
try {
const jsonContent = item.contentJson || item.content_json;
const contentData = jsonContent ? JSON.parse(jsonContent) : {};
const medicineName = contentData.adviceName || contentData.advice_name || item.adviceName || item.advice_name || '';
const spec = contentData.volume || contentData.specification || item.volume || item.specification || '';
const specMatch = spec.match(/(\d+)(\D+)/)
const specValue = specMatch ? parseInt(specMatch[1]) : 1
const specUnit = specMatch ? specMatch[2] : 'ml'
const dosage = specValue * (contentData.quantity || item.quantity || 1)
let usageCode = contentData.methodCode || 'iv'
let usageLabel = getUsageLabel(usageCode)
if (usageCode === 'iv' && medicineName.includes('注射液')) { usageLabel = '静脉注射' }
else if (usageCode === 'po' && (medicineName.includes('片') || medicineName.includes('胶囊'))) { usageLabel = '口服' }
return {
id: index + 1, adviceName: medicineName, dosage, unit: specUnit,
usage: usageCode, usageLabel, frequency: '临时',
executeTime: new Date().toLocaleString('zh-CN'),
originalMedicine: {
...item,
medicineName: medicineName,
specification: spec,
quantity: contentData.quantity || item.quantity || 1,
encounterId: temporaryPatientInfo.value.visitId
}
}
} catch (e) {
return {
id: index + 1, adviceName: item.adviceName || item.advice_name || '',
dosage: 1, unit: 'ml', usage: 'iv', usageLabel: '静脉注射',
frequency: '临时', executeTime: new Date().toLocaleString('zh-CN'),
originalMedicine: {
...item,
medicineName: item.adviceName || item.advice_name || '',
specification: item.volume || item.specification || '',
quantity: item.quantity || 1,
encounterId: temporaryPatientInfo.value.visitId
}
}
}
})
// 再提取草稿项目(statusEnum=1)填充待生成列表
const filteredItems = res.data.filter(item => { const filteredItems = res.data.filter(item => {
// 匹配 encounterId
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false; if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
const at = Number(item.adviceType ?? item.advice_type); // 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3)
if (at !== 1 && at !== 2) return false; if (item.adviceType !== 1) return false;
if (item.statusEnum !== 1) return false; // 过滤掉名称为空的项目
const medicineName = item.adviceName || item.advice_name; const medicineName = item.adviceName || item.advice_name;
if (!medicineName || medicineName.trim() === '') return false; if (!medicineName || medicineName.trim() === '') return false;
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影']; // 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目(已有 requestId
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false; if (item.requestId) return false;
return true; return true;
}) })
// 🔧 修复限制返回数量最多显示前100条避免数据过多导致页面卡死 // 🔧 修复限制返回数量最多显示前100条避免数据过多导致页面卡死
@@ -1896,6 +1874,7 @@ function handleQuoteBilling() {
// 将过滤后的数据转换为临时医嘱需要的格式 // 将过滤后的数据转换为临时医嘱需要的格式
temporaryBillingMedicines.value = filteredItems.map(item => { temporaryBillingMedicines.value = filteredItems.map(item => {
try { try {
// 从 contentJson 或 content_json 中解析详细数据 - 兼容下划线和驼峰命名
const jsonContent = item.contentJson || item.content_json; const jsonContent = item.contentJson || item.content_json;
const contentData = jsonContent ? JSON.parse(jsonContent) : {}; const contentData = jsonContent ? JSON.parse(jsonContent) : {};
return { return {
@@ -1914,9 +1893,10 @@ function handleQuoteBilling() {
definitionDetailId: contentData.definitionDetailId || item.definitionDetailId definitionDetailId: contentData.definitionDetailId || item.definitionDetailId
} }
} catch (e) { } catch (e) {
// 如果解析失败,使用顶层数据 - 兼容 snake_case 和 camelCase
return { return {
medicineName: item.adviceName || item.advice_name || '', medicineName: item.adviceName || item.advice_name || '',
specification: item.specification || item.volume || '', specification: item.specification || item.specification || item.volume || '',
quantity: item.quantity || item.quantity_value || 0, quantity: item.quantity || item.quantity_value || 0,
batchNumber: item.lotNumber || item.lot_number || '', batchNumber: item.lotNumber || item.lot_number || '',
unitPrice: item.unitPrice || item.unit_price || 0, unitPrice: item.unitPrice || item.unit_price || 0,
@@ -1932,6 +1912,89 @@ function handleQuoteBilling() {
} }
}) })
// 将计费药品转换为临时医嘱数据
temporaryAdvices.value = temporaryBillingMedicines.value.map((medicine, index) => {
// 解析规格中的数值和单位
const specMatch = medicine.specification ? medicine.specification.match(/(\d+)(\D+)/) : null
const specValue = specMatch ? parseInt(specMatch[1]) : 1
const specUnit = specMatch ? specMatch[2] : 'ml'
// 计算剂量 = 规格数值 × 数量
const dosage = specValue * (medicine.quantity || 1)
// 🔧 修复:优先从 contentJson 中读取已有的用法,如果没有则根据药品名称判断
let usageCode = 'iv' // 默认静脉注射编码
let usageLabel = '静脉注射' // 默认显示名称
// 尝试从 contentJson 中读取用法
try {
const jsonContent = medicine.contentJson || medicine.content_json;
if (jsonContent) {
const contentData = JSON.parse(jsonContent);
if (contentData.methodCode) {
usageCode = contentData.methodCode;
usageLabel = getUsageLabel(contentData.methodCode);
}
}
} catch (e) {
// 解析失败,继续使用默认值
}
// 如果没有从 contentJson 中读取到用法,根据药品名称判断
if (!usageCode || usageCode === 'iv') {
if (medicine.medicineName && medicine.medicineName.includes('注射液')) {
usageCode = 'iv'
usageLabel = '静脉注射'
} else if (medicine.medicineName && medicine.medicineName.includes('片')) {
usageCode = 'po'
usageLabel = '口服'
} else if (medicine.medicineName && medicine.medicineName.includes('胶囊')) {
usageCode = 'po'
usageLabel = '口服'
}
}
return {
id: index + 1,
adviceName: medicine.medicineName || '',
dosage: dosage,
unit: specUnit,
usage: usageCode, // 🔧 修复:保存的是编码
usageLabel: usageLabel, // 🔧 新增:保存显示名称
frequency: '临时',
executeTime: new Date().toLocaleString('zh-CN'),
// 🔧 关键修复:确保 originalMedicine 中包含 encounterId以便后续判断是否为同一患者
originalMedicine: {
...medicine,
encounterId: temporaryPatientInfo.value.visitId // 添加 encounterId 字段
}
}
})
// 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目,避免"引用计费"后已提交项目重新出现在"待生成"列表
// 原因:后端返回的计费数据中,已生成医嘱的项目可能没有 requestId 字段
// 方案:用 chargeItemId/requestId/id 与已有的 temporaryAdvices 做匹配,排除已生成项目
if (temporaryAdvices.value.length > 0) {
const existingAdviceIds = new Set()
temporaryAdvices.value.forEach(a => {
const om = a.originalMedicine || {}
if (om.requestId) existingAdviceIds.add(String(om.requestId))
if (om.chargeItemId) existingAdviceIds.add(String(om.chargeItemId))
if (om.id) existingAdviceIds.add(String(om.id))
})
if (existingAdviceIds.size > 0) {
temporaryBillingMedicines.value = temporaryBillingMedicines.value.filter(m => {
const mRequestId = m.requestId != null ? String(m.requestId) : null
const mChargeItemId = m.chargeItemId != null ? String(m.chargeItemId) : null
const mId = m.id != null ? String(m.id) : null
if (mRequestId && existingAdviceIds.has(mRequestId)) return false
if (mChargeItemId && existingAdviceIds.has(mChargeItemId)) return false
if (mId && existingAdviceIds.has(mId)) return false
return true
})
}
}
temporaryMedicalLoading.value = false // 🔧 新增:加载完成 temporaryMedicalLoading.value = false // 🔧 新增:加载完成
ElMessage.success('已成功引用最新计费药品信息!') ElMessage.success('已成功引用最新计费药品信息!')
} else { } else {
@@ -2340,35 +2403,19 @@ function getRowClassName({ row, rowIndex }) {
margin-left: 10px; margin-left: 10px;
} }
/* 手术申请查询弹窗 — flex 布局确保分页不溢出 */ /* 手术申请查询弹窗 — 分页与footer间距 */
.surgery-apply-dialog :deep(.el-dialog__body) { .surgery-apply-dialog :deep(.el-dialog__body) {
display: flex;
flex-direction: column;
padding-bottom: 16px; padding-bottom: 16px;
overflow: hidden;
} }
.surgery-apply-dialog :deep(.el-dialog__footer) { .surgery-apply-dialog :deep(.el-dialog__footer) {
padding-top: 0; padding-top: 8px;
}
.surgery-apply-dialog :deep(.apply-card) {
flex: 1;
overflow: hidden;
min-height: 0;
}
.surgery-apply-dialog :deep(.apply-card .el-card__body) {
overflow-y: auto;
} }
.surgery-apply-dialog :deep(.apply-pagination) { .surgery-apply-dialog :deep(.apply-pagination) {
display: flex; padding-top: 12px;
justify-content: flex-end; padding-bottom: 16px;
padding-top: 8px;
border-top: 1px solid #ebeef5;
}
.surgery-apply-dialog :deep(.apply-pagination .pagination-container) {
margin-top: 0;
} }
.surgery-apply-dialog :deep(.apply-pagination .el-pagination) { .surgery-apply-dialog :deep(.apply-pagination .el-pagination) {
position: static; margin-right: 80px;
} }
/* 选中行样式 */ /* 选中行样式 */
@@ -2384,33 +2431,17 @@ function getRowClassName({ row, rowIndex }) {
<style> <style>
/* 手术申请查询弹窗 — 非 scoped 确保穿透 teleport */ /* 手术申请查询弹窗 — 非 scoped 确保穿透 teleport */
.surgery-apply-dialog .el-dialog__body {
display: flex !important;
flex-direction: column !important;
padding-bottom: 16px !important;
overflow: hidden !important;
}
.surgery-apply-dialog .el-dialog__footer {
padding-top: 0 !important;
}
.surgery-apply-dialog .apply-card {
flex: 1 !important;
overflow: hidden !important;
min-height: 0 !important;
}
.surgery-apply-dialog .apply-card .el-card__body {
overflow-y: auto !important;
}
.surgery-apply-dialog .apply-pagination { .surgery-apply-dialog .apply-pagination {
display: flex !important; padding-top: 12px !important;
justify-content: flex-end !important; padding-bottom: 16px !important;
padding-top: 8px !important;
border-top: 1px solid #ebeef5 !important;
}
.surgery-apply-dialog .apply-pagination .pagination-container {
margin-top: 0 !important;
} }
.surgery-apply-dialog .apply-pagination .el-pagination { .surgery-apply-dialog .apply-pagination .el-pagination {
position: static !important; margin-right: 80px !important;
}
.surgery-apply-dialog .el-dialog__body {
padding-bottom: 16px !important;
}
.surgery-apply-dialog .el-dialog__footer {
padding-top: 8px !important;
} }
</style> </style>

View File

@@ -48,12 +48,9 @@
<div class="medicine-section"> <div class="medicine-section">
<div class="section-title"> <div class="section-title">
已引用计费药品待生成医嘱 已引用计费药品待生成医嘱
<span v-if="(billingMedicines || []).length >= PAGE_SIZE" style="margin-left:auto;font-size:14px;color:#4a8bc9;cursor:pointer;white-space:nowrap;" @click="billingExpanded = !billingExpanded">
{{ billingExpanded ? '收起' : `展开全部(${(billingMedicines || []).length})` }}
</span>
</div> </div>
<el-table <el-table
:data="displayBillingMedicines" :data="billingMedicines"
stripe stripe
border border
style="width: 100%;" style="width: 100%;"
@@ -101,12 +98,9 @@
<div class="advice-section"> <div class="advice-section">
<div class="section-title"> <div class="section-title">
临时医嘱预览已生成 临时医嘱预览已生成
<span v-if="(displayAdvices || []).length >= PAGE_SIZE" style="margin-left:auto;font-size:14px;color:#4a8bc9;cursor:pointer;white-space:nowrap;" @click="advicesExpanded = !advicesExpanded">
{{ advicesExpanded ? '收起' : `展开全部(${(displayAdvices || []).length})` }}
</span>
</div> </div>
<el-table <el-table
:data="displayAdvicesList" :data="displayAdvices"
stripe stripe
border border
style="width: 100%;" style="width: 100%;"
@@ -156,7 +150,7 @@
:disabled="allItemsSubmitted" :disabled="allItemsSubmitted"
@click="handleSignAndSubmit" @click="handleSignAndSubmit"
> >
{{ allItemsSubmitted ? '已签发' : '一键签名并生成医嘱' }} {{ allItemsSubmitted ? '已签发' : (isSigned ? '提交医嘱' : '一键签名并生成医嘱') }}
</el-button> </el-button>
</div> </div>
</div> </div>
@@ -323,19 +317,6 @@ const allItemsSubmitted = computed(() => {
return meds.length > 0 && meds.every(m => m.requestId) return meds.length > 0 && meds.every(m => m.requestId)
}) })
// 展开/收起控制
const PAGE_SIZE = 3
const billingExpanded = ref(false)
const advicesExpanded = ref(false)
const displayBillingMedicines = computed(() => {
const all = props.billingMedicines || []
return billingExpanded.value ? all : all.slice(0, PAGE_SIZE)
})
const displayAdvicesList = computed(() => {
const all = displayAdvices.value || []
return advicesExpanded.value ? all : all.slice(0, PAGE_SIZE)
})
// 响应式数据 - isSigned 从父组件传入的 prop 初始化 // 响应式数据 - isSigned 从父组件传入的 prop 初始化
const isSigned = ref(props.isSignedProp) const isSigned = ref(props.isSignedProp)
@@ -1064,21 +1045,6 @@ const editFormUsageLabel = computed(() => {
padding-bottom: 12px; padding-bottom: 12px;
margin-bottom: 16px; margin-bottom: 16px;
border-bottom: 2px solid #e4e7ed; border-bottom: 2px solid #e4e7ed;
display: flex;
align-items: center;
gap: 12px;
}
.expand-btn {
font-size: 0.85rem;
color: #4a8bc9;
cursor: pointer;
font-weight: 400;
margin-left: auto;
}
.expand-btn:hover {
color: #2a6ba9;
text-decoration: underline;
} }
.medicine-summary { .medicine-summary {

View File

@@ -88,16 +88,16 @@
</el-table> </el-table>
</div> </div>
<div class="candidate-actions"> <div class="candidate-actions">
<el-button <el-button
type="primary" type="primary"
:disabled="selectedCandidates.length === 0 || isQueryingHistory" :disabled="selectedCandidates.length === 0"
@click="handleAddToQueue" @click="handleAddToQueue"
> >
加入队列 >> 加入队列 >>
</el-button> </el-button>
<el-button <el-button
type="primary" type="primary"
:disabled="filteredCandidatePoolList.length === 0 || isQueryingHistory" :disabled="filteredCandidatePoolList.length === 0"
@click="handleAddAllToQueue" @click="handleAddAllToQueue"
> >
一键加入队列 一键加入队列
@@ -109,19 +109,6 @@
<div class="right-panel"> <div class="right-panel">
<div class="panel-header"> <div class="panel-header">
<span class="panel-title"> 智能队列 (全科)</span> <span class="panel-title"> 智能队列 (全科)</span>
<div class="history-query">
<el-date-picker
v-model="queryDate"
type="date"
placeholder="选择日期"
format="YYYY-MM-DD"
value-format="YYYY-MM-DD"
size="small"
style="width: 150px"
/>
<el-button type="primary" size="small" @click="handleHistoryQuery">查询</el-button>
<el-button size="small" @click="handleTodayQuery">今天</el-button>
</div>
</div> </div>
<div class="table-container"> <div class="table-container">
<el-table <el-table
@@ -186,25 +173,25 @@
</div> </div>
<div class="display-options"> <div class="display-options">
<div class="queue-actions-left"> <div class="queue-actions-left">
<el-button <el-button
type="danger" type="danger"
:disabled="!selectedQueueRow || isQueryingHistory" :disabled="!selectedQueueRow"
size="small" size="small"
@click="handleRemoveFromQueue" @click="handleRemoveFromQueue"
> >
&lt;&lt; 移出队列 &lt;&lt; 移出队列
</el-button> </el-button>
<el-button <el-button
type="info" type="info"
:disabled="!selectedQueueRow || !canMoveUp || isQueryingHistory" :disabled="!selectedQueueRow || !canMoveUp"
size="small" size="small"
@click="handleMoveUp" @click="handleMoveUp"
> >
</el-button> </el-button>
<el-button <el-button
type="info" type="info"
:disabled="!selectedQueueRow || !canMoveDown || isQueryingHistory" :disabled="!selectedQueueRow || !canMoveDown"
size="small" size="small"
@click="handleMoveDown" @click="handleMoveDown"
> >
@@ -272,35 +259,30 @@
<div class="control-buttons"> <div class="control-buttons">
<el-button <el-button
type="primary" type="primary"
:disabled="isQueryingHistory"
@click="handleSelectCall" @click="handleSelectCall"
> >
选呼 选呼
</el-button> </el-button>
<el-button <el-button
type="success" type="success"
:disabled="isQueryingHistory"
@click="handleNextPatient" @click="handleNextPatient"
> >
下一患者 下一患者
</el-button> </el-button>
<el-button <el-button
type="warning" type="warning"
:disabled="isQueryingHistory"
@click="handleSkip" @click="handleSkip"
> >
跳过 跳过
</el-button> </el-button>
<el-button <el-button
type="primary" type="primary"
:disabled="isQueryingHistory"
@click="handleComplete" @click="handleComplete"
> >
完成 完成
</el-button> </el-button>
<el-button <el-button
type="info" type="info"
:disabled="isQueryingHistory"
@click="handleRequeue" @click="handleRequeue"
> >
过号重排 过号重排
@@ -700,14 +682,6 @@ const showOnlyWaiting = ref(false)
// Bug #411诊室过滤替代原来的科室下拉框selectedDept/departmentList 已移除) // Bug #411诊室过滤替代原来的科室下拉框selectedDept/departmentList 已移除)
const selectedRoom = ref('all') const selectedRoom = ref('all')
// 历史队列查询日期 (默认当天)
const getTodayStr = () => {
const now = new Date()
return `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, '0')}-${String(now.getDate()).padStart(2, '0')}`
}
const queryDate = ref(getTodayStr())
const isQueryingHistory = computed(() => queryDate.value !== getTodayStr())
// 修复【#397】动态获取当前科室名称 // 修复【#397】动态获取当前科室名称
const currentDeptName = computed(() => { const currentDeptName = computed(() => {
return userStore.deptName || userStore.orgName || '心内科' return userStore.deptName || userStore.orgName || '心内科'
@@ -927,12 +901,14 @@ const mapFrontendStatusToBackend = (status) => {
} }
// 从数据库加载队列 // 从数据库加载队列
const loadQueueFromDb = async (dateStr) => { const loadQueueFromDb = async () => {
try { try {
// Bug #411不再按科室选筛加载后端默认按当前登录人科室查询 // Bug #411不再按科室选筛加载后端默认按当前登录人科室查询
const organizationId = undefined const organizationId = undefined
const queryDateStr = dateStr || queryDate.value // 只查询今天的患者
const res = await getTriageQueueList({ organizationId, date: queryDateStr }).catch((err) => { const today = new Date()
const todayStr = `${today.getFullYear()}-${String(today.getMonth() + 1).padStart(2, '0')}-${String(today.getDate()).padStart(2, '0')}`
const res = await getTriageQueueList({ organizationId, date: todayStr }).catch((err) => {
console.error('【心内科】loadQueueFromDb 请求异常:', err) console.error('【心内科】loadQueueFromDb 请求异常:', err)
return { code: 500, msg: err?.message || '请求失败', data: null } return { code: 500, msg: err?.message || '请求失败', data: null }
}) })
@@ -955,6 +931,10 @@ const loadQueueFromDb = async (dateStr) => {
originalQueueList.value = list originalQueueList.value = list
.map((it) => { .map((it) => {
const frontendStatus = mapBackendStatusToFrontend(it.status) const frontendStatus = mapBackendStatusToFrontend(it.status)
// 调试日志:检查状态映射
if (list.length <= 5) {
console.log('【心内科】状态映射:后端状态=', it.status, '-> 前端状态=', frontendStatus, '患者=', it.patientName)
}
// 计算等待时间基于创建时间createTime // 计算等待时间基于创建时间createTime
let waitingTime = '00:00' let waitingTime = '00:00'
if (it.createTime) { if (it.createTime) {
@@ -992,7 +972,15 @@ const loadQueueFromDb = async (dateStr) => {
organizationId: it.organizationId organizationId: it.organizationId
} }
}) })
.filter((item) => {
// 过滤掉"已完成"状态的患者,不显示在队列中
if (item.status === '已完成') {
console.log('【心内科】过滤掉已完成状态的患者:', item.patientName)
return false
}
return true
})
// 调试日志:检查查找结果 // 调试日志:检查查找结果
const callingCount = originalQueueList.value.filter(i => i.status === '叫号中').length const callingCount = originalQueueList.value.filter(i => i.status === '叫号中').length
const waitingCount = originalQueueList.value.filter(i => i.status === '等待').length const waitingCount = originalQueueList.value.filter(i => i.status === '等待').length
@@ -1208,6 +1196,9 @@ const formatSecondsToMmSs = (totalSeconds) => {
const filteredQueueList = computed(() => { const filteredQueueList = computed(() => {
let filtered = originalQueueList.value let filtered = originalQueueList.value
// 先过滤掉"已完成"状态的患者(无论什么情况都不显示)
filtered = filtered.filter(item => item.status !== '已完成')
// 再按诊室过滤 // 再按诊室过滤
if (selectedRoom.value !== 'all') { if (selectedRoom.value !== 'all') {
filtered = filtered.filter(item => item.room === selectedRoom.value) filtered = filtered.filter(item => item.room === selectedRoom.value)
@@ -1636,26 +1627,6 @@ const handleRefresh = async () => {
ElMessage.success('已刷新(已从数据库恢复队列)') ElMessage.success('已刷新(已从数据库恢复队列)')
} }
// 历史队列查询
const handleHistoryQuery = async () => {
if (!queryDate.value) {
ElMessage.warning('请选择查询日期')
return
}
console.log('【心内科】历史队列查询:', queryDate.value)
await loadQueueFromDb(queryDate.value)
if (isQueryingHistory.value) {
ElMessage.success(`已加载 ${queryDate.value} 的队列数据`)
}
}
// 回到今天
const handleTodayQuery = async () => {
queryDate.value = getTodayStr()
await loadQueueFromDb(getTodayStr())
ElMessage.success('已切换到今天队列')
}
// 退出 // 退出
const handleExit = () => { const handleExit = () => {
ElMessage.info('退出功能待实现') ElMessage.info('退出功能待实现')
@@ -2194,21 +2165,12 @@ onUnmounted(() => {
padding: 15px 20px; padding: 15px 20px;
border-bottom: 2px solid #409eff; border-bottom: 2px solid #409eff;
background-color: #f8f9fa; background-color: #f8f9fa;
display: flex;
justify-content: space-between;
align-items: center;
.panel-title { .panel-title {
font-size: 16px; font-size: 16px;
font-weight: bold; font-weight: bold;
color: #333; color: #333;
} }
.history-query {
display: flex;
gap: 8px;
align-items: center;
}
} }
.table-container { .table-container {