- Add new Docs Generator agent component (docs_generator.py) - Support markdown-to-PDF conversion with tables, lists, code blocks - Add PDF download button component for chat interface - Add component form and configuration UI - Add translations for all supported languages - Add component documentation guide - Support multiple output formats: PDF, DOCX, TXT - Configurable styling, logo, orientaton.. - Download button integration with Message component
191 lines
6 KiB
TypeScript
191 lines
6 KiB
TypeScript
import { ReactComponent as AssistantIcon } from '@/assets/svg/assistant.svg';
|
|
import { MessageType } from '@/constants/chat';
|
|
import {
|
|
IMessage,
|
|
IReference,
|
|
IReferenceChunk,
|
|
UploadResponseDataType,
|
|
} from '@/interfaces/database/chat';
|
|
import classNames from 'classnames';
|
|
import { memo, useCallback, useMemo } from 'react';
|
|
|
|
import { IRegenerateMessage, IRemoveMessageById } from '@/hooks/logic-hooks';
|
|
import { cn } from '@/lib/utils';
|
|
import MarkdownContent from '../markdown-content';
|
|
import { ReferenceDocumentList } from '../next-message-item/reference-document-list';
|
|
import { UploadedMessageFiles } from '../next-message-item/uploaded-message-files';
|
|
import {
|
|
PDFDownloadButton,
|
|
extractPDFDownloadInfo,
|
|
removePDFDownloadInfo,
|
|
} from '../pdf-download-button';
|
|
import { RAGFlowAvatar } from '../ragflow-avatar';
|
|
import { useTheme } from '../theme-provider';
|
|
import { AssistantGroupButton, UserGroupButton } from './group-button';
|
|
import styles from './index.less';
|
|
|
|
interface IProps extends Partial<IRemoveMessageById>, IRegenerateMessage {
|
|
item: IMessage;
|
|
reference: IReference;
|
|
loading?: boolean;
|
|
sendLoading?: boolean;
|
|
visibleAvatar?: boolean;
|
|
nickname?: string;
|
|
avatar?: string;
|
|
avatarDialog?: string | null;
|
|
clickDocumentButton?: (documentId: string, chunk: IReferenceChunk) => void;
|
|
index: number;
|
|
showLikeButton?: boolean;
|
|
showLoudspeaker?: boolean;
|
|
}
|
|
|
|
const MessageItem = ({
|
|
item,
|
|
reference,
|
|
loading = false,
|
|
avatar,
|
|
avatarDialog,
|
|
sendLoading = false,
|
|
clickDocumentButton,
|
|
index,
|
|
removeMessageById,
|
|
regenerateMessage,
|
|
showLikeButton = true,
|
|
showLoudspeaker = true,
|
|
visibleAvatar = true,
|
|
}: IProps) => {
|
|
const { theme } = useTheme();
|
|
const isAssistant = item.role === MessageType.Assistant;
|
|
const isUser = item.role === MessageType.User;
|
|
|
|
const uploadedFiles = useMemo(() => {
|
|
return item?.files ?? [];
|
|
}, [item?.files]);
|
|
|
|
const referenceDocumentList = useMemo(() => {
|
|
return reference?.doc_aggs ?? [];
|
|
}, [reference?.doc_aggs]);
|
|
|
|
// Extract PDF download info from message content
|
|
const pdfDownloadInfo = useMemo(
|
|
() => extractPDFDownloadInfo(item.content),
|
|
[item.content],
|
|
);
|
|
|
|
// If we have PDF download info, extract the remaining text
|
|
const messageContent = useMemo(() => {
|
|
if (!pdfDownloadInfo) return item.content;
|
|
|
|
// Remove the JSON part from the content to avoid showing it
|
|
return removePDFDownloadInfo(item.content, pdfDownloadInfo);
|
|
}, [item.content, pdfDownloadInfo]);
|
|
|
|
const handleRegenerateMessage = useCallback(() => {
|
|
regenerateMessage?.(item);
|
|
}, [regenerateMessage, item]);
|
|
|
|
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,
|
|
})}
|
|
>
|
|
{visibleAvatar &&
|
|
(item.role === MessageType.User ? (
|
|
<RAGFlowAvatar
|
|
className="size-10"
|
|
avatar={avatar ?? '/logo.svg'}
|
|
isPerson
|
|
/>
|
|
) : avatarDialog ? (
|
|
<RAGFlowAvatar
|
|
className="size-10"
|
|
avatar={avatarDialog}
|
|
isPerson
|
|
/>
|
|
) : (
|
|
<AssistantIcon />
|
|
))}
|
|
|
|
<section className="flex gap-2 flex-1 flex-col">
|
|
{isAssistant ? (
|
|
index !== 0 && (
|
|
<AssistantGroupButton
|
|
messageId={item.id}
|
|
content={item.content}
|
|
prompt={item.prompt}
|
|
showLikeButton={showLikeButton}
|
|
audioBinary={item.audio_binary}
|
|
showLoudspeaker={showLoudspeaker}
|
|
></AssistantGroupButton>
|
|
)
|
|
) : (
|
|
<UserGroupButton
|
|
content={item.content}
|
|
messageId={item.id}
|
|
removeMessageById={removeMessageById}
|
|
regenerateMessage={regenerateMessage && handleRegenerateMessage}
|
|
sendLoading={sendLoading}
|
|
></UserGroupButton>
|
|
)}
|
|
|
|
{/* Show PDF download button if download info is present */}
|
|
{pdfDownloadInfo && (
|
|
<PDFDownloadButton
|
|
downloadInfo={pdfDownloadInfo}
|
|
className="mb-2"
|
|
/>
|
|
)}
|
|
|
|
{/* Show message content if there's any text besides the download */}
|
|
{messageContent && (
|
|
<div
|
|
className={cn(
|
|
isAssistant
|
|
? theme === 'dark'
|
|
? styles.messageTextDark
|
|
: styles.messageText
|
|
: styles.messageUserText,
|
|
{ '!bg-bg-card': !isAssistant },
|
|
)}
|
|
>
|
|
<MarkdownContent
|
|
loading={loading}
|
|
content={messageContent}
|
|
reference={reference}
|
|
clickDocumentButton={clickDocumentButton}
|
|
></MarkdownContent>
|
|
</div>
|
|
)}
|
|
{isAssistant && referenceDocumentList.length > 0 && (
|
|
<ReferenceDocumentList
|
|
list={referenceDocumentList}
|
|
></ReferenceDocumentList>
|
|
)}
|
|
{isUser &&
|
|
Array.isArray(uploadedFiles) &&
|
|
uploadedFiles.length > 0 && (
|
|
<UploadedMessageFiles
|
|
files={uploadedFiles as UploadResponseDataType[]}
|
|
></UploadedMessageFiles>
|
|
)}
|
|
</section>
|
|
</div>
|
|
</section>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default memo(MessageItem);
|