373 lines
9.7 KiB
Vue
373 lines
9.7 KiB
Vue
<template>
|
|
<div class="chat-view">
|
|
<div v-if="!conversation" 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">claw</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">{{ conversation.title || '新对话' }}</h2>
|
|
<span class="model-badge">{{ formatModelName(conversation.model) }}</span>
|
|
<span v-if="conversation.thinking_enabled" class="thinking-badge">思考</span>
|
|
</div>
|
|
<div class="chat-actions">
|
|
<button class="btn-icon" @click="$emit('toggleStats')" title="使用统计">
|
|
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
<path d="M18 20V10"/>
|
|
<path d="M12 20V4"/>
|
|
<path d="M6 20v-6"/>
|
|
</svg>
|
|
</button>
|
|
<button class="btn-icon" @click="$emit('toggleSettings')" title="设置">
|
|
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
<circle cx="12" cy="12" r="3"></circle>
|
|
<path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06A1.65 1.65 0 0 0 4.68 15a1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06A1.65 1.65 0 0 0 9 4.68a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"></path>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div ref="scrollContainer" class="messages-container">
|
|
<div v-if="hasMoreMessages" class="load-more-top">
|
|
<button @click="$emit('loadMoreMessages')" :disabled="loadingMore">
|
|
{{ loadingMore ? '加载中...' : '加载更早的消息' }}
|
|
</button>
|
|
</div>
|
|
|
|
<div class="messages-list">
|
|
<MessageBubble
|
|
v-for="msg in messages"
|
|
:key="msg.id"
|
|
:role="msg.role"
|
|
:text="msg.text"
|
|
:thinking-content="msg.thinking"
|
|
: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="$emit('deleteMessage', msg.id)"
|
|
@regenerate="$emit('regenerateMessage', msg.id)"
|
|
/>
|
|
|
|
<div v-if="streaming" class="message-bubble assistant streaming">
|
|
<div class="avatar">claw</div>
|
|
<div class="message-body">
|
|
<ProcessBlock
|
|
:thinking-content="streamingThinking"
|
|
:tool-calls="streamingToolCalls"
|
|
:process-steps="streamingProcessSteps"
|
|
:streaming-content="streamingContent"
|
|
:streaming="streaming"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<MessageInput
|
|
ref="inputRef"
|
|
:disabled="streaming"
|
|
:tools-enabled="toolsEnabled"
|
|
@send="handleSend"
|
|
@toggle-tools="$emit('toggleTools', $event)"
|
|
/>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, watch, nextTick, onMounted } from 'vue'
|
|
import MessageBubble from './MessageBubble.vue'
|
|
import MessageInput from './MessageInput.vue'
|
|
import ProcessBlock from './ProcessBlock.vue'
|
|
import { modelApi } from '../api'
|
|
|
|
const props = defineProps({
|
|
conversation: { type: Object, default: null },
|
|
messages: { type: Array, required: true },
|
|
streaming: { type: Boolean, default: false },
|
|
streamingContent: { type: String, default: '' },
|
|
streamingThinking: { type: String, default: '' },
|
|
streamingToolCalls: { type: Array, default: () => [] },
|
|
streamingProcessSteps: { type: Array, default: () => [] },
|
|
hasMoreMessages: { type: Boolean, default: false },
|
|
loadingMore: { type: Boolean, default: false },
|
|
toolsEnabled: { type: Boolean, default: true },
|
|
})
|
|
|
|
const emit = defineEmits(['sendMessage', 'deleteMessage', 'regenerateMessage', 'toggleSettings', 'toggleStats', 'loadMoreMessages', 'toggleTools'])
|
|
|
|
const scrollContainer = ref(null)
|
|
const inputRef = ref(null)
|
|
const modelNameMap = ref({})
|
|
|
|
function formatModelName(modelId) {
|
|
return modelNameMap.value[modelId] || modelId
|
|
}
|
|
|
|
onMounted(async () => {
|
|
try {
|
|
const res = await modelApi.getCached()
|
|
const map = {}
|
|
for (const m of res.data) {
|
|
if (m.id && m.name) map[m.id] = m.name
|
|
}
|
|
modelNameMap.value = map
|
|
} catch (e) {
|
|
console.warn('Failed to load model names:', e)
|
|
}
|
|
})
|
|
|
|
function handleSend(data) {
|
|
emit('sendMessage', data)
|
|
}
|
|
|
|
function scrollToBottom(smooth = true) {
|
|
nextTick(() => {
|
|
const el = scrollContainer.value
|
|
if (el) {
|
|
el.scrollTo({ top: el.scrollHeight, behavior: smooth ? 'smooth' : 'instant' })
|
|
}
|
|
})
|
|
}
|
|
|
|
watch(() => props.messages.length, () => {
|
|
scrollToBottom()
|
|
})
|
|
|
|
watch(() => props.streamingContent, () => {
|
|
scrollToBottom()
|
|
})
|
|
|
|
watch(() => props.conversation?.id, () => {
|
|
if (props.conversation) {
|
|
nextTick(() => inputRef.value?.focus())
|
|
}
|
|
})
|
|
|
|
defineExpose({ scrollToBottom })
|
|
</script>
|
|
|
|
<style scoped>
|
|
.chat-view {
|
|
flex: 1 1 0;
|
|
display: flex;
|
|
flex-direction: column;
|
|
height: 100vh;
|
|
background: var(--bg-secondary);
|
|
min-width: 0;
|
|
overflow: hidden;
|
|
}
|
|
|
|
.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: var(--bg-primary);
|
|
backdrop-filter: blur(8px);
|
|
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;
|
|
}
|
|
|
|
.badge {
|
|
font-size: 11px;
|
|
padding: 2px 8px;
|
|
border-radius: 10px;
|
|
flex-shrink: 0;
|
|
}
|
|
|
|
.model-badge {
|
|
background: var(--accent-primary-medium);
|
|
color: var(--accent-primary);
|
|
font-size: 13px;
|
|
padding: 4px 12px;
|
|
border-radius: 12px;
|
|
font-weight: 500;
|
|
flex-shrink: 0;
|
|
display: inline-flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
line-height: 1;
|
|
}
|
|
|
|
.thinking-badge {
|
|
background: var(--success-bg);
|
|
color: var(--success-color);
|
|
}
|
|
|
|
.chat-actions {
|
|
display: flex;
|
|
gap: 4px;
|
|
}
|
|
|
|
.btn-icon {
|
|
width: 36px;
|
|
height: 36px;
|
|
border-radius: 8px;
|
|
border: none;
|
|
background: none;
|
|
color: var(--text-tertiary);
|
|
cursor: pointer;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
transition: all 0.15s;
|
|
}
|
|
|
|
.btn-icon:hover {
|
|
background: var(--bg-hover);
|
|
color: var(--accent-primary);
|
|
}
|
|
|
|
.messages-container {
|
|
flex: 1 1 auto;
|
|
overflow-y: auto;
|
|
padding: 16px 0;
|
|
width: 100%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
}
|
|
|
|
.messages-container::-webkit-scrollbar {
|
|
width: 6px;
|
|
}
|
|
|
|
.messages-container::-webkit-scrollbar-thumb {
|
|
background: var(--scrollbar-thumb);
|
|
border-radius: 3px;
|
|
}
|
|
|
|
.messages-container::-webkit-scrollbar-thumb:hover {
|
|
background: var(--text-tertiary);
|
|
}
|
|
|
|
.load-more-top {
|
|
text-align: center;
|
|
padding: 12px 0;
|
|
}
|
|
|
|
.load-more-top button {
|
|
background: none;
|
|
border: 1px solid var(--border-medium);
|
|
color: var(--text-secondary);
|
|
padding: 6px 16px;
|
|
border-radius: 16px;
|
|
cursor: pointer;
|
|
font-size: 13px;
|
|
transition: all 0.15s;
|
|
}
|
|
|
|
.load-more-top button:hover {
|
|
background: var(--bg-hover);
|
|
color: var(--accent-primary);
|
|
}
|
|
|
|
.messages-list {
|
|
width: 80%;
|
|
margin: 0 auto;
|
|
padding: 0 16px;
|
|
}
|
|
|
|
.message-bubble {
|
|
display: flex;
|
|
gap: 12px;
|
|
margin-bottom: 16px;
|
|
width: 100%;
|
|
}
|
|
|
|
.message-bubble .message-container {
|
|
display: flex;
|
|
flex-direction: column;
|
|
max-width: 85%;
|
|
min-width: 200px;
|
|
}
|
|
|
|
.message-bubble.user .message-container {
|
|
align-items: flex-end;
|
|
}
|
|
|
|
.message-bubble.assistant .message-container {
|
|
align-items: flex-start;
|
|
}
|
|
|
|
.message-bubble .avatar {
|
|
width: 32px;
|
|
height: 32px;
|
|
border-radius: 8px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
font-size: 12px;
|
|
font-weight: 700;
|
|
letter-spacing: -0.3px;
|
|
flex-shrink: 0;
|
|
background: var(--avatar-gradient);
|
|
color: white;
|
|
}
|
|
|
|
.message-bubble .message-body {
|
|
flex: 1;
|
|
min-width: 0;
|
|
overflow: hidden;
|
|
padding: 16px;
|
|
border: 1px solid var(--border-light);
|
|
border-radius: 12px;
|
|
background: var(--bg-primary);
|
|
transition: background 0.2s, border-color 0.2s;
|
|
}
|
|
|
|
</style>
|