106 lines
3.1 KiB
Vue
106 lines
3.1 KiB
Vue
|
<!-- components/UrlSelect.vue -->
|
|||
|
<template>
|
|||
|
<el-select
|
|||
|
v-bind="$attrs"
|
|||
|
v-model="internalValue"
|
|||
|
@change="$emit('change', $event)"
|
|||
|
@blur="$emit('blur', $event)"
|
|||
|
@focus="$emit('focus', $event)"
|
|||
|
@clear="$emit('clear', $event)"
|
|||
|
@visible-change="$emit('visible-change', $event)"
|
|||
|
@remove-tag="$emit('remove-tag', $event)"
|
|||
|
@scroll="$emit('scroll', $event)"
|
|||
|
>
|
|||
|
<!-- 1. 原生 prefix 插槽 -->
|
|||
|
<slot name="prefix" />
|
|||
|
<!-- 2. 如果要实现分组插槽,也照搬 el-option-group 的结构 -->
|
|||
|
<slot name="option-group">
|
|||
|
<!-- 默认选项渲染 -->
|
|||
|
<el-option v-for="item in options" :key="item[valueKey]" :label="item[labelKey]" :value="item[valueKey]" />
|
|||
|
</slot>
|
|||
|
<!-- 3. 其他自定义插槽,不指定 name 的默认为 default slot -->
|
|||
|
<slot />
|
|||
|
<!-- 4. 下面这几个插槽是 el-select 也常见的: -->
|
|||
|
<slot name="empty" />
|
|||
|
<slot name="no-match" />
|
|||
|
<slot name="footer" />
|
|||
|
</el-select>
|
|||
|
</template>
|
|||
|
|
|||
|
<script setup>
|
|||
|
import { ref, watch, onMounted } from 'vue';
|
|||
|
import request from '@/utils/axios';
|
|||
|
|
|||
|
// -------------- ① 定义要继承/自定义的 Props --------------
|
|||
|
const props = defineProps({
|
|||
|
modelValue: {
|
|||
|
type: [String, Number, Array],
|
|||
|
default: null,
|
|||
|
},
|
|||
|
url: {
|
|||
|
type: String,
|
|||
|
default: null,
|
|||
|
},
|
|||
|
params: {
|
|||
|
type: Object,
|
|||
|
default: () => ({}),
|
|||
|
},
|
|||
|
// 可按需让上层指定 label 字段名、value 字段名
|
|||
|
labelKey: {
|
|||
|
type: String,
|
|||
|
default: 'label',
|
|||
|
},
|
|||
|
valueKey: {
|
|||
|
type: String,
|
|||
|
default: 'value',
|
|||
|
},
|
|||
|
});
|
|||
|
|
|||
|
// -------------- ② 定义要透传回父组件的 Events --------------
|
|||
|
const emit = defineEmits(['update:modelValue', 'change', 'blur', 'focus', 'clear', 'visible-change', 'remove-tag', 'scroll']);
|
|||
|
|
|||
|
// -------------- ③ 内部双向绑定 --------------
|
|||
|
// 用一个 ref 存储内部的值
|
|||
|
const internalValue = ref(props.modelValue);
|
|||
|
// 当 prop 变化时,同步到 internalValue
|
|||
|
watch(
|
|||
|
() => props.modelValue,
|
|||
|
(newVal) => {
|
|||
|
internalValue.value = newVal;
|
|||
|
}
|
|||
|
);
|
|||
|
// 当 internalValue 变化时,emit update:modelValue
|
|||
|
watch(internalValue, (newVal) => {
|
|||
|
emit('update:modelValue', newVal);
|
|||
|
});
|
|||
|
|
|||
|
// -------------- ④ 选项列表 --------------
|
|||
|
const options = ref([]);
|
|||
|
|
|||
|
// -------------- ⑤ 请求数据的函数 --------------
|
|||
|
async function fetchOptions() {
|
|||
|
console.log('fetchOptions :>> ', props.url, props.params);
|
|||
|
if (!props.url) return;
|
|||
|
try {
|
|||
|
const res = await request.get(props.url, { params: props.params });
|
|||
|
// 假设后端返回格式: { code: 200, data: { records: [ {...}, {...} ] } }
|
|||
|
// 也可能是直接 data: [{...}, {...}]
|
|||
|
const records = res.data.records;
|
|||
|
if (Array.isArray(records)) {
|
|||
|
options.value = records;
|
|||
|
// console.log('option', options.value);
|
|||
|
} else {
|
|||
|
options.value = [];
|
|||
|
console.log('UrlSelect:接口返回数据格式不是数组,无法解析成 options');
|
|||
|
}
|
|||
|
} catch (err) {
|
|||
|
console.error('UrlSelect:拉取选项失败', err);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// -------------- ⑥ 生命周期:组件挂载后立即拉一次数据 --------------
|
|||
|
onMounted(() => {
|
|||
|
fetchOptions();
|
|||
|
});
|
|||
|
</script>
|