415 lines
12 KiB
Vue
415 lines
12 KiB
Vue
|
|
<template>
|
|||
|
|
<div class="mosquito-leaderboard">
|
|||
|
|
<!-- 加载状态 -->
|
|||
|
|
<div v-if="loading" class="loading-state">
|
|||
|
|
<div class="loading-skeleton">
|
|||
|
|
<div v-for="i in 5" :key="i" class="skeleton-item"></div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<!-- 错误状态 -->
|
|||
|
|
<div v-else-if="error" class="error-state">
|
|||
|
|
<div class="error-content">
|
|||
|
|
<svg class="w-8 h-8 text-red-400 mb-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|||
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4m0 4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"></path>
|
|||
|
|
</svg>
|
|||
|
|
<p class="text-gray-900 font-medium">加载失败</p>
|
|||
|
|
<p class="text-gray-600 text-sm mt-1">{{ error.message }}</p>
|
|||
|
|
<button
|
|||
|
|
class="retry-button mt-3 px-4 py-2 bg-blue-500 text-white rounded-md hover:bg-blue-600 transition-colors text-sm"
|
|||
|
|
@click="retryLoad"
|
|||
|
|
>
|
|||
|
|
重试
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<!-- 空状态 -->
|
|||
|
|
<div v-else-if="entries.length === 0" class="empty-state">
|
|||
|
|
<div class="empty-content">
|
|||
|
|
<svg class="w-12 h-12 text-gray-400 mb-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|||
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"></path>
|
|||
|
|
</svg>
|
|||
|
|
<p class="text-gray-900 font-medium">暂无排行榜数据</p>
|
|||
|
|
<p class="text-gray-600 text-sm mt-1">邀请好友加入,您将出现在排行榜中</p>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<!-- 排行榜内容 -->
|
|||
|
|
<div v-else class="leaderboard-content">
|
|||
|
|
<!-- Top N 显示 -->
|
|||
|
|
<div v-if="topN" class="top-section">
|
|||
|
|
<div
|
|||
|
|
v-for="(entry, index) in topEntries"
|
|||
|
|
:key="entry.userId"
|
|||
|
|
class="leaderboard-item top-item"
|
|||
|
|
:class="`top-${index + 1}`"
|
|||
|
|
>
|
|||
|
|
<div class="rank">
|
|||
|
|
<div class="rank-number">{{ index + 1 }}</div>
|
|||
|
|
<div v-if="index < 3" class="rank-badge">
|
|||
|
|
<svg v-if="index === 0" class="w-4 h-4" fill="currentColor" viewBox="0 0 20 20">
|
|||
|
|
<path d="M9.049 2.927c.3-.921 1.603-.921 1.902 0l1.07 3.292a1 1 0 00.95.69h3.462c.969 0 1.371 1.24.588 1.81l-2.8 2.034a1 1 0 00-.364 1.118l1.07 3.292c.3.921-.755 1.688-1.54 1.118l-2.8-2.034a1 1 0 00-1.175 0l-2.8 2.034c-.784.57-1.838-.197-1.539-1.118l1.07-3.292a1 1 0 00-.364-1.118L2.98 8.72c-.783-.57-.38-1.81.588-1.81h3.461a1 1 0 00.951-.69l1.07-3.292z"></path>
|
|||
|
|
</svg>
|
|||
|
|
<svg v-else-if="index === 1" class="w-4 h-4" fill="currentColor" viewBox="0 0 20 20">
|
|||
|
|
<path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm1-12a1 1 0 10-2 0v4a1 1 0 00.293.707l2.828 2.829a1 1 0 101.415-1.415L11 9.586V6z" clip-rule="evenodd"></path>
|
|||
|
|
</svg>
|
|||
|
|
<svg v-else class="w-4 h-4" fill="currentColor" viewBox="0 0 20 20">
|
|||
|
|
<path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM7 9a1 1 0 000 2h6a1 1 0 100-2H7z" clip-rule="evenodd"></path>
|
|||
|
|
</svg>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
<div class="user-info">
|
|||
|
|
<div class="user-name">{{ entry.userName }}</div>
|
|||
|
|
<div class="user-meta">{{ entry.inviteCount || entry.score }} 个好友</div>
|
|||
|
|
</div>
|
|||
|
|
<div class="score">
|
|||
|
|
<div class="score-number">{{ entry.score }}分</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<!-- 完整排行榜 -->
|
|||
|
|
<div class="full-list">
|
|||
|
|
<div
|
|||
|
|
v-for="(entry, index) in displayEntries"
|
|||
|
|
:key="entry.userId"
|
|||
|
|
class="leaderboard-item"
|
|||
|
|
:class="{ 'current-user': isCurrentUser(entry) }"
|
|||
|
|
>
|
|||
|
|
<div class="rank">
|
|||
|
|
<div class="rank-number">{{ startIndex + index + 1 }}</div>
|
|||
|
|
</div>
|
|||
|
|
<div class="user-info">
|
|||
|
|
<div class="user-name" :class="{ 'font-bold': isCurrentUser(entry) }">
|
|||
|
|
{{ entry.userName }}
|
|||
|
|
<span v-if="isCurrentUser(entry)" class="user-badge">我</span>
|
|||
|
|
</div>
|
|||
|
|
<div class="user-meta">{{ entry.inviteCount || entry.score }} 个好友</div>
|
|||
|
|
</div>
|
|||
|
|
<div class="score">
|
|||
|
|
<div class="score-number">{{ entry.score }}分</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<!-- 分页控制 -->
|
|||
|
|
<div v-if="hasPagination" class="pagination">
|
|||
|
|
<button
|
|||
|
|
class="pagination-button"
|
|||
|
|
:disabled="page === 0"
|
|||
|
|
@click="prevPage"
|
|||
|
|
>
|
|||
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|||
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 19l-7-7 7-7"></path>
|
|||
|
|
</svg>
|
|||
|
|
</button>
|
|||
|
|
|
|||
|
|
<div class="pagination-info">
|
|||
|
|
第 {{ page + 1 }} 页,共 {{ totalPages }} 页
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<button
|
|||
|
|
class="pagination-button"
|
|||
|
|
:disabled="page >= totalPages - 1"
|
|||
|
|
@click="nextPage"
|
|||
|
|
>
|
|||
|
|
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|||
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7"></path>
|
|||
|
|
</svg>
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
|
|||
|
|
<!-- 导出按钮 -->
|
|||
|
|
<div class="export-section">
|
|||
|
|
<button
|
|||
|
|
class="export-button"
|
|||
|
|
@click="exportLeaderboard"
|
|||
|
|
:disabled="loading"
|
|||
|
|
>
|
|||
|
|
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|||
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"></path>
|
|||
|
|
</svg>
|
|||
|
|
导出CSV
|
|||
|
|
</button>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</template>
|
|||
|
|
|
|||
|
|
<script setup lang="ts">
|
|||
|
|
import { ref, computed, watch } from 'vue'
|
|||
|
|
import { useMosquito } from '../index'
|
|||
|
|
|
|||
|
|
interface Props {
|
|||
|
|
activityId: number
|
|||
|
|
page?: number
|
|||
|
|
size?: number
|
|||
|
|
topN?: number
|
|||
|
|
currentUserId?: number
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const props = withDefaults(defineProps<Props>(), {
|
|||
|
|
page: 0,
|
|||
|
|
size: 20,
|
|||
|
|
topN: 10,
|
|||
|
|
currentUserId: undefined
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const emit = defineEmits<{
|
|||
|
|
loaded: [entries: any[]]
|
|||
|
|
error: [error: Error]
|
|||
|
|
}>()
|
|||
|
|
|
|||
|
|
const { getLeaderboard, exportLeaderboardCsv } = useMosquito()
|
|||
|
|
const loading = ref(false)
|
|||
|
|
const error = ref<Error | null>(null)
|
|||
|
|
const entries = ref<any[]>([])
|
|||
|
|
const pagination = ref({
|
|||
|
|
page: props.page,
|
|||
|
|
size: props.size,
|
|||
|
|
total: 0,
|
|||
|
|
totalPages: 0
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
// 计算属性
|
|||
|
|
const topEntries = computed(() => {
|
|||
|
|
return props.topN ? entries.value.slice(0, props.topN) : entries.value
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const displayEntries = computed(() => {
|
|||
|
|
if (props.topN) {
|
|||
|
|
return entries.value.slice(props.topN)
|
|||
|
|
}
|
|||
|
|
return entries.value
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const startIndex = computed(() => {
|
|||
|
|
return pagination.value.page * pagination.value.size
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const totalPages = computed(() => {
|
|||
|
|
return pagination.value.totalPages || Math.ceil(pagination.value.total / pagination.value.size)
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
const hasPagination = computed(() => {
|
|||
|
|
return totalPages.value > 1
|
|||
|
|
})
|
|||
|
|
|
|||
|
|
// 检查是否为当前用户
|
|||
|
|
const isCurrentUser = (entry: any) => {
|
|||
|
|
return props.currentUserId && entry.userId === props.currentUserId
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 加载排行榜数据
|
|||
|
|
const loadLeaderboard = async () => {
|
|||
|
|
if (loading.value) return
|
|||
|
|
|
|||
|
|
loading.value = true
|
|||
|
|
error.value = null
|
|||
|
|
|
|||
|
|
try {
|
|||
|
|
const result = await getLeaderboard(
|
|||
|
|
props.activityId,
|
|||
|
|
props.page,
|
|||
|
|
props.size
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
entries.value = result?.data || []
|
|||
|
|
const meta = result?.meta?.pagination
|
|||
|
|
pagination.value = {
|
|||
|
|
page: meta?.page ?? props.page,
|
|||
|
|
size: meta?.size ?? props.size,
|
|||
|
|
total: meta?.total ?? entries.value.length,
|
|||
|
|
totalPages: meta?.totalPages ?? Math.ceil((meta?.total ?? entries.value.length) / props.size)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
emit('loaded', entries.value)
|
|||
|
|
} catch (err) {
|
|||
|
|
console.error('加载排行榜失败:', err)
|
|||
|
|
error.value = err as Error
|
|||
|
|
emit('error', error.value)
|
|||
|
|
} finally {
|
|||
|
|
loading.value = false
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 重试加载
|
|||
|
|
const retryLoad = () => {
|
|||
|
|
loadLeaderboard()
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 分页控制
|
|||
|
|
const prevPage = () => {
|
|||
|
|
if (pagination.value.page > 0) {
|
|||
|
|
pagination.value.page--
|
|||
|
|
loadLeaderboard()
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
const nextPage = () => {
|
|||
|
|
if (pagination.value.page < totalPages.value - 1) {
|
|||
|
|
pagination.value.page++
|
|||
|
|
loadLeaderboard()
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 导出排行榜
|
|||
|
|
const exportLeaderboard = async () => {
|
|||
|
|
try {
|
|||
|
|
const csvData = await exportLeaderboardCsv(props.activityId)
|
|||
|
|
const blob = new Blob([csvData], { type: 'text/csv;charset=utf-8;' })
|
|||
|
|
const link = document.createElement('a')
|
|||
|
|
|
|||
|
|
if (link.download !== undefined) {
|
|||
|
|
const url = URL.createObjectURL(blob)
|
|||
|
|
link.setAttribute('href', url)
|
|||
|
|
link.setAttribute('download', `leaderboard-${props.activityId}.csv`)
|
|||
|
|
link.style.visibility = 'hidden'
|
|||
|
|
document.body.appendChild(link)
|
|||
|
|
link.click()
|
|||
|
|
document.body.removeChild(link)
|
|||
|
|
}
|
|||
|
|
} catch (err) {
|
|||
|
|
console.error('导出排行榜失败:', err)
|
|||
|
|
error.value = err as Error
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// 组件挂载时加载数据
|
|||
|
|
loadLeaderboard()
|
|||
|
|
|
|||
|
|
// 监听参数变化
|
|||
|
|
watch(() => [props.activityId, props.page, props.size, props.topN], () => {
|
|||
|
|
loadLeaderboard()
|
|||
|
|
}, { deep: true })
|
|||
|
|
</script>
|
|||
|
|
|
|||
|
|
<style scoped>
|
|||
|
|
.mosquito-leaderboard {
|
|||
|
|
@apply rounded-2xl border border-mosquito-line bg-mosquito-surface shadow-soft;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.loading-state {
|
|||
|
|
@apply p-6;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.loading-skeleton {
|
|||
|
|
@apply space-y-3;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.skeleton-item {
|
|||
|
|
@apply h-12 bg-gray-200 rounded animate-pulse;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.error-state,
|
|||
|
|
.empty-state {
|
|||
|
|
@apply p-8 text-center;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.error-content,
|
|||
|
|
.empty-content {
|
|||
|
|
@apply inline-block;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.retry-button {
|
|||
|
|
@apply px-4 py-2 bg-mosquito-accent text-white rounded-md hover:bg-mosquito-accent/90 transition-colors text-sm;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-content {
|
|||
|
|
@apply divide-y divide-gray-200;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.top-section {
|
|||
|
|
@apply p-4 bg-gradient-to-r from-mosquito-accent/10 to-mosquito-accent2/10;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-item {
|
|||
|
|
@apply flex items-center justify-between p-4 hover:bg-gray-50 transition-colors;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-item.current-user {
|
|||
|
|
@apply bg-mosquito-accent/10 border-l-4 border-mosquito-accent;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-item.top-item {
|
|||
|
|
@apply p-6 border-b border-gray-200;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-item.top-item.top-1 {
|
|||
|
|
@apply bg-gradient-to-r from-mosquito-accent2/20 to-mosquito-accent/10 border-b-2 border-mosquito-accent/40;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-item.top-item.top-2 {
|
|||
|
|
@apply bg-gradient-to-r from-mosquito-accent2/15 to-mosquito-accent2/5 border-b-2 border-mosquito-accent2/40;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.leaderboard-item.top-item.top-3 {
|
|||
|
|
@apply bg-gradient-to-r from-mosquito-accent/15 to-mosquito-accent/5 border-b-2 border-mosquito-accent/40;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.rank {
|
|||
|
|
@apply flex items-center justify-center w-12 h-12 rounded-full bg-mosquito-surface shadow-sm;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.rank-number {
|
|||
|
|
@apply font-bold text-lg text-mosquito-ink;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.rank-badge {
|
|||
|
|
@absolute -top-1 -right-1;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.user-info {
|
|||
|
|
@apply flex-1 ml-4;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.user-name {
|
|||
|
|
@apply font-medium text-mosquito-ink;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.user-badge {
|
|||
|
|
@apply ml-2 px-2 py-1 text-xs bg-mosquito-accent/15 text-mosquito-brand rounded-full;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.user-meta {
|
|||
|
|
@apply text-sm text-mosquito-ink/60;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.score {
|
|||
|
|
@apply text-right;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.score-number {
|
|||
|
|
@apply font-bold text-lg text-mosquito-ink;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.pagination {
|
|||
|
|
@apply flex items-center justify-between p-4 bg-mosquito-bg border-t border-mosquito-line;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.pagination-button {
|
|||
|
|
@apply p-2 text-mosquito-ink/60 hover:text-mosquito-ink hover:bg-mosquito-accent/10 rounded-md transition-colors;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.pagination-button:disabled {
|
|||
|
|
@apply text-mosquito-ink/30 cursor-not-allowed hover:text-mosquito-ink/30 hover:bg-transparent;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.pagination-info {
|
|||
|
|
@apply text-sm text-mosquito-ink/60;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.export-section {
|
|||
|
|
@apply p-4 bg-mosquito-bg border-t border-mosquito-line;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.export-button {
|
|||
|
|
@apply inline-flex items-center px-4 py-2 bg-mosquito-accent text-white rounded-md hover:bg-mosquito-accent/90 transition-colors text-sm;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
.export-button:disabled {
|
|||
|
|
@apply bg-mosquito-ink/30 cursor-not-allowed;
|
|||
|
|
}
|
|||
|
|
</style>
|