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 */}
{loading && (
)}
{error && (
)}
{job && !loading && (
{/* ── Left column: image + read-only info ── */}
Source Image

{ 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
)}
{/* ── Right column: all editable fields ── */}
{/* Metadata */}
{/* OCR / reviewed text */}
{/* Reviewer + save */}
setReviewerName(e.target.value)}
placeholder="Your name"
className={inputClass}
/>
{saveResult && (
{saveResult.success
? (isReviewed ? 'Changes saved!' : 'Job marked as reviewed!')
: saveResult.error}
)}
{submitting ? (
<> Saving...>
) : isReviewed ? (
<> Save Changes>
) : (
<> Mark Reviewed>
)}
)}
)
}
export default function JobsPanel() {
const [search, setSearch] = useState('')
const [filterStatus, setFilterStatus] = useState('')
const [filterAuthor, setFilterAuthor] = useState('')
const [filterBook, setFilterBook] = useState('')
const [jobs, setJobs] = useState([])
const [total, setTotal] = useState(0)
const [page, setPage] = useState(0)
const [loading, setLoading] = useState(false)
const [error, setError] = useState(null)
const [selectedJobId, setSelectedJobId] = useState(null)
const LIMIT = 20
const fetchJobs = useCallback(async (pageNum = 0) => {
setLoading(true)
setError(null)
try {
const params = new URLSearchParams()
if (search.trim()) params.set('search', search.trim())
if (filterStatus) params.set('status', filterStatus)
if (filterAuthor.trim()) params.set('author', filterAuthor.trim())
if (filterBook.trim()) params.set('book', filterBook.trim())
params.set('limit', LIMIT)
params.set('offset', pageNum * LIMIT)
const res = await axios.get(`${API_BASE}/jobs?${params}`)
setJobs(res.data.jobs)
setTotal(res.data.total)
setPage(pageNum)
} catch (err) {
setError(err.response?.data?.detail || err.message)
} finally {
setLoading(false)
}
}, [search, filterStatus, filterAuthor, filterBook])
useEffect(() => {
fetchJobs(0)
}, []) // eslint-disable-line react-hooks/exhaustive-deps
const handleSearch = (e) => {
e.preventDefault()
fetchJobs(0)
}
const handleReviewed = (updatedJob) => {
setJobs(prev => prev.map(j => j.id === updatedJob.id ? { ...j, ...updatedJob } : j))
}
const totalPages = Math.ceil(total / LIMIT)
const inputClass =
'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'
return (
// 1/3 list — 2/3 detail on large screens
{/* ── Left: Search + List ── */}
{loading && (
)}
{error && (
)}
{!loading && !error && jobs.length === 0 && (
No jobs found
Commit your first OCR job from the New Job tab
)}
{jobs.map(job => (
setSelectedJobId(job.id === selectedJobId ? null : job.id)}
className={`w-full text-left glass p-4 rounded-xl transition-all border ${
selectedJobId === job.id
? 'border-purple-500/50 bg-purple-500/5'
: 'border-white/5 hover:border-white/20 hover:bg-white/5'
}`}
initial={{ opacity: 0, y: 10 }}
animate={{ opacity: 1, y: 0 }}
exit={{ opacity: 0 }}
layout
>
{job.book && (
{job.book}
)}
{job.chapter && (
Ch. {job.chapter}
)}
{job.page && (
p. {job.page}
)}
{job.author && (
{job.author}
)}
{new Date(job.submitted_at).toLocaleString()}
))}
{totalPages > 1 && (
Page {page + 1} of {totalPages}
)}
{/* ── Right: Detail panel ── */}
{selectedJobId ? (
setSelectedJobId(null)}
onReviewed={handleReviewed}
/>
) : (
Select a job to view and edit details
)}
)
}