mcp server card switcher

This commit is contained in:
jZonG 2025-05-27 15:38:37 +08:00
parent a77e7ab177
commit 01d4768d2f
4 changed files with 97 additions and 51 deletions

View File

@ -60,7 +60,7 @@ const MCPServerModal = ({
</div> </div>
<div> <div>
<div className='mb-1 flex items-center gap-2'> <div className='mb-1 flex items-center gap-2'>
<div className='system-xs-medium-uppercase text-text-primary'>{t('tools.mcp.server.modal.parameters')}</div> <div className='system-xs-medium-uppercase shrink-0 text-text-primary'>{t('tools.mcp.server.modal.parameters')}</div>
<Divider type='horizontal' className='!m-0 !h-px grow bg-divider-subtle' /> <Divider type='horizontal' className='!m-0 !h-px grow bg-divider-subtle' />
</div> </div>
<div className='body-xs-regular mb-2 text-text-tertiary'>{t('tools.mcp.server.modal.parametersTip')}</div> <div className='body-xs-regular mb-2 text-text-tertiary'>{t('tools.mcp.server.modal.parametersTip')}</div>

View File

@ -1,5 +1,5 @@
'use client' 'use client'
import React, { useState } from 'react' import React, { useEffect, useState } from 'react'
import { useTranslation } from 'react-i18next' import { useTranslation } from 'react-i18next'
import { import {
RiLoopLeftLine, RiLoopLeftLine,
@ -10,7 +10,6 @@ import {
import Button from '@/app/components/base/button' import Button from '@/app/components/base/button'
import Tooltip from '@/app/components/base/tooltip' import Tooltip from '@/app/components/base/tooltip'
import { asyncRunSafe } from '@/utils' import { asyncRunSafe } from '@/utils'
import { basePath } from '@/utils/var'
import Switch from '@/app/components/base/switch' import Switch from '@/app/components/base/switch'
import Divider from '@/app/components/base/divider' import Divider from '@/app/components/base/divider'
import CopyFeedback from '@/app/components/base/copy-feedback' import CopyFeedback from '@/app/components/base/copy-feedback'
@ -21,6 +20,9 @@ import type { AppSSO } from '@/types/app'
import Indicator from '@/app/components/header/indicator' import Indicator from '@/app/components/header/indicator'
import MCPServerModal from '@/app/components/tools/mcp/mcp-server-modal' import MCPServerModal from '@/app/components/tools/mcp/mcp-server-modal'
import { useAppWorkflow } from '@/service/use-workflow' import { useAppWorkflow } from '@/service/use-workflow'
import {
useMCPServerDetail,
} from '@/service/use-tools'
import cn from '@/utils/classnames' import cn from '@/utils/classnames'
export type IAppCardProps = { export type IAppCardProps = {
@ -36,14 +38,20 @@ function MCPServiceCard({
const { isCurrentWorkspaceManager, isCurrentWorkspaceEditor } = useAppContext() const { isCurrentWorkspaceManager, isCurrentWorkspaceEditor } = useAppContext()
const [genLoading, setGenLoading] = useState(false) const [genLoading, setGenLoading] = useState(false)
const [showConfirmDelete, setShowConfirmDelete] = useState(false) const [showConfirmDelete, setShowConfirmDelete] = useState(false)
const [showMCPServerModal, setShowMCPServerModal] = useState(false)
const { data: currentWorkflow } = useAppWorkflow(appInfo.id) const { data: currentWorkflow } = useAppWorkflow(appInfo.id)
const { data: detail } = useMCPServerDetail(appInfo.id)
const { id, status, server_code } = detail ?? {}
const toggleDisabled = !isCurrentWorkspaceEditor || !currentWorkflow?.graph const appUnpublished = !currentWorkflow?.graph
const runningStatus = appInfo.enable_site // TODO const serverPublished = !!id
const { app_base_url, access_token } = appInfo.site ?? {} const serverActivated = status === 'active'
const appMode = (appInfo.mode !== 'completion' && appInfo.mode !== 'workflow') ? 'chat' : appInfo.mode const serverURL = serverPublished ? `${globalThis.location.protocol}//${globalThis.location.host}/api/server/${server_code}/mcp` : '***********'
const appUrl = `${app_base_url}${basePath}/${appMode}/${access_token}`
const toggleDisabled = !isCurrentWorkspaceEditor || appUnpublished
const [activated, setActivated] = useState(serverActivated)
const onGenCode = async () => { const onGenCode = async () => {
if (onGenerateCode) { if (onGenerateCode) {
@ -53,17 +61,36 @@ function MCPServiceCard({
} }
} }
const onChangeStatus = async (status: boolean) => { const onChangeStatus = async (state: boolean) => {
// TODO if (state) {
if (!serverPublished) {
setActivated(true)
setShowMCPServerModal(true)
}
// TODO handle server activation
}
else {
// TODO handle server activation
}
} }
const [showMCPServerModal, setShowMCPServerModal] = useState(false) const handleServerModalHide = () => {
setShowMCPServerModal(false)
if (!serverActivated)
setActivated(false)
}
const handleServerModalConfirm = () => {
setShowMCPServerModal(false)
}
useEffect(() => {
setActivated(serverActivated)
}, [serverActivated])
if (!currentWorkflow) if (!currentWorkflow)
return null return null
// TODO: show disabled state if workflow not published
return ( return (
<> <>
<div className={cn('w-full max-w-full rounded-xl border-l-[0.5px] border-t border-effects-highlight')}> <div className={cn('w-full max-w-full rounded-xl border-l-[0.5px] border-t border-effects-highlight')}>
@ -81,14 +108,20 @@ function MCPServiceCard({
</div> </div>
</div> </div>
<div className='flex items-center gap-1'> <div className='flex items-center gap-1'>
<Indicator color={runningStatus ? 'green' : 'yellow'} /> <Indicator color={serverActivated ? 'green' : 'yellow'} />
<div className={`${runningStatus ? 'text-text-success' : 'text-text-warning'} system-xs-semibold-uppercase`}> <div className={`${serverActivated ? 'text-text-success' : 'text-text-warning'} system-xs-semibold-uppercase`}>
{runningStatus {serverActivated
? t('appOverview.overview.status.running') ? t('appOverview.overview.status.running')
: t('appOverview.overview.status.disable')} : t('appOverview.overview.status.disable')}
</div> </div>
</div> </div>
<Switch defaultValue={runningStatus} onChange={onChangeStatus} disabled={toggleDisabled} /> <Tooltip
popupContent={appUnpublished ? t('tools.mcp.server.publishTip') : ''}
>
<div>
<Switch defaultValue={activated} onChange={onChangeStatus} disabled={toggleDisabled} />
</div>
</Tooltip>
</div> </div>
<div className='flex flex-col items-start justify-center self-stretch'> <div className='flex flex-col items-start justify-center self-stretch'>
<div className="system-xs-medium pb-1 text-text-tertiary"> <div className="system-xs-medium pb-1 text-text-tertiary">
@ -97,53 +130,64 @@ function MCPServiceCard({
<div className="inline-flex h-9 w-full items-center gap-0.5 rounded-lg bg-components-input-bg-normal p-1 pl-2"> <div className="inline-flex h-9 w-full items-center gap-0.5 rounded-lg bg-components-input-bg-normal p-1 pl-2">
<div className="flex h-4 min-w-0 flex-1 items-start justify-start gap-2 px-1"> <div className="flex h-4 min-w-0 flex-1 items-start justify-start gap-2 px-1">
<div className="overflow-hidden text-ellipsis whitespace-nowrap text-xs font-medium text-text-secondary"> <div className="overflow-hidden text-ellipsis whitespace-nowrap text-xs font-medium text-text-secondary">
{appUrl} {serverURL}
</div> </div>
</div> </div>
<CopyFeedback {serverPublished && (
content={appUrl} <>
className={'!size-6'} <CopyFeedback
/> content={serverURL}
<Divider type="vertical" className="!mx-0.5 !h-3.5 shrink-0" /> className={'!size-6'}
{/* button copy link/ button regenerate */} />
{showConfirmDelete && ( <Divider type="vertical" className="!mx-0.5 !h-3.5 shrink-0" />
<Confirm {isCurrentWorkspaceManager && (
type='warning' <Tooltip
title={t('appOverview.overview.appInfo.regenerate')} popupContent={t('appOverview.overview.appInfo.regenerate') || ''}
content={t('tools.mcp.server.reGen')} >
isShow={showConfirmDelete} <div
onConfirm={() => { className="cursor-pointer rounded-md p-1 hover:bg-state-base-hover"
onGenCode() onClick={() => setShowConfirmDelete(true)}
setShowConfirmDelete(false) >
}} <RiLoopLeftLine className={cn('h-4 w-4 text-text-tertiary hover:text-text-secondary', genLoading && 'animate-spin')}/>
onCancel={() => setShowConfirmDelete(false)} </div>
/> </Tooltip>
)} )}
{isCurrentWorkspaceManager && ( </>
<Tooltip
popupContent={t('appOverview.overview.appInfo.regenerate') || ''}
>
<div
className="cursor-pointer rounded-md p-1 hover:bg-state-base-hover"
onClick={() => setShowConfirmDelete(true)}
>
<RiLoopLeftLine className={cn('h-4 w-4 text-text-tertiary hover:text-text-secondary', genLoading && 'animate-spin')}/>
</div>
</Tooltip>
)} )}
</div> </div>
</div> </div>
</div> </div>
<div className='flex items-center gap-1 self-stretch p-3'> <div className='flex items-center gap-1 self-stretch p-3'>
<Button disabled={toggleDisabled} size='small' variant='ghost'>{t('tools.mcp.server.addDescription')}</Button> <Button
disabled={toggleDisabled}
size='small'
variant='ghost'
onClick={() => setShowMCPServerModal(true)}
>
{serverPublished ? t('tools.mcp.server.editDescription') : t('tools.mcp.server.addDescription')}
</Button>
</div> </div>
</div> </div>
</div> </div>
{showMCPServerModal && ( {showMCPServerModal && (
<MCPServerModal <MCPServerModal
show={showMCPServerModal} show={showMCPServerModal}
onConfirm={() => setShowMCPServerModal(false)} onConfirm={handleServerModalConfirm}
onHide={() => setShowMCPServerModal(false)} onHide={handleServerModalHide}
/>
)}
{/* button copy link/ button regenerate */}
{showConfirmDelete && (
<Confirm
type='warning'
title={t('appOverview.overview.appInfo.regenerate')}
content={t('tools.mcp.server.reGen')}
isShow={showConfirmDelete}
onConfirm={() => {
onGenCode()
setShowConfirmDelete(false)
}}
onCancel={() => setShowConfirmDelete(false)}
/> />
)} )}
</> </>

View File

@ -206,6 +206,7 @@ const translation = {
parametersPlaceholder: 'Parameter purpose and constraints', parametersPlaceholder: 'Parameter purpose and constraints',
confirm: 'Enable MCP Server', confirm: 'Enable MCP Server',
}, },
publishTip: 'App not published. Please publish the app first.',
}, },
}, },
} }

View File

@ -206,6 +206,7 @@ const translation = {
parametersPlaceholder: '参数的用途和约束条件', parametersPlaceholder: '参数的用途和约束条件',
confirm: '启用 MCP 服务', confirm: '启用 MCP 服务',
}, },
publishTip: '应用未发布。请先发布应用。',
}, },
}, },
} }