You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

config-credentials.tsx 6.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React from 'react'
  4. import { useTranslation } from 'react-i18next'
  5. import Tooltip from '@/app/components/base/tooltip'
  6. import cn from '@/utils/classnames'
  7. import type { Credential } from '@/app/components/tools/types'
  8. import Input from '@/app/components/base/input'
  9. import Drawer from '@/app/components/base/drawer-plus'
  10. import Button from '@/app/components/base/button'
  11. import Radio from '@/app/components/base/radio/ui'
  12. import { AuthHeaderPrefix, AuthType } from '@/app/components/tools/types'
  13. type Props = {
  14. positionCenter?: boolean
  15. credential: Credential
  16. onChange: (credential: Credential) => void
  17. onHide: () => void
  18. }
  19. type ItemProps = {
  20. text: string
  21. value: AuthType | AuthHeaderPrefix
  22. isChecked: boolean
  23. onClick: (value: AuthType | AuthHeaderPrefix) => void
  24. }
  25. const SelectItem: FC<ItemProps> = ({ text, value, isChecked, onClick }) => {
  26. return (
  27. <div
  28. className={cn(isChecked ? 'border-[2px] border-util-colors-indigo-indigo-600 bg-components-panel-on-panel-item-bg shadow-sm' : 'border border-components-card-border', 'mb-2 flex h-9 w-[150px] cursor-pointer items-center space-x-2 rounded-xl bg-components-panel-on-panel-item-bg pl-3 hover:bg-components-panel-on-panel-item-bg-hover')}
  29. onClick={() => onClick(value)}
  30. >
  31. <Radio isChecked={isChecked} />
  32. <div className='system-sm-regular text-text-primary'>{text}</div>
  33. </div>
  34. )
  35. }
  36. const ConfigCredential: FC<Props> = ({
  37. positionCenter,
  38. credential,
  39. onChange,
  40. onHide,
  41. }) => {
  42. const { t } = useTranslation()
  43. const [tempCredential, setTempCredential] = React.useState<Credential>(credential)
  44. return (
  45. <Drawer
  46. isShow
  47. positionCenter={positionCenter}
  48. onHide={onHide}
  49. title={t('tools.createTool.authMethod.title')!}
  50. dialogClassName='z-[60]'
  51. dialogBackdropClassName='z-[70]'
  52. panelClassName='mt-2 !w-[520px] h-fit z-[80]'
  53. maxWidthClassName='!max-w-[520px]'
  54. height={'fit-content'}
  55. headerClassName='!border-b-divider-regular'
  56. body={
  57. <div className='px-6 pt-2'>
  58. <div className='space-y-4'>
  59. <div>
  60. <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.authMethod.type')}</div>
  61. <div className='flex space-x-3'>
  62. <SelectItem
  63. text={t('tools.createTool.authMethod.types.none')}
  64. value={AuthType.none}
  65. isChecked={tempCredential.auth_type === AuthType.none}
  66. onClick={value => setTempCredential({ ...tempCredential, auth_type: value as AuthType })}
  67. />
  68. <SelectItem
  69. text={t('tools.createTool.authMethod.types.api_key')}
  70. value={AuthType.apiKey}
  71. isChecked={tempCredential.auth_type === AuthType.apiKey}
  72. onClick={value => setTempCredential({
  73. ...tempCredential,
  74. auth_type: value as AuthType,
  75. api_key_header: tempCredential.api_key_header || 'Authorization',
  76. api_key_value: tempCredential.api_key_value || '',
  77. api_key_header_prefix: tempCredential.api_key_header_prefix || AuthHeaderPrefix.custom,
  78. })}
  79. />
  80. </div>
  81. </div>
  82. {tempCredential.auth_type === AuthType.apiKey && (
  83. <>
  84. <div>
  85. <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.authHeaderPrefix.title')}</div>
  86. <div className='flex space-x-3'>
  87. <SelectItem
  88. text={t('tools.createTool.authHeaderPrefix.types.basic')}
  89. value={AuthHeaderPrefix.basic}
  90. isChecked={tempCredential.api_key_header_prefix === AuthHeaderPrefix.basic}
  91. onClick={value => setTempCredential({ ...tempCredential, api_key_header_prefix: value as AuthHeaderPrefix })}
  92. />
  93. <SelectItem
  94. text={t('tools.createTool.authHeaderPrefix.types.bearer')}
  95. value={AuthHeaderPrefix.bearer}
  96. isChecked={tempCredential.api_key_header_prefix === AuthHeaderPrefix.bearer}
  97. onClick={value => setTempCredential({ ...tempCredential, api_key_header_prefix: value as AuthHeaderPrefix })}
  98. />
  99. <SelectItem
  100. text={t('tools.createTool.authHeaderPrefix.types.custom')}
  101. value={AuthHeaderPrefix.custom}
  102. isChecked={tempCredential.api_key_header_prefix === AuthHeaderPrefix.custom}
  103. onClick={value => setTempCredential({ ...tempCredential, api_key_header_prefix: value as AuthHeaderPrefix })}
  104. />
  105. </div>
  106. </div>
  107. <div>
  108. <div className='system-sm-medium flex items-center py-2 text-text-primary'>
  109. {t('tools.createTool.authMethod.key')}
  110. <Tooltip
  111. popupContent={
  112. <div className='w-[261px] text-text-tertiary'>
  113. {t('tools.createTool.authMethod.keyTooltip')}
  114. </div>
  115. }
  116. triggerClassName='ml-0.5 w-4 h-4'
  117. />
  118. </div>
  119. <Input
  120. value={tempCredential.api_key_header}
  121. onChange={e => setTempCredential({ ...tempCredential, api_key_header: e.target.value })}
  122. placeholder={t('tools.createTool.authMethod.types.apiKeyPlaceholder')!}
  123. />
  124. </div>
  125. <div>
  126. <div className='system-sm-medium py-2 text-text-primary'>{t('tools.createTool.authMethod.value')}</div>
  127. <Input
  128. value={tempCredential.api_key_value}
  129. onChange={e => setTempCredential({ ...tempCredential, api_key_value: e.target.value })}
  130. placeholder={t('tools.createTool.authMethod.types.apiValuePlaceholder')!}
  131. />
  132. </div>
  133. </>)}
  134. </div>
  135. <div className='mt-4 flex shrink-0 justify-end space-x-2 py-4'>
  136. <Button onClick={onHide}>{t('common.operation.cancel')}</Button>
  137. <Button variant='primary' onClick={() => {
  138. onChange(tempCredential)
  139. onHide()
  140. }}>{t('common.operation.save')}</Button>
  141. </div>
  142. </div>
  143. }
  144. />
  145. )
  146. }
  147. export default React.memo(ConfigCredential)