Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| cfa073bdba | |||
| 021f7180c0 | |||
|
|
ee59fd5ea0 | ||
| 20a372268b | |||
| 31f288c0dd | |||
| 5c97380a78 | |||
| 30e5c92f0b | |||
| 7c6e35dcc3 | |||
| 74b67401f3 | |||
| 81d954fceb | |||
| 7adb3b3ea4 | |||
| 1e6704928a | |||
| 75e49f0237 | |||
| 2b6b00b6c2 | |||
| 1ddf8a2ccd | |||
| 0a37b05aab | |||
| 20817d6dc4 |
44
bug444_analysis.md
Normal file
44
bug444_analysis.md
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
# Bug #444 分析报告
|
||||||
|
|
||||||
|
## Bug 描述
|
||||||
|
【手术管理-门诊手术安排】生成临时医嘱界面,"已引用计费药品"列表未正常显示药品详细名称信息,且错误地带出了非药品类的计费信息(如手术诊疗项目"小腿烧伤扩创交腿皮瓣修复术"、检查项目"心脏彩色多普勒超声")。
|
||||||
|
|
||||||
|
## 根因分析
|
||||||
|
|
||||||
|
### 数据流
|
||||||
|
1. 用户点击"医嘱"按钮 → `handleMedicalAdvice()` → 调用 `getPrescriptionList()` 获取计费数据
|
||||||
|
2. 用户对数据进行过滤后展示在"已引用计费药品"列表
|
||||||
|
3. 用户点击"引用计费"按钮 → `handleQuoteBilling()` → 再次调用 `getPrescriptionList()` 获取最新计费数据
|
||||||
|
|
||||||
|
### 根因定位
|
||||||
|
**`handleQuoteBilling()` 方法(index.vue:1866-1877)缺少非药品关键词过滤逻辑。**
|
||||||
|
|
||||||
|
`handleMedicalAdvice()` 中有两层过滤:
|
||||||
|
1. `adviceType !== 1` 过滤(只保留药品类型)
|
||||||
|
2. **关键词排除过滤**(排除名称中包含"术"、"超声"、"检查"等非药品关键词的项目)
|
||||||
|
|
||||||
|
但 `handleQuoteBilling()` 中只有第一层过滤(`adviceType !== 1`),**缺少关键词排除过滤**。
|
||||||
|
|
||||||
|
当后端返回的计费数据中某些非药品项目被错误标注为 `adviceType=1` 时:
|
||||||
|
- `handleMedicalAdvice()` 能通过关键词过滤排除这些项目
|
||||||
|
- `handleQuoteBilling()` 无法排除,导致非药品项目出现在"已引用计费药品"列表中
|
||||||
|
|
||||||
|
### 代码对比
|
||||||
|
|
||||||
|
| 过滤条件 | handleMedicalAdvice (L1576-1597) | handleQuoteBilling (L1866-1877) |
|
||||||
|
|---------|:---:|:---:|
|
||||||
|
| encounterId 匹配 | ✓ | ✓ |
|
||||||
|
| adviceType === 1 | ✓ | ✓ |
|
||||||
|
| 名称非空 | ✓ | ✓ |
|
||||||
|
| **关键词排除** | ✓ **有** | ✗ **缺失** |
|
||||||
|
| requestId 过滤 | ✓ | ✓ |
|
||||||
|
|
||||||
|
## 修复方案
|
||||||
|
|
||||||
|
在 `handleQuoteBilling()` 方法的过滤逻辑中,添加与 `handleMedicalAdvice()` 一致的关键词排除逻辑:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// 🔧 修复 Bug #444: 排除名称中包含手术/检查/诊疗关键词的非药品项目
|
||||||
|
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
||||||
|
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
|
||||||
|
```
|
||||||
@@ -5,37 +5,39 @@
|
|||||||
|
|
||||||
## 根因定位
|
## 根因定位
|
||||||
|
|
||||||
**核心问题在 `OrganizationLocationAppServiceImpl.java:161-174`**
|
**核心问题在 `OrganizationLocationAppServiceImpl.java:161-162`**
|
||||||
|
|
||||||
时间冲突检测的查询逻辑存在两个缺陷:
|
时间冲突检测使用了 `getOrgLocListByActivityDefinitionId()` 跨科室查询同一诊疗定义下的**所有**科室配置记录。
|
||||||
|
|
||||||
|
### 缺陷:跨科室误报冲突
|
||||||
|
|
||||||
### 缺陷1:查询范围过窄
|
|
||||||
```java
|
```java
|
||||||
// 只查同一科室 + 同一诊疗的记录
|
// 修复前:查询同一诊疗的所有科室配置(跨科室)
|
||||||
getOrgLocListByOrgIdAndActivityDefinitionId(orgLoc.getOrganizationId(), orgLoc.getActivityDefinitionId());
|
organizationLocationService.getOrgLocListByActivityDefinitionId(orgLoc.getActivityDefinitionId());
|
||||||
```
|
```
|
||||||
只查询**同一科室**的记录。如果同一诊疗项目在其他科室已有配置且时间重叠,不会被当前查询检测到。但系统本应阻止同一诊疗在多个科室同时段执行。
|
|
||||||
|
|
||||||
### 缺陷2:"未知科室"错误提示
|
该查询返回**所有科室**中同一诊疗项目的配置记录。当其他科室(非当前操作科室)已配置了相同诊疗且时间重叠时,会被误判为冲突。
|
||||||
|
|
||||||
|
"执行科室配置"的业务语义是:为某个科室配置它可执行的诊疗项目及时段。不同科室配置同一诊疗的不同时段是完全合理的(如检验科 08:00-12:00,放射科 14:00-18:00)。跨科室时间重叠不应视为冲突。
|
||||||
|
|
||||||
|
### 附带缺陷:"未知科室"错误提示
|
||||||
当冲突记录关联的科室被软删除(`delete_flag='1'`)时,`organizationService.getById()` 受 `@TableLogic` 注解影响查不到该科室,返回 null,错误提示变成"与未知科室时间冲突"。
|
当冲突记录关联的科室被软删除(`delete_flag='1'`)时,`organizationService.getById()` 受 `@TableLogic` 注解影响查不到该科室,返回 null,错误提示变成"与未知科室时间冲突"。
|
||||||
|
|
||||||
数据库验证发现确实存在软删除科室的组织位置记录(内科门诊、上海学校医院、信息科等,共9条)。
|
|
||||||
|
|
||||||
### 数据流
|
|
||||||
|
|
||||||
1. 前端选择科室 → 点击"添加新项目" → 填写诊疗和时间 → 点击"保存"
|
|
||||||
2. 后端 `addOrEditOrgLoc()` 接收请求
|
|
||||||
3. 查询现有冲突记录(**当前只查同科室**)
|
|
||||||
4. 对冲突记录检查时间重叠
|
|
||||||
5. 查找冲突科室名称 → 若科室被软删除则返回 null → "未知科室"
|
|
||||||
|
|
||||||
## 修复方案
|
## 修复方案
|
||||||
|
|
||||||
1. **修改冲突检测范围**:查询同一 `activityDefinitionId` 的所有记录(跨科室检测),而非仅限当前科室
|
**修改冲突检测范围**:将 `getOrgLocListByActivityDefinitionId` 改为 `getOrgLocListByOrgIdAndActivityDefinitionId`,仅检测**同一科室内**的时间冲突。
|
||||||
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`
|
|
||||||
|
**改动行数**: +3/-1(`OrganizationLocationAppServiceImpl.java`)
|
||||||
|
|
||||||
|
**变更内容**:
|
||||||
|
```diff
|
||||||
|
-organizationLocationService.getOrgLocListByActivityDefinitionId(orgLoc.getActivityDefinitionId());
|
||||||
|
+organizationLocationService.getOrgLocListByOrgIdAndActivityDefinitionId(orgLoc.getOrganizationId(),
|
||||||
|
+ orgLoc.getActivityDefinitionId());
|
||||||
|
```
|
||||||
|
|
||||||
|
编译验证通过。
|
||||||
|
|||||||
@@ -158,8 +158,10 @@ public class OrganizationLocationAppServiceImpl implements IOrganizationLocation
|
|||||||
? activityDefinitionMapper.selectById(activityDefinitionId) : null;
|
? activityDefinitionMapper.selectById(activityDefinitionId) : null;
|
||||||
String activityName = activityDef != null ? activityDef.getName() : "";
|
String activityName = activityDef != null ? activityDef.getName() : "";
|
||||||
|
|
||||||
|
// Only check for time conflicts within the same department
|
||||||
List<OrganizationLocation> organizationLocationList =
|
List<OrganizationLocation> organizationLocationList =
|
||||||
organizationLocationService.getOrgLocListByActivityDefinitionId(orgLoc.getActivityDefinitionId());
|
organizationLocationService.getOrgLocListByOrgIdAndActivityDefinitionId(orgLoc.getOrganizationId(),
|
||||||
|
orgLoc.getActivityDefinitionId());
|
||||||
organizationLocationList = (orgLoc.getId() != null)
|
organizationLocationList = (orgLoc.getId() != null)
|
||||||
? organizationLocationList.stream().filter(item -> !orgLoc.getId().equals(item.getId())).toList()
|
? organizationLocationList.stream().filter(item -> !orgLoc.getId().equals(item.getId())).toList()
|
||||||
: organizationLocationList;
|
: organizationLocationList;
|
||||||
|
|||||||
@@ -31,9 +31,4 @@ public class OrgLocQueryParam implements Serializable {
|
|||||||
/** 发放类别 */
|
/** 发放类别 */
|
||||||
private String distributionCategoryCode;
|
private String distributionCategoryCode;
|
||||||
|
|
||||||
/**
|
|
||||||
* 项目编码 | 药品:1 耗材:2
|
|
||||||
*/
|
|
||||||
private String itemCode;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -63,21 +63,17 @@ public interface IDoctorStationEmrAppService {
|
|||||||
* 获取待写病历列表
|
* 获取待写病历列表
|
||||||
*
|
*
|
||||||
* @param doctorId 医生ID
|
* @param doctorId 医生ID
|
||||||
* @param pageNo 当前页码
|
* @return 待写病历列表
|
||||||
* @param pageSize 每页条数
|
|
||||||
* @param patientName 患者姓名(可选)
|
|
||||||
* @return 待写病历分页数据
|
|
||||||
*/
|
*/
|
||||||
R<?> getPendingEmrList(Long doctorId, Integer pageNo, Integer pageSize, String patientName);
|
R<?> getPendingEmrList(Long doctorId);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取待写病历数量
|
* 获取待写病历数量
|
||||||
*
|
*
|
||||||
* @param doctorId 医生ID
|
* @param doctorId 医生ID
|
||||||
* @param patientName 患者姓名(可选)
|
|
||||||
* @return 待写病历数量
|
* @return 待写病历数量
|
||||||
*/
|
*/
|
||||||
R<?> getPendingEmrCount(Long doctorId, String patientName);
|
R<?> getPendingEmrCount(Long doctorId);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 检查患者是否需要写病历
|
* 检查患者是否需要写病历
|
||||||
|
|||||||
@@ -2205,10 +2205,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
|
|||||||
// 收费状态
|
// 收费状态
|
||||||
requestBaseDto.setChargeStatus_enumText(
|
requestBaseDto.setChargeStatus_enumText(
|
||||||
EnumUtils.getInfoByValue(ChargeItemStatus.class, requestBaseDto.getChargeStatus()));
|
EnumUtils.getInfoByValue(ChargeItemStatus.class, requestBaseDto.getChargeStatus()));
|
||||||
// 单位字典翻译失败时回退使用原始值(如手术申请硬编码了中文单位名)
|
|
||||||
if (StringUtils.isNotBlank(requestBaseDto.getUnitCode()) && StringUtils.isBlank(requestBaseDto.getUnitCode_dictText())) {
|
|
||||||
requestBaseDto.setUnitCode_dictText(requestBaseDto.getUnitCode());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return R.ok(requestBaseInfo);
|
return R.ok(requestBaseInfo);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,6 @@ import com.openhis.document.service.IEmrTemplateService;
|
|||||||
import com.openhis.web.doctorstation.appservice.IDoctorStationEmrAppService;
|
import com.openhis.web.doctorstation.appservice.IDoctorStationEmrAppService;
|
||||||
import com.openhis.web.doctorstation.dto.EmrTemplateDto;
|
import com.openhis.web.doctorstation.dto.EmrTemplateDto;
|
||||||
import com.openhis.web.doctorstation.dto.PatientEmrDto;
|
import com.openhis.web.doctorstation.dto.PatientEmrDto;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
|
||||||
import org.springframework.beans.BeanUtils;
|
import org.springframework.beans.BeanUtils;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@@ -42,7 +41,6 @@ import java.util.stream.Collectors;
|
|||||||
/**
|
/**
|
||||||
* 医生站-电子病历 应用实现类
|
* 医生站-电子病历 应用实现类
|
||||||
*/
|
*/
|
||||||
@Slf4j
|
|
||||||
@Service
|
@Service
|
||||||
public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppService {
|
public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppService {
|
||||||
|
|
||||||
@@ -62,7 +60,13 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
|
|||||||
IDocRecordService docRecordService;
|
IDocRecordService docRecordService;
|
||||||
|
|
||||||
@Resource
|
@Resource
|
||||||
private com.openhis.web.doctorstation.mapper.DoctorStationEmrAppMapper doctorStationEmrAppMapper;
|
private EncounterMapper encounterMapper;
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
private PatientMapper patientMapper;
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
private com.openhis.administration.mapper.EncounterParticipantMapper encounterParticipantMapper;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 添加病人病历信息
|
* 添加病人病历信息
|
||||||
@@ -219,35 +223,52 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
|
|||||||
* @return 待写病历列表
|
* @return 待写病历列表
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public R<?> getPendingEmrList(Long doctorId, Integer pageNo, Integer pageSize, String patientName) {
|
public R<?> getPendingEmrList(Long doctorId) {
|
||||||
List<Map<String, Object>> allRows = doctorStationEmrAppMapper.getPendingEmrList(doctorId, patientName);
|
// 由于Encounter实体中没有jzPractitionerUserId字段,我们需要通过关联查询来获取相关信息
|
||||||
int total = allRows.size();
|
// 使用医生工作站的mapper来查询相关数据
|
||||||
|
// 这里我们直接使用医生工作站的查询逻辑
|
||||||
|
|
||||||
// 分页截取
|
// 查询当前医生负责的、状态为"就诊中"但还没有写病历的患者
|
||||||
int fromIndex = (pageNo - 1) * pageSize;
|
// 需要通过EncounterParticipant表来关联医生信息
|
||||||
int toIndex = Math.min(fromIndex + pageSize, total);
|
List<Encounter> encounters = encounterMapper.selectList(
|
||||||
List<Map<String, Object>> pageRows;
|
new LambdaQueryWrapper<Encounter>()
|
||||||
if (fromIndex >= total) {
|
.eq(Encounter::getStatusEnum, EncounterStatus.IN_PROGRESS.getValue())
|
||||||
pageRows = new ArrayList<>();
|
);
|
||||||
} else {
|
|
||||||
pageRows = allRows.subList(fromIndex, toIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 计算年龄列
|
// 过滤出由指定医生负责且还没有写病历的就诊记录
|
||||||
for (Map<String, Object> row : pageRows) {
|
List<Map<String, Object>> pendingEmrs = new ArrayList<>();
|
||||||
Object birthDate = row.get("birthDate");
|
for (Encounter encounter : encounters) {
|
||||||
if (birthDate instanceof Date) {
|
// 检查该就诊记录是否已经有病历
|
||||||
row.put("age", calculateAge((Date) birthDate));
|
Emr existingEmr = emrService.getOne(
|
||||||
} else {
|
new LambdaQueryWrapper<Emr>().eq(Emr::getEncounterId, encounter.getId())
|
||||||
row.put("age", null);
|
);
|
||||||
|
|
||||||
|
// 检查该就诊是否由指定医生负责
|
||||||
|
boolean isAssignedToDoctor = isEncounterAssignedToDoctor(encounter.getId(), doctorId);
|
||||||
|
|
||||||
|
if (existingEmr == null && isAssignedToDoctor) {
|
||||||
|
// 如果没有病历且由该医生负责,则添加到待写病历列表
|
||||||
|
Map<String, Object> pendingEmr = new java.util.HashMap<>();
|
||||||
|
|
||||||
|
// 获取患者信息
|
||||||
|
Patient patient = patientMapper.selectById(encounter.getPatientId());
|
||||||
|
|
||||||
|
pendingEmr.put("encounterId", encounter.getId());
|
||||||
|
pendingEmr.put("patientId", encounter.getPatientId());
|
||||||
|
pendingEmr.put("patientName", patient != null ? patient.getName() : "未知");
|
||||||
|
pendingEmr.put("gender", patient != null ? patient.getGenderEnum() : null);
|
||||||
|
// 使用出生日期计算年龄
|
||||||
|
pendingEmr.put("age", patient != null && patient.getBirthDate() != null ?
|
||||||
|
calculateAge(patient.getBirthDate()) : null);
|
||||||
|
// 使用创建时间作为挂号时间
|
||||||
|
pendingEmr.put("registerTime", encounter.getCreateTime());
|
||||||
|
pendingEmr.put("busNo", encounter.getBusNo()); // 病历号
|
||||||
|
|
||||||
|
pendingEmrs.add(pendingEmr);
|
||||||
}
|
}
|
||||||
row.remove("birthDate");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Map<String, Object> result = new java.util.HashMap<>();
|
return R.ok(pendingEmrs);
|
||||||
result.put("rows", pageRows);
|
|
||||||
result.put("total", total);
|
|
||||||
return R.ok(result);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -257,9 +278,14 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
|
|||||||
* @return 待写病历数量
|
* @return 待写病历数量
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public R<?> getPendingEmrCount(Long doctorId, String patientName) {
|
public R<?> getPendingEmrCount(Long doctorId) {
|
||||||
Long count = doctorStationEmrAppMapper.getPendingEmrCount(doctorId, patientName);
|
// 获取待写病历列表,然后返回数量
|
||||||
return R.ok(count != null ? count.intValue() : 0);
|
R<?> result = getPendingEmrList(doctorId);
|
||||||
|
if (result.getCode() == 200) {
|
||||||
|
List<?> pendingEmrs = (List<?>) result.getData();
|
||||||
|
return R.ok(pendingEmrs.size());
|
||||||
|
}
|
||||||
|
return R.ok(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -280,6 +306,24 @@ public class DoctorStationEmrAppServiceImpl implements IDoctorStationEmrAppServi
|
|||||||
return R.ok(needWrite);
|
return R.ok(needWrite);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 检查就诊是否分配给指定医生
|
||||||
|
*
|
||||||
|
* @param encounterId 就诊ID
|
||||||
|
* @param doctorId 医生ID
|
||||||
|
* @return 是否分配给指定医生
|
||||||
|
*/
|
||||||
|
private boolean isEncounterAssignedToDoctor(Long encounterId, Long doctorId) {
|
||||||
|
// 查询就诊参与者表,检查是否有指定医生的接诊记录
|
||||||
|
com.openhis.administration.domain.EncounterParticipant participant =
|
||||||
|
encounterParticipantMapper.selectOne(
|
||||||
|
new com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper<com.openhis.administration.domain.EncounterParticipant>()
|
||||||
|
.eq(com.openhis.administration.domain.EncounterParticipant::getEncounterId, encounterId)
|
||||||
|
.eq(com.openhis.administration.domain.EncounterParticipant::getPractitionerId, doctorId)
|
||||||
|
);
|
||||||
|
|
||||||
|
return participant != null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 根据出生日期计算年龄
|
* 根据出生日期计算年龄
|
||||||
|
|||||||
@@ -26,36 +26,34 @@ public class PendingEmrController {
|
|||||||
* 获取待写病历列表
|
* 获取待写病历列表
|
||||||
*
|
*
|
||||||
* @param doctorId 医生ID
|
* @param doctorId 医生ID
|
||||||
* @param pageNo 当前页码
|
* @return 待写病历列表
|
||||||
* @param pageSize 每页条数
|
|
||||||
* @param patientName 患者姓名(可选)
|
|
||||||
* @return 待写病历分页数据
|
|
||||||
*/
|
*/
|
||||||
@GetMapping("/pending-list")
|
@GetMapping("/pending-list")
|
||||||
public R<?> getPendingEmrList(@RequestParam(required = false) Long doctorId,
|
public R<?> getPendingEmrList(@RequestParam(required = false) Long doctorId) {
|
||||||
@RequestParam(defaultValue = "1") Integer pageNum,
|
// 如果没有传递医生ID,则使用当前登录用户ID
|
||||||
@RequestParam(defaultValue = "10") Integer pageSize,
|
|
||||||
@RequestParam(required = false) String patientName) {
|
|
||||||
if (doctorId == null) {
|
if (doctorId == null) {
|
||||||
doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getPractitionerId();
|
doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getUserId();
|
||||||
}
|
}
|
||||||
return iDoctorStationEmrAppService.getPendingEmrList(doctorId, pageNum, pageSize, patientName);
|
|
||||||
|
// 调用服务获取待写病历列表
|
||||||
|
return iDoctorStationEmrAppService.getPendingEmrList(doctorId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取待写病历数量
|
* 获取待写病历数量
|
||||||
*
|
*
|
||||||
* @param doctorId 医生ID
|
* @param doctorId 医生ID
|
||||||
* @param patientName 患者姓名(可选)
|
|
||||||
* @return 待写病历数量
|
* @return 待写病历数量
|
||||||
*/
|
*/
|
||||||
@GetMapping("/pending-count")
|
@GetMapping("/pending-count")
|
||||||
public R<?> getPendingEmrCount(@RequestParam(required = false) Long doctorId,
|
public R<?> getPendingEmrCount(@RequestParam(required = false) Long doctorId) {
|
||||||
@RequestParam(required = false) String patientName) {
|
// 如果没有传递医生ID,则使用当前登录用户ID
|
||||||
if (doctorId == null) {
|
if (doctorId == null) {
|
||||||
doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getPractitionerId();
|
doctorId = com.core.common.utils.SecurityUtils.getLoginUser().getUserId();
|
||||||
}
|
}
|
||||||
return iDoctorStationEmrAppService.getPendingEmrCount(doctorId, patientName);
|
|
||||||
|
// 调用服务获取待写病历数量
|
||||||
|
return iDoctorStationEmrAppService.getPendingEmrCount(doctorId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -198,10 +198,8 @@ public class AdviceBaseDto {
|
|||||||
/**
|
/**
|
||||||
* 所属科室
|
* 所属科室
|
||||||
*/
|
*/
|
||||||
@Dict(dictTable = "adm_organization", dictCode = "id", dictText = "name")
|
|
||||||
@JsonSerialize(using = ToStringSerializer.class)
|
@JsonSerialize(using = ToStringSerializer.class)
|
||||||
private Long orgId;
|
private Long orgId;
|
||||||
private String orgId_dictText;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 所在位置
|
* 所在位置
|
||||||
|
|||||||
@@ -1,20 +1,11 @@
|
|||||||
package com.openhis.web.doctorstation.mapper;
|
package com.openhis.web.doctorstation.mapper;
|
||||||
|
|
||||||
import org.apache.ibatis.annotations.Param;
|
|
||||||
import org.springframework.stereotype.Repository;
|
import org.springframework.stereotype.Repository;
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 医生站-电子病历 应用Mapper
|
* 医生站-电子病历 应用Mapper
|
||||||
*/
|
*/
|
||||||
@Repository
|
@Repository
|
||||||
public interface DoctorStationEmrAppMapper {
|
public interface DoctorStationEmrAppMapper {
|
||||||
|
|
||||||
List<Map<String, Object>> getPendingEmrList(@Param("doctorId") Long doctorId,
|
|
||||||
@Param("patientName") String patientName);
|
|
||||||
|
|
||||||
Long getPendingEmrCount(@Param("doctorId") Long doctorId,
|
|
||||||
@Param("patientName") String patientName);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -359,24 +359,6 @@ public class AdviceProcessAppServiceImpl implements IAdviceProcessAppService {
|
|||||||
medRequestList.add(item);
|
medRequestList.add(item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 校验医嘱是否已执行,已执行的医嘱需要先取消执行后才能退回
|
|
||||||
List<Long> allRequestIds = performInfoList.stream().map(PerformInfoDto::getRequestId).toList();
|
|
||||||
List<Procedure> allProcedures = procedureService.list(
|
|
||||||
new LambdaQueryWrapper<Procedure>()
|
|
||||||
.in(Procedure::getRequestId, allRequestIds)
|
|
||||||
.eq(Procedure::getDeleteFlag, "0"));
|
|
||||||
Set<Long> executedIds = allProcedures.stream()
|
|
||||||
.filter(p -> EventStatus.COMPLETED.getValue().equals(p.getStatusEnum()))
|
|
||||||
.map(Procedure::getId)
|
|
||||||
.collect(Collectors.toSet());
|
|
||||||
Set<Long> cancelledRefundIds = allProcedures.stream()
|
|
||||||
.filter(p -> EventStatus.CANCEL.getValue().equals(p.getStatusEnum()) && p.getRefundId() != null)
|
|
||||||
.map(Procedure::getRefundId)
|
|
||||||
.collect(Collectors.toSet());
|
|
||||||
executedIds.removeAll(cancelledRefundIds);
|
|
||||||
if (!executedIds.isEmpty()) {
|
|
||||||
return R.fail("该医嘱已执行,请先取消执行后再退回");
|
|
||||||
}
|
|
||||||
// 校验药品医嘱是否已发药,已发药的医嘱不允许退回
|
// 校验药品医嘱是否已发药,已发药的医嘱不允许退回
|
||||||
if (!medRequestList.isEmpty()) {
|
if (!medRequestList.isEmpty()) {
|
||||||
List<Long> medReqIds = medRequestList.stream().map(PerformInfoDto::getRequestId).toList();
|
List<Long> medReqIds = medRequestList.stream().map(PerformInfoDto::getRequestId).toList();
|
||||||
|
|||||||
@@ -78,10 +78,12 @@ public class MedicineSummaryAppServiceImpl implements IMedicineSummaryAppService
|
|||||||
.map(notPerformedReason -> new DispenseInitDto.NotPerformedReasonOption(notPerformedReason.getValue(),
|
.map(notPerformedReason -> new DispenseInitDto.NotPerformedReasonOption(notPerformedReason.getValue(),
|
||||||
notPerformedReason.getInfo()))
|
notPerformedReason.getInfo()))
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
// 发药状态(汇总单:待配药→已提交,已发放→已发药)
|
// 发药状态
|
||||||
List<DispenseStatusOption> dispenseStatusOptions = new ArrayList<>();
|
List<DispenseStatusOption> dispenseStatusOptions = new ArrayList<>();
|
||||||
dispenseStatusOptions.add(new DispenseStatusOption(DispenseStatus.PREPARATION.getValue(), "已提交"));
|
dispenseStatusOptions.add(new DispenseStatusOption(DispenseStatus.PREPARATION.getValue(),
|
||||||
dispenseStatusOptions.add(new DispenseStatusOption(DispenseStatus.COMPLETED.getValue(), "已发药"));
|
DispenseStatus.PREPARATION.getInfo()));
|
||||||
|
dispenseStatusOptions.add(new DispenseStatusOption(DispenseStatus.COMPLETED.getValue(),
|
||||||
|
DispenseStatus.COMPLETED.getInfo()));
|
||||||
|
|
||||||
initDto.setNotPerformedReasonOptions(notPerformedReasonOptions).setDispenseStatusOptions(dispenseStatusOptions);
|
initDto.setNotPerformedReasonOptions(notPerformedReasonOptions).setDispenseStatusOptions(dispenseStatusOptions);
|
||||||
return R.ok(initDto);
|
return R.ok(initDto);
|
||||||
@@ -159,8 +161,8 @@ public class MedicineSummaryAppServiceImpl implements IMedicineSummaryAppService
|
|||||||
new Page<>(pageNo, pageSize), queryWrapper, DispenseStatus.COMPLETED.getValue(),
|
new Page<>(pageNo, pageSize), queryWrapper, DispenseStatus.COMPLETED.getValue(),
|
||||||
DispenseStatus.PREPARATION.getValue(), SupplyType.SUMMARY_DISPENSE.getValue());
|
DispenseStatus.PREPARATION.getValue(), SupplyType.SUMMARY_DISPENSE.getValue());
|
||||||
medicineSummaryFormPage.getRecords().forEach(e -> {
|
medicineSummaryFormPage.getRecords().forEach(e -> {
|
||||||
// 发药状态(汇总单展示文案)
|
// 发药状态
|
||||||
e.setStatusEnum_enumText(getSummaryFormStatusText(e.getStatusEnum()));
|
e.setStatusEnum_enumText(EnumUtils.getInfoByValue(DispenseStatus.class, e.getStatusEnum()));
|
||||||
});
|
});
|
||||||
return R.ok(medicineSummaryFormPage);
|
return R.ok(medicineSummaryFormPage);
|
||||||
}
|
}
|
||||||
@@ -290,17 +292,4 @@ public class MedicineSummaryAppServiceImpl implements IMedicineSummaryAppService
|
|||||||
}
|
}
|
||||||
return R.ok(MessageUtils.createMessage(PromptMsgConstant.Common.M00004, new Object[]{"取消"}));
|
return R.ok(MessageUtils.createMessage(PromptMsgConstant.Common.M00004, new Object[]{"取消"}));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* 汇总发药单状态展示文案(药品医嘱状态映射表:汇总申请→已提交,发药→已发药)
|
|
||||||
*/
|
|
||||||
private String getSummaryFormStatusText(Integer statusEnum) {
|
|
||||||
if (DispenseStatus.PREPARATION.getValue().equals(statusEnum)) {
|
|
||||||
return "已提交";
|
|
||||||
}
|
|
||||||
if (DispenseStatus.COMPLETED.getValue().equals(statusEnum)) {
|
|
||||||
return "已发药";
|
|
||||||
}
|
|
||||||
return EnumUtils.getInfoByValue(DispenseStatus.class, statusEnum);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
|
|||||||
import com.baomidou.mybatisplus.core.metadata.IPage;
|
import com.baomidou.mybatisplus.core.metadata.IPage;
|
||||||
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
||||||
import com.core.common.core.domain.R;
|
import com.core.common.core.domain.R;
|
||||||
import com.core.common.enums.DelFlag;
|
|
||||||
import com.core.common.exception.ServiceException;
|
import com.core.common.exception.ServiceException;
|
||||||
import com.core.common.utils.AssignSeqUtil;
|
import com.core.common.utils.AssignSeqUtil;
|
||||||
import com.core.common.utils.MessageUtils;
|
import com.core.common.utils.MessageUtils;
|
||||||
@@ -18,8 +17,6 @@ import com.openhis.common.constant.PromptMsgConstant;
|
|||||||
import com.openhis.common.enums.*;
|
import com.openhis.common.enums.*;
|
||||||
import com.openhis.document.domain.RequestForm;
|
import com.openhis.document.domain.RequestForm;
|
||||||
import com.openhis.document.service.IRequestFormService;
|
import com.openhis.document.service.IRequestFormService;
|
||||||
import com.openhis.lab.domain.Specimen;
|
|
||||||
import com.openhis.lab.service.ISpecimenService;
|
|
||||||
import com.openhis.web.doctorstation.dto.ActivityChildrenJsonParams;
|
import com.openhis.web.doctorstation.dto.ActivityChildrenJsonParams;
|
||||||
import com.openhis.web.doctorstation.utils.AdviceUtils;
|
import com.openhis.web.doctorstation.utils.AdviceUtils;
|
||||||
import com.openhis.web.regdoctorstation.appservice.IRequestFormManageAppService;
|
import com.openhis.web.regdoctorstation.appservice.IRequestFormManageAppService;
|
||||||
@@ -70,39 +67,6 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
|
|||||||
@Resource
|
@Resource
|
||||||
IActivityDefinitionService iActivityDefinitionService;
|
IActivityDefinitionService iActivityDefinitionService;
|
||||||
|
|
||||||
@Resource
|
|
||||||
ISpecimenService iSpecimenService;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 校验当前用户是否有权操作该申请单(申请者本人或管理员)
|
|
||||||
*/
|
|
||||||
private R<?> validateRequestFormPermission(RequestForm requestForm) {
|
|
||||||
if (SecurityUtils.isAdmin(SecurityUtils.getUserId())) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
Long currentPractitionerId = SecurityUtils.getLoginUser().getPractitionerId();
|
|
||||||
Long requesterId = requestForm.getRequesterId();
|
|
||||||
if (currentPractitionerId == null || requesterId == null
|
|
||||||
|| !currentPractitionerId.equals(requesterId)) {
|
|
||||||
return R.fail("无操作权限,仅申请开立者或管理员可操作");
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 校验关联医嘱是否已采证(存在已采集/已接收标本则不可撤回)
|
|
||||||
*/
|
|
||||||
private boolean hasCollectedSpecimen(List<Long> serviceRequestIds) {
|
|
||||||
if (serviceRequestIds == null || serviceRequestIds.isEmpty()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
long count = iSpecimenService.count(
|
|
||||||
new LambdaQueryWrapper<Specimen>()
|
|
||||||
.in(Specimen::getServiceId, serviceRequestIds)
|
|
||||||
.ge(Specimen::getCollectionStatusEnum, SpecCollectStatus.COLLECTED.getValue()));
|
|
||||||
return count > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 保存申请单
|
* 保存申请单
|
||||||
*
|
*
|
||||||
@@ -564,17 +528,12 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
|
|||||||
if (requestForm == null) {
|
if (requestForm == null) {
|
||||||
return R.fail("申请单不存在");
|
return R.fail("申请单不存在");
|
||||||
}
|
}
|
||||||
R<?> permissionResult = validateRequestFormPermission(requestForm);
|
|
||||||
if (permissionResult != null) {
|
|
||||||
return permissionResult;
|
|
||||||
}
|
|
||||||
String prescriptionNo = requestForm.getPrescriptionNo();
|
String prescriptionNo = requestForm.getPrescriptionNo();
|
||||||
|
|
||||||
// 查询该申请单下所有 ServiceRequest(含子项)
|
// 查询该申请单下所有 ServiceRequest(含子项)
|
||||||
List<ServiceRequest> serviceRequests = iServiceRequestService.list(
|
List<ServiceRequest> serviceRequests = iServiceRequestService.list(
|
||||||
new LambdaQueryWrapper<ServiceRequest>()
|
new LambdaQueryWrapper<ServiceRequest>()
|
||||||
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo)
|
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo));
|
||||||
.eq(ServiceRequest::getDeleteFlag, DelFlag.NO.getCode()));
|
|
||||||
if (serviceRequests == null || serviceRequests.isEmpty()) {
|
if (serviceRequests == null || serviceRequests.isEmpty()) {
|
||||||
return R.fail("未找到关联的诊疗医嘱");
|
return R.fail("未找到关联的诊疗医嘱");
|
||||||
}
|
}
|
||||||
@@ -604,7 +563,7 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
|
|||||||
// 4. 删除申请单
|
// 4. 删除申请单
|
||||||
iRequestFormService.removeById(requestFormId);
|
iRequestFormService.removeById(requestFormId);
|
||||||
|
|
||||||
log.info("检验申请单删除成功,requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
|
log.info("检查申请单删除成功,requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
|
||||||
return R.ok("删除成功");
|
return R.ok("删除成功");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -617,45 +576,32 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
|
|||||||
if (requestForm == null) {
|
if (requestForm == null) {
|
||||||
return R.fail("申请单不存在");
|
return R.fail("申请单不存在");
|
||||||
}
|
}
|
||||||
R<?> permissionResult = validateRequestFormPermission(requestForm);
|
|
||||||
if (permissionResult != null) {
|
|
||||||
return permissionResult;
|
|
||||||
}
|
|
||||||
String prescriptionNo = requestForm.getPrescriptionNo();
|
String prescriptionNo = requestForm.getPrescriptionNo();
|
||||||
|
|
||||||
// 查询该申请单下所有 ServiceRequest
|
// 查询该申请单下所有 ServiceRequest
|
||||||
List<ServiceRequest> serviceRequests = iServiceRequestService.list(
|
List<ServiceRequest> serviceRequests = iServiceRequestService.list(
|
||||||
new LambdaQueryWrapper<ServiceRequest>()
|
new LambdaQueryWrapper<ServiceRequest>()
|
||||||
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo)
|
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo));
|
||||||
.eq(ServiceRequest::getDeleteFlag, DelFlag.NO.getCode()));
|
|
||||||
if (serviceRequests == null || serviceRequests.isEmpty()) {
|
if (serviceRequests == null || serviceRequests.isEmpty()) {
|
||||||
return R.fail("未找到关联的诊疗医嘱");
|
return R.fail("未找到关联的诊疗医嘱");
|
||||||
}
|
}
|
||||||
|
|
||||||
// 筛选出ACTIVE状态的ServiceRequest,与SQL的EXISTS逻辑一致
|
// 校验:只有已签发(status=2)的申请单可撤回
|
||||||
List<Long> activeServiceRequestIds = serviceRequests.stream()
|
boolean allActive = serviceRequests.stream()
|
||||||
.filter(sr -> RequestStatus.ACTIVE.getValue().equals(sr.getStatusEnum()))
|
.allMatch(sr -> RequestStatus.ACTIVE.getValue().equals(sr.getStatusEnum()));
|
||||||
|
if (!allActive) {
|
||||||
|
return R.fail("只有已签发状态的申请单可撤回");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将所有 ServiceRequest 状态改回待签发(DRAFT=0)
|
||||||
|
List<Long> serviceRequestIds = serviceRequests.stream()
|
||||||
.map(ServiceRequest::getId).collect(Collectors.toList());
|
.map(ServiceRequest::getId).collect(Collectors.toList());
|
||||||
if (activeServiceRequestIds.isEmpty()) {
|
iServiceRequestService.update(
|
||||||
return R.fail("只有已签发且未采证的申请单可撤回");
|
|
||||||
}
|
|
||||||
|
|
||||||
// 校验:仅检查ACTIVE状态医嘱的标本采集情况,与SQL的computed_status逻辑一致
|
|
||||||
if (hasCollectedSpecimen(activeServiceRequestIds)) {
|
|
||||||
return R.fail("标本已采集,无法撤回");
|
|
||||||
}
|
|
||||||
|
|
||||||
// 将所有已签发的 ServiceRequest 状态改回待签发,与申请单展示状态同步
|
|
||||||
boolean updated = iServiceRequestService.update(
|
|
||||||
new ServiceRequest().setStatusEnum(RequestStatus.DRAFT.getValue()),
|
new ServiceRequest().setStatusEnum(RequestStatus.DRAFT.getValue()),
|
||||||
new LambdaUpdateWrapper<ServiceRequest>()
|
new LambdaUpdateWrapper<ServiceRequest>()
|
||||||
.in(ServiceRequest::getId, activeServiceRequestIds)
|
.in(ServiceRequest::getId, serviceRequestIds));
|
||||||
.eq(ServiceRequest::getStatusEnum, RequestStatus.ACTIVE.getValue()));
|
|
||||||
if (!updated) {
|
|
||||||
return R.fail("撤回失败,医嘱状态已变更,请刷新后重试");
|
|
||||||
}
|
|
||||||
|
|
||||||
log.info("检验申请单撤回成功,requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
|
log.info("检查申请单撤回成功,requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
|
||||||
return R.ok("撤回成功");
|
return R.ok("撤回成功");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -194,8 +194,8 @@ public class RequestFormManageController {
|
|||||||
* @return 结果
|
* @return 结果
|
||||||
*/
|
*/
|
||||||
@PostMapping(value = "/delete")
|
@PostMapping(value = "/delete")
|
||||||
public R<?> deleteRequestForm(@RequestBody Map<String, Object> data) {
|
public R<?> deleteRequestForm(@RequestBody Map<String, Long> data) {
|
||||||
return iRequestFormManageAppService.deleteRequestForm(parseLong(data.get("requestFormId")));
|
return iRequestFormManageAppService.deleteRequestForm(data.get("requestFormId"));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -205,24 +205,7 @@ public class RequestFormManageController {
|
|||||||
* @return 结果
|
* @return 结果
|
||||||
*/
|
*/
|
||||||
@PostMapping(value = "/withdraw")
|
@PostMapping(value = "/withdraw")
|
||||||
public R<?> withdrawRequestForm(@RequestBody Map<String, Object> data) {
|
public R<?> withdrawRequestForm(@RequestBody Map<String, Long> data) {
|
||||||
return iRequestFormManageAppService.withdrawRequestForm(parseLong(data.get("requestFormId")));
|
return iRequestFormManageAppService.withdrawRequestForm(data.get("requestFormId"));
|
||||||
}
|
|
||||||
|
|
||||||
private Long parseLong(Object value) {
|
|
||||||
if (value == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
if (value instanceof Long) {
|
|
||||||
return (Long) value;
|
|
||||||
}
|
|
||||||
if (value instanceof Number) {
|
|
||||||
return ((Number) value).longValue();
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
return Long.parseLong(value.toString());
|
|
||||||
} catch (NumberFormatException e) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,8 +31,8 @@ public class HomeController {
|
|||||||
HomeStatisticsDto statisticsDto = homeStatisticsService.getHomeStatistics();
|
HomeStatisticsDto statisticsDto = homeStatisticsService.getHomeStatistics();
|
||||||
|
|
||||||
// 获取待写病历数量
|
// 获取待写病历数量
|
||||||
Long practitionerId = SecurityUtils.getLoginUser().getPractitionerId();
|
Long userId = SecurityUtils.getLoginUser().getUserId();
|
||||||
R<?> pendingEmrCount = doctorStationEmrAppService.getPendingEmrCount(practitionerId, null);
|
R<?> pendingEmrCount = doctorStationEmrAppService.getPendingEmrCount(userId);
|
||||||
|
|
||||||
// 将待写病历数量添加到统计数据中
|
// 将待写病历数量添加到统计数据中
|
||||||
statisticsDto.setPendingEmr((Integer) pendingEmrCount.getData());
|
statisticsDto.setPendingEmr((Integer) pendingEmrCount.getData());
|
||||||
|
|||||||
@@ -74,6 +74,7 @@ public class TriageQueueAppServiceImpl implements TriageQueueAppService {
|
|||||||
.eq(TriageQueueItem::getTenantId, tenantId)
|
.eq(TriageQueueItem::getTenantId, tenantId)
|
||||||
.eq(TriageQueueItem::getQueueDate, qd)
|
.eq(TriageQueueItem::getQueueDate, qd)
|
||||||
.eq(TriageQueueItem::getDeleteFlag, "0")
|
.eq(TriageQueueItem::getDeleteFlag, "0")
|
||||||
|
.ne(TriageQueueItem::getStatus, TriageQueueStatus.COMPLETED.getValue())
|
||||||
.orderByAsc(TriageQueueItem::getQueueOrder);
|
.orderByAsc(TriageQueueItem::getQueueOrder);
|
||||||
|
|
||||||
// 如果指定了科室,按科室过滤;否则查询所有科室(全科模式)
|
// 如果指定了科室,按科室过滤;否则查询所有科室(全科模式)
|
||||||
@@ -91,6 +92,14 @@ public class TriageQueueAppServiceImpl implements TriageQueueAppService {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 双重保险:再次过滤掉 COMPLETED 状态的患者(防止数据库中有异常数据)
|
||||||
|
if (list != null && !list.isEmpty()) {
|
||||||
|
int beforeSize = list.size();
|
||||||
|
list = list.stream()
|
||||||
|
.filter(item -> !TriageQueueStatus.COMPLETED.getValue().equals(item.getStatus()))
|
||||||
|
.collect(java.util.stream.Collectors.toList());
|
||||||
|
}
|
||||||
return R.ok(list);
|
return R.ok(list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -894,6 +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 #{limit} OFFSET #{offset}
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<!-- 检查/检验项目专用分页查询:仅查指定 category_code + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
<!-- 检查/检验项目专用分页查询:仅查指定 category_code + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
||||||
|
|||||||
@@ -4,38 +4,4 @@
|
|||||||
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
|
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
|
||||||
<mapper namespace="com.openhis.web.doctorstation.mapper.DoctorStationEmrAppMapper">
|
<mapper namespace="com.openhis.web.doctorstation.mapper.DoctorStationEmrAppMapper">
|
||||||
|
|
||||||
<select id="getPendingEmrList" resultType="java.util.HashMap">
|
</mapper>
|
||||||
SELECT e.id AS "encounterId",
|
|
||||||
e.patient_id AS "patientId",
|
|
||||||
p.name AS "patientName",
|
|
||||||
p.gender_enum AS "gender",
|
|
||||||
p.birth_date AS "birthDate",
|
|
||||||
e.create_time AS "registerTime",
|
|
||||||
e.bus_no AS "busNo"
|
|
||||||
FROM adm_encounter e
|
|
||||||
INNER JOIN adm_encounter_participant ep ON e.id = ep.encounter_id AND ep.practitioner_id = #{doctorId}
|
|
||||||
LEFT JOIN adm_patient p ON e.patient_id = p.id
|
|
||||||
LEFT JOIN doc_emr emr ON e.id = emr.encounter_id
|
|
||||||
WHERE e.status_enum = 2
|
|
||||||
AND emr.id IS NULL
|
|
||||||
<if test="patientName != null and patientName != ''">
|
|
||||||
AND p.name LIKE CONCAT('%', #{patientName}, '%')
|
|
||||||
</if>
|
|
||||||
ORDER BY e.create_time DESC
|
|
||||||
</select>
|
|
||||||
|
|
||||||
<select id="getPendingEmrCount" resultType="java.lang.Long">
|
|
||||||
SELECT COUNT(*)
|
|
||||||
FROM adm_encounter e
|
|
||||||
INNER JOIN adm_encounter_participant ep ON e.id = ep.encounter_id AND ep.practitioner_id = #{doctorId}
|
|
||||||
LEFT JOIN doc_emr emr ON e.id = emr.encounter_id
|
|
||||||
WHERE e.status_enum = 2
|
|
||||||
AND emr.id IS NULL
|
|
||||||
<if test="patientName != null and patientName != ''">
|
|
||||||
AND e.patient_id IN (
|
|
||||||
SELECT id FROM adm_patient WHERE name LIKE CONCAT('%', #{patientName}, '%')
|
|
||||||
)
|
|
||||||
</if>
|
|
||||||
</select>
|
|
||||||
|
|
||||||
</mapper>
|
|
||||||
@@ -35,27 +35,21 @@
|
|||||||
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
||||||
AND ws.status_enum = 8
|
AND ws.status_enum = 8
|
||||||
) THEN 6
|
) THEN 6
|
||||||
WHEN EXISTS (
|
|
||||||
SELECT 1 FROM wor_service_request ws
|
|
||||||
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
|
||||||
AND ws.status_enum = 5
|
|
||||||
) THEN 7
|
|
||||||
WHEN EXISTS (
|
WHEN EXISTS (
|
||||||
SELECT 1 FROM wor_service_request ws
|
SELECT 1 FROM wor_service_request ws
|
||||||
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
||||||
AND ws.status_enum = 3
|
AND ws.status_enum = 3
|
||||||
) THEN 5
|
) THEN 5
|
||||||
WHEN EXISTS (
|
|
||||||
SELECT 1 FROM wor_service_request ws
|
|
||||||
INNER JOIN lab_specimen ls ON ls.service_id = ws.id
|
|
||||||
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
|
||||||
AND ls.collection_status_enum >= 1
|
|
||||||
) THEN 4
|
|
||||||
WHEN EXISTS (
|
WHEN EXISTS (
|
||||||
SELECT 1 FROM wor_service_request ws
|
SELECT 1 FROM wor_service_request ws
|
||||||
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
||||||
AND ws.status_enum = 2
|
AND ws.status_enum = 2
|
||||||
) THEN 1
|
) THEN 1
|
||||||
|
WHEN EXISTS (
|
||||||
|
SELECT 1 FROM wor_service_request ws
|
||||||
|
WHERE ws.prescription_no = drf.prescription_no AND ws.delete_flag = '0'
|
||||||
|
AND ws.status_enum = 5
|
||||||
|
) THEN 7
|
||||||
ELSE 0
|
ELSE 0
|
||||||
END AS computed_status
|
END AS computed_status
|
||||||
FROM doc_request_form AS drf
|
FROM doc_request_form AS drf
|
||||||
|
|||||||
@@ -36,58 +36,197 @@
|
|||||||
<el-button type="primary" plain icon="Search" @click="getValue">查询</el-button>
|
<el-button type="primary" plain icon="Search" @click="getValue">查询</el-button>
|
||||||
<el-button type="primary" plain icon="Printer" @click="print">打印</el-button>
|
<el-button type="primary" plain icon="Printer" @click="print">打印</el-button>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
<!-- <el-form-item label="科室:" prop="sourceLocationId">
|
||||||
|
<el-select
|
||||||
|
v-model="queryParams.sourceLocationId"
|
||||||
|
placeholder=""
|
||||||
|
clearable
|
||||||
|
style="width: 150px"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="issueDepartment in issueDepartmentDto"
|
||||||
|
:key="issueDepartment.id"
|
||||||
|
:label="issueDepartment.name"
|
||||||
|
:value="issueDepartment.id"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item> -->
|
||||||
</el-form>
|
</el-form>
|
||||||
|
|
||||||
<div v-loading="loading" class="report-container">
|
<!-- <el-row :gutter="10" class="mb8">
|
||||||
<div class="report-title">门诊收费日结单</div>
|
<el-col :span="1.5">
|
||||||
<div class="report-section">
|
<el-button type="primary" plain icon="Search" @click="getValue">查询</el-button>
|
||||||
<div class="section-title">基本信息</div>
|
</el-col>
|
||||||
<el-row :gutter="20" class="report-row">
|
<el-col :span="1.5">
|
||||||
<el-col :span="6"><span class="label">经办人姓名:</span><span class="value">{{ userStore.nickName }}</span></el-col>
|
<el-button type="warning" plain icon="CircleClose" @click="handleClear">重置</el-button>
|
||||||
<el-col :span="6"><span class="label">科室:</span><span class="value">{{ userStore.orgName }}</span></el-col>
|
</el-col>
|
||||||
<el-col :span="12"><span class="label">时间:</span><span class="value">{{ queryTime[0] + '~' + queryTime[1] }}</span></el-col>
|
</el-row> -->
|
||||||
</el-row>
|
<div v-loading="loading">
|
||||||
</div>
|
<el-row
|
||||||
<div class="divider"></div>
|
:gutter="10"
|
||||||
<div class="report-section">
|
outpatientNo="mb8"
|
||||||
<div class="section-title">收费汇总</div>
|
style="
|
||||||
<el-row :gutter="20" class="report-row">
|
margin: 20px 0;
|
||||||
<el-col :span="6"><span class="label">实际现金收入:</span><span class="value">{{ formatValue(reportValue.cashSum) }}</span></el-col>
|
display: flex;
|
||||||
<el-col :span="6"><span class="label">现金:</span><span class="value">{{ formatValue(reportValue.rmbCashSum) }}</span></el-col>
|
align-items: center;
|
||||||
<el-col :span="6"><span class="label">微信:</span><span class="value">{{ formatValue(reportValue.vxCashSum) }}</span></el-col>
|
justify-content: flex-start;
|
||||||
<el-col :span="6"><span class="label">支付宝:</span><span class="value">{{ formatValue(reportValue.aliCashSum) }}</span></el-col>
|
padding: 0 20px;
|
||||||
</el-row>
|
"
|
||||||
</div>
|
>
|
||||||
<div class="divider"></div>
|
<!-- <el-col :span="3">
|
||||||
<div class="report-section">
|
<span>经办人编号:</span>
|
||||||
<div class="section-title">医保支付</div>
|
</el-col> -->
|
||||||
<el-row :gutter="20" class="report-row">
|
<el-col :span="3">
|
||||||
<el-col :span="6"><span class="label">统筹支付:</span><span class="value">{{ formatValue(reportValue.tcSum) }}</span></el-col>
|
<span class="label">经办人姓名:</span>
|
||||||
<el-col :span="6"><span class="label">账户支付:</span><span class="value">{{ formatValue(reportValue.zhSum) }}</span></el-col>
|
<span class="value"> {{ userStore.nickName }}</span>
|
||||||
<el-col :span="12"><span class="label">基金支付总额:</span><span class="value">{{ formatValue(reportValue.fundSum) }}</span></el-col>
|
</el-col>
|
||||||
</el-row>
|
<el-col :span="3">
|
||||||
</div>
|
<span class="label">科室:</span>
|
||||||
<div class="divider"></div>
|
<span class="value">{{ userStore.orgName }}</span>
|
||||||
<div class="report-section">
|
</el-col>
|
||||||
<div class="section-title">费用明细</div>
|
<el-col :span="4.5">
|
||||||
<el-row :gutter="20" class="report-row">
|
<span class="label">时间:</span>
|
||||||
<el-col :span="6"><span class="label">诊查费:</span><span class="value">{{ formatValue(reportValue.DIAGNOSTIC_FEE) }}</span></el-col>
|
<span class="value"> {{ queryTime[0] + '~' + queryTime[1] }} </span>
|
||||||
<el-col :span="6"><span class="label">检查费:</span><span class="value">{{ formatValue(reportValue.CHECK_FEE) }}</span></el-col>
|
</el-col>
|
||||||
<el-col :span="6"><span class="label">化验费:</span><span class="value">{{ formatValue(reportValue.DIAGNOSTIC_TEST_FEE) }}</span></el-col>
|
</el-row>
|
||||||
<el-col :span="6"><span class="label">治疗费:</span><span class="value">{{ formatValue(reportValue.MEDICAL_EXPENSE_FEE) }}</span></el-col>
|
<el-row
|
||||||
</el-row>
|
:gutter="10"
|
||||||
<el-row :gutter="20" class="report-row">
|
outpatientNo="mb8"
|
||||||
<el-col :span="6"><span class="label">西药费:</span><span class="value">{{ formatValue(reportValue.WEST_MEDICINE) }}</span></el-col>
|
style="
|
||||||
<el-col :span="6"><span class="label">中药饮片费:</span><span class="value">{{ formatValue(reportValue.CHINESE_MEDICINE_SLICES_FEE) }}</span></el-col>
|
margin: 20px 0;
|
||||||
<el-col :span="6"><span class="label">中成药费:</span><span class="value">{{ formatValue(reportValue.CHINESE_MEDICINE_FEE) }}</span></el-col>
|
display: flex;
|
||||||
<el-col :span="6"><span class="label">卫生材料费:</span><span class="value">{{ formatValue(reportValue.SANITARY_MATERIALS_FEE) }}</span></el-col>
|
align-items: center;
|
||||||
</el-row>
|
justify-content: flex-start;
|
||||||
<el-row :gutter="20" class="report-row">
|
padding: 0 20px;
|
||||||
<el-col :span="6"><span class="label">诊疗费:</span><span class="value">{{ formatValue(reportValue.GENERAL_CONSULTATION_FEE) }}</span></el-col>
|
"
|
||||||
<el-col :span="6"><span class="label">挂号费:</span><span class="value">{{ formatValue(reportValue.REGISTRATION_FEE) }}</span></el-col>
|
>
|
||||||
<el-col :span="12"><span class="label">其他费用:</span><span class="value">{{ formatValue(reportValue.OTHER_FEE) }}</span></el-col>
|
<el-col :span="3">
|
||||||
</el-row>
|
<span class="label">实际现金收入:</span>
|
||||||
</div>
|
<span class="value"> {{ formatValue(reportValue.cashSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">现金:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.rmbCashSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">微信:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.vxCashSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">支付宝:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.aliCashSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row
|
||||||
|
:gutter="10"
|
||||||
|
outpatientNo="mb8"
|
||||||
|
style="
|
||||||
|
margin: 20px 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-start;
|
||||||
|
padding: 0 20px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">统筹支付:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.tcSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">账户支付:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.zhSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">基金支付总额:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.fundSum) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<!-- <el-col :span="3">
|
||||||
|
<span>医保人次:{{ reportValue.aliCashSum }}</span>
|
||||||
|
</el-col> -->
|
||||||
|
</el-row>
|
||||||
|
<el-row
|
||||||
|
:gutter="10"
|
||||||
|
outpatientNo="mb8"
|
||||||
|
style="
|
||||||
|
margin: 20px 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-start;
|
||||||
|
padding: 0 20px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">诊查费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.DIAGNOSTIC_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">检查费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.CHECK_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">化验费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.DIAGNOSTIC_TEST_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">治疗费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.MEDICAL_EXPENSE_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row
|
||||||
|
:gutter="10"
|
||||||
|
outpatientNo="mb8"
|
||||||
|
style="
|
||||||
|
margin: 20px 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-start;
|
||||||
|
padding: 0 20px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">西药费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.WEST_MEDICINE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">中药饮片费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.CHINESE_MEDICINE_SLICES_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">中成药费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.CHINESE_MEDICINE_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">卫生材料费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.SANITARY_MATERIALS_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row
|
||||||
|
:gutter="10"
|
||||||
|
outpatientNo="mb8"
|
||||||
|
style="
|
||||||
|
margin: 20px 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-start;
|
||||||
|
padding: 0 20px;
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">诊疗费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.GENERAL_CONSULTATION_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">挂号费:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.REGISTRATION_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="3">
|
||||||
|
<span class="label">其他费用:</span>
|
||||||
|
<span class="value">{{ formatValue(reportValue.OTHER_FEE) }}</span>
|
||||||
|
</el-col>
|
||||||
|
<!-- <el-col :span="3">
|
||||||
|
<span>现金:</span>
|
||||||
|
</el-col> -->
|
||||||
|
</el-row>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -99,11 +238,12 @@ import {formatDateStr} from '@/utils/index';
|
|||||||
import Decimal from 'decimal.js';
|
import Decimal from 'decimal.js';
|
||||||
|
|
||||||
const userStore = useUserStore();
|
const userStore = useUserStore();
|
||||||
|
// import Dialog from "./components/Dialog";
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const { proxy } = getCurrentInstance();
|
const { proxy } = getCurrentInstance();
|
||||||
|
|
||||||
const purchaseinventoryRef = ref(null);
|
const purchaseinventoryRef = ref(null); // 初始化 ref
|
||||||
const purchaseinventoryList = ref([]);
|
const purchaseinventoryList = ref([]);
|
||||||
const open = ref(false);
|
const open = ref(false);
|
||||||
const loading = ref(true);
|
const loading = ref(true);
|
||||||
@@ -160,6 +300,10 @@ function getContract() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function getPharmacyCabinetLists() {
|
function getPharmacyCabinetLists() {
|
||||||
|
// occurrenceTime.value =
|
||||||
|
// getDepartmentList().then((response) => {
|
||||||
|
// issueDepartmentDto.value = response.data
|
||||||
|
// })
|
||||||
}
|
}
|
||||||
/** 查询调拨管理项目列表 */
|
/** 查询调拨管理项目列表 */
|
||||||
function getList() {
|
function getList() {
|
||||||
@@ -187,6 +331,7 @@ function handleQuery() {
|
|||||||
|
|
||||||
/** 清空条件按钮操作 */
|
/** 清空条件按钮操作 */
|
||||||
function handleClear() {
|
function handleClear() {
|
||||||
|
// 清空查询条件
|
||||||
queryParams.value.approvalTimeSTime = '';
|
queryParams.value.approvalTimeSTime = '';
|
||||||
queryParams.value.approvalTimeETime = '';
|
queryParams.value.approvalTimeETime = '';
|
||||||
occurrenceTime.value = '';
|
occurrenceTime.value = '';
|
||||||
@@ -203,11 +348,12 @@ function handleSelectionChange(selection) {
|
|||||||
|
|
||||||
/** 打印门诊日结 */
|
/** 打印门诊日结 */
|
||||||
async function print() {
|
async function print() {
|
||||||
|
// const selectedRows = proxy.$refs['tableRef'].getSelectionRows();
|
||||||
console.log(reportValue.value, '==reportValue.value==');
|
console.log(reportValue.value, '==reportValue.value==');
|
||||||
const result = {
|
const result = {
|
||||||
data: [
|
data: [
|
||||||
{
|
{
|
||||||
...reportValue.value,
|
...reportValue.value, // 将 reportValue.value 中的所有属性展开到 result 中
|
||||||
nickName: userStore.nickName,
|
nickName: userStore.nickName,
|
||||||
orgName: userStore.orgName,
|
orgName: userStore.orgName,
|
||||||
fixmedinsName: userStore.hospitalName,
|
fixmedinsName: userStore.hospitalName,
|
||||||
@@ -228,12 +374,14 @@ async function print() {
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
console.log(result, '==result.data==');
|
console.log(result, '==result.data==');
|
||||||
|
// 将对象转换为 JSON 字符串
|
||||||
let jsonString = JSON.stringify(result, null, 2);
|
let jsonString = JSON.stringify(result, null, 2);
|
||||||
console.log(jsonString, 'jsonstring');
|
console.log(jsonString, 'jsonstring');
|
||||||
await CefSharp.BindObjectAsync('boundAsync');
|
await CefSharp.BindObjectAsync('boundAsync');
|
||||||
await boundAsync
|
await boundAsync
|
||||||
.printReport(getPrintFileName(contractNo.value), jsonString)
|
.printReport(getPrintFileName(contractNo.value), jsonString)
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
|
//返回结果是jsonString,可判断其调用是否成功
|
||||||
console.log(response, 'response');
|
console.log(response, 'response');
|
||||||
var res = JSON.parse(response);
|
var res = JSON.parse(response);
|
||||||
if (!res.IsSuccess) {
|
if (!res.IsSuccess) {
|
||||||
@@ -249,9 +397,9 @@ function getPrintFileName(value) {
|
|||||||
switch (value) {
|
switch (value) {
|
||||||
case '0000':
|
case '0000':
|
||||||
return '门诊日结单(按登录角色查询)自费.grf';
|
return '门诊日结单(按登录角色查询)自费.grf';
|
||||||
case '229900':
|
case '229900': // 省医保
|
||||||
return '门诊日结单(按登录角色查询)省医保.grf';
|
return '门诊日结单(按登录角色查询)省医保.grf';
|
||||||
case '220100':
|
case '220100': // 市医保
|
||||||
return '门诊日结单(按登录角色查询)市医保.grf';
|
return '门诊日结单(按登录角色查询)市医保.grf';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -264,53 +412,24 @@ getList();
|
|||||||
getPharmacyCabinetLists();
|
getPharmacyCabinetLists();
|
||||||
</script>
|
</script>
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.report-container {
|
.custom-tree-node {
|
||||||
width: 100%;
|
|
||||||
max-width: 1200px;
|
|
||||||
margin: 0 auto;
|
|
||||||
box-sizing: border-box;
|
|
||||||
padding: 24px 32px;
|
|
||||||
background: #fff;
|
|
||||||
border-radius: 4px;
|
|
||||||
box-shadow: 0 1px 4px rgba(0, 0, 0, 0.08);
|
|
||||||
}
|
|
||||||
.report-title {
|
|
||||||
text-align: center;
|
|
||||||
font-size: 18px;
|
|
||||||
font-weight: bold;
|
|
||||||
margin: 16px 0;
|
|
||||||
}
|
|
||||||
.report-section {
|
|
||||||
margin: 8px 0;
|
|
||||||
}
|
|
||||||
.section-title {
|
|
||||||
font-size: 15px;
|
|
||||||
font-weight: bold;
|
|
||||||
color: #303133;
|
|
||||||
margin: 8px 0;
|
|
||||||
padding-left: 8px;
|
|
||||||
border-left: 3px solid #409eff;
|
|
||||||
}
|
|
||||||
.report-row .el-col {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 6px 0;
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: large;
|
||||||
|
margin-bottom: 10px;
|
||||||
}
|
}
|
||||||
.label {
|
.label {
|
||||||
min-width: 110px;
|
display: inline-block;
|
||||||
flex-shrink: 0;
|
width: 120px !important;
|
||||||
color: #606266;
|
|
||||||
white-space: nowrap;
|
|
||||||
text-align: right;
|
|
||||||
}
|
}
|
||||||
.value {
|
.value {
|
||||||
color: #303133;
|
float: right;
|
||||||
font-weight: 500;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
}
|
||||||
.divider {
|
.el-col {
|
||||||
height: 1px;
|
margin-right: 50px;
|
||||||
background-color: #dcdfe6;
|
|
||||||
margin: 12px 0;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -749,26 +749,22 @@ function handleInfectiousDiseaseReport() {
|
|||||||
'手足口病': '0311',
|
'手足口病': '0311',
|
||||||
};
|
};
|
||||||
|
|
||||||
// 获取所有命中传染病映射的诊断,但跳过已有已提交报卡的诊断
|
// 获取所有诊断名称对应的报卡编码,但跳过已有已提交报卡的诊断
|
||||||
const infectiousDiagnoses = form.value.diagnosisList
|
const allSelectedDiseases = form.value.diagnosisList
|
||||||
.map(d => ({
|
.filter(d => d.name && d.hasInfectiousReport !== 1)
|
||||||
diagnosis: d,
|
.map(d => diseaseNameToCode[d.name] || null)
|
||||||
diseaseCode: d.name && d.hasInfectiousReport !== 1 ? diseaseNameToCode[d.name] : null
|
.filter(code => code);
|
||||||
}))
|
|
||||||
.filter(item => item.diseaseCode);
|
|
||||||
|
|
||||||
const allSelectedDiseases = infectiousDiagnoses.map(item => item.diseaseCode);
|
|
||||||
|
|
||||||
if (allSelectedDiseases.length === 0) {
|
if (allSelectedDiseases.length === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 优先使用命中传染病映射的主诊断,否则使用第一条命中的传染病诊断
|
// 优先使用主诊断(同样跳过已有报卡的)
|
||||||
const mainInfectiousDiagnosis = infectiousDiagnoses.find(item => item.diagnosis.maindiseFlag === 1)?.diagnosis;
|
const mainDiagnosis = form.value.diagnosisList.find(d => d.maindiseFlag === 1 && d.hasInfectiousReport !== 1);
|
||||||
const firstInfectiousDiagnosis = infectiousDiagnoses[0].diagnosis;
|
const firstDiagnosis = form.value.diagnosisList.find(d => d.hasInfectiousReport !== 1) || form.value.diagnosisList[0];
|
||||||
|
|
||||||
const diagnosisToShow = {
|
const diagnosisToShow = {
|
||||||
...(mainInfectiousDiagnosis || firstInfectiousDiagnosis),
|
...(mainDiagnosis || firstDiagnosis),
|
||||||
selectedDiseases: allSelectedDiseases
|
selectedDiseases: allSelectedDiseases
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1442,7 +1442,7 @@ async function buildSubmitData() {
|
|||||||
const submitData = {
|
const submitData = {
|
||||||
cardNo: formData.cardNo,
|
cardNo: formData.cardNo,
|
||||||
visitId: props.patientInfo?.encounterId || formData.encounterId || null,
|
visitId: props.patientInfo?.encounterId || formData.encounterId || null,
|
||||||
diagId: formData.diagnosisId || null,
|
diagId: formData.diagnosisId ? Number(formData.diagnosisId) : null,
|
||||||
patId: formData.patientId || null,
|
patId: formData.patientId || null,
|
||||||
idType: 1, // 默认身份证
|
idType: 1, // 默认身份证
|
||||||
idNo: formData.idNo,
|
idNo: formData.idNo,
|
||||||
|
|||||||
@@ -397,165 +397,117 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- 右侧:已选择(检查项目、检查方法为两类独立选择结果) -->
|
<!-- 右侧:已选择 项目卡片(可展开显示检查方法) -->
|
||||||
<div class="selected-panel">
|
<div class="selected-panel">
|
||||||
<div class="panel-label">已选择:</div>
|
<div class="panel-label">已选择:</div>
|
||||||
<div class="selected-tags">
|
<div class="selected-tags">
|
||||||
<template v-if="selectedItems.length === 0 && selectedMethods.length === 0 && methodsForActiveCategory.length === 0">
|
<div v-if="selectedItems.length === 0" class="empty-selected">–</div>
|
||||||
<div class="empty-selected">–</div>
|
|
||||||
</template>
|
|
||||||
<template v-else>
|
|
||||||
<div
|
<div
|
||||||
|
v-else
|
||||||
v-for="(item, idx) in selectedItems"
|
v-for="(item, idx) in selectedItems"
|
||||||
:key="'project-' + item.id"
|
:key="idx"
|
||||||
class="selected-item-card"
|
class="selected-item-card"
|
||||||
:class="{ 'is-expanded': item.projectFoldExpanded }"
|
:class="{ 'is-expanded': item.expanded }"
|
||||||
>
|
>
|
||||||
<div
|
<!-- 项目卡片头部:项目和检查方法解耦,点击展开查看方法/明细 -->
|
||||||
class="fold-strip fold-strip-project"
|
<div class="card-header" @click="toggleItemExpand(item)">
|
||||||
:class="{ 'is-open': item.projectFoldExpanded }"
|
<el-tooltip :content="getDisplayItemName(item)" placement="top" :show-after="400">
|
||||||
>
|
<span class="card-name">{{ getDisplayItemName(item) }}</span>
|
||||||
<div class="fold-strip-header" @click="toggleProjectFold(item)">
|
</el-tooltip>
|
||||||
<el-icon :class="['fold-chevron', { open: item.projectFoldExpanded }]">
|
<span class="card-price">¥{{ formatDetailAmount(getSelectedItemAmount(item)) }}</span>
|
||||||
<ArrowDown />
|
<el-icon :class="['expand-icon', { expanded: item.expanded }]">
|
||||||
</el-icon>
|
<ArrowDown />
|
||||||
<div class="fold-header-main">
|
</el-icon>
|
||||||
<span class="fold-kicker">检查项目</span>
|
<!-- 删除按钮 -->
|
||||||
<el-tooltip :content="getDisplayItemName(item)" placement="top" :show-after="400">
|
<el-button link type="danger" size="small" @click.stop="handleRemoveItem(idx, item)">
|
||||||
<span class="fold-title line-clamp-2">{{ getDisplayItemName(item) }}</span>
|
<el-icon><Close /></el-icon>
|
||||||
</el-tooltip>
|
</el-button>
|
||||||
|
</div>
|
||||||
|
<div v-if="item.expanded" 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>
|
||||||
<span class="fold-price-strong">¥{{ formatDetailAmount(item.price || 0) }}</span>
|
<div v-show="item.itemPackageExpanded">
|
||||||
<el-button link type="danger" size="small" @click.stop="handleRemoveItem(idx, item)">
|
|
||||||
<el-icon><Close /></el-icon>
|
|
||||||
</el-button>
|
|
||||||
</div>
|
|
||||||
<div v-show="item.projectFoldExpanded" class="fold-strip-body">
|
|
||||||
<div v-if="shouldShowItemPackageBody(item)" class="fold-package-wrap">
|
|
||||||
<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
|
<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}`"
|
||||||
class="detail-row"
|
class="detail-row"
|
||||||
>
|
>
|
||||||
<el-tooltip :content="detail.name" placement="top" :show-after="500">
|
<el-tooltip :content="detail.name" placement="top" :show-after="500">
|
||||||
<span class="detail-name">{{ detail.name }}</span>
|
<span class="detail-name">{{ detail.name }}</span>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
<div class="detail-meta">
|
<div class="detail-meta">
|
||||||
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
|
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
|
||||||
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
|
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="fold-strip-muted">暂无项目套餐明细</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
<div class="selected-card-section">
|
||||||
</div>
|
<div class="selected-section-title">检查方法</div>
|
||||||
|
<div v-if="!item.methods || item.methods.length === 0" class="selected-method-empty">
|
||||||
<div
|
暂无检查方法
|
||||||
v-for="(method, idx) in selectedMethods"
|
|
||||||
:key="'method-' + method.id"
|
|
||||||
class="selected-item-card"
|
|
||||||
:class="{ 'is-expanded': method.expanded }"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="fold-strip fold-strip-method"
|
|
||||||
:class="{ 'is-open': method.expanded }"
|
|
||||||
>
|
|
||||||
<div class="fold-strip-header" @click="toggleSelectedMethodFold(method)">
|
|
||||||
<el-icon :class="['fold-chevron', { open: method.expanded }]">
|
|
||||||
<ArrowDown />
|
|
||||||
</el-icon>
|
|
||||||
<div class="fold-header-main">
|
|
||||||
<span class="fold-kicker">检查方法</span>
|
|
||||||
<span
|
|
||||||
class="fold-title fold-title-plain line-clamp-2"
|
|
||||||
:title="getDisplayMethodName(method)"
|
|
||||||
>
|
|
||||||
{{ getDisplayMethodName(method) }}
|
|
||||||
</span>
|
|
||||||
</div>
|
</div>
|
||||||
<span
|
<div
|
||||||
v-if="hasStandaloneMethodPackage(method)"
|
v-for="method in item.methods"
|
||||||
class="fold-price-strong warn"
|
:key="method.id"
|
||||||
|
class="selected-method-option"
|
||||||
>
|
>
|
||||||
¥{{ formatDetailAmount(method.packagePrice || method.price || 0) }}
|
<el-checkbox
|
||||||
</span>
|
:model-value="item.selectedMethod?.id === method.id"
|
||||||
<el-button link type="danger" size="small" @click.stop="handleRemoveMethod(idx)">
|
@change="(val) => selectMethodCheckbox(val, item, method)"
|
||||||
<el-icon><Close /></el-icon>
|
class="method-checkbox"
|
||||||
</el-button>
|
>
|
||||||
|
{{ method.name }}
|
||||||
|
</el-checkbox>
|
||||||
|
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-show="method.expanded" class="fold-strip-body">
|
<div v-if="shouldShowMethodPackageBody(item)">
|
||||||
<template v-if="hasStandaloneMethodPackage(method)">
|
<div class="package-toggle" @click.stop="toggleMethodPackageExpand(item)">
|
||||||
<div class="fold-package-wrap fold-method-package-wrap">
|
<span>检查方法套餐明细</span>
|
||||||
<div v-if="method.packageLoading" class="package-details-loading">加载中...</div>
|
<el-icon :class="['expand-icon', { expanded: item.methodPackageExpanded }]">
|
||||||
<template v-else>
|
<ArrowDown />
|
||||||
<div v-if="getStandaloneMethodPackageDetailsList(method).length === 0" class="package-details-empty">
|
</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 v-else class="package-details-list method-package-list">
|
</div>
|
||||||
<div
|
|
||||||
v-for="(detail, dIdx) in getStandaloneMethodPackageDetailsList(method)"
|
|
||||||
: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>
|
||||||
</template>
|
</template>
|
||||||
<template v-else>
|
</div>
|
||||||
<div class="fold-strip-muted">无单独的检查方法套餐明细。</div>
|
|
||||||
</template>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- 底部:独立勾选检查方法,样式与左侧项目选择一致 -->
|
|
||||||
<div
|
|
||||||
v-if="methodsForActiveCategory.length > 0"
|
|
||||||
class="selected-global-method-picker"
|
|
||||||
@click.stop
|
|
||||||
>
|
|
||||||
<div class="method-picker-collapse-title" @click="methodPickerExpanded = !methodPickerExpanded">
|
|
||||||
<span class="method-picker-title-main">检查方法</span>
|
|
||||||
<span v-if="activeCategoryName" class="global-method-picker-scope">{{ activeCategoryName }}</span>
|
|
||||||
<el-icon :class="['method-picker-arrow', { expanded: methodPickerExpanded }]">
|
|
||||||
<ArrowDown />
|
|
||||||
</el-icon>
|
|
||||||
</div>
|
|
||||||
<div v-show="methodPickerExpanded" class="global-method-picker-list">
|
|
||||||
<div
|
|
||||||
v-for="method in methodsForActiveCategory"
|
|
||||||
:key="'g-m-' + method.id"
|
|
||||||
class="item-row method-picker-row"
|
|
||||||
>
|
|
||||||
<el-checkbox
|
|
||||||
:model-value="isStandaloneMethodSelected(method)"
|
|
||||||
@change="(val) => onStandaloneMethodChange(!!val, method)"
|
|
||||||
class="item-checkbox"
|
|
||||||
>
|
|
||||||
<span class="method-label-inner">{{ formatExamMethodCaption(method.name) }}</span>
|
|
||||||
</el-checkbox>
|
|
||||||
<span class="item-price">¥{{ formatDetailAmount(method.packagePrice || method.price || 0) }}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -587,8 +539,6 @@ const dictLoading = ref(false);
|
|||||||
const activeDetailTab = ref('applyForm');
|
const activeDetailTab = ref('applyForm');
|
||||||
const applicationList = ref([]);
|
const applicationList = ref([]);
|
||||||
const selectedItems = ref([]);
|
const selectedItems = ref([]);
|
||||||
const selectedMethods = ref([]);
|
|
||||||
const methodPickerExpanded = ref(true);
|
|
||||||
|
|
||||||
// Bug #499: 查询过滤状态
|
// Bug #499: 查询过滤状态
|
||||||
const searchForm = reactive({
|
const searchForm = reactive({
|
||||||
@@ -755,25 +705,13 @@ function getDisplayItemName(item) {
|
|||||||
return String(item?.name || '').replace(/^套餐[::\-\s]*/, '');
|
return String(item?.name || '').replace(/^套餐[::\-\s]*/, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 检查方法展示:避免与后端文案重复出现「(方法)(方法)」 */
|
|
||||||
function formatExamMethodCaption(name) {
|
|
||||||
const raw = String(name || '').trim();
|
|
||||||
if (!raw) return '';
|
|
||||||
if (/^\(方法\)/.test(raw) || /^(方法)/.test(raw)) {
|
|
||||||
return raw;
|
|
||||||
}
|
|
||||||
return `(方法) ${raw}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** 已选方法纯文本(用于标题下级展示,不包含「勾选」前缀,去掉后端自带的 (方法) 前缀) */
|
|
||||||
function getDisplaySelectedMethodName(item) {
|
|
||||||
const raw = String(item?.selectedMethod?.name || '').trim();
|
|
||||||
if (!raw) return '';
|
|
||||||
return raw.replace(/^\(方法\)\s*/, '').replace(/^(方法)\s*/, '').trim();
|
|
||||||
}
|
|
||||||
|
|
||||||
function getSelectedItemAmount(item) {
|
function getSelectedItemAmount(item) {
|
||||||
return Number(item?.price || 0);
|
const itemPrice = Number(item?.price || 0);
|
||||||
|
const methodPrice = Number(item?.selectedMethod?.packagePrice || 0);
|
||||||
|
if (!hasMethodPackage(item) || isSamePackage(item)) {
|
||||||
|
return itemPrice;
|
||||||
|
}
|
||||||
|
return itemPrice + methodPrice;
|
||||||
}
|
}
|
||||||
|
|
||||||
function parsePackageDetailsPayload(res) {
|
function parsePackageDetailsPayload(res) {
|
||||||
@@ -905,19 +843,6 @@ const currentActiveCategory = ref(null); // Bug #500: 记录当前激活的分
|
|||||||
|
|
||||||
const allMethods = ref([]);
|
const allMethods = ref([]);
|
||||||
|
|
||||||
const activeCategory = computed(() => {
|
|
||||||
const id = activeNames.value;
|
|
||||||
if (id === '' || id === null || id === undefined) return null;
|
|
||||||
return categoryList.value.find((cat) => String(cat.typeId) === String(id)) || null;
|
|
||||||
});
|
|
||||||
|
|
||||||
const activeCategoryName = computed(() => activeCategory.value?.typeName || activeCategory.value?.categoryName || '');
|
|
||||||
|
|
||||||
const methodsForActiveCategory = computed(() => {
|
|
||||||
const arr = activeCategory.value?.methods;
|
|
||||||
return Array.isArray(arr) ? arr : [];
|
|
||||||
});
|
|
||||||
|
|
||||||
// ====== 科室下拉(来源:科室管理)======
|
// ====== 科室下拉(来源:科室管理)======
|
||||||
const orgLoading = ref(false);
|
const orgLoading = ref(false);
|
||||||
const orgOptions = ref([]); // { label, value }
|
const orgOptions = ref([]); // { label, value }
|
||||||
@@ -1028,44 +953,6 @@ const availableMethods = computed(() => {
|
|||||||
return allMethods.value;
|
return allMethods.value;
|
||||||
});
|
});
|
||||||
|
|
||||||
function isStandaloneMethodSelected(method) {
|
|
||||||
return selectedMethods.value.some((m) => String(m.id) === String(method?.id));
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDisplayMethodName(method) {
|
|
||||||
const raw = String(method?.name || '').trim();
|
|
||||||
if (!raw) return '';
|
|
||||||
return raw.replace(/^\(方法\)\s*/, '').replace(/^(方法)\s*/, '').trim();
|
|
||||||
}
|
|
||||||
|
|
||||||
function hasStandaloneMethodPackage(method) {
|
|
||||||
return !!(method?.packageId || method?.packageName);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getStandaloneMethodPackageDetailsList(method) {
|
|
||||||
return Array.isArray(method?.packageDetails) ? method.packageDetails : [];
|
|
||||||
}
|
|
||||||
|
|
||||||
async function onStandaloneMethodChange(checked, method) {
|
|
||||||
if (!method) return;
|
|
||||||
if (checked) {
|
|
||||||
if (!isStandaloneMethodSelected(method)) {
|
|
||||||
selectedMethods.value.push({
|
|
||||||
...method,
|
|
||||||
expanded: false,
|
|
||||||
packageLoading: false,
|
|
||||||
packageDetails: []
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
const idx = selectedMethods.value.findIndex((m) => String(m.id) === String(method.id));
|
|
||||||
if (idx > -1) selectedMethods.value.splice(idx, 1);
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
|
||||||
await nextTick();
|
|
||||||
form.totalAmount = totalAmountCalc.value;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 当可选方法列表改变时,如果当前选中的方法不在新列表中,则清空
|
// 当可选方法列表改变时,如果当前选中的方法不在新列表中,则清空
|
||||||
// #428: 分类展开时联动加载检查方法
|
// #428: 分类展开时联动加载检查方法
|
||||||
// Bug #500: 使用 categoryLoadingSet 替代 dictLoading,避免切换分类时整个区域闪烁
|
// Bug #500: 使用 categoryLoadingSet 替代 dictLoading,避免切换分类时整个区域闪烁
|
||||||
@@ -1109,8 +996,6 @@ async function handleCategoryExpand(cat) {
|
|||||||
function handleCollapseChange(activeName) {
|
function handleCollapseChange(activeName) {
|
||||||
// 始终记录当前激活的分类,确保 handleCategoryExpand 能正确忽略过期请求
|
// 始终记录当前激活的分类,确保 handleCategoryExpand 能正确忽略过期请求
|
||||||
currentActiveCategory.value = activeName || null;
|
currentActiveCategory.value = activeName || null;
|
||||||
// 底部「检查方法」勾选区默认展开,不因切换左侧分类而收起
|
|
||||||
methodPickerExpanded.value = true;
|
|
||||||
|
|
||||||
if (activeName) {
|
if (activeName) {
|
||||||
// Bug #428修复: 直接从 categoryList(原始响应式数组)查找分类,
|
// Bug #428修复: 直接从 categoryList(原始响应式数组)查找分类,
|
||||||
@@ -1120,7 +1005,6 @@ function handleCollapseChange(activeName) {
|
|||||||
handleCategoryExpand(cat); // 异步加载,不 await
|
handleCategoryExpand(cat); // 异步加载,不 await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
updateMethodDisplay();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
watch(availableMethods, (newMethods) => {
|
watch(availableMethods, (newMethods) => {
|
||||||
@@ -1246,26 +1130,17 @@ const filteredCategoryList = computed(() => {
|
|||||||
|
|
||||||
// ====== 合计 ======
|
// ====== 合计 ======
|
||||||
const totalAmountCalc = computed(() => {
|
const totalAmountCalc = computed(() => {
|
||||||
const itemTotal = selectedItems.value.reduce((sum, item) => {
|
const total = selectedItems.value.reduce((sum, item) => {
|
||||||
const effectivePrice = getSelectedItemAmount(item);
|
const effectivePrice = getSelectedItemAmount(item);
|
||||||
return sum + (effectivePrice * (item.quantity || 1));
|
return sum + (effectivePrice * (item.quantity || 1));
|
||||||
}, 0);
|
}, 0);
|
||||||
const methodTotal = selectedMethods.value.reduce((sum, method) => {
|
|
||||||
return sum + Number(method?.packagePrice ?? method?.price ?? 0);
|
|
||||||
}, 0);
|
|
||||||
const total = itemTotal + methodTotal;
|
|
||||||
return total.toFixed(2);
|
return total.toFixed(2);
|
||||||
});
|
});
|
||||||
|
|
||||||
// 监听已选项:自动更新申检部位
|
// 监听已选项:自动更新申检部位
|
||||||
watch(selectedItems, () => {
|
watch(selectedItems, () => {
|
||||||
form.inspectionArea = selectedItems.value.map(i => i.name).join('+');
|
form.inspectionArea = selectedItems.value.map(i => i.name).join('+');
|
||||||
form.isCharged = selectedItems.value.length > 0 || selectedMethods.value.length > 0 ? 1 : 0;
|
form.isCharged = selectedItems.value.length > 0 ? 1 : 0;
|
||||||
}, { deep: true });
|
|
||||||
|
|
||||||
watch(selectedMethods, () => {
|
|
||||||
form.isCharged = selectedItems.value.length > 0 || selectedMethods.value.length > 0 ? 1 : 0;
|
|
||||||
updateMethodDisplay();
|
|
||||||
}, { deep: true });
|
}, { deep: true });
|
||||||
|
|
||||||
// 监听患者变化
|
// 监听患者变化
|
||||||
@@ -1356,7 +1231,6 @@ function handleAdd() {
|
|||||||
selectedMethodDisplay: '' // Bug #384修复: 重置检查方法显示
|
selectedMethodDisplay: '' // Bug #384修复: 重置检查方法显示
|
||||||
});
|
});
|
||||||
selectedItems.value = [];
|
selectedItems.value = [];
|
||||||
selectedMethods.value = [];
|
|
||||||
resetCategoryChecked();
|
resetCategoryChecked();
|
||||||
activeDetailTab.value = 'applyForm';
|
activeDetailTab.value = 'applyForm';
|
||||||
// 自动加载临床诊断
|
// 自动加载临床诊断
|
||||||
@@ -1370,27 +1244,22 @@ function handleSave() {
|
|||||||
ElMessage.warning('请至少选择一个检查明细项目');
|
ElMessage.warning('请至少选择一个检查明细项目');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (selectedMethods.value.length === 0) {
|
// 检查每个项目是否已选择检查方法
|
||||||
ElMessage.warning('请选择检查方法');
|
const itemsWithoutMethod = selectedItems.value.filter(item => !item.selectedMethod);
|
||||||
|
if (itemsWithoutMethod.length > 0) {
|
||||||
|
const names = itemsWithoutMethod.map(item => item.name).join('、');
|
||||||
|
ElMessage.warning(`以下项目未选择检查方法:${names},请在右侧勾选后再保存`);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// 从已选项目推导检查类型编码(取第一个项目的 checkType,如 CT / ECG / GI)
|
// 从已选项目推导检查类型编码(取第一个项目的 checkType,如 CT / ECG / GI)
|
||||||
const firstCheckType = selectedItems.value[0]?.checkType || 'unknown';
|
const firstCheckType = selectedItems.value[0]?.checkType || 'unknown';
|
||||||
form.examTypeCode = firstCheckType;
|
form.examTypeCode = firstCheckType;
|
||||||
form.totalAmount = totalAmountCalc.value;
|
|
||||||
|
|
||||||
|
|
||||||
const primaryMethod = selectedMethods.value[0] || null;
|
|
||||||
const payload = {
|
const payload = {
|
||||||
...form,
|
...form,
|
||||||
encounterId: props.patientInfo?.encounterId || null,
|
encounterId: props.patientInfo?.encounterId || null,
|
||||||
patientIdNum: props.patientInfo?.patientId || null,
|
patientIdNum: props.patientInfo?.patientId || null,
|
||||||
checkMethods: selectedMethods.value.map((method) => ({
|
|
||||||
checkMethodId: method.id || null,
|
|
||||||
checkMethodName: method.name || null,
|
|
||||||
checkMethodCode: method.code || null,
|
|
||||||
checkMethodPackageName: method.packageName || null
|
|
||||||
})),
|
|
||||||
items: selectedItems.value.map((item, index) => ({
|
items: selectedItems.value.map((item, index) => ({
|
||||||
itemCode: String(item.id),
|
itemCode: String(item.id),
|
||||||
itemName: item.name,
|
itemName: item.name,
|
||||||
@@ -1400,10 +1269,10 @@ function handleSave() {
|
|||||||
itemStatus: 0,
|
itemStatus: 0,
|
||||||
itemSeq: index + 1,
|
itemSeq: index + 1,
|
||||||
// 检查方法信息
|
// 检查方法信息
|
||||||
checkMethodId: primaryMethod?.id || null,
|
checkMethodId: item.selectedMethod?.id || null,
|
||||||
checkMethodName: primaryMethod?.name || null,
|
checkMethodName: item.selectedMethod?.name || null,
|
||||||
checkMethodCode: primaryMethod?.code || null,
|
checkMethodCode: item.selectedMethod?.code || null,
|
||||||
checkMethodPackageName: primaryMethod?.packageName || null // Bug #384修复: 保存套餐名称
|
checkMethodPackageName: item.selectedMethod?.packageName || null // Bug #384修复: 保存套餐名称
|
||||||
}))
|
}))
|
||||||
};
|
};
|
||||||
request({
|
request({
|
||||||
@@ -1424,7 +1293,6 @@ function handleRowClick(row) {
|
|||||||
Object.assign(form, row);
|
Object.assign(form, row);
|
||||||
form.selectedMethodDisplay = ''; // Bug #384修复: 先清空,后面根据回充数据更新
|
form.selectedMethodDisplay = ''; // Bug #384修复: 先清空,后面根据回充数据更新
|
||||||
selectedItems.value = [];
|
selectedItems.value = [];
|
||||||
selectedMethods.value = [];
|
|
||||||
activeDetailTab.value = 'applyForm';
|
activeDetailTab.value = 'applyForm';
|
||||||
request({ url: `/exam/apply/${row.applyNo}`, method: 'get' }).then(async res => {
|
request({ url: `/exam/apply/${row.applyNo}`, method: 'get' }).then(async res => {
|
||||||
// 响应结构: Axios拦截器对code===200返回res.data(AjaxResult体)
|
// 响应结构: Axios拦截器对code===200返回res.data(AjaxResult体)
|
||||||
@@ -1449,9 +1317,8 @@ function handleRowClick(row) {
|
|||||||
methods: [],
|
methods: [],
|
||||||
selectedMethod: null,
|
selectedMethod: null,
|
||||||
expanded: false,
|
expanded: false,
|
||||||
projectFoldExpanded: false,
|
itemPackageExpanded: true,
|
||||||
methodFoldExpanded: false,
|
methodPackageExpanded: true,
|
||||||
methodPackageExpanded: false,
|
|
||||||
packageDetailsLoading: false,
|
packageDetailsLoading: false,
|
||||||
isPackage: false,
|
isPackage: false,
|
||||||
packageId: null,
|
packageId: null,
|
||||||
@@ -1495,21 +1362,7 @@ function handleRowClick(row) {
|
|||||||
return item;
|
return item;
|
||||||
}));
|
}));
|
||||||
// Bug #408修复: 确保明细数据正确加载到selectedItems
|
// Bug #408修复: 确保明细数据正确加载到selectedItems
|
||||||
const methodMap = new Map();
|
|
||||||
for (const item of itemsWithMethods) {
|
|
||||||
if (item.selectedMethod && !methodMap.has(String(item.selectedMethod.id))) {
|
|
||||||
methodMap.set(String(item.selectedMethod.id), {
|
|
||||||
...item.selectedMethod,
|
|
||||||
expanded: false,
|
|
||||||
packageLoading: false,
|
|
||||||
packageDetails: []
|
|
||||||
});
|
|
||||||
}
|
|
||||||
item.selectedMethod = null;
|
|
||||||
item.methodPackageDetails = [];
|
|
||||||
}
|
|
||||||
selectedItems.value = itemsWithMethods;
|
selectedItems.value = itemsWithMethods;
|
||||||
selectedMethods.value = Array.from(methodMap.values());
|
|
||||||
// 加载套餐明细(单个失败不影响其他项目和明细显示)
|
// 加载套餐明细(单个失败不影响其他项目和明细显示)
|
||||||
for (const it of selectedItems.value) {
|
for (const it of selectedItems.value) {
|
||||||
if (hasItemPackage(it)) {
|
if (hasItemPackage(it)) {
|
||||||
@@ -1519,17 +1372,14 @@ function handleRowClick(row) {
|
|||||||
console.error('加载套餐明细失败:', it.name, e);
|
console.error('加载套餐明细失败:', it.name, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
it.methodFoldExpanded = false;
|
if (hasMethodPackage(it) && !isSamePackage(it)) {
|
||||||
syncItemExpandedFlag(it);
|
|
||||||
}
|
|
||||||
for (const method of selectedMethods.value) {
|
|
||||||
if (hasStandaloneMethodPackage(method)) {
|
|
||||||
try {
|
try {
|
||||||
await loadStandaloneMethodPackageDetails(method);
|
await loadMethodPackageDetails(it, it.selectedMethod);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('加载检查方法套餐明细失败:', method.name, e);
|
console.error('加载检查方法套餐明细失败:', it.name, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
it.expanded = shouldShowPackageBody(it);
|
||||||
}
|
}
|
||||||
syncCategoryChecked();
|
syncCategoryChecked();
|
||||||
// Bug #384修复: 回充后更新检查方法显示
|
// Bug #384修复: 回充后更新检查方法显示
|
||||||
@@ -1615,9 +1465,8 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
methods: methods,
|
methods: methods,
|
||||||
selectedMethod: null,
|
selectedMethod: null,
|
||||||
expanded: false,
|
expanded: false,
|
||||||
projectFoldExpanded: false,
|
itemPackageExpanded: true,
|
||||||
methodFoldExpanded: false,
|
methodPackageExpanded: true,
|
||||||
methodPackageExpanded: false,
|
|
||||||
isPackage: !!(item.packageId || item.packageName),
|
isPackage: !!(item.packageId || item.packageName),
|
||||||
packageName: item.packageName || null,
|
packageName: item.packageName || null,
|
||||||
packageDetailsLoading: false,
|
packageDetailsLoading: false,
|
||||||
@@ -1626,13 +1475,19 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
};
|
};
|
||||||
selectedItems.value.push(newRow);
|
selectedItems.value.push(newRow);
|
||||||
// 必须用数组里的响应式行,不能继续改局部 newRow:push 后列表内是 proxy,改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
|
// 必须用数组里的响应式行,不能继续改局部 newRow:push 后列表内是 proxy,改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
|
||||||
|
const row = selectedItems.value[selectedItems.value.length - 1];
|
||||||
|
|
||||||
const rowJustAdded = selectedItems.value[selectedItems.value.length - 1];
|
// 勾选项目只加入项目列表,检查方法由用户在“检查方法”区域手动选择
|
||||||
syncItemExpandedFlag(rowJustAdded);
|
row.selectedMethod = null;
|
||||||
|
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
await nextTick();
|
|
||||||
form.totalAmount = totalAmountCalc.value;
|
// 新勾选项目后默认展开,直接展示检查方法状态和套餐明细
|
||||||
|
row.expanded = true;
|
||||||
|
row.itemPackageExpanded = true;
|
||||||
|
row.methodPackageExpanded = true;
|
||||||
|
if (hasItemPackage(row)) {
|
||||||
|
await loadPackageDetailsForItem(row);
|
||||||
|
}
|
||||||
|
|
||||||
// 自动回填执行科室:按检查项目类型 → 检查类型管理里配置的执行科室
|
// 自动回填执行科室:按检查项目类型 → 检查类型管理里配置的执行科室
|
||||||
if (selectedItems.value.length === 1 && cat?.performDeptName) {
|
if (selectedItems.value.length === 1 && cat?.performDeptName) {
|
||||||
@@ -1653,16 +1508,25 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
// Bug #382 修复:移除自动切换页签逻辑,保持当前页签状态
|
// Bug #382 修复:移除自动切换页签逻辑,保持当前页签状态
|
||||||
}
|
}
|
||||||
|
|
||||||
/** expanded 与各折叠条保持一致(明细表等仍可依赖 expanded) */
|
// Bug #384修复 + #426修复: 展开/收起项目卡片
|
||||||
function syncItemExpandedFlag(row) {
|
async function toggleItemExpand(item) {
|
||||||
if (!row) return;
|
item.expanded = !item.expanded;
|
||||||
row.expanded = !!(row.projectFoldExpanded || row.methodFoldExpanded);
|
if (item.expanded && hasItemPackage(item) && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
||||||
|
await loadPackageDetailsForItem(item);
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
item.expanded &&
|
||||||
|
shouldShowMethodPackageBody(item) &&
|
||||||
|
getMethodPackageDetailsList(item).length === 0 &&
|
||||||
|
!item.methodPackageLoading
|
||||||
|
) {
|
||||||
|
await loadMethodPackageDetails(item, item.selectedMethod);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function toggleProjectFold(item) {
|
async function toggleItemPackageExpand(item) {
|
||||||
item.projectFoldExpanded = !item.projectFoldExpanded;
|
item.itemPackageExpanded = !item.itemPackageExpanded;
|
||||||
syncItemExpandedFlag(item);
|
if (item.itemPackageExpanded && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
||||||
if (item.projectFoldExpanded && hasItemPackage(item) && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
|
||||||
await loadPackageDetailsForItem(item);
|
await loadPackageDetailsForItem(item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1679,64 +1543,25 @@ async function toggleMethodPackageExpand(item) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function toggleSelectedMethodFold(method) {
|
// Bug #384修复: 勾选框选择检查方法(单选逻辑)
|
||||||
method.expanded = !method.expanded;
|
async function selectMethodCheckbox(checked, item, method) {
|
||||||
if (
|
if (checked) {
|
||||||
method.expanded &&
|
item.selectedMethod = method;
|
||||||
hasStandaloneMethodPackage(method) &&
|
item.expanded = true;
|
||||||
getStandaloneMethodPackageDetailsList(method).length === 0 &&
|
item.methodPackageExpanded = true;
|
||||||
!method.packageLoading
|
// 动态加载该方法对应的套餐明细
|
||||||
) {
|
await loadMethodPackageDetails(item, method);
|
||||||
await loadStandaloneMethodPackageDetails(method);
|
} else {
|
||||||
|
item.selectedMethod = null;
|
||||||
|
item.methodPackageDetails = [];
|
||||||
}
|
}
|
||||||
}
|
// 联动更新表单检查方法显示字段
|
||||||
|
|
||||||
function handleRemoveMethod(idx) {
|
|
||||||
selectedMethods.value.splice(idx, 1);
|
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
}
|
|
||||||
|
|
||||||
async function loadStandaloneMethodPackageDetails(method) {
|
// #430: 套餐金额实时同步到申请单
|
||||||
method.packageLoading = true;
|
nextTick(() => {
|
||||||
method.packageDetails = [];
|
form.totalAmount = totalAmountCalc.value;
|
||||||
try {
|
});
|
||||||
let packageId = method.packageId;
|
|
||||||
if (!packageId && !method.packageName) {
|
|
||||||
method.packageLoading = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!packageId && method.packageName) {
|
|
||||||
const pkgRes = await listCheckPackage({ packageName: method.packageName });
|
|
||||||
let packages = pkgRes?.data || [];
|
|
||||||
if (!Array.isArray(packages)) {
|
|
||||||
packages = packages.records || packages.data || [];
|
|
||||||
}
|
|
||||||
if (packages.length === 0) {
|
|
||||||
method.packageLoading = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
packageId = packages[0].id;
|
|
||||||
method.packageId = packageId;
|
|
||||||
}
|
|
||||||
const detailRes = await request({
|
|
||||||
url: `/system/check-type/package/${packageId}/details`,
|
|
||||||
method: 'get'
|
|
||||||
});
|
|
||||||
method.packageDetails = parsePackageDetailsPayload(detailRes).map(d => ({
|
|
||||||
id: d.id,
|
|
||||||
name: d.name || d.itemName,
|
|
||||||
quantity: d.quantity || 1,
|
|
||||||
unit: d.unit || '次',
|
|
||||||
price: d.price ?? d.unitPrice ?? d.itemPrice ?? 0,
|
|
||||||
amount: d.amount || d.total || 0,
|
|
||||||
checked: true
|
|
||||||
}));
|
|
||||||
} catch (err) {
|
|
||||||
console.error('加载检查方法套餐明细失败:', err);
|
|
||||||
method.packageDetails = [];
|
|
||||||
} finally {
|
|
||||||
method.packageLoading = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 根据检查方法的packageName加载对应的套餐明细
|
// 根据检查方法的packageName加载对应的套餐明细
|
||||||
@@ -1798,12 +1623,9 @@ async function onDetailMethodChange(row, val) {
|
|||||||
}
|
}
|
||||||
row.methodPackageDetails = [];
|
row.methodPackageDetails = [];
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
const open = shouldShowPackageBody(row);
|
row.expanded = shouldShowPackageBody(row);
|
||||||
row.expanded = open;
|
row.itemPackageExpanded = true;
|
||||||
row.projectFoldExpanded = shouldShowItemPackageBody(row) && open;
|
row.methodPackageExpanded = true;
|
||||||
row.methodFoldExpanded = shouldShowMethodPackageBody(row) && open;
|
|
||||||
row.methodPackageExpanded = false;
|
|
||||||
syncItemExpandedFlag(row);
|
|
||||||
if (hasItemPackage(row)) {
|
if (hasItemPackage(row)) {
|
||||||
await loadPackageDetailsForItem(row);
|
await loadPackageDetailsForItem(row);
|
||||||
}
|
}
|
||||||
@@ -1815,13 +1637,26 @@ async function onDetailMethodChange(row, val) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bug #384修复: 更新检查方法显示字段(取自独立已选检查方法)
|
// Bug #384修复: 更新检查方法显示字段(联动)
|
||||||
function updateMethodDisplay() {
|
function updateMethodDisplay() {
|
||||||
if (selectedMethods.value.length > 0) {
|
// 找到第一个有选中检查方法的项目
|
||||||
form.selectedMethodDisplay = selectedMethods.value.map((method) => method.name).join('、');
|
const itemWithMethod = selectedItems.value.find(item => item.selectedMethod);
|
||||||
return;
|
if (itemWithMethod?.selectedMethod) {
|
||||||
|
form.selectedMethodDisplay = itemWithMethod.selectedMethod.name; // 显示检查方法名称,不显示套餐名称
|
||||||
|
} else {
|
||||||
|
form.selectedMethodDisplay = '';
|
||||||
}
|
}
|
||||||
form.selectedMethodDisplay = '';
|
}
|
||||||
|
|
||||||
|
// 选择检查方法
|
||||||
|
function selectMethod(item, method) {
|
||||||
|
if (item.selectedMethod?.id === method.id) {
|
||||||
|
item.selectedMethod = null;
|
||||||
|
} else {
|
||||||
|
item.selectedMethod = method;
|
||||||
|
}
|
||||||
|
// Bug #384修复: 联动更新表单检查方法显示字段
|
||||||
|
updateMethodDisplay();
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleRemoveItem(idx, item) {
|
function handleRemoveItem(idx, item) {
|
||||||
@@ -1835,7 +1670,7 @@ function handleRemoveItem(idx, item) {
|
|||||||
if (selectedItems.value.length === 0) {
|
if (selectedItems.value.length === 0) {
|
||||||
form.performDeptCode = '';
|
form.performDeptCode = '';
|
||||||
form.examTypeCode = '';
|
form.examTypeCode = '';
|
||||||
updateMethodDisplay();
|
form.selectedMethodDisplay = ''; // Bug #384修复: 清空检查方法显示
|
||||||
} else {
|
} else {
|
||||||
// Bug #384修复: 移除后重新计算检查方法显示
|
// Bug #384修复: 移除后重新计算检查方法显示
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
@@ -2141,167 +1976,39 @@ defineExpose({ getList });
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 项目上 / 方法下:各自独立下拉条 */
|
.selected-item-card .card-header {
|
||||||
.fold-strip {
|
|
||||||
border-bottom: 1px solid var(--el-border-color-lighter);
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-strip:last-child {
|
|
||||||
border-bottom: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-strip-header {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: flex-start;
|
align-items: center;
|
||||||
gap: 8px;
|
|
||||||
padding: 10px 10px;
|
padding: 10px 10px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
gap: 8px;
|
||||||
background: linear-gradient(180deg, #f8fafc 0%, #f0f4f8 100%);
|
background: linear-gradient(180deg, #f8fafc 0%, #f0f4f8 100%);
|
||||||
|
border-bottom: 1px solid transparent;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fold-strip-header:hover {
|
.selected-item-card .card-header:hover {
|
||||||
background: linear-gradient(180deg, #ecf5ff 0%, #e3eef8 100%);
|
background: linear-gradient(180deg, #ecf5ff 0%, #e3eef8 100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.fold-strip-method.is-method-target .fold-strip-header {
|
.selected-item-card.is-expanded .card-header {
|
||||||
background: linear-gradient(180deg, #e8f3ff 0%, #dceaff 100%);
|
border-bottom-color: #ebeef5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fold-chevron {
|
.card-name {
|
||||||
font-size: 14px;
|
|
||||||
color: #909399;
|
|
||||||
transition: transform 0.2s ease;
|
|
||||||
flex-shrink: 0;
|
|
||||||
margin-top: 2px;
|
|
||||||
transform: rotate(-90deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-chevron.open {
|
|
||||||
transform: rotate(0deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-header-main {
|
|
||||||
flex: 1;
|
flex: 1;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-kicker {
|
|
||||||
font-size: 11px;
|
|
||||||
font-weight: 600;
|
|
||||||
color: #909399;
|
|
||||||
letter-spacing: 0.03em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-title {
|
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
font-weight: 600;
|
font-weight: 500;
|
||||||
color: #303133;
|
color: #303133;
|
||||||
line-height: 1.35;
|
line-height: 1.4;
|
||||||
word-break: break-word;
|
word-break: break-word;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fold-title-plain {
|
.card-price {
|
||||||
font-weight: 500;
|
|
||||||
color: #606266;
|
|
||||||
}
|
|
||||||
|
|
||||||
.line-clamp-2 {
|
|
||||||
display: -webkit-box;
|
|
||||||
-webkit-line-clamp: 2;
|
|
||||||
-webkit-box-orient: vertical;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-price-strong {
|
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
color: #409eff;
|
color: #409eff;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
margin-top: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-price-strong.warn {
|
|
||||||
color: #e6a23c;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-strip-body {
|
|
||||||
background: #fafbfc;
|
|
||||||
padding: 0 10px 10px 36px;
|
|
||||||
border-top: 1px dashed var(--el-border-color-lighter);
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-package-wrap {
|
|
||||||
padding-top: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.fold-strip-muted {
|
|
||||||
font-size: 12px;
|
|
||||||
color: #909399;
|
|
||||||
padding: 10px 0 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.selected-global-method-picker {
|
|
||||||
flex-shrink: 0;
|
|
||||||
margin-top: 8px;
|
|
||||||
border-radius: 6px;
|
|
||||||
border: 1px solid #e4e7ed;
|
|
||||||
background: #fff;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.method-picker-collapse-title {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 10px;
|
|
||||||
padding: 10px 10px;
|
|
||||||
cursor: pointer;
|
|
||||||
background: #fff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.method-picker-collapse-title:hover {
|
|
||||||
background: #f5f7fa;
|
|
||||||
}
|
|
||||||
|
|
||||||
.method-picker-title-main {
|
|
||||||
flex: 1;
|
|
||||||
min-width: 0;
|
|
||||||
font-size: 13px;
|
|
||||||
font-weight: 600;
|
|
||||||
color: #303133;
|
|
||||||
}
|
|
||||||
|
|
||||||
.global-method-picker-scope {
|
|
||||||
font-size: 12px;
|
|
||||||
color: #909399;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.method-picker-arrow {
|
|
||||||
font-size: 14px;
|
|
||||||
color: #909399;
|
|
||||||
transition: transform 0.2s ease;
|
|
||||||
flex-shrink: 0;
|
|
||||||
transform: rotate(-90deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
.method-picker-arrow.expanded {
|
|
||||||
transform: rotate(0deg);
|
|
||||||
}
|
|
||||||
|
|
||||||
.global-method-picker-list {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 0;
|
|
||||||
padding: 6px 8px 8px;
|
|
||||||
border-top: 1px solid #ebeef5;
|
|
||||||
}
|
|
||||||
|
|
||||||
.method-picker-row {
|
|
||||||
padding: 6px 4px;
|
|
||||||
border-radius: 3px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.expand-icon {
|
.expand-icon {
|
||||||
@@ -2351,6 +2058,11 @@ defineExpose({ getList });
|
|||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* 展开区域 */
|
||||||
|
.selected-card-body {
|
||||||
|
background: #fafbfc;
|
||||||
|
}
|
||||||
|
|
||||||
.selected-card-section {
|
.selected-card-section {
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
border-bottom: 1px solid #ebeef5;
|
border-bottom: 1px solid #ebeef5;
|
||||||
@@ -2400,49 +2112,6 @@ defineExpose({ getList });
|
|||||||
color: #409eff;
|
color: #409eff;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 收起态:仅展示折叠箭头,不显示「套餐明细」等冗余标题 */
|
|
||||||
.package-toggle-minimal {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: flex-start;
|
|
||||||
padding: 8px 10px;
|
|
||||||
font-size: 12px;
|
|
||||||
color: var(--el-text-color-secondary);
|
|
||||||
cursor: pointer;
|
|
||||||
border-bottom: 1px dashed #e4e7ed;
|
|
||||||
background: #fafafa;
|
|
||||||
}
|
|
||||||
|
|
||||||
.package-toggle-minimal:hover {
|
|
||||||
color: #409eff;
|
|
||||||
background: #f5f9ff;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
.nested-empty-text {
|
|
||||||
font-size: 12px;
|
|
||||||
color: var(--el-text-color-placeholder);
|
|
||||||
padding-left: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nested-label-row {
|
|
||||||
margin-bottom: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nested-label {
|
|
||||||
font-size: 11px;
|
|
||||||
font-weight: 600;
|
|
||||||
color: var(--el-text-color-secondary);
|
|
||||||
letter-spacing: 0.03em;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
.method-label-inner {
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.package-details-loading,
|
.package-details-loading,
|
||||||
.package-details-empty {
|
.package-details-empty {
|
||||||
padding: 12px 10px;
|
padding: 12px 10px;
|
||||||
|
|||||||
@@ -179,8 +179,8 @@
|
|||||||
type="datetime"
|
type="datetime"
|
||||||
placeholder="选择执行时间"
|
placeholder="选择执行时间"
|
||||||
size="small"
|
size="small"
|
||||||
format="YYYY-MM-DD HH:mm"
|
format="YYYY-MM-DD HH:mm:ss"
|
||||||
value-format="YYYY-MM-DD HH:mm"
|
value-format="YYYY-MM-DD HH:mm:ss"
|
||||||
style="width: 100%"
|
style="width: 100%"
|
||||||
/>
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
@@ -445,6 +445,7 @@
|
|||||||
>
|
>
|
||||||
<el-table-column label="项目名称" prop="itemName" min-width="180">
|
<el-table-column label="项目名称" prop="itemName" min-width="180">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
|
<el-tag v-if="scope.row.isPackage" size="small" type="warning" style="margin-right: 4px">套餐</el-tag>
|
||||||
<span :style="{ fontWeight: scope.row.isPackage ? 'bold' : 'normal' }">
|
<span :style="{ fontWeight: scope.row.isPackage ? 'bold' : 'normal' }">
|
||||||
{{ scope.row.itemName }}
|
{{ scope.row.itemName }}
|
||||||
</span>
|
</span>
|
||||||
@@ -562,6 +563,7 @@
|
|||||||
@change="toggleInspectionItem(item)"
|
@change="toggleInspectionItem(item)"
|
||||||
@click.stop
|
@click.stop
|
||||||
/>
|
/>
|
||||||
|
<el-tag v-if="item.isPackage" size="small" type="warning" style="margin-right: 4px">套餐</el-tag>
|
||||||
<span class="item-itemName">{{ item.itemName }}</span>
|
<span class="item-itemName">{{ item.itemName }}</span>
|
||||||
<span class="item-price">¥{{ item.itemPrice }}/{{ item.unit || "次" }}</span>
|
<span class="item-price">¥{{ item.itemPrice }}/{{ item.unit || "次" }}</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -612,6 +614,7 @@
|
|||||||
<template v-if="item.isPackage">{{ item.expanded ? '▼' : '▶' }}</template>
|
<template v-if="item.isPackage">{{ item.expanded ? '▼' : '▶' }}</template>
|
||||||
<template v-else>•</template>
|
<template v-else>•</template>
|
||||||
</span>
|
</span>
|
||||||
|
<el-tag v-if="item.isPackage" size="small" type="warning" style="margin-right: 4px">套餐</el-tag>
|
||||||
<span class="item-itemName">{{ item.itemName }}</span>
|
<span class="item-itemName">{{ item.itemName }}</span>
|
||||||
<span class="item-price">¥{{ item.itemPrice }}/{{ item.unit || "次" }}</span>
|
<span class="item-price">¥{{ item.itemPrice }}/{{ item.unit || "次" }}</span>
|
||||||
<el-button
|
<el-button
|
||||||
@@ -872,30 +875,6 @@ let applyTimeTimer = null
|
|||||||
const userStore = useUserStore()
|
const userStore = useUserStore()
|
||||||
const { id: userId, name: userName, nickName: userNickName } = storeToRefs(userStore)
|
const { id: userId, name: userName, nickName: userNickName } = storeToRefs(userStore)
|
||||||
|
|
||||||
/** 执行时间默认值:当前系统时间,精确到分钟 */
|
|
||||||
const getDefaultExecuteTime = () => {
|
|
||||||
const d = new Date()
|
|
||||||
const year = d.getFullYear()
|
|
||||||
const month = String(d.getMonth() + 1).padStart(2, '0')
|
|
||||||
const day = String(d.getDate()).padStart(2, '0')
|
|
||||||
const hours = String(d.getHours()).padStart(2, '0')
|
|
||||||
const minutes = String(d.getMinutes()).padStart(2, '0')
|
|
||||||
return `${year}-${month}-${day} ${hours}:${minutes}`
|
|
||||||
}
|
|
||||||
|
|
||||||
/** 将后端时间规范为 YYYY-MM-DD HH:mm */
|
|
||||||
const normalizeExecuteTime = (value) => {
|
|
||||||
if (!value) return getDefaultExecuteTime()
|
|
||||||
const d = new Date(String(value).replace(/-/g, '/'))
|
|
||||||
if (Number.isNaN(d.getTime())) return getDefaultExecuteTime()
|
|
||||||
const year = d.getFullYear()
|
|
||||||
const month = String(d.getMonth() + 1).padStart(2, '0')
|
|
||||||
const day = String(d.getDate()).padStart(2, '0')
|
|
||||||
const hours = String(d.getHours()).padStart(2, '0')
|
|
||||||
const minutes = String(d.getMinutes()).padStart(2, '0')
|
|
||||||
return `${year}-${month}-${day} ${hours}:${minutes}`
|
|
||||||
}
|
|
||||||
|
|
||||||
// 修改 initData 函数
|
// 修改 initData 函数
|
||||||
const initData = async () => {
|
const initData = async () => {
|
||||||
// 先初始化患者信息(如果有)
|
// 先初始化患者信息(如果有)
|
||||||
@@ -918,10 +897,6 @@ const initData = async () => {
|
|||||||
formData.applyNo = '自动生成'
|
formData.applyNo = '自动生成'
|
||||||
// 申请日期实时更新(启动定时器)
|
// 申请日期实时更新(启动定时器)
|
||||||
startApplyTimeTimer()
|
startApplyTimeTimer()
|
||||||
// 执行时间默认当前系统时间(精确到分钟)
|
|
||||||
if (!formData.executeTime) {
|
|
||||||
formData.executeTime = getDefaultExecuteTime()
|
|
||||||
}
|
|
||||||
|
|
||||||
// 执行时间默认填充当前系统时间
|
// 执行时间默认填充当前系统时间
|
||||||
formData.executeTime = formatDateTime(new Date())
|
formData.executeTime = formatDateTime(new Date())
|
||||||
@@ -1003,7 +978,7 @@ const formData = reactive({
|
|||||||
applyDeptCode: '',
|
applyDeptCode: '',
|
||||||
specimenName: '血液',
|
specimenName: '血液',
|
||||||
encounterId: '',
|
encounterId: '',
|
||||||
executeTime: getDefaultExecuteTime(),
|
executeTime: null,
|
||||||
applicationType: 0
|
applicationType: 0
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -1575,7 +1550,7 @@ const resetForm = async () => {
|
|||||||
visitNo: '',
|
visitNo: '',
|
||||||
specimenName: '血液',
|
specimenName: '血液',
|
||||||
encounterId: props.patientInfo.encounterId || '',
|
encounterId: props.patientInfo.encounterId || '',
|
||||||
executeTime: getDefaultExecuteTime(),
|
executeTime: null,
|
||||||
applicationType: 0,
|
applicationType: 0,
|
||||||
})
|
})
|
||||||
selectedInspectionItems.value = []
|
selectedInspectionItems.value = []
|
||||||
@@ -2013,7 +1988,7 @@ const loadApplicationToForm = async (row) => {
|
|||||||
visitNo: detail.visitNo,
|
visitNo: detail.visitNo,
|
||||||
specimenName: detail.specimenName,
|
specimenName: detail.specimenName,
|
||||||
encounterId: detail.encounterId,
|
encounterId: detail.encounterId,
|
||||||
executeTime: normalizeExecuteTime(detail.executeTime),
|
executeTime: detail.executeTime || null,
|
||||||
applicationType: detail.applicationType ?? 0
|
applicationType: detail.applicationType ?? 0
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -89,14 +89,8 @@ const getList = async () => {
|
|||||||
const response = await listPendingEmr(queryParams)
|
const response = await listPendingEmr(queryParams)
|
||||||
// 根据后端返回的数据结构调整
|
// 根据后端返回的数据结构调整
|
||||||
if (response.code === 200) {
|
if (response.code === 200) {
|
||||||
const data = response.data
|
emrList.value = response.data || []
|
||||||
if (data && data.rows !== undefined) {
|
total.value = Array.isArray(response.data) ? response.data.length : 0
|
||||||
emrList.value = data.rows || []
|
|
||||||
total.value = data.total || 0
|
|
||||||
} else {
|
|
||||||
emrList.value = Array.isArray(data) ? data : []
|
|
||||||
total.value = emrList.value.length
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
ElMessage.error(response.msg || '获取待写病历列表失败')
|
ElMessage.error(response.msg || '获取待写病历列表失败')
|
||||||
emrList.value = []
|
emrList.value = []
|
||||||
|
|||||||
@@ -18,12 +18,16 @@
|
|||||||
<!-- <el-table-column label="组套类型" align="center" prop="typeEnum_enumText" /> -->
|
<!-- <el-table-column label="组套类型" align="center" prop="typeEnum_enumText" /> -->
|
||||||
<el-table-column label="单次剂量" align="center" prop="rangeCode_dictText">
|
<el-table-column label="单次剂量" align="center" prop="rangeCode_dictText">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
{{ formatHistoryDose(scope.row) }}
|
{{
|
||||||
|
scope.row.dose
|
||||||
|
? formatNumber(scope.row.dose) + ' ' + scope.row.doseUnitCode_dictText
|
||||||
|
: ''
|
||||||
|
}}
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="总量" align="center" prop="rangeCode_dictText">
|
<el-table-column label="总量" align="center" prop="rangeCode_dictText">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
{{ formatHistoryTotalQuantity(scope.row) }}
|
{{ scope.row.quantity ? scope.row.quantity + ' ' + scope.row.unitCode_dictText : '' }}
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="频次/用法" align="center" prop="rangeCode_dictText" width="200">
|
<el-table-column label="频次/用法" align="center" prop="rangeCode_dictText" width="200">
|
||||||
@@ -86,31 +90,6 @@ const queryParams = ref({
|
|||||||
typeEnum: 1,
|
typeEnum: 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
function formatHistoryTotalQuantity(row) {
|
|
||||||
if (!row || row.quantity == null || row.quantity === '') return '';
|
|
||||||
const fromDict = row.unitCode_dictText ?? row.minUnitCode_dictText ?? row.unitCodeName;
|
|
||||||
let u =
|
|
||||||
fromDict != null && String(fromDict).trim() !== ''
|
|
||||||
&& String(fromDict).toLowerCase() !== 'null'
|
|
||||||
? String(fromDict).trim()
|
|
||||||
: '';
|
|
||||||
if (!u) {
|
|
||||||
const t = Number(row.adviceType);
|
|
||||||
if (t === 3 || t === 6 || t === 23 || t === 5) u = '次';
|
|
||||||
else if (t === 4) u = '个';
|
|
||||||
}
|
|
||||||
return u ? `${row.quantity} ${u}` : String(row.quantity);
|
|
||||||
}
|
|
||||||
|
|
||||||
function formatHistoryDose(row) {
|
|
||||||
if (!row?.dose) return '';
|
|
||||||
const du = row.doseUnitCode_dictText;
|
|
||||||
if (du != null && String(du).trim() !== '' && String(du).toLowerCase() !== 'null') {
|
|
||||||
return formatNumber(row.dose) + ' ' + du;
|
|
||||||
}
|
|
||||||
return formatNumber(row.dose);
|
|
||||||
}
|
|
||||||
|
|
||||||
function handleOpen() {
|
function handleOpen() {
|
||||||
drawer.value = true;
|
drawer.value = true;
|
||||||
getList();
|
getList();
|
||||||
|
|||||||
@@ -82,7 +82,7 @@
|
|||||||
<span>{{ index + 1 + '. ' }}</span>
|
<span>{{ index + 1 + '. ' }}</span>
|
||||||
<span>{{ medItem.adviceName }}</span>
|
<span>{{ medItem.adviceName }}</span>
|
||||||
<span>{{ '(' + medItem.volume + ')' }}</span>
|
<span>{{ '(' + medItem.volume + ')' }}</span>
|
||||||
<span>{{ formatPrintLineQuantity(medItem) }}</span>
|
<span>{{ medItem.quantity + ' ' + medItem.unitCode_dictText }}</span>
|
||||||
<span>{{ '批次号:' + medItem.lotNumber }}</span>
|
<span>{{ '批次号:' + medItem.lotNumber }}</span>
|
||||||
<div>
|
<div>
|
||||||
<span>用法用量:</span>
|
<span>用法用量:</span>
|
||||||
@@ -161,22 +161,6 @@ const props = defineProps({
|
|||||||
const emit = defineEmits(['close']);
|
const emit = defineEmits(['close']);
|
||||||
|
|
||||||
//合计
|
//合计
|
||||||
function formatPrintLineQuantity(row) {
|
|
||||||
if (row == null || row.quantity == null || row.quantity === '') return '';
|
|
||||||
const fromDict = row.unitCode_dictText ?? row.minUnitCode_dictText ?? row.unitCodeName;
|
|
||||||
let u =
|
|
||||||
fromDict != null && String(fromDict).trim() !== ''
|
|
||||||
&& String(fromDict).toLowerCase() !== 'null'
|
|
||||||
? String(fromDict).trim()
|
|
||||||
: '';
|
|
||||||
if (!u) {
|
|
||||||
const t = Number(row.adviceType);
|
|
||||||
if (t === 3 || t === 6 || t === 23 || t === 5) u = '次';
|
|
||||||
else if (t === 4) u = '个';
|
|
||||||
}
|
|
||||||
return u ? `${row.quantity} ${u}` : String(row.quantity);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTotalPrice(item) {
|
function getTotalPrice(item) {
|
||||||
let totalPrice = new Decimal(0);
|
let totalPrice = new Decimal(0);
|
||||||
item.prescriptionInfoDetailList.forEach((medItem) => {
|
item.prescriptionInfoDetailList.forEach((medItem) => {
|
||||||
|
|||||||
@@ -686,15 +686,7 @@
|
|||||||
<span style="margin-left: 4px">{{ scope.row.doseUnitCode_dictText }}</span>
|
<span style="margin-left: 4px">{{ scope.row.doseUnitCode_dictText }}</span>
|
||||||
</template>
|
</template>
|
||||||
<span v-else>
|
<span v-else>
|
||||||
{{
|
{{ scope.row.dose ? scope.row.dose + ' ' + scope.row.doseUnitCode_dictText : '' }}
|
||||||
scope.row.dose
|
|
||||||
? scope.row.dose +
|
|
||||||
(scope.row.doseUnitCode_dictText &&
|
|
||||||
String(scope.row.doseUnitCode_dictText).toLowerCase() !== 'null'
|
|
||||||
? ' ' + scope.row.doseUnitCode_dictText
|
|
||||||
: '')
|
|
||||||
: ''
|
|
||||||
}}
|
|
||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
@@ -711,10 +703,10 @@
|
|||||||
@change="calculateTotalPrice(scope.row, scope.$index)"
|
@change="calculateTotalPrice(scope.row, scope.$index)"
|
||||||
@input="calculateTotalPrice(scope.row, scope.$index)"
|
@input="calculateTotalPrice(scope.row, scope.$index)"
|
||||||
/>
|
/>
|
||||||
<span style="margin-left: 4px">{{ resolveTotalQuantityUnit(scope.row) }}</span>
|
<span style="margin-left: 4px">{{ scope.row.unitCode_dictText }}</span>
|
||||||
</template>
|
</template>
|
||||||
<span v-else>
|
<span v-else>
|
||||||
{{ formatTotalQuantityWithUnit(scope.row) }}
|
{{ scope.row.quantity ? scope.row.quantity + ' ' + scope.row.unitCode_dictText : '' }}
|
||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
@@ -925,39 +917,6 @@ const unitMap = ref({
|
|||||||
minUnit: 'minUnit',
|
minUnit: 'minUnit',
|
||||||
unit: 'unit',
|
unit: 'unit',
|
||||||
});
|
});
|
||||||
|
|
||||||
/** 解析总量单位文案(无字典时:诊疗/手术/检查默认「次」,耗材默认「个」) */
|
|
||||||
const resolveTotalQuantityUnit = (row) => {
|
|
||||||
if (row == null) return '';
|
|
||||||
const fromDict =
|
|
||||||
row.unitCode_dictText ?? row.minUnitCode_dictText ?? row.unitCodeName;
|
|
||||||
let unitStr =
|
|
||||||
fromDict != null && String(fromDict).trim() !== ''
|
|
||||||
&& String(fromDict).toLowerCase() !== 'null'
|
|
||||||
? String(fromDict).trim()
|
|
||||||
: '';
|
|
||||||
if (!unitStr) {
|
|
||||||
const t = Number(row.adviceType);
|
|
||||||
// drord_doctor_type: 3=诊疗 4=耗材 5=会诊 6=手术;23=检查(特殊)
|
|
||||||
// 注意:2=中成药(药品),不可用「次」作为默认单位
|
|
||||||
if (t === 3 || t === 6 || t === 23 || t === 5) {
|
|
||||||
unitStr = '次';
|
|
||||||
} else if (t === 4) {
|
|
||||||
unitStr = '个';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return unitStr;
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 总量列展示:避免 unitCode_dictText 为空时显示「1 null」 */
|
|
||||||
const formatTotalQuantityWithUnit = (row) => {
|
|
||||||
if (row == null) return '';
|
|
||||||
const q = row.quantity;
|
|
||||||
if (q === undefined || q === null || q === '') return '';
|
|
||||||
const unitStr = resolveTotalQuantityUnit(row);
|
|
||||||
return unitStr ? `${q} ${unitStr}` : String(q);
|
|
||||||
};
|
|
||||||
|
|
||||||
const buttonDisabled = computed(() => {
|
const buttonDisabled = computed(() => {
|
||||||
return !props.patientInfo;
|
return !props.patientInfo;
|
||||||
});
|
});
|
||||||
@@ -2755,8 +2714,7 @@ function handleEmrTreatment() {
|
|||||||
treatment += '诊疗[' + (index + 1) + ']' + ' ';
|
treatment += '诊疗[' + (index + 1) + ']' + ' ';
|
||||||
treatment += item.adviceName + ' ';
|
treatment += item.adviceName + ' ';
|
||||||
if (item.quantity) {
|
if (item.quantity) {
|
||||||
const u = resolveTotalQuantityUnit(item);
|
treatment += '数量:' + item.quantity + item.unitCode_dictText + ' ';
|
||||||
treatment += '数量:' + item.quantity + (u ? ' ' + u : '') + ' ';
|
|
||||||
}
|
}
|
||||||
treatment += '频次:' + item.rateCode_dictText + ' ';
|
treatment += '频次:' + item.rateCode_dictText + ' ';
|
||||||
if (item.methodCode_dictText) {
|
if (item.methodCode_dictText) {
|
||||||
|
|||||||
@@ -113,17 +113,10 @@ const getList = async () => {
|
|||||||
loading.value = true
|
loading.value = true
|
||||||
try {
|
try {
|
||||||
const response = await listPendingEmr(queryParams)
|
const response = await listPendingEmr(queryParams)
|
||||||
|
// 根据后端返回的数据结构调整
|
||||||
if (response.code === 200) {
|
if (response.code === 200) {
|
||||||
const data = response.data
|
emrList.value = response.data || []
|
||||||
if (data && data.rows !== undefined) {
|
total.value = Array.isArray(response.data) ? response.data.length : 0
|
||||||
// 新分页格式 {rows, total}
|
|
||||||
emrList.value = data.rows || []
|
|
||||||
total.value = data.total || 0
|
|
||||||
} else {
|
|
||||||
// 兼容旧格式(数组)
|
|
||||||
emrList.value = Array.isArray(data) ? data : []
|
|
||||||
total.value = emrList.value.length
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
ElMessage.error(response.msg || '获取待写病历列表失败')
|
ElMessage.error(response.msg || '获取待写病历列表失败')
|
||||||
emrList.value = []
|
emrList.value = []
|
||||||
|
|||||||
@@ -58,11 +58,7 @@
|
|||||||
<el-table-column prop="busNo" label="单据号" align="center" width="150" />
|
<el-table-column prop="busNo" label="单据号" align="center" width="150" />
|
||||||
<el-table-column prop="applicantName" label="申请人" align="center" width="100" />
|
<el-table-column prop="applicantName" label="申请人" align="center" width="100" />
|
||||||
<el-table-column prop="locationName" label="发药药房" align="center" />
|
<el-table-column prop="locationName" label="发药药房" align="center" />
|
||||||
<el-table-column prop="statusEnum_enumText" label="状态" align="center">
|
<el-table-column prop="statusEnum_enumText" label="状态" align="center" />
|
||||||
<template #default="scope">
|
|
||||||
{{ formatSummaryStatusText(scope.row) }}
|
|
||||||
</template>
|
|
||||||
</el-table-column>
|
|
||||||
<el-table-column prop="applyTime" label="汇总日期" align="center" width="140">
|
<el-table-column prop="applyTime" label="汇总日期" align="center" width="140">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
{{ scope.row.applyTime ? parseTime(scope.row.applyTime, '{y}-{m}-{d}') : '-' }}
|
{{ scope.row.applyTime ? parseTime(scope.row.applyTime, '{y}-{m}-{d}') : '-' }}
|
||||||
@@ -143,32 +139,6 @@ import {getCurrentInstance, ref} from 'vue';
|
|||||||
import {getFromSummaryDetails, getFromSummaryInit, getFromSummaryList, totalSendDrug,} from './api.js';
|
import {getFromSummaryDetails, getFromSummaryInit, getFromSummaryList, totalSendDrug,} from './api.js';
|
||||||
|
|
||||||
const { proxy } = getCurrentInstance();
|
const { proxy } = getCurrentInstance();
|
||||||
|
|
||||||
/** 发药汇总单状态展示(汇总申请→已提交,发药→已发药) */
|
|
||||||
const SUMMARY_STATUS_DISPLAY = {
|
|
||||||
2: '已提交',
|
|
||||||
4: '已发药',
|
|
||||||
};
|
|
||||||
const LEGACY_SUMMARY_STATUS_TEXT = {
|
|
||||||
待配药: '已提交',
|
|
||||||
已发放: '已发药',
|
|
||||||
};
|
|
||||||
|
|
||||||
function formatSummaryStatusText(row) {
|
|
||||||
const code = Number(row?.statusEnum);
|
|
||||||
if (SUMMARY_STATUS_DISPLAY[code]) {
|
|
||||||
return SUMMARY_STATUS_DISPLAY[code];
|
|
||||||
}
|
|
||||||
return LEGACY_SUMMARY_STATUS_TEXT[row?.statusEnum_enumText] || row?.statusEnum_enumText || '-';
|
|
||||||
}
|
|
||||||
|
|
||||||
function mapSummaryStatusOptions(options = []) {
|
|
||||||
return options.map((item) => ({
|
|
||||||
...item,
|
|
||||||
label: SUMMARY_STATUS_DISPLAY[item.value] ?? LEGACY_SUMMARY_STATUS_TEXT[item.label] ?? item.label,
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
const statusEnumOptions = ref([]);
|
const statusEnumOptions = ref([]);
|
||||||
const summaryList = ref([]);
|
const summaryList = ref([]);
|
||||||
const queryParams = ref({
|
const queryParams = ref({
|
||||||
@@ -252,7 +222,7 @@ function handleSend(row) {
|
|||||||
const getStatusOption = async () => {
|
const getStatusOption = async () => {
|
||||||
try {
|
try {
|
||||||
const res = await getFromSummaryInit();
|
const res = await getFromSummaryInit();
|
||||||
statusEnumOptions.value = mapSummaryStatusOptions(res.data.dispenseStatusOptions);
|
statusEnumOptions.value = res.data.dispenseStatusOptions;
|
||||||
} catch (error) {}
|
} catch (error) {}
|
||||||
};
|
};
|
||||||
getStatusOption();
|
getStatusOption();
|
||||||
|
|||||||
@@ -23,7 +23,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import {computed, nextTick, ref} from 'vue';
|
import {computed, nextTick, onMounted, ref} from 'vue';
|
||||||
import {throttle} from 'lodash-es';
|
import {throttle} from 'lodash-es';
|
||||||
import Table from '@/components/TableLayout/Table.vue';
|
import Table from '@/components/TableLayout/Table.vue';
|
||||||
import {getAdviceBaseInfo} from './api';
|
import {getAdviceBaseInfo} from './api';
|
||||||
@@ -204,6 +204,11 @@ defineExpose({
|
|||||||
handleKeyDown,
|
handleKeyDown,
|
||||||
refresh,
|
refresh,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 组件挂载时自动加载数据(el-popover 懒渲染,父组件 refresh 可能因时序问题未生效,onMounted 最可靠)
|
||||||
|
onMounted(() => {
|
||||||
|
getList();
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
|
|||||||
@@ -250,11 +250,10 @@ export function getContract(params) {
|
|||||||
/**
|
/**
|
||||||
* 获取科室列表
|
* 获取科室列表
|
||||||
*/
|
*/
|
||||||
export function getOrgTree(params = {}) {
|
export function getOrgTree() {
|
||||||
return request({
|
return request({
|
||||||
url: '/base-data-manage/organization/organization',
|
url: '/base-data-manage/organization/organization',
|
||||||
method: 'get',
|
method: 'get',
|
||||||
params: { pageNo: 1, pageSize: 5000, ...params },
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -115,18 +115,20 @@
|
|||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column prop="requesterId_dictText" label="申请者" width="120" />
|
<el-table-column prop="requesterId_dictText" label="申请者" width="120" />
|
||||||
<el-table-column label="操作" align="center" fixed="right" width="280">
|
<el-table-column label="操作" align="center" fixed="right" width="220">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
|
<!-- 待签发:可修改、删除 -->
|
||||||
<template v-if="canManageRow(scope.row) && isPendingStatus(scope.row)">
|
<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>
|
||||||
<template v-if="canManageRow(scope.row) && isWithdrawableStatus(scope.row)">
|
<!-- 已签发:可撤回 -->
|
||||||
|
<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>
|
||||||
<template v-if="isReportStatus(scope.row)">
|
<!-- 已采证、已送检、报告已出、已作废:仅查看详情 -->
|
||||||
<el-button link type="success" @click="handleViewReport(scope.row)">查看报告</el-button>
|
<template v-else>
|
||||||
|
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
|
||||||
</template>
|
</template>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
@@ -226,10 +228,7 @@ import {patientInfo} from '../../store/patient.js';
|
|||||||
import {getInspection, deleteRequestForm, withdrawRequestForm, getProofResult} 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 useUserStore from '@/store/modules/user';
|
import {saveInspection} from '../order/applicationForm/api.js';
|
||||||
import auth from '@/plugins/auth';
|
|
||||||
|
|
||||||
const userStore = useUserStore();
|
|
||||||
|
|
||||||
const { proxy } = getCurrentInstance();
|
const { proxy } = getCurrentInstance();
|
||||||
|
|
||||||
@@ -379,25 +378,10 @@ const isPendingStatus = (row) => {
|
|||||||
return status === undefined || status === null || status === '' || String(status) === '0';
|
return status === undefined || status === null || status === '' || String(status) === '0';
|
||||||
};
|
};
|
||||||
|
|
||||||
const isWithdrawableStatus = (row) => String(getBillStatus(row)) === '1';
|
const isIssuedStatus = (row) => String(getBillStatus(row)) === '1';
|
||||||
|
|
||||||
const isReportStatus = (row) => ['6', '8'].includes(String(getBillStatus(row)));
|
const isReportStatus = (row) => ['6', '8'].includes(String(getBillStatus(row)));
|
||||||
|
|
||||||
/**
|
|
||||||
* 是否可管理该申请单:申请者本人或管理员
|
|
||||||
*/
|
|
||||||
const canManageRow = (row) => {
|
|
||||||
if (auth.hasRole('admin')) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const currentPractitionerId = userStore.practitionerId;
|
|
||||||
const requesterId = row?.requesterId;
|
|
||||||
if (!currentPractitionerId || !requesterId) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return String(currentPractitionerId) === String(requesterId);
|
|
||||||
};
|
|
||||||
|
|
||||||
const sortByCreateTimeDesc = (a, b) => {
|
const sortByCreateTimeDesc = (a, b) => {
|
||||||
const aTime = a?.createTime ? new Date(a.createTime).getTime() : 0;
|
const aTime = a?.createTime ? new Date(a.createTime).getTime() : 0;
|
||||||
const bTime = b?.createTime ? new Date(b.createTime).getTime() : 0;
|
const bTime = b?.createTime ? new Date(b.createTime).getTime() : 0;
|
||||||
@@ -602,9 +586,9 @@ const submitEditForm = () => {
|
|||||||
*/
|
*/
|
||||||
const handleDelete = async (row) => {
|
const handleDelete = async (row) => {
|
||||||
try {
|
try {
|
||||||
await proxy.$modal?.confirm?.('确认作废该申请单吗?作废后不可撤销');
|
await proxy.$modal?.confirm?.(`确定要删除申请单 "${row.prescriptionNo}" 吗?此操作不可恢复。`);
|
||||||
} catch {
|
} catch {
|
||||||
return;
|
return; // 用户取消
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -615,21 +599,19 @@ const handleDelete = async (row) => {
|
|||||||
} else {
|
} else {
|
||||||
proxy.$modal?.msgError?.(res?.msg || '删除失败');
|
proxy.$modal?.msgError?.(res?.msg || '删除失败');
|
||||||
}
|
}
|
||||||
} catch {
|
} catch (e) {
|
||||||
// 响应拦截器已处理错误提示,此处静默
|
proxy.$modal?.msgError?.(e.message || '删除异常');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 撤回检验申请单(已签发且未采证状态可撤回)
|
* 撤回检验申请单(已签发状态撤回至待签发)
|
||||||
*/
|
*/
|
||||||
const handleWithdraw = async (row) => {
|
const handleWithdraw = async (row) => {
|
||||||
try {
|
try {
|
||||||
await proxy.$modal?.confirm?.(
|
await proxy.$modal?.confirm?.(`确定要撤回申请单 "${row.prescriptionNo}" 吗?撤回后将恢复为待签发状态。`);
|
||||||
'确认撤回该申请单吗?撤回后申请单及关联医嘱将恢复为待签发状态,护士站将同步更新。'
|
|
||||||
);
|
|
||||||
} catch {
|
} catch {
|
||||||
return;
|
return; // 用户取消
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -640,8 +622,8 @@ const handleWithdraw = async (row) => {
|
|||||||
} else {
|
} else {
|
||||||
proxy.$modal?.msgError?.(res?.msg || '撤回失败');
|
proxy.$modal?.msgError?.(res?.msg || '撤回失败');
|
||||||
}
|
}
|
||||||
} catch {
|
} catch (e) {
|
||||||
// 响应拦截器已处理错误提示,此处静默
|
proxy.$modal?.msgError?.(e.message || '撤回异常');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -758,10 +740,6 @@ defineExpose({
|
|||||||
|
|
||||||
.report-status-tag {
|
.report-status-tag {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background-color: #f0f9eb !important;
|
|
||||||
border-color: #67c23a !important;
|
|
||||||
color: #529b2e !important;
|
|
||||||
font-weight: 600;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@keyframes rotating {
|
@keyframes rotating {
|
||||||
|
|||||||
@@ -633,11 +633,11 @@ const calculateTotalAmount = () => {
|
|||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
const row = props.row;
|
const row = props.row;
|
||||||
const qty = new Decimal(row.doseQuantity || 0);
|
const qty = new Decimal(row.doseQuantity || 0);
|
||||||
// 根据首次用量单位类型决定使用哪个单价
|
const isMinUnit = row.unitCode == row.minUnitCode;
|
||||||
const unitType = row.unitCodeList?.find((k) => k.value == row.doseUnitCode)?.type;
|
const price = isMinUnit ? row.minUnitPrice : row.unitPrice;
|
||||||
const price = unitType == 'unit' ? row.unitPrice : row.minUnitPrice;
|
// 四舍五入到2位再算,与页面显示的单价一致
|
||||||
const roundedPrice = new Decimal(price || 0).toDecimalPlaces(2, Decimal.ROUND_HALF_UP);
|
const roundedPrice = new Decimal(price || 0).toDecimalPlaces(2, Decimal.ROUND_HALF_UP);
|
||||||
row.totalPrice = qty.mul(roundedPrice).toDecimalPlaces(2, Decimal.ROUND_HALF_UP).toString();
|
row.totalPrice = qty.mul(roundedPrice).toFixed(6);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
const setInputRef = props.handlers.setInputRef;
|
const setInputRef = props.handlers.setInputRef;
|
||||||
|
|||||||
@@ -24,20 +24,22 @@
|
|||||||
</el-col> -->
|
</el-col> -->
|
||||||
<el-col :span="12">
|
<el-col :span="12">
|
||||||
<el-form-item label="发往科室" prop="targetDepartment" style="width: 100%">
|
<el-form-item label="发往科室" prop="targetDepartment" style="width: 100%">
|
||||||
<el-select
|
<!-- <el-input v-model="form.targetDepartment" autocomplete="off" /> -->
|
||||||
|
<el-tree-select
|
||||||
|
clearable
|
||||||
|
style="width: 100%"
|
||||||
v-model="form.targetDepartment"
|
v-model="form.targetDepartment"
|
||||||
filterable
|
filterable
|
||||||
clearable
|
:data="orgOptions"
|
||||||
|
:props="{
|
||||||
|
value: 'id',
|
||||||
|
label: 'name',
|
||||||
|
children: 'children',
|
||||||
|
}"
|
||||||
|
value-key="id"
|
||||||
|
check-strictly
|
||||||
placeholder="请选择科室"
|
placeholder="请选择科室"
|
||||||
style="width: 100%"
|
/>
|
||||||
>
|
|
||||||
<el-option
|
|
||||||
v-for="opt in flatOrgOptions"
|
|
||||||
:key="opt.value"
|
|
||||||
:label="opt.label"
|
|
||||||
:value="opt.value"
|
|
||||||
/>
|
|
||||||
</el-select>
|
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
<el-col :span="12">
|
<el-col :span="12">
|
||||||
@@ -76,33 +78,18 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup name="BloodTransfusion">
|
<script setup name="BloodTransfusion">
|
||||||
import {computed, getCurrentInstance, nextTick, onBeforeMount, onMounted, reactive, ref, watch} from 'vue';
|
import {getCurrentInstance, onBeforeMount, onMounted, reactive, ref} from 'vue';
|
||||||
import {ElMessage} from 'element-plus';
|
|
||||||
import {patientInfo} from '../../../store/patient.js';
|
import {patientInfo} from '../../../store/patient.js';
|
||||||
import {getDepartmentList} from '@/api/public.js';
|
import {getDepartmentList} from '@/api/public.js';
|
||||||
import request from '@/utils/request';
|
|
||||||
import {getDiagnosisTreatmentOne} from '@/views/catalog/diagnosistreatment/components/diagnosistreatment';
|
|
||||||
import {getEncounterDiagnosis} from '../../api.js';
|
import {getEncounterDiagnosis} from '../../api.js';
|
||||||
import {getApplicationList, saveBloodTransfusio} from './api';
|
import {getApplicationList, saveBloodTransfusio} from './api';
|
||||||
|
|
||||||
const { proxy } = getCurrentInstance();
|
const { proxy } = getCurrentInstance();
|
||||||
|
|
||||||
/** 科室树节点 id 统一为字符串,避免大整数精度丢失导致 tree-select 无法匹配 */
|
|
||||||
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 findTreeItem = (list, id) => {
|
const findTreeItem = (list, id) => {
|
||||||
if (!list || list.length === 0 || id == null || id === '') return null;
|
if (!list || list.length === 0) return null;
|
||||||
const strId = String(id);
|
|
||||||
for (const item of list) {
|
for (const item of list) {
|
||||||
if (String(item.id) === strId) return item;
|
if (item.id == id) return item;
|
||||||
if (item.children && item.children.length > 0) {
|
if (item.children && item.children.length > 0) {
|
||||||
const found = findTreeItem(item.children, id);
|
const found = findTreeItem(item.children, id);
|
||||||
if (found) return found;
|
if (found) return found;
|
||||||
@@ -110,149 +97,11 @@ const findTreeItem = (list, id) => {
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
|
|
||||||
/** 在科室树中解析 orgId(兼容 Long 转 Number 后的精度丢失) */
|
|
||||||
const resolveOrgIdInTree = (rawOrgId) => {
|
|
||||||
if (rawOrgId == null || rawOrgId === '') return '';
|
|
||||||
const strOrgId = String(rawOrgId);
|
|
||||||
const findInTree = (nodes) => {
|
|
||||||
if (!nodes?.length) return null;
|
|
||||||
for (const node of nodes) {
|
|
||||||
if (String(node.id) === strOrgId) return String(node.id);
|
|
||||||
if (
|
|
||||||
typeof node.id === 'string' &&
|
|
||||||
node.id.length >= 16 &&
|
|
||||||
strOrgId.length >= 16 &&
|
|
||||||
node.id.substring(0, 15) === strOrgId.substring(0, 15)
|
|
||||||
) {
|
|
||||||
return String(node.id);
|
|
||||||
}
|
|
||||||
if (node.children?.length) {
|
|
||||||
const found = findInTree(node.children);
|
|
||||||
if (found) return found;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
};
|
|
||||||
return findInTree(orgOptions.value) || strOrgId;
|
|
||||||
};
|
|
||||||
|
|
||||||
const resolveTargetDepartmentId = (rawId) => {
|
|
||||||
if (rawId == null || rawId === '') return '';
|
|
||||||
const resolved = resolveOrgIdInTree(rawId);
|
|
||||||
const node = findTreeItem(orgOptions.value, resolved);
|
|
||||||
return node ? String(node.id) : resolved;
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 诊疗目录「所属科室」→ AdviceBaseDto.orgId */
|
|
||||||
const getBelongingOrgId = (item) => {
|
|
||||||
if (!item) return null;
|
|
||||||
return item.orgId ?? item.org_id ?? null;
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 诊疗目录所属科室名称(字典翻译字段) */
|
|
||||||
const getBelongingOrgName = (item) => {
|
|
||||||
if (!item) return '';
|
|
||||||
return item.orgId_dictText || item.orgName || item.org_name || '';
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 按机构 ID 拉取科室名称(树中无节点时兜底) */
|
|
||||||
const fetchOrgNameById = async (orgId) => {
|
|
||||||
if (orgId == null || orgId === '') return '';
|
|
||||||
const fromTree = findOrgName(orgId);
|
|
||||||
if (fromTree) return fromTree;
|
|
||||||
try {
|
|
||||||
const res = await request({
|
|
||||||
url: '/base-data-manage/organization/organization-getById',
|
|
||||||
method: 'get',
|
|
||||||
params: { orgId },
|
|
||||||
});
|
|
||||||
if (res.code === 200 && res.data?.name) {
|
|
||||||
return res.data.name;
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
console.error('查询科室名称失败', e);
|
|
||||||
}
|
|
||||||
return '';
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 从机构树解析科室名称 */
|
|
||||||
const findOrgName = (orgId) => {
|
|
||||||
if (orgId == null || orgId === '') return '';
|
|
||||||
const node = findTreeItem(orgOptions.value, orgId);
|
|
||||||
if (node?.name) return node.name;
|
|
||||||
const resolved = resolveOrgIdInTree(orgId);
|
|
||||||
const resolvedNode = findTreeItem(orgOptions.value, resolved);
|
|
||||||
return resolvedNode?.name || '';
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 自动填充时缓存的科室名称 */
|
|
||||||
const targetDepartmentName = ref('');
|
|
||||||
|
|
||||||
/** 扁平化科室选项,保证 el-select 能稳定显示 label */
|
|
||||||
const flatOrgOptions = computed(() => {
|
|
||||||
const options = [];
|
|
||||||
const seen = new Set();
|
|
||||||
const walk = (nodes) => {
|
|
||||||
for (const node of nodes || []) {
|
|
||||||
if (node?.id == null) continue;
|
|
||||||
const value = String(node.id);
|
|
||||||
if (seen.has(value)) continue;
|
|
||||||
seen.add(value);
|
|
||||||
options.push({ value, label: node.name || value });
|
|
||||||
if (node.children?.length) walk(node.children);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
walk(orgOptions.value);
|
|
||||||
const curId = form.targetDepartment;
|
|
||||||
const curName = targetDepartmentName.value || findOrgName(curId);
|
|
||||||
if (curId && curName && !seen.has(String(curId))) {
|
|
||||||
options.unshift({ value: String(curId), label: curName });
|
|
||||||
}
|
|
||||||
return options;
|
|
||||||
});
|
|
||||||
|
|
||||||
/** 从诊疗目录详情补全所属科室(医嘱下拉接口不带 orgId_dictText) */
|
|
||||||
const resolveProjectOrgInfo = async (item) => {
|
|
||||||
if (!item) return { orgId: null, orgName: '' };
|
|
||||||
let orgId = getBelongingOrgId(item);
|
|
||||||
let orgName = getBelongingOrgName(item);
|
|
||||||
if ((!orgId || !orgName) && item.adviceDefinitionId) {
|
|
||||||
try {
|
|
||||||
const res = await getDiagnosisTreatmentOne(item.adviceDefinitionId);
|
|
||||||
const detail = res?.data;
|
|
||||||
if (detail) {
|
|
||||||
orgId = orgId ?? detail.orgId ?? detail.org_id ?? null;
|
|
||||||
orgName = orgName || detail.orgId_dictText || detail.orgName || '';
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
console.error('查询诊疗目录所属科室失败', e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (orgId && !orgName) {
|
|
||||||
orgName = await fetchOrgNameById(orgId);
|
|
||||||
}
|
|
||||||
return { orgId, orgName };
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 写入发往科室 */
|
|
||||||
const applyTargetDepartment = async (belongOrgId, nameHint = '') => {
|
|
||||||
if (belongOrgId == null || belongOrgId === '') {
|
|
||||||
form.targetDepartment = '';
|
|
||||||
targetDepartmentName.value = '';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const resolvedDeptId = resolveTargetDepartmentId(belongOrgId);
|
|
||||||
const deptName =
|
|
||||||
nameHint || findOrgName(belongOrgId) || findOrgName(resolvedDeptId) || (await fetchOrgNameById(belongOrgId));
|
|
||||||
targetDepartmentName.value = deptName;
|
|
||||||
form.targetDepartment = resolvedDeptId;
|
|
||||||
};
|
|
||||||
const emits = defineEmits(['submitOk']);
|
const emits = defineEmits(['submitOk']);
|
||||||
const props = defineProps({});
|
const props = defineProps({});
|
||||||
const state = reactive({});
|
const state = reactive({});
|
||||||
const applicationListAll = ref([]);
|
const applicationListAll = ref();
|
||||||
const applicationList = ref([]);
|
const applicationList = ref();
|
||||||
const loading = ref(false);
|
const loading = ref(false);
|
||||||
const orgOptions = ref([]); // 科室选项
|
const orgOptions = ref([]); // 科室选项
|
||||||
const getList = () => {
|
const getList = () => {
|
||||||
@@ -269,48 +118,29 @@ const getList = () => {
|
|||||||
adviceTypes: [3], //1 药品 2耗材 3诊疗
|
adviceTypes: [3], //1 药品 2耗材 3诊疗
|
||||||
})
|
})
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
if (res.code === 200 && Array.isArray(res.data?.records)) {
|
if (res.code === 200) {
|
||||||
const records = res.data.records.filter((item) => item.adviceDefinitionId != null);
|
applicationListAll.value = res.data.records;
|
||||||
applicationListAll.value = records;
|
applicationList.value = res.data.records.map((item) => {
|
||||||
applicationList.value = records.map((item) => {
|
|
||||||
const priceInfo = item.priceList?.[0] || {};
|
const priceInfo = item.priceList?.[0] || {};
|
||||||
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
|
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
|
||||||
const unit = item.unitCode_dictText || item.unitCode || '';
|
const unit = item.unitCode_dictText || item.unitCode || '';
|
||||||
const id = item.adviceDefinitionId;
|
|
||||||
return {
|
return {
|
||||||
adviceDefinitionId: id,
|
adviceDefinitionId: item.adviceDefinitionId,
|
||||||
orgId: item.orgId,
|
orgId: item.orgId,
|
||||||
label: item.adviceName + ' (¥' + price + '/' + unit + ')',
|
label: item.adviceName + ' (¥' + price + '/' + unit + ')',
|
||||||
key: id,
|
key: item.adviceDefinitionId,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
proxy.$message.error(res.message || '加载输血项目失败');
|
proxy.$message.error(res.message);
|
||||||
applicationListAll.value = [];
|
|
||||||
applicationList.value = [];
|
applicationList.value = [];
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.finally(() => {
|
.finally(() => {
|
||||||
loading.value = false;
|
loading.value = false;
|
||||||
if (transferValue.value.length > 0) {
|
|
||||||
nextTick(async () => {
|
|
||||||
const valid = await validateTransferOrgConsistency(transferValue.value);
|
|
||||||
if (valid) {
|
|
||||||
lastValidTransferValue.value = [...transferValue.value];
|
|
||||||
fillTargetDepartmentFromSelection(transferValue.value, 1);
|
|
||||||
} else {
|
|
||||||
transferValue.value = [];
|
|
||||||
lastValidTransferValue.value = [];
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
const transferValue = ref([]);
|
const transferValue = ref([]);
|
||||||
/** 上一次通过校验的已选项目(科室不一致时回滚到此状态) */
|
|
||||||
const lastValidTransferValue = ref([]);
|
|
||||||
const isRevertingTransfer = ref(false);
|
|
||||||
let transferValidateSeq = 0;
|
|
||||||
const form = reactive({
|
const form = reactive({
|
||||||
// categoryType: '', // 项目类别
|
// categoryType: '', // 项目类别
|
||||||
targetDepartment: '', // 发往科室
|
targetDepartment: '', // 发往科室
|
||||||
@@ -327,140 +157,86 @@ const rules = reactive({});
|
|||||||
onBeforeMount(() => {});
|
onBeforeMount(() => {});
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
getList();
|
getList();
|
||||||
getLocationInfo();
|
|
||||||
});
|
});
|
||||||
const collectSelectedProjects = (selectProjectIds) => {
|
|
||||||
return (selectProjectIds || [])
|
|
||||||
.map((element) =>
|
|
||||||
applicationListAll.value.find((item) => String(item.adviceDefinitionId) === String(element))
|
|
||||||
)
|
|
||||||
.filter(Boolean);
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 校验已选项目的所属科室是否一致(超过 1 项时才校验) */
|
|
||||||
const validateTransferOrgConsistency = async (selectProjectIds) => {
|
|
||||||
const arr = collectSelectedProjects(selectProjectIds);
|
|
||||||
if (arr.length <= 1) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const orgInfoList = await Promise.all(arr.map((item) => resolveProjectOrgInfo(item)));
|
|
||||||
const firstOrgId = orgInfoList[0]?.orgId;
|
|
||||||
return orgInfoList.every((info) => String(info?.orgId ?? '') === String(firstOrgId ?? ''));
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* type(1:watch监听类型 2:点击保存类型)
|
* type(1:watch监听类型 2:点击保存类型)
|
||||||
*/
|
* selectProjectIds(选中项目的id数组)
|
||||||
const fillTargetDepartmentFromSelection = async (selectProjectIds, type) => {
|
* */
|
||||||
const manualDept = type === 2 && form.targetDepartment ? form.targetDepartment : '';
|
const projectWithDepartment = (selectProjectIds, type) => {
|
||||||
const arr = collectSelectedProjects(selectProjectIds);
|
//1.获取选中的项目 2.判断项目的执行科室是否相同 3.判断执行科室是否配置 4.将项目的执行科室复值到执行科室下拉选位置
|
||||||
|
let isRelease = true;
|
||||||
if (arr.length === 0) {
|
// 选中项目的数组
|
||||||
// 项目列表尚未加载完时,已选 ID 存在则先不清空(避免误清发往科室)
|
const arr = [];
|
||||||
if ((selectProjectIds || []).length > 0 && applicationListAll.value.length === 0) {
|
// 根据选中的项目id查找对应的项目
|
||||||
return type === 2 ? !!manualDept : true;
|
selectProjectIds.forEach((element) => {
|
||||||
|
const searchData = applicationList.value.find((item) => {
|
||||||
|
return element == item.adviceDefinitionId;
|
||||||
|
});
|
||||||
|
arr.push(searchData);
|
||||||
|
});
|
||||||
|
// 清空科室
|
||||||
|
form.targetDepartment = '';
|
||||||
|
if (arr.length > 0) {
|
||||||
|
const obj = arr[0];
|
||||||
|
// 判断科室是否相同
|
||||||
|
const isCompare = arr.every((item) => {
|
||||||
|
return item.orgId == obj.orgId;
|
||||||
|
});
|
||||||
|
if (!isCompare) {
|
||||||
|
ElMessage({
|
||||||
|
type: 'error',
|
||||||
|
message: '执行科室不同',
|
||||||
|
});
|
||||||
|
isRelease = false;
|
||||||
}
|
}
|
||||||
form.targetDepartment = '';
|
// 选中项目中的执行科室id与全部科室数据做匹配
|
||||||
targetDepartmentName.value = '';
|
const findItem = findTreeItem(orgOptions.value, obj.orgId);
|
||||||
return type === 2 ? !!manualDept : true;
|
|
||||||
}
|
|
||||||
|
|
||||||
const orgInfoList = await Promise.all(arr.map((item) => resolveProjectOrgInfo(item)));
|
if (!findItem) {
|
||||||
const firstOrg = orgInfoList[0];
|
isRelease = false;
|
||||||
const belongOrgId = firstOrg?.orgId;
|
ElMessage({
|
||||||
const allSameOrg = orgInfoList.every((info) => String(info?.orgId ?? '') === String(belongOrgId ?? ''));
|
type: 'error',
|
||||||
if (!allSameOrg) {
|
message: '未找到项目执行的科室',
|
||||||
if (type === 2) {
|
|
||||||
ElMessage.error('所选项目的所属科室不一致,请分开申请');
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (belongOrgId == null || belongOrgId === '') {
|
|
||||||
if (type === 2 && manualDept) {
|
|
||||||
await applyTargetDepartment(manualDept, findOrgName(manualDept));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (type === 2) {
|
|
||||||
ElMessage.warning('所选项目未在诊疗目录配置所属科室,请手动选择发往科室');
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
form.targetDepartment = '';
|
|
||||||
targetDepartmentName.value = '';
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type === 2 && manualDept) {
|
|
||||||
await applyTargetDepartment(manualDept, findOrgName(manualDept));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
await applyTargetDepartment(belongOrgId, firstOrg?.orgName || '');
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
// 选中项目:先校验所属科室一致,不通过则回滚穿梭框,不允许进入「已选择」
|
|
||||||
watch(
|
|
||||||
() => transferValue.value,
|
|
||||||
async (newValue) => {
|
|
||||||
if (isRevertingTransfer.value) return;
|
|
||||||
|
|
||||||
const seq = ++transferValidateSeq;
|
|
||||||
const valid = await validateTransferOrgConsistency(newValue);
|
|
||||||
if (seq !== transferValidateSeq) return;
|
|
||||||
|
|
||||||
if (!valid) {
|
|
||||||
ElMessage.error('所选项目的所属科室不一致,请分开申请');
|
|
||||||
isRevertingTransfer.value = true;
|
|
||||||
transferValue.value = [...lastValidTransferValue.value];
|
|
||||||
await nextTick();
|
|
||||||
isRevertingTransfer.value = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
lastValidTransferValue.value = [...newValue];
|
|
||||||
await fillTargetDepartmentFromSelection(newValue, 1);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
watch(
|
|
||||||
() => orgOptions.value,
|
|
||||||
() => {
|
|
||||||
if (transferValue.value.length > 0) {
|
|
||||||
nextTick(() => {
|
|
||||||
fillTargetDepartmentFromSelection(transferValue.value, 1);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
if (type == 1) {
|
||||||
{ deep: true }
|
if (isRelease) {
|
||||||
|
form.targetDepartment = findItem.id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return isRelease;
|
||||||
|
};
|
||||||
|
// 监听选择项目变化
|
||||||
|
watch(
|
||||||
|
() => transferValue.value,
|
||||||
|
(newValue) => {
|
||||||
|
projectWithDepartment(newValue, 1);
|
||||||
|
}
|
||||||
);
|
);
|
||||||
const submit = async () => {
|
const submit = () => {
|
||||||
if (transferValue.value.length == 0) {
|
if (transferValue.value.length == 0) {
|
||||||
return proxy.$message.error('请选择申请单');
|
return proxy.$message.error('请选择申请单');
|
||||||
}
|
}
|
||||||
if (!(await fillTargetDepartmentFromSelection(transferValue.value, 2))) {
|
if (!projectWithDepartment(transferValue.value, 2)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!form.targetDepartment) {
|
|
||||||
return proxy.$message.error('请选择发往科室');
|
|
||||||
}
|
|
||||||
let applicationListAllFilter = applicationListAll.value.filter((item) => {
|
let applicationListAllFilter = applicationListAll.value.filter((item) => {
|
||||||
return transferValue.value.some((id) => String(id) === String(item.adviceDefinitionId));
|
return transferValue.value.includes(item.adviceDefinitionId);
|
||||||
});
|
});
|
||||||
applicationListAllFilter = applicationListAllFilter.map((item) => {
|
applicationListAllFilter = applicationListAllFilter.map((item) => {
|
||||||
const priceInfo = item.priceList?.[0] || {};
|
|
||||||
return {
|
return {
|
||||||
adviceDefinitionId: item.adviceDefinitionId /** 诊疗定义id */,
|
adviceDefinitionId: item.adviceDefinitionId /** 诊疗定义id */,
|
||||||
quantity: 1, // /** 请求数量 */
|
quantity: 1, // /** 请求数量 */
|
||||||
unitCode: priceInfo.unitCode /** 请求单位编码 */,
|
unitCode: item.priceList[0].unitCode /** 请求单位编码 */,
|
||||||
unitPrice: priceInfo.price /** 单价 */,
|
unitPrice: item.priceList[0].price /** 单价 */,
|
||||||
totalPrice: priceInfo.price /** 总价 */,
|
totalPrice: item.priceList[0].price /** 总价 */,
|
||||||
positionId: form.targetDepartment || item.positionId, //执行科室id
|
positionId: item.positionId, //执行科室id
|
||||||
ybClassEnum: item.ybClassEnum, //类别医保编码
|
ybClassEnum: item.ybClassEnum, //类别医保编码
|
||||||
conditionId: item.conditionId, //诊断ID
|
conditionId: item.conditionId, //诊断ID
|
||||||
encounterDiagnosisId: item.encounterDiagnosisId, //就诊诊断id
|
encounterDiagnosisId: item.encounterDiagnosisId, //就诊诊断id
|
||||||
adviceType: item.adviceType, ///** 医嘱类型 */
|
adviceType: item.adviceType, ///** 医嘱类型 */
|
||||||
definitionId: priceInfo.definitionId, //费用定价主表ID */
|
definitionId: item.priceList[0].definitionId, //费用定价主表ID */
|
||||||
definitionDetailId: item.definitionDetailId, //费用定价子表ID */
|
definitionDetailId: item.definitionDetailId, //费用定价子表ID */
|
||||||
accountId: patientInfo.value.accountId, // // 账户id
|
accountId: patientInfo.value.accountId, // // 账户id
|
||||||
};
|
};
|
||||||
@@ -478,22 +254,16 @@ const submit = async () => {
|
|||||||
if (res.code === 200) {
|
if (res.code === 200) {
|
||||||
proxy.$message.success(res.msg);
|
proxy.$message.success(res.msg);
|
||||||
applicationList.value = [];
|
applicationList.value = [];
|
||||||
applicationListAll.value = [];
|
|
||||||
transferValue.value = [];
|
|
||||||
lastValidTransferValue.value = [];
|
|
||||||
emits('submitOk');
|
emits('submitOk');
|
||||||
} else {
|
} else {
|
||||||
proxy.$message.error(res.message);
|
proxy.$message.error(res.message);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
/** 查询科室(与检验申请单一致) */
|
/** 查询科室 */
|
||||||
const getLocationInfo = () => {
|
const getLocationInfo = () => {
|
||||||
return getDepartmentList().then((res) => {
|
getDepartmentList().then((res) => {
|
||||||
orgOptions.value = normalizeOrgTreeIds(res?.data || []);
|
orgOptions.value = res.data || [];
|
||||||
if (transferValue.value.length > 0) {
|
|
||||||
nextTick(() => fillTargetDepartmentFromSelection(transferValue.value, 1));
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
// 获取诊断目录
|
// 获取诊断目录
|
||||||
@@ -530,7 +300,7 @@ function getDiagnosisList() {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
defineExpose({ state, submit, getLocationInfo, getDiagnosisList, getList });
|
defineExpose({ state, submit, getLocationInfo, getDiagnosisList });
|
||||||
</script>
|
</script>
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.bloodTransfusion-container {
|
.bloodTransfusion-container {
|
||||||
@@ -542,22 +312,8 @@ defineExpose({ state, submit, getLocationInfo, getDiagnosisList, getList });
|
|||||||
min-height: 300px;
|
min-height: 300px;
|
||||||
}
|
}
|
||||||
|
|
||||||
:deep(.el-transfer) {
|
.el-transfer {
|
||||||
--el-transfer-panel-width: 480px !important;
|
--el-transfer-panel-width: 480px !important;
|
||||||
display: flex !important;
|
|
||||||
flex-direction: row !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
:deep(.el-transfer__buttons) {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
padding: 0 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
:deep(.el-transfer__button) {
|
|
||||||
margin: 4px 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.bloodTransfusion-form {
|
.bloodTransfusion-form {
|
||||||
|
|||||||
@@ -910,11 +910,6 @@ function handleFocus(row, index) {
|
|||||||
categoryCode = selectedItem ? selectedItem.categoryCode : (row.categoryCode || '');
|
categoryCode = selectedItem ? selectedItem.categoryCode : (row.categoryCode || '');
|
||||||
}
|
}
|
||||||
adviceQueryParams.value = { adviceType, categoryCode, searchKey: '' };
|
adviceQueryParams.value = { adviceType, categoryCode, searchKey: '' };
|
||||||
// handleFocus 打开 popover 时也要加载数据
|
|
||||||
const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
|
|
||||||
if (tableRef && tableRef.refresh) {
|
|
||||||
tableRef.refresh(adviceType, categoryCode, '');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleBlur(row) {
|
function handleBlur(row) {
|
||||||
|
|||||||
@@ -46,8 +46,7 @@
|
|||||||
<div style="display: flex; gap: 20px; height: 70vh">
|
<div style="display: flex; gap: 20px; height: 70vh">
|
||||||
<div
|
<div
|
||||||
style="
|
style="
|
||||||
width: 350px;
|
width: 250px;
|
||||||
min-width: 350px;
|
|
||||||
border: 1px solid #e4e7ed;
|
border: 1px solid #e4e7ed;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -71,35 +70,21 @@
|
|||||||
<span class="status-dot"></span>
|
<span class="status-dot"></span>
|
||||||
{{ getItemType_Text(item.adviceType) }}
|
{{ getItemType_Text(item.adviceType) }}
|
||||||
</div>
|
</div>
|
||||||
<el-tooltip :content="item.adviceName" placement="top" :show-after="500">
|
<div class="item-name">{{ item.adviceName }}</div>
|
||||||
<div class="item-name">{{ item.adviceName }}</div>
|
<div class="item-name">
|
||||||
</el-tooltip>
|
{{
|
||||||
<el-tooltip
|
|
||||||
:content="
|
|
||||||
item.priceList && item.priceList.length > 0
|
item.priceList && item.priceList.length > 0
|
||||||
? (item.priceList[0].price / item.partPercent).toFixed(2) + '元/' + item.minUnitCode_dictText
|
? (item.priceList[0].price / item.partPercent).toFixed(2) +
|
||||||
|
'元' +
|
||||||
|
'/' +
|
||||||
|
item.minUnitCode_dictText
|
||||||
: ''
|
: ''
|
||||||
"
|
}}
|
||||||
placement="top"
|
</div>
|
||||||
:show-after="500"
|
<div class="item-name" v-if="item.adviceType === 2">
|
||||||
>
|
库存数量:
|
||||||
<div class="item-name">
|
{{ handleQuantity(item) }}
|
||||||
{{
|
</div>
|
||||||
item.priceList && item.priceList.length > 0
|
|
||||||
? (item.priceList[0].price / item.partPercent).toFixed(2) +
|
|
||||||
'元' +
|
|
||||||
'/' +
|
|
||||||
item.minUnitCode_dictText
|
|
||||||
: ''
|
|
||||||
}}
|
|
||||||
</div>
|
|
||||||
</el-tooltip>
|
|
||||||
<el-tooltip v-if="item.adviceType === 2" :content="'库存数量:' + handleQuantity(item)" placement="top" :show-after="500">
|
|
||||||
<div class="item-name">
|
|
||||||
库存数量:
|
|
||||||
{{ handleQuantity(item) }}
|
|
||||||
</div>
|
|
||||||
</el-tooltip>
|
|
||||||
</div>
|
</div>
|
||||||
<!-- 只显示暂无数据文本 -->
|
<!-- 只显示暂无数据文本 -->
|
||||||
<div
|
<div
|
||||||
@@ -323,7 +308,7 @@
|
|||||||
import {computed, getCurrentInstance, onMounted, reactive, ref, watch} from 'vue';
|
import {computed, getCurrentInstance, onMounted, reactive, ref, watch} from 'vue';
|
||||||
import {ElMessage} from 'element-plus';
|
import {ElMessage} from 'element-plus';
|
||||||
import {formatDateStr} from '@/utils/index';
|
import {formatDateStr} from '@/utils/index';
|
||||||
import {getAdviceBaseInfo, getDiseaseTreatmentInitLoc, getOrgList, getOrgLocConfig} from './api.js';
|
import {getAdviceBaseInfo, getDiseaseTreatmentInitLoc, getOrgList} from './api.js';
|
||||||
import {getOrderGroup} from '@/views/doctorstation/components/api.js';
|
import {getOrderGroup} from '@/views/doctorstation/components/api.js';
|
||||||
import useUserStore from '@/store/modules/user';
|
import useUserStore from '@/store/modules/user';
|
||||||
|
|
||||||
@@ -381,7 +366,6 @@ const executeTime = ref('');
|
|||||||
const departmentOptions = ref([]);
|
const departmentOptions = ref([]);
|
||||||
const AdviceBaseInfoList = ref([]);
|
const AdviceBaseInfoList = ref([]);
|
||||||
const locationOptions = ref([]);
|
const locationOptions = ref([]);
|
||||||
const consumableDefaultLocId = ref(null); // 患者科室耗材默认库房ID(来自取药科室配置)
|
|
||||||
const searchText = ref('');
|
const searchText = ref('');
|
||||||
const userId = ref('');
|
const userId = ref('');
|
||||||
const orgId = ref('');
|
const orgId = ref('');
|
||||||
@@ -487,8 +471,11 @@ onMounted(() => {
|
|||||||
const userStore = useUserStore();
|
const userStore = useUserStore();
|
||||||
userId.value = userStore.id;
|
userId.value = userStore.id;
|
||||||
orgId.value = userStore.orgId;
|
orgId.value = userStore.orgId;
|
||||||
|
console.log(props.patientInfo, 'patientInfo in FeeDialog');
|
||||||
console.log('initialData in FeeDialog');
|
console.log('initialData in FeeDialog');
|
||||||
// 数据加载由 watch(visible) 统一触发,避免 patientInfo 未就绪时调用报错
|
loadDepartmentOptions();
|
||||||
|
getAdviceBaseInfos();
|
||||||
|
getDiseaseInitLoc();
|
||||||
});
|
});
|
||||||
|
|
||||||
// 监听弹窗显示状态
|
// 监听弹窗显示状态
|
||||||
@@ -497,12 +484,10 @@ 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');
|
||||||
consumableDefaultLocId.value = null; // 重置耗材默认库房,避免复用上次患者配置
|
|
||||||
// 弹窗打开时按当前患者科室重新加载,避免复用上一次患者/登录科室的结果
|
// 弹窗打开时按当前患者科室重新加载,避免复用上一次患者/登录科室的结果
|
||||||
loadDepartmentOptions();
|
loadDepartmentOptions();
|
||||||
getAdviceBaseInfos();
|
getAdviceBaseInfos();
|
||||||
getDiseaseInitLoc(16);
|
getDiseaseInitLoc(16);
|
||||||
loadConsumableDefaultLoc();
|
|
||||||
} else {
|
} else {
|
||||||
resetData();
|
resetData();
|
||||||
}
|
}
|
||||||
@@ -531,16 +516,7 @@ watch(
|
|||||||
if (!locs || locs.length === 0) return;
|
if (!locs || locs.length === 0) return;
|
||||||
feeItemsList.value.forEach(item => {
|
feeItemsList.value.forEach(item => {
|
||||||
if (item.adviceType === 2 && !item.positionId) {
|
if (item.adviceType === 2 && !item.positionId) {
|
||||||
if (consumableDefaultLocId.value) {
|
item.positionId = String(locs[0].value);
|
||||||
const matched = locs.find(d => String(d.value) === consumableDefaultLocId.value);
|
|
||||||
if (matched) {
|
|
||||||
item.positionId = String(matched.value);
|
|
||||||
} else {
|
|
||||||
ElMessage.warning(`"${item.adviceName}" 未找到匹配的执行科室,请手动选择`);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
ElMessage.warning(`"${item.adviceName}" 所在科室未配置耗材执行科室,请手动选择`);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -610,33 +586,14 @@ function getAdviceBaseInfos() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
function getDiseaseInitLoc() {
|
function getDiseaseInitLoc() {
|
||||||
// 16=药房,17=耗材库,合并后作为耗材执行科室下拉选项
|
getDiseaseTreatmentInitLoc(16)
|
||||||
Promise.all([
|
.then((response) => {
|
||||||
getDiseaseTreatmentInitLoc(16).catch(() => ({ data: { locationOptions: [] } })),
|
console.log('Disease Treatment Init Loc:', response);
|
||||||
getDiseaseTreatmentInitLoc(17).catch(() => ({ data: { locationOptions: [] } })),
|
locationOptions.value = response.data.locationOptions;
|
||||||
]).then(([pharmacyRes, warehouseRes]) => {
|
|
||||||
const pharmacies = pharmacyRes.data?.locationOptions || [];
|
|
||||||
const warehouses = warehouseRes.data?.locationOptions || [];
|
|
||||||
locationOptions.value = [...pharmacies, ...warehouses];
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 查询患者科室的耗材默认库房(取药科室配置 itemCode=2)
|
|
||||||
*/
|
|
||||||
function loadConsumableDefaultLoc() {
|
|
||||||
const deptId = props.patientInfo?.organizationId;
|
|
||||||
if (!deptId) {
|
|
||||||
consumableDefaultLocId.value = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
getOrgLocConfig({ organizationId: deptId, itemCode: '2', pageNo: 1, pageSize: 100 })
|
|
||||||
.then((res) => {
|
|
||||||
const records = res.data?.records || [];
|
|
||||||
consumableDefaultLocId.value = records.length > 0 ? String(records[0].defLocationId) : null;
|
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
consumableDefaultLocId.value = null;
|
console.warn('位置列表加载失败(可能无权限)');
|
||||||
|
locationOptions.value = [];
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// 下拉框模糊搜索过滤(自定义filter-method,配合element-plus filterable使用)
|
// 下拉框模糊搜索过滤(自定义filter-method,配合element-plus filterable使用)
|
||||||
@@ -787,19 +744,8 @@ function selectChange(row) {
|
|||||||
defaultPositionId = String(departmentOptions.value[0].id);
|
defaultPositionId = String(departmentOptions.value[0].id);
|
||||||
}
|
}
|
||||||
} else if (row.adviceType === 2 && locationOptions.value.length > 0) {
|
} else if (row.adviceType === 2 && locationOptions.value.length > 0) {
|
||||||
// 耗材:必须从取药科室配置中匹配默认库房,未配置则提示用户
|
// 耗材:默认取第一个药房/耗材房
|
||||||
if (consumableDefaultLocId.value) {
|
defaultPositionId = String(locationOptions.value[0].value);
|
||||||
const matched = locationOptions.value.find(
|
|
||||||
d => String(d.value) === consumableDefaultLocId.value
|
|
||||||
);
|
|
||||||
if (matched) {
|
|
||||||
defaultPositionId = String(matched.value);
|
|
||||||
} else {
|
|
||||||
ElMessage.warning(`"${row.adviceName}" 未找到匹配的执行科室,请手动选择`);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
ElMessage.warning(`"${row.adviceName}" 所在科室未配置耗材执行科室,请手动选择`);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
//插入费用列表
|
//插入费用列表
|
||||||
feeItemsList.value.push({
|
feeItemsList.value.push({
|
||||||
@@ -1077,8 +1023,6 @@ function applyGroupSet() {
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: #303133;
|
color: #303133;
|
||||||
margin-bottom: 8px;
|
margin-bottom: 8px;
|
||||||
overflow: hidden;
|
word-break: break-word;
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -111,16 +111,6 @@ export function getDiseaseTreatmentInitLoc(id) {
|
|||||||
method: 'get',
|
method: 'get',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
/**
|
|
||||||
* 查询科室取药配置(耗材默认库房)
|
|
||||||
*/
|
|
||||||
export function getOrgLocConfig(params) {
|
|
||||||
return request({
|
|
||||||
url: '/base-data-manage/org-loc/org-loc',
|
|
||||||
method: 'get',
|
|
||||||
params: params,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// 住院护士站费用明细
|
// 住院护士站费用明细
|
||||||
export function getCostDetail(queryParams) {
|
export function getCostDetail(queryParams) {
|
||||||
return request({
|
return request({
|
||||||
|
|||||||
@@ -90,13 +90,6 @@
|
|||||||
</span>
|
</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="医嘱状态" width="100" align="center">
|
|
||||||
<template #default="scope">
|
|
||||||
<el-tag :type="getStatusType(scope.row)" size="small">
|
|
||||||
{{ getStatusDisplayText(scope.row) }}
|
|
||||||
</el-tag>
|
|
||||||
</template>
|
|
||||||
</el-table-column>
|
|
||||||
<el-table-column label="医嘱内容" prop="adviceName">
|
<el-table-column label="医嘱内容" prop="adviceName">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<span>
|
<span>
|
||||||
@@ -176,43 +169,6 @@ import {formatDateStr} from '@/utils/index';
|
|||||||
import {getCurrentInstance, ref} from 'vue';
|
import {getCurrentInstance, ref} from 'vue';
|
||||||
import useUserStore from '@/store/modules/user';
|
import useUserStore from '@/store/modules/user';
|
||||||
|
|
||||||
/** 发药状态 → 医嘱状态(药品医嘱状态映射表) */
|
|
||||||
const DISPENSE_STATUS_TO_ADVICE_TEXT = {
|
|
||||||
2: '已执行',
|
|
||||||
8: '已提交',
|
|
||||||
4: '已发药',
|
|
||||||
};
|
|
||||||
|
|
||||||
function getStatusDisplayText(row) {
|
|
||||||
const params = row?.medicineSummaryParamList || [];
|
|
||||||
const pending = params.filter((p) => Number(p.dispenseStatus) !== 8);
|
|
||||||
if (pending.length === 0) {
|
|
||||||
return params.length ? '已提交' : '已执行';
|
|
||||||
}
|
|
||||||
const texts = [
|
|
||||||
...new Set(
|
|
||||||
pending
|
|
||||||
.map((p) => DISPENSE_STATUS_TO_ADVICE_TEXT[Number(p.dispenseStatus)])
|
|
||||||
.filter(Boolean),
|
|
||||||
),
|
|
||||||
];
|
|
||||||
if (texts.length === 1) {
|
|
||||||
return texts[0];
|
|
||||||
}
|
|
||||||
return '已执行';
|
|
||||||
}
|
|
||||||
|
|
||||||
function getStatusType(row) {
|
|
||||||
const text = getStatusDisplayText(row);
|
|
||||||
if (text === '已发药') {
|
|
||||||
return 'success';
|
|
||||||
}
|
|
||||||
if (text === '已提交') {
|
|
||||||
return 'warning';
|
|
||||||
}
|
|
||||||
return 'primary';
|
|
||||||
}
|
|
||||||
|
|
||||||
const activeNames = ref([]);
|
const activeNames = ref([]);
|
||||||
|
|
||||||
const userStore = useUserStore();
|
const userStore = useUserStore();
|
||||||
@@ -334,7 +290,6 @@ function handleMedicineSummary() {
|
|||||||
medicineSummary(ids).then((res) => {
|
medicineSummary(ids).then((res) => {
|
||||||
if (res.code == 200) {
|
if (res.code == 200) {
|
||||||
proxy.$message.success('操作成功');
|
proxy.$message.success('操作成功');
|
||||||
handleGetPrescription();
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,13 +16,7 @@
|
|||||||
<el-table-column label="药房" align="center" prop="locationName" />
|
<el-table-column label="药房" align="center" prop="locationName" />
|
||||||
<el-table-column label="申请人" align="center" prop="applicantName" />
|
<el-table-column label="申请人" align="center" prop="applicantName" />
|
||||||
<el-table-column label="领药人" align="center" prop="receiverName" />
|
<el-table-column label="领药人" align="center" prop="receiverName" />
|
||||||
<el-table-column label="医嘱状态" align="center" prop="statusEnum_enumText">
|
<el-table-column label="发药状态" align="center" prop="statusEnum_enumText" />
|
||||||
<template #default="scope">
|
|
||||||
<el-tag :type="getSummaryStatusType(scope.row)" size="small">
|
|
||||||
{{ formatSummaryStatusText(scope.row) }}
|
|
||||||
</el-tag>
|
|
||||||
</template>
|
|
||||||
</el-table-column>
|
|
||||||
<el-table-column label="操作" width="100" align="center">
|
<el-table-column label="操作" width="100" align="center">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<el-button link type="primary" @click="getDetail(scope.row)">详情</el-button>
|
<el-button link type="primary" @click="getDetail(scope.row)">详情</el-button>
|
||||||
@@ -58,35 +52,6 @@ import {getMedicineSummary, getMedicineSummaryDetail} from './api';
|
|||||||
import {patientInfoList} from '../../components/store/patient.js';
|
import {patientInfoList} from '../../components/store/patient.js';
|
||||||
import {getCurrentInstance, ref} from 'vue';
|
import {getCurrentInstance, ref} from 'vue';
|
||||||
|
|
||||||
const SUMMARY_STATUS_DISPLAY = {
|
|
||||||
2: '已提交',
|
|
||||||
4: '已发药',
|
|
||||||
};
|
|
||||||
|
|
||||||
const LEGACY_SUMMARY_STATUS_TEXT = {
|
|
||||||
待配药: '已提交',
|
|
||||||
已发放: '已发药',
|
|
||||||
};
|
|
||||||
|
|
||||||
function formatSummaryStatusText(row) {
|
|
||||||
const code = Number(row?.statusEnum);
|
|
||||||
if (SUMMARY_STATUS_DISPLAY[code]) {
|
|
||||||
return SUMMARY_STATUS_DISPLAY[code];
|
|
||||||
}
|
|
||||||
return LEGACY_SUMMARY_STATUS_TEXT[row?.statusEnum_enumText] || row?.statusEnum_enumText || '-';
|
|
||||||
}
|
|
||||||
|
|
||||||
function getSummaryStatusType(row) {
|
|
||||||
const text = formatSummaryStatusText(row);
|
|
||||||
if (text === '已发药') {
|
|
||||||
return 'success';
|
|
||||||
}
|
|
||||||
if (text === '已提交') {
|
|
||||||
return 'warning';
|
|
||||||
}
|
|
||||||
return 'info';
|
|
||||||
}
|
|
||||||
|
|
||||||
const medicineSummaryFormList = ref([]);
|
const medicineSummaryFormList = ref([]);
|
||||||
const medicineSummaryFormDetails = ref([]);
|
const medicineSummaryFormDetails = ref([]);
|
||||||
const dialogVisible = ref(false);
|
const dialogVisible = ref(false);
|
||||||
|
|||||||
@@ -184,11 +184,6 @@
|
|||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="开始/终止" prop="requestTime" width="200" />
|
<el-table-column label="开始/终止" prop="requestTime" width="200" />
|
||||||
<el-table-column label="医嘱状态" align="center" width="100">
|
|
||||||
<template #default="scope">
|
|
||||||
<el-tag v-if="props.exeStatus === 6" type="success" size="small">已执行</el-tag>
|
|
||||||
</template>
|
|
||||||
</el-table-column>
|
|
||||||
<el-table-column :label="props.exeStatus == 1 ? '预计执行' : '执行时间'" prop="times">
|
<el-table-column :label="props.exeStatus == 1 ? '预计执行' : '执行时间'" prop="times">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<div
|
<div
|
||||||
@@ -234,80 +229,8 @@ import {adviceCancel, adviceExecute, adviceNoExecute, getPrescriptionList} from
|
|||||||
import {patientInfoList} from '../../components/store/patient.js';
|
import {patientInfoList} from '../../components/store/patient.js';
|
||||||
import {lotNumberMatch} from '@/api/public';
|
import {lotNumberMatch} from '@/api/public';
|
||||||
import {formatDateStr} from '@/utils/index';
|
import {formatDateStr} from '@/utils/index';
|
||||||
import {RequestStatus} from '@/utils/medicalConstants';
|
|
||||||
import {getCurrentInstance, nextTick, ref, provide} from 'vue';
|
import {getCurrentInstance, nextTick, ref, provide} from 'vue';
|
||||||
|
|
||||||
/** 请求状态 → 医嘱状态映射表(开具/签发/校对) */
|
|
||||||
const REQUEST_STATUS_DISPLAY = {
|
|
||||||
[RequestStatus.DRAFT]: '待签发',
|
|
||||||
[RequestStatus.ACTIVE]: '已签发',
|
|
||||||
[RequestStatus.COMPLETED]: '已校对',
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 发药状态 → 医嘱状态映射表(汇总申请/发药) */
|
|
||||||
const DISPENSE_STATUS_DISPLAY = {
|
|
||||||
8: '已提交',
|
|
||||||
4: '已发药',
|
|
||||||
};
|
|
||||||
|
|
||||||
/** 执行页签对应的医嘱状态展示 */
|
|
||||||
const STATUS_DISPLAY_BY_EXE_TAB = {
|
|
||||||
1: { text: '已校对', type: 'success' },
|
|
||||||
6: { text: '已执行', type: 'success' },
|
|
||||||
5: { text: '不执行', type: 'warning' },
|
|
||||||
9: { text: '取消执行', type: 'info' },
|
|
||||||
};
|
|
||||||
|
|
||||||
const LEGACY_STATUS_TEXT = {
|
|
||||||
待发送: '待签发',
|
|
||||||
已发送: '已签发',
|
|
||||||
'已发送/待执行': '已签发',
|
|
||||||
已完成: '已校对',
|
|
||||||
待配药: '已提交',
|
|
||||||
已汇总: '已提交',
|
|
||||||
已发放: '已发药',
|
|
||||||
};
|
|
||||||
|
|
||||||
function getStatusDisplayText(row) {
|
|
||||||
const dispenseCode = Number(row?.dispenseStatus);
|
|
||||||
if (DISPENSE_STATUS_DISPLAY[dispenseCode]) {
|
|
||||||
return DISPENSE_STATUS_DISPLAY[dispenseCode];
|
|
||||||
}
|
|
||||||
const tabText = STATUS_DISPLAY_BY_EXE_TAB[props.exeStatus]?.text;
|
|
||||||
if (tabText) {
|
|
||||||
return tabText;
|
|
||||||
}
|
|
||||||
const requestCode = Number(row?.requestStatus);
|
|
||||||
if (REQUEST_STATUS_DISPLAY[requestCode]) {
|
|
||||||
return REQUEST_STATUS_DISPLAY[requestCode];
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
LEGACY_STATUS_TEXT[row?.requestStatus_enumText] ||
|
|
||||||
LEGACY_STATUS_TEXT[row?.dispenseStatus_enumText] ||
|
|
||||||
row?.requestStatus_enumText ||
|
|
||||||
row?.dispenseStatus_enumText ||
|
|
||||||
'-'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getStatusType(row) {
|
|
||||||
const tabType = STATUS_DISPLAY_BY_EXE_TAB[props.exeStatus]?.type;
|
|
||||||
if (tabType) {
|
|
||||||
return tabType;
|
|
||||||
}
|
|
||||||
const status = row?.requestStatus;
|
|
||||||
const map = {
|
|
||||||
1: 'info',
|
|
||||||
2: 'primary',
|
|
||||||
3: 'success',
|
|
||||||
4: 'warning',
|
|
||||||
5: 'danger',
|
|
||||||
6: 'danger',
|
|
||||||
7: 'warning',
|
|
||||||
};
|
|
||||||
return map[status] || 'info';
|
|
||||||
}
|
|
||||||
|
|
||||||
const activeNames = ref([]);
|
const activeNames = ref([]);
|
||||||
const prescriptionList = ref([]);
|
const prescriptionList = ref([]);
|
||||||
const deadline = ref(formatDateStr(new Date(), 'YYYY-MM-DD') + ' 23:59:59');
|
const deadline = ref(formatDateStr(new Date(), 'YYYY-MM-DD') + ' 23:59:59');
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ function handleClick(tabName) {
|
|||||||
// 执行状态待执行
|
// 执行状态待执行
|
||||||
exeStatus.value = 1;
|
exeStatus.value = 1;
|
||||||
// 请求状态已校对
|
// 请求状态已校对
|
||||||
requestStatus.value = RequestStatus.COMPLETED;
|
requestStatus.value = 3;
|
||||||
break;
|
break;
|
||||||
case 'completed':
|
case 'completed':
|
||||||
exeStatus.value = 6;
|
exeStatus.value = 6;
|
||||||
|
|||||||
@@ -142,10 +142,10 @@
|
|||||||
</template>
|
</template>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="医嘱状态" prop="requestStatus_enumText" width="100" align="center">
|
<el-table-column label="医嘱状态" prop="requestStatus_enumText" width="100">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<el-tag :type="getStatusType(scope.row.requestStatus)" size="small">
|
<el-tag :type="getStatusType(scope.row.requestStatus)" size="small">
|
||||||
{{ getStatusDisplayText(scope.row) }}
|
{{ scope.row.requestStatus_enumText }}
|
||||||
</el-tag>
|
</el-tag>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
@@ -163,7 +163,6 @@ 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';
|
||||||
import {RequestStatus} from '@/utils/medicalConstants';
|
|
||||||
|
|
||||||
const activeNames = ref([]);
|
const activeNames = ref([]);
|
||||||
const prescriptionList = ref([]);
|
const prescriptionList = ref([]);
|
||||||
@@ -174,34 +173,18 @@ const loading = ref(false);
|
|||||||
const chooseAll = ref(false);
|
const chooseAll = ref(false);
|
||||||
const selectionTrigger = ref(0);
|
const selectionTrigger = ref(0);
|
||||||
|
|
||||||
/** 各页签对应的医嘱状态展示文案(与后端枚举值解耦,贴合校对业务语义) */
|
|
||||||
const STATUS_DISPLAY_BY_TAB = {
|
|
||||||
[RequestStatus.ACTIVE]: { text: '已签发', type: 'primary' },
|
|
||||||
[RequestStatus.COMPLETED]: { text: '已校对', type: 'success' },
|
|
||||||
[RequestStatus.DRAFT]: { text: '待签发', type: 'info' },
|
|
||||||
[RequestStatus.STOPPED]: { text: '已停止', type: 'danger' },
|
|
||||||
};
|
|
||||||
|
|
||||||
const getStatusType = (status) => {
|
const getStatusType = (status) => {
|
||||||
const tabType = STATUS_DISPLAY_BY_TAB[props.requestStatus]?.type;
|
|
||||||
if (tabType) return tabType;
|
|
||||||
const map = {
|
const map = {
|
||||||
1: 'info',
|
1: 'info', // 待发送
|
||||||
2: 'primary',
|
2: 'primary', // 已发送
|
||||||
3: 'success',
|
3: 'success', // 已完成
|
||||||
4: 'warning',
|
4: 'warning', // 暂停
|
||||||
5: 'danger',
|
5: 'danger', // 取消/待退
|
||||||
6: 'danger',
|
6: 'danger', // 停嘱
|
||||||
7: 'info',
|
7: 'info' // 不执行
|
||||||
};
|
}
|
||||||
return map[status] || 'info';
|
return map[status] || 'info'
|
||||||
};
|
}
|
||||||
|
|
||||||
const getStatusDisplayText = (row) => {
|
|
||||||
const tabText = STATUS_DISPLAY_BY_TAB[props.requestStatus]?.text;
|
|
||||||
if (tabText) return tabText;
|
|
||||||
return row.requestStatus_enumText || '';
|
|
||||||
};
|
|
||||||
const hasDispensedSelected = computed(() => {
|
const hasDispensedSelected = computed(() => {
|
||||||
selectionTrigger.value;
|
selectionTrigger.value;
|
||||||
return getSelectRows().some(item => item.dispenseStatus === 4);
|
return getSelectRows().some(item => item.dispenseStatus === 4);
|
||||||
@@ -211,10 +194,6 @@ const props = defineProps({
|
|||||||
type: Number,
|
type: Number,
|
||||||
default: 2,
|
default: 2,
|
||||||
},
|
},
|
||||||
activeTab: {
|
|
||||||
type: String,
|
|
||||||
default: 'unverified',
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
function handleRadioChange() {
|
function handleRadioChange() {
|
||||||
|
|||||||
@@ -41,7 +41,6 @@
|
|||||||
<!-- 使用模板引用 -->
|
<!-- 使用模板引用 -->
|
||||||
<PrescriptionList
|
<PrescriptionList
|
||||||
:requestStatus="requestStatus"
|
:requestStatus="requestStatus"
|
||||||
:activeTab="activeName"
|
|
||||||
:ref="(el) => setPrescriptionRef(el, tab.name)"
|
:ref="(el) => setPrescriptionRef(el, tab.name)"
|
||||||
/>
|
/>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
@@ -83,16 +82,16 @@ function handleTabClick(tabName) {
|
|||||||
|
|
||||||
switch (activeTabName) {
|
switch (activeTabName) {
|
||||||
case 'unverified':
|
case 'unverified':
|
||||||
requestStatus.value = RequestStatus.ACTIVE;
|
requestStatus.value = 2;
|
||||||
break;
|
break;
|
||||||
case 'verified':
|
case 'verified':
|
||||||
requestStatus.value = RequestStatus.COMPLETED;
|
requestStatus.value = 3;
|
||||||
break;
|
break;
|
||||||
case 'stopped':
|
case 'stopped':
|
||||||
requestStatus.value = RequestStatus.STOPPED;
|
requestStatus.value = 6;
|
||||||
break;
|
break;
|
||||||
case 'cancelled':
|
case 'cancelled':
|
||||||
requestStatus.value = RequestStatus.DRAFT;
|
requestStatus.value = 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
// 调用子组件方法
|
// 调用子组件方法
|
||||||
|
|||||||
@@ -1818,11 +1818,14 @@ function handleQuoteBilling() {
|
|||||||
temporaryBillingMedicines.value = []
|
temporaryBillingMedicines.value = []
|
||||||
temporaryAdvices.value = []
|
temporaryAdvices.value = []
|
||||||
|
|
||||||
// 🔧 修复 Bug #445: 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3/6)
|
// 🔧 修复 Bug #444: 统一过滤逻辑,与 handleMedicalAdvice 保持一致
|
||||||
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
|
// 1. 使用 Number() + snake_case 回退,避免类型转换导致过滤失效
|
||||||
// 先提取已签发项目(statusEnum=2)填充已生成列表
|
// 2. 增加关键词二次过滤,排除手术/检查/诊疗等非药品项目
|
||||||
const activeItems = res.data.filter(item => {
|
const filteredItems = res.data.filter(item => {
|
||||||
|
// 匹配 encounterId
|
||||||
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
||||||
|
// 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3/6)
|
||||||
|
// 🔧 修复 Bug #444: 使用 Number() 显式转换,增加 snake_case 回退
|
||||||
const at = Number(item.adviceType ?? item.advice_type);
|
const at = Number(item.adviceType ?? item.advice_type);
|
||||||
if (at !== 1 && at !== 2) return false;
|
if (at !== 1 && at !== 2) return false;
|
||||||
if (item.statusEnum !== 2) return false;
|
if (item.statusEnum !== 2) return false;
|
||||||
|
|||||||
@@ -88,16 +88,16 @@
|
|||||||
</el-table>
|
</el-table>
|
||||||
</div>
|
</div>
|
||||||
<div class="candidate-actions">
|
<div class="candidate-actions">
|
||||||
<el-button
|
<el-button
|
||||||
type="primary"
|
type="primary"
|
||||||
:disabled="selectedCandidates.length === 0 || isQueryingHistory"
|
:disabled="selectedCandidates.length === 0"
|
||||||
@click="handleAddToQueue"
|
@click="handleAddToQueue"
|
||||||
>
|
>
|
||||||
加入队列 >>
|
加入队列 >>
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="primary"
|
type="primary"
|
||||||
:disabled="filteredCandidatePoolList.length === 0 || isQueryingHistory"
|
:disabled="filteredCandidatePoolList.length === 0"
|
||||||
@click="handleAddAllToQueue"
|
@click="handleAddAllToQueue"
|
||||||
>
|
>
|
||||||
一键加入队列
|
一键加入队列
|
||||||
@@ -109,19 +109,6 @@
|
|||||||
<div class="right-panel">
|
<div class="right-panel">
|
||||||
<div class="panel-header">
|
<div class="panel-header">
|
||||||
<span class="panel-title">② 智能队列 (全科)</span>
|
<span class="panel-title">② 智能队列 (全科)</span>
|
||||||
<div class="history-query">
|
|
||||||
<el-date-picker
|
|
||||||
v-model="queryDate"
|
|
||||||
type="date"
|
|
||||||
placeholder="选择日期"
|
|
||||||
format="YYYY-MM-DD"
|
|
||||||
value-format="YYYY-MM-DD"
|
|
||||||
size="small"
|
|
||||||
style="width: 150px"
|
|
||||||
/>
|
|
||||||
<el-button type="primary" size="small" @click="handleHistoryQuery">查询</el-button>
|
|
||||||
<el-button size="small" @click="handleTodayQuery">今天</el-button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="table-container">
|
<div class="table-container">
|
||||||
<el-table
|
<el-table
|
||||||
@@ -186,25 +173,25 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="display-options">
|
<div class="display-options">
|
||||||
<div class="queue-actions-left">
|
<div class="queue-actions-left">
|
||||||
<el-button
|
<el-button
|
||||||
type="danger"
|
type="danger"
|
||||||
:disabled="!selectedQueueRow || isQueryingHistory"
|
:disabled="!selectedQueueRow"
|
||||||
size="small"
|
size="small"
|
||||||
@click="handleRemoveFromQueue"
|
@click="handleRemoveFromQueue"
|
||||||
>
|
>
|
||||||
<< 移出队列
|
<< 移出队列
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="info"
|
type="info"
|
||||||
:disabled="!selectedQueueRow || !canMoveUp || isQueryingHistory"
|
:disabled="!selectedQueueRow || !canMoveUp"
|
||||||
size="small"
|
size="small"
|
||||||
@click="handleMoveUp"
|
@click="handleMoveUp"
|
||||||
>
|
>
|
||||||
↑
|
↑
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="info"
|
type="info"
|
||||||
:disabled="!selectedQueueRow || !canMoveDown || isQueryingHistory"
|
:disabled="!selectedQueueRow || !canMoveDown"
|
||||||
size="small"
|
size="small"
|
||||||
@click="handleMoveDown"
|
@click="handleMoveDown"
|
||||||
>
|
>
|
||||||
@@ -272,35 +259,30 @@
|
|||||||
<div class="control-buttons">
|
<div class="control-buttons">
|
||||||
<el-button
|
<el-button
|
||||||
type="primary"
|
type="primary"
|
||||||
:disabled="isQueryingHistory"
|
|
||||||
@click="handleSelectCall"
|
@click="handleSelectCall"
|
||||||
>
|
>
|
||||||
选呼
|
选呼
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="success"
|
type="success"
|
||||||
:disabled="isQueryingHistory"
|
|
||||||
@click="handleNextPatient"
|
@click="handleNextPatient"
|
||||||
>
|
>
|
||||||
下一患者
|
下一患者
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="warning"
|
type="warning"
|
||||||
:disabled="isQueryingHistory"
|
|
||||||
@click="handleSkip"
|
@click="handleSkip"
|
||||||
>
|
>
|
||||||
跳过
|
跳过
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="primary"
|
type="primary"
|
||||||
:disabled="isQueryingHistory"
|
|
||||||
@click="handleComplete"
|
@click="handleComplete"
|
||||||
>
|
>
|
||||||
完成
|
完成
|
||||||
</el-button>
|
</el-button>
|
||||||
<el-button
|
<el-button
|
||||||
type="info"
|
type="info"
|
||||||
:disabled="isQueryingHistory"
|
|
||||||
@click="handleRequeue"
|
@click="handleRequeue"
|
||||||
>
|
>
|
||||||
过号重排
|
过号重排
|
||||||
@@ -700,14 +682,6 @@ const showOnlyWaiting = ref(false)
|
|||||||
// Bug #411:诊室过滤,替代原来的科室下拉框(selectedDept/departmentList 已移除)
|
// Bug #411:诊室过滤,替代原来的科室下拉框(selectedDept/departmentList 已移除)
|
||||||
const selectedRoom = ref('all')
|
const selectedRoom = ref('all')
|
||||||
|
|
||||||
// 历史队列查询日期 (默认当天)
|
|
||||||
const getTodayStr = () => {
|
|
||||||
const now = new Date()
|
|
||||||
return `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, '0')}-${String(now.getDate()).padStart(2, '0')}`
|
|
||||||
}
|
|
||||||
const queryDate = ref(getTodayStr())
|
|
||||||
const isQueryingHistory = computed(() => queryDate.value !== getTodayStr())
|
|
||||||
|
|
||||||
// 修复【#397】:动态获取当前科室名称
|
// 修复【#397】:动态获取当前科室名称
|
||||||
const currentDeptName = computed(() => {
|
const currentDeptName = computed(() => {
|
||||||
return userStore.deptName || userStore.orgName || '心内科'
|
return userStore.deptName || userStore.orgName || '心内科'
|
||||||
@@ -927,12 +901,14 @@ const mapFrontendStatusToBackend = (status) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 从数据库加载队列
|
// 从数据库加载队列
|
||||||
const loadQueueFromDb = async (dateStr) => {
|
const loadQueueFromDb = async () => {
|
||||||
try {
|
try {
|
||||||
// Bug #411:不再按科室选筛加载,后端默认按当前登录人科室查询
|
// Bug #411:不再按科室选筛加载,后端默认按当前登录人科室查询
|
||||||
const organizationId = undefined
|
const organizationId = undefined
|
||||||
const queryDateStr = dateStr || queryDate.value
|
// 只查询今天的患者
|
||||||
const res = await getTriageQueueList({ organizationId, date: queryDateStr }).catch((err) => {
|
const today = new Date()
|
||||||
|
const todayStr = `${today.getFullYear()}-${String(today.getMonth() + 1).padStart(2, '0')}-${String(today.getDate()).padStart(2, '0')}`
|
||||||
|
const res = await getTriageQueueList({ organizationId, date: todayStr }).catch((err) => {
|
||||||
console.error('【心内科】loadQueueFromDb 请求异常:', err)
|
console.error('【心内科】loadQueueFromDb 请求异常:', err)
|
||||||
return { code: 500, msg: err?.message || '请求失败', data: null }
|
return { code: 500, msg: err?.message || '请求失败', data: null }
|
||||||
})
|
})
|
||||||
@@ -955,6 +931,10 @@ const loadQueueFromDb = async (dateStr) => {
|
|||||||
originalQueueList.value = list
|
originalQueueList.value = list
|
||||||
.map((it) => {
|
.map((it) => {
|
||||||
const frontendStatus = mapBackendStatusToFrontend(it.status)
|
const frontendStatus = mapBackendStatusToFrontend(it.status)
|
||||||
|
// 调试日志:检查状态映射
|
||||||
|
if (list.length <= 5) {
|
||||||
|
console.log('【心内科】状态映射:后端状态=', it.status, '-> 前端状态=', frontendStatus, '患者=', it.patientName)
|
||||||
|
}
|
||||||
// 计算等待时间:基于创建时间(createTime)
|
// 计算等待时间:基于创建时间(createTime)
|
||||||
let waitingTime = '00:00'
|
let waitingTime = '00:00'
|
||||||
if (it.createTime) {
|
if (it.createTime) {
|
||||||
@@ -992,7 +972,15 @@ const loadQueueFromDb = async (dateStr) => {
|
|||||||
organizationId: it.organizationId
|
organizationId: it.organizationId
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
.filter((item) => {
|
||||||
|
// 过滤掉"已完成"状态的患者,不显示在队列中
|
||||||
|
if (item.status === '已完成') {
|
||||||
|
console.log('【心内科】过滤掉已完成状态的患者:', item.patientName)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
|
||||||
// 调试日志:检查查找结果
|
// 调试日志:检查查找结果
|
||||||
const callingCount = originalQueueList.value.filter(i => i.status === '叫号中').length
|
const callingCount = originalQueueList.value.filter(i => i.status === '叫号中').length
|
||||||
const waitingCount = originalQueueList.value.filter(i => i.status === '等待').length
|
const waitingCount = originalQueueList.value.filter(i => i.status === '等待').length
|
||||||
@@ -1208,6 +1196,9 @@ const formatSecondsToMmSs = (totalSeconds) => {
|
|||||||
const filteredQueueList = computed(() => {
|
const filteredQueueList = computed(() => {
|
||||||
let filtered = originalQueueList.value
|
let filtered = originalQueueList.value
|
||||||
|
|
||||||
|
// 先过滤掉"已完成"状态的患者(无论什么情况都不显示)
|
||||||
|
filtered = filtered.filter(item => item.status !== '已完成')
|
||||||
|
|
||||||
// 再按诊室过滤
|
// 再按诊室过滤
|
||||||
if (selectedRoom.value !== 'all') {
|
if (selectedRoom.value !== 'all') {
|
||||||
filtered = filtered.filter(item => item.room === selectedRoom.value)
|
filtered = filtered.filter(item => item.room === selectedRoom.value)
|
||||||
@@ -1636,26 +1627,6 @@ const handleRefresh = async () => {
|
|||||||
ElMessage.success('已刷新(已从数据库恢复队列)')
|
ElMessage.success('已刷新(已从数据库恢复队列)')
|
||||||
}
|
}
|
||||||
|
|
||||||
// 历史队列查询
|
|
||||||
const handleHistoryQuery = async () => {
|
|
||||||
if (!queryDate.value) {
|
|
||||||
ElMessage.warning('请选择查询日期')
|
|
||||||
return
|
|
||||||
}
|
|
||||||
console.log('【心内科】历史队列查询:', queryDate.value)
|
|
||||||
await loadQueueFromDb(queryDate.value)
|
|
||||||
if (isQueryingHistory.value) {
|
|
||||||
ElMessage.success(`已加载 ${queryDate.value} 的队列数据`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 回到今天
|
|
||||||
const handleTodayQuery = async () => {
|
|
||||||
queryDate.value = getTodayStr()
|
|
||||||
await loadQueueFromDb(getTodayStr())
|
|
||||||
ElMessage.success('已切换到今天队列')
|
|
||||||
}
|
|
||||||
|
|
||||||
// 退出
|
// 退出
|
||||||
const handleExit = () => {
|
const handleExit = () => {
|
||||||
ElMessage.info('退出功能待实现')
|
ElMessage.info('退出功能待实现')
|
||||||
@@ -2194,21 +2165,12 @@ onUnmounted(() => {
|
|||||||
padding: 15px 20px;
|
padding: 15px 20px;
|
||||||
border-bottom: 2px solid #409eff;
|
border-bottom: 2px solid #409eff;
|
||||||
background-color: #f8f9fa;
|
background-color: #f8f9fa;
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
.panel-title {
|
.panel-title {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: #333;
|
color: #333;
|
||||||
}
|
}
|
||||||
|
|
||||||
.history-query {
|
|
||||||
display: flex;
|
|
||||||
gap: 8px;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.table-container {
|
.table-container {
|
||||||
|
|||||||
Reference in New Issue
Block a user