实现消息级操作栏

本次更新

支持:

  • 复制单条消息
  • 删除单条消息
  • 重新生成上一条 AI 回复

1)改 web/src/App.vue

新增方法

javascript 复制代码
const copyMessage = async content => {
  try {
    await navigator.clipboard.writeText(content || '')
    window.alert('消息已复制')
  } catch (e) {
    console.error(e)
    window.alert('复制失败')
  }
}

const handleDeleteMessage = index => {
  if (!currentSession.value) return

  const nextMessages = currentSession.value.messages.filter((_, i) => i !== index)

  sessions.value = sortSessions(
    sessions.value.map(item =>
      item.id === currentSessionId.value
        ? {
            ...item,
            updatedAt: Date.now(),
            messages: nextMessages,
          }
        : item
    )
  )
}

const handleRegenerate = async index => {
  if (!currentSession.value || loading.value) return

  const messages = [...currentSession.value.messages]
  const current = messages[index]
  const prev = messages[index - 1]

  if (!current || current.role !== 'assistant') return
  if (!prev || prev.role !== 'user') {
    window.alert('只能重新生成上一条 AI 回复')
    return
  }

  const nextMessages = messages.slice(0, index)

  sessions.value = sortSessions(
    sessions.value.map(item =>
      item.id === currentSessionId.value
        ? {
            ...item,
            updatedAt: Date.now(),
            messages: nextMessages,
          }
        : item
    )
  )

  loading.value = true
  try {
    await sendMessageStream(nextMessages)
  } catch (error) {
    updateCurrentSession(session => ({
      ...session,
      updatedAt: Date.now(),
      messages: [
        ...session.messages,
        {
          role: 'assistant',
          content: '重新生成失败,请稍后再试。',
        },
      ],
    }))
    console.error(error)
  } finally {
    loading.value = false
    abortController.value = null
  }
}

改消息模板

ini 复制代码
<div
  v-for="(item, index) in currentMessages"
  :key="index"
  :class="['msg', item.role]"
>
  <div class="role-row">
    <div class="role">
      {{
        item.role === 'user'
          ? '我'
          : item.role === 'assistant'
          ? 'AI'
          : 'system'
      }}
    </div>

    <div v-if="item.role !== 'system'" class="msg-actions">
      <span class="msg-action-btn" @click="copyMessage(item.content)">复制</span>
      <span class="msg-action-btn delete" @click="handleDeleteMessage(index)">删除</span>
      <span
        v-if="item.role === 'assistant'"
        class="msg-action-btn"
        @click="handleRegenerate(index)"
      >
        重新生成
      </span>
    </div>
  </div>

  <div
    v-if="item.role === 'assistant'"
    class="content markdown-body"
    v-html="renderMarkdown(item.content)"
  />
  <div v-else class="content">{{ item.content }}</div>
</div>

新增样式

css 复制代码
.role-row {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
  margin-bottom: 6px;
}

.msg-actions {
  display: flex;
  align-items: center;
  gap: 10px;
  flex-shrink: 0;
}

.msg-action-btn {
  font-size: 12px;
  color: #6b7280;
  cursor: pointer;
  user-select: none;
}

.msg-action-btn:hover {
  color: #111827;
}

.msg-action-btn.delete:hover {
  color: #dc2626;
}

2)验证

复制消息

点任意一条消息右侧的 复制

删除消息

删除 后,这条消息会从当前会话里移除

重新生成

找一条 AI 回复,点 重新生成

它会:

  1. 删除当前这条 AI 回复
  2. 保留前一条用户消息
  3. 重新走流式生成

nice 这一节非常简单

本节仓库提交地址

github.com/fhj414/ai-c...

相关推荐
ZC跨境爬虫20 小时前
3D 地球卫星轨道可视化平台开发 Day5(简介接口对接+规划AI自动化卫星数据生成工作流)
前端·人工智能·3d·ai·自动化
毛骗导演20 小时前
Claude Code Agent 实现原理深度剖析
前端·架构
木卫二号Coding20 小时前
第八十四篇-V100-32G+Easyclaw+Ollama+Qwopus3.5-27B-V3
人工智能
Rust研习社20 小时前
Rust 写时克隆智能指针 Cow
后端·rust·编程语言
xiaoxiang960920 小时前
TDD测试驱动开发:从理论到实战的完整指南(含AI增强工作流)
人工智能·驱动开发·tdd
星晨雪海20 小时前
若依框架原有页面功能进行了点位管理模块完整改造(3)
开发语言·前端·javascript
董董灿是个攻城狮20 小时前
库克不再担任苹果 CEO,附全员信
后端
小张同学82420 小时前
Python 封神技巧:1 行代码搞定 90% 日常数据处理,效率直接拉满
开发语言·人工智能·python
伞伞悦读20 小时前
Docker 安装 Redis 教程(重点避坑版)
后端
源码之家20 小时前
计算机毕业设计:Python棉花产业数据可视化与预测系统 Django框架 ARIMA算法 数据分析 可视化 爬虫 大数据 大模型(建议收藏)✅
人工智能·python·算法·信息可视化·数据挖掘·django·课程设计