fix: show uninstalled plugin nodes in workflow checklist (#29630)

This commit is contained in:
yyh 2025-12-15 10:11:23 +08:00 committed by GitHub
parent b8d54d745e
commit 59137f1d05
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
24 changed files with 1315 additions and 1270 deletions

View File

@ -37,9 +37,13 @@ import useNodes from '@/app/components/workflow/store/workflow/use-nodes'
type WorkflowChecklistProps = { type WorkflowChecklistProps = {
disabled: boolean disabled: boolean
showGoTo?: boolean
onItemClick?: (item: ChecklistItem) => void
} }
const WorkflowChecklist = ({ const WorkflowChecklist = ({
disabled, disabled,
showGoTo = true,
onItemClick,
}: WorkflowChecklistProps) => { }: WorkflowChecklistProps) => {
const { t } = useTranslation() const { t } = useTranslation()
const [open, setOpen] = useState(false) const [open, setOpen] = useState(false)
@ -49,9 +53,13 @@ const WorkflowChecklist = ({
const { handleNodeSelect } = useNodesInteractions() const { handleNodeSelect } = useNodesInteractions()
const handleChecklistItemClick = (item: ChecklistItem) => { const handleChecklistItemClick = (item: ChecklistItem) => {
if (!item.canNavigate) const goToEnabled = showGoTo && item.canNavigate && !item.disableGoTo
if (!goToEnabled)
return return
handleNodeSelect(item.id) if (onItemClick)
onItemClick(item)
else
handleNodeSelect(item.id)
setOpen(false) setOpen(false)
} }
@ -116,7 +124,7 @@ const WorkflowChecklist = ({
key={node.id} key={node.id}
className={cn( className={cn(
'group mb-2 rounded-lg border-[0.5px] border-components-panel-border bg-components-panel-bg shadow-xs last-of-type:mb-0', 'group mb-2 rounded-lg border-[0.5px] border-components-panel-border bg-components-panel-bg shadow-xs last-of-type:mb-0',
node.canNavigate ? 'cursor-pointer' : 'cursor-default opacity-80', showGoTo && node.canNavigate && !node.disableGoTo ? 'cursor-pointer' : 'cursor-default opacity-80',
)} )}
onClick={() => handleChecklistItemClick(node)} onClick={() => handleChecklistItemClick(node)}
> >
@ -130,7 +138,7 @@ const WorkflowChecklist = ({
{node.title} {node.title}
</span> </span>
{ {
node.canNavigate && ( (showGoTo && node.canNavigate && !node.disableGoTo) && (
<div className='flex h-4 w-[60px] shrink-0 items-center justify-center gap-1 opacity-0 transition-opacity duration-200 group-hover:opacity-100'> <div className='flex h-4 w-[60px] shrink-0 items-center justify-center gap-1 opacity-0 transition-opacity duration-200 group-hover:opacity-100'>
<span className='whitespace-nowrap text-xs font-medium leading-4 text-primary-600'> <span className='whitespace-nowrap text-xs font-medium leading-4 text-primary-600'>
{t('workflow.panel.goTo')} {t('workflow.panel.goTo')}

View File

@ -66,6 +66,7 @@ export type ChecklistItem = {
unConnected?: boolean unConnected?: boolean
errorMessage?: string errorMessage?: string
canNavigate: boolean canNavigate: boolean
disableGoTo?: boolean
} }
const START_NODE_TYPES: BlockEnum[] = [ const START_NODE_TYPES: BlockEnum[] = [
@ -75,6 +76,13 @@ const START_NODE_TYPES: BlockEnum[] = [
BlockEnum.TriggerPlugin, BlockEnum.TriggerPlugin,
] ]
// Node types that depend on plugins
const PLUGIN_DEPENDENT_TYPES: BlockEnum[] = [
BlockEnum.Tool,
BlockEnum.DataSource,
BlockEnum.TriggerPlugin,
]
export const useChecklist = (nodes: Node[], edges: Edge[]) => { export const useChecklist = (nodes: Node[], edges: Edge[]) => {
const { t } = useTranslation() const { t } = useTranslation()
const language = useGetLanguage() const language = useGetLanguage()
@ -157,7 +165,14 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => {
if (node.type === CUSTOM_NODE) { if (node.type === CUSTOM_NODE) {
const checkData = getCheckData(node.data) const checkData = getCheckData(node.data)
const validator = nodesExtraData?.[node.data.type as BlockEnum]?.checkValid const validator = nodesExtraData?.[node.data.type as BlockEnum]?.checkValid
let errorMessage = validator ? validator(checkData, t, moreDataForCheckValid).errorMessage : undefined const isPluginMissing = PLUGIN_DEPENDENT_TYPES.includes(node.data.type as BlockEnum) && node.data._pluginInstallLocked
// Check if plugin is installed for plugin-dependent nodes first
let errorMessage: string | undefined
if (isPluginMissing)
errorMessage = t('workflow.nodes.common.pluginNotInstalled')
else if (validator)
errorMessage = validator(checkData, t, moreDataForCheckValid).errorMessage
if (!errorMessage) { if (!errorMessage) {
const availableVars = map[node.id].availableVars const availableVars = map[node.id].availableVars
@ -194,7 +209,8 @@ export const useChecklist = (nodes: Node[], edges: Edge[]) => {
toolIcon, toolIcon,
unConnected: isUnconnected && !canSkipConnectionCheck, unConnected: isUnconnected && !canSkipConnectionCheck,
errorMessage, errorMessage,
canNavigate: true, canNavigate: !isPluginMissing,
disableGoTo: isPluginMissing,
}) })
} }
} }

File diff suppressed because it is too large Load Diff

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'Verwende die Variable', variable: 'Verwende die Variable',
}, },
inputVars: 'Eingabevariablen', inputVars: 'Eingabevariablen',
pluginNotInstalled: 'Plugin ist nicht installiert',
}, },
start: { start: {
required: 'erforderlich', required: 'erforderlich',

View File

@ -464,6 +464,7 @@ const translation = {
variable: 'Use variable', variable: 'Use variable',
}, },
inputVars: 'Input Variables', inputVars: 'Input Variables',
pluginNotInstalled: 'Plugin is not installed',
}, },
start: { start: {
required: 'required', required: 'required',

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'Usa la variable', variable: 'Usa la variable',
}, },
inputVars: 'Variables de entrada', inputVars: 'Variables de entrada',
pluginNotInstalled: 'El complemento no está instalado',
}, },
start: { start: {
required: 'requerido', required: 'requerido',

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'از متغیر استفاده کن', variable: 'از متغیر استفاده کن',
}, },
inputVars: 'متغیرهای ورودی', inputVars: 'متغیرهای ورودی',
pluginNotInstalled: 'افزونه نصب نشده است',
}, },
start: { start: {
required: 'الزامی', required: 'الزامی',

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'Utilisez une variable', variable: 'Utilisez une variable',
}, },
inputVars: 'Variables dentrée', inputVars: 'Variables dentrée',
pluginNotInstalled: 'Le plugin n\'est pas installé',
}, },
start: { start: {
required: 'requis', required: 'requis',

View File

@ -449,6 +449,7 @@ const translation = {
variable: 'चर का प्रयोग करें', variable: 'चर का प्रयोग करें',
}, },
inputVars: 'इनपुट चर', inputVars: 'इनपुट चर',
pluginNotInstalled: 'प्लगइन इंस्टॉल नहीं है',
}, },
start: { start: {
required: 'आवश्यक', required: 'आवश्यक',

View File

@ -444,6 +444,7 @@ const translation = {
insertVarTip: 'Sisipkan Variabel', insertVarTip: 'Sisipkan Variabel',
outputVars: 'Variabel Keluaran', outputVars: 'Variabel Keluaran',
inputVars: 'Variabel Masukan', inputVars: 'Variabel Masukan',
pluginNotInstalled: 'Plugin tidak terpasang',
}, },
start: { start: {
outputVars: { outputVars: {

View File

@ -452,6 +452,7 @@ const translation = {
variable: 'Usa la variabile', variable: 'Usa la variabile',
}, },
inputVars: 'Variabili di input', inputVars: 'Variabili di input',
pluginNotInstalled: 'Il plugin non è installato',
}, },
start: { start: {
required: 'richiesto', required: 'richiesto',

View File

@ -464,6 +464,7 @@ const translation = {
variable: '変数を使用する', variable: '変数を使用する',
}, },
inputVars: '入力変数', inputVars: '入力変数',
pluginNotInstalled: 'プラグインがインストールされていません',
}, },
start: { start: {
required: '必須', required: '必須',

View File

@ -461,6 +461,7 @@ const translation = {
variable: '변수를 사용하세요', variable: '변수를 사용하세요',
}, },
inputVars: '입력 변수', inputVars: '입력 변수',
pluginNotInstalled: '플러그인이 설치되지 않았습니다',
}, },
start: { start: {
required: '필수', required: '필수',

View File

@ -437,6 +437,7 @@ const translation = {
input: 'Wartość wejściowa', input: 'Wartość wejściowa',
}, },
inputVars: 'Zmienne wejściowe', inputVars: 'Zmienne wejściowe',
pluginNotInstalled: 'Wtyczka nie jest zainstalowana',
}, },
start: { start: {
required: 'wymagane', required: 'wymagane',

View File

@ -437,6 +437,7 @@ const translation = {
input: 'Valor de entrada', input: 'Valor de entrada',
}, },
inputVars: 'Variáveis de entrada', inputVars: 'Variáveis de entrada',
pluginNotInstalled: 'O plugin não está instalado',
}, },
start: { start: {
required: 'requerido', required: 'requerido',

View File

@ -437,6 +437,7 @@ const translation = {
input: 'Valoare de intrare', input: 'Valoare de intrare',
}, },
inputVars: 'Variabile de intrare', inputVars: 'Variabile de intrare',
pluginNotInstalled: 'Pluginul nu este instalat',
}, },
start: { start: {
required: 'necesar', required: 'necesar',

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'Используйте переменную', variable: 'Используйте переменную',
}, },
inputVars: 'Входные переменные', inputVars: 'Входные переменные',
pluginNotInstalled: 'Плагин не установлен',
}, },
start: { start: {
required: 'обязательно', required: 'обязательно',

View File

@ -444,6 +444,7 @@ const translation = {
input: 'Vhodna vrednost', input: 'Vhodna vrednost',
}, },
inputVars: 'Vhodne spremenljivke', inputVars: 'Vhodne spremenljivke',
pluginNotInstalled: 'Vtičnik ni nameščen',
}, },
start: { start: {
outputVars: { outputVars: {

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'ใช้ตัวแปร', variable: 'ใช้ตัวแปร',
}, },
inputVars: 'ตัวแปรอินพุต', inputVars: 'ตัวแปรอินพุต',
pluginNotInstalled: 'ปลั๊กอินไม่ได้ติดตั้ง',
}, },
start: { start: {
required: 'ต้องระบุ', required: 'ต้องระบุ',

View File

@ -437,6 +437,7 @@ const translation = {
input: 'Girdi değeri', input: 'Girdi değeri',
}, },
inputVars: 'Giriş Değişkenleri', inputVars: 'Giriş Değişkenleri',
pluginNotInstalled: 'Eklenti yüklü değil',
}, },
start: { start: {
required: 'gerekli', required: 'gerekli',

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'Використовуйте змінну', variable: 'Використовуйте змінну',
}, },
inputVars: 'Вхідні змінні', inputVars: 'Вхідні змінні',
pluginNotInstalled: 'Плагін не встановлений',
}, },
start: { start: {
required: 'обов\'язковий', required: 'обов\'язковий',

View File

@ -437,6 +437,7 @@ const translation = {
variable: 'Sử dụng biến', variable: 'Sử dụng biến',
}, },
inputVars: 'Biến đầu vào', inputVars: 'Biến đầu vào',
pluginNotInstalled: 'Plugin chưa được cài đặt',
}, },
start: { start: {
required: 'bắt buộc', required: 'bắt buộc',

View File

@ -464,6 +464,7 @@ const translation = {
variable: '使用变量', variable: '使用变量',
}, },
inputVars: '输入变量', inputVars: '输入变量',
pluginNotInstalled: '插件未安装',
}, },
start: { start: {
required: '必填', required: '必填',

View File

@ -442,6 +442,7 @@ const translation = {
variable: '使用變數', variable: '使用變數',
}, },
inputVars: '輸入變數', inputVars: '輸入變數',
pluginNotInstalled: '插件未安裝',
}, },
start: { start: {
required: '必填', required: '必填',