161 lines
4.3 KiB
TypeScript
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>
|
|
)
|
|
}
|