Files
swimmingUni/src/pages/dataAnalyze/paragraphAnalyze.vue
2026-04-06 09:31:25 +08:00

754 lines
19 KiB
Vue

<template>
<view class="paragraph-container">
<view class="header-section">
<view class="header-title">
<text class="title">">分段数据</text>
<text class="subtitle">学生分段训练成绩分析</text>
</view>
</view>
<view class="select-section">
<view class="select-label">
<text class="label-text">">选择项目</text>
</view>
<picker mode="selector" :range="projectOptions" range-key="name" :value="selectedProjectIndex"
@change="handleProjectChange">
<view class="picker-wrapper">
<text class="picker-text">{{ selectedProjectName }}</text>
<u-icon name="arrow-down" size="18" color="#999"></u-icon>
</view>
</picker>
</view>
<view class="calendar-wrapper" v-if="selectedProjectId">
<uni-calendar
:insert="true"
:lunar="false"
:show-month="true"
:selected="selectedDates"
@monthSwitch="handleMonthSwitch"
@change="handleDateChange">
</uni-calendar>
</view>
<view class="select-section" v-if="selectedProjectId && selectedDate">
<view class="select-label">
<text class="label-text">选择学生</text>
</view>
<view class="picker-wrapper" @click="showStudentSelect = true">
<text class="picker-text">{{ selectedStudentDisplay }}</text>
<u-icon name="arrow-down" size="18" color="#999"></u-icon>
</view>
</view>
<view class="modal-overlay" v-if="showStudentSelect" @click="closeStudentSelect"></view>
<view class="student-picker-modal" v-if="showStudentSelect">
<view class="picker-header">
<text class="header-title">选择学生</text>
<view class="header-actions">
<text class="action-btn" @click="selectAllStudents">全选</text>
<text class="action-btn primary" @click="confirmStudentSelect">确定</text>
</view>
</view>
<scroll-view class="student-list" scroll-y>
<view v-for="(student, index) in studentList" :key="student.id" class="student-item"
:class="{ 'selected': selectedStudentIndexes.includes(index) }"
@click="toggleStudentSelect(index)">
<view class="item-checkbox">
<view class="checkbox-inner" :class="{ 'checked': selectedStudentIndexes.includes(index) }">
<u-icon v-if="selectedStudentIndexes.includes(index)" name="checkmark" size="14"
color="#fff"></u-icon>
</view>
</view>
<view class="item-avatar">
<view class="avatar-circle"
:class="{ 'male': student.gender === '男', 'female': student.gender === '女' }">
<text class="avatar-text">{{ student.name.charAt(0) }}</text>
</view>
</view>
<view class="item-info">
<text class="item-name">{{ student.name }}</text>
<view class="item-meta">
<text class="gender-badge"
:class="{ 'male': student.gender === '男', 'female': student.gender === '女' }">
{{ student.gender }}
</text>
<text class="age-text">{{ student.age }}</text>
</view>
</view>
</view>
</scroll-view>
</view>
<view class="table-section" v-if="selectedDate && selectedStudentIndexes.length > 0 && tableData.length > 0">
<view class="table-wrapper-scroll">
<view class="table-header">
<view class="header-row">
<view class="header-cell" style="width: 100px;">日期</view>
<view class="header-cell" style="width: 100px;">姓名</view>
<view class="header-cell" style="width: 120px;">项目</view>
<view class="header-cell" style="width: 100px;">20</view>
<view class="header-cell" style="width: 100px;">70</view>
<view class="header-cell" style="width: 100px;">120</view>
<view class="header-cell" style="width: 100px;">170</view>
</view>
</view>
<view class="table-body">
<view v-for="(row, rowIndex) in tableData" :key="rowIndex" class="body-row" :class="{ 'even': rowIndex % 2 === 0, 'odd': rowIndex % 2 === 1 }">
<view class="body-cell" style="width: 100px;">{{ row.date }}</view>
<view class="body-cell" style="width: 100px;">{{ row.name }}</view>
<view class="body-cell" style="width: 120px;">{{ row.projectName }}</view>
<view class="body-cell" style="width: 100px;">{{ row.segment20 }}</view>
<view class="body-cell" style="width: 100px;">{{ row.segment70 }}</view>
<view class="body-cell" style="width: 100px;">{{ row.segment120 }}</view>
<view class="body-cell" style="width: 100px;">{{ row.segment170 }}</view>
</view>
</view>
</view>
</view>
<view class="empty-container" v-if="selectedDate && selectedStudentIndexes.length > 0 && tableData.length === 0">
<view class="empty-icon">
<u-icon name="file-text" size="80" color="#ccc"></u-icon>
</view>
<text class="empty-text">该日期暂无分段数据</text>
</view>
</view>
</template>
<script setup lang="ts">
import { onShow, onLoad } from "@dcloudio/uni-app"
import { Service } from '@/Service/Service'
import { ref, computed } from 'vue'
interface Project {
id: string
name: string
}
interface Student {
id: string
name: string
gender: string
age: number
}
interface TableDataItem {
date: string
name: string
projectName: string
segment20: string
segment70: string
segment120: string
segment170: string
}
const currentYear = ref(new Date().getFullYear())
const currentMonth = ref(new Date().getMonth() + 1)
const selectedDate = ref('')
const projectList = ref<Project[]>([
{ id: '1', name: '100米自由泳' },
{ id: '2', name: '200米自由泳' },
{ id: '3', name: '50米蛙泳' },
{ id: '4', name: '100米蝶泳' }
])
const projectOptions = ref(projectList.value)
const selectedProjectIndex = ref(-1)
const selectedProjectId = ref('')
const selectedProjectName = computed(() => {
if (selectedProjectIndex.value === -1) {
return '请选择项目'
}
return projectList.value[selectedProjectIndex.value]?.name || '请选择项目'
})
const selectedDates = ref([
{ date: '2026-03-12', info: '分段' },
{ date: '2026-03-14', info: '分段' },
{ date: '2026-03-16', info: '分段' },
{ date: '2026-03-19', info: '分段' },
{ date: '2026-03-24', info: '分段' }
])
const studentList = ref<Student[]>([])
const selectedStudentIndexes = ref<number[]>([])
const selectedStudentIds = ref<string[]>([])
const showStudentSelect = ref(false)
const selectedStudentDisplay = computed(() => {
if (selectedStudentIndexes.value.length === 0) {
return '请选择学生'
} else {
const names = selectedStudentIndexes.value.map(index => studentList.value[index].name)
return names.length > 2 ? `${names.slice(0, 2).join(', ')}${names.length}` : names.join(', ')
}
})
const tableData = ref<TableDataItem[]>([])
const mockData: Record<string, Record<string, any[]>> = {
'1': {
'2026-03-12': [
{ name: '王小明', projectName: '100米'自由泳', segment20: 0.037, segment70: 0.128, segment120: 0.219, segment170: 0.311 },
{ name: '张小红', projectName: '100'自由泳', segment20: 0.042, segment70: 0.145, segment120: 0.248, segment170: 0.352 },
{ name: '李小龙', projectName: '100米'自由泳', segment20: 0.032, segment70: 0.112, segment120: 0.192, segment170: 0.272 }
],
'2026-03-14': [
{ name: '王小明', projectName: '100'自由泳', segment20: 0.035, segment70: 0.122, segment120: 0.209, segment170: 0.296 },
{ name: '张小红', projectName: '100米'自由泳', segment20: 0.040, segment70: 0.138, segment120: 0.236, segment170: 0.334 }
]
},
'2': {
'2026-03-16': [
{ name: '赵小芳', projectName: '200'自由泳', segment20: 0.038, segment70: 0.133, segment120: 0.228, segment170: 0.323 }
]
},
'3': {
'2026-03-19': [
{ name: '陈小刚', projectName: '50米蛙泳', segment20: 0.044, segment70: 0.154, segment120: 0.264, segment170: 0.374 }
]
},
'4': {
'2026-03-24': [
{ name: '周小丽', projectName: '100米'蝶泳', segment20: 0.041, segment70: 0.143, segment120: 0.245, segment170: 0.347 }
]
}
}
const mockStudents: Record<string, Student[]> = {
'1': [
{ id: 's1', name: '王小明', gender: '', age: 12 },
{ id: 's2', name: '张小红', gender: '', age: 11 },
{ id: 's3', name: '李小龙', gender: '', age: 13 }
],
'2': [
{ id: 's4', name: '赵小芳', gender: '', age: 10 }
],
'3': [
{ id: 's5', name: '陈小刚', gender: '', age: 14 }
],
'4': [
{ id: 's6', name: '周小丽', gender: '', age: 12 }
]
}
onLoad(() => {
loadData()
})
onShow(() => {
})
const loadData = () => {
}
const handleProjectChange = (e: any) => {
const index = e.detail.value
selectedProjectIndex.value = index
const selectedProject = projectList.value[index]
if (selectedProject) {
selectedProjectId.value = selectedProject.id
selectedDate.value = ''
selectedStudentIndexes.value = []
selectedStudentIds.value = []
tableData.value = []
loadProjectStudents(selectedProject.id)
}
}
const handleMonthSwitch = (e: any) => {
currentYear.value = e.year
currentMonth.value = e.month
loadData()
selectedDate.value = ''
selectedStudentIndexes.value = []
selectedStudentIds.value = []
tableData.value = []
}
const handleDateChange = (e: any) => {
const date = e.fulldate
selectedDate.value = date
selectedStudentIndexes.value = []
selectedStudentIds.value = []
tableData.value = []
}
const loadProjectStudents = (projectId: string) => {
const students = mockStudents[projectId] || []
studentList.value = students
console.log(`项目 ${projectId} 的学生列表加载完成,共 ${students.length} 位学生`)
}
const selectAllStudents = () => {
selectedStudentIndexes.value = studentList.value.map((_, index) => index)
}
const toggleStudentSelect = (index: number) => {
const idx = selectedStudentIndexes.value.indexOf(index)
if (idx > -1) {
selectedStudentIndexes.value.splice(idx, 1)
} else {
selectedStudentIndexes.value.push(index)
}
}
const confirmStudentSelect = () => {
selectedStudentIds.value = selectedStudentIndexes.value.map(index => studentList.value[index].id)
showStudentSelect.value = false
filterAndLoadData()
}
const closeStudentSelect = () => {
showStudentSelect.value = false
}
const filterAndLoadData = () => {
if (!selectedDate.value || selectedStudentIndexes.value.length === 0) {
tableData.value = []
return
}
const projectData = mockData[selectedProjectId.value]
if (!projectData) {
tableData.value = []
return
}
const dateData = projectData[selectedDate.value]
if (!dateData) {
tableData.value = []
return
}
const selectedStudentNames = selectedStudentIndexes.value.map(index => studentList.value[index].name)
const filteredData = dateData.filter(item => selectedStudentNames.includes(item.name))
tableData.value = filteredData.map(item => ({
date: selectedDate.value,
name: item.name,
projectName: item.projectName,
segment20: item.segment20.toFixed(3),
segment70: item.segment70.toFixed(3),
segment120: item.segment120.toFixed(3),
segment170: item.segment170.toFixed(3)
}))
}
</script>
<style lang="scss" scoped>
page {
background-color: #f5f5f5;
}
.paragraph-container {
min-height: 100vh;
padding-bottom: 40rpx;
}
/* ==================== 页面标题区域) ==================== */
.header-section {
background-color: #fff;
padding: 32rpx 28rpx 24rpx;
margin-bottom: 20rpx;
.header-title {
.title {
font-size: 36rpx;
font-weight: 700;
color: #333;
display: block;
margin-bottom: 8rpx;
}
.subtitle {
font-size: 24rpx;
color: #999;
}
}
}
/* ==================== 选择区域通用样式 ==================== */
.select-section {
background-color: #fff;
margin: 0 20rpx 20rpx;
border-radius: 20rpx;
padding: 28rpx;
box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.05);
position: relative;
overflow: hidden;
transition: all 0.3s ease;
.select-label {
margin-bottom: 16rpx;
.label-text {
font-size: 26rpx;
font-weight: 600;
color: #666;
}
}
.picker-wrapper {
display: flex;
align-items: center;
justify-content: space-between;
padding: 20rpx 24rpx;
background-color: #f8f8f8;
border-radius: 12rpx;
border: 1rpx solid #e8e8e8;
transition: all 0.3s ease;
&:active {
background-color: #f0f0f0;
border-color: #1890ff;
transform: scale(0.98);
}
.picker-text {
font-size: 28rpx;
color: #333;
}
}
}
/* ==================== 日历组件包装 ==================== */
.calendar-wrapper {
background-color: #fff;
margin: 0 20rpx 20rpx;
padding: 20rpx 0;
border-radius: 20rpx;
box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.05);
}
/* ==================== 学生选择器弹窗 ==================== */
.student-picker-modal {
position: fixed;
left: 0;
right: 0;
bottom: 0;
height: 70vh;
background: linear-gradient(180deg, #fafbfc 0%, #fff 100%);
border-radius: 32rpx 32rpx 0 0;
z-index: 999;
animation: slideUp 0.35s cubic-bezier(0.34, 1.56, 0.64, 1);
box-shadow: 0 -8rpx 40rpx rgba(0, 0, 0, 0.12);
display: flex;
flex-direction: column;
overflow: hidden;
.picker-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 36rpx 32rpx 28rpx;
background: linear-gradient(180deg, #fafbfc 0%, #fff 100%);
border-bottom: 1rpx solid rgba(0, 0, 0, 0.06);
flex-shrink: 0;
.header-title {
font-size: 36rpx;
font-weight: 700;
background: linear-gradient(135deg, #1890ff 0%, #40a9ff 100%);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.header-actions {
display: flex;
align-items: center;
gap: 16rpx;
.action-btn {
font-size: 26rpx;
color: #666;
padding: 14rpx 28rpx;
border-radius: 24rpx;
background: linear-gradient(135deg, #f5f5f5 0%, #f0f0f0 100%);
font-weight: 600;
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.08);
&.primary {
color: #fff;
background: linear-gradient(135deg, #1890ff 0%, #40a9ff 50%, #096dd9 100%);
box-shadow: 0 4rpx 16rpx rgba(24, 144, 255, 0.3);
}
&:active {
transform: scale(0.92);
}
}
}
}
.student-list {
flex: 1;
overflow-y: auto;
padding: 0 20rpx 40rpx;
.student-item {
display: flex;
align-items: center;
gap: 20rpx;
padding: 24rpx;
background: linear-gradient(135deg, #fff 0%, #fafbfc 100%);
border-radius: 20rpx;
margin-top: 16rpx;
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.06), 0 0 0 1rpx rgba(0, 0kt, 0, 0.04) inset;
position: relative;
overflow: hidden;
&:active {
transform: scale(0.98);
}
.item-checkbox {
flex-shrink: 0;
position: relative;
z-index: 1;
.checkbox-inner {
width: 40rpx;
height: 40rpx;
border-radius: 10rpx;
border: 2rpx solid #d9d9d9;
background-color: #fff;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
&.checked {
border-color: #1890ff;
background: linear-gradient(135deg, #1890ff 0%, #40a9ff 100%);
box-shadow: 0 2rpx 8rpx rgba(24, 144, 255, 0.3);
}
}
}
.item-avatar {
flex-shrink: 0;
position: relative;
z-index: 1;
.avatar-circle {
width: 72rpx;
height: 72rpx;
border-radius: 18rpx;
display: flex;
align-items: center;
justify-content: center;
box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.1);
position: relative;
overflow: hidden;
transition: all 0.3s ease;
&.male {
background: linear-gradient(135deg, #1890ffkt 0%, #40a9ff 50%, #096dd9 100%);
}
&.female {
background: linear-gradient(135deg, #fa8c16 0%, #ffa940 50%, #d46b08 100%);
}
&::before {
content: '';
position: absolute;
top: 8rpx;
right: 8rpx;
width: 12rpx;
height: 12rpx;
background: rgba(255, 255, 255, 0.3);
border-radius: 50%;
}
.avatar-text {
font-size: 34rpx;
color: #fff;
font-weight: 700;
letter-spacing: 2rpx;
}
}
}
.item-info {
flex: 1;
min-width: 0;
display: flex;
flex-direction: column;
gap: 10rpx;
position: relative;
z-index: 1;
.item-name {
font-size: 30rpx;
font-weight: 600;
color: #333;
letter-spacing: 1rpx;
}
.item-meta {
display: flex;
align-items: center;
gap: 12rpx;
.gender-badge {
font-size: 22rpx;
padding: 6rpx 14rpx;
border-radius: 10rpx;
font-weight: 500;
&.male {
color: #1890ff;
background: linear-gradient(135deg, rgba(24, 144, 255, 0.1) 0%, rgba(64, 169, 255, 0.05) 100%);
}
&.female {
color color: #fa8c16;
background: linear-gradient(135deg, rgba(250, 140, 22, 0.1) 0%, rgba(255, 169, 64, 0.05) 100%);
}
}
.age-text {
font-size: 22rpx;
color: #999;
}
}
}
}
}
}
.modal-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: linear-gradient(180deg, rgba(0, 0, 0, 0.6) 0%, rgba(0, 0, 0, 0.4) 100%);
backdrop-filter: blur(10rpx);
z-index: 998;
animation: fadeIn 0.3s ease;
}
/* ==================== 表格区域 ==================== */
.table-section {
margin: 0 20rpx;
background-color: #fff;
border-radius: 20rpx;
overflow: hidden;
box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.05);
.table-wrapper-scroll {
width: 100%;
overflow-x: auto;
}
.table-header {
background: linear-gradient(135deg, #1890ff 0%, #096dd9 100%);
.header-row {
display: flex;
.header-cell {
min-width: 100px;
padding: 24rpx 16rpx;
font-size: 28rpx;
font-weight: 700;
color: #fff;
text-align: center;
border-right: 1rpx solid rgba(255, 255, 255, 0.1);
flex-shrink: 0;
&:last-child {
border-right: none;
}
}
}
}
.table-body {
.body-row {
display: flex;
transition: all 0.3s ease;
&.even {
background-color: #fafafa;
}
&.odd {
background-color: #fff;
}
&:active {
background-color: #e6f7ff;
}
.body-cell {
min-width: 100px;
padding: 24rpx 16rpx;
font-size: 26rpx;
color: #333;
text-align: center;
border-right: 1rpx solid #f0f0f0;
flex-shrink: 0;
&:last-child {
border-right: none;
}
}
}
}
}
/* ==================== 空状态容器 ==================== */
.empty-container {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
margin: 0 20rpx;
background-color: #fff;
border-radius: 20rpx;
padding: 80rpx 40rpx;
box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.05);
.empty-icon {
margin-bottom: 24rpx;
}
.empty-text {
font-size: 28rpx;
color: #999;
}
}
@keyframes fadeIn {
from {
opacity: 0;
}
to {
opacity: 1;
}
}
@keyframes slideUp {
from {
transform: translateY(100%);
}
to {
transform: translateY(0);
}
}
</style>