| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | /* eslint-disable @typescript-eslint/no-use-before-define */ | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 'use client' | 
					
						
							|  |  |  | import type { FC } from 'react' | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  | import React, { useCallback, useEffect, useRef, useState } from 'react' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | import cn from 'classnames' | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  | import useSWR from 'swr' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							|  |  |  | import { useContext } from 'use-context-selector' | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  | import produce, { setAutoFreeze } from 'immer' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | import { useBoolean, useGetState } from 'ahooks' | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | import AppUnavailable from '../../base/app-unavailable' | 
					
						
							| 
									
										
										
										
											2023-07-11 15:21:20 +08:00
										 |  |  | import { checkOrSetAccessToken } from '../utils' | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  | import { addFileInfos, sortAgentSorts } from '../../tools/utils' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | import useConversation from './hooks/use-conversation' | 
					
						
							|  |  |  | import { ToastContext } from '@/app/components/base/toast' | 
					
						
							|  |  |  | import Sidebar from '@/app/components/share/chat/sidebar' | 
					
						
							|  |  |  | import ConfigSence from '@/app/components/share/chat/config-scence' | 
					
						
							|  |  |  | import Header from '@/app/components/share/header' | 
					
						
							| 
									
										
										
										
											2023-07-11 15:21:20 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   delConversation, | 
					
						
							|  |  |  |   fetchAppInfo, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |   fetchAppMeta, | 
					
						
							| 
									
										
										
										
											2023-07-11 15:21:20 +08:00
										 |  |  |   fetchAppParams, | 
					
						
							|  |  |  |   fetchChatList, | 
					
						
							|  |  |  |   fetchConversations, | 
					
						
							|  |  |  |   fetchSuggestedQuestions, | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   generationConversationName, | 
					
						
							| 
									
										
										
										
											2023-07-11 15:21:20 +08:00
										 |  |  |   pinConversation, | 
					
						
							|  |  |  |   sendChatMessage, | 
					
						
							|  |  |  |   stopChatMessageResponding, | 
					
						
							|  |  |  |   unpinConversation, | 
					
						
							|  |  |  |   updateFeedback, | 
					
						
							|  |  |  | } from '@/service/share' | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  | import type { AppMeta, ConversationItem, SiteInfo } from '@/models/share' | 
					
						
							| 
									
										
										
										
											2024-01-24 01:05:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import type { | 
					
						
							|  |  |  |   CitationConfig, | 
					
						
							|  |  |  |   PromptConfig, | 
					
						
							|  |  |  |   SpeechToTextConfig, | 
					
						
							|  |  |  |   SuggestedQuestionsAfterAnswerConfig, | 
					
						
							|  |  |  |   TextToSpeechConfig, | 
					
						
							|  |  |  | } from '@/models/debug' | 
					
						
							| 
									
										
										
										
											2023-08-22 10:58:06 +08:00
										 |  |  | import type { Feedbacktype, IChatItem } from '@/app/components/app/chat/type' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | import Chat from '@/app/components/app/chat' | 
					
						
							|  |  |  | import { changeLanguage } from '@/i18n/i18next-config' | 
					
						
							|  |  |  | import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints' | 
					
						
							|  |  |  | import Loading from '@/app/components/base/loading' | 
					
						
							|  |  |  | import { replaceStringWithValues } from '@/app/components/app/configuration/prompt-value-panel' | 
					
						
							|  |  |  | import { userInputsFormToPromptVariables } from '@/utils/model-config' | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | import type { InstalledApp } from '@/models/explore' | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  | import Confirm from '@/app/components/base/confirm' | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  | import type { VisionFile, VisionSettings } from '@/types/app' | 
					
						
							|  |  |  | import { Resolution, TransferMethod } from '@/types/app' | 
					
						
							|  |  |  | import { fetchFileUploadConfig } from '@/service/common' | 
					
						
							| 
									
										
										
										
											2024-01-03 00:05:08 +08:00
										 |  |  | import type { Annotation as AnnotationType } from '@/models/log' | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | export type IMainProps = { | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |   isInstalledApp?: boolean | 
					
						
							|  |  |  |   installedAppInfo?: InstalledApp | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |   isSupportPlugin?: boolean | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  | const Main: FC<IMainProps> = ({ | 
					
						
							|  |  |  |   isInstalledApp = false, | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |   installedAppInfo, | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  | }) => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const { t } = useTranslation() | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const { notify } = useContext(ToastContext) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const media = useBreakpoints() | 
					
						
							|  |  |  |   const isMobile = media === MediaType.mobile | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* | 
					
						
							|  |  |  |   * app info | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  |   const [appUnavailable, setAppUnavailable] = useState<boolean>(false) | 
					
						
							|  |  |  |   const [isUnknwonReason, setIsUnknwonReason] = useState<boolean>(false) | 
					
						
							|  |  |  |   const [appId, setAppId] = useState<string>('') | 
					
						
							|  |  |  |   const [isPublicVersion, setIsPublicVersion] = useState<boolean>(true) | 
					
						
							|  |  |  |   const [siteInfo, setSiteInfo] = useState<SiteInfo | null>() | 
					
						
							|  |  |  |   const [promptConfig, setPromptConfig] = useState<PromptConfig | null>(null) | 
					
						
							|  |  |  |   const [inited, setInited] = useState<boolean>(false) | 
					
						
							|  |  |  |   const [plan, setPlan] = useState<string>('basic') // basic/plus/pro
 | 
					
						
							| 
									
										
										
										
											2023-12-13 20:21:11 +08:00
										 |  |  |   const [canReplaceLogo, setCanReplaceLogo] = useState<boolean>(false) | 
					
						
							| 
									
										
										
										
											2023-12-18 16:25:37 +08:00
										 |  |  |   const [customConfig, setCustomConfig] = useState<any>(null) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |   const [appMeta, setAppMeta] = useState<AppMeta | null>(null) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   // in mobile, show sidebar by click button
 | 
					
						
							|  |  |  |   const [isShowSidebar, { setTrue: showSidebar, setFalse: hideSidebar }] = useBoolean(false) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |   // Can Use metadata(https://beta.nextjs.org/docs/api-reference/metadata) to set title. But it only works in server side client.
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (siteInfo?.title) { | 
					
						
							| 
									
										
										
										
											2023-12-13 20:21:11 +08:00
										 |  |  |       if (canReplaceLogo) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         document.title = `${siteInfo.title}` | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         document.title = `${siteInfo.title} - Powered by Dify` | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-12-13 20:21:11 +08:00
										 |  |  |   }, [siteInfo?.title, canReplaceLogo]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* | 
					
						
							|  |  |  |   * conversation info | 
					
						
							|  |  |  |   */ | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |   const [allConversationList, setAllConversationList] = useState<ConversationItem[]>([]) | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |   const [isClearConversationList, { setTrue: clearConversationListTrue, setFalse: clearConversationListFalse }] = useBoolean(false) | 
					
						
							|  |  |  |   const [isClearPinnedConversationList, { setTrue: clearPinnedConversationListTrue, setFalse: clearPinnedConversationListFalse }] = useBoolean(false) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const { | 
					
						
							|  |  |  |     conversationList, | 
					
						
							|  |  |  |     setConversationList, | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     pinnedConversationList, | 
					
						
							|  |  |  |     setPinnedConversationList, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     currConversationId, | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |     getCurrConversationId, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     setCurrConversationId, | 
					
						
							|  |  |  |     getConversationIdFromStorage, | 
					
						
							|  |  |  |     isNewConversation, | 
					
						
							|  |  |  |     currConversationInfo, | 
					
						
							|  |  |  |     currInputs, | 
					
						
							|  |  |  |     newConversationInputs, | 
					
						
							|  |  |  |     // existConversationInputs,
 | 
					
						
							|  |  |  |     resetNewConversationInputs, | 
					
						
							|  |  |  |     setCurrInputs, | 
					
						
							|  |  |  |     setNewConversationInfo, | 
					
						
							| 
									
										
										
										
											2023-08-30 17:32:32 +08:00
										 |  |  |     existConversationInfo, | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     setExistConversationInfo, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   } = useConversation() | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |   const [suggestedQuestions, setSuggestQuestions] = useState<string[]>([]) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |   const [hasMore, setHasMore] = useState<boolean>(true) | 
					
						
							|  |  |  |   const [hasPinnedMore, setHasPinnedMore] = useState<boolean>(true) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const [isShowSuggestion, setIsShowSuggestion] = useState(false) | 
					
						
							|  |  |  |   const onMoreLoaded = useCallback(({ data: conversations, has_more }: any) => { | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |     setHasMore(has_more) | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |     if (isClearConversationList) { | 
					
						
							|  |  |  |       setConversationList(conversations) | 
					
						
							|  |  |  |       clearConversationListFalse() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       setConversationList([...conversationList, ...conversations]) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [conversationList, setConversationList, isClearConversationList, clearConversationListFalse]) | 
					
						
							|  |  |  |   const onPinnedMoreLoaded = useCallback(({ data: conversations, has_more }: any) => { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     setHasPinnedMore(has_more) | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |     if (isClearPinnedConversationList) { | 
					
						
							|  |  |  |       setPinnedConversationList(conversations) | 
					
						
							|  |  |  |       clearPinnedConversationListFalse() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       setPinnedConversationList([...pinnedConversationList, ...conversations]) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [pinnedConversationList, setPinnedConversationList, isClearPinnedConversationList, clearPinnedConversationListFalse]) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |   const [controlUpdateConversationList, setControlUpdateConversationList] = useState(0) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const noticeUpdateList = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     setHasMore(true) | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |     clearConversationListTrue() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     setHasPinnedMore(true) | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |     clearPinnedConversationListTrue() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     setControlUpdateConversationList(Date.now()) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [clearConversationListTrue, clearPinnedConversationListTrue]) | 
					
						
							|  |  |  |   const handlePin = useCallback(async (id: string) => { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     await pinConversation(isInstalledApp, installedAppInfo?.id, id) | 
					
						
							|  |  |  |     notify({ type: 'success', message: t('common.api.success') }) | 
					
						
							|  |  |  |     noticeUpdateList() | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [isInstalledApp, installedAppInfo?.id, t, notify, noticeUpdateList]) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const handleUnpin = useCallback(async (id: string) => { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     await unpinConversation(isInstalledApp, installedAppInfo?.id, id) | 
					
						
							|  |  |  |     notify({ type: 'success', message: t('common.api.success') }) | 
					
						
							|  |  |  |     noticeUpdateList() | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [isInstalledApp, installedAppInfo?.id, t, notify, noticeUpdateList]) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |   const [isShowConfirm, { setTrue: showConfirm, setFalse: hideConfirm }] = useBoolean(false) | 
					
						
							|  |  |  |   const [toDeleteConversationId, setToDeleteConversationId] = useState('') | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const handleDelete = useCallback((id: string) => { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     setToDeleteConversationId(id) | 
					
						
							|  |  |  |     hideSidebar() // mobile
 | 
					
						
							|  |  |  |     showConfirm() | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [hideSidebar, showConfirm]) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const didDelete = async () => { | 
					
						
							|  |  |  |     await delConversation(isInstalledApp, installedAppInfo?.id, toDeleteConversationId) | 
					
						
							|  |  |  |     notify({ type: 'success', message: t('common.api.success') }) | 
					
						
							|  |  |  |     hideConfirm() | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |     if (currConversationId === toDeleteConversationId) | 
					
						
							|  |  |  |       handleConversationIdChange('-1') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |     noticeUpdateList() | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const [suggestedQuestionsAfterAnswerConfig, setSuggestedQuestionsAfterAnswerConfig] = useState<SuggestedQuestionsAfterAnswerConfig | null>(null) | 
					
						
							| 
									
										
										
										
											2024-01-24 01:05:37 +08:00
										 |  |  |   const [speechToTextConfig, setSpeechToTextConfig] = useState<SpeechToTextConfig | null>(null) | 
					
						
							|  |  |  |   const [textToSpeechConfig, setTextToSpeechConfig] = useState<TextToSpeechConfig | null>(null) | 
					
						
							|  |  |  |   const [citationConfig, setCitationConfig] = useState<CitationConfig | null>(null) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const [chatList, setChatList, getChatList] = useGetState<IChatItem[]>([]) | 
					
						
							|  |  |  |   const chatListDomRef = useRef<HTMLDivElement>(null) | 
					
						
							|  |  |  |   const [isResponsing, { setTrue: setResponsingTrue, setFalse: setResponsingFalse }] = useBoolean(false) | 
					
						
							|  |  |  |   const [abortController, setAbortController] = useState<AbortController | null>(null) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const [conversationIdChangeBecauseOfNew, setConversationIdChangeBecauseOfNew, getConversationIdChangeBecauseOfNew] = useGetState(false) | 
					
						
							|  |  |  |   const [isChatStarted, { setTrue: setChatStarted, setFalse: setChatNotStarted }] = useBoolean(false) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const conversationIntroduction = currConversationInfo?.introduction || '' | 
					
						
							|  |  |  |   const createNewChat = useCallback(async () => { | 
					
						
							|  |  |  |     // if new chat is already exist, do not create new chat
 | 
					
						
							|  |  |  |     abortController?.abort() | 
					
						
							|  |  |  |     setResponsingFalse() | 
					
						
							|  |  |  |     if (conversationList.some(item => item.id === '-1')) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setConversationList(produce(conversationList, (draft) => { | 
					
						
							|  |  |  |       draft.unshift({ | 
					
						
							|  |  |  |         id: '-1', | 
					
						
							|  |  |  |         name: t('share.chat.newChatDefaultName'), | 
					
						
							|  |  |  |         inputs: newConversationInputs, | 
					
						
							|  |  |  |         introduction: conversationIntroduction, | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     })) | 
					
						
							|  |  |  |   }, [ | 
					
						
							|  |  |  |     abortController, | 
					
						
							|  |  |  |     setResponsingFalse, | 
					
						
							|  |  |  |     setConversationList, | 
					
						
							|  |  |  |     conversationList, | 
					
						
							|  |  |  |     newConversationInputs, | 
					
						
							|  |  |  |     conversationIntroduction, | 
					
						
							|  |  |  |     t, | 
					
						
							|  |  |  |   ]) | 
					
						
							|  |  |  |   const handleStartChat = useCallback((inputs: Record<string, any>) => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     createNewChat() | 
					
						
							|  |  |  |     setConversationIdChangeBecauseOfNew(true) | 
					
						
							|  |  |  |     setCurrInputs(inputs) | 
					
						
							|  |  |  |     setChatStarted() | 
					
						
							|  |  |  |     // parse variables in introduction
 | 
					
						
							|  |  |  |     setChatList(generateNewChatListWithOpenstatement('', inputs)) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [ | 
					
						
							|  |  |  |     createNewChat, | 
					
						
							|  |  |  |     setConversationIdChangeBecauseOfNew, | 
					
						
							|  |  |  |     setCurrInputs, | 
					
						
							|  |  |  |     setChatStarted, | 
					
						
							|  |  |  |     setChatList, | 
					
						
							|  |  |  |   ]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const hasSetInputs = (() => { | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (!isNewConversation) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       return true | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     return isChatStarted | 
					
						
							|  |  |  |   })() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const conversationName = currConversationInfo?.name || t('share.chat.newChatDefaultName') as string | 
					
						
							| 
									
										
										
										
											2023-08-30 17:32:32 +08:00
										 |  |  |   const [controlChatUpdateAllConversation, setControlChatUpdateAllConversation] = useState(0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |   // onData change thought (the produce obj). https://github.com/immerjs/immer/issues/576
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     setAutoFreeze(false) | 
					
						
							|  |  |  |     return () => { | 
					
						
							|  |  |  |       setAutoFreeze(true) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 17:32:32 +08:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     (async () => { | 
					
						
							|  |  |  |       if (controlChatUpdateAllConversation && !isNewConversation) { | 
					
						
							|  |  |  |         const { data: allConversations } = await fetchAllConversations() as { data: ConversationItem[]; has_more: boolean } | 
					
						
							|  |  |  |         const item = allConversations.find(item => item.id === currConversationId) | 
					
						
							|  |  |  |         setAllConversationList(allConversations) | 
					
						
							|  |  |  |         if (item) { | 
					
						
							|  |  |  |           setExistConversationInfo({ | 
					
						
							|  |  |  |             ...existConversationInfo, | 
					
						
							|  |  |  |             name: item?.name || '', | 
					
						
							|  |  |  |           } as any) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     })() | 
					
						
							|  |  |  |   }, [controlChatUpdateAllConversation]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleConversationSwitch = () => { | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (!inited) | 
					
						
							|  |  |  |       return | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     if (!appId) { | 
					
						
							|  |  |  |       // wait for appId
 | 
					
						
							|  |  |  |       setTimeout(handleConversationSwitch, 100) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // update inputs of current conversation
 | 
					
						
							|  |  |  |     let notSyncToStateIntroduction = '' | 
					
						
							|  |  |  |     let notSyncToStateInputs: Record<string, any> | undefined | null = {} | 
					
						
							|  |  |  |     if (!isNewConversation) { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |       const item = allConversationList.find(item => item.id === currConversationId) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       notSyncToStateInputs = item?.inputs || {} | 
					
						
							|  |  |  |       setCurrInputs(notSyncToStateInputs) | 
					
						
							|  |  |  |       notSyncToStateIntroduction = item?.introduction || '' | 
					
						
							|  |  |  |       setExistConversationInfo({ | 
					
						
							|  |  |  |         name: item?.name || '', | 
					
						
							|  |  |  |         introduction: notSyncToStateIntroduction, | 
					
						
							|  |  |  |       }) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       notSyncToStateInputs = newConversationInputs | 
					
						
							|  |  |  |       setCurrInputs(notSyncToStateInputs) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     // update chat list of current conversation
 | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |     if (!isNewConversation && !conversationIdChangeBecauseOfNew) { | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |       fetchChatList(currConversationId, isInstalledApp, installedAppInfo?.id).then((res: any) => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         const { data } = res | 
					
						
							|  |  |  |         const newChatList: IChatItem[] = generateNewChatListWithOpenstatement(notSyncToStateIntroduction, notSyncToStateInputs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         data.forEach((item: any) => { | 
					
						
							|  |  |  |           newChatList.push({ | 
					
						
							|  |  |  |             id: `question-${item.id}`, | 
					
						
							|  |  |  |             content: item.query, | 
					
						
							|  |  |  |             isAnswer: false, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |             message_files: item.message_files?.filter((file: any) => file.belongs_to === 'user') || [], | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           }) | 
					
						
							|  |  |  |           newChatList.push({ | 
					
						
							|  |  |  |             id: item.id, | 
					
						
							|  |  |  |             content: item.answer, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |             agent_thoughts: addFileInfos(item.agent_thoughts ? sortAgentSorts(item.agent_thoughts) : item.agent_thoughts, item.message_files), | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |             feedback: item.feedback, | 
					
						
							|  |  |  |             isAnswer: true, | 
					
						
							| 
									
										
										
										
											2023-09-09 19:17:12 +08:00
										 |  |  |             citation: item.retriever_resources, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |             message_files: item.message_files?.filter((file: any) => file.belongs_to === 'assistant') || [], | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           }) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |         setChatList(newChatList) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (isNewConversation && isChatStarted) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       setChatList(generateNewChatListWithOpenstatement()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setControlFocus(Date.now()) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   useEffect(handleConversationSwitch, [currConversationId, inited]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* | 
					
						
							|  |  |  |   * chat info. chat is under conversation. | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     // scroll to bottom
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (chatListDomRef.current) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       chatListDomRef.current.scrollTop = chatListDomRef.current.scrollHeight | 
					
						
							|  |  |  |   }, [chatList, currConversationId]) | 
					
						
							|  |  |  |   // user can not edit inputs if user had send message
 | 
					
						
							|  |  |  |   const canEditInpus = !chatList.some(item => item.isAnswer === false) && isNewConversation | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const handleConversationIdChange = useCallback((id: string) => { | 
					
						
							|  |  |  |     if (id === '-1') { | 
					
						
							|  |  |  |       createNewChat() | 
					
						
							|  |  |  |       setConversationIdChangeBecauseOfNew(true) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       setConversationIdChangeBecauseOfNew(false) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // trigger handleConversationSwitch
 | 
					
						
							|  |  |  |     setCurrConversationId(id, appId) | 
					
						
							|  |  |  |     setIsShowSuggestion(false) | 
					
						
							|  |  |  |     hideSidebar() | 
					
						
							|  |  |  |   }, [ | 
					
						
							|  |  |  |     appId, | 
					
						
							|  |  |  |     createNewChat, | 
					
						
							|  |  |  |     hideSidebar, | 
					
						
							|  |  |  |     setCurrConversationId, | 
					
						
							|  |  |  |     setIsShowSuggestion, | 
					
						
							|  |  |  |     setConversationIdChangeBecauseOfNew, | 
					
						
							|  |  |  |   ]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // sometime introduction is not applied to state
 | 
					
						
							|  |  |  |   const generateNewChatListWithOpenstatement = (introduction?: string, inputs?: Record<string, any> | null) => { | 
					
						
							|  |  |  |     let caculatedIntroduction = introduction || conversationIntroduction || '' | 
					
						
							|  |  |  |     const caculatedPromptVariables = inputs || currInputs || null | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (caculatedIntroduction && caculatedPromptVariables) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       caculatedIntroduction = replaceStringWithValues(caculatedIntroduction, promptConfig?.prompt_variables || [], caculatedPromptVariables) | 
					
						
							| 
									
										
										
										
											2024-01-24 01:05:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     const openstatement = { | 
					
						
							|  |  |  |       id: `${Date.now()}`, | 
					
						
							|  |  |  |       content: caculatedIntroduction, | 
					
						
							|  |  |  |       isAnswer: true, | 
					
						
							|  |  |  |       feedbackDisabled: true, | 
					
						
							| 
									
										
										
										
											2023-09-04 10:26:46 +08:00
										 |  |  |       isOpeningStatement: true, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |       suggestedQuestions: openingSuggestedQuestions, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (caculatedIntroduction) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       return [openstatement] | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     return [] | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |   const fetchAllConversations = () => { | 
					
						
							|  |  |  |     return fetchConversations(isInstalledApp, installedAppInfo?.id, undefined, undefined, 100) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-11 15:21:20 +08:00
										 |  |  |   const fetchInitData = async () => { | 
					
						
							| 
									
										
										
										
											2023-07-11 17:07:52 +08:00
										 |  |  |     if (!isInstalledApp) | 
					
						
							|  |  |  |       await checkOrSetAccessToken() | 
					
						
							| 
									
										
										
										
											2023-07-11 15:21:20 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     return Promise.all([isInstalledApp | 
					
						
							|  |  |  |       ? { | 
					
						
							|  |  |  |         app_id: installedAppInfo?.id, | 
					
						
							|  |  |  |         site: { | 
					
						
							|  |  |  |           title: installedAppInfo?.app.name, | 
					
						
							| 
									
										
										
										
											2023-10-26 10:18:38 +08:00
										 |  |  |           icon: installedAppInfo?.app.icon, | 
					
						
							|  |  |  |           icon_background: installedAppInfo?.app.icon_background, | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |           prompt_public: false, | 
					
						
							|  |  |  |           copyright: '', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         plan: 'basic', | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |       : fetchAppInfo(), fetchAllConversations(), fetchAppParams(isInstalledApp, installedAppInfo?.id), fetchAppMeta(isInstalledApp, installedAppInfo?.id)]) | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   const { data: fileUploadConfigResponse } = useSWR(isInstalledApp ? { url: '/files/upload' } : null, fetchFileUploadConfig) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   // init
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     (async () => { | 
					
						
							|  |  |  |       try { | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |         const [appData, conversationData, appParams, appMeta]: any = await fetchInitData() | 
					
						
							|  |  |  |         setAppMeta(appMeta) | 
					
						
							| 
									
										
										
										
											2023-12-18 16:25:37 +08:00
										 |  |  |         const { app_id: appId, site: siteInfo, plan, can_replace_logo, custom_config }: any = appData | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         setAppId(appId) | 
					
						
							|  |  |  |         setPlan(plan) | 
					
						
							| 
									
										
										
										
											2023-12-13 20:21:11 +08:00
										 |  |  |         setCanReplaceLogo(can_replace_logo) | 
					
						
							| 
									
										
										
										
											2023-12-18 16:25:37 +08:00
										 |  |  |         setCustomConfig(custom_config) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         const tempIsPublicVersion = siteInfo.prompt_public | 
					
						
							|  |  |  |         setIsPublicVersion(tempIsPublicVersion) | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         const prompt_template = '' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         // handle current conversation id
 | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         const { data: allConversations } = conversationData as { data: ConversationItem[]; has_more: boolean } | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         const _conversationId = getConversationIdFromStorage(appId) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         const isNotNewConversation = allConversations.some(item => item.id === _conversationId) | 
					
						
							|  |  |  |         setAllConversationList(allConversations) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         // fetch new conversation info
 | 
					
						
							| 
									
										
										
										
											2024-01-24 01:05:37 +08:00
										 |  |  |         const { user_input_form, opening_statement: introduction, suggested_questions, suggested_questions_after_answer, speech_to_text, text_to_speech, retriever_resource, file_upload, sensitive_word_avoidance }: any = appParams | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |         setVisionConfig({ | 
					
						
							|  |  |  |           ...file_upload.image, | 
					
						
							|  |  |  |           image_file_size_limit: appParams?.system_parameters?.image_file_size_limit, | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         const prompt_variables = userInputsFormToPromptVariables(user_input_form) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         if (siteInfo.default_language) | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |           changeLanguage(siteInfo.default_language) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         setNewConversationInfo({ | 
					
						
							|  |  |  |           name: t('share.chat.newChatDefaultName'), | 
					
						
							|  |  |  |           introduction, | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |         setOpeningSuggestedQuestions(suggested_questions || []) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         setSiteInfo(siteInfo as SiteInfo) | 
					
						
							|  |  |  |         setPromptConfig({ | 
					
						
							|  |  |  |           prompt_template, | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |           prompt_variables, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         } as PromptConfig) | 
					
						
							|  |  |  |         setSuggestedQuestionsAfterAnswerConfig(suggested_questions_after_answer) | 
					
						
							| 
									
										
										
										
											2023-07-07 17:50:42 +08:00
										 |  |  |         setSpeechToTextConfig(speech_to_text) | 
					
						
							| 
									
										
										
										
											2024-01-24 01:05:37 +08:00
										 |  |  |         setTextToSpeechConfig(text_to_speech) | 
					
						
							| 
									
										
										
										
											2023-09-09 19:17:12 +08:00
										 |  |  |         setCitationConfig(retriever_resource) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         // setConversationList(conversations as ConversationItem[])
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         if (isNotNewConversation) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           setCurrConversationId(_conversationId, appId, false) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         setInited(true) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |       } | 
					
						
							|  |  |  |       catch (e: any) { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         if (e.status === 404) { | 
					
						
							|  |  |  |           setAppUnavailable(true) | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           setIsUnknwonReason(true) | 
					
						
							|  |  |  |           setAppUnavailable(true) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     })() | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const logError = useCallback((message: string) => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     notify({ type: 'error', message }) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [notify]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const checkCanSend = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2023-07-04 14:33:41 +08:00
										 |  |  |     if (currConversationId !== '-1') | 
					
						
							|  |  |  |       return true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     const prompt_variables = promptConfig?.prompt_variables | 
					
						
							|  |  |  |     const inputs = currInputs | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |     if (!inputs || !prompt_variables || prompt_variables?.length === 0) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       return true | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-19 12:41:35 +08:00
										 |  |  |     let hasEmptyInput = '' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     const requiredVars = prompt_variables?.filter(({ key, name, required }) => { | 
					
						
							|  |  |  |       const res = (!key || !key.trim()) || (!name || !name.trim()) || (required || required === undefined || required === null) | 
					
						
							|  |  |  |       return res | 
					
						
							|  |  |  |     }) || [] // compatible with old version
 | 
					
						
							| 
									
										
										
										
											2023-07-19 12:41:35 +08:00
										 |  |  |     requiredVars.forEach(({ key, name }) => { | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |       if (hasEmptyInput) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         return | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (!inputs?.[key]) | 
					
						
							| 
									
										
										
										
											2023-07-19 12:41:35 +08:00
										 |  |  |         hasEmptyInput = name | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hasEmptyInput) { | 
					
						
							| 
									
										
										
										
											2023-07-19 12:41:35 +08:00
										 |  |  |       logError(t('appDebug.errorMessage.valueOfVarRequired', { key: hasEmptyInput })) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       return false | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return !hasEmptyInput | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [currConversationId, currInputs, promptConfig, t, logError]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const [controlFocus, setControlFocus] = useState(0) | 
					
						
							|  |  |  |   const doShowSuggestion = isShowSuggestion && !isResponsing | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |   const [openingSuggestedQuestions, setOpeningSuggestedQuestions] = useState<string[]>([]) | 
					
						
							| 
									
										
										
										
											2023-06-12 16:37:03 +08:00
										 |  |  |   const [messageTaskId, setMessageTaskId] = useState('') | 
					
						
							| 
									
										
										
										
											2023-06-13 16:34:53 +08:00
										 |  |  |   const [hasStopResponded, setHasStopResponded, getHasStopResponded] = useGetState(false) | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |   const [isResponsingConIsCurrCon, setIsResponsingConCurrCon, getIsResponsingConIsCurrCon] = useGetState(true) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |   const [userQuery, setUserQuery] = useState('') | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   const [visionConfig, setVisionConfig] = useState<VisionSettings>({ | 
					
						
							|  |  |  |     enabled: false, | 
					
						
							|  |  |  |     number_limits: 2, | 
					
						
							|  |  |  |     detail: Resolution.low, | 
					
						
							|  |  |  |     transfer_methods: [TransferMethod.local_file], | 
					
						
							|  |  |  |   }) | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const updateCurrentQA = ({ | 
					
						
							|  |  |  |     responseItem, | 
					
						
							|  |  |  |     questionId, | 
					
						
							|  |  |  |     placeholderAnswerId, | 
					
						
							|  |  |  |     questionItem, | 
					
						
							|  |  |  |   }: { | 
					
						
							|  |  |  |     responseItem: IChatItem | 
					
						
							|  |  |  |     questionId: string | 
					
						
							|  |  |  |     placeholderAnswerId: string | 
					
						
							|  |  |  |     questionItem: IChatItem | 
					
						
							|  |  |  |   }) => { | 
					
						
							|  |  |  |     // closesure new list is outdated.
 | 
					
						
							|  |  |  |     const newListWithAnswer = produce( | 
					
						
							|  |  |  |       getChatList().filter(item => item.id !== responseItem.id && item.id !== placeholderAnswerId), | 
					
						
							|  |  |  |       (draft) => { | 
					
						
							|  |  |  |         if (!draft.find(item => item.id === questionId)) | 
					
						
							|  |  |  |           draft.push({ ...questionItem }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         draft.push({ ...responseItem }) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     setChatList(newListWithAnswer) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |   const handleSend = async (message: string, files?: VisionFile[]) => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     if (isResponsing) { | 
					
						
							|  |  |  |       notify({ type: 'info', message: t('appDebug.errorMessage.waitForResponse') }) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (files?.find(item => item.transfer_method === TransferMethod.local_file && !item.upload_file_id)) { | 
					
						
							|  |  |  |       notify({ type: 'info', message: t('appDebug.errorMessage.waitForImgUpload') }) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const data: Record<string, any> = { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       inputs: currInputs, | 
					
						
							|  |  |  |       query: message, | 
					
						
							|  |  |  |       conversation_id: isNewConversation ? null : currConversationId, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |     if (visionConfig.enabled && files && files?.length > 0) { | 
					
						
							|  |  |  |       data.files = files.map((item) => { | 
					
						
							|  |  |  |         if (item.transfer_method === TransferMethod.local_file) { | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             ...item, | 
					
						
							|  |  |  |             url: '', | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return item | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     // qustion
 | 
					
						
							|  |  |  |     const questionId = `question-${Date.now()}` | 
					
						
							|  |  |  |     const questionItem = { | 
					
						
							|  |  |  |       id: questionId, | 
					
						
							|  |  |  |       content: message, | 
					
						
							|  |  |  |       isAnswer: false, | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |       message_files: files, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const placeholderAnswerId = `answer-placeholder-${Date.now()}` | 
					
						
							|  |  |  |     const placeholderAnswerItem = { | 
					
						
							|  |  |  |       id: placeholderAnswerId, | 
					
						
							|  |  |  |       content: '', | 
					
						
							|  |  |  |       isAnswer: true, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const newList = [...getChatList(), questionItem, placeholderAnswerItem] | 
					
						
							|  |  |  |     setChatList(newList) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |     let isAgentMode = false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     // answer
 | 
					
						
							| 
									
										
										
										
											2023-09-09 19:17:12 +08:00
										 |  |  |     const responseItem: IChatItem = { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       id: `${Date.now()}`, | 
					
						
							|  |  |  |       content: '', | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |       agent_thoughts: [], | 
					
						
							|  |  |  |       message_files: [], | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       isAnswer: true, | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |     let hasSetResponseId = false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |     const prevTempNewConversationId = getCurrConversationId() || '-1' | 
					
						
							|  |  |  |     let tempNewConversationId = prevTempNewConversationId | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-13 16:34:53 +08:00
										 |  |  |     setHasStopResponded(false) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     setResponsingTrue() | 
					
						
							|  |  |  |     setIsShowSuggestion(false) | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |     setIsResponsingConCurrCon(true) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     sendChatMessage(data, { | 
					
						
							|  |  |  |       getAbortController: (abortController) => { | 
					
						
							|  |  |  |         setAbortController(abortController) | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2023-06-12 16:37:03 +08:00
										 |  |  |       onData: (message: string, isFirstMessage: boolean, { conversationId: newConversationId, messageId, taskId }: any) => { | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |         if (!isAgentMode) { | 
					
						
							|  |  |  |           responseItem.content = responseItem.content + message | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           const lastThought = responseItem.agent_thoughts?.[responseItem.agent_thoughts?.length - 1] | 
					
						
							|  |  |  |           if (lastThought) | 
					
						
							|  |  |  |             lastThought.thought = lastThought.thought + message // need immer setAutoFreeze
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (messageId && !hasSetResponseId) { | 
					
						
							|  |  |  |           responseItem.id = messageId | 
					
						
							|  |  |  |           hasSetResponseId = true | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         if (isFirstMessage && newConversationId) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           tempNewConversationId = newConversationId | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-12 16:37:03 +08:00
										 |  |  |         setMessageTaskId(taskId) | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |         // has switched to other conversation
 | 
					
						
							|  |  |  |         if (prevTempNewConversationId !== getCurrConversationId()) { | 
					
						
							|  |  |  |           setIsResponsingConCurrCon(false) | 
					
						
							|  |  |  |           return | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |         updateCurrentQA({ | 
					
						
							|  |  |  |           responseItem, | 
					
						
							|  |  |  |           questionId, | 
					
						
							|  |  |  |           placeholderAnswerId, | 
					
						
							|  |  |  |           questionItem, | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       }, | 
					
						
							|  |  |  |       async onCompleted(hasError?: boolean) { | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         if (hasError) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           return | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         if (getConversationIdChangeBecauseOfNew()) { | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |           const { data: allConversations }: any = await fetchAllConversations() | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |           const newItem: any = await generationConversationName(isInstalledApp, installedAppInfo?.id, allConversations[0].id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           const newAllConversations = produce(allConversations, (draft: any) => { | 
					
						
							|  |  |  |             draft[0].name = newItem.name | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           setAllConversationList(newAllConversations as any) | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |           noticeUpdateList() | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         setConversationIdChangeBecauseOfNew(false) | 
					
						
							|  |  |  |         resetNewConversationInputs() | 
					
						
							|  |  |  |         setChatNotStarted() | 
					
						
							|  |  |  |         setCurrConversationId(tempNewConversationId, appId, true) | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |         if (getIsResponsingConIsCurrCon() && suggestedQuestionsAfterAnswerConfig?.enabled && !getHasStopResponded()) { | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |           const { data }: any = await fetchSuggestedQuestions(responseItem.id, isInstalledApp, installedAppInfo?.id) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           setSuggestQuestions(data) | 
					
						
							|  |  |  |           setIsShowSuggestion(true) | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |         setResponsingFalse() | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       }, | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |       onFile(file) { | 
					
						
							|  |  |  |         const lastThought = responseItem.agent_thoughts?.[responseItem.agent_thoughts?.length - 1] | 
					
						
							|  |  |  |         if (lastThought) | 
					
						
							|  |  |  |           lastThought.message_files = [...(lastThought as any).message_files, { ...file }] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         updateCurrentQA({ | 
					
						
							|  |  |  |           responseItem, | 
					
						
							|  |  |  |           questionId, | 
					
						
							|  |  |  |           placeholderAnswerId, | 
					
						
							|  |  |  |           questionItem, | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       onThought(thought) { | 
					
						
							|  |  |  |         isAgentMode = true | 
					
						
							|  |  |  |         const response = responseItem as any | 
					
						
							|  |  |  |         if (thought.message_id && !hasSetResponseId) { | 
					
						
							|  |  |  |           response.id = thought.message_id | 
					
						
							|  |  |  |           hasSetResponseId = true | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // responseItem.id = thought.message_id;
 | 
					
						
							|  |  |  |         if (response.agent_thoughts.length === 0) { | 
					
						
							|  |  |  |           response.agent_thoughts.push(thought) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           const lastThought = response.agent_thoughts[response.agent_thoughts.length - 1] | 
					
						
							|  |  |  |           // thought changed but still the same thought, so update.
 | 
					
						
							|  |  |  |           if (lastThought.id === thought.id) { | 
					
						
							|  |  |  |             thought.thought = lastThought.thought | 
					
						
							|  |  |  |             thought.message_files = lastThought.message_files | 
					
						
							|  |  |  |             responseItem.agent_thoughts![response.agent_thoughts.length - 1] = thought | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           else { | 
					
						
							|  |  |  |             responseItem.agent_thoughts!.push(thought) | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // has switched to other conversation
 | 
					
						
							|  |  |  |         if (prevTempNewConversationId !== getCurrConversationId()) { | 
					
						
							|  |  |  |           setIsResponsingConCurrCon(false) | 
					
						
							|  |  |  |           return false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         updateCurrentQA({ | 
					
						
							|  |  |  |           responseItem, | 
					
						
							|  |  |  |           questionId, | 
					
						
							|  |  |  |           placeholderAnswerId, | 
					
						
							|  |  |  |           questionItem, | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2024-01-03 00:05:08 +08:00
										 |  |  |       onMessageEnd: (messageEnd) => { | 
					
						
							|  |  |  |         if (messageEnd.metadata?.annotation_reply) { | 
					
						
							|  |  |  |           responseItem.id = messageEnd.id | 
					
						
							|  |  |  |           responseItem.annotation = ({ | 
					
						
							|  |  |  |             id: messageEnd.metadata.annotation_reply.id, | 
					
						
							|  |  |  |             authorName: messageEnd.metadata.annotation_reply.account.name, | 
					
						
							|  |  |  |           } as AnnotationType) | 
					
						
							| 
									
										
										
										
											2023-09-09 19:17:12 +08:00
										 |  |  |           const newListWithAnswer = produce( | 
					
						
							|  |  |  |             getChatList().filter(item => item.id !== responseItem.id && item.id !== placeholderAnswerId), | 
					
						
							|  |  |  |             (draft) => { | 
					
						
							|  |  |  |               if (!draft.find(item => item.id === questionId)) | 
					
						
							|  |  |  |                 draft.push({ ...questionItem }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-03 00:05:08 +08:00
										 |  |  |               draft.push({ | 
					
						
							|  |  |  |                 ...responseItem, | 
					
						
							|  |  |  |               }) | 
					
						
							| 
									
										
										
										
											2023-09-09 19:17:12 +08:00
										 |  |  |             }) | 
					
						
							|  |  |  |           setChatList(newListWithAnswer) | 
					
						
							| 
									
										
										
										
											2024-01-03 00:05:08 +08:00
										 |  |  |           return | 
					
						
							| 
									
										
										
										
											2023-09-09 19:17:12 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-01-03 00:05:08 +08:00
										 |  |  |         // not support show citation
 | 
					
						
							|  |  |  |         // responseItem.citation = messageEnd.retriever_resources
 | 
					
						
							|  |  |  |         if (!isInstalledApp) | 
					
						
							|  |  |  |           return | 
					
						
							|  |  |  |         const newListWithAnswer = produce( | 
					
						
							|  |  |  |           getChatList().filter(item => item.id !== responseItem.id && item.id !== placeholderAnswerId), | 
					
						
							|  |  |  |           (draft) => { | 
					
						
							|  |  |  |             if (!draft.find(item => item.id === questionId)) | 
					
						
							|  |  |  |               draft.push({ ...questionItem }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             draft.push({ ...responseItem }) | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         setChatList(newListWithAnswer) | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2023-11-06 19:36:32 +08:00
										 |  |  |       onMessageReplace: (messageReplace) => { | 
					
						
							|  |  |  |         if (isInstalledApp) { | 
					
						
							|  |  |  |           responseItem.content = messageReplace.answer | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           setChatList(produce( | 
					
						
							|  |  |  |             getChatList(), | 
					
						
							|  |  |  |             (draft) => { | 
					
						
							|  |  |  |               const current = draft.find(item => item.id === messageReplace.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               if (current) | 
					
						
							|  |  |  |                 current.content = messageReplace.answer | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           )) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       onError() { | 
					
						
							|  |  |  |         setResponsingFalse() | 
					
						
							|  |  |  |         // role back placeholder answer
 | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         setChatList(produce(getChatList(), (draft) => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           draft.splice(draft.findIndex(item => item.id === placeholderAnswerId), 1) | 
					
						
							|  |  |  |         })) | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |     }, isInstalledApp, installedAppInfo?.id) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const handleFeedback = useCallback(async (messageId: string, feedback: Feedbacktype) => { | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |     await updateFeedback({ url: `/messages/${messageId}/feedbacks`, body: { rating: feedback.rating } }, isInstalledApp, installedAppInfo?.id) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     const newChatList = chatList.map((item) => { | 
					
						
							|  |  |  |       if (item.id === messageId) { | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |           ...item, | 
					
						
							|  |  |  |           feedback, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return item | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     setChatList(newChatList) | 
					
						
							|  |  |  |     notify({ type: 'success', message: t('common.api.success') }) | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   }, [isInstalledApp, installedAppInfo?.id, chatList, t, notify, setChatList]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleListChanged = useCallback((list: ConversationItem[]) => { | 
					
						
							|  |  |  |     setConversationList(list) | 
					
						
							|  |  |  |     setControlChatUpdateAllConversation(Date.now()) | 
					
						
							|  |  |  |   }, [setConversationList, setControlChatUpdateAllConversation]) | 
					
						
							|  |  |  |   const handlePinnedListChanged = useCallback((list: ConversationItem[]) => { | 
					
						
							|  |  |  |     setPinnedConversationList(list) | 
					
						
							|  |  |  |     setControlChatUpdateAllConversation(Date.now()) | 
					
						
							|  |  |  |   }, [setPinnedConversationList, setControlChatUpdateAllConversation]) | 
					
						
							|  |  |  |   const handleStartChatOnSidebar = useCallback(() => { | 
					
						
							|  |  |  |     handleConversationIdChange('-1') | 
					
						
							|  |  |  |   }, [handleConversationIdChange]) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const renderSidebar = () => { | 
					
						
							|  |  |  |     if (!appId || !siteInfo || !promptConfig) | 
					
						
							|  |  |  |       return null | 
					
						
							|  |  |  |     return ( | 
					
						
							|  |  |  |       <Sidebar | 
					
						
							|  |  |  |         list={conversationList} | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |         onListChanged={handleListChanged} | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |         isClearConversationList={isClearConversationList} | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         pinnedList={pinnedConversationList} | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |         onPinnedListChanged={handlePinnedListChanged} | 
					
						
							| 
									
										
										
										
											2023-06-28 17:53:23 +08:00
										 |  |  |         isClearPinnedConversationList={isClearPinnedConversationList} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         onMoreLoaded={onMoreLoaded} | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         onPinnedMoreLoaded={onPinnedMoreLoaded} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         isNoMore={!hasMore} | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         isPinnedNoMore={!hasPinnedMore} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         onCurrentIdChange={handleConversationIdChange} | 
					
						
							|  |  |  |         currentId={currConversationId} | 
					
						
							|  |  |  |         copyRight={siteInfo.copyright || siteInfo.title} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         isInstalledApp={isInstalledApp} | 
					
						
							|  |  |  |         installedAppId={installedAppInfo?.id} | 
					
						
							|  |  |  |         siteInfo={siteInfo} | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  |         onPin={handlePin} | 
					
						
							|  |  |  |         onUnpin={handleUnpin} | 
					
						
							|  |  |  |         controlUpdateList={controlUpdateConversationList} | 
					
						
							|  |  |  |         onDelete={handleDelete} | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |         onStartChat={handleStartChatOnSidebar} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       /> | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |   const handleAbortResponsing = useCallback(async () => { | 
					
						
							|  |  |  |     await stopChatMessageResponding(appId, messageTaskId, isInstalledApp, installedAppInfo?.id) | 
					
						
							|  |  |  |     setHasStopResponded(true) | 
					
						
							|  |  |  |     setResponsingFalse() | 
					
						
							|  |  |  |   }, [appId, messageTaskId, isInstalledApp, installedAppInfo?.id]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   if (appUnavailable) | 
					
						
							|  |  |  |     return <AppUnavailable isUnknwonReason={isUnknwonReason} /> | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-24 23:24:33 +08:00
										 |  |  |   if (!appId || !siteInfo || !promptConfig) { | 
					
						
							|  |  |  |     return <div className='flex h-screen w-full'> | 
					
						
							|  |  |  |       <Loading type='app' /> | 
					
						
							|  |  |  |     </div> | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							| 
									
										
										
										
											2023-11-28 20:05:19 +08:00
										 |  |  |     <div className='bg-gray-100 h-full flex flex-col'> | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |       {!isInstalledApp && ( | 
					
						
							|  |  |  |         <Header | 
					
						
							|  |  |  |           title={siteInfo.title} | 
					
						
							|  |  |  |           icon={siteInfo.icon || ''} | 
					
						
							| 
									
										
										
										
											2024-02-04 16:10:46 +08:00
										 |  |  |           icon_background={siteInfo.icon_background || ''} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |           isMobile={isMobile} | 
					
						
							|  |  |  |           onShowSideBar={showSidebar} | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |           onCreateNewChat={handleStartChatOnSidebar} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         /> | 
					
						
							|  |  |  |       )} | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       <div | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         className={cn( | 
					
						
							| 
									
										
										
										
											2023-08-24 17:57:25 +08:00
										 |  |  |           'flex rounded-t-2xl bg-white overflow-hidden h-full w-full', | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |           isInstalledApp && 'rounded-b-2xl', | 
					
						
							|  |  |  |         )} | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         style={isInstalledApp | 
					
						
							|  |  |  |           ? { | 
					
						
							|  |  |  |             boxShadow: '0px 12px 16px -4px rgba(16, 24, 40, 0.08), 0px 4px 6px -2px rgba(16, 24, 40, 0.03)', | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           : {}} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |       > | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         {/* sidebar */} | 
					
						
							|  |  |  |         {!isMobile && renderSidebar()} | 
					
						
							|  |  |  |         {isMobile && isShowSidebar && ( | 
					
						
							|  |  |  |           <div className='fixed inset-0 z-50' | 
					
						
							|  |  |  |             style={{ backgroundColor: 'rgba(35, 56, 118, 0.2)' }} | 
					
						
							|  |  |  |             onClick={hideSidebar} | 
					
						
							|  |  |  |           > | 
					
						
							|  |  |  |             <div className='inline-block' onClick={e => e.stopPropagation()}> | 
					
						
							|  |  |  |               {renderSidebar()} | 
					
						
							|  |  |  |             </div> | 
					
						
							|  |  |  |           </div> | 
					
						
							|  |  |  |         )} | 
					
						
							|  |  |  |         {/* main */} | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         <div className={cn( | 
					
						
							| 
									
										
										
										
											2023-11-28 20:05:19 +08:00
										 |  |  |           'h-full flex-grow flex flex-col overflow-y-auto', | 
					
						
							| 
									
										
										
										
											2023-05-30 16:15:08 +08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-05-25 16:59:47 +08:00
										 |  |  |         }> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           <ConfigSence | 
					
						
							|  |  |  |             conversationName={conversationName} | 
					
						
							|  |  |  |             hasSetInputs={hasSetInputs} | 
					
						
							|  |  |  |             isPublicVersion={isPublicVersion} | 
					
						
							|  |  |  |             siteInfo={siteInfo} | 
					
						
							|  |  |  |             promptConfig={promptConfig} | 
					
						
							|  |  |  |             onStartChat={handleStartChat} | 
					
						
							|  |  |  |             canEidtInpus={canEditInpus} | 
					
						
							|  |  |  |             savedInputs={currInputs as Record<string, any>} | 
					
						
							|  |  |  |             onInputsChange={setCurrInputs} | 
					
						
							|  |  |  |             plan={plan} | 
					
						
							| 
									
										
										
										
											2023-12-13 20:21:11 +08:00
										 |  |  |             canReplaceLogo={canReplaceLogo} | 
					
						
							| 
									
										
										
										
											2023-12-18 16:25:37 +08:00
										 |  |  |             customConfig={customConfig} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |           ></ConfigSence> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             hasSetInputs && ( | 
					
						
							| 
									
										
										
										
											2023-07-17 00:14:32 +08:00
										 |  |  |               <div className={cn(doShowSuggestion ? 'pb-[140px]' : (isResponsing ? 'pb-[113px]' : 'pb-[76px]'), 'relative grow h-[200px] pc:w-[794px] max-w-full mobile:w-full mx-auto mb-3.5 overflow-hidden')}> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                 <div className='h-full overflow-y-auto' ref={chatListDomRef}> | 
					
						
							|  |  |  |                   <Chat | 
					
						
							|  |  |  |                     chatList={chatList} | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |                     query={userQuery} | 
					
						
							|  |  |  |                     onQueryChange={setUserQuery} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                     onSend={handleSend} | 
					
						
							|  |  |  |                     isHideFeedbackEdit | 
					
						
							|  |  |  |                     onFeedback={handleFeedback} | 
					
						
							|  |  |  |                     isResponsing={isResponsing} | 
					
						
							| 
									
										
										
										
											2023-07-27 13:27:34 +08:00
										 |  |  |                     canStopResponsing={!!messageTaskId && isResponsingConIsCurrCon} | 
					
						
							| 
									
										
										
										
											2024-01-25 14:55:12 +08:00
										 |  |  |                     abortResponsing={handleAbortResponsing} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                     checkCanSend={checkCanSend} | 
					
						
							|  |  |  |                     controlFocus={controlFocus} | 
					
						
							|  |  |  |                     isShowSuggestion={doShowSuggestion} | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |                     suggestionList={suggestedQuestions} | 
					
						
							| 
									
										
										
										
											2023-07-07 17:50:42 +08:00
										 |  |  |                     isShowSpeechToText={speechToTextConfig?.enabled} | 
					
						
							| 
									
										
										
										
											2024-01-24 01:05:37 +08:00
										 |  |  |                     isShowTextToSpeech={textToSpeechConfig?.enabled} | 
					
						
							| 
									
										
										
										
											2024-01-24 13:57:11 +08:00
										 |  |  |                     isShowCitation={citationConfig?.enabled} | 
					
						
							| 
									
										
										
										
											2023-11-13 22:32:39 +08:00
										 |  |  |                     visionConfig={{ | 
					
						
							|  |  |  |                       ...visionConfig, | 
					
						
							|  |  |  |                       image_file_size_limit: fileUploadConfigResponse ? fileUploadConfigResponse.image_file_size_limit : visionConfig.image_file_size_limit, | 
					
						
							|  |  |  |                     }} | 
					
						
							| 
									
										
										
										
											2024-01-23 19:31:56 +08:00
										 |  |  |                     allToolIcons={appMeta?.tool_icons || {}} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                   /> | 
					
						
							|  |  |  |                 </div> | 
					
						
							|  |  |  |               </div>) | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2023-06-28 11:16:54 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |           {isShowConfirm && ( | 
					
						
							|  |  |  |             <Confirm | 
					
						
							|  |  |  |               title={t('share.chat.deleteConversation.title')} | 
					
						
							|  |  |  |               content={t('share.chat.deleteConversation.content')} | 
					
						
							|  |  |  |               isShow={isShowConfirm} | 
					
						
							|  |  |  |               onClose={hideConfirm} | 
					
						
							|  |  |  |               onConfirm={didDelete} | 
					
						
							|  |  |  |               onCancel={hideConfirm} | 
					
						
							|  |  |  |             /> | 
					
						
							|  |  |  |           )} | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         </div> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     </div> | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | export default React.memo(Main) |