mirror of
https://github.com/infiniflow/ragflow.git
synced 2025-12-08 20:42:30 +08:00
### What problem does this PR solve? some chunk method pictures are not in English #437 feat: set the height of both html and body to 100% feat: add SharedChat feat: add shared hooks ### Type of change - [x] Bug Fix (non-breaking change which fixes an issue)
217 lines
6.6 KiB
TypeScript
217 lines
6.6 KiB
TypeScript
import { ReactComponent as AssistantIcon } from '@/assets/svg/assistant.svg';
|
|
import NewDocumentLink from '@/components/new-document-link';
|
|
import DocumentPreviewer from '@/components/pdf-previewer';
|
|
import { MessageType } from '@/constants/chat';
|
|
import { useSelectFileThumbnails } from '@/hooks/knowledgeHook';
|
|
import { useSelectUserInfo } from '@/hooks/userSettingHook';
|
|
import { IReference, Message } from '@/interfaces/database/chat';
|
|
import { IChunk } from '@/interfaces/database/knowledge';
|
|
import {
|
|
Avatar,
|
|
Button,
|
|
Drawer,
|
|
Flex,
|
|
Input,
|
|
List,
|
|
Skeleton,
|
|
Spin,
|
|
} from 'antd';
|
|
import classNames from 'classnames';
|
|
import { useMemo } from 'react';
|
|
import {
|
|
useClickDrawer,
|
|
useFetchConversationOnMount,
|
|
useGetFileIcon,
|
|
useGetSendButtonDisabled,
|
|
useSelectConversationLoading,
|
|
useSendMessage,
|
|
} from '../hooks';
|
|
import MarkdownContent from '../markdown-content';
|
|
|
|
import SvgIcon from '@/components/svg-icon';
|
|
import { useTranslate } from '@/hooks/commonHooks';
|
|
import { getExtension, isPdf } from '@/utils/documentUtils';
|
|
import styles from './index.less';
|
|
|
|
const MessageItem = ({
|
|
item,
|
|
reference,
|
|
clickDocumentButton,
|
|
}: {
|
|
item: Message;
|
|
reference: IReference;
|
|
clickDocumentButton: (documentId: string, chunk: IChunk) => void;
|
|
}) => {
|
|
const userInfo = useSelectUserInfo();
|
|
const fileThumbnails = useSelectFileThumbnails();
|
|
|
|
const isAssistant = item.role === MessageType.Assistant;
|
|
|
|
const referenceDocumentList = useMemo(() => {
|
|
return reference?.doc_aggs ?? [];
|
|
}, [reference?.doc_aggs]);
|
|
|
|
return (
|
|
<div
|
|
className={classNames(styles.messageItem, {
|
|
[styles.messageItemLeft]: item.role === MessageType.Assistant,
|
|
[styles.messageItemRight]: item.role === MessageType.User,
|
|
})}
|
|
>
|
|
<section
|
|
className={classNames(styles.messageItemSection, {
|
|
[styles.messageItemSectionLeft]: item.role === MessageType.Assistant,
|
|
[styles.messageItemSectionRight]: item.role === MessageType.User,
|
|
})}
|
|
>
|
|
<div
|
|
className={classNames(styles.messageItemContent, {
|
|
[styles.messageItemContentReverse]: item.role === MessageType.User,
|
|
})}
|
|
>
|
|
{item.role === MessageType.User ? (
|
|
<Avatar
|
|
size={40}
|
|
src={
|
|
userInfo.avatar ??
|
|
'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png'
|
|
}
|
|
/>
|
|
) : (
|
|
<AssistantIcon></AssistantIcon>
|
|
)}
|
|
<Flex vertical gap={8} flex={1}>
|
|
<b>{isAssistant ? '' : userInfo.nickname}</b>
|
|
<div className={styles.messageText}>
|
|
{item.content !== '' ? (
|
|
<MarkdownContent
|
|
content={item.content}
|
|
reference={reference}
|
|
clickDocumentButton={clickDocumentButton}
|
|
></MarkdownContent>
|
|
) : (
|
|
<Skeleton active className={styles.messageEmpty} />
|
|
)}
|
|
</div>
|
|
{isAssistant && referenceDocumentList.length > 0 && (
|
|
<List
|
|
bordered
|
|
dataSource={referenceDocumentList}
|
|
renderItem={(item) => {
|
|
const fileThumbnail = fileThumbnails[item.doc_id];
|
|
const fileExtension = getExtension(item.doc_name);
|
|
return (
|
|
<List.Item>
|
|
<Flex gap={'small'} align="center">
|
|
{fileThumbnail ? (
|
|
<img src={fileThumbnail}></img>
|
|
) : (
|
|
<SvgIcon
|
|
name={`file-icon/${fileExtension}`}
|
|
width={24}
|
|
></SvgIcon>
|
|
)}
|
|
|
|
<NewDocumentLink
|
|
documentId={item.doc_id}
|
|
preventDefault={!isPdf(item.doc_name)}
|
|
>
|
|
{item.doc_name}
|
|
</NewDocumentLink>
|
|
</Flex>
|
|
</List.Item>
|
|
);
|
|
}}
|
|
/>
|
|
)}
|
|
</Flex>
|
|
</div>
|
|
</section>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
const ChatContainer = () => {
|
|
const {
|
|
ref,
|
|
currentConversation: conversation,
|
|
addNewestConversation,
|
|
removeLatestMessage,
|
|
} = useFetchConversationOnMount();
|
|
const {
|
|
handleInputChange,
|
|
handlePressEnter,
|
|
value,
|
|
loading: sendLoading,
|
|
} = useSendMessage(conversation, addNewestConversation, removeLatestMessage);
|
|
const { visible, hideModal, documentId, selectedChunk, clickDocumentButton } =
|
|
useClickDrawer();
|
|
const disabled = useGetSendButtonDisabled();
|
|
useGetFileIcon();
|
|
const loading = useSelectConversationLoading();
|
|
const { t } = useTranslate('chat');
|
|
|
|
return (
|
|
<>
|
|
<Flex flex={1} className={styles.chatContainer} vertical>
|
|
<Flex flex={1} vertical className={styles.messageContainer}>
|
|
<div>
|
|
<Spin spinning={loading}>
|
|
{conversation?.message?.map((message) => {
|
|
const assistantMessages = conversation?.message
|
|
?.filter((x) => x.role === MessageType.Assistant)
|
|
.slice(1);
|
|
const referenceIndex = assistantMessages.findIndex(
|
|
(x) => x.id === message.id,
|
|
);
|
|
const reference = conversation.reference[referenceIndex];
|
|
return (
|
|
<MessageItem
|
|
key={message.id}
|
|
item={message}
|
|
reference={reference}
|
|
clickDocumentButton={clickDocumentButton}
|
|
></MessageItem>
|
|
);
|
|
})}
|
|
</Spin>
|
|
</div>
|
|
<div ref={ref} />
|
|
</Flex>
|
|
<Input
|
|
size="large"
|
|
placeholder={t('sendPlaceholder')}
|
|
value={value}
|
|
disabled={disabled}
|
|
suffix={
|
|
<Button
|
|
type="primary"
|
|
onClick={handlePressEnter}
|
|
loading={sendLoading}
|
|
disabled={disabled}
|
|
>
|
|
{t('send')}
|
|
</Button>
|
|
}
|
|
onPressEnter={handlePressEnter}
|
|
onChange={handleInputChange}
|
|
/>
|
|
</Flex>
|
|
<Drawer
|
|
title="Document Previewer"
|
|
onClose={hideModal}
|
|
open={visible}
|
|
width={'50vw'}
|
|
>
|
|
<DocumentPreviewer
|
|
documentId={documentId}
|
|
chunk={selectedChunk}
|
|
visible={visible}
|
|
></DocumentPreviewer>
|
|
</Drawer>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default ChatContainer;
|