Compare commits
47 Commits
zhugeliang
...
zhaoyun
| Author | SHA1 | Date | |
|---|---|---|---|
| d554048158 | |||
| 183e5195a5 | |||
| e747f5b8e9 | |||
| bbf5029194 | |||
| fd6e97285f | |||
|
|
f84940fa5f | ||
| e3db810972 | |||
| 85e95420b7 | |||
| 206a0f4083 | |||
| 5e9aaebc7a | |||
| f925731f6f | |||
|
|
156a3f0f24 | ||
| 85d254990f | |||
| c75b8038ec | |||
| af17d1f460 | |||
| efc1c100aa | |||
|
|
d9c975a950 | ||
| 0874012dae | |||
|
|
cbad13bddc | ||
| a91ee66368 | |||
| 871e2de574 | |||
| 3d279548f0 | |||
| c4a5932a5d | |||
| e9953cd037 | |||
| 798c5e19e2 | |||
| fa18e94cd9 | |||
| 69bb887d19 | |||
| b89f41048b | |||
| e13e328627 | |||
| 9cac8c3e41 | |||
| d7ca64e023 | |||
|
|
0e974129eb | ||
| 4972ca64da | |||
| 1b0028e62f | |||
| c8876dd890 | |||
| 707cfc63df | |||
| 2cddc00d22 | |||
| ea5da8d2bc | |||
| 09353c11ca | |||
| c49ec61e18 | |||
| 8081f3ac7f | |||
| 5bdedd84e0 | |||
| 69ac346ff3 | |||
| 549d2529bc | |||
| 9d3f44bafc | |||
| 0228cba94e | |||
| 3a97f5ce02 |
29
.agentforge/analysis/556.md
Normal file
29
.agentforge/analysis/556.md
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
# 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
|
||||||
|
|
||||||
|
修复结果:✅ 成功,6行改动(含linter自动补充的就诊卡号字段回退逻辑)
|
||||||
27
.agentforge/analysis/bug545_analysis.md
Normal file
27
.agentforge/analysis/bug545_analysis.md
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# 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行已有定义)
|
||||||
79
BUG540_ANALYSIS.md
Normal file
79
BUG540_ANALYSIS.md
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
# Bug #540 分析报告
|
||||||
|
|
||||||
|
## Bug 描述
|
||||||
|
【住院医生站-检查申请】详情页弹窗中"申请单描述"区域缺少临床必要信息显示
|
||||||
|
|
||||||
|
## 数据流分析
|
||||||
|
|
||||||
|
### 前端组件
|
||||||
|
- 入口: `src/views/inpatientDoctor/home/index.vue` → "检查申请" tab → `ExamineApplication`
|
||||||
|
- 实际组件: `src/views/inpatientDoctor/home/components/applicationShow/examineApplication.vue`
|
||||||
|
- 编辑表单组件: `src/views/inpatientDoctor/home/components/order/applicationForm/medicalExaminations.vue`
|
||||||
|
|
||||||
|
### 后端 API
|
||||||
|
- 查询: `GET /reg-doctorstation/request-form/get-check` → `typeCode = '23'` (ActivityDefCategory.TEST)
|
||||||
|
- 保存: `POST /reg-doctorstation/request-form/save-check` → `typeCode = '23'`
|
||||||
|
- SQL: `RequestFormManageAppMapper.xml` 的 `getRequestForm` 查询,SELECT `drf.desc_json`
|
||||||
|
- DTO: `RequestFormQueryDto` 有 `descJson` 字段 (String 类型)
|
||||||
|
|
||||||
|
### 数据库
|
||||||
|
- 表: `doc_request_form`,type_code = '23' 的记录 desc_json 均有数据
|
||||||
|
- descJson 包含: targetDepartment, urgencyLevel, symptom, sign, clinicalDiagnosis, otherDiagnosis, relatedResult, attention, examinationPurpose, medicalHistorySummary, allergyHistory, expectedExaminationTime 等
|
||||||
|
|
||||||
|
## 根因定位
|
||||||
|
|
||||||
|
对比检验申请 (testApplication.vue) 和检查申请 (examineApplication.vue) 的详情弹窗中"申请单描述"区域的渲染逻辑:
|
||||||
|
|
||||||
|
**testApplication.vue (检验申请) - 正确:**
|
||||||
|
```vue
|
||||||
|
<template v-for="(value, key) in descJsonData" :key="key">
|
||||||
|
<el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)">
|
||||||
|
{{ value || '-' }}
|
||||||
|
</el-descriptions-item>
|
||||||
|
</template>
|
||||||
|
```
|
||||||
|
- 遍历 `descJsonData` 的所有 key,只要 key 在 labelMap 中就显示
|
||||||
|
- 空值显示为 '-'
|
||||||
|
|
||||||
|
**examineApplication.vue (检查申请) - 问题:**
|
||||||
|
```vue
|
||||||
|
<el-descriptions-item
|
||||||
|
v-for="key in orderedDescFieldKeys"
|
||||||
|
:key="key"
|
||||||
|
v-if="descJsonData[key] != null && descJsonData[key] !== ''"
|
||||||
|
:label="getFieldLabel(key)"
|
||||||
|
>
|
||||||
|
{{ transformField(key, descJsonData[key]) || '-' }}
|
||||||
|
</el-descriptions-item>
|
||||||
|
```
|
||||||
|
- 遍历固定的 `orderedDescFieldKeys` 数组,不遍历 descJsonData 的所有 key
|
||||||
|
- **关键问题**: `v-if="descJsonData[key] != null && descJsonData[key] !== ''"` 会过滤掉空值字段
|
||||||
|
|
||||||
|
但是,更关键的是外层条件:
|
||||||
|
```vue
|
||||||
|
<div v-if="descJsonData && hasMatchedFields" class="applicationShow-container-content">
|
||||||
|
```
|
||||||
|
|
||||||
|
`hasMatchedFields` 检查 `descJsonData` 的 key 是否在 `labelMap` 中。`labelMap` 包含所有需要显示的字段。
|
||||||
|
|
||||||
|
**实际根因**:通过对比 testApplication.vue 与 examineApplication.vue,发现两个组件在 "申请单描述" 区域的渲染方式不同。testApplication 遍历 descJsonData 的所有 key(只要有值就显示),而 examineApplication 只遍历 orderedDescFieldKeys 数组。
|
||||||
|
|
||||||
|
**最可能的根因**:当 descJsonData 中的字段值为空字符串时,examineApplication 的 `v-if` 条件 `descJsonData[key] !== ''` 会过滤掉该字段(整行不显示),而 testApplication 会显示该字段标签并填入 `-`。
|
||||||
|
|
||||||
|
对于 `targetDepartment` 字段,`recursionFun` 函数在科室列表中找不到对应 ID 时会返回空字符串 `''`,导致 `targetDepartment` 被过滤不显示。
|
||||||
|
|
||||||
|
**但核心问题是**:如果 descJsonData 存在但某些字段为空,这些字段会被完全隐藏而不是显示 `-`。用户期望看到的是字段标签+占位符 `-`,而不是整个字段不显示。
|
||||||
|
|
||||||
|
## 修复方案
|
||||||
|
|
||||||
|
将 examineApplication.vue 中"申请单描述"区域的渲染方式改为与 testApplication.vue 一致:
|
||||||
|
1. 遍历 `descJsonData` 的所有 key(而非固定 orderedDescFieldKeys)
|
||||||
|
2. 使用 `isFieldMatched(key)` 过滤需要显示的字段
|
||||||
|
3. 空值显示为 `-`(而非完全隐藏)
|
||||||
|
|
||||||
|
同时保留 `orderedDescFieldKeys` 用于打印功能(已有代码使用)。
|
||||||
|
|
||||||
|
## 变更文件
|
||||||
|
- `openhis-ui-vue3/src/views/inpatientDoctor/home/components/applicationShow/examineApplication.vue`(前端模板修改)
|
||||||
|
|
||||||
|
修复结果:✅ 成功,5行改动(+5/-8)
|
||||||
55
BUG_556_ANALYSIS.md
Normal file
55
BUG_556_ANALYSIS.md
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
# Bug #556 分析报告
|
||||||
|
|
||||||
|
## Bug 描述
|
||||||
|
【门诊医生站-检验】新增检验申请单时就诊卡号/执行时间未自动回显,且项目列表冗余显示"套餐"文字
|
||||||
|
|
||||||
|
## 三个子问题分析
|
||||||
|
|
||||||
|
### 问题1:就诊卡号未自动回显
|
||||||
|
**位置**: `inspectionApplication.vue` 第 146 行 `formData.medicalrecordNumber`
|
||||||
|
|
||||||
|
**根因**:
|
||||||
|
- `initData()` 第 886 行使用 `props.patientInfo.identifierNo` 填充
|
||||||
|
- `resetForm()` 第 1526 行同样使用 `props.patientInfo.identifierNo` 填充
|
||||||
|
- `handleSave()` 第 1598-1607 行同步患者信息时,只同步了 `visitNo`(对应 `busNo`),**没有同步 `medicalrecordNumber`**
|
||||||
|
- `props.patientInfo` 中就诊卡号对应的字段是 `busNo`(就诊流水号/卡号),而 `identifierNo` 是身份证号/标识号,可能为空
|
||||||
|
- **修复**: `medicalrecordNumber` 应使用 `props.patientInfo.busNo` 填充,与 `visitNo` 保持一致
|
||||||
|
|
||||||
|
### 问题2:执行时间未自动填充
|
||||||
|
**位置**: `inspectionApplication.vue` 第 978 行 `executeTime: null`
|
||||||
|
|
||||||
|
**根因**:
|
||||||
|
- `formData.executeTime` 初始值为 `null`
|
||||||
|
- `initData()` 和 `resetForm()` 都**没有**设置 `executeTime` 默认值
|
||||||
|
- 第 176-186 行的日期选择器 placeholder 为"选择执行时间",无默认值
|
||||||
|
- **修复**: 在 `initData()` 和 `resetForm()` 中将 `executeTime` 设置为当前时间(格式与 `applyTime` 一致)
|
||||||
|
|
||||||
|
### 问题3:项目列表冗余显示"套餐"文字
|
||||||
|
**位置**: `inspectionApplication.vue` 第 566 行 `el-tag v-if="item.isPackage"`
|
||||||
|
|
||||||
|
**根因**:
|
||||||
|
- `loadCategoryItems()` 第 1190 行: `const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null'`
|
||||||
|
- 数据库中 ALL 12条 lab_activity_definition 记录都有 `fee_package_id` 值
|
||||||
|
- 但通过 `inspection_package_detail` 比对,只有 **2条** 记录的名称与套餐名称匹配("肝功能12项"和"免疫组织化学染色诊断")
|
||||||
|
- 其余 10 条是套餐子项或普通项目,不应显示"套餐"标签
|
||||||
|
- **修复**: 改为检查 `item.packageName && item.itemName === item.packageName`,即只有当项目名称与套餐名称匹配时才标记为套餐
|
||||||
|
|
||||||
|
## 修复方案
|
||||||
|
|
||||||
|
### 修复1: 就诊卡号
|
||||||
|
- `initData()`: `formData.medicalrecordNumber = props.patientInfo.busNo || ''`(替代 `identifierNo`)
|
||||||
|
- `resetForm()`: 同样改为 `busNo`
|
||||||
|
- `handleSave()`: 同步时增加 `formData.medicalrecordNumber` 同步
|
||||||
|
|
||||||
|
### 修复2: 执行时间
|
||||||
|
- `initData()`: 新增 `formData.executeTime = formatDateTime(new Date())`
|
||||||
|
- `resetForm()`: 将 `executeTime: null` 改为 `executeTime: formatDateTime(new Date())`
|
||||||
|
|
||||||
|
### 修复3: 套餐标签
|
||||||
|
- `loadCategoryItems()` 第1190行: 将 `isPackage` 判断改为 `item.packageName && item.name === item.packageName`
|
||||||
|
|
||||||
|
## 验证门禁
|
||||||
|
- Gate A: 根因已定位到具体代码行
|
||||||
|
- Gate B: 已读取 inspectionApplication.vue 全部代码,理解数据流
|
||||||
|
- Gate C: 修复方案与验收标准一致
|
||||||
|
- Gate D: 不涉及数据库 DDL 变更
|
||||||
42
analysis_469.md
Normal file
42
analysis_469.md
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
# 分析报告 — 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 置回 0,needRefund/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` - 前端API(deleteRequestForm, 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实现
|
||||||
41
bug547_analysis.md
Normal file
41
bug547_analysis.md
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
# 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`
|
||||||
2
his-repo
2
his-repo
Submodule his-repo updated: 414c204578...5de8a22418
30
md/bug-analysis/bug444-analysis.md
Normal file
30
md/bug-analysis/bug444-analysis.md
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
# 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. 药品名称正常显示
|
||||||
@@ -132,7 +132,22 @@ temporaryAdvices.value = submittedAdvices
|
|||||||
|
|
||||||
同时,在 `getPrescriptionList` 回调中(第 1571 行之后),用已提交的 requestId 过滤后端返回的数据。
|
同时,在 `getPrescriptionList` 回调中(第 1571 行之后),用已提交的 requestId 过滤后端返回的数据。
|
||||||
|
|
||||||
## 总结
|
## 修复结果
|
||||||
|
|
||||||
- **根因**:`handleMedicalAdvice` 每次打开都清空 `temporaryAdvices`,然后从后端重新拉取数据。但后端返回的新创建医嘱项可能没有 `requestId`,导致无法过滤。
|
### 实际根因
|
||||||
- **修复**:保留已提交(有 requestId)的医嘱数据,不清空;同时用这些 requestId 过滤后端返回的新数据。
|
`handleQuoteBilling` 函数中:
|
||||||
|
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)
|
||||||
|
|||||||
@@ -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.constant.CommonConstants;
|
import com.openhis.common.enums.SlotStatus;
|
||||||
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", CommonConstants.SlotStatus.BOOKED, CommonConstants.SlotStatus.LOCKED,
|
.in("status", SlotStatus.BOOKED.getValue(), SlotStatus.LOCKED.getValue(),
|
||||||
CommonConstants.SlotStatus.CHECKED_IN));
|
SlotStatus.CHECKED_IN.getValue()));
|
||||||
if (appointmentCount > 0) {
|
if (appointmentCount > 0) {
|
||||||
return R.fail("该排班已有患者预约,禁止删除!如需取消请先处理患者退预约或使用'停诊'功能。");
|
return R.fail("该排班已有患者预约,禁止删除!如需取消请先处理患者退预约或使用'停诊'功能。");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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.constant.CommonConstants.SlotStatus;
|
import com.openhis.common.enums.SlotStatus;
|
||||||
import com.openhis.common.enums.OrderStatus;
|
import com.openhis.common.enums.OrderStatus;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@@ -193,25 +193,24 @@ public class TicketAppServiceImpl implements ITicketAppService {
|
|||||||
if (Boolean.TRUE.equals(raw.getIsStopped())) {
|
if (Boolean.TRUE.equals(raw.getIsStopped())) {
|
||||||
dto.setStatus("已停诊");
|
dto.setStatus("已停诊");
|
||||||
} else {
|
} else {
|
||||||
Integer slotStatus = raw.getSlotStatus();
|
SlotStatus status = SlotStatus.getByValue(raw.getSlotStatus());
|
||||||
if (slotStatus != null) {
|
if (status != null) {
|
||||||
if (SlotStatus.CHECKED_IN.equals(slotStatus)) {
|
if (status == SlotStatus.LOCKED) {
|
||||||
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("已预约");
|
|
||||||
}
|
|
||||||
} else if (SlotStatus.RETURNED.equals(slotStatus)) {
|
|
||||||
dto.setStatus("已退号");
|
|
||||||
} else if (SlotStatus.CANCELLED.equals(slotStatus)) {
|
|
||||||
dto.setStatus("已停诊");
|
|
||||||
} else if (SlotStatus.LOCKED.equals(slotStatus)) {
|
|
||||||
dto.setStatus("已锁定");
|
dto.setStatus("已锁定");
|
||||||
|
}
|
||||||
|
} else if (status == SlotStatus.BOOKED) {
|
||||||
|
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("已退号");
|
||||||
} else {
|
} else {
|
||||||
dto.setStatus("未预约");
|
dto.setStatus("未预约");
|
||||||
}
|
}
|
||||||
@@ -237,6 +236,10 @@ 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) {
|
||||||
@@ -263,28 +266,31 @@ 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 "2":
|
case "3":
|
||||||
case "已取号":
|
case "已取号":
|
||||||
query.setStatus("checked");
|
query.setStatus("checked");
|
||||||
break;
|
break;
|
||||||
case "cancelled":
|
case "cancelled":
|
||||||
case "canceled":
|
case "canceled":
|
||||||
case "3":
|
case "4":
|
||||||
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;
|
||||||
}
|
}
|
||||||
@@ -367,26 +373,25 @@ 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: 已锁定...)
|
||||||
Integer slotStatus = raw.getSlotStatus();
|
SlotStatus status = SlotStatus.getByValue(raw.getSlotStatus());
|
||||||
if (slotStatus != null) {
|
if (status != null) {
|
||||||
if (SlotStatus.CHECKED_IN.equals(slotStatus)) {
|
if (status == SlotStatus.LOCKED) {
|
||||||
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("已预约");
|
|
||||||
}
|
|
||||||
} else if (SlotStatus.RETURNED.equals(slotStatus)) {
|
|
||||||
dto.setStatus("已退号");
|
|
||||||
} else if (SlotStatus.CANCELLED.equals(slotStatus)) {
|
|
||||||
dto.setStatus("已停诊");
|
|
||||||
} else if (SlotStatus.LOCKED.equals(slotStatus)) {
|
|
||||||
dto.setStatus("已锁定");
|
dto.setStatus("已锁定");
|
||||||
|
}
|
||||||
|
} else if (status == SlotStatus.BOOKED) {
|
||||||
|
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("已退号");
|
||||||
} else {
|
} else {
|
||||||
dto.setStatus("未预约");
|
dto.setStatus("未预约");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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.getOrgLocListByOrgIdAndActivityDefinitionId(orgLoc.getActivityDefinitionId());
|
organizationLocationService.getOrgLocListByActivityDefinitionId(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,9 +169,11 @@ 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() : "未知科室";
|
if (org == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
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() + "与" + org.getName() + "时间冲突");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (orgLocQueryDto.getId() != null) {
|
if (orgLocQueryDto.getId() != null) {
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ 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;
|
||||||
@@ -643,8 +644,7 @@ 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,18 +660,28 @@ public class OutpatientRegistrationAppServiceImpl implements IOutpatientRegistra
|
|||||||
return appointmentOrder.getId();
|
return appointmentOrder.getId();
|
||||||
}
|
}
|
||||||
|
|
||||||
int slotRows = scheduleSlotMapper.updateSlotStatus(slotId, CommonConstants.SlotStatus.AVAILABLE);
|
// 只有已预约(1)的号源才能退号,对应签到后的 BOOKED 状态
|
||||||
if (slotRows > 0) {
|
ScheduleSlot slot = scheduleSlotMapper.selectById(slotId);
|
||||||
|
if (slot == null || !SlotStatus.BOOKED.getValue().equals(slot.getStatus())) {
|
||||||
|
log.warn("退号跳过:槽位非已预约状态, slotId={}, status={}", slotId,
|
||||||
|
slot != null ? slot.getStatus() : null);
|
||||||
|
return appointmentOrder.getId();
|
||||||
|
}
|
||||||
|
|
||||||
|
int slotRows = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.AVAILABLE.getValue());
|
||||||
|
if (slotRows == 0) {
|
||||||
|
log.warn("退号时更新槽位状态未影响任何行, slotId={}", slotId);
|
||||||
|
return appointmentOrder.getId();
|
||||||
|
}
|
||||||
|
|
||||||
Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId);
|
Long poolId = scheduleSlotMapper.selectPoolIdBySlotId(slotId);
|
||||||
if (poolId != null) {
|
if (poolId != null) {
|
||||||
schedulePoolMapper.refreshPoolStats(poolId);
|
|
||||||
schedulePoolMapper.update(null,
|
schedulePoolMapper.update(null,
|
||||||
new LambdaUpdateWrapper<SchedulePool>()
|
new LambdaUpdateWrapper<SchedulePool>()
|
||||||
.setSql("version = version + 1")
|
.setSql("booked_num = booked_num - 1, version = version + 1")
|
||||||
.set(SchedulePool::getUpdateTime, new Date())
|
.set(SchedulePool::getUpdateTime, new Date())
|
||||||
.eq(SchedulePool::getId, poolId));
|
.eq(SchedulePool::getId, poolId));
|
||||||
}
|
}
|
||||||
}
|
|
||||||
return appointmentOrder.getId();
|
return appointmentOrder.getId();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.warn("同步预约号源已退号状态失败, encounterId={}", encounter.getId(), e);
|
log.warn("同步预约号源已退号状态失败, encounterId={}", encounter.getId(), e);
|
||||||
|
|||||||
@@ -35,6 +35,9 @@ 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;
|
||||||
@@ -51,6 +54,7 @@ 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;
|
||||||
@@ -69,6 +73,7 @@ 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;
|
||||||
|
|
||||||
@@ -132,6 +137,20 @@ 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:";
|
||||||
// 缓存过期时间(小时)
|
// 缓存过期时间(小时)
|
||||||
@@ -1750,6 +1769,7 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
|
|||||||
return StringUtils.isBlank(applyNo) ? null : applyNo;
|
return StringUtils.isBlank(applyNo) ? null : applyNo;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 处理诊疗
|
* 处理诊疗
|
||||||
*/
|
*/
|
||||||
@@ -1798,31 +1818,50 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 检验申请单在医嘱 contentJson 中写入 applyNo;从医嘱删除时需先级联作废检验单,避免检验页签仍显示孤儿申请
|
// 🔧 级联作废:在删除 ServiceRequest 之前,先读取所有待删除记录的级联信息
|
||||||
|
// 检验申请单: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;
|
||||||
}
|
}
|
||||||
iServiceRequestService.removeById(requestId);// 删除诊疗
|
ServiceRequest existing = iServiceRequestService.getById(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(adviceSaveDto.getRequestId());
|
.add(requestId);
|
||||||
|
}
|
||||||
|
// 手术申请单级联(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)) {
|
||||||
labCascadeSkippedRequestIds.addAll(e.getValue());
|
cascadeSkippedRequestIds.addAll(e.getValue());
|
||||||
log.info("handService - 级联作废检验申请单 applyNo={},已跳过重复删除的医嘱 requestIds={}",
|
log.info("handService - 级联作废检验申请单 applyNo={},已跳过重复删除的医嘱 requestIds={}",
|
||||||
e.getKey(), e.getValue());
|
e.getKey(), e.getValue());
|
||||||
} else {
|
} else {
|
||||||
@@ -1831,8 +1870,41 @@ 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 (labCascadeSkippedRequestIds.contains(adviceSaveDto.getRequestId())) {
|
if (cascadeSkippedRequestIds.contains(adviceSaveDto.getRequestId())) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
Long requestId = adviceSaveDto.getRequestId();
|
Long requestId = adviceSaveDto.getRequestId();
|
||||||
@@ -1842,7 +1914,6 @@ 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);
|
||||||
}
|
}
|
||||||
@@ -2121,11 +2192,6 @@ 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
|
||||||
@@ -2481,21 +2547,17 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 使用直接 LIMIT 查询,不触发 MyBatis Plus 的 COUNT 开销
|
// 使用 MyBatis Plus 分页查询
|
||||||
List<SurgeryItemDto> records = doctorStationAdviceAppMapper.getSurgeryPage(
|
IPage<SurgeryItemDto> result = 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) {
|
if (useCache && result instanceof com.baomidou.mybatisplus.extension.plugins.pagination.Page) {
|
||||||
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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -300,16 +300,12 @@ public class DoctorStationMainAppServiceImpl implements IDoctorStationMainAppSer
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3. 获取 pool_id 和 slot_id:优先使用 triage_queue_item(挂号时录入的号源信息,为权威来源)
|
// 3. 获取 pool_id 和 slot_id:优先使用 encounter.orderId → order_main → adm_schedule_slot 链路
|
||||||
// 队列项不存在或值缺失时,回退使用 encounter → order_main → adm_schedule_slot 链路
|
// (order_main.slot_id 为挂号时实际锁定的号源,是最权威的数据来源)
|
||||||
|
// 当无 orderId 或订单无 slot_id 时,回退使用 triage_queue_item 的 poolId/slotId
|
||||||
Long divPoolId = null;
|
Long divPoolId = null;
|
||||||
Long divSlotId = null;
|
Long divSlotId = null;
|
||||||
if (queueItem != null && queueItem.getPoolId() != null && queueItem.getSlotId() != null) {
|
if (encounter.getOrderId() != null) {
|
||||||
divPoolId = queueItem.getPoolId();
|
|
||||||
divSlotId = queueItem.getSlotId();
|
|
||||||
}
|
|
||||||
// 队列项 poolId/slotId 缺失时,通过 encounter.orderId → order_main.slot_id → adm_schedule_slot.pool_id 回退获取
|
|
||||||
if ((divPoolId == null || divSlotId == null) && encounter.getOrderId() != null) {
|
|
||||||
try {
|
try {
|
||||||
Order order = iOrderService.getById(encounter.getOrderId());
|
Order order = iOrderService.getById(encounter.getOrderId());
|
||||||
if (order != null && order.getSlotId() != null) {
|
if (order != null && order.getSlotId() != null) {
|
||||||
@@ -320,7 +316,16 @@ public class DoctorStationMainAppServiceImpl implements IDoctorStationMainAppSer
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.warn("回退获取完诊div_log的pool_id/slot_id失败,encounterId={}", encounterId, e);
|
log.warn("完诊获取div_log的pool_id/slot_id失败(order链路),encounterId={}", encounterId, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 订单链路无数据时,回退使用 triage_queue_item 的 poolId/slotId
|
||||||
|
if ((divPoolId == null || divSlotId == null) && queueItem != null) {
|
||||||
|
if (queueItem.getPoolId() != null) {
|
||||||
|
divPoolId = queueItem.getPoolId();
|
||||||
|
}
|
||||||
|
if (queueItem.getSlotId() != null) {
|
||||||
|
divSlotId = queueItem.getSlotId();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -96,6 +96,11 @@ public class DiagnosisQueryDto {
|
|||||||
*/
|
*/
|
||||||
private String diagnosisDoctor;
|
private String diagnosisDoctor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 长效诊断标识
|
||||||
|
*/
|
||||||
|
private Integer longTermFlag;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 是否已有传染病报卡(0-无,1-有)
|
* 是否已有传染病报卡(0-无,1-有)
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -23,6 +23,9 @@ 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;
|
||||||
|
|||||||
@@ -195,7 +195,7 @@ public interface DoctorStationAdviceAppMapper {
|
|||||||
* @param searchKey 模糊查询关键字(可选)
|
* @param searchKey 模糊查询关键字(可选)
|
||||||
* @return 手术项目分页数据
|
* @return 手术项目分页数据
|
||||||
*/
|
*/
|
||||||
List<SurgeryItemDto> getSurgeryPage(@Param("page") Page<SurgeryItemDto> page,
|
IPage<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);
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ 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;
|
||||||
@@ -202,62 +203,70 @@ 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();
|
||||||
for (ProductTransferDto dto : productTransferDtoList) {
|
Date now = DateUtils.getNowDate();
|
||||||
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> requestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
|
List<SupplyRequest> oldRequestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
|
||||||
if (!requestList.isEmpty()) {
|
if (!oldRequestList.isEmpty()) {
|
||||||
List<Long> requestIdList = requestList.stream().map(SupplyRequest::getId).collect(Collectors.toList());
|
// 恢复旧记录已预划扣的库存
|
||||||
// 单据信息删除
|
for (SupplyRequest oldReq : oldRequestList) {
|
||||||
supplyRequestService.removeByIds(requestIdList);
|
if (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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
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());
|
||||||
@@ -265,33 +274,58 @@ 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);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -332,33 +366,63 @@ 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();
|
||||||
for (ProductTransferDto dto : productTransferDtoList) {
|
Date now = DateUtils.getNowDate();
|
||||||
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> requestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
|
List<SupplyRequest> oldRequestList = supplyRequestService.getSupplyByBusNo(busNoList.get(0));
|
||||||
if (!requestList.isEmpty()) {
|
Map<String, BigDecimal> oldDeductionMap = new HashMap<>();
|
||||||
// 请求id取得
|
if (!oldRequestList.isEmpty()) {
|
||||||
List<Long> requestIdList = requestList.stream().map(SupplyRequest::getId).collect(Collectors.toList());
|
for (SupplyRequest oldReq : oldRequestList) {
|
||||||
// 单据信息删除
|
if (oldReq.getItemId() != null && oldReq.getLotNumber() != null && oldReq.getSourceLocationId() != null
|
||||||
supplyRequestService.removeByIds(requestIdList);
|
&& oldReq.getItemQuantity() != null) {
|
||||||
|
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<>();
|
||||||
@@ -405,6 +469,22 @@ 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))
|
||||||
|
|||||||
@@ -519,58 +519,8 @@ 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 查询源仓库库存表信息
|
// 🔧 源仓库库存已在保存时预划扣,审批通过时不再重复扣减
|
||||||
// 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);
|
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(
|
||||||
|
|||||||
@@ -269,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 || CommonConstants.Common.AREA_CODE.equals(idCard.substring(0, 6))) {
|
if (idCard == null || idCard.length() < 6 || 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);
|
||||||
|
|||||||
@@ -36,6 +36,8 @@ 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;
|
||||||
|
|
||||||
@@ -85,6 +87,9 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
|
|||||||
@Resource
|
@Resource
|
||||||
IDeviceDispenseService iDeviceDispenseService;
|
IDeviceDispenseService iDeviceDispenseService;
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
IRequestFormService iRequestFormService;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 查询住院患者信息
|
* 查询住院患者信息
|
||||||
*
|
*
|
||||||
@@ -266,6 +271,38 @@ 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();
|
||||||
|
|||||||
@@ -81,16 +81,7 @@ 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<>();
|
||||||
@@ -102,16 +93,17 @@ 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) {
|
||||||
throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室");
|
|
||||||
}
|
|
||||||
activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), configPositionId);
|
activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), configPositionId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// 诊疗处方号
|
// 诊疗处方号
|
||||||
String prescriptionNo;
|
String prescriptionNo;
|
||||||
@@ -176,6 +168,9 @@ 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 块生成手术医嘱,避免重复
|
||||||
|
boolean isSurgeryRequest = ActivityDefCategory.PROCEDURE.getCode().equals(typeCode);
|
||||||
|
if (!isSurgeryRequest) {
|
||||||
for (ActivitySaveDto activitySaveDto : activityList) {
|
for (ActivitySaveDto activitySaveDto : activityList) {
|
||||||
serviceRequest = new ServiceRequest();
|
serviceRequest = new ServiceRequest();
|
||||||
serviceRequest.setStatusEnum(RequestStatus.DRAFT.getValue());
|
serviceRequest.setStatusEnum(RequestStatus.DRAFT.getValue());
|
||||||
@@ -245,6 +240,7 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
|
|||||||
adviceUtils.handleActivityChild(childrenJson, organizationId, activityChildrenJsonParams);
|
adviceUtils.handleActivityChild(childrenJson, organizationId, activityChildrenJsonParams);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// 如果是手术申请单,需要额外生成手术医嘱
|
// 如果是手术申请单,需要额外生成手术医嘱
|
||||||
log.info("【调试】判断手术医嘱生成条件: typeCode={}, PROCEDURE.code={}, typeCode类型={}, PROCEDURE.code类型={}",
|
log.info("【调试】判断手术医嘱生成条件: typeCode={}, PROCEDURE.code={}, typeCode类型={}, PROCEDURE.code类型={}",
|
||||||
@@ -326,6 +322,13 @@ 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);
|
||||||
}
|
}
|
||||||
@@ -368,9 +371,10 @@ 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
|
// 优先从 activityList 获取 productId 和 definitionId
|
||||||
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);
|
||||||
@@ -409,6 +413,10 @@ 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("次");
|
||||||
@@ -425,6 +433,18 @@ 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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,11 @@ import java.math.BigDecimal;
|
|||||||
@Accessors(chain = true)
|
@Accessors(chain = true)
|
||||||
public class RequestFormDetailQueryDto {
|
public class RequestFormDetailQueryDto {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 诊疗活动定义ID(wor_service_request.activity_id,与开立检验时项目字典的 id / adviceDefinitionId 一致,用于编辑回显)
|
||||||
|
*/
|
||||||
|
private Long activityId;
|
||||||
|
|
||||||
/** 医嘱名称 */
|
/** 医嘱名称 */
|
||||||
private String adviceName;
|
private String adviceName;
|
||||||
|
|
||||||
|
|||||||
@@ -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,6 +871,7 @@
|
|||||||
</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,
|
||||||
@@ -893,7 +894,7 @@
|
|||||||
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 #{page.size} OFFSET ${(page.current - 1) * page.size}
|
LIMIT #{limit} OFFSET #{offset}
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<!-- 检查项目专用分页查询:仅查检查(23) + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
<!-- 检查项目专用分页查询:仅查检查(23) + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
||||||
@@ -902,6 +903,7 @@
|
|||||||
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,
|
||||||
@@ -913,6 +915,9 @@
|
|||||||
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 = '23'
|
AND t1.category_code = '23'
|
||||||
<if test="searchKey != null and searchKey != ''">
|
<if test="searchKey != null and searchKey != ''">
|
||||||
|
|||||||
@@ -135,6 +135,7 @@
|
|||||||
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
|
||||||
|
|||||||
@@ -57,8 +57,6 @@
|
|||||||
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}
|
||||||
|
|||||||
@@ -768,36 +768,4 @@ 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;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,57 @@
|
|||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -36,6 +36,14 @@ public interface IOrganizationLocationService extends IService<OrganizationLocat
|
|||||||
* @param activityDefinitionId 诊疗定义id
|
* @param activityDefinitionId 诊疗定义id
|
||||||
* @return 诊疗的执行科室列表
|
* @return 诊疗的执行科室列表
|
||||||
*/
|
*/
|
||||||
List<OrganizationLocation> getOrgLocListByOrgIdAndActivityDefinitionId(Long activityDefinitionId);
|
List<OrganizationLocation> getOrgLocListByOrgIdAndActivityDefinitionId(Long organizationId, Long activityDefinitionId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据诊疗定义id查询所有执行科室列表(跨科室)
|
||||||
|
*
|
||||||
|
* @param activityDefinitionId 诊疗定义id
|
||||||
|
* @return 执行科室列表
|
||||||
|
*/
|
||||||
|
List<OrganizationLocation> getOrgLocListByActivityDefinitionId(Long activityDefinitionId);
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -53,11 +53,25 @@ public class OrganizationLocationServiceImpl extends ServiceImpl<OrganizationLoc
|
|||||||
/**
|
/**
|
||||||
* 查询诊疗的执行科室列表
|
* 查询诊疗的执行科室列表
|
||||||
*
|
*
|
||||||
|
* @param organizationId 机构id
|
||||||
* @param activityDefinitionId 诊疗定义id
|
* @param activityDefinitionId 诊疗定义id
|
||||||
* @return 诊疗的执行科室列表
|
* @return 诊疗的执行科室列表
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public List<OrganizationLocation> getOrgLocListByOrgIdAndActivityDefinitionId(Long activityDefinitionId) {
|
public List<OrganizationLocation> getOrgLocListByOrgIdAndActivityDefinitionId(Long organizationId, Long activityDefinitionId) {
|
||||||
|
return baseMapper.selectList(new LambdaQueryWrapper<OrganizationLocation>()
|
||||||
|
.eq(OrganizationLocation::getOrganizationId, organizationId)
|
||||||
|
.eq(OrganizationLocation::getActivityDefinitionId, activityDefinitionId));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据诊疗定义id查询所有执行科室列表(跨科室)
|
||||||
|
*
|
||||||
|
* @param activityDefinitionId 诊疗定义id
|
||||||
|
* @return 执行科室列表
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public List<OrganizationLocation> getOrgLocListByActivityDefinitionId(Long activityDefinitionId) {
|
||||||
return baseMapper.selectList(new LambdaQueryWrapper<OrganizationLocation>()
|
return baseMapper.selectList(new LambdaQueryWrapper<OrganizationLocation>()
|
||||||
.eq(OrganizationLocation::getActivityDefinitionId, activityDefinitionId));
|
.eq(OrganizationLocation::getActivityDefinitionId, activityDefinitionId));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,10 +10,11 @@ import org.springframework.stereotype.Repository;
|
|||||||
public interface SchedulePoolMapper extends BaseMapper<SchedulePool> {
|
public interface SchedulePoolMapper extends BaseMapper<SchedulePool> {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 按号源池实时重算统计值,避免并发场景下计数漂移。
|
* 按号源池实时重算统计值。
|
||||||
*
|
*
|
||||||
* 说明:available_num 在当前项目中可能为数据库生成列,因此这里仅维护
|
* @param poolId 号源池ID
|
||||||
* booked_num / locked_num,剩余号由数据库或查询逻辑计算。
|
* @param bookedStatus 已约状态值,由 SlotStatus.BOOKED.getValue() 传入
|
||||||
|
* @param lockedStatus 锁定状态值,由 SlotStatus.LOCKED.getValue() 传入
|
||||||
*/
|
*/
|
||||||
@Update("""
|
@Update("""
|
||||||
UPDATE adm_schedule_pool p
|
UPDATE adm_schedule_pool p
|
||||||
@@ -23,20 +24,22 @@ 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 = 1
|
AND s.status = #{bookedStatus}
|
||||||
), 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 = 3
|
AND s.status = #{lockedStatus}
|
||||||
), 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
|
||||||
|
|||||||
@@ -22,9 +22,12 @@ public interface ScheduleSlotMapper extends BaseMapper<ScheduleSlot> {
|
|||||||
TicketSlotDTO selectTicketSlotById(@Param("id") Long id);
|
TicketSlotDTO selectTicketSlotById(@Param("id") Long id);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 原子抢占槽位:仅当当前状态=0(可用)时,更新为1(已预约)。
|
* 原子抢占槽位:仅当当前状态=0(待约)时,更新为目标锁定状态。
|
||||||
|
*
|
||||||
|
* @param slotId 槽位ID
|
||||||
|
* @param lockedStatus 锁定状态值,由 SlotStatus.LOCKED.getValue() 传入
|
||||||
*/
|
*/
|
||||||
int lockSlotForBooking(@Param("slotId") Long slotId);
|
int lockSlotForBooking(@Param("slotId") Long slotId, @Param("lockedStatus") Integer lockedStatus);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 按主键更新槽位状态。
|
* 按主键更新槽位状态。
|
||||||
@@ -35,11 +38,15 @@ public interface ScheduleSlotMapper extends BaseMapper<ScheduleSlot> {
|
|||||||
* 更新槽位状态并记录签到时间
|
* 更新槽位状态并记录签到时间
|
||||||
*
|
*
|
||||||
* @param slotId 槽位ID
|
* @param slotId 槽位ID
|
||||||
* @param status 状态
|
* @param status 目标状态,由 SlotStatus.BOOKED.getValue() 传入
|
||||||
* @param checkInTime 签到时间
|
* @param checkInTime 签到时间
|
||||||
|
* @param requiredStatus 前置状态,由 SlotStatus.LOCKED.getValue() 传入
|
||||||
* @return 结果
|
* @return 结果
|
||||||
*/
|
*/
|
||||||
int updateSlotStatusAndCheckInTime(@Param("slotId") Long slotId, @Param("status") Integer status, @Param("checkInTime") Date checkInTime);
|
int updateSlotStatusAndCheckInTime(@Param("slotId") Long slotId,
|
||||||
|
@Param("status") Integer status,
|
||||||
|
@Param("checkInTime") Date checkInTime,
|
||||||
|
@Param("requiredStatus") Integer requiredStatus);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 根据槽位ID查询所属号源池ID。
|
* 根据槽位ID查询所属号源池ID。
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
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;
|
||||||
@@ -13,7 +15,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.constant.CommonConstants.SlotStatus;
|
import com.openhis.common.enums.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;
|
||||||
@@ -177,7 +179,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.AVAILABLE.equals(slot.getSlotStatus())) {
|
if (slot.getSlotStatus() != null && SlotStatus.getByValue(slot.getSlotStatus()) != SlotStatus.AVAILABLE) {
|
||||||
throw new RuntimeException("手慢了!该号源已刚刚被他人抢占");
|
throw new RuntimeException("手慢了!该号源已刚刚被他人抢占");
|
||||||
}
|
}
|
||||||
if (Boolean.TRUE.equals(slot.getIsStopped())) {
|
if (Boolean.TRUE.equals(slot.getIsStopped())) {
|
||||||
@@ -205,7 +207,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 原子抢占:避免并发下同一槽位被重复预约
|
// 原子抢占:避免并发下同一槽位被重复预约
|
||||||
int lockRows = scheduleSlotMapper.lockSlotForBooking(slotId);
|
int lockRows = scheduleSlotMapper.lockSlotForBooking(slotId, SlotStatus.LOCKED.getValue());
|
||||||
if (lockRows <= 0) {
|
if (lockRows <= 0) {
|
||||||
throw new RuntimeException("手慢了!该号源已刚刚被他人抢占");
|
throw new RuntimeException("手慢了!该号源已刚刚被他人抢占");
|
||||||
}
|
}
|
||||||
@@ -260,7 +262,15 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
|
|||||||
throw new RuntimeException("预约成功但号源回填订单失败,请重试");
|
throw new RuntimeException("预约成功但号源回填订单失败,请重试");
|
||||||
}
|
}
|
||||||
|
|
||||||
refreshPoolStatsBySlotId(slotId);
|
// 6. 预约成功后 locked_num+1(原子递增替代全量 recount,避免并发计数漂移)
|
||||||
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -277,7 +287,8 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
|
|||||||
if (slot == null) {
|
if (slot == null) {
|
||||||
throw new RuntimeException("号源槽位不存在");
|
throw new RuntimeException("号源槽位不存在");
|
||||||
}
|
}
|
||||||
if (slot.getSlotStatus() == null || !SlotStatus.BOOKED.equals(slot.getSlotStatus())) {
|
// 只有锁定态(2)的号源可以取消预约
|
||||||
|
if (slot.getSlotStatus() == null || SlotStatus.getByValue(slot.getSlotStatus()) != SlotStatus.LOCKED) {
|
||||||
throw new RuntimeException("号源不可取消预约");
|
throw new RuntimeException("号源不可取消预约");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -292,7 +303,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
|
|||||||
orderService.cancelAppointmentOrder(order.getId(), "患者取消预约");
|
orderService.cancelAppointmentOrder(order.getId(), "患者取消预约");
|
||||||
}
|
}
|
||||||
|
|
||||||
int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.AVAILABLE);
|
int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.AVAILABLE.getValue());
|
||||||
if (updated > 0) {
|
if (updated > 0) {
|
||||||
refreshPoolStatsBySlotId(slotId);
|
refreshPoolStatsBySlotId(slotId);
|
||||||
}
|
}
|
||||||
@@ -318,11 +329,14 @@ 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)的号源才能签到,签到时 2→1(LOCKED→BOOKED)
|
||||||
ScheduleSlot slot = scheduleSlotMapper.selectById(slotId);
|
ScheduleSlot slot = scheduleSlotMapper.selectById(slotId);
|
||||||
|
if (slot == null || !SlotStatus.LOCKED.getValue().equals(slot.getStatus())) {
|
||||||
|
throw new RuntimeException("号源状态异常,无法签到");
|
||||||
|
}
|
||||||
|
|
||||||
// 3. 更新号源槽位状态为已签到,记录签到时间
|
// 3. 更新号源槽位状态 2→1(LOCKED→BOOKED,已预约=已签到)
|
||||||
scheduleSlotMapper.updateSlotStatusAndCheckInTime(slotId, SlotStatus.CHECKED_IN, new Date());
|
scheduleSlotMapper.updateSlotStatusAndCheckInTime(slotId, SlotStatus.BOOKED.getValue(), new Date(), SlotStatus.LOCKED.getValue());
|
||||||
|
|
||||||
// 4. 更新号源池统计:锁定数-1,已预约数+1
|
// 4. 更新号源池统计:锁定数-1,已预约数+1
|
||||||
if (slot != null && slot.getPoolId() != null) {
|
if (slot != null && slot.getPoolId() != null) {
|
||||||
@@ -351,7 +365,7 @@ public class TicketServiceImpl extends ServiceImpl<TicketMapper, Ticket> impleme
|
|||||||
orderService.cancelAppointmentOrder(order.getId(), "医生停诊");
|
orderService.cancelAppointmentOrder(order.getId(), "医生停诊");
|
||||||
}
|
}
|
||||||
|
|
||||||
int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.CANCELLED);
|
int updated = scheduleSlotMapper.updateSlotStatus(slotId, SlotStatus.CANCELLED.getValue());
|
||||||
if (updated > 0) {
|
if (updated > 0) {
|
||||||
refreshPoolStatsBySlotId(slotId);
|
refreshPoolStatsBySlotId(slotId);
|
||||||
}
|
}
|
||||||
@@ -364,7 +378,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);
|
schedulePoolMapper.refreshPoolStats(poolId, SlotStatus.BOOKED.getValue(), SlotStatus.LOCKED.getValue());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -79,11 +79,13 @@ public class OpSchedule extends HisBaseEntity {
|
|||||||
private String surgerySite;
|
private String surgerySite;
|
||||||
|
|
||||||
/** 入院时间 */
|
/** 入院时间 */
|
||||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
|
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
private LocalDateTime admissionTime;
|
private LocalDateTime admissionTime;
|
||||||
|
|
||||||
/** 入手术室时间 */
|
/** 入手术室时间 */
|
||||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
|
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
private LocalDateTime entryTime;
|
private LocalDateTime entryTime;
|
||||||
|
|
||||||
/** 手术室编码 */
|
/** 手术室编码 */
|
||||||
@@ -142,19 +144,23 @@ public class OpSchedule extends HisBaseEntity {
|
|||||||
private String assistant3Code;
|
private String assistant3Code;
|
||||||
|
|
||||||
/** 手术开始时间 */
|
/** 手术开始时间 */
|
||||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
|
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
private LocalDateTime startTime;
|
private LocalDateTime startTime;
|
||||||
|
|
||||||
/** 手术结束时间 */
|
/** 手术结束时间 */
|
||||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
|
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
private LocalDateTime endTime;
|
private LocalDateTime endTime;
|
||||||
|
|
||||||
/** 麻醉开始时间 */
|
/** 麻醉开始时间 */
|
||||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
|
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
private LocalDateTime anesStart;
|
private LocalDateTime anesStart;
|
||||||
|
|
||||||
/** 麻醉结束时间 */
|
/** 麻醉结束时间 */
|
||||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
|
@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
||||||
private LocalDateTime anesEnd;
|
private LocalDateTime anesEnd;
|
||||||
|
|
||||||
/** 手术状态 */
|
/** 手术状态 */
|
||||||
|
|||||||
@@ -4,14 +4,17 @@
|
|||||||
"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', 'cancelled', 'canceled', 'stopped') THEN 2
|
WHEN LOWER(CONCAT('', s.status)) IN ('2', 'locked') 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', 'locked') THEN 4
|
WHEN LOWER(CONCAT('', s.status)) IN ('4', 'cancelled', 'canceled', 'stopped') 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
|
||||||
@@ -31,9 +34,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', 'cancelled', 'canceled', 'stopped') THEN 2
|
WHEN LOWER(CONCAT('', p.status)) IN ('2', 'locked') 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', 'locked') THEN 4
|
WHEN LOWER(CONCAT('', p.status)) IN ('4', 'cancelled', 'canceled', 'stopped') 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
|
||||||
@@ -149,10 +152,11 @@
|
|||||||
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 = 1,
|
status = #{lockedStatus},
|
||||||
update_time = now()
|
update_time = now()
|
||||||
WHERE
|
WHERE
|
||||||
id = #{slotId}
|
id = #{slotId}
|
||||||
@@ -174,6 +178,7 @@
|
|||||||
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
|
||||||
@@ -182,6 +187,7 @@
|
|||||||
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>
|
||||||
|
|
||||||
@@ -202,7 +208,7 @@
|
|||||||
update_time = now()
|
update_time = now()
|
||||||
WHERE
|
WHERE
|
||||||
id = #{slotId}
|
id = #{slotId}
|
||||||
AND status = 1
|
AND status = 2
|
||||||
AND delete_flag = '0'
|
AND delete_flag = '0'
|
||||||
</update>
|
</update>
|
||||||
|
|
||||||
@@ -299,15 +305,16 @@
|
|||||||
<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. 按系统时间过滤(Bug #398 #399 修复:仅未预约受时间过滤,已预约/已取号/已退号不受影响) -->
|
<!-- 5. 时间过滤: 仅待约(0)受时间限制,已锁定(2)/已约(1)/已签到(3)/已退号(5)不受影响 -->
|
||||||
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. 状态过滤 -->
|
<!-- 6. 状态筛选: unbooked(0) locked(2) booked(2) checked(1) cancelled(4) returned(5) -->
|
||||||
<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)">
|
||||||
@@ -318,7 +325,15 @@
|
|||||||
)
|
)
|
||||||
</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" /> = 1
|
AND <include refid="slotStatusNormExpr" /> = 2
|
||||||
|
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
|
||||||
@@ -326,13 +341,7 @@
|
|||||||
)
|
)
|
||||||
</when>
|
</when>
|
||||||
<when test="'checked'.equals(query.status) or '已取号'.equals(query.status)">
|
<when test="'checked'.equals(query.status) or '已取号'.equals(query.status)">
|
||||||
AND (
|
AND <include refid="slotStatusNormExpr" /> = 1
|
||||||
<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
|
||||||
@@ -340,7 +349,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" /> = 2
|
<include refid="slotStatusNormExpr" /> = 4
|
||||||
OR d.is_stopped = TRUE
|
OR d.is_stopped = TRUE
|
||||||
)
|
)
|
||||||
</when>
|
</when>
|
||||||
|
|||||||
@@ -172,12 +172,12 @@ export const SlotStatus = {
|
|||||||
AVAILABLE: 0,
|
AVAILABLE: 0,
|
||||||
/** 已预约 */
|
/** 已预约 */
|
||||||
BOOKED: 1,
|
BOOKED: 1,
|
||||||
/** 已取消 / 已停诊 */
|
/** 已锁定 */
|
||||||
CANCELLED: 2,
|
LOCKED: 2,
|
||||||
/** 已签到 / 已取号 */
|
/** 已签到 / 已取号 */
|
||||||
CHECKED_IN: 3,
|
CHECKED_IN: 3,
|
||||||
/** 已锁定 */
|
/** 已取消 / 已停诊 */
|
||||||
LOCKED: 4,
|
CANCELLED: 4,
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -185,10 +185,10 @@ export const SlotStatus = {
|
|||||||
*/
|
*/
|
||||||
export const SlotStatusDescriptions = {
|
export const SlotStatusDescriptions = {
|
||||||
0: '未预约',
|
0: '未预约',
|
||||||
1: '已预约',
|
1: '已取号',
|
||||||
2: '已停诊',
|
2: '已锁定',
|
||||||
3: '已取号',
|
3: '已取号',
|
||||||
4: '已锁定',
|
4: '已停诊',
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -34,6 +34,7 @@
|
|||||||
<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>
|
||||||
@@ -253,6 +254,7 @@ 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',
|
||||||
@@ -774,6 +776,7 @@ export default {
|
|||||||
// 🔧 BugFix#399: 确保已取号状态正确匹配
|
// 🔧 BugFix#399: 确保已取号状态正确匹配
|
||||||
const statusMap = {
|
const statusMap = {
|
||||||
unbooked: ['未预约'],
|
unbooked: ['未预约'],
|
||||||
|
locked: ['已锁定'],
|
||||||
booked: ['已预约'],
|
booked: ['已预约'],
|
||||||
checked: ['已取号', '已签到'],
|
checked: ['已取号', '已签到'],
|
||||||
cancelled: ['已停诊', '已取消'],
|
cancelled: ['已停诊', '已取消'],
|
||||||
|
|||||||
@@ -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: 'booked',
|
status: 'locked',
|
||||||
name: checkInSearchKey.value, // 支持姓名等模糊查询,后端需适配
|
name: checkInSearchKey.value, // 支持姓名等模糊查询,后端需适配
|
||||||
page: checkInPage.value,
|
page: checkInPage.value,
|
||||||
limit: checkInLimit.value
|
limit: checkInLimit.value
|
||||||
|
|||||||
@@ -461,6 +461,10 @@ 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")
|
||||||
@@ -1025,7 +1029,9 @@ 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){//待签发
|
||||||
if(Number(k.bizRequestFlag)==1||!k.bizRequestFlag){
|
// 手术计费场景(generateSourceEnum=6)不限制 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
|
||||||
@@ -1040,7 +1046,9 @@ function changeCheck(value,index,row){
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if(k.statusEnum == 2){ //已签发
|
if(k.statusEnum == 2){ //已签发
|
||||||
if(Number(k.bizRequestFlag)==1||!k.bizRequestFlag){
|
// 手术计费场景(generateSourceEnum=6)不限制 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
|
||||||
@@ -1067,28 +1075,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
|
// 无可签发项目时提前返回,避免后端报"医嘱列表为空"
|
||||||
// .filter((item) => {
|
if (saveList.length == 0) {
|
||||||
// return item.check;
|
proxy.$modal.msgWarning('当前无可签发处方');
|
||||||
// }).filter((item) => {
|
return;
|
||||||
// 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,
|
||||||
dbOpType: '1',
|
// 已有 requestId 的记录走 UPDATE 路径,新记录走 INSERT 路径
|
||||||
|
dbOpType: item.requestId ? '2' : '1',
|
||||||
groupId: item.groupId,
|
groupId: item.groupId,
|
||||||
// 🔧 Bug #443: 补充顶层关键字段(这些不在 contentJson 中,需从 API 响应顶层提取)
|
// 补充顶层关键字段(这些可能不在 contentJson 中,需从 API 响应顶层提取)
|
||||||
encounterId: item.encounterId,
|
encounterId: item.encounterId,
|
||||||
patientId: item.patientId,
|
patientId: item.patientId,
|
||||||
locationId: item.positionId,
|
locationId: item.positionId,
|
||||||
@@ -1096,9 +1104,14 @@ function handleSave() {
|
|||||||
adviceTableName: item.adviceTableName,
|
adviceTableName: item.adviceTableName,
|
||||||
adviceDefinitionId: item.adviceDefinitionId,
|
adviceDefinitionId: item.adviceDefinitionId,
|
||||||
chargeItemId: item.chargeItemId,
|
chargeItemId: item.chargeItemId,
|
||||||
// 🔧 Bug Fix: 签发时显式设置手术计费关键字段,避免后端 prescription_no / generateSourceEnum 回退为默认值导致查询无法匹配
|
// 补充数量、单位、批号等字段(后端 handDevice 需要这些字段)
|
||||||
generateSourceEnum: props.generateSourceEnum ?? parsedContent.generateSourceEnum,
|
quantity: item.quantity,
|
||||||
sourceBillNo: props.sourceBillNo ?? parsedContent.sourceBillNo,
|
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)
|
||||||
@@ -1160,8 +1173,9 @@ function handleSaveSign(row, index) {
|
|||||||
cleanRow.generateSourceEnum = 6; // 手术计费
|
cleanRow.generateSourceEnum = 6; // 手术计费
|
||||||
cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
|
cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
|
||||||
}
|
}
|
||||||
console.log('cleanRow', cleanRow)
|
// 🔧 门诊计费场景:保存为草稿,让药品出现在临时医嘱弹窗"已引用计费药品(待生成医嘱)"中
|
||||||
savePrescription({ adviceSaveList: [cleanRow] }, '1').then((res) => {
|
const adviceOpType = props.patientInfo.sourceBillNo ? '0' : '1'
|
||||||
|
savePrescription({ adviceSaveList: [cleanRow] }, adviceOpType).then((res) => {
|
||||||
if (res.code === 200) {
|
if (res.code === 200) {
|
||||||
proxy.$modal.msgSuccess('保存成功');
|
proxy.$modal.msgSuccess('保存成功');
|
||||||
getListInfo(false);
|
getListInfo(false);
|
||||||
@@ -1185,6 +1199,10 @@ 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) => {
|
||||||
|
|||||||
@@ -354,6 +354,11 @@ 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];
|
||||||
|
|||||||
@@ -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">
|
||||||
{{ scope.row.selectedMethod?.packagePrice || scope.row.price }}
|
{{ formatDetailAmount(getSelectedItemAmount(scope.row)) }}
|
||||||
</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">
|
||||||
{{ ((scope.row.selectedMethod?.packagePrice || scope.row.price || 0) * (scope.row.quantity || 1)).toFixed(2) }}
|
{{ formatDetailAmount(getSelectedItemAmount(scope.row) * (scope.row.quantity || 1)) }}
|
||||||
</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,37 +392,6 @@
|
|||||||
<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>
|
||||||
@@ -440,31 +409,35 @@
|
|||||||
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-tag v-if="item.isPackage || item.packageName" size="small" type="warning" style="margin-right: 4px; flex-shrink: 0;">套餐</el-tag>
|
<el-tooltip :content="getDisplayItemName(item)" placement="top" :show-after="400">
|
||||||
<el-tooltip :content="item.name" placement="top" :show-after="400">
|
<span class="card-name">{{ getDisplayItemName(item) }}</span>
|
||||||
<span class="card-name">{{ item.name }}</span>
|
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
<span class="card-price">¥{{ formatDetailAmount(item.price) }}</span>
|
<span class="card-price">¥{{ formatDetailAmount(getSelectedItemAmount(item)) }}</span>
|
||||||
<el-icon :class="['expand-icon', { expanded: item.expanded }]">
|
<el-icon :class="['expand-icon', { expanded: item.expanded }]">
|
||||||
<ArrowDown v-if="!item.expanded" />
|
<ArrowDown />
|
||||||
<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>
|
||||||
<!-- Bug #428: 有套餐 ID 时默认展开;加载中/空/明细均在本区域展示 -->
|
<div v-if="item.expanded" class="selected-card-body">
|
||||||
<div v-if="item.expanded && shouldShowPackageBody(item)" class="selected-card-body">
|
<div v-if="shouldShowItemPackageBody(item)">
|
||||||
|
<div class="package-toggle" @click.stop="toggleItemPackageExpand(item)">
|
||||||
|
<span>项目套餐明细</span>
|
||||||
|
<el-icon :class="['expand-icon', { expanded: item.itemPackageExpanded }]">
|
||||||
|
<ArrowDown />
|
||||||
|
</el-icon>
|
||||||
|
</div>
|
||||||
|
<div v-show="item.itemPackageExpanded">
|
||||||
<div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div>
|
<div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div>
|
||||||
<template v-else>
|
<template v-else>
|
||||||
<div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty">
|
<div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty">
|
||||||
暂无套餐明细
|
暂无套餐明细
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="package-details-list">
|
<div v-else class="package-details-list">
|
||||||
<div class="package-details-head">套餐明细</div>
|
|
||||||
<div
|
<div
|
||||||
v-for="(detail, dIdx) in getPackageDetailsList(item)"
|
v-for="(detail, dIdx) in getPackageDetailsList(item)"
|
||||||
:key="detail.id ?? detail.itemCode ?? `d-${dIdx}`"
|
:key="detail.id ?? detail.itemCode ?? `d-${dIdx}`"
|
||||||
@@ -482,6 +455,59 @@
|
|||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
</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-checkbox>
|
||||||
|
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div v-if="shouldShowMethodPackageBody(item)">
|
||||||
|
<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>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -493,7 +519,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, ArrowUp, Close } from '@element-plus/icons-vue';
|
import { Printer, Delete, ArrowDown, 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';
|
||||||
@@ -624,24 +650,48 @@ 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;
|
||||||
}
|
}
|
||||||
const carrier = getPackageCarrier(item);
|
return Array.isArray(item?.packageDetails) ? item.packageDetails : [];
|
||||||
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 !!(getPackageCarrier(item)?.packageId || item.packageName || item.packageId);
|
return shouldShowItemPackageBody(item) || shouldShowMethodPackageBody(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 金额展示:统一两位小数 */
|
/** 金额展示:统一两位小数 */
|
||||||
@@ -650,20 +700,18 @@ function formatDetailAmount(value) {
|
|||||||
return Number.isFinite(n) ? n.toFixed(2) : '0.00';
|
return Number.isFinite(n) ? n.toFixed(2) : '0.00';
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 默认检查方法:优先与部位 packageId 一致的方法,否则取首个带套餐的方法,否则取第一个 */
|
/** 已选卡片名称:去掉 UI 上冗余的“套餐”前缀,完整名称通过 tooltip 展示 */
|
||||||
function pickDefaultMethod(methods, partItem) {
|
function getDisplayItemName(item) {
|
||||||
if (!methods?.length) return null;
|
return String(item?.name || '').replace(/^套餐[::\-\s]*/, '');
|
||||||
if (methods.length === 1) return methods[0];
|
}
|
||||||
const pid = partItem?.packageId ?? null;
|
|
||||||
if (pid != null && pid !== '') {
|
function getSelectedItemAmount(item) {
|
||||||
const matched = methods.find(
|
const itemPrice = Number(item?.price || 0);
|
||||||
(x) => x.packageId != null && String(x.packageId) === String(pid)
|
const methodPrice = Number(item?.selectedMethod?.packagePrice || 0);
|
||||||
);
|
if (!hasMethodPackage(item) || isSamePackage(item)) {
|
||||||
if (matched) return matched;
|
return itemPrice;
|
||||||
}
|
}
|
||||||
const withPkg = methods.find((x) => x.packageId != null);
|
return itemPrice + methodPrice;
|
||||||
if (withPkg) return withPkg;
|
|
||||||
return methods[0];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function parsePackageDetailsPayload(res) {
|
function parsePackageDetailsPayload(res) {
|
||||||
@@ -679,15 +727,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) {
|
||||||
const carrier = getPackageCarrier(item);
|
let packageId = item.packageId;
|
||||||
let packageId = item.packageId || carrier?.packageId;
|
const packageName = item.packageName;
|
||||||
if (!packageId && !item.packageName) {
|
if (!packageId && !packageName) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
item.packageDetailsLoading = true;
|
item.packageDetailsLoading = true;
|
||||||
try {
|
try {
|
||||||
if (!packageId && item.packageName) {
|
if (!packageId && packageName) {
|
||||||
const pkgRes = await listCheckPackage({ packageName: item.packageName });
|
const pkgRes = await listCheckPackage({ 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 || [];
|
||||||
@@ -699,6 +747,7 @@ 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 = [];
|
||||||
@@ -718,7 +767,6 @@ 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) => ({
|
||||||
@@ -735,7 +783,6 @@ 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;
|
||||||
@@ -1058,7 +1105,10 @@ async function loadCategoryList() {
|
|||||||
|
|
||||||
// 默认展开第一个
|
// 默认展开第一个
|
||||||
if (categoryList.value.length > 0) {
|
if (categoryList.value.length > 0) {
|
||||||
activeNames.value = categoryList.value[0].typeId;
|
const firstCat = categoryList.value[0];
|
||||||
|
activeNames.value = firstCat.typeId;
|
||||||
|
await nextTick();
|
||||||
|
await handleCategoryExpand(firstCat);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('加载检查项目分类失败', err);
|
console.error('加载检查项目分类失败', err);
|
||||||
@@ -1079,10 +1129,9 @@ 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 = item.selectedMethod?.packagePrice || item.price;
|
const effectivePrice = getSelectedItemAmount(item);
|
||||||
return sum + (effectivePrice * (item.quantity || 1));
|
return sum + (effectivePrice * (item.quantity || 1));
|
||||||
}, 0);
|
}, 0);
|
||||||
return total.toFixed(2);
|
return total.toFixed(2);
|
||||||
@@ -1215,8 +1264,7 @@ function handleSave() {
|
|||||||
itemCode: String(item.id),
|
itemCode: String(item.id),
|
||||||
itemName: item.name,
|
itemName: item.name,
|
||||||
bodyPartCode: item.checkType || 'unknown',
|
bodyPartCode: item.checkType || 'unknown',
|
||||||
// Bug #384修复: 如果选中了检查方法且有套餐价格,使用套餐价格;否则使用部位价格
|
itemFee: getSelectedItemAmount(item),
|
||||||
itemFee: item.selectedMethod?.packagePrice || item.price,
|
|
||||||
performDeptCode: form.performDeptCode || '',
|
performDeptCode: form.performDeptCode || '',
|
||||||
itemStatus: 0,
|
itemStatus: 0,
|
||||||
itemSeq: index + 1,
|
itemSeq: index + 1,
|
||||||
@@ -1269,6 +1317,8 @@ 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,
|
||||||
@@ -1298,17 +1348,10 @@ 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修复
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1322,14 +1365,21 @@ function handleRowClick(row) {
|
|||||||
selectedItems.value = itemsWithMethods;
|
selectedItems.value = itemsWithMethods;
|
||||||
// 加载套餐明细(单个失败不影响其他项目和明细显示)
|
// 加载套餐明细(单个失败不影响其他项目和明细显示)
|
||||||
for (const it of selectedItems.value) {
|
for (const it of selectedItems.value) {
|
||||||
if (getPackageCarrier(it)?.packageId) {
|
if (hasItemPackage(it)) {
|
||||||
try {
|
try {
|
||||||
await loadPackageDetailsForItem(it);
|
await loadPackageDetailsForItem(it);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('加载套餐明细失败:', it.name, e);
|
console.error('加载套餐明细失败:', it.name, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
it.expanded = !!getPackageCarrier(it)?.packageId;
|
if (hasMethodPackage(it) && !isSamePackage(it)) {
|
||||||
|
try {
|
||||||
|
await loadMethodPackageDetails(it, it.selectedMethod);
|
||||||
|
} catch (e) {
|
||||||
|
console.error('加载检查方法套餐明细失败:', it.name, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
it.expanded = shouldShowPackageBody(it);
|
||||||
}
|
}
|
||||||
syncCategoryChecked();
|
syncCategoryChecked();
|
||||||
// Bug #384修复: 回充后更新检查方法显示
|
// Bug #384修复: 回充后更新检查方法显示
|
||||||
@@ -1359,97 +1409,6 @@ 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) {
|
||||||
@@ -1506,6 +1465,8 @@ 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,
|
||||||
@@ -1516,15 +1477,15 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
// 必须用数组里的响应式行,不能继续改局部 newRow:push 后列表内是 proxy,改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
|
// 必须用数组里的响应式行,不能继续改局部 newRow:push 后列表内是 proxy,改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
|
||||||
const row = selectedItems.value[selectedItems.value.length - 1];
|
const row = selectedItems.value[selectedItems.value.length - 1];
|
||||||
|
|
||||||
// 右侧不再展示「检查方法」列表:自动选默认方法(保存、计价仍依赖 selectedMethod)
|
// 勾选项目只加入项目列表,检查方法由用户在“检查方法”区域手动选择
|
||||||
if (methods.length >= 1) {
|
row.selectedMethod = null;
|
||||||
row.selectedMethod = pickDefaultMethod(methods, item);
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
|
|
||||||
// 有套餐 ID 时默认展开(先显示加载区,明细写入行对象 packageDetailsDisplay)
|
// 新勾选项目后默认展开,直接展示检查方法状态和套餐明细
|
||||||
row.expanded = !!getPackageCarrier(row)?.packageId;
|
row.expanded = true;
|
||||||
if (getPackageCarrier(row)?.packageId) {
|
row.itemPackageExpanded = true;
|
||||||
|
row.methodPackageExpanded = true;
|
||||||
|
if (hasItemPackage(row)) {
|
||||||
await loadPackageDetailsForItem(row);
|
await loadPackageDetailsForItem(row);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1550,13 +1511,35 @@ 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 && (item.isPackage || item.packageName) && (!item.packageDetails || item.packageDetails.length === 0) && !item.packageDetailsLoading) {
|
if (item.expanded && hasItemPackage(item) && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
||||||
await loadPackageDetailsForItem(item);
|
await loadPackageDetailsForItem(item);
|
||||||
}
|
}
|
||||||
if (item.expanded && shouldShowPackageBody(item)) {
|
if (
|
||||||
if (getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
item.expanded &&
|
||||||
|
shouldShowMethodPackageBody(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);
|
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1564,9 +1547,8 @@ async function toggleItemExpand(item) {
|
|||||||
async function selectMethodCheckbox(checked, item, method) {
|
async function selectMethodCheckbox(checked, item, method) {
|
||||||
if (checked) {
|
if (checked) {
|
||||||
item.selectedMethod = method;
|
item.selectedMethod = method;
|
||||||
if (item.expanded && (method.packageId || method.packageName)) {
|
item.expanded = true;
|
||||||
loadPackageDetailsForItem(item);
|
item.methodPackageExpanded = true;
|
||||||
}
|
|
||||||
// 动态加载该方法对应的套餐明细
|
// 动态加载该方法对应的套餐明细
|
||||||
await loadMethodPackageDetails(item, method);
|
await loadMethodPackageDetails(item, method);
|
||||||
} else {
|
} else {
|
||||||
@@ -1587,11 +1569,13 @@ async function loadMethodPackageDetails(item, method) {
|
|||||||
item.methodPackageLoading = true;
|
item.methodPackageLoading = true;
|
||||||
item.methodPackageDetails = [];
|
item.methodPackageDetails = [];
|
||||||
try {
|
try {
|
||||||
if (!method.packageName) {
|
let packageId = method.packageId;
|
||||||
|
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)) {
|
||||||
@@ -1601,22 +1585,27 @@ async function loadMethodPackageDetails(item, method) {
|
|||||||
item.methodPackageLoading = false;
|
item.methodPackageLoading = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const packageId = packages[0].id;
|
packageId = packages[0].id;
|
||||||
|
method.packageId = packageId;
|
||||||
|
}
|
||||||
// 查询套餐明细
|
// 查询套餐明细
|
||||||
const detailRes = await request({
|
const detailRes = await request({
|
||||||
url: `/system/package/${packageId}/details`,
|
url: `/system/check-type/package/${packageId}/details`,
|
||||||
method: 'get'
|
method: 'get'
|
||||||
});
|
});
|
||||||
if (detailRes.code === 200 && detailRes.data) {
|
const detailList = parsePackageDetailsPayload(detailRes);
|
||||||
item.methodPackageDetails = detailRes.data.map(d => ({
|
if (detailList.length > 0) {
|
||||||
|
const mapped = detailList.map(d => ({
|
||||||
id: d.id,
|
id: d.id,
|
||||||
name: d.itemName || d.name,
|
name: d.name || d.itemName,
|
||||||
quantity: d.quantity || 1,
|
quantity: d.quantity || 1,
|
||||||
unit: d.unit || '次',
|
unit: d.unit || '次',
|
||||||
price: d.unitPrice || d.price || 0,
|
price: d.price ?? d.unitPrice ?? d.itemPrice ?? 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);
|
||||||
@@ -1630,21 +1619,19 @@ 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.packageDetailsDisplay = undefined;
|
row.methodPackageDetails = [];
|
||||||
const carrier = getPackageCarrier(row);
|
|
||||||
if (carrier) {
|
|
||||||
carrier.packageDetails = undefined;
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
row.expanded = !!getPackageCarrier(row)?.packageId;
|
row.expanded = shouldShowPackageBody(row);
|
||||||
if (getPackageCarrier(row)?.packageId) {
|
row.itemPackageExpanded = true;
|
||||||
|
row.methodPackageExpanded = true;
|
||||||
|
if (hasItemPackage(row)) {
|
||||||
await loadPackageDetailsForItem(row);
|
await loadPackageDetailsForItem(row);
|
||||||
}
|
}
|
||||||
|
if (val?.packageId || val?.packageName) {
|
||||||
|
await loadMethodPackageDetails(row, val);
|
||||||
|
}
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
form.totalAmount = totalAmountCalc.value;
|
form.totalAmount = totalAmountCalc.value;
|
||||||
});
|
});
|
||||||
@@ -1794,7 +1781,7 @@ defineExpose({ getList });
|
|||||||
|
|
||||||
/* 右:分类面板 */
|
/* 右:分类面板 */
|
||||||
.category-panel {
|
.category-panel {
|
||||||
width: 420px;
|
width: 560px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
background: #fff;
|
background: #fff;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
@@ -1951,9 +1938,9 @@ defineExpose({ getList });
|
|||||||
/* 已选择 tags */
|
/* 已选择 tags */
|
||||||
/* 已选择:加宽,避免套餐明细挤成一团 */
|
/* 已选择:加宽,避免套餐明细挤成一团 */
|
||||||
.selected-panel {
|
.selected-panel {
|
||||||
width: 220px;
|
width: 260px;
|
||||||
min-width: 200px;
|
min-width: 240px;
|
||||||
max-width: 280px;
|
max-width: 320px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -2013,9 +2000,8 @@ defineExpose({ getList });
|
|||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #303133;
|
color: #303133;
|
||||||
overflow: hidden;
|
line-height: 1.4;
|
||||||
text-overflow: ellipsis;
|
word-break: break-word;
|
||||||
white-space: nowrap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.card-price {
|
.card-price {
|
||||||
@@ -2030,12 +2016,11 @@ defineExpose({ getList });
|
|||||||
color: #909399;
|
color: #909399;
|
||||||
transition: transform 0.2s ease;
|
transition: transform 0.2s ease;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
transition: transform 0.2s;
|
transform: rotate(-90deg);
|
||||||
transform: rotate(0deg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.expand-icon.expanded {
|
.expand-icon.expanded {
|
||||||
transform: rotate(90deg);
|
transform: rotate(0deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Bug #428修复: 套餐明细列表样式 */
|
/* Bug #428修复: 套餐明细列表样式 */
|
||||||
@@ -2078,6 +2063,55 @@ 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;
|
||||||
|
|||||||
@@ -883,7 +883,7 @@ const initData = async () => {
|
|||||||
formData.visitNo = props.patientInfo.busNo || ''
|
formData.visitNo = props.patientInfo.busNo || ''
|
||||||
formData.patientId = props.patientInfo.patientId || ''
|
formData.patientId = props.patientInfo.patientId || ''
|
||||||
formData.patientName = props.patientInfo.patientName || ''
|
formData.patientName = props.patientInfo.patientName || ''
|
||||||
formData.medicalrecordNumber = props.patientInfo.identifierNo || ''
|
formData.medicalrecordNumber = props.patientInfo.busNo || ''
|
||||||
formData.applyDepartment = props.patientInfo.organizationName || ''
|
formData.applyDepartment = props.patientInfo.organizationName || ''
|
||||||
formData.applyDocName = userNickName.value || userName.value || ''
|
formData.applyDocName = userNickName.value || userName.value || ''
|
||||||
formData.applyDocCode = userId.value || ''
|
formData.applyDocCode = userId.value || ''
|
||||||
@@ -893,6 +893,9 @@ const initData = async () => {
|
|||||||
formData.applyOrganizationId = props.patientInfo.orgId || ''
|
formData.applyOrganizationId = props.patientInfo.orgId || ''
|
||||||
formData.encounterId = props.patientInfo.encounterId
|
formData.encounterId = props.patientInfo.encounterId
|
||||||
|
|
||||||
|
// 执行时间默认填充当前系统时间
|
||||||
|
formData.executeTime = formatDateTime(new Date())
|
||||||
|
|
||||||
// 申请单号在保存时由后端生成,此处显示"自动生成"
|
// 申请单号在保存时由后端生成,此处显示"自动生成"
|
||||||
formData.applyNo = '自动生成'
|
formData.applyNo = '自动生成'
|
||||||
// 申请日期实时更新(启动定时器)
|
// 申请日期实时更新(启动定时器)
|
||||||
@@ -1185,9 +1188,9 @@ const loadCategoryItems = async (categoryKey, loadMore = false) => {
|
|||||||
|
|
||||||
// 映射数据格式(从检验项目维护页面的数据结构映射)
|
// 映射数据格式(从检验项目维护页面的数据结构映射)
|
||||||
const mappedItems = records.map(item => {
|
const mappedItems = records.map(item => {
|
||||||
// 套餐项目处理:套餐项目使用套餐金额,普通项目使用零售价
|
// 套餐项目处理:需同时满足 feePackageId 有效且 packageName 非空
|
||||||
// BugFix#404: 增加对空字符串的判断,避免空字符串被误认为有效套餐ID
|
// BugFix#556: 增加 packageName 联合判断,避免普通项目因 feePackageId 有值被误标为套餐
|
||||||
const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null'
|
const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null' && item.packageName
|
||||||
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))
|
||||||
@@ -1523,7 +1526,7 @@ const resetForm = async () => {
|
|||||||
applicationId: null,
|
applicationId: null,
|
||||||
applyOrganizationId: props.patientInfo.orgId || '',
|
applyOrganizationId: props.patientInfo.orgId || '',
|
||||||
patientName: props.patientInfo.patientName || '',
|
patientName: props.patientInfo.patientName || '',
|
||||||
medicalrecordNumber: props.patientInfo.identifierNo || '',
|
medicalrecordNumber: props.patientInfo.busNo || '',
|
||||||
natureofCost: 'self',
|
natureofCost: 'self',
|
||||||
applyTime: '', // 申请日期由定时器实时更新
|
applyTime: '', // 申请日期由定时器实时更新
|
||||||
applyDepartment: props.patientInfo.organizationName || '',
|
applyDepartment: props.patientInfo.organizationName || '',
|
||||||
@@ -1547,7 +1550,7 @@ const resetForm = async () => {
|
|||||||
visitNo: '',
|
visitNo: '',
|
||||||
specimenName: '血液',
|
specimenName: '血液',
|
||||||
encounterId: props.patientInfo.encounterId || '',
|
encounterId: props.patientInfo.encounterId || '',
|
||||||
executeTime: null,
|
executeTime: formatDateTime(new Date()),
|
||||||
applicationType: 0,
|
applicationType: 0,
|
||||||
})
|
})
|
||||||
selectedInspectionItems.value = []
|
selectedInspectionItems.value = []
|
||||||
@@ -1597,7 +1600,7 @@ const handleSave = () => {
|
|||||||
// 修复【#406】:保存前尝试从 props 同步患者信息,避免因加载时序导致信息缺失
|
// 修复【#406】:保存前尝试从 props 同步患者信息,避免因加载时序导致信息缺失
|
||||||
if ((!formData.patientName?.trim() || !formData.medicalrecordNumber?.trim()) && props.patientInfo && props.patientInfo.encounterId) {
|
if ((!formData.patientName?.trim() || !formData.medicalrecordNumber?.trim()) && props.patientInfo && props.patientInfo.encounterId) {
|
||||||
formData.patientName = props.patientInfo.patientName || ''
|
formData.patientName = props.patientInfo.patientName || ''
|
||||||
formData.medicalrecordNumber = props.patientInfo.identifierNo || ''
|
formData.medicalrecordNumber = props.patientInfo.busNo || ''
|
||||||
formData.encounterId = props.patientInfo.encounterId || ''
|
formData.encounterId = props.patientInfo.encounterId || ''
|
||||||
formData.visitNo = props.patientInfo.busNo || ''
|
formData.visitNo = props.patientInfo.busNo || ''
|
||||||
formData.patientId = props.patientInfo.patientId || ''
|
formData.patientId = props.patientInfo.patientId || ''
|
||||||
@@ -1997,7 +2000,7 @@ const loadApplicationToForm = async (row) => {
|
|||||||
// Bug #387修复: 套餐项目默认展开,并自动加载明细
|
// Bug #387修复: 套餐项目默认展开,并自动加载明细
|
||||||
selectedInspectionItems.value = detail.labApplyItemList.map(item => {
|
selectedInspectionItems.value = detail.labApplyItemList.map(item => {
|
||||||
const itemId = item.activityId || item.itemId || item.id || Math.random().toString(36).substring(2, 11)
|
const itemId = item.activityId || item.itemId || item.id || Math.random().toString(36).substring(2, 11)
|
||||||
const isPackage = item.feePackageId != null || item.itemName?.includes('套餐')
|
const isPackage = item.feePackageId != null && item.feePackageId !== '' && item.feePackageId !== 'null' && item.packageName
|
||||||
|
|
||||||
return {
|
return {
|
||||||
itemId: itemId,
|
itemId: itemId,
|
||||||
|
|||||||
@@ -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="申请单名称" width="140">
|
<el-table-column label="申请单名称" min-width="140">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<span>{{ buildApplicationName(scope.row) }}</span>
|
<span>{{ buildApplicationName(scope.row) }}</span>
|
||||||
</template>
|
</template>
|
||||||
@@ -179,14 +179,11 @@
|
|||||||
|
|
||||||
<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">
|
||||||
<el-descriptions-item
|
<template v-for="(value, key) in descJsonData" :key="key">
|
||||||
v-for="key in orderedDescFieldKeys"
|
<el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)">
|
||||||
:key="key"
|
{{ transformField(key, value) || '-' }}
|
||||||
v-if="descJsonData[key] != null && descJsonData[key] !== ''"
|
|
||||||
:label="getFieldLabel(key)"
|
|
||||||
>
|
|
||||||
{{ transformField(key, descJsonData[key]) || '-' }}
|
|
||||||
</el-descriptions-item>
|
</el-descriptions-item>
|
||||||
|
</template>
|
||||||
</el-descriptions>
|
</el-descriptions>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -447,11 +444,9 @@ const buildApplicationName = (row) => {
|
|||||||
if (!details || details.length === 0) {
|
if (!details || details.length === 0) {
|
||||||
return row.name || '-';
|
return row.name || '-';
|
||||||
}
|
}
|
||||||
if (details.length === 1) {
|
const names = details.map(d => d.adviceName).filter(Boolean);
|
||||||
return details[0].adviceName || row.name || '-';
|
if (names.length === 0) return row.name || '-';
|
||||||
}
|
return names.join(' + ');
|
||||||
const first = details[0];
|
|
||||||
return `${first.adviceName || ''}等${details.length}项`;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -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 {getOrgList} from '@/views/doctorstation/components/api.js';
|
import {getDepartmentList} from '@/api/public.js';
|
||||||
|
|
||||||
const { proxy } = getCurrentInstance();
|
const { proxy } = getCurrentInstance();
|
||||||
|
|
||||||
@@ -182,25 +182,32 @@ const hasMatchedFields = computed(() => {
|
|||||||
|
|
||||||
/** 查询科室 */
|
/** 查询科室 */
|
||||||
const getLocationInfo = async () => {
|
const getLocationInfo = async () => {
|
||||||
const res = await getOrgList();
|
const res = await getDepartmentList();
|
||||||
orgOptions.value = res.data.records;
|
orgOptions.value = res.data || [];
|
||||||
};
|
};
|
||||||
|
|
||||||
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 obj = orgOptions.value[index];
|
const findInList = (list) => {
|
||||||
if (obj.id == targetDepartment) {
|
for (const node of list) {
|
||||||
name = obj.name;
|
if (String(node.id) === String(targetDepartment)) {
|
||||||
|
name = node.name;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
const subObjArray = obj['children'];
|
// 树形结构:递归查找 children
|
||||||
for (let index = 0; index < subObjArray.length; index++) {
|
if (node.children && node.children.length > 0) {
|
||||||
const item = subObjArray[index];
|
if (findInList(node.children)) {
|
||||||
if (item.id == targetDepartment) {
|
return true;
|
||||||
name = item.name;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
findInList(orgOptions.value);
|
||||||
return name;
|
return name;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -41,7 +41,9 @@
|
|||||||
<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="6" />
|
<el-option label="已采证" value="4" />
|
||||||
|
<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>
|
||||||
@@ -91,7 +93,15 @@
|
|||||||
<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">
|
||||||
<span>{{ parseBillStatus(scope.row.billStatus ?? scope.row.status) }}</span>
|
<el-tag
|
||||||
|
: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">
|
||||||
@@ -107,16 +117,16 @@
|
|||||||
<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="220">
|
<el-table-column label="操作" align="center" fixed="right" width="220">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<!-- 待签发(status=0或null/undefined):可修改、删除 -->
|
<!-- 待签发:可修改、删除 -->
|
||||||
<template v-if="!scope.row.status || 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>
|
||||||
<!-- 已签发(status=1):可撤回 -->
|
<!-- 已签发:可撤回 -->
|
||||||
<template v-else-if="scope.row.status == 1">
|
<template v-else-if="isIssuedStatus(scope.row)">
|
||||||
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
|
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
|
||||||
</template>
|
</template>
|
||||||
<!-- 已校对(2)、待接收(3)、已收样(4)、已出报告(6)、已作废(7):仅查看详情 -->
|
<!-- 已采证、已送检、报告已出、已作废:仅查看详情 -->
|
||||||
<template v-else>
|
<template v-else>
|
||||||
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
|
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
|
||||||
</template>
|
</template>
|
||||||
@@ -212,10 +222,10 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import {computed, getCurrentInstance, ref, watch} from 'vue';
|
import {computed, getCurrentInstance, nextTick, 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} from './api';
|
import {getInspection, deleteRequestForm, withdrawRequestForm, getProofResult} 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 {saveInspection} from '../order/applicationForm/api.js';
|
import {saveInspection} from '../order/applicationForm/api.js';
|
||||||
@@ -270,7 +280,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);
|
tableData.value = list.filter(Boolean).sort(sortByCreateTimeDesc);
|
||||||
} else {
|
} else {
|
||||||
tableData.value = [];
|
tableData.value = [];
|
||||||
}
|
}
|
||||||
@@ -329,19 +339,95 @@ 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': '已采证',
|
||||||
'6': '已出报告',
|
'5': '已送检',
|
||||||
|
'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 isIssuedStatus = (row) => String(getBillStatus(row)) === '1';
|
||||||
|
|
||||||
|
const isReportStatus = (row) => ['6', '8'].includes(String(getBillStatus(row)));
|
||||||
|
|
||||||
|
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字符串
|
||||||
@@ -443,7 +529,13 @@ const handleViewDetail = async (row) => {
|
|||||||
if (row.descJson) {
|
if (row.descJson) {
|
||||||
try {
|
try {
|
||||||
const obj = JSON.parse(row.descJson);
|
const obj = JSON.parse(row.descJson);
|
||||||
obj.targetDepartment = recursionFun(obj.targetDepartment);
|
// 将发往科室 ID 转换为名称
|
||||||
|
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 = '急诊';
|
||||||
@@ -462,12 +554,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?.();
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -646,6 +738,10 @@ defineExpose({
|
|||||||
animation: rotating 2s linear infinite;
|
animation: rotating 2s linear infinite;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.report-status-tag {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
@keyframes rotating {
|
@keyframes rotating {
|
||||||
0% {
|
0% {
|
||||||
transform: rotate(0deg);
|
transform: rotate(0deg);
|
||||||
|
|||||||
@@ -134,10 +134,10 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup name="LaboratoryTests">
|
<script setup name="LaboratoryTests">
|
||||||
import {getCurrentInstance, onMounted, reactive, ref, watch, computed} from 'vue';
|
import {getCurrentInstance, nextTick, onMounted, reactive, ref, watch, computed} from 'vue';
|
||||||
import {patientInfo} from '../../../store/patient.js';
|
import {patientInfo} from '../../../store/patient.js';
|
||||||
import {getApplicationList, saveInspection} from './api';
|
import {getApplicationList, saveInspection} from './api';
|
||||||
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';
|
||||||
|
|
||||||
@@ -168,6 +168,7 @@ 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) => {
|
||||||
@@ -207,6 +208,8 @@ 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;
|
||||||
|
// 检验项目列表为异步加载,编辑回显必须在数据就绪后执行,否则已选区一直为空
|
||||||
|
applyEditTransferSelection()
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
proxy.$message.error('获取检验项目列表失败');
|
proxy.$message.error('获取检验项目列表失败');
|
||||||
applicationListAll.value = [];
|
applicationListAll.value = [];
|
||||||
@@ -242,6 +245,8 @@ const getList = async () => {
|
|||||||
const handleSearch = () => {
|
const handleSearch = () => {
|
||||||
// 搜索时保持已选中的项目不受影响
|
// 搜索时保持已选中的项目不受影响
|
||||||
};
|
};
|
||||||
|
// 编辑初始化标志:避免 applyEditTransferSelection 设置 transferValue 时触发 projectWithDepartment 覆盖 descJson 中的科室值
|
||||||
|
const isInitializing = ref(false);
|
||||||
const transferValue = ref([]);
|
const transferValue = ref([]);
|
||||||
const form = reactive({
|
const form = reactive({
|
||||||
// categoryType: '', // 项目类别
|
// categoryType: '', // 项目类别
|
||||||
@@ -259,7 +264,31 @@ 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();
|
||||||
});
|
});
|
||||||
/**
|
/**
|
||||||
@@ -288,8 +317,9 @@ const projectWithDepartment = (selectProjectIds, type) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// 保存用户手动选择的发往科室(提交时需要保留)
|
// 保存用户手动选择/回显的发往科室(提交、编辑回显时需要保留)
|
||||||
const manualDept = type === 2 ? form.targetDepartment : '';
|
const manualDept =
|
||||||
|
type === 2 || (isEditMode.value && form.targetDepartment) ? form.targetDepartment : '';
|
||||||
// 清空科室
|
// 清空科室
|
||||||
form.targetDepartment = '';
|
form.targetDepartment = '';
|
||||||
if (arr.length > 0) {
|
if (arr.length > 0) {
|
||||||
@@ -309,8 +339,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 && manualDept) {
|
if ((type === 2 || isEditMode.value) && manualDept) {
|
||||||
form.targetDepartment = manualDept;
|
form.targetDepartment = resolveTargetDepartmentId(manualDept);
|
||||||
isRelease = true;
|
isRelease = true;
|
||||||
} else if (type === 2 && !manualDept) {
|
} else if (type === 2 && !manualDept) {
|
||||||
// 提交时用户未手动选择科室,才提示错误
|
// 提交时用户未手动选择科室,才提示错误
|
||||||
@@ -326,10 +356,10 @@ const projectWithDepartment = (selectProjectIds, type) => {
|
|||||||
}
|
}
|
||||||
if (findItem && isRelease) {
|
if (findItem && isRelease) {
|
||||||
// 提交时若用户已选「发往科室」,不得用项目默认执行科室覆盖
|
// 提交时若用户已选「发往科室」,不得用项目默认执行科室覆盖
|
||||||
if (type === 2 && manualDept) {
|
if ((type === 2 || isEditMode.value) && manualDept) {
|
||||||
form.targetDepartment = manualDept;
|
form.targetDepartment = resolveTargetDepartmentId(manualDept);
|
||||||
} else {
|
} else {
|
||||||
form.targetDepartment = findItem.id;
|
form.targetDepartment = String(findItem.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -339,49 +369,93 @@ const projectWithDepartment = (selectProjectIds, type) => {
|
|||||||
watch(
|
watch(
|
||||||
() => transferValue.value,
|
() => transferValue.value,
|
||||||
(newValue) => {
|
(newValue) => {
|
||||||
|
if (skipDeptAutoFill.value) return;
|
||||||
|
if (isInitializing.value) return;
|
||||||
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 }
|
{ immediate: true, deep: true }
|
||||||
);
|
)
|
||||||
|
|
||||||
// 编辑模式下,applicationListAll 加载完成后重新回显已选项目
|
watch(
|
||||||
|
() => orgOptions.value,
|
||||||
|
() => {
|
||||||
|
applyTargetDepartmentEcho()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// 编辑模式下,项目字典加载完成后重新回显已选项目
|
||||||
watch(
|
watch(
|
||||||
() => applicationListAll.value,
|
() => applicationListAll.value,
|
||||||
() => {
|
() => {
|
||||||
@@ -398,7 +472,10 @@ watch(
|
|||||||
selectedIds.push(matched.adviceDefinitionId);
|
selectedIds.push(matched.adviceDefinitionId);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
isInitializing.value = true;
|
||||||
transferValue.value = selectedIds;
|
transferValue.value = selectedIds;
|
||||||
|
isInitializing.value = false;
|
||||||
|
applyEditTransferSelection();
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -451,9 +528,9 @@ const submit = () => {
|
|||||||
};
|
};
|
||||||
/** 查询科室 */
|
/** 查询科室 */
|
||||||
const getLocationInfo = () => {
|
const getLocationInfo = () => {
|
||||||
getOrgList().then((res) => {
|
return getDepartmentList().then((res) => {
|
||||||
orgOptions.value = res.data.records;
|
orgOptions.value = normalizeOrgTreeIds(res.data || []);
|
||||||
console.log('科室========>', JSON.stringify(orgOptions.value));
|
applyTargetDepartmentEcho();
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
// 获取诊断目录
|
// 获取诊断目录
|
||||||
|
|||||||
@@ -428,11 +428,52 @@ 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 更新完成后再设置值
|
||||||
|
nextTick(() => {
|
||||||
projectWithDepartment(newValue);
|
projectWithDepartment(newValue);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const getPriorityCode = () => {
|
const getPriorityCode = () => {
|
||||||
|
|||||||
@@ -232,64 +232,21 @@ onMounted(() => {
|
|||||||
* type(1:watch监听类型 2:点击保存类型)
|
* type(1:watch监听类型 2:点击保存类型)
|
||||||
* selectProjectIds(选中项目的id数组)
|
* selectProjectIds(选中项目的id数组)
|
||||||
* */
|
* */
|
||||||
const projectWithDepartment = (selectProjectIds, type) => {
|
const projectWithDepartment = (selectProjectIds) => {
|
||||||
//1.获取选中的项目 2.判断项目的执行科室是否相同 3.判断执行科室是否配置 4.将项目的执行科室复值到执行科室下拉选位置
|
if (!selectProjectIds || selectProjectIds.length === 0) {
|
||||||
let isRelease = true;
|
|
||||||
// 选中项目的数组
|
|
||||||
const arr = [];
|
|
||||||
// 根据选中的项目id查找对应的项目
|
|
||||||
selectProjectIds.forEach((element) => {
|
|
||||||
const searchData = applicationList.value.find((item) => {
|
|
||||||
return element == item.adviceDefinitionId;
|
|
||||||
});
|
|
||||||
arr.push(searchData);
|
|
||||||
});
|
|
||||||
// 清空科室
|
|
||||||
form.targetDepartment = '';
|
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(
|
watch(() => transferValue.value, (newValue) => {
|
||||||
() => transferValue.value,
|
projectWithDepartment(newValue);
|
||||||
(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 (!projectWithDepartment(transferValue.value, 2)) {
|
if (!form.targetDepartment) {
|
||||||
return;
|
return proxy.$message.error('请选择发往科室');
|
||||||
}
|
}
|
||||||
let applicationListAllFilter = applicationListAll.value.filter((item) => {
|
let applicationListAllFilter = applicationListAll.value.filter((item) => {
|
||||||
return transferValue.value.includes(item.adviceDefinitionId);
|
return transferValue.value.includes(item.adviceDefinitionId);
|
||||||
@@ -302,7 +259,7 @@ const submit = () => {
|
|||||||
unitCode: item.unitCode,
|
unitCode: item.unitCode,
|
||||||
unitPrice: item.price,
|
unitPrice: item.price,
|
||||||
totalPrice: item.price,
|
totalPrice: item.price,
|
||||||
positionId: item.positionId,
|
positionId: form.targetDepartment || item.positionId, // 用户手动选择的发往科室优先于项目默认执行科室
|
||||||
definitionId: item.chargeItemDefinitionId,
|
definitionId: item.chargeItemDefinitionId,
|
||||||
accountId: patientInfo.value.accountId,
|
accountId: patientInfo.value.accountId,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -429,6 +429,8 @@ 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([]);
|
||||||
@@ -804,7 +806,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')) {
|
||||||
@@ -815,14 +817,18 @@ function clickRowDb(row, column, event) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
row.showPopover = false;
|
row.showPopover = false;
|
||||||
// “待签发(已保存 requestId存在)”不允许再编辑;仅“待保存(无requestId)”允许编辑
|
if (row.statusEnum == 1) {
|
||||||
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;
|
||||||
|
if (index !== -1) {
|
||||||
prescriptionList.value[index] = row;
|
prescriptionList.value[index] = row;
|
||||||
|
}
|
||||||
expandOrder.value = [row.uniqueKey];
|
expandOrder.value = [row.uniqueKey];
|
||||||
|
} else {
|
||||||
|
proxy.$modal.msgWarning('仅待保存或待签发医嘱允许编辑');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1195,7 +1201,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 = JSON.parse(item.contentJson);
|
const parsedContent = item.contentJson ? JSON.parse(item.contentJson) : {};
|
||||||
return {
|
return {
|
||||||
...parsedContent,
|
...parsedContent,
|
||||||
adviceType: item.adviceType,
|
adviceType: item.adviceType,
|
||||||
@@ -1393,7 +1399,9 @@ function handleSaveSign(row, index) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
if (prescriptionList.value[0].adviceName) {
|
// 仅通过【新增】按钮创建的医嘱保存后才自动添加下一行空医嘱
|
||||||
|
// 双击编辑已有"待保存"医嘱保存时,不应自动添加空行
|
||||||
|
if (isAdding.value && prescriptionList.value[0].adviceName) {
|
||||||
handleAddPrescription();
|
handleAddPrescription();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -348,13 +348,14 @@ const adviceTypeList = computed(() => {
|
|||||||
return val === 3 || val === 4;
|
return val === 3 || val === 4;
|
||||||
}).map(item => ({
|
}).map(item => ({
|
||||||
label: item.label,
|
label: item.label,
|
||||||
value: parseInt(item.value)
|
// drord_doctor_type 中耗材是 4,但 /advice-base-info 后端耗材类型是 2
|
||||||
|
value: parseInt(item.value) === 4 ? 2 : parseInt(item.value)
|
||||||
}));
|
}));
|
||||||
return [...filtered, { label: '全部', value: '' }];
|
return [...filtered, { label: '全部', value: '' }];
|
||||||
}
|
}
|
||||||
// 默认值
|
// 默认值
|
||||||
return [
|
return [
|
||||||
{ label: '耗材', value: 4 },
|
{ label: '耗材', value: 2 },
|
||||||
{ label: '诊疗', value: 3 },
|
{ label: '诊疗', value: 3 },
|
||||||
{ label: '全部', value: '' },
|
{ label: '全部', value: '' },
|
||||||
];
|
];
|
||||||
@@ -373,7 +374,8 @@ const filterKeywords = ref({});
|
|||||||
const queryParams = ref({
|
const queryParams = ref({
|
||||||
pageSize: 100,
|
pageSize: 100,
|
||||||
pageNum: 1,
|
pageNum: 1,
|
||||||
adviceTypes: '2,3',
|
// 默认加载全部类型(药品1+耗材2+诊疗3)
|
||||||
|
adviceTypes: [1, 2, 3],
|
||||||
});
|
});
|
||||||
/**
|
/**
|
||||||
* 医嘱提交数据模型
|
* 医嘱提交数据模型
|
||||||
@@ -482,8 +484,9 @@ 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();
|
||||||
@@ -555,12 +558,28 @@ function loadDepartmentOptions() {
|
|||||||
function getAdviceBaseInfos() {
|
function getAdviceBaseInfos() {
|
||||||
adviceLoading.value = true;
|
adviceLoading.value = true;
|
||||||
queryParams.value.searchKey = searchText.value;
|
queryParams.value.searchKey = searchText.value;
|
||||||
queryParams.value.adviceType = adviceType.value;
|
// 字典值(3=诊疗,4=耗材)映射为后端adviceType(2=耗材,3=诊疗)
|
||||||
|
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) => {
|
||||||
AdviceBaseInfoList.value = res.data?.records || [];
|
const list = 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;
|
||||||
@@ -605,6 +624,12 @@ 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 = [];
|
||||||
// 大单位:优先用 code,code 缺失时用字典文本兜底
|
// 大单位:优先用 code,code 缺失时用字典文本兜底
|
||||||
|
|||||||
@@ -262,7 +262,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import {computed, nextTick, onMounted, ref, watch} from 'vue';
|
import {nextTick, onMounted, ref} 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,24 +381,6 @@ 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位小数)
|
||||||
@@ -447,16 +429,19 @@ const handleTableSelectionChange = (index, val) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 日期Tab切换
|
* 按 Tab 同步日期范围(避免 date-picker @change 与 Tab v-model 互相覆盖)
|
||||||
* @param {Object} tab - 标签页
|
* @param {string} rangeType - today | yesterday | custom
|
||||||
*/
|
*/
|
||||||
const handleDateTabClick = (tab) => {
|
const applyDateRangeByTab = (rangeType) => {
|
||||||
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');
|
||||||
|
|
||||||
switch (safeGet(tab, 'paneName')) {
|
syncingDateFromTab.value = true;
|
||||||
|
dateRange.value = rangeType;
|
||||||
|
|
||||||
|
switch (rangeType) {
|
||||||
case 'today':
|
case 'today':
|
||||||
dateRangeValue.value = [format(today), format(today)];
|
dateRangeValue.value = [format(today), format(today)];
|
||||||
break;
|
break;
|
||||||
@@ -464,27 +449,54 @@ const handleDateTabClick = (tab) => {
|
|||||||
dateRangeValue.value = [format(yesterday), format(yesterday)];
|
dateRangeValue.value = [format(yesterday), format(yesterday)];
|
||||||
break;
|
break;
|
||||||
case 'custom':
|
case 'custom':
|
||||||
if (!dateRangeValue.value.length) {
|
if (safeArray(dateRangeValue.value).length < 2) {
|
||||||
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) {
|
if (dateVal.length !== 2) return;
|
||||||
dateRange.value = 'custom';
|
|
||||||
const start = new Date(dateVal[0]);
|
const start = new Date(dateVal[0]);
|
||||||
const end = new Date(dateVal[1]);
|
const end = new Date(dateVal[1]);
|
||||||
if (start > end) {
|
if (start > end) {
|
||||||
ElMessage.warning('开始日期不能晚于结束日期');
|
ElMessage.warning('开始日期不能晚于结束日期');
|
||||||
|
syncingDateFromTab.value = true;
|
||||||
dateRangeValue.value = [dateVal[1], dateVal[0]];
|
dateRangeValue.value = [dateVal[1], dateVal[0]];
|
||||||
|
nextTick(() => {
|
||||||
|
syncingDateFromTab.value = false;
|
||||||
|
});
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (dateRange.value !== 'custom') {
|
||||||
|
dateRange.value = 'custom';
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -714,24 +726,7 @@ 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>
|
||||||
|
|
||||||
|
|||||||
@@ -142,6 +142,13 @@
|
|||||||
</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>
|
||||||
@@ -152,7 +159,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {ref, computed} from 'vue';
|
import {ref, computed, getCurrentInstance} 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';
|
||||||
@@ -165,6 +172,19 @@ 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);
|
||||||
|
|||||||
@@ -386,7 +386,7 @@
|
|||||||
:disabled="viewStatus == 'view'"
|
:disabled="viewStatus == 'view'"
|
||||||
v-model="scope.row.itemQuantityDisplay"
|
v-model="scope.row.itemQuantityDisplay"
|
||||||
placeholder=""
|
placeholder=""
|
||||||
@change="(value) => handleItemQuantityChange(scope.row, value)"
|
@input="(value) => handleItemQuantityChange(scope.row, value)"
|
||||||
:class="{ 'error-border': scope.row.error }"
|
:class="{ 'error-border': scope.row.error }"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -971,6 +971,10 @@ 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() {
|
||||||
@@ -1262,128 +1266,108 @@ function editBatchTransfer(index) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function handleSave(row, index) {
|
function handleSave(row, index) {
|
||||||
rowList.value = [];
|
// 过滤出未保存的行,已保存的行不重复提交
|
||||||
if (route.query.supplyBusNo) {
|
const listToCheck = route.query.supplyBusNo
|
||||||
// 编辑
|
? forms.purchaseinventoryList
|
||||||
forms.purchaseinventoryList.map((row, index) => {
|
: form.purchaseinventoryList;
|
||||||
if (row) {
|
const unsavedList = listToCheck.filter(item => !item.isSave);
|
||||||
proxy.$refs['receiptHeaderRef'].validate((valid) => {
|
if (unsavedList.length === 0) {
|
||||||
if (valid) {
|
proxy.$modal.msgWarning('所有行均已保存,无需重复提交');
|
||||||
proxy.$refs['formRef'].validate((valid) => {
|
return;
|
||||||
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) {
|
proxy.$refs['receiptHeaderRef'].validate((headerValid) => {
|
||||||
row.unitCode = row.unitList.minUnitCode;
|
if (!headerValid) return;
|
||||||
} else {
|
|
||||||
row.unitCode = row.unitList.unitCode;
|
// 逐行校验(避免异步回调导致重复提交)
|
||||||
row.unitCode_dictText = row.unitList.unitCode_dictText;
|
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 (row.unitCode == row.unitList.unitCode) {
|
if (!rowValid) {
|
||||||
row.unitCode_dictText = row.unitList.unitCode_dictText;
|
proxy.$modal.msgWarning('第' + (i + 1) + '行数据不完整,请检查');
|
||||||
} 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;
|
return;
|
||||||
}
|
}
|
||||||
forms.purchaseinventoryList[index].totalPrice =
|
|
||||||
forms.purchaseinventoryList[index].price * forms.purchaseinventoryList[index].itemQuantity;
|
// 单价校验
|
||||||
rowList.value.push(JSON.parse(JSON.stringify(row)));
|
if (!r.price || r.price <= 0) {
|
||||||
if (
|
proxy.$modal.msgWarning('第' + (i + 1) + '行调拨单价不能为空或为0');
|
||||||
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;
|
return;
|
||||||
}
|
}
|
||||||
form.purchaseinventoryList[index].totalPrice =
|
|
||||||
form.purchaseinventoryList[index].price * form.purchaseinventoryList[index].itemQuantity;
|
// 单位处理
|
||||||
rowList.value.push(JSON.parse(JSON.stringify(rows)));
|
const rowData = route.query.supplyBusNo
|
||||||
if (
|
? JSON.parse(JSON.stringify(forms.purchaseinventoryList[i]))
|
||||||
rowList._rawValue &&
|
: JSON.parse(JSON.stringify(r));
|
||||||
rowList._rawValue.length == form.purchaseinventoryList.length
|
delete rowData.itemMaxQuantity;
|
||||||
) {
|
|
||||||
addTransferProducts(rowList._rawValue);
|
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) => {
|
||||||
form.purchaseinventoryList[index].id = res.data[index];
|
// 只有未保存的行才会拿到新 id(和提交顺序一致)
|
||||||
|
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) => {
|
||||||
forms.purchaseinventoryList[index].id = res.data[index];
|
if (!row.isSave && res.data[newIdIdx]) {
|
||||||
|
forms.purchaseinventoryList[index].id = res.data[newIdIdx];
|
||||||
|
newIdIdx++;
|
||||||
|
}
|
||||||
forms.purchaseinventoryList[index].isSave = true;
|
forms.purchaseinventoryList[index].isSave = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1067,15 +1067,6 @@ 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) // 🔧 新增:签名状态,用于保持按钮名称一致性
|
||||||
|
|
||||||
@@ -1499,9 +1490,6 @@ async function closeChargeDialog() {
|
|||||||
chargeSurgeryInfo.value = {}
|
chargeSurgeryInfo.value = {}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 🔧 新增:标志位,用于区分是"打开"还是"刷新"
|
|
||||||
const isRefreshAction = ref(false)
|
|
||||||
|
|
||||||
// 处理医嘱按钮点击事件
|
// 处理医嘱按钮点击事件
|
||||||
function handleMedicalAdvice(row) {
|
function handleMedicalAdvice(row) {
|
||||||
// 如果没有传入行数据,使用选中的行
|
// 如果没有传入行数据,使用选中的行
|
||||||
@@ -1529,31 +1517,7 @@ 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 = []
|
||||||
@@ -1565,41 +1529,40 @@ function handleMedicalAdvice(row) {
|
|||||||
temporaryMedicalLoading.value = true // 🔧 新增:开始加载
|
temporaryMedicalLoading.value = true // 🔧 新增:开始加载
|
||||||
|
|
||||||
// 调用计费接口获取数据
|
// 调用计费接口获取数据
|
||||||
getPrescriptionList(row.visitId).then((res) => {
|
getPrescriptionList(row.visitId, 6, row.operCode).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;
|
||||||
// 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3)
|
// 只保留药品(1)和耗材(2),屏蔽诊疗(3)和手术(6)
|
||||||
if (item.adviceType !== 1) return false;
|
const at = Number(item.adviceType ?? item.advice_type);
|
||||||
|
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 的不应出现在"待生成"列表)
|
// 排除名称中包含手术/检查/诊疗关键词的非药品项目
|
||||||
if (item.requestId) return false;
|
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
||||||
|
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 (filteredItems.length > maxItems) {
|
if (draftItems.length > maxItems) {
|
||||||
ElMessage.warning(`待签发医嘱数量过多(${filteredItems.length}条),仅显示前${maxItems}条`)
|
ElMessage.warning(`待签发医嘱数量过多(${draftItems.length}条),仅显示前${maxItems}条`)
|
||||||
filteredItems.length = maxItems
|
draftItems.length = maxItems
|
||||||
}
|
}
|
||||||
|
|
||||||
// 将过滤后的数据转换为临时医嘱需要的格式 - 兼容驼峰和下划线命名
|
// === 待生成列表:statusEnum=1 草稿状态的项目 ===
|
||||||
// 对于从 adm_charge_item(计费项目表)查询来的项目,特殊处理
|
temporaryBillingMedicines.value = draftItems.map(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;
|
||||||
@@ -1646,69 +1609,65 @@ function handleMedicalAdvice(row) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
|
||||||
// 如果没有数据或接口调用失败,初始化空列表
|
|
||||||
temporaryBillingMedicines.value = []
|
|
||||||
}
|
|
||||||
|
|
||||||
// 将计费药品转换为临时医嘱数据
|
// === 已生成列表:statusEnum=2 已签发状态的项目,直接转为医嘱格式 ===
|
||||||
temporaryAdvices.value = temporaryBillingMedicines.value.map((medicine, index) => {
|
temporaryAdvices.value = activeItems.map((item, index) => {
|
||||||
// 解析规格中的数值和单位
|
try {
|
||||||
const specMatch = medicine.specification ? medicine.specification.match(/(\d+)(\D+)/) : null
|
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 specValue = specMatch ? parseInt(specMatch[1]) : 1
|
||||||
const specUnit = specMatch ? specMatch[2] : 'ml'
|
const specUnit = specMatch ? specMatch[2] : 'ml'
|
||||||
|
const dosage = specValue * (contentData.quantity || item.quantity || 1)
|
||||||
|
|
||||||
// 计算剂量 = 规格数值 × 数量
|
let usageCode = contentData.methodCode || 'iv'
|
||||||
const dosage = specValue * (medicine.quantity || 1)
|
let usageLabel = getUsageLabel(usageCode)
|
||||||
|
if (usageCode === 'iv') {
|
||||||
// 🔧 修复:优先从 contentJson 中读取已有的用法,如果没有则根据药品名称判断
|
if (medicineName.includes('注射液')) { usageCode = 'iv'; usageLabel = '静脉注射' }
|
||||||
let usageCode = 'iv' // 默认静脉注射编码
|
} else if (usageCode === 'po') {
|
||||||
let usageLabel = '静脉注射' // 默认显示名称
|
if (medicineName.includes('片') || medicineName.includes('胶囊')) { usageCode = 'po'; 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 {
|
return {
|
||||||
id: index + 1,
|
id: index + 1,
|
||||||
adviceName: medicine.medicineName || '',
|
adviceName: medicineName,
|
||||||
dosage: dosage,
|
dosage,
|
||||||
unit: specUnit,
|
unit: specUnit,
|
||||||
usage: usageCode, // 🔧 修复:保存的是编码
|
usage: usageCode,
|
||||||
usageLabel: usageLabel, // 🔧 新增:保存显示名称
|
usageLabel,
|
||||||
frequency: '临时',
|
frequency: '临时',
|
||||||
executeTime: new Date().toLocaleString('zh-CN'),
|
executeTime: new Date().toLocaleString('zh-CN'),
|
||||||
// 🔧 关键修复:确保 originalMedicine 中包含 encounterId,以便后续判断是否为同一患者
|
|
||||||
originalMedicine: {
|
originalMedicine: {
|
||||||
...medicine,
|
...item,
|
||||||
encounterId: row.visitId // 添加 encounterId 字段
|
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
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
} else {
|
||||||
|
temporaryBillingMedicines.value = []
|
||||||
|
temporaryAdvices.value = []
|
||||||
|
}
|
||||||
|
|
||||||
// 打开临时医嘱弹窗
|
// 打开临时医嘱弹窗
|
||||||
showTemporaryMedical.value = true
|
showTemporaryMedical.value = true
|
||||||
@@ -1734,11 +1693,6 @@ 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) {
|
||||||
@@ -1794,8 +1748,6 @@ 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 = []
|
||||||
@@ -1843,6 +1795,21 @@ 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) {
|
||||||
@@ -1850,18 +1817,72 @@ function handleQuoteBilling() {
|
|||||||
temporaryBillingMedicines.value = []
|
temporaryBillingMedicines.value = []
|
||||||
temporaryAdvices.value = []
|
temporaryAdvices.value = []
|
||||||
|
|
||||||
// 🔧 修复 Bug #445: 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3)
|
// 🔧 修复 Bug #445: 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3/6)
|
||||||
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
|
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
|
||||||
const filteredItems = res.data.filter(item => {
|
// 先提取已签发项目(statusEnum=2)填充已生成列表
|
||||||
// 匹配 encounterId
|
const activeItems = res.data.filter(item => {
|
||||||
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
||||||
// 只保留药品类型(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;
|
||||||
// 过滤掉名称为空的项目
|
if (item.statusEnum !== 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;
|
||||||
|
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 => {
|
||||||
|
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 !== 1) 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;
|
return true;
|
||||||
})
|
})
|
||||||
// 🔧 修复:限制返回数量,最多显示前100条,避免数据过多导致页面卡死
|
// 🔧 修复:限制返回数量,最多显示前100条,避免数据过多导致页面卡死
|
||||||
@@ -1874,7 +1895,6 @@ 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 {
|
||||||
@@ -1893,10 +1913,9 @@ 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.specification || item.volume || '',
|
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,
|
||||||
@@ -1912,89 +1931,6 @@ 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 {
|
||||||
|
|||||||
@@ -48,9 +48,12 @@
|
|||||||
<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="billingMedicines"
|
:data="displayBillingMedicines"
|
||||||
stripe
|
stripe
|
||||||
border
|
border
|
||||||
style="width: 100%;"
|
style="width: 100%;"
|
||||||
@@ -98,9 +101,12 @@
|
|||||||
<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="displayAdvices"
|
:data="displayAdvicesList"
|
||||||
stripe
|
stripe
|
||||||
border
|
border
|
||||||
style="width: 100%;"
|
style="width: 100%;"
|
||||||
@@ -150,7 +156,7 @@
|
|||||||
:disabled="allItemsSubmitted"
|
:disabled="allItemsSubmitted"
|
||||||
@click="handleSignAndSubmit"
|
@click="handleSignAndSubmit"
|
||||||
>
|
>
|
||||||
{{ allItemsSubmitted ? '已签发' : (isSigned ? '提交医嘱' : '一键签名并生成医嘱') }}
|
{{ allItemsSubmitted ? '已签发' : '一键签名并生成医嘱' }}
|
||||||
</el-button>
|
</el-button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -317,6 +323,19 @@ 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)
|
||||||
|
|
||||||
@@ -1045,6 +1064,21 @@ 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 {
|
||||||
|
|||||||
Reference in New Issue
Block a user