feat(ai): 添加 AI 绘图和思维导图功能
- 新增 AI 绘图管理页面,包括绘画列表、搜索筛选和操作功能 - 实现 AI 思维导图生成功能,支持流式生成和已有内容生成 - 添加 AI 音乐和写作相关的 API 接口 - 更新常量文件,增加 AI 平台、图像生成状态等枚举 - 优化 AI 绘图和思维导图的组件结构,提高可维护性pull/145/head
							parent
							
								
									3ef362508a
								
							
						
					
					
						commit
						1b236e89bf
					
				|  | @ -0,0 +1,112 @@ | ||||||
|  | import type { PageParam, PageResult } from '@vben/request'; | ||||||
|  | 
 | ||||||
|  | import { requestClient } from '#/api/request'; | ||||||
|  | 
 | ||||||
|  | export namespace AiImageApi { | ||||||
|  |   export interface ImageMidjourneyButtonsVO { | ||||||
|  |     customId: string; // MJ::JOB::upsample::1::85a4b4c1-8835-46c5-a15c-aea34fad1862 动作标识
 | ||||||
|  |     emoji: string; // 图标 emoji
 | ||||||
|  |     label: string; // Make Variations 文本
 | ||||||
|  |     style: number; // 样式: 2(Primary)、3(Green)
 | ||||||
|  |   } | ||||||
|  |   // AI 绘图 VO
 | ||||||
|  |   export interface ImageVO { | ||||||
|  |     id: number; // 编号
 | ||||||
|  |     platform: string; // 平台
 | ||||||
|  |     model: string; // 模型
 | ||||||
|  |     prompt: string; // 提示词
 | ||||||
|  |     width: number; // 图片宽度
 | ||||||
|  |     height: number; // 图片高度
 | ||||||
|  |     status: number; // 状态
 | ||||||
|  |     publicStatus: boolean; // 公开状态
 | ||||||
|  |     picUrl: string; // 任务地址
 | ||||||
|  |     errorMessage: string; // 错误信息
 | ||||||
|  |     options: any; // 配置 Map<string, string>
 | ||||||
|  |     taskId: number; // 任务编号
 | ||||||
|  |     buttons: ImageMidjourneyButtonsVO[]; // mj 操作按钮
 | ||||||
|  |     createTime: Date; // 创建时间
 | ||||||
|  |     finishTime: Date; // 完成时间
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   export interface ImageDrawReqVO { | ||||||
|  |     prompt: string; // 提示词
 | ||||||
|  |     modelId: number; // 模型
 | ||||||
|  |     style: string; // 图像生成的风格
 | ||||||
|  |     width: string; // 图片宽度
 | ||||||
|  |     height: string; // 图片高度
 | ||||||
|  |     options: object; // 绘制参数,Map<String, String>
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   export interface ImageMidjourneyImagineReqVO { | ||||||
|  |     prompt: string; // 提示词
 | ||||||
|  |     modelId: number; // 模型
 | ||||||
|  |     base64Array: string[]; // size不能为空
 | ||||||
|  |     width: string; // 图片宽度
 | ||||||
|  |     height: string; // 图片高度
 | ||||||
|  |     version: string; // 版本
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   export interface ImageMidjourneyActionVO { | ||||||
|  |     id: number; // 图片编号
 | ||||||
|  |     customId: string; // MJ::JOB::upsample::1::85a4b4c1-8835-46c5-a15c-aea34fad1862 动作标识
 | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 获取【我的】绘图分页
 | ||||||
|  | export function getImagePageMy(params: PageParam) { | ||||||
|  |   return requestClient.get<PageResult<AiImageApi.ImageVO>>( | ||||||
|  |     '/ai/image/my-page', | ||||||
|  |     { params }, | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 获取【我的】绘图记录
 | ||||||
|  | export function getImageMy(id: number) { | ||||||
|  |   return requestClient.get<AiImageApi.ImageVO[]>(`/ai/image/get-my?id=${id}`); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 获取【我的】绘图记录列表
 | ||||||
|  | export function getImageListMyByIds(ids: number[]) { | ||||||
|  |   return requestClient.get<AiImageApi.ImageVO[]>(`/ai/image/my-list-by-ids`, { | ||||||
|  |     params: { ids: ids.join(',') }, | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 生成图片
 | ||||||
|  | export function drawImage(data: AiImageApi.ImageDrawReqVO) { | ||||||
|  |   return requestClient.post(`/ai/image/draw`, data); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 删除【我的】绘画记录
 | ||||||
|  | export function deleteImageMy(id: number) { | ||||||
|  |   return requestClient.delete(`/ai/image/delete-my?id=${id}`); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ================ midjourney 专属 ================
 | ||||||
|  | // 【Midjourney】生成图片
 | ||||||
|  | export function midjourneyImagine( | ||||||
|  |   data: AiImageApi.ImageMidjourneyImagineReqVO, | ||||||
|  | ) { | ||||||
|  |   return requestClient.post(`/ai/image/midjourney/imagine`, data); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 【Midjourney】Action 操作(二次生成图片)
 | ||||||
|  | export function midjourneyAction(data: AiImageApi.ImageMidjourneyActionVO) { | ||||||
|  |   return requestClient.post(`/ai/image/midjourney/action`, data); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ================ 绘图管理 ================
 | ||||||
|  | // 查询绘画分页
 | ||||||
|  | export function getImagePage(params: any) { | ||||||
|  |   return requestClient.get<AiImageApi.ImageVO[]>(`/ai/image/page`, { params }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 更新绘画发布状态
 | ||||||
|  | export function updateImage(data: any) { | ||||||
|  |   return requestClient.put(`/ai/image/update`, data); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 删除绘画
 | ||||||
|  | export function deleteImage(id: number) { | ||||||
|  |   return requestClient.delete(`/ai/image/delete?id=${id}`); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,65 @@ | ||||||
|  | import { fetchEventSource } from '@vben/request'; | ||||||
|  | import { useAccessStore } from '@vben/stores'; | ||||||
|  | 
 | ||||||
|  | import { requestClient } from '#/api/request'; | ||||||
|  | 
 | ||||||
|  | const accessStore = useAccessStore(); | ||||||
|  | export namespace AiMindmapApi { | ||||||
|  |   // AI 思维导图 VO
 | ||||||
|  |   export interface MindMapVO { | ||||||
|  |     id: number; // 编号
 | ||||||
|  |     userId: number; // 用户编号
 | ||||||
|  |     prompt: string; // 生成内容提示
 | ||||||
|  |     generatedContent: string; // 生成的思维导图内容
 | ||||||
|  |     platform: string; // 平台
 | ||||||
|  |     model: string; // 模型
 | ||||||
|  |     errorMessage: string; // 错误信息
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // AI 思维导图生成 VO
 | ||||||
|  |   export interface AiMindMapGenerateReqVO { | ||||||
|  |     prompt: string; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function generateMindMap({ | ||||||
|  |   data, | ||||||
|  |   onClose, | ||||||
|  |   onMessage, | ||||||
|  |   onError, | ||||||
|  |   ctrl, | ||||||
|  | }: { | ||||||
|  |   ctrl: AbortController; | ||||||
|  |   data: AiMindmapApi.AiMindMapGenerateReqVO; | ||||||
|  |   onClose?: (...args: any[]) => void; | ||||||
|  |   onError?: (...args: any[]) => void; | ||||||
|  |   onMessage?: (res: any) => void; | ||||||
|  | }) { | ||||||
|  |   const token = accessStore.accessToken; | ||||||
|  |   return fetchEventSource( | ||||||
|  |     `${import.meta.env.VITE_BASE_URL}/ai/mind-map/generate-stream`, | ||||||
|  |     { | ||||||
|  |       method: 'post', | ||||||
|  |       headers: { | ||||||
|  |         'Content-Type': 'application/json', | ||||||
|  |         Authorization: `Bearer ${token}`, | ||||||
|  |       }, | ||||||
|  |       openWhenHidden: true, | ||||||
|  |       body: JSON.stringify(data), | ||||||
|  |       onmessage: onMessage, | ||||||
|  |       onerror: onError, | ||||||
|  |       onclose: onClose, | ||||||
|  |       signal: ctrl.signal, | ||||||
|  |     }, | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 查询思维导图分页
 | ||||||
|  | export function getMindMapPage(params: any) { | ||||||
|  |   return requestClient.get(`/ai/mind-map/page`, { params }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 删除思维导图
 | ||||||
|  | export function deleteMindMap(id: number) { | ||||||
|  |   return requestClient.delete(`/ai/mind-map/delete?id=${id}`); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,44 @@ | ||||||
|  | import type { PageParam, PageResult } from '@vben/request'; | ||||||
|  | 
 | ||||||
|  | import { requestClient } from '#/api/request'; | ||||||
|  | 
 | ||||||
|  | export namespace AiMusicApi { | ||||||
|  |   // AI 音乐 VO
 | ||||||
|  |   export interface MusicVO { | ||||||
|  |     id: number; // 编号
 | ||||||
|  |     userId: number; // 用户编号
 | ||||||
|  |     title: string; // 音乐名称
 | ||||||
|  |     lyric: string; // 歌词
 | ||||||
|  |     imageUrl: string; // 图片地址
 | ||||||
|  |     audioUrl: string; // 音频地址
 | ||||||
|  |     videoUrl: string; // 视频地址
 | ||||||
|  |     status: number; // 音乐状态
 | ||||||
|  |     gptDescriptionPrompt: string; // 描述词
 | ||||||
|  |     prompt: string; // 提示词
 | ||||||
|  |     platform: string; // 模型平台
 | ||||||
|  |     model: string; // 模型
 | ||||||
|  |     generateMode: number; // 生成模式
 | ||||||
|  |     tags: string; // 音乐风格标签
 | ||||||
|  |     duration: number; // 音乐时长
 | ||||||
|  |     publicStatus: boolean; // 是否发布
 | ||||||
|  |     taskId: string; // 任务id
 | ||||||
|  |     errorMessage: string; // 错误信息
 | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 查询音乐分页
 | ||||||
|  | export function getMusicPage(params: PageParam) { | ||||||
|  |   return requestClient.get<PageResult<AiMusicApi.MusicVO>>(`/ai/music/page`, { | ||||||
|  |     params, | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 更新音乐
 | ||||||
|  | export function updateMusic(data: any) { | ||||||
|  |   return requestClient.put('/ai/music/update', data); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 删除音乐
 | ||||||
|  | export function deleteMusic(id: number) { | ||||||
|  |   return requestClient.delete(`/ai/music/delete?id=${id}`); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,90 @@ | ||||||
|  | import type { PageParam, PageResult } from '@vben/request'; | ||||||
|  | 
 | ||||||
|  | import type { AiWriteTypeEnum } from '#/utils/constants'; | ||||||
|  | 
 | ||||||
|  | import { fetchEventSource } from '@vben/request'; | ||||||
|  | import { useAccessStore } from '@vben/stores'; | ||||||
|  | 
 | ||||||
|  | import { requestClient } from '#/api/request'; | ||||||
|  | 
 | ||||||
|  | const accessStore = useAccessStore(); | ||||||
|  | export namespace AiWriteApi { | ||||||
|  |   export interface WriteVO { | ||||||
|  |     type: AiWriteTypeEnum.REPLY | AiWriteTypeEnum.WRITING; // 1:撰写 2:回复
 | ||||||
|  |     prompt: string; // 写作内容提示 1。撰写 2回复
 | ||||||
|  |     originalContent: string; // 原文
 | ||||||
|  |     length: number; // 长度
 | ||||||
|  |     format: number; // 格式
 | ||||||
|  |     tone: number; // 语气
 | ||||||
|  |     language: number; // 语言
 | ||||||
|  |     userId?: number; // 用户编号
 | ||||||
|  |     platform?: string; // 平台
 | ||||||
|  |     model?: string; // 模型
 | ||||||
|  |     generatedContent?: string; // 生成的内容
 | ||||||
|  |     errorMessage?: string; // 错误信息
 | ||||||
|  |     createTime?: Date; // 创建时间
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   export interface AiWritePageReqVO extends PageParam { | ||||||
|  |     userId?: number; // 用户编号
 | ||||||
|  |     type?: AiWriteTypeEnum; //  写作类型
 | ||||||
|  |     platform?: string; // 平台
 | ||||||
|  |     createTime?: [string, string]; // 创建时间
 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   export interface AiWriteRespVo { | ||||||
|  |     id: number; | ||||||
|  |     userId: number; | ||||||
|  |     type: number; | ||||||
|  |     platform: string; | ||||||
|  |     model: string; | ||||||
|  |     prompt: string; | ||||||
|  |     generatedContent: string; | ||||||
|  |     originalContent: string; | ||||||
|  |     length: number; | ||||||
|  |     format: number; | ||||||
|  |     tone: number; | ||||||
|  |     language: number; | ||||||
|  |     errorMessage: string; | ||||||
|  |     createTime: string; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function writeStream( | ||||||
|  |   data: any, | ||||||
|  |   onClose: any, | ||||||
|  |   onMessage: any, | ||||||
|  |   onError: any, | ||||||
|  |   ctrl: any, | ||||||
|  | ) { | ||||||
|  |   const token = accessStore.accessToken; | ||||||
|  |   return fetchEventSource( | ||||||
|  |     `${import.meta.env.VITE_BASE_URL}/ai/write/generate-stream`, | ||||||
|  |     { | ||||||
|  |       method: 'post', | ||||||
|  |       headers: { | ||||||
|  |         'Content-Type': 'application/json', | ||||||
|  |         Authorization: `Bearer ${token}`, | ||||||
|  |       }, | ||||||
|  |       openWhenHidden: true, | ||||||
|  |       body: JSON.stringify(data), | ||||||
|  |       onmessage: onMessage, | ||||||
|  |       onerror: onError, | ||||||
|  |       onclose: onClose, | ||||||
|  |       signal: ctrl.signal, | ||||||
|  |     }, | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 获取写作列表
 | ||||||
|  | export function getWritePage(params: any) { | ||||||
|  |   return requestClient.get<PageResult<AiWriteApi.AiWritePageReqVO>>( | ||||||
|  |     `/ai/write/page`, | ||||||
|  |     { params }, | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 删除音乐
 | ||||||
|  | export function deleteWrite(id: number) { | ||||||
|  |   return requestClient.delete(`/ai/write/delete`, { params: { id } }); | ||||||
|  | } | ||||||
|  | @ -5,6 +5,23 @@ | ||||||
|  * 枚举类 |  * 枚举类 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * AI 平台的枚举 | ||||||
|  |  */ | ||||||
|  | export const AiPlatformEnum = { | ||||||
|  |   TONG_YI: 'TongYi', // 阿里
 | ||||||
|  |   YI_YAN: 'YiYan', // 百度
 | ||||||
|  |   DEEP_SEEK: 'DeepSeek', // DeepSeek
 | ||||||
|  |   ZHI_PU: 'ZhiPu', // 智谱 AI
 | ||||||
|  |   XING_HUO: 'XingHuo', // 讯飞
 | ||||||
|  |   SiliconFlow: 'SiliconFlow', // 硅基流动
 | ||||||
|  |   OPENAI: 'OpenAI', | ||||||
|  |   Ollama: 'Ollama', | ||||||
|  |   STABLE_DIFFUSION: 'StableDiffusion', // Stability AI
 | ||||||
|  |   MIDJOURNEY: 'Midjourney', // Midjourney
 | ||||||
|  |   SUNO: 'Suno', // Suno AI
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| export const AiModelTypeEnum = { | export const AiModelTypeEnum = { | ||||||
|   CHAT: 1, // 聊天
 |   CHAT: 1, // 聊天
 | ||||||
|   IMAGE: 2, // 图像
 |   IMAGE: 2, // 图像
 | ||||||
|  | @ -13,6 +30,31 @@ export const AiModelTypeEnum = { | ||||||
|   EMBEDDING: 5, // 向量
 |   EMBEDDING: 5, // 向量
 | ||||||
|   RERANK: 6, // 重排
 |   RERANK: 6, // 重排
 | ||||||
| }; | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * AI 图像生成状态的枚举 | ||||||
|  |  */ | ||||||
|  | export const AiImageStatusEnum = { | ||||||
|  |   IN_PROGRESS: 10, // 进行中
 | ||||||
|  |   SUCCESS: 20, // 已完成
 | ||||||
|  |   FAIL: 30, // 已失败
 | ||||||
|  | }; | ||||||
|  | /** | ||||||
|  |  * AI 音乐生成状态的枚举 | ||||||
|  |  */ | ||||||
|  | export const AiMusicStatusEnum = { | ||||||
|  |   IN_PROGRESS: 10, // 进行中
 | ||||||
|  |   SUCCESS: 20, // 已完成
 | ||||||
|  |   FAIL: 30, // 已失败
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * AI 写作类型的枚举 | ||||||
|  |  */ | ||||||
|  | export enum AiWriteTypeEnum { | ||||||
|  |   WRITING = 1, // 撰写
 | ||||||
|  |   REPLY, // 回复
 | ||||||
|  | } | ||||||
| // ========== COMMON 模块 ==========
 | // ========== COMMON 模块 ==========
 | ||||||
| // 全局通用状态枚举
 | // 全局通用状态枚举
 | ||||||
| export const CommonStatusEnum = { | export const CommonStatusEnum = { | ||||||
|  | @ -733,3 +775,82 @@ OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.DELEGATE, '委派'); | ||||||
| OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.ADD_SIGN, '加签'); | OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.ADD_SIGN, '加签'); | ||||||
| OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.RETURN, '退回'); | OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.RETURN, '退回'); | ||||||
| OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.COPY, '抄送'); | OPERATION_BUTTON_NAME.set(BpmTaskOperationButtonTypeEnum.COPY, '抄送'); | ||||||
|  | 
 | ||||||
|  | // ========== 【写作 UI】相关的枚举 ==========
 | ||||||
|  | 
 | ||||||
|  | /** 写作点击示例时的数据 */ | ||||||
|  | export const WriteExample = { | ||||||
|  |   write: { | ||||||
|  |     prompt: 'vue', | ||||||
|  |     data: 'Vue.js 是一种用于构建用户界面的渐进式 JavaScript 框架。它的核心库只关注视图层,易于上手,同时也便于与其他库或已有项目整合。\n\nVue.js 的特点包括:\n- 响应式的数据绑定:Vue.js 会自动将数据与 DOM 同步,使得状态管理变得更加简单。\n- 组件化:Vue.js 允许开发者通过小型、独立和通常可复用的组件构建大型应用。\n- 虚拟 DOM:Vue.js 使用虚拟 DOM 实现快速渲染,提高了性能。\n\n在 Vue.js 中,一个典型的应用结构可能包括:\n1. 根实例:每个 Vue 应用都需要一个根实例作为入口点。\n2. 组件系统:可以创建自定义的可复用组件。\n3. 指令:特殊的带有前缀 v- 的属性,为 DOM 元素提供特殊的行为。\n4. 插值:用于文本内容,将数据动态地插入到 HTML。\n5. 计算属性和侦听器:用于处理数据的复杂逻辑和响应数据变化。\n6. 条件渲染:根据条件决定元素的渲染。\n7. 列表渲染:用于显示列表数据。\n8. 事件处理:响应用户交互。\n9. 表单输入绑定:处理表单输入和验证。\n10. 组件生命周期钩子:在组件的不同阶段执行特定的函数。\n\nVue.js 还提供了官方的路由器 Vue Router 和状态管理库 Vuex,以支持构建复杂的单页应用(SPA)。\n\n在开发过程中,开发者通常会使用 Vue CLI,这是一个强大的命令行工具,用于快速生成 Vue 项目脚手架,集成了诸如 Babel、Webpack 等现代前端工具,以及热重载、代码检测等开发体验优化功能。\n\nVue.js 的生态系统还包括大量的第三方库和插件,如 Vuetify(UI 组件库)、Vue Test Utils(测试工具)等,这些都极大地丰富了 Vue.js 的开发生态。\n\n总的来说,Vue.js 是一个灵活、高效的前端框架,适合从小型项目到大型企业级应用的开发。它的易用性、灵活性和强大的社区支持使其成为许多开发者的首选框架之一。', | ||||||
|  |   }, | ||||||
|  |   reply: { | ||||||
|  |     originalContent: '领导,我想请假', | ||||||
|  |     prompt: '不批', | ||||||
|  |     data: '您的请假申请已收悉,经核实和考虑,暂时无法批准您的请假申请。\n\n如有特殊情况或紧急事务,请及时与我联系。\n\n祝工作顺利。\n\n谢谢。', | ||||||
|  |   }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // ========== 【思维导图 UI】相关的枚举 ==========
 | ||||||
|  | 
 | ||||||
|  | /** 思维导图已有内容生成示例 */ | ||||||
|  | export const MindMapContentExample = `# Java 技术栈
 | ||||||
|  | 
 | ||||||
|  | ## 核心技术 | ||||||
|  | ### Java SE | ||||||
|  | ### Java EE | ||||||
|  | 
 | ||||||
|  | ## 框架 | ||||||
|  | ### Spring | ||||||
|  | #### Spring Boot | ||||||
|  | #### Spring MVC | ||||||
|  | #### Spring Data | ||||||
|  | ### Hibernate | ||||||
|  | ### MyBatis | ||||||
|  | 
 | ||||||
|  | ## 构建工具 | ||||||
|  | ### Maven | ||||||
|  | ### Gradle | ||||||
|  | 
 | ||||||
|  | ## 版本控制 | ||||||
|  | ### Git | ||||||
|  | ### SVN | ||||||
|  | 
 | ||||||
|  | ## 测试工具 | ||||||
|  | ### JUnit | ||||||
|  | ### Mockito | ||||||
|  | ### Selenium | ||||||
|  | 
 | ||||||
|  | ## 应用服务器 | ||||||
|  | ### Tomcat | ||||||
|  | ### Jetty | ||||||
|  | ### WildFly | ||||||
|  | 
 | ||||||
|  | ## 数据库 | ||||||
|  | ### MySQL | ||||||
|  | ### PostgreSQL | ||||||
|  | ### Oracle | ||||||
|  | ### MongoDB | ||||||
|  | 
 | ||||||
|  | ## 消息队列 | ||||||
|  | ### Kafka | ||||||
|  | ### RabbitMQ | ||||||
|  | ### ActiveMQ | ||||||
|  | 
 | ||||||
|  | ## 微服务 | ||||||
|  | ### Spring Cloud | ||||||
|  | ### Dubbo | ||||||
|  | 
 | ||||||
|  | ## 容器化 | ||||||
|  | ### Docker | ||||||
|  | ### Kubernetes | ||||||
|  | 
 | ||||||
|  | ## 云服务 | ||||||
|  | ### AWS | ||||||
|  | ### Azure | ||||||
|  | ### Google Cloud | ||||||
|  | 
 | ||||||
|  | ## 开发工具 | ||||||
|  | ### IntelliJ IDEA | ||||||
|  | ### Eclipse | ||||||
|  | ### Visual Studio Code`;
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,146 @@ | ||||||
|  | import type { VbenFormSchema } from '#/adapter/form'; | ||||||
|  | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | 
 | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
|  | import { DICT_TYPE, getDictOptions } from '#/utils'; | ||||||
|  | 
 | ||||||
|  | /** 列表的搜索表单 */ | ||||||
|  | export function useGridFormSchema(): VbenFormSchema[] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       fieldName: 'userId', | ||||||
|  |       label: '用户编号', | ||||||
|  |       component: 'ApiSelect', | ||||||
|  |       componentProps: { | ||||||
|  |         api: getSimpleUserList, | ||||||
|  |         labelField: 'nickname', | ||||||
|  |         valueField: 'id', | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'platform', | ||||||
|  |       label: '平台', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         allowClear: true, | ||||||
|  |         options: getDictOptions(DICT_TYPE.AI_PLATFORM, 'string'), | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'status', | ||||||
|  |       label: '绘画状态', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         allowClear: true, | ||||||
|  |         options: getDictOptions(DICT_TYPE.AI_IMAGE_STATUS, 'number'), | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'publicStatus', | ||||||
|  |       label: '是否发布', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         options: getDictOptions(DICT_TYPE.INFRA_BOOLEAN_STRING, 'boolean'), | ||||||
|  |         allowClear: true, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'createTime', | ||||||
|  |       label: '创建时间', | ||||||
|  |       component: 'RangePicker', | ||||||
|  |       componentProps: { | ||||||
|  |         placeholder: ['开始时间', '结束时间'], | ||||||
|  |         valueFormat: 'YYYY-MM-DD HH:mm:ss', | ||||||
|  |         allowClear: true, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 列表的字段 */ | ||||||
|  | export function useGridColumns(): VxeTableGridOptions['columns'] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       field: 'id', | ||||||
|  |       title: '编号', | ||||||
|  |       minWidth: 180, | ||||||
|  |       fixed: 'left', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '图片', | ||||||
|  |       minWidth: 110, | ||||||
|  |       fixed: 'left', | ||||||
|  |       slots: { default: 'picUrl' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       minWidth: 180, | ||||||
|  |       title: '用户', | ||||||
|  |       slots: { default: 'userId' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'platform', | ||||||
|  |       title: '平台', | ||||||
|  |       minWidth: 120, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_PLATFORM }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'model', | ||||||
|  |       title: '模型', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'status', | ||||||
|  |       title: '绘画状态', | ||||||
|  |       minWidth: 100, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_IMAGE_STATUS }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       minWidth: 100, | ||||||
|  |       title: '是否发布', | ||||||
|  |       slots: { default: 'publicStatus' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'prompt', | ||||||
|  |       title: '提示词', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'createTime', | ||||||
|  |       title: '创建时间', | ||||||
|  |       minWidth: 180, | ||||||
|  |       formatter: 'formatDateTime', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'width', | ||||||
|  |       title: '宽度', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'height', | ||||||
|  |       title: '高度', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'errorMessage', | ||||||
|  |       title: '错误信息', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'taskId', | ||||||
|  |       title: '任务编号', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '操作', | ||||||
|  |       width: 130, | ||||||
|  |       fixed: 'right', | ||||||
|  |       slots: { default: 'actions' }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | @ -1,31 +1,135 @@ | ||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import { Page } from '@vben/common-ui'; | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | import type { AiImageApi } from '#/api/ai/image'; | ||||||
|  | import type { SystemUserApi } from '#/api/system/user'; | ||||||
| 
 | 
 | ||||||
| import { Button } from 'ant-design-vue'; | import { onMounted, ref } from 'vue'; | ||||||
| 
 | 
 | ||||||
|  | import { confirm, Page } from '@vben/common-ui'; | ||||||
|  | 
 | ||||||
|  | import { Image, message, Switch } from 'ant-design-vue'; | ||||||
|  | 
 | ||||||
|  | import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; | ||||||
|  | import { deleteImage, getImagePage, updateImage } from '#/api/ai/image'; | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
| import { DocAlert } from '#/components/doc-alert'; | import { DocAlert } from '#/components/doc-alert'; | ||||||
|  | import { $t } from '#/locales'; | ||||||
|  | import { AiImageStatusEnum } from '#/utils/constants'; | ||||||
|  | 
 | ||||||
|  | import { useGridColumns, useGridFormSchema } from './data'; | ||||||
|  | 
 | ||||||
|  | const userList = ref<SystemUserApi.User[]>([]); // 用户列表 | ||||||
|  | /** 刷新表格 */ | ||||||
|  | function onRefresh() { | ||||||
|  |   gridApi.query(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 删除 */ | ||||||
|  | async function handleDelete(row: AiImageApi.ImageVO) { | ||||||
|  |   const hideLoading = message.loading({ | ||||||
|  |     content: $t('ui.actionMessage.deleting', [row.id]), | ||||||
|  |     key: 'action_key_msg', | ||||||
|  |   }); | ||||||
|  |   try { | ||||||
|  |     await deleteImage(row.id as number); | ||||||
|  |     message.success({ | ||||||
|  |       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||||
|  |       key: 'action_key_msg', | ||||||
|  |     }); | ||||||
|  |     onRefresh(); | ||||||
|  |   } finally { | ||||||
|  |     hideLoading(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | /** 修改是否发布 */ | ||||||
|  | const handleUpdatePublicStatusChange = async (row: AiImageApi.ImageVO) => { | ||||||
|  |   try { | ||||||
|  |     // 修改状态的二次确认 | ||||||
|  |     const text = row.publicStatus ? '公开' : '私有'; | ||||||
|  |     await confirm(`确认要"${text}"该图片吗?`).then(async () => { | ||||||
|  |       await updateImage({ | ||||||
|  |         id: row.id, | ||||||
|  |         publicStatus: row.publicStatus, | ||||||
|  |       }); | ||||||
|  |       onRefresh(); | ||||||
|  |     }); | ||||||
|  |   } catch { | ||||||
|  |     row.publicStatus = !row.publicStatus; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | const [Grid, gridApi] = useVbenVxeGrid({ | ||||||
|  |   formOptions: { | ||||||
|  |     schema: useGridFormSchema(), | ||||||
|  |   }, | ||||||
|  |   gridOptions: { | ||||||
|  |     columns: useGridColumns(), | ||||||
|  |     height: 'auto', | ||||||
|  |     keepSource: true, | ||||||
|  |     proxyConfig: { | ||||||
|  |       ajax: { | ||||||
|  |         query: async ({ page }, formValues) => { | ||||||
|  |           return await getImagePage({ | ||||||
|  |             pageNo: page.currentPage, | ||||||
|  |             pageSize: page.pageSize, | ||||||
|  |             ...formValues, | ||||||
|  |           }); | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     rowConfig: { | ||||||
|  |       keyField: 'id', | ||||||
|  |     }, | ||||||
|  |     toolbarConfig: { | ||||||
|  |       refresh: { code: 'query' }, | ||||||
|  |       search: true, | ||||||
|  |     }, | ||||||
|  |   } as VxeTableGridOptions<AiImageApi.ImageVO>, | ||||||
|  | }); | ||||||
|  | onMounted(async () => { | ||||||
|  |   // 获得下拉数据 | ||||||
|  |   userList.value = await getSimpleUserList(); | ||||||
|  | }); | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <Page> |   <Page auto-content-height> | ||||||
|     <DocAlert title="AI 绘图创作" url="https://doc.iocoder.cn/ai/image/" /> |     <DocAlert title="AI 绘图创作" url="https://doc.iocoder.cn/ai/image/" /> | ||||||
|     <Button |     <Grid table-title="绘画管理列表"> | ||||||
|       danger |       <template #toolbar-tools> | ||||||
|       type="link" |         <TableAction :actions="[]" /> | ||||||
|       target="_blank" |       </template> | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3" |       <template #picUrl="{ row }"> | ||||||
|     > |         <Image :src="row.picUrl" class="h-80px w-80px" /> | ||||||
|       该功能支持 Vue3 + element-plus 版本! |       </template> | ||||||
|     </Button> |       <template #userId="{ row }"> | ||||||
|     <br /> |         <span>{{ | ||||||
|     <Button |           userList.find((item) => item.id === row.userId)?.nickname | ||||||
|       type="link" |         }}</span> | ||||||
|       target="_blank" |       </template> | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/image/manager/index.vue" |       <template #publicStatus="{ row }"> | ||||||
|     > |         <Switch | ||||||
|       可参考 |           v-model:checked="row.publicStatus" | ||||||
|       https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/image/manager/index.vue |           @change="handleUpdatePublicStatusChange(row)" | ||||||
|       代码,pull request 贡献给我们! |           :disabled="row.status !== AiImageStatusEnum.SUCCESS" | ||||||
|     </Button> |         /> | ||||||
|  |       </template> | ||||||
|  |       <template #actions="{ row }"> | ||||||
|  |         <TableAction | ||||||
|  |           :actions="[ | ||||||
|  |             { | ||||||
|  |               label: $t('common.delete'), | ||||||
|  |               type: 'link', | ||||||
|  |               danger: true, | ||||||
|  |               icon: ACTION_ICON.DELETE, | ||||||
|  |               auth: ['ai:image:delete'], | ||||||
|  |               popConfirm: { | ||||||
|  |                 title: $t('ui.actionMessage.deleteConfirm', [row.id]), | ||||||
|  |                 confirm: handleDelete.bind(null, row), | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           ]" | ||||||
|  |         /> | ||||||
|  |       </template> | ||||||
|  |     </Grid> | ||||||
|   </Page> |   </Page> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
|  | @ -1,28 +1,85 @@ | ||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import { Page } from '@vben/common-ui'; | import type { AiMindmapApi } from '#/api/ai/mindmap'; | ||||||
| 
 | 
 | ||||||
| import { Button } from 'ant-design-vue'; | import { nextTick, onMounted, ref } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import { alert, Page } from '@vben/common-ui'; | ||||||
|  | 
 | ||||||
|  | import { generateMindMap } from '#/api/ai/mindmap'; | ||||||
|  | import { MindMapContentExample } from '#/utils/constants'; | ||||||
|  | import Left from './modules/Left.vue'; | ||||||
|  | 
 | ||||||
|  | const ctrl = ref<AbortController>(); // 请求控制 | ||||||
|  | const isGenerating = ref(false); // 是否正在生成思维导图 | ||||||
|  | const isStart = ref(false); // 开始生成,用来清空思维导图 | ||||||
|  | const isEnd = ref(true); // 用来判断结束的时候渲染思维导图 | ||||||
|  | const generatedContent = ref(''); // 生成思维导图结果 | ||||||
|  | 
 | ||||||
|  | const leftRef = ref<InstanceType<typeof Left>>(); // 左边组件 | ||||||
|  | const rightRef = ref(); // 右边组件 | ||||||
|  | 
 | ||||||
|  | /** 使用已有内容直接生成 */ | ||||||
|  | const directGenerate = (existPrompt: string) => { | ||||||
|  |   isEnd.value = false; // 先设置为 false 再设置为 true,让子组建的 watch 能够监听到 | ||||||
|  |   generatedContent.value = existPrompt; | ||||||
|  |   isEnd.value = true; | ||||||
|  | }; | ||||||
|  | /** 提交生成 */ | ||||||
|  | const submit = (data: AiMindmapApi.AiMindMapGenerateReqVO) => { | ||||||
|  |   isGenerating.value = true; | ||||||
|  |   isStart.value = true; | ||||||
|  |   isEnd.value = false; | ||||||
|  |   ctrl.value = new AbortController(); // 请求控制赋值 | ||||||
|  |   generatedContent.value = ''; // 清空生成数据 | ||||||
|  |   generateMindMap({ | ||||||
|  |     data, | ||||||
|  |     onMessage: async (res: any) => { | ||||||
|  |       const { code, data, msg } = JSON.parse(res.data); | ||||||
|  |       if (code !== 0) { | ||||||
|  |         alert(`生成思维导图异常! ${msg}`); | ||||||
|  |         stopStream(); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       generatedContent.value = generatedContent.value + data; | ||||||
|  |       await nextTick(); | ||||||
|  |       rightRef.value?.scrollBottom(); | ||||||
|  |     }, | ||||||
|  |     onClose() { | ||||||
|  |       isEnd.value = true; | ||||||
|  |       leftRef.value?.setGeneratedContent(generatedContent.value); | ||||||
|  |       stopStream(); | ||||||
|  |     }, | ||||||
|  |     onError(err) { | ||||||
|  |       console.error('生成思维导图失败', err); | ||||||
|  |       stopStream(); | ||||||
|  |       // 需要抛出异常,禁止重试 | ||||||
|  |       throw err; | ||||||
|  |     }, | ||||||
|  |     ctrl: ctrl.value, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  | /** 停止 stream 生成 */ | ||||||
|  | const stopStream = () => { | ||||||
|  |   isGenerating.value = false; | ||||||
|  |   isStart.value = false; | ||||||
|  |   ctrl.value?.abort(); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** 初始化 */ | ||||||
|  | onMounted(() => { | ||||||
|  |   generatedContent.value = MindMapContentExample; | ||||||
|  | }); | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <Page> |   <Page auto-content-height> | ||||||
|     <Button |     <div class="absolute bottom-0 left-0 right-0 top-0 flex"> | ||||||
|       danger |       <Left | ||||||
|       type="link" |         ref="leftRef" | ||||||
|       target="_blank" |         :is-generating="isGenerating" | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3" |         @submit="submit" | ||||||
|     > |         @direct-generate="directGenerate" | ||||||
|       该功能支持 Vue3 + element-plus 版本! |       /> | ||||||
|     </Button> |     </div> | ||||||
|     <br /> |  | ||||||
|     <Button |  | ||||||
|       type="link" |  | ||||||
|       target="_blank" |  | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/mindmap/index/index.vue" |  | ||||||
|     > |  | ||||||
|       可参考 |  | ||||||
|       https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/mindmap/index/index.vue |  | ||||||
|       代码,pull request 贡献给我们! |  | ||||||
|     </Button> |  | ||||||
|   </Page> |   </Page> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
|  | @ -0,0 +1,77 @@ | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { reactive, ref } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import { Button, Textarea } from 'ant-design-vue'; | ||||||
|  | 
 | ||||||
|  | import { MindMapContentExample } from '#/utils/constants'; | ||||||
|  | 
 | ||||||
|  | defineProps<{ | ||||||
|  |   isGenerating: boolean; | ||||||
|  | }>(); | ||||||
|  | const emits = defineEmits(['submit', 'directGenerate']); | ||||||
|  | const formData = reactive({ | ||||||
|  |   prompt: '', | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const generatedContent = ref(MindMapContentExample); // 已有的内容 | ||||||
|  | 
 | ||||||
|  | defineExpose({ | ||||||
|  |   setGeneratedContent(newContent: string) { | ||||||
|  |     // 设置已有的内容,在生成结束的时候将结果赋值给该值 | ||||||
|  |     generatedContent.value = newContent; | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  | <template> | ||||||
|  |   <div class="flex w-[350px] flex-col bg-[#f5f7f9] p-5"> | ||||||
|  |     <h3 class="title w-full text-center leading-[28px]">思维导图创作中心</h3> | ||||||
|  |     <div class="flex-grow overflow-y-auto"> | ||||||
|  |       <div> | ||||||
|  |         <b>您的需求?</b> | ||||||
|  |         <Textarea | ||||||
|  |           v-model:value="formData.prompt" | ||||||
|  |           :maxlength="1024" | ||||||
|  |           :rows="8" | ||||||
|  |           class="w-100% mt-15px" | ||||||
|  |           placeholder="请输入提示词,让AI帮你完善" | ||||||
|  |           show-count | ||||||
|  |         /> | ||||||
|  |         <Button | ||||||
|  |           class="mt-[15px] !w-full" | ||||||
|  |           type="primary" | ||||||
|  |           :loading="isGenerating" | ||||||
|  |           @click="emits('submit', formData)" | ||||||
|  |         > | ||||||
|  |           智能生成思维导图 | ||||||
|  |         </Button> | ||||||
|  |       </div> | ||||||
|  |       <div class="mt-[30px]"> | ||||||
|  |         <b>使用已有内容生成?</b> | ||||||
|  |         <Textarea | ||||||
|  |           v-model:value="generatedContent" | ||||||
|  |           :maxlength="1024" | ||||||
|  |           :rows="8" | ||||||
|  |           class="w-100% mt-15px" | ||||||
|  |           placeholder="例如:童话里的小屋应该是什么样子?" | ||||||
|  |           show-count | ||||||
|  |         /> | ||||||
|  |         <Button | ||||||
|  |           class="mt-[15px] !w-full" | ||||||
|  |           type="primary" | ||||||
|  |           @click="emits('directGenerate', generatedContent)" | ||||||
|  |           :disabled="isGenerating" | ||||||
|  |         > | ||||||
|  |           直接生成 | ||||||
|  |         </Button> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <style lang="scss" scoped> | ||||||
|  | .title { | ||||||
|  |   height: 1.75rem; | ||||||
|  |   font-size: 1.25rem; | ||||||
|  |   color: hsl(var(--primary)); | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | @ -0,0 +1,167 @@ | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import { Button, Card, message } from 'ant-design-vue'; | ||||||
|  | import markdownit from 'markdown-it'; | ||||||
|  | import { Markmap } from 'markmap-view' | ||||||
|  | import { Transformer } from 'markmap-lib' | ||||||
|  | import { Toolbar } from 'markmap-toolbar' | ||||||
|  | import { nextTick, onMounted, ref, watch } from 'vue'; | ||||||
|  | 
 | ||||||
|  | const md = markdownit(); | ||||||
|  | const props = defineProps<{ | ||||||
|  |   generatedContent: string // 生成结果 | ||||||
|  |   isEnd: boolean // 是否结束 | ||||||
|  |   isGenerating: boolean // 是否正在生成 | ||||||
|  |   isStart: boolean // 开始状态,开始时需要清除 html | ||||||
|  | }>() | ||||||
|  | const contentRef = ref<HTMLDivElement>() // 右侧出来 header 以下的区域 | ||||||
|  | const mdContainerRef = ref<HTMLDivElement>() // markdown 的容器,用来滚动到底下的 | ||||||
|  | const mindMapRef = ref<HTMLDivElement>() // 思维导图的容器 | ||||||
|  | const svgRef = ref<SVGElement>() // 思维导图的渲染 svg | ||||||
|  | const toolBarRef = ref<HTMLDivElement>() // 思维导图右下角的工具栏,缩放等 | ||||||
|  | const html = ref('') // 生成过程中的文本 | ||||||
|  | const contentAreaHeight = ref(0) // 生成区域的高度,出去 header 部分 | ||||||
|  | let markMap: Markmap | null = null | ||||||
|  | const transformer = new Transformer() | ||||||
|  | 
 | ||||||
|  | onMounted(() => { | ||||||
|  |   contentAreaHeight.value = contentRef.value?.clientHeight || 0 // 获取区域高度 | ||||||
|  |   /** 初始化思维导图 **/ | ||||||
|  |   try { | ||||||
|  |     markMap = Markmap.create(svgRef.value!) | ||||||
|  |     const { el } = Toolbar.create(markMap) | ||||||
|  |     toolBarRef.value?.append(el) | ||||||
|  |     nextTick(update) | ||||||
|  |   } catch (e) { | ||||||
|  |     message.error('思维导图初始化失败') | ||||||
|  |   } | ||||||
|  | }) | ||||||
|  | watch(props, ({ generatedContent, isGenerating, isEnd, isStart }) => { | ||||||
|  |   // 开始生成的时候清空一下 markdown 的内容 | ||||||
|  |   if (isStart) { | ||||||
|  |     html.value = '' | ||||||
|  |   } | ||||||
|  |   // 生成内容的时候使用 markdown 来渲染 | ||||||
|  |   if (isGenerating) { | ||||||
|  |     html.value = md.render(generatedContent) | ||||||
|  |   } | ||||||
|  |   // 生成结束时更新思维导图 | ||||||
|  |   if (isEnd) { | ||||||
|  |     update() | ||||||
|  |   } | ||||||
|  | }) | ||||||
|  | /** 更新思维导图的展示 */ | ||||||
|  | const update = () => { | ||||||
|  |   try { | ||||||
|  |     const { root } = transformer.transform(processContent(props.generatedContent)) | ||||||
|  |     markMap?.setData(root) | ||||||
|  |     markMap?.fit() | ||||||
|  |   } catch (e) { | ||||||
|  |     console.error(e) | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 处理内容 */ | ||||||
|  | const processContent = (text: string) => { | ||||||
|  |   const arr: string[] = [] | ||||||
|  |   const lines = text.split('\n') | ||||||
|  |   for (let line of lines) { | ||||||
|  |     if (line.indexOf('```') !== -1) { | ||||||
|  |       continue | ||||||
|  |     } | ||||||
|  |     line = line.replace(/([*_~`>])|(\d+\.)\s/g, '') | ||||||
|  |     arr.push(line) | ||||||
|  |   } | ||||||
|  |   return arr.join('\n') | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 下载图片:download SVG to png file */ | ||||||
|  | const downloadImage = () => { | ||||||
|  |   const svgElement = mindMapRef.value | ||||||
|  |   // 将 SVG 渲染到图片对象 | ||||||
|  |   const serializer = new XMLSerializer() | ||||||
|  |   const source = `<?xml version="1.0" standalone="no"?>\r\n${serializer.serializeToString(svgRef.value!)}` | ||||||
|  |   const base64Url = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(source)}` | ||||||
|  |   download.image({ | ||||||
|  |     url: base64Url, | ||||||
|  |     canvasWidth: svgElement?.offsetWidth, | ||||||
|  |     canvasHeight: svgElement?.offsetHeight, | ||||||
|  |     drawWithImageSize: false | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | defineExpose({ | ||||||
|  |   scrollBottom() { | ||||||
|  |     mdContainerRef.value?.scrollTo(0, mdContainerRef.value?.scrollHeight) | ||||||
|  |   } | ||||||
|  | }) | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <template> | ||||||
|  |   <Card class="my-card h-full flex-grow"> | ||||||
|  |     <template #title> | ||||||
|  |       <h3 class="m-0 flex shrink-0 items-center justify-between px-7"> | ||||||
|  |         <span>思维导图预览</span> | ||||||
|  |         <!-- 展示在右上角 --> | ||||||
|  |         <Button v-show="isEnd" size="small" type="primary" style="display: flex;" @click="downloadImage"> | ||||||
|  |           <template #icon> | ||||||
|  |             <div class="flex items-center justify-center"> | ||||||
|  |               <span class="icon-[ant-design--copy-twotone]"></span> | ||||||
|  |             </div> | ||||||
|  |           </template> | ||||||
|  |           下载图片 | ||||||
|  |         </Button> | ||||||
|  |       </h3> | ||||||
|  |     </template> | ||||||
|  |     <div ref="contentRef" class="hide-scroll-bar h-full box-border"> | ||||||
|  |       <!--展示 markdown 的容器,最终生成的是 html 字符串,直接用 v-html 嵌入--> | ||||||
|  |       <div v-if="isGenerating" ref="mdContainerRef" class="wh-full overflow-y-auto"> | ||||||
|  |         <div class="flex flex-col items-center justify-center" v-html="html"></div> | ||||||
|  |       </div> | ||||||
|  | 
 | ||||||
|  |       <div ref="mindMapRef" class="wh-full"> | ||||||
|  |         <svg ref="svgRef" :style="{ height: `${contentAreaHeight}px` }" class="w-full" /> | ||||||
|  |         <div ref="toolBarRef" class="absolute bottom-[10px] right-5"></div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </Card> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <style lang="scss" scoped> | ||||||
|  | .hide-scroll-bar { | ||||||
|  |   -ms-overflow-style: none; | ||||||
|  |   scrollbar-width: none; | ||||||
|  | 
 | ||||||
|  |   &::-webkit-scrollbar { | ||||||
|  |     width: 0; | ||||||
|  |     height: 0; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .my-card { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: column; | ||||||
|  | 
 | ||||||
|  |   :deep(.el-card__body) { | ||||||
|  |     @extend .hide-scroll-bar; | ||||||
|  | 
 | ||||||
|  |     padding: 0; | ||||||
|  |     overflow-y: auto; | ||||||
|  |     box-sizing: border-box; | ||||||
|  |     flex-grow: 1; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // markmap的tool样式覆盖 | ||||||
|  | :deep(.markmap) { | ||||||
|  |   width: 100%; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | :deep(.mm-toolbar-brand) { | ||||||
|  |   display: none; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | :deep(.mm-toolbar) { | ||||||
|  |   display: flex; | ||||||
|  |   flex-direction: row; | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | @ -0,0 +1,84 @@ | ||||||
|  | import type { VbenFormSchema } from '#/adapter/form'; | ||||||
|  | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | 
 | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
|  | 
 | ||||||
|  | /** 列表的搜索表单 */ | ||||||
|  | export function useGridFormSchema(): VbenFormSchema[] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       fieldName: 'userId', | ||||||
|  |       label: '用户编号', | ||||||
|  |       component: 'ApiSelect', | ||||||
|  |       componentProps: { | ||||||
|  |         api: getSimpleUserList, | ||||||
|  |         labelField: 'nickname', | ||||||
|  |         valueField: 'id', | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'prompt', | ||||||
|  |       label: '提示词', | ||||||
|  |       component: 'Input', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'createTime', | ||||||
|  |       label: '创建时间', | ||||||
|  |       component: 'RangePicker', | ||||||
|  |       componentProps: { | ||||||
|  |         placeholder: ['开始时间', '结束时间'], | ||||||
|  |         valueFormat: 'YYYY-MM-DD HH:mm:ss', | ||||||
|  |         allowClear: true, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 列表的字段 */ | ||||||
|  | export function useGridColumns(): VxeTableGridOptions['columns'] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       field: 'id', | ||||||
|  |       title: '编号', | ||||||
|  |       minWidth: 180, | ||||||
|  |       fixed: 'left', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       minWidth: 180, | ||||||
|  |       title: '用户', | ||||||
|  |       slots: { default: 'userId' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'prompt', | ||||||
|  |       title: '提示词', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'generatedContent', | ||||||
|  |       title: '思维导图', | ||||||
|  |       minWidth: 300, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'model', | ||||||
|  |       title: '模型', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'createTime', | ||||||
|  |       title: '创建时间', | ||||||
|  |       minWidth: 180, | ||||||
|  |       formatter: 'formatDateTime', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'errorMessage', | ||||||
|  |       title: '错误信息', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '操作', | ||||||
|  |       width: 130, | ||||||
|  |       fixed: 'right', | ||||||
|  |       slots: { default: 'actions' }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | @ -1,31 +1,108 @@ | ||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
|  | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | import type { AiMindmapApi } from '#/api/ai/mindmap'; | ||||||
|  | import type { SystemUserApi } from '#/api/system/user'; | ||||||
|  | 
 | ||||||
|  | import { onMounted, ref } from 'vue'; | ||||||
|  | 
 | ||||||
| import { Page } from '@vben/common-ui'; | import { Page } from '@vben/common-ui'; | ||||||
| 
 | 
 | ||||||
| import { Button } from 'ant-design-vue'; | import { message } from 'ant-design-vue'; | ||||||
| 
 | 
 | ||||||
|  | import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; | ||||||
|  | import { deleteMindMap, getMindMapPage } from '#/api/ai/mindmap'; | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
| import { DocAlert } from '#/components/doc-alert'; | import { DocAlert } from '#/components/doc-alert'; | ||||||
|  | import { $t } from '#/locales'; | ||||||
|  | 
 | ||||||
|  | import { useGridColumns, useGridFormSchema } from './data'; | ||||||
|  | 
 | ||||||
|  | const userList = ref<SystemUserApi.User[]>([]); // 用户列表 | ||||||
|  | /** 刷新表格 */ | ||||||
|  | function onRefresh() { | ||||||
|  |   gridApi.query(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 删除 */ | ||||||
|  | async function handleDelete(row: AiMindmapApi.MindMapVO) { | ||||||
|  |   const hideLoading = message.loading({ | ||||||
|  |     content: $t('ui.actionMessage.deleting', [row.id]), | ||||||
|  |     key: 'action_key_msg', | ||||||
|  |   }); | ||||||
|  |   try { | ||||||
|  |     await deleteMindMap(row.id as number); | ||||||
|  |     message.success({ | ||||||
|  |       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||||
|  |       key: 'action_key_msg', | ||||||
|  |     }); | ||||||
|  |     onRefresh(); | ||||||
|  |   } finally { | ||||||
|  |     hideLoading(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | const [Grid, gridApi] = useVbenVxeGrid({ | ||||||
|  |   formOptions: { | ||||||
|  |     schema: useGridFormSchema(), | ||||||
|  |   }, | ||||||
|  |   gridOptions: { | ||||||
|  |     columns: useGridColumns(), | ||||||
|  |     height: 'auto', | ||||||
|  |     keepSource: true, | ||||||
|  |     proxyConfig: { | ||||||
|  |       ajax: { | ||||||
|  |         query: async ({ page }, formValues) => { | ||||||
|  |           return await getMindMapPage({ | ||||||
|  |             pageNo: page.currentPage, | ||||||
|  |             pageSize: page.pageSize, | ||||||
|  |             ...formValues, | ||||||
|  |           }); | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     rowConfig: { | ||||||
|  |       keyField: 'id', | ||||||
|  |     }, | ||||||
|  |     toolbarConfig: { | ||||||
|  |       refresh: { code: 'query' }, | ||||||
|  |       search: true, | ||||||
|  |     }, | ||||||
|  |   } as VxeTableGridOptions<AiMindmapApi.MindMapVO>, | ||||||
|  | }); | ||||||
|  | onMounted(async () => { | ||||||
|  |   // 获得下拉数据 | ||||||
|  |   userList.value = await getSimpleUserList(); | ||||||
|  | }); | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <Page> |   <Page auto-content-height> | ||||||
|     <DocAlert title="AI 思维导图" url="https://doc.iocoder.cn/ai/mindmap/" /> |     <DocAlert title="AI 思维导图" url="https://doc.iocoder.cn/ai/mindmap/" /> | ||||||
|     <Button |     <Grid table-title="思维导图管理列表"> | ||||||
|       danger |       <template #toolbar-tools> | ||||||
|       type="link" |         <TableAction :actions="[]" /> | ||||||
|       target="_blank" |       </template> | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3" |       <template #userId="{ row }"> | ||||||
|     > |         <span>{{ | ||||||
|       该功能支持 Vue3 + element-plus 版本! |           userList.find((item) => item.id === row.userId)?.nickname | ||||||
|     </Button> |         }}</span> | ||||||
|     <br /> |       </template> | ||||||
|     <Button |       <template #actions="{ row }"> | ||||||
|       type="link" |         <TableAction | ||||||
|       target="_blank" |           :actions="[ | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/mindmap/manager/index" |             { | ||||||
|     > |               label: $t('common.delete'), | ||||||
|       可参考 |               type: 'link', | ||||||
|       https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/mindmap/manager/index |               danger: true, | ||||||
|       代码,pull request 贡献给我们! |               icon: ACTION_ICON.DELETE, | ||||||
|     </Button> |               auth: ['ai:mind-map:delete'], | ||||||
|  |               popConfirm: { | ||||||
|  |                 title: $t('ui.actionMessage.deleteConfirm', [row.id]), | ||||||
|  |                 confirm: handleDelete.bind(null, row), | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           ]" | ||||||
|  |         /> | ||||||
|  |       </template> | ||||||
|  |     </Grid> | ||||||
|   </Page> |   </Page> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
|  | @ -0,0 +1,175 @@ | ||||||
|  | import type { VbenFormSchema } from '#/adapter/form'; | ||||||
|  | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | 
 | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
|  | import { DICT_TYPE, getDictOptions } from '#/utils'; | ||||||
|  | 
 | ||||||
|  | /** 列表的搜索表单 */ | ||||||
|  | export function useGridFormSchema(): VbenFormSchema[] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       fieldName: 'userId', | ||||||
|  |       label: '用户编号', | ||||||
|  |       component: 'ApiSelect', | ||||||
|  |       componentProps: { | ||||||
|  |         api: getSimpleUserList, | ||||||
|  |         labelField: 'nickname', | ||||||
|  |         valueField: 'id', | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'title', | ||||||
|  |       label: '音乐名称', | ||||||
|  |       component: 'Input', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'status', | ||||||
|  |       label: '绘画状态', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         allowClear: true, | ||||||
|  |         options: getDictOptions(DICT_TYPE.AI_MUSIC_STATUS, 'number'), | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'generateMode', | ||||||
|  |       label: '生成模式', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         allowClear: true, | ||||||
|  |         options: getDictOptions(DICT_TYPE.AI_GENERATE_MODE, 'number'), | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'createTime', | ||||||
|  |       label: '创建时间', | ||||||
|  |       component: 'RangePicker', | ||||||
|  |       componentProps: { | ||||||
|  |         placeholder: ['开始时间', '结束时间'], | ||||||
|  |         valueFormat: 'YYYY-MM-DD HH:mm:ss', | ||||||
|  |         allowClear: true, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'publicStatus', | ||||||
|  |       label: '是否发布', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         options: getDictOptions(DICT_TYPE.INFRA_BOOLEAN_STRING, 'boolean'), | ||||||
|  |         allowClear: true, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 列表的字段 */ | ||||||
|  | export function useGridColumns(): VxeTableGridOptions['columns'] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       field: 'id', | ||||||
|  |       title: '编号', | ||||||
|  |       minWidth: 180, | ||||||
|  |       fixed: 'left', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '音乐名称', | ||||||
|  |       minWidth: 180, | ||||||
|  |       fixed: 'left', | ||||||
|  |       field: 'title', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       minWidth: 180, | ||||||
|  |       title: '用户', | ||||||
|  |       slots: { default: 'userId' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'status', | ||||||
|  |       title: '音乐状态', | ||||||
|  |       minWidth: 100, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_MUSIC_STATUS }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'model', | ||||||
|  |       title: '模型', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '内容', | ||||||
|  |       minWidth: 180, | ||||||
|  |       slots: { default: 'content' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'status', | ||||||
|  |       title: '绘画状态', | ||||||
|  |       minWidth: 100, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_IMAGE_STATUS }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'duration', | ||||||
|  |       title: '时长(秒)', | ||||||
|  |       minWidth: 100, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'prompt', | ||||||
|  |       title: '提示词', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'lyric', | ||||||
|  |       title: '歌词', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'gptDescriptionPrompt', | ||||||
|  |       title: '描述', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'generateMode', | ||||||
|  |       title: '生成模式', | ||||||
|  |       minWidth: 100, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_GENERATE_MODE }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '风格标签', | ||||||
|  |       minWidth: 180, | ||||||
|  |       slots: { default: 'tags' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       minWidth: 100, | ||||||
|  |       title: '是否发布', | ||||||
|  |       slots: { default: 'publicStatus' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'taskId', | ||||||
|  |       title: '任务编号', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'errorMessage', | ||||||
|  |       title: '错误信息', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'createTime', | ||||||
|  |       title: '创建时间', | ||||||
|  |       minWidth: 180, | ||||||
|  |       formatter: 'formatDateTime', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '操作', | ||||||
|  |       width: 130, | ||||||
|  |       fixed: 'right', | ||||||
|  |       slots: { default: 'actions' }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | @ -1,31 +1,169 @@ | ||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import { Page } from '@vben/common-ui'; | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | import type { AiMusicApi } from '#/api/ai/music'; | ||||||
|  | import type { SystemUserApi } from '#/api/system/user'; | ||||||
| 
 | 
 | ||||||
| import { Button } from 'ant-design-vue'; | import { onMounted, ref } from 'vue'; | ||||||
| 
 | 
 | ||||||
|  | import { confirm, Page } from '@vben/common-ui'; | ||||||
|  | 
 | ||||||
|  | import { Button, message, Switch, Tag } from 'ant-design-vue'; | ||||||
|  | 
 | ||||||
|  | import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; | ||||||
|  | import { deleteMusic, getMusicPage, updateMusic } from '#/api/ai/music'; | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
| import { DocAlert } from '#/components/doc-alert'; | import { DocAlert } from '#/components/doc-alert'; | ||||||
|  | import { $t } from '#/locales'; | ||||||
|  | import { AiMusicStatusEnum } from '#/utils/constants'; | ||||||
|  | 
 | ||||||
|  | import { useGridColumns, useGridFormSchema } from './data'; | ||||||
|  | 
 | ||||||
|  | const userList = ref<SystemUserApi.User[]>([]); // 用户列表 | ||||||
|  | /** 刷新表格 */ | ||||||
|  | function onRefresh() { | ||||||
|  |   gridApi.query(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 删除 */ | ||||||
|  | async function handleDelete(row: AiMusicApi.MusicVO) { | ||||||
|  |   const hideLoading = message.loading({ | ||||||
|  |     content: $t('ui.actionMessage.deleting', [row.id]), | ||||||
|  |     key: 'action_key_msg', | ||||||
|  |   }); | ||||||
|  |   try { | ||||||
|  |     await deleteMusic(row.id as number); | ||||||
|  |     message.success({ | ||||||
|  |       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||||
|  |       key: 'action_key_msg', | ||||||
|  |     }); | ||||||
|  |     onRefresh(); | ||||||
|  |   } finally { | ||||||
|  |     hideLoading(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | /** 修改是否发布 */ | ||||||
|  | const handleUpdatePublicStatusChange = async (row: AiMusicApi.MusicVO) => { | ||||||
|  |   try { | ||||||
|  |     // 修改状态的二次确认 | ||||||
|  |     const text = row.publicStatus ? '公开' : '私有'; | ||||||
|  |     await confirm(`确认要"${text}"该图片吗?`).then(async () => { | ||||||
|  |       await updateMusic({ | ||||||
|  |         id: row.id, | ||||||
|  |         publicStatus: row.publicStatus, | ||||||
|  |       }); | ||||||
|  |       onRefresh(); | ||||||
|  |     }); | ||||||
|  |   } catch { | ||||||
|  |     row.publicStatus = !row.publicStatus; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | const [Grid, gridApi] = useVbenVxeGrid({ | ||||||
|  |   formOptions: { | ||||||
|  |     schema: useGridFormSchema(), | ||||||
|  |   }, | ||||||
|  |   gridOptions: { | ||||||
|  |     columns: useGridColumns(), | ||||||
|  |     height: 'auto', | ||||||
|  |     keepSource: true, | ||||||
|  |     proxyConfig: { | ||||||
|  |       ajax: { | ||||||
|  |         query: async ({ page }, formValues) => { | ||||||
|  |           return await getMusicPage({ | ||||||
|  |             pageNo: page.currentPage, | ||||||
|  |             pageSize: page.pageSize, | ||||||
|  |             ...formValues, | ||||||
|  |           }); | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     rowConfig: { | ||||||
|  |       keyField: 'id', | ||||||
|  |     }, | ||||||
|  |     toolbarConfig: { | ||||||
|  |       refresh: { code: 'query' }, | ||||||
|  |       search: true, | ||||||
|  |     }, | ||||||
|  |   } as VxeTableGridOptions<AiMusicApi.MusicVO>, | ||||||
|  | }); | ||||||
|  | onMounted(async () => { | ||||||
|  |   // 获得下拉数据 | ||||||
|  |   userList.value = await getSimpleUserList(); | ||||||
|  | }); | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <Page> |   <Page auto-content-height> | ||||||
|     <DocAlert title="AI 音乐创作" url="https://doc.iocoder.cn/ai/music/" /> |     <DocAlert title="AI 音乐创作" url="https://doc.iocoder.cn/ai/music/" /> | ||||||
|     <Button |     <Grid table-title="音乐管理列表"> | ||||||
|       danger |       <template #toolbar-tools> | ||||||
|       type="link" |         <TableAction :actions="[]" /> | ||||||
|       target="_blank" |       </template> | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3" | 
 | ||||||
|     > |       <template #userId="{ row }"> | ||||||
|       该功能支持 Vue3 + element-plus 版本! |         <span>{{ | ||||||
|     </Button> |           userList.find((item) => item.id === row.userId)?.nickname | ||||||
|     <br /> |         }}</span> | ||||||
|     <Button |       </template> | ||||||
|       type="link" |       <template #content="{ row }"> | ||||||
|       target="_blank" |         <Button | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/music/manager/index.vue" |           type="link" | ||||||
|     > |           v-if="row.audioUrl?.length > 0" | ||||||
|       可参考 |           :href="row.audioUrl" | ||||||
|       https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/music/manager/index.vue |           target="_blank" | ||||||
|       代码,pull request 贡献给我们! |           style="padding: 0" | ||||||
|     </Button> |         > | ||||||
|  |           音乐 | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           type="link" | ||||||
|  |           v-if="row.videoUrl?.length > 0" | ||||||
|  |           :href="row.videoUrl" | ||||||
|  |           target="_blank" | ||||||
|  |           class="!pl-5px" | ||||||
|  |           style="padding: 0" | ||||||
|  |         > | ||||||
|  |           视频 | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           type="link" | ||||||
|  |           v-if="row.imageUrl?.length > 0" | ||||||
|  |           :href="row.imageUrl" | ||||||
|  |           target="_blank" | ||||||
|  |           class="!pl-5px" | ||||||
|  |           style="padding: 0" | ||||||
|  |         > | ||||||
|  |           封面 | ||||||
|  |         </Button> | ||||||
|  |       </template> | ||||||
|  |       <template #publicStatus="{ row }"> | ||||||
|  |         <Switch | ||||||
|  |           v-model:checked="row.publicStatus" | ||||||
|  |           @change="handleUpdatePublicStatusChange(row)" | ||||||
|  |           :disabled="row.status !== AiMusicStatusEnum.SUCCESS" | ||||||
|  |         /> | ||||||
|  |       </template> | ||||||
|  |       <template #tags="{ row }"> | ||||||
|  |         <Tag v-for="tag in row.tags" :key="tag" class="ml-2px"> | ||||||
|  |           {{ tag }} | ||||||
|  |         </Tag> | ||||||
|  |       </template> | ||||||
|  |       <template #actions="{ row }"> | ||||||
|  |         <TableAction | ||||||
|  |           :actions="[ | ||||||
|  |             { | ||||||
|  |               label: $t('common.delete'), | ||||||
|  |               type: 'link', | ||||||
|  |               danger: true, | ||||||
|  |               icon: ACTION_ICON.DELETE, | ||||||
|  |               auth: ['ai:music:delete'], | ||||||
|  |               popConfirm: { | ||||||
|  |                 title: $t('ui.actionMessage.deleteConfirm', [row.id]), | ||||||
|  |                 confirm: handleDelete.bind(null, row), | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           ]" | ||||||
|  |         /> | ||||||
|  |       </template> | ||||||
|  |     </Grid> | ||||||
|   </Page> |   </Page> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
|  | @ -0,0 +1,157 @@ | ||||||
|  | import type { VbenFormSchema } from '#/adapter/form'; | ||||||
|  | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | 
 | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
|  | import { DICT_TYPE, getDictOptions } from '#/utils'; | ||||||
|  | 
 | ||||||
|  | /** 列表的搜索表单 */ | ||||||
|  | export function useGridFormSchema(): VbenFormSchema[] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       fieldName: 'userId', | ||||||
|  |       label: '用户编号', | ||||||
|  |       component: 'ApiSelect', | ||||||
|  |       componentProps: { | ||||||
|  |         api: getSimpleUserList, | ||||||
|  |         labelField: 'nickname', | ||||||
|  |         valueField: 'id', | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'type', | ||||||
|  |       label: '写作类型', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         allowClear: true, | ||||||
|  |         options: getDictOptions(DICT_TYPE.AI_WRITE_TYPE, 'number'), | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'platform', | ||||||
|  |       label: '平台', | ||||||
|  |       component: 'Select', | ||||||
|  |       componentProps: { | ||||||
|  |         allowClear: true, | ||||||
|  |         options: getDictOptions(DICT_TYPE.AI_PLATFORM, 'number'), | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       fieldName: 'createTime', | ||||||
|  |       label: '创建时间', | ||||||
|  |       component: 'RangePicker', | ||||||
|  |       componentProps: { | ||||||
|  |         placeholder: ['开始时间', '结束时间'], | ||||||
|  |         valueFormat: 'YYYY-MM-DD HH:mm:ss', | ||||||
|  |         allowClear: true, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 列表的字段 */ | ||||||
|  | export function useGridColumns(): VxeTableGridOptions['columns'] { | ||||||
|  |   return [ | ||||||
|  |     { | ||||||
|  |       field: 'id', | ||||||
|  |       title: '编号', | ||||||
|  |       minWidth: 180, | ||||||
|  |       fixed: 'left', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       minWidth: 180, | ||||||
|  |       title: '用户', | ||||||
|  |       slots: { default: 'userId' }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'type', | ||||||
|  |       title: '写作类型', | ||||||
|  |       minWidth: 100, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_WRITE_TYPE }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'platform', | ||||||
|  |       title: '平台', | ||||||
|  |       minWidth: 120, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_WRITE_TYPE }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'model', | ||||||
|  |       title: '模型', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'prompt', | ||||||
|  |       title: '生成内容提示', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'generatedContent', | ||||||
|  |       title: '生成的内容', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'originalContent', | ||||||
|  |       title: '原文', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'length', | ||||||
|  |       title: '长度', | ||||||
|  |       minWidth: 120, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_WRITE_LENGTH }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'format', | ||||||
|  |       title: '格式', | ||||||
|  |       minWidth: 120, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_WRITE_FORMAT }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'tone', | ||||||
|  |       title: '语气', | ||||||
|  |       minWidth: 120, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_WRITE_TONE }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'language', | ||||||
|  |       title: '语言', | ||||||
|  |       minWidth: 120, | ||||||
|  |       cellRender: { | ||||||
|  |         name: 'CellDict', | ||||||
|  |         props: { type: DICT_TYPE.AI_WRITE_LANGUAGE }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'createTime', | ||||||
|  |       title: '创建时间', | ||||||
|  |       minWidth: 180, | ||||||
|  |       formatter: 'formatDateTime', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       field: 'errorMessage', | ||||||
|  |       title: '错误信息', | ||||||
|  |       minWidth: 180, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '操作', | ||||||
|  |       width: 130, | ||||||
|  |       fixed: 'right', | ||||||
|  |       slots: { default: 'actions' }, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  | @ -1,31 +1,108 @@ | ||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
|  | import type { VxeTableGridOptions } from '#/adapter/vxe-table'; | ||||||
|  | import type { AiWriteApi } from '#/api/ai/write'; | ||||||
|  | import type { SystemUserApi } from '#/api/system/user'; | ||||||
|  | 
 | ||||||
|  | import { onMounted, ref } from 'vue'; | ||||||
|  | 
 | ||||||
| import { Page } from '@vben/common-ui'; | import { Page } from '@vben/common-ui'; | ||||||
| 
 | 
 | ||||||
| import { Button } from 'ant-design-vue'; | import { message } from 'ant-design-vue'; | ||||||
| 
 | 
 | ||||||
|  | import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table'; | ||||||
|  | import { deleteWrite, getWritePage } from '#/api/ai/write'; | ||||||
|  | import { getSimpleUserList } from '#/api/system/user'; | ||||||
| import { DocAlert } from '#/components/doc-alert'; | import { DocAlert } from '#/components/doc-alert'; | ||||||
|  | import { $t } from '#/locales'; | ||||||
|  | 
 | ||||||
|  | import { useGridColumns, useGridFormSchema } from './data'; | ||||||
|  | 
 | ||||||
|  | const userList = ref<SystemUserApi.User[]>([]); // 用户列表 | ||||||
|  | /** 刷新表格 */ | ||||||
|  | function onRefresh() { | ||||||
|  |   gridApi.query(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** 删除 */ | ||||||
|  | async function handleDelete(row: AiWriteApi.AiWritePageReqVO) { | ||||||
|  |   const hideLoading = message.loading({ | ||||||
|  |     content: $t('ui.actionMessage.deleting', [row.id]), | ||||||
|  |     key: 'action_key_msg', | ||||||
|  |   }); | ||||||
|  |   try { | ||||||
|  |     await deleteWrite(row.id as number); | ||||||
|  |     message.success({ | ||||||
|  |       content: $t('ui.actionMessage.deleteSuccess', [row.id]), | ||||||
|  |       key: 'action_key_msg', | ||||||
|  |     }); | ||||||
|  |     onRefresh(); | ||||||
|  |   } finally { | ||||||
|  |     hideLoading(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | const [Grid, gridApi] = useVbenVxeGrid({ | ||||||
|  |   formOptions: { | ||||||
|  |     schema: useGridFormSchema(), | ||||||
|  |   }, | ||||||
|  |   gridOptions: { | ||||||
|  |     columns: useGridColumns(), | ||||||
|  |     height: 'auto', | ||||||
|  |     keepSource: true, | ||||||
|  |     proxyConfig: { | ||||||
|  |       ajax: { | ||||||
|  |         query: async ({ page }, formValues) => { | ||||||
|  |           return await getWritePage({ | ||||||
|  |             pageNo: page.currentPage, | ||||||
|  |             pageSize: page.pageSize, | ||||||
|  |             ...formValues, | ||||||
|  |           }); | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     rowConfig: { | ||||||
|  |       keyField: 'id', | ||||||
|  |     }, | ||||||
|  |     toolbarConfig: { | ||||||
|  |       refresh: { code: 'query' }, | ||||||
|  |       search: true, | ||||||
|  |     }, | ||||||
|  |   } as VxeTableGridOptions<AiWriteApi.AiWritePageReqVO>, | ||||||
|  | }); | ||||||
|  | onMounted(async () => { | ||||||
|  |   // 获得下拉数据 | ||||||
|  |   userList.value = await getSimpleUserList(); | ||||||
|  | }); | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <Page> |   <Page auto-content-height> | ||||||
|     <DocAlert title="AI 写作助手" url="https://doc.iocoder.cn/ai/write/" /> |     <DocAlert title="AI 写作助手" url="https://doc.iocoder.cn/ai/write/" /> | ||||||
|     <Button |     <Grid table-title="写作管理列表"> | ||||||
|       danger |       <template #toolbar-tools> | ||||||
|       type="link" |         <TableAction :actions="[]" /> | ||||||
|       target="_blank" |       </template> | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3" |       <template #userId="{ row }"> | ||||||
|     > |         <span>{{ | ||||||
|       该功能支持 Vue3 + element-plus 版本! |           userList.find((item) => item.id === row.userId)?.nickname | ||||||
|     </Button> |         }}</span> | ||||||
|     <br /> |       </template> | ||||||
|     <Button |       <template #actions="{ row }"> | ||||||
|       type="link" |         <TableAction | ||||||
|       target="_blank" |           :actions="[ | ||||||
|       href="https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/write/manager/index.vue" |             { | ||||||
|     > |               label: $t('common.delete'), | ||||||
|       可参考 |               type: 'link', | ||||||
|       https://github.com/yudaocode/yudao-ui-admin-vue3/blob/master/src/views/ai/write/manager/index.vue |               danger: true, | ||||||
|       代码,pull request 贡献给我们! |               icon: ACTION_ICON.DELETE, | ||||||
|     </Button> |               auth: ['ai:write:delete'], | ||||||
|  |               popConfirm: { | ||||||
|  |                 title: $t('ui.actionMessage.deleteConfirm', [row.id]), | ||||||
|  |                 confirm: handleDelete.bind(null, row), | ||||||
|  |               }, | ||||||
|  |             }, | ||||||
|  |           ]" | ||||||
|  |         /> | ||||||
|  |       </template> | ||||||
|  |     </Grid> | ||||||
|   </Page> |   </Page> | ||||||
| </template> | </template> | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 gjd
						gjd