mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-01-15 06:58:51 +01:00
* ✨ feat: Enhance Spinner component with customizable properties and improved animation * 🔧 fix: Replace Loader with Spinner in RunCode component and update FilePreview to use Spinner for progress indication * ✨ feat: Refactor icons in CodeProgress and CancelledIcon components; enhance animation and styling in ExecuteCode and ProgressText components * ✨ feat: Refactor attachment handling in ExecuteCode component; replace individual attachment rendering with AttachmentGroup for improved structure * ✨ feat: Refactor dialog components for improved accessibility and styling; integrate Skeleton loading state in Image component * ✨ feat: Refactor ToolCall component to use ToolCallInfo for better structure; replace ToolPopover with AttachmentGroup; enhance ProgressText with error handling and improved UI elements * 🔧 fix: Remove unnecessary whitespace in ProgressText * 🔧 fix: Remove unnecessary margin from AgentFooter and AgentPanel components; clean up SidePanel imports * ✨ feat: Enhance ToolCall and ToolCallInfo components with improved styling; update translations and add warning text color to Tailwind config * 🔧 fix: Update import statement for useLocalize in ToolCallInfo component; fix: chatform transition * ✨ feat: Refactor ToolCall and ToolCallInfo components for improved structure and styling; add optimized code block for better output display * ✨ feat: Implement OpenAI image generation component; add progress tracking and localization for user feedback * 🔧 fix: Adjust base duration values for image generation; optimize timing for quality settings * chore: remove unnecessary space * ✨ feat: Enhance OpenAI image generation with editing capabilities; update localization for progress feedback * ✨ feat: Add download functionality to images; enhance DialogImage component with download button * ✨ feat: Enhance image resizing functionality; support custom percentage and pixel dimensions in resizeImageBuffer
153 lines
4.6 KiB
TypeScript
153 lines
4.6 KiB
TypeScript
import { memo, useMemo, useState } from 'react';
|
|
import { useRecoilValue, useRecoilState } from 'recoil';
|
|
import { ContentTypes } from 'librechat-data-provider';
|
|
import type { TMessageContentParts, TAttachment, Agents } from 'librechat-data-provider';
|
|
import { ThinkingButton } from '~/components/Artifacts/Thinking';
|
|
import useLocalize from '~/hooks/useLocalize';
|
|
import { mapAttachments } from '~/utils/map';
|
|
import { MessageContext } from '~/Providers';
|
|
import { EditTextPart } from './Parts';
|
|
import store from '~/store';
|
|
import Part from './Part';
|
|
|
|
type ContentPartsProps = {
|
|
content: Array<TMessageContentParts | undefined> | undefined;
|
|
messageId: string;
|
|
conversationId?: string | null;
|
|
attachments?: TAttachment[];
|
|
isCreatedByUser: boolean;
|
|
isLast: boolean;
|
|
isSubmitting: boolean;
|
|
edit?: boolean;
|
|
enterEdit?: (cancel?: boolean) => void | null | undefined;
|
|
siblingIdx?: number;
|
|
setSiblingIdx?:
|
|
| ((value: number) => void | React.Dispatch<React.SetStateAction<number>>)
|
|
| null
|
|
| undefined;
|
|
};
|
|
|
|
const ContentParts = memo(
|
|
({
|
|
content,
|
|
messageId,
|
|
conversationId,
|
|
attachments,
|
|
isCreatedByUser,
|
|
isLast,
|
|
isSubmitting,
|
|
edit,
|
|
enterEdit,
|
|
siblingIdx,
|
|
setSiblingIdx,
|
|
}: ContentPartsProps) => {
|
|
const localize = useLocalize();
|
|
const [showThinking, setShowThinking] = useRecoilState<boolean>(store.showThinking);
|
|
const [isExpanded, setIsExpanded] = useState(showThinking);
|
|
const messageAttachmentsMap = useRecoilValue(store.messageAttachmentsMap);
|
|
const attachmentMap = useMemo(
|
|
() => mapAttachments(attachments ?? messageAttachmentsMap[messageId] ?? []),
|
|
[attachments, messageAttachmentsMap, messageId],
|
|
);
|
|
|
|
const hasReasoningParts = useMemo(() => {
|
|
const hasThinkPart = content?.some((part) => part?.type === ContentTypes.THINK) ?? false;
|
|
const allThinkPartsHaveContent =
|
|
content?.every((part) => {
|
|
if (part?.type !== ContentTypes.THINK) {
|
|
return true;
|
|
}
|
|
|
|
if (typeof part.think === 'string') {
|
|
const cleanedContent = part.think.replace(/<\/?think>/g, '').trim();
|
|
return cleanedContent.length > 0;
|
|
}
|
|
|
|
return false;
|
|
}) ?? false;
|
|
|
|
return hasThinkPart && allThinkPartsHaveContent;
|
|
}, [content]);
|
|
if (!content) {
|
|
return null;
|
|
}
|
|
if (edit === true && enterEdit && setSiblingIdx) {
|
|
return (
|
|
<>
|
|
{content.map((part, idx) => {
|
|
if (part?.type !== ContentTypes.TEXT || typeof part.text !== 'string') {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<EditTextPart
|
|
index={idx}
|
|
text={part.text}
|
|
messageId={messageId}
|
|
isSubmitting={isSubmitting}
|
|
enterEdit={enterEdit}
|
|
siblingIdx={siblingIdx ?? null}
|
|
setSiblingIdx={setSiblingIdx}
|
|
key={`edit-${messageId}-${idx}`}
|
|
/>
|
|
);
|
|
})}
|
|
</>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{hasReasoningParts && (
|
|
<div className="mb-5">
|
|
<ThinkingButton
|
|
isExpanded={isExpanded}
|
|
onClick={() =>
|
|
setIsExpanded((prev) => {
|
|
const val = !prev;
|
|
setShowThinking(val);
|
|
return val;
|
|
})
|
|
}
|
|
label={
|
|
isSubmitting && isLast ? localize('com_ui_thinking') : localize('com_ui_thoughts')
|
|
}
|
|
/>
|
|
</div>
|
|
)}
|
|
{content
|
|
.filter((part) => part)
|
|
.map((part, idx) => {
|
|
const toolCallId =
|
|
(part?.[ContentTypes.TOOL_CALL] as Agents.ToolCall | undefined)?.id ?? '';
|
|
const attachments = attachmentMap[toolCallId];
|
|
|
|
return (
|
|
<MessageContext.Provider
|
|
key={`provider-${messageId}-${idx}`}
|
|
value={{
|
|
messageId,
|
|
conversationId,
|
|
partIndex: idx,
|
|
isExpanded,
|
|
nextType: content[idx + 1]?.type,
|
|
}}
|
|
>
|
|
<Part
|
|
part={part}
|
|
attachments={attachments}
|
|
isSubmitting={isSubmitting}
|
|
key={`part-${messageId}-${idx}`}
|
|
isCreatedByUser={isCreatedByUser}
|
|
isLast={idx === content.length - 1}
|
|
showCursor={idx === content.length - 1 && isLast}
|
|
/>
|
|
</MessageContext.Provider>
|
|
);
|
|
})}
|
|
</>
|
|
);
|
|
},
|
|
);
|
|
|
|
export default ContentParts;
|