2026-02-03 01:46:03 +08:00
|
|
|
|
<!--
|
|
|
|
|
|
SelectiveContextDemo.vue
|
|
|
|
|
|
选择性上下文保留演示
|
|
|
|
|
|
|
|
|
|
|
|
用途:
|
|
|
|
|
|
展示如何通过 "Pinning" (钉住) 机制来保护关键信息不被滑动窗口移除。
|
|
|
|
|
|
演示 System Prompt 和关键用户指令如何长期保留。
|
|
|
|
|
|
|
|
|
|
|
|
交互功能:
|
|
|
|
|
|
- 发送消息:添加新内容。
|
|
|
|
|
|
- 钉住/取消钉住:手动选择要保留的消息。
|
|
|
|
|
|
- 自动管理:演示当窗口满时,未钉住的消息优先被移除。
|
|
|
|
|
|
-->
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="selective-context-demo">
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div class="control-panel">
|
|
|
|
|
|
<div class="stat-group">
|
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
|
<span class="value">{{ totalMessages }}</span>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<span class="label">现在一共记了几条</span>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="stat-divider">/</div>
|
|
|
|
|
|
<div class="stat-item">
|
|
|
|
|
|
<span class="value">{{ maxSlots }}</span>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<span class="label">黑板最多能记几条</span>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="usage-bar">
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="usage-fill"
|
|
|
|
|
|
:style="{ width: `${(totalMessages / maxSlots) * 100}%` }"
|
|
|
|
|
|
:class="{ full: totalMessages >= maxSlots }"
|
|
|
|
|
|
></div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div class="visualization-area">
|
|
|
|
|
|
<!-- Pinned Section -->
|
|
|
|
|
|
<div class="context-section pinned-section">
|
|
|
|
|
|
<div class="section-header">
|
|
|
|
|
|
<span class="icon">📌</span>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<span class="title">钉住区(永远保留的重要信息)</span>
|
|
|
|
|
|
<span class="count">当前 {{ pinnedMessages.length }} 条</span>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="message-list">
|
|
|
|
|
|
<transition-group name="list">
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-for="msg in pinnedMessages"
|
|
|
|
|
|
:key="msg.id"
|
|
|
|
|
|
class="message-card pinned"
|
|
|
|
|
|
:class="msg.role.toLowerCase()"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="card-header">
|
|
|
|
|
|
<span class="role-badge">{{ msg.role }}</span>
|
|
|
|
|
|
<button
|
|
|
|
|
|
class="pin-btn active"
|
|
|
|
|
|
@click="togglePin(msg)"
|
|
|
|
|
|
:disabled="msg.role === 'System'"
|
2026-02-03 19:41:14 +08:00
|
|
|
|
title="取消钉住"
|
2026-02-03 01:46:03 +08:00
|
|
|
|
>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<span v-if="msg.role === 'System'">🔒 系统信息固定在这</span>
|
|
|
|
|
|
<span v-else>📌 取消钉住</span>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="card-content">{{ msg.content }}</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</transition-group>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<!-- Scrolling Section -->
|
|
|
|
|
|
<div class="context-section scrolling-section">
|
|
|
|
|
|
<div class="section-header">
|
|
|
|
|
|
<span class="icon">📜</span>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<span class="title">会被“挤走”的普通对话(先进先出)</span>
|
|
|
|
|
|
<span class="count">当前 {{ scrollingMessages.length }} 条</span>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div class="message-list">
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<transition-group name="list">
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-for="msg in scrollingMessages"
|
|
|
|
|
|
:key="msg.id"
|
2026-02-03 01:46:03 +08:00
|
|
|
|
class="message-card scrolling"
|
|
|
|
|
|
:class="msg.role.toLowerCase()"
|
2026-01-16 19:10:21 +08:00
|
|
|
|
>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div class="card-header">
|
|
|
|
|
|
<span class="role-badge">{{ msg.role }}</span>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<button class="pin-btn" @click="togglePin(msg)" title="把这条钉在黑板上">
|
|
|
|
|
|
📌 钉住这条
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</button>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div class="card-content">{{ msg.content }}</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</transition-group>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div v-if="scrollingMessages.length === 0" class="empty-state">
|
2026-02-03 19:41:14 +08:00
|
|
|
|
这里是“普通对话区”,暂时还空着
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div class="input-section">
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<div class="input-group">
|
|
|
|
|
|
<input
|
|
|
|
|
|
v-model="newMessage"
|
|
|
|
|
|
@keyup.enter="sendMessage"
|
2026-02-03 19:41:14 +08:00
|
|
|
|
placeholder="在这里输入一条新的信息,比如“我叫小明”"
|
2026-01-16 19:10:21 +08:00
|
|
|
|
/>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<button class="send-btn" @click="sendMessage" :disabled="!newMessage.trim()">
|
2026-02-03 19:41:14 +08:00
|
|
|
|
添加到黑板
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</button>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
<div class="presets">
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<button class="preset-btn" @click="addPreset('我的名字叫 Alice。')">用户:我的名字叫 Alice</button>
|
|
|
|
|
|
<button class="preset-btn" @click="addPreset('系统密码是 1234。')">用户:系统密码是 1234</button>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-02-03 01:46:03 +08:00
|
|
|
|
|
|
|
|
|
|
<div class="info-box">
|
|
|
|
|
|
<p>
|
|
|
|
|
|
<span class="icon">💡</span>
|
2026-02-03 19:41:14 +08:00
|
|
|
|
<strong>说明:</strong>
|
|
|
|
|
|
“选择性保留”就是:重要的就钉在黑板上,普通的让它自己滑走。
|
|
|
|
|
|
系统提示通常会永久钉住,用户的关键信息(比如名字、账号、重要偏好)也可以通过记忆模块或 RAG 钉在这里,避免被新对话挤掉。
|
2026-02-03 01:46:03 +08:00
|
|
|
|
</p>
|
|
|
|
|
|
</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
|
|
|
|
|
import { ref, computed } from 'vue'
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
const maxSlots = 6
|
2026-01-16 19:10:21 +08:00
|
|
|
|
const messages = ref([
|
|
|
|
|
|
{
|
|
|
|
|
|
id: 1,
|
|
|
|
|
|
role: 'System',
|
2026-02-03 01:46:03 +08:00
|
|
|
|
content: 'You are a helpful AI assistant focused on coding.',
|
2026-01-16 19:10:21 +08:00
|
|
|
|
pinned: true
|
|
|
|
|
|
},
|
2026-02-03 01:46:03 +08:00
|
|
|
|
{ id: 2, role: 'User', content: 'Hi, I want to learn Vue.', pinned: false },
|
|
|
|
|
|
{ id: 3, role: 'AI', content: 'Sure! Vue is a progressive framework.', pinned: false }
|
2026-01-16 19:10:21 +08:00
|
|
|
|
])
|
|
|
|
|
|
const newMessage = ref('')
|
|
|
|
|
|
let msgId = 4
|
|
|
|
|
|
|
|
|
|
|
|
const pinnedMessages = computed(() => messages.value.filter((m) => m.pinned))
|
2026-02-03 01:46:03 +08:00
|
|
|
|
const scrollingMessages = computed(() => messages.value.filter((m) => !m.pinned))
|
|
|
|
|
|
const totalMessages = computed(() => messages.value.length)
|
2026-01-16 19:10:21 +08:00
|
|
|
|
|
|
|
|
|
|
const sendMessage = () => {
|
|
|
|
|
|
if (!newMessage.value.trim()) return
|
2026-02-03 01:46:03 +08:00
|
|
|
|
addMessage('User', newMessage.value)
|
2026-01-16 19:10:21 +08:00
|
|
|
|
newMessage.value = ''
|
2026-02-03 01:46:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const addPreset = (text) => {
|
|
|
|
|
|
addMessage('User', text)
|
|
|
|
|
|
}
|
2026-01-16 19:10:21 +08:00
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
const addMessage = (role, content) => {
|
|
|
|
|
|
// If full, remove oldest unpinned message
|
|
|
|
|
|
if (messages.value.length >= maxSlots) {
|
|
|
|
|
|
const firstUnpinnedIndex = messages.value.findIndex(m => !m.pinned)
|
|
|
|
|
|
if (firstUnpinnedIndex !== -1) {
|
|
|
|
|
|
messages.value.splice(firstUnpinnedIndex, 1)
|
2026-01-16 19:10:21 +08:00
|
|
|
|
} else {
|
2026-02-03 01:46:03 +08:00
|
|
|
|
// If all are pinned (rare edge case), we might force remove or block
|
|
|
|
|
|
// For demo, we'll block adding
|
|
|
|
|
|
alert("Context window full of pinned messages! Unpin something first.")
|
|
|
|
|
|
return
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-02-03 01:46:03 +08:00
|
|
|
|
|
|
|
|
|
|
messages.value.push({
|
|
|
|
|
|
id: msgId++,
|
|
|
|
|
|
role,
|
|
|
|
|
|
content,
|
|
|
|
|
|
pinned: false
|
|
|
|
|
|
})
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const togglePin = (msg) => {
|
2026-02-03 01:46:03 +08:00
|
|
|
|
if (msg.role === 'System') return // System prompt is always pinned
|
|
|
|
|
|
|
|
|
|
|
|
// If pinning would exceed capacity (unlikely in this logic but possible if we change rules)
|
|
|
|
|
|
// Logic: Pinning just changes state, doesn't add new msg.
|
2026-01-16 19:10:21 +08:00
|
|
|
|
msg.pinned = !msg.pinned
|
|
|
|
|
|
}
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.selective-context-demo {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
background-color: var(--vp-c-bg-soft);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
margin: 0.5rem 0;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.control-panel {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
gap: 1rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
background: var(--vp-c-bg);
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0.75rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-group {
|
2026-01-16 19:10:21 +08:00
|
|
|
|
display: flex;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
align-items: baseline;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
min-width: 120px;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.stat-item {
|
2026-01-16 19:10:21 +08:00
|
|
|
|
display: flex;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-item .value {
|
|
|
|
|
|
font-size: 1.2rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
font-weight: bold;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-item .label {
|
|
|
|
|
|
font-size: 0.7rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.stat-divider {
|
|
|
|
|
|
font-size: 1.2rem;
|
|
|
|
|
|
color: var(--vp-c-divider);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.usage-bar {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
height: 8px;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.usage-fill {
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
background-color: var(--vp-c-brand);
|
|
|
|
|
|
transition: width 0.3s ease;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.usage-fill.full {
|
|
|
|
|
|
background-color: var(--vp-c-warning-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.visualization-area {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
gap: 0.75rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.context-section {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.pinned-section {
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
background: rgba(var(--vp-c-brand-rgb), 0.02);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.section-header {
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0.4rem 0.8rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
background: var(--vp-c-bg-alt);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
border-bottom: 1px solid var(--vp-c-divider);
|
2026-02-03 01:46:03 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
font-weight: bold;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.pinned-section .section-header {
|
|
|
|
|
|
background: rgba(var(--vp-c-brand-rgb), 0.1);
|
|
|
|
|
|
color: var(--vp-c-brand-dark);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.section-header .count {
|
|
|
|
|
|
margin-left: auto;
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
opacity: 0.7;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.message-list {
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0.5rem;
|
|
|
|
|
|
min-height: 60px;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.message-card {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
border-radius: 6px;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0.5rem;
|
|
|
|
|
|
margin-bottom: 0.5rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
transition: all 0.3s ease;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.message-card:last-child {
|
|
|
|
|
|
margin-bottom: 0;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.message-card.pinned {
|
|
|
|
|
|
border-left: 3px solid var(--vp-c-brand);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.message-card.scrolling {
|
|
|
|
|
|
border-left: 3px solid var(--vp-c-text-3);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.card-header {
|
2026-01-16 19:10:21 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: center;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
margin-bottom: 0.25rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.role-badge {
|
2026-02-03 19:41:14 +08:00
|
|
|
|
font-size: 0.65rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
text-transform: uppercase;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
padding: 2px 6px;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.pin-btn {
|
2026-02-03 01:46:03 +08:00
|
|
|
|
background: transparent;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
border-radius: 4px;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 2px 6px;
|
|
|
|
|
|
font-size: 0.7rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
cursor: pointer;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.pin-btn:hover:not(:disabled) {
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
color: var(--vp-c-brand);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.pin-btn.active {
|
2026-02-03 01:46:03 +08:00
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.pin-btn:disabled {
|
|
|
|
|
|
opacity: 0.5;
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-content {
|
2026-02-03 19:41:14 +08:00
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
line-height: 1.3;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.empty-state {
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
color: var(--vp-c-text-3);
|
|
|
|
|
|
font-style: italic;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
font-size: 0.8rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.input-section {
|
2026-02-03 19:41:14 +08:00
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.input-group {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
input {
|
|
|
|
|
|
flex: 1;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0.5rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-03 01:46:03 +08:00
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
input:focus {
|
|
|
|
|
|
outline: none;
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.send-btn {
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0 1rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
border: none;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
cursor: pointer;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
font-size: 0.9rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.send-btn:disabled {
|
|
|
|
|
|
opacity: 0.5;
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.presets {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.preset-btn {
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
padding: 4px 8px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
border-radius: 4px;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
background: transparent;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.preset-btn:hover {
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
color: var(--vp-c-brand);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.info-box {
|
|
|
|
|
|
background-color: var(--vp-c-bg-alt);
|
2026-02-03 19:41:14 +08:00
|
|
|
|
padding: 0.75rem;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
border-radius: 6px;
|
2026-02-03 19:41:14 +08:00
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
line-height: 1.4;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-03 01:46:03 +08:00
|
|
|
|
.info-box .icon {
|
|
|
|
|
|
margin-right: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Animations */
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.list-enter-active,
|
|
|
|
|
|
.list-leave-active {
|
2026-02-03 01:46:03 +08:00
|
|
|
|
transition: all 0.4s ease;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
.list-enter-from,
|
|
|
|
|
|
.list-leave-to {
|
|
|
|
|
|
opacity: 0;
|
2026-02-03 01:46:03 +08:00
|
|
|
|
transform: scale(0.95);
|
|
|
|
|
|
}
|
|
|
|
|
|
.list-move {
|
|
|
|
|
|
transition: transform 0.4s ease;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
</style>
|