import { useState, useEffect, useCallback } from 'react' import { motion, AnimatePresence } from 'framer-motion' import { Search, ChevronLeft, ChevronRight, CheckCircle2, Clock, User, BookOpen, FileText, Calendar, Hash, Loader2, X, Save, RefreshCw, Image as ImageIcon, } from 'lucide-react' import axios from 'axios' const API_BASE = import.meta.env.VITE_API_URL || '/api' const STATUS_COLORS = { unreviewed: 'text-amber-400 bg-amber-400/10 border-amber-400/30', reviewed: 'text-green-400 bg-green-400/10 border-green-400/30', } const STATUS_ICONS = { unreviewed: Clock, reviewed: CheckCircle2, } function StatusBadge({ status }) { const Icon = STATUS_ICONS[status] || Clock return ( {status} ) } function JobDetail({ jobId, onClose, onReviewed }) { const [job, setJob] = useState(null) const [loading, setLoading] = useState(true) const [error, setError] = useState(null) // Editable fields const [editedText, setEditedText] = useState('') const [editAuthor, setEditAuthor] = useState('') const [editBook, setEditBook] = useState('') const [editChapter, setEditChapter] = useState('') const [editPage, setEditPage] = useState('') const [reviewerName, setReviewerName] = useState('') const [submitting, setSubmitting] = useState(false) const [saveResult, setSaveResult] = useState(null) useEffect(() => { let cancelled = false setLoading(true) setError(null) setSaveResult(null) axios.get(`${API_BASE}/jobs/${jobId}`) .then(res => { if (!cancelled) { const d = res.data setJob(d) setEditedText(d.reviewed_text ?? d.ocr_text ?? '') setEditAuthor(d.author || '') setEditBook(d.book || '') setEditChapter(d.chapter || '') setEditPage(d.page || '') setReviewerName(d.reviewer_name || '') } }) .catch(err => { if (!cancelled) setError(err.response?.data?.detail || err.message) }) .finally(() => { if (!cancelled) setLoading(false) }) return () => { cancelled = true } }, [jobId]) const handleSave = async () => { if (!reviewerName.trim()) { setSaveResult({ success: false, error: 'Reviewer name is required.' }) return } setSubmitting(true) setSaveResult(null) try { const res = await axios.put(`${API_BASE}/jobs/${jobId}/review`, { reviewed_text: editedText, reviewer_name: reviewerName.trim(), author: editAuthor, book: editBook, chapter: editChapter, page: editPage, }) setJob(res.data) setSaveResult({ success: true }) onReviewed(res.data) } catch (err) { setSaveResult({ success: false, error: err.response?.data?.detail || err.message }) } finally { setSubmitting(false) } } const inputClass = 'w-full bg-white/5 border border-white/10 rounded-lg px-3 py-2 text-sm text-gray-200 ' + 'placeholder-gray-600 focus:outline-none focus:border-purple-500/50 transition-colors' const isReviewed = job?.status === 'reviewed' return (
{/* Header */}
{job && }

Job Detail

{loading && (
)} {error && (

{error}

)} {job && !loading && (
{/* ── Left column: image + read-only info ── */}

Source Image

Job source { e.target.style.display = 'none' }} />
{/* Read-only job info */}

{job.id}

Submitted: {new Date(job.submitted_at).toLocaleString()}

{job.mode &&

Mode: {job.mode}

} {isReviewed && job.reviewed_at && (

Last reviewed: {new Date(job.reviewed_at).toLocaleString()}

)}
{/* Original OCR text (collapsed, for reviewed jobs) */} {isReviewed && job.ocr_text && (
Original OCR Text
{job.ocr_text}
)}
{/* ── Right column: all editable fields ── */}
{/* Metadata */}

Metadata

setEditAuthor(e.target.value)} placeholder="Author name" className={inputClass} />
setEditBook(e.target.value)} placeholder="Book title" className={inputClass} />
setEditChapter(e.target.value)} placeholder="Chapter" className={inputClass} />
setEditPage(e.target.value)} placeholder="Page" className={inputClass} />
{/* OCR / reviewed text */}