162 lines
4.2 KiB
TypeScript
162 lines
4.2 KiB
TypeScript
import { Button } from "@/components/ui/button"
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuTrigger
|
|
} from "@/components/ui/dropdown-menu"
|
|
import { Input } from "@/components/ui/input"
|
|
import { ChatbotUIContext } from "@/context/context"
|
|
import { Tables } from "@/supabase/types"
|
|
import {
|
|
IconBolt,
|
|
IconChevronDown,
|
|
IconCircleCheckFilled
|
|
} from "@tabler/icons-react"
|
|
import { FC, useContext, useEffect, useRef, useState } from "react"
|
|
|
|
interface AssistantToolSelectProps {
|
|
selectedAssistantTools: Tables<"tools">[]
|
|
onAssistantToolsSelect: (tool: Tables<"tools">) => void
|
|
}
|
|
|
|
export const AssistantToolSelect: FC<AssistantToolSelectProps> = ({
|
|
selectedAssistantTools,
|
|
onAssistantToolsSelect
|
|
}) => {
|
|
const { tools } = useContext(ChatbotUIContext)
|
|
|
|
const inputRef = useRef<HTMLInputElement>(null)
|
|
const triggerRef = useRef<HTMLButtonElement>(null)
|
|
|
|
const [isOpen, setIsOpen] = useState(false)
|
|
const [search, setSearch] = useState("")
|
|
|
|
useEffect(() => {
|
|
if (isOpen) {
|
|
setTimeout(() => {
|
|
inputRef.current?.focus()
|
|
}, 100) // FIX: hacky
|
|
}
|
|
}, [isOpen])
|
|
|
|
const handleToolSelect = (tool: Tables<"tools">) => {
|
|
onAssistantToolsSelect(tool)
|
|
}
|
|
|
|
if (!tools) return null
|
|
|
|
return (
|
|
<DropdownMenu
|
|
open={isOpen}
|
|
onOpenChange={isOpen => {
|
|
setIsOpen(isOpen)
|
|
setSearch("")
|
|
}}
|
|
>
|
|
<DropdownMenuTrigger
|
|
className="bg-background w-full justify-start border-2 px-3 py-5"
|
|
asChild
|
|
>
|
|
<Button
|
|
ref={triggerRef}
|
|
className="flex items-center justify-between"
|
|
variant="ghost"
|
|
>
|
|
<div className="flex items-center">
|
|
<div className="ml-2 flex items-center">
|
|
{selectedAssistantTools.length} tools selected
|
|
</div>
|
|
</div>
|
|
|
|
<IconChevronDown />
|
|
</Button>
|
|
</DropdownMenuTrigger>
|
|
|
|
<DropdownMenuContent
|
|
style={{ width: triggerRef.current?.offsetWidth }}
|
|
className="space-y-2 overflow-auto p-2"
|
|
align="start"
|
|
>
|
|
<Input
|
|
ref={inputRef}
|
|
placeholder="Search tools..."
|
|
value={search}
|
|
onChange={e => setSearch(e.target.value)}
|
|
onKeyDown={e => e.stopPropagation()}
|
|
/>
|
|
|
|
{selectedAssistantTools
|
|
.filter(item =>
|
|
item.name.toLowerCase().includes(search.toLowerCase())
|
|
)
|
|
.map(tool => (
|
|
<AssistantToolItem
|
|
key={tool.id}
|
|
tool={tool}
|
|
selected={selectedAssistantTools.some(
|
|
selectedAssistantRetrieval =>
|
|
selectedAssistantRetrieval.id === tool.id
|
|
)}
|
|
onSelect={handleToolSelect}
|
|
/>
|
|
))}
|
|
|
|
{tools
|
|
.filter(
|
|
tool =>
|
|
!selectedAssistantTools.some(
|
|
selectedAssistantRetrieval =>
|
|
selectedAssistantRetrieval.id === tool.id
|
|
) && tool.name.toLowerCase().includes(search.toLowerCase())
|
|
)
|
|
.map(tool => (
|
|
<AssistantToolItem
|
|
key={tool.id}
|
|
tool={tool}
|
|
selected={selectedAssistantTools.some(
|
|
selectedAssistantRetrieval =>
|
|
selectedAssistantRetrieval.id === tool.id
|
|
)}
|
|
onSelect={handleToolSelect}
|
|
/>
|
|
))}
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
)
|
|
}
|
|
|
|
interface AssistantToolItemProps {
|
|
tool: Tables<"tools">
|
|
selected: boolean
|
|
onSelect: (tool: Tables<"tools">) => void
|
|
}
|
|
|
|
const AssistantToolItem: FC<AssistantToolItemProps> = ({
|
|
tool,
|
|
selected,
|
|
onSelect
|
|
}) => {
|
|
const handleSelect = () => {
|
|
onSelect(tool)
|
|
}
|
|
|
|
return (
|
|
<div
|
|
className="flex cursor-pointer items-center justify-between py-0.5 hover:opacity-50"
|
|
onClick={handleSelect}
|
|
>
|
|
<div className="flex grow items-center truncate">
|
|
<div className="mr-2 min-w-[24px]">
|
|
<IconBolt size={24} />
|
|
</div>
|
|
|
|
<div className="truncate">{tool.name}</div>
|
|
</div>
|
|
|
|
{selected && (
|
|
<IconCircleCheckFilled size={20} className="min-w-[30px] flex-none" />
|
|
)}
|
|
</div>
|
|
)
|
|
}
|