| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112 | 'use client'import type { FC } from 'react'import React, { useEffect } from 'react'import { useTranslation } from 'react-i18next'import Item from './item'import FeaturePanel from '@/app/components/app/configuration/base/feature-panel'import { Google, WebReader, Wikipedia } from '@/app/components/base/icons/src/public/plugins'import { getToolProviders } from '@/service/explore'import Loading from '@/app/components/base/loading'import AccountSetting from '@/app/components/header/account-setting'export type IPluginsProps = {  readonly?: boolean  config: Record<string, boolean>  onChange?: (key: string, value: boolean) => void}const plugins = [  { key: 'google_search', icon: <Google /> },  { key: 'web_reader', icon: <WebReader /> },  { key: 'wikipedia', icon: <Wikipedia /> },] as constconst Plugins: FC<IPluginsProps> = ({  readonly,  config,  onChange,}) => {  const { t } = useTranslation()  const [isLoading, setIsLoading] = React.useState(!readonly)  const [isSerpApiValid, setIsSerpApiValid] = React.useState(false)  const checkSerpApiKey = async () => {    if (readonly)      return    const provides: any = await getToolProviders()    const isSerpApiValid = !!provides.find((v: any) => v.tool_name === 'serpapi' && v.is_enabled)    setIsSerpApiValid(isSerpApiValid)    setIsLoading(false)  }  useEffect(() => {    checkSerpApiKey()  }, [])  const [showSetSerpAPIKeyModal, setShowSetAPIKeyModal] = React.useState(false)  const itemConfigs = plugins.map((plugin) => {    const res: Record<string, any> = { ...plugin }    const { key } = plugin    res.name = t(`explore.universalChat.plugins.${key}.name`)    if (key === 'web_reader')      res.description = t(`explore.universalChat.plugins.${key}.description`)    if (key === 'google_search' && !isSerpApiValid && !readonly) {      res.readonly = true      res.more = (        <div className='border-t border-[#FEF0C7] flex items-center h-[34px] pl-2 bg-[#FFFAEB] text-gray-700 text-xs '>          <span className='whitespace-pre'>{t('explore.universalChat.plugins.google_search.more.left')}</span>          <span className='cursor-pointer text-[#155EEF]' onClick={() => setShowSetAPIKeyModal(true)}>{t('explore.universalChat.plugins.google_search.more.link')}</span>          <span className='whitespace-pre'>{t('explore.universalChat.plugins.google_search.more.right')}</span>        </div>      )    }    return res  })  const enabledPluginNum = Object.values(config).filter(v => v).length  return (    <>      <FeaturePanel        className='mt-3'        title={          <div className='flex space-x-1'>            <div>{t('explore.universalChat.plugins.name')}</div>            <div className='text-[13px] font-normal text-gray-500'>({enabledPluginNum}/{plugins.length})</div>          </div>}        hasHeaderBottomBorder={false}      >        {isLoading          ? (            <div className='flex items-center h-[166px]'>              <Loading type='area' />            </div>          )          : (<div className='space-y-2'>            {itemConfigs.map(item => (              <Item                key={item.key}                icon={item.icon}                name={item.name}                description={item.description}                more={item.more}                enabled={config[item.key]}                onChange={enabled => onChange?.(item.key, enabled)}                readonly={readonly || item.readonly}              />            ))}          </div>)}      </FeaturePanel>      {        showSetSerpAPIKeyModal && (          <AccountSetting activeTab="plugin" onCancel={async () => {            setShowSetAPIKeyModal(false)            await checkSerpApiKey()          }} />        )      }    </>  )}export default React.memo(Plugins)
 |