Files
2026-04-13 15:34:57 -04:00

133 lines
3.3 KiB
TypeScript

"use client";
import {
Collapsible,
CollapsibleContent,
CollapsibleTrigger,
} from "@/components/ui/collapsible";
import {
Tabs,
TabsContent,
TabsList,
TabsTrigger,
} from "@/components/ui/tabs";
import { cn } from "@/lib/utils";
import type { ToolUIPart } from "ai";
import { ChevronDownIcon, Code } from "lucide-react";
import type { ComponentProps } from "react";
import { getStatusBadge } from "./tool";
export type SandboxRootProps = ComponentProps<typeof Collapsible>;
export const Sandbox = ({ className, ...props }: SandboxRootProps) => (
<Collapsible
className={cn(
"not-prose group mb-4 w-full overflow-hidden rounded-md border",
className
)}
defaultOpen
{...props}
/>
);
export interface SandboxHeaderProps {
title?: string;
state: ToolUIPart["state"];
className?: string;
}
export const SandboxHeader = ({
className,
title,
state,
...props
}: SandboxHeaderProps) => (
<CollapsibleTrigger
className={cn(
"flex w-full items-center justify-between gap-4 p-3",
className
)}
{...props}
>
<div className="flex items-center gap-2">
<Code className="size-4 text-muted-foreground" />
<span className="font-medium text-sm">{title}</span>
{getStatusBadge(state)}
</div>
<ChevronDownIcon className="size-4 text-muted-foreground transition-transform group-data-[state=open]:rotate-180" />
</CollapsibleTrigger>
);
export type SandboxContentProps = ComponentProps<typeof CollapsibleContent>;
export const SandboxContent = ({
className,
...props
}: SandboxContentProps) => (
<CollapsibleContent
className={cn(
"data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 outline-none data-[state=closed]:animate-out data-[state=open]:animate-in",
className
)}
{...props}
/>
);
export type SandboxTabsProps = ComponentProps<typeof Tabs>;
export const SandboxTabs = ({ className, ...props }: SandboxTabsProps) => (
<Tabs className={cn("w-full gap-0", className)} {...props} />
);
export type SandboxTabsBarProps = ComponentProps<"div">;
export const SandboxTabsBar = ({
className,
...props
}: SandboxTabsBarProps) => (
<div
className={cn(
"flex w-full items-center border-border border-t border-b",
className
)}
{...props}
/>
);
export type SandboxTabsListProps = ComponentProps<typeof TabsList>;
export const SandboxTabsList = ({
className,
...props
}: SandboxTabsListProps) => (
<TabsList
className={cn("h-auto rounded-none border-0 bg-transparent p-0", className)}
{...props}
/>
);
export type SandboxTabsTriggerProps = ComponentProps<typeof TabsTrigger>;
export const SandboxTabsTrigger = ({
className,
...props
}: SandboxTabsTriggerProps) => (
<TabsTrigger
className={cn(
"rounded-none border-0 border-transparent border-b-2 px-4 py-2 font-medium text-muted-foreground text-sm transition-colors data-[state=active]:border-primary data-[state=active]:bg-transparent data-[state=active]:text-foreground data-[state=active]:shadow-none",
className
)}
{...props}
/>
);
export type SandboxTabContentProps = ComponentProps<typeof TabsContent>;
export const SandboxTabContent = ({
className,
...props
}: SandboxTabContentProps) => (
<TabsContent className={cn("mt-0 text-sm", className)} {...props} />
);