mirror of
https://github.com/ostris/ai-toolkit.git
synced 2026-05-11 08:20:35 +00:00
Improve dataset uploader. Upload the files one at a time instead of one huge chunk. Show progress for each file.
This commit is contained in:
@@ -2,9 +2,7 @@
|
||||
import { createGlobalState } from 'react-global-hooks';
|
||||
import { Dialog, DialogBackdrop, DialogPanel, DialogTitle } from '@headlessui/react';
|
||||
import { FaUpload } from 'react-icons/fa';
|
||||
import { useCallback, useState } from 'react';
|
||||
import { useDropzone } from 'react-dropzone';
|
||||
import { apiClient } from '@/utils/api';
|
||||
import { useEffect, useRef } from 'react';
|
||||
|
||||
export interface AddImagesModalState {
|
||||
datasetName: string;
|
||||
@@ -19,69 +17,34 @@ export const openImagesModal = (datasetName: string, onComplete: () => void) =>
|
||||
|
||||
export default function AddImagesModal() {
|
||||
const [addImagesModalInfo, setAddImagesModalInfo] = addImagesModalState.use();
|
||||
const [uploadProgress, setUploadProgress] = useState<number>(0);
|
||||
const [isUploading, setIsUploading] = useState<boolean>(false);
|
||||
const open = addImagesModalInfo !== null;
|
||||
const panelRef = useRef<HTMLDivElement>(null);
|
||||
|
||||
const onCancel = () => {
|
||||
if (!isUploading) {
|
||||
setAddImagesModalInfo(null);
|
||||
}
|
||||
setAddImagesModalInfo(null);
|
||||
};
|
||||
|
||||
const onDone = () => {
|
||||
if (addImagesModalInfo?.onComplete && !isUploading) {
|
||||
if (addImagesModalInfo?.onComplete) {
|
||||
addImagesModalInfo.onComplete();
|
||||
setAddImagesModalInfo(null);
|
||||
}
|
||||
setAddImagesModalInfo(null);
|
||||
};
|
||||
|
||||
const onDrop = useCallback(
|
||||
async (acceptedFiles: File[]) => {
|
||||
if (acceptedFiles.length === 0) return;
|
||||
// Close modal as soon as files are dragged in so the FullscreenDropOverlay can handle the drop
|
||||
useEffect(() => {
|
||||
if (!open) return;
|
||||
|
||||
setIsUploading(true);
|
||||
setUploadProgress(0);
|
||||
|
||||
const formData = new FormData();
|
||||
acceptedFiles.forEach(file => {
|
||||
formData.append('files', file);
|
||||
});
|
||||
formData.append('datasetName', addImagesModalInfo?.datasetName || '');
|
||||
|
||||
try {
|
||||
await apiClient.post(`/api/datasets/upload`, formData, {
|
||||
headers: {
|
||||
'Content-Type': 'multipart/form-data',
|
||||
},
|
||||
onUploadProgress: progressEvent => {
|
||||
const percentCompleted = Math.round((progressEvent.loaded * 100) / (progressEvent.total || 100));
|
||||
setUploadProgress(percentCompleted);
|
||||
},
|
||||
timeout: 0, // Disable timeout
|
||||
});
|
||||
|
||||
onDone();
|
||||
} catch (error) {
|
||||
console.error('Upload failed:', error);
|
||||
} finally {
|
||||
setIsUploading(false);
|
||||
setUploadProgress(0);
|
||||
const handleDragEnter = (e: DragEvent) => {
|
||||
const types = e?.dataTransfer?.types;
|
||||
if (types && Array.from(types).includes('Files')) {
|
||||
setAddImagesModalInfo(null);
|
||||
}
|
||||
},
|
||||
[addImagesModalInfo],
|
||||
);
|
||||
};
|
||||
|
||||
const { getRootProps, getInputProps, isDragActive } = useDropzone({
|
||||
onDrop,
|
||||
accept: {
|
||||
'image/*': ['.png', '.jpg', '.jpeg', '.gif', '.bmp', '.webp'],
|
||||
'video/*': ['.mp4', '.avi', '.mov', '.mkv', '.wmv', '.m4v', '.flv'],
|
||||
'audio/*': ['.mp3', '.wav'],
|
||||
'text/*': ['.txt'],
|
||||
},
|
||||
multiple: true,
|
||||
});
|
||||
window.addEventListener('dragenter', handleDragEnter);
|
||||
return () => window.removeEventListener('dragenter', handleDragEnter);
|
||||
}, [open, setAddImagesModalInfo]);
|
||||
|
||||
return (
|
||||
<Dialog open={open} onClose={onCancel} className="relative z-10">
|
||||
@@ -93,6 +56,7 @@ export default function AddImagesModal() {
|
||||
<div className="fixed inset-0 z-10 w-screen overflow-y-auto">
|
||||
<div className="flex min-h-full items-end justify-center p-4 text-center sm:items-center sm:p-0">
|
||||
<DialogPanel
|
||||
ref={panelRef}
|
||||
transition
|
||||
className="relative transform overflow-hidden rounded-lg bg-gray-800 text-left shadow-xl transition-all data-closed:translate-y-4 data-closed:opacity-0 data-enter:duration-300 data-enter:ease-out data-leave:duration-200 data-leave:ease-in sm:my-8 sm:w-full sm:max-w-lg data-closed:sm:translate-y-0 data-closed:sm:scale-95"
|
||||
>
|
||||
@@ -103,24 +67,13 @@ export default function AddImagesModal() {
|
||||
</DialogTitle>
|
||||
<div className="w-full">
|
||||
<div
|
||||
{...getRootProps()}
|
||||
className={`h-40 w-full flex flex-col items-center justify-center border-2 border-dashed rounded-lg cursor-pointer transition-colors duration-200
|
||||
${isDragActive ? 'border-blue-500 bg-blue-50/10' : 'border-gray-600'}`}
|
||||
className="h-40 w-full flex flex-col items-center justify-center border-2 border-dashed rounded-lg border-gray-600"
|
||||
>
|
||||
<input {...getInputProps()} />
|
||||
<FaUpload className="size-8 mb-3 text-gray-400" />
|
||||
<p className="text-sm text-gray-200 text-center">
|
||||
{isDragActive ? 'Drop the files here...' : 'Drag & drop files here, or click to select files'}
|
||||
Drag & drop files anywhere on the page to upload
|
||||
</p>
|
||||
</div>
|
||||
{isUploading && (
|
||||
<div className="mt-4">
|
||||
<div className="w-full bg-gray-700 rounded-full h-2.5">
|
||||
<div className="bg-blue-600 h-2.5 rounded-full" style={{ width: `${uploadProgress}%` }}></div>
|
||||
</div>
|
||||
<p className="text-sm text-gray-300 mt-2 text-center">Uploading... {uploadProgress}%</p>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -128,9 +81,7 @@ export default function AddImagesModal() {
|
||||
<button
|
||||
type="button"
|
||||
onClick={onDone}
|
||||
disabled={isUploading}
|
||||
className={`inline-flex w-full justify-center rounded-md bg-slate-600 px-3 py-2 text-sm font-semibold text-white shadow-xs sm:ml-3 sm:w-auto
|
||||
${isUploading ? 'opacity-50 cursor-not-allowed' : ''}`}
|
||||
className="inline-flex w-full justify-center rounded-md bg-slate-600 px-3 py-2 text-sm font-semibold text-white shadow-xs sm:ml-3 sm:w-auto"
|
||||
>
|
||||
Done
|
||||
</button>
|
||||
@@ -138,9 +89,7 @@ export default function AddImagesModal() {
|
||||
type="button"
|
||||
data-autofocus
|
||||
onClick={onCancel}
|
||||
disabled={isUploading}
|
||||
className={`mt-3 inline-flex w-full justify-center rounded-md bg-gray-800 px-3 py-2 text-sm font-semibold text-gray-200 hover:bg-gray-800 sm:mt-0 sm:w-auto ring-0
|
||||
${isUploading ? 'opacity-50 cursor-not-allowed' : ''}`}
|
||||
className="mt-3 inline-flex w-full justify-center rounded-md bg-gray-800 px-3 py-2 text-sm font-semibold text-gray-200 hover:bg-gray-800 sm:mt-0 sm:w-auto ring-0"
|
||||
>
|
||||
Cancel
|
||||
</button>
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||
import { useDropzone } from 'react-dropzone';
|
||||
import { FaUpload } from 'react-icons/fa';
|
||||
import { FaUpload, FaTimesCircle, FaSpinner } from 'react-icons/fa';
|
||||
import { apiClient } from '@/utils/api';
|
||||
|
||||
type AcceptMap = {
|
||||
@@ -10,12 +10,28 @@ type AcceptMap = {
|
||||
};
|
||||
|
||||
interface FullscreenDropOverlayProps {
|
||||
datasetName: string; // where to upload
|
||||
onComplete?: () => void; // called after successful upload
|
||||
accept?: AcceptMap; // optional override
|
||||
multiple?: boolean; // default true
|
||||
datasetName: string;
|
||||
onComplete?: () => void;
|
||||
accept?: AcceptMap;
|
||||
multiple?: boolean;
|
||||
}
|
||||
|
||||
type FileStatus = 'pending' | 'uploading' | 'error';
|
||||
|
||||
interface FileEntry {
|
||||
id: number;
|
||||
file: File;
|
||||
status: FileStatus;
|
||||
progress: number;
|
||||
error?: string;
|
||||
}
|
||||
|
||||
const MAX_CONCURRENT = 3;
|
||||
const ROW_HEIGHT = 32;
|
||||
const VISIBLE_ROWS = 8;
|
||||
|
||||
let nextId = 0;
|
||||
|
||||
export default function FullscreenDropOverlay({
|
||||
datasetName,
|
||||
onComplete,
|
||||
@@ -24,16 +40,18 @@ export default function FullscreenDropOverlay({
|
||||
}: FullscreenDropOverlayProps) {
|
||||
const [visible, setVisible] = useState(false);
|
||||
const [isUploading, setIsUploading] = useState(false);
|
||||
const [uploadProgress, setUploadProgress] = useState(0);
|
||||
const dragDepthRef = useRef(0); // drag-enter/leave tracking
|
||||
const [fileEntries, setFileEntries] = useState<FileEntry[]>([]);
|
||||
const [totalCount, setTotalCount] = useState(0);
|
||||
const [doneCount, setDoneCount] = useState(0);
|
||||
const [errorCount, setErrorCount] = useState(0);
|
||||
const dragDepthRef = useRef(0);
|
||||
const abortRef = useRef(false);
|
||||
|
||||
// Only show the overlay for real file drags (not text, images from page, etc)
|
||||
const isFileDrag = (e: DragEvent) => {
|
||||
const types = e?.dataTransfer?.types;
|
||||
return !!types && Array.from(types).includes('Files');
|
||||
};
|
||||
|
||||
// Window-level drag listeners to toggle visibility
|
||||
useEffect(() => {
|
||||
const onDragEnter = (e: DragEvent) => {
|
||||
if (!isFileDrag(e)) return;
|
||||
@@ -43,7 +61,6 @@ export default function FullscreenDropOverlay({
|
||||
};
|
||||
const onDragOver = (e: DragEvent) => {
|
||||
if (!isFileDrag(e)) return;
|
||||
// Must preventDefault to allow dropping in the browser
|
||||
e.preventDefault();
|
||||
if (!visible) setVisible(true);
|
||||
};
|
||||
@@ -54,63 +71,131 @@ export default function FullscreenDropOverlay({
|
||||
setVisible(false);
|
||||
}
|
||||
};
|
||||
const onDrop = (e: DragEvent) => {
|
||||
const onWindowDrop = (e: DragEvent) => {
|
||||
if (!isFileDrag(e)) return;
|
||||
// Prevent browser from opening the file
|
||||
e.preventDefault();
|
||||
dragDepthRef.current = 0;
|
||||
// We do NOT hide here; the dropzone onDrop will handle workflow visibility.
|
||||
};
|
||||
|
||||
window.addEventListener('dragenter', onDragEnter);
|
||||
window.addEventListener('dragover', onDragOver);
|
||||
window.addEventListener('dragleave', onDragLeave);
|
||||
window.addEventListener('drop', onDrop);
|
||||
window.addEventListener('drop', onWindowDrop);
|
||||
|
||||
return () => {
|
||||
window.removeEventListener('dragenter', onDragEnter);
|
||||
window.removeEventListener('dragover', onDragOver);
|
||||
window.removeEventListener('dragleave', onDragLeave);
|
||||
window.removeEventListener('drop', onDrop);
|
||||
window.removeEventListener('drop', onWindowDrop);
|
||||
};
|
||||
}, [visible, isUploading]);
|
||||
|
||||
const uploadSingleFile = useCallback(
|
||||
async (entry: FileEntry): Promise<'done' | 'error'> => {
|
||||
if (abortRef.current) return 'error';
|
||||
|
||||
const id = entry.id;
|
||||
|
||||
setFileEntries(prev =>
|
||||
prev.map(e => (e.id === id ? { ...e, status: 'uploading' as FileStatus, progress: 0 } : e)),
|
||||
);
|
||||
|
||||
const formData = new FormData();
|
||||
formData.append('files', entry.file);
|
||||
formData.append('datasetName', datasetName || '');
|
||||
|
||||
try {
|
||||
await apiClient.post('/api/datasets/upload', formData, {
|
||||
headers: { 'Content-Type': 'multipart/form-data' },
|
||||
onUploadProgress: pe => {
|
||||
const percent = Math.round(((pe.loaded || 0) * 100) / (pe.total || pe.loaded || 1));
|
||||
setFileEntries(prev => prev.map(e => (e.id === id ? { ...e, progress: percent } : e)));
|
||||
},
|
||||
timeout: 0,
|
||||
});
|
||||
// Remove from list on success
|
||||
setFileEntries(prev => prev.filter(e => e.id !== id));
|
||||
setDoneCount(prev => prev + 1);
|
||||
return 'done';
|
||||
} catch (err) {
|
||||
console.error(`Upload failed for ${entry.file.name}:`, err);
|
||||
setFileEntries(prev =>
|
||||
prev.map(e =>
|
||||
e.id === id
|
||||
? { ...e, status: 'error' as FileStatus, error: err instanceof Error ? err.message : 'Upload failed' }
|
||||
: e,
|
||||
),
|
||||
);
|
||||
setErrorCount(prev => prev + 1);
|
||||
return 'error';
|
||||
}
|
||||
},
|
||||
[datasetName],
|
||||
);
|
||||
|
||||
const processQueue = useCallback(
|
||||
async (entries: FileEntry[]) => {
|
||||
setIsUploading(true);
|
||||
abortRef.current = false;
|
||||
|
||||
let nextIndex = 0;
|
||||
|
||||
const runNext = async (): Promise<void> => {
|
||||
while (nextIndex < entries.length) {
|
||||
if (abortRef.current) return;
|
||||
const idx = nextIndex++;
|
||||
await uploadSingleFile(entries[idx]);
|
||||
}
|
||||
};
|
||||
|
||||
const workers = Array.from({ length: Math.min(MAX_CONCURRENT, entries.length) }, () => runNext());
|
||||
await Promise.all(workers);
|
||||
|
||||
setIsUploading(false);
|
||||
if (!abortRef.current) {
|
||||
onComplete?.();
|
||||
setFileEntries([]);
|
||||
setVisible(false);
|
||||
setTotalCount(0);
|
||||
setDoneCount(0);
|
||||
setErrorCount(0);
|
||||
}
|
||||
},
|
||||
[uploadSingleFile, onComplete],
|
||||
);
|
||||
|
||||
const onDrop = useCallback(
|
||||
async (acceptedFiles: File[]) => {
|
||||
(acceptedFiles: File[]) => {
|
||||
if (acceptedFiles.length === 0) {
|
||||
// no accepted files; hide overlay cleanly
|
||||
setVisible(false);
|
||||
return;
|
||||
}
|
||||
|
||||
setIsUploading(true);
|
||||
setUploadProgress(0);
|
||||
|
||||
const formData = new FormData();
|
||||
acceptedFiles.forEach(file => formData.append('files', file));
|
||||
formData.append('datasetName', datasetName || '');
|
||||
|
||||
try {
|
||||
await apiClient.post(`/api/datasets/upload`, formData, {
|
||||
headers: { 'Content-Type': 'multipart/form-data' },
|
||||
onUploadProgress: pe => {
|
||||
const percent = Math.round(((pe.loaded || 0) * 100) / (pe.total || pe.loaded || 1));
|
||||
setUploadProgress(percent);
|
||||
},
|
||||
timeout: 0,
|
||||
});
|
||||
onComplete?.();
|
||||
} catch (err) {
|
||||
console.error('Upload failed:', err);
|
||||
} finally {
|
||||
setIsUploading(false);
|
||||
setUploadProgress(0);
|
||||
setVisible(false);
|
||||
}
|
||||
const entries: FileEntry[] = acceptedFiles.map(file => ({
|
||||
id: nextId++,
|
||||
file,
|
||||
status: 'pending' as FileStatus,
|
||||
progress: 0,
|
||||
}));
|
||||
setFileEntries(entries);
|
||||
setTotalCount(entries.length);
|
||||
setDoneCount(0);
|
||||
setErrorCount(0);
|
||||
processQueue(entries);
|
||||
},
|
||||
[datasetName, onComplete],
|
||||
[processQueue],
|
||||
);
|
||||
|
||||
const handleClose = useCallback(() => {
|
||||
abortRef.current = true;
|
||||
setIsUploading(false);
|
||||
setFileEntries([]);
|
||||
setVisible(false);
|
||||
setTotalCount(0);
|
||||
setDoneCount(0);
|
||||
setErrorCount(0);
|
||||
}, []);
|
||||
|
||||
const dropAccept = useMemo<AcceptMap>(
|
||||
() =>
|
||||
accept || {
|
||||
@@ -128,56 +213,134 @@ export default function FullscreenDropOverlay({
|
||||
multiple,
|
||||
noClick: true,
|
||||
noKeyboard: true,
|
||||
// Prevent "folder opens" by browser if someone drags outside the overlay mid-drop:
|
||||
preventDropOnDocument: true,
|
||||
});
|
||||
|
||||
const overallPercent = totalCount > 0 ? Math.round(((doneCount + errorCount) / totalCount) * 100) : 0;
|
||||
|
||||
return (
|
||||
<div
|
||||
// When hidden: opacity-0 + pointer-events-none so the page is fully interactive
|
||||
// When visible or uploading: fade in and capture the drop
|
||||
className={`fixed inset-0 z-[9999] transition-opacity duration-200 ${
|
||||
visible || isUploading ? 'opacity-100 pointer-events-auto' : 'opacity-0 pointer-events-none'
|
||||
}`}
|
||||
aria-hidden={!visible && !isUploading}
|
||||
{...getRootProps()}
|
||||
>
|
||||
{/* Fullscreen capture layer */}
|
||||
<input {...getInputProps()} />
|
||||
|
||||
{/* Backdrop: keep it subtle so context remains visible */}
|
||||
<div className={`absolute inset-0 ${isUploading ? 'bg-gray-900/70' : 'bg-gray-900/40'}`} />
|
||||
|
||||
{/* Center drop target UI */}
|
||||
<div className="absolute inset-0 flex items-center justify-center p-6">
|
||||
<div
|
||||
className={`w-full max-w-2xl rounded-2xl border-2 border-dashed px-8 py-10 text-center shadow-2xl backdrop-blur-sm
|
||||
${isDragActive ? 'border-blue-400 bg-white/10' : 'border-white/30 bg-white/5'}`}
|
||||
>
|
||||
<div className="flex flex-col items-center gap-4">
|
||||
<FaUpload className="size-10 opacity-80" />
|
||||
{!isUploading ? (
|
||||
<>
|
||||
<p className="text-lg font-semibold">Drop files to upload</p>
|
||||
<p className="text-sm opacity-80">
|
||||
Destination: <span className="font-mono">{datasetName || 'unknown'}</span>
|
||||
</p>
|
||||
<p className="text-xs opacity-70 mt-1">Images, videos, or .txt supported</p>
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<p className="text-lg font-semibold">Uploading… {uploadProgress}%</p>
|
||||
<div className="w-full h-2.5 bg-white/20 rounded-full overflow-hidden">
|
||||
<div
|
||||
className="h-2.5 bg-blue-500 rounded-full transition-[width] duration-150 ease-linear"
|
||||
style={{ width: `${uploadProgress}%` }}
|
||||
/>
|
||||
</div>
|
||||
</>
|
||||
)}
|
||||
<div className="w-full max-w-2xl flex flex-col gap-3">
|
||||
{/* Drop target / status box */}
|
||||
<div
|
||||
className={`rounded-2xl border-2 border-dashed px-8 py-10 text-center shadow-2xl backdrop-blur-sm
|
||||
${isDragActive ? 'border-blue-400 bg-white/10' : 'border-white/30 bg-white/5'}`}
|
||||
>
|
||||
<div className="flex flex-col items-center gap-4">
|
||||
<FaUpload className="size-10 opacity-80" />
|
||||
{!isUploading ? (
|
||||
<>
|
||||
<p className="text-lg font-semibold">Drop files to upload</p>
|
||||
<p className="text-sm opacity-80">
|
||||
Destination: <span className="font-mono">{datasetName || 'unknown'}</span>
|
||||
</p>
|
||||
<p className="text-xs opacity-70 mt-1">Images, videos, or .txt supported</p>
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<p className="text-lg font-semibold">
|
||||
Uploading… {doneCount + errorCount} / {totalCount}
|
||||
</p>
|
||||
<div className="w-full h-2.5 bg-white/20 rounded-full overflow-hidden">
|
||||
<div
|
||||
className="h-2.5 bg-blue-500 rounded-full transition-[width] duration-150 ease-linear"
|
||||
style={{ width: `${overallPercent}%` }}
|
||||
/>
|
||||
</div>
|
||||
{errorCount > 0 && (
|
||||
<p className="text-xs text-red-400">
|
||||
{errorCount} file{errorCount !== 1 ? 's' : ''} failed
|
||||
</p>
|
||||
)}
|
||||
<button
|
||||
type="button"
|
||||
onClick={e => {
|
||||
e.stopPropagation();
|
||||
handleClose();
|
||||
}}
|
||||
className="mt-2 px-4 py-1.5 text-sm rounded-lg bg-white/10 hover:bg-white/20 transition-colors"
|
||||
>
|
||||
Cancel
|
||||
</button>
|
||||
</>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* File progress list — only shows pending, uploading, and errored files */}
|
||||
{fileEntries.length > 0 && <FileProgressList entries={fileEntries} />}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
/** Virtualized file progress list for handling thousands of files */
|
||||
function FileProgressList({ entries }: { entries: FileEntry[] }) {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [scrollTop, setScrollTop] = useState(0);
|
||||
|
||||
const totalHeight = entries.length * ROW_HEIGHT;
|
||||
const containerHeight = Math.min(entries.length, VISIBLE_ROWS) * ROW_HEIGHT;
|
||||
|
||||
const startIdx = Math.floor(scrollTop / ROW_HEIGHT);
|
||||
const endIdx = Math.min(entries.length, startIdx + VISIBLE_ROWS + 2);
|
||||
const visibleEntries = entries.slice(startIdx, endIdx);
|
||||
const offsetY = startIdx * ROW_HEIGHT;
|
||||
|
||||
const onScroll = useCallback(() => {
|
||||
if (containerRef.current) {
|
||||
setScrollTop(containerRef.current.scrollTop);
|
||||
}
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={containerRef}
|
||||
onScroll={onScroll}
|
||||
onClick={e => e.stopPropagation()}
|
||||
className="rounded-xl bg-black/60 backdrop-blur-sm border border-white/10 overflow-y-auto"
|
||||
style={{ height: containerHeight + 2 }}
|
||||
>
|
||||
<div style={{ height: totalHeight, position: 'relative' }}>
|
||||
<div style={{ position: 'absolute', top: offsetY, left: 0, right: 0 }}>
|
||||
{visibleEntries.map(entry => (
|
||||
<FileRow key={entry.id} entry={entry} />
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function FileRow({ entry }: { entry: FileEntry }) {
|
||||
return (
|
||||
<div className="flex items-center gap-2 px-3 text-xs font-mono" style={{ height: ROW_HEIGHT }}>
|
||||
<span className="flex-shrink-0 w-4 text-center">
|
||||
{entry.status === 'error' && <FaTimesCircle className="text-red-400 inline" />}
|
||||
{entry.status === 'uploading' && <FaSpinner className="text-blue-400 inline animate-spin" />}
|
||||
{entry.status === 'pending' && <span className="inline-block w-2 h-2 rounded-full bg-white/30" />}
|
||||
</span>
|
||||
|
||||
<span className="truncate flex-1 opacity-80" title={entry.file.name}>
|
||||
{entry.file.name}
|
||||
</span>
|
||||
|
||||
<span className="flex-shrink-0 w-16 text-right">
|
||||
{entry.status === 'uploading' && <span className="text-blue-300">{entry.progress}%</span>}
|
||||
{entry.status === 'error' && <span className="text-red-400">Failed</span>}
|
||||
{entry.status === 'pending' && <span className="text-white/30">Queued</span>}
|
||||
</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user