Files
sionrui/frontend/app/web-gold/src/views/content-style/components/BenchmarkTable.vue

373 lines
10 KiB
Vue
Raw Normal View History

2025-11-13 01:06:28 +08:00
<script setup>
import { reactive, computed, ref } from 'vue'
2025-11-13 01:06:28 +08:00
import { formatTime } from '../utils/benchmarkUtils'
2025-11-14 02:15:14 +08:00
import GradientButton from '@/components/GradientButton.vue'
import { Button } from '@/components/ui/button'
import { Checkbox } from '@/components/ui/checkbox'
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from '@/components/ui/table'
import { Spinner } from '@/components/ui/spinner'
const props = defineProps({
2026-02-25 21:57:05 +08:00
data: { type: Array, required: true },
selectedRowKeys: { type: Array, required: true },
loading: { type: Boolean, default: false },
loadingMore: { type: Boolean, default: false },
hasMore: { type: Boolean, default: false },
2025-11-13 01:06:28 +08:00
})
const emit = defineEmits(['update:selectedRowKeys', 'export', 'loadMore', 'createAsyncTask'])
2025-11-13 01:06:28 +08:00
// 列定义
const columns = [
{ key: 'cover', title: '封面', width: '100px' },
{ key: 'desc', title: '描述', width: '200px' },
{ key: 'digg_count', title: '点赞', width: '80px', sortable: true },
{ key: 'comment_count', title: '评论', width: '80px', sortable: true },
{ key: 'share_count', title: '分享', width: '80px', sortable: true },
{ key: 'collect_count', title: '收藏', width: '80px', sortable: true },
{ key: 'duration_s', title: '时长', width: '80px', sortable: true },
{ key: 'create_time', title: '创建时间', width: '140px', sortable: true },
2025-11-13 01:06:28 +08:00
]
// 排序状态
const sortKey = ref('digg_count')
const sortOrder = ref('desc')
// 全选状态
const isAllSelected = computed(() => {
return props.data.length > 0 && props.selectedRowKeys.length === props.data.length
})
// 切换排序
function handleSort(key) {
if (sortKey.value === key) {
sortOrder.value = sortOrder.value === 'asc' ? 'desc' : 'asc'
} else {
sortKey.value = key
sortOrder.value = 'desc'
}
emit('update:selectedRowKeys', [])
}
2025-11-13 01:06:28 +08:00
// 选择切换
function handleSelectAll() {
if (isAllSelected.value) {
emit('update:selectedRowKeys', [])
} else {
emit('update:selectedRowKeys', props.data.map(item => String(item.id)))
}
2025-11-13 01:06:28 +08:00
}
2026-02-25 21:57:05 +08:00
function handleSelect(id) {
const key = String(id)
const newKeys = [...props.selectedRowKeys]
const index = newKeys.indexOf(key)
if (index > -1) {
newKeys.splice(index, 1)
} else {
newKeys.push(key)
}
emit('update:selectedRowKeys', newKeys)
}
// 排序后的数据
const sortedData = computed(() => {
const data = [...props.data]
const key = sortKey.value
if (!key) return data
return data.sort((a, b) => {
const aVal = a[key] || 0
const bVal = b[key] || 0
return sortOrder.value === 'asc' ? aVal - bVal : bVal - aVal
})
})
// 格式化数字
2026-02-25 21:57:05 +08:00
function formatNumber(value) {
return value ? value.toLocaleString('zh-CN') : '0'
}
2025-11-13 01:06:28 +08:00
</script>
<template>
<section class="card results-card" v-if="data.length > 0">
<div class="section-header">
<div class="section-title">分析结果</div>
2026-01-17 19:33:59 +08:00
<div class="button-group">
<GradientButton
:text="`导出Excel (${selectedRowKeys.length}/20)`"
size="small"
2025-11-13 01:06:28 +08:00
@click="$emit('export')"
2026-01-17 19:33:59 +08:00
:disabled="data.length === 0 || selectedRowKeys.length === 0 || selectedRowKeys.length > 20"
icon="download"
/>
2025-11-14 02:15:14 +08:00
<GradientButton
2026-01-17 19:33:59 +08:00
:text="`批量分析 (${selectedRowKeys.length}/20)`"
2025-11-14 02:15:14 +08:00
size="small"
2026-03-15 15:36:29 +08:00
@click="$emit('createAsyncTask')"
:disabled="data.length === 0 || selectedRowKeys.length === 0 || selectedRowKeys.length > 20"
2026-03-15 15:36:29 +08:00
icon="clock-circle"
/>
2026-01-17 19:33:59 +08:00
</div>
2025-11-13 01:06:28 +08:00
</div>
<!-- 表格 -->
<div class="table-wrapper">
<Table>
<TableHeader>
<TableRow>
<TableHead class="w-[40px]">
<Checkbox
:checked="isAllSelected"
@update:checked="handleSelectAll"
class="checkbox-small"
/>
</TableHead>
<TableHead
v-for="col in columns"
:key="col.key"
:class="[col.width && `w-[${col.width}]`, col.sortable && 'cursor-pointer']"
@click="col.sortable && handleSort(col.key)"
>
<div class="flex items-center gap-1">
{{ col.title }}
<span v-if="col.sortable" class="sort-icon">
<svg v-if="sortKey !== col.key" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M7 15l5 5 5-5M7 9l5-5 5 5"/>
</svg>
<svg v-else-if="sortOrder === 'asc'" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M12 19V5M5 12l7-7 7 7"/>
</svg>
<svg v-else width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M12 5v14M5 12l7 7 7-7"/>
</svg>
</span>
</div>
</TableHead>
</TableRow>
</TableHeader>
<TableBody>
<TableRow v-if="loading">
<TableCell :col-span="columns.length + 1" class="h-32 text-center">
<Spinner class="mx-auto" />
</TableCell>
</TableRow>
<TableRow
v-else
v-for="record in sortedData"
:key="record.id"
class="table-row"
2026-01-27 00:39:12 +08:00
>
<TableCell class="w-[40px]">
<Checkbox
:checked="selectedRowKeys.includes(String(record.id))"
@update:checked="handleSelect(record.id)"
class="checkbox-small"
/>
</TableCell>
<TableCell>
<div class="cover-wrapper">
<a v-if="record.cover && record.share_url" :href="record.share_url" target="_blank" class="cover-link">
<img :src="record.cover" alt="cover" loading="lazy" class="cover-img" />
<span v-if="record.is_top" class="top-tag">置顶</span>
</a>
<template v-else-if="record.cover">
<img :src="record.cover" alt="cover" loading="lazy" class="cover-img" />
<span v-if="record.is_top" class="top-tag">置顶</span>
</template>
<span v-else class="no-cover">无封面</span>
</div>
</TableCell>
<TableCell>
<span :title="record.desc" class="truncate-text">{{ record.desc || '-' }}</span>
</TableCell>
<TableCell>{{ formatNumber(record.digg_count) }}</TableCell>
<TableCell>{{ formatNumber(record.comment_count) }}</TableCell>
<TableCell>{{ formatNumber(record.share_count) }}</TableCell>
<TableCell>{{ formatNumber(record.collect_count) }}</TableCell>
<TableCell>{{ record.duration_s ? `${record.duration_s}s` : '-' }}</TableCell>
<TableCell>{{ formatTime(record.create_time) }}</TableCell>
</TableRow>
</TableBody>
</Table>
</div>
<!-- 加载更多 -->
<div v-if="hasMore" class="load-more-footer">
<Button
:disabled="loadingMore"
size="lg"
variant="outline"
@click="$emit('loadMore')"
class="load-more-btn"
>
<Spinner v-if="loadingMore" class="mr-2 h-4 w-4" />
{{ loadingMore ? '加载中...' : '加载更多内容' }}
</Button>
<div class="load-more-hint">已加载 {{ data.length }} </div>
</div>
<div v-else-if="data.length > 0" class="no-more-hint">
已加载全部内容
</div>
2025-11-13 01:06:28 +08:00
</section>
</template>
<style scoped>
.card {
2026-02-25 23:44:01 +08:00
background: var(--color-bg-card);
border-radius: var(--radius-md);
padding: var(--space-4);
box-shadow: var(--shadow-sm);
border: 1px solid var(--color-gray-200);
2026-02-25 21:57:05 +08:00
overflow: hidden;
2025-11-13 01:06:28 +08:00
}
.results-card {
min-height: 420px;
}
.section-header {
display: flex;
justify-content: space-between;
align-items: center;
2026-02-25 23:44:01 +08:00
margin-bottom: var(--space-3);
2025-11-13 01:06:28 +08:00
}
2026-01-17 19:33:59 +08:00
.button-group {
2025-11-13 01:06:28 +08:00
display: flex;
align-items: center;
2026-02-25 23:44:01 +08:00
gap: var(--space-3);
2025-11-13 01:06:28 +08:00
}
.section-title {
2026-02-25 23:44:01 +08:00
font-size: var(--font-size-xs);
color: var(--color-gray-600);
margin-bottom: var(--space-3);
2025-11-13 01:06:28 +08:00
}
.table-wrapper {
overflow-x: auto;
max-height: calc(100vh - 400px);
overflow-y: auto;
}
.table-wrapper::-webkit-scrollbar {
width: 6px;
height: 6px;
2025-11-13 01:06:28 +08:00
}
.table-wrapper::-webkit-scrollbar-track {
background: transparent;
}
.table-wrapper::-webkit-scrollbar-thumb {
background: var(--color-gray-300);
border-radius: 3px;
}
.table-row:hover {
background: var(--color-gray-50);
}
.checkbox-small {
transform: scale(0.85);
2025-11-13 01:06:28 +08:00
}
2026-02-25 21:57:05 +08:00
.cover-wrapper {
position: relative;
display: inline-block;
}
.cover-link {
display: block;
cursor: pointer;
}
.cover-link:hover .cover-img {
transform: scale(1.02);
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
}
.top-tag {
position: absolute;
2026-02-25 23:44:01 +08:00
top: var(--space-1);
left: var(--space-1);
2026-02-25 21:57:05 +08:00
padding: 1px 6px;
font-size: 10px;
color: #fff;
2026-02-25 23:44:01 +08:00
background: var(--color-error-500);
2026-02-25 21:57:05 +08:00
border-radius: 2px;
2025-11-13 01:06:28 +08:00
}
2026-02-25 21:57:05 +08:00
.cover-img {
width: 80px;
height: 45px;
object-fit: cover;
2026-02-25 23:44:01 +08:00
border-radius: var(--radius-sm);
border: 1px solid var(--color-gray-200);
2026-02-25 21:57:05 +08:00
transition: transform 0.2s, box-shadow 0.2s;
}
.no-cover {
2026-02-25 23:44:01 +08:00
color: var(--color-gray-500);
font-size: var(--font-size-xs);
2025-11-13 01:06:28 +08:00
}
.truncate-text {
display: block;
max-width: 180px;
2026-02-25 21:57:05 +08:00
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
2026-02-25 21:57:05 +08:00
}
.sort-icon {
opacity: 0.4;
transition: opacity var(--duration-fast);
2026-02-25 21:57:05 +08:00
}
.table-row:hover .sort-icon,
[onclick]:hover .sort-icon {
opacity: 1;
2025-11-13 01:06:28 +08:00
}
2026-01-27 00:39:12 +08:00
2026-02-25 21:57:05 +08:00
.load-more-footer,
.no-more-hint {
2026-01-27 00:39:12 +08:00
text-align: center;
2026-02-25 23:44:01 +08:00
border-top: 1px solid var(--color-gray-200);
2026-01-27 00:39:12 +08:00
}
2026-02-25 21:57:05 +08:00
.load-more-footer {
2026-02-25 23:44:01 +08:00
padding: var(--space-4);
2026-02-25 21:57:05 +08:00
}
.load-more-btn {
2026-01-27 00:39:12 +08:00
max-width: 200px;
height: 40px;
2026-02-25 23:44:01 +08:00
font-size: var(--font-size-base);
border-radius: var(--radius-md);
2026-01-27 00:39:12 +08:00
}
2026-02-25 21:57:05 +08:00
.load-more-hint,
.no-more-hint {
2026-02-25 23:44:01 +08:00
font-size: var(--font-size-xs);
color: var(--color-gray-600);
2026-01-27 00:39:12 +08:00
}
2026-02-25 21:57:05 +08:00
.load-more-hint {
2026-02-25 23:44:01 +08:00
margin-top: var(--space-2);
2026-02-25 21:57:05 +08:00
}
2026-01-27 00:39:12 +08:00
.no-more-hint {
2026-02-25 23:44:01 +08:00
padding: var(--space-3);
2026-01-27 00:39:12 +08:00
}
2025-11-13 01:06:28 +08:00
</style>