mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-01-13 05: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
101 lines
2.8 KiB
TypeScript
101 lines
2.8 KiB
TypeScript
import React, { useState, useRef, useMemo } from 'react';
|
|
import { LazyLoadImage } from 'react-lazy-load-image-component';
|
|
import { cn, scaleImage } from '~/utils';
|
|
import DialogImage from './DialogImage';
|
|
import { Skeleton } from '~/components';
|
|
|
|
const Image = ({
|
|
imagePath,
|
|
altText,
|
|
height,
|
|
width,
|
|
placeholderDimensions,
|
|
className,
|
|
}: {
|
|
imagePath: string;
|
|
altText: string;
|
|
height: number;
|
|
width: number;
|
|
placeholderDimensions?: {
|
|
height?: string;
|
|
width?: string;
|
|
};
|
|
className?: string;
|
|
}) => {
|
|
const [isOpen, setIsOpen] = useState(false);
|
|
const [isLoaded, setIsLoaded] = useState(false);
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
|
|
const handleImageLoad = () => setIsLoaded(true);
|
|
|
|
const { width: scaledWidth, height: scaledHeight } = useMemo(
|
|
() =>
|
|
scaleImage({
|
|
originalWidth: Number(placeholderDimensions?.width?.split('px')[0] ?? width),
|
|
originalHeight: Number(placeholderDimensions?.height?.split('px')[0] ?? height),
|
|
containerRef,
|
|
}),
|
|
[placeholderDimensions, height, width],
|
|
);
|
|
|
|
const downloadImage = () => {
|
|
const link = document.createElement('a');
|
|
link.href = imagePath;
|
|
link.download = altText;
|
|
document.body.appendChild(link);
|
|
link.click();
|
|
document.body.removeChild(link);
|
|
};
|
|
|
|
return (
|
|
<div ref={containerRef}>
|
|
<div
|
|
className={cn(
|
|
'relative mt-1 flex h-auto w-full max-w-lg items-center justify-center overflow-hidden rounded-lg border border-border-light text-text-secondary-alt shadow-md',
|
|
className,
|
|
)}
|
|
>
|
|
<button
|
|
type="button"
|
|
aria-label={`View ${altText} in dialog`}
|
|
onClick={() => setIsOpen(true)}
|
|
className="cursor-pointer focus:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2"
|
|
>
|
|
<LazyLoadImage
|
|
alt={altText}
|
|
onLoad={handleImageLoad}
|
|
visibleByDefault={true}
|
|
className={cn(
|
|
'opacity-100 transition-opacity duration-100',
|
|
isLoaded ? 'opacity-100' : 'opacity-0',
|
|
)}
|
|
src={imagePath}
|
|
style={{
|
|
width: `${scaledWidth}`,
|
|
height: 'auto',
|
|
color: 'transparent',
|
|
display: 'block',
|
|
}}
|
|
placeholder={
|
|
<Skeleton
|
|
className={cn('h-auto w-full', `h-[${scaledHeight}] w-[${scaledWidth}]`)}
|
|
aria-label="Loading image"
|
|
aria-busy="true"
|
|
/>
|
|
}
|
|
/>
|
|
</button>
|
|
{isLoaded && (
|
|
<DialogImage
|
|
isOpen={isOpen}
|
|
onOpenChange={setIsOpen}
|
|
src={imagePath}
|
|
downloadImage={downloadImage}
|
|
/>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default Image;
|