feat(mes): 优化物料产品选择器 V2 组件(完善 tooltip 展示)

pull/871/MERGE
YunaiV 2026-04-05 12:50:00 +08:00
parent 07cd4c47ed
commit 35e003de64
1 changed files with 24 additions and 44 deletions

View File

@ -4,7 +4,6 @@
交互显示为只读 el-input点击打开弹窗单选模式进行选择
Props:
modelValue 绑定的物料 IDv-model
itemName 备用显示名称父组件已有名称时传入可跳过 getItem 请求避免 N+1
disabled 是否禁用
clearable 是否允许清空鼠标悬停时显示清除图标
placeholder 占位文字
@ -14,25 +13,33 @@
-->
<template>
<div
v-bind="attrs"
class="w-full"
:class="disabled ? 'cursor-not-allowed' : 'cursor-pointer'"
@click="handleClick"
@mouseenter="hovering = true"
@mouseleave="hovering = false"
>
<el-input
ref="inputRef"
:model-value="displayLabel"
:placeholder="placeholder"
:disabled="disabled"
readonly
:suffix-icon="suffixIcon"
:class="disabled ? 'is-select-disabled' : 'is-select-clickable'"
/>
<el-tooltip :disabled="!selectedItem" placement="top" :show-after="500">
<template #content>
<div v-if="selectedItem" class="leading-6">
<div>编码{{ selectedItem.code }}</div>
<div>名称{{ selectedItem.name }}</div>
<div>规格{{ selectedItem.specification || '-' }}</div>
<div>单位{{ selectedItem.unitMeasureName || '-' }}</div>
</div>
</template>
<el-input
:model-value="displayLabel"
:placeholder="placeholder"
:disabled="disabled"
readonly
:suffix-icon="suffixIcon"
:class="disabled ? 'is-select-disabled' : 'is-select-clickable'"
/>
</el-tooltip>
<!-- 弹窗选择器单选模式放在 div 内部保持单根节点Vue 自动继承 attrs -->
<MdItemSelectDialogV2 ref="dialogRef" :multiple="false" @selected="handleSelected" />
</div>
<!-- 弹窗选择器单选模式 -->
<MdItemSelectDialogV2 ref="dialogRef" :multiple="false" @selected="handleSelected" />
</template>
<script setup lang="ts">
@ -40,14 +47,11 @@ import { MdItemApi, MdItemVO } from '@/api/mes/md/item'
import { Search, CircleClose } from '@element-plus/icons-vue'
import MdItemSelectDialogV2 from './MdItemSelectDialogV2.vue'
const attrs = useAttrs() // TODO @AIattrs
defineOptions({ name: 'MdItemSelectV2', inheritAttrs: false })
defineOptions({ name: 'MdItemSelectV2' })
const props = withDefaults(
defineProps<{
modelValue?: number // ID
itemName?: string // getItem
disabled?: boolean //
clearable?: boolean //
placeholder?: string //
@ -65,24 +69,14 @@ const emit = defineEmits<{
}>()
const dialogRef = ref() // Ref
const inputRef = ref() // Ref
const hovering = ref(false) //
// ==================== ====================
const selectedItem = ref<MdItemVO | undefined>() //
const resolving = ref(false) //
let resolveSeq = 0 //
/** 显示文本:[物料编码] 物料名称 */
/** 输入框显示文本:只展示物料名称,保持简洁 */
const displayLabel = computed(() => {
if (selectedItem.value) {
return `[${selectedItem.value.code}] ${selectedItem.value.name}`
}
// itemName selectedItem
if (props.itemName && props.modelValue != null) {
return props.itemName
}
return ''
return selectedItem.value?.name ?? ''
})
/** 是否显示清除图标 */
@ -104,24 +98,10 @@ const resolveItemById = async (id: number | undefined) => {
if (selectedItem.value?.id === id) {
return
}
// itemName N+1
if (props.itemName) {
return
}
const seq = ++resolveSeq
resolving.value = true
try {
const item = await MdItemApi.getItem(id)
//
if (seq !== resolveSeq) return
selectedItem.value = item
selectedItem.value = await MdItemApi.getItem(id)
} catch (e) {
if (seq !== resolveSeq) return
console.error('[MdItemSelectV2] resolveItemById failed:', e)
} finally {
if (seq === resolveSeq) {
resolving.value = false
}
}
}