mirror of
https://github.com/DayuanJiang/next-ai-draw-io.git
synced 2026-01-02 22:32:27 +08:00
- Disable noisy biome rules (noExplicitAny, useExhaustiveDependencies, etc.) - Fix memory leak in file-preview-list.tsx with useRef pattern - Separate unmount cleanup into dedicated useEffect - Add ToolPartLike interface for type safety in chat-message-display - Add accessibility attributes (role, tabIndex, onKeyDown) - Replace autoFocus with useEffect focus pattern - Minor syntax improvements (optional chaining, key fixes)
134 lines
5.1 KiB
TypeScript
134 lines
5.1 KiB
TypeScript
"use client"
|
|
|
|
import { X } from "lucide-react"
|
|
import Image from "next/image"
|
|
import { useEffect, useRef, useState } from "react"
|
|
|
|
interface FilePreviewListProps {
|
|
files: File[]
|
|
onRemoveFile: (fileToRemove: File) => void
|
|
}
|
|
|
|
export function FilePreviewList({ files, onRemoveFile }: FilePreviewListProps) {
|
|
const [selectedImage, setSelectedImage] = useState<string | null>(null)
|
|
const [imageUrls, setImageUrls] = useState<Map<File, string>>(new Map())
|
|
const imageUrlsRef = useRef<Map<File, string>>(new Map())
|
|
|
|
// Create and cleanup object URLs when files change
|
|
useEffect(() => {
|
|
const currentUrls = imageUrlsRef.current
|
|
const newUrls = new Map<File, string>()
|
|
|
|
files.forEach((file) => {
|
|
if (file.type.startsWith("image/")) {
|
|
// Reuse existing URL if file is already tracked
|
|
const existingUrl = currentUrls.get(file)
|
|
if (existingUrl) {
|
|
newUrls.set(file, existingUrl)
|
|
} else {
|
|
newUrls.set(file, URL.createObjectURL(file))
|
|
}
|
|
}
|
|
})
|
|
|
|
// Revoke URLs for files that are no longer in the list
|
|
currentUrls.forEach((url, file) => {
|
|
if (!newUrls.has(file)) {
|
|
URL.revokeObjectURL(url)
|
|
}
|
|
})
|
|
|
|
imageUrlsRef.current = newUrls
|
|
setImageUrls(newUrls)
|
|
}, [files])
|
|
|
|
// Cleanup all URLs on unmount only
|
|
useEffect(() => {
|
|
return () => {
|
|
imageUrlsRef.current.forEach((url) => {
|
|
URL.revokeObjectURL(url)
|
|
})
|
|
}
|
|
}, [])
|
|
|
|
// Clear selected image if its URL was revoked
|
|
useEffect(() => {
|
|
if (
|
|
selectedImage &&
|
|
!Array.from(imageUrls.values()).includes(selectedImage)
|
|
) {
|
|
setSelectedImage(null)
|
|
}
|
|
}, [imageUrls, selectedImage])
|
|
|
|
if (files.length === 0) return null
|
|
|
|
return (
|
|
<>
|
|
<div className="flex flex-wrap gap-2 mt-2 p-2 bg-muted/50 rounded-md">
|
|
{files.map((file, index) => {
|
|
const imageUrl = imageUrls.get(file) || null
|
|
return (
|
|
<div key={file.name + index} className="relative group">
|
|
<div
|
|
className="w-20 h-20 border rounded-md overflow-hidden bg-muted cursor-pointer"
|
|
onClick={() =>
|
|
imageUrl && setSelectedImage(imageUrl)
|
|
}
|
|
>
|
|
{file.type.startsWith("image/") && imageUrl ? (
|
|
<Image
|
|
src={imageUrl}
|
|
alt={file.name}
|
|
width={80}
|
|
height={80}
|
|
className="object-cover w-full h-full"
|
|
/>
|
|
) : (
|
|
<div className="flex items-center justify-center h-full text-xs text-center p-1">
|
|
{file.name}
|
|
</div>
|
|
)}
|
|
</div>
|
|
<button
|
|
type="button"
|
|
onClick={() => onRemoveFile(file)}
|
|
className="absolute -top-2 -right-2 bg-destructive rounded-full p-1 opacity-0 group-hover:opacity-100 transition-opacity"
|
|
aria-label="Remove file"
|
|
>
|
|
<X className="h-3 w-3" />
|
|
</button>
|
|
</div>
|
|
)
|
|
})}
|
|
</div>
|
|
|
|
{/* Image Modal/Lightbox */}
|
|
{selectedImage && (
|
|
<div
|
|
className="fixed inset-0 z-50 bg-black/80 flex items-center justify-center p-4"
|
|
onClick={() => setSelectedImage(null)}
|
|
>
|
|
<button
|
|
className="absolute top-4 right-4 z-10 bg-white rounded-full p-2 hover:bg-gray-200 transition-colors"
|
|
onClick={() => setSelectedImage(null)}
|
|
aria-label="Close"
|
|
>
|
|
<X className="h-6 w-6" />
|
|
</button>
|
|
<div className="relative w-auto h-auto max-w-[90vw] max-h-[90vh]">
|
|
<Image
|
|
src={selectedImage}
|
|
alt="Full size preview of uploaded diagram or image"
|
|
width={1200}
|
|
height={900}
|
|
className="object-contain max-w-full max-h-[90vh] w-auto h-auto"
|
|
onClick={(e) => e.stopPropagation()}
|
|
/>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</>
|
|
)
|
|
}
|