tabby/ee/tabby-ui/components/ui/combobox.tsx

249 lines
6.6 KiB
TypeScript

import * as React from 'react'
import {
Popover,
PopoverAnchor,
PopoverPortal,
PopoverContent,
PopoverClose
} from '@/components/ui/popover'
import { cn } from '@/lib/utils'
import { useCombobox } from 'downshift'
import type {
UseComboboxReturnValue,
UseComboboxState,
UseComboboxStateChangeOptions
} from 'downshift'
import Textarea from 'react-textarea-autosize'
import { isNil, omitBy } from 'lodash-es'
interface ComboboxContextValue<T = any> extends UseComboboxReturnValue<T> {
open: boolean
inputRef: React.RefObject<HTMLInputElement | HTMLTextAreaElement>
anchorRef: React.RefObject<HTMLElement>
}
export const ComboboxContext = React.createContext({} as ComboboxContextValue)
export const ComboboxClose = PopoverClose
export const ComboboxAnchor = React.forwardRef<
React.ElementRef<typeof PopoverAnchor>,
React.ComponentPropsWithoutRef<typeof PopoverAnchor>
>((props, forwardRef) => {
return <PopoverAnchor {...props} ref={forwardRef} />
})
ComboboxAnchor.displayName = 'ComboboxAnchor'
export const ComboboxTextarea = React.forwardRef<
React.ElementRef<typeof Textarea>,
React.ComponentPropsWithoutRef<typeof Textarea>
>((props, forwardRef) => {
const { getInputProps, open } = React.useContext(ComboboxContext)
const { onKeyDown, onChange, onInput, onBlur, onClick, ...rest } = props
return (
<Textarea
{...getInputProps(
omitBy(
{
onKeyDown: (e: React.KeyboardEvent<HTMLTextAreaElement>) => {
if (e.key === 'Enter' && open) {
e.preventDefault()
} else if (!open) {
onKeyDown?.(e)
}
},
onChange,
onInput,
onBlur,
onClick,
ref: forwardRef
},
isNil
)
)}
{...rest}
/>
)
})
ComboboxTextarea.displayName = 'ComboboxTextarea'
export const ComboboxContent = React.forwardRef<
React.ElementRef<typeof PopoverContent>,
React.ComponentPropsWithoutRef<typeof PopoverContent> & {
popupMatchAnchorWidth?: boolean
}
>(({ children, style, popupMatchAnchorWidth, ...rest }, forwardRef) => {
const { getMenuProps, anchorRef } = React.useContext(ComboboxContext)
const popupWidth = React.useRef<number | undefined>(undefined)
React.useLayoutEffect(() => {
if (popupMatchAnchorWidth) {
const anchor = anchorRef.current
if (anchor) {
const rect = anchor.getBoundingClientRect()
popupWidth.current = rect.width
}
}
}, [])
return (
<PopoverPortal>
<PopoverContent
align="start"
onOpenAutoFocus={e => e.preventDefault()}
style={{
width: popupWidth.current,
...style
}}
{...getMenuProps({ ref: forwardRef }, { suppressRefError: true })}
{...rest}
>
{children}
</PopoverContent>
</PopoverPortal>
)
})
ComboboxContent.displayName = 'ComboboxContent'
interface ComboboxOptionProps<T = any> {
item: T
index: number
className?: string
disabled?: boolean
children?:
| React.ReactNode
| React.ReactNode[]
| ((p: { selected: boolean; highlighted: boolean }) => React.ReactNode)
}
export const ComboboxOption = React.forwardRef<
React.RefObject<HTMLDivElement>,
ComboboxOptionProps
>(({ item, index, className, children, disabled, ...rest }, forwardRef) => {
const { highlightedIndex, selectedItem, getItemProps } =
React.useContext(ComboboxContext)
const highlighted = highlightedIndex === index
const selected = selectedItem === item
return (
<ComboboxClose key={item.id} asChild>
<div
className={cn(
'relative flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none',
highlighted && 'text-accent-foreground bg-accent',
selected && 'font-bold',
disabled && 'pointer-events-none opacity-50',
className
)}
{...getItemProps({
item,
index,
onMouseLeave: e => e.preventDefault(),
onMouseOut: e => e.preventDefault()
})}
{...rest}
>
{typeof children === 'function'
? children({ highlighted, selected })
: children}
</div>
</ComboboxClose>
)
})
ComboboxOption.displayName = 'ComboboxOption'
interface ComboboxProps<T> {
options: T[] | undefined
onSelect?: (
ref: React.RefObject<HTMLTextAreaElement | HTMLInputElement>,
data: T
) => void
inputRef?: React.RefObject<HTMLTextAreaElement | HTMLInputElement>
children?:
| React.ReactNode
| React.ReactNode[]
| ((contextValue: ComboboxContextValue) => React.ReactNode)
}
export function Combobox<T extends { id: number }>({
options,
onSelect,
inputRef: propsInputRef,
children
}: ComboboxProps<T>) {
const [manualOpen, setManualOpen] = React.useState(false)
const internalInputRef = React.useRef<HTMLTextAreaElement | HTMLInputElement>(
null
)
const inputRef = propsInputRef ?? internalInputRef
const anchorRef = React.useRef<HTMLElement>(null)
const stateReducer = React.useCallback(
(
state: UseComboboxState<T>,
actionAndChanges: UseComboboxStateChangeOptions<T>
) => {
const { type, changes } = actionAndChanges
switch (type) {
case useCombobox.stateChangeTypes.MenuMouseLeave:
return {
...changes,
highlightedIndex: state.highlightedIndex
}
default:
return changes
}
},
[]
)
const comboboxValue = useCombobox({
items: options ?? [],
isOpen: manualOpen,
onSelectedItemChange({ selectedItem }) {
if (selectedItem) {
onSelect?.(inputRef, selectedItem)
setManualOpen(false)
}
},
onIsOpenChange: ({ isOpen }) => {
if (!isOpen) {
setManualOpen(false)
}
},
stateReducer
})
const { setHighlightedIndex, highlightedIndex } = comboboxValue
const open = manualOpen && !!options?.length
React.useEffect(() => {
if (open && !!options.length && highlightedIndex === -1) {
setHighlightedIndex(0)
}
if (open && !options.length) {
setManualOpen(false)
}
}, [open, options])
React.useEffect(() => {
if (options?.length) {
setManualOpen(true)
} else {
setManualOpen(false)
}
}, [options])
const contextValue = React.useMemo(() => {
return { ...comboboxValue, open, inputRef, anchorRef }
}, [comboboxValue, open, inputRef, anchorRef])
return (
<ComboboxContext.Provider value={contextValue}>
<Popover open={open}>
{typeof children === 'function' ? children(contextValue) : children}
</Popover>
</ComboboxContext.Provider>
)
}