Chore/remove python dependencies selector (#7494)
parent
715eb8fa32
commit
784b11ce19
@ -1,6 +0,0 @@
|
|||||||
from pydantic import BaseModel
|
|
||||||
|
|
||||||
|
|
||||||
class CodeDependency(BaseModel):
|
|
||||||
name: str
|
|
||||||
version: str
|
|
||||||
@ -1,97 +0,0 @@
|
|||||||
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)
|
|
||||||
@ -1,36 +0,0 @@
|
|||||||
import type { FC } from 'react'
|
|
||||||
import React from 'react'
|
|
||||||
import RemoveButton from '../_base/components/remove-button'
|
|
||||||
import type { CodeDependency } from './types'
|
|
||||||
import DependencyPicker from './dependency-picker'
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
available_dependencies: CodeDependency[]
|
|
||||||
dependencies: CodeDependency[]
|
|
||||||
handleRemove: (index: number) => void
|
|
||||||
handleChange: (index: number, dependency: CodeDependency) => void
|
|
||||||
}
|
|
||||||
|
|
||||||
const Dependencies: FC<Props> = ({
|
|
||||||
available_dependencies, dependencies, handleRemove, handleChange,
|
|
||||||
}) => {
|
|
||||||
return (
|
|
||||||
<div className='space-y-2'>
|
|
||||||
{dependencies.map((dependency, index) => (
|
|
||||||
<div className='flex items-center space-x-1' key={index}>
|
|
||||||
<DependencyPicker
|
|
||||||
value={dependency}
|
|
||||||
available_dependencies={available_dependencies}
|
|
||||||
onChange={dependency => handleChange(index, dependency)}
|
|
||||||
/>
|
|
||||||
<RemoveButton
|
|
||||||
className='!p-2 !bg-gray-100 hover:!bg-gray-200'
|
|
||||||
onClick={() => handleRemove(index)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
export default React.memo(Dependencies)
|
|
||||||
Loading…
Reference in New Issue