您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

mcp-server-modal.tsx 4.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. 'use client'
  2. import React from 'react'
  3. import { useTranslation } from 'react-i18next'
  4. import { RiCloseLine } from '@remixicon/react'
  5. import Modal from '@/app/components/base/modal'
  6. import Button from '@/app/components/base/button'
  7. import Textarea from '@/app/components/base/textarea'
  8. import Divider from '@/app/components/base/divider'
  9. import MCPServerParamItem from '@/app/components/tools/mcp/mcp-server-param-item'
  10. import type {
  11. MCPServerDetail,
  12. } from '@/app/components/tools/types'
  13. import {
  14. useCreateMCPServer,
  15. useInvalidateMCPServerDetail,
  16. useUpdateMCPServer,
  17. } from '@/service/use-tools'
  18. import cn from '@/utils/classnames'
  19. export type ModalProps = {
  20. appID: string
  21. latestParams?: any[]
  22. data?: MCPServerDetail
  23. show: boolean
  24. onHide: () => void
  25. }
  26. const MCPServerModal = ({
  27. appID,
  28. latestParams = [],
  29. data,
  30. show,
  31. onHide,
  32. }: ModalProps) => {
  33. const { t } = useTranslation()
  34. const { mutateAsync: createMCPServer, isPending: creating } = useCreateMCPServer()
  35. const { mutateAsync: updateMCPServer, isPending: updating } = useUpdateMCPServer()
  36. const invalidateMCPServerDetail = useInvalidateMCPServerDetail()
  37. const [description, setDescription] = React.useState(data?.description || '')
  38. const [params, setParams] = React.useState(data?.parameters || {})
  39. const handleParamChange = (variable: string, value: string) => {
  40. setParams(prev => ({
  41. ...prev,
  42. [variable]: value,
  43. }))
  44. }
  45. const getParamValue = () => {
  46. const res = {} as any
  47. latestParams.map((param) => {
  48. res[param.variable] = params[param.variable]
  49. return param
  50. })
  51. return res
  52. }
  53. const submit = async () => {
  54. if (!data) {
  55. await createMCPServer({
  56. appID,
  57. description,
  58. parameters: getParamValue(),
  59. })
  60. invalidateMCPServerDetail(appID)
  61. onHide()
  62. }
  63. else {
  64. await updateMCPServer({
  65. appID,
  66. id: data.id,
  67. description,
  68. parameters: getParamValue(),
  69. })
  70. invalidateMCPServerDetail(appID)
  71. onHide()
  72. }
  73. }
  74. return (
  75. <Modal
  76. isShow={show}
  77. onClose={onHide}
  78. className={cn('relative !max-w-[520px] !p-0')}
  79. >
  80. <div className='absolute right-5 top-5 z-10 cursor-pointer p-1.5' onClick={onHide}>
  81. <RiCloseLine className='h-5 w-5 text-text-tertiary' />
  82. </div>
  83. <div className='title-2xl-semi-bold relative p-6 pb-3 text-xl text-text-primary'>
  84. {!data ? t('tools.mcp.server.modal.addTitle') : t('tools.mcp.server.modal.editTitle')}
  85. </div>
  86. <div className='space-y-5 px-6 py-3'>
  87. <div className='space-y-0.5'>
  88. <div className='flex h-6 items-center gap-1'>
  89. <div className='system-sm-medium text-text-secondary'>{t('tools.mcp.server.modal.description')}</div>
  90. <div className='system-xs-regular text-text-destructive-secondary'>*</div>
  91. </div>
  92. <Textarea
  93. className='h-[96px] resize-none'
  94. value={description}
  95. placeholder={t('tools.mcp.server.modal.descriptionPlaceholder')}
  96. onChange={e => setDescription(e.target.value)}
  97. ></Textarea>
  98. </div>
  99. {latestParams.length > 0 && (
  100. <div>
  101. <div className='mb-1 flex items-center gap-2'>
  102. <div className='system-xs-medium-uppercase shrink-0 text-text-primary'>{t('tools.mcp.server.modal.parameters')}</div>
  103. <Divider type='horizontal' className='!m-0 !h-px grow bg-divider-subtle' />
  104. </div>
  105. <div className='body-xs-regular mb-2 text-text-tertiary'>{t('tools.mcp.server.modal.parametersTip')}</div>
  106. <div className='space-y-3'>
  107. {latestParams.map(paramItem => (
  108. <MCPServerParamItem
  109. key={paramItem.variable}
  110. data={paramItem}
  111. value={params[paramItem.variable] || ''}
  112. onChange={value => handleParamChange(paramItem.variable, value)}
  113. />
  114. ))}
  115. </div>
  116. </div>
  117. )}
  118. </div>
  119. <div className='flex flex-row-reverse p-6 pt-5'>
  120. <Button disabled={!description || creating || updating} className='ml-2' variant='primary' onClick={submit}>{data ? t('tools.mcp.modal.save') : t('tools.mcp.server.modal.confirm')}</Button>
  121. <Button onClick={onHide}>{t('tools.mcp.modal.cancel')}</Button>
  122. </div>
  123. </Modal>
  124. )
  125. }
  126. export default MCPServerModal