2026-05-25 16:38:31 +08:00

237 lines
5.1 KiB
Vue

<template>
<div class="token-usage-page">
<div class="usage-shell">
<div class="page-header">
<div>
<div class="title-line">
<span class="title-icon">
<i class="el-icon-data-line"></i>
</span>
<h2>Token用量</h2>
</div>
<p>查看模型调用 Token 消耗调用次数和费用趋势</p>
</div>
<el-button size="small" icon="el-icon-refresh" class="refresh-btn">刷新</el-button>
</div>
<div class="stat-row">
<div v-for="item in statCards" :key="item.label" class="stat-card" :class="item.type">
<span>{{ item.label }}</span>
<strong>{{ item.value }}</strong>
<em>{{ item.desc }}</em>
</div>
</div>
<div class="content-card">
<div class="card-header">
<div>
<h3>用量明细</h3>
<p>按模型维度统计 Token 输入输出与费用</p>
</div>
<el-date-picker
v-model="dateRange"
type="daterange"
size="small"
range-separator=""
start-placeholder="开始日期"
end-placeholder="结束日期"
></el-date-picker>
</div>
<el-table :data="usageList" class="usage-table" style="width: 100%">
<el-table-column prop="modelName" label="模型名称" min-width="180"></el-table-column>
<el-table-column prop="calls" label="调用次数" width="120"></el-table-column>
<el-table-column prop="inputTokens" label="输入Token" min-width="140"></el-table-column>
<el-table-column prop="outputTokens" label="输出Token" min-width="140"></el-table-column>
<el-table-column prop="totalTokens" label="总Token" min-width="140"></el-table-column>
<el-table-column prop="cost" label="预估费用" width="120"></el-table-column>
<el-table-column prop="updatedAt" label="更新时间" min-width="170"></el-table-column>
</el-table>
</div>
</div>
</div>
</template>
<script>
import { reqTokenUsage } from '@/api/model/model'
export default {
name: 'TokenUsage',
data() {
return {
dateRange: [],
statCards: [
{ label: '总消耗 Token', value: '1,286,400', desc: '较昨日 +12.6%', type: 'primary' },
{ label: '调用次数', value: '3,482', desc: '今日累计调用', type: 'success' },
{ label: '预估费用', value: '¥ 128.64', desc: '按当前单价估算', type: 'warning' },
{ label: '活跃模型', value: '8', desc: '最近 7 天有调用', type: 'purple' }
],
usageList: [
]
}
},
created() {
this.getTokenList()
},
methods: {
getCurrentUserId() {
return sessionStorage.getItem('userId') || localStorage.getItem('userId') || ''
},
async getTokenList() {
const userid = this.getCurrentUserId()
const res = await reqTokenUsage({ userid:userid })
console.log('token用量',res);
}
}
}
</script>
<style lang="less" scoped>
.token-usage-page {
min-height: 100vh;
padding: 24px;
background: linear-gradient(180deg, #f3f7ff 0%, #f7f9fc 48%, #ffffff 100%);
}
.usage-shell {
min-height: calc(100vh - 48px);
padding: 24px;
background: #ffffff;
border: 1px solid #edf1f7;
border-radius: 18px;
box-shadow: 0 12px 30px rgba(31, 45, 61, 0.06);
}
.page-header,
.card-header {
display: flex;
align-items: flex-start;
justify-content: space-between;
gap: 20px;
}
.page-header {
margin-bottom: 24px;
}
.title-line {
display: flex;
align-items: center;
gap: 10px;
margin-bottom: 8px;
h2 {
margin: 0;
color: #1f2d3d;
font-size: 24px;
}
}
.title-icon {
display: flex;
align-items: center;
justify-content: center;
width: 36px;
height: 36px;
color: #409eff;
font-size: 18px;
background: #eef5ff;
border-radius: 12px;
}
.page-header p,
.card-header p {
margin: 0;
color: #909399;
font-size: 13px;
}
.refresh-btn {
border-radius: 10px;
}
.stat-row {
display: grid;
grid-template-columns: repeat(4, minmax(0, 1fr));
gap: 16px;
margin-bottom: 20px;
}
.stat-card {
padding: 20px;
border-radius: 16px;
border: 1px solid #edf1f7;
background: #ffffff;
span,
em {
display: block;
color: #909399;
font-style: normal;
font-size: 13px;
}
strong {
display: block;
margin: 10px 0 8px;
color: #1f2d3d;
font-size: 26px;
}
&.primary {
background: #eef5ff;
}
&.success {
background: #f0fdf4;
}
&.warning {
background: #fff7ed;
}
&.purple {
background: #f5f3ff;
}
}
.content-card {
padding: 22px;
border: 1px solid #edf1f7;
border-radius: 16px;
}
.card-header {
margin-bottom: 18px;
h3 {
margin: 0 0 8px;
color: #1f2d3d;
}
}
.usage-table {
border-radius: 12px;
overflow: hidden;
}
@media (max-width: 1100px) {
.stat-row {
grid-template-columns: repeat(2, minmax(0, 1fr));
}
}
@media (max-width: 700px) {
.stat-row {
grid-template-columns: 1fr;
}
.page-header,
.card-header {
flex-direction: column;
}
}
</style>