mirror of
https://git.mirrors.martin98.com/https://github.com/langgenius/dify.git
synced 2025-05-30 01:55:17 +08:00

Signed-off-by: yihong0618 <zouzou0208@gmail.com> Signed-off-by: -LAN- <laipz8200@outlook.com> Signed-off-by: xhe <xw897002528@gmail.com> Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: takatost <takatost@gmail.com> Co-authored-by: kurokobo <kuro664@gmail.com> Co-authored-by: Novice Lee <novicelee@NoviPro.local> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: AkaraChen <akarachen@outlook.com> Co-authored-by: Yi <yxiaoisme@gmail.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: JzoNg <jzongcode@gmail.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: Hiroshi Fujita <fujita-h@users.noreply.github.com> Co-authored-by: AkaraChen <85140972+AkaraChen@users.noreply.github.com> Co-authored-by: NFish <douxc512@gmail.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: 非法操作 <hjlarry@163.com> Co-authored-by: Novice <857526207@qq.com> Co-authored-by: Hiroki Nagai <82458324+nagaihiroki-git@users.noreply.github.com> Co-authored-by: Gen Sato <52241300+halogen22@users.noreply.github.com> Co-authored-by: eux <euxuuu@gmail.com> Co-authored-by: huangzhuo1949 <167434202+huangzhuo1949@users.noreply.github.com> Co-authored-by: huangzhuo <huangzhuo1@xiaomi.com> Co-authored-by: lotsik <lotsik@mail.ru> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: gakkiyomi <gakkiyomi@aliyun.com> Co-authored-by: CN-P5 <heibai2006@gmail.com> Co-authored-by: CN-P5 <heibai2006@qq.com> Co-authored-by: Chuehnone <1897025+chuehnone@users.noreply.github.com> Co-authored-by: yihong <zouzou0208@gmail.com> Co-authored-by: Kevin9703 <51311316+Kevin9703@users.noreply.github.com> Co-authored-by: -LAN- <laipz8200@outlook.com> Co-authored-by: Boris Feld <lothiraldan@gmail.com> Co-authored-by: mbo <himabo@gmail.com> Co-authored-by: mabo <mabo@aeyes.ai> Co-authored-by: Warren Chen <warren.chen830@gmail.com> Co-authored-by: JzoNgKVO <27049666+JzoNgKVO@users.noreply.github.com> Co-authored-by: jiandanfeng <chenjh3@wangsu.com> Co-authored-by: zhu-an <70234959+xhdd123321@users.noreply.github.com> Co-authored-by: zhaoqingyu.1075 <zhaoqingyu.1075@bytedance.com> Co-authored-by: 海狸大師 <86974027+yenslife@users.noreply.github.com> Co-authored-by: Xu Song <xusong.vip@gmail.com> Co-authored-by: rayshaw001 <396301947@163.com> Co-authored-by: Ding Jiatong <dingjiatong@gmail.com> Co-authored-by: Bowen Liang <liangbowen@gf.com.cn> Co-authored-by: JasonVV <jasonwangiii@outlook.com> Co-authored-by: le0zh <newlight@qq.com> Co-authored-by: zhuxinliang <zhuxinliang@didiglobal.com> Co-authored-by: k-zaku <zaku99@outlook.jp> Co-authored-by: luckylhb90 <luckylhb90@gmail.com> Co-authored-by: hobo.l <hobo.l@binance.com> Co-authored-by: jiangbo721 <365065261@qq.com> Co-authored-by: 刘江波 <jiangbo721@163.com> Co-authored-by: Shun Miyazawa <34241526+miya@users.noreply.github.com> Co-authored-by: EricPan <30651140+Egfly@users.noreply.github.com> Co-authored-by: crazywoola <427733928@qq.com> Co-authored-by: sino <sino2322@gmail.com> Co-authored-by: Jhvcc <37662342+Jhvcc@users.noreply.github.com> Co-authored-by: lowell <lowell.hu@zkteco.in> Co-authored-by: Boris Polonsky <BorisPolonsky@users.noreply.github.com> Co-authored-by: Ademílson Tonato <ademilsonft@outlook.com> Co-authored-by: Ademílson Tonato <ademilson.tonato@refurbed.com> Co-authored-by: IWAI, Masaharu <iwaim.sub@gmail.com> Co-authored-by: Yueh-Po Peng (Yabi) <94939112+y10ab1@users.noreply.github.com> Co-authored-by: Jason <ggbbddjm@gmail.com> Co-authored-by: Xin Zhang <sjhpzx@gmail.com> Co-authored-by: yjc980121 <3898524+yjc980121@users.noreply.github.com> Co-authored-by: heyszt <36215648+hieheihei@users.noreply.github.com> Co-authored-by: Abdullah AlOsaimi <osaimiacc@gmail.com> Co-authored-by: Abdullah AlOsaimi <189027247+osaimi@users.noreply.github.com> Co-authored-by: Yingchun Lai <laiyingchun@apache.org> Co-authored-by: Hash Brown <hi@xzd.me> Co-authored-by: zuodongxu <192560071+zuodongxu@users.noreply.github.com> Co-authored-by: Masashi Tomooka <tmokmss@users.noreply.github.com> Co-authored-by: aplio <ryo.091219@gmail.com> Co-authored-by: Obada Khalili <54270856+obadakhalili@users.noreply.github.com> Co-authored-by: Nam Vu <zuzoovn@gmail.com> Co-authored-by: Kei YAMAZAKI <1715090+kei-yamazaki@users.noreply.github.com> Co-authored-by: TechnoHouse <13776377+deephbz@users.noreply.github.com> Co-authored-by: Riddhimaan-Senapati <114703025+Riddhimaan-Senapati@users.noreply.github.com> Co-authored-by: MaFee921 <31881301+2284730142@users.noreply.github.com> Co-authored-by: te-chan <t-nakanome@sakura-is.co.jp> Co-authored-by: HQidea <HQidea@users.noreply.github.com> Co-authored-by: Joshbly <36315710+Joshbly@users.noreply.github.com> Co-authored-by: xhe <xw897002528@gmail.com> Co-authored-by: weiwenyan-dev <154779315+weiwenyan-dev@users.noreply.github.com> Co-authored-by: ex_wenyan.wei <ex_wenyan.wei@tcl.com> Co-authored-by: engchina <12236799+engchina@users.noreply.github.com> Co-authored-by: engchina <atjapan2015@gmail.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: 呆萌闷油瓶 <253605712@qq.com> Co-authored-by: Kemal <kemalmeler@outlook.com> Co-authored-by: Lazy_Frog <4590648+lazyFrogLOL@users.noreply.github.com> Co-authored-by: Yi Xiao <54782454+YIXIAO0@users.noreply.github.com> Co-authored-by: Steven sun <98230804+Tuyohai@users.noreply.github.com> Co-authored-by: steven <sunzwj@digitalchina.com> Co-authored-by: Kalo Chin <91766386+fdb02983rhy@users.noreply.github.com> Co-authored-by: Katy Tao <34019945+KatyTao@users.noreply.github.com> Co-authored-by: depy <42985524+h4ckdepy@users.noreply.github.com> Co-authored-by: 胡春东 <gycm520@gmail.com> Co-authored-by: Junjie.M <118170653@qq.com> Co-authored-by: MuYu <mr.muzea@gmail.com> Co-authored-by: Naoki Takashima <39912547+takatea@users.noreply.github.com> Co-authored-by: Summer-Gu <37869445+gubinjie@users.noreply.github.com> Co-authored-by: Fei He <droxer.he@gmail.com> Co-authored-by: ybalbert001 <120714773+ybalbert001@users.noreply.github.com> Co-authored-by: Yuanbo Li <ybalbert@amazon.com> Co-authored-by: douxc <7553076+douxc@users.noreply.github.com> Co-authored-by: liuzhenghua <1090179900@qq.com> Co-authored-by: Wu Jiayang <62842862+Wu-Jiayang@users.noreply.github.com> Co-authored-by: Your Name <you@example.com> Co-authored-by: kimjion <45935338+kimjion@users.noreply.github.com> Co-authored-by: AugNSo <song.tiankai@icloud.com> Co-authored-by: llinvokerl <38915183+llinvokerl@users.noreply.github.com> Co-authored-by: liusurong.lsr <liusurong.lsr@alibaba-inc.com> Co-authored-by: Vasu Negi <vasu-negi@users.noreply.github.com> Co-authored-by: Hundredwz <1808096180@qq.com> Co-authored-by: Xiyuan Chen <52963600+GareArc@users.noreply.github.com>
458 lines
19 KiB
TypeScript
458 lines
19 KiB
TypeScript
'use client'
|
|
import type { FC } from 'react'
|
|
import React, { useMemo, useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import Link from 'next/link'
|
|
import {
|
|
RiArrowLeftLine,
|
|
RiArrowRightUpLine,
|
|
} from '@remixicon/react'
|
|
import {
|
|
PortalToFollowElem,
|
|
PortalToFollowElemContent,
|
|
PortalToFollowElemTrigger,
|
|
} from '@/app/components/base/portal-to-follow-elem'
|
|
import ToolTrigger from '@/app/components/plugins/plugin-detail-panel/tool-selector/tool-trigger'
|
|
import ToolItem from '@/app/components/plugins/plugin-detail-panel/tool-selector/tool-item'
|
|
import ToolPicker from '@/app/components/workflow/block-selector/tool-picker'
|
|
import Button from '@/app/components/base/button'
|
|
import Indicator from '@/app/components/header/indicator'
|
|
import ToolCredentialForm from '@/app/components/plugins/plugin-detail-panel/tool-selector/tool-credentials-form'
|
|
import Toast from '@/app/components/base/toast'
|
|
import Textarea from '@/app/components/base/textarea'
|
|
import Divider from '@/app/components/base/divider'
|
|
import TabSlider from '@/app/components/base/tab-slider-plain'
|
|
import ReasoningConfigForm from '@/app/components/plugins/plugin-detail-panel/tool-selector/reasoning-config-form'
|
|
import Form from '@/app/components/header/account-setting/model-provider-page/model-modal/Form'
|
|
import { generateFormValue, getPlainValue, getStructureValue, toolParametersToFormSchemas } from '@/app/components/tools/utils/to-form-schema'
|
|
|
|
import { useAppContext } from '@/context/app-context'
|
|
import {
|
|
useAllBuiltInTools,
|
|
useAllCustomTools,
|
|
useAllWorkflowTools,
|
|
useInvalidateAllBuiltInTools,
|
|
useUpdateProviderCredentials,
|
|
} from '@/service/use-tools'
|
|
import { useInvalidateInstalledPluginList } from '@/service/use-plugins'
|
|
import { usePluginInstalledCheck } from '@/app/components/plugins/plugin-detail-panel/tool-selector/hooks'
|
|
import { CollectionType } from '@/app/components/tools/types'
|
|
import type { ToolDefaultValue, ToolValue } from '@/app/components/workflow/block-selector/types'
|
|
import type {
|
|
OffsetOptions,
|
|
Placement,
|
|
} from '@floating-ui/react'
|
|
import { MARKETPLACE_API_PREFIX } from '@/config'
|
|
import type { Node } from 'reactflow'
|
|
import type { NodeOutPutVar } from '@/app/components/workflow/types'
|
|
import cn from '@/utils/classnames'
|
|
|
|
type Props = {
|
|
disabled?: boolean
|
|
placement?: Placement
|
|
offset?: OffsetOptions
|
|
scope?: string
|
|
value?: ToolValue
|
|
selectedTools?: ToolValue[]
|
|
onSelect: (tool: {
|
|
provider_name: string
|
|
tool_name: string
|
|
tool_label: string
|
|
settings?: Record<string, any>
|
|
parameters?: Record<string, any>
|
|
extra?: Record<string, any>
|
|
}) => void
|
|
onDelete?: () => void
|
|
supportEnableSwitch?: boolean
|
|
supportAddCustomTool?: boolean
|
|
trigger?: React.ReactNode
|
|
controlledState?: boolean
|
|
onControlledStateChange?: (state: boolean) => void
|
|
panelShowState?: boolean
|
|
onPanelShowStateChange?: (state: boolean) => void
|
|
nodeOutputVars: NodeOutPutVar[],
|
|
availableNodes: Node[],
|
|
nodeId?: string,
|
|
}
|
|
const ToolSelector: FC<Props> = ({
|
|
value,
|
|
selectedTools,
|
|
disabled,
|
|
placement = 'left',
|
|
offset = 4,
|
|
onSelect,
|
|
onDelete,
|
|
scope,
|
|
supportEnableSwitch,
|
|
trigger,
|
|
controlledState,
|
|
onControlledStateChange,
|
|
panelShowState,
|
|
onPanelShowStateChange,
|
|
nodeOutputVars,
|
|
availableNodes,
|
|
nodeId = '',
|
|
}) => {
|
|
const { t } = useTranslation()
|
|
const [isShow, onShowChange] = useState(false)
|
|
const handleTriggerClick = () => {
|
|
if (disabled) return
|
|
onShowChange(true)
|
|
}
|
|
|
|
const { data: buildInTools } = useAllBuiltInTools()
|
|
const { data: customTools } = useAllCustomTools()
|
|
const { data: workflowTools } = useAllWorkflowTools()
|
|
const invalidateAllBuiltinTools = useInvalidateAllBuiltInTools()
|
|
const invalidateInstalledPluginList = useInvalidateInstalledPluginList()
|
|
|
|
// plugin info check
|
|
const { inMarketPlace, manifest } = usePluginInstalledCheck(value?.provider_name)
|
|
|
|
const currentProvider = useMemo(() => {
|
|
const mergedTools = [...(buildInTools || []), ...(customTools || []), ...(workflowTools || [])]
|
|
return mergedTools.find((toolWithProvider) => {
|
|
return toolWithProvider.id === value?.provider_name
|
|
})
|
|
}, [value, buildInTools, customTools, workflowTools])
|
|
|
|
const [isShowChooseTool, setIsShowChooseTool] = useState(false)
|
|
const handleSelectTool = (tool: ToolDefaultValue) => {
|
|
const settingValues = generateFormValue(tool.params, toolParametersToFormSchemas(tool.paramSchemas.filter(param => param.form !== 'llm') as any))
|
|
const paramValues = generateFormValue(tool.params, toolParametersToFormSchemas(tool.paramSchemas.filter(param => param.form === 'llm') as any), true)
|
|
const toolValue = {
|
|
provider_name: tool.provider_id,
|
|
type: tool.provider_type,
|
|
tool_name: tool.tool_name,
|
|
tool_label: tool.tool_label,
|
|
settings: settingValues,
|
|
parameters: paramValues,
|
|
enabled: tool.is_team_authorization,
|
|
extra: {
|
|
description: '',
|
|
},
|
|
schemas: tool.paramSchemas,
|
|
}
|
|
onSelect(toolValue)
|
|
// setIsShowChooseTool(false)
|
|
}
|
|
|
|
const handleDescriptionChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {
|
|
onSelect({
|
|
...value,
|
|
extra: {
|
|
...value?.extra,
|
|
description: e.target.value || '',
|
|
},
|
|
} as any)
|
|
}
|
|
|
|
// tool settings & params
|
|
const currentToolSettings = useMemo(() => {
|
|
if (!currentProvider) return []
|
|
return currentProvider.tools.find(tool => tool.name === value?.tool_name)?.parameters.filter(param => param.form !== 'llm') || []
|
|
}, [currentProvider, value])
|
|
const currentToolParams = useMemo(() => {
|
|
if (!currentProvider) return []
|
|
return currentProvider.tools.find(tool => tool.name === value?.tool_name)?.parameters.filter(param => param.form === 'llm') || []
|
|
}, [currentProvider, value])
|
|
const [currType, setCurrType] = useState('settings')
|
|
const showTabSlider = currentToolSettings.length > 0 && currentToolParams.length > 0
|
|
const userSettingsOnly = currentToolSettings.length > 0 && !currentToolParams.length
|
|
const reasoningConfigOnly = currentToolParams.length > 0 && !currentToolSettings.length
|
|
|
|
const settingsFormSchemas = useMemo(() => toolParametersToFormSchemas(currentToolSettings), [currentToolSettings])
|
|
const paramsFormSchemas = useMemo(() => toolParametersToFormSchemas(currentToolParams), [currentToolParams])
|
|
|
|
const handleSettingsFormChange = (v: Record<string, any>) => {
|
|
const newValue = getStructureValue(v)
|
|
|
|
const toolValue = {
|
|
...value,
|
|
settings: newValue,
|
|
}
|
|
onSelect(toolValue as any)
|
|
}
|
|
const handleParamsFormChange = (v: Record<string, any>) => {
|
|
const toolValue = {
|
|
...value,
|
|
parameters: v,
|
|
}
|
|
onSelect(toolValue as any)
|
|
}
|
|
|
|
const handleEnabledChange = (state: boolean) => {
|
|
onSelect({
|
|
...value,
|
|
enabled: state,
|
|
} as any)
|
|
}
|
|
|
|
// authorization
|
|
const { isCurrentWorkspaceManager } = useAppContext()
|
|
const [isShowSettingAuth, setShowSettingAuth] = useState(false)
|
|
const handleCredentialSettingUpdate = () => {
|
|
invalidateAllBuiltinTools()
|
|
Toast.notify({
|
|
type: 'success',
|
|
message: t('common.api.actionSuccess'),
|
|
})
|
|
setShowSettingAuth(false)
|
|
onShowChange(false)
|
|
}
|
|
|
|
const { mutate: updatePermission } = useUpdateProviderCredentials({
|
|
onSuccess: handleCredentialSettingUpdate,
|
|
})
|
|
|
|
// install from marketplace
|
|
const currentTool = useMemo(() => {
|
|
return currentProvider?.tools.find(tool => tool.name === value?.tool_name)
|
|
}, [currentProvider?.tools, value?.tool_name])
|
|
const manifestIcon = useMemo(() => {
|
|
if (!manifest)
|
|
return ''
|
|
return `${MARKETPLACE_API_PREFIX}/plugins/${(manifest as any).plugin_id}/icon`
|
|
}, [manifest])
|
|
const handleInstall = async () => {
|
|
invalidateAllBuiltinTools()
|
|
invalidateInstalledPluginList()
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<PortalToFollowElem
|
|
placement={placement}
|
|
offset={offset}
|
|
open={trigger ? controlledState : isShow}
|
|
onOpenChange={trigger ? onControlledStateChange : onShowChange}
|
|
>
|
|
<PortalToFollowElemTrigger
|
|
className='w-full'
|
|
onClick={() => {
|
|
if (!currentProvider || !currentTool) return
|
|
handleTriggerClick()
|
|
}}
|
|
>
|
|
{trigger}
|
|
{!trigger && !value?.provider_name && (
|
|
<ToolTrigger
|
|
isConfigure
|
|
open={isShow}
|
|
value={value}
|
|
provider={currentProvider}
|
|
/>
|
|
)}
|
|
{!trigger && value?.provider_name && (
|
|
<ToolItem
|
|
open={isShow}
|
|
icon={currentProvider?.icon || manifestIcon}
|
|
providerName={value.provider_name}
|
|
toolLabel={value.tool_label || value.tool_name}
|
|
showSwitch={supportEnableSwitch}
|
|
switchValue={value.enabled}
|
|
onSwitchChange={handleEnabledChange}
|
|
onDelete={onDelete}
|
|
noAuth={currentProvider && currentTool && !currentProvider.is_team_authorization}
|
|
onAuth={() => setShowSettingAuth(true)}
|
|
uninstalled={!currentProvider && inMarketPlace}
|
|
versionMismatch={currentProvider && inMarketPlace && !currentTool}
|
|
installInfo={manifest?.latest_package_identifier}
|
|
onInstall={() => handleInstall()}
|
|
isError={(!currentProvider || !currentTool) && !inMarketPlace}
|
|
errorTip={
|
|
<div className='space-y-1 max-w-[240px] text-xs'>
|
|
<h3 className='text-text-primary font-semibold'>{currentTool ? t('plugin.detailPanel.toolSelector.uninstalledTitle') : t('plugin.detailPanel.toolSelector.unsupportedTitle')}</h3>
|
|
<p className='text-text-secondary tracking-tight'>{currentTool ? t('plugin.detailPanel.toolSelector.uninstalledContent') : t('plugin.detailPanel.toolSelector.unsupportedContent')}</p>
|
|
<p>
|
|
<Link href={'/plugins'} className='text-text-accent tracking-tight'>{t('plugin.detailPanel.toolSelector.uninstalledLink')}</Link>
|
|
</p>
|
|
</div>
|
|
}
|
|
/>
|
|
)}
|
|
</PortalToFollowElemTrigger>
|
|
<PortalToFollowElemContent className='z-[1000]'>
|
|
<div className={cn('relative w-[361px] min-h-20 max-h-[642px] pb-4 rounded-xl backdrop-blur-sm bg-components-panel-bg-blur border-[0.5px] border-components-panel-border shadow-lg', !isShowSettingAuth && 'overflow-y-auto pb-2')}>
|
|
{!isShowSettingAuth && (
|
|
<>
|
|
<div className='px-4 pt-3.5 pb-1 text-text-primary system-xl-semibold'>{t('plugin.detailPanel.toolSelector.title')}</div>
|
|
{/* base form */}
|
|
<div className='px-4 py-2 flex flex-col gap-3'>
|
|
<div className='flex flex-col gap-1'>
|
|
<div className='h-6 flex items-center system-sm-semibold text-text-secondary'>{t('plugin.detailPanel.toolSelector.toolLabel')}</div>
|
|
<ToolPicker
|
|
panelClassName='w-[328px]'
|
|
placement='bottom'
|
|
offset={offset}
|
|
trigger={
|
|
<ToolTrigger
|
|
open={panelShowState || isShowChooseTool}
|
|
value={value}
|
|
provider={currentProvider}
|
|
/>
|
|
}
|
|
isShow={panelShowState || isShowChooseTool}
|
|
onShowChange={trigger ? onPanelShowStateChange as any : setIsShowChooseTool}
|
|
disabled={false}
|
|
supportAddCustomTool
|
|
onSelect={handleSelectTool}
|
|
scope={scope}
|
|
selectedTools={selectedTools}
|
|
/>
|
|
</div>
|
|
<div className='flex flex-col gap-1'>
|
|
<div className='h-6 flex items-center system-sm-semibold text-text-secondary'>{t('plugin.detailPanel.toolSelector.descriptionLabel')}</div>
|
|
<Textarea
|
|
className='resize-none'
|
|
placeholder={t('plugin.detailPanel.toolSelector.descriptionPlaceholder')}
|
|
value={value?.extra?.description || ''}
|
|
onChange={handleDescriptionChange}
|
|
disabled={!value?.provider_name}
|
|
/>
|
|
</div>
|
|
</div>
|
|
{/* authorization */}
|
|
{currentProvider && currentProvider.type === CollectionType.builtIn && currentProvider.allow_delete && (
|
|
<>
|
|
<Divider className='my-1 w-full' />
|
|
<div className='px-4 py-2'>
|
|
{!currentProvider.is_team_authorization && (
|
|
<Button
|
|
variant='primary'
|
|
className={cn('shrink-0 w-full')}
|
|
onClick={() => setShowSettingAuth(true)}
|
|
disabled={!isCurrentWorkspaceManager}
|
|
>
|
|
{t('tools.auth.unauthorized')}
|
|
</Button>
|
|
)}
|
|
{currentProvider.is_team_authorization && (
|
|
<Button
|
|
variant='secondary'
|
|
className={cn('shrink-0 w-full')}
|
|
onClick={() => setShowSettingAuth(true)}
|
|
disabled={!isCurrentWorkspaceManager}
|
|
>
|
|
<Indicator className='mr-2' color={'green'} />
|
|
{t('tools.auth.authorized')}
|
|
</Button>
|
|
)}
|
|
</div>
|
|
</>
|
|
)}
|
|
{/* tool settings */}
|
|
{(currentToolSettings.length > 0 || currentToolParams.length > 0) && currentProvider?.is_team_authorization && (
|
|
<>
|
|
<Divider className='my-1 w-full' />
|
|
{/* tabs */}
|
|
{nodeId && showTabSlider && (
|
|
<TabSlider
|
|
className='shrink-0 mt-1 px-4'
|
|
itemClassName='py-3'
|
|
noBorderBottom
|
|
smallItem
|
|
value={currType}
|
|
onChange={(value) => {
|
|
setCurrType(value)
|
|
}}
|
|
options={[
|
|
{ value: 'settings', text: t('plugin.detailPanel.toolSelector.settings')! },
|
|
{ value: 'params', text: t('plugin.detailPanel.toolSelector.params')! },
|
|
]}
|
|
/>
|
|
)}
|
|
{nodeId && showTabSlider && currType === 'params' && (
|
|
<div className='px-4 py-2'>
|
|
<div className='text-text-tertiary system-xs-regular'>{t('plugin.detailPanel.toolSelector.paramsTip1')}</div>
|
|
<div className='text-text-tertiary system-xs-regular'>{t('plugin.detailPanel.toolSelector.paramsTip2')}</div>
|
|
</div>
|
|
)}
|
|
{/* user settings only */}
|
|
{userSettingsOnly && (
|
|
<div className='p-4 pb-1'>
|
|
<div className='text-text-primary system-sm-semibold-uppercase'>{t('plugin.detailPanel.toolSelector.settings')}</div>
|
|
</div>
|
|
)}
|
|
{/* reasoning config only */}
|
|
{nodeId && reasoningConfigOnly && (
|
|
<div className='mb-1 p-4 pb-1'>
|
|
<div className='text-text-primary system-sm-semibold-uppercase'>{t('plugin.detailPanel.toolSelector.params')}</div>
|
|
<div className='pb-1'>
|
|
<div className='text-text-tertiary system-xs-regular'>{t('plugin.detailPanel.toolSelector.paramsTip1')}</div>
|
|
<div className='text-text-tertiary system-xs-regular'>{t('plugin.detailPanel.toolSelector.paramsTip2')}</div>
|
|
</div>
|
|
</div>
|
|
)}
|
|
{/* user settings form */}
|
|
{(currType === 'settings' || userSettingsOnly) && (
|
|
<div className='px-4 py-2'>
|
|
<Form
|
|
value={getPlainValue(value?.settings || {})}
|
|
onChange={handleSettingsFormChange}
|
|
formSchemas={settingsFormSchemas as any}
|
|
isEditMode={true}
|
|
showOnVariableMap={{}}
|
|
validating={false}
|
|
inputClassName='bg-components-input-bg-normal hover:bg-components-input-bg-hover'
|
|
fieldMoreInfo={item => item.url
|
|
? (<a
|
|
href={item.url}
|
|
target='_blank' rel='noopener noreferrer'
|
|
className='inline-flex items-center text-xs text-text-accent'
|
|
>
|
|
{t('tools.howToGet')}
|
|
<RiArrowRightUpLine className='ml-1 w-3 h-3' />
|
|
</a>)
|
|
: null}
|
|
/>
|
|
</div>
|
|
)}
|
|
{/* reasoning config form */}
|
|
{nodeId && (currType === 'params' || reasoningConfigOnly) && (
|
|
<ReasoningConfigForm
|
|
value={value?.parameters || {}}
|
|
onChange={handleParamsFormChange}
|
|
schemas={paramsFormSchemas as any}
|
|
nodeOutputVars={nodeOutputVars}
|
|
availableNodes={availableNodes}
|
|
nodeId={nodeId}
|
|
/>
|
|
)}
|
|
</>
|
|
)}
|
|
</>
|
|
)}
|
|
{/* authorization panel */}
|
|
{isShowSettingAuth && currentProvider && (
|
|
<>
|
|
<div className='relative pt-3.5 flex flex-col gap-1'>
|
|
<div className='absolute -top-2 left-2 w-[345px] pt-2 rounded-t-xl backdrop-blur-sm bg-components-panel-bg-blur border-[0.5px] border-components-panel-border'></div>
|
|
<div
|
|
className='px-3 h-6 flex items-center gap-1 text-text-accent-secondary system-xs-semibold-uppercase cursor-pointer'
|
|
onClick={() => setShowSettingAuth(false)}
|
|
>
|
|
<RiArrowLeftLine className='w-4 h-4' />
|
|
BACK
|
|
</div>
|
|
<div className='px-4 text-text-primary system-xl-semibold'>{t('tools.auth.setupModalTitle')}</div>
|
|
<div className='px-4 text-text-tertiary system-xs-regular'>{t('tools.auth.setupModalTitleDescription')}</div>
|
|
</div>
|
|
<ToolCredentialForm
|
|
collection={currentProvider}
|
|
onCancel={() => setShowSettingAuth(false)}
|
|
onSaved={async value => updatePermission({
|
|
providerName: currentProvider.name,
|
|
credentials: value,
|
|
})}
|
|
/>
|
|
</>
|
|
)}
|
|
</div>
|
|
</PortalToFollowElemContent>
|
|
</PortalToFollowElem>
|
|
</>
|
|
)
|
|
}
|
|
export default React.memo(ToolSelector)
|