chatai/chatbot-ui/components/workspace/assign-workspaces.tsx

161 lines
4.3 KiB
TypeScript

import { ChatbotUIContext } from "@/context/context"
import { Tables } from "@/supabase/types"
import { IconChevronDown, IconCircleCheckFilled } from "@tabler/icons-react"
import { FC, useContext, useEffect, useRef, useState } from "react"
import { Button } from "../ui/button"
import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuTrigger
} from "../ui/dropdown-menu"
import { Input } from "../ui/input"
import { toast } from "sonner"
interface AssignWorkspaces {
selectedWorkspaces: Tables<"workspaces">[]
onSelectWorkspace: (workspace: Tables<"workspaces">) => void
}
export const AssignWorkspaces: FC<AssignWorkspaces> = ({
selectedWorkspaces,
onSelectWorkspace
}) => {
const { workspaces } = 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 handleWorkspaceSelect = (workspace: Tables<"workspaces">) => {
onSelectWorkspace(workspace)
}
if (!workspaces) 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">
{selectedWorkspaces.length} workspaces 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 workspaces..."
value={search}
onChange={e => setSearch(e.target.value)}
onKeyDown={e => e.stopPropagation()}
/>
{selectedWorkspaces
.filter(workspace =>
workspace.name.toLowerCase().includes(search.toLowerCase())
)
.map(workspace => (
<WorkspaceItem
key={workspace.id}
selectedWorkspaces={selectedWorkspaces}
workspace={workspace}
selected={selectedWorkspaces.some(
selectedWorkspace => selectedWorkspace.id === workspace.id
)}
onSelect={handleWorkspaceSelect}
/>
))}
{workspaces
.filter(
workspace =>
!selectedWorkspaces.some(
selectedWorkspace => selectedWorkspace.id === workspace.id
) && workspace.name.toLowerCase().includes(search.toLowerCase())
)
.map(workspace => (
<WorkspaceItem
key={workspace.id}
selectedWorkspaces={selectedWorkspaces}
workspace={workspace}
selected={selectedWorkspaces.some(
selectedWorkspace => selectedWorkspace.id === workspace.id
)}
onSelect={handleWorkspaceSelect}
/>
))}
</DropdownMenuContent>
</DropdownMenu>
)
}
interface WorkspaceItemProps {
selectedWorkspaces: Tables<"workspaces">[]
workspace: Tables<"workspaces">
selected: boolean
onSelect: (workspace: Tables<"workspaces">) => void
}
const WorkspaceItem: FC<WorkspaceItemProps> = ({
selectedWorkspaces,
workspace,
selected,
onSelect
}) => {
const handleSelect = () => {
if (selected && selectedWorkspaces.length === 1) {
toast.info("You must select at least one workspace")
return
}
onSelect(workspace)
}
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="truncate">{workspace.name}</div>
</div>
{selected && (
<IconCircleCheckFilled size={20} className="min-w-[30px] flex-none" />
)}
</div>
)
}