diff --git a/src/api/ai/writer/index.ts b/src/api/ai/writer/index.ts
new file mode 100644
index 00000000..01664b3d
--- /dev/null
+++ b/src/api/ai/writer/index.ts
@@ -0,0 +1,64 @@
+import { fetchEventSource } from '@microsoft/fetch-event-source'
+
+import { getAccessToken } from '@/utils/auth'
+import { config } from '@/config/axios/config'
+
+export interface WriteParams {
+ /**
+ * 1:撰写 2:回复
+ */
+ type: 1 | 2
+ /**
+ * 写作内容提示 1。撰写 2回复
+ */
+ prompt: string
+ /**
+ * 原文
+ */
+ originalContent: string
+ /**
+ * 长度
+ */
+ length: number
+ /**
+ * 格式
+ */
+ format: number
+ /**
+ * 语气
+ */
+ tone: number
+ /**
+ * 语言
+ */
+ language: number
+}
+export const writeStream = ({
+ data,
+ onClose,
+ onMessage,
+ onError,
+ ctrl
+}: {
+ data: WriteParams
+ onMessage?: (res: any) => void
+ onError?: (...args: any[]) => void
+ onClose?: (...args: any[]) => void
+ ctrl: AbortController
+}) => {
+ // return request.post({ url: '/ai/write/generate-stream', data })
+ const token = getAccessToken()
+ return fetchEventSource(`${config.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
+ })
+}
diff --git a/src/views/ai/writer/components/Left.vue b/src/views/ai/writer/components/Left.vue
new file mode 100644
index 00000000..af808b15
--- /dev/null
+++ b/src/views/ai/writer/components/Left.vue
@@ -0,0 +1,195 @@
+
+
+
+
+ {{ text }}
+
+
+
+
+
+ {{ label }}
+
+
+ {{ hint }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 重置
+ 生成
+
+
+
+
+
+
+
diff --git a/src/views/ai/writer/components/Right.vue b/src/views/ai/writer/components/Right.vue
new file mode 100644
index 00000000..30b74cdc
--- /dev/null
+++ b/src/views/ai/writer/components/Right.vue
@@ -0,0 +1,86 @@
+
+
+
+
+
+ 复制
+
+
+
+ 终止生成
+
+
+
+
+
+
+
+
+
diff --git a/src/views/ai/writer/components/Tag.vue b/src/views/ai/writer/components/Tag.vue
new file mode 100644
index 00000000..4a32e572
--- /dev/null
+++ b/src/views/ai/writer/components/Tag.vue
@@ -0,0 +1,32 @@
+
+
+
+ {{ tag.label }}
+
+
+
+
+
+
+
diff --git a/src/views/ai/writer/data.json b/src/views/ai/writer/data.json
new file mode 100644
index 00000000..3f20951b
--- /dev/null
+++ b/src/views/ai/writer/data.json
@@ -0,0 +1,11 @@
+{
+ "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谢谢。"
+ }
+}
diff --git a/src/views/ai/writer/index.vue b/src/views/ai/writer/index.vue
new file mode 100644
index 00000000..8875830f
--- /dev/null
+++ b/src/views/ai/writer/index.vue
@@ -0,0 +1,61 @@
+
+
+
+
+