| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 | import {  memo,  useCallback,} from 'react'import { useTranslation } from 'react-i18next'import {  RiAddLine,} from '@remixicon/react'import cn from '@/utils/classnames'import { ArrowUpRight } from '@/app/components/base/icons/src/vender/line/arrows'import { Check } from '@/app/components/base/icons/src/vender/line/general'import { Tag01 } from '@/app/components/base/icons/src/vender/line/financeAndECommerce'import type { ToolWithProvider } from '@/app/components/workflow/types'import { BlockEnum } from '@/app/components/workflow/types'import BlockIcon from '@/app/components/workflow/block-icon'import Tooltip from '@/app/components/base/tooltip'import Button from '@/app/components/base/button'import { useGetLanguage } from '@/context/i18n'import { useStore as useLabelStore } from '@/app/components/tools/labels/store'import Empty from '@/app/components/tools/add-tool-modal/empty'import type { Tool } from '@/app/components/tools/types'import { CollectionType } from '@/app/components/tools/types'import type { AgentTool } from '@/types/app'import { MAX_TOOLS_NUM } from '@/config'type ToolsProps = {  showWorkflowEmpty: boolean  tools: ToolWithProvider[]  addedTools: AgentTool[]  onSelect: (provider: ToolWithProvider, tool: Tool) => void  onAuthSetup: (provider: ToolWithProvider) => void}const Blocks = ({  showWorkflowEmpty,  tools,  addedTools,  onSelect,  onAuthSetup,}: ToolsProps) => {  const { t } = useTranslation()  const language = useGetLanguage()  const labelList = useLabelStore(s => s.labelList)  const addable = addedTools.length < MAX_TOOLS_NUM  const renderGroup = useCallback((toolWithProvider: ToolWithProvider) => {    const list = toolWithProvider.tools    const needAuth = toolWithProvider.allow_delete && !toolWithProvider.is_team_authorization && toolWithProvider.type === CollectionType.builtIn    return (      <div        key={toolWithProvider.id}        className='group mb-1 last-of-type:mb-0'      >        <div className='flex h-[22px] w-full items-center justify-between pl-3 pr-1 text-xs font-medium text-gray-500'>          {toolWithProvider.label[language]}          <a className='hidden cursor-pointer items-center group-hover:flex' href={`/tools?category=${toolWithProvider.type}`} target='_blank'>{t('tools.addToolModal.manageInTools')}<ArrowUpRight className='ml-0.5 h-3 w-3' /></a>        </div>        {list.map((tool) => {          const labelContent = (() => {            if (!tool.labels)              return ''            return tool.labels.map((name) => {              const label = labelList.find(item => item.name === name)              return label?.label[language]            }).filter(Boolean).join(', ')          })()          const added = !!addedTools?.find(v => v.provider_id === toolWithProvider.id && v.provider_type === toolWithProvider.type && v.tool_name === tool.name)          return (            <Tooltip              key={tool.name}              position='bottom'              popupClassName='!p-0 !px-3 !py-2.5 !w-[210px] !leading-[18px] !text-xs !text-gray-700 !border-[0.5px] !border-black/5 !bg-transparent !rounded-xl !shadow-lg translate-x-[108px]'              popupContent={(                <div>                  <BlockIcon                    size='md'                    className='mb-2'                    type={BlockEnum.Tool}                    toolIcon={toolWithProvider.icon}                  />                  <div className='mb-1 text-sm leading-5 text-gray-900'>{tool.label[language]}</div>                  <div className='text-xs leading-[18px] text-gray-700'>{tool.description[language]}</div>                  {tool.labels?.length > 0 && (                    <div className='mt-1 flex shrink-0 items-center'>                      <div className='relative flex w-full items-center gap-1 rounded-md py-1 text-gray-500' title={labelContent}>                        <Tag01 className='h-3 w-3 shrink-0 text-gray-500' />                        <div className='grow truncate text-start text-xs font-normal leading-[18px]'>{labelContent}</div>                      </div>                    </div>                  )}                </div>              )}            >              <div className='group/item flex h-8 w-full cursor-pointer items-center rounded-lg pl-3 pr-1 hover:bg-gray-50'>                <BlockIcon                  className={cn('mr-2 shrink-0', needAuth && 'opacity-30')}                  type={BlockEnum.Tool}                  toolIcon={toolWithProvider.icon}                />                <div className={cn('grow truncate text-sm text-gray-900', needAuth && 'opacity-30')}>{tool.label[language]}</div>                {!needAuth && added && (                  <div className='flex items-center gap-1 rounded-[6px] border border-gray-100 bg-white px-2 py-[3px] text-xs font-medium leading-[18px] text-gray-300'>                    <Check className='h-3 w-3' />                    {t('tools.addToolModal.added').toLocaleUpperCase()}                  </div>                )}                {!needAuth && !added && addable && (                  <Button                    variant='secondary-accent'                    size='small'                    className={cn('hidden shrink-0 items-center group-hover/item:flex')}                    onClick={() => onSelect(toolWithProvider, tool)}                  >                    <RiAddLine className='h-3 w-3' />                    {t('tools.addToolModal.add').toLocaleUpperCase()}                  </Button>                )}                {needAuth && (                  <Button                    variant='secondary-accent'                    size='small'                    className={cn('hidden shrink-0 group-hover/item:flex')}                    onClick={() => onAuthSetup(toolWithProvider)}                  >{t('tools.auth.setup')}</Button>                )}              </div>            </Tooltip>          )        })}      </div>    )  }, [addable, language, t, labelList, addedTools, onAuthSetup, onSelect])  return (    <div className='max-w-[440px] p-1 pb-6'>      {!tools.length && !showWorkflowEmpty && (        <div className='flex h-[22px] items-center px-3 text-xs font-medium text-gray-500'>{t('workflow.tabs.noResult')}</div>      )}      {!tools.length && showWorkflowEmpty && (        <div className='pt-[280px]'>          <Empty />        </div>      )}      {!!tools.length && tools.map(renderGroup)}    </div>  )}export default memo(Blocks)
 |