dependency-picker.tsx 3.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. import type { FC } from 'react'
  2. import React, { useCallback, useState } from 'react'
  3. import { t } from 'i18next'
  4. import {
  5. RiArrowDownSLine,
  6. RiSearchLine,
  7. } from '@remixicon/react'
  8. import type { CodeDependency } from './types'
  9. import { PortalToFollowElem, PortalToFollowElemContent, PortalToFollowElemTrigger } from '@/app/components/base/portal-to-follow-elem'
  10. import { Check } from '@/app/components/base/icons/src/vender/line/general'
  11. import { XCircle } from '@/app/components/base/icons/src/vender/solid/general'
  12. type Props = {
  13. value: CodeDependency
  14. available_dependencies: CodeDependency[]
  15. onChange: (dependency: CodeDependency) => void
  16. }
  17. const DependencyPicker: FC<Props> = ({
  18. available_dependencies,
  19. value,
  20. onChange,
  21. }) => {
  22. const [open, setOpen] = useState(false)
  23. const [searchText, setSearchText] = useState('')
  24. const handleChange = useCallback((dependency: CodeDependency) => {
  25. return () => {
  26. setOpen(false)
  27. onChange(dependency)
  28. }
  29. }, [onChange])
  30. return (
  31. <PortalToFollowElem
  32. open={open}
  33. onOpenChange={setOpen}
  34. placement='bottom-start'
  35. offset={4}
  36. >
  37. <PortalToFollowElemTrigger onClick={() => setOpen(!open)} className='flex-grow cursor-pointer'>
  38. <div className='flex items-center h-8 justify-between px-2.5 rounded-lg border-0 bg-gray-100 text-gray-900 text-[13px]'>
  39. <div className='grow w-0 truncate' title={value.name}>{value.name}</div>
  40. <RiArrowDownSLine className='shrink-0 w-3.5 h-3.5 text-gray-700' />
  41. </div>
  42. </PortalToFollowElemTrigger>
  43. <PortalToFollowElemContent style={{
  44. zIndex: 100,
  45. }}>
  46. <div className='p-1 bg-white rounded-lg shadow-sm' style={{
  47. width: 350,
  48. }}>
  49. <div
  50. className='shadow-sm bg-white mb-2 mx-1 flex items-center px-2 rounded-lg bg-gray-100'
  51. >
  52. <RiSearchLine className='shrink-0 ml-[1px] mr-[5px] w-3.5 h-3.5 text-gray-400' />
  53. <input
  54. value={searchText}
  55. 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'
  56. placeholder={t('workflow.nodes.code.searchDependencies') || ''}
  57. onChange={e => setSearchText(e.target.value)}
  58. autoFocus
  59. />
  60. {
  61. searchText && (
  62. <div
  63. className='flex items-center justify-center ml-[5px] w-[18px] h-[18px] cursor-pointer'
  64. onClick={() => setSearchText('')}
  65. >
  66. <XCircle className='w-[14px] h-[14px] text-gray-400' />
  67. </div>
  68. )
  69. }
  70. </div>
  71. <div className='max-h-[30vh] overflow-y-auto'>
  72. {available_dependencies.filter((v) => {
  73. if (!searchText)
  74. return true
  75. return v.name.toLowerCase().includes(searchText.toLowerCase())
  76. }).map(dependency => (
  77. <div
  78. key={dependency.name}
  79. 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'
  80. onClick={handleChange(dependency)}
  81. >
  82. <div className='w-0 grow truncate'>{dependency.name}</div>
  83. {dependency.name === value.name && <Check className='shrink-0 w-4 h-4 text-primary-600' />}
  84. </div>
  85. ))}
  86. </div>
  87. </div>
  88. </PortalToFollowElemContent>
  89. </PortalToFollowElem>
  90. )
  91. }
  92. export default React.memo(DependencyPicker)