Spaces:
Running
Running
| import React from 'react'; | |
| import useFilesList from '@/hooks/useFilesList'; | |
| import Link from 'next/link'; | |
| import { Loader2, AlertCircle, Download, Box, Brain } from 'lucide-react'; | |
| export default function FilesWidget({ jobID }: { jobID: string }) { | |
| const { files, status, refreshFiles } = useFilesList(jobID, 5000); | |
| const cleanSize = (size: number) => { | |
| if (size < 1024) { | |
| return `${size} B`; | |
| } else if (size < 1024 * 1024) { | |
| return `${(size / 1024).toFixed(1)} KB`; | |
| } else if (size < 1024 * 1024 * 1024) { | |
| return `${(size / (1024 * 1024)).toFixed(1)} MB`; | |
| } else { | |
| return `${(size / (1024 * 1024 * 1024)).toFixed(1)} GB`; | |
| } | |
| }; | |
| return ( | |
| <div className="col-span-2 bg-gray-900 rounded-xl shadow-lg overflow-hidden hover:shadow-2xl transition-all duration-300 border border-gray-800"> | |
| <div className="bg-gray-800 px-4 py-3 flex items-center justify-between"> | |
| <div className="flex items-center space-x-2"> | |
| <Brain className="w-5 h-5 text-purple-400" /> | |
| <h2 className="font-semibold text-gray-100">Checkpoints</h2> | |
| <span className="px-2 py-0.5 bg-gray-700 rounded-full text-xs text-gray-300">{files.length}</span> | |
| </div> | |
| </div> | |
| <div className="p-2"> | |
| {status === 'loading' && ( | |
| <div className="flex items-center justify-center py-4"> | |
| <Loader2 className="w-5 h-5 text-gray-400 animate-spin" /> | |
| </div> | |
| )} | |
| {status === 'error' && ( | |
| <div className="flex items-center justify-center py-4 text-rose-400 space-x-2"> | |
| <AlertCircle className="w-4 h-4" /> | |
| <span className="text-sm">Error loading checkpoints</span> | |
| </div> | |
| )} | |
| {['success', 'refreshing'].includes(status) && ( | |
| <div className="space-y-1"> | |
| {files.map((file, index) => { | |
| const fileName = file.path.split('/').pop() || ''; | |
| const nameWithoutExt = fileName.replace('.safetensors', ''); | |
| return ( | |
| <a | |
| key={index} | |
| target="_blank" | |
| href={`/api/files/${encodeURIComponent(file.path)}`} | |
| className="group flex items-center justify-between px-2 py-1.5 rounded-lg hover:bg-gray-800 transition-all duration-200" | |
| > | |
| <div className="flex items-center space-x-2 min-w-0"> | |
| <Box className="w-4 h-4 text-purple-400 flex-shrink-0" /> | |
| <div className="flex flex-col min-w-0"> | |
| <div className="flex text-sm text-gray-200"> | |
| <span className="overflow-hidden text-ellipsis direction-rtl whitespace-nowrap"> | |
| {nameWithoutExt} | |
| </span> | |
| </div> | |
| <span className="text-xs text-gray-500">.safetensors</span> | |
| </div> | |
| </div> | |
| <div className="flex items-center space-x-3 flex-shrink-0"> | |
| <span className="text-xs text-gray-400">{cleanSize(file.size)}</span> | |
| <div className="bg-purple-500 bg-opacity-0 group-hover:bg-opacity-10 rounded-full p-1 transition-all"> | |
| <Download className="w-3 h-3 text-purple-400" /> | |
| </div> | |
| </div> | |
| </a> | |
| ); | |
| })} | |
| </div> | |
| )} | |
| {['success', 'refreshing'].includes(status) && files.length === 0 && ( | |
| <div className="text-center py-4 text-gray-400 text-sm">No checkpoints available</div> | |
| )} | |
| </div> | |
| </div> | |
| ); | |
| } | |