* feat: Add portal management related icons * feat: Add portal configuration pages and related translations * feat: Add new gateway configuration components and icons - Introduced `ConfigButtons` component for save and share actions with new SVG icons. - Added `CopyrightTable` and `HomeTable` components for managing copyright and home settings. - Implemented `SectionHeader` for consistent section titles in the gateway configuration. - Updated `FillRowTabs` to support new tabs for home and copyright configurations. - Modified translations for gateway-related terms in English, Simplified Chinese, and Traditional Chinese. - Removed unused gateway tab from `AccountContainer`. * feat(gate): add API and schema for team gate configurations - Introduced new TypeScript definitions for gate configuration parameters and data structures. - Created constants for gate status and tools. - Implemented MongoDB schema for team gate configurations. - Added API functions for getting, creating, updating, and deleting team gate configurations and logos. - Developed ShareGateModal component for sharing portal links and custom domains. - Updated ConfigButtons component to handle saving configurations and opening the share modal. - Added new icons for gate functionalities. - Updated English and Chinese translations for gateway-related texts. * feat(gate): refactor gate configuration API and remove unused logo handling * feat(gate): enhance team gate configuration with new error handling and chat features - Added new error codes to CommonErrEnum for method not allowed, system error, and unauthorized access. - Updated datasetErr to include corresponding error messages for new error codes. - Refactored API to support updating team gate configurations and copyright information. - Introduced ChatInputBox component for chat functionalities, including file and image uploads. - Enhanced HomeTable and CopyrightTable components to manage settings more effectively. - Updated translations for new terms in English and Chinese. - Improved layout and user experience in the gateway configuration pages. * feat: Refactor gateway configuration and chat components - Replaced direct API calls with Zustand store for gate configuration management. - Introduced `useGateStore` for managing gate and copyright configurations. - Updated `GatewayConfig` component to utilize the new store and remove redundant state management. - Enhanced chat functionality in `application.tsx` and `index.tsx` to support gate model. - Created new `application.tsx` for handling chat interactions with the gate application. - Improved error handling and loading states in chat components. - Added dynamic imports for better performance and code splitting. * feat(gate): update GateSideBar to conditionally render recent apps based on chat page state * fix(HomeTable): comment out unused FormControl for better readability * feat(gate): enhance copyright configuration and file upload functionality - Updated ConfigButtons to handle team avatar updates and save copyright configurations. - Refactored CopyrightTable to integrate file selection for team avatars and improve form handling. - Added animations and hover effects for better user experience during file uploads. - Improved toast notifications for success and error handling in configuration processes. * feat(gate): add gate service availability check and update translations - Implemented gate service availability check in application and index pages, redirecting users if the service is unavailable. - Added new translation keys for gate service status in English and Chinese. - Refactored GateSideBar to improve rendering logic for recent apps based on gate status. * feat(chat): add route check to ToolMenu for app detail visibility - Implemented a check to prevent displaying app details when the current route starts with '/chat/gate'. - Updated menu rendering logic to conditionally show app details based on the new route check. * feat(constants): add 'gate' type to AppTypeEnum * refactor: rename "Portal" to "Gate" across the application - Updated schema to remove the slogan field from GateConfigSchema. - Modified SVG icon dimensions for gateLight.svg. - Changed localization keys and values from "Portal" to "Gate" in various JSON files. - Added support for gate applications in the app creation and management logic. - Enhanced ChatBox component to handle gate-specific routes and configurations. - Updated ConfigButtons to manage gate configurations and intros. - Adjusted ShareGateModal to generate correct gate URLs. - Expanded emptyTemplates to include gate-specific templates and configurations. - Refactored chatItemContext to include intro for gate applications. - Updated useGateStore to initialize gate configurations with intros from existing gate applications. * fix: add isResponseDetail prop to ChatItemContextProvider * feat: refactor gate-related API and components for improved functionality * feat: 添加工具选择和工具选择模态框组件 * refactor: Update GateConfig related types, remove unnecessary constants and enums * feat: Enhance Gate configuration components and API integration - Updated ConfigButtons and HomeTable to use string arrays for tools instead of GateTool type. - Implemented batch plugin loading in HomeTable with error handling. - Added ToolSelect and ToolSelectModal components for improved tool management. - Introduced AppCard and ChatTest components for app detail editing. - Enhanced Edit and EditForm components for better app configuration management. - Added new API endpoint for batch plugin retrieval. - Improved overall structure and styling for better user experience. * fix: Update ChatBoxDataType to make intro optional in chatItemContext.tsx * fix: Add isResponseDetail prop to ChatItemContextProvider in ChatPage component * feat: Enhance ToolSelectModal and GatewayConfig with new functionalities - Updated ToolSelectModal to handle tool selection and configuration, integrating new props for selected tools and chat configuration. - Implemented loading and error handling for Gate applications in GatewayConfig, including a retry mechanism for fetching apps. - Added selectedTool parameter to chat completions API to enable tool activation during chat. - Refactored chat component to support app form context and debug mode for testing. - Enhanced useGateStore to manage gate applications, including loading and updating functionalities. * feat: Refactor GateSideBar to enhance recent apps display and add resource selection * refactor: 移除门户删除确认功能 * feat: 更新 Chat 组件以使用 AppContextProvider 并修正 localAppDetail 的类型 * refactor: Remove the tool menu logic in the GateChatInput component to simplify the code structure * refactor: Remove the tool menu logic from the GateChatInput component to simplify the code structure * feat: Simplify the ShareGateModal component by removing unused states and logic * fix: Update chatGray.svg to remove fill attributes for paths, improving SVG structure * feat: Added new chat icons and updated internationalized text to support new chat features * feat: Refactor chat components and introduce GateChat functionality - Updated ChatHistorySlider to remove isGateRoute check for PC view. - Added new GateChatHistorySlider component for handling chat history in gate context. - Removed obsolete ChatPage component related to gate chat. - Modified GateSideBar styles for improved UI consistency. - Implemented new API endpoint for chat gate functionality. - Refactored chat gate index page to utilize GateChatHistorySlider and streamline chat initialization. - Cleaned up unused imports and code related to debugging and legacy chat handling. * feat: Update GateSideBar styles for improved responsiveness and animation - Adjusted width and padding for collapsed and expanded states. - Enhanced transition effects for smoother UI interactions. - Modified alignment and positioning of navigation items and user profile for better layout consistency. - Improved accessibility by ensuring elements are centered when collapsed. * feat: 添加新的聊天图标和更新分享门户组件样式以提升用户体验 * Refactor chat gate components and implement sidebar functionality - Updated ChatGate component to use ChatItemContextProvider and ChatRecordContextProvider for better context management. - Introduced FoldButton component for sidebar collapsing functionality. - Created GateNavBar component to replace GateSideBar for improved navigation. - Refactored GateSideBar to handle folding state and external triggers. - Updated application and index pages to integrate new components and manage sidebar state. - Enhanced useChatGate hook to include appDetail.intro. * feat: Updated team structure, set default banner image and refactored LogoBox component to support diagonal background * feat: Enhance GateNavBar with user popover functionality and logout feature - Added user popover for displaying user information and logout option. - Implemented mouse enter/leave handlers for popover visibility. - Updated user profile section to include popover and improved layout. - Modified index page to include 'account' in server-side props for better context management. * feat: Add a bottom line statement in the ChatBox component to remind users that the content is generated by third-party AI * feat: Update placeholder text in ChatBox and GateChatInput components for better user guidance - Added internationalized placeholder text for user input in both English and Chinese. - Updated ChatBox and GateChatInput components to utilize the new placeholder text from localization files. * feat: Add upload icon and enhance ChatBox layout for better user experience - Introduced a new upload icon in the Icon component for improved visual representation. - Updated ChatBox layout to enhance responsiveness and user interaction, including adjustments to padding and structure. - Added hover overlay effect for logo upload areas in the CopyrightTable component to improve user guidance. * feat: Refactor Chat component to integrate GateSideBar and GateChatHistorySlider for improved layout and functionality * refactor: Update imports to use 'import type' for type-only imports across multiple files - Changed standard imports to type imports for better clarity and performance in TypeScript. - Updated files in the global support, service, and app components to reflect this change. * feat: Update localization strings and improve toast messages for better user feedback - Added new success and failure messages for create, delete, save, and update actions in English and Chinese localization files. - Refactored toast message keys in the ConfigButtons, CopyrightTable, HomeTable, ToolSelect, and other components to use updated localization keys for consistency. - Enhanced user experience by providing clearer feedback on actions performed within the application. * feat: Implement tag management functionality with CRUD operations - Added new Tag schema and controller for managing application tags. - Implemented API endpoints for creating, updating, deleting, and listing tags. - Enhanced the App schema to include a reference to tags. - Updated localization files for new tag-related messages. - Improved user experience by providing clear feedback on tag operations. * feat: Enhance ChatWelcome and GateNavBar components with conditional rendering for team avatars - Updated ChatWelcome and GateNavBar components to conditionally render avatars based on availability. - Improved layout by using Flex components for better alignment and responsiveness. - Ensured consistent styling and structure for avatar display across both components. * fix: Update parameter name in getBatchPlugins API for consistency - Changed parameter name from 'id' to 'appId' in getChildAppPreviewNode function call for better clarity and consistency with the rest of the codebase. * feat: Enhance ToolSelectModal with gate plugins integration and improved filtering - Added useEffect to load plugins from gateStore and set them in state. - Introduced ExtendedNodeTemplateItemType to include cost-related properties. - Updated filtering logic for plugins based on search input. - Refactored RenderList to display plugins with cost information and improved layout. * refactor: Update ToolSelect and ToolSelectModal components for improved UI and state management - Replaced Button with Flex component in ToolSelect for better styling and hover effects. - Adjusted layout and styling in ToolSelect for a more responsive design. - Removed ExtendedNodeTemplateItemType and reverted to NodeTemplateListItemType in ToolSelectModal for simplified state management. - Updated RenderList to reflect changes in template type and maintain consistency. * refactor: Replace Flex with Button for add tool action and enhance loading state UI * feat: Enhance application tag management and localization support - Added 'tags' property to AppListItemType for better tag management. - Updated localization files for English and Chinese to include new tag-related strings. - Implemented new AppTable component in the gateway for managing applications. - Adjusted routes and components to support the new app management features. * feat: Update localization and refactor chat components - Added new localization strings for "enlarge" in English, Simplified Chinese, and Traditional Chinese. - Refactored chat components to replace `quoteData` with `datasetCiteData` for improved state management. - Enhanced `ToolSelect` and related components by removing error handling logic for a cleaner UI. - Updated `AppTable` component to remove unnecessary props for better clarity. * feat: Initialize copyright configuration in GateNavBar component * feat: Add appDetail property to ChatGate component and update related logic * feat: Update GateNavBar routing logic for chat page refresh and enhance avatar display * feat: Enhance tag management and app detail handling in Chat component * feat: 更新聊天组件中的国际化文本和输入逻辑,优化用户体验 * feat: Refactor gate configuration management - Updated API endpoints for fetching and updating gate configurations. - Changed `avatar` field to `logo` and added `banner` in gate configuration types. - Implemented new controller methods for creating, retrieving, updating, and deleting gate configurations. - Enhanced `ConfigButtons` and `CopyrightTable` components to handle new configuration fields. - Added new SVG icon for sidebar collapse button. - Improved internationalization support by adding new translation keys. - Refactored `HomeTable` to manage gate configuration state and handle updates. - Updated `ShareGateModal` to accept gate configuration as props. - Cleaned up unused imports and optimized component structures. * feat: 加载和管理 Gate 配置及版权信息,优化相关组件逻辑 * feat: 更新国际化文本,优化聊天组件中的配置和状态检查逻辑 * feat: Update template configuration and adjust default open state to improve user experience * feat: Enhance gate management features and update related components - Added `featuredApps` and `quickApps` fields to `GateSchemaType` for better app management. - Implemented new methods for updating and managing featured and quick apps in the `controller` and `featureApp` modules. - Introduced `AddFeatureAppModal` for selecting and adding featured apps. - Updated `AppTable` and `HomeTable` components to integrate new app management functionalities. - Enhanced internationalization support by adding new translation keys for app management features. - Refactored existing components to improve code clarity and maintainability. * feat: Enhance chat tool selection and quick app management features - Added `selectedToolIds` and `onSelectedToolIdsChange` props to `ChatBox` and `GateChatInput` components for better tool management. - Introduced `GateToolSelect` component for selecting tools with improved UI and functionality. - Implemented `AddQuickAppModal` for managing quick apps, including selection and drag-and-drop functionality. - Updated `HomeTable` to integrate quick app management and display selected apps. - Refactored related components to improve code clarity and maintainability. * refactor: Remove unused AppContext import in useChatGate.tsx to clean up code * refactor: Update plugin ID handling and clean up unused imports - Renamed `splitCombinePluginId` to `splitCombineToolId` for consistency in plugin ID processing. - Removed unused `checkNode` import from `featureApp/detail.ts` and `quickApp/detail.ts` files to streamline the code. - Added `ownerTmbId` to the parameters in `rewriteAppWorkflowToDetail` for better context management. * refactor: Rename storeEdgesRenderEdge to storeEdge2RenderEdge for consistency - Updated the function name from `storeEdgesRenderEdge` to `storeEdge2RenderEdge` in the Header component to maintain naming consistency. - Adjusted the mapping of edges to use the new function name for improved clarity in the workflow processing.
329 lines
10 KiB
TypeScript
329 lines
10 KiB
TypeScript
import { Box, Button, Flex, Grid, useDisclosure, Text } from '@chakra-ui/react';
|
|
import React, { useMemo, useState, useCallback, useEffect, useRef } from 'react';
|
|
import MyIcon from '@fastgpt/web/components/common/Icon';
|
|
import { useTranslation } from 'next-i18next';
|
|
import { SmallAddIcon } from '@chakra-ui/icons';
|
|
import type { AppSimpleEditFormType } from '@fastgpt/global/core/app/type';
|
|
import MyTooltip from '@fastgpt/web/components/common/MyTooltip';
|
|
import { theme } from '@fastgpt/web/styles/theme';
|
|
import {
|
|
FlowNodeInputTypeEnum,
|
|
FlowNodeTypeEnum
|
|
} from '@fastgpt/global/core/workflow/node/constant';
|
|
import Avatar from '@fastgpt/web/components/common/Avatar';
|
|
import { keyframes } from '@emotion/react';
|
|
import QuestionTip from '@fastgpt/web/components/common/MyTooltip/QuestionTip';
|
|
|
|
import { getWebLLMModel } from '@/web/common/system/utils';
|
|
import ToolSelectModal, {
|
|
childAppSystemKey
|
|
} from '@/pageComponents/app/detail/Gate/components/ToolSelectModal';
|
|
import ConfigToolModal from '@/pageComponents/app/detail/Gate/components/ConfigToolModal';
|
|
|
|
// 定义粉碎动画关键帧
|
|
const shatterKeyframes = keyframes`
|
|
0% {
|
|
opacity: 1;
|
|
transform: scale(1);
|
|
filter: blur(0);
|
|
}
|
|
50% {
|
|
opacity: 0.5;
|
|
transform: scale(0.7) rotate(5deg) translateY(10px);
|
|
filter: blur(2px);
|
|
}
|
|
100% {
|
|
opacity: 0;
|
|
transform: scale(0.2) rotate(-5deg) translateY(15px);
|
|
filter: blur(4px);
|
|
}
|
|
`;
|
|
|
|
// 定义淡入动画关键帧
|
|
const fadeInKeyframes = keyframes`
|
|
0% {
|
|
opacity: 0;
|
|
}
|
|
100% {
|
|
opacity: 1;
|
|
}
|
|
`;
|
|
|
|
// 样式常量
|
|
const spacing = {
|
|
xs: 2
|
|
};
|
|
|
|
const formStyles = {
|
|
fontWeight: 500,
|
|
fontSize: '14px',
|
|
lineHeight: '20px',
|
|
letterSpacing: '0.1px'
|
|
};
|
|
|
|
const ToolSelect = ({
|
|
appForm,
|
|
setAppForm
|
|
}: {
|
|
appForm: AppSimpleEditFormType;
|
|
setAppForm: (newAppForm: AppSimpleEditFormType) => void;
|
|
}) => {
|
|
const { t } = useTranslation();
|
|
|
|
const [configTool, setConfigTool] = useState<
|
|
AppSimpleEditFormType['selectedTools'][number] | null
|
|
>(null);
|
|
|
|
// 添加删除状态管理
|
|
const [deletingToolIds, setDeletingToolIds] = useState<Set<string>>(new Set());
|
|
|
|
const {
|
|
isOpen: isOpenToolsSelect,
|
|
onOpen: onOpenToolsSelect,
|
|
onClose: onCloseToolsSelect
|
|
} = useDisclosure();
|
|
const selectedModel = getWebLLMModel(appForm.aiSettings.model);
|
|
|
|
// 使用 useCallback 缓存删除函数
|
|
const handleDeleteTool = useCallback(
|
|
(toolId: string) => {
|
|
// 先设置删除标记,触发动画
|
|
setDeletingToolIds((prev) => new Set([...prev, toolId]));
|
|
|
|
// 设置延时,等待动画完成后再从数组中移除
|
|
setTimeout(() => {
|
|
const newAppForm = {
|
|
...appForm,
|
|
selectedTools: appForm.selectedTools.filter((tool) => tool.id !== toolId)
|
|
};
|
|
setAppForm(newAppForm);
|
|
|
|
// 清除删除标记
|
|
setDeletingToolIds((prev) => {
|
|
const newSet = new Set(prev);
|
|
newSet.delete(toolId);
|
|
return newSet;
|
|
});
|
|
}, 150); // 动画持续时间缩短到150ms
|
|
},
|
|
[appForm, setAppForm]
|
|
);
|
|
|
|
return (
|
|
<>
|
|
{/* 标题区域 */}
|
|
<Flex alignItems="center" justifyContent="space-between" width="100%">
|
|
<Flex alignItems="center" gap={spacing.xs}>
|
|
<Text
|
|
ml={2}
|
|
fontWeight={formStyles.fontWeight}
|
|
fontSize={formStyles.fontSize}
|
|
lineHeight={formStyles.lineHeight}
|
|
letterSpacing={formStyles.letterSpacing}
|
|
color="myGray.700"
|
|
>
|
|
{t('common:core.app.Tool call')}
|
|
</Text>
|
|
<QuestionTip ml={1} label={t('app:plugin_dispatch_tip')} />
|
|
</Flex>
|
|
|
|
{/* 已有工具时显示新增按钮 */}
|
|
{appForm.selectedTools.length > 0 && (
|
|
<Button
|
|
size="sm"
|
|
colorScheme="primary"
|
|
variant="outline"
|
|
leftIcon={<SmallAddIcon />}
|
|
onClick={onOpenToolsSelect}
|
|
_hover={{ bg: 'blue.50' }}
|
|
>
|
|
{t('common:Add')}
|
|
</Button>
|
|
)}
|
|
</Flex>
|
|
|
|
{/* 工具容器 */}
|
|
{appForm.selectedTools.length > 0 ? (
|
|
<Box mt={2}>
|
|
<Grid gridTemplateColumns={'repeat(3, minmax(0, 1fr))'} gridGap={[2, 4]}>
|
|
{appForm.selectedTools.map((item) => {
|
|
const isDeleting = deletingToolIds.has(item.id);
|
|
|
|
return (
|
|
<MyTooltip key={item.id} label={item.intro}>
|
|
<Flex
|
|
overflow={'hidden'}
|
|
display={'flex'}
|
|
height={'40px'}
|
|
padding={'8px 12px'}
|
|
flexDirection={'row'}
|
|
justifyContent={'flex-start'}
|
|
alignItems={'center'}
|
|
flex={'1 0 0'}
|
|
borderRadius={'6px'}
|
|
border={'0.5px solid var(--Gray-Modern-200, #E8EBF0)'}
|
|
background={'#FFF'}
|
|
boxShadow={
|
|
'0px 4px 4px 0px rgba(19, 51, 107, 0.05), 0px 0px 1px 0px rgba(19, 51, 107, 0.08)'
|
|
}
|
|
_hover={{
|
|
transform: 'translateY(-2px)',
|
|
borderRadius: '6px',
|
|
border: '0.5px solid var(--Gray-Modern-200, #E8EBF0)',
|
|
background: '#FFF',
|
|
boxShadow:
|
|
'0px 4px 4px 0px rgba(19, 51, 107, 0.05), 0px 0px 1px 0px rgba(19, 51, 107, 0.08)'
|
|
}}
|
|
cursor={'pointer'}
|
|
transition="all 0.2s ease"
|
|
position="relative"
|
|
role="group"
|
|
animation={isDeleting ? `${shatterKeyframes} 0.15s ease forwards` : undefined}
|
|
onClick={() => {
|
|
if (
|
|
item.inputs
|
|
.filter((input) => !childAppSystemKey.includes(input.key))
|
|
.every(
|
|
(input) =>
|
|
input.toolDescription ||
|
|
input.renderTypeList.includes(FlowNodeInputTypeEnum.selectLLMModel) ||
|
|
input.renderTypeList.includes(FlowNodeInputTypeEnum.fileSelect)
|
|
) ||
|
|
item.flowNodeType === FlowNodeTypeEnum.tool ||
|
|
item.flowNodeType === FlowNodeTypeEnum.toolSet
|
|
) {
|
|
return;
|
|
}
|
|
setConfigTool(item);
|
|
}}
|
|
>
|
|
<Flex alignItems="center" width="100%">
|
|
<Avatar src={item.avatar} borderRadius={'6px'} w={'20px'} h={'20px'} />
|
|
<Box
|
|
ml={'6px'}
|
|
className={'textEllipsis'}
|
|
fontSize={'sm'}
|
|
fontWeight="medium"
|
|
color={'myGray.900'}
|
|
flex="1"
|
|
>
|
|
{item.name}
|
|
</Box>
|
|
<Flex
|
|
className="delete"
|
|
alignItems="center"
|
|
justifyContent="center"
|
|
ml={2}
|
|
w="22px"
|
|
h="22px"
|
|
borderRadius="sm"
|
|
cursor="pointer"
|
|
transition="all 0.2s"
|
|
_hover={{
|
|
background: 'rgba(17, 24, 36, 0.05)',
|
|
color: 'red.600'
|
|
}}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
handleDeleteTool(item.id);
|
|
}}
|
|
opacity="0"
|
|
_groupHover={{
|
|
opacity: 1,
|
|
animation: `${fadeInKeyframes} 0.2s ease`
|
|
}}
|
|
>
|
|
<MyIcon
|
|
className="delete"
|
|
name={'delete' as any}
|
|
w={'16px'}
|
|
h={'16px'}
|
|
color={'inherit'}
|
|
/>
|
|
</Flex>
|
|
</Flex>
|
|
</Flex>
|
|
</MyTooltip>
|
|
);
|
|
})}
|
|
</Grid>
|
|
</Box>
|
|
) : (
|
|
<Box
|
|
mt={2}
|
|
display="flex"
|
|
width="100%"
|
|
height="80px"
|
|
justifyContent="center"
|
|
alignItems="center"
|
|
borderRadius="4px"
|
|
border="1px dashed var(--Gray-Modern-250, #DFE2EA)"
|
|
cursor="pointer"
|
|
onClick={onOpenToolsSelect}
|
|
_hover={{
|
|
borderColor: 'primary.300',
|
|
bg: 'gray.100',
|
|
'.hoverContent': { color: 'primary.500' }
|
|
}}
|
|
transition="all 0.2s"
|
|
position="relative"
|
|
>
|
|
<Flex
|
|
className="hoverContent"
|
|
alignItems="center"
|
|
justifyContent="center"
|
|
flexDirection="row"
|
|
gap={'6px'}
|
|
color="gray.500"
|
|
>
|
|
<SmallAddIcon boxSize={5} />
|
|
<Box fontSize="sm" fontWeight="medium">
|
|
{t('common:Choose')}
|
|
</Box>
|
|
</Flex>
|
|
</Box>
|
|
)}
|
|
|
|
{isOpenToolsSelect && (
|
|
<ToolSelectModal
|
|
selectedTools={appForm.selectedTools}
|
|
chatConfig={appForm.chatConfig}
|
|
selectedModel={selectedModel}
|
|
onAddTool={(e) => {
|
|
const newAppForm = {
|
|
...appForm,
|
|
selectedTools: [...appForm.selectedTools, e]
|
|
};
|
|
setAppForm(newAppForm);
|
|
}}
|
|
onRemoveTool={(e) => {
|
|
const newAppForm = {
|
|
...appForm,
|
|
selectedTools: appForm.selectedTools.filter((item) => item.pluginId !== e.id)
|
|
};
|
|
setAppForm(newAppForm);
|
|
}}
|
|
onClose={onCloseToolsSelect}
|
|
/>
|
|
)}
|
|
{configTool && (
|
|
<ConfigToolModal
|
|
configTool={configTool}
|
|
onCloseConfigTool={() => setConfigTool(null)}
|
|
onAddTool={(e) => {
|
|
const newAppForm = {
|
|
...appForm,
|
|
selectedTools: appForm.selectedTools.map((item) =>
|
|
item.pluginId === configTool.pluginId ? e : item
|
|
)
|
|
};
|
|
setAppForm(newAppForm);
|
|
}}
|
|
/>
|
|
)}
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default React.memo(ToolSelect);
|