update
This commit is contained in:
parent
319d21c0c8
commit
69cc3703a4
@ -3,7 +3,7 @@ services:
|
||||
build:
|
||||
context: .
|
||||
dockerfile: Dockerfile
|
||||
image: tradus-web:1.4.7
|
||||
image: tradus-web:1.4.9
|
||||
container_name: tradus-web
|
||||
ports:
|
||||
- '6000:80'
|
||||
|
||||
@ -53,6 +53,15 @@ const router = createRouter({
|
||||
title: 'AI 交易智能体',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/quick-analysis',
|
||||
name: 'quick-analysis',
|
||||
component: () => import('../views/QuickAnalysisView.vue'),
|
||||
meta: {
|
||||
requiresAuth: true,
|
||||
title: '快速行情分析',
|
||||
},
|
||||
},
|
||||
{
|
||||
path: '/subscription',
|
||||
name: 'subscription',
|
||||
|
||||
@ -771,13 +771,45 @@ const stopStreaming = async () => {
|
||||
<span class="quick-icon">Ξ</span>
|
||||
<span>ETH</span>
|
||||
</button>
|
||||
<button
|
||||
class="quick-btn"
|
||||
@click="sendExampleMessage('分析SOL行情')"
|
||||
:disabled="isLoading"
|
||||
>
|
||||
<span class="quick-icon">◎</span>
|
||||
<span>SOL</span>
|
||||
</button>
|
||||
<button
|
||||
class="quick-btn"
|
||||
@click="sendExampleMessage('分析DOGE行情')"
|
||||
:disabled="isLoading"
|
||||
>
|
||||
<span class="quick-icon">Ð</span>
|
||||
<span>DOGE</span>
|
||||
</button>
|
||||
<button
|
||||
class="quick-btn"
|
||||
@click="sendExampleMessage('分析BNB行情')"
|
||||
:disabled="isLoading"
|
||||
>
|
||||
<span class="quick-icon">🔶</span>
|
||||
<span>BNB</span>
|
||||
</button>
|
||||
<button
|
||||
class="quick-btn"
|
||||
@click="sendExampleMessage('分析ADA行情')"
|
||||
:disabled="isLoading"
|
||||
>
|
||||
<span class="quick-icon">♠</span>
|
||||
<span>ADA</span>
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div class="input-container">
|
||||
<textarea
|
||||
v-model="messageInput"
|
||||
class="message-input"
|
||||
placeholder="输入您的问题,例如:分析BTC、分析ETH、分析加密货币行情"
|
||||
placeholder="输入您的问题,例如:分析BTC、分析ETH、分析SOL、分析DOGE等加密货币行情"
|
||||
@keydown="handleKeydown"
|
||||
:disabled="isLoading"
|
||||
rows="1"
|
||||
|
||||
684
src/views/QuickAnalysisView.vue
Normal file
684
src/views/QuickAnalysisView.vue
Normal file
@ -0,0 +1,684 @@
|
||||
<script setup lang="ts">
|
||||
import { ref, computed, onMounted } from 'vue'
|
||||
import { useUserStore } from '../stores/user'
|
||||
import { http } from '../services/api'
|
||||
import { marked } from 'marked'
|
||||
import { useRouter } from 'vue-router'
|
||||
|
||||
const router = useRouter()
|
||||
const userStore = useUserStore()
|
||||
const userInfo = computed(() => userStore.userInfo)
|
||||
|
||||
// 主流加密货币配置
|
||||
const cryptoCoins = [
|
||||
{ symbol: 'BTC', name: 'Bitcoin', icon: '₿', color: '#f7931a' },
|
||||
{ symbol: 'ETH', name: 'Ethereum', icon: 'Ξ', color: '#627eea' },
|
||||
{ symbol: 'SOL', name: 'Solana', icon: '◎', color: '#9945ff' },
|
||||
{ symbol: 'BNB', name: 'BNB', icon: '🔶', color: '#f3ba2f' },
|
||||
{ symbol: 'DOGE', name: 'Dogecoin', icon: 'Ð', color: '#c2a633' },
|
||||
{ symbol: 'ADA', name: 'Cardano', icon: '♠', color: '#0033ad' },
|
||||
{ symbol: 'XRP', name: 'XRP', icon: '◈', color: '#23292f' },
|
||||
{ symbol: 'MATIC', name: 'Polygon', icon: '⬢', color: '#8247e5' },
|
||||
]
|
||||
|
||||
// 状态管理
|
||||
const isAnalyzing = ref(false)
|
||||
const currentAnalyzing = ref('')
|
||||
const analysisResults = ref<{ [key: string]: string }>({})
|
||||
const recentAnalysis = ref<Array<{ symbol: string; timestamp: Date; preview: string }>>([])
|
||||
|
||||
// API基础URL
|
||||
const apiBaseUrl =
|
||||
import.meta.env.MODE === 'development' ? 'http://127.0.0.1:8000' : 'https://api.ibtc.work'
|
||||
|
||||
// 配置marked选项
|
||||
onMounted(() => {
|
||||
marked.setOptions({
|
||||
breaks: true,
|
||||
gfm: true,
|
||||
})
|
||||
})
|
||||
|
||||
// 一键快速分析
|
||||
const quickAnalyze = async (symbol: string, name: string) => {
|
||||
if (isAnalyzing.value) return
|
||||
|
||||
isAnalyzing.value = true
|
||||
currentAnalyzing.value = symbol
|
||||
|
||||
const message = `分析${symbol}行情`
|
||||
|
||||
try {
|
||||
const response = await http.post(`${apiBaseUrl}/analysis/chat-messages`, {
|
||||
message: message,
|
||||
})
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error('分析请求失败')
|
||||
}
|
||||
|
||||
const reader = response.body?.getReader()
|
||||
if (!reader) {
|
||||
throw new Error('无法获取响应流')
|
||||
}
|
||||
|
||||
const decoder = new TextDecoder()
|
||||
let buffer = ''
|
||||
let fullContent = ''
|
||||
|
||||
while (true) {
|
||||
const { done, value } = await reader.read()
|
||||
if (done) break
|
||||
|
||||
const chunk = decoder.decode(value, { stream: true })
|
||||
buffer += chunk
|
||||
|
||||
const lines = buffer.split('\n')
|
||||
buffer = lines.pop() || ''
|
||||
|
||||
for (const line of lines) {
|
||||
if (!line.trim() || !line.startsWith('data: ')) continue
|
||||
|
||||
try {
|
||||
const jsonStr = line.slice(6)
|
||||
const data = JSON.parse(jsonStr)
|
||||
|
||||
if (data.event === 'text_chunk' && data.data?.text) {
|
||||
fullContent += data.data.text
|
||||
analysisResults.value[symbol] = fullContent
|
||||
} else if (data.event === 'message' && data.answer) {
|
||||
fullContent += data.answer
|
||||
analysisResults.value[symbol] = fullContent
|
||||
} else if (data.event === 'workflow_finished') {
|
||||
if (data.data?.outputs?.text) {
|
||||
fullContent = data.data.outputs.text
|
||||
analysisResults.value[symbol] = fullContent
|
||||
}
|
||||
|
||||
// 添加到最近分析记录
|
||||
const preview = fullContent.substring(0, 100) + '...'
|
||||
recentAnalysis.value.unshift({
|
||||
symbol,
|
||||
timestamp: new Date(),
|
||||
preview
|
||||
})
|
||||
|
||||
// 只保留最近10条记录
|
||||
if (recentAnalysis.value.length > 10) {
|
||||
recentAnalysis.value = recentAnalysis.value.slice(0, 10)
|
||||
}
|
||||
|
||||
break
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('解析响应数据出错:', error)
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('分析失败:', error)
|
||||
analysisResults.value[symbol] = '分析失败,请稍后重试'
|
||||
} finally {
|
||||
isAnalyzing.value = false
|
||||
currentAnalyzing.value = ''
|
||||
}
|
||||
}
|
||||
|
||||
// 解析markdown
|
||||
const parseMarkdown = (content: string) => {
|
||||
if (!content) return ''
|
||||
try {
|
||||
return marked(content) as string
|
||||
} catch (error) {
|
||||
return content.replace(/\n/g, '<br>')
|
||||
}
|
||||
}
|
||||
|
||||
// 跳转到详细聊天页面
|
||||
const goToDetailedChat = (symbol: string) => {
|
||||
router.push({
|
||||
path: '/ai-agent',
|
||||
query: { symbol }
|
||||
})
|
||||
}
|
||||
|
||||
// 格式化时间
|
||||
const formatTime = (date: Date) => {
|
||||
return date.toLocaleString('zh-CN', {
|
||||
month: '2-digit',
|
||||
day: '2-digit',
|
||||
hour: '2-digit',
|
||||
minute: '2-digit'
|
||||
})
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="quick-analysis-view">
|
||||
<!-- 页面标题 -->
|
||||
<div class="page-header">
|
||||
<h1 class="page-title">快速行情分析</h1>
|
||||
<p class="page-subtitle">一键获取AI加密货币行情分析报告</p>
|
||||
</div>
|
||||
|
||||
<!-- 使用限制提示 -->
|
||||
<div v-if="!userInfo?.is_subscribed" class="usage-notice">
|
||||
<div class="notice-icon">💡</div>
|
||||
<div class="notice-content">
|
||||
<span class="notice-text">非订阅用户每天只有 1 次体验分析,</span>
|
||||
<router-link to="/subscription" class="notice-link">订阅后解锁无限次分析</router-link>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 加密货币网格 -->
|
||||
<div class="crypto-grid">
|
||||
<div
|
||||
v-for="coin in cryptoCoins"
|
||||
:key="coin.symbol"
|
||||
class="crypto-card"
|
||||
:class="{
|
||||
analyzing: currentAnalyzing === coin.symbol,
|
||||
analyzed: analysisResults[coin.symbol]
|
||||
}"
|
||||
>
|
||||
<div class="card-header">
|
||||
<div class="coin-info">
|
||||
<div class="coin-icon" :style="{ color: coin.color }">{{ coin.icon }}</div>
|
||||
<div class="coin-details">
|
||||
<div class="coin-symbol">{{ coin.symbol }}</div>
|
||||
<div class="coin-name">{{ coin.name }}</div>
|
||||
</div>
|
||||
</div>
|
||||
<button
|
||||
class="analyze-btn"
|
||||
@click="quickAnalyze(coin.symbol, coin.name)"
|
||||
:disabled="isAnalyzing"
|
||||
:class="{ active: currentAnalyzing === coin.symbol }"
|
||||
>
|
||||
<div v-if="currentAnalyzing === coin.symbol" class="loading-spinner"></div>
|
||||
<span v-else>{{ analysisResults[coin.symbol] ? '重新分析' : '快速分析' }}</span>
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<!-- 分析结果 -->
|
||||
<div v-if="analysisResults[coin.symbol]" class="analysis-result">
|
||||
<div class="result-header">
|
||||
<h4>分析结果</h4>
|
||||
<button class="detail-btn" @click="goToDetailedChat(coin.symbol)">
|
||||
详细讨论
|
||||
</button>
|
||||
</div>
|
||||
<div
|
||||
class="result-content markdown-content"
|
||||
v-html="parseMarkdown(analysisResults[coin.symbol])"
|
||||
></div>
|
||||
</div>
|
||||
|
||||
<!-- 分析中状态 -->
|
||||
<div v-else-if="currentAnalyzing === coin.symbol" class="analyzing-state">
|
||||
<div class="analyzing-icon">
|
||||
<div class="pulse-dot"></div>
|
||||
</div>
|
||||
<p>AI正在分析 {{ coin.symbol }} 行情...</p>
|
||||
</div>
|
||||
|
||||
<!-- 默认状态 -->
|
||||
<div v-else class="default-state">
|
||||
<div class="placeholder-icon">📊</div>
|
||||
<p>点击"快速分析"获取 {{ coin.symbol }} 行情分析</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 最近分析历史 -->
|
||||
<div v-if="recentAnalysis.length > 0" class="recent-analysis">
|
||||
<h3 class="section-title">最近分析</h3>
|
||||
<div class="analysis-list">
|
||||
<div
|
||||
v-for="(item, index) in recentAnalysis"
|
||||
:key="`${item.symbol}-${index}`"
|
||||
class="analysis-item"
|
||||
@click="goToDetailedChat(item.symbol)"
|
||||
>
|
||||
<div class="item-header">
|
||||
<span class="item-symbol">{{ item.symbol }}</span>
|
||||
<span class="item-time">{{ formatTime(item.timestamp) }}</span>
|
||||
</div>
|
||||
<div class="item-preview">{{ item.preview }}</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.quick-analysis-view {
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
padding: 2rem;
|
||||
min-height: 100vh;
|
||||
background-color: var(--color-bg-primary);
|
||||
}
|
||||
|
||||
.page-header {
|
||||
text-align: center;
|
||||
margin-bottom: 2rem;
|
||||
}
|
||||
|
||||
.page-title {
|
||||
font-size: 2rem;
|
||||
font-weight: 700;
|
||||
color: var(--color-text-primary);
|
||||
margin-bottom: 0.5rem;
|
||||
background: linear-gradient(135deg, var(--color-accent) 0%, #667eea 100%);
|
||||
-webkit-background-clip: text;
|
||||
-webkit-text-fill-color: transparent;
|
||||
background-clip: text;
|
||||
}
|
||||
|
||||
.page-subtitle {
|
||||
color: var(--color-text-secondary);
|
||||
font-size: 1.1rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* 使用限制提示 */
|
||||
.usage-notice {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 0.75rem;
|
||||
padding: 1rem 1.5rem;
|
||||
background: linear-gradient(135deg, #fff3cd, #ffeaa7);
|
||||
border: 1px solid #ffd60a;
|
||||
border-radius: 12px;
|
||||
margin-bottom: 2rem;
|
||||
box-shadow: 0 2px 8px rgba(255, 214, 10, 0.1);
|
||||
}
|
||||
|
||||
.notice-icon {
|
||||
font-size: 1.5rem;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.notice-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 0.25rem;
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.notice-text {
|
||||
font-size: 0.95rem;
|
||||
color: #856404;
|
||||
line-height: 1.4;
|
||||
}
|
||||
|
||||
.notice-link {
|
||||
font-size: 0.95rem;
|
||||
color: #0066cc;
|
||||
text-decoration: none;
|
||||
font-weight: 600;
|
||||
transition: color 0.2s ease;
|
||||
}
|
||||
|
||||
.notice-link:hover {
|
||||
color: #004080;
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
/* 加密货币网格 */
|
||||
.crypto-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
|
||||
gap: 1.5rem;
|
||||
margin-bottom: 3rem;
|
||||
}
|
||||
|
||||
.crypto-card {
|
||||
background: var(--color-bg-secondary);
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: 16px;
|
||||
padding: 1.5rem;
|
||||
transition: all 0.3s ease;
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.crypto-card:hover {
|
||||
transform: translateY(-2px);
|
||||
box-shadow: 0 8px 25px rgba(0, 0, 0, 0.1);
|
||||
border-color: var(--color-accent);
|
||||
}
|
||||
|
||||
.crypto-card.analyzing {
|
||||
border-color: var(--color-accent);
|
||||
background: rgba(51, 85, 255, 0.02);
|
||||
}
|
||||
|
||||
.crypto-card.analyzed {
|
||||
background: var(--color-bg-primary);
|
||||
}
|
||||
|
||||
.card-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.coin-info {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 1rem;
|
||||
}
|
||||
|
||||
.coin-icon {
|
||||
font-size: 2rem;
|
||||
font-weight: bold;
|
||||
width: 50px;
|
||||
height: 50px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
background: rgba(255, 255, 255, 0.1);
|
||||
border-radius: 12px;
|
||||
}
|
||||
|
||||
.coin-symbol {
|
||||
font-size: 1.2rem;
|
||||
font-weight: 700;
|
||||
color: var(--color-text-primary);
|
||||
}
|
||||
|
||||
.coin-name {
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-text-secondary);
|
||||
}
|
||||
|
||||
.analyze-btn {
|
||||
padding: 0.75rem 1.5rem;
|
||||
border: none;
|
||||
border-radius: 8px;
|
||||
background: var(--color-accent);
|
||||
color: white;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
transition: all 0.2s ease;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 0.5rem;
|
||||
min-width: 100px;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.analyze-btn:hover:not(:disabled) {
|
||||
background: var(--color-accent-hover);
|
||||
transform: translateY(-1px);
|
||||
}
|
||||
|
||||
.analyze-btn:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
.analyze-btn.active {
|
||||
background: #ff6b35;
|
||||
}
|
||||
|
||||
.loading-spinner {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
border: 2px solid rgba(255, 255, 255, 0.3);
|
||||
border-top-color: white;
|
||||
border-radius: 50%;
|
||||
animation: spin 1s linear infinite;
|
||||
}
|
||||
|
||||
@keyframes spin {
|
||||
to { transform: rotate(360deg); }
|
||||
}
|
||||
|
||||
/* 分析结果 */
|
||||
.analysis-result {
|
||||
margin-top: 1rem;
|
||||
padding-top: 1rem;
|
||||
border-top: 1px solid var(--color-border);
|
||||
}
|
||||
|
||||
.result-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.result-header h4 {
|
||||
margin: 0;
|
||||
color: var(--color-text-primary);
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.detail-btn {
|
||||
padding: 0.4rem 0.8rem;
|
||||
border: 1px solid var(--color-accent);
|
||||
border-radius: 6px;
|
||||
background: transparent;
|
||||
color: var(--color-accent);
|
||||
font-size: 0.85rem;
|
||||
cursor: pointer;
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.detail-btn:hover {
|
||||
background: var(--color-accent);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.result-content {
|
||||
max-height: 300px;
|
||||
overflow-y: auto;
|
||||
line-height: 1.6;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
/* 分析中状态 */
|
||||
.analyzing-state {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
text-align: center;
|
||||
padding: 2rem;
|
||||
}
|
||||
|
||||
.analyzing-icon {
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.pulse-dot {
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
background: var(--color-accent);
|
||||
border-radius: 50%;
|
||||
animation: pulse 1.5s ease-in-out infinite;
|
||||
}
|
||||
|
||||
@keyframes pulse {
|
||||
0% {
|
||||
transform: scale(0.95);
|
||||
opacity: 0.7;
|
||||
}
|
||||
70% {
|
||||
transform: scale(1);
|
||||
opacity: 1;
|
||||
}
|
||||
100% {
|
||||
transform: scale(0.95);
|
||||
opacity: 0.7;
|
||||
}
|
||||
}
|
||||
|
||||
/* 默认状态 */
|
||||
.default-state {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
text-align: center;
|
||||
padding: 2rem;
|
||||
color: var(--color-text-secondary);
|
||||
}
|
||||
|
||||
.placeholder-icon {
|
||||
font-size: 3rem;
|
||||
margin-bottom: 1rem;
|
||||
opacity: 0.5;
|
||||
}
|
||||
|
||||
/* 最近分析 */
|
||||
.recent-analysis {
|
||||
margin-top: 3rem;
|
||||
}
|
||||
|
||||
.section-title {
|
||||
font-size: 1.5rem;
|
||||
font-weight: 600;
|
||||
color: var(--color-text-primary);
|
||||
margin-bottom: 1rem;
|
||||
}
|
||||
|
||||
.analysis-list {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 0.75rem;
|
||||
}
|
||||
|
||||
.analysis-item {
|
||||
padding: 1rem 1.5rem;
|
||||
background: var(--color-bg-secondary);
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: 12px;
|
||||
cursor: pointer;
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.analysis-item:hover {
|
||||
border-color: var(--color-accent);
|
||||
background: var(--color-bg-primary);
|
||||
transform: translateY(-1px);
|
||||
}
|
||||
|
||||
.item-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 0.5rem;
|
||||
}
|
||||
|
||||
.item-symbol {
|
||||
font-weight: 600;
|
||||
color: var(--color-accent);
|
||||
}
|
||||
|
||||
.item-time {
|
||||
font-size: 0.85rem;
|
||||
color: var(--color-text-secondary);
|
||||
}
|
||||
|
||||
.item-preview {
|
||||
color: var(--color-text-secondary);
|
||||
font-size: 0.9rem;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* 响应式设计 */
|
||||
@media (max-width: 768px) {
|
||||
.quick-analysis-view {
|
||||
padding: 1rem;
|
||||
}
|
||||
|
||||
.crypto-grid {
|
||||
grid-template-columns: 1fr;
|
||||
gap: 1rem;
|
||||
}
|
||||
|
||||
.crypto-card {
|
||||
padding: 1rem;
|
||||
}
|
||||
|
||||
.page-title {
|
||||
font-size: 1.5rem;
|
||||
}
|
||||
|
||||
.page-subtitle {
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.card-header {
|
||||
flex-direction: column;
|
||||
gap: 1rem;
|
||||
align-items: stretch;
|
||||
}
|
||||
|
||||
.analyze-btn {
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
<style>
|
||||
/* Markdown 样式 */
|
||||
.quick-analysis-view .markdown-content h1,
|
||||
.quick-analysis-view .markdown-content h2,
|
||||
.quick-analysis-view .markdown-content h3,
|
||||
.quick-analysis-view .markdown-content h4 {
|
||||
color: var(--color-text-primary);
|
||||
margin-top: 1rem;
|
||||
margin-bottom: 0.5rem;
|
||||
font-size: 0.95rem;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content p {
|
||||
margin-bottom: 0.75rem;
|
||||
line-height: 1.6;
|
||||
color: var(--color-text-primary);
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content ul,
|
||||
.quick-analysis-view .markdown-content ol {
|
||||
margin-bottom: 0.75rem;
|
||||
padding-left: 1.5rem;
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content li {
|
||||
margin-bottom: 0.25rem;
|
||||
color: var(--color-text-primary);
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content strong {
|
||||
color: var(--color-accent);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content code {
|
||||
background: rgba(0, 0, 0, 0.1);
|
||||
padding: 0.2rem 0.4rem;
|
||||
border-radius: 4px;
|
||||
font-size: 0.85rem;
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content pre {
|
||||
background: rgba(0, 0, 0, 0.05);
|
||||
padding: 0.75rem;
|
||||
border-radius: 8px;
|
||||
overflow-x: auto;
|
||||
margin-bottom: 0.75rem;
|
||||
}
|
||||
|
||||
.quick-analysis-view .markdown-content blockquote {
|
||||
border-left: 4px solid var(--color-accent);
|
||||
padding-left: 1rem;
|
||||
margin: 0.75rem 0;
|
||||
color: var(--color-text-secondary);
|
||||
}
|
||||
</style>
|
||||
Loading…
Reference in New Issue
Block a user