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.

index.tsx 6.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useEffect, useState } from 'react'
  4. import { ChevronRightIcon } from '@heroicons/react/20/solid'
  5. import Link from 'next/link'
  6. import { Trans, useTranslation } from 'react-i18next'
  7. import s from './style.module.css'
  8. import Modal from '@/app/components/base/modal'
  9. import Button from '@/app/components/base/button'
  10. import AppIcon from '@/app/components/base/app-icon'
  11. import { SimpleSelect } from '@/app/components/base/select'
  12. import type { AppDetailResponse } from '@/models/app'
  13. import type { Language } from '@/types/app'
  14. import EmojiPicker from '@/app/components/base/emoji-picker'
  15. import { languages } from '@/utils/language'
  16. export type ISettingsModalProps = {
  17. appInfo: AppDetailResponse
  18. isShow: boolean
  19. defaultValue?: string
  20. onClose: () => void
  21. onSave?: (params: ConfigParams) => Promise<void>
  22. }
  23. export type ConfigParams = {
  24. title: string
  25. description: string
  26. default_language: string
  27. prompt_public: boolean
  28. copyright: string
  29. privacy_policy: string
  30. icon: string
  31. icon_background: string
  32. }
  33. const prefixSettings = 'appOverview.overview.appInfo.settings'
  34. const SettingsModal: FC<ISettingsModalProps> = ({
  35. appInfo,
  36. isShow = false,
  37. onClose,
  38. onSave,
  39. }) => {
  40. const [isShowMore, setIsShowMore] = useState(false)
  41. const { icon, icon_background } = appInfo
  42. const { title, description, copyright, privacy_policy, default_language } = appInfo.site
  43. const [inputInfo, setInputInfo] = useState({ title, desc: description, copyright, privacyPolicy: privacy_policy })
  44. const [language, setLanguage] = useState(default_language)
  45. const [saveLoading, setSaveLoading] = useState(false)
  46. const { t } = useTranslation()
  47. // Emoji Picker
  48. const [showEmojiPicker, setShowEmojiPicker] = useState(false)
  49. const [emoji, setEmoji] = useState({ icon, icon_background })
  50. useEffect(() => {
  51. setInputInfo({ title, desc: description, copyright, privacyPolicy: privacy_policy })
  52. setLanguage(default_language)
  53. setEmoji({ icon, icon_background })
  54. }, [appInfo])
  55. const onHide = () => {
  56. onClose()
  57. setTimeout(() => {
  58. setIsShowMore(false)
  59. }, 200)
  60. }
  61. const onClickSave = async () => {
  62. setSaveLoading(true)
  63. const params = {
  64. title: inputInfo.title,
  65. description: inputInfo.desc,
  66. default_language: language,
  67. prompt_public: false,
  68. copyright: inputInfo.copyright,
  69. privacy_policy: inputInfo.privacyPolicy,
  70. icon: emoji.icon,
  71. icon_background: emoji.icon_background,
  72. }
  73. await onSave?.(params)
  74. setSaveLoading(false)
  75. onHide()
  76. }
  77. const onChange = (field: string) => {
  78. return (e: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) => {
  79. setInputInfo(item => ({ ...item, [field]: e.target.value }))
  80. }
  81. }
  82. return (
  83. <>
  84. <Modal
  85. title={t(`${prefixSettings}.title`)}
  86. isShow={isShow}
  87. onClose={onHide}
  88. className={`${s.settingsModal}`}
  89. >
  90. <div className={`mt-6 font-medium ${s.settingTitle} text-gray-900`}>{t(`${prefixSettings}.webName`)}</div>
  91. <div className='flex mt-2'>
  92. <AppIcon size='large'
  93. onClick={() => { setShowEmojiPicker(true) }}
  94. className='cursor-pointer !mr-3 self-center'
  95. icon={emoji.icon}
  96. background={emoji.icon_background}
  97. />
  98. <input className={`flex-grow rounded-lg h-10 box-border px-3 ${s.projectName} bg-gray-100`}
  99. value={inputInfo.title}
  100. onChange={onChange('title')}
  101. placeholder={t('app.appNamePlaceholder') || ''}
  102. />
  103. </div>
  104. <div className={`mt-6 font-medium ${s.settingTitle} text-gray-900 `}>{t(`${prefixSettings}.webDesc`)}</div>
  105. <p className={`mt-1 ${s.settingsTip} text-gray-500`}>{t(`${prefixSettings}.webDescTip`)}</p>
  106. <textarea
  107. rows={3}
  108. className={`mt-2 pt-2 pb-2 px-3 rounded-lg bg-gray-100 w-full ${s.settingsTip} text-gray-900`}
  109. value={inputInfo.desc}
  110. onChange={onChange('desc')}
  111. placeholder={t(`${prefixSettings}.webDescPlaceholder`) as string}
  112. />
  113. <div className={`mt-6 mb-2 font-medium ${s.settingTitle} text-gray-900 `}>{t(`${prefixSettings}.language`)}</div>
  114. <SimpleSelect
  115. items={languages}
  116. defaultValue={language}
  117. onSelect={item => setLanguage(item.value as Language)}
  118. />
  119. {!isShowMore && <div className='w-full cursor-pointer mt-8' onClick={() => setIsShowMore(true)}>
  120. <div className='flex justify-between'>
  121. <div className={`font-medium ${s.settingTitle} flex-grow text-gray-900`}>{t(`${prefixSettings}.more.entry`)}</div>
  122. <div className='flex-shrink-0 w-4 h-4 text-gray-500'>
  123. <ChevronRightIcon />
  124. </div>
  125. </div>
  126. <p className={`mt-1 ${s.policy} text-gray-500`}>{t(`${prefixSettings}.more.copyright`)} & {t(`${prefixSettings}.more.privacyPolicy`)}</p>
  127. </div>}
  128. {isShowMore && <>
  129. <hr className='w-full mt-6' />
  130. <div className={`mt-6 font-medium ${s.settingTitle} text-gray-900`}>{t(`${prefixSettings}.more.copyright`)}</div>
  131. <input className={`w-full mt-2 rounded-lg h-10 box-border px-3 ${s.projectName} bg-gray-100`}
  132. value={inputInfo.copyright}
  133. onChange={onChange('copyright')}
  134. placeholder={t(`${prefixSettings}.more.copyRightPlaceholder`) as string}
  135. />
  136. <div className={`mt-8 font-medium ${s.settingTitle} text-gray-900`}>{t(`${prefixSettings}.more.privacyPolicy`)}</div>
  137. <p className={`mt-1 ${s.settingsTip} text-gray-500`}>
  138. <Trans
  139. i18nKey={`${prefixSettings}.more.privacyPolicyTip`}
  140. components={{ privacyPolicyLink: <Link href={'https://docs.dify.ai/user-agreement/privacy-policy'} target='_blank' rel='noopener noreferrer' className='text-primary-600' /> }}
  141. />
  142. </p>
  143. <input className={`w-full mt-2 rounded-lg h-10 box-border px-3 ${s.projectName} bg-gray-100`}
  144. value={inputInfo.privacyPolicy}
  145. onChange={onChange('privacyPolicy')}
  146. placeholder={t(`${prefixSettings}.more.privacyPolicyPlaceholder`) as string}
  147. />
  148. </>}
  149. <div className='mt-10 flex justify-end'>
  150. <Button className='mr-2 flex-shrink-0 !text-sm' onClick={onHide}>{t('common.operation.cancel')}</Button>
  151. <Button type='primary' className='flex-shrink-0 !text-sm' onClick={onClickSave} loading={saveLoading}>{t('common.operation.save')}</Button>
  152. </div>
  153. {showEmojiPicker && <EmojiPicker
  154. onSelect={(icon, icon_background) => {
  155. setEmoji({ icon, icon_background })
  156. setShowEmojiPicker(false)
  157. }}
  158. onClose={() => {
  159. setEmoji({ icon: appInfo.site.icon, icon_background: appInfo.site.icon_background })
  160. setShowEmojiPicker(false)
  161. }}
  162. />}
  163. </Modal >
  164. </>
  165. )
  166. }
  167. export default React.memo(SettingsModal)