Compare commits
21 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| cfa073bdba | |||
| 021f7180c0 | |||
|
|
ee59fd5ea0 | ||
| 20a372268b | |||
| 31f288c0dd | |||
| 5c97380a78 | |||
| 30e5c92f0b | |||
| 7c6e35dcc3 | |||
| 74b67401f3 | |||
| 81d954fceb | |||
| c7da7440f6 | |||
|
|
232a0db810 | ||
|
|
3394aa54d7 | ||
| dc94978187 | |||
| 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());
|
||||||
|
```
|
||||||
|
|
||||||
|
编译验证通过。
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package com.core.framework.config;
|
package com.core.framework.config;
|
||||||
|
|
||||||
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
|
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
|
||||||
|
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
|
||||||
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
|
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
|
||||||
import org.mybatis.spring.annotation.MapperScan;
|
import org.mybatis.spring.annotation.MapperScan;
|
||||||
import org.springframework.boot.autoconfigure.jackson.Jackson2ObjectMapperBuilderCustomizer;
|
import org.springframework.boot.autoconfigure.jackson.Jackson2ObjectMapperBuilderCustomizer;
|
||||||
@@ -34,7 +35,9 @@ public class ApplicationConfig {
|
|||||||
// 设置日期格式为 yyyy/M/d HH:mm:ss,支持多种格式反序列化
|
// 设置日期格式为 yyyy/M/d HH:mm:ss,支持多种格式反序列化
|
||||||
builder.simpleDateFormat("yyyy/M/d HH:mm:ss");
|
builder.simpleDateFormat("yyyy/M/d HH:mm:ss");
|
||||||
// 添加JavaTimeModule支持,用于LocalDateTime
|
// 添加JavaTimeModule支持,用于LocalDateTime
|
||||||
builder.modules(new JavaTimeModule());
|
JavaTimeModule javaTimeModule = new JavaTimeModule();
|
||||||
|
javaTimeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
|
||||||
|
builder.modules(javaTimeModule);
|
||||||
builder.serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern("yyyy/M/d HH:mm:ss")));
|
builder.serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern("yyyy/M/d HH:mm:ss")));
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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;
|
||||||
@@ -169,11 +171,9 @@ public class OrganizationLocationAppServiceImpl implements IOrganizationLocation
|
|||||||
if (DateTimeUtils.isOverlap(organizationLocation.getStartTime(), organizationLocation.getEndTime(),
|
if (DateTimeUtils.isOverlap(organizationLocation.getStartTime(), organizationLocation.getEndTime(),
|
||||||
orgLoc.getStartTime(), orgLoc.getEndTime())) {
|
orgLoc.getStartTime(), orgLoc.getEndTime())) {
|
||||||
Organization org = organizationService.getById(organizationLocation.getOrganizationId());
|
Organization org = organizationService.getById(organizationLocation.getOrganizationId());
|
||||||
if (org == null) {
|
String organizationName = org != null ? org.getName() : ("科室[" + organizationLocation.getOrganizationId() + "]已删除");
|
||||||
continue;
|
|
||||||
}
|
|
||||||
return R.fail("当前诊疗:" + activityName + CommonConstants.Common.DASH + orgLoc.getStartTime()
|
return R.fail("当前诊疗:" + activityName + CommonConstants.Common.DASH + orgLoc.getStartTime()
|
||||||
+ CommonConstants.Common.DASH + orgLoc.getEndTime() + "与" + org.getName() + "时间冲突");
|
+ CommonConstants.Common.DASH + orgLoc.getEndTime() + "与" + organizationName + "时间冲突");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (orgLocQueryDto.getId() != null) {
|
if (orgLocQueryDto.getId() != null) {
|
||||||
|
|||||||
@@ -215,6 +215,9 @@ public class SurgicalScheduleAppServiceImpl implements ISurgicalScheduleAppServi
|
|||||||
if (surgery != null) {
|
if (surgery != null) {
|
||||||
surgery.setStatusEnum(1); // 1 = 已排期
|
surgery.setStatusEnum(1); // 1 = 已排期
|
||||||
surgery.setUpdateTime(new Date());
|
surgery.setUpdateTime(new Date());
|
||||||
|
// Bug #558: 手术安排时同步写入手术室确认时间和确认人
|
||||||
|
surgery.setOperatingRoomConfirmTime(new Date());
|
||||||
|
surgery.setOperatingRoomConfirmUser(loginUser.getUsername());
|
||||||
|
|
||||||
// 填充缺失的申请科室和主刀医生名称
|
// 填充缺失的申请科室和主刀医生名称
|
||||||
fillSurgeryMissingNames(surgery);
|
fillSurgeryMissingNames(surgery);
|
||||||
|
|||||||
@@ -147,6 +147,6 @@ public interface IDoctorStationAdviceAppService {
|
|||||||
*/
|
*/
|
||||||
IPage<SurgeryItemDto> getSurgeryPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey);
|
IPage<SurgeryItemDto> getSurgeryPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey);
|
||||||
|
|
||||||
IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey);
|
IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey, String categoryCode);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2566,12 +2566,13 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey) {
|
public IPage<SurgeryItemDto> getExaminationPage(Long organizationId, Integer pageNo, Integer pageSize, String searchKey, String categoryCode) {
|
||||||
IPage<SurgeryItemDto> result = doctorStationAdviceAppMapper.getExaminationPage(
|
IPage<SurgeryItemDto> result = doctorStationAdviceAppMapper.getExaminationPage(
|
||||||
new Page<>(pageNo, pageSize),
|
new Page<>(pageNo, pageSize),
|
||||||
PublicationStatus.ACTIVE.getValue(),
|
PublicationStatus.ACTIVE.getValue(),
|
||||||
organizationId,
|
organizationId,
|
||||||
searchKey);
|
searchKey,
|
||||||
|
categoryCode);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -226,8 +226,9 @@ public class DoctorStationAdviceController {
|
|||||||
@RequestParam(value = "organizationId", required = false) Long organizationId,
|
@RequestParam(value = "organizationId", required = false) Long organizationId,
|
||||||
@RequestParam(value = "pageNo", defaultValue = "1") Integer pageNo,
|
@RequestParam(value = "pageNo", defaultValue = "1") Integer pageNo,
|
||||||
@RequestParam(value = "pageSize", defaultValue = "500") Integer pageSize,
|
@RequestParam(value = "pageSize", defaultValue = "500") Integer pageSize,
|
||||||
@RequestParam(value = "searchKey", defaultValue = "") String searchKey) {
|
@RequestParam(value = "searchKey", defaultValue = "") String searchKey,
|
||||||
return R.ok(iDoctorStationAdviceAppService.getExaminationPage(organizationId, pageNo, pageSize, searchKey));
|
@RequestParam(value = "categoryCode", defaultValue = "23") String categoryCode) {
|
||||||
|
return R.ok(iDoctorStationAdviceAppService.getExaminationPage(organizationId, pageNo, pageSize, searchKey, categoryCode));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -203,6 +203,7 @@ public interface DoctorStationAdviceAppMapper {
|
|||||||
IPage<SurgeryItemDto> getExaminationPage(@Param("page") Page<SurgeryItemDto> page,
|
IPage<SurgeryItemDto> getExaminationPage(@Param("page") Page<SurgeryItemDto> page,
|
||||||
@Param("statusEnum") Integer statusEnum,
|
@Param("statusEnum") Integer statusEnum,
|
||||||
@Param("organizationId") Long organizationId,
|
@Param("organizationId") Long organizationId,
|
||||||
@Param("searchKey") String searchKey);
|
@Param("searchKey") String searchKey,
|
||||||
|
@Param("categoryCode") String categoryCode);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -133,47 +133,13 @@ public class PatientInformationServiceImpl implements IPatientInformationService
|
|||||||
@Override
|
@Override
|
||||||
public IPage<PatientBaseInfoDto> getPatientInfo(PatientBaseInfoDto patientBaseInfoDto, String searchKey,
|
public IPage<PatientBaseInfoDto> getPatientInfo(PatientBaseInfoDto patientBaseInfoDto, String searchKey,
|
||||||
Integer pageNo, Integer pageSize, HttpServletRequest request) {
|
Integer pageNo, Integer pageSize, HttpServletRequest request) {
|
||||||
// 获取登录者信息
|
// 构建基础查询条件
|
||||||
LoginUser loginUser = SecurityUtils.getLoginUser();
|
LoginUser loginUser = SecurityUtils.getLoginUser();
|
||||||
Long userId = loginUser.getUserId();
|
|
||||||
Integer tenantId = loginUser.getTenantId().intValue();
|
|
||||||
|
|
||||||
// 先构建基础查询条件
|
|
||||||
QueryWrapper<PatientBaseInfoDto> queryWrapper = HisQueryUtils.buildQueryWrapper(
|
QueryWrapper<PatientBaseInfoDto> queryWrapper = HisQueryUtils.buildQueryWrapper(
|
||||||
patientBaseInfoDto, searchKey, new HashSet<>(Arrays.asList(CommonConstants.FieldName.Name,
|
patientBaseInfoDto, searchKey, new HashSet<>(Arrays.asList(CommonConstants.FieldName.Name,
|
||||||
CommonConstants.FieldName.BusNo, CommonConstants.FieldName.PyStr, CommonConstants.FieldName.WbStr)),
|
CommonConstants.FieldName.BusNo, CommonConstants.FieldName.PyStr, CommonConstants.FieldName.WbStr)),
|
||||||
request);
|
request);
|
||||||
|
|
||||||
// 检查是否是精确ID查询(从门诊挂号页面跳转时使用)
|
|
||||||
boolean hasExactIdQuery = (patientBaseInfoDto.getId() != null);
|
|
||||||
|
|
||||||
// 只有非精确ID查询时,才添加医生患者过滤条件
|
|
||||||
if (!hasExactIdQuery) {
|
|
||||||
// 查询当前用户对应的医生信息
|
|
||||||
LambdaQueryWrapper<com.openhis.administration.domain.Practitioner> practitionerQuery = new LambdaQueryWrapper<>();
|
|
||||||
practitionerQuery.eq(com.openhis.administration.domain.Practitioner::getUserId, userId);
|
|
||||||
// 使用list()避免TooManyResultsException异常,然后取第一个记录
|
|
||||||
List<com.openhis.administration.domain.Practitioner> practitionerList = practitionerService.list(practitionerQuery);
|
|
||||||
com.openhis.administration.domain.Practitioner practitioner = practitionerList != null && !practitionerList.isEmpty() ? practitionerList.get(0) : null;
|
|
||||||
|
|
||||||
// 如果当前用户是医生,添加医生患者过滤条件
|
|
||||||
if (practitioner != null) {
|
|
||||||
// 查询该医生作为接诊医生(ADMITTER, code="1")和挂号医生(REGISTRATION_DOCTOR, code="12")的所有就诊记录的患者ID
|
|
||||||
List<Long> doctorPatientIds = patientManageMapper.getPatientIdsByPractitionerId(
|
|
||||||
practitioner.getId(),
|
|
||||||
Arrays.asList(ParticipantType.ADMITTER.getCode(), ParticipantType.REGISTRATION_DOCTOR.getCode()),
|
|
||||||
tenantId);
|
|
||||||
|
|
||||||
if (doctorPatientIds != null && !doctorPatientIds.isEmpty()) {
|
|
||||||
// 添加患者ID过滤条件 - 注意:这里使用列名而不是表别名
|
|
||||||
queryWrapper.in("id", doctorPatientIds);
|
|
||||||
} else {
|
|
||||||
// 如果没有相关患者,返回空结果
|
|
||||||
queryWrapper.eq("id", -1); // 设置一个不存在的ID
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// 如果不是医生,查询所有患者
|
|
||||||
}
|
|
||||||
|
|
||||||
IPage<PatientBaseInfoDto> patientInformationPage
|
IPage<PatientBaseInfoDto> patientInformationPage
|
||||||
= patientManageMapper.getPatientPage(new Page<>(pageNo, pageSize), queryWrapper);
|
= patientManageMapper.getPatientPage(new Page<>(pageNo, pageSize), queryWrapper);
|
||||||
|
|||||||
@@ -897,7 +897,7 @@
|
|||||||
LIMIT #{limit} OFFSET #{offset}
|
LIMIT #{limit} OFFSET #{offset}
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<!-- 检查项目专用分页查询:仅查检查(23) + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
<!-- 检查/检验项目专用分页查询:仅查指定 category_code + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
||||||
<select id="getExaminationPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto">
|
<select id="getExaminationPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto">
|
||||||
SELECT DISTINCT ON (t1.ID)
|
SELECT DISTINCT ON (t1.ID)
|
||||||
t1.ID AS advice_definition_id,
|
t1.ID AS advice_definition_id,
|
||||||
@@ -919,7 +919,7 @@
|
|||||||
ON t3.id = t1.org_id
|
ON t3.id = t1.org_id
|
||||||
AND t3.delete_flag = '0'
|
AND t3.delete_flag = '0'
|
||||||
WHERE t1.delete_flag = '0'
|
WHERE t1.delete_flag = '0'
|
||||||
AND t1.category_code = '23'
|
AND t1.category_code = #{categoryCode}
|
||||||
<if test="searchKey != null and searchKey != ''">
|
<if test="searchKey != null and searchKey != ''">
|
||||||
AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
|
AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
|
||||||
</if>
|
</if>
|
||||||
|
|||||||
@@ -220,3 +220,18 @@ export function getSlotStatusDescription(value) {
|
|||||||
export function getSlotStatusClass(status) {
|
export function getSlotStatusClass(status) {
|
||||||
return SlotStatusClassMap[status] || 'status-unbooked';
|
return SlotStatusClassMap[status] || 'status-unbooked';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 诊疗项目分类代码(对应后端 ActivityDefCategory 枚举)
|
||||||
|
* wor_activity_definition.category_code 字段
|
||||||
|
*/
|
||||||
|
export const ActivityCategory = {
|
||||||
|
/** 治疗 */
|
||||||
|
TREATMENT: '21',
|
||||||
|
/** 检验 */
|
||||||
|
PROOF: '22',
|
||||||
|
/** 检查 */
|
||||||
|
TEST: '23',
|
||||||
|
/** 手术 */
|
||||||
|
PROCEDURE: '24',
|
||||||
|
};
|
||||||
|
|||||||
@@ -898,6 +898,9 @@ const initData = async () => {
|
|||||||
// 申请日期实时更新(启动定时器)
|
// 申请日期实时更新(启动定时器)
|
||||||
startApplyTimeTimer()
|
startApplyTimeTimer()
|
||||||
|
|
||||||
|
// 执行时间默认填充当前系统时间
|
||||||
|
formData.executeTime = formatDateTime(new Date())
|
||||||
|
|
||||||
// 获取主诊断信息
|
// 获取主诊断信息
|
||||||
try {
|
try {
|
||||||
const res = await getEncounterDiagnosis(props.patientInfo.encounterId)
|
const res = await getEncounterDiagnosis(props.patientInfo.encounterId)
|
||||||
|
|||||||
@@ -17,17 +17,14 @@
|
|||||||
style="width: 300px; margin-bottom: 10px"
|
style="width: 300px; margin-bottom: 10px"
|
||||||
>
|
>
|
||||||
<template #append>
|
<template #append>
|
||||||
<el-button @click="handleSearch">搜索</el-button>
|
<el-button @click="handleSearch" :loading="loading">搜索</el-button>
|
||||||
</template>
|
</template>
|
||||||
</el-input>
|
</el-input>
|
||||||
<span v-if="!searchKey" class="total-count">共 {{ totalCount }} 项</span>
|
<span class="total-count">共 {{ totalCount }} 项</span>
|
||||||
<span v-else class="total-count">搜索到 {{ filteredCount }} 项 / 共 {{ totalCount }} 项</span>
|
|
||||||
</div>
|
</div>
|
||||||
<el-transfer
|
<el-transfer
|
||||||
v-model="transferValue"
|
v-model="transferValue"
|
||||||
:data="transferData"
|
:data="transferData"
|
||||||
filter-placeholder="项目代码/名称"
|
|
||||||
filterable
|
|
||||||
:titles="['未选择', '已选择']"
|
:titles="['未选择', '已选择']"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -136,7 +133,8 @@
|
|||||||
<script setup name="LaboratoryTests">
|
<script setup name="LaboratoryTests">
|
||||||
import {getCurrentInstance, nextTick, onMounted, reactive, ref, watch, computed} from 'vue';
|
import {getCurrentInstance, nextTick, onMounted, reactive, ref, watch, computed} from 'vue';
|
||||||
import {patientInfo} from '../../../store/patient.js';
|
import {patientInfo} from '../../../store/patient.js';
|
||||||
import {getApplicationList, saveInspection} from './api';
|
import {getExaminationPage, saveInspection} from './api';
|
||||||
|
import {ActivityCategory} from '@/utils/medicalConstants';
|
||||||
import {getDepartmentList} from '@/api/public.js';
|
import {getDepartmentList} from '@/api/public.js';
|
||||||
import {getEncounterDiagnosis} from '../../api.js';
|
import {getEncounterDiagnosis} from '../../api.js';
|
||||||
import {ElMessage} from 'element-plus';
|
import {ElMessage} from 'element-plus';
|
||||||
@@ -173,9 +171,8 @@ const skipDeptAutoFill = ref(false);
|
|||||||
// 将已加载的全部数据转为 transfer 组件所需的格式
|
// 将已加载的全部数据转为 transfer 组件所需的格式
|
||||||
const buildTransferData = (records) => {
|
const buildTransferData = (records) => {
|
||||||
return records.map((item) => {
|
return records.map((item) => {
|
||||||
const priceInfo = item.priceList?.[0] || {};
|
const price = item.price != null ? Number(item.price).toFixed(2) : '0.00';
|
||||||
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
|
const unit = item.unitCodeDictText || item.unitCode || '';
|
||||||
const unit = item.unitCode_dictText || item.unitCode || '';
|
|
||||||
return {
|
return {
|
||||||
adviceDefinitionId: item.adviceDefinitionId,
|
adviceDefinitionId: item.adviceDefinitionId,
|
||||||
orgId: item.orgId,
|
orgId: item.orgId,
|
||||||
@@ -185,7 +182,8 @@ const buildTransferData = (records) => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
// 加载全部数据(不分页,一次性拉取)
|
const selectedItemsCache = ref(new Map());
|
||||||
|
|
||||||
const loadAllData = async () => {
|
const loadAllData = async () => {
|
||||||
if (!patientInfo.value?.inHospitalOrgId) {
|
if (!patientInfo.value?.inHospitalOrgId) {
|
||||||
applicationListAll.value = [];
|
applicationListAll.value = [];
|
||||||
@@ -193,13 +191,12 @@ const loadAllData = async () => {
|
|||||||
}
|
}
|
||||||
loading.value = true;
|
loading.value = true;
|
||||||
try {
|
try {
|
||||||
// 使用大 pageSize 一次性拉取所有启用状态的检验类诊疗项目
|
const res = await getExaminationPage({
|
||||||
const res = await getApplicationList({
|
pageSize: 100,
|
||||||
pageSize: 9999,
|
|
||||||
pageNo: 1,
|
pageNo: 1,
|
||||||
categoryCode: '22',
|
categoryCode: ActivityCategory.PROOF,
|
||||||
organizationId: patientInfo.value.inHospitalOrgId,
|
organizationId: patientInfo.value.inHospitalOrgId,
|
||||||
adviceTypes: [3], // 1 药品 2 耗材 3 诊疗
|
searchKey: searchKey.value,
|
||||||
});
|
});
|
||||||
if (res.code !== 200) {
|
if (res.code !== 200) {
|
||||||
proxy.$message.error(res.message);
|
proxy.$message.error(res.message);
|
||||||
@@ -208,8 +205,9 @@ const loadAllData = async () => {
|
|||||||
}
|
}
|
||||||
applicationListAll.value = res.data?.records || [];
|
applicationListAll.value = res.data?.records || [];
|
||||||
totalCount.value = res.data?.total || 0;
|
totalCount.value = res.data?.total || 0;
|
||||||
// 检验项目列表为异步加载,编辑回显必须在数据就绪后执行,否则已选区一直为空
|
if (!searchKey.value) {
|
||||||
applyEditTransferSelection()
|
applyEditTransferSelection();
|
||||||
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
proxy.$message.error('获取检验项目列表失败');
|
proxy.$message.error('获取检验项目列表失败');
|
||||||
applicationListAll.value = [];
|
applicationListAll.value = [];
|
||||||
@@ -218,32 +216,18 @@ const loadAllData = async () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// 根据搜索关键词过滤数据
|
const transferData = computed(() => buildTransferData(applicationListAll.value));
|
||||||
const filterData = (key) => {
|
|
||||||
if (!key || key.trim() === '') {
|
|
||||||
return applicationListAll.value;
|
|
||||||
}
|
|
||||||
const lowerKey = key.toLowerCase().trim();
|
|
||||||
return applicationListAll.value.filter((item) => {
|
|
||||||
return (
|
|
||||||
item.adviceName?.toLowerCase().includes(lowerKey) ||
|
|
||||||
item.pyStr?.toLowerCase().includes(lowerKey) ||
|
|
||||||
item.adviceBusNo?.toLowerCase().includes(lowerKey)
|
|
||||||
);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
// transfer 组件实际显示的数据(受搜索词影响)
|
|
||||||
const transferData = computed(() => buildTransferData(filterData(searchKey.value)));
|
|
||||||
// 当前显示的条数
|
|
||||||
const filteredCount = computed(() => filterData(searchKey.value).length);
|
|
||||||
|
|
||||||
const getList = async () => {
|
const getList = async () => {
|
||||||
await loadAllData();
|
await loadAllData();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let searchTimer = null;
|
||||||
const handleSearch = () => {
|
const handleSearch = () => {
|
||||||
// 搜索时保持已选中的项目不受影响
|
clearTimeout(searchTimer);
|
||||||
|
searchTimer = setTimeout(() => {
|
||||||
|
loadAllData();
|
||||||
|
}, 300);
|
||||||
};
|
};
|
||||||
// 编辑初始化标志:避免 applyEditTransferSelection 设置 transferValue 时触发 projectWithDepartment 覆盖 descJson 中的科室值
|
// 编辑初始化标志:避免 applyEditTransferSelection 设置 transferValue 时触发 projectWithDepartment 覆盖 descJson 中的科室值
|
||||||
const isInitializing = ref(false);
|
const isInitializing = ref(false);
|
||||||
@@ -302,13 +286,17 @@ const projectWithDepartment = (selectProjectIds, type) => {
|
|||||||
const arr = [];
|
const arr = [];
|
||||||
// 根据选中的项目id查找对应的项目(从全部原始数据中查找)
|
// 根据选中的项目id查找对应的项目(从全部原始数据中查找)
|
||||||
selectProjectIds.forEach((element) => {
|
selectProjectIds.forEach((element) => {
|
||||||
const searchData = applicationListAll.value.find((item) => {
|
let searchData = applicationListAll.value.find((item) => {
|
||||||
return element == item.adviceDefinitionId;
|
return element == item.adviceDefinitionId;
|
||||||
});
|
});
|
||||||
|
if (!searchData) {
|
||||||
|
searchData = selectedItemsCache.value.get(element);
|
||||||
|
}
|
||||||
if (searchData) {
|
if (searchData) {
|
||||||
const priceInfo = searchData.priceList?.[0] || {};
|
const priceInfo = searchData.priceList?.[0] || {};
|
||||||
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
|
const price = searchData.price != null ? Number(searchData.price).toFixed(2)
|
||||||
const unit = searchData.unitCode_dictText || searchData.unitCode || '';
|
: priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
|
||||||
|
const unit = searchData.unitCodeDictText || searchData.unitCode_dictText || searchData.unitCode || '';
|
||||||
arr.push({
|
arr.push({
|
||||||
adviceDefinitionId: searchData.adviceDefinitionId,
|
adviceDefinitionId: searchData.adviceDefinitionId,
|
||||||
orgId: searchData.orgId,
|
orgId: searchData.orgId,
|
||||||
@@ -371,6 +359,12 @@ watch(
|
|||||||
(newValue) => {
|
(newValue) => {
|
||||||
if (skipDeptAutoFill.value) return;
|
if (skipDeptAutoFill.value) return;
|
||||||
if (isInitializing.value) return;
|
if (isInitializing.value) return;
|
||||||
|
newValue.forEach((id) => {
|
||||||
|
if (!selectedItemsCache.value.has(id)) {
|
||||||
|
const item = applicationListAll.value.find((i) => i.adviceDefinitionId == id);
|
||||||
|
if (item) selectedItemsCache.value.set(id, item);
|
||||||
|
}
|
||||||
|
});
|
||||||
projectWithDepartment(newValue, 1);
|
projectWithDepartment(newValue, 1);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@@ -455,13 +449,14 @@ watch(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
// 编辑模式下,项目字典加载完成后重新回显已选项目
|
// 编辑模式下,项目字典首次加载完成后回显已选项目(搜索刷新不重置)
|
||||||
watch(
|
watch(
|
||||||
() => applicationListAll.value,
|
() => applicationListAll.value,
|
||||||
() => {
|
() => {
|
||||||
if (!props.editData?.requestFormId) return;
|
if (!props.editData?.requestFormId) return;
|
||||||
if (!props.editData.requestFormDetailList?.length) return;
|
if (!props.editData.requestFormDetailList?.length) return;
|
||||||
if (!applicationListAll.value.length) return;
|
if (!applicationListAll.value.length) return;
|
||||||
|
if (searchKey.value) return;
|
||||||
|
|
||||||
const selectedIds = [];
|
const selectedIds = [];
|
||||||
props.editData.requestFormDetailList.forEach((detail) => {
|
props.editData.requestFormDetailList.forEach((detail) => {
|
||||||
@@ -486,26 +481,29 @@ const submit = () => {
|
|||||||
if (!projectWithDepartment(transferValue.value, 2)) {
|
if (!projectWithDepartment(transferValue.value, 2)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let applicationListAllFilter = applicationListAll.value.filter((item) => {
|
let applicationListAllFilter = transferValue.value.map((id) => {
|
||||||
return transferValue.value.includes(item.adviceDefinitionId);
|
let item = applicationListAll.value.find((i) => i.adviceDefinitionId == id);
|
||||||
});
|
if (!item) {
|
||||||
applicationListAllFilter = applicationListAllFilter.map((item) => {
|
item = selectedItemsCache.value.get(id);
|
||||||
|
}
|
||||||
|
if (!item) return null;
|
||||||
|
const priceInfo = item.priceList?.[0] || {};
|
||||||
return {
|
return {
|
||||||
adviceDefinitionId: item.adviceDefinitionId /** 诊疗定义id */,
|
adviceDefinitionId: item.adviceDefinitionId /** 诊疗定义id */,
|
||||||
quantity: 1, // /** 请求数量 */
|
quantity: 1, // /** 请求数量 */
|
||||||
unitCode: item.priceList[0].unitCode /** 请求单位编码 */,
|
unitCode: item.unitCode || priceInfo.unitCode || '' /** 请求单位编码 */,
|
||||||
unitPrice: item.priceList[0].price /** 单价 */,
|
unitPrice: item.price ?? priceInfo.price ?? 0 /** 单价 */,
|
||||||
totalPrice: item.priceList[0].price /** 总价 */,
|
totalPrice: item.price ?? priceInfo.price ?? 0 /** 总价 */,
|
||||||
positionId: form.targetDepartment || item.positionId, // 用户指定发往科室优先于项目默认执行科室
|
positionId: form.targetDepartment || item.positionId, // 用户指定发往科室优先于项目默认执行科室
|
||||||
ybClassEnum: item.ybClassEnum, //类别医保编码
|
ybClassEnum: item.ybClassEnum || '', //类别医保编码
|
||||||
conditionId: item.conditionId, //诊断ID
|
conditionId: item.conditionId || '', //诊断ID
|
||||||
encounterDiagnosisId: item.encounterDiagnosisId, //就诊诊断id
|
encounterDiagnosisId: item.encounterDiagnosisId || '', //就诊诊断id
|
||||||
adviceType: item.adviceType, ///** 医嘱类型 */
|
adviceType: item.adviceType || 3, ///** 医嘱类型 */
|
||||||
definitionId: item.priceList[0].definitionId, //费用定价主表ID */
|
definitionId: item.chargeItemDefinitionId || priceInfo.definitionId || '', //费用定价主表ID */
|
||||||
definitionDetailId: item.definitionDetailId, //费用定价子表ID */
|
definitionDetailId: item.definitionDetailId || priceInfo.definitionDetailId || '', //费用定价子表ID */
|
||||||
accountId: patientInfo.value.accountId, // // 账户id
|
accountId: patientInfo.value.accountId, // // 账户id
|
||||||
};
|
};
|
||||||
});
|
}).filter(Boolean);
|
||||||
const params = {
|
const params = {
|
||||||
activityList: applicationListAllFilter,
|
activityList: applicationListAllFilter,
|
||||||
patientId: patientInfo.value.patientId, //患者ID
|
patientId: patientInfo.value.patientId, //患者ID
|
||||||
@@ -520,6 +518,7 @@ const submit = () => {
|
|||||||
if (res.code === 200) {
|
if (res.code === 200) {
|
||||||
proxy.$message.success(isEditMode.value ? '修改成功' : res.msg);
|
proxy.$message.success(isEditMode.value ? '修改成功' : res.msg);
|
||||||
transferValue.value = [];
|
transferValue.value = [];
|
||||||
|
selectedItemsCache.value.clear();
|
||||||
emits('submitOk');
|
emits('submitOk');
|
||||||
} else {
|
} else {
|
||||||
proxy.$message.error(res.message);
|
proxy.$message.error(res.message);
|
||||||
|
|||||||
@@ -207,6 +207,7 @@ import {patientInfo} from '../../../store/patient.js';
|
|||||||
import {getDepartmentList} from '@/api/public.js';
|
import {getDepartmentList} from '@/api/public.js';
|
||||||
import {getEncounterDiagnosis} from '../../api.js';
|
import {getEncounterDiagnosis} from '../../api.js';
|
||||||
import {getExaminationPage, saveCheckd} from './api';
|
import {getExaminationPage, saveCheckd} from './api';
|
||||||
|
import {ActivityCategory} from '@/utils/medicalConstants';
|
||||||
import {ElMessage, ElMessageBox} from 'element-plus';
|
import {ElMessage, ElMessageBox} from 'element-plus';
|
||||||
import {WarningFilled, Warning, Refresh, Files, Document, EditPen, Aim, DocumentCopy} from '@element-plus/icons-vue';
|
import {WarningFilled, Warning, Refresh, Files, Document, EditPen, Aim, DocumentCopy} from '@element-plus/icons-vue';
|
||||||
|
|
||||||
@@ -276,6 +277,7 @@ const getList = () => {
|
|||||||
pageNo: 1,
|
pageNo: 1,
|
||||||
pageSize: 5000,
|
pageSize: 5000,
|
||||||
searchKey: '',
|
searchKey: '',
|
||||||
|
categoryCode: ActivityCategory.TEST,
|
||||||
})
|
})
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
if (res.code === 200 && res.data?.records) {
|
if (res.code === 200 && res.data?.records) {
|
||||||
|
|||||||
@@ -198,7 +198,7 @@
|
|||||||
v-model="scope.row.adviceName"
|
v-model="scope.row.adviceName"
|
||||||
placeholder="请选择项目"
|
placeholder="请选择项目"
|
||||||
@input="handleChange"
|
@input="handleChange"
|
||||||
@click="handleFocus(scope.row, scope.$index)"
|
@focus="handleFocus(scope.row, scope.$index)"
|
||||||
@keyup.enter.stop="handleFocus(scope.row, scope.$index)"
|
@keyup.enter.stop="handleFocus(scope.row, scope.$index)"
|
||||||
@keydown="
|
@keydown="
|
||||||
(e) => {
|
(e) => {
|
||||||
@@ -640,6 +640,10 @@ function getListInfo(addNewRow) {
|
|||||||
};
|
};
|
||||||
})
|
})
|
||||||
.sort((a, b) => {
|
.sort((a, b) => {
|
||||||
|
// 没有 requestTime 的项(新增/组套添加)排在最前面
|
||||||
|
if (!a.requestTime && !b.requestTime) return 0;
|
||||||
|
if (!a.requestTime) return -1;
|
||||||
|
if (!b.requestTime) return 1;
|
||||||
return new Date(b.requestTime) - new Date(a.requestTime);
|
return new Date(b.requestTime) - new Date(a.requestTime);
|
||||||
});
|
});
|
||||||
getGroupMarkers(); // 更新标记
|
getGroupMarkers(); // 更新标记
|
||||||
@@ -896,31 +900,16 @@ function handleDiagnosisChange(item) {
|
|||||||
function handleFocus(row, index) {
|
function handleFocus(row, index) {
|
||||||
rowIndex.value = index;
|
rowIndex.value = index;
|
||||||
row.showPopover = true;
|
row.showPopover = true;
|
||||||
|
// Bug #555: handleFocus 只负责开 popover 和初始化查询参数,搜索由 handleChange 统一处理
|
||||||
|
// 避免异步 refresh 用旧闭包 searchKey 覆盖 handleChange 的搜索结果
|
||||||
const adviceType = row.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
|
const adviceType = row.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
|
||||||
// 用 adviceType + categoryCode 组合查找匹配的选项
|
let categoryCode = '';
|
||||||
|
if (row.adviceType !== undefined) {
|
||||||
const selectValue = (adviceType == 1 && row.categoryCode) ? '1-' + row.categoryCode : adviceType;
|
const selectValue = (adviceType == 1 && row.categoryCode) ? '1-' + row.categoryCode : adviceType;
|
||||||
const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType);
|
const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType);
|
||||||
// If the row has an explicit adviceType (saved/existing row), use its own categoryCode.
|
categoryCode = selectedItem ? selectedItem.categoryCode : (row.categoryCode || '');
|
||||||
// If no type is selected (new row), use empty string for global search across all categories.
|
|
||||||
const categoryCode = selectedItem ? selectedItem.categoryCode : (row.adviceType != null ? (row.categoryCode || '') : '');
|
|
||||||
const searchKey = row.adviceName || '';
|
|
||||||
|
|
||||||
nextTick(() => {
|
|
||||||
nextTick(() => {
|
|
||||||
const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
|
|
||||||
if (tableRef && tableRef.refresh) {
|
|
||||||
tableRef.refresh(adviceType, categoryCode, searchKey);
|
|
||||||
} else {
|
|
||||||
// fallback: 如果双重 nextTick 仍未挂载,延迟 100ms 再试
|
|
||||||
setTimeout(() => {
|
|
||||||
const tableRef2 = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
|
|
||||||
if (tableRef2 && tableRef2.refresh) {
|
|
||||||
tableRef2.refresh(adviceType, categoryCode, searchKey);
|
|
||||||
}
|
}
|
||||||
}, 100);
|
adviceQueryParams.value = { adviceType, categoryCode, searchKey: '' };
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleBlur(row) {
|
function handleBlur(row) {
|
||||||
@@ -929,20 +918,24 @@ function handleBlur(row) {
|
|||||||
|
|
||||||
function handleChange(value) {
|
function handleChange(value) {
|
||||||
adviceQueryParams.value.searchKey = value;
|
adviceQueryParams.value.searchKey = value;
|
||||||
// 搜索词变化时,调用当前行子组件的 refresh 方法
|
// @focus 已先于 @input 执行,rowIndex 必定有效
|
||||||
const index = rowIndex.value;
|
const currentIndex = rowIndex.value;
|
||||||
if (index >= 0) {
|
if (currentIndex < 0) return;
|
||||||
const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[index] : adviceTableRef.value;
|
const row = filterPrescriptionList.value[currentIndex];
|
||||||
|
// popover 被 blur 关闭后,用户继续输入时自行打开
|
||||||
|
if (!row.showPopover) {
|
||||||
|
row.showPopover = true;
|
||||||
|
}
|
||||||
|
const tableRef = Array.isArray(adviceTableRef.value) ? adviceTableRef.value[currentIndex] : adviceTableRef.value;
|
||||||
if (tableRef && tableRef.refresh) {
|
if (tableRef && tableRef.refresh) {
|
||||||
const row = filterPrescriptionList.value[index];
|
|
||||||
const adviceType = row?.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
|
const adviceType = row?.adviceType !== undefined ? row.adviceType : adviceQueryParams.value.adviceType;
|
||||||
// 用 adviceType + categoryCode 组合查找匹配的选项
|
let categoryCode = '';
|
||||||
|
if (row?.adviceType !== undefined) {
|
||||||
const selectValue = (adviceType == 1 && row?.categoryCode) ? '1-' + row.categoryCode : adviceType;
|
const selectValue = (adviceType == 1 && row?.categoryCode) ? '1-' + row.categoryCode : adviceType;
|
||||||
const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType);
|
const selectedItem = adviceTypeList.value.find(item => item.value === selectValue) || adviceTypeList.value.find(item => item.adviceType === adviceType);
|
||||||
// 修复Bug #486:当行没有显式选择医嘱类型时,不传categoryCode,让搜索在全药库中进行
|
categoryCode = selectedItem ? selectedItem.categoryCode : (adviceQueryParams.value.categoryCode || '');
|
||||||
const categoryCode = selectedItem ? selectedItem.categoryCode : (row?.adviceType !== undefined ? (adviceQueryParams.value.categoryCode || '') : '');
|
|
||||||
tableRef.refresh(adviceType, categoryCode, value);
|
|
||||||
}
|
}
|
||||||
|
tableRef.refresh(adviceType, categoryCode, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1579,11 +1572,24 @@ function handleSaveGroup(orderGroupList) {
|
|||||||
|
|
||||||
let successCount = 0;
|
let successCount = 0;
|
||||||
|
|
||||||
|
// 收集所有要添加的新行,最后统一 unshift 到数组开头(置顶显示)
|
||||||
|
const newRows = [];
|
||||||
|
|
||||||
|
// 记录循环前的数组长度,用于清理循环中创建的临时行
|
||||||
|
const originalLength = prescriptionList.value.length;
|
||||||
|
|
||||||
orderGroupList.forEach((item) => {
|
orderGroupList.forEach((item) => {
|
||||||
rowIndex.value = prescriptionList.value.length;
|
// 使用临时索引,先追加到末尾用于 setValue 填充
|
||||||
|
const tempIndex = prescriptionList.value.length;
|
||||||
|
prescriptionList.value[tempIndex] = {
|
||||||
|
uniqueKey: nextId.value++,
|
||||||
|
isEdit: false,
|
||||||
|
statusEnum: 1,
|
||||||
|
};
|
||||||
|
|
||||||
if (!item) {
|
if (!item) {
|
||||||
console.warn('组套中的项目为空');
|
console.warn('组套中的项目为空');
|
||||||
|
prescriptionList.value.splice(tempIndex, 1);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1609,18 +1615,12 @@ function handleSaveGroup(orderGroupList) {
|
|||||||
therapyEnum: item.orderDetailInfos?.therapyEnum || '1',
|
therapyEnum: item.orderDetailInfos?.therapyEnum || '1',
|
||||||
};
|
};
|
||||||
|
|
||||||
// 预初始化空行(组套项带预填值,设为 false 让明细字段在表格中直接展示)
|
rowIndex.value = tempIndex;
|
||||||
prescriptionList.value[rowIndex.value] = {
|
|
||||||
uniqueKey: nextId.value++,
|
|
||||||
isEdit: false,
|
|
||||||
statusEnum: 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
setValue(mergedDetail);
|
setValue(mergedDetail);
|
||||||
|
|
||||||
// 创建新的处方项目
|
// 创建新的处方项目
|
||||||
const newRow = {
|
const newRow = {
|
||||||
...prescriptionList.value[rowIndex.value],
|
...prescriptionList.value[tempIndex],
|
||||||
patientId: patientInfo.value.patientId,
|
patientId: patientInfo.value.patientId,
|
||||||
encounterId: patientInfo.value.encounterId,
|
encounterId: patientInfo.value.encounterId,
|
||||||
accountId: accountId.value,
|
accountId: accountId.value,
|
||||||
@@ -1639,12 +1639,12 @@ function handleSaveGroup(orderGroupList) {
|
|||||||
orgId: resolveOrgId(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '',
|
orgId: resolveOrgId(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '',
|
||||||
// 🔧 修复:同时存储 orgName,确保树匹配不到时仍有中文名称可显示
|
// 🔧 修复:同时存储 orgName,确保树匹配不到时仍有中文名称可显示
|
||||||
orgName: findOrgName(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || mergedDetail.orgName || patientInfo.value?.inHospitalOrgName || '',
|
orgName: findOrgName(mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || mergedDetail.orgName || patientInfo.value?.inHospitalOrgName || '',
|
||||||
dbOpType: prescriptionList.value[rowIndex.value].requestId ? '2' : '1',
|
dbOpType: prescriptionList.value[tempIndex].requestId ? '2' : '1',
|
||||||
conditionId: conditionId.value,
|
conditionId: conditionId.value,
|
||||||
conditionDefinitionId: conditionDefinitionId.value,
|
conditionDefinitionId: conditionDefinitionId.value,
|
||||||
encounterDiagnosisId: encounterDiagnosisId.value,
|
encounterDiagnosisId: encounterDiagnosisId.value,
|
||||||
diagnosisName: diagnosisName.value,
|
diagnosisName: diagnosisName.value,
|
||||||
therapyEnum: prescriptionList.value[rowIndex.value]?.therapyEnum || mergedDetail.therapyEnum || '1',
|
therapyEnum: prescriptionList.value[tempIndex]?.therapyEnum || mergedDetail.therapyEnum || '1',
|
||||||
// 🔧 修复:确保组套医嘱的 categoryEnum 被正确映射,防止后端 NPE
|
// 🔧 修复:确保组套医嘱的 categoryEnum 被正确映射,防止后端 NPE
|
||||||
categoryEnum: mergedDetail?.categoryEnum || mergedDetail?.categoryCode || item?.categoryCode,
|
categoryEnum: mergedDetail?.categoryEnum || mergedDetail?.categoryCode || item?.categoryCode,
|
||||||
};
|
};
|
||||||
@@ -1663,11 +1663,14 @@ function handleSaveGroup(orderGroupList) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
newRow.contentJson = JSON.stringify(newRow);
|
newRow.contentJson = JSON.stringify(newRow);
|
||||||
prescriptionList.value[rowIndex.value] = newRow;
|
newRows.push(newRow);
|
||||||
successCount++;
|
successCount++;
|
||||||
});
|
});
|
||||||
|
|
||||||
if (successCount > 0) {
|
// 清理循环中创建的临时行,统一添加到数组开头(置顶显示)
|
||||||
|
if (newRows.length > 0) {
|
||||||
|
prescriptionList.value.splice(originalLength); // 移除循环中追加到末尾的临时行
|
||||||
|
prescriptionList.value.unshift(...newRows);
|
||||||
proxy.$modal.msgSuccess(`成功添加 ${successCount} 个医嘱项`);
|
proxy.$modal.msgSuccess(`成功添加 ${successCount} 个医嘱项`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -740,7 +740,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-form>
|
</el-form>
|
||||||
|
|
||||||
<!-- 结果表格区 -->
|
<!-- 结果表格卡片 -->
|
||||||
|
<el-card shadow="never" class="apply-card">
|
||||||
<el-table
|
<el-table
|
||||||
ref="applyTableRef"
|
ref="applyTableRef"
|
||||||
v-loading="applyLoading"
|
v-loading="applyLoading"
|
||||||
@@ -749,8 +750,7 @@
|
|||||||
@row-click="handleApplyRowClick"
|
@row-click="handleApplyRowClick"
|
||||||
:row-class-name="tableRowClassName"
|
:row-class-name="tableRowClassName"
|
||||||
style="width: 100%"
|
style="width: 100%"
|
||||||
max-height="340"
|
max-height="320"
|
||||||
:scroll="{ y: 340 }"
|
|
||||||
>
|
>
|
||||||
<el-table-column type="selection" width="55" :selectable="handleSelectable" />
|
<el-table-column type="selection" width="55" :selectable="handleSelectable" />
|
||||||
<el-table-column label="ID" align="center" width="80" fixed>
|
<el-table-column label="ID" align="center" width="80" fixed>
|
||||||
@@ -779,19 +779,20 @@
|
|||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="主刀医生" align="center" width="100" prop="mainSurgeonName" />
|
<el-table-column label="主刀医生" align="center" width="100" prop="mainSurgeonName" />
|
||||||
</el-table>
|
</el-table>
|
||||||
|
<!-- 分页在卡片内部 -->
|
||||||
<!-- 底部分页区 -->
|
<div class="apply-pagination">
|
||||||
<div class="pagination-container apply-pagination">
|
|
||||||
<pagination
|
<pagination
|
||||||
v-show="applyTotal > 0"
|
v-show="applyTotal > 0"
|
||||||
:total="applyTotal"
|
:total="applyTotal"
|
||||||
:page="applyQueryParams.pageNo"
|
:page="applyQueryParams.pageNo"
|
||||||
:limit="applyQueryParams.pageSize"
|
:limit="applyQueryParams.pageSize"
|
||||||
|
layout="total, sizes, prev, pager, next"
|
||||||
@update:page="val => applyQueryParams.pageNo = val"
|
@update:page="val => applyQueryParams.pageNo = val"
|
||||||
@update:limit="val => applyQueryParams.pageSize = val"
|
@update:limit="val => applyQueryParams.pageSize = val"
|
||||||
@pagination="getSurgicalScheduleList"
|
@pagination="getSurgicalScheduleList"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
</el-card>
|
||||||
<!-- 底部操作区 -->
|
<!-- 底部操作区 -->
|
||||||
<template #footer>
|
<template #footer>
|
||||||
<div class="dialog-footer" style="padding-top: 12px; border-top: 1px solid #ebeef5">
|
<div class="dialog-footer" style="padding-top: 12px; border-top: 1px solid #ebeef5">
|
||||||
@@ -1817,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;
|
||||||
@@ -2339,19 +2343,35 @@ function getRowClassName({ row, rowIndex }) {
|
|||||||
margin-left: 10px;
|
margin-left: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 手术申请查询弹窗 — 分页与footer间距 */
|
/* 手术申请查询弹窗 — flex 布局确保分页不溢出 */
|
||||||
.surgery-apply-dialog :deep(.el-dialog__body) {
|
.surgery-apply-dialog :deep(.el-dialog__body) {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
padding-bottom: 16px;
|
padding-bottom: 16px;
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
.surgery-apply-dialog :deep(.el-dialog__footer) {
|
.surgery-apply-dialog :deep(.el-dialog__footer) {
|
||||||
padding-top: 8px;
|
padding-top: 0;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog :deep(.apply-card) {
|
||||||
|
flex: 1;
|
||||||
|
overflow: hidden;
|
||||||
|
min-height: 0;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog :deep(.apply-card .el-card__body) {
|
||||||
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
.surgery-apply-dialog :deep(.apply-pagination) {
|
.surgery-apply-dialog :deep(.apply-pagination) {
|
||||||
padding-top: 12px;
|
display: flex;
|
||||||
padding-bottom: 16px;
|
justify-content: flex-end;
|
||||||
|
padding-top: 8px;
|
||||||
|
border-top: 1px solid #ebeef5;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog :deep(.apply-pagination .pagination-container) {
|
||||||
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
.surgery-apply-dialog :deep(.apply-pagination .el-pagination) {
|
.surgery-apply-dialog :deep(.apply-pagination .el-pagination) {
|
||||||
margin-right: 80px;
|
position: static;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 选中行样式 */
|
/* 选中行样式 */
|
||||||
@@ -2367,17 +2387,33 @@ function getRowClassName({ row, rowIndex }) {
|
|||||||
|
|
||||||
<style>
|
<style>
|
||||||
/* 手术申请查询弹窗 — 非 scoped 确保穿透 teleport */
|
/* 手术申请查询弹窗 — 非 scoped 确保穿透 teleport */
|
||||||
.surgery-apply-dialog .apply-pagination {
|
|
||||||
padding-top: 12px !important;
|
|
||||||
padding-bottom: 16px !important;
|
|
||||||
}
|
|
||||||
.surgery-apply-dialog .apply-pagination .el-pagination {
|
|
||||||
margin-right: 80px !important;
|
|
||||||
}
|
|
||||||
.surgery-apply-dialog .el-dialog__body {
|
.surgery-apply-dialog .el-dialog__body {
|
||||||
|
display: flex !important;
|
||||||
|
flex-direction: column !important;
|
||||||
padding-bottom: 16px !important;
|
padding-bottom: 16px !important;
|
||||||
|
overflow: hidden !important;
|
||||||
}
|
}
|
||||||
.surgery-apply-dialog .el-dialog__footer {
|
.surgery-apply-dialog .el-dialog__footer {
|
||||||
|
padding-top: 0 !important;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog .apply-card {
|
||||||
|
flex: 1 !important;
|
||||||
|
overflow: hidden !important;
|
||||||
|
min-height: 0 !important;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog .apply-card .el-card__body {
|
||||||
|
overflow-y: auto !important;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog .apply-pagination {
|
||||||
|
display: flex !important;
|
||||||
|
justify-content: flex-end !important;
|
||||||
padding-top: 8px !important;
|
padding-top: 8px !important;
|
||||||
|
border-top: 1px solid #ebeef5 !important;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog .apply-pagination .pagination-container {
|
||||||
|
margin-top: 0 !important;
|
||||||
|
}
|
||||||
|
.surgery-apply-dialog .apply-pagination .el-pagination {
|
||||||
|
position: static !important;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
Reference in New Issue
Block a user