'use client' import type { FC } from 'react' import type { Tool as ToolType } from '../../../tools/types' import type { ToolWithProvider } from '../../types' import type { ToolDefaultValue, ToolValue } from '../types' import { RiArrowDownSLine, RiArrowRightSLine } from '@remixicon/react' import { useHover } from 'ahooks' import * as React from 'react' import { useCallback, useEffect, useMemo, useRef } from 'react' import { useTranslation } from 'react-i18next' import { Mcp } from '@/app/components/base/icons/src/vender/other' import { useMCPToolAvailability } from '@/app/components/workflow/nodes/_base/components/mcp-tool-availability' import { useGetLanguage } from '@/context/i18n' import useTheme from '@/hooks/use-theme' import { Theme } from '@/types/app' import { cn } from '@/utils/classnames' import { basePath } from '@/utils/var' import { CollectionType } from '../../../tools/types' import BlockIcon from '../../block-icon' import McpToolNotSupportTooltip from '../../nodes/_base/components/mcp-tool-not-support-tooltip' import { BlockEnum } from '../../types' import { ViewType } from '../view-type-select' import ActionItem from './action-item' const normalizeProviderIcon = (icon?: ToolWithProvider['icon']) => { if (!icon) return icon if (typeof icon === 'string' && basePath && icon.startsWith('/') && !icon.startsWith(`${basePath}/`)) return `${basePath}${icon}` return icon } type Props = { className?: string payload: ToolWithProvider viewType: ViewType hasSearchText: boolean onSelect: (type: BlockEnum, tool: ToolDefaultValue) => void canNotSelectMultiple?: boolean onSelectMultiple?: (type: BlockEnum, tools: ToolDefaultValue[]) => void selectedTools?: ToolValue[] isShowLetterIndex?: boolean } const Tool: FC = ({ className, payload, viewType, hasSearchText, onSelect, canNotSelectMultiple, onSelectMultiple, selectedTools, }) => { const { t } = useTranslation() const { allowed: isMCPToolAllowed } = useMCPToolAvailability() const language = useGetLanguage() const isFlatView = viewType === ViewType.flat const notShowProvider = payload.type === CollectionType.workflow const actions = payload.tools const hasAction = !notShowProvider const [isFold, setFold] = React.useState(true) const ref = useRef(null) const isHovering = useHover(ref) const isMCPTool = payload.type === CollectionType.mcp const isShowCanNotChooseMCPTip = !isMCPToolAllowed && isMCPTool const { theme } = useTheme() const normalizedIcon = useMemo(() => { return normalizeProviderIcon(payload.icon) ?? payload.icon }, [payload.icon]) const normalizedIconDark = useMemo(() => { if (!payload.icon_dark) return undefined return normalizeProviderIcon(payload.icon_dark) ?? payload.icon_dark }, [payload.icon_dark]) const providerIcon = useMemo(() => { if (theme === Theme.dark && normalizedIconDark) return normalizedIconDark return normalizedIcon }, [theme, normalizedIcon, normalizedIconDark]) const getIsDisabled = useCallback((tool: ToolType) => { if (!selectedTools || !selectedTools.length) return false return selectedTools.some(selectedTool => (selectedTool.provider_name === payload.name || selectedTool.provider_name === payload.id) && selectedTool.tool_name === tool.name) }, [payload.id, payload.name, selectedTools]) const totalToolsNum = actions.length const selectedToolsNum = actions.filter(action => getIsDisabled(action)).length const isAllSelected = selectedToolsNum === totalToolsNum const notShowProviderSelectInfo = useMemo(() => { if (isAllSelected) { return ( {t('addToolModal.added', { ns: 'tools' })} ) } }, [isAllSelected, t]) const selectedInfo = useMemo(() => { if (isHovering && !isAllSelected) { return ( { onSelectMultiple?.(BlockEnum.Tool, actions.filter(action => !getIsDisabled(action)).map((tool) => { const params: Record = {} if (tool.parameters) { tool.parameters.forEach((item) => { params[item.name] = '' }) } return { provider_id: payload.id, provider_type: payload.type, provider_name: payload.name, plugin_id: payload.plugin_id, plugin_unique_identifier: payload.plugin_unique_identifier, provider_icon: normalizedIcon, provider_icon_dark: normalizedIconDark, tool_name: tool.name, tool_label: tool.label[language], tool_description: tool.description[language], title: tool.label[language], is_team_authorization: payload.is_team_authorization, paramSchemas: tool.parameters, params, } })) }} > {t('tabs.addAll', { ns: 'workflow' })} ) } if (selectedToolsNum === 0) return <> return ( {isAllSelected ? t('tabs.allAdded', { ns: 'workflow' }) : `${selectedToolsNum} / ${totalToolsNum}`} ) }, [actions, getIsDisabled, isAllSelected, isHovering, language, onSelectMultiple, payload.id, payload.is_team_authorization, payload.name, payload.type, selectedToolsNum, t, totalToolsNum]) useEffect(() => { if (hasSearchText && isFold) { setFold(false) return } if (!hasSearchText && !isFold) setFold(true) }, [hasSearchText]) const FoldIcon = isFold ? RiArrowRightSLine : RiArrowDownSLine const groupName = useMemo(() => { if (payload.type === CollectionType.builtIn) return payload.author if (payload.type === CollectionType.custom) return t('tabs.customTool', { ns: 'workflow' }) if (payload.type === CollectionType.workflow) return t('tabs.workflowTool', { ns: 'workflow' }) return '' }, [payload.author, payload.type, t]) return (
{ if (hasAction) { setFold(!isFold) return } const tool = actions[0] const params: Record = {} if (tool.parameters) { tool.parameters.forEach((item) => { params[item.name] = '' }) } onSelect(BlockEnum.Tool, { provider_id: payload.id, provider_type: payload.type, provider_name: payload.name, plugin_id: payload.plugin_id, plugin_unique_identifier: payload.plugin_unique_identifier, provider_icon: normalizedIcon, provider_icon_dark: normalizedIconDark, tool_name: tool.name, tool_label: tool.label[language], tool_description: tool.description[language], title: tool.label[language], is_team_authorization: payload.is_team_authorization, paramSchemas: tool.parameters, params, }) }} >
{notShowProvider ? actions[0]?.label[language] : payload.label[language]} {isFlatView && groupName && ( {groupName} )} {isMCPTool && }
{!isShowCanNotChooseMCPTip && !canNotSelectMultiple && (notShowProvider ? notShowProviderSelectInfo : selectedInfo)} {isShowCanNotChooseMCPTip && } {hasAction && ( )}
{!notShowProvider && hasAction && !isFold && ( actions.map(action => ( )) )}
) } export default React.memo(Tool)