index.tsx 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. import { useEffect, useState } from 'react'
  2. import { useTranslation } from 'react-i18next'
  3. import Link from 'next/link'
  4. import { ArrowTopRightOnSquareIcon } from '@heroicons/react/24/outline'
  5. import ProviderInput from '../provider-input'
  6. import type { ValidatedStatusState } from '../provider-input/useValidateToken'
  7. import useValidateToken, { ValidatedStatus } from '../provider-input/useValidateToken'
  8. import {
  9. ValidatedErrorIcon,
  10. ValidatedErrorOnOpenaiTip,
  11. ValidatedSuccessIcon,
  12. ValidatingTip,
  13. } from '../provider-input/Validate'
  14. import type { Provider } from '@/models/common'
  15. type IOpenaiProviderProps = {
  16. provider: Provider
  17. onValidatedStatus: (status?: ValidatedStatusState) => void
  18. onTokenChange: (token: string) => void
  19. }
  20. const OpenaiProvider = ({
  21. provider,
  22. onValidatedStatus,
  23. onTokenChange,
  24. }: IOpenaiProviderProps) => {
  25. const { t } = useTranslation()
  26. const [token, setToken] = useState(provider.token as string || '')
  27. const [validating, validatedStatus, setValidatedStatus, validate] = useValidateToken(provider.provider_name)
  28. const handleFocus = () => {
  29. if (token === provider.token) {
  30. setToken('')
  31. onTokenChange('')
  32. setValidatedStatus({})
  33. }
  34. }
  35. const handleChange = (v: string) => {
  36. setToken(v)
  37. onTokenChange(v)
  38. validate(v, {
  39. beforeValidating: () => {
  40. if (!v) {
  41. setValidatedStatus({})
  42. return false
  43. }
  44. return true
  45. },
  46. })
  47. }
  48. useEffect(() => {
  49. if (typeof onValidatedStatus === 'function')
  50. onValidatedStatus(validatedStatus)
  51. }, [validatedStatus])
  52. const getValidatedIcon = () => {
  53. if (validatedStatus?.status === ValidatedStatus.Error || validatedStatus.status === ValidatedStatus.Exceed)
  54. return <ValidatedErrorIcon />
  55. if (validatedStatus.status === ValidatedStatus.Success)
  56. return <ValidatedSuccessIcon />
  57. }
  58. const getValidatedTip = () => {
  59. if (validating)
  60. return <ValidatingTip />
  61. if (validatedStatus?.status === ValidatedStatus.Error)
  62. return <ValidatedErrorOnOpenaiTip errorMessage={validatedStatus.message ?? ''} />
  63. }
  64. return (
  65. <div className='px-4 pt-3 pb-4'>
  66. <ProviderInput
  67. value={token}
  68. name={t('common.provider.apiKey')}
  69. placeholder={t('common.provider.enterYourKey')}
  70. onChange={handleChange}
  71. onFocus={handleFocus}
  72. validatedIcon={getValidatedIcon()}
  73. validatedTip={getValidatedTip()}
  74. />
  75. <Link className="inline-flex items-center mt-3 text-xs font-normal cursor-pointer text-primary-600 w-fit" href="https://platform.openai.com/account/api-keys" target={'_blank'}>
  76. {t('appOverview.welcome.getKeyTip')}
  77. <ArrowTopRightOnSquareIcon className='w-3 h-3 ml-1 text-primary-600' aria-hidden="true" />
  78. </Link>
  79. </div>
  80. )
  81. }
  82. export default OpenaiProvider