| | import { Search, X } from 'lucide-react'; |
| | import React, { useState, useMemo, useCallback, useRef } from 'react'; |
| | import { cn } from '~/utils'; |
| |
|
| | |
| | export default function MultiSearch({ |
| | value, |
| | onChange, |
| | placeholder, |
| | className = '', |
| | }: { |
| | value: string | null; |
| | onChange: (filter: string) => void; |
| | placeholder?: string; |
| | className?: string; |
| | }) { |
| | const inputRef = useRef<HTMLInputElement>(null); |
| |
|
| | const onChangeHandler: React.ChangeEventHandler<HTMLInputElement> = useCallback( |
| | (e) => onChange(e.target.value), |
| | [onChange], |
| | ); |
| |
|
| | const clearSearch = () => { |
| | onChange(''); |
| | setTimeout(() => { |
| | inputRef.current?.focus(); |
| | }, 0); |
| | }; |
| |
|
| | return ( |
| | <div |
| | className={cn( |
| | 'focus:to-surface-primary/50 group sticky left-0 top-0 z-10 flex h-12 items-center gap-2 bg-gradient-to-b from-surface-tertiary-alt from-65% to-transparent px-3 py-2 text-text-primary transition-colors duration-300 focus:bg-gradient-to-b focus:from-surface-primary', |
| | className, |
| | )} |
| | > |
| | <Search |
| | className="h-4 w-4 text-text-secondary-alt transition-colors duration-300" |
| | aria-hidden={'true'} |
| | /> |
| | <input |
| | ref={inputRef} |
| | type="text" |
| | value={value ?? ''} |
| | onChange={onChangeHandler} |
| | placeholder={String(placeholder ?? 'Search...')} |
| | aria-label="Search Model" |
| | className="flex-1 rounded-md border-none bg-transparent px-2.5 py-2 text-sm placeholder-text-secondary focus:outline-none focus:ring-1 focus:ring-ring-primary" |
| | /> |
| | <button |
| | className={cn( |
| | 'relative flex h-5 w-5 items-center justify-end rounded-md text-text-secondary-alt', |
| | (value?.length ?? 0) ? 'cursor-pointer opacity-100' : 'hidden', |
| | )} |
| | aria-label={'Clear search'} |
| | onClick={clearSearch} |
| | tabIndex={0} |
| | > |
| | <X |
| | aria-hidden={'true'} |
| | className={cn( |
| | 'text-text-secondary-alt', |
| | (value?.length ?? 0) ? 'cursor-pointer opacity-100' : 'opacity-0', |
| | )} |
| | /> |
| | </button> |
| | </div> |
| | ); |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | function defaultGetStringKey(node: unknown): string { |
| | if (typeof node === 'string') { |
| | |
| | |
| | |
| | |
| | if (node.startsWith('---') && node.endsWith('---')) { |
| | return ''; |
| | } |
| |
|
| | return node.toUpperCase(); |
| | } |
| | |
| | return ''; |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | export function useMultiSearch<OptionsType extends unknown[]>({ |
| | availableOptions = [] as unknown as OptionsType, |
| | placeholder, |
| | getTextKeyOverride, |
| | className, |
| | disabled = false, |
| | }: { |
| | availableOptions?: OptionsType; |
| | placeholder?: string; |
| | getTextKeyOverride?: (node: OptionsType[0]) => string; |
| | className?: string; |
| | disabled?: boolean; |
| | }): [OptionsType, React.ReactNode] { |
| | const [filterValue, setFilterValue] = useState<string | null>(null); |
| |
|
| | |
| | const shouldShowSearch = availableOptions.length > 10 && !disabled; |
| |
|
| | |
| | |
| | const getTextKeyHelper = getTextKeyOverride || defaultGetStringKey; |
| |
|
| | |
| | const filteredOptions = useMemo(() => { |
| | const currentFilter = filterValue ?? ''; |
| | if (!shouldShowSearch || !currentFilter || !availableOptions.length) { |
| | |
| | return availableOptions; |
| | } |
| | |
| | |
| | const upperFilterValue = currentFilter.toUpperCase(); |
| |
|
| | return availableOptions.filter((value) => |
| | getTextKeyHelper(value).includes(upperFilterValue), |
| | ) as OptionsType; |
| | }, [availableOptions, getTextKeyHelper, filterValue, shouldShowSearch]); |
| |
|
| | const onSearchChange = useCallback( |
| | (nextFilterValue: string) => setFilterValue(nextFilterValue), |
| | [], |
| | ); |
| |
|
| | const searchRender = shouldShowSearch ? ( |
| | <MultiSearch |
| | value={filterValue} |
| | className={className} |
| | onChange={onSearchChange} |
| | placeholder={placeholder} |
| | /> |
| | ) : null; |
| |
|
| | return [filteredOptions, searchRender]; |
| | } |
| |
|