dify/web/app/components/plugins/install-plugin/install-bundle/steps/install.tsx

89 lines
3.4 KiB
TypeScript
Raw Normal View History

2024-11-12 17:32:39 +08:00
'use client'
import type { FC } from 'react'
2024-11-14 16:54:23 +08:00
import React, { useCallback } from 'react'
import type { Dependency, Plugin } from '../../../types'
2024-11-12 17:32:39 +08:00
import Button from '@/app/components/base/button'
import { RiLoader2Line } from '@remixicon/react'
import { useTranslation } from 'react-i18next'
2024-11-14 16:54:23 +08:00
import InstallByDSLList from './install-by-dsl-list'
2024-11-15 12:07:20 +08:00
import { useInstallFromMarketplaceAndGitHub } from '@/service/use-plugins'
2024-11-12 17:32:39 +08:00
const i18nPrefix = 'plugin.installModal'
type Props = {
2024-11-14 16:54:23 +08:00
fromDSLPayload: Dependency[]
2024-11-12 17:32:39 +08:00
onCancel: () => void
}
const Install: FC<Props> = ({
2024-11-14 16:54:23 +08:00
fromDSLPayload,
2024-11-12 17:32:39 +08:00
onCancel,
}) => {
const { t } = useTranslation()
2024-11-14 16:54:23 +08:00
const [selectedPlugins, setSelectedPlugins] = React.useState<Plugin[]>([])
2024-11-15 12:07:20 +08:00
const [selectedIndexes, setSelectedIndexes] = React.useState<number[]>([])
2024-11-12 17:32:39 +08:00
const selectedPluginsNum = selectedPlugins.length
2024-11-14 16:54:23 +08:00
2024-11-15 12:07:20 +08:00
const handleSelect = (plugin: Plugin, selectedIndex: number) => {
2024-11-14 18:31:10 +08:00
const isSelected = !!selectedPlugins.find(p => p.plugin_id === plugin.plugin_id)
let nextSelectedPlugins
if (isSelected)
nextSelectedPlugins = selectedPlugins.filter(p => p.plugin_id !== plugin.plugin_id)
else
nextSelectedPlugins = [...selectedPlugins, plugin]
setSelectedPlugins(nextSelectedPlugins)
2024-11-15 12:07:20 +08:00
const nextSelectedIndexes = isSelected ? selectedIndexes.filter(i => i !== selectedIndex) : [...selectedIndexes, selectedIndex]
setSelectedIndexes(nextSelectedIndexes)
2024-11-12 17:32:39 +08:00
}
2024-11-14 16:54:23 +08:00
const [canInstall, setCanInstall] = React.useState(false)
const handleLoadedAllPlugin = useCallback(() => {
setCanInstall(true)
2024-11-15 12:07:20 +08:00
}, [selectedPlugins, selectedIndexes])
2024-11-12 17:32:39 +08:00
2024-11-15 12:07:20 +08:00
// Install from marketplace and github
const { mutate: installFromMarketplaceAndGitHub, isPending: isInstalling } = useInstallFromMarketplaceAndGitHub({
onSuccess: () => {
console.log('success!')
},
})
console.log(canInstall, !isInstalling, selectedPlugins.length === 0)
const handleInstall = () => {
installFromMarketplaceAndGitHub(fromDSLPayload.filter((_d, index) => selectedIndexes.includes(index)))
2024-11-12 17:32:39 +08:00
}
return (
<>
<div className='flex flex-col px-6 py-3 justify-center items-start gap-4 self-stretch'>
<div className='text-text-secondary system-md-regular'>
<p>{t(`${i18nPrefix}.${selectedPluginsNum > 1 ? 'readyToInstallPackages' : 'readyToInstallPackage'}`, { num: selectedPluginsNum })}</p>
</div>
<div className='w-full p-2 rounded-2xl bg-background-section-burn space-y-1'>
2024-11-14 16:54:23 +08:00
<InstallByDSLList
fromDSLPayload={fromDSLPayload}
selectedPlugins={selectedPlugins}
2024-11-15 12:07:20 +08:00
onSelect={handleSelect}
2024-11-14 16:54:23 +08:00
onLoadedAllPlugin={handleLoadedAllPlugin}
/>
2024-11-12 17:32:39 +08:00
</div>
</div>
{/* Action Buttons */}
<div className='flex p-6 pt-5 justify-end items-center gap-2 self-stretch'>
2024-11-14 16:54:23 +08:00
{!canInstall && (
2024-11-12 17:32:39 +08:00
<Button variant='secondary' className='min-w-[72px]' onClick={onCancel}>
{t('common.operation.cancel')}
</Button>
)}
<Button
variant='primary'
className='min-w-[72px] flex space-x-0.5'
2024-11-15 12:07:20 +08:00
disabled={!canInstall || isInstalling || selectedPlugins.length === 0}
2024-11-12 17:32:39 +08:00
onClick={handleInstall}
>
2024-11-15 12:07:20 +08:00
{isInstalling && <RiLoader2Line className='w-4 h-4 animate-spin-slow' />}
<span>{t(`${i18nPrefix}.${isInstalling ? 'installing' : 'install'}`)}</span>
2024-11-12 17:32:39 +08:00
</Button>
</div>
</>
)
}
export default React.memo(Install)