Side-by-side image/text layout and editable metadata on review

New Job page:
- OCR result now shows source image and editable textarea side by side
- Grounding-box overlay preview moved into the non-commit branch

Browse Jobs / Review page:
- JobDetail uses a 2-column layout: image + read-only info on left,
  all editable fields on right
- Author, book, chapter, and page are now editable inputs (not read-only)
- Text textarea is always editable (for both unreviewed and reviewed jobs)
- Reviewer name pre-filled for reviewed jobs; button becomes "Save Changes"
- Outer grid changed to 1/3 list + 2/3 detail for more review space

Backend:
- PUT /api/jobs/{id}/review now accepts and saves author, book,
  chapter, page alongside reviewed_text and reviewer_name

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Aaron Roberts
2026-06-09 17:38:36 +01:00
parent da7957d7d5
commit 9356ba6d1b
3 changed files with 281 additions and 204 deletions

View File

@@ -603,6 +603,10 @@ async def process_pdf(
class ReviewRequest(BaseModel): class ReviewRequest(BaseModel):
reviewed_text: str reviewed_text: str
reviewer_name: str reviewer_name: str
author: Optional[str] = None
book: Optional[str] = None
chapter: Optional[str] = None
page: Optional[str] = None
def _job_row_to_dict(row) -> Dict[str, Any]: def _job_row_to_dict(row) -> Dict[str, Any]:
@@ -811,11 +815,23 @@ async def review_job(job_id: str, body: ReviewRequest):
SET status = 'reviewed', SET status = 'reviewed',
reviewed_text = %s, reviewed_text = %s,
reviewer_name = %s, reviewer_name = %s,
reviewed_at = NOW() reviewed_at = NOW(),
author = %s,
book = %s,
chapter = %s,
page = %s
WHERE id = %s WHERE id = %s
RETURNING * RETURNING *
""", """,
(body.reviewed_text, body.reviewer_name, job_id), (
body.reviewed_text,
body.reviewer_name,
body.author or None,
body.book or None,
body.chapter or None,
body.page or None,
job_id,
),
) )
row = cur.fetchone() row = cur.fetchone()
except Exception as exc: except Exception as exc:

View File

@@ -29,37 +29,39 @@ function StatusBadge({ status }) {
) )
} }
function MetaRow({ icon: Icon, label, value }) {
if (!value) return null
return (
<div className="flex items-start gap-2 text-sm">
<Icon className="w-4 h-4 text-purple-400 mt-0.5 flex-shrink-0" />
<span className="text-gray-400 flex-shrink-0">{label}:</span>
<span className="text-gray-200">{value}</span>
</div>
)
}
function JobDetail({ jobId, onClose, onReviewed }) { function JobDetail({ jobId, onClose, onReviewed }) {
const [job, setJob] = useState(null) const [job, setJob] = useState(null)
const [loading, setLoading] = useState(true) const [loading, setLoading] = useState(true)
const [error, setError] = useState(null) const [error, setError] = useState(null)
// Editable fields
const [editedText, setEditedText] = useState('') 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 [reviewerName, setReviewerName] = useState('')
const [submitting, setSubmitting] = useState(false) const [submitting, setSubmitting] = useState(false)
const [reviewResult, setReviewResult] = useState(null) const [saveResult, setSaveResult] = useState(null)
useEffect(() => { useEffect(() => {
let cancelled = false let cancelled = false
setLoading(true) setLoading(true)
setError(null) setError(null)
setReviewResult(null) setSaveResult(null)
axios.get(`${API_BASE}/jobs/${jobId}`) axios.get(`${API_BASE}/jobs/${jobId}`)
.then(res => { .then(res => {
if (!cancelled) { if (!cancelled) {
setJob(res.data) const d = res.data
setEditedText(res.data.reviewed_text ?? res.data.ocr_text ?? '') 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 => { .catch(err => {
@@ -72,23 +74,27 @@ function JobDetail({ jobId, onClose, onReviewed }) {
return () => { cancelled = true } return () => { cancelled = true }
}, [jobId]) }, [jobId])
const handleMarkReviewed = async () => { const handleSave = async () => {
if (!reviewerName.trim()) { if (!reviewerName.trim()) {
setReviewResult({ success: false, error: 'Reviewer name is required.' }) setSaveResult({ success: false, error: 'Reviewer name is required.' })
return return
} }
setSubmitting(true) setSubmitting(true)
setReviewResult(null) setSaveResult(null)
try { try {
const res = await axios.put(`${API_BASE}/jobs/${jobId}/review`, { const res = await axios.put(`${API_BASE}/jobs/${jobId}/review`, {
reviewed_text: editedText, reviewed_text: editedText,
reviewer_name: reviewerName.trim(), reviewer_name: reviewerName.trim(),
author: editAuthor,
book: editBook,
chapter: editChapter,
page: editPage,
}) })
setJob(res.data) setJob(res.data)
setReviewResult({ success: true }) setSaveResult({ success: true })
onReviewed(res.data) onReviewed(res.data)
} catch (err) { } catch (err) {
setReviewResult({ success: false, error: err.response?.data?.detail || err.message }) setSaveResult({ success: false, error: err.response?.data?.detail || err.message })
} finally { } finally {
setSubmitting(false) setSubmitting(false)
} }
@@ -98,48 +104,38 @@ function JobDetail({ jobId, onClose, onReviewed }) {
'w-full bg-white/5 border border-white/10 rounded-lg px-3 py-2 text-sm text-gray-200 ' + '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' 'placeholder-gray-600 focus:outline-none focus:border-purple-500/50 transition-colors'
const isReviewed = job?.status === 'reviewed'
return ( return (
<div className="glass rounded-2xl flex flex-col h-full overflow-hidden"> <div className="glass rounded-2xl flex flex-col overflow-hidden" style={{ minHeight: '600px' }}>
{/* Header */} {/* Header */}
<div className="flex items-center justify-between px-5 py-4 border-b border-white/10 flex-shrink-0"> <div className="flex items-center justify-between px-5 py-4 border-b border-white/10 flex-shrink-0">
<h3 className="font-semibold text-gray-200">Job Detail</h3> <div className="flex items-center gap-3">
{job && <StatusBadge status={job.status} />}
<h3 className="font-semibold text-gray-200">Job Detail</h3>
</div>
<button onClick={onClose} className="glass glass-hover p-1.5 rounded-lg"> <button onClick={onClose} className="glass glass-hover p-1.5 rounded-lg">
<X className="w-4 h-4" /> <X className="w-4 h-4" />
</button> </button>
</div> </div>
<div className="flex-1 overflow-y-auto p-5 space-y-5"> {loading && (
{loading && ( <div className="flex justify-center py-16">
<div className="flex justify-center py-12"> <Loader2 className="w-8 h-8 animate-spin text-purple-400" />
<Loader2 className="w-8 h-8 animate-spin text-purple-400" /> </div>
</div> )}
)}
{error && ( {error && (
<div className="glass p-4 rounded-xl border-red-500/30 bg-red-500/10"> <div className="m-5 glass p-4 rounded-xl border-red-500/30 bg-red-500/10">
<p className="text-sm text-red-400">{error}</p> <p className="text-sm text-red-400">{error}</p>
</div> </div>
)} )}
{job && !loading && ( {job && !loading && (
<> <div className="grid grid-cols-2 divide-x divide-white/10 flex-1 min-h-0">
{/* Status + IDs */}
<div className="flex items-center gap-3 flex-wrap">
<StatusBadge status={job.status} />
<span className="text-xs text-gray-500 font-mono">{job.id}</span>
</div>
{/* Metadata */} {/* ── Left column: image + read-only info ── */}
<div className="glass p-4 rounded-xl space-y-2"> <div className="overflow-y-auto p-5 space-y-4">
<MetaRow icon={User} label="Author" value={job.author} />
<MetaRow icon={BookOpen} label="Book" value={job.book} />
<MetaRow icon={Hash} label="Chapter" value={job.chapter} />
<MetaRow icon={FileText} label="Page" value={job.page} />
<MetaRow icon={Calendar} label="Submitted" value={job.submitted_at ? new Date(job.submitted_at).toLocaleString() : null} />
{job.mode && <MetaRow icon={FileText} label="Mode" value={job.mode} />}
</div>
{/* Image */}
<div> <div>
<p className="text-xs text-gray-400 mb-2 flex items-center gap-1"> <p className="text-xs text-gray-400 mb-2 flex items-center gap-1">
<ImageIcon className="w-3.5 h-3.5" /> Source Image <ImageIcon className="w-3.5 h-3.5" /> Source Image
@@ -152,93 +148,141 @@ function JobDetail({ jobId, onClose, onReviewed }) {
/> />
</div> </div>
{/* OCR / Reviewed text */} {/* Read-only job info */}
<div> <div className="space-y-1.5 text-xs text-gray-500">
<p className="text-xs text-gray-400 mb-2"> <p className="font-mono break-all">{job.id}</p>
{job.status === 'reviewed' ? 'Reviewed Text' : 'OCR Text (editable)'} <p>Submitted: {new Date(job.submitted_at).toLocaleString()}</p>
</p> {job.mode && <p>Mode: {job.mode}</p>}
{job.status === 'reviewed' ? ( {isReviewed && job.reviewed_at && (
<div className="bg-white/5 border border-white/10 rounded-xl p-4 max-h-60 overflow-y-auto"> <p>Last reviewed: {new Date(job.reviewed_at).toLocaleString()}</p>
<pre className="text-sm text-gray-200 whitespace-pre-wrap font-mono">
{job.reviewed_text}
</pre>
</div>
) : (
<textarea
value={editedText}
onChange={e => setEditedText(e.target.value)}
rows={8}
className={`${inputClass} resize-y font-mono`}
placeholder="OCR text will appear here for editing..."
/>
)} )}
</div> </div>
{/* Original OCR text (collapsed) for reviewed jobs */} {/* Original OCR text (collapsed, for reviewed jobs) */}
{job.status === 'reviewed' && job.ocr_text && ( {isReviewed && job.ocr_text && (
<details className="glass rounded-xl overflow-hidden"> <details className="glass rounded-xl overflow-hidden">
<summary className="px-4 py-3 cursor-pointer text-sm text-gray-400 hover:bg-white/5 transition-colors"> <summary className="px-3 py-2 cursor-pointer text-xs text-gray-500 hover:bg-white/5 transition-colors">
Original OCR Text Original OCR Text
</summary> </summary>
<div className="px-4 py-3 border-t border-white/10"> <div className="px-3 py-3 border-t border-white/10">
<pre className="text-sm text-gray-500 whitespace-pre-wrap font-mono"> <pre className="text-xs text-gray-500 whitespace-pre-wrap font-mono">{job.ocr_text}</pre>
{job.ocr_text}
</pre>
</div> </div>
</details> </details>
)} )}
</div>
{/* Review info for reviewed jobs */} {/* ── Right column: all editable fields ── */}
{job.status === 'reviewed' && ( <div className="overflow-y-auto p-5 space-y-5">
<div className="glass p-4 rounded-xl space-y-2">
<MetaRow icon={User} label="Reviewer" value={job.reviewer_name} /> {/* Metadata */}
<MetaRow icon={Calendar} label="Reviewed" value={job.reviewed_at ? new Date(job.reviewed_at).toLocaleString() : null} /> <div className="space-y-3">
<p className="text-xs font-semibold text-gray-400 uppercase tracking-wider">Metadata</p>
<div>
<label className="text-xs text-gray-400 mb-1 block">Author</label>
<input
type="text"
value={editAuthor}
onChange={e => setEditAuthor(e.target.value)}
placeholder="Author name"
className={inputClass}
/>
</div> </div>
)} <div>
<label className="text-xs text-gray-400 mb-1 block">Book</label>
{/* Mark Reviewed form */} <input
{job.status === 'unreviewed' && ( type="text"
<div className="glass p-4 rounded-xl space-y-3 border border-purple-500/20"> value={editBook}
<p className="text-sm font-medium text-gray-300">Mark as Reviewed</p> onChange={e => setEditBook(e.target.value)}
placeholder="Book title"
className={inputClass}
/>
</div>
<div className="grid grid-cols-2 gap-2">
<div> <div>
<label className="text-xs text-gray-400 mb-1 block">Reviewer Name</label> <label className="text-xs text-gray-400 mb-1 block">Chapter</label>
<input <input
type="text" type="text"
value={reviewerName} value={editChapter}
onChange={e => setReviewerName(e.target.value)} onChange={e => setEditChapter(e.target.value)}
placeholder="Your name" placeholder="Chapter"
className={inputClass}
/>
</div>
<div>
<label className="text-xs text-gray-400 mb-1 block">Page</label>
<input
type="text"
value={editPage}
onChange={e => setEditPage(e.target.value)}
placeholder="Page"
className={inputClass} className={inputClass}
/> />
</div> </div>
{reviewResult && (
<div className={`p-3 rounded-lg text-sm ${reviewResult.success ? 'bg-green-500/10 text-green-400' : 'bg-red-500/10 text-red-400'}`}>
{reviewResult.success ? 'Job marked as reviewed!' : reviewResult.error}
</div>
)}
<motion.button
onClick={handleMarkReviewed}
disabled={submitting || !reviewerName.trim()}
className={`w-full flex items-center justify-center gap-2 px-4 py-3 rounded-xl font-medium text-sm transition-all ${
submitting || !reviewerName.trim()
? 'opacity-50 cursor-not-allowed bg-white/5'
: 'bg-gradient-to-r from-green-600 to-emerald-600 hover:from-green-500 hover:to-emerald-500'
}`}
whileHover={!submitting && reviewerName.trim() ? { scale: 1.02 } : {}}
whileTap={!submitting && reviewerName.trim() ? { scale: 0.98 } : {}}
>
{submitting ? (
<><Loader2 className="w-4 h-4 animate-spin" /> Saving...</>
) : (
<><Save className="w-4 h-4" /> Mark Reviewed</>
)}
</motion.button>
</div> </div>
)} </div>
</>
)} {/* OCR / reviewed text */}
</div> <div className="space-y-1">
<label className="text-xs text-gray-400 block">
{isReviewed ? 'Reviewed Text' : 'OCR Text'}
<span className="text-purple-400 ml-1">(editable)</span>
</label>
<textarea
value={editedText}
onChange={e => setEditedText(e.target.value)}
rows={12}
className={`${inputClass} resize-y font-mono`}
placeholder="Text content..."
/>
</div>
{/* Reviewer + save */}
<div className="space-y-3 pt-3 border-t border-white/10">
<div>
<label className="text-xs text-gray-400 mb-1 block">Reviewer Name</label>
<input
type="text"
value={reviewerName}
onChange={e => setReviewerName(e.target.value)}
placeholder="Your name"
className={inputClass}
/>
</div>
{saveResult && (
<div className={`p-3 rounded-lg text-sm ${saveResult.success ? 'bg-green-500/10 text-green-400' : 'bg-red-500/10 text-red-400'}`}>
{saveResult.success
? (isReviewed ? 'Changes saved!' : 'Job marked as reviewed!')
: saveResult.error}
</div>
)}
<motion.button
onClick={handleSave}
disabled={submitting || !reviewerName.trim()}
className={`w-full flex items-center justify-center gap-2 px-4 py-3 rounded-xl font-medium text-sm transition-all ${
submitting || !reviewerName.trim()
? 'opacity-50 cursor-not-allowed bg-white/5'
: isReviewed
? 'bg-gradient-to-r from-blue-600 to-indigo-600 hover:from-blue-500 hover:to-indigo-500'
: 'bg-gradient-to-r from-green-600 to-emerald-600 hover:from-green-500 hover:to-emerald-500'
}`}
whileHover={!submitting && reviewerName.trim() ? { scale: 1.02 } : {}}
whileTap={!submitting && reviewerName.trim() ? { scale: 0.98 } : {}}
>
{submitting ? (
<><Loader2 className="w-4 h-4 animate-spin" /> Saving...</>
) : isReviewed ? (
<><Save className="w-4 h-4" /> Save Changes</>
) : (
<><CheckCircle2 className="w-4 h-4" /> Mark Reviewed</>
)}
</motion.button>
</div>
</div>
</div>
)}
</div> </div>
) )
} }
@@ -280,7 +324,6 @@ export default function JobsPanel() {
} }
}, [search, filterStatus, filterAuthor, filterBook]) }, [search, filterStatus, filterAuthor, filterBook])
// Initial load
useEffect(() => { useEffect(() => {
fetchJobs(0) fetchJobs(0)
}, []) // eslint-disable-line react-hooks/exhaustive-deps }, []) // eslint-disable-line react-hooks/exhaustive-deps
@@ -301,10 +344,11 @@ export default function JobsPanel() {
'placeholder-gray-600 focus:outline-none focus:border-purple-500/50 transition-colors' 'placeholder-gray-600 focus:outline-none focus:border-purple-500/50 transition-colors'
return ( return (
<div className="grid lg:grid-cols-2 gap-6 h-full"> // 1/3 list — 2/3 detail on large screens
{/* Left: Search + List */} <div className="grid grid-cols-1 lg:grid-cols-3 gap-6">
<div className="space-y-4">
{/* Search form */} {/* ── Left: Search + List ── */}
<div className="lg:col-span-1 space-y-4">
<div className="glass p-4 rounded-2xl space-y-3"> <div className="glass p-4 rounded-2xl space-y-3">
<form onSubmit={handleSearch} className="flex gap-2"> <form onSubmit={handleSearch} className="flex gap-2">
<input <input
@@ -325,12 +369,11 @@ export default function JobsPanel() {
</motion.button> </motion.button>
</form> </form>
{/* Filters */}
<div className="grid grid-cols-3 gap-2"> <div className="grid grid-cols-3 gap-2">
<select <select
value={filterStatus} value={filterStatus}
onChange={e => setFilterStatus(e.target.value)} onChange={e => setFilterStatus(e.target.value)}
className={`${inputClass} col-span-1`} className={inputClass}
> >
<option value="">All statuses</option> <option value="">All statuses</option>
<option value="unreviewed">Unreviewed</option> <option value="unreviewed">Unreviewed</option>
@@ -341,14 +384,14 @@ export default function JobsPanel() {
value={filterAuthor} value={filterAuthor}
onChange={e => setFilterAuthor(e.target.value)} onChange={e => setFilterAuthor(e.target.value)}
placeholder="Author..." placeholder="Author..."
className={`${inputClass} col-span-1`} className={inputClass}
/> />
<input <input
type="text" type="text"
value={filterBook} value={filterBook}
onChange={e => setFilterBook(e.target.value)} onChange={e => setFilterBook(e.target.value)}
placeholder="Book..." placeholder="Book..."
className={`${inputClass} col-span-1`} className={inputClass}
/> />
</div> </div>
@@ -366,28 +409,27 @@ export default function JobsPanel() {
</div> </div>
</div> </div>
{/* Results */} {loading && (
<div className="flex justify-center py-8">
<Loader2 className="w-6 h-6 animate-spin text-purple-400" />
</div>
)}
{error && (
<div className="glass p-4 rounded-xl border-red-500/30 bg-red-500/10">
<p className="text-sm text-red-400">{error}</p>
</div>
)}
{!loading && !error && jobs.length === 0 && (
<div className="glass p-8 rounded-2xl text-center">
<FileText className="w-10 h-10 mx-auto mb-3 text-gray-600" />
<p className="text-gray-400">No jobs found</p>
<p className="text-xs text-gray-500 mt-1">Commit your first OCR job from the New Job tab</p>
</div>
)}
<div className="space-y-2"> <div className="space-y-2">
{loading && (
<div className="flex justify-center py-8">
<Loader2 className="w-6 h-6 animate-spin text-purple-400" />
</div>
)}
{error && (
<div className="glass p-4 rounded-xl border-red-500/30 bg-red-500/10">
<p className="text-sm text-red-400">{error}</p>
</div>
)}
{!loading && !error && jobs.length === 0 && (
<div className="glass p-8 rounded-2xl text-center">
<FileText className="w-10 h-10 mx-auto mb-3 text-gray-600" />
<p className="text-gray-400">No jobs found</p>
<p className="text-xs text-gray-500 mt-1">Commit your first OCR job from the New Job tab</p>
</div>
)}
<AnimatePresence> <AnimatePresence>
{jobs.map(job => ( {jobs.map(job => (
<motion.button <motion.button
@@ -432,7 +474,6 @@ export default function JobsPanel() {
</AnimatePresence> </AnimatePresence>
</div> </div>
{/* Pagination */}
{totalPages > 1 && ( {totalPages > 1 && (
<div className="flex items-center justify-center gap-3"> <div className="flex items-center justify-center gap-3">
<button <button
@@ -456,8 +497,8 @@ export default function JobsPanel() {
)} )}
</div> </div>
{/* Right: Detail panel */} {/* ── Right: Detail panel ── */}
<div> <div className="lg:col-span-2">
<AnimatePresence mode="wait"> <AnimatePresence mode="wait">
{selectedJobId ? ( {selectedJobId ? (
<motion.div <motion.div
@@ -465,7 +506,6 @@ export default function JobsPanel() {
initial={{ opacity: 0, x: 20 }} initial={{ opacity: 0, x: 20 }}
animate={{ opacity: 1, x: 0 }} animate={{ opacity: 1, x: 0 }}
exit={{ opacity: 0, x: 20 }} exit={{ opacity: 0, x: 20 }}
className="h-full"
> >
<JobDetail <JobDetail
jobId={selectedJobId} jobId={selectedJobId}
@@ -479,14 +519,16 @@ export default function JobsPanel() {
initial={{ opacity: 0 }} initial={{ opacity: 0 }}
animate={{ opacity: 1 }} animate={{ opacity: 1 }}
exit={{ opacity: 0 }} exit={{ opacity: 0 }}
className="glass p-8 rounded-2xl flex flex-col items-center justify-center text-center h-full min-h-64" className="glass p-8 rounded-2xl flex flex-col items-center justify-center text-center"
style={{ minHeight: '300px' }}
> >
<Search className="w-10 h-10 mb-3 text-gray-600" /> <Search className="w-10 h-10 mb-3 text-gray-600" />
<p className="text-gray-400">Select a job to view details</p> <p className="text-gray-400">Select a job to view and edit details</p>
</motion.div> </motion.div>
)} )}
</AnimatePresence> </AnimatePresence>
</div> </div>
</div> </div>
) )
} }

View File

@@ -205,57 +205,76 @@ export default function ResultPanel({ result, loading, imagePreview, onCopy, onD
exit={{ opacity: 0, y: -20 }} exit={{ opacity: 0, y: -20 }}
className="space-y-4" className="space-y-4"
> >
{/* Preview with boxes */} {/* plain_ocr commit mode: image + editable textarea side by side */}
{imagePreview && result.boxes && result.boxes.length > 0 && (
<div className="relative rounded-xl overflow-hidden border border-white/10 bg-black">
<img
ref={imgRef}
src={imagePreview}
alt="Result"
className="w-full block"
onLoad={() => {
console.log('🖼️ Image loaded, triggering draw')
setImageLoaded(true)
}}
/>
<canvas
ref={canvasRef}
className="absolute top-0 left-0 w-full h-full pointer-events-none"
style={{ display: 'block' }}
/>
</div>
)}
{/* Text result — editable textarea in plain_ocr/commit mode, rendered otherwise */}
{onCommitJob ? ( {onCommitJob ? (
<div className="space-y-1"> <div className="grid grid-cols-2 gap-4 items-start">
<p className="text-xs text-gray-400">OCR Text <span className="text-purple-400">(editable correct before committing)</span></p> {imagePreview && typeof imagePreview === 'string' ? (
<textarea <div className="space-y-1">
value={editedOcrText} <p className="text-xs text-gray-400">Source Image</p>
onChange={e => onOcrTextChange(e.target.value)} <img
rows={10} src={imagePreview}
className="w-full bg-white/5 border border-white/10 rounded-xl px-4 py-3 text-sm text-gray-200 font-mono resize-y focus:outline-none focus:border-purple-500/50 transition-colors" alt="Source"
placeholder="OCR text will appear here..." className="w-full rounded-xl border border-white/10 bg-black/30"
/> />
</div>
) : (
<div className="bg-white/5 border border-white/10 rounded-xl p-4 max-h-96 overflow-y-auto">
{isHTML ? (
<div
className="prose prose-invert prose-sm max-w-none"
dangerouslySetInnerHTML={{ __html: DOMPurify.sanitize(result.text) }}
style={{ color: '#e5e7eb' }}
/>
) : isMarkdown ? (
<div className="prose prose-invert prose-sm max-w-none">
<ReactMarkdown>{result.text}</ReactMarkdown>
</div> </div>
) : ( ) : (
<pre className="text-sm text-gray-200 whitespace-pre-wrap font-mono"> <div />
{result.text}
</pre>
)} )}
<div className={`space-y-1 ${(!imagePreview || typeof imagePreview !== 'string') ? 'col-span-2' : ''}`}>
<p className="text-xs text-gray-400">
OCR Text <span className="text-purple-400">(edit before committing)</span>
</p>
<textarea
value={editedOcrText}
onChange={e => onOcrTextChange(e.target.value)}
className="w-full bg-white/5 border border-white/10 rounded-xl px-4 py-3 text-sm text-gray-200 font-mono resize-y focus:outline-none focus:border-purple-500/50 transition-colors"
style={{ minHeight: '240px' }}
placeholder="OCR text will appear here..."
/>
</div>
</div> </div>
) : (
<>
{/* Preview with boxes (grounding modes) */}
{imagePreview && result.boxes && result.boxes.length > 0 && (
<div className="relative rounded-xl overflow-hidden border border-white/10 bg-black">
<img
ref={imgRef}
src={imagePreview}
alt="Result"
className="w-full block"
onLoad={() => {
console.log('🖼️ Image loaded, triggering draw')
setImageLoaded(true)
}}
/>
<canvas
ref={canvasRef}
className="absolute top-0 left-0 w-full h-full pointer-events-none"
style={{ display: 'block' }}
/>
</div>
)}
{/* Rendered text result */}
<div className="bg-white/5 border border-white/10 rounded-xl p-4 max-h-96 overflow-y-auto">
{isHTML ? (
<div
className="prose prose-invert prose-sm max-w-none"
dangerouslySetInnerHTML={{ __html: DOMPurify.sanitize(result.text) }}
style={{ color: '#e5e7eb' }}
/>
) : isMarkdown ? (
<div className="prose prose-invert prose-sm max-w-none">
<ReactMarkdown>{result.text}</ReactMarkdown>
</div>
) : (
<pre className="text-sm text-gray-200 whitespace-pre-wrap font-mono">
{result.text}
</pre>
)}
</div>
</>
)} )}
{/* Raw Response Viewer */} {/* Raw Response Viewer */}