web/src/views/player/index.vue
@@ -1,25 +1,27 @@
<template>
  <div class="player-page">
    <div class="page-card">
      <h3 class="card-title">报名审核</h3>
      <!-- 搜索和操作栏 -->
      <div class="toolbar">
    <!-- 页面标题区域 -->
    <div class="page-header">
      <div class="title-section">
        <h1 class="page-title">报名审核</h1>
        <p class="page-subtitle">管理参赛选手的报名申请,审核参赛资格和信息</p>
      </div>
    </div>
    <!-- 搜索工具栏 -->
    <div class="search-toolbar">
      <div class="search-form">
        <el-input
          v-model="searchForm.name"
          placeholder="请输入学员名称"
          style="width: 200px"
          style="width: 180px"
          clearable
          @keyup.enter="handleSearch"
        >
          <template #prefix>
            <el-icon><Search /></el-icon>
          </template>
        </el-input>
          @clear="handleClear"
        />
        <el-select
          v-model="searchForm.activityId"
          placeholder="选择比赛"
          style="width: 300px"
          style="width: 200px"
          clearable
          filterable
        >
@@ -35,7 +37,7 @@
        <el-select
          v-model="searchForm.state"
          placeholder="选择状态"
          style="width: 150px"
          style="width: 120px"
          clearable
        >
          <el-option label="待审核" value="0" />
@@ -44,58 +46,58 @@
        </el-select>
        <el-button type="primary" @click="handleSearch">
          <el-icon><Search /></el-icon>
          搜索
          查询
        </el-button>
        <el-button type="warning" @click="debugAPI">
          调试API
        </el-button>
        <span style="margin-left: 10px; color: #666; font-size: 12px;">
          活动选项数量: {{ activityOptions.length }}
        </span>
      </div>
    </div>
      <!-- 学员列表 -->
      <el-table :data="tableData" style="width: 100%" v-loading="loading">
        <el-table-column label="头像" width="80" align="center">
          <template #default="{ row }">
            <el-avatar :src="row.avatar" :size="40">
              <el-icon><User /></el-icon>
            </el-avatar>
          </template>
        </el-table-column>
        <el-table-column prop="name" label="学员名称" min-width="120" />
        <el-table-column prop="activityName" label="报名项目" min-width="200" />
        <el-table-column prop="phone" label="联系电话" width="140" />
        <el-table-column prop="applyTime" label="申请时间" width="180" />
        <el-table-column prop="state" label="状态" width="100" align="center">
          <template #default="{ row }">
            <el-tag :type="getStateType(row.state)">{{ getStateText(row.state) }}</el-tag>
          </template>
        </el-table-column>
        <el-table-column label="操作" width="120" fixed="right">
          <template #default="{ row }">
            <div class="table-actions">
              <!-- 只保留详情按钮 -->
              <el-button type="primary" size="small" @click="handleViewDetail(row)">
                详情
              </el-button>
            </div>
          </template>
        </el-table-column>
      </el-table>
    <!-- 学员列表 -->
    <el-table :data="tableData" style="width: 100%" v-loading="loading">
      <el-table-column label="头像" width="80" align="center">
        <template #default="{ row }">
          <el-avatar :src="row.avatar" :size="40">
            <el-icon><User /></el-icon>
          </el-avatar>
        </template>
      </el-table-column>
      <el-table-column label="学员名称" min-width="120">
        <template #default="{ row }">
          {{ row.name }}
        </template>
      </el-table-column>
      <el-table-column prop="projectName" label="项目名称" min-width="150" />
      <el-table-column prop="activityName" label="比赛名称" min-width="200" />
      <el-table-column prop="phone" label="联系电话" width="140" />
      <el-table-column prop="applyTime" label="申请时间" width="180" />
      <el-table-column prop="state" label="状态" width="100" align="center">
        <template #default="{ row }">
          <el-tag :type="getStateType(row.state)">{{ getStateText(row.state) }}</el-tag>
        </template>
      </el-table-column>
      <el-table-column label="操作" width="80" align="center">
        <template #default="{ row }">
          <el-button
            text
            :icon="View"
            @click="handleViewDetail(row)"
            class="action-btn view-btn"
            title="查看详情"
          />
        </template>
      </el-table-column>
    </el-table>
      <!-- 分页 -->
      <div class="pagination">
        <el-pagination
          v-model:current-page="pagination.page"
          v-model:page-size="pagination.size"
          :page-sizes="[10, 20, 50, 100]"
          :total="pagination.total"
          layout="total, sizes, prev, pager, next, jumper"
          @size-change="handleSizeChange"
          @current-change="handleCurrentChange"
        />
      </div>
    <!-- 分页 -->
    <div class="pagination">
      <el-pagination
        v-model:current-page="pagination.page"
        v-model:page-size="pagination.size"
        :page-sizes="[10, 20, 50, 100]"
        :total="pagination.total"
        layout="total, sizes, prev, pager, next, jumper"
        @size-change="handleSizeChange"
        @current-change="handleCurrentChange"
      />
    </div>
  </div>
</template>
@@ -104,6 +106,7 @@
import { reactive, ref, onMounted } from 'vue'
import { useRouter } from 'vue-router'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Search, User, View } from '@element-plus/icons-vue'
import { PlayerApi } from '@/api/player'
import { getAllActivities } from '@/api/activity'
@@ -128,44 +131,7 @@
})
// 表格数据
const tableData = ref([
  {
    id: 1,
    name: '张三',
    avatar: '',
    activityName: '2024年创新创业大赛',
    phone: '13800138001',
    applyTime: '2024-01-05 14:30:00',
    state: 0 // 0-未审核
  },
  {
    id: 2,
    name: '李四',
    avatar: '',
    activityName: '书法比赛',
    phone: '13900139002',
    applyTime: '2024-01-16 10:30:00',
    state: 1 // 1-审核通过
  },
  {
    id: 3,
    name: '王五',
    avatar: '',
    activityName: '绘画比赛',
    phone: '13900139003',
    applyTime: '2024-01-17 14:20:00',
    state: 2 // 2-审核驳回
  },
  {
    id: 4,
    name: '赵六',
    avatar: '',
    activityName: '音乐比赛',
    phone: '13900139004',
    applyTime: '2024-01-18 09:15:00',
    state: 0 // 0-未审核
  }
])
const tableData = ref([])
// 获取状态标签类型
const getStateType = (state: number | null | undefined) => {
@@ -195,7 +161,11 @@
  loadData()
}
const handleClear = () => {
  searchForm.name = ''
  pagination.page = 1
  loadData()
}
// 查看详情(跳转到详情页面,只读模式)
const handleViewDetail = (row: any) => {
@@ -236,110 +206,49 @@
  return activity.name
}
// 加载活动选项
const loadActivityOptions = async () => {
  try {
    console.log('=== 开始加载活动选项 ===')
    console.log('调用getAllActivities API...')
    const activities = await getAllActivities()
    console.log('API返回的原始数据:', activities)
    console.log('数据类型:', typeof activities)
    console.log('是否为数组:', Array.isArray(activities))
    
    if (activities && Array.isArray(activities)) {
      console.log('活动数量:', activities.length)
      activities.forEach((activity, index) => {
        console.log(`活动${index + 1}:`, {
          id: activity.id,
          name: activity.name,
          state: activity.state,
          pid: activity.pid
        })
      })
      // 过滤出正在进行的比赛(不是阶段)
      const filtered = activities.filter(activity => 
        activity.state === 1 && (activity.pid === 0 || activity.pid === "0")
      )
      console.log('过滤条件: state === 1 && (pid === 0 || pid === "0")')
      console.log('过滤后的活动:', filtered)
      
      activityOptions.value = filtered
      console.log('设置到activityOptions.value:', activityOptions.value)
      console.log('activityOptions.value.length:', activityOptions.value.length)
      // 强制触发响应式更新
      setTimeout(() => {
        console.log('延迟检查activityOptions.value:', activityOptions.value)
        console.log('延迟检查activityOptions.value.length:', activityOptions.value.length)
      }, 100)
    } else {
      console.error('API返回的数据格式不正确:', activities)
    }
  } catch (error) {
    console.error('=== 加载活动选项失败 ===')
    console.error('错误详情:', error)
    console.error('错误消息:', error.message)
    console.error('错误堆栈:', error.stack)
    ElMessage.error('加载活动选项失败: ' + error.message)
    ElMessage.error('加载活动选项失败')
  }
}
// 调试API函数
const debugAPI = async () => {
  console.log('=== 开始API调试 ===')
  // 检查认证状态
  const token = localStorage.getItem('auth_token')
  const userInfo = localStorage.getItem('user_info')
  console.log('认证状态检查:')
  console.log('Token:', token ? '已存在' : '不存在')
  console.log('Token内容:', token)
  console.log('用户信息:', userInfo ? '已存在' : '不存在')
  console.log('用户信息内容:', userInfo)
  if (!token) {
    ElMessage.error('未找到认证token,请先登录')
    return
  }
  // 测试API调用
  try {
    console.log('开始测试getAllActivities API...')
    const activities = await getAllActivities()
    console.log('API调用成功,返回数据:', activities)
    ElMessage.success(`API调用成功,获取到${activities?.length || 0}个活动`)
  } catch (error) {
    console.error('API调用失败:', error)
    ElMessage.error('API调用失败: ' + error.message)
  }
}
// 加载数据
const loadData = async () => {
  loading.value = true
  try {
    const list = await PlayerApi.getApplications(
    const response = await PlayerApi.getApplications(
      searchForm.name || '', 
      searchForm.activityId || null, 
      searchForm.state !== '' ? parseInt(searchForm.state) : null,
      pagination.page, 
      pagination.size
    )
    tableData.value = (list || []).map((item: any) => ({
    const list = response.content || []
    tableData.value = list.map((item: any) => ({
      id: item.id,
      name: item.playerName,
      projectName: item.projectName,
      avatar: '',
      activityName: item.activityName,
      phone: item.phone,
      applyTime: item.applyTime,
      state: item.state
    }))
    pagination.total = tableData.value.length
    pagination.total = response.totalElements || 0
  } catch (e: any) {
    ElMessage.error(String(e?.message || e))
  } finally {
@@ -348,49 +257,102 @@
}
onMounted(() => {
  console.log('=== Player页面onMounted被调用 ===')
  console.log('当前时间:', new Date().toLocaleTimeString())
  console.log('activityOptions初始值:', activityOptions.value)
  console.log('直接加载活动选项进行测试...')
  // 立即调用API测试
  loadActivityOptions().catch(error => {
    console.error('loadActivityOptions调用失败:', error)
  })
  // loadData() // 暂时注释掉,专注于活动选项加载
  loadActivityOptions()
  loadData()
})
</script>
<style lang="scss" scoped>
.player-page {
  .card-title {
    margin-bottom: 20px;
    color: #303133;
    font-size: 16px;
    font-weight: 500;
  }
  .toolbar {
    display: flex;
    gap: 12px;
    margin-bottom: 20px;
    align-items: center;
  }
  .table-actions {
    display: flex;
    gap: 8px;
    flex-wrap: wrap;
    align-items: center;
  }
  padding: 20px;
}
/* 页面标题区域 */
.page-header {
  margin-bottom: 24px;
}
.title-section {
  text-align: left;
}
.page-title {
  font-size: 24px;
  font-weight: 600;
  color: #1f2937;
  margin: 0 0 8px 0;
}
.page-subtitle {
  font-size: 14px;
  color: #6b7280;
  margin: 0;
  line-height: 1.5;
}
/* 搜索工具栏 */
.search-toolbar {
  display: flex;
  justify-content: flex-end;
  margin-bottom: 20px;
}
.search-form {
  display: flex;
  gap: 12px;
  align-items: center;
}
.search-area {
  display: flex;
  align-items: center;
  gap: 12px;
}
.action-area {
  display: flex;
  align-items: center;
  gap: 12px;
}
/* 操作按钮样式 */
.action-btn {
  padding: 8px !important;
  margin: 0 6px;
  border-radius: 6px;
  transition: all 0.2s ease;
}
.view-btn {
  color: #3b82f6 !important;
}
.view-btn:hover {
  background-color: rgba(59, 130, 246, 0.1) !important;
  transform: scale(1.2);
}
.pagination {
  margin-top: 20px;
  display: flex;
  justify-content: center;
}
/* 响应式适配 */
@media (max-width: 768px) {
  .search-toolbar {
    flex-direction: column;
    gap: 12px;
    align-items: stretch;
  }
  
  .pagination {
    margin-top: 20px;
    display: flex;
    justify-content: flex-end;
  .search-area {
    justify-content: center;
    flex-wrap: wrap;
  }
  .action-area {
    justify-content: center;
  }
}
</style>