142 lines
3.7 KiB
Vue
142 lines
3.7 KiB
Vue
<script lang="ts" setup>
|
||
import type { MesTmToolApi } from '#/api/mes/tm/tool';
|
||
|
||
import { computed, ref, useAttrs, watch } from 'vue';
|
||
|
||
import { IconifyIcon } from '@vben/icons';
|
||
|
||
import { Input, Tooltip } from 'ant-design-vue';
|
||
|
||
import { getTool } from '#/api/mes/tm/tool';
|
||
|
||
import TmToolSelectDialog from './select-dialog.vue';
|
||
|
||
defineOptions({ name: 'TmToolSelect', inheritAttrs: false });
|
||
|
||
const props = withDefaults(
|
||
defineProps<{
|
||
allowClear?: boolean;
|
||
disabled?: boolean;
|
||
modelValue?: number;
|
||
placeholder?: string;
|
||
}>(),
|
||
{
|
||
allowClear: true,
|
||
disabled: false,
|
||
modelValue: undefined,
|
||
placeholder: '请选择工具',
|
||
},
|
||
);
|
||
const emit = defineEmits<{
|
||
change: [item: MesTmToolApi.Tool | undefined];
|
||
'update:modelValue': [value: number | undefined];
|
||
}>();
|
||
const attrs = useAttrs(); // 透传属性
|
||
const dialogRef = ref<InstanceType<typeof TmToolSelectDialog>>(); // 工具选择弹窗
|
||
const hovering = ref(false); // 是否悬停
|
||
const selectedItem = ref<MesTmToolApi.Tool>(); // 当前选中工具
|
||
|
||
const displayLabel = computed(() => selectedItem.value?.name ?? ''); // 选择器展示名称
|
||
const showClear = computed(
|
||
() =>
|
||
props.allowClear &&
|
||
!props.disabled &&
|
||
hovering.value &&
|
||
props.modelValue != null,
|
||
);
|
||
|
||
/** 根据工具编号回显选择器 */
|
||
async function resolveItemById(id: number | undefined) {
|
||
if (id == null) {
|
||
selectedItem.value = undefined;
|
||
return;
|
||
}
|
||
if (selectedItem.value?.id === id) {
|
||
return;
|
||
}
|
||
selectedItem.value = await getTool(id);
|
||
}
|
||
|
||
watch(
|
||
() => props.modelValue,
|
||
(value) => {
|
||
resolveItemById(value);
|
||
},
|
||
{ immediate: true },
|
||
);
|
||
|
||
/** 清空已选工具 */
|
||
function clearSelected() {
|
||
selectedItem.value = undefined;
|
||
emit('update:modelValue', undefined);
|
||
emit('change', undefined);
|
||
}
|
||
|
||
/** 打开工具选择弹窗 */
|
||
function handleClick(event: MouseEvent) {
|
||
if (props.disabled) {
|
||
return;
|
||
}
|
||
const target = event.target as HTMLElement;
|
||
if (showClear.value && target.closest('.ant-input-suffix')) {
|
||
event.stopPropagation();
|
||
clearSelected();
|
||
return;
|
||
}
|
||
const selectedIds = props.modelValue == null ? [] : [props.modelValue];
|
||
dialogRef.value?.open(selectedIds, { multiple: false });
|
||
}
|
||
|
||
/** 回填选中的工具 */
|
||
function handleSelected(rows: MesTmToolApi.Tool[]) {
|
||
const item = rows[0];
|
||
if (!item) {
|
||
return;
|
||
}
|
||
selectedItem.value = item;
|
||
emit('update:modelValue', item.id);
|
||
emit('change', item);
|
||
}
|
||
</script>
|
||
|
||
<template>
|
||
<div
|
||
v-bind="attrs"
|
||
class="w-full"
|
||
:class="disabled ? 'cursor-not-allowed' : 'cursor-pointer'"
|
||
@click="handleClick"
|
||
@mouseenter="hovering = true"
|
||
@mouseleave="hovering = false"
|
||
>
|
||
<Tooltip :mouse-enter-delay="0.5" :open="selectedItem ? undefined : false">
|
||
<template #title>
|
||
<div v-if="selectedItem" class="leading-6">
|
||
<div>工具编码:{{ selectedItem.code || '-' }}</div>
|
||
<div>工具名称:{{ selectedItem.name || '-' }}</div>
|
||
<div v-if="selectedItem.brand">品牌:{{ selectedItem.brand }}</div>
|
||
<div v-if="selectedItem.specification">
|
||
型号规格:{{ selectedItem.specification }}
|
||
</div>
|
||
<div v-if="selectedItem.toolTypeName">
|
||
工具类型:{{ selectedItem.toolTypeName }}
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<Input
|
||
:disabled="disabled"
|
||
:placeholder="placeholder"
|
||
:value="displayLabel"
|
||
readonly
|
||
>
|
||
<template #suffix>
|
||
<IconifyIcon
|
||
class="size-4"
|
||
:icon="showClear ? 'lucide:circle-x' : 'lucide:search'"
|
||
/>
|
||
</template>
|
||
</Input>
|
||
</Tooltip>
|
||
</div>
|
||
<TmToolSelectDialog ref="dialogRef" @selected="handleSelected" />
|
||
</template>
|