1 Commits

Author SHA1 Message Date
赵云
80695d90f9 Fix Bug #475: 【住院医生工作站】开立检查申请单报错"请先配置当前时间段的执行科室"后,系统仍生成申请记录
根因:saveRequestForm 方法中,执行科室配置校验(activityOrganizationConfig.isEmpty)位于 saveOrUpdate(requestForm) 之后,导致即使校验失败抛出异常,RequestForm 记录已被写入数据库。
修复:将校验逻辑移至方法开头,在任何数据库操作之前执行,确保校验失败时不产生任何脏数据。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-13 18:12:18 +08:00
61 changed files with 1025 additions and 3353 deletions

View File

@@ -85,13 +85,18 @@ public class SysDictTypeServiceImpl implements ISysDictTypeService {
String trimmedKey = searchKey.trim(); String trimmedKey = searchKey.trim();
return dictDataMapper.selectDictDataByTypeWithSearch(dictType, trimmedKey); return dictDataMapper.selectDictDataByTypeWithSearch(dictType, trimmedKey);
} }
// 直接查询数据库,避免缓存中为空数据导致前端下拉框显示"无数据" // 否则使用原有方法(带缓存)
List<SysDictData> dictDatas = dictDataMapper.selectDictDataByType(dictType); List<SysDictData> dictDatas = DictUtils.getDictCache(dictType);
if (StringUtils.isNotEmpty(dictDatas)) {
return dictDatas;
}
dictDatas = dictDataMapper.selectDictDataByType(dictType);
if (StringUtils.isNotEmpty(dictDatas)) { if (StringUtils.isNotEmpty(dictDatas)) {
DictUtils.setDictCache(dictType, dictDatas); DictUtils.setDictCache(dictType, dictDatas);
return dictDatas;
} }
return dictDatas; return null;
} }
/** /**

View File

@@ -366,7 +366,7 @@ public class SurgeryAppServiceImpl implements ISurgeryAppService {
serviceRequest.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue());// 治疗类型 serviceRequest.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue());// 治疗类型
serviceRequest.setQuantity(BigDecimal.valueOf(1)); // 请求数量 serviceRequest.setQuantity(BigDecimal.valueOf(1)); // 请求数量
serviceRequest.setUnitCode(""); // 请求单位编码 serviceRequest.setUnitCode(""); // 请求单位编码
serviceRequest.setCategoryEnum(24); // 请求类型:24-手术(新值域,避开 adviceType 碰撞) serviceRequest.setCategoryEnum(4); // 请求类型4-手术
serviceRequest.setActivityId(surgeryId); // 手术ID作为诊疗定义id serviceRequest.setActivityId(surgeryId); // 手术ID作为诊疗定义id
serviceRequest.setPatientId(surgeryDto.getPatientId()); // 患者 serviceRequest.setPatientId(surgeryDto.getPatientId()); // 患者
serviceRequest.setRequesterId(practitionerId); // 开方医生 serviceRequest.setRequesterId(practitionerId); // 开方医生

View File

@@ -7,6 +7,7 @@ import com.core.common.core.domain.R;
import com.core.common.core.domain.model.LoginUser; import com.core.common.core.domain.model.LoginUser;
import com.core.common.utils.SecurityUtils; import com.core.common.utils.SecurityUtils;
import com.openhis.administration.domain.Patient; import com.openhis.administration.domain.Patient;
import com.openhis.administration.service.IOrganizationService;
import com.openhis.administration.service.IPatientService; import com.openhis.administration.service.IPatientService;
import com.openhis.clinical.domain.Surgery; import com.openhis.clinical.domain.Surgery;
import com.openhis.clinical.service.ISurgeryService; import com.openhis.clinical.service.ISurgeryService;
@@ -27,6 +28,7 @@ import java.io.IOException;
import java.io.OutputStream; import java.io.OutputStream;
import java.io.PrintWriter; import java.io.PrintWriter;
import java.net.URLEncoder; import java.net.URLEncoder;
import java.time.LocalDate;
import java.time.LocalDateTime; import java.time.LocalDateTime;
import java.util.Date; import java.util.Date;
import java.util.List; import java.util.List;
@@ -202,8 +204,6 @@ public class SurgicalScheduleAppServiceImpl implements ISurgicalScheduleAppServi
return R.fail("新增手术安排失败"); return R.fail("新增手术安排失败");
} }
syncSurgeryIncisionLevel(opSchedule.getOperCode(), opCreateScheduleDto.getIncisionLevel());
// Bug #247 修复:更新手术申请单状态为已排期 (1) // Bug #247 修复:更新手术申请单状态为已排期 (1)
if (opCreateScheduleDto.getApplyId() != null) { if (opCreateScheduleDto.getApplyId() != null) {
try { try {
@@ -300,8 +300,6 @@ public class SurgicalScheduleAppServiceImpl implements ISurgicalScheduleAppServi
return R.fail("修改手术安排失败"); return R.fail("修改手术安排失败");
} }
syncSurgeryIncisionLevel(opScheduleDto.getOperCode(), opScheduleDto.getIncisionLevel());
return R.ok("修改手术安排成功"); return R.ok("修改手术安排成功");
} }
@@ -435,28 +433,6 @@ public class SurgicalScheduleAppServiceImpl implements ISurgicalScheduleAppServi
return scheduleDate.format(java.time.format.DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); return scheduleDate.format(java.time.format.DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
} }
/**
* 同步手术申请表中的切口类型
*/
private void syncSurgeryIncisionLevel(String surgeryNo, Integer incisionLevel) {
if (surgeryNo == null || surgeryNo.isEmpty() || incisionLevel == null) {
return;
}
LambdaQueryWrapper<Surgery> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(Surgery::getSurgeryNo, surgeryNo)
.eq(Surgery::getDeleteFlag, "0");
Surgery surgery = surgeryService.getOne(queryWrapper);
if (surgery == null) {
log.warn("未找到需要同步切口类型的手术申请记录 - surgeryNo: {}", surgeryNo);
return;
}
surgery.setIncisionLevel(incisionLevel);
surgery.setUpdateTime(new Date());
surgeryService.updateById(surgery);
}
/** /**
* 填充手术申请中缺失的名称字段 * 填充手术申请中缺失的名称字段
* 在创建手术安排时调用确保关联的cli_surgery表中的名称字段有值 * 在创建手术安排时调用确保关联的cli_surgery表中的名称字段有值

View File

@@ -4,6 +4,7 @@ import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data; import lombok.Data;
import java.math.BigDecimal; import java.math.BigDecimal;
import java.time.LocalDate;
import java.time.LocalDateTime; import java.time.LocalDateTime;
@Data @Data
@@ -84,11 +85,6 @@ public class OpCreateScheduleDto {
*/ */
private String surgerySite; private String surgerySite;
/**
* 切口类型
*/
private Integer incisionLevel;
/** /**
* 入院时间 * 入院时间
*/ */
@@ -265,11 +261,6 @@ public class OpCreateScheduleDto {
*/ */
private String remark; private String remark;
/**
* 费用类别
*/
private String feeType;
/** /**
* 创建时间 * 创建时间
*/ */

View File

@@ -4,7 +4,6 @@ import com.fasterxml.jackson.annotation.JsonFormat;
import com.openhis.surgicalschedule.domain.OpSchedule; import com.openhis.surgicalschedule.domain.OpSchedule;
import lombok.Data; import lombok.Data;
import lombok.EqualsAndHashCode; import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;
import org.springframework.format.annotation.DateTimeFormat; import org.springframework.format.annotation.DateTimeFormat;
import java.time.LocalDate; import java.time.LocalDate;
@@ -94,12 +93,6 @@ public class OpScheduleDto extends OpSchedule {
* 手术类型 * 手术类型
*/ */
private String surgeryType; private String surgeryType;
/**
* 切口类型
*/
private Integer incisionLevel;
/** /**
* 申请科室 * 申请科室
*/ */
@@ -114,4 +107,8 @@ public class OpScheduleDto extends OpSchedule {
*/ */
private String createByName; private String createByName;
/**
* 费用类别
*/
private String feeType;
} }

View File

@@ -35,7 +35,6 @@ import com.openhis.medication.service.IMedicationDispenseService;
import com.openhis.medication.service.IMedicationRequestService; import com.openhis.medication.service.IMedicationRequestService;
import com.openhis.web.chargemanage.mapper.OutpatientRegistrationAppMapper; import com.openhis.web.chargemanage.mapper.OutpatientRegistrationAppMapper;
import com.openhis.web.doctorstation.appservice.IDoctorStationAdviceAppService; import com.openhis.web.doctorstation.appservice.IDoctorStationAdviceAppService;
import com.openhis.web.doctorstation.appservice.IDoctorStationInspectionLabApplyService;
import com.openhis.web.doctorstation.dto.*; import com.openhis.web.doctorstation.dto.*;
import com.openhis.web.doctorstation.mapper.DoctorStationAdviceAppMapper; import com.openhis.web.doctorstation.mapper.DoctorStationAdviceAppMapper;
import com.openhis.web.doctorstation.utils.AdviceUtils; import com.openhis.web.doctorstation.utils.AdviceUtils;
@@ -48,15 +47,12 @@ import com.openhis.workflow.domain.InventoryItem;
import com.openhis.workflow.domain.ServiceRequest; import com.openhis.workflow.domain.ServiceRequest;
import com.openhis.workflow.service.*; import com.openhis.workflow.service.*;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
import javax.annotation.Resource; import javax.annotation.Resource;
import java.math.BigDecimal; import java.math.BigDecimal;
import java.util.*; import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors; import java.util.stream.Collectors;
/** /**
@@ -66,9 +62,6 @@ import java.util.stream.Collectors;
@Service @Service
public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAppService { public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAppService {
private static final Pattern INSPECTION_APPLY_NO_JSON =
Pattern.compile("\"applyNo\"\\s*:\\s*\"([^\"]+)\"");
@Resource @Resource
AssignSeqUtil assignSeqUtil; AssignSeqUtil assignSeqUtil;
@@ -125,13 +118,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
@Resource @Resource
IInventoryItemService inventoryItemService; IInventoryItemService inventoryItemService;
/**
* 与检验申请实现存在循环依赖,需延迟注入;删除诊疗医嘱时按 contentJson 级联作废检验申请单。
*/
@Resource
@Lazy
private IDoctorStationInspectionLabApplyService iDoctorStationInspectionLabApplyService;
// 缓存 key 前缀 // 缓存 key 前缀
private static final String ADVICE_BASE_INFO_CACHE_PREFIX = "advice:base:info:"; private static final String ADVICE_BASE_INFO_CACHE_PREFIX = "advice:base:info:";
// 缓存过期时间(小时) // 缓存过期时间(小时)
@@ -1543,44 +1529,21 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
deviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.DEVICE_RES_NO.getPrefix(), 4)); deviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.DEVICE_RES_NO.getPrefix(), 4));
} }
deviceRequest.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源 deviceRequest.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源
deviceRequest.setPrescriptionNo(adviceSaveDto.getSourceBillNo()); // 来源业务单据号(手术单号)
deviceRequest.setQuantity(adviceSaveDto.getQuantity()); // 请求数量 deviceRequest.setQuantity(adviceSaveDto.getQuantity()); // 请求数量
deviceRequest.setUnitCode(adviceSaveDto.getUnitCode()); // 请求单位编码 deviceRequest.setUnitCode(adviceSaveDto.getUnitCode()); // 请求单位编码
deviceRequest.setLotNumber(adviceSaveDto.getLotNumber());// 产品批号 deviceRequest.setLotNumber(adviceSaveDto.getLotNumber());// 产品批号
deviceRequest.setCategoryEnum(adviceSaveDto.getCategoryEnum()); // 请求类型 deviceRequest.setCategoryEnum(adviceSaveDto.getCategoryEnum()); // 请求类型
// 🔧 BugFix #498: categoryEnum=22(检查) 走 ServiceRequest不走 DeviceRequest deviceRequest.setDeviceDefId(adviceSaveDto.getAdviceDefinitionId());// 耗材定义id
// 检查申请单的诊疗定义ID存在 activityId不在 adviceDefinitionId
// deviceDefId 对应耗材定义ID不能用诊疗定义ID填充
if (adviceSaveDto.getCategoryEnum() == 22) {
log.info("handDevice skip - 检查申请单(categoryEnum=22) 走 ServiceRequest 路径,跳过 DeviceRequest 保存");
continue; // 跳过本次循环,不走耗材请求路径
} else if (adviceSaveDto.getAdviceDefinitionId() != null) {
deviceRequest.setDeviceDefId(adviceSaveDto.getAdviceDefinitionId());// 耗材定义id
} else {
log.warn("handDevice - deviceDefId 为空adviceDefinitionId=null, categoryEnum={}", adviceSaveDto.getCategoryEnum());
}
deviceRequest.setPatientId(adviceSaveDto.getPatientId()); // 患者 deviceRequest.setPatientId(adviceSaveDto.getPatientId()); // 患者
deviceRequest.setRequesterId(adviceSaveDto.getPractitionerId()); // 开方医生 deviceRequest.setRequesterId(adviceSaveDto.getPractitionerId()); // 开方医生
deviceRequest.setOrgId(adviceSaveDto.getFounderOrgId());// 开方人科室 deviceRequest.setOrgId(adviceSaveDto.getFounderOrgId());// 开方人科室
deviceRequest.setReqAuthoredTime(curDate); // 请求开始时间 deviceRequest.setReqAuthoredTime(curDate); // 请求开始时间
// 发放耗材房若前端未传locationId优先沿用已有DeviceRequest的performLocation否则使用登录用户科室) // 发放耗材房若前端未传locationId使用登录用户科室作为默认值
Long locId = adviceSaveDto.getLocationId(); Long locId = adviceSaveDto.getLocationId();
if (locId == null) { if (locId == null) {
// 尝试从已有DeviceRequest获取原始的performLocation locId = SecurityUtils.getLoginUser().getOrgId();
if (adviceSaveDto.getRequestId() != null) { log.info("耗材locationId为空使用登录用户科室作为默认值: locationId={}", locId);
DeviceRequest existingDevice = iDeviceRequestService.getById(adviceSaveDto.getRequestId());
if (existingDevice != null && existingDevice.getPerformLocation() != null) {
locId = existingDevice.getPerformLocation();
log.info("耗材locationId为空使用已有DeviceRequest的performLocation: locationId={}", locId);
}
}
// 如果已有记录也没有performLocation则使用登录用户科室作为兜底
if (locId == null) {
locId = SecurityUtils.getLoginUser().getOrgId();
log.info("耗材locationId为空且无已有记录使用登录用户科室作为默认值: locationId={}", locId);
}
} }
deviceRequest.setPerformLocation(locId); deviceRequest.setPerformLocation(locId);
deviceRequest.setEncounterId(adviceSaveDto.getEncounterId()); // 就诊id deviceRequest.setEncounterId(adviceSaveDto.getEncounterId()); // 就诊id
@@ -1721,21 +1684,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
} }
} }
/**
* 从诊疗医嘱 contentJson 中解析检验申请单号(检验保存时写入形如 {"applyNo":"..."})。
*/
private String extractInspectionApplyNoFromContentJson(String contentJson) {
if (StringUtils.isBlank(contentJson) || !contentJson.contains("applyNo")) {
return null;
}
Matcher m = INSPECTION_APPLY_NO_JSON.matcher(contentJson);
if (!m.find()) {
return null;
}
String applyNo = m.group(1).trim();
return StringUtils.isBlank(applyNo) ? null : applyNo;
}
/** /**
* 处理诊疗 * 处理诊疗
*/ */
@@ -1784,8 +1732,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
} }
} }
} }
// 检验申请单在医嘱 contentJson 中写入 applyNo从医嘱删除时需先级联作废检验单避免检验页签仍显示孤儿申请
Map<String, List<Long>> labApplyNoToRequestIds = new LinkedHashMap<>();
for (AdviceSaveDto adviceSaveDto : deleteList) { for (AdviceSaveDto adviceSaveDto : deleteList) {
Long requestId = adviceSaveDto.getRequestId(); Long requestId = adviceSaveDto.getRequestId();
// 🔧 Bug #442: 跳过 requestId 为 null 的记录,避免删除不存在的诊疗请求 // 🔧 Bug #442: 跳过 requestId 为 null 的记录,避免删除不存在的诊疗请求
@@ -1794,35 +1740,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
continue; continue;
} }
iServiceRequestService.removeById(requestId);// 删除诊疗 iServiceRequestService.removeById(requestId);// 删除诊疗
ServiceRequest existing = iServiceRequestService.getById(adviceSaveDto.getRequestId());
if (existing == null) {
continue;
}
String applyNo = extractInspectionApplyNoFromContentJson(existing.getContentJson());
if (StringUtils.isNotBlank(applyNo)) {
labApplyNoToRequestIds.computeIfAbsent(applyNo, k -> new ArrayList<>())
.add(adviceSaveDto.getRequestId());
}
}
Set<Long> labCascadeSkippedRequestIds = new HashSet<>();
for (Map.Entry<String, List<Long>> e : labApplyNoToRequestIds.entrySet()) {
R<?> delLab = iDoctorStationInspectionLabApplyService.deleteInspectionLabApply(e.getKey());
if (delLab != null && R.isSuccess(delLab)) {
labCascadeSkippedRequestIds.addAll(e.getValue());
log.info("handService - 级联作废检验申请单 applyNo={},已跳过重复删除的医嘱 requestIds={}",
e.getKey(), e.getValue());
} else {
String msg = delLab != null && StringUtils.isNotEmpty(delLab.getMsg()) ? delLab.getMsg() : "删除检验申请单失败";
log.warn("handService - 级联作废检验申请单未成功 applyNo={} msg={},将回退为仅删除当前医嘱记录",
e.getKey(), msg);
}
}
for (AdviceSaveDto adviceSaveDto : deleteList) {
if (labCascadeSkippedRequestIds.contains(adviceSaveDto.getRequestId())) {
continue;
}
Long requestId = adviceSaveDto.getRequestId();
iServiceRequestService.removeById(requestId);// 删除诊疗
iServiceRequestService.remove( iServiceRequestService.remove(
new LambdaQueryWrapper<ServiceRequest>().eq(ServiceRequest::getParentId, new LambdaQueryWrapper<ServiceRequest>().eq(ServiceRequest::getParentId,
requestId));// 删除诊疗套餐对应的子项 requestId));// 删除诊疗套餐对应的子项
@@ -1907,7 +1824,6 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
serviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.SERVICE_RES_NO.getPrefix(), 4)); serviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.SERVICE_RES_NO.getPrefix(), 4));
} }
serviceRequest.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源 serviceRequest.setGenerateSourceEnum(GenerateSource.DOCTOR_PRESCRIPTION.getValue()); // 生成来源
serviceRequest.setPrescriptionNo(adviceSaveDto.getSourceBillNo()); // 来源业务单据号(手术单号)
serviceRequest.setQuantity(adviceSaveDto.getQuantity()); // 请求数量 serviceRequest.setQuantity(adviceSaveDto.getQuantity()); // 请求数量
serviceRequest.setUnitCode(adviceSaveDto.getUnitCode()); // 请求单位编码 serviceRequest.setUnitCode(adviceSaveDto.getUnitCode()); // 请求单位编码
@@ -2103,10 +2019,10 @@ public class DoctorStationAdviceAppServiceImpl implements IDoctorStationAdviceAp
CommonConstants.TableName.MED_MEDICATION_REQUEST, CommonConstants.TableName.WOR_DEVICE_REQUEST, CommonConstants.TableName.MED_MEDICATION_REQUEST, CommonConstants.TableName.WOR_DEVICE_REQUEST,
CommonConstants.TableName.WOR_SERVICE_REQUEST, practitionerId, Whether.NO.getCode(), CommonConstants.TableName.WOR_SERVICE_REQUEST, practitionerId, Whether.NO.getCode(),
sourceEnum, sourceBillNo); sourceEnum, sourceBillNo);
// 手术计费场景sourceBillNo 不为空时过滤掉药品1,保留耗材2和诊疗3/6 // 手术计费场景sourceBillNo 不为空时,只保留诊疗请求3/6过滤掉药品1耗材2
if (sourceBillNo != null && !sourceBillNo.isEmpty()) { if (sourceBillNo != null && !sourceBillNo.isEmpty()) {
requestBaseInfo.removeIf(dto -> dto.getAdviceType() != null requestBaseInfo.removeIf(dto -> dto.getAdviceType() != null
&& dto.getAdviceType() == 1); && (dto.getAdviceType() == 1 || dto.getAdviceType() == 2));
} }
for (RequestBaseDto requestBaseDto : requestBaseInfo) { for (RequestBaseDto requestBaseDto : requestBaseInfo) {
// 请求状态 // 请求状态

View File

@@ -261,10 +261,8 @@ public class DoctorStationDiagnosisAppServiceImpl implements IDoctorStationDiagn
// 设置创建时间,避免数据库约束错误 // 设置创建时间,避免数据库约束错误
encounterDiagnosis.setCreateTime(new Date()); encounterDiagnosis.setCreateTime(new Date());
iEncounterDiagnosisService.saveOrUpdate(encounterDiagnosis); iEncounterDiagnosisService.saveOrUpdate(encounterDiagnosis);
// 回写就诊诊断ID供前端后续更新使用
saveDiagnosisChildParam.setEncounterDiagnosisId(encounterDiagnosis.getId());
} }
return R.ok(saveDiagnosisParam, MessageUtils.createMessage(PromptMsgConstant.Common.M00002, new Object[] {"诊断"})); return R.ok(null, MessageUtils.createMessage(PromptMsgConstant.Common.M00002, new Object[] {"诊断"}));
} }
@@ -582,11 +580,7 @@ public class DoctorStationDiagnosisAppServiceImpl implements IDoctorStationDiagn
@Override @Override
public R<?> saveInfectiousDiseaseReport(InfectiousDiseaseReportDto infectiousDiseaseReportDto) { public R<?> saveInfectiousDiseaseReport(InfectiousDiseaseReportDto infectiousDiseaseReportDto) {
// 检查卡片编号唯一性(新增时检查,编辑时排除当前记录) // 检查卡片编号唯一性(新增时检查,编辑时排除当前记录)
String cardNo = infectiousDiseaseReportDto.getCardNo(); String cardNo = infectiousDiseaseReportDto.getCardNo().trim();
if (cardNo == null || cardNo.trim().isEmpty()) {
return R.fail("卡片编号不能为空");
}
cardNo = cardNo.trim();
LambdaQueryWrapper<InfectiousDiseaseReport> queryWrapper = new LambdaQueryWrapper<>(); LambdaQueryWrapper<InfectiousDiseaseReport> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(InfectiousDiseaseReport::getCardNo, cardNo); queryWrapper.eq(InfectiousDiseaseReport::getCardNo, cardNo);
long count = iInfectiousDiseaseReportService.count(queryWrapper); long count = iInfectiousDiseaseReportService.count(queryWrapper);

View File

@@ -326,9 +326,6 @@ public class DoctorStationMainAppServiceImpl implements IDoctorStationMainAppSer
// 如果队列项存在且未完成,更新队列状态为已完成 // 如果队列项存在且未完成,更新队列状态为已完成
// 使用排除法而非白名单:只要不是"已完成"就可以完诊,覆盖跳过、等待等非标准流转状态 // 使用排除法而非白名单:只要不是"已完成"就可以完诊,覆盖跳过、等待等非标准流转状态
// Bug #401在更新前记录队列原始完成状态用于判断是否需要写入 div_log
boolean queueWasAlreadyCompleted = queueItem != null
&& TriageQueueStatus.COMPLETED.getValue().equals(queueItem.getStatus());
if (queueItem != null && if (queueItem != null &&
!TriageQueueStatus.COMPLETED.getValue().equals(queueItem.getStatus())) { !TriageQueueStatus.COMPLETED.getValue().equals(queueItem.getStatus())) {
java.time.LocalDateTime nowLocal = java.time.LocalDateTime.now().truncatedTo(ChronoUnit.SECONDS); java.time.LocalDateTime nowLocal = java.time.LocalDateTime.now().truncatedTo(ChronoUnit.SECONDS);
@@ -346,22 +343,19 @@ public class DoctorStationMainAppServiceImpl implements IDoctorStationMainAppSer
} }
// 写入 div_log 审计日志(独立于队列项,确保每次完诊都生成记录) // 写入 div_log 审计日志(独立于队列项,确保每次完诊都生成记录)
// Bug #401使用更新前记录的原始状态判断避免自身更新后将状态改为 COMPLETED 导致误判为"已完成" try {
if (!queueWasAlreadyCompleted) { LoginUser loginUser = SecurityUtils.getLoginUser();
try { DivLog divLog = new DivLog()
LoginUser loginUser = SecurityUtils.getLoginUser(); .setPoolId(divPoolId)
DivLog divLog = new DivLog() .setSlotId(divSlotId)
.setPoolId(divPoolId) .setOpUserId(loginUser != null ? loginUser.getUserId() : null)
.setSlotId(divSlotId) .setAction("COMPLETE")
.setOpUserId(loginUser != null ? loginUser.getUserId() : null) .setCreateTime(LocalDateTime.now())
.setAction("COMPLETE") .setUpdateAt(LocalDateTime.now())
.setCreateTime(LocalDateTime.now()) .setCreatedAt(LocalDateTime.now());
.setUpdateAt(LocalDateTime.now()) divLogService.save(divLog);
.setCreatedAt(LocalDateTime.now()); } catch (Exception e) {
divLogService.save(divLog); log.error("写入div_log审计日志失败", e);
} catch (Exception e) {
log.error("写入div_log审计日志失败", e);
}
} }
// 4. 更新状态、完成时间以及初复诊标识 // 4. 更新状态、完成时间以及初复诊标识

View File

@@ -178,26 +178,15 @@ public class AdviceUtils {
// 生命提示信息集合 // 生命提示信息集合
List<String> tipsList = new ArrayList<>(); List<String> tipsList = new ArrayList<>();
for (MedicationRequestUseExe medicationRequestUseExe : medUseExeList) { for (MedicationRequestUseExe medicationRequestUseExe : medUseExeList) {
// 第一步:按 performLocation 匹配指定药房的库存 // 聚合同一位置所有批次的库存总量
List<AdviceInventoryDto> matchedInventories = adviceInventory.stream() List<AdviceInventoryDto> matchedInventories = adviceInventory.stream()
.filter(inventoryDto -> medicationRequestUseExe.getMedicationId().equals(inventoryDto.getItemId()) .filter(inventoryDto -> medicationRequestUseExe.getMedicationId().equals(inventoryDto.getItemId())
&& CommonConstants.TableName.MED_MEDICATION_DEFINITION.equals(inventoryDto.getItemTable()) && CommonConstants.TableName.MED_MEDICATION_DEFINITION.equals(inventoryDto.getItemTable())
&& (medicationRequestUseExe.getPerformLocation() == null && medicationRequestUseExe.getPerformLocation().equals(inventoryDto.getLocationId())
|| medicationRequestUseExe.getPerformLocation().equals(inventoryDto.getLocationId()))
// 如果选择了具体的批次号,校验库存时需要加上批次号的匹配条件 // 如果选择了具体的批次号,校验库存时需要加上批次号的匹配条件
&& (StringUtils.isEmpty(medicationRequestUseExe.getLotNumber()) && (StringUtils.isEmpty(medicationRequestUseExe.getLotNumber())
|| medicationRequestUseExe.getLotNumber().equals(inventoryDto.getLotNumber()))) || medicationRequestUseExe.getLotNumber().equals(inventoryDto.getLotNumber())))
.collect(Collectors.toList()); .collect(Collectors.toList());
// 第二步:如果指定药房没有匹配到库存,则放宽条件查询所有药房的库存
if (matchedInventories.isEmpty()) {
matchedInventories = adviceInventory.stream()
.filter(inventoryDto -> medicationRequestUseExe.getMedicationId().equals(inventoryDto.getItemId())
&& CommonConstants.TableName.MED_MEDICATION_DEFINITION.equals(inventoryDto.getItemTable())
// 如果选择了具体的批次号,校验库存时需要加上批次号的匹配条件
&& (StringUtils.isEmpty(medicationRequestUseExe.getLotNumber())
|| medicationRequestUseExe.getLotNumber().equals(inventoryDto.getLotNumber())))
.collect(Collectors.toList());
}
// 匹配到库存信息 // 匹配到库存信息
if (!matchedInventories.isEmpty()) { if (!matchedInventories.isEmpty()) {
// 聚合所有批次的可用库存 // 聚合所有批次的可用库存

View File

@@ -178,8 +178,6 @@ public class AdviceProcessAppServiceImpl implements IAdviceProcessAppService {
inpatientAdviceParam.setEncounterIds(null); inpatientAdviceParam.setEncounterIds(null);
Integer exeStatus = inpatientAdviceParam.getExeStatus(); Integer exeStatus = inpatientAdviceParam.getExeStatus();
inpatientAdviceParam.setExeStatus(null); inpatientAdviceParam.setExeStatus(null);
// requestStatus由前端tab控制后端SQL已通过CASE条件处理校对状态过滤无需再作为SQL条件
inpatientAdviceParam.setRequestStatus(null);
// 构建查询条件 // 构建查询条件
QueryWrapper<InpatientAdviceParam> queryWrapper QueryWrapper<InpatientAdviceParam> queryWrapper
= HisQueryUtils.buildQueryWrapper(inpatientAdviceParam, null, null, null); = HisQueryUtils.buildQueryWrapper(inpatientAdviceParam, null, null, null);

View File

@@ -31,7 +31,6 @@ public class NursingRecordController {
* 获取住院患者信息 分页显示 * 获取住院患者信息 分页显示
* *
* @param nursingSearchParam 查询参数 * @param nursingSearchParam 查询参数
*
* @param searchKey 模糊查询 * @param searchKey 模糊查询
* @param pageNo 当前页码 * @param pageNo 当前页码
* @param pageSize 查询条数 * @param pageSize 查询条数

View File

@@ -63,20 +63,4 @@ public interface IRequestFormManageAppService {
* @return 申请单 * @return 申请单
*/ */
IPage<RequestFormPageDto> getRequestFormPage(RequestFormDto requestFormDto); IPage<RequestFormPageDto> getRequestFormPage(RequestFormDto requestFormDto);
/**
* 删除申请单(仅待签发状态可删除)
*
* @param requestFormId 申请单ID
* @return 结果
*/
R<?> deleteRequestForm(Long requestFormId);
/**
* 撤回申请单(已签发状态撤回至待签发)
*
* @param requestFormId 申请单ID
* @return 结果
*/
R<?> withdrawRequestForm(Long requestFormId);
} }

View File

@@ -341,7 +341,7 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
&& (DbOpType.INSERT.getCode().equals(e.getDbOpType()) || DbOpType.UPDATE.getCode().equals(e.getDbOpType()))) && (DbOpType.INSERT.getCode().equals(e.getDbOpType()) || DbOpType.UPDATE.getCode().equals(e.getDbOpType())))
.collect(Collectors.toList()); .collect(Collectors.toList());
// 防重复保存:对所有医嘱进行去重(包括 INSERT 和 UPDATE 混合场景),避免签发单条医嘱时产生重复记录 // 防重复保存:对新增医嘱进行去重,避免签发单条长期医嘱时产生重复记录
Set<String> longUniqueKeySet = new HashSet<>(); Set<String> longUniqueKeySet = new HashSet<>();
List<RegAdviceSaveDto> longUniqueList = new ArrayList<>(); List<RegAdviceSaveDto> longUniqueList = new ArrayList<>();
for (RegAdviceSaveDto adviceSaveDto : longInsertOrUpdateList) { for (RegAdviceSaveDto adviceSaveDto : longInsertOrUpdateList) {
@@ -351,10 +351,10 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
+ adviceSaveDto.getDose() + "_" + adviceSaveDto.getDose() + "_"
+ adviceSaveDto.getMethodCode() + "_" + adviceSaveDto.getMethodCode() + "_"
+ adviceSaveDto.getRateCode(); + adviceSaveDto.getRateCode();
if (longUniqueKeySet.contains(uniqueKey)) { if (DbOpType.INSERT.getCode().equals(adviceSaveDto.getDbOpType()) && longUniqueKeySet.contains(uniqueKey)) {
log.warn("防重复保存:检测到重复长期医嘱(跨操作类型),跳过 - patientId={}, encounterId={}, adviceDefinitionId={}, dbOpType={}", log.warn("防重复保存:检测到重复长期医嘱,跳过保存 - patientId={}, encounterId={}, adviceDefinitionId={}, dose={}",
adviceSaveDto.getPatientId(), adviceSaveDto.getEncounterId(), adviceSaveDto.getPatientId(), adviceSaveDto.getEncounterId(),
adviceSaveDto.getAdviceDefinitionId(), adviceSaveDto.getDbOpType()); adviceSaveDto.getAdviceDefinitionId(), adviceSaveDto.getDose());
continue; continue;
} }
longUniqueKeySet.add(uniqueKey); longUniqueKeySet.add(uniqueKey);
@@ -429,7 +429,7 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
&& (DbOpType.INSERT.getCode().equals(e.getDbOpType()) || DbOpType.UPDATE.getCode().equals(e.getDbOpType()))) && (DbOpType.INSERT.getCode().equals(e.getDbOpType()) || DbOpType.UPDATE.getCode().equals(e.getDbOpType())))
.collect(Collectors.toList()); .collect(Collectors.toList());
// 防重复保存:对所有医嘱进行去重(包括 INSERT 和 UPDATE 混合场景),避免签发时产生重复记录 // 防重复保存:对新增医嘱进行去重
Set<String> tempUniqueKeySet = new HashSet<>(); Set<String> tempUniqueKeySet = new HashSet<>();
List<RegAdviceSaveDto> tempUniqueList = new ArrayList<>(); List<RegAdviceSaveDto> tempUniqueList = new ArrayList<>();
for (RegAdviceSaveDto adviceSaveDto : tempInsertOrUpdateList) { for (RegAdviceSaveDto adviceSaveDto : tempInsertOrUpdateList) {
@@ -439,10 +439,10 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
+ adviceSaveDto.getDose() + "_" + adviceSaveDto.getDose() + "_"
+ adviceSaveDto.getMethodCode() + "_" + adviceSaveDto.getMethodCode() + "_"
+ adviceSaveDto.getRateCode(); + adviceSaveDto.getRateCode();
if (tempUniqueKeySet.contains(uniqueKey)) { if (DbOpType.INSERT.getCode().equals(adviceSaveDto.getDbOpType()) && tempUniqueKeySet.contains(uniqueKey)) {
log.warn("防重复保存:检测到重复临时医嘱(跨操作类型),跳过 - patientId={}, encounterId={}, adviceDefinitionId={}, dbOpType={}", log.warn("防重复保存:检测到重复临时医嘱,跳过保存 - patientId={}, encounterId={}, adviceDefinitionId={}, dose={}",
adviceSaveDto.getPatientId(), adviceSaveDto.getEncounterId(), adviceSaveDto.getPatientId(), adviceSaveDto.getEncounterId(),
adviceSaveDto.getAdviceDefinitionId(), adviceSaveDto.getDbOpType()); adviceSaveDto.getAdviceDefinitionId(), adviceSaveDto.getDose());
continue; continue;
} }
tempUniqueKeySet.add(uniqueKey); tempUniqueKeySet.add(uniqueKey);
@@ -710,21 +710,11 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
// 批量更新诊疗医嘱状态(使用 update 确保状态字段必定更新) // 批量更新诊疗医嘱状态(使用 update 确保状态字段必定更新)
if (!processedRequestIds.isEmpty()) { if (!processedRequestIds.isEmpty()) {
// 🔧 Bug #487 修复:签发时额外设置 authoredTime确保签发时间被记录 iServiceRequestService.update(null,
if (is_sign) { new LambdaUpdateWrapper<ServiceRequest>()
iServiceRequestService.update(null, .set(ServiceRequest::getStatusEnum,
new LambdaUpdateWrapper<ServiceRequest>() is_save ? RequestStatus.DRAFT.getValue() : RequestStatus.ACTIVE.getValue())
.set(ServiceRequest::getStatusEnum, RequestStatus.ACTIVE.getValue()) .in(ServiceRequest::getId, processedRequestIds));
.set(ServiceRequest::getAuthoredTime, authoredTime)
.set(ServiceRequest::getSignCode, signCode)
.in(ServiceRequest::getId, processedRequestIds));
log.info("签发诊疗医嘱成功requestIds: {}, signCode: {}", processedRequestIds, signCode);
} else {
iServiceRequestService.update(null,
new LambdaUpdateWrapper<ServiceRequest>()
.set(ServiceRequest::getStatusEnum, RequestStatus.DRAFT.getValue())
.in(ServiceRequest::getId, processedRequestIds));
}
} }
} }
@@ -761,8 +751,6 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
if (is_sign) { if (is_sign) {
deviceRequest.setReqAuthoredTime(authoredTime); // 医嘱签发时间 deviceRequest.setReqAuthoredTime(authoredTime); // 医嘱签发时间
} }
// 保存或签发时都需要设置耗材定义ID防止 sign 分支 deviceDefId 为空触发 NOT NULL 约束)
deviceRequest.setDeviceDefId(regAdviceSaveDto.getAdviceDefinitionId());
// 保存时处理的字段属性 // 保存时处理的字段属性
if (is_save) { if (is_save) {
deviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.DEVICE_RES_NO.getPrefix(), 4)); deviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.DEVICE_RES_NO.getPrefix(), 4));
@@ -800,8 +788,6 @@ public class AdviceManageAppServiceImpl implements IAdviceManageAppService {
if (is_sign) { if (is_sign) {
deviceRequest.setReqAuthoredTime(authoredTime); // 医嘱签发时间 deviceRequest.setReqAuthoredTime(authoredTime); // 医嘱签发时间
} }
// 保存或签发时都需要设置耗材定义ID防止 sign 分支 deviceDefId 为空触发 NOT NULL 约束)
deviceRequest.setDeviceDefId(regAdviceSaveDto.getAdviceDefinitionId());
// 保存时处理的字段属性 // 保存时处理的字段属性
if (is_save) { if (is_save) {
deviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.DEVICE_RES_NO.getPrefix(), 4)); deviceRequest.setBusNo(assignSeqUtil.getSeqByDay(AssignSeqEnum.DEVICE_RES_NO.getPrefix(), 4));

View File

@@ -1,7 +1,6 @@
package com.openhis.web.regdoctorstation.appservice.impl; package com.openhis.web.regdoctorstation.appservice.impl;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
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;
@@ -77,10 +76,6 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
@Override @Override
@Transactional(rollbackFor = Exception.class) @Transactional(rollbackFor = Exception.class)
public R<?> saveRequestForm(RequestFormSaveDto requestFormSaveDto, String typeCode) { public R<?> saveRequestForm(RequestFormSaveDto requestFormSaveDto, String typeCode) {
// 申请单ID前端空字符串可能反序列化为0L需同时判0
Long requestFormId = requestFormSaveDto.getRequestFormId();
boolean isEdit = requestFormId != null && requestFormId != 0L;
// 诊疗执行科室配置校验(必须在任何数据库操作之前) // 诊疗执行科室配置校验(必须在任何数据库操作之前)
List<ActivityOrganizationConfigDto> activityOrganizationConfig = List<ActivityOrganizationConfigDto> activityOrganizationConfig =
requestFormManageAppMapper.getActivityOrganizationConfig(typeCode); requestFormManageAppMapper.getActivityOrganizationConfig(typeCode);
@@ -88,27 +83,12 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
throw new ServiceException("请先配置当前时间段的执行科室"); throw new ServiceException("请先配置当前时间段的执行科室");
} }
// 逐个校验activityList中的项目是否都配置了执行科室并收集positionId供后续使用
// 必须在任何数据库操作之前完成全部校验,避免部分保存后异常导致脏数据
List<ActivitySaveDto> activityList = requestFormSaveDto.getActivityList();
// 缓存校验结果,避免主循环中重复查询和可能出现的数据不一致
java.util.Map<Long, Long> activityIdToPositionIdMap = new java.util.HashMap<>();
if (activityList != null && !activityList.isEmpty()) {
for (ActivitySaveDto activitySaveDto : activityList) {
Long positionId = activityOrganizationConfig.stream()
.filter(dto -> activitySaveDto.getAdviceDefinitionId().equals(dto.getActivityDefinitionId()))
.map(ActivityOrganizationConfigDto::getOrganizationId).findFirst().orElse(null);
if (positionId == null) {
throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室");
}
activityIdToPositionIdMap.put(activitySaveDto.getAdviceDefinitionId(), positionId);
}
}
// 诊疗处方号 // 诊疗处方号
String prescriptionNo; String prescriptionNo;
// 申请单ID
Long requestFormId = requestFormSaveDto.getRequestFormId();
// 编辑场景 // 编辑场景
if (isEdit) { if (requestFormId != null) {
RequestForm requestFormInfo = iRequestFormService.getById(requestFormId); RequestForm requestFormInfo = iRequestFormService.getById(requestFormId);
prescriptionNo = requestFormInfo.getPrescriptionNo(); prescriptionNo = requestFormInfo.getPrescriptionNo();
// 该申请单存在的待发送医嘱个数 // 该申请单存在的待发送医嘱个数
@@ -152,7 +132,7 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
iRequestFormService.saveOrUpdate(requestForm); iRequestFormService.saveOrUpdate(requestForm);
// 编辑场景时,先删除掉原有诊疗项目及账单再新增 // 编辑场景时,先删除掉原有诊疗项目及账单再新增
if (isEdit) { if (requestFormId != null) {
List<Long> serviceRequestIds = iServiceRequestService List<Long> serviceRequestIds = iServiceRequestService
.list(new LambdaQueryWrapper<ServiceRequest>().eq(ServiceRequest::getPrescriptionNo, prescriptionNo)) .list(new LambdaQueryWrapper<ServiceRequest>().eq(ServiceRequest::getPrescriptionNo, prescriptionNo))
.stream().map(ServiceRequest::getId).collect(Collectors.toList()); .stream().map(ServiceRequest::getId).collect(Collectors.toList());
@@ -166,6 +146,8 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
ServiceRequest serviceRequest; ServiceRequest serviceRequest;
ChargeItem chargeItem; ChargeItem chargeItem;
// 诊疗集合
List<ActivitySaveDto> activityList = requestFormSaveDto.getActivityList();
log.info("保存申请单typeCode={}, activityListSize={}, encounterId={}", typeCode, activityList != null ? activityList.size() : 0, encounterId); log.info("保存申请单typeCode={}, activityListSize={}, encounterId={}", typeCode, activityList != null ? activityList.size() : 0, encounterId);
for (ActivitySaveDto activitySaveDto : activityList) { for (ActivitySaveDto activitySaveDto : activityList) {
@@ -184,7 +166,9 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
serviceRequest.setEncounterId(encounterId); // 就诊id serviceRequest.setEncounterId(encounterId); // 就诊id
serviceRequest.setAuthoredTime(curDate); // 请求签发时间 serviceRequest.setAuthoredTime(curDate); // 请求签发时间
Long positionId = activityIdToPositionIdMap.get(activitySaveDto.getAdviceDefinitionId()); Long positionId = activityOrganizationConfig.stream()
.filter(dto -> activitySaveDto.getAdviceDefinitionId().equals(dto.getActivityDefinitionId()))
.map(ActivityOrganizationConfigDto::getOrganizationId).findFirst().orElse(null);
if (positionId == null) { if (positionId == null) {
throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室"); throw new ServiceException(activitySaveDto.getAdviceDefinitionName() + "未配置当前时间段的执行科室");
} }
@@ -295,7 +279,7 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
surgeryServiceRequest.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue()); surgeryServiceRequest.setTherapyEnum(TherapyTimeType.TEMPORARY.getValue());
surgeryServiceRequest.setQuantity(BigDecimal.valueOf(1)); surgeryServiceRequest.setQuantity(BigDecimal.valueOf(1));
surgeryServiceRequest.setUnitCode(""); surgeryServiceRequest.setUnitCode("");
surgeryServiceRequest.setCategoryEnum(24); // 24-手术(新值域,避开 adviceType 碰撞) surgeryServiceRequest.setCategoryEnum(4); // 4-手术
// 优先从 activityList 获取手术 ID // 优先从 activityList 获取手术 ID
if (activityList != null && !activityList.isEmpty()) { if (activityList != null && !activityList.isEmpty()) {
Long activityId = activityList.get(0).getAdviceDefinitionId(); Long activityId = activityList.get(0).getAdviceDefinitionId();
@@ -491,90 +475,4 @@ public class RequestFormManageAppServiceImpl implements IRequestFormManageAppSer
return requestFormManageAppMapper.getRequestFormPage(requestFormDto, page); return requestFormManageAppMapper.getRequestFormPage(requestFormDto, page);
} }
@Override
public R<?> deleteRequestForm(Long requestFormId) {
if (requestFormId == null) {
return R.fail("申请单ID不能为空");
}
RequestForm requestForm = iRequestFormService.getById(requestFormId);
if (requestForm == null) {
return R.fail("申请单不存在");
}
String prescriptionNo = requestForm.getPrescriptionNo();
// 查询该申请单下所有 ServiceRequest含子项
List<ServiceRequest> serviceRequests = iServiceRequestService.list(
new LambdaQueryWrapper<ServiceRequest>()
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo));
if (serviceRequests == null || serviceRequests.isEmpty()) {
return R.fail("未找到关联的诊疗医嘱");
}
// 校验:只有待签发(status=0)的申请单可删除
boolean allDraft = serviceRequests.stream()
.allMatch(sr -> RequestStatus.DRAFT.getValue().equals(sr.getStatusEnum()));
if (!allDraft) {
return R.fail("只有待签发状态的申请单可删除");
}
List<Long> serviceRequestIds = serviceRequests.stream()
.map(ServiceRequest::getId).collect(Collectors.toList());
// 1. 删除关联的费用项
for (Long srId : serviceRequestIds) {
iChargeItemService.deleteByServiceTableAndId(
CommonConstants.TableName.WOR_SERVICE_REQUEST, srId);
}
// 2. 删除子项 ServiceRequestparentId 非空)
iServiceRequestService.remove(
new LambdaQueryWrapper<ServiceRequest>()
.in(ServiceRequest::getId, serviceRequestIds)
.isNotNull(ServiceRequest::getParentId));
// 3. 删除主项 ServiceRequest
iServiceRequestService.removeByIds(serviceRequestIds);
// 4. 删除申请单
iRequestFormService.removeById(requestFormId);
log.info("检查申请单删除成功requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
return R.ok("删除成功");
}
@Override
public R<?> withdrawRequestForm(Long requestFormId) {
if (requestFormId == null) {
return R.fail("申请单ID不能为空");
}
RequestForm requestForm = iRequestFormService.getById(requestFormId);
if (requestForm == null) {
return R.fail("申请单不存在");
}
String prescriptionNo = requestForm.getPrescriptionNo();
// 查询该申请单下所有 ServiceRequest
List<ServiceRequest> serviceRequests = iServiceRequestService.list(
new LambdaQueryWrapper<ServiceRequest>()
.eq(ServiceRequest::getPrescriptionNo, prescriptionNo));
if (serviceRequests == null || serviceRequests.isEmpty()) {
return R.fail("未找到关联的诊疗医嘱");
}
// 校验:只有已签发(status=2)的申请单可撤回
boolean allActive = serviceRequests.stream()
.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());
iServiceRequestService.update(
new ServiceRequest().setStatusEnum(RequestStatus.DRAFT.getValue()),
new LambdaUpdateWrapper<ServiceRequest>()
.in(ServiceRequest::getId, serviceRequestIds));
log.info("检查申请单撤回成功requestFormId={}, prescriptionNo={}", requestFormId, prescriptionNo);
return R.ok("撤回成功");
}
} }

View File

@@ -18,7 +18,6 @@ import org.springframework.web.bind.annotation.*;
import java.time.LocalDate; import java.time.LocalDate;
import java.util.List; import java.util.List;
import java.util.Map;
/** /**
* 申请单管理 controller * 申请单管理 controller
@@ -83,23 +82,14 @@ public class RequestFormManageController {
* 查询检查申请单 * 查询检查申请单
* *
* @param encounterId 就诊id * @param encounterId 就诊id
* @param startDate 开始日期可选格式yyyy-MM-dd
* @param endDate 结束日期可选格式yyyy-MM-dd
* @param status 单据状态(可选)
* @param keyword 关键字(可选,申请单号/检查项目名称模糊匹配)
* @return 检查申请单 * @return 检查申请单
*/ */
@GetMapping(value = "/get-check") @GetMapping(value = "/get-check")
public R<?> getCheckRequestForm( public R<?> getCheckRequestForm(@RequestParam(required = false) Long encounterId) {
@RequestParam(required = false) Long encounterId,
@RequestParam(required = false) String startDate,
@RequestParam(required = false) String endDate,
@RequestParam(required = false) String status,
@RequestParam(required = false) String keyword) {
if (encounterId == null) { if (encounterId == null) {
return R.fail("就诊ID不能为空"); return R.fail("就诊ID不能为空");
} }
return R.ok(iRequestFormManageAppService.getRequestForm(encounterId, ActivityDefCategory.TEST.getCode(), startDate, endDate, status, keyword)); return R.ok(iRequestFormManageAppService.getRequestForm(encounterId, ActivityDefCategory.TEST.getCode()));
} }
/** /**
@@ -186,26 +176,4 @@ public class RequestFormManageController {
public R<IPage<RequestFormPageDto>> getRequestFormPage(@RequestBody RequestFormDto requestFormDto) { public R<IPage<RequestFormPageDto>> getRequestFormPage(@RequestBody RequestFormDto requestFormDto) {
return R.ok(iRequestFormManageAppService.getRequestFormPage(requestFormDto)); return R.ok(iRequestFormManageAppService.getRequestFormPage(requestFormDto));
} }
/**
* 删除申请单(仅待签发状态可删除)
*
* @param data 包含 requestFormId 的请求体
* @return 结果
*/
@PostMapping(value = "/delete")
public R<?> deleteRequestForm(@RequestBody Map<String, Long> data) {
return iRequestFormManageAppService.deleteRequestForm(data.get("requestFormId"));
}
/**
* 撤回申请单(已签发状态撤回至待签发)
*
* @param data 包含 requestFormId 的请求体
* @return 结果
*/
@PostMapping(value = "/withdraw")
public R<?> withdrawRequestForm(@RequestBody Map<String, Long> data) {
return iRequestFormManageAppService.withdrawRequestForm(data.get("requestFormId"));
}
} }

View File

@@ -138,12 +138,4 @@ public class CurrentDayEncounterTencentDto {
*/ */
private String englishName; private String englishName;
/** 号源池ID用于分诊队列 div_log 审计日志) */
@JsonSerialize(using = ToStringSerializer.class)
private Long poolId;
/** 号源槽位ID用于分诊队列 div_log 审计日志) */
@JsonSerialize(using = ToStringSerializer.class)
private Long slotId;
} }

View File

@@ -97,10 +97,6 @@
CASE CASE
WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.surgery_name WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.surgery_name
WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.surgery_name WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.surgery_name
WHEN T1.context_enum = 6 AND T1.service_table = 'wor_service_request' THEN COALESCE(
wsr.content_json::json->>'surgeryName',
wsr.content_json::json->>'adviceName',
T9sr.surgery_name)
WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name") WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name")
WHEN T1.context_enum = 6 THEN T2."name" WHEN T1.context_enum = 6 THEN T2."name"
WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name") WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name")
@@ -112,7 +108,6 @@
CASE CASE
WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN NULL WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN NULL
WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN NULL WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN NULL
WHEN T1.context_enum = 6 AND T1.service_table = 'wor_service_request' THEN NULL
WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL
WHEN T1.context_enum = 6 THEN T2.yb_no WHEN T1.context_enum = 6 THEN T2.yb_no
WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL
@@ -123,7 +118,6 @@
CASE CASE
WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.id WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.id
WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.id WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.id
WHEN T1.context_enum = 6 AND T1.service_table = 'wor_service_request' THEN COALESCE(T9sr.id, wsr.activity_id)
WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0 WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0
WHEN T1.context_enum = 6 THEN T2.id WHEN T1.context_enum = 6 THEN T2.id
WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0 WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0
@@ -165,11 +159,6 @@
LEFT JOIN med_medication_request AS mmr ON mmr.id = T1.service_id AND mmr.delete_flag = '0' LEFT JOIN med_medication_request AS mmr ON mmr.id = T1.service_id AND mmr.delete_flag = '0'
LEFT JOIN wor_device_request AS wdr ON wdr.id = T1.service_id AND wdr.delete_flag = '0' LEFT JOIN wor_device_request AS wdr ON wdr.id = T1.service_id AND wdr.delete_flag = '0'
LEFT JOIN wor_service_request AS wsr ON wsr.id = T1.service_id AND wsr.delete_flag = '0' LEFT JOIN wor_service_request AS wsr ON wsr.id = T1.service_id AND wsr.delete_flag = '0'
LEFT JOIN cli_surgery AS T9sr ON T1.context_enum = 6
AND T1.service_table = 'wor_service_request'
AND wsr.activity_id IS NOT NULL
AND wsr.activity_id = T9sr.id
AND T9sr.delete_flag = '0'
LEFT JOIN wor_service_request AS wsrp ON wsrp.id = wsr.parent_id AND wsrp.delete_flag = '0' LEFT JOIN wor_service_request AS wsrp ON wsrp.id = wsr.parent_id AND wsrp.delete_flag = '0'
WHERE T1.encounter_id = #{encounterId} WHERE T1.encounter_id = #{encounterId}
AND T1.status_enum IN (0 AND T1.status_enum IN (0
@@ -234,10 +223,6 @@
CASE CASE
WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.surgery_name WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.surgery_name
WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.surgery_name WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.surgery_name
WHEN T1.context_enum = 6 AND T1.service_table = 'wor_service_request' THEN COALESCE(
wsr.content_json::json->>'surgeryName',
wsr.content_json::json->>'adviceName',
T9sr.surgery_name)
WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name") WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name")
WHEN T1.context_enum = 6 THEN T2."name" WHEN T1.context_enum = 6 THEN T2."name"
WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name") WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN COALESCE(wsr.content_json::json->>'adviceName', T2."name")
@@ -249,7 +234,6 @@
CASE CASE
WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN NULL WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN NULL
WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN NULL WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN NULL
WHEN T1.context_enum = 6 AND T1.service_table = 'wor_service_request' THEN NULL
WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL
WHEN T1.context_enum = 6 THEN T2.yb_no WHEN T1.context_enum = 6 THEN T2.yb_no
WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN NULL
@@ -260,7 +244,6 @@
CASE CASE
WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.id WHEN T1.context_enum = #{activity} AND T1.product_table = 'cli_surgery' THEN T9.id
WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.id WHEN T1.context_enum = 6 AND T1.product_table = 'cli_surgery' THEN T9.id
WHEN T1.context_enum = 6 AND T1.service_table = 'wor_service_request' THEN COALESCE(T9sr.id, wsr.activity_id)
WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0 WHEN T1.context_enum = 6 AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0
WHEN T1.context_enum = 6 THEN T2.id WHEN T1.context_enum = 6 THEN T2.id
WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0 WHEN T1.context_enum = #{activity} AND T1.product_id = 0 AND T1.service_table = 'wor_service_request' THEN 0
@@ -303,11 +286,6 @@
LEFT JOIN med_medication_request AS mmr ON mmr.id = T1.service_id AND mmr.delete_flag = '0' LEFT JOIN med_medication_request AS mmr ON mmr.id = T1.service_id AND mmr.delete_flag = '0'
LEFT JOIN wor_device_request AS wdr ON wdr.id = T1.service_id AND wdr.delete_flag = '0' LEFT JOIN wor_device_request AS wdr ON wdr.id = T1.service_id AND wdr.delete_flag = '0'
LEFT JOIN wor_service_request AS wsr ON wsr.id = T1.service_id AND wsr.delete_flag = '0' LEFT JOIN wor_service_request AS wsr ON wsr.id = T1.service_id AND wsr.delete_flag = '0'
LEFT JOIN cli_surgery AS T9sr ON T1.context_enum = 6
AND T1.service_table = 'wor_service_request'
AND wsr.activity_id IS NOT NULL
AND wsr.activity_id = T9sr.id
AND T9sr.delete_flag = '0'
WHERE T1.encounter_id = #{encounterId} WHERE T1.encounter_id = #{encounterId}
AND T1.status_enum IN (0 AND T1.status_enum IN (0
, #{planned} , #{planned}

View File

@@ -71,7 +71,7 @@
</if> </if>
AND os.delete_flag = '0' AND os.delete_flag = '0'
</where> </where>
ORDER BY os.create_time DESC, os.schedule_id DESC ORDER BY os.create_time DESC
</select> </select>
<!-- 根据ID查询手术安排详情--> <!-- 根据ID查询手术安排详情-->
<select id="getSurgeryScheduleDetail" resultType="com.openhis.web.clinicalmanage.dto.OpScheduleDto"> <select id="getSurgeryScheduleDetail" resultType="com.openhis.web.clinicalmanage.dto.OpScheduleDto">
@@ -89,14 +89,15 @@
cs.apply_doctor_name AS apply_doctor_name, cs.apply_doctor_name AS apply_doctor_name,
drf.create_time AS apply_time, drf.create_time AS apply_time,
os.surgery_nature AS surgeryType, os.surgery_nature AS surgeryType,
cs.incision_level AS incisionLevel,
fc.contract_name AS feeType, fc.contract_name AS feeType,
os.fee_type AS feeType,
COALESCE(pi.identifier_no, ap.bus_no, '') AS identifierNo COALESCE(pi.identifier_no, ap.bus_no, '') AS identifierNo
FROM op_schedule os FROM op_schedule os
LEFT JOIN adm_patient ap ON os.patient_id = ap.id LEFT JOIN adm_patient ap ON os.patient_id = ap.id
INNER JOIN cli_surgery cs ON os.oper_code = cs.surgery_no AND cs.delete_flag = '0' INNER JOIN cli_surgery cs ON os.oper_code = cs.surgery_no AND cs.delete_flag = '0'
LEFT JOIN adm_organization o ON cs.org_id = o.id LEFT JOIN adm_organization o ON cs.org_id = o.id
LEFT JOIN adm_encounter ae ON ae.id = cs.encounter_id AND ae.delete_flag = '0'
LEFT JOIN adm_account aa ON aa.encounter_id = ae.id AND aa.delete_flag = '0'
LEFT JOIN fin_contract fc ON fc.bus_no = aa.contract_no AND fc.delete_flag = '0'
LEFT JOIN doc_request_form drf ON drf.prescription_no=cs.surgery_no LEFT JOIN doc_request_form drf ON drf.prescription_no=cs.surgery_no
LEFT JOIN ( LEFT JOIN (
SELECT patient_id, identifier_no SELECT patient_id, identifier_no
@@ -185,7 +186,7 @@
<if test="dto.applyDeptId != null and dto.applyDeptId != ''"> AND cs.apply_dept_id = #{dto.applyDeptId}</if> <if test="dto.applyDeptId != null and dto.applyDeptId != ''"> AND cs.apply_dept_id = #{dto.applyDeptId}</if>
<if test="dto.patientName != null and dto.patientName != ''"> AND ap.name LIKE CONCAT('%', #{dto.patientName}, '%')</if> <if test="dto.patientName != null and dto.patientName != ''"> AND ap.name LIKE CONCAT('%', #{dto.patientName}, '%')</if>
</where> </where>
ORDER BY os.create_time DESC, os.schedule_id DESC ORDER BY os.create_time DESC
</select> </select>
<!-- 查询时间段内该手术室是否被占用--> <!-- 查询时间段内该手术室是否被占用-->
<select id="isScheduleConflict" resultType="java.lang.Boolean"> <select id="isScheduleConflict" resultType="java.lang.Boolean">

View File

@@ -239,7 +239,7 @@
NULL AS activity_type_dictText, NULL AS activity_type_dictText,
-- 前端"包装单位"列显示使用单位permitted_unit_code -- 前端"包装单位"列显示使用单位permitted_unit_code
T1.permitted_unit_code AS unit_code, T1.permitted_unit_code AS unit_code,
T1.permitted_unit_code AS min_unit_code, '' AS min_unit_code,
'' AS volume, '' AS volume,
'' AS method_code, '' AS method_code,
'' AS rate_code, '' AS rate_code,
@@ -539,8 +539,7 @@
AND T1.refund_medicine_id IS NULL AND T1.refund_medicine_id IS NULL
ORDER BY T1.status_enum,T1.sort_number) ORDER BY T1.status_enum,T1.sort_number)
UNION ALL UNION ALL
-- 🔧 查询仅存在于 adm_charge_item 的"孤儿"耗材数据DeviceRequest 缺失或 generate_source_enum 未设置 -- 🔧 新增:查询门诊术中计费生成的耗材数据(这些数据存在于 adm_charge_item 和 wor_device_request
-- 正常 DeviceRequestgenerate_source_enum 已赋值)由下方 Part 3 统一负责,此处不做重复覆盖避免 UNION ALL 重复行
(SELECT 2 AS advice_type, (SELECT 2 AS advice_type,
CI.service_id AS request_id, CI.service_id AS request_id,
CI.service_id || '-ci-dev' AS unique_key, CI.service_id || '-ci-dev' AS unique_key,
@@ -584,9 +583,6 @@
LEFT JOIN adm_location AS AL ON AL.id = DR.perform_location AND AL.delete_flag = '0' LEFT JOIN adm_location AS AL ON AL.id = DR.perform_location AND AL.delete_flag = '0'
WHERE CI.delete_flag = '0' WHERE CI.delete_flag = '0'
AND CI.service_table = 'wor_device_request' AND CI.service_table = 'wor_device_request'
<if test="generateSourceEnum != null">
AND DR.generate_source_enum IS NULL <!-- 仅匹配孤儿记录normal DeviceRequest 由 Part 3 负责,避免 UNION ALL 重复 -->
</if>
<if test="historyFlag == '0'.toString()"> <if test="historyFlag == '0'.toString()">
AND CI.encounter_id = #{encounterId} AND CI.encounter_id = #{encounterId}
</if> </if>

View File

@@ -264,6 +264,7 @@
WHERE ael.status_enum = #{active} WHERE ael.status_enum = #{active}
AND ael.delete_flag = '0' AND ael.delete_flag = '0'
AND ael.form_enum = #{bed} AND ael.form_enum = #{bed}
LIMIT 1
) AS bed ON bed.encounter_id = ae.id ) AS bed ON bed.encounter_id = ae.id
LEFT JOIN ( LEFT JOIN (
SELECT ael.encounter_id, SELECT ael.encounter_id,
@@ -274,6 +275,7 @@
WHERE ael.status_enum = #{active} WHERE ael.status_enum = #{active}
AND ael.delete_flag = '0' AND ael.delete_flag = '0'
AND ael.form_enum = #{house} AND ael.form_enum = #{house}
LIMIT 1
) AS house ON house.encounter_id = ae.id ) AS house ON house.encounter_id = ae.id
LEFT JOIN ( LEFT JOIN (
SELECT ael.encounter_id, SELECT ael.encounter_id,
@@ -284,6 +286,7 @@
WHERE ael.status_enum = #{active} WHERE ael.status_enum = #{active}
AND ael.delete_flag = '0' AND ael.delete_flag = '0'
AND ael.form_enum = #{ward} AND ael.form_enum = #{ward}
LIMIT 1
) AS ward ON ward.encounter_id = ae.id ) AS ward ON ward.encounter_id = ae.id
LEFT JOIN ( LEFT JOIN (
SELECT aep.encounter_id, SELECT aep.encounter_id,

View File

@@ -155,7 +155,7 @@
ii.performer_check_id, ii.performer_check_id,
ii.category_code, ii.category_code,
ii.dispense_status ii.dispense_status
FROM (( SELECT DISTINCT T1.encounter_id, FROM (( SELECT T1.encounter_id,
T1.tenant_id, T1.tenant_id,
#{medMedicationRequest} AS advice_table, #{medMedicationRequest} AS advice_table,
T1.id AS request_id, T1.id AS request_id,
@@ -280,13 +280,9 @@
aa.balance_amount aa.balance_amount
) AS personal_account ) AS personal_account
ON personal_account.encounter_id = ae.id ON personal_account.encounter_id = ae.id
LEFT JOIN LATERAL ( LEFT JOIN med_medication_dispense mmd
SELECT status_enum ON mmd.med_req_id = T1.id
FROM med_medication_dispense AND mmd.delete_flag = '0'
WHERE med_req_id = T1.id AND delete_flag = '0'
ORDER BY create_time DESC
LIMIT 1
) mmd ON true
WHERE T1.delete_flag = '0' WHERE T1.delete_flag = '0'
AND T1.refund_medicine_id IS NULL AND T1.refund_medicine_id IS NULL
AND T1.generate_source_enum = #{doctorPrescription} AND T1.generate_source_enum = #{doctorPrescription}
@@ -297,7 +293,7 @@
T1.sort_number, T1.sort_number,
T1.group_id ) T1.group_id )
UNION UNION
( SELECT DISTINCT T1.encounter_id, ( SELECT T1.encounter_id,
T1.tenant_id, T1.tenant_id,
#{worServiceRequest} AS advice_table, #{worServiceRequest} AS advice_table,
T1.id AS request_id, T1.id AS request_id,

View File

@@ -288,7 +288,7 @@
AND T1.refund_device_id IS NULL AND T1.refund_device_id IS NULL
ORDER BY T1.status_enum) ORDER BY T1.status_enum)
UNION ALL UNION ALL
(SELECT CASE WHEN T1.category_enum IN (4, 24) THEN 6 ELSE 3 END AS advice_type, (SELECT CASE WHEN T1.category_enum = 4 THEN 6 ELSE 3 END AS advice_type,
T1.id AS request_id, T1.id AS request_id,
T1.id || '-3' AS unique_key, T1.id || '-3' AS unique_key,
T1.requester_id AS requester_id, T1.requester_id AS requester_id,

View File

@@ -8,27 +8,21 @@
SELECT drf.id AS request_form_id, SELECT drf.id AS request_form_id,
drf.encounter_id, drf.encounter_id,
drf.prescription_no, drf.prescription_no,
COALESCE( drf.NAME,
(SELECT STRING_AGG(DISTINCT wad.name, '、')
FROM wor_service_request wsr2
LEFT JOIN wor_activity_definition wad ON wad.id = wsr2.activity_id AND wad.delete_flag = '0'
WHERE wsr2.prescription_no = drf.prescription_no AND wsr2.delete_flag = '0'),
drf.name
) AS name,
drf.desc_json, drf.desc_json,
drf.requester_id, drf.requester_id,
drf.create_time, drf.create_time,
ap.NAME AS patient_name, ap.NAME AS patient_name,
CASE CASE MIN(wsr.status_enum)
WHEN MIN(wsr.status_enum) = 1 THEN 0 WHEN 1 THEN 0
WHEN MIN(wsr.status_enum) = 2 THEN 1 WHEN 2 THEN 1
WHEN MIN(wsr.status_enum) = 3 AND MAX(CASE WHEN wsr.performer_check_id IS NOT NULL THEN 1 ELSE 0 END) = 1 THEN 2 WHEN 3 THEN 4
WHEN MIN(wsr.status_enum) = 3 THEN 4 WHEN 4 THEN 4
WHEN MIN(wsr.status_enum) = 4 THEN 3 WHEN 5 THEN 5
WHEN MIN(wsr.status_enum) = 5 OR MIN(wsr.status_enum) = 6 OR MIN(wsr.status_enum) = 7 THEN 7 WHEN 6 THEN 5
WHEN MIN(wsr.status_enum) = 8 THEN 6 WHEN 7 THEN 5
ELSE NULL ELSE NULL
END AS status END AS status
FROM doc_request_form AS drf FROM doc_request_form AS drf
LEFT JOIN adm_encounter AS ae ON ae.ID = drf.encounter_id LEFT JOIN adm_encounter AS ae ON ae.ID = drf.encounter_id
AND ae.delete_flag = '0' AND ae.delete_flag = '0'
@@ -46,16 +40,16 @@
AND drf.create_time &lt;= (#{endDate}::date + INTERVAL '1 day' - INTERVAL '1 second') AND drf.create_time &lt;= (#{endDate}::date + INTERVAL '1 day' - INTERVAL '1 second')
</if> </if>
<if test="status != null and status != ''"> <if test="status != null and status != ''">
AND CASE AND CASE MIN(wsr.status_enum)
WHEN MIN(wsr.status_enum) = 1 THEN 0 WHEN 1 THEN 0
WHEN MIN(wsr.status_enum) = 2 THEN 1 WHEN 2 THEN 1
WHEN MIN(wsr.status_enum) = 3 AND MAX(CASE WHEN wsr.performer_check_id IS NOT NULL THEN 1 ELSE 0 END) = 1 THEN 2 WHEN 3 THEN 4
WHEN MIN(wsr.status_enum) = 3 THEN 4 WHEN 4 THEN 4
WHEN MIN(wsr.status_enum) = 4 THEN 3 WHEN 5 THEN 5
WHEN MIN(wsr.status_enum) = 5 OR MIN(wsr.status_enum) = 6 OR MIN(wsr.status_enum) = 7 THEN 7 WHEN 6 THEN 5
WHEN MIN(wsr.status_enum) = 8 THEN 6 WHEN 7 THEN 5
ELSE NULL ELSE NULL
END = #{status}::integer END = #{status}::integer
</if> </if>
<if test="keyword != null and keyword != ''"> <if test="keyword != null and keyword != ''">
AND (drf.prescription_no ILIKE '%' || #{keyword} || '%' AND (drf.prescription_no ILIKE '%' || #{keyword} || '%'
@@ -168,7 +162,7 @@
AND drf.prescription_no LIKE CONCAT('%', #{requestFormDto.surgeryNo}, '%') AND drf.prescription_no LIKE CONCAT('%', #{requestFormDto.surgeryNo}, '%')
</if> </if>
<if test="requestFormDto.typeCode != null and requestFormDto.typeCode != ''"> <if test="requestFormDto.typeCode != null and requestFormDto.typeCode != ''">
AND drf.type_code IN (#{requestFormDto.typeCode}, 'SURGERY') AND drf.type_code = #{requestFormDto.typeCode}
</if> </if>
<if test="requestFormDto.applyTimeStart != null"> <if test="requestFormDto.applyTimeStart != null">
AND drf.create_time >= #{requestFormDto.applyTimeStart} AND drf.create_time >= #{requestFormDto.applyTimeStart}

View File

@@ -27,9 +27,7 @@
T9.payment_id, T9.payment_id,
T9.picture_url, T9.picture_url,
T9.birth_date, T9.birth_date,
t9.english_name, t9.english_name
t9.slot_id,
t9.pool_id
from ( from (
SELECT T1.tenant_id AS tenant_id, SELECT T1.tenant_id AS tenant_id,
T1.id AS encounter_id, T1.id AS encounter_id,
@@ -53,9 +51,7 @@
T13.id AS payment_id, T13.id AS payment_id,
ai.picture_url AS picture_url, ai.picture_url AS picture_url,
T8.birth_date AS birth_date, T8.birth_date AS birth_date,
tx.staff_english_name AS english_name, tx.staff_english_name AS english_name
om_slot.slot_id AS slot_id,
om_slot.pool_id AS pool_id
FROM adm_encounter AS T1 FROM adm_encounter AS T1
LEFT JOIN adm_organization AS T2 ON T1.organization_id = T2.ID AND T2.delete_flag = '0' LEFT JOIN adm_organization AS T2 ON T1.organization_id = T2.ID AND T2.delete_flag = '0'
LEFT JOIN adm_healthcare_service AS T3 ON T1.service_type_id = T3.ID AND T3.delete_flag = '0' LEFT JOIN adm_healthcare_service AS T3 ON T1.service_type_id = T3.ID AND T3.delete_flag = '0'
@@ -95,8 +91,6 @@
AND T13.status_enum = ${paymentStatus} AND T13.status_enum = ${paymentStatus}
LEFT JOIN adm_invoice AS ai LEFT JOIN adm_invoice AS ai
ON ai.reconciliation_id = T13.id AND ai.delete_flag = '0' ON ai.reconciliation_id = T13.id AND ai.delete_flag = '0'
LEFT JOIN order_main AS om ON T1.order_id = om.id AND om.delete_flag = '0'
LEFT JOIN adm_schedule_slot AS om_slot ON om.slot_id = om_slot.id
WHERE T1.delete_flag = '0' WHERE T1.delete_flag = '0'
AND T1.class_enum = #{classEnum} AND T1.class_enum = #{classEnum}
AND T10.context_enum = #{register} AND T10.context_enum = #{register}

View File

@@ -49,11 +49,6 @@ public enum RequestStatus implements HisEnumInterface {
*/ */
ENDED(7, "ended", "不执行"), ENDED(7, "ended", "不执行"),
/**
* 已出报告
*/
COMPLETED_REPORT(8, "completed_report", "已出报告"),
/** /**
* 未知 * 未知
*/ */

View File

@@ -193,9 +193,6 @@ public class OpSchedule extends HisBaseEntity {
/** 外请专家姓名 */ /** 外请专家姓名 */
private String externalExpertName; private String externalExpertName;
/** 费用类别 */
private String feeType;
/** 备注信息 */ /** 备注信息 */
private String remark; private String remark;

View File

@@ -1,5 +1,3 @@
import useDictStore from '@/store/modules/dict';
// 日期格式化 // 日期格式化
export function parseTime(time, pattern) { export function parseTime(time, pattern) {
if (arguments.length === 0 || !time) { if (arguments.length === 0 || !time) {
@@ -277,13 +275,30 @@ export function blobValidate(data) {
// 按照频次天数计算总数量 // 按照频次天数计算总数量
export function calculateQuantityByDays(frequency, days) { export function calculateQuantityByDays(frequency, days) {
const dicts = useDictStore().getDict('rate_code'); // const dict = useDict('rate_code').rate_code.value
if (!dicts) return; // const rate = dict.find(item => item.value === frequency).remark
const dict = dicts.find(item => item.value === frequency); // if(rate){
if (!dict?.remark) return; // return Math.floor(Number(rate) * days)
const rate = Number(dict.remark); // } else {
if (isNaN(rate) || !rate) return; // return undefined
const quantity = rate * days; // }
const frequencyMap = {
ST: 1,
QD: 1, // 每日一次
BID: 2, // 每日两次
TID: 3, // 每日三次
QID: 4, // 每日四次
QN: 1, // 每晚一次
QOD: 1 / 2, // 每隔一日一次
QW: 1 / 7, // 每周一次
BIW: 2 / 7, // 每周两次
TIW: 3 / 7, // 每周三次
QOW: 1 / 14, // 隔周一次
};
if (!frequencyMap[frequency]) {
return;
}
const quantity = frequencyMap[frequency] * days;
return quantity < 1 ? 1 : Math.ceil(quantity); return quantity < 1 ? 1 : Math.ceil(quantity);
} }

View File

@@ -178,25 +178,22 @@ service.interceptors.request.use(config => {
} }
return Promise.reject('无效的会话,或者会话已过期,请重新登录。') return Promise.reject('无效的会话,或者会话已过期,请重新登录。')
} else if (code === 500) { } else if (code === 500) {
// 检查是否需要跳过错误提示(静默请求:返回响应让.then()处理) // 检查是否需要跳过错误提示
if (res.config?.skipErrorMsg) { if (!res.config?.skipErrorMsg) {
return Promise.resolve(res.data) ElMessage({ message: msg, type: 'error' })
} }
ElMessage({ message: msg, type: 'error' })
return Promise.reject(new Error(msg)) return Promise.reject(new Error(msg))
} else if (code === 601) { } else if (code === 601) {
// 检查是否需要跳过错误提示(静默请求:返回响应让.then()处理) // 检查是否需要跳过错误提示
if (res.config?.skipErrorMsg) { if (!res.config?.skipErrorMsg) {
return Promise.resolve(res.data) ElMessage({ message: msg, type: 'warning' })
} }
ElMessage({ message: msg, type: 'warning' })
return Promise.reject(new Error(msg)) return Promise.reject(new Error(msg))
} else if (code !== 200) { } else if (code !== 200) {
// 检查是否需要跳过错误提示(静默请求:返回响应让.then()处理) // 检查是否需要跳过错误提示
if (res.config?.skipErrorMsg) { if (!res.config?.skipErrorMsg) {
return Promise.resolve(res.data) ElNotification.error({ title: msg })
} }
ElNotification.error({ title: msg })
return Promise.reject('error') return Promise.reject('error')
} else { } else {
return Promise.resolve(res.data) return Promise.resolve(res.data)

View File

@@ -226,14 +226,8 @@ function getList() {
getDiagnosisTreatmentList(queryParams.value).then((res) => { getDiagnosisTreatmentList(queryParams.value).then((res) => {
loading.value = false; loading.value = false;
catagoryList.value = res.data.records.map(record => { catagoryList.value = res.data.records.map(record => {
// 为每一行初始化 filteredOptions确保显示框能正确显示项目名称
const filteredOptions = allImplementDepartmentList.value.slice(0, 100); const filteredOptions = allImplementDepartmentList.value.slice(0, 100);
// 确保后端返回的项目名称选项存在于 filteredOptions 中,避免 el-select 因找不到选项而回显为 ID
if (record.activityDefinitionId && !filteredOptions.some(o => o.value === record.activityDefinitionId)) {
filteredOptions.push({
value: record.activityDefinitionId,
label: record.activityDefinitionId_dictText || record.activityDefinitionId
});
}
return { return {
...record, ...record,
loading: false, loading: false,

View File

@@ -473,12 +473,15 @@ function calculateTotalPrice() {
} }
}); });
totalPrice.value = sum.toFixed(2); totalPrice.value = sum.toFixed(2);
// Bug #464: 零售价与诊疗子项合计总价实时同步直接赋值不使用nextTick避免多调用方竞争 // Bug #464: 零售价与诊疗子项合计总价实时同步
const hasValidItem = treatmentItems.value.some( const hasValidItem = treatmentItems.value.some(
(item) => item.adviceDefinitionId && item.adviceDefinitionId !== '' (item) => item.adviceDefinitionId && item.adviceDefinitionId !== ''
); );
if (hasValidItem) { if (hasValidItem) {
form.value.retailPrice = parseFloat(totalPrice.value) || 0; // 使用 nextTick 确保总价更新后零售价才更新,避免 Vue 响应式时序问题
nextTick(() => {
form.value.retailPrice = parseFloat(totalPrice.value) || 0;
});
} else { } else {
form.value.retailPrice = undefined; form.value.retailPrice = undefined;
} }
@@ -760,7 +763,10 @@ function selectRow(row, index) {
treatmentItems.value[index].adviceDefinitionId = row.id; treatmentItems.value[index].adviceDefinitionId = row.id;
treatmentItems.value[index].retailPrice = row.retailPrice || 0; treatmentItems.value[index].retailPrice = row.retailPrice || 0;
medicineSearchKey.value = ''; medicineSearchKey.value = '';
calculateTotalPrice(); // 使用 nextTick 确保 DOM 更新后再计算总价
nextTick(() => {
calculateTotalPrice();
});
} }
// 清空诊疗子项 // 清空诊疗子项

View File

@@ -115,44 +115,22 @@ function getList() {
console.log('[adviceBaseList] getList() 跳过:未选择患者'); console.log('[adviceBaseList] getList() 跳过:未选择患者');
return; // 不执行API调用 return; // 不执行API调用
} }
// 只有在弹窗打开时才执行查询 // 只有在弹窗打开时才执行查询
if (!props.popoverVisible) { if (!props.popoverVisible) {
console.log('[adviceBaseList] getList() 跳过:弹窗未打开'); console.log('[adviceBaseList] getList() 跳过:弹窗未打开');
return; return;
} }
// 🔧 Bug #448 修复:显式构建请求参数,确保 adviceType 正确传递 queryParams.value.organizationId = props.patientInfo.orgId;
// 不直接使用 queryParams.value避免 undefined 值被发送到后端导致过滤失效 console.log('[adviceBaseList] getList() 请求参数:', JSON.stringify(queryParams.value));
const requestParams = {
pageSize: queryParams.value.pageSize, getAdviceBaseInfo(queryParams.value).then((res) => {
pageNum: queryParams.value.pageNum,
organizationId: props.patientInfo.orgId,
};
// 只在 adviceType 有值时添加0 是无效值undefined/null 会导致后端查询所有类型)
if (queryParams.value.adviceType != null && queryParams.value.adviceType !== 0) {
requestParams.adviceType = queryParams.value.adviceType;
}
// 只在 categoryCode 有值时添加
if (queryParams.value.categoryCode) {
requestParams.categoryCode = queryParams.value.categoryCode;
}
// 只在 searchKey 有值时添加
if (queryParams.value.searchKey) {
requestParams.searchKey = queryParams.value.searchKey;
}
console.log('[adviceBaseList] getList() 请求参数:', JSON.stringify(requestParams));
getAdviceBaseInfo(requestParams).then((res) => {
console.log('[adviceBaseList] getList() 响应数据:', { console.log('[adviceBaseList] getList() 响应数据:', {
total: res.data?.total, total: res.data?.total,
recordsCount: res.data?.records?.length || 0, recordsCount: res.data?.records?.length || 0,
firstRecord: res.data?.records?.[0]?.adviceName || '无数据', firstRecord: res.data?.records?.[0]?.adviceName || '无数据',
adviceType: requestParams.adviceType adviceType: queryParams.value.adviceType
}); });
adviceBaseList.value = res.data.records || []; adviceBaseList.value = res.data.records || [];
total.value = res.data.total || 0; total.value = res.data.total || 0;

View File

@@ -461,7 +461,7 @@ watch(
console.log(prescriptionList.value,"prescriptionList.value") console.log(prescriptionList.value,"prescriptionList.value")
if(newValue&&newValue.length>0){ if(newValue&&newValue.length>0){
let saveList = prescriptionList.value.filter((item) => { let saveList = prescriptionList.value.filter((item) => {
return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag) return item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
}) })
console.log(saveList,"prescriptionList.value") console.log(saveList,"prescriptionList.value")
if (saveList.length == 0) { if (saveList.length == 0) {
@@ -887,10 +887,6 @@ function handleDelete() {
if (item.statusEnum != 1 || item.chargeStatus == 5) { if (item.statusEnum != 1 || item.chargeStatus == 5) {
return null; return null;
} }
// 🔧 Bug #442: 非本人创建的医嘱不允许删除(与签发/签退逻辑保持一致)
if (Number(item.bizRequestFlag) !== 1 && item.bizRequestFlag) {
return null;
}
// 🔧 Bug #442: 已保存的行必须有有效的 requestId否则跳过避免后端删除不存在的记录 // 🔧 Bug #442: 已保存的行必须有有效的 requestId否则跳过避免后端删除不存在的记录
if (item.requestId == null || item.requestId === undefined || item.requestId === '') { if (item.requestId == null || item.requestId === undefined || item.requestId === '') {
return null; return null;
@@ -900,7 +896,7 @@ function handleDelete() {
dbOpType: '3', dbOpType: '3',
adviceType: item.adviceType, adviceType: item.adviceType,
}; };
}).filter(item => item !== null); // 过滤掉已签发、已收费、非本人创建或无 requestId 的项目 }).filter(item => item !== null); // 过滤掉已签发、已收费或无 requestId 的项目
if (deleteList.length == 0) { if (deleteList.length == 0) {
proxy.$modal.msgWarning('只能删除待签发且未收费的项目'); proxy.$modal.msgWarning('只能删除待签发且未收费的项目');
@@ -1015,7 +1011,7 @@ function handleSave() {
return; return;
} }
let saveList = prescriptionList.value.filter((item) => { let saveList = prescriptionList.value.filter((item) => {
return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag) return item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
}); });
// let saveList = prescriptionList.value // let saveList = prescriptionList.value
// .filter((item) => { // .filter((item) => {
@@ -1036,14 +1032,6 @@ function handleSave() {
requestId: item.requestId, requestId: item.requestId,
dbOpType: '1', dbOpType: '1',
groupId: item.groupId, groupId: item.groupId,
// 🔧 Bug #443: 补充顶层关键字段(这些不在 contentJson 中,需从 API 响应顶层提取)
encounterId: item.encounterId,
patientId: item.patientId,
locationId: item.positionId,
adviceType: item.adviceType,
adviceTableName: item.adviceTableName,
adviceDefinitionId: item.adviceDefinitionId,
chargeItemId: item.chargeItemId,
}; };
}); });
savePrescriptionSign({ savePrescriptionSign({
@@ -1059,12 +1047,7 @@ function handleSave() {
groupIndexList.value = [] groupIndexList.value = []
groupList.value = [] groupList.value = []
nextId.value = 1; nextId.value = 1;
} else {
proxy.$modal.msgError(res?.msg || '签发失败,请重试');
} }
}).catch((error) => {
console.error('签发失败:', error);
proxy.$modal.msgError(error?.response?.data?.msg || error?.message || '签发失败,请重试');
}); });
} }
@@ -1080,44 +1063,42 @@ function handleSaveSign(row, index) {
proxy.$modal.msgWarning('诊疗项目必须选择执行科室'); proxy.$modal.msgWarning('诊疗项目必须选择执行科室');
return; return;
} }
isSaving.value = true; // #437 立即加锁,消除 TOCTOU 竞态
proxy.$refs['formRef' + index].validate((valid) => { proxy.$refs['formRef' + index].validate((valid) => {
if (!valid) { if (valid) {
isSaving.value = false; // 验证失败释放 isSaving.value = true; // #437 加
return; row.isEdit = false;
} isAdding.value = false;
row.isEdit = false; expandOrder.value = [];
isAdding.value = false; row.patientId = props.patientInfo.patientId;
expandOrder.value = []; row.encounterId = props.patientInfo.encounterId;
row.patientId = props.patientInfo.patientId; row.accountId = props.patientInfo.accountId;
row.encounterId = props.patientInfo.encounterId; const cleanRow = JSON.parse(JSON.stringify(row));
row.accountId = props.patientInfo.accountId; cleanRow.contentJson = JSON.stringify(cleanRow);
const cleanRow = JSON.parse(JSON.stringify(row)); cleanRow.dbOpType = cleanRow.requestId ? '2' : '1';
cleanRow.contentJson = JSON.stringify(cleanRow); cleanRow.minUnitQuantity = cleanRow.quantity * cleanRow.partPercent;
cleanRow.dbOpType = cleanRow.requestId ? '2' : '1'; cleanRow.categoryEnum = cleanRow.adviceType
cleanRow.minUnitQuantity = cleanRow.quantity * cleanRow.partPercent; // 如果是手术计费,设置生成来源和来源业务单据号
cleanRow.categoryEnum = cleanRow.adviceType if (props.patientInfo.sourceBillNo) {
// 如果是手术计费,设置生成来源和来源业务单据号 cleanRow.generateSourceEnum = 6; // 手术计费
if (props.patientInfo.sourceBillNo) { cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
cleanRow.generateSourceEnum = 6; // 手术计费
cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
}
console.log('cleanRow', cleanRow)
savePrescription({ adviceSaveList: [cleanRow] }, '1').then((res) => {
if (res.code === 200) {
proxy.$modal.msgSuccess('保存成功');
getListInfo(false);
nextId.value = 1;
// 🔧 Bug Fix #238: 如果诊疗项目缺少执行科室,标记为需要修复的脏数据
if (row.adviceType === 3 && !row.orgId) {
console.warn('Bug #238: 检测到诊疗项目保存时缺少执行科室,请手动编辑修正:', cleanRow);
proxy.$modal.msgWarning('诊疗项目执行科室信息不完整,请编辑后重新保存');
}
} }
}).finally(() => { console.log('cleanRow', cleanRow)
isSaving.value = false; // #437 释放锁 savePrescription({ adviceSaveList: [cleanRow] }).then((res) => {
}); if (res.code === 200) {
}) proxy.$modal.msgSuccess('保存成功');
getListInfo(false);
nextId.value = 1;
// 🔧 Bug Fix #238: 如果诊疗项目缺少执行科室,标记为需要修复的脏数据
if (row.adviceType === 3 && !row.orgId) {
console.warn('Bug #238: 检测到诊疗项目保存时缺少执行科室,请手动编辑修正:', cleanRow);
proxy.$modal.msgWarning('诊疗项目执行科室信息不完整,请编辑后重新保存');
}
}
}).finally(() => {
isSaving.value = false; // #437 释放锁
});
}
});
} }
// 签退 // 签退

View File

@@ -49,40 +49,6 @@
</el-col> </el-col>
</el-row> </el-row>
<!-- 性别出生日期或实足年龄 -->
<el-row :gutter="16" class="form-row">
<el-col :span="7" class="form-item">
<span class="form-label required">性别</span>
<el-radio-group v-model="form.sex" class="gender-radio-group">
<el-radio value="男"></el-radio>
<el-radio value="女"></el-radio>
<el-radio value="未知">未知</el-radio>
</el-radio-group>
</el-col>
<el-col :span="10" class="form-item">
<span class="form-label required">出生日期</span>
<div class="birth-input-group">
<el-input v-model="form.birthYear" class="birth-input year" placeholder="年" maxlength="4" />
<span class="birth-separator"></span>
<el-input v-model="form.birthMonth" class="birth-input month" placeholder="月" maxlength="2" />
<span class="birth-separator"></span>
<el-input v-model="form.birthDay" class="birth-input day" placeholder="日" maxlength="2" />
<span class="birth-separator"></span>
</div>
</el-col>
<el-col :span="7" class="form-item">
<span class="form-label"> 实足年龄</span>
<div class="age-input-group">
<el-input v-model="form.age" class="age-input" placeholder="年龄" />
<el-select v-model="form.ageUnit" class="age-unit-select">
<el-option label="岁" value="岁" />
<el-option label="月" value="月" />
<el-option label="天" value="天" />
</el-select>
</div>
</el-col>
</el-row>
<!-- 联系电话紧急联系人电话 --> <!-- 联系电话紧急联系人电话 -->
<el-row :gutter="16" class="form-row"> <el-row :gutter="16" class="form-row">
<el-col :span="12" class="form-item"> <el-col :span="12" class="form-item">
@@ -1034,17 +1000,6 @@ function normalizeSex(value) {
return '未知'; return '未知';
} }
function normalizeSexFromPatientInfo(patientInfo) {
// 优先使用文本字段
if (patientInfo.genderEnum_enumText) return patientInfo.genderEnum_enumText;
if (patientInfo.genderName) return patientInfo.genderName;
if (patientInfo.sex) return normalizeSex(patientInfo.sex);
// 使用数字枚举字段
if (patientInfo.genderEnum === 1) return '男';
if (patientInfo.genderEnum === 2) return '女';
return '未知';
}
function normalizeAgeUnit(value) { function normalizeAgeUnit(value) {
const ageUnitMap = { const ageUnitMap = {
1: '岁', 1: '岁',
@@ -1306,7 +1261,7 @@ async function show(diagnosisData) {
patName: patientInfo.patientName || patientInfo.name || '', // 患者姓名 patName: patientInfo.patientName || patientInfo.name || '', // 患者姓名
parentName: '', // 家长姓名14岁以下患者必填 parentName: '', // 家长姓名14岁以下患者必填
idNo: patientInfo.idCard, // 身份证号 idNo: patientInfo.idCard, // 身份证号
sex: normalizeSexFromPatientInfo(patientInfo), // 性别 sex: patientInfo.sex || patientInfo.genderName || '男', // 性别
// 出生日期信息 // 出生日期信息
birthYear: birthInfo.year, // 出生年份 birthYear: birthInfo.year, // 出生年份
@@ -1411,9 +1366,6 @@ async function buildSubmitData() {
} else if (formData.otherDisease) { } else if (formData.otherDisease) {
// 其他传染病使用自定义编码 // 其他传染病使用自定义编码
diseaseCode = 'OTHER'; diseaseCode = 'OTHER';
} else if (formData.selectedDiseases && formData.selectedDiseases.length > 0) {
// 兜底:如果 ClassA/B/C 都为空但 selectedDiseases 有值,取第一个作为 diseaseCode
diseaseCode = formData.selectedDiseases[0];
} }
// 转换年龄单位:岁=1, 月=2, 天=3 // 转换年龄单位:岁=1, 月=2, 天=3
@@ -1823,33 +1775,6 @@ defineExpose({ show, showReport, close: handleClose });
color: #999; color: #999;
} }
/* 输入框下划线样式(与 underline-select 保持一致) */
.underline-input :deep(.el-input__wrapper) {
border: none;
border-bottom: 1px solid #dcdfe6;
border-radius: 0;
box-shadow: none;
background: transparent;
}
.underline-input :deep(.el-input__wrapper:hover) {
border-bottom-color: #c0c4cc;
}
.underline-input :deep(.el-input__wrapper.is-focus) {
border-bottom-color: #409eff;
}
.underline-input :deep(.el-input__inner) {
font-size: 12px;
color: #666;
}
.underline-input :deep(.el-input__inner::placeholder) {
font-size: 12px;
color: #999;
}
/* 街道下拉框下划线样式 */ /* 街道下拉框下划线样式 */
.underline-select { .underline-select {
width: 100%; width: 100%;
@@ -2037,53 +1962,4 @@ defineExpose({ show, showReport, close: handleClose });
display: flex !important; display: flex !important;
justify-content: center !important; justify-content: center !important;
} }
/* 性别单选按钮组 */
.gender-radio-group {
display: flex;
gap: 12px;
padding-top: 4px;
}
/* 出生日期输入组 */
.birth-input-group {
display: flex;
align-items: center;
gap: 2px;
}
.birth-input {
text-align: center;
}
.birth-input.year {
width: 70px;
}
.birth-input.month,
.birth-input.day {
width: 50px;
}
.birth-separator {
color: #606266;
font-size: 13px;
margin: 0 2px;
}
/* 年龄输入组 */
.age-input-group {
display: flex;
align-items: center;
gap: 6px;
}
.age-input {
width: 70px;
text-align: center;
}
.age-unit-select {
width: 65px;
}
</style> </style>

View File

@@ -3,56 +3,15 @@
<!-- ====== 顶部卡片申请单列表 ====== --> <!-- ====== 顶部卡片申请单列表 ====== -->
<div class="top-section"> <div class="top-section">
<div class="section-header"> <div class="section-header">
<span class="section-title">检查项目 ({{ filteredApplicationList.length }})</span> <span class="section-title">检查项目 ({{ applicationList.length }})</span>
<div class="header-actions"> <div class="header-actions">
<el-button type="primary" @click="handleAdd" icon="Plus">新增</el-button> <el-button type="primary" @click="handleAdd" icon="Plus">新增</el-button>
<el-button type="success" @click="handleSave" icon="Finished">保存</el-button> <el-button type="success" @click="handleSave" icon="Finished">保存</el-button>
</div> </div>
</div> </div>
<!-- Bug #499: 查询过滤工具栏 -->
<div class="search-toolbar">
<el-form :inline="true" size="small">
<el-form-item label="日期范围">
<el-date-picker
v-model="searchForm.dateRange"
type="daterange"
range-separator=""
start-placeholder="开始日期"
end-placeholder="结束日期"
value-format="YYYY-MM-DD"
style="width: 240px"
/>
</el-form-item>
<el-form-item label="状态">
<el-select v-model="searchForm.applyStatus" placeholder="全部" clearable style="width: 140px">
<el-option
v-for="opt in statusOptions"
:key="opt.value"
:label="opt.label"
:value="opt.value"
/>
</el-select>
</el-form-item>
<el-form-item label="关键字">
<el-input
v-model="searchForm.keyword"
placeholder="申请单号 / 检查项目"
clearable
style="width: 200px"
@keyup.enter="handleSearch"
/>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleSearch" icon="Search">搜索</el-button>
<el-button @click="handleResetSearch" icon="Refresh">重置</el-button>
</el-form-item>
</el-form>
</div>
<el-table <el-table
v-loading="loading" v-loading="loading"
:data="filteredApplicationList" :data="applicationList"
:max-height="200" :max-height="200"
highlight-current-row highlight-current-row
@row-click="handleRowClick" @row-click="handleRowClick"
@@ -278,33 +237,16 @@
<el-input v-model="scope.row.applyPart" size="small" /> <el-input v-model="scope.row.applyPart" size="small" />
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="检查方法" min-width="160"> <el-table-column label="检查方法" min-width="120">
<template #default="scope"> <template #default="scope">
<el-select <!-- Bug #384修复: 显示检查方法名称不显示套餐名称 -->
v-if="scope.row.methods && scope.row.methods.length > 1" <span v-if="scope.row.selectedMethod">
:model-value="scope.row.selectedMethod" {{ scope.row.selectedMethod.name }}
value-key="id" </span>
size="small" <span v-else-if="scope.row.methods && scope.row.methods.length > 0" style="color: #909399;">
style="width: 100%" 未选择
placeholder="选择方法" </span>
@update:model-value="(val) => onDetailMethodChange(scope.row, val)" <span v-else style="color: #c0c4cc;">-</span>
>
<el-option
v-for="meth in scope.row.methods"
:key="meth.id"
:label="`${meth.name}${meth.packagePrice != null ? ' ¥' + formatDetailAmount(meth.packagePrice) : ''}`"
:value="meth"
/>
</el-select>
<template v-else>
<span v-if="scope.row.selectedMethod">
{{ scope.row.selectedMethod.name }}
</span>
<span v-else-if="scope.row.methods && scope.row.methods.length > 0" style="color: #909399;">
未选择
</span>
<span v-else style="color: #c0c4cc;">-</span>
</template>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="单位" prop="unit" width="55" align="center" /> <el-table-column label="单位" prop="unit" width="55" align="center" />
@@ -393,9 +335,8 @@
加载中... 加载中...
</div> </div>
<!-- Bug #428修复: 渲染分类联动加载的检查方法列表 --> <!-- Bug #428修复: 渲染分类联动加载的检查方法列表 -->
<!-- Bug #500修复: v-if 改为 v-show避免方法列表加载时 DOM 突然插入导致高度跳变 -->
<div <div
v-show="cat.methods && cat.methods.length > 0" v-if="cat.methods && cat.methods.length > 0"
class="method-section" class="method-section"
> >
<div class="method-section-title">检查方法</div> <div class="method-section-title">检查方法</div>
@@ -429,15 +370,12 @@
v-for="(item, idx) in selectedItems" v-for="(item, idx) in selectedItems"
:key="idx" :key="idx"
class="selected-item-card" class="selected-item-card"
:class="{ 'is-expanded': item.expanded }"
> >
<!-- Bug #384修复 + #426修复: 项目卡片头部,可展开/收起 --> <!-- Bug #384修复: 项目卡片头部可展开/收起 -->
<div class="card-header" @click="toggleItemExpand(item)"> <div class="card-header" @click="toggleItemExpand(item)">
<el-tag v-if="item.isPackage || item.packageName" size="small" type="warning" style="margin-right: 4px; flex-shrink: 0;">套餐</el-tag> <span class="card-name">{{ item.name }}</span>
<el-tooltip :content="item.name" placement="top" :show-after="400"> <span class="card-price">¥{{ item.price }}</span>
<span class="card-name">{{ item.name }}</span> <!-- 展开图标 -->
</el-tooltip>
<span class="card-price">¥{{ formatDetailAmount(item.price) }}</span>
<el-icon :class="['expand-icon', { expanded: item.expanded }]"> <el-icon :class="['expand-icon', { expanded: item.expanded }]">
<ArrowDown v-if="!item.expanded" /> <ArrowDown v-if="!item.expanded" />
<ArrowUp v-if="item.expanded" /> <ArrowUp v-if="item.expanded" />
@@ -447,30 +385,39 @@
<el-icon><Close /></el-icon> <el-icon><Close /></el-icon>
</el-button> </el-button>
</div> </div>
<!-- Bug #428: 有套餐 ID 时默认展开;加载中/空/明细均在本区域展示 --> <!-- Bug #428修复: 展开后显示套餐明细或检查方法 -->
<div v-if="item.expanded && shouldShowPackageBody(item)" class="selected-card-body"> <div v-if="item.expanded">
<div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div> <!-- 显示套餐明细 -->
<template v-else> <div v-if="item.packageDetails && item.packageDetails.length > 0" class="package-details-list">
<div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty"> <div class="detail-row" v-for="detail in item.packageDetails" :key="detail.id">
暂无套餐明细 <span class="detail-name">{{ detail.name }}</span>
<span class="detail-info">数量: {{ detail.quantity }} 单价: ¥{{ detail.price }}</span>
</div> </div>
<div v-else class="package-details-list"> </div>
<div class="package-details-head">套餐明细</div> <!-- 显示检查方法 -->
<div <div v-else-if="item.methods && item.methods.length > 0" class="method-list">
v-for="(detail, dIdx) in getPackageDetailsList(item)" <div v-for="method in item.methods" :key="method.id" class="method-option">
:key="detail.id ?? detail.itemCode ?? `d-${dIdx}`" <el-checkbox :model-value="item.selectedMethod?.id === method.id" @change="(val) => selectMethodCheckbox(val, item, method)">
class="detail-row" <span class="method-name">{{ method.name }}</span>
> <span class="method-price">¥{{ method.packagePrice || item.price }}</span>
<el-tooltip :content="detail.name" placement="top" :show-after="500"> </el-checkbox>
<span class="detail-name">{{ detail.name }}</span> </div>
</el-tooltip> <!-- 选中方法后显示对应的套餐明细 -->
<div class="detail-meta"> <div v-if="item.selectedMethod && item.methodPackageDetails && item.methodPackageDetails.length > 0" class="method-package-details">
<span class="detail-qty">×{{ detail.quantity || 1 }}</span> <div class="method-package-header">
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span> <span class="method-package-title">套餐明细 - {{ item.selectedMethod.name }}</span>
</div> </div>
<div v-for="detail in item.methodPackageDetails" :key="detail.id" class="method-option">
<el-checkbox v-model="detail.checked">
<span class="method-name">{{ detail.name }}</span>
<span class="method-price">数量: {{ detail.quantity }} ¥{{ detail.price }}</span>
</el-checkbox>
</div> </div>
</div> </div>
</template> <div v-if="item.selectedMethod && item.methodPackageLoading" class="method-package-loading">
加载套餐明细中...
</div>
</div>
</div> </div>
</div> </div>
</div> </div>
@@ -505,92 +452,23 @@ const activeDetailTab = ref('applyForm');
const applicationList = ref([]); const applicationList = ref([]);
const selectedItems = ref([]); const selectedItems = ref([]);
// Bug #499: 查询过滤状态
const searchForm = reactive({
dateRange: [],
applyStatus: '',
keyword: ''
});
// 申请单状态选项
const statusOptions = [
{ label: '已开单', value: 0 },
{ label: '已收费', value: 1 },
{ label: '已预约', value: 2 },
{ label: '已签到', value: 3 },
{ label: '部分报告', value: 4 },
{ label: '已完告', value: 5 },
{ label: '已作废', value: 6 }
];
// Bug #499: 过滤后的申请单列表
const filteredApplicationList = computed(() => {
let result = applicationList.value;
// 日期范围过滤
if (searchForm.dateRange && searchForm.dateRange.length === 2) {
const start = searchForm.dateRange[0];
const end = searchForm.dateRange[1];
result = result.filter(item => {
const d = item.applyTime;
if (!d) return false;
const dateStr = d.length > 10 ? d.substring(0, 10) : d;
return dateStr >= start && dateStr <= end;
});
}
// 状态过滤
if (searchForm.applyStatus !== '' && searchForm.applyStatus !== null && searchForm.applyStatus !== undefined) {
result = result.filter(item => item.applyStatus === searchForm.applyStatus);
}
// 关键字过滤(申请单号、申检部位、检查项目名)
if (searchForm.keyword) {
const kw = searchForm.keyword.toLowerCase();
result = result.filter(item => {
return (item.applyNo || '').toLowerCase().includes(kw)
|| (item.inspectionArea || '').toLowerCase().includes(kw);
});
}
return result;
});
// Bug #499: 搜索与重置
function handleSearch() {
// 过滤逻辑由 computed 自动处理
}
function handleResetSearch() {
const now = new Date();
const end = now.toISOString().substring(0, 10);
const start = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000).toISOString().substring(0, 10);
searchForm.dateRange = [start, end];
searchForm.applyStatus = '';
searchForm.keyword = '';
}
// 初始化默认日期范围为近一周
handleResetSearch();
// 🔧 BugFix#426: 懒加载套餐明细 // 🔧 BugFix#426: 懒加载套餐明细
async function loadPackageDetails(row, treeNode, resolve) { async function loadPackageDetails(row, treeNode, resolve) {
if (!row.packageId) { if (!row.isPackage || !row.packageId) {
resolve([]); resolve([]);
return; return;
} }
try { try {
const res = await request({ const res = await request({
url: `/system/check-type/package/${row.packageId}/details`, url: `/system/package/${row.packageId}/details`,
method: 'get' method: 'get'
}); });
if (res.code === 200) { if (res.code === 200 && res.data) {
const list = parsePackageDetailsPayload(res); const children = res.data.map(item => ({
const children = list.map((child) => ({ ...item,
...child, name: item.name || item.itemName,
name: child.name || child.itemName, unit: item.unit || '次',
unit: child.unit || '次', price: item.price || item.itemPrice || 0,
price: child.price ?? child.unitPrice ?? child.itemPrice ?? 0,
quantity: row.quantity || 1, quantity: row.quantity || 1,
isPackageDetail: true isPackageDetail: true
})); }));
@@ -604,69 +482,15 @@ async function loadPackageDetails(row, treeNode, resolve) {
} }
} }
// #428修复 + #426修复: 为已选择项目加载套餐明细通过packageId或packageName查询 // #428修复: 为已选择项目加载套餐明细通过packageId或packageName查询
/** 套餐明细挂在「部位」或已选的「检查方法」上(方法可带 packageId */
function getPackageCarrier(item) {
return item?.selectedMethod?.packageId ? item.selectedMethod : item;
}
function getPackageDetailsList(item) {
// 明细挂在行对象上,避免仅写入 methods 内嵌对象时首帧不触发视图更新(体感需点两次才展开)
if (Array.isArray(item?.packageDetailsDisplay)) {
return item.packageDetailsDisplay;
}
const carrier = getPackageCarrier(item);
return Array.isArray(carrier?.packageDetails) ? carrier.packageDetails : [];
}
/** 有套餐 ID 的已选行才展示右侧套餐区(加载中 / 空 / 明细列表) */
function shouldShowPackageBody(item) {
return !!getPackageCarrier(item)?.packageId;
}
/** 金额展示:统一两位小数 */
function formatDetailAmount(value) {
const n = Number(value ?? 0);
return Number.isFinite(n) ? n.toFixed(2) : '0.00';
}
/** 默认检查方法:优先与部位 packageId 一致的方法,否则取首个带套餐的方法,否则取第一个 */
function pickDefaultMethod(methods, partItem) {
if (!methods?.length) return null;
if (methods.length === 1) return methods[0];
const pid = partItem?.packageId ?? null;
if (pid != null && pid !== '') {
const matched = methods.find(
(x) => x.packageId != null && String(x.packageId) === String(pid)
);
if (matched) return matched;
}
const withPkg = methods.find((x) => x.packageId != null);
if (withPkg) return withPkg;
return methods[0];
}
function parsePackageDetailsPayload(res) {
const raw =
res?.data?.data ??
res?.data?.records ??
res?.data ??
res?.rows ??
res;
if (!Array.isArray(raw)) return [];
return raw;
}
// #428: 为已选择项目加载套餐明细后端CheckTypeController /system/check-type/package/{id}/details
async function loadPackageDetailsForItem(item) { async function loadPackageDetailsForItem(item) {
const carrier = getPackageCarrier(item); if (!item.isPackage || (!item.packageId && !item.packageName)) {
let packageId = item.packageId || carrier?.packageId;
if (!packageId && !item.packageName) {
return; return;
} }
item.packageDetailsLoading = true;
try { try {
let packageId = item.packageId;
if (!packageId && item.packageName) { if (!packageId && item.packageName) {
// CheckPart 没有 packageId 字段,需要通过 packageName 查询获取
const pkgRes = await listCheckPackage({ packageName: item.packageName }); const pkgRes = await listCheckPackage({ packageName: item.packageName });
let packages = pkgRes?.data || []; let packages = pkgRes?.data || [];
if (!Array.isArray(packages)) { if (!Array.isArray(packages)) {
@@ -674,51 +498,28 @@ async function loadPackageDetailsForItem(item) {
} }
if (packages.length === 0) { if (packages.length === 0) {
item.packageDetails = []; item.packageDetails = [];
item.packageDetailsDisplay = [];
return; return;
} }
packageId = packages[0].id; packageId = packages[0].id;
item.packageId = packageId;
}
if (!packageId) {
item.packageDetails = [];
item.packageDetailsDisplay = [];
return;
} }
const res = await request({ const res = await request({
url: `/system/check-type/package/${packageId}/details`, url: `/system/package/${packageId}/details`,
method: 'get' method: 'get'
}); });
const list = parsePackageDetailsPayload(res);
const mapped = list.map((detail) => ({
...detail,
name: detail.name || detail.itemName,
unit: detail.unit || '次',
price: detail.price ?? detail.unitPrice ?? detail.itemPrice ?? 0,
quantity: detail.quantity || 1
}));
item.packageDetailsDisplay = mapped;
carrier.packageDetails = mapped;
if (res.code === 200 && res.data) { if (res.code === 200 && res.data) {
item.packageDetails = Array.isArray(res.data) item.packageDetails = res.data.map(detail => ({
? res.data.map((detail) => ({ ...detail,
...detail, name: detail.name || detail.itemName,
name: detail.name || detail.itemName, unit: detail.unit || '次',
unit: detail.unit || '次', price: detail.price || detail.unitPrice || 0,
price: detail.price || detail.unitPrice || 0, quantity: detail.quantity || 1
quantity: detail.quantity || 1 }));
}))
: mapped;
} else { } else {
item.packageDetails = mapped; item.packageDetails = [];
} }
} catch (err) { } catch (err) {
console.error('加载套餐明细失败:', err); console.error('加载套餐明细失败:', err);
item.packageDetailsDisplay = [];
carrier.packageDetails = [];
item.packageDetails = []; item.packageDetails = [];
} finally {
item.packageDetailsLoading = false;
} }
} }
const detailTableRef = ref(null); const detailTableRef = ref(null);
@@ -772,7 +573,7 @@ const categoryList = ref([]); // 原始分类+项目数据
const dictSearchKey = ref(''); const dictSearchKey = ref('');
const activeNames = ref(''); // 当前展开的折叠项 const activeNames = ref(''); // 当前展开的折叠项
const categoryLoadingSet = ref(new Set()); // Bug #500: 正在加载方法的分类集合 const categoryLoadingSet = ref(new Set()); // Bug #500: 正在加载方法的分类集合
const currentActiveCategory = ref(null); // Bug #500: 记录当前激活的分类,忽略过期请求响应 const isAnimating = ref(false); // Bug #500: 防止快速切换时折叠动画重叠导致抖动
const allMethods = ref([]); const allMethods = ref([]);
@@ -889,18 +690,15 @@ const availableMethods = computed(() => {
// 当可选方法列表改变时,如果当前选中的方法不在新列表中,则清空 // 当可选方法列表改变时,如果当前选中的方法不在新列表中,则清空
// #428: 分类展开时联动加载检查方法 // #428: 分类展开时联动加载检查方法
// Bug #500: 使用 categoryLoadingSet 替代 dictLoading避免切换分类时整个区域闪烁 // Bug #500: 使用 categoryLoadingSet 替代 dictLoading避免切换分类时整个区域闪烁
// Bug #500: 添加 currentActiveCategory 守卫,忽略过期请求响应,防止快速切换时数据闪烁
async function handleCategoryExpand(cat) { async function handleCategoryExpand(cat) {
if (!cat || !cat.typeName) return; if (!cat || !cat.typeName) return;
// 如果已加载过或正在加载中,跳过
if ((cat.methods && cat.methods.length > 0) || categoryLoadingSet.value.has(cat.typeId)) return; if ((cat.methods && cat.methods.length > 0) || categoryLoadingSet.value.has(cat.typeId)) return;
categoryLoadingSet.value.add(cat.typeId); categoryLoadingSet.value.add(cat.typeId);
currentActiveCategory.value = cat.typeId;
try { try {
const res = await searchCheckMethod({ checkType: cat.typeName }); const res = await searchCheckMethod({ checkType: cat.typeName });
// 忽略过期请求:用户已切换到其他分类,丢弃当前响应
if (currentActiveCategory.value !== cat.typeId) return;
let data = res?.data?.data || res?.data || res?.rows || res; let data = res?.data?.data || res?.data || res?.rows || res;
if (!Array.isArray(data) && res?.data && Array.isArray(res.data.data)) { if (!Array.isArray(data) && res?.data && Array.isArray(res.data.data)) {
data = res.data.data; data = res.data.data;
@@ -919,16 +717,17 @@ async function handleCategoryExpand(cat) {
})); }));
} }
} catch (err) { } catch (err) {
if (currentActiveCategory.value !== cat.typeId) return;
console.error('加载分类检查方法失败', err); console.error('加载分类检查方法失败', err);
} finally { } finally {
categoryLoadingSet.value.delete(cat.typeId); categoryLoadingSet.value.delete(cat.typeId);
} }
} }
// Bug #500修复: 不阻塞 accordion 状态更新,仅防止重复加载同一分类的方法 // Bug #500: 添加防抖逻辑,快速切换时跳过中间状态的动画,避免高度跳变和白屏闪烁
function handleCollapseChange(activeName) { function handleCollapseChange(activeName) {
// 始终记录当前激活的分类,确保 handleCategoryExpand 能正确忽略过期请求 if (isAnimating.value) return; // 动画进行中,忽略后续点击
currentActiveCategory.value = activeName || null;
isAnimating.value = true;
setTimeout(() => { isAnimating.value = false; }, 300); // 与 CSS 过渡时长一致
if (activeName) { if (activeName) {
// Bug #428修复: 直接从 categoryList原始响应式数组查找分类 // Bug #428修复: 直接从 categoryList原始响应式数组查找分类
@@ -1001,8 +800,7 @@ async function loadCategoryList() {
categoryName: t.name, categoryName: t.name,
// “检查类型管理”里配置的执行科室(图三) // “检查类型管理”里配置的执行科室(图三)
performDeptName: t.department || '', performDeptName: t.department || '',
items: [], items: []
methods: [] // #428修复: 初始化 methods 数组,确保 Vue 响应式追踪
}); });
} }
const unclassified = []; const unclassified = [];
@@ -1226,19 +1024,17 @@ function handleRowClick(row) {
selectedItems.value = []; selectedItems.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.dataAjaxResult体 const resp = res.data || res;
// 但某些情况下可能返回完整 Axios 响应 {data: AjaxResult}。 // Bug #408修复: items 在 AjaxResult 顶层(res.items / resp.items),不在 ExamApply 对象内
// 用 res.code 判定是否已是 AjaxResult 体,避免二次解包导致 items 丢失。 // 防御性提取:优先取顶层 items兼容嵌套在 resp.data.items 的情况
const isAjaxResult = res && typeof res === 'object' && res.code !== undefined; let rawItems = res.items || resp.items;
const ajaxBody = isAjaxResult ? res : (res.data || res); if (!rawItems && resp.data && typeof resp.data === 'object') {
rawItems = resp.data.items;
// items 在 AjaxResult 顶层data 字段是 ExamApply 实体 }
const rawItems = Array.isArray(ajaxBody.items) ? ajaxBody.items : []; rawItems = rawItems || [];
const detailData = ajaxBody.data || {}; const d = resp.data || resp;
if (d) Object.assign(form, d);
if (detailData && typeof detailData === 'object') Object.assign(form, detailData); if (Array.isArray(rawItems) && rawItems.length > 0) {
if (rawItems.length > 0) {
try { try {
// 为每个项目加载检查方法 // 为每个项目加载检查方法
const itemsWithMethods = await Promise.all(rawItems.map(async m => { const itemsWithMethods = await Promise.all(rawItems.map(async m => {
@@ -1251,10 +1047,7 @@ function handleRowClick(row) {
nationalCode: '', checked: true, nationalCode: '', checked: true,
methods: [], methods: [],
selectedMethod: null, selectedMethod: null,
expanded: false, expanded: false // Bug #384修复: 添加展开状态
packageDetailsLoading: false,
isPackage: false,
packageId: null
}; };
// 加载该项目的检查方法 // 加载该项目的检查方法
if (m.bodyPartCode) { if (m.bodyPartCode) {
@@ -1285,13 +1078,6 @@ function handleRowClick(row) {
item.packageId = item.selectedMethod.packageId; item.packageId = item.selectedMethod.packageId;
} }
} }
if (!item.selectedMethod && item.methods.length) {
item.selectedMethod = pickDefaultMethod(item.methods, { packageId: item.packageId });
}
if (item.selectedMethod?.packageId) {
item.packageId = item.selectedMethod.packageId;
item.isPackage = true;
}
} }
} catch (err) { } catch (err) {
console.error('加载检查方法失败', err); console.error('加载检查方法失败', err);
@@ -1301,12 +1087,6 @@ function handleRowClick(row) {
return item; return item;
})); }));
selectedItems.value = itemsWithMethods; selectedItems.value = itemsWithMethods;
for (const it of selectedItems.value) {
if (getPackageCarrier(it)?.packageId) {
await loadPackageDetailsForItem(it);
}
it.expanded = !!getPackageCarrier(it)?.packageId;
}
syncCategoryChecked(); syncCategoryChecked();
// Bug #384修复: 回充后更新检查方法显示 // Bug #384修复: 回充后更新检查方法显示
updateMethodDisplay(); updateMethodDisplay();
@@ -1361,7 +1141,6 @@ async function handleMethodSelect(checked, method, cat) {
if (method.packageId) { if (method.packageId) {
existingItem.isPackage = true; existingItem.isPackage = true;
existingItem.packageId = method.packageId; existingItem.packageId = method.packageId;
existingItem.packageName = method.packageName || existingItem.packageName; // #428修复: 确保 packageName 同步
// 预加载套餐明细 // 预加载套餐明细
loadPackageDetailsForItem(existingItem); loadPackageDetailsForItem(existingItem);
} }
@@ -1389,13 +1168,12 @@ async function handleMethodSelect(checked, method, cat) {
checkType: cat.typeName, checkType: cat.typeName,
nationalCode: targetItem.nationalCode || '', nationalCode: targetItem.nationalCode || '',
checked: true, checked: true,
methods: cat.methods || [method], // #428修复: 复制分类下全部方法,允许用户切换 methods: [method],
selectedMethod: method, selectedMethod: method,
expanded: false, expanded: false,
// 从方法或项目中获取套餐信息 // 从方法中获取套餐信息(优先级高于项目本身的 packageName
isPackage: !!method.packageId || !!targetItem.packageName, isPackage: !!method.packageId || !!targetItem.packageName,
packageId: method.packageId || targetItem.packageId || null, packageId: method.packageId || targetItem.packageId || null
packageName: method.packageName || targetItem.packageName || null // #428修复: 复制 packageName确保套餐明细可加载
}; };
selectedItems.value.push(newItem); selectedItems.value.push(newItem);
@@ -1468,7 +1246,7 @@ async function handleItemSelect(checked, item, cat) {
} }
} }
const newRow = { selectedItems.value.push({
id: item.id, name: item.name, id: item.id, name: item.name,
price: item.price, quantity: 1, price: item.price, quantity: 1,
serviceFee: item.serviceFee || 0, serviceFee: item.serviceFee || 0,
@@ -1479,27 +1257,11 @@ async function handleItemSelect(checked, item, cat) {
checked: true, checked: true,
methods: methods, methods: methods,
selectedMethod: null, selectedMethod: null,
expanded: false, expanded: false, // Bug #384修复: 新增展开状态,默认不展开
isPackage: !!(item.packageId || item.packageName), isPackage: !!item.packageName, // Bug #428修复: 标记是否为套餐
packageName: item.packageName || null, packageName: item.packageName || null, // Bug #426修复: 套餐名称用于查找packageId
packageDetailsLoading: false, packageId: item.packageId || null // Bug #428修复: 套餐ID
packageId: item.packageId || null });
};
selectedItems.value.push(newRow);
// 必须用数组里的响应式行,不能继续改局部 newRowpush 后列表内是 proxy改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
const row = selectedItems.value[selectedItems.value.length - 1];
// 右侧不再展示「检查方法」列表:自动选默认方法(保存、计价仍依赖 selectedMethod
if (methods.length >= 1) {
row.selectedMethod = pickDefaultMethod(methods, item);
}
updateMethodDisplay();
// 有套餐 ID 时默认展开(先显示加载区,明细写入行对象 packageDetailsDisplay
row.expanded = !!getPackageCarrier(row)?.packageId;
if (getPackageCarrier(row)?.packageId) {
await loadPackageDetailsForItem(row);
}
// 自动回填执行科室:按检查项目类型 → 检查类型管理里配置的执行科室 // 自动回填执行科室:按检查项目类型 → 检查类型管理里配置的执行科室
if (selectedItems.value.length === 1 && cat?.performDeptName) { if (selectedItems.value.length === 1 && cat?.performDeptName) {
@@ -1520,26 +1282,19 @@ async function handleItemSelect(checked, item, cat) {
// Bug #382 修复:移除自动切换页签逻辑,保持当前页签状态 // Bug #382 修复:移除自动切换页签逻辑,保持当前页签状态
} }
// Bug #384修复 + #426修复: 展开/收起项目卡片 // Bug #384修复: 展开/收起项目卡片
async function toggleItemExpand(item) { async function toggleItemExpand(item) {
item.expanded = !item.expanded; item.expanded = !item.expanded;
if (item.expanded && (item.isPackage || item.packageName) && (!item.packageDetails || item.packageDetails.length === 0) && !item.packageDetailsLoading) { // 如果是展开且该项目是套餐,加载套餐明细
if (item.expanded && item.isPackage && (!item.packageDetails || item.packageDetails.length === 0)) {
await loadPackageDetailsForItem(item); await loadPackageDetailsForItem(item);
} }
if (item.expanded && shouldShowPackageBody(item)) {
if (getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
await loadPackageDetailsForItem(item);
}
}
} }
// Bug #384修复: 勾选框选择检查方法(单选逻辑) // Bug #384修复: 勾选框选择检查方法(单选逻辑)
async function selectMethodCheckbox(checked, item, method) { async function selectMethodCheckbox(checked, item, method) {
if (checked) { if (checked) {
item.selectedMethod = method; item.selectedMethod = method;
if (item.expanded && method.packageId) {
loadPackageDetailsForItem(item);
}
// 动态加载该方法对应的套餐明细 // 动态加载该方法对应的套餐明细
await loadMethodPackageDetails(item, method); await loadMethodPackageDetails(item, method);
} else { } else {
@@ -1599,28 +1354,6 @@ async function loadMethodPackageDetails(item, method) {
} }
} }
/** 检查明细表格中切换检查方法 */
async function onDetailMethodChange(row, val) {
row.selectedMethod = val || null;
if (val?.packageId) {
row.packageId = val.packageId;
row.isPackage = true;
}
row.packageDetailsDisplay = undefined;
const carrier = getPackageCarrier(row);
if (carrier) {
carrier.packageDetails = undefined;
}
updateMethodDisplay();
row.expanded = !!getPackageCarrier(row)?.packageId;
if (getPackageCarrier(row)?.packageId) {
await loadPackageDetailsForItem(row);
}
nextTick(() => {
form.totalAmount = totalAmountCalc.value;
});
}
// Bug #384修复: 更新检查方法显示字段(联动) // Bug #384修复: 更新检查方法显示字段(联动)
function updateMethodDisplay() { function updateMethodDisplay() {
// 找到第一个有选中检查方法的项目 // 找到第一个有选中检查方法的项目
@@ -1710,19 +1443,6 @@ defineExpose({ getList });
gap: 8px; gap: 8px;
} }
/* Bug #499: 查询过滤工具栏 */
.search-toolbar {
margin-bottom: 10px;
padding: 8px 0;
border-bottom: 1px solid #ebeef5;
}
.search-toolbar :deep(.el-form-item) {
margin-bottom: 8px;
}
.search-toolbar :deep(.el-form-item__label) {
font-size: 12px;
}
/* 底部区域:左表单 + 右分类 */ /* 底部区域:左表单 + 右分类 */
.bottom-section { .bottom-section {
display: flex; display: flex;
@@ -1765,7 +1485,7 @@ defineExpose({ getList });
/* 右:分类面板 */ /* 右:分类面板 */
.category-panel { .category-panel {
width: 420px; width: 380px;
flex-shrink: 0; flex-shrink: 0;
background: #fff; background: #fff;
border-radius: 4px; border-radius: 4px;
@@ -1800,7 +1520,6 @@ defineExpose({ getList });
flex: 1; flex: 1;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; /* Bug #500: 防止切换时水平方向溢出导致抖动 */ overflow-x: hidden; /* Bug #500: 防止切换时水平方向溢出导致抖动 */
min-height: 120px; /* Bug #500: 固定最小高度,避免分类切换时 flex 容器高度突变 */
} }
.empty-hint { .empty-hint {
color: #909399; color: #909399;
@@ -1901,11 +1620,8 @@ defineExpose({ getList });
} }
/* 已选择 tags */ /* 已选择 tags */
/* 已选择:加宽,避免套餐明细挤成一团 */
.selected-panel { .selected-panel {
width: 220px; width: 140px; /* Bug #384修复: 加宽以适应展开内容 */
min-width: 200px;
max-width: 280px;
flex-shrink: 0; flex-shrink: 0;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -1913,11 +1629,9 @@ defineExpose({ getList });
.selected-tags { .selected-tags {
flex: 1; flex: 1;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 8px; gap: 6px;
padding-right: 2px;
} }
.selected-tag { .selected-tag {
max-width: 100%; max-width: 100%;
@@ -1930,40 +1644,30 @@ defineExpose({ getList });
font-size: 12px; font-size: 12px;
} }
/* 已选择项目卡片 */ /* Bug #384修复: 已选择项目卡片(可展开) */
.selected-item-card { .selected-item-card {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
background: #fff; background: #F5F5F5;
border-radius: 6px; border-radius: 4px;
border: 1px solid #e4e7ed; border: 1px solid #e4e7ed;
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.04);
overflow: hidden;
} }
.selected-item-card .card-header { .selected-item-card .card-header {
display: flex; display: flex;
align-items: center; align-items: center;
padding: 10px 10px; padding: 8px 10px;
cursor: pointer; cursor: pointer;
gap: 8px; gap: 4px;
background: linear-gradient(180deg, #f8fafc 0%, #f0f4f8 100%);
border-bottom: 1px solid transparent;
} }
.selected-item-card .card-header:hover { .selected-item-card .card-header:hover {
background: linear-gradient(180deg, #ecf5ff 0%, #e3eef8 100%); background: #E6F7FF;
}
.selected-item-card.is-expanded .card-header {
border-bottom-color: #ebeef5;
} }
.card-name { .card-name {
flex: 1; flex: 1;
min-width: 0; font-size: 12px;
font-size: 13px;
font-weight: 500;
color: #303133; color: #303133;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
@@ -1971,23 +1675,19 @@ defineExpose({ getList });
} }
.card-price { .card-price {
font-size: 13px; font-size: 12px;
color: #409eff; color: #1890FF;
font-weight: 600; font-weight: 500;
flex-shrink: 0;
} }
.expand-icon { .expand-icon {
font-size: 14px; font-size: 12px;
color: #909399; color: #909399;
transition: transform 0.2s ease;
flex-shrink: 0;
transition: transform 0.2s; transition: transform 0.2s;
transform: rotate(0deg);
} }
.expand-icon.expanded { .expand-icon.expanded {
transform: rotate(90deg); transform: rotate(180deg);
} }
/* Bug #428修复: 套餐明细列表样式 */ /* Bug #428修复: 套餐明细列表样式 */
@@ -2025,83 +1725,61 @@ defineExpose({ getList });
white-space: nowrap; white-space: nowrap;
} }
/* 展开区域 */ /* Bug #384修复: 检查方法勾选框列表 */
.selected-card-body { .method-list {
background: #fafbfc; padding: 6px 10px;
} background: #fff;
border-top: 1px solid #e4e7ed;
.package-details-loading,
.package-details-empty {
padding: 12px 10px;
font-size: 12px;
color: #909399;
text-align: center;
}
.package-details-empty {
color: #c0c4cc;
}
.package-details-list {
padding: 10px 10px 12px;
}
.package-details-head {
font-size: 11px;
font-weight: 600;
color: #909399;
letter-spacing: 0.02em;
margin-bottom: 8px;
padding-bottom: 6px;
border-bottom: 1px dashed #dcdfe6;
}
.detail-row {
display: grid;
grid-template-columns: minmax(0, 1fr) auto;
gap: 8px 12px;
align-items: start;
padding: 10px 0;
border-bottom: 1px solid #ebeef5;
}
.detail-row:last-of-type {
border-bottom: none;
padding-bottom: 2px;
}
.detail-name {
font-size: 12px;
color: #303133;
line-height: 1.5;
word-break: break-word;
display: -webkit-box;
-webkit-box-orient: vertical;
-webkit-line-clamp: 2;
line-clamp: 2;
overflow: hidden;
}
.detail-meta {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
align-items: flex-end;
gap: 4px; gap: 4px;
flex-shrink: 0;
text-align: right;
} }
.detail-qty { .method-option {
display: flex;
align-items: center;
}
.method-option :deep(.el-checkbox__label) {
display: flex;
align-items: center;
justify-content: space-between;
width: 100%;
}
.method-option .method-name {
font-size: 11px; font-size: 11px;
color: #909399; color: #606266;
font-variant-numeric: tabular-nums;
} }
.detail-price { .method-option .method-price {
font-size: 12px; font-size: 11px;
font-weight: 600;
color: #e6a23c; color: #e6a23c;
font-variant-numeric: tabular-nums; font-weight: 500;
margin-left: 8px;
}
/* 选中方法后显示的套餐明细 */
.method-package-details {
margin-top: 4px;
padding: 4px 0;
border-top: 1px dashed #dcdfe6;
}
.method-package-header {
padding: 2px 0 4px 24px;
}
.method-package-title {
font-size: 10px;
color: #909399;
font-weight: 500;
}
.method-package-loading {
padding: 4px 0 4px 24px;
font-size: 10px;
color: #c0c4cc;
} }
/* 折叠组件细节 */ /* 折叠组件细节 */
@@ -2114,10 +1792,10 @@ defineExpose({ getList });
height: auto; height: auto;
line-height: 1.5; line-height: 1.5;
} }
/* Bug #500修复: 折叠内容使用明确属性过渡,避免 transition: all 导致子元素意外动画 */ /* Bug #500: 折叠内容添加平滑过渡动画,避免切换时高度跳变 */
:deep(.el-collapse-item__content) { :deep(.el-collapse-item__content) {
padding-bottom: 4px; padding-bottom: 4px;
transition: height 0.3s ease, max-height 0.3s ease; transition: all 0.3s ease;
} }
/* Bug #500: 折叠面板动画容器,添加 overflow:hidden 防止展开时内容溢出导致闪烁 */ /* Bug #500: 折叠面板动画容器,添加 overflow:hidden 防止展开时内容溢出导致闪烁 */
:deep(.el-collapse-item__wrap) { :deep(.el-collapse-item__wrap) {

View File

@@ -56,13 +56,6 @@
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="申请单号" prop="applyNo" min-width="160" align="center" header-align="center" /> <el-table-column label="申请单号" prop="applyNo" min-width="160" align="center" header-align="center" />
<el-table-column label="单据状态" prop="applyStatus" width="100" align="center" header-align="center">
<template #default="scope">
<el-tag :type="getStatusType(scope.row.applyStatus)" size="small">
{{ getStatusLabel(scope.row.applyStatus, scope.row) }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="检验项目" prop="itemName" min-width="170px" align="center" header-align="center"> <el-table-column label="检验项目" prop="itemName" min-width="170px" align="center" header-align="center">
<template #default="scope"> <template #default="scope">
<span>{{ scope.row.itemName }}</span> <span>{{ scope.row.itemName }}</span>
@@ -1452,26 +1445,6 @@ const formatAmount = (amount) => {
return num.toFixed(2) return num.toFixed(2)
} }
// 单据状态标签文字
const getStatusLabel = (applyStatus, row) => {
// applyStatus: 0=待开立, 1=已开立(已签发)
// 结合收费/执行标记推导更丰富的状态
if (applyStatus === 1) {
// 已收费后根据执行标记判断
if (row.needExecute === true) {
return '已执行'
}
return '已开立'
}
return '待开立'
}
// 单据状态标签颜色
const getStatusType = (applyStatus) => {
if (applyStatus === 1) return 'success'
return 'info'
}
// 格式化日期时间为字符串 YYYY-MM-DD HH:mm:ss // 格式化日期时间为字符串 YYYY-MM-DD HH:mm:ss
const formatDateTime = (date) => { const formatDateTime = (date) => {
if (!date) return '' if (!date) return ''

View File

@@ -589,46 +589,36 @@ function handleUseOrderGroup(row) {
minUnitPrice: orderDetail.minUnitPrice, minUnitPrice: orderDetail.minUnitPrice,
inventoryList: orderDetail.inventoryList || [], inventoryList: orderDetail.inventoryList || [],
priceList: orderDetail.priceList || [], priceList: orderDetail.priceList || [],
partPercent: orderDetail.partPercent ?? 1, partPercent: orderDetail.partPercent || 1,
partAttributeEnum: orderDetail.partAttributeEnum,
unitConversionRatio: orderDetail.unitConversionRatio,
// 🔧 Bug #218 修复positionId 可能存储在 item 本身,优先使用 item.positionId // 🔧 Bug #218 修复positionId 可能存储在 item 本身,优先使用 item.positionId
positionId: item.positionId ?? orderDetail.positionId, positionId: item.positionId || orderDetail.positionId,
defaultLotNumber: orderDetail.defaultLotNumber, defaultLotNumber: orderDetail.defaultLotNumber,
// 单位信息 // 单位信息
unitCode: item.unitCode ?? orderDetail.unitCode, unitCode: item.unitCode || orderDetail.unitCode,
categoryCode: item.categoryCode ?? orderDetail.categoryCode,
unitCodeName: item.unitCodeName || orderDetail.unitCode_dictText, unitCodeName: item.unitCodeName || orderDetail.unitCode_dictText,
minUnitCode: orderDetail.minUnitCode, minUnitCode: orderDetail.minUnitCode,
doseUnitCode: orderDetail.doseUnitCode, doseUnitCode: orderDetail.doseUnitCode,
// 合并后的完整对象(用于 setValue // 合并后的完整对象(用于 setValue
// 先展开 orderDetail 获取所有药品基础字段categoryCode、minUnitCode、doseUnitCode、
// partPercent、partAttributeEnum、unitConversionRatio、defaultLotNumber 等),
// 再用组套用户覆盖值覆盖,确保单次剂量/频次/用法/用药天数/总量等不被丢失
mergedDetail: { mergedDetail: {
...orderDetail, ...orderDetail,
adviceName: orderDetail.adviceName || item.orderDefinitionName || '未知项目', adviceName: orderDetail.adviceName || item.orderDefinitionName || '未知项目',
adviceType: orderDetail.adviceType, adviceType: orderDetail.adviceType,
quantity: item.quantity, quantity: item.quantity,
unitCode: item.unitCode ?? orderDetail.unitCode, unitCode: item.unitCode || orderDetail.unitCode,
categoryCode: item.categoryCode ?? orderDetail.categoryCode,
unitCodeName: item.unitCodeName, unitCodeName: item.unitCodeName,
dose: item.dose ?? orderDetail.dose, dose: item.dose || orderDetail.dose,
rateCode: item.rateCode ?? orderDetail.rateCode, rateCode: item.rateCode || orderDetail.rateCode,
methodCode: item.methodCode ?? orderDetail.methodCode, methodCode: item.methodCode || orderDetail.methodCode,
dispensePerDuration: item.dispensePerDuration ?? orderDetail.dispensePerDuration, dispensePerDuration: item.dispensePerDuration || orderDetail.dispensePerDuration,
doseQuantity: item.doseQuantity ?? orderDetail.doseQuantity, doseQuantity: item.doseQuantity,
// 🔧 Bug #218 / #403 修复positionId 可能存储在 item 本身,优先使用 item.positionId inventoryList: orderDetail.inventoryList || [],
positionId: item.positionId ?? orderDetail.positionId, priceList: orderDetail.priceList || [],
// 执行科室:优先使用组套明细中保存的 orgId partPercent: orderDetail.partPercent || 1,
orgId: item.orgId ?? orderDetail.orgId, // 🔧 Bug #218 修复positionId 可能存储在 item 本身,优先使用 item.positionId
orgName: item.orgName ?? orderDetail.orgName, positionId: item.positionId || orderDetail.positionId,
// 组号(保留组套中的分组信息) defaultLotNumber: orderDetail.defaultLotNumber,
groupId: item.groupId,
groupOrder: item.groupOrder,
therapyEnum: item.therapyEnum ?? orderDetail.therapyEnum ?? '1',
} }
}; };
}); });

View File

@@ -315,7 +315,6 @@
data-prop="dispensePerDuration"> data-prop="dispensePerDuration">
<el-input-number v-model="scope.row.dispensePerDuration" style="width: 80px" :min="1" <el-input-number v-model="scope.row.dispensePerDuration" style="width: 80px" :min="1"
controls-position="right" :controls="false" :ref="(el) => (inputRefs.dispensePerDuration = el)" controls-position="right" :controls="false" :ref="(el) => (inputRefs.dispensePerDuration = el)"
@input="calculateTotalAmount(scope.row, scope.$index)"
@change="calculateTotalAmount(scope.row, scope.$index)" @change="calculateTotalAmount(scope.row, scope.$index)"
@keyup.enter.prevent=" @keyup.enter.prevent="
handleEnter('dispensePerDuration', scope.row, scope.$index) handleEnter('dispensePerDuration', scope.row, scope.$index)
@@ -875,7 +874,7 @@ import { ArrowDown, Search, Memo, Minus, Plus, Edit, Delete } from '@element-plu
import printUtils, { getPrinterList, PRINT_TEMPLATE, savePrinterToCache, } from '@/utils/printUtils'; import printUtils, { getPrinterList, PRINT_TEMPLATE, savePrinterToCache, } from '@/utils/printUtils';
import Template from "@/views/inpatientDoctor/home/emr/components/template.vue"; import Template from "@/views/inpatientDoctor/home/emr/components/template.vue";
const emit = defineEmits(['selectDiagnosis', 'inspectionListRefresh']); const emit = defineEmits(['selectDiagnosis']);
const total = ref(0); const total = ref(0);
const queryParams = ref({}); const queryParams = ref({});
const prescriptionList = ref([]); const prescriptionList = ref([]);
@@ -2084,21 +2083,6 @@ function getOrgList() {
}); });
} }
/** 诊疗医嘱关联检验申请时 contentJson 含 applyNo */
function getInspectionApplyNoFromAdviceRow(row) {
if (!row || row.adviceType !== 3) {
return null;
}
try {
const raw = row.contentJson;
const j = raw ? (typeof raw === 'string' ? JSON.parse(raw) : raw) : {};
const no = j && j.applyNo != null ? String(j.applyNo).trim() : '';
return no || null;
} catch (e) {
return null;
}
}
function handleDelete() { function handleDelete() {
let selectRows = prescriptionRef.value.getSelectionRows(); let selectRows = prescriptionRef.value.getSelectionRows();
console.log('BugFix#219: handleDelete called, selectRows=', selectRows); console.log('BugFix#219: handleDelete called, selectRows=', selectRows);
@@ -2278,31 +2262,12 @@ function handleDelete() {
} }
if (deleteList.length > 0) { if (deleteList.length > 0) {
const hasLabLinked = deleteList.some((d) => { savePrescription({ adviceSaveList: deleteList }).then((res) => {
const row = normalRows.find((r) => r.requestId === d.requestId); if (res.code == 200) {
return row && getInspectionApplyNoFromAdviceRow(row); proxy.$modal.msgSuccess('删除成功');
getListInfo(false);
}
}); });
const runApiDelete = () => {
savePrescription({ adviceSaveList: deleteList }).then((res) => {
if (res.code == 200) {
proxy.$modal.msgSuccess('删除成功');
getListInfo(false);
emit('inspectionListRefresh');
}
});
};
if (hasLabLinked) {
proxy.$modal
.confirm(
'删除此医嘱将同时作废关联的检验申请单(检验页签中的同单申请及同单下相关医嘱)。是否继续?',
'删除确认',
{ type: 'warning' }
)
.then(runApiDelete)
.catch(() => {});
} else {
runApiDelete();
}
} else if (consultationRows.length == 0) { } else if (consultationRows.length == 0) {
proxy.$modal.msgWarning('所选医嘱不可删除,请先撤回后再删除'); proxy.$modal.msgWarning('所选医嘱不可删除,请先撤回后再删除');
return; return;
@@ -3453,12 +3418,7 @@ async function setValue(row) {
console.log('[BugFix] setValue - prescriptionList[rowIndex].adviceType_dictText:', prescriptionList.value[rowIndex.value].adviceType_dictText); console.log('[BugFix] setValue - prescriptionList[rowIndex].adviceType_dictText:', prescriptionList.value[rowIndex.value].adviceType_dictText);
// 🔧 Bug #455: 诊疗医嘱(adviceType=3)的执行科室默认使用患者就诊科室, // 🔧 Bug #455: 诊疗医嘱(adviceType=3)的执行科室默认使用患者就诊科室,
// 不使用positionId(诊疗目录配置的执行科室)避免配置ID不在机构树中导致显示原始ID // 不使用positionId(诊疗目录配置的执行科室)避免配置ID不在机构树中导致显示原始ID
if (Number(row.adviceType) == 3) { if (Number(row.adviceType) != 3) {
// 覆盖 catalog 传来的 positionId/orgId使用患者就诊科室
prescriptionList.value[rowIndex.value].orgId = props.patientInfo?.orgId;
prescriptionList.value[rowIndex.value].positionId = props.patientInfo?.orgId;
prescriptionList.value[rowIndex.value].positionName = findOrgNameById(props.patientInfo?.orgId) || props.patientInfo?.orgName || '';
} else {
prescriptionList.value[rowIndex.value].orgId = row.positionId || props.patientInfo?.orgId; prescriptionList.value[rowIndex.value].orgId = row.positionId || props.patientInfo?.orgId;
} }
prescriptionList.value[rowIndex.value].dose = row.dose || row.doseQuantity; prescriptionList.value[rowIndex.value].dose = row.dose || row.doseQuantity;
@@ -3591,10 +3551,13 @@ async function setValue(row) {
prescriptionList.value[rowIndex.value].categoryEnum = 31; // 会诊的category_enum设置为31 prescriptionList.value[rowIndex.value].categoryEnum = 31; // 会诊的category_enum设置为31
} else { } else {
// 诊疗类型adviceType == 3 // 诊疗类型adviceType == 3
// 🔧 Bug #455: 诊疗项目执行科室强制使用患者就诊科室 // 🔧 Bug Fix #238: 诊疗项目默认使用患者就诊科室
// 不使用目录配置的执行科室可能是错误ID或占位符导致显示原始ID if (!prescriptionList.value[rowIndex.value].orgId) {
prescriptionList.value[rowIndex.value].orgId = props.patientInfo.orgId; prescriptionList.value[rowIndex.value].orgId = props.patientInfo.orgId;
prescriptionList.value[rowIndex.value].positionName = findOrgNameById(props.patientInfo.orgId) || props.patientInfo.orgName || ''; }
if (!prescriptionList.value[rowIndex.value].positionName) {
prescriptionList.value[rowIndex.value].positionName = findOrgNameById(prescriptionList.value[rowIndex.value].orgId) || props.patientInfo.orgName || '';
}
// 🔧 Bug #218 修复使用组套中维护的quantity如果没有则默认1 // 🔧 Bug #218 修复使用组套中维护的quantity如果没有则默认1
prescriptionList.value[rowIndex.value].quantity = row.quantity || 1; prescriptionList.value[rowIndex.value].quantity = row.quantity || 1;
// 🔧 Bug #144 修复:安全访问 priceList防止 orderDetailInfos 为空时出错 // 🔧 Bug #144 修复:安全访问 priceList防止 orderDetailInfos 为空时出错
@@ -3689,10 +3652,7 @@ function handleSaveGroup(orderGroupList) {
unitCode_dictText: item.unitCodeName || '', unitCode_dictText: item.unitCodeName || '',
statusEnum: 1, statusEnum: 1,
// 🔧 修复执行科室逻辑:优先使用 orgId(所属科室),其次 positionId // 🔧 修复执行科室逻辑:优先使用 orgId(所属科室),其次 positionId
// 🔧 Bug #455: 诊疗类(adviceType=3)使用患者就诊科室不使用目录配置的ID orgId: item.orderDetailInfos?.orgId || mergedDetail.orgId || item.positionId || item.orderDetailInfos?.positionId || mergedDetail.positionId,
orgId: item.adviceType === 3
? props.patientInfo?.orgId
: (item.orderDetailInfos?.orgId || mergedDetail.orgId || item.positionId || item.orderDetailInfos?.positionId || mergedDetail.positionId),
dbOpType: prescriptionList.value[rowIndex.value].requestId ? '2' : '1', dbOpType: prescriptionList.value[rowIndex.value].requestId ? '2' : '1',
conditionId: conditionId.value, conditionId: conditionId.value,
conditionDefinitionId: conditionDefinitionId.value, conditionDefinitionId: conditionDefinitionId.value,

View File

@@ -626,19 +626,14 @@ function getList() {
loading.value = false loading.value = false
return return
} }
loading.value = true loading.value = true
getSurgeryPage({ getSurgeryPage({
pageNo: 1, pageNo: 1,
pageSize: 100, pageSize: 100,
encounterId: props.patientInfo.encounterId encounterId: props.patientInfo.encounterId
}).then((res) => { }).then((res) => {
if (res.code === 200) { surgeryList.value = res.data.records || []
surgeryList.value = res.data?.records || []
} else {
proxy.$modal.msgError(res.msg || '数据加载失败,请稍后重试')
surgeryList.value = []
}
}).catch(error => { }).catch(error => {
console.error('获取手术列表失败:', error) console.error('获取手术列表失败:', error)
proxy.$modal.msgError('数据加载失败,请稍后重试') proxy.$modal.msgError('数据加载失败,请稍后重试')
@@ -1138,12 +1133,13 @@ function submitForm() {
// 新增手术 // 新增手术
addSurgery(form.value).then((res) => { addSurgery(form.value).then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$modal.msgSuccess('手术申请提交成功!') proxy.$modal.msgSuccess(res.msg || '手术申请提交成功!')
// 保存麻醉方式 // 保存麻醉方式
sessionStorage.setItem('anesthesiaType', form.value.anesthesiaTypeEnum) sessionStorage.setItem('anesthesiaType', form.value.anesthesiaTypeEnum)
open.value = false open.value = false
getList() // 提交成功后直接刷新列表
emit('saved') // 通知父组件刷新医嘱列表 emit('saved') // 通知父组件刷新医嘱列表
// 刷新手术申请列表
getList()
} else { } else {
proxy.$modal.msgError(res.msg || '新增手术失败,请检查表单信息') proxy.$modal.msgError(res.msg || '新增手术失败,请检查表单信息')
} }
@@ -1155,12 +1151,13 @@ function submitForm() {
// 修改手术 // 修改手术
updateSurgery(form.value).then((res) => { updateSurgery(form.value).then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$modal.msgSuccess('手术申请修改成功!') proxy.$modal.msgSuccess(res.msg || '手术申请修改成功!')
// 保存麻醉方式 // 保存麻醉方式
sessionStorage.setItem('anesthesiaType', form.value.anesthesiaTypeEnum) sessionStorage.setItem('anesthesiaType', form.value.anesthesiaTypeEnum)
open.value = false open.value = false
getList() // 修改成功后直接刷新列表
emit('saved') // 通知父组件刷新医嘱列表 emit('saved') // 通知父组件刷新医嘱列表
// 刷新手术申请列表
getList()
} else { } else {
proxy.$modal.msgError(res.msg || '更新手术失败,请检查表单信息') proxy.$modal.msgError(res.msg || '更新手术失败,请检查表单信息')
} }

View File

@@ -138,8 +138,7 @@
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="医嘱" name="prescription"> <el-tab-pane label="医嘱" name="prescription">
<prescriptionlist :patientInfo="patientInfo" ref="prescriptionRef" :activeTab="activeTab" <prescriptionlist :patientInfo="patientInfo" ref="prescriptionRef" :activeTab="activeTab"
:outpatientEmrSaved="outpatientEmrSaved" :outpatientEmrSaved="outpatientEmrSaved" />
@inspectionListRefresh="refreshInspectionListFromAdvice" />
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="中医" name="tcm"> <el-tab-pane label="中医" name="tcm">
<tcmAdvice :patientInfo="patientInfo" ref="tcmRef" /> <tcmAdvice :patientInfo="patientInfo" ref="tcmRef" />
@@ -152,8 +151,8 @@
@saved="() => prescriptionRef?.getListInfo()" /> @saved="() => prescriptionRef?.getListInfo()" />
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="手术申请" name="surgery"> <el-tab-pane label="手术申请" name="surgery">
<surgeryApplication :patientInfo="patientInfo" :activeTab="activeTab" ref="surgeryRef" <surgeryApplication :patientInfo="patientInfo" :activeTab="activeTab" ref="surgeryRef"
@saved="() => { prescriptionRef?.getListInfo(); surgeryRef?.getList() }" /> @saved="() => prescriptionRef?.getListInfo()" />
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="电子处方" name="eprescription"> <el-tab-pane label="电子处方" name="eprescription">
<eprescriptionlist :patientInfo="patientInfo" ref="eprescriptionRef" /> <eprescriptionlist :patientInfo="patientInfo" ref="eprescriptionRef" />
@@ -313,9 +312,6 @@ const patientDrawerRef = ref();
const prescriptionRef = ref(); const prescriptionRef = ref();
const tcmRef = ref(); const tcmRef = ref();
const inspectionRef = ref(); const inspectionRef = ref();
function refreshInspectionListFromAdvice() {
inspectionRef.value?.getList?.();
}
const examinationRef = ref(); const examinationRef = ref();
const surgeryRef = ref(); const surgeryRef = ref();
const emrRef = ref(); const emrRef = ref();

View File

@@ -51,7 +51,7 @@ const currentSelectRow = ref<any>({});
const queryParams = ref({ const queryParams = ref({
pageSize: 100, pageSize: 100,
pageNo: 1, pageNo: 1,
adviceTypes: [1, 2, 3, 6], adviceTypes: '1,2,3,6',
searchKey: '', searchKey: '',
organizationId: '', organizationId: '',
categoryCode: '', categoryCode: '',
@@ -88,10 +88,10 @@ const tableColumns = computed<TableColumn[]>(() => [
function refresh(adviceType: any, categoryCode: string, searchKey: string) { function refresh(adviceType: any, categoryCode: string, searchKey: string) {
// 有搜索词时跨类型搜索,避免用户输入"级护理"但因当前adviceType为药品而搜不到诊疗类护理项目 // 有搜索词时跨类型搜索,避免用户输入"级护理"但因当前adviceType为药品而搜不到诊疗类护理项目
if (searchKey) { if (searchKey) {
queryParams.value.adviceTypes = [1, 2, 3, 6]; queryParams.value.adviceTypes = '1,2,3,6';
} else { } else {
queryParams.value.adviceTypes = queryParams.value.adviceTypes =
adviceType !== undefined && adviceType !== '' ? [parseInt(adviceType)] : [1, 2, 3, 6]; adviceType !== undefined && adviceType !== '' ? String(adviceType) : '1,2,3,6';
} }
queryParams.value.categoryCode = categoryCode || ''; queryParams.value.categoryCode = categoryCode || '';
queryParams.value.searchKey = searchKey || ''; queryParams.value.searchKey = searchKey || '';
@@ -131,8 +131,7 @@ function getList() {
} }
}); });
}) })
.catch((err) => { .catch(() => {
console.warn('医嘱基础信息加载失败:', err);
adviceBaseList.value = []; adviceBaseList.value = [];
}) })
.finally(() => { .finally(() => {

View File

@@ -49,15 +49,6 @@
<el-option label="已作废" value="7" /> <el-option label="已作废" value="7" />
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item label="关键字">
<el-input
v-model="filterForm.keyword"
placeholder="申请单号 / 检查项目名称"
clearable
style="width: 220px"
@keyup.enter="handleSearch"
/>
</el-form-item>
<el-form-item> <el-form-item>
<el-button type="primary" @click="handleSearch" :loading="loading"> <el-button type="primary" @click="handleSearch" :loading="loading">
<el-icon><Search /></el-icon> <el-icon><Search /></el-icon>
@@ -86,53 +77,18 @@
</template> </template>
<el-table-column type="index" label="序号" width="60" align="center" /> <el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="patientName" label="患者姓名" width="120" /> <el-table-column prop="patientName" label="患者姓名" width="120" />
<el-table-column label="申请单名称" width="140"> <el-table-column prop="name" label="申请单名称" width="140" />
<template #default="scope">
<span>{{ buildApplicationName(scope.row) }}</span>
</template>
</el-table-column>
<el-table-column prop="createTime" label="创建时间" width="160" /> <el-table-column prop="createTime" label="创建时间" width="160" />
<el-table-column prop="prescriptionNo" label="申请单号" width="140" /> <el-table-column prop="prescriptionNo" label="申请单号" width="140" />
<el-table-column prop="requesterId_dictText" label="申请者" width="120" /> <el-table-column prop="requesterId_dictText" label="申请者" width="120" />
<el-table-column label="申请单状态" width="120" align="center"> <el-table-column label="申请单状态" width="120" align="center">
<template #default="scope"> <template #default="scope">
<el-tag :type="getStatusTagType(scope.row.status)" effect="plain" round> <span>{{ parseStatus(scope.row.status) }}</span>
{{ parseStatus(scope.row.status) }}
</el-tag>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="操作" width="280" align="center" fixed="right"> <el-table-column label="操作" align="center" fixed="right">
<template #default="scope"> <template #default="scope">
<!-- 详情 - 所有状态都显示 -->
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button> <el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
<!-- 待签发修改删除 -->
<template v-if="scope.row.status === '0' || scope.row.status === 0">
<el-button link type="primary" @click="handleEdit(scope.row)">修改</el-button>
<el-button link type="danger" @click="handleDelete(scope.row)">删除</el-button>
</template>
<!-- 已签发撤回 -->
<template v-else-if="scope.row.status === '1' || scope.row.status === 1">
<el-button link type="warning" @click="handleRecall(scope.row)">撤回</el-button>
</template>
<!-- 已校对/待接收打印 -->
<template v-else-if="scope.row.status === '2' || scope.row.status === 2 || scope.row.status === '3' || scope.row.status === 3">
<el-button link type="primary" @click="handlePrint(scope.row)">打印</el-button>
</template>
<!-- 已接收/已检查看报告 -->
<template v-else-if="scope.row.status === '4' || scope.row.status === 4 || scope.row.status === '5' || scope.row.status === 5">
<el-button link type="success" @click="handleViewReport(scope.row)">看报告</el-button>
</template>
<!-- 已出报告打印看报告 -->
<template v-else-if="scope.row.status === '6' || scope.row.status === 6">
<el-button link type="primary" @click="handlePrint(scope.row)">打印</el-button>
<el-button link type="success" @click="handleViewReport(scope.row)">看报告</el-button>
</template>
<!-- 已作废无额外按钮 -->
<template v-else-if="scope.row.status === '7' || scope.row.status === 7">
</template>
<!-- 其他/未知状态仅详情 -->
<template v-else>
</template>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
@@ -181,7 +137,7 @@
<el-descriptions title="申请单描述" :column="2"> <el-descriptions title="申请单描述" :column="2">
<template v-for="(value, key) in descJsonData" :key="key"> <template v-for="(value, key) in descJsonData" :key="key">
<el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)"> <el-descriptions-item v-if="isFieldMatched(key)" :label="getFieldLabel(key)">
{{ transformField(key, value) || '-' }} {{ value || '-' }}
</el-descriptions-item> </el-descriptions-item>
</template> </template>
</el-descriptions> </el-descriptions>
@@ -204,102 +160,15 @@
<el-button @click="detailDialogVisible = false">关闭</el-button> <el-button @click="detailDialogVisible = false">关闭</el-button>
</template> </template>
</el-dialog> </el-dialog>
<!-- 修改申请单弹窗 - 复用检查申请单组件 -->
<el-dialog
v-model="editDialogVisible"
title="修改检查申请"
width="1200px"
destroy-on-close
top="5vh"
:close-on-click-modal="false"
>
<MedicalExaminations
v-if="editDialogVisible"
ref="editFormRef"
:is-edit-mode="true"
:edit-data="editingRow"
:external-patient-info="patientInfo"
/>
<template #footer>
<el-button @click="editDialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleEditSubmit">确认</el-button>
</template>
</el-dialog>
<!-- 查看报告弹窗 -->
<el-dialog
v-model="reportDialogVisible"
:title="'检查报告 - ' + (reportData?.prescriptionNo || '')"
width="900px"
destroy-on-close
top="5vh"
:close-on-click-modal="false"
>
<div v-loading="reportLoading" class="report-viewer">
<!-- 报告基本信息 -->
<div v-if="reportData" class="report-viewer-container">
<el-descriptions title="报告信息" :column="2" border size="small">
<el-descriptions-item label="患者姓名">{{ reportData.patientName || reportRow?.patientName || '-' }}</el-descriptions-item>
<el-descriptions-item label="申请单号">{{ reportData.prescriptionNo || reportRow?.prescriptionNo || '-' }}</el-descriptions-item>
<el-descriptions-item label="申请单名称">{{ reportData.name || reportRow?.name || '-' }}</el-descriptions-item>
<el-descriptions-item label="报告时间">{{ reportData.reportTime || '-' }}</el-descriptions-item>
<el-descriptions-item label="诊断意见" :span="2">{{ reportData.diagnosis || reportData.conclusion || '-' }}</el-descriptions-item>
</el-descriptions>
<!-- 报告详细内容 -->
<div v-if="reportData.content" class="report-content-section">
<div class="section-title">报告内容</div>
<div class="report-content-text">{{ reportData.content }}</div>
</div>
<!-- 影像预览 - PACS链接 -->
<div v-if="reportData.imageUrl || reportData.pacsUrl" class="report-image-section">
<div class="section-title">
影像预览
<el-button
v-if="reportData.imageUrl || reportData.pacsUrl"
type="primary"
size="small"
plain
style="margin-left: 12px;"
@click="openPacsLink"
>
<el-icon><Link /></el-icon>
打开PACS影像
</el-button>
</div>
<iframe
v-if="reportData.imageUrl"
:src="reportData.imageUrl"
class="report-iframe"
frameborder="0"
/>
<el-empty v-else-if="!reportData.imageUrl && reportData.pacsUrl" description="点击上方按钮打开PACS影像" :image-size="60" />
</div>
<!-- 完全无数据时的提示 -->
<el-empty v-if="!reportData.content && !reportData.imageUrl && !reportData.pacsUrl" description="暂无详细报告数据" :image-size="60" />
</div>
<!-- 未获取到报告 -->
<el-empty v-else description="暂未生成报告" :image-size="80" />
</div>
<template #footer>
<el-button @click="reportDialogVisible = false">关闭</el-button>
</template>
</el-dialog>
</div> </div>
</template> </template>
<script setup> <script setup>
import {computed, getCurrentInstance, ref, watch, nextTick} from 'vue'; import {computed, getCurrentInstance, ref, watch} from 'vue';
import {Refresh, Search, Link} from '@element-plus/icons-vue'; import {Refresh, Search} from '@element-plus/icons-vue';
import {patientInfo} from '../../store/patient.js'; import {patientInfo} from '../../store/patient.js';
import {getCheck, deleteRequestForm, withdrawRequestForm, getTestResult} from './api'; import {getCheck} from './api';
import {getDepartmentList} from '@/api/public.js'; import {getDepartmentList} from '@/api/public.js';
import {getApplicationList, saveCheckd} from '../order/applicationForm/api';
import MedicalExaminations from '../order/applicationForm/medicalExaminations.vue';
const { proxy } = getCurrentInstance(); const { proxy } = getCurrentInstance();
@@ -309,38 +178,11 @@ const detailDialogVisible = ref(false);
const currentDetail = ref(null); const currentDetail = ref(null);
const descJsonData = ref(null); const descJsonData = ref(null);
const orgOptions = ref([]); const orgOptions = ref([]);
const editForm = ref({
name: '',
targetDepartment: '',
symptom: '',
sign: '',
clinicalDiagnosis: '',
otherDiagnosis: '',
relatedResult: '',
attention: '',
examinationPurpose: '',
medicalHistorySummary: '',
});
// 报告弹窗相关
const reportDialogVisible = ref(false);
const reportLoading = ref(false);
const reportData = ref(null);
const reportRow = ref(null);
// 获取近7天的日期范围作为默认值
const getDefaultDateRange = () => {
const now = new Date();
const endDate = now.toISOString().split('T')[0];
const startDate = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000).toISOString().split('T')[0];
return [startDate, endDate];
};
// 筛选表单数据 // 筛选表单数据
const filterForm = ref({ const filterForm = ref({
dateRange: getDefaultDateRange(), // 默认近一周 dateRange: [], // [startDate, endDate]
status: '', // 申请单状态 status: '', // 申请单状态
keyword: '', // 关键字搜索
}); });
const fetchData = async () => { const fetchData = async () => {
@@ -365,18 +207,11 @@ const fetchData = async () => {
params.status = filterForm.value.status; params.status = filterForm.value.status;
} }
// 添加关键字搜索
if (filterForm.value.keyword && filterForm.value.keyword.trim()) {
params.keyword = filterForm.value.keyword.trim();
}
const res = await getCheck(params); const res = await getCheck(params);
if (res.code === 200 && res.data) { if (res.code === 200 && res.data) {
const raw = res.data?.records || res.data; const raw = res.data?.records || res.data;
const list = Array.isArray(raw) ? raw : [raw]; const list = Array.isArray(raw) ? raw : [raw];
console.log('API返回的原始数据:', JSON.stringify(list, null, 2));
tableData.value = list.filter(Boolean); tableData.value = list.filter(Boolean);
console.log('tableData设置后的第一条:', tableData.value[0]);
} else { } else {
tableData.value = []; tableData.value = [];
} }
@@ -408,9 +243,8 @@ const handleSearch = async () => {
* 重置按钮处理 * 重置按钮处理
*/ */
const handleReset = () => { const handleReset = () => {
filterForm.value.dateRange = getDefaultDateRange(); filterForm.value.dateRange = [];
filterForm.value.status = ''; filterForm.value.status = '';
filterForm.value.keyword = '';
fetchData(); fetchData();
}; };
@@ -433,51 +267,9 @@ const parseStatus = (status) => {
return statusMap[String(status)] || '-'; return statusMap[String(status)] || '-';
}; };
/**
* 根据申请单详情构建申请单名称
* 单一项目:显示项目名称
* 多个项目:显示首个项目名称+"等X项"
*/
const buildApplicationName = (row) => {
const details = row.requestFormDetailList;
if (!details || details.length === 0) {
return row.name || '-';
}
if (details.length === 1) {
return details[0].adviceName || row.name || '-';
}
const first = details[0];
return `${first.adviceName || ''}${details.length}`;
};
/**
* 获取状态标签类型 - 参考临床医嘱样式
* @param {string|number} status - 状态码
* @returns {string} el-tag type
*/
const getStatusTagType = (status) => {
const typeMap = {
'0': 'primary', // 待签发 - 蓝色
'1': 'success', // 已签发 - 绿色
'2': 'success', // 已校对 - 绿色
'3': 'primary', // 待接收 - 蓝色
'4': 'primary', // 已接收 - 蓝色
'5': 'success', // 已检查 - 绿色
'6': 'success', // 已出报告 - 绿色
'7': 'danger', // 已作废 - 红色
};
return typeMap[String(status)] || 'info';
};
const labelMap = { const labelMap = {
categoryType: '项目类别', categoryType: '项目类别',
targetDepartment: '发往科室', targetDepartment: '发往科室',
urgencyLevel: '紧急程度',
allergyHistory: '过敏史',
examinationPurpose: '检查目的',
expectedExaminationTime: '期望检查时间',
medicalHistorySummary: '病史摘要',
allergyConfirmed: '过敏确认',
symptom: '症状', symptom: '症状',
sign: '体征', sign: '体征',
clinicalDiagnosis: '临床诊断', clinicalDiagnosis: '临床诊断',
@@ -486,17 +278,6 @@ const labelMap = {
attention: '注意事项', attention: '注意事项',
}; };
// Fields that need value transformation before display
const transformField = (key, value) => {
if (key === 'urgencyLevel') {
return value === 'emergency' ? '急诊' : '普通';
}
if (key === 'allergyConfirmed') {
return value === true || value === 'true' ? '已口头确认' : '未确认';
}
return value;
};
const isFieldMatched = (key) => { const isFieldMatched = (key) => {
return key in labelMap; return key in labelMap;
}; };
@@ -511,68 +292,50 @@ const hasMatchedFields = computed(() => {
}); });
/** 查询科室 */ /** 查询科室 */
const getLocationInfo = async () => { const getLocationInfo = () => {
try { getDepartmentList().then((res) => {
const res = await getDepartmentList(); orgOptions.value = res.data || [];
orgOptions.value = Array.isArray(res.data) ? res.data : []; });
} catch (e) {
console.warn('科室列表加载失败:', e.message);
orgOptions.value = [];
}
};
// 递归查找树形科室节点
const findTreeItem = (list, id) => {
if (!list || list.length === 0) return null;
for (const item of list) {
if (item.id == id) return item;
if (item.children && item.children.length > 0) {
const found = findTreeItem(item.children, id);
if (found) return found;
}
}
return null;
}; };
const recursionFun = (targetDepartment) => { const recursionFun = (targetDepartment) => {
if (!targetDepartment) return '';
let name = ''; let name = '';
for (let index = 0; index < orgOptions.value.length; index++) { for (let index = 0; index < orgOptions.value.length; index++) {
const obj = orgOptions.value[index]; const obj = orgOptions.value[index];
if (obj.id == targetDepartment) { if (obj.id == targetDepartment) {
name = obj.name; name = obj.name;
break;
} }
const subObjArray = obj['children']; const subObjArray = obj['children'];
if (subObjArray && subObjArray.length > 0) { if (subObjArray && subObjArray.length > 0) {
for (let i = 0; i < subObjArray.length; i++) { for (let index = 0; index < subObjArray.length; index++) {
const item = subObjArray[i]; const item = subObjArray[index];
if (item.id == targetDepartment) { if (item.id == targetDepartment) {
name = item.name; name = item.name;
break;
} }
} }
} }
if (name) break;
} }
return name; return name;
}; };
const handleViewDetail = async (row) => { const handleViewDetail = async (row) => {
// 确保科室数据已加载,以便将 ID 解析为名称 console.log('targetDepartment========>', JSON.stringify(row));
if (!orgOptions.value || orgOptions.value.length === 0) {
await getLocationInfo();
}
currentDetail.value = row; currentDetail.value = row;
// 解析 descJson // 解析 descJson
if (row.descJson) { if (row.descJson) {
try { try {
const obj = JSON.parse(row.descJson); const obj = JSON.parse(row.descJson);
// 将发往科室 ID 转换为名称 // 确保科室数据已加载
if (obj.targetDepartment) { if (!orgOptions.value || orgOptions.value.length === 0) {
obj.targetDepartment = recursionFun(obj.targetDepartment); await new Promise((resolve) => {
getDepartmentList().then((res) => {
orgOptions.value = res.data || [];
resolve();
});
});
} }
obj.targetDepartment = recursionFun(obj.targetDepartment);
descJsonData.value = obj; descJsonData.value = obj;
} catch (e) { } catch (e) {
console.error('解析 descJson 失败:', e); console.error('解析 descJson 失败:', e);
@@ -584,395 +347,6 @@ const handleViewDetail = async (row) => {
detailDialogVisible.value = true; detailDialogVisible.value = true;
}; };
/**
* 删除申请单 - 仅待签发状态可用
*/
const handleDelete = async (row) => {
try {
await proxy.$modal?.confirm?.('确认删除该检查申请单?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
});
const res = await deleteRequestForm({ requestFormId: row.requestFormId });
if (res.code === 200) {
proxy.$modal?.msgSuccess?.('删除成功');
fetchData();
} else {
proxy.$modal?.msgError?.(res.msg || '删除失败');
}
} catch (e) {
// 用户取消操作,不做处理
}
};
/**
* 撤回申请单 - 仅已签发状态可用
*/
const handleRecall = async (row) => {
try {
await proxy.$modal?.confirm?.('确认撤回该申请单?撤回后状态将变更为"待签发"。', '提示', {
confirmButtonText: '确定撤回',
cancelButtonText: '取消',
type: 'warning',
});
const res = await withdrawRequestForm({ requestFormId: row.requestFormId });
if (res.code === 200) {
proxy.$modal?.msgSuccess?.('撤回成功');
fetchData();
} else {
proxy.$modal?.msgError?.(res.msg || '撤回失败');
}
} catch (e) {
// 用户取消操作,不做处理
}
};
/**
* 打印申请单 - 已校对/待接收/已接收/已检查状态可用
* 打印内容与详情展示一致,并包含申请单条码
*/
const handlePrint = async (row) => {
try {
proxy.$modal?.msgInfo?.('正在生成打印预览...');
// 确保科室数据已加载,用于解析发往科室名称
if (!orgOptions.value || orgOptions.value.length === 0) {
await new Promise((resolve) => {
getDepartmentList().then((res) => {
orgOptions.value = res.data || [];
resolve();
});
});
}
// 解析 descJson
let descData = {};
if (row.descJson) {
try {
const obj = JSON.parse(row.descJson);
// 将发往科室ID转换为名称
if (obj.targetDepartment) {
obj.targetDepartment = recursionFun(obj.targetDepartment);
}
descData = obj;
} catch (e) {
console.error('解析 descJson 失败:', e);
}
}
// 构建诊疗项目表格行
let detailRowsHtml = '';
const detailList = row.requestFormDetailList || [];
if (detailList.length > 0) {
detailList.forEach((item, index) => {
detailRowsHtml += `
<tr>
<td style="text-align:center;padding:6px 8px;border:1px solid #ddd;">${index + 1}</td>
<td style="padding:6px 8px;border:1px solid #ddd;">${item.adviceName || '-'}</td>
<td style="text-align:center;padding:6px 8px;border:1px solid #ddd;">${item.quantity || '-'}</td>
<td style="padding:6px 8px;border:1px solid #ddd;">${item.unitCode_dictText || '-'}</td>
<td style="text-align:right;padding:6px 8px;border:1px solid #ddd;">${item.totalPrice != null ? '¥' + Number(item.totalPrice).toFixed(2) : '-'}</td>
</tr>`;
});
}
// 构建 descJson 字段行(与详情弹窗展示的字段一致)
const fieldKeys = ['targetDepartment', 'symptom', 'sign', 'clinicalDiagnosis', 'otherDiagnosis', 'relatedResult', 'attention'];
let descFieldsHtml = '';
fieldKeys.forEach((key) => {
const label = labelMap[key] || key;
if (descData[key] != null && descData[key] !== '') {
descFieldsHtml += `
<div class="info-row">
<span class="label">${label}</span>
<span class="value">${descData[key]}</span>
</div>`;
}
});
// 构建完整打印HTML
const printContent = `
<div class="print-wrapper">
<!-- 标题 -->
<div class="print-header">
<div class="print-title">检查申请单</div>
<div class="print-meta">打印时间:${new Date().toLocaleString()}</div>
</div>
<!-- 基本信息 -->
<div class="print-section">
<div class="section-title">基本信息</div>
<div class="info-grid">
<div class="info-row"><span class="label">患者姓名:</span><span class="value">${row.patientName || '-'}</span></div>
<div class="info-row"><span class="label">申请单名称:</span><span class="value">${row.name || '-'}</span></div>
<div class="info-row"><span class="label">申请单状态:</span><span class="value">${parseStatus(row.status)}</span></div>
<div class="info-row"><span class="label">创建时间:</span><span class="value">${row.createTime || '-'}</span></div>
<div class="info-row"><span class="label">申请单号:</span><span class="value">${row.prescriptionNo || '-'}</span></div>
<div class="info-row"><span class="label">申请者:</span><span class="value">${row.requesterId_dictText || '-'}</span></div>
<div class="info-row"><span class="label">就诊ID</span><span class="value">${row.encounterId || '-'}</span></div>
<div class="info-row"><span class="label">申请单ID</span><span class="value">${row.requestFormId || '-'}</span></div>
</div>
</div>
${descFieldsHtml ? `
<!-- 申请单描述 -->
<div class="print-section">
<div class="section-title">申请单描述</div>
${descFieldsHtml}
</div>` : ''}
${detailRowsHtml ? `
<!-- 诊疗项目 -->
<div class="print-section">
<div class="section-title">诊疗项目</div>
<table class="detail-table">
<thead>
<tr>
<th style="width:50px;padding:8px;border:1px solid #ddd;background:#f5f7fa;text-align:center;">序号</th>
<th style="padding:8px;border:1px solid #ddd;background:#f5f7fa;text-align:left;">医嘱名称</th>
<th style="width:60px;padding:8px;border:1px solid #ddd;background:#f5f7fa;text-align:center;">数量</th>
<th style="width:60px;padding:8px;border:1px solid #ddd;background:#f5f7fa;text-align:center;">单位</th>
<th style="width:80px;padding:8px;border:1px solid #ddd;background:#f5f7fa;text-align:right;">总价</th>
</tr>
</thead>
<tbody>${detailRowsHtml}</tbody>
</table>
</div>` : ''}
<!-- 条码区 -->
<div class="barcode-section">
<div class="barcode-container">
<div class="barcode-number">${row.prescriptionNo || ''}</div>
<div class="barcode-label">申请单号 / 扫码核验</div>
</div>
</div>
<div class="print-footer">
<div class="footer-line">本申请单仅供院内使用,请勿外传</div>
</div>
</div>
`;
// 打开新窗口打印
const printWindow = window.open('', '_blank');
if (!printWindow) {
proxy.$modal?.msgError?.('无法打开打印窗口,请检查浏览器弹窗设置');
return;
}
printWindow.document.write(`
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>检查申请单 - 打印</title>
<style>
* { margin: 0; padding: 0; box-sizing: border-box; }
body {
font-family: -apple-system, BlinkMacSystemFont, 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', sans-serif;
font-size: 12px;
color: #333;
padding: 20px;
}
.print-wrapper {
max-width: 210mm;
margin: 0 auto;
}
.print-header {
text-align: center;
padding-bottom: 12px;
margin-bottom: 16px;
border-bottom: 2px solid #333;
}
.print-title {
font-size: 20px;
font-weight: bold;
letter-spacing: 4px;
margin-bottom: 6px;
}
.print-meta {
font-size: 11px;
color: #666;
}
.print-section {
margin-bottom: 16px;
}
.section-title {
font-size: 14px;
font-weight: bold;
margin-bottom: 10px;
padding-bottom: 4px;
border-bottom: 1px solid #ddd;
}
.info-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 6px 16px;
}
.info-row {
font-size: 12px;
line-height: 1.8;
}
.info-row .label {
font-weight: 600;
color: #555;
}
.info-row .value {
color: #333;
}
.detail-table {
width: 100%;
border-collapse: collapse;
margin-top: 6px;
}
.detail-table th {
font-size: 11px;
font-weight: 600;
color: #555;
}
.detail-table td {
font-size: 12px;
}
.barcode-section {
margin-top: 24px;
padding-top: 16px;
border-top: 1px dashed #ccc;
text-align: center;
}
.barcode-container {
display: inline-block;
padding: 12px 24px;
border: 2px solid #333;
border-radius: 6px;
background: #fff;
}
.barcode-number {
font-family: 'Libre Barcode 128', 'Libre Barcode 39', 'Code128', 'C', monospace;
font-size: 42px;
letter-spacing: 4px;
color: #000;
line-height: 1.2;
}
.barcode-label {
font-size: 10px;
color: #888;
margin-top: 4px;
letter-spacing: 2px;
}
.print-footer {
margin-top: 20px;
text-align: center;
}
.footer-line {
font-size: 10px;
color: #aaa;
}
@media print {
body { padding: 0; }
.print-wrapper { max-width: none; }
.barcode-section { page-break-inside: avoid; }
}
</style>
</head>
<body>
${printContent}
</body>
</html>
`);
printWindow.document.close();
// 等待内容渲染后打印
printWindow.onload = function() {
setTimeout(() => {
printWindow.print();
proxy.$modal?.closeAll?.();
}, 300);
};
} catch (error) {
console.error('打印失败:', error);
proxy.$modal?.msgError?.('打印失败: ' + (error.message || '未知错误'));
}
};
/**
* 查看报告 - 仅已出报告状态可用
* 调用报告查询接口弹窗展示结构化报告或影像预览PACS链接
*/
const handleViewReport = async (row) => {
reportRow.value = row;
reportDialogVisible.value = true;
reportLoading.value = true;
reportData.value = null;
try {
const res = await getTestResult({ prescriptionNo: row.prescriptionNo });
if (res.code === 200) {
if (res.data) {
// 支持两种返回格式:
// 1. res.data 为对象(结构化报告):含 patientName, prescriptionNo, reportTime, diagnosis/content, imageUrl/pacsUrl
// 2. res.data 为字符串报告URL映射到 imageUrl 以支持iframe预览
if (typeof res.data === 'string') {
reportData.value = {
prescriptionNo: row.prescriptionNo,
imageUrl: res.data,
};
} else if (typeof res.data === 'object') {
reportData.value = {
...res.data,
prescriptionNo: res.data.prescriptionNo || row.prescriptionNo,
};
}
} else {
reportData.value = null;
}
} else {
reportData.value = null;
proxy.$modal?.msgWarning?.(res.msg || '暂未生成报告');
}
} catch (e) {
console.error('获取报告失败:', e);
reportData.value = null;
proxy.$modal?.msgError?.('获取报告失败');
} finally {
reportLoading.value = false;
}
};
/**
* 打开PACS影像链接
*/
const openPacsLink = () => {
const url = reportData.value?.pacsUrl || reportData.value?.imageUrl;
if (url) {
window.open(url, '_blank');
}
};
// ========== 修改申请单相关 ==========
const editDialogVisible = ref(false);
const editFormRef = ref(null);
const editingRow = ref(null);
// 修改申请单 - 复用检查申请单弹窗
const handleEdit = (row) => {
editingRow.value = { ...row };
editDialogVisible.value = true;
// 弹窗打开后手动调用getList确保数据加载
nextTick(() => {
editFormRef.value?.getList?.();
editFormRef.value?.getLocationInfo?.();
editFormRef.value?.getDiagnosisList?.();
});
};
// 编辑弹窗确认提交
const handleEditSubmit = () => {
// 调用MedicalExaminations组件的submit方法
if (editFormRef.value?.submit) {
editFormRef.value.submit();
}
};
watch( watch(
() => patientInfo.value?.encounterId, () => patientInfo.value?.encounterId,
(val) => { (val) => {
@@ -981,9 +355,8 @@ watch(
getLocationInfo(); getLocationInfo();
} else { } else {
tableData.value = []; tableData.value = [];
filterForm.value.dateRange = getDefaultDateRange(); filterForm.value.dateRange = [];
filterForm.value.status = ''; filterForm.value.status = '';
filterForm.value.keyword = '';
} }
}, },
{ immediate: true } { immediate: true }
@@ -1097,96 +470,4 @@ defineExpose({
overflow: auto; overflow: auto;
} }
} }
// 报告弹窗样式
.report-viewer {
min-height: 200px;
padding: 8px 0;
}
.report-viewer-container {
.report-content-section {
margin-top: 16px;
padding: 12px;
background: #fafafa;
border-radius: 6px;
border: 1px solid #eee;
.section-title {
font-size: 14px;
font-weight: 600;
margin-bottom: 10px;
color: #303133;
display: flex;
align-items: center;
}
.report-content-text {
font-size: 13px;
line-height: 1.8;
color: #606266;
white-space: pre-wrap;
}
}
.report-image-section {
margin-top: 16px;
.section-title {
font-size: 14px;
font-weight: 600;
margin-bottom: 10px;
color: #303133;
display: flex;
align-items: center;
}
.report-iframe {
width: 100%;
height: 480px;
border: 1px solid #dcdfe6;
border-radius: 4px;
}
}
}
// 状态标签样式 - 参考临床医嘱
:deep(.el-tag) {
border-radius: 2px;
padding: 0 8px;
height: 24px;
line-height: 22px;
font-size: 12px;
border-width: 1px;
}
:deep(.el-tag--info.is-plain) {
background: #f4f4f5;
border-color: #e9e9eb;
color: #909399;
}
:deep(.el-tag--primary.is-plain) {
background: #ecf5ff;
border-color: #d9ecff;
color: #409eff;
}
:deep(.el-tag--success.is-plain) {
background: #f0f9eb;
border-color: #e1f3d8;
color: #67c23a;
}
:deep(.el-tag--warning.is-plain) {
background: #fdf6ec;
border-color: #faecd8;
color: #e6a23c;
}
:deep(.el-tag--danger.is-plain) {
background: #fef0f0;
border-color: #fde2e2;
color: #f56c6c;
}
</style> </style>

View File

@@ -82,11 +82,7 @@
</template> </template>
<el-table-column type="index" label="序号" width="60" align="center" /> <el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="patientName" label="患者姓名" width="120" /> <el-table-column prop="patientName" label="患者姓名" width="120" />
<el-table-column label="申请单名称" width="140"> <el-table-column prop="name" label="申请单名称" width="140" />
<template #default="scope">
<span>{{ buildApplicationName(scope.row) }}</span>
</template>
</el-table-column>
<el-table-column prop="createTime" label="创建时间" width="160" /> <el-table-column prop="createTime" label="创建时间" width="160" />
<el-table-column prop="prescriptionNo" label="申请单号" width="140" /> <el-table-column prop="prescriptionNo" label="申请单号" width="140" />
<el-table-column label="单据状态" width="100" align="center"> <el-table-column label="单据状态" width="100" align="center">
@@ -107,11 +103,11 @@
<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="160"> <el-table-column label="操作" align="center" fixed="right" width="160">
<template #default="scope"> <template #default="scope">
<template v-if="scope.row.status == 0"> <template v-if="scope.row.billStatus == 0 || scope.row.status == 0">
<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-else-if="scope.row.status == 1"> <template v-else-if="scope.row.billStatus == 1 || scope.row.status == 1">
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button> <el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
</template> </template>
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button> <el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
@@ -141,7 +137,7 @@
<el-descriptions-item label="创建时间">{{ <el-descriptions-item label="创建时间">{{
currentDetail.createTime || '-' currentDetail.createTime || '-'
}}</el-descriptions-item> }}</el-descriptions-item>
<el-descriptions-item label="申请单号">{{ <el-descriptions-item label="处方号">{{
currentDetail.prescriptionNo || '-' currentDetail.prescriptionNo || '-'
}}</el-descriptions-item> }}</el-descriptions-item>
<el-descriptions-item label="申请者">{{ <el-descriptions-item label="申请者">{{
@@ -183,26 +179,6 @@
<el-button @click="detailDialogVisible = false">关闭</el-button> <el-button @click="detailDialogVisible = false">关闭</el-button>
</template> </template>
</el-dialog> </el-dialog>
<!-- 编辑检验申请单弹窗 -->
<el-dialog
v-model="editDialogVisible"
title="编辑检验申请单"
width="1200px"
destroy-on-close
top="5vh"
:close-on-click-modal="false"
>
<LaboratoryTests
ref="editFormRef"
@submitOk="handleEditSubmitOk"
:editData="editRowData"
/>
<template #footer>
<el-button @click="editDialogVisible = false">取消</el-button>
<el-button type="primary" @click="submitEditForm">确认</el-button>
</template>
</el-dialog>
</div> </div>
</template> </template>
@@ -211,18 +187,13 @@ import {computed, getCurrentInstance, ref, watch} from 'vue';
import {Refresh, Search} from '@element-plus/icons-vue'; import {Refresh, Search} from '@element-plus/icons-vue';
import {patientInfo} from '../../store/patient.js'; import {patientInfo} from '../../store/patient.js';
import {getInspection, deleteRequestForm, withdrawRequestForm} from './api'; import {getInspection, deleteRequestForm, withdrawRequestForm} from './api';
import {getDepartmentList} from '@/api/public.js'; import {getOrgList} from '@/views/doctorstation/components/api.js';
import LaboratoryTests from '../order/applicationForm/laboratoryTests.vue';
import {saveInspection} from '../order/applicationForm/api.js';
const { proxy } = getCurrentInstance(); const { proxy } = getCurrentInstance();
const tableData = ref([]); const tableData = ref([]);
const loading = ref(false); const loading = ref(false);
const detailDialogVisible = ref(false); const detailDialogVisible = ref(false);
const editDialogVisible = ref(false);
const editRowData = ref(null);
const editFormRef = ref(null);
const currentDetail = ref(null); const currentDetail = ref(null);
const descJsonData = ref(null); const descJsonData = ref(null);
const orgOptions = ref([]); const orgOptions = ref([]);
@@ -314,9 +285,6 @@ const labelMap = {
otherDiagnosis: '其他诊断', otherDiagnosis: '其他诊断',
relatedResult: '相关结果', relatedResult: '相关结果',
attention: '注意事项', attention: '注意事项',
applicationType: '申请类型',
specimenName: '标本类型',
executeTime: '执行时间',
}; };
/** /**
@@ -343,8 +311,8 @@ const parsePriorityCode = (descJson) => {
if (!descJson) return '-'; if (!descJson) return '-';
try { try {
const obj = JSON.parse(descJson); const obj = JSON.parse(descJson);
// applicationType: 0-普通, 1-急 // priorityCode: 0-普通, 1-急
return obj.applicationType === 1 ? '急' : '普通'; return obj.priorityCode === 1 ? '急' : '普通';
} catch (e) { } catch (e) {
console.error('解析 descJson 失败:', e); console.error('解析 descJson 失败:', e);
return '-'; return '-';
@@ -368,24 +336,6 @@ const parseSpecimenType = (descJson) => {
} }
}; };
/**
* 根据申请单详情构建申请单名称
* 单一项目:显示项目名称+数量
* 多个项目:显示首个项目名称+数量+"等X项"
*/
const buildApplicationName = (row) => {
const details = row.requestFormDetailList;
if (!details || details.length === 0) {
return row.name || '-';
}
if (details.length === 1) {
const item = details[0];
return `${item.adviceName}${item.quantity || ''}`;
}
const first = details[0];
return `${first.adviceName}${first.quantity || ''}${details.length}`;
};
const isFieldMatched = (key) => { const isFieldMatched = (key) => {
return key in labelMap; return key in labelMap;
}; };
@@ -441,9 +391,6 @@ const handleViewDetail = async (row) => {
try { try {
const obj = JSON.parse(row.descJson); const obj = JSON.parse(row.descJson);
obj.targetDepartment = recursionFun(obj.targetDepartment); obj.targetDepartment = recursionFun(obj.targetDepartment);
// 转换申请类型编码为可读文本
if (obj.applicationType === 0) obj.applicationType = '普通';
else if (obj.applicationType === 1) obj.applicationType = '急诊';
descJsonData.value = obj; descJsonData.value = obj;
} catch (e) { } catch (e) {
console.error('解析 descJson 失败:', e); console.error('解析 descJson 失败:', e);
@@ -458,32 +405,10 @@ const handleViewDetail = async (row) => {
/** /**
* 修改检验申请单(待签发状态) * 修改检验申请单(待签发状态)
*/ */
const handleEdit = async (row) => { const handleEdit = (row) => {
// 确保科室数据已加载 // 复用详情查看逻辑,后续可扩展为打开编辑弹窗
if (!orgOptions.value || orgOptions.value.length === 0) { handleViewDetail(row);
await getLocationInfo(); proxy.$modal?.msgInfo?.('修改功能待接入,请通过详情弹窗查看后重新开立');
}
editRowData.value = row;
editDialogVisible.value = true;
};
/**
* 编辑弹窗提交成功回调
*/
const handleEditSubmitOk = async () => {
editDialogVisible.value = false;
editRowData.value = null;
proxy.$modal?.msgSuccess?.('修改成功');
await fetchData();
};
/**
* 编辑弹窗提交按钮
*/
const submitEditForm = () => {
if (editFormRef.value?.submit) {
editFormRef.value.submit();
}
}; };
/** /**
@@ -497,7 +422,7 @@ const handleDelete = async (row) => {
} }
try { try {
const res = await deleteRequestForm({ requestFormId: row.requestFormId }); const res = await deleteRequestForm({ prescriptionNo: row.prescriptionNo });
if (res?.code === 200) { if (res?.code === 200) {
proxy.$modal?.msgSuccess?.('删除成功'); proxy.$modal?.msgSuccess?.('删除成功');
await fetchData(); await fetchData();
@@ -520,7 +445,7 @@ const handleWithdraw = async (row) => {
} }
try { try {
const res = await withdrawRequestForm({ requestFormId: row.requestFormId }); const res = await withdrawRequestForm({ prescriptionNo: row.prescriptionNo });
if (res?.code === 200) { if (res?.code === 200) {
proxy.$modal?.msgSuccess?.('撤回成功'); proxy.$modal?.msgSuccess?.('撤回成功');
await fetchData(); await fetchData();

View File

@@ -568,11 +568,6 @@ function handleMaindise(value, index) {
* 保存诊断 * 保存诊断
*/ */
function handleSaveDiagnosis() { function handleSaveDiagnosis() {
// 防止重复点击保存
if (isSaving.value) {
return;
}
for (let index = 0; index < (form.value.diagnosisList || []).length; index++) { for (let index = 0; index < (form.value.diagnosisList || []).length; index++) {
const item = form.value.diagnosisList[index]; const item = form.value.diagnosisList[index];
if (!item.diagSrtNo) { if (!item.diagSrtNo) {
@@ -605,7 +600,7 @@ function handleSaveDiagnosis() {
// 步骤2重新分配连续的序号从1开始 // 步骤2重新分配连续的序号从1开始
sortedList.forEach((item, index) => { sortedList.forEach((item, index) => {
item.diagSrtNo = index + 1; // 这里是关键!把诊断排序”改成新顺序 item.diagSrtNo = index + 1; // 这里是关键!把诊断排序”改成新顺序
}); });
// 步骤3提交排序后的数据 // 步骤3提交排序后的数据
@@ -615,12 +610,12 @@ function handleSaveDiagnosis() {
diagnosisChildList: sortedList, diagnosisChildList: sortedList,
}).then((res) => { }).then((res) => {
if (res.code === 200) { if (res.code === 200) {
// 步骤4更新本地数据使用全新对象防止响应式问题
form.value.diagnosisList = sortedList.map(item => ({ ...item }));
emits('diagnosisSave', false); emits('diagnosisSave', false);
proxy.$modal.msgSuccess('诊断已保存'); proxy.$modal.msgSuccess('诊断已保存');
// 保存成功后从服务器重新加载数据,确保前后端数据一致
getList();
// 食源性疾病逻辑 // 食源性疾病逻辑
isFoodDiseasesNew({ encounterId: props.patientInfo.encounterId }).then((res2) => { isFoodDiseasesNew({ encounterId: props.patientInfo.encounterId }).then((res2) => {
if (res2.code === 20 && res2.data) { if (res2.code === 20 && res2.data) {

View File

@@ -249,7 +249,7 @@ const submit = () => {
requestFormId: '', // 申请单ID requestFormId: '', // 申请单ID
name: '输血申请单', name: '输血申请单',
descJson: JSON.stringify(form), descJson: JSON.stringify(form),
categoryEnum: '23', // 21 检验 22 检查 23 输血 24 手术(避开 adviceType 1-6 碰撞) categoryEnum: '3', // 1 检验 2 检查 3 输血 4 手术
}).then((res) => { }).then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$message.success(res.msg); proxy.$message.success(res.msg);

View File

@@ -6,26 +6,9 @@
<template> <template>
<div class="LaboratoryTests-container"> <div class="LaboratoryTests-container">
<div v-loading="loading" class="transfer-wrapper"> <div v-loading="loading" class="transfer-wrapper">
<!-- 远程搜索框 -->
<div class="search-bar">
<el-input
v-model="searchKey"
placeholder="输入项目代码/名称搜索"
clearable
@keyup.enter="handleSearch"
@clear="handleSearch"
style="width: 300px; margin-bottom: 10px"
>
<template #append>
<el-button @click="handleSearch">搜索</el-button>
</template>
</el-input>
<span v-if="!searchKey" class="total-count"> {{ totalCount }} </span>
<span v-else class="total-count">搜索到 {{ filteredCount }} / {{ totalCount }} </span>
</div>
<el-transfer <el-transfer
v-model="transferValue" v-model="transferValue"
:data="transferData" :data="applicationList"
filter-placeholder="项目代码/名称" filter-placeholder="项目代码/名称"
filterable filterable
:titles="['未选择', '已选择']" :titles="['未选择', '已选择']"
@@ -134,7 +117,7 @@
</div> </div>
</template> </template>
<script setup name="LaboratoryTests"> <script setup name="LaboratoryTests">
import {getCurrentInstance, onMounted, reactive, ref, watch, computed} from 'vue'; import {getCurrentInstance, onBeforeMount, onMounted, reactive, watch} from 'vue';
import {patientInfo} from '../../../store/patient.js'; import {patientInfo} from '../../../store/patient.js';
import {getApplicationList, saveInspection} from './api'; import {getApplicationList, saveInspection} from './api';
import {getOrgList} from '@/views/doctorstation/components/api.js'; import {getOrgList} from '@/views/doctorstation/components/api.js';
@@ -155,92 +138,48 @@ const findTreeItem = (list, id) => {
return null; return null;
}; };
const emits = defineEmits(['submitOk']); const emits = defineEmits(['submitOk']);
const props = defineProps({ const props = defineProps({});
editData: {
type: Object,
default: null,
},
});
const isEditMode = computed(() => !!props.editData?.requestFormId);
const state = reactive({}); const state = reactive({});
const applicationListAll = ref([]); const applicationListAll = ref();
const applicationList = ref();
const loading = ref(false); const loading = ref(false);
const orgOptions = ref([]); const orgOptions = ref([]); // 科室选项
const searchKey = ref(''); const getList = () => {
const totalCount = ref(0);
// 将已加载的全部数据转为 transfer 组件所需的格式
const buildTransferData = (records) => {
return records.map((item) => {
const priceInfo = item.priceList?.[0] || {};
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
const unit = item.unitCode_dictText || item.unitCode || '';
return {
adviceDefinitionId: item.adviceDefinitionId,
orgId: item.orgId,
label: item.adviceName + ' (¥' + price + '/' + unit + ')',
key: item.adviceDefinitionId,
};
});
};
// 加载全部数据(不分页,一次性拉取)
const loadAllData = async () => {
if (!patientInfo.value?.inHospitalOrgId) { if (!patientInfo.value?.inHospitalOrgId) {
applicationListAll.value = []; applicationList.value = [];
return; return;
} }
loading.value = true; loading.value = true;
try { getApplicationList({
// 使用大 pageSize 一次性拉取所有启用状态的检验类诊疗项目 pageSize: 9999,
const res = await getApplicationList({ pageNum: 1,
pageSize: 9999, categoryCode: '22',
pageNo: 1, organizationId: patientInfo.value.inHospitalOrgId,
categoryCode: '22', adviceTypes: [3], //1 药品 2耗材 3诊疗
organizationId: patientInfo.value.inHospitalOrgId, })
adviceTypes: [3], // 1 药品 2 耗材 3 诊疗 .then((res) => {
if (res.code === 200) {
applicationListAll.value = res.data.records;
applicationList.value = res.data.records.map((item) => {
const priceInfo = item.priceList?.[0] || {};
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
const unit = item.unitCode_dictText || item.unitCode || '';
return {
adviceDefinitionId: item.adviceDefinitionId,
orgId: item.orgId,
label: item.adviceName + ' (¥' + price + '/' + unit + ')',
key: item.adviceDefinitionId,
};
});
console.log('applicationList========>', JSON.stringify(res.data.records));
} else {
proxy.$message.error(res.message);
applicationList.value = [];
}
})
.finally(() => {
loading.value = false;
}); });
if (res.code !== 200) {
proxy.$message.error(res.message);
applicationListAll.value = [];
return;
}
applicationListAll.value = res.data?.records || [];
totalCount.value = res.data?.total || 0;
} catch (e) {
proxy.$message.error('获取检验项目列表失败');
applicationListAll.value = [];
} finally {
loading.value = false;
}
};
// 根据搜索关键词过滤数据
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 () => {
await loadAllData();
};
const handleSearch = () => {
// 搜索时保持已选中的项目不受影响
}; };
const transferValue = ref([]); const transferValue = ref([]);
const form = reactive({ const form = reactive({
@@ -259,6 +198,7 @@ const form = reactive({
otherDiagnosisList: [], //其他断目录 otherDiagnosisList: [], //其他断目录
}); });
const rules = reactive({}); const rules = reactive({});
onBeforeMount(() => {});
onMounted(() => { onMounted(() => {
getList(); getList();
}); });
@@ -271,22 +211,12 @@ const projectWithDepartment = (selectProjectIds, type) => {
let isRelease = true; let isRelease = true;
// 选中项目的数组 // 选中项目的数组
const arr = []; const arr = [];
// 根据选中的项目id查找对应的项目(从全部原始数据中查找) // 根据选中的项目id查找对应的项目
selectProjectIds.forEach((element) => { selectProjectIds.forEach((element) => {
const searchData = applicationListAll.value.find((item) => { const searchData = applicationList.value.find((item) => {
return element == item.adviceDefinitionId; return element == item.adviceDefinitionId;
}); });
if (searchData) { arr.push(searchData);
const priceInfo = searchData.priceList?.[0] || {};
const price = priceInfo.price != null ? Number(priceInfo.price).toFixed(2) : '0.00';
const unit = searchData.unitCode_dictText || searchData.unitCode || '';
arr.push({
adviceDefinitionId: searchData.adviceDefinitionId,
orgId: searchData.orgId,
label: searchData.adviceName + ' (¥' + price + '/' + unit + ')',
key: searchData.adviceDefinitionId,
});
}
}); });
// 保存用户手动选择的发往科室(提交时需要保留) // 保存用户手动选择的发往科室(提交时需要保留)
const manualDept = type === 2 ? form.targetDepartment : ''; const manualDept = type === 2 ? form.targetDepartment : '';
@@ -312,16 +242,12 @@ const projectWithDepartment = (selectProjectIds, type) => {
if (type === 2 && manualDept) { if (type === 2 && manualDept) {
form.targetDepartment = manualDept; form.targetDepartment = manualDept;
isRelease = true; isRelease = true;
} else if (type === 2 && !manualDept) { } else {
// 提交时用户未手动选择科室,才提示错误
isRelease = false; isRelease = false;
ElMessage({ ElMessage({
type: 'error', type: 'error',
message: '未找到项目执行的科室', message: '未找到项目执行的科室',
}); });
} else {
// type=1(选择项目变化)时,不弹窗,仅清空科室让用户自行选择
isRelease = false;
} }
} }
if (findItem && isRelease) { if (findItem && isRelease) {
@@ -337,44 +263,6 @@ watch(
projectWithDepartment(newValue, 1); projectWithDepartment(newValue, 1);
} }
); );
// 编辑模式下,回显已有数据
watch(
() => props.editData,
(newData) => {
if (!newData || !newData.requestFormId) return;
// 解析 descJson 回填表单
if (newData.descJson) {
try {
const obj = JSON.parse(newData.descJson);
Object.keys(form).forEach((key) => {
if (obj[key] !== undefined) {
form[key] = obj[key];
}
});
} catch (e) {
console.error('解析 descJson 失败:', e);
}
}
// 回填已选项目
if (newData.requestFormDetailList && newData.requestFormDetailList.length > 0) {
// 从全部数据中匹配已选项目
const selectedIds = [];
newData.requestFormDetailList.forEach((detail) => {
const matched = applicationListAll.value.find(
(item) => item.adviceName === detail.adviceName
);
if (matched) {
selectedIds.push(matched.adviceDefinitionId);
}
});
transferValue.value = selectedIds;
}
},
{ immediate: true }
);
const submit = () => { const submit = () => {
if (transferValue.value.length == 0) { if (transferValue.value.length == 0) {
return proxy.$message.error('请选择申请单'); return proxy.$message.error('请选择申请单');
@@ -407,15 +295,15 @@ const submit = () => {
patientId: patientInfo.value.patientId, //患者ID patientId: patientInfo.value.patientId, //患者ID
encounterId: patientInfo.value.encounterId, // 就诊ID encounterId: patientInfo.value.encounterId, // 就诊ID
organizationId: patientInfo.value.inHospitalOrgId, // 医疗机构ID organizationId: patientInfo.value.inHospitalOrgId, // 医疗机构ID
requestFormId: isEditMode.value ? props.editData.requestFormId : '', // 申请单ID(编辑模式传入,新增为空) requestFormId: '', // 申请单ID
name: '检验申请单', name: '检验申请单',
descJson: JSON.stringify(form), descJson: JSON.stringify(form),
categoryEnum: '21', // 21 检验 22 检查 23 输血 24 手术(避开 adviceType 1-6 碰撞) categoryEnum: '1', // 1 检验 2 检查 3 输血 4 手术
}; };
saveInspection(params).then((res) => { saveInspection(params).then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$message.success(isEditMode.value ? '修改成功' : res.msg); proxy.$message.success(res.msg);
transferValue.value = []; applicationList.value = [];
emits('submitOk'); emits('submitOk');
} else { } else {
proxy.$message.error(res.message); proxy.$message.error(res.message);
@@ -472,19 +360,6 @@ defineExpose({ state, submit, getLocationInfo, getDiagnosisList, getList });
.transfer-wrapper { .transfer-wrapper {
position: relative; position: relative;
min-height: 300px; min-height: 300px;
.search-bar {
display: flex;
align-items: center;
gap: 12px;
margin-bottom: 8px;
.total-count {
font-size: 13px;
color: #909399;
white-space: nowrap;
}
}
} }
.el-transfer { .el-transfer {

View File

@@ -5,25 +5,36 @@
--> -->
<template> <template>
<div class="medicalExaminations-container"> <div class="medicalExaminations-container">
<!-- 主体内容 --> <!-- 顶部标题栏 -->
<div class="form-body"> <div class="form-header">
<!-- 右上角紧急程度 --> <div class="header-left">
<div class="urgency-bar"> <el-icon class="header-icon"><Files /></el-icon>
<span class="urgency-bar-label">紧急程度</span> <span class="header-title">检查申请单</span>
<el-radio-group v-model="form.urgencyLevel" @change="handleUrgencyChange" size="small"> </div>
<div class="header-right">
<span class="urgency-label">紧急程度</span>
<el-radio-group v-model="form.urgencyLevel" @change="handleUrgencyChange" class="urgency-radio-group">
<el-radio-button label="routine">普通</el-radio-button> <el-radio-button label="routine">普通</el-radio-button>
<el-radio-button label="emergency">急诊</el-radio-button> <el-radio-button label="emergency">急诊</el-radio-button>
</el-radio-group> </el-radio-group>
<transition name="el-fade-in-linear"> <transition name="el-fade-in-linear">
<span v-if="form.urgencyLevel === 'emergency'" class="emergency-tip-inline"> <span v-if="form.urgencyLevel === 'emergency'" class="emergency-tip">
<el-icon><WarningFilled /></el-icon> <el-icon><WarningFilled /></el-icon>
绿色通道 急诊单将进入绿色通道
</span> </span>
</transition> </transition>
</div> </div>
</div>
<!-- 主体内容区 -->
<div class="form-body">
<!-- 选择检查项目 --> <!-- 选择检查项目 -->
<div class="section-card"> <div class="section-card">
<div class="transfer-wrapper"> <div class="section-header">
<el-icon><Document /></el-icon>
<span>选择检查项目</span>
</div>
<div v-loading="loading" class="transfer-wrapper">
<el-transfer <el-transfer
v-model="transferValue" v-model="transferValue"
:data="applicationList" :data="applicationList"
@@ -34,150 +45,165 @@
</div> </div>
</div> </div>
<el-form :model="form" :rules="rules" ref="formRef" label-position="top" class="info-form"> <!-- 申请信息 -->
<!-- 第一行发往科室 + 紧急程度 + 期望检查时间 --> <div class="section-card">
<el-row :gutter="16"> <div class="section-header">
<el-col :span="8"> <el-icon><EditPen /></el-icon>
<el-form-item label="发往科室" prop="targetDepartment"> <span>申请信息</span>
<el-tree-select </div>
clearable
style="width: 100%"
v-model="form.targetDepartment"
filterable
:data="orgOptions"
:props="{ value: 'id', label: 'name', children: 'children' }"
value-key="id"
check-strictly
placeholder="请选择执行科室"
/>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="期望检查时间">
<el-date-picker
v-model="form.expectedExaminationTime"
type="datetime"
placeholder="默认当前时间"
style="width: 100%"
value-format="YYYY-MM-DD HH:mm:ss"
format="YYYY-MM-DD HH:mm"
:disabled-date="disabledFutureDate"
:default-value="new Date()"
/>
</el-form-item>
</el-col>
</el-row>
<!-- 第二行症状 + 体征 --> <el-form :model="form" :rules="rules" ref="formRef" label-position="top" class="info-form">
<el-row :gutter="16"> <!-- 第一行发往科室 + 期望检查时间 -->
<el-col :span="12"> <el-row :gutter="16">
<el-form-item label="症状"> <el-col :span="12">
<el-input v-model="form.symptom" autocomplete="off" type="textarea" :rows="2" placeholder="请输入患者症状" /> <el-form-item label="发往科室" prop="targetDepartment">
</el-form-item> <el-tree-select
</el-col> clearable
<el-col :span="12"> style="width: 100%"
<el-form-item label="体征"> v-model="form.targetDepartment"
<el-input v-model="form.sign" autocomplete="off" type="textarea" :rows="2" placeholder="请输入患者体征" /> filterable
</el-form-item> :data="orgOptions"
</el-col> :props="{ value: 'id', label: 'name', children: 'children' }"
</el-row> value-key="id"
check-strictly
placeholder="请选择执行科室"
/>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="期望检查时间">
<el-date-picker
v-model="form.expectedExaminationTime"
type="datetime"
placeholder="默认当前时间"
style="width: 100%"
value-format="YYYY-MM-DD HH:mm:ss"
format="YYYY-MM-DD HH:mm"
:disabled-date="disabledFutureDate"
:default-value="new Date()"
/>
</el-form-item>
</el-col>
</el-row>
<!-- 临床诊断 + 其他诊断 --> <!-- 症状 + 体征 -->
<el-row :gutter="16"> <el-row :gutter="16">
<el-col :span="12"> <el-col :span="12">
<el-form-item label="临床诊断"> <el-form-item label="症状">
<el-input disabled v-model="form.clinicalDiagnosis" placeholder="自动带入主诊断" /> <el-input v-model="form.symptom" autocomplete="off" type="textarea" :rows="2" placeholder="请输入患者症状" />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="其他诊断"> <el-form-item label="体征">
<el-input disabled v-model="form.otherDiagnosis" placeholder="自动带入其他诊断" /> <el-input v-model="form.sign" autocomplete="off" type="textarea" :rows="2" placeholder="请输入患者体征" />
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> </el-row>
<!-- 相关结果 + 注意事项 --> <!-- 临床诊断 + 其他诊断 -->
<el-row :gutter="16"> <el-row :gutter="16">
<el-col :span="12"> <el-col :span="12">
<el-form-item label="相关结果"> <el-form-item label="临床诊断">
<el-input v-model="form.relatedResult" autocomplete="off" type="textarea" :rows="2" placeholder="请输入相关检验结果" /> <el-input disabled v-model="form.clinicalDiagnosis" placeholder="自动带入主诊断" />
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="注意事项"> <el-form-item label="其他诊断">
<el-input v-model="form.attention" autocomplete="off" type="textarea" :rows="2" placeholder="请输入检查注意事项" /> <el-input disabled v-model="form.otherDiagnosis" placeholder="自动带入其他诊断" />
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> </el-row>
<!-- 检查目的 + 病史摘要 --> <!-- 相关结果 + 注意事项 -->
<el-row :gutter="16"> <el-row :gutter="16">
<el-col :span="12"> <el-col :span="12">
<el-form-item label="检查目的" prop="examinationPurpose"> <el-form-item label="相关结果">
<el-input v-model="form.relatedResult" autocomplete="off" type="textarea" :rows="2" placeholder="请输入相关检验结果" />
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="注意事项">
<el-input v-model="form.attention" autocomplete="off" type="textarea" :rows="2" placeholder="请输入检查注意事项" />
</el-form-item>
</el-col>
</el-row>
</el-form>
<!-- 过敏史卡片 -->
<div class="section-card allergy-card">
<div class="section-header">
<el-icon><Warning /></el-icon>
<span>过敏史</span>
<span v-if="form.allergyHistory" class="header-count">{{ form.allergyHistory.length }}</span>
</div>
<div class="allergy-content">
<div class="allergy-input-row">
<el-input <el-input
v-model="form.examinationPurpose" v-model="form.allergyHistory"
autocomplete="off" autocomplete="off"
type="textarea" type="textarea"
:rows="2" :rows="2"
maxlength="200" :class="{ 'allergy-danger': isSevereAllergy }"
show-word-limit placeholder="如:造影剂过敏史等(系统将自动从患者档案带入)"
placeholder="请输入检查目的,如:明确诊断、术后复查、疗效评估等"
/> />
</el-form-item> <span v-if="isSevereAllergy" class="allergy-severe-tag">
</el-col> <el-icon><WarningFilled /></el-icon>
<el-col :span="12"> 严重过敏
<el-form-item label="病史摘要" prop="medicalHistorySummary"> </span>
<div class="history-field-wrapper"> </div>
<el-input <div class="allergy-confirm">
v-model="form.medicalHistorySummary" <el-checkbox v-model="form.allergyConfirmed" size="small">
autocomplete="off" 已通过口头询问确认无过敏史
type="textarea" </el-checkbox>
:rows="2" </div>
placeholder="请简要描述患者病史摘要" </div>
/> </div>
<el-button
type="primary"
plain
size="small"
class="history-sync-btn"
@click="handleSyncHistory"
:loading="syncingHistory"
>
<el-icon><Refresh /></el-icon>
同步
</el-button>
</div>
</el-form-item>
</el-col>
</el-row>
<!-- 第六行过敏史 --> <!-- 检查目的卡片 -->
<el-row :gutter="16"> <div class="section-card purpose-card">
<el-col :span="24"> <div class="section-header">
<el-form-item label="过敏史"> <el-icon><Aim /></el-icon>
<div class="allergy-wrapper"> <span>检查目的</span>
<el-input <span class="required-mark">必填</span>
v-model="form.allergyHistory" </div>
autocomplete="off" <el-input
type="textarea" v-model="form.examinationPurpose"
:rows="1" autocomplete="off"
:class="{ 'allergy-danger': isSevereAllergy }" type="textarea"
placeholder="如:造影剂过敏史等(系统将自动从患者档案带入)" :rows="2"
/> maxlength="200"
<div class="allergy-actions"> show-word-limit
<span v-if="isSevereAllergy" class="allergy-severe-tag-inline"> placeholder="请输入检查目的,如:明确诊断、术后复查、疗效评估等"
<el-icon><WarningFilled /></el-icon> />
严重过敏 </div>
</span>
<el-checkbox v-model="form.allergyConfirmed" size="small"> <!-- 病史摘要卡片 -->
已通过口头询问确认无过敏史 <div class="section-card history-card">
</el-checkbox> <div class="section-header">
</div> <el-icon><DocumentCopy /></el-icon>
</div> <span>病史摘要</span>
</el-form-item> <span class="required-mark">必填</span>
</el-col> <el-button
</el-row> type="primary"
</el-form> plain
size="small"
class="sync-btn"
@click="handleSyncHistory"
:loading="syncingHistory"
>
<el-icon><Refresh /></el-icon>
同步现病史/体征
</el-button>
</div>
<el-input
v-model="form.medicalHistorySummary"
autocomplete="off"
type="textarea"
:rows="3"
placeholder="请简要描述患者病史摘要"
/>
</div>
</div>
</div> </div>
<!-- 急诊确认弹窗 --> <!-- 急诊确认弹窗 -->
@@ -201,8 +227,7 @@
</template> </template>
<script setup name="MedicalExaminations"> <script setup name="MedicalExaminations">
import {getCurrentInstance, onMounted, reactive, ref, watch, computed, nextTick} from 'vue'; import {getCurrentInstance, onMounted, reactive, ref, watch, computed} from 'vue';
import dayjs from 'dayjs';
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 {getEncounterDiagnosis} from '../../api.js'; import {getEncounterDiagnosis} from '../../api.js';
@@ -226,26 +251,7 @@ const findTreeItem = (list, id) => {
}; };
const emits = defineEmits(['submitOk']); const emits = defineEmits(['submitOk']);
const props = defineProps({ const props = defineProps({});
isEditMode: {
type: Boolean,
default: false,
},
editData: {
type: Object,
default: () => ({}),
},
// 支持通过props传入patientInfo
externalPatientInfo: {
type: Object,
default: null,
},
});
// 优先使用外部传入的patientInfo否则使用store中的
const effectivePatientInfo = computed(() => {
return props.externalPatientInfo || patientInfo.value;
});
const orgOptions = ref([]); const orgOptions = ref([]);
const state = reactive({}); const state = reactive({});
const applicationListAll = ref(); const applicationListAll = ref();
@@ -266,9 +272,7 @@ const isSevereAllergy = computed(() => {
}); });
const getList = () => { const getList = () => {
console.log('getList called, effectivePatientInfo:', effectivePatientInfo.value); if (!patientInfo.value?.inHospitalOrgId) {
if (!effectivePatientInfo.value?.inHospitalOrgId) {
console.log('inHospitalOrgId is missing, setting empty list');
applicationList.value = []; applicationList.value = [];
return; return;
} }
@@ -277,7 +281,7 @@ const getList = () => {
pageSize: 500, pageSize: 500,
pageNum: 1, pageNum: 1,
categoryCode: '23', categoryCode: '23',
organizationId: effectivePatientInfo.value.inHospitalOrgId, organizationId: patientInfo.value.inHospitalOrgId,
adviceTypes: [3], adviceTypes: [3],
}) })
.then((res) => { .then((res) => {
@@ -294,24 +298,6 @@ const getList = () => {
key: item.adviceDefinitionId, key: item.adviceDefinitionId,
}; };
}); });
// 编辑模式下,加载完数据后设置已选项目
if (props.isEditMode && props.editData?.requestFormDetailList?.length) {
nextTick(() => {
// 使用 adviceName 匹配
const selectedNames = props.editData.requestFormDetailList.map(item => item.adviceName);
console.log('getList completed, selectedNames:', selectedNames);
const selectedIds = [];
applicationList.value?.forEach(app => {
// 匹配时去掉价格部分,只比较名称
const appName = app.label?.split(' (')[0];
if (selectedNames.includes(appName)) {
selectedIds.push(app.key);
}
});
console.log('getList completed, matched selectedIds:', selectedIds);
transferValue.value = selectedIds;
});
}
} else { } else {
proxy.$message.error(res.message); proxy.$message.error(res.message);
applicationList.value = []; applicationList.value = [];
@@ -330,7 +316,7 @@ const form = reactive({
allergyHistory: '', allergyHistory: '',
examinationPurpose: '', examinationPurpose: '',
medicalHistorySummary: '', medicalHistorySummary: '',
expectedExaminationTime: dayjs(new Date()).format('YYYY-MM-DD HH:mm:ss'), expectedExaminationTime: '',
symptom: '', symptom: '',
sign: '', sign: '',
clinicalDiagnosis: '', clinicalDiagnosis: '',
@@ -401,65 +387,12 @@ const handleSyncHistory = async () => {
// 自动带入患者过敏史 // 自动带入患者过敏史
const loadPatientAllergyHistory = () => { const loadPatientAllergyHistory = () => {
if (!effectivePatientInfo.value?.patientId) return; if (!patientInfo.value?.patientId) return;
};
// 加载编辑数据
const loadEditData = () => {
if (!props.isEditMode || !props.editData?.requestFormId) return;
console.log('loadEditData called, editData:', props.editData);
// 解析已有的descJson填充表单
if (props.editData.descJson) {
try {
const obj = JSON.parse(props.editData.descJson);
form.targetDepartment = obj.targetDepartment || '';
form.urgencyLevel = obj.urgencyLevel || 'routine';
form.allergyHistory = obj.allergyHistory || '';
form.examinationPurpose = obj.examinationPurpose || '';
form.medicalHistorySummary = obj.medicalHistorySummary || '';
form.expectedExaminationTime = obj.expectedExaminationTime || '';
form.symptom = obj.symptom || '';
form.sign = obj.sign || '';
form.clinicalDiagnosis = obj.clinicalDiagnosis || '';
form.otherDiagnosis = obj.otherDiagnosis || '';
form.relatedResult = obj.relatedResult || '';
form.attention = obj.attention || '';
form.allergyConfirmed = obj.allergyConfirmed || false;
} catch (e) {
console.error('解析descJson失败:', e);
}
}
// 设置已选项目从requestFormDetailList获取
// 注意:后端返回的字段是 adviceName不是 adviceDefinitionId
console.log('requestFormDetailList:', props.editData.requestFormDetailList);
if (props.editData.requestFormDetailList && props.editData.requestFormDetailList.length) {
// 使用 adviceName 匹配
const selectedNames = props.editData.requestFormDetailList.map(item => item.adviceName);
console.log('setting transferValue by adviceName to:', selectedNames);
// 通过名称匹配找到对应的 key
const selectedIds = [];
applicationList.value?.forEach(app => {
if (selectedNames.includes(app.label?.split(' (')[0])) {
selectedIds.push(app.key);
}
});
console.log('matched selectedIds:', selectedIds);
transferValue.value = selectedIds;
} else {
console.log('requestFormDetailList is empty or undefined');
}
}; };
const projectWithDepartment = (selectProjectIds, type) => { const projectWithDepartment = (selectProjectIds, type) => {
let isRelease = true; let isRelease = true;
const arr = []; const arr = [];
// 确保 applicationList 存在
if (!applicationList.value || !Array.isArray(applicationList.value)) {
return true;
}
selectProjectIds.forEach((element) => { selectProjectIds.forEach((element) => {
const searchData = applicationList.value.find((item) => { const searchData = applicationList.value.find((item) => {
return element == item.adviceDefinitionId; return element == item.adviceDefinitionId;
@@ -496,28 +429,9 @@ const projectWithDepartment = (selectProjectIds, type) => {
}; };
watch(() => transferValue.value, (newValue) => { watch(() => transferValue.value, (newValue) => {
console.log('transferValue changed:', newValue);
console.log('applicationList length:', applicationList.value?.length);
projectWithDepartment(newValue, 1); projectWithDepartment(newValue, 1);
}); });
// 监听 applicationList 加载完成,重新设置已选项目
watch(() => applicationList.value, (newList) => {
if (newList && newList.length > 0 && props.isEditMode && props.editData?.requestFormDetailList?.length) {
console.log('applicationList loaded, re-setting transferValue');
// 使用 adviceName 匹配
const selectedNames = props.editData.requestFormDetailList.map(item => item.adviceName);
const selectedIds = [];
newList.forEach(app => {
const appName = app.label?.split(' (')[0];
if (selectedNames.includes(appName)) {
selectedIds.push(app.key);
}
});
transferValue.value = selectedIds;
}
});
const getPriorityCode = () => { const getPriorityCode = () => {
return form.urgencyLevel === 'emergency' ? 1 : 0; return form.urgencyLevel === 'emergency' ? 1 : 0;
}; };
@@ -545,7 +459,6 @@ const submit = () => {
applicationListAllFilter = applicationListAllFilter.map((item) => { applicationListAllFilter = applicationListAllFilter.map((item) => {
return { return {
adviceDefinitionId: item.adviceDefinitionId, adviceDefinitionId: item.adviceDefinitionId,
adviceName: item.adviceName,
quantity: 1, quantity: 1,
unitCode: item.priceList[0].unitCode, unitCode: item.priceList[0].unitCode,
unitPrice: item.priceList[0].price, unitPrice: item.priceList[0].price,
@@ -557,32 +470,25 @@ const submit = () => {
adviceType: item.adviceType, adviceType: item.adviceType,
definitionId: item.priceList[0].definitionId, definitionId: item.priceList[0].definitionId,
definitionDetailId: item.priceList[0].definitionDetailId, definitionDetailId: item.priceList[0].definitionDetailId,
accountId: effectivePatientInfo.value.accountId, accountId: patientInfo.value.accountId,
}; };
}); });
const submitForm = { ...form, priorityCode: getPriorityCode() }; const submitForm = { ...form, priorityCode: getPriorityCode() };
console.log('提交 descJson:', JSON.stringify(submitForm)); console.log('提交 descJson:', JSON.stringify(submitForm));
// 如果是编辑模式带上requestFormId
const requestFormId = props.isEditMode ? props.editData?.requestFormId : '';
saveCheckd({ saveCheckd({
activityList: applicationListAllFilter, activityList: applicationListAllFilter,
patientId: effectivePatientInfo.value.patientId, patientId: patientInfo.value.patientId,
encounterId: effectivePatientInfo.value.encounterId, encounterId: patientInfo.value.encounterId,
organizationId: effectivePatientInfo.value.inHospitalOrgId, organizationId: patientInfo.value.inHospitalOrgId,
requestFormId: requestFormId, requestFormId: '',
name: applicationListAllFilter.map(item => item.adviceName).join('、'), name: applicationListAllFilter.map(item => item.adviceName).join('、'),
descJson: JSON.stringify(submitForm), descJson: JSON.stringify(submitForm),
categoryEnum: '22', categoryEnum: '2',
}).then((res) => { }).then((res) => {
if (res.code === 200) { if (res.code === 200) {
if (props.isEditMode) { proxy.$message.success(res.msg);
proxy.$message.success(res.msg || '修改成功');
} else {
proxy.$message.success(res.msg);
}
applicationList.value = []; applicationList.value = [];
resetForm(); resetForm();
emits('submitOk'); emits('submitOk');
@@ -598,7 +504,7 @@ const resetForm = () => {
form.allergyHistory = ''; form.allergyHistory = '';
form.examinationPurpose = ''; form.examinationPurpose = '';
form.medicalHistorySummary = ''; form.medicalHistorySummary = '';
form.expectedExaminationTime = dayjs(new Date()).format('YYYY-MM-DD HH:mm:ss'); form.expectedExaminationTime = '';
form.symptom = ''; form.symptom = '';
form.sign = ''; form.sign = '';
form.clinicalDiagnosis = ''; form.clinicalDiagnosis = '';
@@ -618,7 +524,7 @@ const getLocationInfo = () => {
}; };
function getDiagnosisList() { function getDiagnosisList() {
getEncounterDiagnosis(effectivePatientInfo.value.encounterId).then((res) => { getEncounterDiagnosis(patientInfo.value.encounterId).then((res) => {
if (res.code == 200) { if (res.code == 200) {
const datas = (res.data || []).map((item) => { const datas = (res.data || []).map((item) => {
let obj = { ...item }; let obj = { ...item };
@@ -639,24 +545,9 @@ onMounted(() => {
getList(); getList();
getLocationInfo(); getLocationInfo();
loadPatientAllergyHistory(); loadPatientAllergyHistory();
loadEditData();
}); });
// 监听编辑模式变化,重新加载数据 defineExpose({ state, submit, getLocationInfo, getDiagnosisList, resetForm });
watch(
() => props.isEditMode,
(newVal) => {
if (newVal) {
nextTick(() => {
getList();
getLocationInfo();
loadEditData();
});
}
}
);
defineExpose({ state, submit, getLocationInfo, getDiagnosisList, resetForm, getList });
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@@ -681,13 +572,81 @@ $bg-color: #f5f7fa;
background: $bg-color; background: $bg-color;
font-family: -apple-system, BlinkMacSystemFont, 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', sans-serif; font-family: -apple-system, BlinkMacSystemFont, 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', sans-serif;
// 主体内容区 - 紧凑布局 // 顶部标题栏
.form-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 14px 20px;
background: linear-gradient(135deg, #fff 0%, #f0f7ff 100%);
border-bottom: 1px solid $border-color;
.header-left {
display: flex;
align-items: center;
gap: 10px;
.header-icon {
font-size: 24px;
color: $primary-color;
}
.header-title {
font-size: 18px;
font-weight: 600;
color: $text-primary;
letter-spacing: 1px;
}
}
.header-right {
display: flex;
align-items: center;
gap: 12px;
.urgency-label {
font-size: 13px;
color: $text-secondary;
font-weight: 500;
}
.urgency-radio-group {
:deep(.el-radio-button__inner) {
border-radius: 4px;
margin: 0;
}
:deep(.el-radio-button:first-child .el-radio-button__inner) {
border-radius: 4px;
}
:deep(.el-radio-button:last-child .el-radio-button__inner) {
border-radius: 4px;
}
}
.emergency-tip {
display: flex;
align-items: center;
gap: 4px;
color: $danger-color;
font-size: 13px;
font-weight: 500;
background: #fef0f0;
padding: 4px 10px;
border-radius: 4px;
border: 1px solid #fde2e2;
}
}
}
// 主体内容区
.form-body { .form-body {
flex: 1; flex: 1;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 8px; gap: 12px;
padding: 8px 12px; padding: 16px;
overflow-y: auto; overflow-y: auto;
&::-webkit-scrollbar { &::-webkit-scrollbar {
@@ -704,30 +663,47 @@ $bg-color: #f5f7fa;
} }
} }
// 紧急程度栏 - 右上角 // 卡片通用样式
.urgency-bar {
display: flex;
align-items: center;
justify-content: flex-end;
gap: 8px;
padding: 4px 0;
margin-bottom: 4px;
}
.urgency-bar-label {
font-size: 13px;
font-weight: 500;
color: $text-regular;
white-space: nowrap;
}
// 卡片通用样式 - 紧凑
.section-card { .section-card {
background: #fff; background: #fff;
border-radius: 6px; border-radius: 8px;
padding: 8px; padding: 16px;
border: 1px solid #e4e7ed; box-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);
margin-bottom: 4px; border: 1px solid rgba(0, 0, 0, 0.04);
.section-header {
display: flex;
align-items: center;
gap: 8px;
padding-bottom: 12px;
margin-bottom: 12px;
border-bottom: 1px dashed $border-color;
font-size: 14px;
font-weight: 600;
color: $text-primary;
> i {
font-size: 16px;
color: $primary-color;
}
.header-count {
margin-left: auto;
font-size: 12px;
font-weight: 400;
color: $text-secondary;
}
.required-mark {
font-size: 12px;
font-weight: 500;
color: #fff;
background: $danger-color;
padding: 2px 8px;
border-radius: 10px;
margin-left: 4px;
}
}
} }
.transfer-wrapper { .transfer-wrapper {
@@ -741,23 +717,10 @@ $bg-color: #f5f7fa;
display: flex !important; display: flex !important;
flex-direction: row !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;
}
// 信息表单 - 紧凑
.info-form { .info-form {
:deep(.el-form-item) { :deep(.el-form-item) {
margin-bottom: 6px; margin-bottom: 14px;
.el-form-item__label { .el-form-item__label {
font-size: 13px; font-size: 13px;
@@ -787,10 +750,53 @@ $bg-color: #f5f7fa;
} }
} }
// 过敏史危险输入样式 // 过敏史卡片
:deep(.el-textarea__inner.allergy-danger) { .allergy-card {
border-color: $danger-color !important; .allergy-content {
background-color: #fef0f0; .allergy-input-row {
position: relative;
:deep(.el-textarea) {
.el-textarea__inner.allergy-danger {
border-color: $danger-color !important;
background-color: #fef0f0;
}
}
}
.allergy-severe-tag {
position: absolute;
right: 12px;
top: 8px;
display: flex;
align-items: center;
gap: 4px;
color: $danger-color;
font-size: 13px;
font-weight: 600;
background: #fef0f0;
padding: 3px 10px;
border-radius: 12px;
border: 1px solid #fde2e2;
}
.allergy-confirm {
margin-top: 10px;
padding-left: 4px;
}
}
}
// 病史摘要卡片
.history-card {
.section-header {
.sync-btn {
margin-left: auto;
font-size: 12px;
padding: 6px 12px;
border-radius: 16px;
}
}
} }
// 急诊确认弹窗 // 急诊确认弹窗
@@ -829,64 +835,4 @@ $bg-color: #f5f7fa;
.fade-in-linear-leave-to { .fade-in-linear-leave-to {
opacity: 0; opacity: 0;
} }
/* 紧急程度行内布局 */
.urgency-inline {
display: flex;
align-items: center;
gap: 8px;
width: 100%;
}
.emergency-tip-inline {
display: inline-flex;
align-items: center;
gap: 2px;
color: $danger-color;
font-size: 11px;
font-weight: 500;
background: #fef0f0;
padding: 2px 6px;
border-radius: 3px;
white-space: nowrap;
}
/* 过敏史包装 */
.allergy-wrapper {
width: 100%;
}
.allergy-actions {
display: flex;
align-items: center;
gap: 12px;
margin-top: 4px;
}
.allergy-severe-tag-inline {
display: inline-flex;
align-items: center;
gap: 2px;
color: $danger-color;
font-size: 11px;
font-weight: 600;
background: #fef0f0;
padding: 2px 8px;
border-radius: 3px;
}
/* 病史摘要同步按钮 */
.history-field-wrapper {
position: relative;
width: 100%;
}
.history-sync-btn {
position: absolute;
right: 4px;
top: -28px;
font-size: 11px;
padding: 2px 8px;
height: 24px;
}
</style> </style>

View File

@@ -86,9 +86,6 @@ import {getApplicationList, saveSurgery} from './api';
import {ElMessage} from 'element-plus'; import {ElMessage} from 'element-plus';
const { proxy } = getCurrentInstance(); const { proxy } = getCurrentInstance();
// 模块级缓存:避免每次打开弹窗都重新请求手术项目列表
let surgeryRecordsCache = null; // 原始 API 记录
let surgeryMappedCache = null; // 映射后的 el-transfer 数据
// 递归查找树形科室节点 // 递归查找树形科室节点
const findTreeItem = (list, id) => { const findTreeItem = (list, id) => {
if (!list || list.length === 0) return null; if (!list || list.length === 0) return null;
@@ -113,12 +110,6 @@ const getList = () => {
applicationList.value = []; applicationList.value = [];
return; return;
} }
// 命中缓存时直接使用,避免重复请求导致加载缓慢
if (surgeryMappedCache && surgeryMappedCache.length > 0) {
applicationList.value = surgeryMappedCache;
applicationListAll.value = surgeryRecordsCache;
return;
}
loading.value = true; loading.value = true;
getApplicationList({ getApplicationList({
pageSize: 500, pageSize: 500,
@@ -141,9 +132,6 @@ const getList = () => {
key: item.adviceDefinitionId, key: item.adviceDefinitionId,
}; };
}); });
// 写入模块缓存,后续打开弹窗直接复用
surgeryRecordsCache = res.data.records;
surgeryMappedCache = applicationList.value;
} else { } else {
console.warn('获取手术项目列表失败:', res.message); console.warn('获取手术项目列表失败:', res.message);
applicationList.value = []; applicationList.value = [];
@@ -267,7 +255,7 @@ const submit = () => {
requestFormId: '', // 申请单ID requestFormId: '', // 申请单ID
name: '手术申请单', name: '手术申请单',
descJson: JSON.stringify(form), descJson: JSON.stringify(form),
categoryEnum: '24', // 21 检验 22 检查 23 输血 24 手术(避开 adviceType 1-6 碰撞) categoryEnum: '4', // 1 检验 2 检查 3 输血 4 手术
}).then((res) => { }).then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$message.success(res.msg); proxy.$message.success(res.msg);

View File

@@ -197,7 +197,6 @@
style="width: 62%" style="width: 62%"
v-model="scope.row.adviceName" v-model="scope.row.adviceName"
placeholder="请选择项目" placeholder="请选择项目"
@input="handleChange"
@click="handleFocus(scope.row, scope.$index)" @click="handleFocus(scope.row, scope.$index)"
@keyup.enter.stop="handleFocus(scope.row, scope.$index)" @keyup.enter.stop="handleFocus(scope.row, scope.$index)"
@keydown=" @keydown="
@@ -533,7 +532,6 @@ const statusOption = [
let loadingInstance = undefined; let loadingInstance = undefined;
onMounted(() => { onMounted(() => {
document.addEventListener('keydown', escKeyListener); document.addEventListener('keydown', escKeyListener);
getList();
}); });
onBeforeUnmount(() => { onBeforeUnmount(() => {
@@ -574,6 +572,7 @@ function handleTotalAmount() {
} }
}, new Decimal(0)); }, new Decimal(0));
} }
getList();
function getList() { function getList() {
getDiagnosisDefinitionList(queryParams.value).then((res) => { getDiagnosisDefinitionList(queryParams.value).then((res) => {
// prescriptionList.value = res.data.records; // prescriptionList.value = res.data.records;
@@ -585,11 +584,6 @@ function refresh() {
} }
// 获取列表信息 // 获取列表信息
function getListInfo(addNewRow) { function getListInfo(addNewRow) {
// 守护:未选择患者时不发起 API 请求,避免页面加载时循环报错
if (!patientInfo.value || !patientInfo.value.encounterId) {
console.warn('⚠️ getListInfo 跳过:未选择患者');
return;
}
loadingInstance = ElLoading.service({ fullscreen: true }); loadingInstance = ElLoading.service({ fullscreen: true });
setTimeout(() => { setTimeout(() => {
loadingInstance.close(); loadingInstance.close();
@@ -715,17 +709,9 @@ function loadConfiguredCategories() {
// 数据过滤 // 数据过滤
const filterPrescriptionList = computed(() => { const filterPrescriptionList = computed(() => {
const pList = prescriptionList.value.filter((item) => { const pList = prescriptionList.value.filter((item) => {
// 修复 Bug #488orderClassCode 可能是复合值 '1-2',需提取 adviceType 部分进行比较
let matchAdviceType = true;
if (orderClassCode.value) {
const filterAdviceType = String(orderClassCode.value).includes('-')
? parseInt(String(orderClassCode.value).split('-')[0])
: orderClassCode.value;
matchAdviceType = filterAdviceType == item.adviceType;
}
return ( return (
(!therapyEnum.value || therapyEnum.value == item.therapyEnum) && (!therapyEnum.value || therapyEnum.value == item.therapyEnum) &&
matchAdviceType && (!orderClassCode.value || orderClassCode.value == item.adviceType) &&
(!orderStatus.value || (orderStatus.value == item.statusEnum && item.requestId)) (!orderStatus.value || (orderStatus.value == item.statusEnum && item.requestId))
); );
}); });
@@ -753,28 +739,12 @@ function getRowDisabled(row) {
/** /**
* 将行的 adviceType + categoryCode 映射为 el-select 的选中值 * 将行的 adviceType + categoryCode 映射为 el-select 的选中值
* 药品子分类使用复合值如 '1-2'adviceType-categoryCode诊疗/手术/全部使用原始值 * 药品子分类使用复合值如 '1-2'adviceType-categoryCode诊疗/手术/全部使用原始值
* 修复 Bug #488当行的 adviceType 在当前配置中找不到匹配选项时,返回最接近的可用值,避免回显为纯数字
*/ */
function getRowSelectValue(row) { function getRowSelectValue(row) {
if (row.adviceType == 1 && row.categoryCode) { if (row.adviceType == 1 && row.categoryCode) {
const compositeValue = '1-' + row.categoryCode; return '1-' + row.categoryCode;
// 检查复合值是否在选项列表中
if (adviceTypeList.value.some(item => item.value === compositeValue)) {
return compositeValue;
}
// 配置的 categoryCode 已变更,回退到第一个药品选项
const firstPharmacy = adviceTypeList.value.find(item => String(item.value).startsWith('1-'));
if (firstPharmacy) {
return firstPharmacy.value;
}
return row.adviceType;
} }
// 诊疗/手术等非药品类型,检查其值是否在选项列表中 return row.adviceType;
if (adviceTypeList.value.some(item => item.value === row.adviceType)) {
return row.adviceType;
}
// 不在选项中的值(如已废弃的 adviceType返回 undefined 让 el-select 显示为空
return undefined;
} }
// 新增医嘱 // 新增医嘱
@@ -952,9 +922,7 @@ function handleChange(value) {
// 修复Bug #486当行没有显式选择医嘱类型时row?.adviceType为undefined // 修复Bug #486当行没有显式选择医嘱类型时row?.adviceType为undefined
// 不传categoryCode让搜索在全药库中进行只有行已选择类型时才用对应categoryCode过滤 // 不传categoryCode让搜索在全药库中进行只有行已选择类型时才用对应categoryCode过滤
const categoryCode = row?.adviceType !== undefined ? (selectedItem ? selectedItem.categoryCode : '') : ''; const categoryCode = row?.adviceType !== undefined ? (selectedItem ? selectedItem.categoryCode : '') : '';
// 修复Bug #453当adviceType为空字符串或NaN时不传具体类型让refresh函数根据searchKey决定搜索范围 tableRef.refresh(adviceType, categoryCode, value);
const effectiveAdviceType = (adviceType && !isNaN(Number(adviceType))) ? adviceType : '';
tableRef.refresh(effectiveAdviceType, categoryCode, value);
} }
} }
} }
@@ -1225,7 +1193,7 @@ function handleSave() {
groupId: item.groupId, groupId: item.groupId,
uniqueKey: undefined, uniqueKey: undefined,
// 确保 therapyEnum 被正确传递 // 确保 therapyEnum 被正确传递
therapyEnum: parsedContent?.therapyEnum || item.therapyEnum || '1', therapyEnum: parsedContent.therapyEnum || item.therapyEnum || '1',
}; };
}); });
} catch (error) { } catch (error) {
@@ -1248,13 +1216,6 @@ function handleSave() {
if (res.code === 200) { if (res.code === 200) {
proxy.$modal.msgSuccess('签发成功'); proxy.$modal.msgSuccess('签发成功');
isSaving.value = false; isSaving.value = false;
// 乐观更新:立即将已签发医嘱的状态设为"已签发",确保列表实时刷新
saveList.forEach((item) => {
const row = prescriptionList.value.find((r) => r.requestId && r.requestId === item.requestId);
if (row) {
row.statusEnum = 2;
}
});
getListInfo(false); getListInfo(false);
bindMethod.value = {}; bindMethod.value = {};
nextId.value = 1; nextId.value = 1;
@@ -1358,12 +1319,11 @@ function handleCancelEdit(row, index) {
function handleSaveSign(row, index) { function handleSaveSign(row, index) {
if (row.adviceType != 2) { if (row.adviceType != 2) {
// 修复 Bug #488严格校验 itemNo确保非空且为有效字符串才发起请求
let itemNo = row.adviceType == 1 ? row.methodCode : row.adviceDefinitionId; let itemNo = row.adviceType == 1 ? row.methodCode : row.adviceDefinitionId;
if (!itemNo || String(itemNo).trim() === '') { if (!itemNo) {
console.warn('绑定设备检查跳过itemNo为空adviceType=' + row.adviceType + ', adviceName=' + row.adviceName + ''); console.warn('绑定设备检查跳过itemNo为空adviceType=' + row.adviceType + ', adviceName=' + row.adviceName + '');
} else { } else {
getBindDevice({ typeCode: row.adviceType, itemNo: String(itemNo) }).then((res) => { getBindDevice({ typeCode: row.adviceType, itemNo: itemNo }).then((res) => {
if (res.data.length == 0) { if (res.data.length == 0) {
return; return;
} }
@@ -1479,18 +1439,11 @@ function handleSaveBatch() {
.then((res) => { .then((res) => {
if (res.code === 200) { if (res.code === 200) {
proxy.$modal.msgSuccess('保存成功'); proxy.$modal.msgSuccess('保存成功');
// 修复 Bug #405保存成功后锁定所有待保存行,避免医嘱条目仍处于可编辑状态 // 修复#405:保存成功后重置所有待保存行的 isEdit 为 false锁定医嘱不再编辑
// saveList 中的 item 与 prescriptionList 是同一对象引用,直接修改即可
saveList.forEach(item => { saveList.forEach(item => {
item.isEdit = false; const row = prescriptionList.value.find(r => r.uniqueKey === item.uniqueKey);
if (row) row.isEdit = false;
}); });
// 兜底:锁定所有 statusEnum == 1 的行,确保没有遗漏
prescriptionList.value.forEach(row => {
if (row.statusEnum == 1) {
row.isEdit = false;
}
});
expandOrder.value = [];
getListInfo(false); getListInfo(false);
nextId.value = 1; nextId.value = 1;
isSaving.value = false; isSaving.value = false;
@@ -1660,14 +1613,20 @@ function handleSaveGroup(orderGroupList) {
// 创建新的处方项目 // 创建新的处方项目
// 🔧 Bug #403 修复:关键字段使用 null-safe 回退到 mergedDetail已由 setValue 填充完整数据) // 🔧 Bug #403 修复:关键字段使用 null-safe 回退到 mergedDetail已由 setValue 填充完整数据)
// 先取 setValue 填充的行数据作为基础
const baseRow = prescriptionList.value[rowIndex.value];
const newRow = { const newRow = {
...baseRow, ...prescriptionList.value[rowIndex.value],
patientId: patientInfo.value.patientId, patientId: patientInfo.value.patientId,
encounterId: patientInfo.value.encounterId, encounterId: patientInfo.value.encounterId,
accountId: accountId.value, accountId: accountId.value,
quantity: item.quantity ?? mergedDetail.quantity,
methodCode: item.methodCode ?? mergedDetail.methodCode,
rateCode: item.rateCode ?? mergedDetail.rateCode,
dispensePerDuration: item.dispensePerDuration ?? mergedDetail.dispensePerDuration,
dose: item.dose ?? mergedDetail.dose,
doseQuantity: item.doseQuantity ?? mergedDetail.doseQuantity,
executeNum: 1, executeNum: 1,
unitCode: item.unitCode ?? mergedDetail.unitCode,
unitCode_dictText: item.unitCodeName || mergedDetail.unitCodeName || '',
statusEnum: 1, statusEnum: 1,
orgId: resolveOrgId(item.orderDetailInfos?.orgId || mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '', orgId: resolveOrgId(item.orderDetailInfos?.orgId || mergedDetail.orgId || patientInfo.value?.inHospitalOrgId) || '',
// 🔧 修复:同时保存 orgName确保树匹配不到时仍有中文名称可显示 // 🔧 修复:同时保存 orgName确保树匹配不到时仍有中文名称可显示
@@ -1676,33 +1635,19 @@ function handleSaveGroup(orderGroupList) {
conditionId: conditionId.value, conditionId: conditionId.value,
conditionDefinitionId: conditionDefinitionId.value, conditionDefinitionId: conditionDefinitionId.value,
encounterDiagnosisId: encounterDiagnosisId.value, encounterDiagnosisId: encounterDiagnosisId.value,
therapyEnum: baseRow?.therapyEnum || '1', therapyEnum: prescriptionList.value[rowIndex.value]?.therapyEnum || '1',
}; };
// 覆盖关键字段:优先使用 item 的值,其次 mergedDetail已由 setValue 填充),最后 baseRow
newRow.quantity = item.quantity ?? mergedDetail.quantity ?? baseRow.quantity;
newRow.methodCode = item.methodCode ?? mergedDetail.methodCode ?? baseRow.methodCode;
newRow.rateCode = item.rateCode ?? mergedDetail.rateCode ?? baseRow.rateCode;
newRow.dispensePerDuration = item.dispensePerDuration ?? mergedDetail.dispensePerDuration ?? baseRow.dispensePerDuration;
newRow.dose = item.dose ?? mergedDetail.dose ?? baseRow.dose;
newRow.doseQuantity = item.doseQuantity ?? mergedDetail.doseQuantity ?? baseRow.doseQuantity;
newRow.unitCode = item.unitCode ?? mergedDetail.unitCode ?? baseRow.unitCode;
newRow.unitCode_dictText = item.unitCodeName || mergedDetail.unitCodeName || baseRow.unitCode_dictText || '';
// 计算价格和总量 // 计算价格和总量
// 🔧 Bug #403 修复:使用 newRow.unitCode已由 setValue 填充)而非 item.unitCode const unitInfo = unitCodeList.value.find((k) => k.value == item.unitCode);
// 使用 ?? 替代 || 计算 partPercent确保值为 0 时不会被错误替换
const finalUnitCode = newRow.unitCode;
const unitInfo = unitCodeList.value.find((k) => k.value == finalUnitCode);
const finalQuantity = newRow.quantity;
const partPercent = item.orderDetailInfos?.partPercent ?? mergedDetail.partPercent ?? baseRow.partPercent ?? 1;
if (unitInfo && unitInfo.type == 'minUnit') { if (unitInfo && unitInfo.type == 'minUnit') {
newRow.price = newRow.minUnitPrice; newRow.price = newRow.minUnitPrice;
newRow.totalPrice = ((finalQuantity || 0) * newRow.minUnitPrice).toFixed(6); newRow.totalPrice = (item.quantity * newRow.minUnitPrice).toFixed(6);
newRow.minUnitQuantity = finalQuantity || 0; newRow.minUnitQuantity = item.quantity;
} else { } else {
newRow.price = newRow.unitPrice; newRow.price = newRow.unitPrice;
newRow.totalPrice = ((finalQuantity || 0) * newRow.unitPrice).toFixed(6); newRow.totalPrice = (item.quantity * newRow.unitPrice).toFixed(6);
newRow.minUnitQuantity = (finalQuantity || 0) * partPercent; newRow.minUnitQuantity = item.quantity * (item.orderDetailInfos?.partPercent || mergedDetail.partPercent || 1);
} }
newRow.contentJson = JSON.stringify(newRow); newRow.contentJson = JSON.stringify(newRow);

View File

@@ -19,9 +19,11 @@
<el-tab-pane label="检验申请" name="test"> <el-tab-pane label="检验申请" name="test">
<TestApplication ref="testApplicationRef" :show-status-column="true" /> <TestApplication ref="testApplicationRef" :show-status-column="true" />
</el-tab-pane> </el-tab-pane>
```vue
<el-tab-pane label="检查申请" name="examine"> <el-tab-pane label="检查申请" name="examine">
<ExamineApplication ref="examineApplicationRef" /> <ExamineApplication ref="examineApplicationRef" />
</el-tab-pane> </el-tab-pane>
```
<el-tab-pane label="汇总发药申请" name="summaryDrug"> <el-tab-pane label="汇总发药申请" name="summaryDrug">
<SummaryDrugApplication ref="summaryDrugApplicationRef" /> <SummaryDrugApplication ref="summaryDrugApplicationRef" />
</el-tab-pane> </el-tab-pane>
@@ -47,7 +49,6 @@
import {computed, onBeforeMount, onMounted, provide, reactive, ref, watch,} from 'vue'; import {computed, onBeforeMount, onMounted, provide, reactive, ref, watch,} from 'vue';
import Emr from './emr/index.vue'; import Emr from './emr/index.vue';
import SummaryDrugApplication from './components/applicationShow/summaryDrugApplication.vue';
import inPatientBarDoctorFold from '@/components/patientBar/inPatientBarDoctorFold.vue'; import inPatientBarDoctorFold from '@/components/patientBar/inPatientBarDoctorFold.vue';
import PatientList from '@/components/PatientList/patient-list.vue'; import PatientList from '@/components/PatientList/patient-list.vue';
import {localPatientInfo, updateLocalPatientInfo} from './store/localPatient'; import {localPatientInfo, updateLocalPatientInfo} from './store/localPatient';
@@ -83,7 +84,6 @@ const currentPatientInfo = ref({});
const testApplicationRef = ref(); const testApplicationRef = ref();
const examineApplicationRef = ref(); const examineApplicationRef = ref();
const surgeryApplicationRef = ref(); const surgeryApplicationRef = ref();
const summaryDrugApplicationRef = ref();
const bloodTtransfusionAapplicationRef = ref(); const bloodTtransfusionAapplicationRef = ref();
// 患者列表相关逻辑 // 患者列表相关逻辑

View File

@@ -251,7 +251,7 @@
</div> </div>
</el-dialog> </el-dialog>
<!-- 划价组套选择对话框 --> <!-- 划价组套选择对话框 -->
<el-dialog v-model="groupSetDialogVisible" title="划价组套选择" width="600px" :close-on-click-modal="false" append-to-body :z-index="3000"> <el-dialog v-model="groupSetDialogVisible" title="划价组套选择" width="600px" :close-on-click-modal="false" append-to-body>
<div style="margin-bottom: 15px; display: flex; align-items: center; gap: 10px"> <div style="margin-bottom: 15px; display: flex; align-items: center; gap: 10px">
<el-input <el-input
v-model="groupSetSearchText" v-model="groupSetSearchText"
@@ -296,7 +296,6 @@
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<el-empty v-if="!groupSetLoading && groupSetList.length === 0" description="暂无划价组套数据" :image-size="80" />
<div style="margin-top: 15px; text-align: right"> <div style="margin-top: 15px; text-align: right">
<el-button @click="groupSetDialogVisible = false">取消</el-button> <el-button @click="groupSetDialogVisible = false">取消</el-button>
<el-button type="primary" @click="applyGroupSet" :disabled="!selectedGroupSet">应用</el-button> <el-button type="primary" @click="applyGroupSet" :disabled="!selectedGroupSet">应用</el-button>
@@ -482,68 +481,19 @@ watch(
(visible) => { (visible) => {
if (visible) { if (visible) {
executeTime.value = formatDateStr(new Date(), 'YYYY-MM-DD HH:mm:ss'); executeTime.value = formatDateStr(new Date(), 'YYYY-MM-DD HH:mm:ss');
// 弹窗打开时重新加载科室和位置选项,确保数据最新
loadDepartmentOptions();
getDiseaseInitLoc(16);
} else { } else {
resetData(); resetData();
} }
} }
); );
// 监听科室选项加载完成,为已添加的诊疗项目设置默认执行科室 // 加载科室选项
watch(
() => departmentOptions.value,
(depts) => {
if (!depts || depts.length === 0) return;
feeItemsList.value.forEach(item => {
if (item.adviceType === 3 && !item.positionId) {
const patientOrgId = props.patientInfo.organizationId;
const matched = depts.find(d => String(d.id) === String(patientOrgId));
item.positionId = matched ? String(matched.id) : String(depts[0].id);
}
});
}
);
// 监听位置选项加载完成,为已添加的耗材项目设置默认位置
watch(
() => locationOptions.value,
(locs) => {
if (!locs || locs.length === 0) return;
feeItemsList.value.forEach(item => {
if (item.adviceType === 2 && !item.positionId) {
item.positionId = String(locs[0].value);
}
});
}
);
// 加载科室选项(支持树形/扁平两种数据结构)
function loadDepartmentOptions() { function loadDepartmentOptions() {
getOrgList() getOrgList()
.then((res) => { .then((res) => {
if (res.data) { if (res.data && res.data.records && res.data.records.length > 0) {
// 尝试从树形结构中取:records[0].children departmentOptions.value = res.data.records[0].children || [];
if (res.data.records && res.data.records.length > 0) {
if (res.data.records[0].children && res.data.records[0].children.length > 0) {
departmentOptions.value = res.data.records[0].children;
return;
}
// 如果 records[0] 有 id 和 name非树根节点直接用所有 records
if (res.data.records[0].id) {
departmentOptions.value = res.data.records;
return;
}
}
// 兜底:如果 records 不存在或为空,尝试直接使用 data 本身
if (Array.isArray(res.data)) {
departmentOptions.value = res.data;
return;
}
} }
// 所有方式都失败,置空
departmentOptions.value = [];
}) })
.catch(() => { .catch(() => {
console.warn('科室列表加载失败(可能无权限)'); console.warn('科室列表加载失败(可能无权限)');
@@ -577,27 +527,23 @@ function getDiseaseInitLoc() {
locationOptions.value = []; locationOptions.value = [];
}); });
} }
// 下拉框模糊搜索过滤自定义filter-method配合element-plus filterable使用 // 下拉框模糊搜索过滤
function filterOptions(val, row, optionsKey) { function filterOptions(val, row, optionsKey) {
const key = row.adviceDefinitionId + '_' + optionsKey; const key = row.adviceDefinitionId + '_' + optionsKey;
if (!val || val.trim() === '') { filterKeywords.value[key] = val;
delete filterKeywords.value[key];
} else {
filterKeywords.value[key] = val.toLowerCase();
}
} }
function getFilteredOptions(row, optionsKey) { function getFilteredOptions(row, optionsKey) {
const key = row.adviceDefinitionId + '_' + optionsKey; const key = row.adviceDefinitionId + '_' + optionsKey;
const keyword = filterKeywords.value[key]; const keyword = filterKeywords.value[key];
const options = optionsKey === 'departmentOptions' ? departmentOptions.value : locationOptions.value; const options = optionsKey === 'departmentOptions' ? departmentOptions.value : locationOptions.value;
if (!keyword) { if (!keyword || keyword.trim() === '') {
return options; return options;
} }
const lower = keyword.toLowerCase();
return options.filter(item => { return options.filter(item => {
const name = (item.name || item.label || '').toLowerCase(); const name = (item.name || item.label || '').toLowerCase();
const id = String(item.id || item.value || '').toLowerCase(); const id = String(item.id || item.value || '').toLowerCase();
const py = (item.pyStr || '').toLowerCase(); return name.includes(lower) || id.includes(lower);
return name.includes(keyword) || id.includes(keyword) || py.includes(keyword);
}); });
} }
// 获取组套类型文本 // 获取组套类型文本
@@ -606,26 +552,19 @@ function getItemType_Text(type) {
return map[type] || '其他'; return map[type] || '其他';
} }
function getUnitCodeOptions(row) { function getUnitCodeOptions(row) {
const unitCodes = []; const unitCodes = [
// 大单位:优先用 codecode 缺失时用字典文本兜底 { code: row.unitCode, codeText: row.unitCode_dictText },
if (row.unitCode != null && String(row.unitCode) !== '') { { code: row.minUnitCode, codeText: row.minUnitCode_dictText },
unitCodes.push({ code: String(row.unitCode), codeText: row.unitCode_dictText }); ];
} else if (row.unitCode_dictText) { // 使用 Set 来跟踪已经存在的 code
unitCodes.push({ code: row.unitCode_dictText, codeText: row.unitCode_dictText });
}
// 小单位:同上
if (row.minUnitCode != null && String(row.minUnitCode) !== '') {
unitCodes.push({ code: String(row.minUnitCode), codeText: row.minUnitCode_dictText });
} else if (row.minUnitCode_dictText) {
unitCodes.push({ code: row.minUnitCode_dictText, codeText: row.minUnitCode_dictText });
}
// 去重
const seenCodes = new Set(); const seenCodes = new Set();
const uniqueUnitCodes = unitCodes.filter((item) => { const uniqueUnitCodes = unitCodes.filter((item) => {
// 如果 Set 中没有这个 code就保留它并把它加入 Set
if (!seenCodes.has(item.code)) { if (!seenCodes.has(item.code)) {
seenCodes.add(item.code); seenCodes.add(item.code);
return true; return true;
} }
// 如果已经存在,就过滤掉
return false; return false;
}); });
return uniqueUnitCodes; return uniqueUnitCodes;
@@ -636,11 +575,11 @@ function unitCodeChange(row) {
// 获取价格 // 获取价格
const price = row.priceList?.[0]?.price || 0; const price = row.priceList?.[0]?.price || 0;
// 根据选择的单位调整单价(统一用字符串比较) // 根据选择的单位调整单价
if (String(row.selectUnitCode) === String(row.unitCode)) { if (row.selectUnitCode === row.unitCode) {
// 如果选择的是大单位 (如 "") // 如果选择的是大单位 (如 "")
row.unitPrice = price.toFixed(6); // 单价就是原价 row.unitPrice = price.toFixed(6); // 单价就是原价
} else if (String(row.selectUnitCode) === String(row.minUnitCode)) { } else if (row.selectUnitCode === row.minUnitCode) {
// 如果选择的是小单位 (如 "") // 如果选择的是小单位 (如 "")
row.unitPrice = (price / (row.partPercent || 1)).toFixed(6); // 单价 = 原价 / 拆零比 row.unitPrice = (price / (row.partPercent || 1)).toFixed(6); // 单价 = 原价 / 拆零比
} }
@@ -688,36 +627,14 @@ function selectChange(row) {
defaultUnitCode = String(row.minUnitCode); defaultUnitCode = String(row.minUnitCode);
} else if (row.unitCode) { } else if (row.unitCode) {
defaultUnitCode = String(row.unitCode); defaultUnitCode = String(row.unitCode);
} else if (row.minUnitCode_dictText) {
// 兜底:如果 minUnitCode 为空但字典文本存在,使用文本作为选项值
defaultUnitCode = row.minUnitCode_dictText;
} else if (row.unitCode_dictText) {
defaultUnitCode = row.unitCode_dictText;
}
// 如果默认单位不在 uniqueUnitCodes 中,添加兜底选项
if (defaultUnitCode && !uniqueUnitCodes.some(u => u.code === defaultUnitCode)) {
uniqueUnitCodes.push({ code: defaultUnitCode, codeText: defaultUnitCode });
} }
// 设置默认执行科室/位置(统一转为字符串,避免 el-select 类型不匹配) // 设置默认执行科室/位置(统一转为字符串,避免 el-select 类型不匹配)
let defaultPositionId = undefined; let defaultPositionId = undefined;
if (row.adviceType === 3 && departmentOptions.value.length > 0) { if (row.adviceType === 3 && departmentOptions.value.length > 0) {
// 诊疗: // 诊疗:优先使用患者所在科室,否则取第一个科室
// 1. 优先使用诊疗目录项目的"所属科室"row.orgId const patientOrgId = props.patientInfo.organizationId;
// 2. 其次使用患者当前病房科室patientInfo.organizationId const matched = departmentOptions.value.find(d => String(d.id) === String(patientOrgId));
// 3. 最后取第一个科室 defaultPositionId = matched ? String(matched.id) : String(departmentOptions.value[0].id);
const orgIdPriority = [row.orgId, props.patientInfo.organizationId];
for (const id of orgIdPriority) {
if (id) {
const matched = departmentOptions.value.find(d => String(d.id) === String(id));
if (matched) {
defaultPositionId = String(matched.id);
break;
}
}
}
if (!defaultPositionId) {
defaultPositionId = String(departmentOptions.value[0].id);
}
} else if (row.adviceType === 2 && locationOptions.value.length > 0) { } else if (row.adviceType === 2 && locationOptions.value.length > 0) {
// 耗材:默认取第一个药房/耗材房 // 耗材:默认取第一个药房/耗材房
defaultPositionId = String(locationOptions.value[0].value); defaultPositionId = String(locationOptions.value[0].value);
@@ -846,7 +763,6 @@ function resetData() {
// 划价组套相关功能 // 划价组套相关功能
function openGroupSetDialog() { function openGroupSetDialog() {
console.log('openGroupSetDialog called');
groupSetDialogVisible.value = true; groupSetDialogVisible.value = true;
groupSetSearchText.value = ''; groupSetSearchText.value = '';
selectedGroupSet.value = null; selectedGroupSet.value = null;
@@ -855,37 +771,19 @@ function openGroupSetDialog() {
function loadGroupSets() { function loadGroupSets() {
groupSetLoading.value = true; groupSetLoading.value = true;
const params = { organizationId: orgId.value }; getOrderGroup({ organizationId: orgId.value })
// 传递搜索关键字,后端 /group-package-for-order 虽不直接支持 searchKey
// 但保持参数传递以便后续扩展
if (groupSetSearchText.value && groupSetSearchText.value.trim()) {
params.searchKey = groupSetSearchText.value.trim();
}
getOrderGroup(params)
.then((res) => { .then((res) => {
const data = res?.data || {}; const data = res?.data || {};
let rawList = [];
if (groupSetRange.value === 1) { if (groupSetRange.value === 1) {
rawList = data.personalList || []; groupSetList.value = data.personalList || [];
} else if (groupSetRange.value === 2) { } else if (groupSetRange.value === 2) {
rawList = data.organizationList || []; groupSetList.value = data.organizationList || [];
} else { } else {
rawList = data.hospitalList || []; groupSetList.value = data.hospitalList || [];
}
// 客户端过滤:根据搜索关键字过滤组套名称
const keyword = groupSetSearchText.value?.trim()?.toLowerCase();
if (keyword) {
groupSetList.value = rawList.filter(item => {
const name = (item.name || item.Name || '').toLowerCase();
return name.includes(keyword);
});
} else {
groupSetList.value = rawList;
} }
}) })
.catch((err) => { .catch(() => {
console.warn('组套列表加载失败(可能无权限):', err); console.warn('组套列表加载失败(可能无权限)');
ElMessage.warning('组套列表加载失败,当前暂无可用组套');
groupSetList.value = []; groupSetList.value = [];
}) })
.finally(() => { .finally(() => {

View File

@@ -28,7 +28,6 @@ export function getOrgList() {
return request({ return request({
url: '/base-data-manage/organization/organization', url: '/base-data-manage/organization/organization',
method: 'get', method: 'get',
params: { pageSize: 100, pageNum: 1 },
}); });
} }
/** /**

View File

@@ -463,45 +463,20 @@ function watchPatientSelection() {
}, 300); }, 300);
} }
/** 查询科室(支持树形/扁平多种响应结构) */ /** 查询科室 */
const getLocationInfo = () => { const getLocationInfo = () => {
getOrgList().then((res) => { getOrgList().then((res) => {
if (!res.data) { orgOptions.value = res.data?.records[0]?.children;
orgOptions.value = [];
return;
}
// 尝试从树形结构取records[0].children
if (res.data.records && res.data.records.length > 0) {
if (res.data.records[0].children && res.data.records[0].children.length > 0) {
orgOptions.value = res.data.records[0].children;
return;
}
// 如果 records[0] 有 id 和 name非树根节点直接用所有 records
if (res.data.records[0].id) {
orgOptions.value = res.data.records;
return;
}
}
// 兜底:如果 data 本身是数组
if (Array.isArray(res.data)) {
orgOptions.value = res.data;
return;
}
orgOptions.value = [];
}).catch(() => {
console.warn('科室列表加载失败(可能无权限)');
orgOptions.value = [];
}); });
}; };
getLocationInfo(); getLocationInfo();
// 映射(查找失败时返回 '-' 而非显示内码) // 映射
const selectOrg = (itemid) => { const selectOrg = (itemid) => {
if (!itemid) return '-';
const item = orgOptions.value.find((item) => { const item = orgOptions.value.find((item) => {
return item.id == itemid; return item.id == itemid;
}); });
return item?.name || '-'; return item?.name;
}; };
// 重置 // 重置
const onReset = () => { const onReset = () => {

View File

@@ -66,21 +66,13 @@ const props = defineProps({
type: Number, type: Number,
default: 1, default: 1,
}, },
therapyEnum: {
type: Number,
default: undefined,
},
}); });
handleGetPrescription(); handleGetPrescription();
function handleGetPrescription() { function handleGetPrescription() {
loading.value = true; loading.value = true;
let encounterIds = patientInfoList.value.map((i) => i.encounterId).join(','); let encounterIds = patientInfoList.value.map((i) => i.encounterId).join(',');
const params = {}; getMedicineSummary({}).then((res) => {
if (props.therapyEnum !== undefined) {
params.therapyEnum = props.therapyEnum;
}
getMedicineSummary(params).then((res) => {
medicineSummaryFormList.value = res.data.records; medicineSummaryFormList.value = res.data.records;
loading.value = false; loading.value = false;
}); });

View File

@@ -85,7 +85,7 @@
:deadline="deadline" :deadline="deadline"
:therapyEnum="therapyEnum" :therapyEnum="therapyEnum"
/> />
<SummaryMedicineList v-else ref="summaryMedicineRefs" :therapyEnum="therapyEnum" /> <SummaryMedicineList v-else />
<!-- <el-tabs v-model="activeName" class="demo-tabs centered-tabs" @tab-change="handleClick"> <!-- <el-tabs v-model="activeName" class="demo-tabs centered-tabs" @tab-change="handleClick">
<el-tab-pane <el-tab-pane
v-for="tab in prescriptionTabs" v-for="tab in prescriptionTabs"
@@ -129,7 +129,6 @@ const therapyEnum = ref(undefined);
// 存储子组件引用的对象 // 存储子组件引用的对象
const prescriptionRefs = ref(); const prescriptionRefs = ref();
const summaryMedicineRefs = ref();
const navigationButtons = inpatientNurseNavs; const navigationButtons = inpatientNurseNavs;
@@ -166,11 +165,7 @@ function handleClick(tabName) {
function handleGetPrescription() { function handleGetPrescription() {
chooseAll.value = false; chooseAll.value = false;
if (isDetails.value == '1') { prescriptionRefs.value?.handleGetPrescription();
prescriptionRefs.value?.handleGetPrescription();
} else {
summaryMedicineRefs.value?.handleGetPrescription();
}
} }
function handelSwicthChange(value) { function handelSwicthChange(value) {

View File

@@ -458,9 +458,7 @@ const loadPatientInfo = () => {
'YYYY-MM-DD HH:mm:ss' 'YYYY-MM-DD HH:mm:ss'
); );
} else { } else {
// 已有患者entranceType == 1不自动填充当前时间避免覆盖历史数据 interventionForm.value.startTime = dayjs(new Date()).format('YYYY-MM-DD HH:mm:ss');
// 新入科患者由后端默认返回当前时间,或由用户手动选择
interventionForm.value.startTime = '';
} }
interventionForm.value.height = res.data.height; interventionForm.value.height = res.data.height;
interventionForm.value.weight = res.data.weight; interventionForm.value.weight = res.data.weight;

View File

@@ -459,15 +459,10 @@ function handleGetPrescription(skipAutoSelectAll = false) {
// 执行 // 执行
function handleExecute() { function handleExecute() {
let list = getSelectRows(); let list = getSelectRows();
if (list.length === 0) {
proxy.$modal.msgWarning('请选择需要执行的医嘱');
return;
}
let encounterIds = patientInfoList.value.map((i) => i.encounterId).join(','); let encounterIds = patientInfoList.value.map((i) => i.encounterId).join(',');
list = list.map((item) => { list = list.map((item) => {
return { return {
requestId: item.requestId, requestId: item.requestId,
encounterId: item.encounterId,
accountId: item.accountId, accountId: item.accountId,
adviceTable: item.adviceTable, adviceTable: item.adviceTable,
executeTimes: item.executeTimes, executeTimes: item.executeTimes,
@@ -476,25 +471,14 @@ function handleExecute() {
console.log(list, 'list'); console.log(list, 'list');
adviceExecute({ exeDate: exeDate.value, adviceExecuteDetailList: list }).then((res) => { adviceExecute({ exeDate: exeDate.value, adviceExecuteDetailList: list }).then((res) => {
if (res.code == 200) { if (res.code == 200) {
// 仅当选中医嘱中包含诊疗类医嘱(可能绑定耗材)时,才调用耗材批号匹配 // 仅当选中医嘱中包含耗材类医嘱时,才调用耗材批号匹配(排除纯药品医嘱场景)
// adviceTable 取值为 med_medication_request药品或 wor_service_request诊疗/耗材) const hasDevice = list.some((item) =>
const hasServiceRequest = list.some((item) => String(item.adviceTable || '').includes('device'),
String(item.adviceTable || '') === 'wor_service_request',
); );
if (hasServiceRequest) { if (hasDevice) {
// 仅传入选中医嘱对应的 encounterId避免其他患者的耗材记录干扰 lotNumberMatch({ encounterIdList: encounterIds }, { skipErrorMsg: true }).catch((error) => {
const selectedEncounterIds = [...new Set(list.map((item) => item.encounterId).filter(Boolean))]; console.warn('lotNumberMatch failed after adviceExecute:', error);
if (selectedEncounterIds.length > 0) { });
lotNumberMatch({ encounterIdList: selectedEncounterIds }, { skipErrorMsg: true })
.then((matchRes) => {
if (matchRes && matchRes.code !== 200) {
console.warn('lotNumberMatch returned error:', matchRes.msg);
}
})
.catch((error) => {
console.warn('lotNumberMatch failed after adviceExecute:', error);
});
}
} }
// 刷新列表(不自动全选,保持用户操作前的选择状态) // 刷新列表(不自动全选,保持用户操作前的选择状态)
handleGetPrescription(true); handleGetPrescription(true);
@@ -502,9 +486,6 @@ function handleExecute() {
} else { } else {
proxy.$modal.msgError(res.msg || '医嘱执行失败'); proxy.$modal.msgError(res.msg || '医嘱执行失败');
} }
}).catch((error) => {
console.error('医嘱执行接口调用失败:', error);
proxy.$modal.msgError('医嘱执行失败,请稍后重试');
}); });
} }

View File

@@ -61,11 +61,7 @@ const props = defineProps({
type: String, type: String,
default: '', default: '',
}, },
/** 表头所选出库仓库:传入后药品列表只含该仓有库存的行,避免选到别仓批号导致 inventory-item-info 一直为 0 */
orgLocationId: {
type: [String, Number],
default: undefined,
},
}); });
const emit = defineEmits(['selectRow']); const emit = defineEmits(['selectRow']);
const queryParams = ref({ const queryParams = ref({
@@ -93,14 +89,11 @@ watch(
queryParams.value.searchKey = newValue.searchKey; queryParams.value.searchKey = newValue.searchKey;
queryParams.value.itemType = newValue.itemType; queryParams.value.itemType = newValue.itemType;
queryParams.value.purchaseFlag = 0; queryParams.value.purchaseFlag = 0;
if (newValue.orgLocationId != null && newValue.orgLocationId !== '') { // queryParams.value.sourceLocationId = newValue.sourceLocationId;
queryParams.value.orgLocationId = newValue.orgLocationId; // queryParams.value.purposeLocationId = newValue.purposeLocationId;
} else {
delete queryParams.value.orgLocationId;
}
throttledGetList(); throttledGetList();
}, },
{ immediate: true, deep: true } { immdiate: true, deep: true }
); );
getList(); getList();

View File

@@ -133,7 +133,6 @@
filterable filterable
style="width: 200px" style="width: 200px"
:disabled="data.isEdit" :disabled="data.isEdit"
@change="onHeaderWarehouseChange"
> >
<el-option <el-option
v-for="item in purposeTypeListOptions" v-for="item in purposeTypeListOptions"
@@ -223,7 +222,6 @@
@selectRow="(row) => selectRow(row, scope.$index)" @selectRow="(row) => selectRow(row, scope.$index)"
:searchKey="medicineSearchKey" :searchKey="medicineSearchKey"
:itemType="itemType" :itemType="itemType"
:orgLocationId="receiptHeaderForm.headerLocationId"
/> />
</template> </template>
</PopoverList> </PopoverList>
@@ -485,46 +483,6 @@ import {useStore} from '@/store/store';
import useTagsViewStore from '@/store/modules/tagsView'; import useTagsViewStore from '@/store/modules/tagsView';
import TraceNoDialog from '@/components/OpenHis/TraceNoDialog/index.vue' import TraceNoDialog from '@/components/OpenHis/TraceNoDialog/index.vue'
/** 领用保存 IssueDto后端 Jackson 只认 yyyy-MM-dd HH:mm:ss库存接口可能回传 2025/4/2 00:00:00 等 */
function toIssueDateTimeStr(val) {
if (val == null || val === '') return undefined;
if (typeof val === 'string') {
const s = val.trim();
if (/^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}$/.test(s)) return s;
if (/^\d{4}-\d{2}-\d{2}$/.test(s)) return `${s} 00:00:00`;
}
const d = val instanceof Date ? val : new Date(val);
if (Number.isNaN(d.getTime())) return undefined;
return formatDate(d);
}
/** 总库存 totalQuantity 为最小单位;领用数量按当前计量单位折算成最小单位后再比较 */
function getRequisitionQtyInMinUnit(r) {
const q = Number(r.itemQuantity ?? 0);
if (!Number.isFinite(q) || q < 0) return NaN;
const minCode = r.unitList?.minUnitCode;
if (!minCode || r.unitCode === minCode) return q;
const part = Number(r.partPercent ?? 1);
return q * part;
}
function validateRequisitionQtyVsStock(r, lineNo) {
const cap = Number(r.totalQuantity ?? 0);
const reqMin = getRequisitionQtyInMinUnit(r);
if (!Number.isFinite(reqMin)) {
return `${lineNo}行:领用数量请输入有效数字`;
}
if (cap > 0 && reqMin > cap + 1e-9) {
const name = r.name || `${lineNo}`;
return `${name}:领用数量(折合最小单位)不能超过当前仓库可领库存 ${cap},请修改后再保存。`;
}
if (cap <= 0 && reqMin > 0) {
const name = r.name || `${lineNo}`;
return `${name}:当前仓库可领库存为 0不能填写正数领用数量。`;
}
return null;
}
const tagsViewStore = useTagsViewStore(); const tagsViewStore = useTagsViewStore();
const store = useStore(); const store = useStore();
@@ -534,7 +492,6 @@ const userStore = useUserStore();
const openTraceNoDialog = ref(false) const openTraceNoDialog = ref(false)
const rowData = ref({}) const rowData = ref({})
const ypName = ref('') const ypName = ref('')
const currentIndex = ref(-1)
const { proxy } = getCurrentInstance(); const { proxy } = getCurrentInstance();
const { warehous_type, category_code, service_type_code, specialty_code, purchase_type } = const { warehous_type, category_code, service_type_code, specialty_code, purchase_type } =
@@ -611,8 +568,6 @@ const data = reactive({
medicationType: [{ required: true, message: '请选择药品类型', trigger: 'change' }], medicationType: [{ required: true, message: '请选择药品类型', trigger: 'change' }],
locationId: [{ required: true, message: '请选择领用部门', trigger: 'change' }], locationId: [{ required: true, message: '请选择领用部门', trigger: 'change' }],
practitionerId: [{ required: true, message: '请选择部门经手人', trigger: 'change' }], practitionerId: [{ required: true, message: '请选择部门经手人', trigger: 'change' }],
// 领用出库按「表头仓库」查 /app-common/inventory-item-info未选仓库会查不到库存并误报「仓库数量为0」
headerLocationId: [{ required: true, message: '请先选择仓库(按仓库查询可领用库存)', trigger: 'change' }],
}, },
tableRules: { tableRules: {
name: [{ required: true, message: '项目不能为空', trigger: 'change' }], name: [{ required: true, message: '项目不能为空', trigger: 'change' }],
@@ -1033,9 +988,10 @@ function selectRow(rowValue, index) {
form.purchaseinventoryList[index].unitList = rowValue.unitList[0]; form.purchaseinventoryList[index].unitList = rowValue.unitList[0];
form.purchaseinventoryList[index].lotNumber = rowValue.lotNumber; form.purchaseinventoryList[index].lotNumber = rowValue.lotNumber;
form.purchaseinventoryList[index].ybNo = rowValue.ybNo; form.purchaseinventoryList[index].ybNo = rowValue.ybNo;
// 出库仓库:优先表头当前所选仓库(避免先选药后选仓时行上一直为空) // #439 fix: 不清空sourceLocationId保留handleAddRow设置的仓库ID
form.purchaseinventoryList[index].sourceLocationId = if (!form.purchaseinventoryList[index].sourceLocationId) {
receiptHeaderForm.headerLocationId || form.purchaseinventoryList[index].sourceLocationId || ''; form.purchaseinventoryList[index].sourceLocationId = receiptHeaderForm.headerLocationId || '';
}
getPharmacyCabinetList().then((res) => { getPharmacyCabinetList().then((res) => {
purposeTypeListOptions.value = res.data; purposeTypeListOptions.value = res.data;
handleLocationClick(1, rowValue, index) handleLocationClick(1, rowValue, index)
@@ -1048,125 +1004,78 @@ function selectRow(rowValue, index) {
}); });
} }
/** 多条库存记录时取可领数量最大的一条(避免仅取 res.data[0] 恰好为 0 */ // 选择仓库
function pickBestOrgQuantityRow(list) {
if (!Array.isArray(list) || list.length === 0) return null;
return list.reduce((best, cur) => {
const cq = Number(cur?.orgQuantity ?? 0);
const bq = Number(best?.orgQuantity ?? 0);
return cq > bq ? cur : best;
});
}
/** 表头「仓库」变化:同步每行 sourceLocationId 并重新拉库存(修复先选药品后选仓库行上仍无仓库 ID */
function onHeaderWarehouseChange() {
const hid = receiptHeaderForm.headerLocationId;
form.purchaseinventoryList.forEach((r) => {
r.sourceLocationId = hid || '';
});
form.purchaseinventoryList.forEach((r, idx) => {
if (hid && r.itemId) {
handleLocationClick(1, {}, idx);
}
});
}
// 选择仓库 / 选药品后拉取该仓库存
function handleLocationClick(item, row, index) { function handleLocationClick(item, row, index) {
const r = form.purchaseinventoryList[index]; getCount({
let orgLocationId = r.sourceLocationId || receiptHeaderForm.headerLocationId || ''; itemId: form.purchaseinventoryList[index].itemId,
if (!orgLocationId) { orgLocationId: form.purchaseinventoryList[index].sourceLocationId,
proxy.$message.warning('请先在表头选择「仓库」。库存按仓库维度查询,未选仓库无法匹配您看到的总库存。'); // objLocationId:purposeLocationId,
r.totalQuantity = 0; lotNumber: form.purchaseinventoryList[index].lotNumber,
r.price = 0; }).then((res) => {
return; if (res.data && res.data.length > 0) {
} form.purchaseinventoryList[index].itemTable = res.data[0].itemTable || '';
if (!r.sourceLocationId) { form.purchaseinventoryList[index].totalQuantity = res.data[0].orgQuantity || 0;
r.sourceLocationId = orgLocationId;
}
const lotTrimmed = if (res.data[0].price) {
r.lotNumber != null && String(r.lotNumber).trim() !== '' ? String(r.lotNumber).trim() : null; form.purchaseinventoryList[index].price = res.data[0].price.toFixed(4);
} else {
const runGet = (withLot) => { form.purchaseinventoryList[index].price = 0;
const params = { itemId: r.itemId, orgLocationId }; }
if (withLot && lotTrimmed) { // 获取供应商id
params.lotNumber = lotTrimmed; form.purchaseinventoryList[index].supplierId = res.data[0].supplierId || '';
} // 生产日期
return getCount(params); form.purchaseinventoryList[index].startTime = res.data[0].productionDate;
}; // 有效期
form.purchaseinventoryList[index].endTime = res.data[0].expirationDate;
const applyFromDto = (d, syncLotFromPick) => { form.purchaseinventoryList[index].unitCode =
if (syncLotFromPick && d.lotNumber != null && d.lotNumber !== '') { form.purchaseinventoryList[index].unitList.minUnitCode;
r.lotNumber = d.lotNumber; form.purchaseinventoryList[index].unitCode_dictText =
} form.purchaseinventoryList[index].unitList.minUnitCode_dictText;
r.itemTable = d.itemTable || ''; // 单价 大单位单价
r.totalQuantity = d.orgQuantity || 0; console.log(
if (d.price) { form.purchaseinventoryList[index].unitCode ==
r.price = d.price.toFixed(4); form.purchaseinventoryList[index].unitList.minUnitCode,
} else { 1212121
r.price = 0; );
} if (
r.supplierId = d.supplierId || ''; form.purchaseinventoryList[index].unitCode ==
r.startTime = toIssueDateTimeStr(d.productionDate) || ''; form.purchaseinventoryList[index].unitList.minUnitCode
r.endTime = toIssueDateTimeStr(d.expirationDate) || ''; ) {
r.unitCode = r.unitList.minUnitCode; form.purchaseinventoryList[index].price =
r.unitCode_dictText = r.unitList.minUnitCode_dictText; res.data[0].price / form.purchaseinventoryList[index].partPercent || '';
if (r.unitCode == r.unitList.minUnitCode) { form.purchaseinventoryList[index].price =
r.price = d.price / r.partPercent || ''; form.purchaseinventoryList[index].price.toFixed(4);
r.price = r.price.toFixed(4); // parseFloat(form.purchaseinventoryList[index].price.toFixed(4))
} else if (r.price > 1) { } else {
r.price = r.price.toFixed(4); console.log(
} form.purchaseinventoryList[index].price > 1,
}; 1212,
form.purchaseinventoryList[index].price
const persistStore = () => { );
store.setCurrentDataLYCK({ if (form.purchaseinventoryList[index].price > 1) {
purchaseinventoryList: form.purchaseinventoryList, form.purchaseinventoryList[index].price =
receiptHeaderForm: receiptHeaderForm, form.purchaseinventoryList[index].price.toFixed(4);
});
};
runGet(true)
.then((res) => {
const list = res.data || [];
const d = pickBestOrgQuantityRow(list);
const strictOk = d && Number(d.orgQuantity ?? 0) > 0;
if (strictOk) {
applyFromDto(d, false);
if (Number(r.totalQuantity) <= 0) {
proxy.$message.warning('仓库数量为0无法调用');
} }
persistStore(); }
if (form.purchaseinventoryList[index].totalQuantity == 0) {
proxy.$message.warning('仓库数量为0无法调用');
return; return;
} }
if (lotTrimmed) { } else {
return runGet(false).then((res2) => { form.purchaseinventoryList[index].totalQuantity = 0;
const list2 = res2.data || []; form.purchaseinventoryList[index].price = 0;
const d2 = pickBestOrgQuantityRow(list2); // if(form.purchaseinventoryList[index].totalQuantity==0){
if (d2 && Number(d2.orgQuantity ?? 0) > 0) {
applyFromDto(d2, true);
proxy.$message.info(
'所选批号在本仓库无对应库存或批号不一致,已按仓库实物回显批号与可领数量,请核对。'
);
} else {
r.totalQuantity = 0;
r.price = 0;
proxy.$message.warning('仓库数量为0无法调用');
}
persistStore();
});
}
r.totalQuantity = 0;
r.price = 0;
proxy.$message.warning('仓库数量为0无法调用'); proxy.$message.warning('仓库数量为0无法调用');
persistStore(); // }
}) }
.catch(() => { }).catch(() => {
r.totalQuantity = 0; form.purchaseinventoryList[index].totalQuantity = 0;
r.price = 0; form.purchaseinventoryList[index].price = 0;
persistStore(); });
}); store.setCurrentDataLYCK({
purchaseinventoryList: form.purchaseinventoryList,
receiptHeaderForm: receiptHeaderForm,
});
} }
// 切换仓库类型获取药房/药库列表 // 切换仓库类型获取药房/药库列表
// function handleChangeLocationType(value) { // function handleChangeLocationType(value) {
@@ -1323,19 +1232,20 @@ function getMaxCounts(row, index, counts) {
} }
// 计算总价 // 计算总价
function handleTotalPrice(index) { function handleTotalPrice(index) {
const r = form.purchaseinventoryList[index]; form.purchaseinventoryList[index].olditemQuantity =
r.olditemQuantity = r.itemQuantity * (r.partPercent ?? 1); form.purchaseinventoryList[index].itemQuantity * row.partPercent;
r.itemMaxQuantity = r.itemQuantity; form.purchaseinventoryList[index].itemMaxQuantity =
let purchaseItem = r; form.purchaseinventoryList[index].itemQuantity;
let purchaseItem = form.purchaseinventoryList[index];
if (purchaseItem.price > 0 && purchaseItem.itemQuantity > 0) { if (purchaseItem.price > 0 && purchaseItem.itemQuantity > 0) {
r.totalPrice = purchaseItem.price * purchaseItem.itemQuantity; form.purchaseinventoryList[index].totalPrice = purchaseItem.price * purchaseItem.itemQuantity;
r.totalPrice = r.totalPrice.toFixed(4); form.purchaseinventoryList[index].totalPrice =
form.purchaseinventoryList[index].totalPrice.toFixed(4);
// parseFloat(form.purchaseinventoryList[index].totalPrice.toFixed(4))
} }
if (r.itemQuantity == 0) { if (form.purchaseinventoryList[index].itemQuantity == 0) {
r.totalPrice = 0; form.purchaseinventoryList[index].totalPrice = 0;
} }
const qtyErr = validateRequisitionQtyVsStock(r, index + 1);
r.error = !!qtyErr;
store.setCurrentDataLYCK({ store.setCurrentDataLYCK({
purchaseinventoryList: form.purchaseinventoryList, purchaseinventoryList: form.purchaseinventoryList,
receiptHeaderForm: receiptHeaderForm, receiptHeaderForm: receiptHeaderForm,
@@ -1344,15 +1254,6 @@ function handleTotalPrice(index) {
// 保存 // 保存
function handleSave(row, index) { function handleSave(row, index) {
rowList.value = []; rowList.value = [];
for (let i = 0; i < form.purchaseinventoryList.length; i++) {
const line = form.purchaseinventoryList[i];
if (!line) continue;
const err = validateRequisitionQtyVsStock(line, i + 1);
if (err) {
proxy.$message.warning(err);
return;
}
}
form.purchaseinventoryList.map((row, index) => { form.purchaseinventoryList.map((row, index) => {
if (row) { if (row) {
// 触发校验 // 触发校验
@@ -1400,13 +1301,7 @@ function handleSave(row, index) {
}); });
} }
function addTransferProducts(rowList) { function addTransferProducts(rowList) {
const payload = (Array.isArray(rowList) ? rowList : []).map((item) => ({ addTransferProduct(JSON.parse(JSON.stringify(rowList))).then((res) => {
...item,
startTime: toIssueDateTimeStr(item.startTime),
endTime: toIssueDateTimeStr(item.endTime),
occurrenceTime: toIssueDateTimeStr(item.occurrenceTime) ?? item.occurrenceTime,
}));
addTransferProduct(JSON.parse(JSON.stringify(payload))).then((res) => {
// 当前行没有id视为首次新增 // 当前行没有id视为首次新增
// if (!row.id) { // if (!row.id) {
// data.isAdding = false; // 允许新增下一行 // data.isAdding = false; // 允许新增下一行
@@ -1475,7 +1370,7 @@ function handleScan(row,index){
rowData.value = row rowData.value = row
rowData.value.itemType = receiptHeaderForm.medicationType rowData.value.itemType = receiptHeaderForm.medicationType
ypName.value = row.name ypName.value = row.name
openTraceNoDialog.value = true; openTraceNoDialog .value = true;
currentIndex.value = index currentIndex.value = index
} }
@@ -1698,17 +1593,6 @@ const exportRequiredParams = ref({
pageSize: 10, pageSize: 10,
busNo: route.query.supplyBusNo busNo: route.query.supplyBusNo
}); });
// 追溯码对话框提交处理
function submit(traceNoData) {
if (currentIndex.value >= 0 && form.purchaseinventoryList[currentIndex.value]) {
form.purchaseinventoryList[currentIndex.value].traceNo = traceNoData.traceNo;
form.purchaseinventoryList[currentIndex.value].ybNo = traceNoData.ybNo;
proxy.$message.success('追溯码保存成功');
}
openTraceNoDialog.value = false;
}
function handleExport() { function handleExport() {
proxy.downloadGet( proxy.downloadGet(
'/issue-manage/requisition/excel-out', '/issue-manage/requisition/excel-out',

View File

@@ -76,7 +76,7 @@
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar> <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row> </el-row>
<el-table v-loading="loading" :data="surgeryList" row-key="id" :row-class-name="getRowClassName" highlight-current-row @current-change="handleCurrentChange"> <el-table v-loading="loading" :data="surgeryList" row-key="id" :row-class-name="getRowClassName">
<!-- 申请日期datetime - 2025-09-19 14:15:00 - 不可操作 --> <!-- 申请日期datetime - 2025-09-19 14:15:00 - 不可操作 -->
<el-table-column label="申请日期" align="center" prop="createTime" width="160"> <el-table-column label="申请日期" align="center" prop="createTime" width="160">
<template #default="scope"> <template #default="scope">
@@ -1405,12 +1405,6 @@ function getRowClassName({ row }) {
return '' return ''
} }
// 当前选中行(高亮)
const currentRow = ref(null)
function handleCurrentChange(row) {
currentRow.value = row
}
// 时间格式化函数 // 时间格式化函数
function parseTime(time, pattern) { function parseTime(time, pattern) {
if (!time) return '' if (!time) return ''

View File

@@ -803,7 +803,7 @@
</el-dialog> </el-dialog>
<!-- 手术计费弹窗 --> <!-- 手术计费弹窗 -->
<el-dialog :title="chargeDialogTitle" v-model="showChargeDialog" width="1400px" @close="closeChargeDialog" append-to-body destroy-on-close> <el-dialog :title="chargeDialogTitle" v-model="showChargeDialog" width="1400px" @close="closeChargeDialog" append-to-body>
<div style="display: flex; justify-content: space-between; height: 80vh"> <div style="display: flex; justify-content: space-between; height: 80vh">
<div style="width: 100%; border: 1px solid #eee; position: relative"> <div style="width: 100%; border: 1px solid #eee; position: relative">
<div style="padding: 10px; border: 1px solid #eee; height: 50px; border-left: 0"> <div style="padding: 10px; border: 1px solid #eee; height: 50px; border-left: 0">
@@ -829,7 +829,7 @@
</el-descriptions> </el-descriptions>
</div> </div>
<div style="padding: 10px"> <div style="padding: 10px">
<prescriptionlist :patientInfo="chargePatientInfo" ref="prescriptionRef" <prescriptionlist v-if="showChargeDialog" :patientInfo="chargePatientInfo" ref="prescriptionRef"
:generateSourceEnum="1" :generateSourceEnum="1"
:sourceBillNo="chargePatientInfo.sourceBillNo" /> :sourceBillNo="chargePatientInfo.sourceBillNo" />
<div class="overlay" v-if="disabled"></div> <div class="overlay" v-if="disabled"></div>
@@ -881,13 +881,16 @@ import {
deleteSurgerySchedule, deleteSurgerySchedule,
getSurgeryScheduleDetail getSurgeryScheduleDetail
} from '@/api/surgicalschedule' } from '@/api/surgicalschedule'
import { listUser } from '@/api/system/user'
import { deptTreeSelect } from '@/api/system/user'
import { listOperatingRoom } from '@/api/operatingroom'
import { getSurgeryPage} from '@/views/inpatientDoctor/home/components/applicationShow/api.js' import { getSurgeryPage} from '@/views/inpatientDoctor/home/components/applicationShow/api.js'
import { getContract } from '@/views/inpatientDoctor/home/components/api.js' import { getContract } from '@/views/inpatientDoctor/home/components/api.js'
import request from '@/utils/request' import request from '@/utils/request'
import SurgeryCharge from '../charge/surgerycharge/index.vue' import SurgeryCharge from '../charge/surgerycharge/index.vue'
import TemporaryMedical from './temporaryMedical.vue' import TemporaryMedical from './temporaryMedical.vue'
// 静默获取字典列表(跳过拦截器错误提示,手术室护士等角色可能无此权限) // 静默获取卫生机构列表(跳过拦截器错误提示,手术室护士等角色可能无此权限)
function getTenantPageSilent(query) { function getTenantPageSilent(query) {
return request({ return request({
url: '/system/tenant/page', url: '/system/tenant/page',
@@ -897,36 +900,6 @@ function getTenantPageSilent(query) {
}) })
} }
// 静默获取科室树(跳过拦截器错误提示)
function deptTreeSelectSilent(params = {}) {
return request({
url: '/base-data-manage/organization/organization',
method: 'get',
params: { typeEnum: 2, ...params },
skipErrorMsg: true
})
}
// 静默获取用户列表(跳过拦截器错误提示)
function listUserSilent(query) {
return request({
url: '/base-data-manage/practitioner/user-practitioner-page',
method: 'get',
params: query,
skipErrorMsg: true
})
}
// 静默获取手术室列表(跳过拦截器错误提示)
function listOperatingRoomSilent(query) {
return request({
url: '/base-data-manage/operating-room/list',
method: 'get',
params: query,
skipErrorMsg: true
})
}
const { proxy } = getCurrentInstance() const { proxy } = getCurrentInstance()
const userStore = useUserStore() const userStore = useUserStore()
const loading = ref(true) const loading = ref(true)
@@ -966,7 +939,7 @@ const form = reactive({
allergyRemark: undefined, allergyRemark: undefined,
surgeryNature: undefined, surgeryNature: undefined,
surgerySite: undefined, surgerySite: undefined,
incisionType: undefined, incisionLevel: undefined,
surgeryLevel: undefined, surgeryLevel: undefined,
admissionTime: undefined, admissionTime: undefined,
@@ -1175,7 +1148,7 @@ function loadOrgList() {
// 加载科室列表 // 加载科室列表
function loadDeptList() { function loadDeptList() {
deptTreeSelectSilent() deptTreeSelect()
.then(res => { .then(res => {
if (res.code === 200) { if (res.code === 200) {
const tree = res.data?.records || res.data || [] const tree = res.data?.records || res.data || []
@@ -1195,7 +1168,7 @@ function loadDeptList() {
// 加载医生列表 // 加载医生列表
function loadDoctorList() { function loadDoctorList() {
listUserSilent({ pageNo: 1, pageSize: 1000 }) listUser({ pageNo: 1, pageSize: 1000 })
.then(res => { .then(res => {
if (res.code === 200) { if (res.code === 200) {
const records = res.data?.records || [] const records = res.data?.records || []
@@ -1215,7 +1188,7 @@ function loadDoctorList() {
// 加载护士列表 // 加载护士列表
function loadNurseList() { function loadNurseList() {
listUserSilent({ pageNo: 1, pageSize: 1000 }) listUser({ pageNo: 1, pageSize: 1000 })
.then(res => { .then(res => {
if (res.code === 200) { if (res.code === 200) {
const records = res.data?.records || [] const records = res.data?.records || []
@@ -1235,7 +1208,7 @@ function loadNurseList() {
// 加载手术室列表 // 加载手术室列表
function loadOperatingRoomList() { function loadOperatingRoomList() {
listOperatingRoomSilent({ pageNo: 1, pageSize: 1000, statusEnum: 1 }) listOperatingRoom({ pageNo: 1, pageSize: 1000, statusEnum: 1 })
.then(res => { .then(res => {
if (res.code === 200) { if (res.code === 200) {
const records = res.data?.records || [] const records = res.data?.records || []
@@ -1456,18 +1429,17 @@ async function handleChargeCharge(row) {
} }
// 关闭计费弹窗 // 关闭计费弹窗
async function closeChargeDialog() { function closeChargeDialog() {
// 先关闭 prescriptionlist 内所有已打开的项目字典 popover // 先关闭 prescriptionlist 内所有已打开的项目字典 popover
if (prescriptionRef.value && prescriptionRef.value.closeAllPopovers) { if (prescriptionRef.value && prescriptionRef.value.closeAllPopovers) {
prescriptionRef.value.closeAllPopovers() prescriptionRef.value.closeAllPopovers()
} }
// 等 Vue 完成 popover 可见性更新的 DOM 操作, // 等 Vue 完成 DOM 更新后再关闭弹窗,确保 popover 先消失
// 因为 el-popover 通过 teleport 渲染在 body 上,需要在 dialog 卸载前完成清理 nextTick(() => {
await nextTick() showChargeDialog.value = false
// 清空数据,避免下次打开时使用缓存 chargePatientInfo.value = {}
showChargeDialog.value = false chargeSurgeryInfo.value = {}
chargePatientInfo.value = {} })
chargeSurgeryInfo.value = {}
} }
// 🔧 新增:标志位,用于区分是"打开"还是"刷新" // 🔧 新增:标志位,用于区分是"打开"还是"刷新"
@@ -1528,11 +1500,7 @@ function handleMedicalAdvice(row) {
// 先清空旧数据 // 先清空旧数据
temporaryBillingMedicines.value = [] temporaryBillingMedicines.value = []
temporaryAdvices.value = [] temporaryAdvices.value = []
// 🔧 修复 Bug #446: 如果是同一 encounter 且已有提交的医嘱(有 requestId保留签名状态 temporarySigned.value = false // 🔧 重置签名状态
const hasSubmittedAdvices = temporaryAdvices.value.length > 0 &&
temporaryAdvices.value[0]?.originalMedicine?.encounterId === row.visitId &&
temporaryAdvices.value.some(a => a.originalMedicine?.requestId);
temporarySigned.value = hasSubmittedAdvices; // 修复:根据已有数据状态设置,而非盲目重置
temporaryMedicalLoading.value = true // 🔧 新增:开始加载 temporaryMedicalLoading.value = true // 🔧 新增:开始加载
// 调用计费接口获取数据 // 调用计费接口获取数据
@@ -1547,13 +1515,9 @@ function handleMedicalAdvice(row) {
const filteredItems = res.data.filter(item => { const filteredItems = res.data.filter(item => {
// 匹配 encounterId // 匹配 encounterId
if (item.encounterId !== row.visitId) return false; if (item.encounterId !== row.visitId) return false;
// 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3)
if (item.adviceType !== 1) return false;
// 过滤掉名称为空的项目 // 过滤掉名称为空的项目
const medicineName = item.adviceName || item.advice_name; const medicineName = item.adviceName || item.advice_name;
if (!medicineName || medicineName.trim() === '') return false; if (!medicineName || medicineName.trim() === '') return false;
// 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目(已有 requestId 的不应出现在"待生成"列表)
if (item.requestId) return false;
// 根据药品请求ID去重避免重复显示 // 根据药品请求ID去重避免重复显示
const itemId = item.requestId || item.id; const itemId = item.requestId || item.id;
if (itemId && seenIds.has(itemId)) return false; if (itemId && seenIds.has(itemId)) return false;
@@ -1741,27 +1705,15 @@ function handleTemporaryMedicalSubmit(data) {
} }
}) })
// 🔧 修复 Bug #445: 使用稳定的字段组合匹配已提交项目,而不是依赖可能为空的 requestId/chargeItemId // 同步更新计费药品列表:移除已生成医嘱的项目,避免数据重复显示
// 构建已提交项目的匹配键集合(药品名称 + 规格 + 数量) const submittedIds = new Set(
const submittedKeys = new Set( (data.temporaryAdvices || []).map(a => a.originalMedicine?.requestId || a.originalMedicine?.chargeItemId).filter(Boolean)
(data.temporaryAdvices || [])
.map(a => {
const om = a.originalMedicine || {}
const name = om.medicineName || om.adviceName || om.advice_name || a.adviceName || ''
const spec = om.specification || om.volume || ''
const qty = om.quantity || 0
return `${name}|||${spec}|||${qty}`
})
.filter(k => k !== '|||0') // 过滤掉空项
) )
if (submittedIds.size > 0) {
if (submittedKeys.size > 0) { temporaryBillingMedicines.value = (data.billingMedicines || []).filter(
temporaryBillingMedicines.value = (temporaryBillingMedicines.value || []).filter(m => { m => !submittedIds.has(m.requestId || m.chargeItemId)
const key = `${m.medicineName || ''}|||${m.specification || ''}|||${m.quantity || 0}` )
return !submittedKeys.has(key)
})
} else { } else {
// 如果没有任何匹配键,清空待生成列表(所有项目都已提交)
temporaryBillingMedicines.value = [] temporaryBillingMedicines.value = []
} }
@@ -1821,19 +1773,13 @@ function handleQuoteBilling() {
temporaryBillingMedicines.value = [] temporaryBillingMedicines.value = []
temporaryAdvices.value = [] temporaryAdvices.value = []
// 🔧 修复 Bug #445: 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3) // 🔧 修复:显示所有药品请求数据,不管有没有计费项目
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
const filteredItems = res.data.filter(item => { const filteredItems = res.data.filter(item => {
// 匹配 encounterId // 匹配 encounterId
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false; if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
// 只保留药品类型adviceType=1过滤掉耗材(2)和诊疗项目(3)
if (item.adviceType !== 1) return false;
// 过滤掉名称为空的项目 // 过滤掉名称为空的项目
const medicineName = item.adviceName || item.advice_name; const medicineName = item.adviceName || item.advice_name;
if (!medicineName || medicineName.trim() === '') return false; return medicineName && medicineName.trim() !== '';
// 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目(已有 requestId
if (item.requestId) return false;
return true;
}) })
// 🔧 修复限制返回数量最多显示前100条避免数据过多导致页面卡死 // 🔧 修复限制返回数量最多显示前100条避免数据过多导致页面卡死
const maxItems = 100 const maxItems = 100
@@ -2050,12 +1996,7 @@ function resetForm() {
function submitForm() { function submitForm() {
proxy.$refs['surgeryRef'].validate((valid) => { proxy.$refs['surgeryRef'].validate((valid) => {
if (valid) { if (valid) {
const submitData = { const submitData = { ...form, orgId: userStore.orgId }
...form,
orgId: userStore.orgId,
incisionLevel: form.incisionType
}
delete submitData.incisionType
if (!form.scheduleId) { if (!form.scheduleId) {
// 新增手术安排 // 新增手术安排
addSurgerySchedule(submitData).then((res) => { addSurgerySchedule(submitData).then((res) => {

View File

@@ -147,10 +147,9 @@
<el-button class="cancel-btn" @click="handleCancel">取消</el-button> <el-button class="cancel-btn" @click="handleCancel">取消</el-button>
<el-button <el-button
class="sign-btn" class="sign-btn"
:disabled="allItemsSubmitted"
@click="handleSignAndSubmit" @click="handleSignAndSubmit"
> >
{{ allItemsSubmitted ? '已签发' : (isSigned ? '提交医嘱' : '一键签名并生成医嘱') }} {{ isSigned ? '提交医嘱' : '一键签名并生成医嘱' }}
</el-button> </el-button>
</div> </div>
</div> </div>
@@ -311,21 +310,8 @@ const getMethodCodeDict = computed(() => {
return dict return dict
}) })
// 🔧 修复 Bug #446: 检查计费药品是否已全部提交(有 requestId用于区分"首次签名"和"已提交重开"
const allItemsSubmitted = computed(() => {
const meds = props.billingMedicines || []
return meds.length > 0 && meds.every(m => m.requestId)
})
// 响应式数据 - isSigned 从父组件传入的 prop 初始化 // 响应式数据 - isSigned 从父组件传入的 prop 初始化
const isSigned = ref(props.isSignedProp) const isSigned = ref(props.isSignedProp)
// 🔧 修复 Bug #446: 同步父组件 isSignedProp 的变化到本地 isSigned
// ref(props.isSignedProp) 只在初始化时读取一次,父组件后续更新不会自动同步
watch(() => props.isSignedProp, (newVal) => {
isSigned.value = newVal
})
const signatureTime = ref('') const signatureTime = ref('')
const showSignDialog = ref(false) const showSignDialog = ref(false)
const signPassword = ref('') const signPassword = ref('')