bug446,468,541,548

This commit is contained in:
2026-05-19 11:59:55 +08:00
parent 871e2de574
commit a91ee66368
5 changed files with 216 additions and 87 deletions

View File

@@ -41,7 +41,9 @@
<el-option label="全部" value="" />
<el-option label="待签发" value="0" />
<el-option label="已签发" value="1" />
<el-option label="已出报告" value="6" />
<el-option label="已采证" value="4" />
<el-option label="已送检" value="5" />
<el-option label="报告已出" value="6" />
<el-option label="已作废" value="7" />
</el-select>
</el-form-item>
@@ -91,7 +93,15 @@
<el-table-column prop="prescriptionNo" label="申请单号" width="140" />
<el-table-column label="单据状态" width="100" align="center">
<template #default="scope">
<span>{{ parseBillStatus(scope.row.billStatus ?? scope.row.status) }}</span>
<el-tag
:type="getBillStatusTagType(scope.row)"
effect="plain"
round
:class="{ 'report-status-tag': isReportStatus(scope.row) }"
@click="handleStatusClick(scope.row)"
>
{{ parseBillStatus(getBillStatus(scope.row)) }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="申请类型" width="100" align="center">
@@ -107,16 +117,16 @@
<el-table-column prop="requesterId_dictText" label="申请者" width="120" />
<el-table-column label="操作" align="center" fixed="right" width="220">
<template #default="scope">
<!-- 待签发status=0或null/undefined可修改删除 -->
<template v-if="!scope.row.status || scope.row.status == 0">
<!-- 待签发可修改删除 -->
<template v-if="isPendingStatus(scope.row)">
<el-button link type="primary" @click="handleEdit(scope.row)">修改</el-button>
<el-button link type="danger" @click="handleDelete(scope.row)">删除</el-button>
</template>
<!-- 已签发status=1可撤回 -->
<template v-else-if="scope.row.status == 1">
<!-- 已签发可撤回 -->
<template v-else-if="isIssuedStatus(scope.row)">
<el-button link type="warning" @click="handleWithdraw(scope.row)">撤回</el-button>
</template>
<!-- 校对(2)待接收(3)已收样(4)已出报告(6)已作废(7)仅查看详情 -->
<!-- 采证已送检报告已出已作废仅查看详情 -->
<template v-else>
<el-button link type="primary" @click="handleViewDetail(scope.row)">详情</el-button>
</template>
@@ -212,10 +222,10 @@
</template>
<script setup>
import {computed, getCurrentInstance, ref, watch} from 'vue';
import {computed, getCurrentInstance, nextTick, ref, watch} from 'vue';
import {Refresh, Search} from '@element-plus/icons-vue';
import {patientInfo} from '../../store/patient.js';
import {getInspection, deleteRequestForm, withdrawRequestForm} from './api';
import {getInspection, deleteRequestForm, withdrawRequestForm, getProofResult} from './api';
import {getDepartmentList} from '@/api/public.js';
import LaboratoryTests from '../order/applicationForm/laboratoryTests.vue';
import {saveInspection} from '../order/applicationForm/api.js';
@@ -270,7 +280,7 @@ const fetchData = async () => {
if (res.code === 200 && res.data) {
const raw = res.data?.records || res.data;
const list = Array.isArray(raw) ? raw : [raw];
tableData.value = list.filter(Boolean);
tableData.value = list.filter(Boolean).sort(sortByCreateTimeDesc);
} else {
tableData.value = [];
}
@@ -329,19 +339,95 @@ const labelMap = {
* @param {string|number} status - 状态码
* @returns {string} 状态文本
*/
const getBillStatus = (row) => {
return row?.billStatus ?? row?.status ?? row?.statusEnum ?? row?.applyStatus;
};
const parseBillStatus = (status) => {
const statusMap = {
'0': '待签发',
'1': '已签发',
'2': '已校对',
'3': '待接收',
'4': '已收样',
'6': '已出报告',
'2': '已采证',
'3': '已送检',
'4': '已采证',
'5': '已送检',
'6': '报告已出',
'8': '报告已出',
'7': '已作废',
};
return statusMap[String(status)] || '-';
};
const getBillStatusTagType = (row) => {
const typeMap = {
'0': 'info',
'1': 'primary',
'2': 'primary',
'3': 'warning',
'4': 'primary',
'5': 'warning',
'6': 'success',
'7': 'danger',
'8': 'success',
};
return typeMap[String(getBillStatus(row))] || 'info';
};
const isPendingStatus = (row) => {
const status = getBillStatus(row);
return status === undefined || status === null || status === '' || String(status) === '0';
};
const isIssuedStatus = (row) => String(getBillStatus(row)) === '1';
const isReportStatus = (row) => ['6', '8'].includes(String(getBillStatus(row)));
const sortByCreateTimeDesc = (a, b) => {
const aTime = a?.createTime ? new Date(a.createTime).getTime() : 0;
const bTime = b?.createTime ? new Date(b.createTime).getTime() : 0;
return bTime - aTime;
};
const handleStatusClick = (row) => {
if (isReportStatus(row)) {
handleViewReport(row);
}
};
const pickReportUrl = (data, row) => {
if (!data) return '';
if (typeof data === 'string') return data;
const raw = data.records || data;
const list = Array.isArray(raw) ? raw : [raw];
const matched =
list.find((item) => {
const reportNo = item.busNo || item.reportNo || item.applyNo || item.prescriptionNo;
return reportNo && row.prescriptionNo && String(reportNo) === String(row.prescriptionNo);
}) || list[0];
return matched?.requestUrl || matched?.pdfUrl || matched?.reportUrl || matched?.url || '';
};
const handleViewReport = async (row) => {
try {
const res = await getProofResult({
encounterId: row.encounterId || patientInfo.value?.encounterId,
prescriptionNo: row.prescriptionNo,
});
if (res?.code === 200) {
const url = pickReportUrl(res.data, row);
if (url) {
window.open(url, '_blank');
return;
}
}
proxy.$modal?.msgWarning?.('暂未获取到检验报告链接');
} catch (e) {
proxy.$modal?.msgError?.(e.message || '获取检验报告失败');
}
};
/**
* 解析申请类型(优先级代码)
* @param {string} descJson - JSON字符串
@@ -462,12 +548,12 @@ const handleViewDetail = async (row) => {
* 修改检验申请单(待签发状态)
*/
const handleEdit = async (row) => {
// 确保科室数据已加载
if (!orgOptions.value || orgOptions.value.length === 0) {
await getLocationInfo();
}
editRowData.value = row;
editDialogVisible.value = true;
await nextTick();
editFormRef.value?.getList?.();
editFormRef.value?.getLocationInfo?.();
editFormRef.value?.getDiagnosisList?.();
};
/**
@@ -646,6 +732,10 @@ defineExpose({
animation: rotating 2s linear infinite;
}
.report-status-tag {
cursor: pointer;
}
@keyframes rotating {
0% {
transform: rotate(0deg);

View File

@@ -134,10 +134,10 @@
</div>
</template>
<script setup name="LaboratoryTests">
import {getCurrentInstance, onMounted, reactive, ref, watch, computed} from 'vue';
import {getCurrentInstance, nextTick, onMounted, reactive, ref, watch, computed} from 'vue';
import {patientInfo} from '../../../store/patient.js';
import {getApplicationList, saveInspection} from './api';
import {getOrgList} from '@/views/doctorstation/components/api.js';
import {getDepartmentList} from '@/api/public.js';
import {getEncounterDiagnosis} from '../../api.js';
import {ElMessage} from 'element-plus';
@@ -168,6 +168,7 @@ const loading = ref(false);
const orgOptions = ref([]);
const searchKey = ref('');
const totalCount = ref(0);
const skipDeptAutoFill = ref(false);
// 将已加载的全部数据转为 transfer 组件所需的格式
const buildTransferData = (records) => {
@@ -263,7 +264,31 @@ const form = reactive({
otherDiagnosisList: [], //其他断目录
});
const rules = reactive({});
const normalizeOrgTreeIds = (nodes) => {
if (!Array.isArray(nodes)) return [];
return nodes.map((node) => ({
...node,
id: node.id != null ? String(node.id) : node.id,
children: node.children?.length ? normalizeOrgTreeIds(node.children) : undefined,
}));
};
const resolveTargetDepartmentId = (rawId) => {
if (rawId == null || rawId === '') return '';
const node = findTreeItem(orgOptions.value, rawId);
return node ? String(node.id) : String(rawId);
};
const applyTargetDepartmentEcho = () => {
if (form.targetDepartment) {
form.targetDepartment = resolveTargetDepartmentId(form.targetDepartment);
}
};
onMounted(() => {
getLocationInfo();
getDiagnosisList();
getList();
});
/**
@@ -292,8 +317,9 @@ const projectWithDepartment = (selectProjectIds, type) => {
});
}
});
// 保存用户手动选择的发往科室(提交时需要保留)
const manualDept = type === 2 ? form.targetDepartment : '';
// 保存用户手动选择/回显的发往科室(提交、编辑回显时需要保留)
const manualDept =
type === 2 || (isEditMode.value && form.targetDepartment) ? form.targetDepartment : '';
// 清空科室
form.targetDepartment = '';
if (arr.length > 0) {
@@ -313,8 +339,8 @@ const projectWithDepartment = (selectProjectIds, type) => {
const findItem = findTreeItem(orgOptions.value, obj.orgId);
if (!findItem) {
// type=2(提交)时,若用户已手动选择发往科室,则允许提交
if (type === 2 && manualDept) {
form.targetDepartment = manualDept;
if ((type === 2 || isEditMode.value) && manualDept) {
form.targetDepartment = resolveTargetDepartmentId(manualDept);
isRelease = true;
} else if (type === 2 && !manualDept) {
// 提交时用户未手动选择科室,才提示错误
@@ -330,10 +356,10 @@ const projectWithDepartment = (selectProjectIds, type) => {
}
if (findItem && isRelease) {
// 提交时若用户已选「发往科室」,不得用项目默认执行科室覆盖
if (type === 2 && manualDept) {
form.targetDepartment = manualDept;
if ((type === 2 || isEditMode.value) && manualDept) {
form.targetDepartment = resolveTargetDepartmentId(manualDept);
} else {
form.targetDepartment = findItem.id;
form.targetDepartment = String(findItem.id);
}
}
}
@@ -343,6 +369,7 @@ const projectWithDepartment = (selectProjectIds, type) => {
watch(
() => transferValue.value,
(newValue) => {
if (skipDeptAutoFill.value) return;
if (isInitializing.value) return;
projectWithDepartment(newValue, 1);
}
@@ -382,7 +409,11 @@ const applyEditTransferSelection = () => {
const uniq = [...new Set(selectedIds)]
// 设置初始化标志,防止 transferValue 变化触发 projectWithDepartment 覆盖 descJson 中的科室值
isInitializing.value = true
skipDeptAutoFill.value = true
transferValue.value = uniq
nextTick(() => {
skipDeptAutoFill.value = false
})
isInitializing.value = false
if (newData.requestFormDetailList.length && uniq.length === 0) {
console.warn(
@@ -406,6 +437,7 @@ watch(
form[key] = obj[key]
}
})
applyTargetDepartmentEcho()
} catch (e) {
console.error('解析 descJson 失败:', e)
}
@@ -416,7 +448,14 @@ watch(
{ immediate: true, deep: true }
)
// 编辑模式下applicationListAll 加载完成后重新回显已选项目
watch(
() => orgOptions.value,
() => {
applyTargetDepartmentEcho()
}
)
// 编辑模式下,项目字典加载完成后重新回显已选项目
watch(
() => applicationListAll.value,
() => {
@@ -436,6 +475,7 @@ watch(
isInitializing.value = true;
transferValue.value = selectedIds;
isInitializing.value = false;
applyEditTransferSelection();
}
);
@@ -488,9 +528,9 @@ const submit = () => {
};
/** 查询科室 */
const getLocationInfo = () => {
getOrgList().then((res) => {
orgOptions.value = res.data.records;
console.log('科室========>', JSON.stringify(orgOptions.value));
return getDepartmentList().then((res) => {
orgOptions.value = normalizeOrgTreeIds(res.data || []);
applyTargetDepartmentEcho();
});
};
// 获取诊断目录

View File

@@ -804,7 +804,7 @@ function checkUnit(item, row) {
}
}
// 行双击打开编辑块"待保存"和"待签发"均可编辑
// 行双击打开编辑块待保存待签发医嘱均可编辑;已签发/已完成/停止不允许编辑
function clickRowDb(row, column, event) {
// 检查点击的是否是复选框
if (event && event.target.closest('.el-checkbox')) {
@@ -815,14 +815,18 @@ function clickRowDb(row, column, event) {
return;
}
row.showPopover = false;
// statusEnum == 1 包含"待保存(无requestId)"和"待签发(有requestId)",均允许编辑
if (row.statusEnum == 1) {
// 确保治疗类型为字符串,方便与单选框 label 对齐,默认为长期医嘱('1')
row.therapyEnum = String(row.therapyEnum ?? '1');
row.isEdit = true;
const index = prescriptionList.value.findIndex((item) => item.uniqueKey === row.uniqueKey);
prescriptionList.value[index] = row;
rowIndex.value = index;
if (index !== -1) {
prescriptionList.value[index] = row;
}
expandOrder.value = [row.uniqueKey];
} else {
proxy.$modal.msgWarning('仅待保存或待签发医嘱允许编辑');
}
}

View File

@@ -262,7 +262,7 @@
</template>
<script setup>
import {computed, nextTick, onMounted, ref, watch} from 'vue';
import {nextTick, onMounted, ref} from 'vue';
import {ElMessage, ElMessageBox} from 'element-plus';
// Element Plus 图标导入
import {User} from '@element-plus/icons-vue';
@@ -366,9 +366,9 @@ const rawPrescriptionList = ref([]); // 原始未分组数据
const groupedPrescriptionList = ref([]); // 按encounterId分组后的数据
const activeCollapseNames = ref([]); // Collapse激活状态
const selectedRows = ref({}); // 选中的行数据
const totalItemsCount = ref(0); // 总医嘱项数
const totalAmount = ref(0); // 总金额保留4位小数
const dialogVisible = ref(false);
/** Tab 切换同步日期时跳过 date-picker change避免与 v-model 循环触发 */
const syncingDateFromTab = ref(false);
const selectedFeeItems = ref([]);
const currentPatientInfo = ref(null);
const queryParams = ref({
@@ -381,24 +381,6 @@ const userStore = useUserStore();
const userId = ref(safeGet(userStore, 'id', ''));
const orgId = ref(safeGet(userStore, 'orgId', ''));
// ========== 计算属性 ==========
// 计算总统计信息(总项数、总金额)
const calculateTotalStats = computed(() => {
let itemsCount = 0;
let amount = 0;
safeArray(groupedPrescriptionList.value).forEach((patientGroup) => {
safeArray(patientGroup).forEach((item) => {
itemsCount++;
// 累加单价保留4位小数精度
amount = Math.round((amount + Number(safeGet(item, 'unitPrice', 0))) * 10000) / 10000;
});
});
totalItemsCount.value = itemsCount;
totalAmount.value = amount;
});
// ========== 方法 ==========
/**
* 计算单个患者的总金额保留4位小数
@@ -447,16 +429,19 @@ const handleTableSelectionChange = (index, val) => {
};
/**
* 日期Tab切换
* @param {Object} tab - 标签页
* 按 Tab 同步日期范围(避免 date-picker @change 与 Tab v-model 互相覆盖)
* @param {string} rangeType - today | yesterday | custom
*/
const handleDateTabClick = (tab) => {
const applyDateRangeByTab = (rangeType) => {
const today = new Date();
const yesterday = new Date(today);
yesterday.setDate(today.getDate() - 1);
const format = (date) => formatDateStr(date, 'YYYY-MM-DD');
switch (safeGet(tab, 'paneName')) {
syncingDateFromTab.value = true;
dateRange.value = rangeType;
switch (rangeType) {
case 'today':
dateRangeValue.value = [format(today), format(today)];
break;
@@ -464,27 +449,54 @@ const handleDateTabClick = (tab) => {
dateRangeValue.value = [format(yesterday), format(yesterday)];
break;
case 'custom':
if (!dateRangeValue.value.length) {
if (safeArray(dateRangeValue.value).length < 2) {
dateRangeValue.value = [format(today), format(today)];
}
break;
default:
break;
}
nextTick(() => {
syncingDateFromTab.value = false;
});
};
/**
* 日期选择器变化
* 日期Tab切换
* @param {Object} tab - 标签页
*/
const handleDateTabClick = (tab) => {
const rangeType = tab?.paneName ?? tab?.props?.name;
if (!rangeType) return;
applyDateRangeByTab(rangeType);
handleQuery();
};
/**
* 日期选择器变化(仅用户手动改日期时切到「自定义」)
* @param {Array} val - 选中日期
*/
const handleDatePickerChange = (val) => {
if (syncingDateFromTab.value) return;
const dateVal = safeArray(val);
if (dateVal.length === 2) {
if (dateVal.length !== 2) return;
const start = new Date(dateVal[0]);
const end = new Date(dateVal[1]);
if (start > end) {
ElMessage.warning('开始日期不能晚于结束日期');
syncingDateFromTab.value = true;
dateRangeValue.value = [dateVal[1], dateVal[0]];
nextTick(() => {
syncingDateFromTab.value = false;
});
return;
}
if (dateRange.value !== 'custom') {
dateRange.value = 'custom';
const start = new Date(dateVal[0]);
const end = new Date(dateVal[1]);
if (start > end) {
ElMessage.warning('开始日期不能晚于结束日期');
dateRangeValue.value = [dateVal[1], dateVal[0]];
}
}
};
@@ -714,24 +726,7 @@ const handleSingleDelete = (row) => {
};
// ========== 初始化 ==========
onMounted(() => {
// 设置默认日期
const today = new Date();
const defaultDate = formatDateStr(today, 'YYYY-MM-DD');
dateRangeValue.value = [defaultDate, defaultDate];
// 监听日期变化自动查询
watch(
[dateRange, dateRangeValue],
([newRange, newVal], [oldRange, oldVal]) => {
if (oldRange !== undefined && safeArray(newVal).length === 2) {
handleQuery();
}
},
{ deep: true }
);
// 初始化统计信息
calculateTotalStats.value;
applyDateRangeByTab('today');
});
</script>

View File

@@ -150,7 +150,7 @@
:disabled="allItemsSubmitted"
@click="handleSignAndSubmit"
>
{{ allItemsSubmitted ? '已签发' : (isSigned ? '提交医嘱' : '一键签名并生成医嘱') }}
{{ allItemsSubmitted ? '已签发' : '一键签名并生成医嘱' }}
</el-button>
</div>
</div>