Merge pull request #362 from SuanmoSuanyangTechnology/feature/model_v2_zy

feat(web): move create custom model  to model list
This commit is contained in:
yingzhao
2026-02-09 10:03:35 +08:00
committed by GitHub
9 changed files with 70 additions and 48 deletions

View File

@@ -66,9 +66,9 @@ export const addModelPlaza = (model_base_id: string) => {
} }
// Create custom model // Create custom model
export const addCustomModel = (data: CustomModelForm) => { export const addCustomModel = (data: CustomModelForm) => {
return request.post('/models/model_plaza', data) return request.post('/models', data)
} }
// Update custom model // Update custom model
export const updateCustomModel = (model_base_id: string, data: CustomModelForm) => { export const updateCustomModel = (model_base_id: string, data: CustomModelForm) => {
return request.put(`/models/model_plaza/${model_base_id}`, data) return request.put(`/models/${model_base_id}`, data)
} }

View File

@@ -93,7 +93,7 @@ const UploadImages = forwardRef<UploadImagesRef, UploadImagesProps>(({
onChange, onChange,
disabled = false, disabled = false,
fileSize, fileSize,
fileType = ['png', 'jpg', 'gif'], fileType = ['png', 'jpg', 'gif', 'svg'],
isAutoUpload = true, isAutoUpload = true,
maxCount = 1, maxCount = 1,
className = 'rb:size-24! rb:leading-1!', className = 'rb:size-24! rb:leading-1!',

View File

@@ -10,11 +10,11 @@
* Shows model tags and allows viewing model details * Shows model tags and allows viewing model details
*/ */
import { useRef, useState, useEffect, type FC } from 'react'; import { useRef, useState, useEffect, forwardRef, useImperativeHandle } from 'react';
import { Button, Flex, Row, Col } from 'antd' import { Button, Flex, Row, Col } from 'antd'
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import type { ProviderModelItem, KeyConfigModalRef, ModelListDetailRef } from './types' import type { ProviderModelItem, KeyConfigModalRef, ModelListDetailRef, ModelListItem, BaseRef } from './types'
import RbCard from '@/components/RbCard/Card' import RbCard from '@/components/RbCard/Card'
import { getModelNewList } from '@/api/models' import { getModelNewList } from '@/api/models'
import PageEmpty from '@/components/Empty/PageEmpty'; import PageEmpty from '@/components/Empty/PageEmpty';
@@ -26,11 +26,12 @@ import { getLogoUrl } from './utils'
/** /**
* Model list component * Model list component
*/ */
const ModelList: FC<{ query: any }> = ({ query }) => { const ModelList = forwardRef<BaseRef, { query: any; handleEdit: (vo?: ModelListItem) => void; }> (({ query, handleEdit }, ref) => {
const { t } = useTranslation(); const { t } = useTranslation();
const keyConfigModalRef = useRef<KeyConfigModalRef>(null) const keyConfigModalRef = useRef<KeyConfigModalRef>(null)
const modelListDetailRef = useRef<ModelListDetailRef>(null) const modelListDetailRef = useRef<ModelListDetailRef>(null)
const [list, setList] = useState<ProviderModelItem[]>([]) const [list, setList] = useState<ProviderModelItem[]>([])
useEffect(() => { useEffect(() => {
getList() getList()
}, [query]) }, [query])
@@ -54,6 +55,11 @@ const ModelList: FC<{ query: any }> = ({ query }) => {
keyConfigModalRef.current?.handleOpen(vo) keyConfigModalRef.current?.handleOpen(vo)
} }
/** Expose methods to parent component */
useImperativeHandle(ref, () => ({
getList,
modelListDetailRefresh: () => modelListDetailRef.current?.handleRefresh()
}));
return ( return (
<> <>
{list.length === 0 {list.length === 0
@@ -96,9 +102,10 @@ const ModelList: FC<{ query: any }> = ({ query }) => {
<ModelListDetail <ModelListDetail
ref={modelListDetailRef} ref={modelListDetailRef}
refresh={getList} refresh={getList}
handleEdit={handleEdit}
/> />
</> </>
) )
} })
export default ModelList export default ModelList

View File

@@ -26,7 +26,7 @@ import { getLogoUrl } from './utils'
/** /**
* Model square component * Model square component
*/ */
const ModelSquare = forwardRef <BaseRef, { query: any; handleEdit: (vo?: ModelPlazaItem) => void; }>(({ query, handleEdit }, ref) => { const ModelSquare = forwardRef <BaseRef, { query: any; }>(({ query }, ref) => {
const { t } = useTranslation(); const { t } = useTranslation();
const { message } = App.useApp() const { message } = App.useApp()
const modelSquareDetailRef = useRef<ModelSquareDetailRef>(null) const modelSquareDetailRef = useRef<ModelSquareDetailRef>(null)
@@ -96,7 +96,6 @@ const ModelSquare = forwardRef <BaseRef, { query: any; handleEdit: (vo?: ModelPl
<Flex justify="space-between"> <Flex justify="space-between">
<Space size={8}><UsergroupAddOutlined /> {item.add_count}</Space> <Space size={8}><UsergroupAddOutlined /> {item.add_count}</Space>
<Space> <Space>
{!item.is_official && <Button type="primary" disabled={item.is_deprecated} onClick={() => handleEdit(item)}>{t('modelNew.edit')}</Button>}
{item.is_added {item.is_added
? <Button type="primary" disabled>{t('modelNew.added')}</Button> ? <Button type="primary" disabled>{t('modelNew.added')}</Button>
: <Button type="primary" ghost disabled={item.is_deprecated} onClick={() => handleAdd(item)}>{item.is_deprecated ? t('modelNew.deprecated') : `+ ${t('common.add')}`}</Button> : <Button type="primary" ghost disabled={item.is_deprecated} onClick={() => handleAdd(item)}>{item.is_deprecated ? t('modelNew.deprecated') : `+ ${t('common.add')}`}</Button>
@@ -114,7 +113,6 @@ const ModelSquare = forwardRef <BaseRef, { query: any; handleEdit: (vo?: ModelPl
<ModelSquareDetail <ModelSquareDetail
ref={modelSquareDetailRef} ref={modelSquareDetailRef}
refresh={getList} refresh={getList}
handleEdit={handleEdit}
/> />
</> </>
) )

View File

@@ -11,10 +11,10 @@
*/ */
import { forwardRef, useImperativeHandle, useState } from 'react'; import { forwardRef, useImperativeHandle, useState } from 'react';
import { Form, Input, App, Select } from 'antd'; import { Form, Input, App } from 'antd';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import type { CustomModelForm, ModelPlazaItem, CustomModelModalRef, CustomModelModalProps } from '../types'; import type { CustomModelForm, ModelListItem, CustomModelModalRef, CustomModelModalProps } from '../types';
import RbModal from '@/components/RbModal' import RbModal from '@/components/RbModal'
import CustomSelect from '@/components/CustomSelect' import CustomSelect from '@/components/CustomSelect'
import UploadImages from '@/components/Upload/UploadImages' import UploadImages from '@/components/Upload/UploadImages'
@@ -30,22 +30,21 @@ const CustomModelModal = forwardRef<CustomModelModalRef, CustomModelModalProps>(
const { t } = useTranslation(); const { t } = useTranslation();
const { message } = App.useApp(); const { message } = App.useApp();
const [visible, setVisible] = useState(false); const [visible, setVisible] = useState(false);
const [model, setModel] = useState<ModelPlazaItem>({} as ModelPlazaItem); const [model, setModel] = useState<ModelListItem>({} as ModelListItem);
const [isEdit, setIsEdit] = useState(false); const [isEdit, setIsEdit] = useState(false);
const [form] = Form.useForm<CustomModelForm>(); const [form] = Form.useForm<CustomModelForm>();
const [loading, setLoading] = useState(false) const [loading, setLoading] = useState(false)
const formValues = Form.useWatch([], form)
/** Close modal and reset state */ /** Close modal and reset state */
const handleClose = () => { const handleClose = () => {
setModel({} as ModelPlazaItem); setModel({} as ModelListItem);
form.resetFields(); form.resetFields();
setLoading(false) setLoading(false)
setVisible(false); setVisible(false);
}; };
/** Open modal with optional model data for editing */ /** Open modal with optional model data for editing */
const handleOpen = (model?: ModelPlazaItem) => { const handleOpen = (model?: ModelListItem) => {
if (model) { if (model) {
setIsEdit(true); setIsEdit(true);
setModel(model); setModel(model);
@@ -66,7 +65,7 @@ const CustomModelModal = forwardRef<CustomModelModalRef, CustomModelModalProps>(
const res = isEdit ? updateCustomModel(model.id, rest) : addCustomModel(data) const res = isEdit ? updateCustomModel(model.id, rest) : addCustomModel(data)
res.then(() => { res.then(() => {
refresh && refresh() refresh && refresh(isEdit)
handleClose() handleClose()
message.success(isEdit ? t('common.updateSuccess') : t('common.createSuccess')) message.success(isEdit ? t('common.updateSuccess') : t('common.createSuccess'))
}) })
@@ -79,12 +78,10 @@ const CustomModelModal = forwardRef<CustomModelModalRef, CustomModelModalProps>(
form form
.validateFields() .validateFields()
.then((values) => { .then((values) => {
setLoading(true)
const { logo, ...rest } = values; const { logo, ...rest } = values;
let formData: CustomModelForm = { let formData: CustomModelForm = {
...rest ...rest
} }
formData.is_official = false;
if (typeof logo === 'object' && logo?.response?.data.file_id) { if (typeof logo === 'object' && logo?.response?.data.file_id) {
getFileLink(logo?.response?.data.file_id) getFileLink(logo?.response?.data.file_id)
@@ -111,8 +108,6 @@ const CustomModelModal = forwardRef<CustomModelModalRef, CustomModelModalProps>(
handleOpen, handleOpen,
})); }));
console.log('formValues', formValues)
return ( return (
<RbModal <RbModal
title={isEdit ? `${model.name} - ${t('modelNew.modelConfiguration')}` : t('modelNew.createCustomModel')} title={isEdit ? `${model.name} - ${t('modelNew.modelConfiguration')}` : t('modelNew.createCustomModel')}
@@ -174,11 +169,22 @@ const CustomModelModal = forwardRef<CustomModelModalRef, CustomModelModalProps>(
> >
<Input.TextArea placeholder={t('common.pleaseEnter')} /> <Input.TextArea placeholder={t('common.pleaseEnter')} />
</Form.Item> </Form.Item>
<Form.Item <Form.Item
name="tags" name={["api_keys", 0, "api_key"]}
label={t('modelNew.tags')} label={t('modelNew.api_key')}
rules={[{ required: true, message: t('common.inputPlaceholder', { title: t('modelNew.api_key') }) }]}
> >
<Select mode="tags" placeholder={t('common.pleaseEnter')} /> <Input.Password placeholder={t('common.pleaseEnter')} />
</Form.Item>
<Form.Item
name={["api_keys", 0, "api_base"]}
label={t('modelNew.api_base')}
rules={[{ required: true, message: t('common.inputPlaceholder', { title: t('modelNew.api_base') }) }]}
>
<Input placeholder="https://api.example.com/v1" />
</Form.Item> </Form.Item>
</Form> </Form>
</RbModal> </RbModal>

View File

@@ -30,12 +30,13 @@ import CustomSelect from '@/components/CustomSelect'
interface ModelListDetailProps { interface ModelListDetailProps {
/** Callback to refresh parent list */ /** Callback to refresh parent list */
refresh?: () => void; refresh?: () => void;
handleEdit: (vo?: ModelListItem) => void;
} }
/** /**
* Model list detail drawer component * Model list detail drawer component
*/ */
const ModelListDetail = forwardRef<ModelListDetailRef, ModelListDetailProps>(({ refresh }, ref) => { const ModelListDetail = forwardRef<ModelListDetailRef, ModelListDetailProps>(({ refresh, handleEdit }, ref) => {
const { t } = useTranslation(); const { t } = useTranslation();
const [open, setOpen] = useState(false); const [open, setOpen] = useState(false);
const [data, setData] = useState<ProviderModelItem>({} as ProviderModelItem) const [data, setData] = useState<ProviderModelItem>({} as ProviderModelItem)
@@ -95,7 +96,8 @@ const ModelListDetail = forwardRef<ModelListDetailRef, ModelListDetailProps>(({
/** Expose methods to parent component */ /** Expose methods to parent component */
useImperativeHandle(ref, () => ({ useImperativeHandle(ref, () => ({
handleOpen, handleOpen,
handleRefresh,
})); }));
/** Filter models by selected type */ /** Filter models by selected type */
@@ -149,7 +151,10 @@ const ModelListDetail = forwardRef<ModelListDetailRef, ModelListDetailProps>(({
</Tooltip> </Tooltip>
<div className="rb:absolute rb:bottom-4 rb:left-6 rb:right-6"> <div className="rb:absolute rb:bottom-4 rb:left-6 rb:right-6">
<Row gutter={12}> <Row gutter={12}>
<Col span={24}> <Col span={12}>
<Button block onClick={() => handleEdit(item)}>{t('modelNew.modelConfiguration')}</Button>
</Col>
<Col span={12}>
<Button type="primary" ghost block onClick={() => handleKeyConfig(item)}>{t('modelNew.keyConfig')}</Button> <Button type="primary" ghost block onClick={() => handleKeyConfig(item)}>{t('modelNew.keyConfig')}</Button>
</Col> </Col>
</Row> </Row>

View File

@@ -29,14 +29,12 @@ import { getLogoUrl } from '../utils'
interface ModelSquareDetailProps { interface ModelSquareDetailProps {
/** Callback to refresh parent list */ /** Callback to refresh parent list */
refresh: () => void; refresh: () => void;
/** Callback to edit model */
handleEdit: (vo: ModelPlazaItem) => void;
} }
/** /**
* Model square detail drawer component * Model square detail drawer component
*/ */
const ModelSquareDetail = forwardRef<ModelSquareDetailRef, ModelSquareDetailProps>(({ refresh, handleEdit }, ref) => { const ModelSquareDetail = forwardRef<ModelSquareDetailRef, ModelSquareDetailProps>(({ refresh }, ref) => {
const { t } = useTranslation(); const { t } = useTranslation();
const { message } = App.useApp() const { message } = App.useApp()
const [model, setModel] = useState<ModelPlaza>({} as ModelPlaza) const [model, setModel] = useState<ModelPlaza>({} as ModelPlaza)
@@ -112,7 +110,6 @@ const ModelSquareDetail = forwardRef<ModelSquareDetailRef, ModelSquareDetailProp
<Flex justify="space-between"> <Flex justify="space-between">
<Space size={8}><UsergroupAddOutlined /> {item.add_count}</Space> <Space size={8}><UsergroupAddOutlined /> {item.add_count}</Space>
<Space> <Space>
{!item.is_official && <Button type="primary" disabled={item.is_deprecated} onClick={() => handleEdit(item)}>{t('modelNew.edit')}</Button>}
{item.is_added {item.is_added
? <Button type="primary" disabled>{t('modelNew.added')}</Button> ? <Button type="primary" disabled>{t('modelNew.added')}</Button>
: <Button type="primary" ghost disabled={item.is_deprecated} onClick={() => handleAdd(item)}>{item.is_deprecated ? t('modelNew.deprecated') : `+ ${t('common.add')}`}</Button> : <Button type="primary" ghost disabled={item.is_deprecated} onClick={() => handleAdd(item)}>{item.is_deprecated ? t('modelNew.deprecated') : `+ ${t('common.add')}`}</Button>

View File

@@ -15,7 +15,7 @@ import { Button, Flex, Space, type SegmentedProps, Form } from 'antd'
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import GroupModelModal from './components/GroupModelModal' import GroupModelModal from './components/GroupModelModal'
import type { ModelListItem, GroupModelModalRef, CustomModelModalRef, ModelPlazaItem, BaseRef, Query } from './types' import type { ModelListItem, GroupModelModalRef, CustomModelModalRef, BaseRef, Query } from './types'
import SearchInput from '@/components/SearchInput' import SearchInput from '@/components/SearchInput'
import PageTabs from '@/components/PageTabs' import PageTabs from '@/components/PageTabs'
import GroupModel from './Group' import GroupModel from './Group'
@@ -38,7 +38,7 @@ const tabKeys = ['group', 'list', 'square']
const configModalRef = useRef<GroupModelModalRef>(null) const configModalRef = useRef<GroupModelModalRef>(null)
const customModelModalRef = useRef<CustomModelModalRef>(null) const customModelModalRef = useRef<CustomModelModalRef>(null)
const groupRef = useRef<BaseRef>(null) const groupRef = useRef<BaseRef>(null)
const squareRef = useRef<BaseRef>(null) const modelListRef = useRef<BaseRef>(null)
const [form] = Form.useForm<Query>() const [form] = Form.useForm<Query>()
const query = Form.useWatch([], form) const query = Form.useWatch([], form)
@@ -56,24 +56,29 @@ const tabKeys = ['group', 'list', 'square']
} }
/** Open edit modal based on active tab */ /** Open edit modal based on active tab */
const handleEdit = (vo?: ModelListItem | ModelPlazaItem) => { const handleEdit = (vo?: ModelListItem | ModelListItem) => {
switch(activeTab) { switch(activeTab) {
case 'group': case 'group':
configModalRef?.current?.handleOpen(vo as ModelListItem) configModalRef?.current?.handleOpen(vo as ModelListItem)
break break
case 'square': case 'list':
customModelModalRef?.current?.handleOpen(vo as ModelPlazaItem) customModelModalRef?.current?.handleOpen(vo as ModelListItem)
break break
} }
} }
/** Refresh list based on active tab */ /** Refresh list based on active tab */
const handleRefresh = () => { const handleRefresh = (isEdit?: boolean) => {
switch (activeTab) { switch (activeTab) {
case 'group': case 'group':
groupRef.current?.getList() groupRef.current?.getList()
break break
case 'square': case 'list':
squareRef.current?.getList() console.log('isEdit', isEdit)
if (isEdit) {
modelListRef.current?.modelListDetailRefresh?.()
} else {
modelListRef.current?.getList()
}
break break
} }
} }
@@ -122,15 +127,15 @@ const tabKeys = ['group', 'list', 'square']
</Form.Item> </Form.Item>
} }
{activeTab === 'group' && <Button type="primary" onClick={() => handleEdit()}>+ {t('modelNew.createGroupModel')}</Button>} {activeTab === 'group' && <Button type="primary" onClick={() => handleEdit()}>+ {t('modelNew.createGroupModel')}</Button>}
{activeTab === 'square' && <Button type="primary" onClick={() => handleEdit()}>+ {t('modelNew.createCustomModel')}</Button>} {activeTab === 'list' && <Button type="primary" onClick={() => handleEdit()}>+ {t('modelNew.createCustomModel')}</Button>}
</Space> </Space>
</Form> </Form>
</Flex> </Flex>
<div className="rb:w-full rb:h-[calc(100%-48px)] rb:my-4"> <div className="rb:w-full rb:h-[calc(100%-48px)] rb:my-4">
{activeTab === 'group' && <GroupModel ref={groupRef} query={query} handleEdit={handleEdit} />} {activeTab === 'group' && <GroupModel ref={groupRef} query={query} handleEdit={handleEdit} />}
{activeTab === 'list' && <ModelList query={query} />} {activeTab === 'list' && <ModelList ref={modelListRef} query={query} handleEdit={handleEdit} />}
{activeTab === 'square' && <ModelSquare ref={squareRef} query={query} handleEdit={handleEdit} />} {activeTab === 'square' && <ModelSquare query={query} />}
</div> </div>
<GroupModelModal <GroupModelModal
ref={configModalRef} ref={configModalRef}

View File

@@ -80,6 +80,7 @@ export interface GroupModelModalProps {
export interface ModelListDetailRef { export interface ModelListDetailRef {
/** Open detail drawer with provider model data */ /** Open detail drawer with provider model data */
handleOpen: (vo: ProviderModelItem) => void; handleOpen: (vo: ProviderModelItem) => void;
handleRefresh: () => void;
} }
/** /**
@@ -284,10 +285,12 @@ export interface CustomModelForm {
logo?: any; logo?: any;
/** Model description */ /** Model description */
description: string; description: string;
/** Whether model is official */ api_keys: Array<{
is_official: boolean; /** API key value */
/** Model tags */ api_key: string;
tags: string[]; /** API base URL */
api_base: string;
}>
} }
/** /**
@@ -295,7 +298,7 @@ export interface CustomModelForm {
*/ */
export interface CustomModelModalRef { export interface CustomModelModalRef {
/** Open modal with optional model plaza item */ /** Open modal with optional model plaza item */
handleOpen: (vo?: ModelPlazaItem) => void; handleOpen: (vo?: ModelListItem) => void;
} }
/** /**
@@ -303,7 +306,7 @@ export interface CustomModelModalRef {
*/ */
export interface CustomModelModalProps { export interface CustomModelModalProps {
/** Callback to refresh model list */ /** Callback to refresh model list */
refresh?: () => void; refresh?: (flag?: boolean) => void;
} }
/** /**
@@ -312,4 +315,5 @@ export interface CustomModelModalProps {
export interface BaseRef { export interface BaseRef {
/** Refresh list data */ /** Refresh list data */
getList: () => void; getList: () => void;
modelListDetailRefresh?: () => void;
} }