Compare commits

...

9 Commits

Author SHA1 Message Date
78e5aff613 Fix Bug #443: 手术计费签发耗材时因 bizRequestFlag 过滤导致签发的项目列表为空
根因:prescriptionlist.vue 中 handleSave()、changeCheck()、watch、handleSingOut()
四处使用 bizRequestFlag 过滤(仅允许操作本人开立的医嘱)。
在手术计费场景下,手术医生创建的手术申请及其耗材的 requester_id 为医生ID,
手术室护士的 practitionerId 与之不匹配,bizRequestFlag='0',导致所有耗材
被过滤掉,saveList 为空,后端返回"医嘱列表为空"错误。

修复:在四处过滤逻辑中增加 isSurgeryChargeBillingContext() 判断(generateSourceEnum=6),
手术计费场景下跳过 bizRequestFlag 限制,允许任何授权用户签发/签退。
门诊划价场景保留 bizRequestFlag 限制,不影响原有安全校验。

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:30 +08:00
40e55bfe61 Fix Bug #545: 清理 handleNodeClick 中重复的 longTermFlag 字段 — 第三次提交时重复添加了 longTermFlag: 0(第887行和第889行各有一处),移除重复项
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:30 +08:00
2ff0243c39 Fix Bug #545: 补全诊断添加处缺失的 longTermFlag 默认值 — 第三个 push 调用缺少 longTermFlag: 0,导致通过此路径添加的诊断该字段为 undefined
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:29 +08:00
2a7d529183 fix bug529 2026-05-18 17:08:29 +08:00
5673a1386a Fix Bug #545: [门诊医生站-诊断-报卡] 长效诊断标识设置保存就清空 — 根因:1) 后端getEncounterDiagnosis查询已补充longTermFlag字段但前端getList()未做类型转换,useDict('long_term_flag')返回字符串字典值而数据库返回整数导致el-select匹配失败下拉框清空;2) 冗余的备份恢复逻辑应移除;修复:1) getList()中新增longTermFlag转字符串处理(String(item.longTermFlag)),保证与useDict字典值类型一致;2) 移除handleSaveDiagnosis中已不再需要的longTermFlagBackup/恢复逻辑
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:29 +08:00
29d8a48c27 Fix Bug #545: 根因+修复方案摘要 2026-05-18 17:08:29 +08:00
86c49a7933 Fix Bug #545: 长效诊断标识设置保存就清空 — 根因:handleSaveDiagnosis保存成功后await getList()刷新列表,后端getEncounterDiagnosis接口不返回longTermFlag字段,导致form.value.diagnosisList中该字段变为undefined,下拉框清空;修复:保存前用longTermFlagBackup备份longTermFlag数组,getList()完成后按索引恢复
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:29 +08:00
cf26980d5e Fix Bug #545: [门诊医生站-诊断-报卡] 长效诊断标识设置保存就清空 — 根因:getEncounterDiagnosis查询SQL(DoctorStationDiagnosisAppMapper.xml)未包含long_term_flag字段且DiagnosisQueryDto缺少对应属性,导致保存成功后刷新列表时后端不返回longTermFlag值,前端接收后下拉框清空;修复:1) SQL新增T1.long_term_flag AS longTermField; 2) DTO新增longTermFlag属性
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:29 +08:00
25c392f8f2 Fix Bug #542: 补费界面耗材类型检索不到数据 — 根因:双重不匹配 (1) getAdviceBaseInfos函数中queryParams.value.adviceType(单数)与后端@RequestParam("adviceTypes")(复数)参数名不匹配导致后端始终使用默认值"1,2,3"而非用户选择的类型; (2) drord_doctor_type字典中耗材值=4但后端SQL查询adviceTypes.contains(2)要求耗材=2; 修复:1) adviceType改为adviceTypes; 2) 默认返回值中耗材值4改为2
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-18 17:08:29 +08:00
8 changed files with 116 additions and 27 deletions

View File

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

View File

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

View File

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

View File

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

View File

@@ -461,6 +461,10 @@ watch(
console.log(prescriptionList.value,"prescriptionList.value")
if(newValue&&newValue.length>0){
let saveList = prescriptionList.value.filter((item) => {
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag
if (isSurgeryChargeBillingContext()) {
return item.check && item.statusEnum == 1
}
return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
})
console.log(saveList,"prescriptionList.value")
@@ -1025,7 +1029,9 @@ function changeCheck(value,index,row){
groupList.value.map(k=>{
if(k.check){
if(k.statusEnum == 1){//待签发
if(Number(k.bizRequestFlag)==1||!k.bizRequestFlag){
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag
const bizAllowed = isSurgeryChargeBillingContext() || Number(k.bizRequestFlag)==1||!k.bizRequestFlag
if(bizAllowed){
if(handleSaveDisabled.value&&!handleSingOutDisabled.value&&groupList.value.length>1){
proxy.$modal.msgWarning('请选择相同的状态的项目进行操作')
return
@@ -1040,7 +1046,9 @@ function changeCheck(value,index,row){
}
}
if(k.statusEnum == 2){ //已签发
if(Number(k.bizRequestFlag)==1||!k.bizRequestFlag){
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag
const bizAllowed = isSurgeryChargeBillingContext() || Number(k.bizRequestFlag)==1||!k.bizRequestFlag
if(bizAllowed){
if(!handleSaveDisabled.value&&handleSingOutDisabled.value&&groupList.value.length>1){
proxy.$modal.msgWarning('请选择相同的状态的项目进行操作')
return
@@ -1067,6 +1075,11 @@ function handleSave() {
return;
}
let saveList = prescriptionList.value.filter((item) => {
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag允许任何授权用户签发
// 门诊划价场景保留 bizRequestFlag 限制,只能操作本人开立的医嘱
if (isSurgeryChargeBillingContext()) {
return item.check && item.statusEnum == 1
}
return item.check && item.statusEnum == 1&&(Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
});
// let saveList = prescriptionList.value
@@ -1185,6 +1198,10 @@ function handleSingOut() {
return item.check;
})
.filter((item) => {
// 手术计费场景generateSourceEnum=6不限制 bizRequestFlag
if (isSurgeryChargeBillingContext()) {
return item.statusEnum == 2 && item.chargeStatus != 5
}
return item.statusEnum == 2 && item.chargeStatus != 5 && (Number(item.bizRequestFlag)==1||!item.bizRequestFlag)
})
.map((item) => {

View File

@@ -354,6 +354,11 @@ async function getList() {
if (!item.classification) {
item.classification = '西医';
}
// 转换 longTermFlag 为字符串,以匹配 useDict 返回的字典值类型(字符串)
// 避免 el-select 因类型不匹配(整数 1 vs 字符串 "1")导致下拉框清空
if (item.longTermFlag != null) {
item.longTermFlag = String(item.longTermFlag);
}
// 如果ybNo诊断编码符合传染病编码格式添加到selectedDiseases
if (item.ybNo && /^(01|02|03)/.test(item.ybNo)) {
item.selectedDiseases = [item.ybNo];

View File

@@ -207,6 +207,8 @@ const loadAllData = async () => {
}
applicationListAll.value = res.data?.records || [];
totalCount.value = res.data?.total || 0;
// 检验项目列表为异步加载,编辑回显必须在数据就绪后执行,否则已选区一直为空
applyEditTransferSelection()
} catch (e) {
proxy.$message.error('获取检验项目列表失败');
applicationListAll.value = [];
@@ -343,43 +345,70 @@ watch(
}
);
// 编辑模式下,回显已有数据
/** 编辑弹窗:根据申请单明细把右侧「已选择」与 transferValue 对齐(依赖 applicationListAll 已加载) */
const applyEditTransferSelection = () => {
const newData = props.editData
if (!newData?.requestFormId || !newData.requestFormDetailList?.length) {
return
}
if (!applicationListAll.value.length) {
return
}
const selectedIds = []
for (const detail of newData.requestFormDetailList) {
const idFromDetail = detail.activityId ?? detail.adviceDefinitionId
let matched = null
if (idFromDetail != null && idFromDetail !== '') {
matched = applicationListAll.value.find(
(item) => String(item.adviceDefinitionId) === String(idFromDetail)
)
}
if (!matched && detail.adviceName) {
matched = applicationListAll.value.find((item) => item.adviceName === detail.adviceName)
}
if (!matched && detail.adviceName) {
const norm = (s) => String(s || '').trim()
matched = applicationListAll.value.find(
(item) => norm(item.adviceName) === norm(detail.adviceName)
)
}
if (matched) {
selectedIds.push(matched.adviceDefinitionId)
}
}
const uniq = [...new Set(selectedIds)]
transferValue.value = uniq
if (newData.requestFormDetailList.length && uniq.length === 0) {
console.warn(
'[LaboratoryTests] 申请单明细未能在项目字典中匹配到项,请核对 activityId / 项目名称',
newData.requestFormDetailList
)
}
}
// 编辑模式下,回显已有数据(表单来自 descJson项目选择在字典加载后由 applyEditTransferSelection 完成)
watch(
() => props.editData,
(newData) => {
if (!newData || !newData.requestFormId) return;
if (!newData || !newData.requestFormId) return
// 解析 descJson 回填表单
if (newData.descJson) {
try {
const obj = JSON.parse(newData.descJson);
const obj = JSON.parse(newData.descJson)
Object.keys(form).forEach((key) => {
if (obj[key] !== undefined) {
form[key] = obj[key];
form[key] = obj[key]
}
});
})
} catch (e) {
console.error('解析 descJson 失败:', 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;
}
applyEditTransferSelection()
},
{ immediate: true }
);
{ immediate: true, deep: true }
)
// 编辑模式下applicationListAll 加载完成后重新回显已选项目
watch(

View File

@@ -354,7 +354,7 @@ const adviceTypeList = computed(() => {
}
// 默认值
return [
{ label: '耗材', value: 4 },
{ label: '耗材', value: 2 },
{ label: '诊疗', value: 3 },
{ label: '全部', value: '' },
];
@@ -555,7 +555,7 @@ function loadDepartmentOptions() {
function getAdviceBaseInfos() {
adviceLoading.value = true;
queryParams.value.searchKey = searchText.value;
queryParams.value.adviceType = adviceType.value;
queryParams.value.adviceTypes = adviceType.value;
queryParams.value.organizationId = orgId.value;
queryParams.value.pricingFlag = 1; // 划价标记
getAdviceBaseInfo(queryParams.value)