mirror of
https://github.com/supermemoryai/supermemory.git
synced 2026-05-18 06:03:49 +00:00
67 lines
1.7 KiB
TypeScript
67 lines
1.7 KiB
TypeScript
"use client"
|
|
|
|
import { useCallback, useState } from "react"
|
|
import { cn } from "@lib/utils"
|
|
|
|
interface ImagePreviewProps {
|
|
url: string
|
|
title?: string | null
|
|
}
|
|
|
|
export function ImagePreview({ url, title }: ImagePreviewProps) {
|
|
const [imageError, setImageError] = useState(false)
|
|
const [isLoading, setIsLoading] = useState(true)
|
|
const [retryKey, setRetryKey] = useState(0)
|
|
|
|
// On first failure, wait briefly then force a re-render with a new key to
|
|
// retry the fetch (covers transient R2 timing issues).
|
|
// On second failure, give up and show the error state.
|
|
const handleImageError = useCallback(() => {
|
|
if (retryKey === 0) {
|
|
setTimeout(() => setRetryKey(1), 500)
|
|
return
|
|
}
|
|
setImageError(true)
|
|
setIsLoading(false)
|
|
}, [retryKey])
|
|
|
|
if (imageError || !url) {
|
|
return (
|
|
<div className="flex items-center justify-center h-full text-[#737373]">
|
|
<p>Failed to load image</p>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<div className="relative size-full overflow-hidden flex items-center justify-center bg-[#0B1017]">
|
|
{isLoading && (
|
|
<div className="absolute inset-0 bg-cover bg-center animate-pulse">
|
|
<div className="size-full bg-[#1B1F24]" />
|
|
</div>
|
|
)}
|
|
<div
|
|
className="absolute inset-0 bg-cover bg-center"
|
|
style={{
|
|
backgroundImage: `url(${url})`,
|
|
filter: "blur(100px)",
|
|
transform: "scale(1.1)",
|
|
opacity: isLoading ? 0.5 : 1,
|
|
}}
|
|
/>
|
|
<div className="absolute inset-0 bg-black/30" />
|
|
<img
|
|
key={retryKey}
|
|
src={url}
|
|
alt={title || "Image preview"}
|
|
className={cn(
|
|
"relative max-w-full max-h-full size-auto object-contain z-10",
|
|
isLoading && "opacity-0",
|
|
)}
|
|
onError={handleImageError}
|
|
onLoad={() => setIsLoading(false)}
|
|
loading="lazy"
|
|
/>
|
|
</div>
|
|
)
|
|
}
|