Merge t3code/expand-inspector-editor: add expandable full-screen editor modal to ResearchInspector

This commit is contained in:
2026-04-27 23:07:57 -04:00

View File

@@ -0,0 +1,367 @@
import React, { useCallback, useEffect, useRef, useState } from 'react';
import { createPortal } from 'react-dom';
import { Pin, Archive, RefreshCw, Sparkles, Maximize2, Minimize2 } from 'lucide-react';
import type {
GhostNote,
NoteAuditTrail,
NoteType,
ResearchNote,
ThesisStatus,
} from '../../types/research';
import { ConfidenceBadge, EvidenceBadge, FreshnessBadge, NoteTypeBadge, TagChip } from './primitives/Badges';
import { GHOST_CLASS_LABELS, NOTE_TYPE_LABELS } from './primitives/researchMeta';
interface ResearchInspectorProps {
note: ResearchNote | null;
ghost: GhostNote | null;
auditTrail: NoteAuditTrail | null;
isLoadingAuditTrail: boolean;
onRefreshAuditTrail: () => void;
onUpdateNote: (noteId: string, patch: {
rawText?: string;
title?: string;
noteType?: NoteType;
pinned?: boolean;
thesisStatus?: ThesisStatus;
}) => Promise<unknown>;
onArchiveNote: (noteId: string, archived: boolean) => Promise<unknown>;
onPromoteNote: (noteId: string, thesisStatus?: ThesisStatus, noteType?: NoteType) => Promise<unknown>;
onReviewGhost: (ghostId: string, action: 'accept' | 'ignore' | 'dismiss' | 'pin') => Promise<unknown>;
}
const editableNoteTypes: NoteType[] = [
'fact',
'management_signal',
'claim',
'thesis',
'sub_thesis',
'risk',
'catalyst',
'valuation_point',
'question',
'contradiction',
'channel_check',
'mosaic_insight',
];
export const ResearchInspector: React.FC<ResearchInspectorProps> = ({
note,
ghost,
auditTrail,
isLoadingAuditTrail,
onRefreshAuditTrail,
onUpdateNote,
onArchiveNote,
onPromoteNote,
onReviewGhost,
}) => {
const [draftTitle, setDraftTitle] = useState('');
const [draftBody, setDraftBody] = useState('');
const [draftType, setDraftType] = useState<NoteType>('claim');
const [isEditorExpanded, setIsEditorExpanded] = useState(false);
const expandedEditorRef = useRef<HTMLTextAreaElement>(null);
useEffect(() => {
if (!note) {
return;
}
setDraftTitle(note.title ?? '');
setDraftBody(note.rawText);
setDraftType(note.noteType);
}, [note]);
useEffect(() => {
if (isEditorExpanded && expandedEditorRef.current) {
expandedEditorRef.current.focus();
expandedEditorRef.current.setSelectionRange(
expandedEditorRef.current.value.length,
expandedEditorRef.current.value.length,
);
}
}, [isEditorExpanded]);
const handleExpandKeyDown = useCallback(
(e: React.KeyboardEvent) => {
if (e.key === 'Escape') {
setIsEditorExpanded(false);
} else if (e.key === 'Enter' && (e.metaKey || e.ctrlKey)) {
e.preventDefault();
setIsEditorExpanded(false);
}
},
[],
);
if (!note && !ghost) {
return (
<div className="flex h-full flex-col justify-between p-4 sm:p-5">
<div>
<h2 className="text-base font-semibold text-[var(--research-text-strong)] sm:text-lg">
Inspector
</h2>
<p className="mt-2 text-xs leading-6 text-[var(--research-text-muted)] sm:mt-2.5 sm:text-sm">
Select a note or ghost note to inspect provenance, links, evidence, and promotion controls.
</p>
</div>
</div>
);
}
if (ghost) {
return (
<div className="space-y-4 p-4 sm:space-y-5 sm:p-5">
<div>
<div className="text-[10px] uppercase tracking-[0.2em] text-term-text-tertiary">Ghost Note</div>
<h2 className="mt-2 text-lg font-semibold text-term-text sm:text-xl">
{ghost.headline}
</h2>
<p className="mt-2.5 text-sm leading-6 text-term-text-muted sm:mt-3 sm:leading-7">
{ghost.body}
</p>
</div>
<div className="flex flex-wrap gap-2">
<TagChip label={GHOST_CLASS_LABELS[ghost.ghostClass]} />
<ConfidenceBadge confidence={ghost.confidence} />
<FreshnessBadge timestamp={ghost.updatedAt} />
</div>
<div className="grid gap-2.5 sm:gap-3">
<button
type="button"
onClick={() => {
void onReviewGhost(ghost.id, 'accept');
}}
className="w-full rounded-2xl border border-info/40 bg-info/10 px-4 py-2.5 text-sm text-info transition-colors hover:border-info hover:text-term-text sm:py-3"
>
Accept ghost
</button>
<button
type="button"
onClick={() => {
void onReviewGhost(ghost.id, 'pin');
}}
className="w-full rounded-2xl border border-term-border bg-term-surface px-4 py-2.5 text-sm text-term-text transition-colors hover:border-info/50 sm:py-3"
>
Pin ghost
</button>
<button
type="button"
onClick={() => {
void onReviewGhost(ghost.id, 'dismiss');
}}
className="w-full rounded-2xl border border-negative/40 bg-negative/10 px-4 py-2.5 text-sm text-negative transition-colors hover:border-negative/70 sm:py-3"
>
Dismiss ghost
</button>
</div>
</div>
);
}
return (
<div className="space-y-4 p-4 sm:space-y-5 sm:p-5">
<div className="flex flex-wrap items-center gap-2">
<NoteTypeBadge noteType={note!.noteType} />
<EvidenceBadge status={note!.evidenceStatus} />
<ConfidenceBadge confidence={note!.confidence} />
<FreshnessBadge timestamp={note!.updatedAt} />
</div>
<div className="space-y-3">
<label className="flex flex-col gap-2">
<span className="text-[10px] uppercase tracking-[0.18em] text-term-text-tertiary">Title</span>
<input
value={draftTitle}
onChange={(event) => setDraftTitle(event.target.value)}
className="w-full rounded-2xl border border-term-border bg-term-bg px-3 py-2.5 text-sm text-term-text outline-none transition-colors focus:border-info sm:py-3"
/>
</label>
<label className="flex flex-col gap-2">
<div className="flex items-center justify-between">
<span className="text-[10px] uppercase tracking-[0.18em] text-term-text-tertiary">Body</span>
<button
type="button"
onClick={() => setIsEditorExpanded(true)}
className="inline-flex items-center gap-1 rounded-lg border border-term-border bg-term-surface px-1.5 py-1 text-term-text-tertiary transition-colors hover:border-info/60 hover:text-term-text"
aria-label="Expand editor"
>
<Maximize2 className="h-3 w-3" />
</button>
</div>
<textarea
value={draftBody}
onChange={(event) => setDraftBody(event.target.value)}
className="min-h-[160px] w-full rounded-2xl border border-term-border bg-term-bg px-3 py-2.5 text-sm leading-6 text-term-text outline-none transition-colors focus:border-info sm:min-h-40 sm:py-3"
/>
</label>
<label className="flex flex-col gap-2">
<span className="text-[10px] uppercase tracking-[0.18em] text-term-text-tertiary">Note Type</span>
<select
value={draftType}
onChange={(event) => setDraftType(event.target.value as NoteType)}
className="w-full rounded-2xl border border-term-border bg-term-bg px-3 py-2.5 text-sm text-term-text outline-none transition-colors focus:border-info sm:py-3"
>
{editableNoteTypes.map((noteType) => (
<option key={noteType} value={noteType}>
{NOTE_TYPE_LABELS[noteType]}
</option>
))}
</select>
</label>
<div className="grid grid-cols-2 gap-2.5 sm:gap-3">
<button
type="button"
onClick={() => {
void onUpdateNote(note!.id, {
title: draftTitle,
rawText: draftBody,
noteType: draftType,
});
}}
className="rounded-2xl border border-info/40 bg-info/10 px-3 py-2.5 text-sm text-info transition-colors hover:border-info hover:text-term-text sm:px-4 sm:py-3"
>
Save note
</button>
<button
type="button"
onClick={() => {
void onPromoteNote(note!.id, 'accepted_core', note!.noteType === 'thesis' ? 'thesis' : 'sub_thesis');
}}
className="inline-flex items-center justify-center gap-1.5 rounded-2xl border border-positive/35 bg-positive/10 px-3 py-2.5 text-sm text-positive transition-colors hover:border-positive/60 sm:gap-2 sm:px-4 sm:py-3"
>
<Sparkles className="h-3.5 w-3.5 sm:h-4 sm:w-4" />
<span className="hidden xs:inline">Promote</span>
<span className="xs:hidden">Promote</span>
</button>
</div>
<div className="grid grid-cols-2 gap-2.5 sm:gap-3">
<button
type="button"
onClick={() => {
void onUpdateNote(note!.id, { pinned: !note!.pinned });
}}
className="inline-flex items-center justify-center gap-1.5 rounded-2xl border border-term-border bg-term-surface px-3 py-2.5 text-sm text-term-text transition-colors hover:border-info/50 sm:gap-2 sm:px-4 sm:py-3"
>
<Pin className="h-3.5 w-3.5 sm:h-4 sm:w-4" />
<span className="hidden xs:inline">{note!.pinned ? 'Unpin' : 'Pin'}</span>
<span className="xs:hidden">{note!.pinned ? 'Unpin' : 'Pin'}</span>
</button>
<button
type="button"
onClick={() => {
void onArchiveNote(note!.id, !note!.archived);
}}
className="inline-flex items-center justify-center gap-1.5 rounded-2xl border border-negative/40 bg-negative/10 px-3 py-2.5 text-sm text-negative transition-colors hover:border-negative/70 sm:gap-2 sm:px-4 sm:py-3"
>
<Archive className="h-3.5 w-3.5 sm:h-4 sm:w-4" />
<span className="hidden xs:inline">{note!.archived ? 'Restore' : 'Archive'}</span>
<span className="xs:hidden">{note!.archived ? 'Restore' : 'Archive'}</span>
</button>
</div>
</div>
<div className="space-y-3 rounded-2xl border border-term-border bg-term-surface p-3 sm:p-4">
<div className="flex items-center justify-between gap-3">
<div className="min-w-0 flex-1">
<div className="text-[10px] uppercase tracking-[0.18em] text-term-text-tertiary">Audit Trail</div>
<div className="mt-1 truncate text-sm text-term-text">
{isLoadingAuditTrail ? 'Loading evidence trace...' : `${auditTrail?.links.length ?? 0} links, ${auditTrail?.sources.length ?? 0} sources`}
</div>
</div>
<button
type="button"
onClick={onRefreshAuditTrail}
className="inline-flex shrink-0 items-center gap-1.5 rounded-xl border border-term-border bg-term-bg px-2.5 py-2 text-xs uppercase tracking-[0.18em] text-term-text-tertiary transition-colors hover:border-info/60 hover:text-term-text sm:gap-2 sm:px-3"
>
<RefreshCw className="h-3.5 w-3.5" />
<span className="hidden sm:inline">Refresh</span>
</button>
</div>
{auditTrail ? (
<div className="space-y-3 text-sm text-term-text-muted">
<div>
<div className="text-[10px] uppercase tracking-[0.18em] text-term-text-tertiary">Sources</div>
<ul className="mt-2 space-y-2">
{auditTrail.sources.slice(0, 4).map((source) => (
<li key={source.id} className="truncate text-xs sm:text-sm">
{source.title}
</li>
))}
</ul>
</div>
<div>
<div className="text-[10px] uppercase tracking-[0.18em] text-term-text-tertiary">Audit Events</div>
<ul className="mt-2 space-y-2">
{auditTrail.auditEvents.slice(0, 5).map((event) => (
<li key={event.id} className="text-xs sm:text-sm">
<span className="text-term-text">{event.action}</span>
<span className="text-term-text-tertiary"> {event.createdAt}</span>
</li>
))}
</ul>
</div>
</div>
) : (
<p className="text-sm text-term-text-tertiary">Select a note to inspect its evidence chain.</p>
)}
</div>
{isEditorExpanded && note && createPortal(
<div
className="fixed inset-0 z-50 flex items-center justify-center bg-black/60 p-4"
onClick={() => setIsEditorExpanded(false)}
role="dialog"
aria-modal="true"
aria-label="Expanded note editor"
>
<div
className="flex max-h-[90vh] w-full max-w-4xl flex-col overflow-hidden rounded-[24px] border border-term-border bg-term-elevated shadow-2xl"
onClick={(e) => e.stopPropagation()}
onKeyDown={handleExpandKeyDown}
>
<div className="flex items-center justify-between border-b border-term-border bg-term-surface px-6 py-4">
<div>
<h2 className="text-lg font-mono font-semibold text-term-text">Edit Note</h2>
<p className="mt-1 text-xs text-term-text-muted">{draftTitle || 'Untitled'}</p>
</div>
<button
type="button"
onClick={() => setIsEditorExpanded(false)}
className="inline-flex items-center gap-1.5 rounded-xl border border-term-border bg-term-bg px-2.5 py-2 text-term-text-tertiary transition-colors hover:border-info/60 hover:text-term-text"
aria-label="Collapse editor"
>
<Minimize2 className="h-4 w-4" />
</button>
</div>
<div className="flex-1 overflow-y-auto p-6">
<textarea
ref={expandedEditorRef}
value={draftBody}
onChange={(e) => setDraftBody(e.target.value)}
className="min-h-[50vh] w-full resize-y rounded-2xl border border-term-border bg-term-bg px-4 py-3 text-sm leading-7 text-term-text outline-none transition-colors focus:border-info"
placeholder="Write your note..."
/>
</div>
<div className="flex items-center justify-between border-t border-term-border bg-term-surface px-6 py-4">
<p className="text-[10px] font-mono text-term-text-tertiary">
Press <kbd className="rounded border border-term-border bg-term-bg px-1 py-0.5">Esc</kbd> or{' '}
<kbd className="rounded border border-term-border bg-term-bg px-1 py-0.5"> Enter</kbd> to collapse
</p>
<button
type="button"
onClick={() => setIsEditorExpanded(false)}
className="rounded border border-info bg-info px-4 py-2 text-xs font-mono font-semibold text-term-bg transition-colors hover:brightness-110"
>
Done
</button>
</div>
</div>
</div>,
document.body,
)}
</div>
);
};