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