Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f84940fa5f | ||
| e3db810972 | |||
| 85e95420b7 | |||
| 206a0f4083 | |||
| 5e9aaebc7a | |||
| f925731f6f | |||
|
|
156a3f0f24 | ||
| 85d254990f |
@@ -169,9 +169,11 @@ public class OrganizationLocationAppServiceImpl implements IOrganizationLocation
|
|||||||
if (DateTimeUtils.isOverlap(organizationLocation.getStartTime(), organizationLocation.getEndTime(),
|
if (DateTimeUtils.isOverlap(organizationLocation.getStartTime(), organizationLocation.getEndTime(),
|
||||||
orgLoc.getStartTime(), orgLoc.getEndTime())) {
|
orgLoc.getStartTime(), orgLoc.getEndTime())) {
|
||||||
Organization org = organizationService.getById(organizationLocation.getOrganizationId());
|
Organization org = organizationService.getById(organizationLocation.getOrganizationId());
|
||||||
String organizationName = org != null ? org.getName() : "已删除科室(ID:" + organizationLocation.getOrganizationId() + ")";
|
if (org == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
return R.fail("当前诊疗:" + activityName + CommonConstants.Common.DASH + orgLoc.getStartTime()
|
return R.fail("当前诊疗:" + activityName + CommonConstants.Common.DASH + orgLoc.getStartTime()
|
||||||
+ CommonConstants.Common.DASH + orgLoc.getEndTime() + "与" + organizationName + "时间冲突");
|
+ CommonConstants.Common.DASH + orgLoc.getEndTime() + "与" + org.getName() + "时间冲突");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (orgLocQueryDto.getId() != null) {
|
if (orgLocQueryDto.getId() != null) {
|
||||||
|
|||||||
@@ -2192,11 +2192,6 @@ 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)
|
|
||||||
if (sourceBillNo != null && !sourceBillNo.isEmpty()) {
|
|
||||||
requestBaseInfo.removeIf(dto -> dto.getAdviceType() != null
|
|
||||||
&& dto.getAdviceType() == 1);
|
|
||||||
}
|
|
||||||
for (RequestBaseDto requestBaseDto : requestBaseInfo) {
|
for (RequestBaseDto requestBaseDto : requestBaseInfo) {
|
||||||
// 请求状态
|
// 请求状态
|
||||||
requestBaseDto
|
requestBaseDto
|
||||||
|
|||||||
@@ -23,6 +23,9 @@ public class SurgeryItemDto {
|
|||||||
@JsonSerialize(using = ToStringSerializer.class)
|
@JsonSerialize(using = ToStringSerializer.class)
|
||||||
private Long orgId;
|
private Long orgId;
|
||||||
|
|
||||||
|
/** 所属科室名称 */
|
||||||
|
private String orgName;
|
||||||
|
|
||||||
/** 执行科室ID */
|
/** 执行科室ID */
|
||||||
@JsonSerialize(using = ToStringSerializer.class)
|
@JsonSerialize(using = ToStringSerializer.class)
|
||||||
private Long positionId;
|
private Long positionId;
|
||||||
|
|||||||
@@ -871,6 +871,7 @@
|
|||||||
</select>
|
</select>
|
||||||
|
|
||||||
<!-- 手术项目专用分页查询:仅查手术 + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
<!-- 手术项目专用分页查询:仅查手术 + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
||||||
|
<!-- 使用 LIMIT/OFFSET 直接查询,避免 MyBatis Plus 分页插件的 COUNT 开销 -->
|
||||||
<select id="getSurgeryPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto">
|
<select id="getSurgeryPage" resultType="com.openhis.web.doctorstation.dto.SurgeryItemDto">
|
||||||
SELECT DISTINCT ON (t1.ID)
|
SELECT DISTINCT ON (t1.ID)
|
||||||
t1.ID AS advice_definition_id,
|
t1.ID AS advice_definition_id,
|
||||||
@@ -893,6 +894,7 @@
|
|||||||
AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
|
AND (t1.name ILIKE '%' || #{searchKey} || '%' OR t1.py_str ILIKE '%' || #{searchKey} || '%')
|
||||||
</if>
|
</if>
|
||||||
ORDER BY t1.ID, t1.name ASC, t2.ID ASC
|
ORDER BY t1.ID, t1.name ASC, t2.ID ASC
|
||||||
|
LIMIT #{limit} OFFSET #{offset}
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<!-- 检查项目专用分页查询:仅查检查(23) + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
<!-- 检查项目专用分页查询:仅查检查(23) + 定价,无库存/草稿库存/取药科室等无关逻辑 -->
|
||||||
@@ -901,6 +903,7 @@
|
|||||||
t1.ID AS advice_definition_id,
|
t1.ID AS advice_definition_id,
|
||||||
t1.NAME AS advice_name,
|
t1.NAME AS advice_name,
|
||||||
t1.org_id AS org_id,
|
t1.org_id AS org_id,
|
||||||
|
t3.name AS org_name,
|
||||||
t1.org_id AS position_id,
|
t1.org_id AS position_id,
|
||||||
t2.ID AS charge_item_definition_id,
|
t2.ID AS charge_item_definition_id,
|
||||||
t2.price AS price,
|
t2.price AS price,
|
||||||
@@ -912,6 +915,9 @@
|
|||||||
AND t2.delete_flag = '0'
|
AND t2.delete_flag = '0'
|
||||||
AND t2.status_enum = #{statusEnum}
|
AND t2.status_enum = #{statusEnum}
|
||||||
AND t2.instance_table = 'wor_activity_definition'
|
AND t2.instance_table = 'wor_activity_definition'
|
||||||
|
LEFT JOIN adm_organization t3
|
||||||
|
ON t3.id = t1.org_id
|
||||||
|
AND t3.delete_flag = '0'
|
||||||
WHERE t1.delete_flag = '0'
|
WHERE t1.delete_flag = '0'
|
||||||
AND t1.category_code = '23'
|
AND t1.category_code = '23'
|
||||||
<if test="searchKey != null and searchKey != ''">
|
<if test="searchKey != null and searchKey != ''">
|
||||||
|
|||||||
@@ -57,8 +57,6 @@
|
|||||||
AND ae.delete_flag = '0'
|
AND ae.delete_flag = '0'
|
||||||
LEFT JOIN adm_patient AS ap ON ap.ID = ae.patient_id
|
LEFT JOIN adm_patient AS ap ON ap.ID = ae.patient_id
|
||||||
AND ap.delete_flag = '0'
|
AND ap.delete_flag = '0'
|
||||||
LEFT JOIN wor_service_request AS wsr ON wsr.prescription_no = drf.prescription_no
|
|
||||||
AND wsr.delete_flag = '0'
|
|
||||||
WHERE drf.delete_flag = '0'
|
WHERE drf.delete_flag = '0'
|
||||||
AND drf.encounter_id = #{encounterId}
|
AND drf.encounter_id = #{encounterId}
|
||||||
AND drf.type_code = #{typeCode}
|
AND drf.type_code = #{typeCode}
|
||||||
|
|||||||
@@ -1173,8 +1173,9 @@ function handleSaveSign(row, index) {
|
|||||||
cleanRow.generateSourceEnum = 6; // 手术计费
|
cleanRow.generateSourceEnum = 6; // 手术计费
|
||||||
cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
|
cleanRow.sourceBillNo = props.patientInfo.sourceBillNo;
|
||||||
}
|
}
|
||||||
console.log('cleanRow', cleanRow)
|
// 🔧 门诊计费场景:保存为草稿,让药品出现在临时医嘱弹窗"已引用计费药品(待生成医嘱)"中
|
||||||
savePrescription({ adviceSaveList: [cleanRow] }, '1').then((res) => {
|
const adviceOpType = props.patientInfo.sourceBillNo ? '0' : '1'
|
||||||
|
savePrescription({ adviceSaveList: [cleanRow] }, adviceOpType).then((res) => {
|
||||||
if (res.code === 200) {
|
if (res.code === 200) {
|
||||||
proxy.$modal.msgSuccess('保存成功');
|
proxy.$modal.msgSuccess('保存成功');
|
||||||
getListInfo(false);
|
getListInfo(false);
|
||||||
|
|||||||
@@ -316,13 +316,13 @@
|
|||||||
<!-- Bug #384修复: 单价显示套餐价格(如果选中)或部位价格 -->
|
<!-- Bug #384修复: 单价显示套餐价格(如果选中)或部位价格 -->
|
||||||
<el-table-column label="单价" width="75" align="right">
|
<el-table-column label="单价" width="75" align="right">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
{{ scope.row.selectedMethod?.packagePrice || scope.row.price }}
|
{{ formatDetailAmount(getSelectedItemAmount(scope.row)) }}
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<!-- Bug #384修复: 金额使用有效价格计算 -->
|
<!-- Bug #384修复: 金额使用有效价格计算 -->
|
||||||
<el-table-column label="金额" width="80" align="right">
|
<el-table-column label="金额" width="80" align="right">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
{{ ((scope.row.selectedMethod?.packagePrice || scope.row.price || 0) * (scope.row.quantity || 1)).toFixed(2) }}
|
{{ formatDetailAmount(getSelectedItemAmount(scope.row) * (scope.row.quantity || 1)) }}
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="类型" prop="checkType" width="70" align="center" />
|
<el-table-column label="类型" prop="checkType" width="70" align="center" />
|
||||||
@@ -392,37 +392,6 @@
|
|||||||
<div v-if="categoryLoadingSet.has(cat.typeId)" class="category-loading-hint">
|
<div v-if="categoryLoadingSet.has(cat.typeId)" class="category-loading-hint">
|
||||||
加载中...
|
加载中...
|
||||||
</div>
|
</div>
|
||||||
<!-- Bug #428修复: 渲染分类联动加载的检查方法列表 -->
|
|
||||||
<!-- Bug #500修复: v-if 改为 v-show,避免方法列表加载时 DOM 突然插入导致高度跳变 -->
|
|
||||||
<div
|
|
||||||
v-show="cat.methods && cat.methods.length > 0"
|
|
||||||
class="method-section"
|
|
||||||
>
|
|
||||||
<div class="method-section-title">检查方法</div>
|
|
||||||
<div
|
|
||||||
v-for="method in cat.methods"
|
|
||||||
:key="method.id"
|
|
||||||
class="method-row"
|
|
||||||
>
|
|
||||||
<el-checkbox
|
|
||||||
:model-value="isMethodSelected(method, cat)"
|
|
||||||
@change="(val) => handleMethodSelect(val, method, cat)"
|
|
||||||
class="method-checkbox"
|
|
||||||
>
|
|
||||||
{{ method.name }}
|
|
||||||
</el-checkbox>
|
|
||||||
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<!-- Bug #500修复: 加载中的方法列表骨架占位,提前预留空间避免高度跳变 -->
|
|
||||||
<div
|
|
||||||
v-if="categoryLoadingSet.has(cat.typeId) && (!cat.methods || cat.methods.length === 0)"
|
|
||||||
class="method-section method-section-skeleton"
|
|
||||||
>
|
|
||||||
<div class="method-section-title">检查方法</div>
|
|
||||||
<div class="skeleton-method-row"></div>
|
|
||||||
<div class="skeleton-method-row"></div>
|
|
||||||
</div>
|
|
||||||
</el-collapse-item>
|
</el-collapse-item>
|
||||||
</el-collapse>
|
</el-collapse>
|
||||||
</div>
|
</div>
|
||||||
@@ -440,46 +409,103 @@
|
|||||||
class="selected-item-card"
|
class="selected-item-card"
|
||||||
:class="{ 'is-expanded': item.expanded }"
|
:class="{ 'is-expanded': item.expanded }"
|
||||||
>
|
>
|
||||||
<!-- Bug #384修复 + #426修复: 项目卡片头部,可展开/收起 -->
|
<!-- 项目卡片头部:项目和检查方法解耦,点击展开查看方法/明细 -->
|
||||||
<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>
|
<el-tooltip :content="getDisplayItemName(item)" placement="top" :show-after="400">
|
||||||
<el-tooltip :content="item.name" placement="top" :show-after="400">
|
<span class="card-name">{{ getDisplayItemName(item) }}</span>
|
||||||
<span class="card-name">{{ item.name }}</span>
|
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
<span class="card-price">¥{{ formatDetailAmount(item.price) }}</span>
|
<span class="card-price">¥{{ formatDetailAmount(getSelectedItemAmount(item)) }}</span>
|
||||||
<el-icon :class="['expand-icon', { expanded: item.expanded }]">
|
<el-icon :class="['expand-icon', { expanded: item.expanded }]">
|
||||||
<ArrowDown v-if="!item.expanded" />
|
<ArrowDown />
|
||||||
<ArrowUp v-if="item.expanded" />
|
|
||||||
</el-icon>
|
</el-icon>
|
||||||
<!-- 删除按钮 -->
|
<!-- 删除按钮 -->
|
||||||
<el-button link type="danger" size="small" @click.stop="handleRemoveItem(idx, item)">
|
<el-button link type="danger" size="small" @click.stop="handleRemoveItem(idx, item)">
|
||||||
<el-icon><Close /></el-icon>
|
<el-icon><Close /></el-icon>
|
||||||
</el-button>
|
</el-button>
|
||||||
</div>
|
</div>
|
||||||
<!-- Bug #428: 有套餐 ID 时默认展开;加载中/空/明细均在本区域展示 -->
|
<div v-if="item.expanded" class="selected-card-body">
|
||||||
<div v-if="item.expanded && shouldShowPackageBody(item)" class="selected-card-body">
|
<div v-if="shouldShowItemPackageBody(item)">
|
||||||
<div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div>
|
<div class="package-toggle" @click.stop="toggleItemPackageExpand(item)">
|
||||||
<template v-else>
|
<span>项目套餐明细</span>
|
||||||
<div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty">
|
<el-icon :class="['expand-icon', { expanded: item.itemPackageExpanded }]">
|
||||||
暂无套餐明细
|
<ArrowDown />
|
||||||
|
</el-icon>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="package-details-list">
|
<div v-show="item.itemPackageExpanded">
|
||||||
<div class="package-details-head">套餐明细</div>
|
<div v-if="item.packageDetailsLoading" class="package-details-loading">加载中...</div>
|
||||||
<div
|
<template v-else>
|
||||||
v-for="(detail, dIdx) in getPackageDetailsList(item)"
|
<div v-if="getPackageDetailsList(item).length === 0" class="package-details-empty">
|
||||||
:key="detail.id ?? detail.itemCode ?? `d-${dIdx}`"
|
暂无套餐明细
|
||||||
class="detail-row"
|
</div>
|
||||||
>
|
<div v-else class="package-details-list">
|
||||||
<el-tooltip :content="detail.name" placement="top" :show-after="500">
|
<div
|
||||||
<span class="detail-name">{{ detail.name }}</span>
|
v-for="(detail, dIdx) in getPackageDetailsList(item)"
|
||||||
</el-tooltip>
|
:key="detail.id ?? detail.itemCode ?? `d-${dIdx}`"
|
||||||
<div class="detail-meta">
|
class="detail-row"
|
||||||
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
|
>
|
||||||
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
|
<el-tooltip :content="detail.name" placement="top" :show-after="500">
|
||||||
|
<span class="detail-name">{{ detail.name }}</span>
|
||||||
|
</el-tooltip>
|
||||||
|
<div class="detail-meta">
|
||||||
|
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
|
||||||
|
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</template>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</div>
|
||||||
|
<div class="selected-card-section">
|
||||||
|
<div class="selected-section-title">检查方法</div>
|
||||||
|
<div v-if="!item.methods || item.methods.length === 0" class="selected-method-empty">
|
||||||
|
暂无检查方法
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-for="method in item.methods"
|
||||||
|
:key="method.id"
|
||||||
|
class="selected-method-option"
|
||||||
|
>
|
||||||
|
<el-checkbox
|
||||||
|
:model-value="item.selectedMethod?.id === method.id"
|
||||||
|
@change="(val) => selectMethodCheckbox(val, item, method)"
|
||||||
|
class="method-checkbox"
|
||||||
|
>
|
||||||
|
{{ method.name }}
|
||||||
|
</el-checkbox>
|
||||||
|
<span class="method-price-tag">¥{{ method.packagePrice || method.price || 0 }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div v-if="shouldShowMethodPackageBody(item)">
|
||||||
|
<div class="package-toggle" @click.stop="toggleMethodPackageExpand(item)">
|
||||||
|
<span>检查方法套餐明细</span>
|
||||||
|
<el-icon :class="['expand-icon', { expanded: item.methodPackageExpanded }]">
|
||||||
|
<ArrowDown />
|
||||||
|
</el-icon>
|
||||||
|
</div>
|
||||||
|
<div v-show="item.methodPackageExpanded">
|
||||||
|
<div v-if="item.methodPackageLoading" class="package-details-loading">加载中...</div>
|
||||||
|
<template v-else>
|
||||||
|
<div v-if="getMethodPackageDetailsList(item).length === 0" class="package-details-empty">
|
||||||
|
暂无检查方法套餐明细
|
||||||
|
</div>
|
||||||
|
<div v-else class="package-details-list method-package-list">
|
||||||
|
<div
|
||||||
|
v-for="(detail, dIdx) in getMethodPackageDetailsList(item)"
|
||||||
|
:key="detail.id ?? detail.itemCode ?? `md-${dIdx}`"
|
||||||
|
class="detail-row"
|
||||||
|
>
|
||||||
|
<el-tooltip :content="detail.name" placement="top" :show-after="500">
|
||||||
|
<span class="detail-name">{{ detail.name }}</span>
|
||||||
|
</el-tooltip>
|
||||||
|
<div class="detail-meta">
|
||||||
|
<span class="detail-qty">×{{ detail.quantity || 1 }}</span>
|
||||||
|
<span class="detail-price">¥{{ formatDetailAmount(detail.price) }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -493,7 +519,7 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import { ref, reactive, computed, watch, onMounted, nextTick } from 'vue';
|
import { ref, reactive, computed, watch, onMounted, nextTick } from 'vue';
|
||||||
import { ElMessage, ElMessageBox } from 'element-plus';
|
import { ElMessage, ElMessageBox } from 'element-plus';
|
||||||
import { Printer, Delete, ArrowDown, ArrowUp, Close } from '@element-plus/icons-vue';
|
import { Printer, Delete, ArrowDown, Close } from '@element-plus/icons-vue';
|
||||||
import useUserStore from '@/store/modules/user';
|
import useUserStore from '@/store/modules/user';
|
||||||
import request from '@/utils/request';
|
import request from '@/utils/request';
|
||||||
import { listCheckMethod, searchCheckMethod, listCheckPackage } from '@/api/system/checkType';
|
import { listCheckMethod, searchCheckMethod, listCheckPackage } from '@/api/system/checkType';
|
||||||
@@ -624,24 +650,48 @@ async function loadPackageDetails(row, treeNode, resolve) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// #428修复 + #426修复: 为已选择项目加载套餐明细(通过packageId或packageName查询)
|
|
||||||
/** 套餐明细挂在「部位」或已选的「检查方法」上(方法可带 packageId) */
|
|
||||||
function getPackageCarrier(item) {
|
|
||||||
return item?.selectedMethod?.packageId ? item.selectedMethod : item;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getPackageDetailsList(item) {
|
function getPackageDetailsList(item) {
|
||||||
// 明细挂在行对象上,避免仅写入 methods 内嵌对象时首帧不触发视图更新(体感需点两次才展开)
|
// 明细挂在行对象上,避免仅写入 methods 内嵌对象时首帧不触发视图更新(体感需点两次才展开)
|
||||||
if (Array.isArray(item?.packageDetailsDisplay)) {
|
if (Array.isArray(item?.packageDetailsDisplay)) {
|
||||||
return item.packageDetailsDisplay;
|
return item.packageDetailsDisplay;
|
||||||
}
|
}
|
||||||
const carrier = getPackageCarrier(item);
|
return Array.isArray(item?.packageDetails) ? item.packageDetails : [];
|
||||||
return Array.isArray(carrier?.packageDetails) ? carrier.packageDetails : [];
|
}
|
||||||
|
|
||||||
|
function getMethodPackageDetailsList(item) {
|
||||||
|
if (Array.isArray(item?.methodPackageDetails)) {
|
||||||
|
return item.methodPackageDetails;
|
||||||
|
}
|
||||||
|
return Array.isArray(item?.selectedMethod?.packageDetails) ? item.selectedMethod.packageDetails : [];
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 有套餐 ID 或 packageName 的已选行才展示右侧套餐区(加载中 / 空 / 明细列表) */
|
/** 有套餐 ID 或 packageName 的已选行才展示右侧套餐区(加载中 / 空 / 明细列表) */
|
||||||
function shouldShowPackageBody(item) {
|
function shouldShowPackageBody(item) {
|
||||||
return !!(getPackageCarrier(item)?.packageId || item.packageName || item.packageId);
|
return shouldShowItemPackageBody(item) || shouldShowMethodPackageBody(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function hasItemPackage(item) {
|
||||||
|
return !!(item?.packageId || item?.packageName);
|
||||||
|
}
|
||||||
|
|
||||||
|
function hasMethodPackage(item) {
|
||||||
|
return !!(item?.selectedMethod?.packageId || item?.selectedMethod?.packageName);
|
||||||
|
}
|
||||||
|
|
||||||
|
function isSamePackage(item) {
|
||||||
|
if (!hasItemPackage(item) || !hasMethodPackage(item)) return false;
|
||||||
|
if (item.packageId && item.selectedMethod?.packageId) {
|
||||||
|
return String(item.packageId) === String(item.selectedMethod.packageId);
|
||||||
|
}
|
||||||
|
return String(item.packageName || '') === String(item.selectedMethod?.packageName || '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldShowItemPackageBody(item) {
|
||||||
|
return hasItemPackage(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldShowMethodPackageBody(item) {
|
||||||
|
return hasMethodPackage(item) && !isSamePackage(item);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 金额展示:统一两位小数 */
|
/** 金额展示:统一两位小数 */
|
||||||
@@ -650,20 +700,18 @@ function formatDetailAmount(value) {
|
|||||||
return Number.isFinite(n) ? n.toFixed(2) : '0.00';
|
return Number.isFinite(n) ? n.toFixed(2) : '0.00';
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 默认检查方法:优先与部位 packageId 一致的方法,否则取首个带套餐的方法,否则取第一个 */
|
/** 已选卡片名称:去掉 UI 上冗余的“套餐”前缀,完整名称通过 tooltip 展示 */
|
||||||
function pickDefaultMethod(methods, partItem) {
|
function getDisplayItemName(item) {
|
||||||
if (!methods?.length) return null;
|
return String(item?.name || '').replace(/^套餐[::\-\s]*/, '');
|
||||||
if (methods.length === 1) return methods[0];
|
}
|
||||||
const pid = partItem?.packageId ?? null;
|
|
||||||
if (pid != null && pid !== '') {
|
function getSelectedItemAmount(item) {
|
||||||
const matched = methods.find(
|
const itemPrice = Number(item?.price || 0);
|
||||||
(x) => x.packageId != null && String(x.packageId) === String(pid)
|
const methodPrice = Number(item?.selectedMethod?.packagePrice || 0);
|
||||||
);
|
if (!hasMethodPackage(item) || isSamePackage(item)) {
|
||||||
if (matched) return matched;
|
return itemPrice;
|
||||||
}
|
}
|
||||||
const withPkg = methods.find((x) => x.packageId != null);
|
return itemPrice + methodPrice;
|
||||||
if (withPkg) return withPkg;
|
|
||||||
return methods[0];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function parsePackageDetailsPayload(res) {
|
function parsePackageDetailsPayload(res) {
|
||||||
@@ -679,15 +727,15 @@ function parsePackageDetailsPayload(res) {
|
|||||||
|
|
||||||
// #428: 为已选择项目加载套餐明细(后端:CheckTypeController /system/check-type/package/{id}/details)
|
// #428: 为已选择项目加载套餐明细(后端:CheckTypeController /system/check-type/package/{id}/details)
|
||||||
async function loadPackageDetailsForItem(item) {
|
async function loadPackageDetailsForItem(item) {
|
||||||
const carrier = getPackageCarrier(item);
|
let packageId = item.packageId;
|
||||||
let packageId = item.packageId || carrier?.packageId;
|
const packageName = item.packageName;
|
||||||
if (!packageId && !item.packageName) {
|
if (!packageId && !packageName) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
item.packageDetailsLoading = true;
|
item.packageDetailsLoading = true;
|
||||||
try {
|
try {
|
||||||
if (!packageId && item.packageName) {
|
if (!packageId && packageName) {
|
||||||
const pkgRes = await listCheckPackage({ packageName: item.packageName });
|
const pkgRes = await listCheckPackage({ packageName });
|
||||||
let packages = pkgRes?.data || [];
|
let packages = pkgRes?.data || [];
|
||||||
if (!Array.isArray(packages)) {
|
if (!Array.isArray(packages)) {
|
||||||
packages = packages.records || packages.data || [];
|
packages = packages.records || packages.data || [];
|
||||||
@@ -699,6 +747,7 @@ async function loadPackageDetailsForItem(item) {
|
|||||||
}
|
}
|
||||||
packageId = packages[0].id;
|
packageId = packages[0].id;
|
||||||
item.packageId = packageId;
|
item.packageId = packageId;
|
||||||
|
item.packageName = item.packageName || packageName;
|
||||||
}
|
}
|
||||||
if (!packageId) {
|
if (!packageId) {
|
||||||
item.packageDetails = [];
|
item.packageDetails = [];
|
||||||
@@ -718,7 +767,6 @@ async function loadPackageDetailsForItem(item) {
|
|||||||
quantity: detail.quantity || 1
|
quantity: detail.quantity || 1
|
||||||
}));
|
}));
|
||||||
item.packageDetailsDisplay = mapped;
|
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 = Array.isArray(res.data)
|
||||||
? res.data.map((detail) => ({
|
? res.data.map((detail) => ({
|
||||||
@@ -735,7 +783,6 @@ async function loadPackageDetailsForItem(item) {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('加载套餐明细失败:', err);
|
console.error('加载套餐明细失败:', err);
|
||||||
item.packageDetailsDisplay = [];
|
item.packageDetailsDisplay = [];
|
||||||
carrier.packageDetails = [];
|
|
||||||
item.packageDetails = [];
|
item.packageDetails = [];
|
||||||
} finally {
|
} finally {
|
||||||
item.packageDetailsLoading = false;
|
item.packageDetailsLoading = false;
|
||||||
@@ -1058,7 +1105,10 @@ async function loadCategoryList() {
|
|||||||
|
|
||||||
// 默认展开第一个
|
// 默认展开第一个
|
||||||
if (categoryList.value.length > 0) {
|
if (categoryList.value.length > 0) {
|
||||||
activeNames.value = categoryList.value[0].typeId;
|
const firstCat = categoryList.value[0];
|
||||||
|
activeNames.value = firstCat.typeId;
|
||||||
|
await nextTick();
|
||||||
|
await handleCategoryExpand(firstCat);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('加载检查项目分类失败', err);
|
console.error('加载检查项目分类失败', err);
|
||||||
@@ -1079,10 +1129,9 @@ const filteredCategoryList = computed(() => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// ====== 合计 ======
|
// ====== 合计 ======
|
||||||
// Bug #384修复: 如果选中了检查方法,使用套餐价格;否则使用部位价格
|
|
||||||
const totalAmountCalc = computed(() => {
|
const totalAmountCalc = computed(() => {
|
||||||
const total = selectedItems.value.reduce((sum, item) => {
|
const total = selectedItems.value.reduce((sum, item) => {
|
||||||
const effectivePrice = item.selectedMethod?.packagePrice || item.price;
|
const effectivePrice = getSelectedItemAmount(item);
|
||||||
return sum + (effectivePrice * (item.quantity || 1));
|
return sum + (effectivePrice * (item.quantity || 1));
|
||||||
}, 0);
|
}, 0);
|
||||||
return total.toFixed(2);
|
return total.toFixed(2);
|
||||||
@@ -1215,8 +1264,7 @@ function handleSave() {
|
|||||||
itemCode: String(item.id),
|
itemCode: String(item.id),
|
||||||
itemName: item.name,
|
itemName: item.name,
|
||||||
bodyPartCode: item.checkType || 'unknown',
|
bodyPartCode: item.checkType || 'unknown',
|
||||||
// Bug #384修复: 如果选中了检查方法且有套餐价格,使用套餐价格;否则使用部位价格
|
itemFee: getSelectedItemAmount(item),
|
||||||
itemFee: item.selectedMethod?.packagePrice || item.price,
|
|
||||||
performDeptCode: form.performDeptCode || '',
|
performDeptCode: form.performDeptCode || '',
|
||||||
itemStatus: 0,
|
itemStatus: 0,
|
||||||
itemSeq: index + 1,
|
itemSeq: index + 1,
|
||||||
@@ -1269,6 +1317,8 @@ function handleRowClick(row) {
|
|||||||
methods: [],
|
methods: [],
|
||||||
selectedMethod: null,
|
selectedMethod: null,
|
||||||
expanded: false,
|
expanded: false,
|
||||||
|
itemPackageExpanded: true,
|
||||||
|
methodPackageExpanded: true,
|
||||||
packageDetailsLoading: false,
|
packageDetailsLoading: false,
|
||||||
isPackage: false,
|
isPackage: false,
|
||||||
packageId: null,
|
packageId: null,
|
||||||
@@ -1298,17 +1348,10 @@ function handleRowClick(row) {
|
|||||||
if (m.checkMethodId) {
|
if (m.checkMethodId) {
|
||||||
item.selectedMethod = item.methods.find(md => String(md.id) === String(m.checkMethodId)) || null;
|
item.selectedMethod = item.methods.find(md => String(md.id) === String(m.checkMethodId)) || null;
|
||||||
if (item.selectedMethod?.packageId) {
|
if (item.selectedMethod?.packageId) {
|
||||||
item.isPackage = true;
|
|
||||||
item.packageId = item.selectedMethod.packageId;
|
|
||||||
item.hasChildren = true; // #426修复
|
item.hasChildren = true; // #426修复
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!item.selectedMethod && item.methods.length) {
|
|
||||||
item.selectedMethod = pickDefaultMethod(item.methods, { packageId: item.packageId });
|
|
||||||
}
|
|
||||||
if (item.selectedMethod?.packageId) {
|
if (item.selectedMethod?.packageId) {
|
||||||
item.packageId = item.selectedMethod.packageId;
|
|
||||||
item.isPackage = true;
|
|
||||||
item.hasChildren = true; // #426修复
|
item.hasChildren = true; // #426修复
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1322,14 +1365,21 @@ function handleRowClick(row) {
|
|||||||
selectedItems.value = itemsWithMethods;
|
selectedItems.value = itemsWithMethods;
|
||||||
// 加载套餐明细(单个失败不影响其他项目和明细显示)
|
// 加载套餐明细(单个失败不影响其他项目和明细显示)
|
||||||
for (const it of selectedItems.value) {
|
for (const it of selectedItems.value) {
|
||||||
if (getPackageCarrier(it)?.packageId) {
|
if (hasItemPackage(it)) {
|
||||||
try {
|
try {
|
||||||
await loadPackageDetailsForItem(it);
|
await loadPackageDetailsForItem(it);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('加载套餐明细失败:', it.name, e);
|
console.error('加载套餐明细失败:', it.name, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
it.expanded = !!getPackageCarrier(it)?.packageId;
|
if (hasMethodPackage(it) && !isSamePackage(it)) {
|
||||||
|
try {
|
||||||
|
await loadMethodPackageDetails(it, it.selectedMethod);
|
||||||
|
} catch (e) {
|
||||||
|
console.error('加载检查方法套餐明细失败:', it.name, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
it.expanded = shouldShowPackageBody(it);
|
||||||
}
|
}
|
||||||
syncCategoryChecked();
|
syncCategoryChecked();
|
||||||
// Bug #384修复: 回充后更新检查方法显示
|
// Bug #384修复: 回充后更新检查方法显示
|
||||||
@@ -1359,97 +1409,6 @@ function handleDelete(row) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bug #428修复: 判断某个检查方法是否已被选中(任意项目关联了该方法)
|
|
||||||
function isMethodSelected(method, cat) {
|
|
||||||
return selectedItems.value.some(item =>
|
|
||||||
item.selectedMethod?.id === method.id && item.checkType === cat.typeName
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bug #428修复: 勾选检查方法
|
|
||||||
async function handleMethodSelect(checked, method, cat) {
|
|
||||||
if (checked) {
|
|
||||||
// 找到该方法所属的第一个检查项目
|
|
||||||
const targetItem = cat.items[0];
|
|
||||||
if (!targetItem) {
|
|
||||||
// 如果分类下没有项目,尝试从其他分类找同名项目或创建
|
|
||||||
console.warn('分类下没有检查项目,无法关联方法');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 如果该项目已存在,只更新 selectedMethod
|
|
||||||
const existingItem = selectedItems.value.find(s => s.id === targetItem.id);
|
|
||||||
if (existingItem) {
|
|
||||||
existingItem.selectedMethod = method;
|
|
||||||
// 从方法中获取套餐信息(支持 packageId 或 packageName 解析)
|
|
||||||
if (method.packageId || method.packageName) {
|
|
||||||
existingItem.isPackage = true;
|
|
||||||
existingItem.packageId = method.packageId || existingItem.packageId;
|
|
||||||
existingItem.hasChildren = true; // #426修复
|
|
||||||
existingItem.packageName = method.packageName || existingItem.packageName; // #428修复: 确保 packageName 同步
|
|
||||||
// 预加载套餐明细
|
|
||||||
loadPackageDetailsForItem(existingItem);
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 如果该项目不存在,创建一个并关联方法
|
|
||||||
if (selectedItems.value.length > 0) {
|
|
||||||
const currentCategory = selectedItems.value[0].checkType;
|
|
||||||
// Bug #428修复: 使用 cat.typeName 进行比较(与 newItem.checkType 保持一致)
|
|
||||||
const newCategory = cat.typeName || '';
|
|
||||||
if (currentCategory !== newCategory) {
|
|
||||||
ElMessage.warning('一个检查单不能同时选择多个项目类型的检查项目');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const newItem = {
|
|
||||||
id: targetItem.id, name: targetItem.name,
|
|
||||||
price: targetItem.price, quantity: 1,
|
|
||||||
serviceFee: targetItem.serviceFee || 0,
|
|
||||||
unit: targetItem.unit || '次',
|
|
||||||
applyPart: targetItem.name,
|
|
||||||
checkType: cat.typeName,
|
|
||||||
nationalCode: targetItem.nationalCode || '',
|
|
||||||
checked: true,
|
|
||||||
methods: cat.methods || [method], // #428修复: 复制分类下全部方法,允许用户切换
|
|
||||||
selectedMethod: method,
|
|
||||||
expanded: false,
|
|
||||||
// 从方法或项目中获取套餐信息
|
|
||||||
isPackage: !!method.packageId || !!targetItem.packageName,
|
|
||||||
packageId: method.packageId || targetItem.packageId || null,
|
|
||||||
packageName: method.packageName || targetItem.packageName || null, // #428修复: 复制 packageName,确保套餐明细可加载
|
|
||||||
hasChildren: !!(method.packageId || method.packageName || targetItem.packageId || targetItem.packageName) // #426修复: 树形表格懒加载展开标记,支持 packageName 解析
|
|
||||||
};
|
|
||||||
selectedItems.value.push(newItem);
|
|
||||||
|
|
||||||
// 如果是套餐,预加载套餐明细
|
|
||||||
if (newItem.isPackage && (newItem.packageId || newItem.packageName)) {
|
|
||||||
loadPackageDetailsForItem(newItem);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 自动回填执行科室
|
|
||||||
if (selectedItems.value.length === 1 && cat?.performDeptName) {
|
|
||||||
form.performDeptCode = cat.performDeptName;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 同时勾选左侧项目的 checkbox
|
|
||||||
targetItem.checked = true;
|
|
||||||
|
|
||||||
} else {
|
|
||||||
// 取消选择方法:将 selectedItems 中关联该方法的项的 selectedMethod 清空
|
|
||||||
const itemsWithMethod = selectedItems.value.filter(
|
|
||||||
item => item.selectedMethod?.id === method.id
|
|
||||||
);
|
|
||||||
for (const item of itemsWithMethod) {
|
|
||||||
item.selectedMethod = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
|
||||||
}
|
|
||||||
|
|
||||||
// ====== 勾选逻辑 ======
|
// ====== 勾选逻辑 ======
|
||||||
async function handleItemSelect(checked, item, cat) {
|
async function handleItemSelect(checked, item, cat) {
|
||||||
if (checked) {
|
if (checked) {
|
||||||
@@ -1506,6 +1465,8 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
methods: methods,
|
methods: methods,
|
||||||
selectedMethod: null,
|
selectedMethod: null,
|
||||||
expanded: false,
|
expanded: false,
|
||||||
|
itemPackageExpanded: true,
|
||||||
|
methodPackageExpanded: true,
|
||||||
isPackage: !!(item.packageId || item.packageName),
|
isPackage: !!(item.packageId || item.packageName),
|
||||||
packageName: item.packageName || null,
|
packageName: item.packageName || null,
|
||||||
packageDetailsLoading: false,
|
packageDetailsLoading: false,
|
||||||
@@ -1516,15 +1477,15 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
// 必须用数组里的响应式行,不能继续改局部 newRow:push 后列表内是 proxy,改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
|
// 必须用数组里的响应式行,不能继续改局部 newRow:push 后列表内是 proxy,改 raw 对象不会触发右侧卡片更新(会一直卡在「加载中」)
|
||||||
const row = selectedItems.value[selectedItems.value.length - 1];
|
const row = selectedItems.value[selectedItems.value.length - 1];
|
||||||
|
|
||||||
// 右侧不再展示「检查方法」列表:自动选默认方法(保存、计价仍依赖 selectedMethod)
|
// 勾选项目只加入项目列表,检查方法由用户在“检查方法”区域手动选择
|
||||||
if (methods.length >= 1) {
|
row.selectedMethod = null;
|
||||||
row.selectedMethod = pickDefaultMethod(methods, item);
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
|
|
||||||
// 有套餐 ID 时默认展开(先显示加载区,明细写入行对象 packageDetailsDisplay)
|
// 新勾选项目后默认展开,直接展示检查方法状态和套餐明细
|
||||||
row.expanded = !!getPackageCarrier(row)?.packageId;
|
row.expanded = true;
|
||||||
if (getPackageCarrier(row)?.packageId) {
|
row.itemPackageExpanded = true;
|
||||||
|
row.methodPackageExpanded = true;
|
||||||
|
if (hasItemPackage(row)) {
|
||||||
await loadPackageDetailsForItem(row);
|
await loadPackageDetailsForItem(row);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1550,13 +1511,35 @@ async function handleItemSelect(checked, item, cat) {
|
|||||||
// Bug #384修复 + #426修复: 展开/收起项目卡片
|
// Bug #384修复 + #426修复: 展开/收起项目卡片
|
||||||
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 && hasItemPackage(item) && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
||||||
await loadPackageDetailsForItem(item);
|
await loadPackageDetailsForItem(item);
|
||||||
}
|
}
|
||||||
if (item.expanded && shouldShowPackageBody(item)) {
|
if (
|
||||||
if (getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
item.expanded &&
|
||||||
await loadPackageDetailsForItem(item);
|
shouldShowMethodPackageBody(item) &&
|
||||||
}
|
getMethodPackageDetailsList(item).length === 0 &&
|
||||||
|
!item.methodPackageLoading
|
||||||
|
) {
|
||||||
|
await loadMethodPackageDetails(item, item.selectedMethod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function toggleItemPackageExpand(item) {
|
||||||
|
item.itemPackageExpanded = !item.itemPackageExpanded;
|
||||||
|
if (item.itemPackageExpanded && getPackageDetailsList(item).length === 0 && !item.packageDetailsLoading) {
|
||||||
|
await loadPackageDetailsForItem(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function toggleMethodPackageExpand(item) {
|
||||||
|
item.methodPackageExpanded = !item.methodPackageExpanded;
|
||||||
|
if (
|
||||||
|
item.methodPackageExpanded &&
|
||||||
|
item.selectedMethod &&
|
||||||
|
getMethodPackageDetailsList(item).length === 0 &&
|
||||||
|
!item.methodPackageLoading
|
||||||
|
) {
|
||||||
|
await loadMethodPackageDetails(item, item.selectedMethod);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1564,9 +1547,8 @@ async function toggleItemExpand(item) {
|
|||||||
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 || method.packageName)) {
|
item.expanded = true;
|
||||||
loadPackageDetailsForItem(item);
|
item.methodPackageExpanded = true;
|
||||||
}
|
|
||||||
// 动态加载该方法对应的套餐明细
|
// 动态加载该方法对应的套餐明细
|
||||||
await loadMethodPackageDetails(item, method);
|
await loadMethodPackageDetails(item, method);
|
||||||
} else {
|
} else {
|
||||||
@@ -1587,36 +1569,43 @@ async function loadMethodPackageDetails(item, method) {
|
|||||||
item.methodPackageLoading = true;
|
item.methodPackageLoading = true;
|
||||||
item.methodPackageDetails = [];
|
item.methodPackageDetails = [];
|
||||||
try {
|
try {
|
||||||
if (!method.packageName) {
|
let packageId = method.packageId;
|
||||||
|
if (!packageId && !method.packageName) {
|
||||||
item.methodPackageLoading = false;
|
item.methodPackageLoading = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// 通过packageName查询套餐获取packageId
|
// 通过packageName查询套餐获取packageId
|
||||||
const pkgRes = await listCheckPackage({ packageName: method.packageName });
|
if (!packageId && method.packageName) {
|
||||||
let packages = pkgRes?.data || [];
|
const pkgRes = await listCheckPackage({ packageName: method.packageName });
|
||||||
if (!Array.isArray(packages)) {
|
let packages = pkgRes?.data || [];
|
||||||
packages = packages.records || packages.data || [];
|
if (!Array.isArray(packages)) {
|
||||||
|
packages = packages.records || packages.data || [];
|
||||||
|
}
|
||||||
|
if (packages.length === 0) {
|
||||||
|
item.methodPackageLoading = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
packageId = packages[0].id;
|
||||||
|
method.packageId = packageId;
|
||||||
}
|
}
|
||||||
if (packages.length === 0) {
|
|
||||||
item.methodPackageLoading = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const packageId = packages[0].id;
|
|
||||||
// 查询套餐明细
|
// 查询套餐明细
|
||||||
const detailRes = await request({
|
const detailRes = await request({
|
||||||
url: `/system/package/${packageId}/details`,
|
url: `/system/check-type/package/${packageId}/details`,
|
||||||
method: 'get'
|
method: 'get'
|
||||||
});
|
});
|
||||||
if (detailRes.code === 200 && detailRes.data) {
|
const detailList = parsePackageDetailsPayload(detailRes);
|
||||||
item.methodPackageDetails = detailRes.data.map(d => ({
|
if (detailList.length > 0) {
|
||||||
|
const mapped = detailList.map(d => ({
|
||||||
id: d.id,
|
id: d.id,
|
||||||
name: d.itemName || d.name,
|
name: d.name || d.itemName,
|
||||||
quantity: d.quantity || 1,
|
quantity: d.quantity || 1,
|
||||||
unit: d.unit || '次',
|
unit: d.unit || '次',
|
||||||
price: d.unitPrice || d.price || 0,
|
price: d.price ?? d.unitPrice ?? d.itemPrice ?? 0,
|
||||||
amount: d.amount || d.total || 0,
|
amount: d.amount || d.total || 0,
|
||||||
checked: true // 默认勾选
|
checked: true // 默认勾选
|
||||||
}));
|
}));
|
||||||
|
item.methodPackageDetails = mapped;
|
||||||
|
method.packageDetails = mapped;
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('加载方法套餐明细失败:', err);
|
console.error('加载方法套餐明细失败:', err);
|
||||||
@@ -1630,21 +1619,19 @@ async function loadMethodPackageDetails(item, method) {
|
|||||||
async function onDetailMethodChange(row, val) {
|
async function onDetailMethodChange(row, val) {
|
||||||
row.selectedMethod = val || null;
|
row.selectedMethod = val || null;
|
||||||
if (val?.packageId || val?.packageName) {
|
if (val?.packageId || val?.packageName) {
|
||||||
row.packageId = val.packageId || row.packageId;
|
|
||||||
row.packageName = val.packageName || row.packageName;
|
|
||||||
row.isPackage = true;
|
|
||||||
row.hasChildren = true; // #426修复
|
row.hasChildren = true; // #426修复
|
||||||
}
|
}
|
||||||
row.packageDetailsDisplay = undefined;
|
row.methodPackageDetails = [];
|
||||||
const carrier = getPackageCarrier(row);
|
|
||||||
if (carrier) {
|
|
||||||
carrier.packageDetails = undefined;
|
|
||||||
}
|
|
||||||
updateMethodDisplay();
|
updateMethodDisplay();
|
||||||
row.expanded = !!getPackageCarrier(row)?.packageId;
|
row.expanded = shouldShowPackageBody(row);
|
||||||
if (getPackageCarrier(row)?.packageId) {
|
row.itemPackageExpanded = true;
|
||||||
|
row.methodPackageExpanded = true;
|
||||||
|
if (hasItemPackage(row)) {
|
||||||
await loadPackageDetailsForItem(row);
|
await loadPackageDetailsForItem(row);
|
||||||
}
|
}
|
||||||
|
if (val?.packageId || val?.packageName) {
|
||||||
|
await loadMethodPackageDetails(row, val);
|
||||||
|
}
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
form.totalAmount = totalAmountCalc.value;
|
form.totalAmount = totalAmountCalc.value;
|
||||||
});
|
});
|
||||||
@@ -1794,7 +1781,7 @@ defineExpose({ getList });
|
|||||||
|
|
||||||
/* 右:分类面板 */
|
/* 右:分类面板 */
|
||||||
.category-panel {
|
.category-panel {
|
||||||
width: 420px;
|
width: 560px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
background: #fff;
|
background: #fff;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
@@ -1951,9 +1938,9 @@ defineExpose({ getList });
|
|||||||
/* 已选择 tags */
|
/* 已选择 tags */
|
||||||
/* 已选择:加宽,避免套餐明细挤成一团 */
|
/* 已选择:加宽,避免套餐明细挤成一团 */
|
||||||
.selected-panel {
|
.selected-panel {
|
||||||
width: 220px;
|
width: 260px;
|
||||||
min-width: 200px;
|
min-width: 240px;
|
||||||
max-width: 280px;
|
max-width: 320px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -2013,9 +2000,8 @@ defineExpose({ getList });
|
|||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: #303133;
|
color: #303133;
|
||||||
overflow: hidden;
|
line-height: 1.4;
|
||||||
text-overflow: ellipsis;
|
word-break: break-word;
|
||||||
white-space: nowrap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.card-price {
|
.card-price {
|
||||||
@@ -2030,12 +2016,11 @@ defineExpose({ getList });
|
|||||||
color: #909399;
|
color: #909399;
|
||||||
transition: transform 0.2s ease;
|
transition: transform 0.2s ease;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
transition: transform 0.2s;
|
transform: rotate(-90deg);
|
||||||
transform: rotate(0deg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.expand-icon.expanded {
|
.expand-icon.expanded {
|
||||||
transform: rotate(90deg);
|
transform: rotate(0deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Bug #428修复: 套餐明细列表样式 */
|
/* Bug #428修复: 套餐明细列表样式 */
|
||||||
@@ -2078,6 +2063,55 @@ defineExpose({ getList });
|
|||||||
background: #fafbfc;
|
background: #fafbfc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.selected-card-section {
|
||||||
|
padding: 10px;
|
||||||
|
border-bottom: 1px solid #ebeef5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.selected-section-title {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #409eff;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
padding-bottom: 6px;
|
||||||
|
border-bottom: 1px dashed #d9ecff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.selected-method-option {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 8px;
|
||||||
|
padding: 4px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.selected-method-option .method-checkbox {
|
||||||
|
flex: 1;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.selected-method-empty {
|
||||||
|
color: #c0c4cc;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.package-toggle {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 10px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #909399;
|
||||||
|
cursor: pointer;
|
||||||
|
border-bottom: 1px dashed #dcdfe6;
|
||||||
|
background: #fffbe6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.package-toggle:hover {
|
||||||
|
color: #409eff;
|
||||||
|
}
|
||||||
|
|
||||||
.package-details-loading,
|
.package-details-loading,
|
||||||
.package-details-empty {
|
.package-details-empty {
|
||||||
padding: 12px 10px;
|
padding: 12px 10px;
|
||||||
|
|||||||
@@ -86,7 +86,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 label="申请单名称" min-width="140">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
<span>{{ buildApplicationName(scope.row) }}</span>
|
<span>{{ buildApplicationName(scope.row) }}</span>
|
||||||
</template>
|
</template>
|
||||||
@@ -444,11 +444,9 @@ const buildApplicationName = (row) => {
|
|||||||
if (!details || details.length === 0) {
|
if (!details || details.length === 0) {
|
||||||
return row.name || '-';
|
return row.name || '-';
|
||||||
}
|
}
|
||||||
if (details.length === 1) {
|
const names = details.map(d => d.adviceName).filter(Boolean);
|
||||||
return details[0].adviceName || row.name || '-';
|
if (names.length === 0) return row.name || '-';
|
||||||
}
|
return names.join(' + ');
|
||||||
const first = details[0];
|
|
||||||
return `${first.adviceName || ''}等${details.length}项`;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -529,7 +529,13 @@ const handleViewDetail = async (row) => {
|
|||||||
if (row.descJson) {
|
if (row.descJson) {
|
||||||
try {
|
try {
|
||||||
const obj = JSON.parse(row.descJson);
|
const obj = JSON.parse(row.descJson);
|
||||||
obj.targetDepartment = recursionFun(obj.targetDepartment);
|
// 将发往科室 ID 转换为名称
|
||||||
|
if (obj.targetDepartment) {
|
||||||
|
const deptName = recursionFun(obj.targetDepartment);
|
||||||
|
if (deptName) {
|
||||||
|
obj.targetDepartment = deptName;
|
||||||
|
}
|
||||||
|
}
|
||||||
// 转换申请类型编码为可读文本
|
// 转换申请类型编码为可读文本
|
||||||
if (obj.applicationType === 0) obj.applicationType = '普通';
|
if (obj.applicationType === 0) obj.applicationType = '普通';
|
||||||
else if (obj.applicationType === 1) obj.applicationType = '急诊';
|
else if (obj.applicationType === 1) obj.applicationType = '急诊';
|
||||||
|
|||||||
@@ -428,11 +428,52 @@ const loadEditData = () => {
|
|||||||
const projectWithDepartment = (selectProjectIds) => {
|
const projectWithDepartment = (selectProjectIds) => {
|
||||||
if (!selectProjectIds || selectProjectIds.length === 0) {
|
if (!selectProjectIds || selectProjectIds.length === 0) {
|
||||||
form.targetDepartment = '';
|
form.targetDepartment = '';
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 获取第一个选中项目的发往科室(orgId)
|
||||||
|
// 优先使用配置的发往科室,如果没有则保留手动选择
|
||||||
|
const selectedProject = applicationListAll.value?.find(
|
||||||
|
item => selectProjectIds.includes(item.adviceDefinitionId)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (selectedProject && selectedProject.orgId) {
|
||||||
|
// 项目配置了发往科室,自动填充
|
||||||
|
const orgId = selectedProject.orgId;
|
||||||
|
const orgName = selectedProject.orgName;
|
||||||
|
|
||||||
|
// 查找树中对应的节点,获取正确的 id 类型
|
||||||
|
const findNode = (nodes, targetId) => {
|
||||||
|
if (!nodes) return null;
|
||||||
|
for (const node of nodes) {
|
||||||
|
if (String(node.id) === String(targetId)) {
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
if (node.children && node.children.length > 0) {
|
||||||
|
const found = findNode(node.children, targetId);
|
||||||
|
if (found) return found;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const treeNode = findNode(orgOptions.value, orgId);
|
||||||
|
if (treeNode) {
|
||||||
|
// 使用树节点的原始 id 值(确保类型匹配)
|
||||||
|
form.targetDepartment = treeNode.id;
|
||||||
|
} else {
|
||||||
|
// 科室不在列表中(可能已删除),留空让用户手动选择
|
||||||
|
form.targetDepartment = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 如果没有配置发往科室,保留手动选择(不修改 form.targetDepartment)
|
||||||
};
|
};
|
||||||
|
|
||||||
watch(() => transferValue.value, (newValue) => {
|
watch(() => transferValue.value, (newValue) => {
|
||||||
projectWithDepartment(newValue);
|
// 使用 nextTick 确保 DOM 更新完成后再设置值
|
||||||
|
nextTick(() => {
|
||||||
|
projectWithDepartment(newValue);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const getPriorityCode = () => {
|
const getPriorityCode = () => {
|
||||||
|
|||||||
@@ -429,6 +429,8 @@ const props = defineProps({
|
|||||||
});
|
});
|
||||||
const isAdding = ref(false);
|
const isAdding = ref(false);
|
||||||
const isSaving = ref(false);
|
const isSaving = ref(false);
|
||||||
|
// 标记双击编辑的是否为已有数据的行(用于保存后是否自动添加下一行)
|
||||||
|
const wasDoubleClickEdit = ref(false);
|
||||||
const prescriptionRef = ref();
|
const prescriptionRef = ref();
|
||||||
const expandOrder = ref([]); //目前的展开行
|
const expandOrder = ref([]); //目前的展开行
|
||||||
const stockList = ref([]);
|
const stockList = ref([]);
|
||||||
@@ -1397,7 +1399,9 @@ function handleSaveSign(row, index) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
if (prescriptionList.value[0].adviceName) {
|
// 仅通过【新增】按钮创建的医嘱保存后才自动添加下一行空医嘱
|
||||||
|
// 双击编辑已有"待保存"医嘱保存时,不应自动添加空行
|
||||||
|
if (isAdding.value && prescriptionList.value[0].adviceName) {
|
||||||
handleAddPrescription();
|
handleAddPrescription();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -142,6 +142,13 @@
|
|||||||
</template>
|
</template>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
|
<el-table-column label="医嘱状态" prop="requestStatus_enumText" width="100">
|
||||||
|
<template #default="scope">
|
||||||
|
<el-tag :type="getStatusType(scope.row.requestStatus)" size="small">
|
||||||
|
{{ scope.row.requestStatus_enumText }}
|
||||||
|
</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
<el-table-column label="执行科室" prop="positionName" width="230" />
|
<el-table-column label="执行科室" prop="positionName" width="230" />
|
||||||
<el-table-column label="签发时间" prop="requestTime" width="230" />
|
<el-table-column label="签发时间" prop="requestTime" width="230" />
|
||||||
</el-table>
|
</el-table>
|
||||||
@@ -152,7 +159,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script setup>
|
<script setup>
|
||||||
import {ref, computed} from 'vue';
|
import {ref, computed, getCurrentInstance} from 'vue';
|
||||||
import {adviceVerify, cancel, getPrescriptionList} from './api';
|
import {adviceVerify, cancel, getPrescriptionList} from './api';
|
||||||
import {patientInfoList} from '../../components/store/patient.js';
|
import {patientInfoList} from '../../components/store/patient.js';
|
||||||
import {formatDateStr} from '@/utils/index';
|
import {formatDateStr} from '@/utils/index';
|
||||||
@@ -165,6 +172,19 @@ const { proxy } = getCurrentInstance();
|
|||||||
const loading = ref(false);
|
const loading = ref(false);
|
||||||
const chooseAll = ref(false);
|
const chooseAll = ref(false);
|
||||||
const selectionTrigger = ref(0);
|
const selectionTrigger = ref(0);
|
||||||
|
|
||||||
|
const getStatusType = (status) => {
|
||||||
|
const map = {
|
||||||
|
1: 'info', // 待发送
|
||||||
|
2: 'primary', // 已发送
|
||||||
|
3: 'success', // 已完成
|
||||||
|
4: 'warning', // 暂停
|
||||||
|
5: 'danger', // 取消/待退
|
||||||
|
6: 'danger', // 停嘱
|
||||||
|
7: 'info' // 不执行
|
||||||
|
}
|
||||||
|
return map[status] || 'info'
|
||||||
|
}
|
||||||
const hasDispensedSelected = computed(() => {
|
const hasDispensedSelected = computed(() => {
|
||||||
selectionTrigger.value;
|
selectionTrigger.value;
|
||||||
return getSelectRows().some(item => item.dispenseStatus === 4);
|
return getSelectRows().some(item => item.dispenseStatus === 4);
|
||||||
|
|||||||
@@ -1067,15 +1067,6 @@ const temporaryPatientInfo = ref({})
|
|||||||
const temporaryBillingMedicines = ref([])
|
const temporaryBillingMedicines = ref([])
|
||||||
const temporaryAdvices = ref([])
|
const temporaryAdvices = ref([])
|
||||||
|
|
||||||
// 🔧 新增:监听 temporaryAdvices 的变化,用于调试
|
|
||||||
watch(temporaryAdvices, (newVal, oldVal) => {
|
|
||||||
console.log('=== temporaryAdvices 变化 ===')
|
|
||||||
console.log('=== 新值 ===', newVal)
|
|
||||||
console.log('=== 新值[1]?.dosage ===', newVal[1]?.dosage)
|
|
||||||
console.log('=== 旧值 ===', oldVal)
|
|
||||||
console.log('=== 旧值[1]?.dosage ===', oldVal[1]?.dosage)
|
|
||||||
}, { deep: true })
|
|
||||||
|
|
||||||
const temporaryMedicalLoading = ref(false) // 🔧 新增:临时医嘱加载状态
|
const temporaryMedicalLoading = ref(false) // 🔧 新增:临时医嘱加载状态
|
||||||
const temporarySigned = ref(false) // 🔧 新增:签名状态,用于保持按钮名称一致性
|
const temporarySigned = ref(false) // 🔧 新增:签名状态,用于保持按钮名称一致性
|
||||||
|
|
||||||
@@ -1499,9 +1490,6 @@ async function closeChargeDialog() {
|
|||||||
chargeSurgeryInfo.value = {}
|
chargeSurgeryInfo.value = {}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 🔧 新增:标志位,用于区分是"打开"还是"刷新"
|
|
||||||
const isRefreshAction = ref(false)
|
|
||||||
|
|
||||||
// 处理医嘱按钮点击事件
|
// 处理医嘱按钮点击事件
|
||||||
function handleMedicalAdvice(row) {
|
function handleMedicalAdvice(row) {
|
||||||
// 如果没有传入行数据,使用选中的行
|
// 如果没有传入行数据,使用选中的行
|
||||||
@@ -1529,31 +1517,7 @@ function handleMedicalAdvice(row) {
|
|||||||
applyId: row.applyId // 手术申请单ID,用于过滤关联医嘱
|
applyId: row.applyId // 手术申请单ID,用于过滤关联医嘱
|
||||||
}
|
}
|
||||||
|
|
||||||
// 🔧 关键修复:如果已有提交的医嘱数据,并且是同一个患者的就诊,则使用保存的数据
|
// 🔧 每次打开临时医嘱都重新拉取最新数据,确保计费弹窗签发后数据自动更新
|
||||||
// 这样可以保留 requestId,避免重复创建医嘱记录
|
|
||||||
console.log('=== 检查是否使用已保存的医嘱数据 ===')
|
|
||||||
console.log('=== temporaryAdvices.value.length ===', temporaryAdvices.value.length)
|
|
||||||
console.log('=== temporaryAdvices.value[0]?.originalMedicine?.encounterId ===', temporaryAdvices.value[0]?.originalMedicine?.encounterId)
|
|
||||||
console.log('=== row.visitId ===', row.visitId)
|
|
||||||
console.log('=== isRefreshAction.value ===', isRefreshAction.value)
|
|
||||||
|
|
||||||
const isSameEncounter = temporaryAdvices.value.length > 0 &&
|
|
||||||
temporaryAdvices.value[0]?.originalMedicine?.encounterId === row.visitId &&
|
|
||||||
!isRefreshAction.value
|
|
||||||
|
|
||||||
console.log('=== isSameEncounter ===', isSameEncounter)
|
|
||||||
|
|
||||||
if (isSameEncounter) {
|
|
||||||
console.log('=== 使用已保存的医嘱数据,避免重复创建 ===')
|
|
||||||
console.log('=== temporaryAdvices.value[0]?.originalMedicine?.requestId ===', temporaryAdvices.value[0]?.originalMedicine?.requestId)
|
|
||||||
// 直接打开弹窗,使用已保存的数据
|
|
||||||
showTemporaryMedical.value = true
|
|
||||||
temporaryMedicalLoading.value = false
|
|
||||||
isRefreshAction.value = false // 重置标志位
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// 🔧 修复:每次打开临时医嘱时都重新加载数据,避免使用缓存数据导致数据重复
|
|
||||||
// 先清空旧数据
|
// 先清空旧数据
|
||||||
temporaryBillingMedicines.value = []
|
temporaryBillingMedicines.value = []
|
||||||
temporaryAdvices.value = []
|
temporaryAdvices.value = []
|
||||||
@@ -1566,46 +1530,39 @@ function handleMedicalAdvice(row) {
|
|||||||
|
|
||||||
// 调用计费接口获取数据
|
// 调用计费接口获取数据
|
||||||
getPrescriptionList(row.visitId, 6, row.operCode).then((res) => {
|
getPrescriptionList(row.visitId, 6, row.operCode).then((res) => {
|
||||||
console.log('=== 拉取计费数据返回结果 ===', res)
|
|
||||||
if (res.code === 200 && res.data) {
|
if (res.code === 200 && res.data) {
|
||||||
// 🔧 修复:显示所有药品请求数据,不管有没有计费项目
|
|
||||||
// 根据用户需求:已引用计费药品(待生成医嘱)和临时医嘱预览(已生成)显示的数据应该相同
|
|
||||||
// 在提交医嘱之前状态应该是"待签发",提交之后变为"已签发"
|
|
||||||
// 再次打开医嘱界面的时候能看到这两个状态的药品
|
|
||||||
const seenIds = new Set();
|
const seenIds = new Set();
|
||||||
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/6)
|
// 只保留药品(1)和耗材(2),屏蔽诊疗(3)和手术(6)
|
||||||
// 🔧 修复 Bug #444: 使用 Number() 显式转换,避免字符串 "1" 被 !== 1 放行
|
|
||||||
const at = Number(item.adviceType ?? item.advice_type);
|
const at = Number(item.adviceType ?? item.advice_type);
|
||||||
if (at !== 1) return false;
|
if (at !== 1 && at !== 2) 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 #444: 二次过滤,排除名称中包含手术/检查/诊疗关键词的非药品项目
|
// 排除名称中包含手术/检查/诊疗关键词的非药品项目
|
||||||
// 某些计费项目可能在 adm_charge_item 中被错误标注为 adviceType=1
|
|
||||||
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
||||||
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
|
if (excludedKeywords.some(kw => medicineName.includes(kw))) 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;
|
||||||
if (itemId) seenIds.add(itemId);
|
if (itemId) seenIds.add(itemId);
|
||||||
return true;
|
return true;
|
||||||
})
|
})
|
||||||
|
// 按 statusEnum 区分:1=草稿(待生成),2=已签发(已生成)
|
||||||
|
const draftItems = filteredItems.filter(item => item.statusEnum === 1)
|
||||||
|
const activeItems = filteredItems.filter(item => item.statusEnum === 2)
|
||||||
|
|
||||||
// 🔧 修复:限制返回数量,最多显示前100条,避免数据过多导致页面卡死
|
// 🔧 修复:限制返回数量,最多显示前100条,避免数据过多导致页面卡死
|
||||||
const maxItems = 100
|
const maxItems = 100
|
||||||
if (filteredItems.length > maxItems) {
|
if (draftItems.length > maxItems) {
|
||||||
ElMessage.warning(`待签发医嘱数量过多(${filteredItems.length}条),仅显示前${maxItems}条`)
|
ElMessage.warning(`待签发医嘱数量过多(${draftItems.length}条),仅显示前${maxItems}条`)
|
||||||
filteredItems.length = maxItems
|
draftItems.length = maxItems
|
||||||
}
|
}
|
||||||
|
|
||||||
// 将过滤后的数据转换为临时医嘱需要的格式 - 兼容驼峰和下划线命名
|
// === 待生成列表:statusEnum=1 草稿状态的项目 ===
|
||||||
// 对于从 adm_charge_item(计费项目表)查询来的项目,特殊处理
|
temporaryBillingMedicines.value = draftItems.map(item => {
|
||||||
temporaryBillingMedicines.value = filteredItems.map(item => {
|
|
||||||
try {
|
try {
|
||||||
// 从 contentJson 或 content_json 中解析详细数据 - 兼容下划线和驼峰命名
|
// 从 contentJson 或 content_json 中解析详细数据 - 兼容下划线和驼峰命名
|
||||||
const jsonContent = item.contentJson || item.content_json;
|
const jsonContent = item.contentJson || item.content_json;
|
||||||
@@ -1652,74 +1609,65 @@ function handleMedicalAdvice(row) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
|
||||||
// 如果没有数据或接口调用失败,初始化空列表
|
|
||||||
temporaryBillingMedicines.value = []
|
|
||||||
}
|
|
||||||
|
|
||||||
// 将计费药品转换为临时医嘱数据
|
|
||||||
temporaryAdvices.value = temporaryBillingMedicines.value.map((medicine, index) => {
|
|
||||||
// 解析规格中的数值和单位
|
|
||||||
const specMatch = medicine.specification ? medicine.specification.match(/(\d+)(\D+)/) : null
|
|
||||||
const specValue = specMatch ? parseInt(specMatch[1]) : 1
|
|
||||||
const specUnit = specMatch ? specMatch[2] : 'ml'
|
|
||||||
|
|
||||||
// 计算剂量 = 规格数值 × 数量
|
// === 已生成列表:statusEnum=2 已签发状态的项目,直接转为医嘱格式 ===
|
||||||
const dosage = specValue * (medicine.quantity || 1)
|
temporaryAdvices.value = activeItems.map((item, index) => {
|
||||||
|
try {
|
||||||
|
const jsonContent = item.contentJson || item.content_json;
|
||||||
|
const contentData = jsonContent ? JSON.parse(jsonContent) : {};
|
||||||
|
const medicineName = contentData.adviceName || contentData.advice_name || item.adviceName || item.advice_name || '';
|
||||||
|
const spec = contentData.volume || contentData.specification || item.volume || item.specification || '';
|
||||||
|
const specMatch = spec.match(/(\d+)(\D+)/)
|
||||||
|
const specValue = specMatch ? parseInt(specMatch[1]) : 1
|
||||||
|
const specUnit = specMatch ? specMatch[2] : 'ml'
|
||||||
|
const dosage = specValue * (contentData.quantity || item.quantity || 1)
|
||||||
|
|
||||||
// 🔧 修复:优先从 contentJson 中读取已有的用法,如果没有则根据药品名称判断
|
let usageCode = contentData.methodCode || 'iv'
|
||||||
let usageCode = 'iv' // 默认静脉注射编码
|
let usageLabel = getUsageLabel(usageCode)
|
||||||
let usageLabel = '静脉注射' // 默认显示名称
|
if (usageCode === 'iv') {
|
||||||
|
if (medicineName.includes('注射液')) { usageCode = 'iv'; usageLabel = '静脉注射' }
|
||||||
|
} else if (usageCode === 'po') {
|
||||||
|
if (medicineName.includes('片') || medicineName.includes('胶囊')) { usageCode = 'po'; usageLabel = '口服' }
|
||||||
|
}
|
||||||
|
|
||||||
// 尝试从 contentJson 中读取用法
|
return {
|
||||||
try {
|
id: index + 1,
|
||||||
const jsonContent = medicine.contentJson || medicine.content_json;
|
adviceName: medicineName,
|
||||||
if (jsonContent) {
|
dosage,
|
||||||
const contentData = JSON.parse(jsonContent);
|
unit: specUnit,
|
||||||
if (contentData.methodCode) {
|
usage: usageCode,
|
||||||
usageCode = contentData.methodCode;
|
usageLabel,
|
||||||
usageLabel = getUsageLabel(contentData.methodCode);
|
frequency: '临时',
|
||||||
|
executeTime: new Date().toLocaleString('zh-CN'),
|
||||||
|
originalMedicine: {
|
||||||
|
...item,
|
||||||
|
medicineName: medicineName,
|
||||||
|
specification: spec,
|
||||||
|
quantity: contentData.quantity || item.quantity || 1,
|
||||||
|
encounterId: row.visitId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return {
|
||||||
|
id: index + 1,
|
||||||
|
adviceName: item.adviceName || item.advice_name || '',
|
||||||
|
dosage: 1, unit: 'ml', usage: 'iv', usageLabel: '静脉注射',
|
||||||
|
frequency: '临时',
|
||||||
|
executeTime: new Date().toLocaleString('zh-CN'),
|
||||||
|
originalMedicine: {
|
||||||
|
...item,
|
||||||
|
medicineName: item.adviceName || item.advice_name || '',
|
||||||
|
specification: item.volume || item.specification || '',
|
||||||
|
quantity: item.quantity || 1,
|
||||||
|
encounterId: row.visitId
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (e) {
|
})
|
||||||
// 解析失败,继续使用默认值
|
} else {
|
||||||
}
|
temporaryBillingMedicines.value = []
|
||||||
|
temporaryAdvices.value = []
|
||||||
// 如果没有从 contentJson 中读取到用法,根据药品名称判断
|
}
|
||||||
if (!usageCode || usageCode === 'iv') {
|
|
||||||
if (medicine.medicineName && medicine.medicineName.includes('注射液')) {
|
|
||||||
usageCode = 'iv'
|
|
||||||
usageLabel = '静脉注射'
|
|
||||||
} else if (medicine.medicineName && medicine.medicineName.includes('片')) {
|
|
||||||
usageCode = 'po'
|
|
||||||
usageLabel = '口服'
|
|
||||||
} else if (medicine.medicineName && medicine.medicineName.includes('胶囊')) {
|
|
||||||
usageCode = 'po'
|
|
||||||
usageLabel = '口服'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
id: index + 1,
|
|
||||||
adviceName: medicine.medicineName || '',
|
|
||||||
dosage: dosage,
|
|
||||||
unit: specUnit,
|
|
||||||
usage: usageCode, // 🔧 修复:保存的是编码
|
|
||||||
usageLabel: usageLabel, // 🔧 新增:保存显示名称
|
|
||||||
frequency: '临时',
|
|
||||||
executeTime: new Date().toLocaleString('zh-CN'),
|
|
||||||
// 🔧 关键修复:确保 originalMedicine 中包含 encounterId 和匹配字段
|
|
||||||
// medicineName/specification/quantity 用于 handleTemporaryMedicalSubmit 中的
|
|
||||||
// 已提交项目匹配过滤(Bug #445),缺少这些字段会导致过滤失效
|
|
||||||
originalMedicine: {
|
|
||||||
...medicine,
|
|
||||||
medicineName: medicine.medicineName,
|
|
||||||
specification: medicine.specification,
|
|
||||||
quantity: medicine.quantity,
|
|
||||||
encounterId: row.visitId // 添加 encounterId 字段
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// 打开临时医嘱弹窗
|
// 打开临时医嘱弹窗
|
||||||
showTemporaryMedical.value = true
|
showTemporaryMedical.value = true
|
||||||
@@ -1745,11 +1693,6 @@ function closeTemporaryMedical() {
|
|||||||
// 处理临时医嘱提交
|
// 处理临时医嘱提交
|
||||||
// 🔧 修复:提交成功后,更新 temporaryAdvices 中的 requestId,以便下次提交时执行更新操作
|
// 🔧 修复:提交成功后,更新 temporaryAdvices 中的 requestId,以便下次提交时执行更新操作
|
||||||
function handleTemporaryMedicalSubmit(data) {
|
function handleTemporaryMedicalSubmit(data) {
|
||||||
console.log('=== handleTemporaryMedicalSubmit 被调用 ===')
|
|
||||||
console.log('=== data ===', data)
|
|
||||||
console.log('=== data.temporaryAdvices ===', data.temporaryAdvices)
|
|
||||||
console.log('=== data.temporaryAdvices[1]?.dosage ===', data.temporaryAdvices[1]?.dosage)
|
|
||||||
|
|
||||||
// 🔧 修复:使用用户修改后的数据,而不是重新加载数据
|
// 🔧 修复:使用用户修改后的数据,而不是重新加载数据
|
||||||
// 这样可以确保用户修改的内容(如剂量)在保存后仍然正确显示
|
// 这样可以确保用户修改的内容(如剂量)在保存后仍然正确显示
|
||||||
if (data.temporaryAdvices && data.temporaryAdvices.length > 0) {
|
if (data.temporaryAdvices && data.temporaryAdvices.length > 0) {
|
||||||
@@ -1804,9 +1747,7 @@ function handleTemporaryMedicalSubmit(data) {
|
|||||||
// 如果没有任何匹配键,清空待生成列表(所有项目都已提交)
|
// 如果没有任何匹配键,清空待生成列表(所有项目都已提交)
|
||||||
temporaryBillingMedicines.value = []
|
temporaryBillingMedicines.value = []
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log('=== 使用用户修改后的临时医嘱数据 ===', temporaryAdvices.value)
|
|
||||||
console.log('=== temporaryAdvices.value[1]?.dosage ===', temporaryAdvices.value[1]?.dosage)
|
|
||||||
} else {
|
} else {
|
||||||
// 如果没有传递数据,则清空
|
// 如果没有传递数据,则清空
|
||||||
temporaryAdvices.value = []
|
temporaryAdvices.value = []
|
||||||
@@ -1878,21 +1819,70 @@ function handleQuoteBilling() {
|
|||||||
|
|
||||||
// 🔧 修复 Bug #445: 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3/6)
|
// 🔧 修复 Bug #445: 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3/6)
|
||||||
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
|
// 同时过滤掉已有 requestId 的项目(已生成医嘱的不需要再次显示在"待生成"列表中)
|
||||||
const filteredItems = res.data.filter(item => {
|
// 先提取已签发项目(statusEnum=2)填充已生成列表
|
||||||
// 匹配 encounterId
|
const activeItems = res.data.filter(item => {
|
||||||
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
||||||
// 只保留药品类型(adviceType=1),过滤掉耗材(2)和诊疗项目(3/6)
|
|
||||||
// 🔧 修复 Bug #444: 使用 Number() 显式转换 + snake_case 回退,避免字符串 "1" 匹配失败
|
|
||||||
const at = Number(item.adviceType ?? item.advice_type);
|
const at = Number(item.adviceType ?? item.advice_type);
|
||||||
if (at !== 1) return false;
|
if (at !== 1 && at !== 2) return false;
|
||||||
// 过滤掉名称为空的项目
|
if (item.statusEnum !== 2) return false;
|
||||||
|
const medicineName = item.adviceName || item.advice_name;
|
||||||
|
if (!medicineName || medicineName.trim() === '') return false;
|
||||||
|
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
||||||
|
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
|
||||||
|
return true;
|
||||||
|
})
|
||||||
|
temporaryAdvices.value = activeItems.map((item, index) => {
|
||||||
|
try {
|
||||||
|
const jsonContent = item.contentJson || item.content_json;
|
||||||
|
const contentData = jsonContent ? JSON.parse(jsonContent) : {};
|
||||||
|
const medicineName = contentData.adviceName || contentData.advice_name || item.adviceName || item.advice_name || '';
|
||||||
|
const spec = contentData.volume || contentData.specification || item.volume || item.specification || '';
|
||||||
|
const specMatch = spec.match(/(\d+)(\D+)/)
|
||||||
|
const specValue = specMatch ? parseInt(specMatch[1]) : 1
|
||||||
|
const specUnit = specMatch ? specMatch[2] : 'ml'
|
||||||
|
const dosage = specValue * (contentData.quantity || item.quantity || 1)
|
||||||
|
let usageCode = contentData.methodCode || 'iv'
|
||||||
|
let usageLabel = getUsageLabel(usageCode)
|
||||||
|
if (usageCode === 'iv' && medicineName.includes('注射液')) { usageLabel = '静脉注射' }
|
||||||
|
else if (usageCode === 'po' && (medicineName.includes('片') || medicineName.includes('胶囊'))) { usageLabel = '口服' }
|
||||||
|
return {
|
||||||
|
id: index + 1, adviceName: medicineName, dosage, unit: specUnit,
|
||||||
|
usage: usageCode, usageLabel, frequency: '临时',
|
||||||
|
executeTime: new Date().toLocaleString('zh-CN'),
|
||||||
|
originalMedicine: {
|
||||||
|
...item,
|
||||||
|
medicineName: medicineName,
|
||||||
|
specification: spec,
|
||||||
|
quantity: contentData.quantity || item.quantity || 1,
|
||||||
|
encounterId: temporaryPatientInfo.value.visitId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return {
|
||||||
|
id: index + 1, adviceName: item.adviceName || item.advice_name || '',
|
||||||
|
dosage: 1, unit: 'ml', usage: 'iv', usageLabel: '静脉注射',
|
||||||
|
frequency: '临时', executeTime: new Date().toLocaleString('zh-CN'),
|
||||||
|
originalMedicine: {
|
||||||
|
...item,
|
||||||
|
medicineName: item.adviceName || item.advice_name || '',
|
||||||
|
specification: item.volume || item.specification || '',
|
||||||
|
quantity: item.quantity || 1,
|
||||||
|
encounterId: temporaryPatientInfo.value.visitId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// 再提取草稿项目(statusEnum=1)填充待生成列表
|
||||||
|
const filteredItems = res.data.filter(item => {
|
||||||
|
if (item.encounterId !== temporaryPatientInfo.value.visitId) return false;
|
||||||
|
const at = Number(item.adviceType ?? item.advice_type);
|
||||||
|
if (at !== 1 && at !== 2) return false;
|
||||||
|
if (item.statusEnum !== 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 #444: 二次过滤,排除名称中包含手术/检查/诊疗关键词的非药品项目
|
|
||||||
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
const excludedKeywords = ['术', '超声', '多普勒', '检查', '检验', '彩超', 'X线', 'CT', 'MRI', '扫描', '造影'];
|
||||||
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
|
if (excludedKeywords.some(kw => medicineName.includes(kw))) return false;
|
||||||
// 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目(已有 requestId)
|
|
||||||
if (item.requestId) return false;
|
|
||||||
return true;
|
return true;
|
||||||
})
|
})
|
||||||
// 🔧 修复:限制返回数量,最多显示前100条,避免数据过多导致页面卡死
|
// 🔧 修复:限制返回数量,最多显示前100条,避免数据过多导致页面卡死
|
||||||
@@ -1905,7 +1895,6 @@ function handleQuoteBilling() {
|
|||||||
// 将过滤后的数据转换为临时医嘱需要的格式
|
// 将过滤后的数据转换为临时医嘱需要的格式
|
||||||
temporaryBillingMedicines.value = filteredItems.map(item => {
|
temporaryBillingMedicines.value = filteredItems.map(item => {
|
||||||
try {
|
try {
|
||||||
// 从 contentJson 或 content_json 中解析详细数据 - 兼容下划线和驼峰命名
|
|
||||||
const jsonContent = item.contentJson || item.content_json;
|
const jsonContent = item.contentJson || item.content_json;
|
||||||
const contentData = jsonContent ? JSON.parse(jsonContent) : {};
|
const contentData = jsonContent ? JSON.parse(jsonContent) : {};
|
||||||
return {
|
return {
|
||||||
@@ -1924,10 +1913,9 @@ function handleQuoteBilling() {
|
|||||||
definitionDetailId: contentData.definitionDetailId || item.definitionDetailId
|
definitionDetailId: contentData.definitionDetailId || item.definitionDetailId
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// 如果解析失败,使用顶层数据 - 兼容 snake_case 和 camelCase
|
|
||||||
return {
|
return {
|
||||||
medicineName: item.adviceName || item.advice_name || '',
|
medicineName: item.adviceName || item.advice_name || '',
|
||||||
specification: item.specification || item.specification || item.volume || '',
|
specification: item.specification || item.volume || '',
|
||||||
quantity: item.quantity || item.quantity_value || 0,
|
quantity: item.quantity || item.quantity_value || 0,
|
||||||
batchNumber: item.lotNumber || item.lot_number || '',
|
batchNumber: item.lotNumber || item.lot_number || '',
|
||||||
unitPrice: item.unitPrice || item.unit_price || 0,
|
unitPrice: item.unitPrice || item.unit_price || 0,
|
||||||
@@ -1943,123 +1931,6 @@ function handleQuoteBilling() {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
// 将计费药品转换为临时医嘱数据
|
|
||||||
temporaryAdvices.value = temporaryBillingMedicines.value.map((medicine, index) => {
|
|
||||||
// 解析规格中的数值和单位
|
|
||||||
const specMatch = medicine.specification ? medicine.specification.match(/(\d+)(\D+)/) : null
|
|
||||||
const specValue = specMatch ? parseInt(specMatch[1]) : 1
|
|
||||||
const specUnit = specMatch ? specMatch[2] : 'ml'
|
|
||||||
|
|
||||||
// 计算剂量 = 规格数值 × 数量
|
|
||||||
const dosage = specValue * (medicine.quantity || 1)
|
|
||||||
|
|
||||||
// 🔧 修复:优先从 contentJson 中读取已有的用法,如果没有则根据药品名称判断
|
|
||||||
let usageCode = 'iv' // 默认静脉注射编码
|
|
||||||
let usageLabel = '静脉注射' // 默认显示名称
|
|
||||||
|
|
||||||
// 尝试从 contentJson 中读取用法
|
|
||||||
try {
|
|
||||||
const jsonContent = medicine.contentJson || medicine.content_json;
|
|
||||||
if (jsonContent) {
|
|
||||||
const contentData = JSON.parse(jsonContent);
|
|
||||||
if (contentData.methodCode) {
|
|
||||||
usageCode = contentData.methodCode;
|
|
||||||
usageLabel = getUsageLabel(contentData.methodCode);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
// 解析失败,继续使用默认值
|
|
||||||
}
|
|
||||||
|
|
||||||
// 如果没有从 contentJson 中读取到用法,根据药品名称判断
|
|
||||||
if (!usageCode || usageCode === 'iv') {
|
|
||||||
if (medicine.medicineName && medicine.medicineName.includes('注射液')) {
|
|
||||||
usageCode = 'iv'
|
|
||||||
usageLabel = '静脉注射'
|
|
||||||
} else if (medicine.medicineName && medicine.medicineName.includes('片')) {
|
|
||||||
usageCode = 'po'
|
|
||||||
usageLabel = '口服'
|
|
||||||
} else if (medicine.medicineName && medicine.medicineName.includes('胶囊')) {
|
|
||||||
usageCode = 'po'
|
|
||||||
usageLabel = '口服'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
id: index + 1,
|
|
||||||
adviceName: medicine.medicineName || '',
|
|
||||||
dosage: dosage,
|
|
||||||
unit: specUnit,
|
|
||||||
usage: usageCode, // 🔧 修复:保存的是编码
|
|
||||||
usageLabel: usageLabel, // 🔧 新增:保存显示名称
|
|
||||||
frequency: '临时',
|
|
||||||
executeTime: new Date().toLocaleString('zh-CN'),
|
|
||||||
// 🔧 关键修复:确保 originalMedicine 中包含 encounterId 和匹配字段
|
|
||||||
// medicineName/specification/quantity 用于已提交项目匹配过滤(Bug #445)
|
|
||||||
originalMedicine: {
|
|
||||||
...medicine,
|
|
||||||
medicineName: medicine.medicineName,
|
|
||||||
specification: medicine.specification,
|
|
||||||
quantity: medicine.quantity,
|
|
||||||
encounterId: temporaryPatientInfo.value.visitId // 添加 encounterId 字段
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
// 🔧 修复 Bug #445: 过滤掉已生成医嘱的项目,避免"引用计费"后已提交项目重新出现在"待生成"列表
|
|
||||||
// 使用清空前提取的 submittedKeys(名称|||规格|||数量复合键)进行匹配
|
|
||||||
if (submittedKeys.size > 0) {
|
|
||||||
temporaryBillingMedicines.value = temporaryBillingMedicines.value.filter(m => {
|
|
||||||
const key = `${m.medicineName || ''}|||${m.specification || ''}|||${m.quantity ?? 0}`
|
|
||||||
return !submittedKeys.has(key)
|
|
||||||
})
|
|
||||||
// 同步更新 temporaryAdvices,保持两份数据一致
|
|
||||||
temporaryAdvices.value = temporaryBillingMedicines.value.map((medicine, index) => {
|
|
||||||
const specMatch = medicine.specification ? medicine.specification.match(/(\d+)(\D+)/) : null
|
|
||||||
const specValue = specMatch ? parseInt(specMatch[1]) : 1
|
|
||||||
const specUnit = specMatch ? specMatch[2] : 'ml'
|
|
||||||
const dosage = specValue * (medicine.quantity || 1)
|
|
||||||
let usageCode = 'iv'
|
|
||||||
let usageLabel = '静脉注射'
|
|
||||||
try {
|
|
||||||
const jsonContent = medicine.contentJson || medicine.content_json;
|
|
||||||
if (jsonContent) {
|
|
||||||
const contentData = JSON.parse(jsonContent);
|
|
||||||
if (contentData.methodCode) {
|
|
||||||
usageCode = contentData.methodCode;
|
|
||||||
usageLabel = getUsageLabel(contentData.methodCode);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {}
|
|
||||||
if (!usageCode || usageCode === 'iv') {
|
|
||||||
if (medicine.medicineName && medicine.medicineName.includes('注射液')) {
|
|
||||||
usageCode = 'iv'; usageLabel = '静脉注射';
|
|
||||||
} else if (medicine.medicineName && medicine.medicineName.includes('片')) {
|
|
||||||
usageCode = 'po'; usageLabel = '口服';
|
|
||||||
} else if (medicine.medicineName && medicine.medicineName.includes('胶囊')) {
|
|
||||||
usageCode = 'po'; usageLabel = '口服';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
id: index + 1,
|
|
||||||
adviceName: medicine.medicineName || '',
|
|
||||||
dosage,
|
|
||||||
unit: specUnit,
|
|
||||||
usage: usageCode,
|
|
||||||
usageLabel,
|
|
||||||
frequency: '临时',
|
|
||||||
executeTime: new Date().toLocaleString('zh-CN'),
|
|
||||||
originalMedicine: {
|
|
||||||
...medicine,
|
|
||||||
medicineName: medicine.medicineName,
|
|
||||||
specification: medicine.specification,
|
|
||||||
quantity: medicine.quantity,
|
|
||||||
encounterId: temporaryPatientInfo.value.visitId
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
temporaryMedicalLoading.value = false // 🔧 新增:加载完成
|
temporaryMedicalLoading.value = false // 🔧 新增:加载完成
|
||||||
ElMessage.success('已成功引用最新计费药品信息!')
|
ElMessage.success('已成功引用最新计费药品信息!')
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -48,9 +48,12 @@
|
|||||||
<div class="medicine-section">
|
<div class="medicine-section">
|
||||||
<div class="section-title">
|
<div class="section-title">
|
||||||
一、已引用计费药品(待生成医嘱)
|
一、已引用计费药品(待生成医嘱)
|
||||||
|
<span v-if="(billingMedicines || []).length >= PAGE_SIZE" style="margin-left:auto;font-size:14px;color:#4a8bc9;cursor:pointer;white-space:nowrap;" @click="billingExpanded = !billingExpanded">
|
||||||
|
{{ billingExpanded ? '收起' : `展开全部(${(billingMedicines || []).length}条)` }}
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<el-table
|
<el-table
|
||||||
:data="billingMedicines"
|
:data="displayBillingMedicines"
|
||||||
stripe
|
stripe
|
||||||
border
|
border
|
||||||
style="width: 100%;"
|
style="width: 100%;"
|
||||||
@@ -98,9 +101,12 @@
|
|||||||
<div class="advice-section">
|
<div class="advice-section">
|
||||||
<div class="section-title">
|
<div class="section-title">
|
||||||
二、临时医嘱预览(已生成)
|
二、临时医嘱预览(已生成)
|
||||||
|
<span v-if="(displayAdvices || []).length >= PAGE_SIZE" style="margin-left:auto;font-size:14px;color:#4a8bc9;cursor:pointer;white-space:nowrap;" @click="advicesExpanded = !advicesExpanded">
|
||||||
|
{{ advicesExpanded ? '收起' : `展开全部(${(displayAdvices || []).length}条)` }}
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<el-table
|
<el-table
|
||||||
:data="displayAdvices"
|
:data="displayAdvicesList"
|
||||||
stripe
|
stripe
|
||||||
border
|
border
|
||||||
style="width: 100%;"
|
style="width: 100%;"
|
||||||
@@ -317,6 +323,19 @@ const allItemsSubmitted = computed(() => {
|
|||||||
return meds.length > 0 && meds.every(m => m.requestId)
|
return meds.length > 0 && meds.every(m => m.requestId)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// 展开/收起控制
|
||||||
|
const PAGE_SIZE = 3
|
||||||
|
const billingExpanded = ref(false)
|
||||||
|
const advicesExpanded = ref(false)
|
||||||
|
const displayBillingMedicines = computed(() => {
|
||||||
|
const all = props.billingMedicines || []
|
||||||
|
return billingExpanded.value ? all : all.slice(0, PAGE_SIZE)
|
||||||
|
})
|
||||||
|
const displayAdvicesList = computed(() => {
|
||||||
|
const all = displayAdvices.value || []
|
||||||
|
return advicesExpanded.value ? all : all.slice(0, PAGE_SIZE)
|
||||||
|
})
|
||||||
|
|
||||||
// 响应式数据 - isSigned 从父组件传入的 prop 初始化
|
// 响应式数据 - isSigned 从父组件传入的 prop 初始化
|
||||||
const isSigned = ref(props.isSignedProp)
|
const isSigned = ref(props.isSignedProp)
|
||||||
|
|
||||||
@@ -1045,6 +1064,21 @@ const editFormUsageLabel = computed(() => {
|
|||||||
padding-bottom: 12px;
|
padding-bottom: 12px;
|
||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
border-bottom: 2px solid #e4e7ed;
|
border-bottom: 2px solid #e4e7ed;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.expand-btn {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
color: #4a8bc9;
|
||||||
|
cursor: pointer;
|
||||||
|
font-weight: 400;
|
||||||
|
margin-left: auto;
|
||||||
|
}
|
||||||
|
.expand-btn:hover {
|
||||||
|
color: #2a6ba9;
|
||||||
|
text-decoration: underline;
|
||||||
}
|
}
|
||||||
|
|
||||||
.medicine-summary {
|
.medicine-summary {
|
||||||
|
|||||||
Reference in New Issue
Block a user