Files
BrowserOS/packages/browseros-agent/apps/agent/components/ai-elements/chain-of-thought.tsx
Dani Akash 290ee91a8b Add 'packages/browseros-agent/' from commit '90bd4be3008285bf3825aad3702aff98f872671a'
git-subtree-dir: packages/browseros-agent
git-subtree-mainline: 8f148d0918
git-subtree-split: 90bd4be300
2026-03-13 21:22:09 +05:30

236 lines
6.2 KiB
TypeScript

'use client'
import { useControllableState } from '@radix-ui/react-use-controllable-state'
import {
BrainIcon,
ChevronDownIcon,
DotIcon,
type LucideIcon,
} from 'lucide-react'
import type { ComponentProps, ReactNode } from 'react'
import { createContext, memo, useContext, useMemo } from 'react'
import { Badge } from '@/components/ui/badge'
import {
Collapsible,
CollapsibleContent,
CollapsibleTrigger,
} from '@/components/ui/collapsible'
import { cn } from '@/lib/utils'
type ChainOfThoughtContextValue = {
isOpen: boolean
setIsOpen: (open: boolean) => void
}
const ChainOfThoughtContext = createContext<ChainOfThoughtContextValue | null>(
null,
)
const useChainOfThought = () => {
const context = useContext(ChainOfThoughtContext)
if (!context) {
throw new Error(
'ChainOfThought components must be used within ChainOfThought',
)
}
return context
}
export type ChainOfThoughtProps = ComponentProps<'div'> & {
open?: boolean
defaultOpen?: boolean
onOpenChange?: (open: boolean) => void
}
/** @public */
export const ChainOfThought = memo(
({
className,
open,
defaultOpen = false,
onOpenChange,
children,
...props
}: ChainOfThoughtProps) => {
const [isOpen, setIsOpen] = useControllableState({
prop: open,
defaultProp: defaultOpen,
onChange: onOpenChange,
})
const chainOfThoughtContext = useMemo(
() => ({ isOpen, setIsOpen }),
[isOpen, setIsOpen],
)
return (
<ChainOfThoughtContext.Provider value={chainOfThoughtContext}>
<div
className={cn('not-prose max-w-prose space-y-4', className)}
{...props}
>
{children}
</div>
</ChainOfThoughtContext.Provider>
)
},
)
export type ChainOfThoughtHeaderProps = ComponentProps<
typeof CollapsibleTrigger
>
/** @public */
export const ChainOfThoughtHeader = memo(
({ className, children, ...props }: ChainOfThoughtHeaderProps) => {
const { isOpen, setIsOpen } = useChainOfThought()
return (
<Collapsible onOpenChange={setIsOpen} open={isOpen}>
<CollapsibleTrigger
className={cn(
'flex w-full items-center gap-2 text-muted-foreground text-sm transition-colors hover:text-foreground',
className,
)}
{...props}
>
<BrainIcon className="size-4" />
<span className="flex-1 text-left">
{children ?? 'Chain of Thought'}
</span>
<ChevronDownIcon
className={cn(
'size-4 transition-transform',
isOpen ? 'rotate-180' : 'rotate-0',
)}
/>
</CollapsibleTrigger>
</Collapsible>
)
},
)
export type ChainOfThoughtStepProps = ComponentProps<'div'> & {
icon?: LucideIcon
label: ReactNode
description?: ReactNode
status?: 'complete' | 'active' | 'pending'
}
/** @public */
export const ChainOfThoughtStep = memo(
({
className,
icon: Icon = DotIcon,
label,
description,
status = 'complete',
children,
...props
}: ChainOfThoughtStepProps) => {
const statusStyles = {
complete: 'text-muted-foreground',
active: 'text-foreground',
pending: 'text-muted-foreground/50',
}
return (
<div
className={cn(
'flex gap-2 text-sm',
statusStyles[status],
'fade-in-0 slide-in-from-top-2 animate-in',
className,
)}
{...props}
>
<div className="relative mt-0.5">
<Icon className="size-4" />
<div className="absolute top-7 bottom-0 left-1/2 -mx-px w-px bg-border" />
</div>
<div className="flex-1 space-y-2">
<div>{label}</div>
{description && (
<div className="text-muted-foreground text-xs">{description}</div>
)}
{children}
</div>
</div>
)
},
)
export type ChainOfThoughtSearchResultsProps = ComponentProps<'div'>
/** @public */
export const ChainOfThoughtSearchResults = memo(
({ className, ...props }: ChainOfThoughtSearchResultsProps) => (
<div className={cn('flex items-center gap-2', className)} {...props} />
),
)
export type ChainOfThoughtSearchResultProps = ComponentProps<typeof Badge>
/** @public */
export const ChainOfThoughtSearchResult = memo(
({ className, children, ...props }: ChainOfThoughtSearchResultProps) => (
<Badge
className={cn('gap-1 px-2 py-0.5 font-normal text-xs', className)}
variant="secondary"
{...props}
>
{children}
</Badge>
),
)
export type ChainOfThoughtContentProps = ComponentProps<
typeof CollapsibleContent
>
/** @public */
export const ChainOfThoughtContent = memo(
({ className, children, ...props }: ChainOfThoughtContentProps) => {
const { isOpen } = useChainOfThought()
return (
<Collapsible open={isOpen}>
<CollapsibleContent
className={cn(
'mt-2 space-y-3',
'data-[state=closed]:fade-out-0 data-[state=closed]:slide-out-to-top-2 data-[state=open]:slide-in-from-top-2 text-popover-foreground outline-none data-[state=closed]:animate-out data-[state=open]:animate-in',
className,
)}
{...props}
>
{children}
</CollapsibleContent>
</Collapsible>
)
},
)
export type ChainOfThoughtImageProps = ComponentProps<'div'> & {
caption?: string
}
/** @public */
export const ChainOfThoughtImage = memo(
({ className, children, caption, ...props }: ChainOfThoughtImageProps) => (
<div className={cn('mt-2 space-y-2', className)} {...props}>
<div className="relative flex max-h-[22rem] items-center justify-center overflow-hidden rounded-lg bg-muted p-3">
{children}
</div>
{caption && <p className="text-muted-foreground text-xs">{caption}</p>}
</div>
),
)
ChainOfThought.displayName = 'ChainOfThought'
ChainOfThoughtHeader.displayName = 'ChainOfThoughtHeader'
ChainOfThoughtStep.displayName = 'ChainOfThoughtStep'
ChainOfThoughtSearchResults.displayName = 'ChainOfThoughtSearchResults'
ChainOfThoughtSearchResult.displayName = 'ChainOfThoughtSearchResult'
ChainOfThoughtContent.displayName = 'ChainOfThoughtContent'
ChainOfThoughtImage.displayName = 'ChainOfThoughtImage'