dify/web/app/components/base/search-input/index.tsx

101 lines
3.4 KiB
TypeScript

import type { ComponentProps } from 'react'
import { cn } from '@langgenius/dify-ui/cn'
import { Input } from '@langgenius/dify-ui/input'
import { useRef, useState } from 'react'
import { useTranslation } from 'react-i18next'
type SearchInputProps = {
value: string
onValueChange: (value: string) => void
placeholder?: string
className?: string
} & Pick<ComponentProps<'input'>, 'aria-label'>
export function SearchInput({
placeholder,
className,
value,
onValueChange,
'aria-label': ariaLabel,
}: SearchInputProps) {
const { t } = useTranslation()
const inputRef = useRef<HTMLInputElement>(null)
const isComposingRef = useRef<boolean>(false)
const compositionCommitRef = useRef<string | null>(null)
const [compositionValue, setCompositionValue] = useState('')
const inputValue = isComposingRef.current ? compositionValue : value
const handleClear = () => {
isComposingRef.current = false
compositionCommitRef.current = null
setCompositionValue('')
onValueChange('')
inputRef.current?.focus()
}
return (
<div className={cn(
'relative',
className,
)}
>
<span className="pointer-events-none absolute top-1/2 left-2 i-ri-search-line size-4 -translate-y-1/2 text-components-input-text-placeholder" aria-hidden="true" />
<Input
ref={inputRef}
type="search"
name="query"
aria-label={ariaLabel ?? t('operation.search', { ns: 'common' })}
className={cn(
'ps-7',
!!inputValue && 'pe-7',
'[&::-webkit-search-cancel-button]:appearance-none [&::-webkit-search-decoration]:appearance-none',
)}
placeholder={placeholder ?? t('operation.search', { ns: 'common' })}
value={inputValue}
onValueChange={(nextValue) => {
if (isComposingRef.current) {
setCompositionValue(nextValue)
return
}
if (compositionCommitRef.current !== null) {
if (compositionCommitRef.current !== nextValue) {
compositionCommitRef.current = null
onValueChange(nextValue)
return
}
compositionCommitRef.current = null
return
}
onValueChange(nextValue)
}}
onCompositionStart={() => {
isComposingRef.current = true
compositionCommitRef.current = null
setCompositionValue(value)
}}
onCompositionEnd={(e) => {
if (!isComposingRef.current)
return
isComposingRef.current = false
setCompositionValue('')
compositionCommitRef.current = e.currentTarget.value
onValueChange(e.currentTarget.value)
}}
autoComplete="off"
enterKeyHint="search"
/>
{!!inputValue && (
<button
type="button"
aria-label={t('operation.clear', { ns: 'common' })}
className="group/clear absolute top-1/2 right-2 flex size-4 -translate-y-1/2 cursor-pointer touch-manipulation items-center justify-center rounded-md border-none bg-transparent p-0 outline-hidden focus-visible:ring-1 focus-visible:ring-components-input-border-active focus-visible:ring-inset"
onClick={handleClear}
>
<span className="i-ri-close-circle-fill size-4 text-text-quaternary group-hover/clear:text-text-tertiary" aria-hidden="true" />
</button>
)}
</div>
)
}