Luxx/dashboard/src/views/ConversationDetailView.vue

461 lines
11 KiB
Vue

<template>
<div class="chat-view main-panel">
<div v-if="!conversationId" class="welcome">
<div class="welcome-icon">
<svg viewBox="0 0 64 64" width="36" height="36">
<rect width="64" height="64" rx="14" fill="url(#favBg)"/>
<defs>
<linearGradient id="favBg" x1="0%" y1="0%" x2="100%" y2="100%">
<stop offset="0%" stop-color="#2563eb"/>
<stop offset="100%" stop-color="#60a5fa"/>
</linearGradient>
</defs>
<text x="32" y="40" text-anchor="middle" font-family="-apple-system,BlinkMacSystemFont,sans-serif" font-size="18" font-weight="800" fill="#fff" letter-spacing="-0.5">Luxx</text>
</svg>
</div>
<h1>Chat</h1>
<p>选择一个对话开始,或创建新对话</p>
</div>
<template v-else>
<div class="chat-header">
<div class="chat-title-area">
<h2 class="chat-title">{{ conversationTitle || '新对话' }}</h2>
</div>
</div>
<div ref="messagesContainer" class="messages-container" @scroll="handleScroll">
<div v-if="loading" class="load-more-top">
<span>加载中...</span>
</div>
<div class="messages-list">
<div
v-for="msg in messages"
:key="msg.id"
:data-msg-id="msg.id"
>
<MessageBubble
:role="msg.role"
:text="msg.text || msg.content"
:tool-calls="msg.tool_calls"
:process-steps="msg.process_steps"
:token-count="msg.token_count"
:created-at="msg.created_at"
:deletable="msg.role === 'user'"
:attachments="msg.attachments"
@delete="deleteMessage(msg.id)"
/>
</div>
<!-- 流式消息 -->
<div v-if="streamingMessage" class="message-bubble assistant streaming">
<div class="avatar">Luxx</div>
<div class="message-body">
<ProcessBlock
:process-steps="streamingMessage.process_steps"
:streaming="true"
/>
</div>
</div>
</div>
</div>
<div class="message-input">
<div class="input-container">
<textarea
ref="textareaRef"
v-model="inputMessage"
:placeholder="sending ? 'AI 正在回复中...' : '输入消息... (Shift+Enter 换行)'"
rows="1"
@input="autoResize"
@keydown="onKeydown"
></textarea>
<div class="input-footer">
<div class="input-actions">
<button
class="btn-send"
:class="{ active: canSend }"
:disabled="!canSend || sending"
@click="sendMessage"
>
<span v-html="sendIcon"></span>
</button>
</div>
</div>
</div>
<div class="input-hint">AI 助手回复内容仅供参考</div>
</div>
</template>
</div>
</template>
<script setup>
import { ref, computed, onMounted, nextTick, watch } from 'vue'
import { useRoute } from 'vue-router'
import { conversationsAPI, messagesAPI } from '../services/api.js'
import ProcessBlock from '../components/ProcessBlock.vue'
import MessageBubble from '../components/MessageBubble.vue'
import { renderMarkdown } from '../utils/markdown.js'
const route = useRoute()
const messages = ref([])
const inputMessage = ref('')
const loading = ref(true)
const sending = ref(false)
const streamingMessage = ref(null)
const messagesContainer = ref(null)
const textareaRef = ref(null)
const autoScroll = ref(true)
const conversationId = ref(route.params.id)
const conversationTitle = ref('')
const canSend = computed(() => inputMessage.value.trim().length > 0)
const sendIcon = `<svg viewBox="0 0 24 24" width="18" height="18" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="22" y1="2" x2="11" y2="13"></line><polygon points="22 2 15 22 11 13 2 9 22 2"></polygon></svg>`
function autoResize() {
const el = textareaRef.value
if (!el) return
el.style.height = 'auto'
el.style.height = Math.min(el.scrollHeight, 200) + 'px'
}
function onKeydown(e) {
if (e.key === 'Enter' && !e.shiftKey) {
e.preventDefault()
sendMessage()
}
}
const loadMessages = async () => {
autoScroll.value = true
loading.value = true
try {
const res = await messagesAPI.list(conversationId.value)
if (res.success) {
messages.value = res.data.messages || []
if (messages.value.length > 0) {
conversationTitle.value = res.data.title || ''
}
}
} catch (e) {
console.error(e)
} finally {
loading.value = false
scrollToBottom()
}
}
const deleteMessage = async (msgId) => {
try {
await messagesAPI.delete(msgId)
messages.value = messages.value.filter(m => m.id !== msgId)
} catch (e) {
console.error(e)
}
}
const sendMessage = async () => {
if (!inputMessage.value.trim() || sending.value) return
const content = inputMessage.value.trim()
inputMessage.value = ''
sending.value = true
// 清空输入框
nextTick(() => {
autoResize()
})
// 添加用户消息
messages.value.push({
id: Date.now(),
role: 'user',
content: content,
text: content,
attachments: [],
process_steps: [],
created_at: new Date().toISOString()
})
scrollToBottom()
// 初始化流式消息
streamingMessage.value = {
id: Date.now() + 1,
role: 'assistant',
process_steps: [],
created_at: new Date().toISOString()
}
// SSE 流式请求
messagesAPI.sendStream(
{ conversation_id: conversationId.value, content },
{
onProcessStep: (step) => {
autoScroll.value = true // 流式开始时启用自动滚动
if (!streamingMessage.value) return
// 按 id 更新或追加步骤
const idx = streamingMessage.value.process_steps.findIndex(s => s.id === step.id)
if (idx >= 0) {
streamingMessage.value.process_steps[idx] = step
} else {
streamingMessage.value.process_steps.push(step)
}
},
onDone: () => {
// 完成,添加到消息列表
autoScroll.value = true
if (streamingMessage.value) {
messages.value.push({
...streamingMessage.value,
created_at: new Date().toISOString()
})
streamingMessage.value = null
}
sending.value = false
},
onError: (error) => {
console.error('Stream error:', error)
if (streamingMessage.value) {
streamingMessage.value.process_steps.push({
id: 'error-' + Date.now(),
index: streamingMessage.value.process_steps.length,
type: 'text',
content: `[错误] ${error}`
})
}
sending.value = false
}
}
)
scrollToBottom()
}
const scrollToBottom = () => {
if (!autoScroll.value) return
nextTick(() => {
if (messagesContainer.value) {
messagesContainer.value.scrollTo({
top: messagesContainer.value.scrollHeight,
behavior: streamingMessage.value ? 'instant' : 'smooth'
})
}
})
}
// 处理滚动事件,检测用户是否手动滚动
const handleScroll = () => {
if (!messagesContainer.value) return
const { scrollTop, scrollHeight, clientHeight } = messagesContainer.value
const distanceToBottom = scrollHeight - scrollTop - clientHeight
// 距离底部超过50px时停止自动跟随
autoScroll.value = distanceToBottom < 50
}
// 监听流式消息变化,自动滚动
watch(() => streamingMessage.value?.process_steps?.length, () => {
if (streamingMessage.value) {
scrollToBottom()
}
})
const formatTime = (time) => {
if (!time) return ''
return new Date(time).toLocaleTimeString('zh-CN', { hour: '2-digit', minute: '2-digit' })
}
onMounted(loadMessages)
</script>
<style scoped>
.chat-view {
flex: 1 1 0;
display: flex;
flex-direction: column;
height: 100vh;
overflow: hidden;
min-width: 0;
}
.welcome {
flex: 1;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
color: var(--text-tertiary);
}
.welcome-icon {
width: 64px;
height: 64px;
border-radius: 16px;
display: flex;
align-items: center;
justify-content: center;
margin-bottom: 20px;
overflow: hidden;
}
.welcome h1 {
font-size: 24px;
color: var(--text-primary);
margin: 0 0 8px;
}
.welcome p {
font-size: 14px;
}
.chat-header {
display: flex;
align-items: center;
justify-content: space-between;
padding: 12px 24px;
border-bottom: 1px solid var(--border-light);
background: color-mix(in srgb, var(--bg-primary) 70%, transparent);
backdrop-filter: blur(40px);
-webkit-backdrop-filter: blur(40px);
transition: background 0.2s, border-color 0.2s;
}
.chat-title-area {
display: flex;
align-items: center;
gap: 8px;
min-width: 0;
}
.chat-title {
font-size: 16px;
font-weight: 600;
color: var(--text-primary);
margin: 0;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.messages-container {
flex: 1 1 auto;
overflow-y: auto;
padding: 16px 0;
width: 100%;
display: flex;
flex-direction: column;
scrollbar-width: none;
-ms-overflow-style: none;
}
.messages-container::-webkit-scrollbar {
display: none;
}
.load-more-top {
text-align: center;
padding: 12px 0;
color: var(--text-tertiary);
font-size: 13px;
}
.messages-list {
width: 80%;
margin: 0 auto;
padding: 0 16px;
}
/* Message Input */
.message-input {
padding: 16px 24px 12px;
background: var(--bg-primary);
border-top: 1px solid var(--border-light);
transition: background 0.2s, border-color 0.2s;
}
.input-container {
display: flex;
flex-direction: column;
background: var(--bg-input);
border: 1px solid var(--border-input);
border-radius: 12px;
padding: 12px;
transition: border-color 0.2s, background 0.2s;
}
.input-container:focus-within {
border-color: var(--accent-primary);
}
textarea {
width: 100%;
background: none;
border: none;
color: var(--text-primary);
font-size: 15px;
line-height: 1.6;
resize: none;
outline: none;
font-family: inherit;
min-height: 36px;
max-height: 200px;
padding: 0;
}
textarea::placeholder {
color: var(--text-tertiary);
}
.input-footer {
display: flex;
justify-content: flex-end;
padding-top: 8px;
margin-top: 4px;
}
.input-actions {
display: flex;
align-items: center;
gap: 8px;
}
.btn-send {
width: 36px;
height: 36px;
border-radius: 8px;
border: none;
background: var(--bg-code);
color: var(--text-tertiary);
cursor: not-allowed;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.15s ease;
}
.btn-send.active {
background: var(--accent-primary);
color: white;
cursor: pointer;
}
.btn-send.active:hover {
background: var(--accent-primary-hover);
transform: translateY(-1px);
box-shadow: 0 2px 8px rgba(37, 99, 235, 0.3);
}
.btn-send.active:active {
transform: translateY(0);
box-shadow: none;
}
.input-hint {
text-align: center;
font-size: 12px;
color: var(--text-tertiary);
margin-top: 8px;
}
</style>