mirror of https://github.com/nocodb/nocodb
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
7.0 KiB
202 lines
7.0 KiB
2 months ago
|
<script lang="ts" setup>
|
||
|
const props = withDefaults(
|
||
|
defineProps<{
|
||
|
fkIntegrationId?: string
|
||
|
model?: string
|
||
|
randomness?: string
|
||
|
workspaceId: string
|
||
|
scope?: string
|
||
|
showTooltip?: boolean
|
||
|
}>(),
|
||
|
{
|
||
|
showTooltip: true,
|
||
|
},
|
||
|
)
|
||
|
|
||
|
const emits = defineEmits(['update:fkIntegrationId', 'update:model', 'update:randomness'])
|
||
|
|
||
|
const vFkIntegrationId = useVModel(props, 'fkIntegrationId', emits)
|
||
|
|
||
|
const vModel = useVModel(props, 'model', emits)
|
||
|
|
||
|
// const vRandomness = useVModel(props, 'randomness', emits)
|
||
|
|
||
|
const { $api } = useNuxtApp()
|
||
|
|
||
|
const { aiIntegrations } = useNocoAi()
|
||
|
|
||
|
const lastIntegrationId = ref<string | null>(null)
|
||
|
|
||
|
const isDropdownOpen = ref(false)
|
||
|
|
||
|
const availableModels = ref<string[]>([])
|
||
|
|
||
|
const isLoadingAvailableModels = ref<boolean>(false)
|
||
|
|
||
|
const onIntegrationChange = async (newFkINtegrationId?: string) => {
|
||
|
if (!vFkIntegrationId.value && !newFkINtegrationId) return
|
||
|
|
||
|
if (!newFkINtegrationId) {
|
||
|
newFkINtegrationId = vFkIntegrationId.value
|
||
|
}
|
||
|
|
||
|
availableModels.value = []
|
||
|
|
||
|
if (lastIntegrationId.value !== newFkINtegrationId) {
|
||
|
vModel.value = undefined
|
||
|
}
|
||
|
|
||
|
isLoadingAvailableModels.value = true
|
||
|
|
||
|
try {
|
||
|
const response = await $api.integrations.endpoint(newFkINtegrationId, 'availableModels', {})
|
||
|
availableModels.value = response as string[]
|
||
|
|
||
|
if (!vModel.value && availableModels.value.length > 0) {
|
||
|
vModel.value = availableModels.value[0]
|
||
|
}
|
||
|
} catch (error) {
|
||
|
console.error(error)
|
||
|
} finally {
|
||
|
isLoadingAvailableModels.value = false
|
||
|
}
|
||
|
}
|
||
|
|
||
|
onMounted(async () => {
|
||
|
if (!vFkIntegrationId.value) {
|
||
|
if (aiIntegrations.value.length > 0 && aiIntegrations.value[0].id) {
|
||
|
vFkIntegrationId.value = aiIntegrations.value[0].id
|
||
|
nextTick(() => {
|
||
|
onIntegrationChange()
|
||
|
})
|
||
|
}
|
||
|
} else {
|
||
|
lastIntegrationId.value = vFkIntegrationId.value
|
||
|
}
|
||
|
})
|
||
|
</script>
|
||
|
|
||
|
<template>
|
||
|
<NcDropdown v-model:visible="isDropdownOpen" :trigger="['click']" placement="bottomRight" overlay-class-name="overflow-hidden">
|
||
|
<slot>
|
||
|
<GeneralIcon icon="ncSettings" class="text-gray-500 cursor-pointer" />
|
||
|
</slot>
|
||
|
|
||
|
<template #overlay>
|
||
|
<div class="flex flex-col w-[320px] overflow-hidden">
|
||
|
<div class="flex items-center justify-between w-full p-3 bg-purple-50">
|
||
|
<span class="text-sm font-bold text-nc-content-gray">Settings</span>
|
||
|
<!-- Todo: add docs link -->
|
||
|
<a
|
||
|
target="_blank"
|
||
|
rel="noopener noreferrer"
|
||
|
class="!no-underline !hover:(underline text-nc-content-purple-dark) text-nc-content-purple-dark"
|
||
|
>{{ $t('title.docs') }}</a
|
||
|
>
|
||
|
</div>
|
||
|
<div class="flex flex-col p-3 text-sm gap-3">
|
||
|
<!-- Integration Select -->
|
||
|
<div class="flex items-center gap-2">
|
||
|
<span class="text-nc-content-gray w-2/6">Integration</span>
|
||
|
<div v-if="showTooltip" class="w-1/6 flex justify-end">
|
||
|
<NcTooltip placement="top">
|
||
|
<template #title>
|
||
|
<span>Integration to use for this operation</span>
|
||
|
</template>
|
||
|
<GeneralIcon icon="info" class="text-sm text-gray-500" />
|
||
|
</NcTooltip>
|
||
|
</div>
|
||
|
<a-form-item class="flex-1 !my-0">
|
||
|
<NcSelect
|
||
|
v-model:value="vFkIntegrationId"
|
||
|
class="w-full nc-select-shadow nc-ai-input"
|
||
|
size="middle"
|
||
|
@change="onIntegrationChange"
|
||
|
>
|
||
|
<a-select-option v-for="integration in aiIntegrations" :key="integration.id" :value="integration.id">
|
||
|
<div class="w-full flex gap-2 items-center">
|
||
|
<GeneralIntegrationIcon v-if="integration?.sub_type" :type="integration.sub_type" />
|
||
|
<NcTooltip class="flex-1 truncate" show-on-truncate-only>
|
||
|
<template #title>
|
||
|
{{ integration.title }}
|
||
|
</template>
|
||
|
{{ integration.title }}
|
||
|
</NcTooltip>
|
||
|
<component
|
||
|
:is="iconMap.check"
|
||
|
v-if="vFkIntegrationId === integration.id"
|
||
|
id="nc-selected-item-icon"
|
||
|
class="text-nc-content-purple-medium w-4 h-4"
|
||
|
/>
|
||
|
</div>
|
||
|
</a-select-option>
|
||
|
</NcSelect>
|
||
|
</a-form-item>
|
||
|
</div>
|
||
|
<!-- Model Select -->
|
||
|
<div class="flex items-center gap-2">
|
||
|
<span class="text-nc-content-gray w-2/6">Model</span>
|
||
|
<div v-if="showTooltip" class="w-1/6 flex justify-end">
|
||
|
<NcTooltip placement="top">
|
||
|
<template #title>
|
||
|
<span>Model to use for this operation</span>
|
||
|
</template>
|
||
|
<GeneralIcon icon="info" class="text-sm text-gray-500" />
|
||
|
</NcTooltip>
|
||
|
</div>
|
||
|
|
||
|
<a-form-item class="flex-1 !my-0">
|
||
|
<NcSelect
|
||
|
v-model:value="vModel"
|
||
|
class="w-full nc-select-shadow nc-ai-input"
|
||
|
size="middle"
|
||
|
:disabled="!vFkIntegrationId || availableModels.length === 0"
|
||
|
:loading="isLoadingAvailableModels"
|
||
|
>
|
||
|
<a-select-option v-for="md in availableModels" :key="md" :value="md">
|
||
|
<div class="w-full flex gap-2 items-center">
|
||
|
<NcTooltip class="flex-1 truncate" show-on-truncate-only>
|
||
|
<template #title>
|
||
|
{{ md }}
|
||
|
</template>
|
||
|
{{ md }}
|
||
|
</NcTooltip>
|
||
|
<component
|
||
|
:is="iconMap.check"
|
||
|
v-if="vModel === md"
|
||
|
id="nc-selected-item-icon"
|
||
|
class="text-nc-content-purple-medium w-4 h-4"
|
||
|
/>
|
||
|
</div>
|
||
|
</a-select-option>
|
||
|
</NcSelect>
|
||
|
</a-form-item>
|
||
|
</div>
|
||
|
<!-- Randomness
|
||
|
<div class="flex items-center gap-2">
|
||
|
<span class="text-nc-content-gray w-2/6">Randomness</span>
|
||
|
<div v-if="showTooltip" class="w-1/6 flex justify-end">
|
||
|
<NcTooltip placement="top">
|
||
|
<template #title>
|
||
|
<span>Randomness of the response</span>
|
||
|
</template>
|
||
|
<GeneralIcon icon="info" class="text-sm text-gray-500" />
|
||
|
</NcTooltip>
|
||
|
</div>
|
||
|
<div class="flex-1">
|
||
|
<NcSelect v-model:value="vModel.randomness" class="w-full" size="middle" :disabled="!vModel.fk_integration_id">
|
||
|
<a-select-option value="high">High</a-select-option>
|
||
|
<a-select-option value="medium">Medium</a-select-option>
|
||
|
<a-select-option value="low">Low (Default)</a-select-option>
|
||
|
</NcSelect>
|
||
|
</div>
|
||
|
</div>
|
||
|
-->
|
||
|
</div>
|
||
|
</div>
|
||
|
</template>
|
||
|
</NcDropdown>
|
||
|
</template>
|
||
|
|
||
|
<style lang="scss" scoped></style>
|