12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- import type { FC } from 'react'
- import React, { useCallback, useState } from 'react'
- import { t } from 'i18next'
- import {
- RiArrowDownSLine,
- RiSearchLine,
- } from '@remixicon/react'
- import type { CodeDependency } from './types'
- import { PortalToFollowElem, PortalToFollowElemContent, PortalToFollowElemTrigger } from '@/app/components/base/portal-to-follow-elem'
- import { Check } from '@/app/components/base/icons/src/vender/line/general'
- import { XCircle } from '@/app/components/base/icons/src/vender/solid/general'
- 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 (
- <PortalToFollowElem
- open={open}
- onOpenChange={setOpen}
- placement='bottom-start'
- offset={4}
- >
- <PortalToFollowElemTrigger onClick={() => setOpen(!open)} className='flex-grow cursor-pointer'>
- <div className='flex items-center h-8 justify-between px-2.5 rounded-lg border-0 bg-gray-100 text-gray-900 text-[13px]'>
- <div className='grow w-0 truncate' title={value.name}>{value.name}</div>
- <RiArrowDownSLine className='shrink-0 w-3.5 h-3.5 text-gray-700' />
- </div>
- </PortalToFollowElemTrigger>
- <PortalToFollowElemContent style={{
- zIndex: 100,
- }}>
- <div className='p-1 bg-white rounded-lg shadow-sm' style={{
- width: 350,
- }}>
- <div
- className='shadow-sm bg-white mb-2 mx-1 flex items-center px-2 rounded-lg bg-gray-100'
- >
- <RiSearchLine className='shrink-0 ml-[1px] mr-[5px] w-3.5 h-3.5 text-gray-400' />
- <input
- value={searchText}
- className='grow px-0.5 py-[7px] text-[13px] text-gray-700 bg-transparent appearance-none outline-none caret-primary-600 placeholder:text-gray-400'
- placeholder={t('workflow.nodes.code.searchDependencies') || ''}
- onChange={e => setSearchText(e.target.value)}
- autoFocus
- />
- {
- searchText && (
- <div
- className='flex items-center justify-center ml-[5px] w-[18px] h-[18px] cursor-pointer'
- onClick={() => setSearchText('')}
- >
- <XCircle className='w-[14px] h-[14px] text-gray-400' />
- </div>
- )
- }
- </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 items-center h-[30px] justify-between pl-3 pr-2 rounded-lg hover:bg-gray-100 text-gray-900 text-[13px] cursor-pointer'
- onClick={handleChange(dependency)}
- >
- <div className='w-0 grow truncate'>{dependency.name}</div>
- {dependency.name === value.name && <Check className='shrink-0 w-4 h-4 text-primary-600' />}
- </div>
- ))}
- </div>
- </div>
- </PortalToFollowElemContent>
- </PortalToFollowElem>
- )
- }
- export default React.memo(DependencyPicker)
|