184 lines
4.7 KiB
Vue
184 lines
4.7 KiB
Vue
<template>
|
|
<div class="history-app">
|
|
<div class="history-header">
|
|
<div class="header-left">
|
|
<h2>{{ fundName }} ({{ fundCode }}) 历史数据</h2>
|
|
</div>
|
|
<a class="back-link" @click="goBack">← 返回看板</a>
|
|
</div>
|
|
|
|
<el-table :data="historyList" v-loading="loading" border height="calc(100vh - 200px)" style="width: 100%">
|
|
<el-table-column prop="date" label="价格日期" align="center" min-width="110" />
|
|
<el-table-column prop="price" label="收盘价" align="center" min-width="90" />
|
|
<el-table-column prop="navDate" label="净值日期" align="center" min-width="110" />
|
|
<el-table-column prop="nav" label="净值" align="center" min-width="90" />
|
|
<el-table-column label="溢价率" align="center" min-width="100">
|
|
<template #default="{ row }">
|
|
<span :class="getRateClass(row.premiumRate)">
|
|
{{ formatPremium(row.premiumRate) }}
|
|
</span>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column prop="turnover" label="成交额(万元)" align="center" min-width="120" />
|
|
<el-table-column prop="shareVolume" label="场内份额(万份)" align="center" min-width="130" />
|
|
<el-table-column label="场内新增(万份)" align="center" min-width="120">
|
|
<template #default="{ row }">
|
|
<span :class="getRateClass(row.changeAmount)">
|
|
{{ row.changeAmount != null ? formatChange(row.changeAmount) : '-' }}
|
|
</span>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column label="份额涨幅" align="center" min-width="100">
|
|
<template #default="{ row }">
|
|
<span :class="getRateClass(row.changePct)">
|
|
{{ row.changePct != null ? formatPct(row.changePct) : '-' }}
|
|
</span>
|
|
</template>
|
|
</el-table-column>
|
|
</el-table>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup name="history">
|
|
import { ref, onMounted } from 'vue'
|
|
import { useRoute, useRouter } from 'vue-router'
|
|
import axios from 'axios'
|
|
import { ElMessage } from 'element-plus'
|
|
|
|
const API_URL = 'http://127.0.0.1:8000/api/lof/history'
|
|
|
|
const route = useRoute()
|
|
const router = useRouter()
|
|
|
|
const fundCode = ref(route.query.fundCode || '')
|
|
const fundName = ref(route.query.fundName || '')
|
|
const historyList = ref([])
|
|
const loading = ref(false)
|
|
|
|
function goBack() {
|
|
router.push('/')
|
|
}
|
|
|
|
async function fetchHistory() {
|
|
if (!fundCode.value) {
|
|
ElMessage.error('缺少基金代码参数')
|
|
return
|
|
}
|
|
loading.value = true
|
|
try {
|
|
const res = await axios.get(API_URL, {
|
|
params: { fund_code: fundCode.value, fund_name: fundName.value }
|
|
})
|
|
if (res.data.code === 200) {
|
|
historyList.value = res.data.data
|
|
fundName.value = res.data.fundName || fundName.value
|
|
} else {
|
|
ElMessage.error(res.data.msg || '获取历史数据失败')
|
|
}
|
|
} catch (err) {
|
|
ElMessage.error('请求失败:请确认 Python 后端已启动')
|
|
} finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
|
|
function formatPremium(rate) {
|
|
if (rate == null) return '-'
|
|
const num = parseFloat(rate)
|
|
if (isNaN(num)) return '-'
|
|
return (num > 0 ? '+' : '') + num + '%'
|
|
}
|
|
|
|
function formatPct(val) {
|
|
if (val == null) return '-'
|
|
const num = parseFloat(val)
|
|
if (isNaN(num)) return '-'
|
|
return (num > 0 ? '+' : '') + num.toFixed(3) + '%'
|
|
}
|
|
|
|
function formatChange(val) {
|
|
if (val == null) return '-'
|
|
const num = parseFloat(val)
|
|
if (isNaN(num)) return '-'
|
|
return (num > 0 ? '+' : '') + num.toFixed(2)
|
|
}
|
|
|
|
function getRateClass(rate) {
|
|
const num = parseFloat(rate) || 0
|
|
if (num > 0) return 'rate-up'
|
|
if (num < 0) return 'rate-down'
|
|
return 'rate-zero'
|
|
}
|
|
|
|
onMounted(() => {
|
|
fetchHistory()
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.history-app {
|
|
padding: 24px;
|
|
box-sizing: border-box;
|
|
width: 100%;
|
|
max-width: 1600px;
|
|
margin: 0 auto;
|
|
overflow-x: auto;
|
|
}
|
|
.history-header {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
margin-bottom: 16px;
|
|
background: var(--bg-elevated);
|
|
border: 1px solid var(--border-color);
|
|
padding: 16px 24px;
|
|
border-radius: 8px;
|
|
gap: 12px;
|
|
}
|
|
.history-header h2 {
|
|
margin: 0;
|
|
font-size: 20px;
|
|
color: var(--text-primary);
|
|
}
|
|
.back-link {
|
|
color: var(--accent);
|
|
cursor: pointer;
|
|
font-size: 14px;
|
|
text-decoration: none;
|
|
transition: color 0.2s;
|
|
white-space: nowrap;
|
|
flex-shrink: 0;
|
|
}
|
|
.back-link:hover {
|
|
color: #66b1ff;
|
|
}
|
|
.rate-up {
|
|
color: var(--rate-up);
|
|
font-weight: bold;
|
|
}
|
|
.rate-down {
|
|
color: var(--rate-down);
|
|
font-weight: bold;
|
|
}
|
|
.rate-zero {
|
|
color: var(--text-muted);
|
|
}
|
|
|
|
@media (max-width: 768px) {
|
|
.history-app {
|
|
padding: 12px;
|
|
}
|
|
.history-header {
|
|
flex-direction: column;
|
|
align-items: flex-start;
|
|
}
|
|
.history-header h2 {
|
|
font-size: 16px;
|
|
word-break: break-all;
|
|
}
|
|
.back-link {
|
|
align-self: flex-end;
|
|
}
|
|
}
|
|
</style>
|