| 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]; |
| } |
|
|