2023-12-15 16:20:50 +08:00

401 lines
11 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<TMagicSelect
v-if="model"
v-model="model[name]"
v-loading="loading"
class="m-select"
ref="tMagicSelect"
:clearable="typeof config.clearable !== 'undefined' ? config.clearable : true"
:filterable="typeof config.filterable !== 'undefined' ? config.filterable : true"
:popper-class="`m-select-popper ${popperClass}`"
:size="size"
:remote="remote"
:placeholder="config.placeholder"
:multiple="config.multiple"
:value-key="config.valueKey || 'value'"
:allow-create="config.allowCreate"
:disabled="disabled"
:remote-method="config.remote && remoteMethod"
@change="changeHandler"
@visible-change="visibleHandler"
>
<template v-if="config.group"><SelectOptionGroups :options="groupOptions"></SelectOptionGroups></template>
<template v-else><SelectOptions :options="itemOptions"></SelectOptions></template>
<div v-loading="true" v-if="moreLoadingVisible"></div>
</TMagicSelect>
</template>
<script lang="ts" setup>
import { inject, nextTick, onBeforeMount, Ref, ref, watch, watchEffect } from 'vue';
import { TMagicSelect } from '@tmagic/design';
import { getValueByKeyPath } from '@tmagic/utils';
import type { FieldProps, FormState, SelectConfig, SelectGroupOption, SelectOption } from '../schema';
import { getConfig } from '../utils/config';
import { useAddField } from '../utils/useAddField';
import SelectOptionGroups from './SelectOptionGroups.vue';
import SelectOptions from './SelectOptions.vue';
defineOptions({
name: 'MFormSelect',
});
const props = defineProps<FieldProps<SelectConfig>>();
const emit = defineEmits(['change']);
if (!props.model) throw new Error('不能没有model');
useAddField(props.prop);
const tMagicSelect = ref<InstanceType<typeof TMagicSelect>>();
const mForm = inject<FormState | undefined>('mForm');
const options = ref<SelectOption[] | SelectGroupOption[]>([]);
const localOptions = ref<SelectOption[] | SelectGroupOption[]>([]);
const loading = ref(false);
const moreLoadingVisible = ref(false);
const total = ref(0);
const pgIndex = ref(0);
const pgSize = ref(20);
const query = ref('');
const remoteData = ref<any[]>([]);
const remote = ref(true);
const equalValue = (value: any, v: any): boolean => {
if (typeof v === 'object') {
const key = props.config.valueKey || 'value';
return v[key] === value[key];
}
return value === v;
};
const mapOptions = (data: any[]) => {
const {
option = {
text: 'text',
value: 'value',
},
} = props.config;
const { text = 'text', value = 'value' } = option;
return data.map((item) => ({
text: typeof text === 'function' ? text(item) : item[text],
value: typeof value === 'function' ? value(item) : item[value],
}));
};
const getOptions = async () => {
if (!props.model) return [];
if (localOptions.value.length) {
return localOptions.value;
}
loading.value = true;
let items: SelectOption[] | SelectGroupOption[] = [];
const { option } = props.config;
if (!option) return [];
const { root = '', totalKey = 'total' } = option;
let { body = {}, url } = option;
if (typeof url === 'function') {
url = await url(mForm, { model: props.model, formValue: mForm?.values });
}
let postOptions: Record<string, any> = {
method: option.method || 'POST',
url,
cache: option.cache,
timeout: option.timeout,
mode: option.mode,
headers: option.headers || {},
json: option.json || false,
};
if (typeof body === 'function') {
body = body(mForm, {
model: props.model,
formValue: mForm?.values,
formValues: mForm?.values,
config: props.config,
}) as Record<string, any>;
}
body.query = query.value;
body.pgSize = pgSize.value;
body.pgIndex = pgIndex.value;
postOptions.data = body;
const requestFuc = getConfig('request') as Function;
if (typeof option.beforeRequest === 'function') {
postOptions = option.beforeRequest(mForm, postOptions, {
model: props.model,
formValue: mForm?.values,
});
}
if (option.method?.toLocaleLowerCase() === 'jsonp') {
postOptions.jsonpCallback = option.jsonpCallback || 'callback';
}
let res = await requestFuc(postOptions);
if (typeof option.afterRequest === 'function') {
res = option.afterRequest(mForm, res, {
model: props.model,
formValue: mForm?.values,
formValues: mForm?.values,
config: props.config,
postOptions,
});
}
const optionsData = getValueByKeyPath(root, res);
const resTotal = globalThis.parseInt(getValueByKeyPath(totalKey, res), 10);
if (resTotal > 0) {
total.value = resTotal;
}
remoteData.value = remoteData.value.concat(optionsData);
if (optionsData) {
if (typeof option.item === 'function') {
items = option.item(optionsData);
} else if (optionsData.map) {
items = mapOptions(optionsData);
}
}
loading.value = false;
// 多选过滤时会导致已选的选项显示不了,所以要把已选的选项保留不要过滤没了
const selectedOptions: SelectOption[] | SelectGroupOption[] = [];
if (props.config.multiple && props.model[props.name]) {
options.value.forEach((o: any) => {
const isInclude = props.model?.[props.name].includes(o.value);
if (isInclude && !(items as any[]).find((op: any) => op.value === o.value)) {
selectedOptions.push(o);
}
});
}
return pgIndex.value === 0 ? (selectedOptions as any[]).concat(items) : (options.value as any).concat(items);
};
const getInitLocalOption = async () => {
if (!props.model) return [];
const value = props.model[props.name];
const { config } = props;
localOptions.value = await getOptions();
remote.value = false;
if (config.group) {
if (config.multiple && value.findIndex) {
return (localOptions.value as SelectGroupOption[]).filter(
(group) => group.options.findIndex((item) => value.find((v: any) => equalValue(item.value, v)) > -1) > -1,
);
}
return (localOptions.value as SelectGroupOption[]).filter(
(group) => group.options.findIndex((item) => equalValue(item.value, value)) > -1,
);
}
if (config.multiple && value.findIndex) {
return (localOptions.value as any[]).filter((item) => value.findIndex((v: any) => equalValue(item.value, v)) > -1);
}
return (localOptions.value as any[]).filter((item) => equalValue(item.value, value));
};
const getInitOption = async () => {
if (!props.model) return [];
const { option } = props.config;
if (!option) return [];
const { root = '', initRoot = '' } = option;
let { initBody = {} } = option;
let options: SelectOption[] | SelectGroupOption[] = [];
let url = option.initUrl;
if (!url) {
return getInitLocalOption();
}
if (typeof url === 'function') {
url = await url(mForm, { model: props.model, formValue: mForm?.values });
}
if (typeof initBody === 'function') {
initBody = initBody(mForm, {
model: props.model,
formValue: mForm?.values,
formValues: mForm?.values,
config: props.config,
}) as Record<string, any>;
}
let postOptions: Record<string, any> = {
method: option.method || 'POST',
url,
data: {
id: props.model[props.name],
...initBody,
},
mode: option.mode,
headers: option.headers || {},
json: option.json || false,
};
if (typeof option.beforeInitRequest === 'function') {
postOptions = option.beforeInitRequest(mForm, postOptions, {
model: props.model,
formValue: mForm?.values,
});
}
if (option.method?.toLocaleLowerCase() === 'jsonp') {
postOptions.jsonpCallback = option.jsonpCallback || 'callback';
}
const requestFuc = getConfig('request') as Function;
let res = await requestFuc(postOptions);
if (typeof option.afterRequest === 'function') {
res = option.afterRequest(mForm, res, {
model: props.model,
formValue: mForm?.values,
formValues: mForm?.values,
config: props.config,
postOptions,
});
}
let initData = getValueByKeyPath(initRoot || root, res);
if (initData) {
if (!Array.isArray(initData)) {
initData = [initData];
}
if (typeof option.item === 'function') {
options = option.item(initData);
} else if (initData.map) {
options = mapOptions(initData);
}
}
return options;
};
if (typeof props.config.options === 'function') {
watchEffect(() => {
typeof props.config.options === 'function' &&
Promise.resolve(
props.config.options(mForm, {
model: props.model,
prop: props.prop,
formValues: mForm?.values,
formValue: mForm?.values,
config: props.config,
}),
).then((data) => {
options.value = data;
});
});
} else if (Array.isArray(props.config.options)) {
watchEffect(() => {
options.value = props.config.options as SelectOption[] | SelectGroupOption[];
});
} else if (props.config.option) {
onBeforeMount(() => {
if (!props.model) return;
const v = props.model[props.name];
if (Array.isArray(v) ? v.length : typeof v !== 'undefined') {
getInitOption().then((data) => {
options.value = data;
});
}
});
}
if (props.config.remote) {
const unWatch = watch(
() => tMagicSelect.value?.scrollbarWrap,
(scrollbarWrap) => {
if (!scrollbarWrap) {
return;
}
nextTick(() => unWatch());
scrollbarWrap.addEventListener('scroll', async (e: Event) => {
const el = e.currentTarget as HTMLDivElement;
if (moreLoadingVisible.value) {
return;
}
if (el.scrollHeight - el.clientHeight - el.scrollTop > 1) {
return;
}
if (total.value <= options.value.length) {
return;
}
moreLoadingVisible.value = true;
pgIndex.value += 1;
options.value = await getOptions();
moreLoadingVisible.value = false;
});
},
{
immediate: true,
},
);
}
const popperClass = mForm?.popperClass;
const changeHandler = (value: any) => {
emit('change', value);
};
const visibleHandler = async (visible: boolean) => {
if (!visible) return;
if (!props.config.remote) return;
if (query.value && tMagicSelect.value) {
tMagicSelect.value.setQuery(query.value);
tMagicSelect.value.setPreviousQuery(query.value);
tMagicSelect.value.setSelectedLabel(query.value);
} else if (options.value.length <= (props.config.multiple ? props.model?.[props.name].length : 1)) {
options.value = await getOptions();
}
};
const remoteMethod = async (q: string) => {
if (!localOptions.value.length) {
query.value = q;
pgIndex.value = 0;
options.value = await getOptions();
// 多选时如果过滤选项会导致已选好的标签异常需要重新刷新一下el-select的状态
if (props.config.multiple)
setTimeout(() => {
tMagicSelect.value?.setSelected();
}, 0);
}
};
const itemOptions = options as Ref<SelectOption[]>;
const groupOptions = options as Ref<SelectGroupOption[]>;
</script>