N-FIN-30: add new auto-complete component

This commit is contained in:
Markus Thielker 2024-03-16 20:37:37 +01:00
parent 586182e69b
commit 3454c8a03a
No known key found for this signature in database

View file

@ -0,0 +1,100 @@
'use client';
import * as React from 'react';
import { useEffect, useState } from 'react';
import { cn } from '@/lib/utils';
export interface AutoCompleteInputProps
extends React.InputHTMLAttributes<HTMLInputElement> {
items: { label: string, value: any }[];
next?: React.RefObject<HTMLInputElement>;
}
const AutoCompleteInput = React.forwardRef<HTMLInputElement, AutoCompleteInputProps>(
({className, type, ...props}, ref) => {
const [value, setValue] = useState(getInitialValue());
const [open, setOpen] = useState(false);
const [lastKey, setLastKey] = useState('');
const [filteredItems, setFilteredItems] = useState(props.items);
function getInitialValue() {
if (!props.items) {
return '';
}
const item = props.items?.find(item => item.value === props.value);
return item?.label || '';
}
function handleChange(e: React.ChangeEvent<HTMLInputElement>) {
props.onChange?.(undefined as any);
const value = e.target.value;
setFilteredItems(props?.items?.filter((item) => {
return item.label.toLowerCase().includes(value.toLowerCase());
}));
setValue(value);
setOpen(value.length > 0);
}
useEffect(() => {
if (filteredItems.length === 1 && /^[a-zA-Z0-9]$/.test(lastKey)) {
setValue(filteredItems[0].label);
setOpen(false);
props.onChange?.({target: {value: filteredItems[0].value}} as any);
props.next && props.next.current?.focus();
}
}, [filteredItems]);
return (
<div className="relative">
<input
type={type}
className={cn(
'flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50',
className,
)}
ref={ref}
value={value}
placeholder={props.placeholder || 'Search...'}
onChange={handleChange}
onKeyDown={(e) => {
if (e.metaKey || e.ctrlKey || e.altKey) {
props.onKeyDown?.(e);
return;
}
setLastKey(e.key);
props.onKeyDown?.(e);
}}
/>
{
open && (
<div
className="z-50 bg-background rounded-md border border-border absolute inset-x-0 top-12 max-h-44 overflow-scroll">
{filteredItems?.map((item) =>
<div
className="px-3 py-3 hover:bg-accent hover:text-accent-foreground cursor-pointer text-sm font-medium"
onClick={() => {
props.onChange?.({target: {value: item.value}} as any);
props.next && props.next.current?.focus();
setValue(item.label);
setOpen(false);
}}
key={item.value}>
{item.label}
</div>,
)}
</div>
)
}
</div>
);
},
);
AutoCompleteInput.displayName = 'Input';
export { AutoCompleteInput };