File size: 3,774 Bytes
4304c6d
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
import type { FC } from 'react'
import React, { useCallback, useState } from 'react'
import { t } from 'i18next'
import type { CodeDependency } from './types'
import { ChevronDown } from '@/app/components/base/icons/src/vender/line/arrows'
import { PortalToFollowElem, PortalToFollowElemContent, PortalToFollowElemTrigger } from '@/app/components/base/portal-to-follow-elem'
import { Check, SearchLg } 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>

          <ChevronDown 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'

          >

            <SearchLg 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)