nexus/ui/src/components/ArtifactsPanel.tsx
scotttong 4b3cda97e4 experiment: filter system output from chat, approval UX, wizard sub-steps
- Strip JSON/system init output from agent messages in chat (no more
  raw tool dumps or session IDs visible to users)
- Filter streaming relay chunks that look like system output
- Create "in progress" artifact when user asks for a hiring plan
- Swap approval button order: Reject (left), Approve (right, green)
- Fix chat/artifact pane height to fill viewport without clipping
- Progressive disclosure in wizard step 1: name first, then mission
- Hide agent comments that are entirely system output

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-19 17:20:32 -07:00

339 lines
13 KiB
TypeScript

import { useState } from "react";
import { useQuery } from "@tanstack/react-query";
import type { IssueWorkProduct } from "@paperclipai/shared";
import { issuesApi } from "../api/issues";
import { queryKeys } from "../lib/queryKeys";
import { MarkdownBody } from "./MarkdownBody";
import { cn } from "../lib/utils";
import {
FileText,
ExternalLink,
GitBranch,
GitCommit,
Globe,
Server,
Package,
Loader2,
ArrowLeft,
X,
CheckCircle2,
XCircle,
RotateCcw,
} from "lucide-react";
import { Button } from "@/components/ui/button";
interface ArtifactsPanelProps {
taskId: string;
isAgentWorking?: boolean;
/** Open the document viewer directly to a specific doc */
openDocKey?: string | null;
openDocTitle?: string | null;
onClearOpenDoc?: () => void;
/** Approval callbacks — called from the document viewer */
onApprove?: () => void;
onReject?: () => void;
}
type FilterValue = "all" | "in_progress" | "for_review" | "completed";
const FILTERS: Array<{ label: string; value: FilterValue }> = [
{ label: "All", value: "all" },
{ label: "In Progress", value: "in_progress" },
{ label: "For Review", value: "for_review" },
{ label: "Completed", value: "completed" },
];
function matchesFilter(wp: IssueWorkProduct, filter: FilterValue): boolean {
if (filter === "all") return true;
if (filter === "in_progress") return wp.status === "active" || wp.status === "draft";
if (filter === "for_review") return wp.status === "ready_for_review";
if (filter === "completed") return wp.status === "approved" || wp.status === "merged";
return true;
}
function typeIcon(type: string) {
switch (type) {
case "document": return FileText;
case "pull_request": return GitBranch;
case "branch": return GitBranch;
case "commit": return GitCommit;
case "preview_url": return Globe;
case "runtime_service": return Server;
case "artifact": return Package;
default: return FileText;
}
}
function statusBadge(status: string) {
switch (status) {
case "active":
case "draft":
return { label: "In Progress", className: "bg-blue-500/10 text-blue-600 dark:text-blue-400" };
case "ready_for_review":
return { label: "For Review", className: "bg-amber-500/10 text-amber-600 dark:text-amber-400" };
case "approved":
case "merged":
return { label: "Completed", className: "bg-green-500/10 text-green-600 dark:text-green-400" };
case "changes_requested":
return { label: "Changes Requested", className: "bg-orange-500/10 text-orange-600 dark:text-orange-400" };
case "failed":
return { label: "Failed", className: "bg-red-500/10 text-red-600 dark:text-red-400" };
default:
return { label: status, className: "bg-muted text-muted-foreground" };
}
}
export function ArtifactsPanel({ taskId, isAgentWorking, openDocKey, openDocTitle, onClearOpenDoc, onApprove, onReject }: ArtifactsPanelProps) {
const [filter, setFilter] = useState<FilterValue>("all");
const [viewingDoc, setViewingDoc] = useState<{ key: string; title: string } | null>(null);
const { data: workProducts, isLoading } = useQuery({
queryKey: queryKeys.issues.workProducts(taskId),
queryFn: () => issuesApi.listWorkProducts(taskId),
refetchInterval: 5000,
});
// Open doc from parent (e.g. clicking plan link in chat)
const effectiveViewingDoc = openDocKey
? { key: openDocKey, title: openDocTitle ?? "Document" }
: viewingDoc;
const handleBack = () => {
setViewingDoc(null);
onClearOpenDoc?.();
};
// Find the work product for the currently viewed doc to know its status
const viewedWorkProduct = effectiveViewingDoc
? (workProducts ?? []).find((wp) => wp.title === effectiveViewingDoc.title)
: null;
const filtered = (workProducts ?? []).filter((wp) => matchesFilter(wp, filter));
// Document viewer
if (effectiveViewingDoc) {
return (
<DocumentViewer
taskId={taskId}
docKey={effectiveViewingDoc.key}
title={effectiveViewingDoc.title}
onBack={handleBack}
status={viewedWorkProduct?.status ?? null}
reviewState={viewedWorkProduct?.reviewState ?? null}
onApprove={onApprove}
onReject={onReject}
/>
);
}
return (
<div className="flex flex-col h-full" data-artifacts-panel>
<div className="px-4 py-3 border-b border-border flex items-center gap-2">
<Package className="h-4 w-4 text-muted-foreground shrink-0" />
<h3 className="text-sm font-semibold">Artifacts</h3>
</div>
{/* Filter chips */}
<div className="px-4 py-2 flex flex-wrap gap-1 border-b border-border">
{FILTERS.map((f) => (
<button
key={f.value}
className={cn(
"rounded-full px-2.5 py-0.5 text-[11px] font-medium transition-colors",
filter === f.value
? "bg-foreground text-background"
: "text-muted-foreground hover:text-foreground hover:bg-accent/50",
)}
onClick={() => setFilter(f.value)}
>
{f.label}
</button>
))}
</div>
{/* Work products list */}
<div className="flex-1 overflow-y-auto scrollbar-auto-hide">
{isLoading ? (
<div className="flex items-center justify-center py-8 text-muted-foreground text-sm">
<Loader2 className="h-4 w-4 mr-2 animate-spin" />
Loading...
</div>
) : filtered.length === 0 ? (
<div className="px-4 py-8 text-center">
<Package className="h-8 w-8 mx-auto text-muted-foreground/40 mb-3" />
<p className="text-sm text-muted-foreground">
{workProducts?.length === 0
? "Your team's deliverables and plans will appear here as they're produced."
: "No artifacts match this filter."}
</p>
</div>
) : (
<div className="divide-y divide-border">
{filtered.map((wp) => {
const Icon = typeIcon(wp.type);
const badge = statusBadge(wp.status);
const isDraft = wp.status === "draft" || wp.status === "active";
const showGenerating = isDraft && isAgentWorking;
return (
<button
key={wp.id}
className={cn(
"w-full text-left px-4 py-3 hover:bg-accent/30 transition-colors",
showGenerating && "bg-muted/30",
)}
onClick={() => {
if (wp.type === "document") {
setViewingDoc({ key: "plan", title: wp.title });
} else if (wp.url) {
window.open(wp.url, "_blank", "noopener,noreferrer");
}
}}
>
<div className="flex items-start gap-2.5">
{showGenerating ? (
<div className="mt-0.5 shrink-0">
<span className="relative flex h-4 w-4">
<span className="animate-ping absolute inline-flex h-full w-full rounded-full bg-cyan-400 opacity-75" />
<span className="relative inline-flex rounded-full h-4 w-4 bg-cyan-500" />
</span>
</div>
) : (
<Icon className="h-4 w-4 mt-0.5 text-muted-foreground shrink-0" />
)}
<div className="min-w-0 flex-1">
<div className="flex items-center gap-2">
<span className="text-sm font-medium truncate">{wp.title}</span>
{wp.url && (
<ExternalLink className="h-3 w-3 text-muted-foreground shrink-0" />
)}
</div>
<div className="flex items-center gap-2 mt-1">
<span className="text-[10px] text-muted-foreground capitalize">
{wp.type.replace(/_/g, " ")}
</span>
{showGenerating ? (
<span className="inline-flex items-center gap-1 text-[10px] font-medium px-1.5 py-0.5 rounded-full bg-cyan-500/10 text-cyan-600 dark:text-cyan-400">
<Loader2 className="h-2.5 w-2.5 animate-spin" />
Generating...
</span>
) : (
<span className={cn("text-[10px] font-medium px-1.5 py-0.5 rounded-full", badge.className)}>
{badge.label}
</span>
)}
</div>
{wp.summary && (
<p className="text-[11px] text-muted-foreground mt-1 line-clamp-2">
{wp.summary}
</p>
)}
</div>
</div>
</button>
);
})}
</div>
)}
</div>
</div>
);
}
function DocumentViewer({
taskId,
docKey,
title,
onBack,
status,
reviewState,
onApprove,
onReject,
}: {
taskId: string;
docKey: string;
title: string;
onBack: () => void;
status: string | null;
reviewState: string | null;
onApprove?: () => void;
onReject?: () => void;
}) {
const { data: doc, isLoading, error } = useQuery({
queryKey: queryKeys.issues.documents(taskId),
queryFn: () => issuesApi.getDocument(taskId, docKey),
});
const needsAction = status === "ready_for_review" || reviewState === "needs_board_review";
const isApproved = status === "approved" || reviewState === "approved";
const isRejected = status === "changes_requested" || reviewState === "changes_requested";
return (
<div className="flex flex-col h-full">
<div className="px-4 py-3 border-b border-border flex items-center gap-2">
<button onClick={onBack} className="text-muted-foreground hover:text-foreground">
<ArrowLeft className="h-4 w-4" />
</button>
<h3 className="text-sm font-semibold flex-1 truncate">{title}</h3>
<button onClick={onBack} className="text-muted-foreground hover:text-foreground">
<X className="h-4 w-4" />
</button>
</div>
<div className="flex-1 overflow-y-auto scrollbar-auto-hide p-4">
{isLoading ? (
<div className="flex items-center justify-center py-8 text-muted-foreground text-sm">
<Loader2 className="h-4 w-4 mr-2 animate-spin" />
Loading document...
</div>
) : error ? (
<p className="text-sm text-muted-foreground">Document not available yet.</p>
) : doc?.body ? (
<div className="prose prose-sm dark:prose-invert max-w-none">
<MarkdownBody>{doc.body}</MarkdownBody>
</div>
) : (
<p className="text-sm text-muted-foreground">Document is empty.</p>
)}
</div>
{/* Sticky action footer */}
{needsAction && (
<div className="border-t border-border px-4 py-3 bg-background shrink-0">
<p className="text-[11px] text-muted-foreground mb-2">This document needs your review.</p>
<div className="flex items-center gap-2">
<Button size="sm" variant="outline" className="h-8 text-xs flex-1 text-destructive hover:text-destructive" onClick={() => {
onReject?.();
onBack();
}}>
<XCircle className="h-3.5 w-3.5 mr-1" />
Reject
</Button>
<Button size="sm" className="h-8 text-xs flex-1 bg-green-600 hover:bg-green-700 text-white" onClick={onApprove}>
<CheckCircle2 className="h-3.5 w-3.5 mr-1" />
Approve
</Button>
</div>
</div>
)}
{isApproved && (
<div className="border-t border-green-500/30 bg-green-500/5 px-4 py-3 shrink-0">
<div className="flex items-center gap-2">
<CheckCircle2 className="h-4 w-4 text-green-500" />
<p className="text-[13px] font-medium text-green-700 dark:text-green-400">
Approved hire tasks created
</p>
</div>
</div>
)}
{isRejected && (
<div className="border-t border-orange-500/30 bg-orange-500/5 px-4 py-3 shrink-0">
<div className="flex items-center gap-2">
<XCircle className="h-4 w-4 text-orange-500" />
<p className="text-[13px] font-medium text-orange-700 dark:text-orange-400">
Changes requested CEO is revising
</p>
</div>
</div>
)}
</div>
);
}