mirror of
https://github.com/csunny/DB-GPT.git
synced 2025-08-03 09:34:04 +00:00
Co-authored-by: Fangyin Cheng <staneyffer@gmail.com> Co-authored-by: 谨欣 <echo.cmy@antgroup.com> Co-authored-by: 严志勇 <yanzhiyong@tiansuixiansheng.com> Co-authored-by: yanzhiyong <932374019@qq.com>
252 lines
8.7 KiB
TypeScript
252 lines
8.7 KiB
TypeScript
import { addApp, apiInterceptors, getAppList, getTeamMode, updateApp } from '@/client/api';
|
||
import { CreateAppParams, TeamMode } from '@/types/app';
|
||
import { useRequest } from 'ahooks';
|
||
import { App, ConfigProvider, Divider, Form, Input, Modal, Spin } from 'antd';
|
||
import classNames from 'classnames';
|
||
import Image from 'next/image';
|
||
import { useRouter } from 'next/router';
|
||
import React, { useMemo, useState } from 'react';
|
||
import { useTranslation } from 'react-i18next';
|
||
import styles from './styles.module.css';
|
||
|
||
interface WorkModeSelectProps {
|
||
disable: boolean;
|
||
options: TeamMode[];
|
||
value?: TeamMode;
|
||
onChange?: (value: TeamMode) => void;
|
||
}
|
||
|
||
// 自定义team_mode选择
|
||
const WorkModeSelect: React.FC<WorkModeSelectProps> = ({ disable = false, options = [], value, onChange }) => {
|
||
const [selected, setSelected] = useState<TeamMode>(value || ({} as TeamMode));
|
||
const { i18n } = useTranslation();
|
||
|
||
const returnOptionStyle = (item: TeamMode) => {
|
||
if (disable) {
|
||
return classNames(
|
||
`flex items-center p-4 border rounded-lg border-[#d9d9d9] cursor-not-allowed relative transition-all duration-500 ease-in-out`,
|
||
{
|
||
'bg-[rgba(0,0,0,0.04)] dark:bg-[#606264]': item.value === selected?.value,
|
||
},
|
||
);
|
||
}
|
||
return `flex items-center p-4 border dark:border-[rgba(217,217,217,0.85)] rounded-lg cursor-pointer hover:border-[#0c75fc] hover:bg-[#f5faff] dark:hover:border-[rgba(12,117,252,0.85)] dark:hover:bg-[#606264] relative transition-all duration-300 ease-in-out ${
|
||
item.value === selected?.value
|
||
? 'border-[#0c75fc] bg-[#f5faff] dark:bg-[#606264] dark:border-[#0c75fc]'
|
||
: 'border-[#d9d9d9]'
|
||
} `;
|
||
};
|
||
const language = i18n.language === 'en';
|
||
|
||
return (
|
||
<div className='grid grid-cols-2 gap-4'>
|
||
{options.map(item => (
|
||
<div
|
||
className={returnOptionStyle(item)}
|
||
key={item.value}
|
||
onClick={() => {
|
||
if (disable) {
|
||
return;
|
||
}
|
||
setSelected(item);
|
||
onChange?.({ ...value, ...item });
|
||
}}
|
||
>
|
||
<Image src={`/icons/app/${item.value}.png`} width={48} height={48} alt={item.value} />
|
||
<div className='flex flex-col ml-3'>
|
||
<span className='text-xs font-medium text-[rgba(0,0,0,0.85)] dark:text-[rgba(255,255,255,0.85)] first-line:leading-6'>
|
||
{language ? item.name_en : item.name_cn}
|
||
</span>
|
||
<span className='text-xs text-[rgba(0,0,0,0.45)] dark:text-[rgba(255,255,255,0.85)]'>
|
||
{language ? item.description_en : item.description}
|
||
</span>
|
||
</div>
|
||
{item.value === selected?.value && (
|
||
<div
|
||
className='w-3 h-3 rounded-tr-md absolute top-[1px] right-[1px] transition-all duration-300 ease-in-out'
|
||
style={{
|
||
background: `linear-gradient(to right top, transparent 50%, transparent 50%, ${disable ? '#d0d0d0' : '#0c75fc'} 0)`,
|
||
}}
|
||
/>
|
||
)}
|
||
</div>
|
||
))}
|
||
</div>
|
||
);
|
||
};
|
||
|
||
const CreateAppModal: React.FC<{
|
||
open: boolean;
|
||
onCancel: () => void;
|
||
refresh?: any;
|
||
type?: 'add' | 'edit';
|
||
}> = ({ open, onCancel, type = 'add', refresh }) => {
|
||
const { t, i18n } = useTranslation();
|
||
const appInfo = JSON.parse(localStorage.getItem('new_app_info') || '{}');
|
||
const { message } = App.useApp();
|
||
const [form] = Form.useForm();
|
||
const teamMode = Form.useWatch('team_mode', form);
|
||
|
||
const router = useRouter();
|
||
const language = i18n.language === 'en';
|
||
|
||
// 获取工作模式列表
|
||
const { data, loading } = useRequest(async () => {
|
||
const [_, res] = await apiInterceptors(getTeamMode());
|
||
return res ?? [];
|
||
});
|
||
|
||
// 创建应用
|
||
const { run: createApp, loading: createLoading } = useRequest(
|
||
async (params: CreateAppParams) => {
|
||
if (type === 'edit') {
|
||
return await apiInterceptors(
|
||
updateApp({
|
||
app_code: appInfo?.app_code,
|
||
language: 'zh',
|
||
...params,
|
||
}),
|
||
);
|
||
} else {
|
||
return await apiInterceptors(
|
||
addApp({
|
||
language: 'zh',
|
||
...params,
|
||
}),
|
||
);
|
||
}
|
||
},
|
||
{
|
||
manual: true,
|
||
onSuccess: async res => {
|
||
const [error, data] = res;
|
||
if (!error) {
|
||
if (type === 'edit') {
|
||
const [, res] = await apiInterceptors(getAppList({}));
|
||
const curApp = res?.app_list?.find(item => item.app_code === appInfo?.app_code);
|
||
localStorage.setItem('new_app_info', JSON.stringify({ ...curApp, isEdit: true }));
|
||
message.success(t('Update_successfully'));
|
||
} else {
|
||
message.success(t('Create_successfully'));
|
||
localStorage.setItem('new_app_info', JSON.stringify(data));
|
||
router.push(`/construct/app/extra`);
|
||
}
|
||
} else {
|
||
message.error(type === 'edit' ? t('Update_failure') : t('Create_failure'));
|
||
}
|
||
await refresh?.();
|
||
onCancel();
|
||
},
|
||
},
|
||
);
|
||
|
||
const mode = useMemo(() => {
|
||
return data?.filter(item => item.value === appInfo?.team_mode)?.[0];
|
||
}, [appInfo, data]);
|
||
|
||
if (loading) {
|
||
return null;
|
||
}
|
||
|
||
return (
|
||
<ConfigProvider
|
||
theme={{
|
||
components: {
|
||
Button: {
|
||
defaultBorderColor: '#d9d9d9',
|
||
},
|
||
},
|
||
}}
|
||
>
|
||
<Modal
|
||
className={styles['create-app-modal-container']}
|
||
title={t('create_app')}
|
||
width={900}
|
||
open={open}
|
||
onOk={async () => {
|
||
form.validateFields().then(async (values: any) => {
|
||
await createApp({
|
||
app_name: values?.app_name,
|
||
app_describe: values?.app_describe,
|
||
team_mode: values?.team_mode?.value,
|
||
});
|
||
});
|
||
}}
|
||
onCancel={onCancel}
|
||
centered={true}
|
||
>
|
||
<Spin spinning={createLoading}>
|
||
<div className='flex flex-1'>
|
||
<Form
|
||
layout='vertical'
|
||
className='w-3/5'
|
||
form={form}
|
||
initialValues={{
|
||
team_mode: mode || data?.[0],
|
||
app_name: appInfo?.app_name,
|
||
app_describe: appInfo?.app_describe,
|
||
}}
|
||
>
|
||
<Form.Item
|
||
label={t('team_modal')}
|
||
name='team_mode'
|
||
required
|
||
rules={[{ required: true, message: t('Please_input_the_work_modal') }]}
|
||
>
|
||
<WorkModeSelect disable={type === 'edit'} options={data || []} />
|
||
</Form.Item>
|
||
<Form.Item
|
||
label={`${t('app_name')}:`}
|
||
name='app_name'
|
||
required
|
||
rules={[{ required: true, message: t('input_app_name') }]}
|
||
>
|
||
<Input placeholder={t('input_app_name')} autoComplete='off' className='h-8' />
|
||
</Form.Item>
|
||
<Form.Item
|
||
label={`${t('Description')}:`}
|
||
name='app_describe'
|
||
required
|
||
rules={[
|
||
{
|
||
required: true,
|
||
message: t('Please_input_the_description'),
|
||
},
|
||
]}
|
||
>
|
||
<Input.TextArea
|
||
autoComplete='off'
|
||
placeholder={t('Please_input_the_description')}
|
||
autoSize={{ minRows: 2.5 }}
|
||
/>
|
||
</Form.Item>
|
||
{/* <Form.Item label="应用图标:" name="app_icon" valuePropName="fileList">
|
||
<Upload listType="picture-card">
|
||
<button style={{ border: 0, background: 'none' }} type="button">
|
||
<PlusOutlined />
|
||
<div style={{ marginTop: 8 }}>上传图标</div>
|
||
</button>
|
||
</Upload>
|
||
</Form.Item> */}
|
||
</Form>
|
||
<Divider type='vertical' className='h-auto mx-6 bg-[rgba(0,0,0,0.06)] dark:bg-[rgba(255,255,255,0.5)] ' />
|
||
<div className='flex flex-col w-2/5 pl-6 pt-8 '>
|
||
<span className='text-base text-[rgba(0,0,0,0.85)] font-medium mb-6 dark:text-[rgba(255,255,255,0.85)]'>
|
||
{language ? teamMode?.name_en : teamMode?.name_cn}
|
||
</span>
|
||
<div className='flex items-start'>
|
||
<span className='flex flex-shrink-0 w-1 h-1 rounded-full bg-[rgba(0,0,0,0.45)] mt-2 mr-1 dark:bg-[rgba(255,255,255,0.65)]' />
|
||
<span className='text-xs leading-5 text-[rgba(0,0,0,0.45)] dark:text-[rgba(255,255,255,0.65)]'>
|
||
{language ? teamMode?.remark_en : teamMode?.remark}
|
||
</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</Spin>
|
||
</Modal>
|
||
</ConfigProvider>
|
||
);
|
||
};
|
||
|
||
export default CreateAppModal;
|