Files
dify/web/app/components/workflow/nodes/code/dependency-picker.tsx
Coding On Star d65a6b4810 refactor: migrate from PortalToFollowElem to Popover component across various components (#35454)
Co-authored-by: CodingOnStar <hanxujiang@dify.com>
Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com>
2026-04-21 10:09:22 +00:00

96 lines
3.0 KiB
TypeScript

import type { FC } from 'react'
import type { CodeDependency } from './types'
import {
Popover,
PopoverContent,
PopoverTrigger,
} from '@langgenius/dify-ui/popover'
import {
RiArrowDownSLine,
} from '@remixicon/react'
import { t } from 'i18next'
import * as React from 'react'
import { useCallback, useState } from 'react'
import { Check } from '@/app/components/base/icons/src/vender/line/general'
import Input from '@/app/components/base/input'
type Props = {
value: CodeDependency
available_dependencies: CodeDependency[]
onChange: (dependency: CodeDependency) => void
}
const DependencyPicker: FC<Props> = ({
available_dependencies,
value,
onChange,
}) => {
const [open, setOpen] = useState(false)
const [searchText, setSearchText] = useState('')
const handleChange = useCallback((dependency: CodeDependency) => {
return () => {
setOpen(false)
onChange(dependency)
}
}, [onChange])
return (
<Popover open={open} onOpenChange={setOpen}>
<PopoverTrigger
render={(
<div className="grow cursor-pointer">
<div className="flex h-8 items-center justify-between rounded-lg border-0 bg-gray-100 px-2.5 text-[13px] text-gray-900">
<div className="w-0 grow truncate" title={value.name}>{value.name}</div>
<RiArrowDownSLine className="h-3.5 w-3.5 shrink-0 text-gray-700" />
</div>
</div>
)}
/>
<PopoverContent
placement="bottom-start"
sideOffset={4}
popupClassName="border-none bg-transparent p-0 shadow-none backdrop-blur-none"
positionerProps={{ style: { zIndex: 100 } }}
>
<div
className="rounded-lg bg-white p-1 shadow-sm"
style={{
width: 350,
}}
>
<div className="mx-1 mb-2">
<Input
showLeftIcon
showClearIcon
value={searchText}
placeholder={t('nodes.code.searchDependencies', { ns: 'workflow' }) || ''}
onChange={e => setSearchText(e.target.value)}
onClear={() => setSearchText('')}
autoFocus
/>
</div>
<div className="max-h-[30vh] overflow-y-auto">
{available_dependencies.filter((v) => {
if (!searchText)
return true
return v.name.toLowerCase().includes(searchText.toLowerCase())
}).map(dependency => (
<div
key={dependency.name}
className="flex h-[30px] cursor-pointer items-center justify-between rounded-lg pr-2 pl-3 text-[13px] text-gray-900 hover:bg-gray-100"
onClick={handleChange(dependency)}
>
<div className="w-0 grow truncate">{dependency.name}</div>
{dependency.name === value.name && <Check className="h-4 w-4 shrink-0 text-primary-600" />}
</div>
))}
</div>
</div>
</PopoverContent>
</Popover>
)
}
export default React.memo(DependencyPicker)