|
| 1 | +<template> |
| 2 | + <div class="w-full"> |
| 3 | + <el-select v-model="modelValue" popper-class="select-model" :clearable="true" v-bind="$attrs"> |
| 4 | + <el-option-group |
| 5 | + v-for="(value, label) in options" |
| 6 | + :key="value" |
| 7 | + :label="relatedObject(providerOptions, label, 'provider')?.name" |
| 8 | + > |
| 9 | + <el-option |
| 10 | + v-for="item in value.filter((v: any) => v.status === 'SUCCESS')" |
| 11 | + :key="item.id" |
| 12 | + :label="item.name" |
| 13 | + :value="item.id" |
| 14 | + class="flex-between" |
| 15 | + > |
| 16 | + <div class="flex"> |
| 17 | + <span |
| 18 | + v-html="relatedObject(providerOptions, label, 'provider')?.icon" |
| 19 | + class="model-icon mr-8" |
| 20 | + ></span> |
| 21 | + <span>{{ item.name }}</span> |
| 22 | + <el-tag v-if="item.permission_type === 'PUBLIC'" type="info" class="info-tag ml-8 mt-4"> |
| 23 | + {{ $t('common.public') }} |
| 24 | + </el-tag> |
| 25 | + </div> |
| 26 | + <el-icon class="check-icon" v-if="item.id === modelValue"> |
| 27 | + <Check /> |
| 28 | + </el-icon> |
| 29 | + </el-option> |
| 30 | + <!-- 不可用 --> |
| 31 | + <el-option |
| 32 | + v-for="item in value.filter((v: any) => v.status !== 'SUCCESS')" |
| 33 | + :key="item.id" |
| 34 | + :label="item.name" |
| 35 | + :value="item.id" |
| 36 | + class="flex-between" |
| 37 | + disabled |
| 38 | + > |
| 39 | + <div class="flex"> |
| 40 | + <span |
| 41 | + v-html="relatedObject(providerOptions, label, 'provider')?.icon" |
| 42 | + class="model-icon mr-8" |
| 43 | + ></span> |
| 44 | + <span>{{ item.name }}</span> |
| 45 | + <span class="danger">{{ $t('common.unavailable') }}</span> |
| 46 | + </div> |
| 47 | + <el-icon class="check-icon" v-if="item.id === modelValue"> |
| 48 | + <Check /> |
| 49 | + </el-icon> |
| 50 | + </el-option> |
| 51 | + </el-option-group> |
| 52 | + <template #footer v-if="showFooter"> |
| 53 | + <slot name="footer"> |
| 54 | + <div class="w-full text-left cursor" @click="openCreateModel()"> |
| 55 | + <el-button type="primary" link> |
| 56 | + <el-icon class="mr-4"> |
| 57 | + <Plus /> |
| 58 | + </el-icon> |
| 59 | + {{ $t('views.application.applicationForm.buttons.addModel') }} |
| 60 | + </el-button> |
| 61 | + </div> |
| 62 | + </slot> |
| 63 | + </template> |
| 64 | + </el-select> |
| 65 | + <!-- 添加模版 --> |
| 66 | + <CreateModelDialog |
| 67 | + v-if="showFooter" |
| 68 | + ref="createModelRef" |
| 69 | + @submit="submitModel" |
| 70 | + @change="openCreateModel($event)" |
| 71 | + ></CreateModelDialog> |
| 72 | + <SelectProviderDialog |
| 73 | + v-if="showFooter" |
| 74 | + ref="selectProviderRef" |
| 75 | + @change="openCreateModel($event)" |
| 76 | + /> |
| 77 | + </div> |
| 78 | +</template> |
| 79 | +<script setup lang="ts"> |
| 80 | +import { computed, ref, onMounted } from 'vue' |
| 81 | +import type { Provider } from '@/api/type/model' |
| 82 | +import { relatedObject } from '@/utils/utils' |
| 83 | +import CreateModelDialog from '@/views/template/component/CreateModelDialog.vue' |
| 84 | +import SelectProviderDialog from '@/views/template/component/SelectProviderDialog.vue' |
| 85 | +
|
| 86 | +import { t } from '@/locales' |
| 87 | +import useStore from '@/stores' |
| 88 | +defineOptions({ name: 'ModelSelect' }) |
| 89 | +const props = defineProps<{ |
| 90 | + modelValue: any |
| 91 | + options: any |
| 92 | + showFooter?: false |
| 93 | +}>() |
| 94 | +
|
| 95 | +const emit = defineEmits(['update:modelValue', 'change', 'submitModel']) |
| 96 | +const modelValue = computed({ |
| 97 | + set: (item) => { |
| 98 | + emit('update:modelValue', item) |
| 99 | + }, |
| 100 | + get: () => { |
| 101 | + return props.modelValue |
| 102 | + } |
| 103 | +}) |
| 104 | +const { model } = useStore() |
| 105 | +
|
| 106 | +const createModelRef = ref<InstanceType<typeof CreateModelDialog>>() |
| 107 | +const selectProviderRef = ref<InstanceType<typeof SelectProviderDialog>>() |
| 108 | +const providerOptions = ref<Array<Provider>>([]) |
| 109 | +const loading = ref(false) |
| 110 | +
|
| 111 | +function getProvider() { |
| 112 | + loading.value = true |
| 113 | + model |
| 114 | + .asyncGetProvider() |
| 115 | + .then((res: any) => { |
| 116 | + providerOptions.value = res?.data |
| 117 | + loading.value = false |
| 118 | + }) |
| 119 | + .catch(() => { |
| 120 | + loading.value = false |
| 121 | + }) |
| 122 | +} |
| 123 | +
|
| 124 | +const openCreateModel = (provider?: Provider) => { |
| 125 | + if (provider && provider.provider) { |
| 126 | + createModelRef.value?.open(provider) |
| 127 | + } else { |
| 128 | + selectProviderRef.value?.open() |
| 129 | + } |
| 130 | +} |
| 131 | +
|
| 132 | +function submitModel() { |
| 133 | + emit('submitModel') |
| 134 | +} |
| 135 | +
|
| 136 | +onMounted(() => { |
| 137 | + getProvider() |
| 138 | +}) |
| 139 | +</script> |
| 140 | +<style lang="scss" scoped> |
| 141 | +// AI模型选择:添加模型hover样式 |
| 142 | +.select-model { |
| 143 | + .el-select-dropdown__footer { |
| 144 | + &:hover { |
| 145 | + background-color: var(--el-fill-color-light); |
| 146 | + } |
| 147 | + } |
| 148 | + .model-icon { |
| 149 | + width: 20px; |
| 150 | + } |
| 151 | + .check-icon { |
| 152 | + position: absolute; |
| 153 | + right: 10px; |
| 154 | + } |
| 155 | +} |
| 156 | +</style> |
0 commit comments