891 lines
24 KiB
TypeScript
891 lines
24 KiB
TypeScript
import dayjs from 'dayjs';
|
||
import { LinearGradient } from 'expo-linear-gradient';
|
||
import { router } from 'expo-router';
|
||
import React, { useEffect, useState } from 'react';
|
||
import {
|
||
ActivityIndicator,
|
||
Animated,
|
||
Dimensions,
|
||
Modal,
|
||
ScrollView,
|
||
StyleSheet,
|
||
Text,
|
||
TouchableOpacity,
|
||
View
|
||
} from 'react-native';
|
||
import Svg, { Circle } from 'react-native-svg';
|
||
import { Ionicons } from '@expo/vector-icons';
|
||
|
||
import { HeaderBar } from '@/components/ui/HeaderBar';
|
||
import { Colors } from '@/constants/Colors';
|
||
import { useColorScheme } from '@/hooks/useColorScheme';
|
||
import {
|
||
fetchSleepDetailForDate,
|
||
formatSleepTime,
|
||
formatTime,
|
||
getSleepStageColor,
|
||
getSleepStageDisplayName,
|
||
SleepDetailData,
|
||
SleepStage
|
||
} from '@/services/sleepService';
|
||
import { ensureHealthPermissions } from '@/utils/health';
|
||
|
||
const { width } = Dimensions.get('window');
|
||
|
||
// 圆形进度条组件
|
||
const CircularProgress = ({
|
||
size,
|
||
strokeWidth,
|
||
progress,
|
||
color,
|
||
backgroundColor = '#E5E7EB'
|
||
}: {
|
||
size: number;
|
||
strokeWidth: number;
|
||
progress: number; // 0-100
|
||
color: string;
|
||
backgroundColor?: string;
|
||
}) => {
|
||
const radius = (size - strokeWidth) / 2;
|
||
const circumference = radius * 2 * Math.PI;
|
||
const strokeDasharray = circumference;
|
||
const strokeDashoffset = circumference - (progress / 100) * circumference;
|
||
|
||
return (
|
||
<Svg width={size} height={size} style={{ transform: [{ rotateZ: '-90deg' }] }}>
|
||
{/* 背景圆环 */}
|
||
<Circle
|
||
cx={size / 2}
|
||
cy={size / 2}
|
||
r={radius}
|
||
stroke={backgroundColor}
|
||
strokeWidth={strokeWidth}
|
||
fill="none"
|
||
/>
|
||
{/* 进度圆环 */}
|
||
<Circle
|
||
cx={size / 2}
|
||
cy={size / 2}
|
||
r={radius}
|
||
stroke={color}
|
||
strokeWidth={strokeWidth}
|
||
fill="none"
|
||
strokeDasharray={strokeDasharray}
|
||
strokeDashoffset={strokeDashoffset}
|
||
strokeLinecap="round"
|
||
/>
|
||
</Svg>
|
||
);
|
||
};
|
||
|
||
// 睡眠阶段图表组件
|
||
const SleepStageChart = ({ sleepData }: { sleepData: SleepDetailData }) => {
|
||
const chartWidth = width - 80;
|
||
const maxHeight = 120;
|
||
|
||
// 生成24小时的睡眠阶段数据(模拟数据,实际应根据真实样本计算)
|
||
const hourlyData = Array.from({ length: 24 }, (_, hour) => {
|
||
// 如果没有数据,显示空状态
|
||
if (sleepData.totalSleepTime === 0) {
|
||
return null;
|
||
}
|
||
|
||
// 根据时间判断可能的睡眠状态
|
||
if (hour >= 0 && hour <= 6) {
|
||
// 凌晨0-6点,主要睡眠时间
|
||
if (hour <= 2) return SleepStage.Core;
|
||
if (hour <= 4) return SleepStage.Deep;
|
||
return SleepStage.REM;
|
||
} else if (hour >= 22) {
|
||
// 晚上10点后开始入睡
|
||
return SleepStage.Core;
|
||
}
|
||
return null; // 清醒时间
|
||
});
|
||
|
||
return (
|
||
<View style={styles.chartContainer}>
|
||
<View style={styles.chartHeader}>
|
||
<View style={styles.chartTimeLabel}>
|
||
<Text style={styles.chartTimeText}>🛏️ {sleepData.totalSleepTime > 0 ? formatTime(sleepData.bedtime) : '--:--'}</Text>
|
||
</View>
|
||
<View style={styles.chartHeartRate}>
|
||
<Text style={styles.chartHeartRateText}>❤️ 平均心率: {sleepData.averageHeartRate || '--'} BPM</Text>
|
||
</View>
|
||
<View style={styles.chartTimeLabel}>
|
||
<Text style={styles.chartTimeText}>☀️ {sleepData.totalSleepTime > 0 ? formatTime(sleepData.wakeupTime) : '--:--'}</Text>
|
||
</View>
|
||
</View>
|
||
|
||
<View style={styles.chartBars}>
|
||
{hourlyData.map((stage, index) => {
|
||
const barHeight = stage ? Math.random() * 0.6 + 0.4 : 0.1; // 随机高度模拟真实数据
|
||
const color = stage ? getSleepStageColor(stage) : '#E5E7EB';
|
||
|
||
return (
|
||
<View
|
||
key={index}
|
||
style={[
|
||
styles.chartBar,
|
||
{
|
||
height: barHeight * maxHeight,
|
||
backgroundColor: color,
|
||
width: chartWidth / 24 - 2,
|
||
}
|
||
]}
|
||
/>
|
||
);
|
||
})}
|
||
</View>
|
||
</View>
|
||
);
|
||
};
|
||
|
||
// Sleep Grade Component 睡眠等级组件
|
||
const SleepGradeCard = ({
|
||
icon,
|
||
grade,
|
||
range,
|
||
isActive = false
|
||
}: {
|
||
icon: string;
|
||
grade: string;
|
||
range: string;
|
||
isActive?: boolean;
|
||
}) => {
|
||
const theme = (useColorScheme() ?? 'light') as 'light' | 'dark';
|
||
const colorTokens = Colors[theme];
|
||
|
||
const getGradeColor = (grade: string) => {
|
||
switch (grade) {
|
||
case '低': return { bg: '#FECACA', text: '#DC2626' };
|
||
case '正常': return { bg: '#D1FAE5', text: '#065F46' };
|
||
case '良好': return { bg: '#D1FAE5', text: '#065F46' };
|
||
case '优秀': return { bg: '#FEF3C7', text: '#92400E' };
|
||
default: return { bg: colorTokens.pageBackgroundEmphasis, text: colorTokens.textSecondary };
|
||
}
|
||
};
|
||
|
||
const colors = getGradeColor(grade);
|
||
|
||
return (
|
||
<View style={[
|
||
styles.gradeCard,
|
||
{
|
||
backgroundColor: isActive ? colors.bg : colorTokens.pageBackgroundEmphasis,
|
||
borderColor: isActive ? colors.text : 'transparent',
|
||
}
|
||
]}>
|
||
<View style={styles.gradeCardLeft}>
|
||
<Text style={[styles.gradeIcon, { color: colors.text }]}>{icon}</Text>
|
||
<Text style={[
|
||
styles.gradeText,
|
||
{ color: isActive ? colors.text : colorTokens.textSecondary }
|
||
]}>
|
||
{grade}
|
||
</Text>
|
||
</View>
|
||
<Text style={[
|
||
styles.gradeRange,
|
||
{ color: isActive ? colors.text : colorTokens.textSecondary }
|
||
]}>
|
||
{range}
|
||
</Text>
|
||
</View>
|
||
);
|
||
};
|
||
|
||
// Info Modal 组件
|
||
const InfoModal = ({
|
||
visible,
|
||
onClose,
|
||
title,
|
||
type
|
||
}: {
|
||
visible: boolean;
|
||
onClose: () => void;
|
||
title: string;
|
||
type: 'sleep-time' | 'sleep-quality';
|
||
}) => {
|
||
const theme = (useColorScheme() ?? 'light') as 'light' | 'dark';
|
||
const colorTokens = Colors[theme];
|
||
const slideAnim = useState(new Animated.Value(0))[0];
|
||
|
||
React.useEffect(() => {
|
||
if (visible) {
|
||
Animated.spring(slideAnim, {
|
||
toValue: 1,
|
||
useNativeDriver: true,
|
||
tension: 100,
|
||
friction: 8,
|
||
}).start();
|
||
} else {
|
||
Animated.spring(slideAnim, {
|
||
toValue: 0,
|
||
useNativeDriver: true,
|
||
tension: 100,
|
||
friction: 8,
|
||
}).start();
|
||
}
|
||
}, [visible]);
|
||
|
||
const translateY = slideAnim.interpolate({
|
||
inputRange: [0, 1],
|
||
outputRange: [300, 0],
|
||
});
|
||
|
||
const opacity = slideAnim.interpolate({
|
||
inputRange: [0, 1],
|
||
outputRange: [0, 1],
|
||
});
|
||
|
||
const sleepTimeGrades = [
|
||
{ icon: '⚠️', grade: '低', range: '< 6h', isActive: false },
|
||
{ icon: '✅', grade: '正常', range: '6h - 7h or > 9h', isActive: false },
|
||
{ icon: '✅', grade: '良好', range: '7h - 8h', isActive: true },
|
||
{ icon: '⭐', grade: '优秀', range: '8h - 9h', isActive: false },
|
||
];
|
||
|
||
const sleepQualityGrades = [
|
||
{ icon: '⚠️', grade: '较差', range: '< 55%', isActive: false },
|
||
{ icon: '✅', grade: '一般', range: '55% - 69%', isActive: false },
|
||
{ icon: '✅', grade: '良好', range: '70% - 84%', isActive: false },
|
||
{ icon: '⭐', grade: '优秀', range: '85% - 100%', isActive: true },
|
||
];
|
||
|
||
const currentGrades = type === 'sleep-time' ? sleepTimeGrades : sleepQualityGrades;
|
||
|
||
const getDescription = () => {
|
||
if (type === 'sleep-time') {
|
||
return '睡眠最重要 - 它占据了你睡眠得分的一半以上。长时间的睡眠可以减少睡眠债务,但是规律的睡眠时间对于高质量的休息至关重要。';
|
||
} else {
|
||
return '睡眠质量综合评估您的睡眠效率、深度睡眠时长、REM睡眠比例等多个指标。高质量的睡眠不仅仅取决于时长,还包括睡眠的连续性和各睡眠阶段的平衡。';
|
||
}
|
||
};
|
||
|
||
return (
|
||
<Modal
|
||
transparent
|
||
visible={visible}
|
||
animationType="none"
|
||
onRequestClose={onClose}
|
||
>
|
||
<TouchableOpacity
|
||
style={styles.modalOverlay}
|
||
activeOpacity={1}
|
||
onPress={onClose}
|
||
>
|
||
<Animated.View style={[
|
||
styles.infoModalContent,
|
||
{
|
||
backgroundColor: colorTokens.background,
|
||
transform: [{ translateY }],
|
||
opacity,
|
||
}
|
||
]}>
|
||
<View style={styles.modalHandle} />
|
||
<View style={styles.infoModalHeader}>
|
||
<Text style={[styles.infoModalTitle, { color: colorTokens.text }]}>
|
||
{title}
|
||
</Text>
|
||
<TouchableOpacity onPress={onClose} style={styles.infoModalCloseButton}>
|
||
<Ionicons name="close" size={20} color={colorTokens.textSecondary} />
|
||
</TouchableOpacity>
|
||
</View>
|
||
|
||
{/* 等级卡片区域 */}
|
||
<View style={styles.gradesContainer}>
|
||
{currentGrades.map((grade, index) => (
|
||
<SleepGradeCard
|
||
key={index}
|
||
icon={grade.icon}
|
||
grade={grade.grade}
|
||
range={grade.range}
|
||
isActive={grade.isActive}
|
||
/>
|
||
))}
|
||
</View>
|
||
|
||
<Text style={[styles.infoModalText, { color: colorTokens.textSecondary }]}>
|
||
{getDescription()}
|
||
</Text>
|
||
</Animated.View>
|
||
</TouchableOpacity>
|
||
</Modal>
|
||
);
|
||
};
|
||
|
||
export default function SleepDetailScreen() {
|
||
const theme = (useColorScheme() ?? 'light') as 'light' | 'dark';
|
||
const colorTokens = Colors[theme];
|
||
const [sleepData, setSleepData] = useState<SleepDetailData | null>(null);
|
||
const [loading, setLoading] = useState(true);
|
||
const [selectedDate] = useState(dayjs().toDate());
|
||
const [infoModal, setInfoModal] = useState<{ visible: boolean; title: string; type: 'sleep-time' | 'sleep-quality' | null }>({
|
||
visible: false,
|
||
title: '',
|
||
type: null
|
||
});
|
||
|
||
useEffect(() => {
|
||
loadSleepData();
|
||
}, [selectedDate]);
|
||
|
||
const loadSleepData = async () => {
|
||
try {
|
||
setLoading(true);
|
||
|
||
// 确保有健康权限
|
||
const hasPermission = await ensureHealthPermissions();
|
||
if (!hasPermission) {
|
||
console.warn('没有健康数据权限');
|
||
return;
|
||
}
|
||
|
||
// 获取睡眠详情数据
|
||
const data = await fetchSleepDetailForDate(selectedDate);
|
||
setSleepData(data);
|
||
|
||
} catch (error) {
|
||
console.error('加载睡眠数据失败:', error);
|
||
} finally {
|
||
setLoading(false);
|
||
}
|
||
};
|
||
|
||
if (loading) {
|
||
return (
|
||
<View style={[styles.container, styles.loadingContainer]}>
|
||
<ActivityIndicator size="large" color={Colors.light.primary} />
|
||
<Text style={styles.loadingText}>加载睡眠数据中...</Text>
|
||
</View>
|
||
);
|
||
}
|
||
|
||
// 如果没有数据,使用默认数据结构
|
||
const displayData: SleepDetailData = sleepData || {
|
||
sleepScore: 0,
|
||
totalSleepTime: 0,
|
||
sleepQualityPercentage: 0,
|
||
bedtime: new Date().toISOString(),
|
||
wakeupTime: new Date().toISOString(),
|
||
timeInBed: 0,
|
||
sleepStages: [],
|
||
averageHeartRate: null,
|
||
sleepHeartRateData: [],
|
||
sleepEfficiency: 0,
|
||
qualityDescription: '暂无睡眠数据',
|
||
recommendation: '请确保在真实iOS设备上运行并授权访问健康数据,或等待有睡眠数据后再查看。'
|
||
};
|
||
|
||
return (
|
||
<View style={styles.container}>
|
||
{/* 背景渐变 */}
|
||
<LinearGradient
|
||
colors={['#f0f4ff', '#e6f2ff', '#ffffff']}
|
||
style={styles.gradientBackground}
|
||
start={{ x: 0, y: 0 }}
|
||
end={{ x: 0, y: 1 }}
|
||
/>
|
||
|
||
{/* 顶部导航 */}
|
||
<HeaderBar
|
||
title={`今天, ${dayjs(selectedDate).format('M月DD日')}`}
|
||
onBack={() => router.back()}
|
||
withSafeTop={true}
|
||
transparent={true}
|
||
/>
|
||
|
||
<ScrollView
|
||
style={styles.scrollView}
|
||
contentContainerStyle={styles.scrollContent}
|
||
showsVerticalScrollIndicator={false}
|
||
>
|
||
{/* 睡眠得分圆形显示 */}
|
||
<View style={styles.scoreContainer}>
|
||
<View style={styles.circularProgressContainer}>
|
||
<CircularProgress
|
||
size={200}
|
||
strokeWidth={12}
|
||
progress={displayData.sleepScore}
|
||
color="#8B5CF6"
|
||
backgroundColor="#E0E7FF"
|
||
/>
|
||
<View style={styles.scoreTextContainer}>
|
||
<Text style={styles.scoreNumber}>{displayData.sleepScore}</Text>
|
||
<Text style={styles.scoreLabel}>睡眠得分</Text>
|
||
</View>
|
||
</View>
|
||
</View>
|
||
|
||
{/* 睡眠质量描述 */}
|
||
<Text style={styles.qualityDescription}>{displayData.qualityDescription}</Text>
|
||
|
||
{/* 建议文本 */}
|
||
<Text style={styles.recommendationText}>{displayData.recommendation}</Text>
|
||
|
||
{/* 睡眠统计卡片 */}
|
||
<View style={styles.statsContainer}>
|
||
<View style={[styles.newStatCard, { backgroundColor: colorTokens.background }]}>
|
||
<View style={styles.statCardHeader}>
|
||
<View style={styles.statCardIcon}>
|
||
<Text style={styles.statIcon}>🌙</Text>
|
||
</View>
|
||
<TouchableOpacity
|
||
style={styles.infoButton}
|
||
onPress={() => setInfoModal({
|
||
visible: true,
|
||
title: '睡眠时间',
|
||
type: 'sleep-time'
|
||
})}
|
||
>
|
||
<Ionicons name="information-circle-outline" size={16} color={colorTokens.textMuted} />
|
||
</TouchableOpacity>
|
||
</View>
|
||
<Text style={[styles.statLabel, { color: colorTokens.textSecondary }]}>睡眠时间</Text>
|
||
<Text style={[styles.newStatValue, { color: colorTokens.text }]}>
|
||
{displayData.totalSleepTime > 0 ? formatSleepTime(displayData.totalSleepTime) : '7h 23m'}
|
||
</Text>
|
||
<View style={[styles.qualityBadge, styles.goodQualityBadge]}>
|
||
<Text style={[styles.qualityBadgeText, styles.goodQualityText]}>✓ 良好</Text>
|
||
</View>
|
||
</View>
|
||
|
||
<View style={[styles.newStatCard, { backgroundColor: colorTokens.background }]}>
|
||
<View style={styles.statCardHeader}>
|
||
<View style={styles.statCardIcon}>
|
||
<Text style={styles.statIcon}>💎</Text>
|
||
</View>
|
||
<TouchableOpacity
|
||
style={styles.infoButton}
|
||
onPress={() => setInfoModal({
|
||
visible: true,
|
||
title: '睡眠质量',
|
||
type: 'sleep-quality'
|
||
})}
|
||
>
|
||
<Ionicons name="information-circle-outline" size={16} color={colorTokens.textMuted} />
|
||
</TouchableOpacity>
|
||
</View>
|
||
<Text style={[styles.statLabel, { color: colorTokens.textSecondary }]}>睡眠质量</Text>
|
||
<Text style={[styles.newStatValue, { color: colorTokens.text }]}>
|
||
{displayData.sleepQualityPercentage > 0 ? `${displayData.sleepQualityPercentage}%` : '94%'}
|
||
</Text>
|
||
<View style={[styles.qualityBadge, styles.excellentQualityBadge]}>
|
||
<Text style={[styles.qualityBadgeText, styles.excellentQualityText]}>★ 优秀</Text>
|
||
</View>
|
||
</View>
|
||
</View>
|
||
|
||
{/* 睡眠阶段图表 */}
|
||
<SleepStageChart sleepData={displayData} />
|
||
|
||
{/* 睡眠阶段统计 */}
|
||
<View style={styles.stagesContainer}>
|
||
{displayData.sleepStages.length > 0 ? displayData.sleepStages.map((stage, index) => (
|
||
<View key={index} style={styles.stageRow}>
|
||
<View style={styles.stageInfo}>
|
||
<View style={[styles.stageColorDot, { backgroundColor: getSleepStageColor(stage.stage) }]} />
|
||
<Text style={styles.stageName}>{getSleepStageDisplayName(stage.stage)}</Text>
|
||
</View>
|
||
<View style={styles.stageStats}>
|
||
<Text style={styles.stagePercentage}>{stage.percentage}%</Text>
|
||
<Text style={styles.stageDuration}>{formatSleepTime(stage.duration)}</Text>
|
||
<Text style={[
|
||
styles.stageQuality,
|
||
{
|
||
color: stage.quality === 'excellent' ? '#10B981' :
|
||
stage.quality === 'good' ? '#059669' :
|
||
stage.quality === 'fair' ? '#F59E0B' : '#EF4444'
|
||
}
|
||
]}>
|
||
{stage.quality === 'excellent' ? '优秀' :
|
||
stage.quality === 'good' ? '良好' :
|
||
stage.quality === 'fair' ? '一般' : '偏低'}
|
||
</Text>
|
||
</View>
|
||
</View>
|
||
)) : (
|
||
<View style={styles.noDataContainer}>
|
||
<Text style={styles.noDataText}>暂无睡眠阶段数据</Text>
|
||
</View>
|
||
)}
|
||
</View>
|
||
</ScrollView>
|
||
|
||
{infoModal.type && (
|
||
<InfoModal
|
||
visible={infoModal.visible}
|
||
onClose={() => setInfoModal({ ...infoModal, visible: false })}
|
||
title={infoModal.title}
|
||
type={infoModal.type}
|
||
/>
|
||
)}
|
||
</View>
|
||
);
|
||
}
|
||
|
||
const styles = StyleSheet.create({
|
||
container: {
|
||
flex: 1,
|
||
backgroundColor: '#F8FAFC',
|
||
},
|
||
gradientBackground: {
|
||
position: 'absolute',
|
||
left: 0,
|
||
right: 0,
|
||
top: 0,
|
||
bottom: 0,
|
||
},
|
||
scrollView: {
|
||
flex: 1,
|
||
},
|
||
scrollContent: {
|
||
paddingHorizontal: 20,
|
||
paddingBottom: 40,
|
||
},
|
||
scoreContainer: {
|
||
alignItems: 'center',
|
||
marginVertical: 20,
|
||
},
|
||
circularProgressContainer: {
|
||
position: 'relative',
|
||
alignItems: 'center',
|
||
justifyContent: 'center',
|
||
},
|
||
scoreTextContainer: {
|
||
position: 'absolute',
|
||
alignItems: 'center',
|
||
justifyContent: 'center',
|
||
},
|
||
scoreNumber: {
|
||
fontSize: 48,
|
||
fontWeight: '800',
|
||
color: '#1F2937',
|
||
lineHeight: 48,
|
||
},
|
||
scoreLabel: {
|
||
fontSize: 14,
|
||
color: '#6B7280',
|
||
marginTop: 4,
|
||
},
|
||
qualityDescription: {
|
||
fontSize: 18,
|
||
fontWeight: '600',
|
||
color: '#1F2937',
|
||
textAlign: 'center',
|
||
marginBottom: 16,
|
||
lineHeight: 24,
|
||
},
|
||
recommendationText: {
|
||
fontSize: 14,
|
||
color: '#6B7280',
|
||
textAlign: 'center',
|
||
lineHeight: 20,
|
||
marginBottom: 32,
|
||
paddingHorizontal: 16,
|
||
},
|
||
statsContainer: {
|
||
flexDirection: 'row',
|
||
gap: 12,
|
||
marginBottom: 32,
|
||
paddingHorizontal: 4,
|
||
},
|
||
newStatCard: {
|
||
flex: 1,
|
||
borderRadius: 20,
|
||
padding: 20,
|
||
shadowColor: '#000',
|
||
shadowOffset: { width: 0, height: 4 },
|
||
shadowOpacity: 0.08,
|
||
shadowRadius: 12,
|
||
elevation: 4,
|
||
borderWidth: 1,
|
||
borderColor: 'rgba(0, 0, 0, 0.06)',
|
||
},
|
||
statCardHeader: {
|
||
flexDirection: 'row',
|
||
justifyContent: 'space-between',
|
||
alignItems: 'flex-start',
|
||
marginBottom: 8,
|
||
},
|
||
statCardIcon: {
|
||
width: 32,
|
||
height: 32,
|
||
borderRadius: 8,
|
||
backgroundColor: 'rgba(120, 120, 128, 0.08)',
|
||
alignItems: 'center',
|
||
justifyContent: 'center',
|
||
},
|
||
infoButton: {
|
||
padding: 4,
|
||
},
|
||
statCard: {
|
||
flex: 1,
|
||
backgroundColor: 'rgba(255, 255, 255, 0.9)',
|
||
borderRadius: 16,
|
||
padding: 16,
|
||
alignItems: 'center',
|
||
shadowColor: '#000',
|
||
shadowOffset: { width: 0, height: 2 },
|
||
shadowOpacity: 0.1,
|
||
shadowRadius: 8,
|
||
elevation: 3,
|
||
},
|
||
statIcon: {
|
||
fontSize: 18,
|
||
},
|
||
statLabel: {
|
||
fontSize: 12,
|
||
fontWeight: '500',
|
||
marginBottom: 8,
|
||
letterSpacing: 0.2,
|
||
},
|
||
newStatValue: {
|
||
fontSize: 28,
|
||
fontWeight: '700',
|
||
marginBottom: 12,
|
||
letterSpacing: -0.5,
|
||
},
|
||
qualityBadge: {
|
||
paddingHorizontal: 8,
|
||
paddingVertical: 4,
|
||
borderRadius: 8,
|
||
alignSelf: 'flex-start',
|
||
},
|
||
goodQualityBadge: {
|
||
backgroundColor: '#D1FAE5',
|
||
},
|
||
excellentQualityBadge: {
|
||
backgroundColor: '#FEF3C7',
|
||
},
|
||
qualityBadgeText: {
|
||
fontSize: 12,
|
||
fontWeight: '600',
|
||
letterSpacing: 0.1,
|
||
},
|
||
goodQualityText: {
|
||
color: '#065F46',
|
||
},
|
||
excellentQualityText: {
|
||
color: '#92400E',
|
||
},
|
||
statValue: {
|
||
fontSize: 18,
|
||
fontWeight: '700',
|
||
color: '#1F2937',
|
||
marginBottom: 4,
|
||
},
|
||
statQuality: {
|
||
fontSize: 12,
|
||
color: '#10B981',
|
||
fontWeight: '500',
|
||
},
|
||
chartContainer: {
|
||
backgroundColor: 'rgba(255, 255, 255, 0.9)',
|
||
borderRadius: 16,
|
||
padding: 16,
|
||
marginBottom: 24,
|
||
shadowColor: '#000',
|
||
shadowOffset: { width: 0, height: 2 },
|
||
shadowOpacity: 0.1,
|
||
shadowRadius: 8,
|
||
elevation: 3,
|
||
},
|
||
chartHeader: {
|
||
flexDirection: 'row',
|
||
justifyContent: 'space-between',
|
||
alignItems: 'center',
|
||
marginBottom: 16,
|
||
},
|
||
chartTimeLabel: {
|
||
alignItems: 'center',
|
||
},
|
||
chartTimeText: {
|
||
fontSize: 12,
|
||
color: '#6B7280',
|
||
fontWeight: '500',
|
||
},
|
||
chartHeartRate: {
|
||
alignItems: 'center',
|
||
},
|
||
chartHeartRateText: {
|
||
fontSize: 12,
|
||
color: '#EF4444',
|
||
fontWeight: '600',
|
||
},
|
||
chartBars: {
|
||
flexDirection: 'row',
|
||
alignItems: 'flex-end',
|
||
height: 120,
|
||
gap: 2,
|
||
},
|
||
chartBar: {
|
||
borderRadius: 2,
|
||
minHeight: 8,
|
||
},
|
||
stagesContainer: {
|
||
backgroundColor: 'rgba(255, 255, 255, 0.9)',
|
||
borderRadius: 16,
|
||
padding: 16,
|
||
shadowColor: '#000',
|
||
shadowOffset: { width: 0, height: 2 },
|
||
shadowOpacity: 0.1,
|
||
shadowRadius: 8,
|
||
elevation: 3,
|
||
},
|
||
stageRow: {
|
||
flexDirection: 'row',
|
||
alignItems: 'center',
|
||
justifyContent: 'space-between',
|
||
paddingVertical: 12,
|
||
borderBottomWidth: 1,
|
||
borderBottomColor: '#F3F4F6',
|
||
},
|
||
stageInfo: {
|
||
flexDirection: 'row',
|
||
alignItems: 'center',
|
||
flex: 1,
|
||
},
|
||
stageColorDot: {
|
||
width: 12,
|
||
height: 12,
|
||
borderRadius: 6,
|
||
marginRight: 12,
|
||
},
|
||
stageName: {
|
||
fontSize: 14,
|
||
color: '#374151',
|
||
fontWeight: '500',
|
||
},
|
||
stageStats: {
|
||
alignItems: 'flex-end',
|
||
},
|
||
stagePercentage: {
|
||
fontSize: 16,
|
||
fontWeight: '700',
|
||
color: '#1F2937',
|
||
},
|
||
stageDuration: {
|
||
fontSize: 12,
|
||
color: '#6B7280',
|
||
marginTop: 2,
|
||
},
|
||
stageQuality: {
|
||
fontSize: 11,
|
||
fontWeight: '600',
|
||
marginTop: 2,
|
||
},
|
||
loadingContainer: {
|
||
justifyContent: 'center',
|
||
alignItems: 'center',
|
||
},
|
||
loadingText: {
|
||
fontSize: 16,
|
||
color: '#6B7280',
|
||
marginTop: 16,
|
||
},
|
||
errorText: {
|
||
fontSize: 16,
|
||
color: '#6B7280',
|
||
marginBottom: 16,
|
||
},
|
||
retryButton: {
|
||
backgroundColor: Colors.light.primary,
|
||
borderRadius: 8,
|
||
paddingHorizontal: 24,
|
||
paddingVertical: 12,
|
||
},
|
||
retryButtonText: {
|
||
color: '#FFFFFF',
|
||
fontSize: 14,
|
||
fontWeight: '600',
|
||
},
|
||
noDataContainer: {
|
||
alignItems: 'center',
|
||
paddingVertical: 24,
|
||
},
|
||
noDataText: {
|
||
fontSize: 14,
|
||
color: '#9CA3AF',
|
||
fontStyle: 'italic',
|
||
},
|
||
// Info Modal 样式
|
||
modalOverlay: {
|
||
flex: 1,
|
||
backgroundColor: 'rgba(0, 0, 0, 0.5)',
|
||
justifyContent: 'flex-end',
|
||
},
|
||
infoModalContent: {
|
||
borderTopLeftRadius: 24,
|
||
borderTopRightRadius: 24,
|
||
paddingTop: 12,
|
||
paddingHorizontal: 20,
|
||
paddingBottom: 34,
|
||
minHeight: 200,
|
||
shadowColor: '#000',
|
||
shadowOffset: { width: 0, height: -4 },
|
||
shadowOpacity: 0.1,
|
||
shadowRadius: 16,
|
||
elevation: 8,
|
||
},
|
||
modalHandle: {
|
||
width: 36,
|
||
height: 4,
|
||
backgroundColor: '#D1D5DB',
|
||
borderRadius: 2,
|
||
alignSelf: 'center',
|
||
marginBottom: 20,
|
||
},
|
||
infoModalHeader: {
|
||
flexDirection: 'row',
|
||
justifyContent: 'space-between',
|
||
alignItems: 'center',
|
||
marginBottom: 16,
|
||
},
|
||
infoModalTitle: {
|
||
fontSize: 18,
|
||
fontWeight: '700',
|
||
letterSpacing: -0.3,
|
||
},
|
||
infoModalCloseButton: {
|
||
padding: 4,
|
||
},
|
||
infoModalText: {
|
||
fontSize: 15,
|
||
lineHeight: 22,
|
||
letterSpacing: -0.1,
|
||
},
|
||
// Grade Cards 样式
|
||
gradesContainer: {
|
||
marginBottom: 20,
|
||
gap: 8,
|
||
},
|
||
gradeCard: {
|
||
flexDirection: 'row',
|
||
justifyContent: 'space-between',
|
||
alignItems: 'center',
|
||
paddingHorizontal: 16,
|
||
paddingVertical: 12,
|
||
borderRadius: 12,
|
||
borderWidth: 1,
|
||
},
|
||
gradeCardLeft: {
|
||
flexDirection: 'row',
|
||
alignItems: 'center',
|
||
gap: 8,
|
||
},
|
||
gradeIcon: {
|
||
fontSize: 16,
|
||
},
|
||
gradeText: {
|
||
fontSize: 16,
|
||
fontWeight: '600',
|
||
letterSpacing: -0.2,
|
||
},
|
||
gradeRange: {
|
||
fontSize: 16,
|
||
fontWeight: '700',
|
||
letterSpacing: -0.3,
|
||
},
|
||
}); |