style: 删除冗余代码, 优化样式

This commit is contained in:
ViperEkura 2026-04-14 11:16:48 +08:00
parent 54d6034f16
commit 35414d99de
3 changed files with 14 additions and 533 deletions

View File

@ -23,13 +23,7 @@ const routes = [
{
path: '/conversations',
name: 'Conversations',
component: () => import('../views/ConversationsView.vue'),
meta: { requiresAuth: true }
},
{
path: '/conversations/:id',
name: 'ConversationDetail',
component: () => import('../views/ConversationDetailView.vue'),
component: () => import('../views/ConversationView.vue'),
meta: { requiresAuth: true }
},
{

View File

@ -1,526 +0,0 @@
<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"
:usage="msg.usage"
:created-at="msg.created_at"
:deletable="msg.role === 'user'"
:attachments="msg.attachments"
@delete="deleteMessage(msg.id)"
/>
</div>
<!-- 流式消息 - 使用 store 中的状态 -->
<div v-if="currentStreamState" class="message-bubble assistant streaming">
<div class="avatar">Luxx</div>
<div class="message-body">
<ProcessBlock
:process-steps="currentStreamState.process_steps"
:streaming="currentStreamState.status === 'streaming'"
/>
</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, onUnmounted, nextTick, watch } from 'vue'
import { useRoute, useRouter } from 'vue-router'
import { conversationsAPI, messagesAPI, toolsAPI } from '../utils/api.js'
import { streamManager } from '../utils/streamManager.js'
import { useStreamStore } from '../utils/streamStore.js'
import ProcessBlock from '../components/ProcessBlock.vue'
import MessageBubble from '../components/MessageBubble.vue'
const route = useRoute()
const router = useRouter()
const streamStore = useStreamStore()
const messages = ref([])
const inputMessage = ref('')
const loading = ref(true)
const messagesContainer = ref(null)
const textareaRef = ref(null)
const autoScroll = ref(true)
const conversationId = ref(route.params.id)
const conversationTitle = ref('')
const enabledTools = ref([])
const canSend = computed(() => inputMessage.value.trim().length > 0)
// store
const currentStreamState = computed(() => {
return streamStore.getStreamState(conversationId.value)
})
// sending
const sending = computed(() => {
const state = currentStreamState.value
return state && state.status === 'streaming'
})
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) {
if (res.data.title) {
conversationTitle.value = res.data.title
} else if (res.data.first_message) {
conversationTitle.value = res.data.first_message
} else {
const userMsg = messages.value.find(m => m.role === 'user')
if (userMsg) {
conversationTitle.value = userMsg.content.slice(0, 30) + (userMsg.content.length > 30 ? '...' : '')
}
}
}
}
} catch (e) {
console.error(e)
} finally {
loading.value = false
scrollToBottom()
}
}
const loadEnabledTools = async () => {
try {
const res = await toolsAPI.list()
if (res.success) {
const tools = res.data?.tools || []
enabledTools.value = tools.map(t => t.function?.name || t.name)
}
} catch (e) {
console.error('Failed to load tools:', e)
}
}
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 = ''
nextTick(() => {
autoResize()
})
//
const userMsgId = 'user-' + Date.now()
messages.value.push({
id: userMsgId,
role: 'user',
content: content,
text: content,
attachments: [],
process_steps: [],
created_at: new Date().toISOString()
})
scrollToBottom()
// 使 StreamManager
await streamManager.startStream(
conversationId.value,
{
conversation_id: conversationId.value,
content,
enabled_tools: enabledTools.value
},
userMsgId
)
scrollToBottom()
}
const scrollToBottom = () => {
if (!autoScroll.value) return
nextTick(() => {
if (messagesContainer.value) {
messagesContainer.value.scrollTo({
top: messagesContainer.value.scrollHeight,
behavior: 'instant'
})
}
})
}
const handleScroll = () => {
if (!messagesContainer.value) return
const { scrollTop, scrollHeight, clientHeight } = messagesContainer.value
const distanceToBottom = scrollHeight - scrollTop - clientHeight
autoScroll.value = distanceToBottom < 50
}
//
watch(
() => currentStreamState.value?.process_steps?.length,
() => {
if (currentStreamState.value) {
scrollToBottom()
}
}
)
// ID
watch(() => route.params.id, (newId) => {
if (newId) {
conversationId.value = newId
loadMessages()
loadEnabledTools()
// watch
setupStreamWatch()
}
})
//
let unwatchStream = null
const setupStreamWatch = () => {
//
if (unwatchStream) {
unwatchStream()
}
unwatchStream = watch(
() => streamStore.getStreamState(conversationId.value),
(state) => {
if (!state) return
if (state.status === 'done') {
//
autoScroll.value = true
const completedMessage = {
id: state.id,
role: 'assistant',
process_steps: state.process_steps,
token_count: state.token_count,
usage: state.usage,
created_at: new Date().toISOString()
}
messages.value.push(completedMessage)
//
if (!conversationTitle.value || conversationTitle.value === '新对话') {
const userMsg = messages.value.find(m => m.role === 'user')
if (userMsg) {
conversationTitle.value = userMsg.content.slice(0, 30) + (userMsg.content.length > 30 ? '...' : '')
conversationsAPI.update(conversationId.value, { title: conversationTitle.value })
}
}
//
streamStore.clearStream(conversationId.value)
} else if (state.status === 'error') {
//
console.error('Stream error:', state.error)
streamStore.clearStream(conversationId.value)
}
},
{ deep: true }
)
}
onMounted(() => {
loadMessages()
loadEnabledTools()
setupStreamWatch()
})
//
onUnmounted(() => {
if (unwatchStream) {
unwatchStream()
}
})
const formatTime = (time) => {
if (!time) return ''
return new Date(time).toLocaleTimeString('zh-CN', { hour: '2-digit', minute: '2-digit' })
}
</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>

View File

@ -425,6 +425,19 @@ const sendMessage = async () => {
}
convMessages.value.push(userMsg)
// 使
const currentTitle = selectedConv.value?.title
const isDefaultTitle = !currentTitle || currentTitle === 'New Conversation' || currentTitle.trim() === ''
if (isDefaultTitle) {
const title = content.slice(0, 30) + (content.length > 30 ? '...' : '')
selectedConv.value.title = title
//
const conv = list.value.find(c => c.id === selectedConv.value.id)
if (conv) conv.title = title
// API
await conversationsAPI.update(selectedConv.value.id, { title })
}
// 使 StreamManager
await streamManager.startStream(
selectedConv.value.id,