Files
vibe-kanban/frontend/src/components/tasks/TaskDetailsPanel.tsx

678 lines
24 KiB
TypeScript
Raw Normal View History

2025-06-21 16:49:16 +01:00
import { useState, useEffect } from "react";
2025-06-21 17:36:07 +01:00
import {
X,
History,
Send,
Clock,
FileText,
Code,
} from "lucide-react";
2025-06-21 16:49:16 +01:00
import { Button } from "@/components/ui/button";
import { Card, CardContent } from "@/components/ui/card";
import { Label } from "@/components/ui/label";
import { Textarea } from "@/components/ui/textarea";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import { makeRequest } from "@/lib/api";
2025-06-21 18:00:28 +01:00
import { getTaskPanelClasses, getBackdropClasses } from "@/lib/responsive-config";
2025-06-21 16:49:16 +01:00
import type {
TaskStatus,
TaskAttempt,
TaskAttemptActivity,
TaskAttemptStatus,
ExecutionProcess,
ExecutionProcessStatus,
ExecutionProcessType,
ApiResponse,
TaskWithAttemptStatus,
} from "shared/types";
interface TaskDetailsPanelProps {
task: TaskWithAttemptStatus | null;
projectId: string;
isOpen: boolean;
onClose: () => void;
}
const statusLabels: Record<TaskStatus, string> = {
todo: "To Do",
2025-06-21 17:36:07 +01:00
inprogress: "In Progress",
2025-06-21 16:49:16 +01:00
inreview: "In Review",
done: "Done",
cancelled: "Cancelled",
};
2025-06-21 17:36:07 +01:00
const getAttemptStatusDisplay = (
status: TaskAttemptStatus
): { label: string; className: string } => {
2025-06-21 16:49:16 +01:00
switch (status) {
case "init":
2025-06-21 17:36:07 +01:00
return {
label: "Init",
className: "bg-status-init text-status-init-foreground",
};
2025-06-21 16:49:16 +01:00
case "setuprunning":
2025-06-21 17:36:07 +01:00
return {
label: "Setup Running",
className: "bg-status-running text-status-running-foreground",
};
2025-06-21 16:49:16 +01:00
case "setupcomplete":
2025-06-21 17:36:07 +01:00
return {
label: "Setup Complete",
className: "bg-status-complete text-status-complete-foreground",
};
2025-06-21 16:49:16 +01:00
case "setupfailed":
2025-06-21 17:36:07 +01:00
return {
label: "Setup Failed",
className: "bg-status-failed text-status-failed-foreground",
};
2025-06-21 16:49:16 +01:00
case "executorrunning":
2025-06-21 17:36:07 +01:00
return {
label: "Executor Running",
className: "bg-status-running text-status-running-foreground",
};
2025-06-21 16:49:16 +01:00
case "executorcomplete":
2025-06-21 17:36:07 +01:00
return {
label: "Executor Complete",
className: "bg-status-complete text-status-complete-foreground",
};
2025-06-21 16:49:16 +01:00
case "executorfailed":
2025-06-21 17:36:07 +01:00
return {
label: "Executor Failed",
className: "bg-status-failed text-status-failed-foreground",
};
2025-06-21 16:49:16 +01:00
case "paused":
2025-06-21 17:36:07 +01:00
return {
label: "Paused",
className: "bg-status-paused text-status-paused-foreground",
};
2025-06-21 16:49:16 +01:00
default:
2025-06-21 17:36:07 +01:00
return {
label: "Unknown",
className: "bg-status-init text-status-init-foreground",
};
2025-06-21 16:49:16 +01:00
}
};
2025-06-21 17:36:07 +01:00
const getProcessStatusDisplay = (
status: ExecutionProcessStatus
): { label: string; className: string } => {
2025-06-21 16:49:16 +01:00
switch (status) {
case "running":
2025-06-21 17:36:07 +01:00
return {
label: "Running",
className: "bg-status-running text-status-running-foreground",
};
2025-06-21 16:49:16 +01:00
case "completed":
2025-06-21 17:36:07 +01:00
return {
label: "Completed",
className: "bg-status-complete text-status-complete-foreground",
};
2025-06-21 16:49:16 +01:00
case "failed":
2025-06-21 17:36:07 +01:00
return {
label: "Failed",
className: "bg-status-failed text-status-failed-foreground",
};
2025-06-21 16:49:16 +01:00
case "killed":
2025-06-21 17:36:07 +01:00
return {
label: "Killed",
className: "bg-status-failed text-status-failed-foreground",
};
2025-06-21 16:49:16 +01:00
default:
2025-06-21 17:36:07 +01:00
return {
label: "Unknown",
className: "bg-status-init text-status-init-foreground",
};
2025-06-21 16:49:16 +01:00
}
};
const getProcessTypeDisplay = (type: ExecutionProcessType): string => {
switch (type) {
case "setupscript":
return "Setup Script";
case "codingagent":
return "Coding Agent";
case "devserver":
return "Dev Server";
default:
return "Unknown";
}
};
2025-06-21 17:36:07 +01:00
export function TaskDetailsPanel({
task,
projectId,
isOpen,
onClose,
}: TaskDetailsPanelProps) {
2025-06-21 16:49:16 +01:00
const [taskAttempts, setTaskAttempts] = useState<TaskAttempt[]>([]);
2025-06-21 17:36:07 +01:00
const [selectedAttempt, setSelectedAttempt] = useState<TaskAttempt | null>(
null
);
const [attemptActivities, setAttemptActivities] = useState<
TaskAttemptActivity[]
>([]);
const [executionProcesses, setExecutionProcesses] = useState<
ExecutionProcess[]
>([]);
2025-06-21 16:49:16 +01:00
const [loading, setLoading] = useState(false);
const [followUpMessage, setFollowUpMessage] = useState("");
const [showAttemptHistory, setShowAttemptHistory] = useState(false);
// Check if the selected attempt is active (not in a final state)
const isAttemptRunning =
selectedAttempt &&
attemptActivities.length > 0 &&
(attemptActivities[0].status === "init" ||
attemptActivities[0].status === "setuprunning" ||
attemptActivities[0].status === "setupcomplete" ||
attemptActivities[0].status === "executorrunning");
// Polling for updates when attempt is running
useEffect(() => {
if (!isAttemptRunning || !task) return;
const interval = setInterval(() => {
if (selectedAttempt) {
fetchAttemptActivities(selectedAttempt.id, true);
fetchExecutionProcesses(selectedAttempt.id, true);
}
}, 2000);
return () => clearInterval(interval);
}, [isAttemptRunning, task?.id, selectedAttempt?.id]);
useEffect(() => {
if (task && isOpen) {
fetchTaskAttempts();
}
}, [task, isOpen]);
const fetchTaskAttempts = async () => {
if (!task) return;
try {
setLoading(true);
const response = await makeRequest(
`/api/projects/${projectId}/tasks/${task.id}/attempts`
);
if (response.ok) {
const result: ApiResponse<TaskAttempt[]> = await response.json();
if (result.success && result.data) {
setTaskAttempts(result.data);
2025-06-21 17:36:07 +01:00
2025-06-21 16:49:16 +01:00
// Auto-select latest attempt
if (result.data.length > 0) {
const latestAttempt = result.data.reduce((latest, current) =>
new Date(current.created_at) > new Date(latest.created_at)
? current
: latest
);
setSelectedAttempt(latestAttempt);
fetchAttemptActivities(latestAttempt.id);
fetchExecutionProcesses(latestAttempt.id);
}
}
}
} catch (err) {
console.error("Failed to fetch task attempts:", err);
} finally {
setLoading(false);
}
};
2025-06-21 17:36:07 +01:00
const fetchAttemptActivities = async (
attemptId: string,
_isBackgroundUpdate = false
) => {
2025-06-21 16:49:16 +01:00
if (!task) return;
try {
const response = await makeRequest(
`/api/projects/${projectId}/tasks/${task.id}/attempts/${attemptId}/activities`
);
if (response.ok) {
2025-06-21 17:36:07 +01:00
const result: ApiResponse<TaskAttemptActivity[]> =
await response.json();
2025-06-21 16:49:16 +01:00
if (result.success && result.data) {
setAttemptActivities(result.data);
}
}
} catch (err) {
console.error("Failed to fetch attempt activities:", err);
}
};
2025-06-21 17:36:07 +01:00
const fetchExecutionProcesses = async (
attemptId: string,
_isBackgroundUpdate = false
) => {
2025-06-21 16:49:16 +01:00
if (!task) return;
try {
const response = await makeRequest(
`/api/projects/${projectId}/tasks/${task.id}/attempts/${attemptId}/execution-processes`
);
if (response.ok) {
const result: ApiResponse<ExecutionProcess[]> = await response.json();
if (result.success && result.data) {
setExecutionProcesses(result.data);
}
}
} catch (err) {
console.error("Failed to fetch execution processes:", err);
}
};
const handleAttemptChange = (attemptId: string) => {
2025-06-21 17:36:07 +01:00
const attempt = taskAttempts.find((a) => a.id === attemptId);
2025-06-21 16:49:16 +01:00
if (attempt) {
setSelectedAttempt(attempt);
fetchAttemptActivities(attempt.id);
fetchExecutionProcesses(attempt.id);
setShowAttemptHistory(false);
}
};
const handleSendFollowUp = () => {
// TODO: Implement follow-up message API
console.log("Follow-up message:", followUpMessage);
setFollowUpMessage("");
};
const stopExecutionProcess = async (processId: string) => {
if (!task || !selectedAttempt) return;
try {
const response = await makeRequest(
`/api/projects/${projectId}/tasks/${task.id}/attempts/${selectedAttempt.id}/execution-processes/${processId}/stop`,
{
method: "POST",
headers: {
"Content-Type": "application/json",
},
}
);
if (response.ok) {
// Refresh the execution processes
fetchExecutionProcesses(selectedAttempt.id);
fetchAttemptActivities(selectedAttempt.id);
}
} catch (err) {
console.error("Failed to stop execution process:", err);
}
};
const openInEditor = async () => {
if (!task || !selectedAttempt) return;
try {
await makeRequest(
`/api/projects/${projectId}/tasks/${task.id}/attempts/${selectedAttempt.id}/open-editor`,
{
method: "POST",
headers: {
"Content-Type": "application/json",
},
}
);
} catch (err) {
console.error("Failed to open editor:", err);
}
};
if (!task) return null;
return (
<>
{isOpen && (
<>
2025-06-21 18:00:28 +01:00
{/* Backdrop - only on smaller screens (overlay mode) */}
<div
className={getBackdropClasses()}
onClick={onClose}
/>
2025-06-21 17:36:07 +01:00
2025-06-21 16:49:16 +01:00
{/* Panel */}
2025-06-21 17:36:07 +01:00
<div
2025-06-21 18:00:28 +01:00
className={getTaskPanelClasses()}
2025-06-21 17:36:07 +01:00
>
2025-06-21 16:49:16 +01:00
<div className="flex flex-col h-full">
2025-06-21 17:36:07 +01:00
{/* Header */}
<div className="p-6 border-b space-y-4">
<div className="flex items-start justify-between">
<div className="flex-1 min-w-0">
<h2 className="text-xl font-bold mb-2 line-clamp-2">
{task.title}
</h2>
<div className="flex items-center gap-2 text-sm text-muted-foreground">
<span
className={`px-2 py-1 rounded-full text-xs font-medium ${
task.status === "todo"
? "bg-neutral text-neutral-foreground"
: task.status === "inprogress"
? "bg-info text-info-foreground"
: task.status === "inreview"
? "bg-warning text-warning-foreground"
: task.status === "done"
? "bg-success text-success-foreground"
: "bg-destructive text-destructive-foreground"
}`}
>
{statusLabels[task.status]}
</span>
</div>
</div>
<div className="flex items-center gap-1">
<Button variant="ghost" size="icon" onClick={onClose}>
<X className="h-4 w-4" />
2025-06-21 16:49:16 +01:00
</Button>
</div>
</div>
2025-06-21 17:36:07 +01:00
{/* Attempt Selection */}
<div className="flex items-center gap-2">
{selectedAttempt && !showAttemptHistory ? (
<div className="flex items-center gap-2 flex-1">
<span className="text-sm text-muted-foreground">
Current attempt:
</span>
<span className="text-sm font-medium">
{new Date(
selectedAttempt.created_at
).toLocaleDateString()}{" "}
{new Date(
selectedAttempt.created_at
).toLocaleTimeString()}
</span>
{taskAttempts.length > 1 && (
<Button
variant="outline"
size="sm"
onClick={() => setShowAttemptHistory(true)}
>
<History className="h-4 w-4 mr-1" />
History
</Button>
)}
</div>
) : (
<div className="flex items-center gap-2 flex-1">
<Select
value={selectedAttempt?.id || ""}
onValueChange={handleAttemptChange}
>
<SelectTrigger className="flex-1">
<SelectValue placeholder="Select an attempt..." />
</SelectTrigger>
<SelectContent>
{taskAttempts.map((attempt) => (
<SelectItem key={attempt.id} value={attempt.id}>
<div className="flex flex-col">
<span className="font-medium">
{new Date(
attempt.created_at
).toLocaleDateString()}{" "}
{new Date(
attempt.created_at
).toLocaleTimeString()}
2025-06-21 16:49:16 +01:00
</span>
2025-06-21 17:36:07 +01:00
<span className="text-xs text-muted-foreground text-left">
{attempt.executor || "executor"}
2025-06-21 16:49:16 +01:00
</span>
</div>
2025-06-21 17:36:07 +01:00
</SelectItem>
))}
</SelectContent>
</Select>
<Button
variant="outline"
size="sm"
onClick={() => setShowAttemptHistory(false)}
>
Close
</Button>
</div>
)}
{selectedAttempt && (
<div className="flex gap-1">
<Button
variant="outline"
size="sm"
onClick={openInEditor}
>
<Code className="h-4 w-4 mr-1" />
Editor
</Button>
<Button
variant="outline"
size="sm"
onClick={() =>
window.open(
`/projects/${projectId}/tasks/${task.id}/attempts/${selectedAttempt.id}/compare`,
"_blank"
)
}
>
<FileText className="h-4 w-4 mr-1" />
Changes
</Button>
</div>
)}
</div>
</div>
{/* Content */}
<div className="flex-1 overflow-y-auto p-6 space-y-6">
{loading ? (
<div className="text-center py-8">
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-foreground mx-auto mb-4"></div>
<p className="text-muted-foreground">Loading...</p>
</div>
) : (
<>
{/* Description */}
<div>
<Label className="text-sm font-medium mb-2 block">
Description
</Label>
<div className="p-3 bg-muted rounded-md min-h-[60px]">
{task.description ? (
<p className="text-sm whitespace-pre-wrap">
{task.description}
</p>
) : (
<p className="text-sm text-muted-foreground italic">
No description provided
</p>
)}
</div>
</div>
{/* Execution Processes */}
{selectedAttempt && executionProcesses.length > 0 && (
<div>
<Label className="text-sm font-medium mb-3 block">
Execution Processes
</Label>
<div className="space-y-3">
{executionProcesses.map((process) => (
<Card key={process.id} className="border">
<CardContent className="p-4 space-y-3">
<div className="flex items-center justify-between">
<div className="flex items-center gap-3">
<span
className={`px-2 py-1 rounded-full text-xs font-medium ${
getProcessStatusDisplay(process.status)
.className
}`}
2025-06-21 16:49:16 +01:00
>
2025-06-21 17:36:07 +01:00
{
getProcessStatusDisplay(process.status)
.label
}
</span>
<span className="font-medium text-sm">
{getProcessTypeDisplay(
process.process_type
)}
</span>
</div>
<div className="flex items-center gap-2">
<span className="text-xs text-muted-foreground">
{new Date(
process.started_at
).toLocaleTimeString()}
</span>
{process.status === "running" && (
<Button
onClick={() =>
stopExecutionProcess(process.id)
}
size="sm"
variant="destructive"
>
Stop
</Button>
)}
</div>
</div>
{(process.stdout || process.stderr) && (
<div className="space-y-2">
{process.stdout && (
<div>
<Label className="text-xs text-muted-foreground mb-1 block">
STDOUT
</Label>
<div
className="bg-black text-green-400 border border-green-400 rounded-md p-2 font-mono text-xs max-h-32 overflow-y-auto whitespace-pre-wrap"
style={{
fontFamily:
'ui-monospace, SFMono-Regular, "SF Mono", Monaco, Consolas, "Liberation Mono", "Courier New", monospace',
}}
>
{process.stdout}
</div>
</div>
)}
{process.stderr && (
<div>
<Label className="text-xs text-muted-foreground mb-1 block">
STDERR
</Label>
<div
className="bg-black text-red-400 border border-red-400 rounded-md p-2 font-mono text-xs max-h-32 overflow-y-auto whitespace-pre-wrap"
style={{
fontFamily:
'ui-monospace, SFMono-Regular, "SF Mono", Monaco, Consolas, "Liberation Mono", "Courier New", monospace',
}}
>
{process.stderr}
</div>
</div>
)}
2025-06-21 16:49:16 +01:00
</div>
)}
2025-06-21 17:36:07 +01:00
</CardContent>
</Card>
))}
</div>
</div>
)}
{/* Activity History */}
{selectedAttempt && (
<div>
<Label className="text-sm font-medium mb-3 block">
Activity History
</Label>
{attemptActivities.length === 0 ? (
<div className="text-center py-4 text-muted-foreground">
No activities found
</div>
) : (
<div className="space-y-3">
{attemptActivities.map((activity) => (
<Card key={activity.id} className="border">
<CardContent className="p-4">
<div className="flex items-center justify-between mb-2">
<span
className={`px-2 py-1 rounded-full text-xs font-medium ${
getAttemptStatusDisplay(activity.status)
.className
}`}
2025-06-21 16:49:16 +01:00
>
2025-06-21 17:36:07 +01:00
{
getAttemptStatusDisplay(activity.status)
.label
}
</span>
<div className="flex items-center gap-1 text-xs text-muted-foreground">
<Clock className="h-3 w-3" />
{new Date(
activity.created_at
).toLocaleString()}
2025-06-21 16:49:16 +01:00
</div>
</div>
2025-06-21 17:36:07 +01:00
{activity.note && (
<p className="text-sm text-muted-foreground">
{activity.note}
</p>
)}
</CardContent>
</Card>
))}
</div>
)}
2025-06-21 16:49:16 +01:00
</div>
)}
2025-06-21 17:36:07 +01:00
</>
2025-06-21 16:49:16 +01:00
)}
2025-06-21 17:36:07 +01:00
</div>
2025-06-21 16:49:16 +01:00
2025-06-21 17:36:07 +01:00
{/* Footer */}
<div className="border-t p-4">
<div className="space-y-2">
<Label className="text-sm font-medium">
Follow-up question
</Label>
<div className="flex gap-2">
<Textarea
placeholder="Ask a follow-up question about this task..."
value={followUpMessage}
onChange={(e) => setFollowUpMessage(e.target.value)}
className="flex-1 min-h-[60px] resize-none"
/>
<Button
onClick={handleSendFollowUp}
disabled={!followUpMessage.trim()}
className="self-end"
>
<Send className="h-4 w-4" />
</Button>
</div>
<p className="text-xs text-muted-foreground">
Follow-up functionality coming soon
</p>
</div>
2025-06-21 16:49:16 +01:00
</div>
</div>
</div>
</>
)}
</>
);
}