index.tsx 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. import type { FC } from 'react'
  2. import {
  3. useEffect,
  4. useState,
  5. } from 'react'
  6. import { useAsyncEffect } from 'ahooks'
  7. import { useThemeContext } from '../embedded-chatbot/theme/theme-context'
  8. import {
  9. ChatWithHistoryContext,
  10. useChatWithHistoryContext,
  11. } from './context'
  12. import { useChatWithHistory } from './hooks'
  13. import Sidebar from './sidebar'
  14. import HeaderInMobile from './header-in-mobile'
  15. import ConfigPanel from './config-panel'
  16. import ChatWrapper from './chat-wrapper'
  17. import type { InstalledApp } from '@/models/explore'
  18. import Loading from '@/app/components/base/loading'
  19. import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints'
  20. import { checkOrSetAccessToken } from '@/app/components/share/utils'
  21. import AppUnavailable from '@/app/components/base/app-unavailable'
  22. type ChatWithHistoryProps = {
  23. className?: string
  24. }
  25. const ChatWithHistory: FC<ChatWithHistoryProps> = ({
  26. className,
  27. }) => {
  28. const {
  29. appInfoError,
  30. appData,
  31. appInfoLoading,
  32. appPrevChatList,
  33. showConfigPanelBeforeChat,
  34. appChatListDataLoading,
  35. chatShouldReloadKey,
  36. isMobile,
  37. themeBuilder,
  38. } = useChatWithHistoryContext()
  39. const chatReady = (!showConfigPanelBeforeChat || !!appPrevChatList.length)
  40. const customConfig = appData?.custom_config
  41. const site = appData?.site
  42. useEffect(() => {
  43. themeBuilder?.buildTheme(site?.chat_color_theme, site?.chat_color_theme_inverted)
  44. if (site) {
  45. if (customConfig)
  46. document.title = `${site.title}`
  47. else
  48. document.title = `${site.title} - Powered by Dify`
  49. }
  50. }, [site, customConfig, themeBuilder])
  51. if (appInfoLoading) {
  52. return (
  53. <Loading type='app' />
  54. )
  55. }
  56. if (appInfoError) {
  57. return (
  58. <AppUnavailable />
  59. )
  60. }
  61. return (
  62. <div className={`h-full flex bg-white ${className} ${isMobile && 'flex-col'}`}>
  63. {
  64. !isMobile && (
  65. <Sidebar />
  66. )
  67. }
  68. {
  69. isMobile && (
  70. <HeaderInMobile />
  71. )
  72. }
  73. <div className={`grow overflow-hidden ${showConfigPanelBeforeChat && !appPrevChatList.length && 'flex items-center justify-center'}`}>
  74. {
  75. showConfigPanelBeforeChat && !appChatListDataLoading && !appPrevChatList.length && (
  76. <div className={`flex w-full items-center justify-center h-full ${isMobile && 'px-4'}`}>
  77. <ConfigPanel />
  78. </div>
  79. )
  80. }
  81. {
  82. appChatListDataLoading && chatReady && (
  83. <Loading type='app' />
  84. )
  85. }
  86. {
  87. chatReady && !appChatListDataLoading && (
  88. <ChatWrapper key={chatShouldReloadKey} />
  89. )
  90. }
  91. </div>
  92. </div>
  93. )
  94. }
  95. export type ChatWithHistoryWrapProps = {
  96. installedAppInfo?: InstalledApp
  97. className?: string
  98. }
  99. const ChatWithHistoryWrap: FC<ChatWithHistoryWrapProps> = ({
  100. installedAppInfo,
  101. className,
  102. }) => {
  103. const media = useBreakpoints()
  104. const isMobile = media === MediaType.mobile
  105. const themeBuilder = useThemeContext()
  106. const {
  107. appInfoError,
  108. appInfoLoading,
  109. appData,
  110. appParams,
  111. appMeta,
  112. appChatListDataLoading,
  113. currentConversationId,
  114. currentConversationItem,
  115. appPrevChatList,
  116. pinnedConversationList,
  117. conversationList,
  118. showConfigPanelBeforeChat,
  119. newConversationInputs,
  120. handleNewConversationInputsChange,
  121. inputsForms,
  122. handleNewConversation,
  123. handleStartChat,
  124. handleChangeConversation,
  125. handlePinConversation,
  126. handleUnpinConversation,
  127. handleDeleteConversation,
  128. conversationRenaming,
  129. handleRenameConversation,
  130. handleNewConversationCompleted,
  131. chatShouldReloadKey,
  132. isInstalledApp,
  133. appId,
  134. handleFeedback,
  135. currentChatInstanceRef,
  136. } = useChatWithHistory(installedAppInfo)
  137. return (
  138. <ChatWithHistoryContext.Provider value={{
  139. appInfoError,
  140. appInfoLoading,
  141. appData,
  142. appParams,
  143. appMeta,
  144. appChatListDataLoading,
  145. currentConversationId,
  146. currentConversationItem,
  147. appPrevChatList,
  148. pinnedConversationList,
  149. conversationList,
  150. showConfigPanelBeforeChat,
  151. newConversationInputs,
  152. handleNewConversationInputsChange,
  153. inputsForms,
  154. handleNewConversation,
  155. handleStartChat,
  156. handleChangeConversation,
  157. handlePinConversation,
  158. handleUnpinConversation,
  159. handleDeleteConversation,
  160. conversationRenaming,
  161. handleRenameConversation,
  162. handleNewConversationCompleted,
  163. chatShouldReloadKey,
  164. isMobile,
  165. isInstalledApp,
  166. appId,
  167. handleFeedback,
  168. currentChatInstanceRef,
  169. themeBuilder,
  170. }}>
  171. <ChatWithHistory className={className} />
  172. </ChatWithHistoryContext.Provider>
  173. )
  174. }
  175. const ChatWithHistoryWrapWithCheckToken: FC<ChatWithHistoryWrapProps> = ({
  176. installedAppInfo,
  177. className,
  178. }) => {
  179. const [initialized, setInitialized] = useState(false)
  180. const [appUnavailable, setAppUnavailable] = useState<boolean>(false)
  181. const [isUnknownReason, setIsUnknownReason] = useState<boolean>(false)
  182. useAsyncEffect(async () => {
  183. if (!initialized) {
  184. if (!installedAppInfo) {
  185. try {
  186. await checkOrSetAccessToken()
  187. }
  188. catch (e: any) {
  189. if (e.status === 404) {
  190. setAppUnavailable(true)
  191. }
  192. else {
  193. setIsUnknownReason(true)
  194. setAppUnavailable(true)
  195. }
  196. }
  197. }
  198. setInitialized(true)
  199. }
  200. }, [])
  201. if (!initialized)
  202. return null
  203. if (appUnavailable)
  204. return <AppUnavailable isUnknownReason={isUnknownReason} />
  205. return (
  206. <ChatWithHistoryWrap
  207. installedAppInfo={installedAppInfo}
  208. className={className}
  209. />
  210. )
  211. }
  212. export default ChatWithHistoryWrapWithCheckToken