mirror of
https://github.com/dograh-hq/dograh.git
synced 2026-06-10 08:05:22 +02:00
116 lines
5.7 KiB
TypeScript
116 lines
5.7 KiB
TypeScript
"use client";
|
|
|
|
import { Brain, ChevronRight, Wrench } from "lucide-react";
|
|
import { useState } from "react";
|
|
|
|
import { Badge } from "@/components/ui/badge";
|
|
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from "@/components/ui/collapsible";
|
|
import { cn } from "@/lib/utils";
|
|
|
|
import { formatConversationValue } from "./utils";
|
|
|
|
interface ToolCallCardProps {
|
|
functionName: string;
|
|
status: "running" | "completed";
|
|
argumentsValue?: unknown;
|
|
resultValue?: unknown;
|
|
reasoningDurationMs?: number;
|
|
}
|
|
|
|
export function ToolCallCard({
|
|
functionName,
|
|
status,
|
|
argumentsValue,
|
|
resultValue,
|
|
reasoningDurationMs,
|
|
}: ToolCallCardProps) {
|
|
const [open, setOpen] = useState(false);
|
|
const hasArguments = argumentsValue !== undefined;
|
|
const hasResult = resultValue !== undefined;
|
|
const hasDetails = hasArguments || hasResult;
|
|
|
|
return (
|
|
<div className="flex justify-center">
|
|
<div className="flex w-full max-w-[85%] flex-col gap-1">
|
|
{reasoningDurationMs !== undefined ? (
|
|
<div className="flex items-center justify-center gap-1.5 text-xs text-muted-foreground">
|
|
<Brain className="h-3 w-3" />
|
|
<span className="font-medium">Reasoning Delay:</span>
|
|
<span>{Math.round(reasoningDurationMs)}ms</span>
|
|
</div>
|
|
) : null}
|
|
<Collapsible
|
|
open={hasDetails ? open : false}
|
|
onOpenChange={hasDetails ? setOpen : undefined}
|
|
className="rounded-2xl border border-amber-500/20 bg-amber-500/10"
|
|
>
|
|
<div className="flex items-start gap-2 px-3.5 py-3 text-sm">
|
|
<Wrench className="mt-0.5 h-4 w-4 shrink-0 text-amber-500" />
|
|
<div className="min-w-0 flex-1">
|
|
<div className="flex flex-wrap items-center gap-2">
|
|
<span className="font-mono text-xs text-amber-700 dark:text-amber-400">
|
|
{functionName}()
|
|
</span>
|
|
<Badge
|
|
variant="outline"
|
|
className={cn(
|
|
"h-5 px-1.5 text-[10px] uppercase tracking-[0.14em]",
|
|
status === "running"
|
|
? "border-amber-400/60 text-amber-700 dark:text-amber-300"
|
|
: "border-emerald-500/30 text-emerald-700 dark:text-emerald-300",
|
|
)}
|
|
>
|
|
{status === "running" ? "Running" : "Completed"}
|
|
</Badge>
|
|
</div>
|
|
{hasDetails ? (
|
|
<div className="mt-2">
|
|
<CollapsibleTrigger asChild>
|
|
<button
|
|
type="button"
|
|
className="inline-flex items-center gap-1 text-xs text-muted-foreground hover:text-foreground"
|
|
>
|
|
<ChevronRight
|
|
className={cn(
|
|
"h-3.5 w-3.5 transition-transform",
|
|
open && "rotate-90",
|
|
)}
|
|
/>
|
|
Details
|
|
</button>
|
|
</CollapsibleTrigger>
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
</div>
|
|
{hasDetails ? (
|
|
<CollapsibleContent className="border-t border-amber-500/20 px-3.5 py-3">
|
|
<div className="space-y-3">
|
|
{hasArguments ? (
|
|
<div className="space-y-1">
|
|
<p className="text-[11px] font-medium uppercase tracking-[0.14em] text-muted-foreground">
|
|
Arguments
|
|
</p>
|
|
<pre className="overflow-x-auto rounded-xl bg-background/70 p-3 text-xs leading-5 text-foreground">
|
|
{formatConversationValue(argumentsValue)}
|
|
</pre>
|
|
</div>
|
|
) : null}
|
|
{hasResult ? (
|
|
<div className="space-y-1">
|
|
<p className="text-[11px] font-medium uppercase tracking-[0.14em] text-muted-foreground">
|
|
Result
|
|
</p>
|
|
<pre className="overflow-x-auto rounded-xl bg-background/70 p-3 text-xs leading-5 text-foreground">
|
|
{formatConversationValue(resultValue)}
|
|
</pre>
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
</CollapsibleContent>
|
|
) : null}
|
|
</Collapsible>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|