Compare commits
12 Commits
1143f8ca02
...
0776517fb7
| Author | SHA1 | Date | |
|---|---|---|---|
| 0776517fb7 | |||
| 064dafad57 | |||
| 5d6166df1b | |||
| aba9b1395b | |||
| c3a41d2b32 | |||
| 254a30ff93 | |||
| 96db3ea3b7 | |||
| 981208ab7b | |||
| a5f761062e | |||
| b1439f059a | |||
| df08ff8950 | |||
| e3a3c3fccf |
Binary file not shown.
@ -7,46 +7,28 @@ import {
|
|||||||
ScrollView,
|
ScrollView,
|
||||||
Alert,
|
Alert,
|
||||||
} from "react-native";
|
} from "react-native";
|
||||||
import { useState, useEffect, useRef } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { useAuth } from "@clerk/clerk-expo";
|
import { useAuth } from "@clerk/clerk-expo";
|
||||||
import { LinearGradient } from "expo-linear-gradient";
|
|
||||||
import { Ionicons } from "@expo/vector-icons";
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
|
import { useTheme } from "../../contexts/ThemeContext";
|
||||||
|
import { MinimalCard } from "../../components/MinimalCard";
|
||||||
|
import { SectionHeader } from "../../components/SectionHeader";
|
||||||
|
import { MinimalButton } from "../../components/MinimalButton";
|
||||||
|
import { Badge } from "../../components/Badge";
|
||||||
|
import { IconContainer } from "../../components/IconContainer";
|
||||||
import { attendanceApi, Attendance } from "../../api/attendance";
|
import { attendanceApi, Attendance } from "../../api/attendance";
|
||||||
import { AttendanceCalendar } from "../../components/AttendanceCalendar";
|
import { AttendanceCalendar } from "../../components/AttendanceCalendar";
|
||||||
import { useStatistics } from "../../contexts/StatisticsContext";
|
import { useStatistics } from "../../contexts/StatisticsContext";
|
||||||
import { theme } from "../../styles/theme";
|
|
||||||
import { Animated } from "react-native";
|
|
||||||
import { getErrorMessage } from "../../utils/error-helpers";
|
import { getErrorMessage } from "../../utils/error-helpers";
|
||||||
import log from "../../utils/logger";
|
import log from "../../utils/logger";
|
||||||
|
|
||||||
export default function AttendanceScreen() {
|
export default function AttendanceScreen() {
|
||||||
const { getToken, userId } = useAuth();
|
const { getToken, userId } = useAuth();
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
const { clearCache: clearStatisticsCache } = useStatistics();
|
const { clearCache: clearStatisticsCache } = useStatistics();
|
||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
const [activeCheckIn, setActiveCheckIn] = useState<Attendance | null>(null);
|
const [activeCheckIn, setActiveCheckIn] = useState<Attendance | null>(null);
|
||||||
const [history, setHistory] = useState<Attendance[]>([]);
|
const [history, setHistory] = useState<Attendance[]>([]);
|
||||||
const pulseAnim = useRef(new Animated.Value(1)).current;
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (activeCheckIn) {
|
|
||||||
const pulse = Animated.loop(
|
|
||||||
Animated.sequence([
|
|
||||||
Animated.timing(pulseAnim, {
|
|
||||||
toValue: 1.05,
|
|
||||||
duration: 1000,
|
|
||||||
useNativeDriver: true,
|
|
||||||
}),
|
|
||||||
Animated.timing(pulseAnim, {
|
|
||||||
toValue: 1,
|
|
||||||
duration: 1000,
|
|
||||||
useNativeDriver: true,
|
|
||||||
}),
|
|
||||||
]),
|
|
||||||
);
|
|
||||||
pulse.start();
|
|
||||||
return () => pulse.stop();
|
|
||||||
}
|
|
||||||
}, [activeCheckIn]);
|
|
||||||
|
|
||||||
const fetchAttendance = async () => {
|
const fetchAttendance = async () => {
|
||||||
try {
|
try {
|
||||||
@ -82,10 +64,7 @@ export default function AttendanceScreen() {
|
|||||||
if (!token) return;
|
if (!token) return;
|
||||||
|
|
||||||
await attendanceApi.checkIn("gym", token);
|
await attendanceApi.checkIn("gym", token);
|
||||||
|
|
||||||
// Clear statistics cache to force refresh on home screen
|
|
||||||
clearStatisticsCache();
|
clearStatisticsCache();
|
||||||
|
|
||||||
fetchAttendance();
|
fetchAttendance();
|
||||||
Alert.alert("Success", "Checked in successfully!");
|
Alert.alert("Success", "Checked in successfully!");
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
@ -100,10 +79,7 @@ export default function AttendanceScreen() {
|
|||||||
if (!token) return;
|
if (!token) return;
|
||||||
|
|
||||||
await attendanceApi.checkOut(token);
|
await attendanceApi.checkOut(token);
|
||||||
|
|
||||||
// Clear statistics cache to force refresh on home screen
|
|
||||||
clearStatisticsCache();
|
clearStatisticsCache();
|
||||||
|
|
||||||
fetchAttendance();
|
fetchAttendance();
|
||||||
Alert.alert("Success", "Checked out successfully!");
|
Alert.alert("Success", "Checked out successfully!");
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
@ -114,143 +90,204 @@ export default function AttendanceScreen() {
|
|||||||
|
|
||||||
if (loading && !history.length) {
|
if (loading && !history.length) {
|
||||||
return (
|
return (
|
||||||
<View style={styles.centered}>
|
<View style={[styles.centered, { backgroundColor: colors.background }]}>
|
||||||
<ActivityIndicator size="large" color={theme.colors.primary} />
|
<ActivityIndicator size="large" color={colors.primary} />
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ScrollView style={styles.container} contentContainerStyle={styles.content}>
|
<ScrollView
|
||||||
<LinearGradient
|
style={[styles.container, { backgroundColor: colors.background }]}
|
||||||
colors={theme.gradients.primary}
|
contentContainerStyle={styles.content}
|
||||||
start={{ x: 0, y: 0 }}
|
>
|
||||||
end={{ x: 1, y: 1 }}
|
{/* Header */}
|
||||||
style={styles.header}
|
<View style={styles.header}>
|
||||||
>
|
<Text
|
||||||
<Text style={styles.title}>Attendance</Text>
|
style={[typography.h1, { color: colors.textPrimary, fontSize: 32 }]}
|
||||||
<Text style={styles.subtitle}>Track your gym visits</Text>
|
>
|
||||||
</LinearGradient>
|
Attendance
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{ color: colors.textSecondary, marginTop: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{activeCheckIn
|
||||||
|
? "You're crushing it today!"
|
||||||
|
: history.length === 0
|
||||||
|
? "Ready to start your fitness journey?"
|
||||||
|
: "Track your gym visits and build streaks!"}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
|
||||||
<View style={styles.actionContainer}>
|
{/* Check In/Out Section */}
|
||||||
|
<View style={styles.section}>
|
||||||
{activeCheckIn ? (
|
{activeCheckIn ? (
|
||||||
<LinearGradient
|
<MinimalCard variant="bordered" style={styles.activeCard}>
|
||||||
colors={["rgba(16, 185, 129, 0.15)", "rgba(5, 150, 105, 0.1)"]}
|
<View style={styles.activeHeader}>
|
||||||
style={[styles.activeCard, theme.shadows.medium]}
|
<View style={styles.activeHeaderLeft}>
|
||||||
>
|
<IconContainer
|
||||||
<View style={styles.activeCardContent}>
|
variant="colored"
|
||||||
<View style={styles.activeIconContainer}>
|
backgroundColor={`${colors.success}20`}
|
||||||
<LinearGradient
|
size="lg"
|
||||||
colors={theme.gradients.success}
|
|
||||||
style={styles.activeIcon}
|
|
||||||
>
|
>
|
||||||
<Ionicons name="checkmark-circle" size={32} color="#fff" />
|
<Ionicons
|
||||||
</LinearGradient>
|
name="checkmark-circle"
|
||||||
</View>
|
size={28}
|
||||||
<View style={styles.activeTextContainer}>
|
color={colors.success}
|
||||||
<Text style={styles.activeText}>Currently Checked In</Text>
|
/>
|
||||||
<Text style={styles.timeText}>
|
</IconContainer>
|
||||||
Since{" "}
|
<View style={{ marginLeft: 12 }}>
|
||||||
{new Date(activeCheckIn.checkInTime).toLocaleTimeString([], {
|
<Text style={[typography.h3, { color: colors.textPrimary }]}>
|
||||||
hour: "2-digit",
|
✅ Currently Checked In
|
||||||
minute: "2-digit",
|
</Text>
|
||||||
})}
|
<Text
|
||||||
</Text>
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textTertiary, marginTop: 2 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Since{" "}
|
||||||
|
{new Date(activeCheckIn.checkInTime).toLocaleTimeString(
|
||||||
|
[],
|
||||||
|
{
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
},
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<TouchableOpacity onPress={handleCheckOut} activeOpacity={0.8}>
|
<MinimalButton
|
||||||
<LinearGradient
|
title="Check Out"
|
||||||
colors={theme.gradients.danger}
|
onPress={handleCheckOut}
|
||||||
start={{ x: 0, y: 0 }}
|
variant="danger"
|
||||||
end={{ x: 1, y: 0 }}
|
size="lg"
|
||||||
style={[styles.checkOutButton, theme.shadows.medium]}
|
style={{ marginTop: 16 }}
|
||||||
>
|
/>
|
||||||
<Ionicons
|
</MinimalCard>
|
||||||
name="log-out-outline"
|
|
||||||
size={20}
|
|
||||||
color="#fff"
|
|
||||||
style={{ marginRight: 8 }}
|
|
||||||
/>
|
|
||||||
<Text style={styles.buttonText}>Check Out</Text>
|
|
||||||
</LinearGradient>
|
|
||||||
</TouchableOpacity>
|
|
||||||
</LinearGradient>
|
|
||||||
) : (
|
) : (
|
||||||
<Animated.View style={{ transform: [{ scale: pulseAnim }] }}>
|
<MinimalButton
|
||||||
<TouchableOpacity onPress={handleCheckIn} activeOpacity={0.8}>
|
title="💪 Check In"
|
||||||
<LinearGradient
|
onPress={handleCheckIn}
|
||||||
colors={theme.gradients.primary}
|
variant="primary"
|
||||||
start={{ x: 0, y: 0 }}
|
size="lg"
|
||||||
end={{ x: 1, y: 0 }}
|
/>
|
||||||
style={[styles.checkInButton, theme.shadows.glow]}
|
|
||||||
>
|
|
||||||
<Ionicons
|
|
||||||
name="log-in-outline"
|
|
||||||
size={24}
|
|
||||||
color="#fff"
|
|
||||||
style={{ marginRight: 8 }}
|
|
||||||
/>
|
|
||||||
<Text style={styles.checkInButtonText}>Check In</Text>
|
|
||||||
</LinearGradient>
|
|
||||||
</TouchableOpacity>
|
|
||||||
</Animated.View>
|
|
||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
{/* Attendance Calendar */}
|
{/* Attendance Calendar */}
|
||||||
{history.length > 0 && <AttendanceCalendar attendanceRecords={history} />}
|
<View style={styles.section}>
|
||||||
|
<SectionHeader title="📅 Calendar" />
|
||||||
|
<MinimalCard variant="default" style={{ borderRadius: 20 }}>
|
||||||
|
<AttendanceCalendar attendanceRecords={history} />
|
||||||
|
</MinimalCard>
|
||||||
|
</View>
|
||||||
|
|
||||||
<Text style={styles.sectionTitle}>Recent History</Text>
|
{/* Recent History */}
|
||||||
{history.map((item) => (
|
<View style={styles.section}>
|
||||||
<LinearGradient
|
<SectionHeader title="📊 Recent History" />
|
||||||
key={item.id}
|
{history.length === 0 ? (
|
||||||
colors={["rgba(255, 255, 255, 1)", "rgba(249, 250, 251, 1)"] as const}
|
<MinimalCard variant="default" style={{ borderRadius: 20 }}>
|
||||||
style={[styles.historyItem, theme.shadows.medium]}
|
<View style={styles.emptyState}>
|
||||||
>
|
<Text style={{ fontSize: 64 }}>📍</Text>
|
||||||
<View style={styles.historyLeft}>
|
<Text
|
||||||
<View style={styles.historyIconContainer}>
|
style={[
|
||||||
<LinearGradient
|
typography.bodyEmphasis,
|
||||||
colors={
|
{ color: colors.textPrimary, marginTop: 16 },
|
||||||
item.checkOutTime
|
]}
|
||||||
? theme.gradients.success
|
|
||||||
: theme.gradients.primary
|
|
||||||
}
|
|
||||||
style={styles.historyIcon}
|
|
||||||
>
|
>
|
||||||
<Ionicons
|
No attendance history yet
|
||||||
name={item.checkOutTime ? "checkmark" : "time-outline"}
|
|
||||||
size={16}
|
|
||||||
color="#fff"
|
|
||||||
/>
|
|
||||||
</LinearGradient>
|
|
||||||
</View>
|
|
||||||
<View>
|
|
||||||
<Text style={styles.dateText}>
|
|
||||||
{new Date(item.checkInTime).toLocaleDateString()}
|
|
||||||
</Text>
|
</Text>
|
||||||
<Text style={styles.typeText}>{item.type.toUpperCase()}</Text>
|
<Text
|
||||||
</View>
|
style={[
|
||||||
</View>
|
typography.body,
|
||||||
<View style={styles.timeContainer}>
|
{
|
||||||
<Text style={styles.historyTime}>
|
color: colors.textSecondary,
|
||||||
In:{" "}
|
marginTop: 8,
|
||||||
{new Date(item.checkInTime).toLocaleTimeString([], {
|
textAlign: "center",
|
||||||
hour: "2-digit",
|
},
|
||||||
minute: "2-digit",
|
]}
|
||||||
})}
|
>
|
||||||
</Text>
|
Check in to start building your streak! 🔥
|
||||||
{item.checkOutTime && (
|
|
||||||
<Text style={styles.historyTime}>
|
|
||||||
Out:{" "}
|
|
||||||
{new Date(item.checkOutTime).toLocaleTimeString([], {
|
|
||||||
hour: "2-digit",
|
|
||||||
minute: "2-digit",
|
|
||||||
})}
|
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
</View>
|
||||||
|
</MinimalCard>
|
||||||
|
) : (
|
||||||
|
<View style={styles.historyList}>
|
||||||
|
{history.slice(0, 10).map((record, index) => {
|
||||||
|
const checkIn = new Date(record.checkInTime);
|
||||||
|
const checkOut = record.checkOutTime
|
||||||
|
? new Date(record.checkOutTime)
|
||||||
|
: null;
|
||||||
|
const duration = checkOut
|
||||||
|
? Math.round((checkOut.getTime() - checkIn.getTime()) / 60000)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MinimalCard
|
||||||
|
key={index}
|
||||||
|
variant="bordered"
|
||||||
|
style={{ borderRadius: 16 }}
|
||||||
|
>
|
||||||
|
<View style={styles.historyItem}>
|
||||||
|
<View style={styles.historyLeft}>
|
||||||
|
<IconContainer
|
||||||
|
variant="colored"
|
||||||
|
backgroundColor={
|
||||||
|
checkOut ? `${colors.success}20` : `${colors.info}20`
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Ionicons
|
||||||
|
name={checkOut ? "checkmark-done" : "time"}
|
||||||
|
size={20}
|
||||||
|
color={checkOut ? colors.success : colors.info}
|
||||||
|
/>
|
||||||
|
</IconContainer>
|
||||||
|
<View style={{ marginLeft: 12, flex: 1 }}>
|
||||||
|
<Text
|
||||||
|
style={[typography.h3, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
{checkIn.toLocaleDateString()}
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textTertiary, marginTop: 2 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{checkIn.toLocaleTimeString([], {
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
})}
|
||||||
|
{checkOut &&
|
||||||
|
` - ${checkOut.toLocaleTimeString([], {
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
})}`}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
{duration && (
|
||||||
|
<Badge
|
||||||
|
label={`${duration}m`}
|
||||||
|
variant="neutral"
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
</MinimalCard>
|
||||||
|
);
|
||||||
|
})}
|
||||||
</View>
|
</View>
|
||||||
</LinearGradient>
|
)}
|
||||||
))}
|
</View>
|
||||||
|
|
||||||
|
{/* Bottom Spacer */}
|
||||||
|
<View style={{ height: 100 }} />
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -258,147 +295,54 @@ export default function AttendanceScreen() {
|
|||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
backgroundColor: theme.colors.background,
|
|
||||||
},
|
|
||||||
content: {
|
|
||||||
paddingBottom: 20,
|
|
||||||
},
|
},
|
||||||
centered: {
|
centered: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
|
content: {
|
||||||
|
paddingBottom: 20,
|
||||||
|
},
|
||||||
header: {
|
header: {
|
||||||
|
paddingHorizontal: 24,
|
||||||
paddingTop: 60,
|
paddingTop: 60,
|
||||||
paddingBottom: 24,
|
paddingBottom: 24,
|
||||||
|
},
|
||||||
|
section: {
|
||||||
paddingHorizontal: 24,
|
paddingHorizontal: 24,
|
||||||
marginBottom: 24,
|
marginBottom: 24,
|
||||||
borderBottomLeftRadius: theme.borderRadius.xl,
|
|
||||||
borderBottomRightRadius: theme.borderRadius.xl,
|
|
||||||
},
|
|
||||||
title: {
|
|
||||||
fontSize: theme.typography.fontSize["3xl"],
|
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.white,
|
|
||||||
marginBottom: 4,
|
|
||||||
},
|
|
||||||
subtitle: {
|
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
color: "rgba(255, 255, 255, 0.9)",
|
|
||||||
},
|
|
||||||
actionContainer: {
|
|
||||||
marginBottom: 32,
|
|
||||||
paddingHorizontal: 20,
|
|
||||||
},
|
|
||||||
checkInButton: {
|
|
||||||
paddingVertical: 20,
|
|
||||||
paddingHorizontal: 24,
|
|
||||||
borderRadius: theme.borderRadius.xl,
|
|
||||||
flexDirection: "row",
|
|
||||||
alignItems: "center",
|
|
||||||
justifyContent: "center",
|
|
||||||
},
|
|
||||||
checkInButtonText: {
|
|
||||||
color: theme.colors.white,
|
|
||||||
fontSize: theme.typography.fontSize.xl,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
},
|
|
||||||
checkOutButton: {
|
|
||||||
paddingVertical: 14,
|
|
||||||
paddingHorizontal: 20,
|
|
||||||
borderRadius: theme.borderRadius.lg,
|
|
||||||
flexDirection: "row",
|
|
||||||
alignItems: "center",
|
|
||||||
justifyContent: "center",
|
|
||||||
marginTop: 16,
|
|
||||||
},
|
|
||||||
buttonText: {
|
|
||||||
color: theme.colors.white,
|
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
},
|
},
|
||||||
activeCard: {
|
activeCard: {
|
||||||
padding: 20,
|
padding: 20,
|
||||||
borderRadius: theme.borderRadius.xl,
|
borderRadius: 20,
|
||||||
borderWidth: 1,
|
|
||||||
borderColor: "rgba(16, 185, 129, 0.2)",
|
|
||||||
},
|
},
|
||||||
activeCardContent: {
|
activeHeader: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
alignItems: "center",
|
justifyContent: "space-between",
|
||||||
marginBottom: 16,
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
activeIconContainer: {
|
activeHeaderLeft: {
|
||||||
marginRight: 16,
|
flexDirection: "row",
|
||||||
},
|
alignItems: "center",
|
||||||
activeIcon: {
|
flex: 1,
|
||||||
width: 56,
|
},
|
||||||
height: 56,
|
emptyState: {
|
||||||
borderRadius: 28,
|
alignItems: "center",
|
||||||
justifyContent: "center",
|
paddingVertical: 40,
|
||||||
alignItems: "center",
|
paddingHorizontal: 20,
|
||||||
},
|
},
|
||||||
activeTextContainer: {
|
historyList: {
|
||||||
flex: 1,
|
gap: 12,
|
||||||
},
|
},
|
||||||
activeText: {
|
historyItem: {
|
||||||
fontSize: theme.typography.fontSize.lg,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
color: theme.colors.gray900,
|
|
||||||
marginBottom: 4,
|
|
||||||
},
|
|
||||||
timeText: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
color: theme.colors.gray600,
|
|
||||||
},
|
|
||||||
sectionTitle: {
|
|
||||||
fontSize: theme.typography.fontSize.xl,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
marginBottom: 16,
|
|
||||||
paddingHorizontal: 20,
|
|
||||||
color: theme.colors.gray900,
|
|
||||||
},
|
|
||||||
historyItem: {
|
|
||||||
padding: 16,
|
|
||||||
borderRadius: theme.borderRadius.xl,
|
|
||||||
marginBottom: 12,
|
|
||||||
marginHorizontal: 20,
|
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
borderWidth: 1,
|
|
||||||
borderColor: "rgba(59, 130, 246, 0.1)",
|
|
||||||
},
|
},
|
||||||
historyLeft: {
|
historyLeft: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
gap: 12,
|
flex: 1,
|
||||||
},
|
|
||||||
historyIconContainer: {
|
|
||||||
marginRight: 4,
|
|
||||||
},
|
|
||||||
historyIcon: {
|
|
||||||
width: 32,
|
|
||||||
height: 32,
|
|
||||||
borderRadius: 16,
|
|
||||||
justifyContent: "center",
|
|
||||||
alignItems: "center",
|
|
||||||
},
|
|
||||||
dateText: {
|
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
color: theme.colors.gray900,
|
|
||||||
},
|
|
||||||
typeText: {
|
|
||||||
fontSize: theme.typography.fontSize.xs,
|
|
||||||
color: theme.colors.gray600,
|
|
||||||
marginTop: 2,
|
|
||||||
},
|
|
||||||
timeContainer: {
|
|
||||||
alignItems: "flex-end",
|
|
||||||
},
|
|
||||||
historyTime: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
color: theme.colors.gray700,
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@ -1,4 +1,4 @@
|
|||||||
import React, { useState, useCallback, useRef } from "react";
|
import React, { useState, useCallback } from "react";
|
||||||
import {
|
import {
|
||||||
View,
|
View,
|
||||||
Text,
|
Text,
|
||||||
@ -6,27 +6,31 @@ import {
|
|||||||
ScrollView,
|
ScrollView,
|
||||||
RefreshControl,
|
RefreshControl,
|
||||||
TouchableOpacity,
|
TouchableOpacity,
|
||||||
Animated,
|
|
||||||
Alert,
|
Alert,
|
||||||
} from "react-native";
|
} from "react-native";
|
||||||
import { Ionicons } from "@expo/vector-icons";
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import { LinearGradient } from "expo-linear-gradient";
|
import { useUser } from "@clerk/clerk-expo";
|
||||||
import { theme } from "../../styles/theme";
|
import { useFocusEffect } from "expo-router";
|
||||||
|
import * as SecureStore from "expo-secure-store";
|
||||||
|
import { useTheme } from "../../contexts/ThemeContext";
|
||||||
|
import { MinimalCard } from "../../components/MinimalCard";
|
||||||
|
import { SectionHeader } from "../../components/SectionHeader";
|
||||||
|
import { MinimalButton } from "../../components/MinimalButton";
|
||||||
|
import { Badge } from "../../components/Badge";
|
||||||
|
import { ProgressBar } from "../../components/ProgressBar";
|
||||||
import { GoalProgressCard } from "../../components/GoalProgressCard";
|
import { GoalProgressCard } from "../../components/GoalProgressCard";
|
||||||
import { GoalCreationModal } from "../../components/GoalCreationModal";
|
import { GoalCreationModal } from "../../components/GoalCreationModal";
|
||||||
import { WeeklyProgressChart } from "../../components/WeeklyProgressChart";
|
import { WeeklyProgressChart } from "../../components/WeeklyProgressChart";
|
||||||
import { GoalTypeBreakdownChart } from "../../components/GoalTypeBreakdownChart";
|
import { GoalTypeBreakdownChart } from "../../components/GoalTypeBreakdownChart";
|
||||||
import { useUser } from "@clerk/clerk-expo";
|
|
||||||
import type { FitnessGoal, CreateGoalData } from "../../services/fitnessGoals";
|
import type { FitnessGoal, CreateGoalData } from "../../services/fitnessGoals";
|
||||||
import { useStatistics } from "../../contexts/StatisticsContext";
|
import { useStatistics } from "../../contexts/StatisticsContext";
|
||||||
import { useFitnessGoals } from "../../contexts/FitnessGoalsContext";
|
import { useFitnessGoals } from "../../contexts/FitnessGoalsContext";
|
||||||
import { useRecommendations } from "../../contexts/RecommendationsContext";
|
import { useRecommendations } from "../../contexts/RecommendationsContext";
|
||||||
import { useFocusEffect } from "expo-router";
|
|
||||||
import * as SecureStore from "expo-secure-store";
|
|
||||||
import log from "../../utils/logger";
|
import log from "../../utils/logger";
|
||||||
|
|
||||||
export default function GoalsScreen() {
|
export default function GoalsScreen() {
|
||||||
const { user } = useUser();
|
const { user } = useUser();
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
const {
|
const {
|
||||||
statistics,
|
statistics,
|
||||||
refetchStatistics,
|
refetchStatistics,
|
||||||
@ -45,10 +49,8 @@ export default function GoalsScreen() {
|
|||||||
const [refreshing, setRefreshing] = useState(false);
|
const [refreshing, setRefreshing] = useState(false);
|
||||||
const [isModalVisible, setIsModalVisible] = useState(false);
|
const [isModalVisible, setIsModalVisible] = useState(false);
|
||||||
const [showAnalytics, setShowAnalytics] = useState(false);
|
const [showAnalytics, setShowAnalytics] = useState(false);
|
||||||
const fabScale = useRef(new Animated.Value(1)).current;
|
|
||||||
|
|
||||||
const loadData = useCallback(async () => {
|
const loadData = useCallback(async () => {
|
||||||
// Load goals and statistics (both cached)
|
|
||||||
await refetchGoals();
|
await refetchGoals();
|
||||||
await refetchStatistics();
|
await refetchStatistics();
|
||||||
}, [refetchGoals, refetchStatistics]);
|
}, [refetchGoals, refetchStatistics]);
|
||||||
@ -64,7 +66,6 @@ export default function GoalsScreen() {
|
|||||||
style: "destructive",
|
style: "destructive",
|
||||||
onPress: async () => {
|
onPress: async () => {
|
||||||
try {
|
try {
|
||||||
// Clear all possible Clerk token keys
|
|
||||||
const keysToDelete = [
|
const keysToDelete = [
|
||||||
"__clerk_client_jwt",
|
"__clerk_client_jwt",
|
||||||
"__clerk_db_jwt",
|
"__clerk_db_jwt",
|
||||||
@ -82,7 +83,6 @@ export default function GoalsScreen() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clear all caches
|
|
||||||
clearStatsCache();
|
clearStatsCache();
|
||||||
clearGoalsCache();
|
clearGoalsCache();
|
||||||
clearRecommendationsCache();
|
clearRecommendationsCache();
|
||||||
@ -128,183 +128,310 @@ export default function GoalsScreen() {
|
|||||||
|
|
||||||
const activeGoals = goals?.filter((g) => g.status === "active") || [];
|
const activeGoals = goals?.filter((g) => g.status === "active") || [];
|
||||||
const completedGoals = goals?.filter((g) => g.status === "completed") || [];
|
const completedGoals = goals?.filter((g) => g.status === "completed") || [];
|
||||||
|
const avgProgress =
|
||||||
|
activeGoals.length > 0
|
||||||
|
? Math.round(
|
||||||
|
activeGoals.reduce((sum, g) => sum + (g.progress || 0), 0) /
|
||||||
|
activeGoals.length,
|
||||||
|
)
|
||||||
|
: 0;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={[styles.container, { backgroundColor: colors.background }]}>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
contentContainerStyle={styles.scrollContent}
|
contentContainerStyle={styles.scrollContent}
|
||||||
refreshControl={
|
refreshControl={
|
||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={onRefresh}
|
onRefresh={onRefresh}
|
||||||
tintColor={theme.colors.primary}
|
tintColor={colors.primary}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<LinearGradient colors={theme.gradients.primary} style={styles.header}>
|
{/* Header */}
|
||||||
<View style={styles.headerContent}>
|
<View style={styles.header}>
|
||||||
<View>
|
<View style={{ flex: 1 }}>
|
||||||
<Text style={styles.headerTitle}>Fitness Goals</Text>
|
<Text
|
||||||
<Text style={styles.headerSubtitle}>
|
style={[
|
||||||
Track your fitness journey progress
|
typography.h1,
|
||||||
</Text>
|
{ color: colors.textPrimary, fontSize: 32 },
|
||||||
</View>
|
]}
|
||||||
<TouchableOpacity
|
|
||||||
onPress={clearClerkCache}
|
|
||||||
style={styles.debugButton}
|
|
||||||
>
|
>
|
||||||
<Ionicons name="refresh-circle-outline" size={24} color="#fff" />
|
Goals
|
||||||
</TouchableOpacity>
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{ color: colors.textSecondary, marginTop: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{activeGoals.length === 0
|
||||||
|
? "Ready to crush some goals?"
|
||||||
|
: activeGoals.length === 1
|
||||||
|
? "You're on a mission! Keep it up!"
|
||||||
|
: `${activeGoals.length} goals in progress. Let's go!`}
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
</LinearGradient>
|
<TouchableOpacity
|
||||||
|
onPress={clearClerkCache}
|
||||||
|
style={styles.debugButton}
|
||||||
|
>
|
||||||
|
<Ionicons
|
||||||
|
name="refresh-circle-outline"
|
||||||
|
size={24}
|
||||||
|
color={colors.textTertiary}
|
||||||
|
/>
|
||||||
|
</TouchableOpacity>
|
||||||
|
</View>
|
||||||
|
|
||||||
{/* Stats Summary */}
|
{/* Stats Summary */}
|
||||||
{goals && goals.length > 0 && (
|
{goals && goals.length > 0 && (
|
||||||
<View style={styles.statsContainer}>
|
<View style={styles.section}>
|
||||||
<View style={styles.statCard}>
|
<View style={styles.statsRow}>
|
||||||
<Text style={styles.statValue}>{activeGoals.length}</Text>
|
<MinimalCard
|
||||||
<Text style={styles.statLabel}>Active</Text>
|
variant="elevated"
|
||||||
</View>
|
style={[styles.statCard, { backgroundColor: colors.primary }]}
|
||||||
<View style={styles.statCard}>
|
>
|
||||||
<Text style={styles.statValue}>{completedGoals.length}</Text>
|
<Text
|
||||||
<Text style={styles.statLabel}>Completed</Text>
|
style={[
|
||||||
</View>
|
typography.statLarge,
|
||||||
<View style={styles.statCard}>
|
{ color: colors.white, fontSize: 36 },
|
||||||
<Text style={styles.statValue}>
|
]}
|
||||||
{activeGoals.length > 0
|
>
|
||||||
? Math.round(
|
{activeGoals.length}
|
||||||
activeGoals.reduce(
|
</Text>
|
||||||
(sum, g) => sum + (g.progress || 0),
|
<Text
|
||||||
0,
|
style={[
|
||||||
) / activeGoals.length,
|
typography.label,
|
||||||
)
|
{ color: "rgba(255,255,255,0.8)", marginTop: 4 },
|
||||||
: 0}
|
]}
|
||||||
%
|
>
|
||||||
</Text>
|
ACTIVE
|
||||||
<Text style={styles.statLabel}>Avg Progress</Text>
|
</Text>
|
||||||
|
</MinimalCard>
|
||||||
|
|
||||||
|
<MinimalCard
|
||||||
|
variant="elevated"
|
||||||
|
style={[styles.statCard, { backgroundColor: colors.success }]}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.statLarge,
|
||||||
|
{ color: colors.white, fontSize: 36 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{completedGoals.length}
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.label,
|
||||||
|
{ color: "rgba(255,255,255,0.8)", marginTop: 4 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
COMPLETED
|
||||||
|
</Text>
|
||||||
|
</MinimalCard>
|
||||||
|
|
||||||
|
<MinimalCard
|
||||||
|
variant="elevated"
|
||||||
|
style={[styles.statCard, { backgroundColor: colors.accent }]}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.statLarge,
|
||||||
|
{ color: colors.white, fontSize: 36 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{avgProgress}%
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.label,
|
||||||
|
{ color: "rgba(255,255,255,0.8)", marginTop: 4 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
PROGRESS
|
||||||
|
</Text>
|
||||||
|
</MinimalCard>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Analytics Section */}
|
{/* Analytics Section */}
|
||||||
{statistics && (
|
{statistics &&
|
||||||
<View style={styles.analyticsSection}>
|
(statistics.weeklyTrend.length > 0 ||
|
||||||
<TouchableOpacity
|
statistics.goals.goalsByType.length > 0) && (
|
||||||
style={styles.analyticsHeader}
|
<View style={styles.section}>
|
||||||
onPress={() => setShowAnalytics(!showAnalytics)}
|
<TouchableOpacity
|
||||||
>
|
onPress={() => setShowAnalytics(!showAnalytics)}
|
||||||
<View style={styles.analyticsHeaderLeft}>
|
activeOpacity={0.85}
|
||||||
<Ionicons
|
>
|
||||||
name="bar-chart-outline"
|
<MinimalCard variant="elevated" style={styles.analyticsCard}>
|
||||||
size={20}
|
<View style={styles.analyticsHeader}>
|
||||||
color={theme.colors.primary}
|
<View style={styles.analyticsHeaderLeft}>
|
||||||
/>
|
<View
|
||||||
<Text style={styles.analyticsTitle}>Progress Analytics</Text>
|
style={[
|
||||||
</View>
|
styles.analyticsIcon,
|
||||||
<Ionicons
|
{ backgroundColor: `${colors.primary}15` },
|
||||||
name={showAnalytics ? "chevron-up" : "chevron-down"}
|
]}
|
||||||
size={20}
|
>
|
||||||
color={theme.colors.gray400}
|
<Ionicons
|
||||||
/>
|
name="bar-chart"
|
||||||
</TouchableOpacity>
|
size={24}
|
||||||
|
color={colors.primary}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View>
|
||||||
|
<Text
|
||||||
|
style={[typography.h3, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
Progress Analytics
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textTertiary, marginTop: 2 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{showAnalytics ? "Tap to collapse" : "Tap to expand"}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.analyticsToggle,
|
||||||
|
{ backgroundColor: colors.surfaceElevated },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Ionicons
|
||||||
|
name={showAnalytics ? "chevron-up" : "chevron-down"}
|
||||||
|
size={20}
|
||||||
|
color={colors.textSecondary}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
|
||||||
{showAnalytics && (
|
{showAnalytics && (
|
||||||
<View style={styles.analyticsContent}>
|
<View style={styles.analyticsContent}>
|
||||||
{statistics.weeklyTrend.length > 0 && (
|
{statistics.weeklyTrend.length > 0 && (
|
||||||
<WeeklyProgressChart
|
<View style={styles.chartSection}>
|
||||||
weeklyData={statistics.weeklyTrend}
|
<Text
|
||||||
title="8-Week Trend"
|
style={[
|
||||||
/>
|
typography.h4,
|
||||||
)}
|
{ color: colors.textPrimary, marginBottom: 16 },
|
||||||
{statistics.goals.goalsByType.length > 0 && (
|
]}
|
||||||
<GoalTypeBreakdownChart
|
>
|
||||||
data={statistics.goals.goalsByType}
|
Weekly Trend
|
||||||
title="Goals by Type"
|
</Text>
|
||||||
/>
|
<WeeklyProgressChart
|
||||||
)}
|
weeklyData={statistics.weeklyTrend}
|
||||||
</View>
|
/>
|
||||||
)}
|
</View>
|
||||||
</View>
|
)}
|
||||||
)}
|
{statistics.goals.goalsByType.length > 0 && (
|
||||||
|
<View style={styles.chartSection}>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.h4,
|
||||||
|
{ color: colors.textPrimary, marginBottom: 16 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Goals by Type
|
||||||
|
</Text>
|
||||||
|
<GoalTypeBreakdownChart
|
||||||
|
data={statistics.goals.goalsByType}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
</MinimalCard>
|
||||||
|
</TouchableOpacity>
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* Active Goals */}
|
{/* Active Goals */}
|
||||||
<View style={styles.section}>
|
<View style={styles.section}>
|
||||||
<Text style={styles.sectionTitle}>
|
<SectionHeader
|
||||||
Active Goals ({activeGoals.length})
|
title={`Active Goals (${activeGoals.length})`}
|
||||||
</Text>
|
subtitle="Keep pushing forward!"
|
||||||
|
actionLabel="+ Add New"
|
||||||
|
onActionPress={() => setIsModalVisible(true)}
|
||||||
|
/>
|
||||||
{activeGoals.length === 0 ? (
|
{activeGoals.length === 0 ? (
|
||||||
<View style={styles.emptyState}>
|
<MinimalCard variant="default">
|
||||||
<Ionicons name="flag-outline" size={48} color="#d1d5db" />
|
<View style={styles.emptyState}>
|
||||||
<Text style={styles.emptyText}>No active goals yet</Text>
|
<Text style={{ fontSize: 64 }}>🎯</Text>
|
||||||
<Text style={styles.emptySubtext}>
|
<Text
|
||||||
Tap the + button to create your first goal
|
style={[
|
||||||
</Text>
|
typography.bodyEmphasis,
|
||||||
</View>
|
{ color: colors.textSecondary, marginTop: 12 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
No active goals yet
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textTertiary, marginTop: 4 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Tap "Add New" to set your first goal! 💪
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</MinimalCard>
|
||||||
) : (
|
) : (
|
||||||
activeGoals.map((goal) => (
|
<View style={styles.goalsList}>
|
||||||
<GoalProgressCard
|
{activeGoals.map((goal) => (
|
||||||
key={goal.id}
|
<GoalProgressCard
|
||||||
goal={goal}
|
key={goal.id}
|
||||||
onComplete={() => handleCompleteGoal(goal)}
|
goal={goal}
|
||||||
onDelete={() => handleDeleteGoal(goal.id)}
|
onComplete={() => handleCompleteGoal(goal)}
|
||||||
/>
|
onDelete={() => handleDeleteGoal(goal.id)}
|
||||||
))
|
/>
|
||||||
|
))}
|
||||||
|
</View>
|
||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
{/* Completed Goals */}
|
{/* Completed Goals */}
|
||||||
{completedGoals.length > 0 && (
|
{completedGoals.length > 0 && (
|
||||||
<View style={styles.section}>
|
<View style={styles.section}>
|
||||||
<Text style={styles.sectionTitle}>
|
<SectionHeader
|
||||||
Completed Goals ({completedGoals.length})
|
title={`Completed (${completedGoals.length})`}
|
||||||
</Text>
|
subtitle="Great work!"
|
||||||
{completedGoals.map((goal) => (
|
/>
|
||||||
<GoalProgressCard
|
<View style={styles.goalsList}>
|
||||||
key={goal.id}
|
{completedGoals.map((goal) => (
|
||||||
goal={goal}
|
<GoalProgressCard
|
||||||
onDelete={() => handleDeleteGoal(goal.id)}
|
key={goal.id}
|
||||||
/>
|
goal={goal}
|
||||||
))}
|
onDelete={() => handleDeleteGoal(goal.id)}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</View>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<View style={styles.footer} />
|
<View style={styles.footer} />
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
|
|
||||||
{/* Floating Action Button */}
|
{/* Floating Action Button - Minimal Style */}
|
||||||
<Animated.View
|
<View style={styles.fabContainer}>
|
||||||
style={[styles.fabContainer, { transform: [{ scale: fabScale }] }]}
|
|
||||||
>
|
|
||||||
<TouchableOpacity
|
<TouchableOpacity
|
||||||
onPress={() => setIsModalVisible(true)}
|
onPress={() => setIsModalVisible(true)}
|
||||||
onPressIn={() => {
|
activeOpacity={0.8}
|
||||||
Animated.spring(fabScale, {
|
style={[
|
||||||
toValue: 0.9,
|
styles.fab,
|
||||||
friction: 8,
|
{
|
||||||
tension: 100,
|
backgroundColor: colors.primary,
|
||||||
useNativeDriver: true,
|
shadowColor: colors.primary,
|
||||||
}).start();
|
},
|
||||||
}}
|
]}
|
||||||
onPressOut={() => {
|
|
||||||
Animated.spring(fabScale, {
|
|
||||||
toValue: 1,
|
|
||||||
friction: 8,
|
|
||||||
tension: 100,
|
|
||||||
useNativeDriver: true,
|
|
||||||
}).start();
|
|
||||||
}}
|
|
||||||
activeOpacity={0.9}
|
|
||||||
>
|
>
|
||||||
<LinearGradient
|
<Ionicons name="add" size={28} color={colors.white} />
|
||||||
colors={theme.gradients.primary}
|
|
||||||
start={{ x: 0, y: 0 }}
|
|
||||||
end={{ x: 1, y: 1 }}
|
|
||||||
style={styles.fab}
|
|
||||||
>
|
|
||||||
<Ionicons name="add" size={28} color="#fff" />
|
|
||||||
</LinearGradient>
|
|
||||||
</TouchableOpacity>
|
</TouchableOpacity>
|
||||||
</Animated.View>
|
</View>
|
||||||
|
|
||||||
{/* Create Goal Modal */}
|
{/* Create Goal Modal */}
|
||||||
<GoalCreationModal
|
<GoalCreationModal
|
||||||
@ -319,129 +446,96 @@ export default function GoalsScreen() {
|
|||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
backgroundColor: theme.colors.background,
|
|
||||||
},
|
},
|
||||||
scrollContent: {
|
scrollContent: {
|
||||||
paddingBottom: 20,
|
paddingBottom: 20,
|
||||||
},
|
},
|
||||||
header: {
|
header: {
|
||||||
padding: 24,
|
|
||||||
paddingTop: 60,
|
|
||||||
paddingBottom: 24,
|
|
||||||
marginBottom: 10,
|
|
||||||
borderBottomLeftRadius: theme.borderRadius.xl,
|
|
||||||
borderBottomRightRadius: theme.borderRadius.xl,
|
|
||||||
},
|
|
||||||
headerContent: {
|
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
alignItems: "center",
|
alignItems: "flex-start",
|
||||||
|
paddingHorizontal: 20,
|
||||||
|
paddingTop: 60,
|
||||||
|
paddingBottom: 20,
|
||||||
},
|
},
|
||||||
debugButton: {
|
debugButton: {
|
||||||
padding: 8,
|
padding: 8,
|
||||||
},
|
},
|
||||||
headerTitle: {
|
section: {
|
||||||
fontSize: theme.typography.fontSize["3xl"],
|
paddingHorizontal: 20,
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
marginBottom: 24,
|
||||||
color: theme.colors.white,
|
|
||||||
},
|
},
|
||||||
headerSubtitle: {
|
statsRow: {
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
color: "rgba(255, 255, 255, 0.9)",
|
|
||||||
marginTop: 4,
|
|
||||||
},
|
|
||||||
statsContainer: {
|
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
padding: 16,
|
|
||||||
gap: 12,
|
gap: 12,
|
||||||
},
|
},
|
||||||
statCard: {
|
statCard: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
backgroundColor: theme.colors.white,
|
|
||||||
padding: 16,
|
|
||||||
borderRadius: theme.borderRadius.xl,
|
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
...theme.shadows.medium,
|
paddingVertical: 20,
|
||||||
borderWidth: 1,
|
paddingHorizontal: 12,
|
||||||
borderColor: "rgba(59, 130, 246, 0.1)",
|
borderRadius: 20,
|
||||||
},
|
},
|
||||||
statValue: {
|
analyticsCard: {
|
||||||
fontSize: theme.typography.fontSize["2xl"],
|
padding: 20,
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.primary,
|
|
||||||
marginBottom: 4,
|
|
||||||
},
|
|
||||||
statLabel: {
|
|
||||||
fontSize: 12,
|
|
||||||
color: "#6b7280",
|
|
||||||
fontWeight: "500",
|
|
||||||
},
|
|
||||||
analyticsSection: {
|
|
||||||
padding: 16,
|
|
||||||
paddingTop: 0,
|
|
||||||
},
|
},
|
||||||
analyticsHeader: {
|
analyticsHeader: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
backgroundColor: theme.colors.white,
|
|
||||||
padding: 16,
|
|
||||||
borderRadius: theme.borderRadius.xl,
|
|
||||||
marginBottom: 12,
|
|
||||||
...theme.shadows.subtle,
|
|
||||||
},
|
},
|
||||||
analyticsHeaderLeft: {
|
analyticsHeaderLeft: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
gap: 8,
|
|
||||||
},
|
},
|
||||||
analyticsTitle: {
|
analyticsIcon: {
|
||||||
fontSize: theme.typography.fontSize.base,
|
width: 48,
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
height: 48,
|
||||||
color: theme.colors.gray700,
|
borderRadius: 14,
|
||||||
|
justifyContent: "center",
|
||||||
|
alignItems: "center",
|
||||||
|
marginRight: 14,
|
||||||
|
},
|
||||||
|
analyticsToggle: {
|
||||||
|
width: 36,
|
||||||
|
height: 36,
|
||||||
|
borderRadius: 10,
|
||||||
|
justifyContent: "center",
|
||||||
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
analyticsContent: {
|
analyticsContent: {
|
||||||
paddingTop: 4,
|
paddingTop: 24,
|
||||||
|
marginTop: 20,
|
||||||
|
borderTopWidth: 1,
|
||||||
|
borderTopColor: "rgba(0,0,0,0.05)",
|
||||||
},
|
},
|
||||||
section: {
|
chartSection: {
|
||||||
padding: 20,
|
marginBottom: 20,
|
||||||
paddingTop: 10,
|
|
||||||
},
|
},
|
||||||
sectionTitle: {
|
goalsList: {
|
||||||
fontSize: 18,
|
gap: 16,
|
||||||
fontWeight: "600",
|
|
||||||
color: "#374151",
|
|
||||||
marginBottom: 12,
|
|
||||||
},
|
},
|
||||||
emptyState: {
|
emptyState: {
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
paddingVertical: 40,
|
paddingVertical: 40,
|
||||||
},
|
},
|
||||||
emptyText: {
|
|
||||||
fontSize: 16,
|
|
||||||
fontWeight: "500",
|
|
||||||
color: "#6b7280",
|
|
||||||
marginTop: 12,
|
|
||||||
},
|
|
||||||
emptySubtext: {
|
|
||||||
fontSize: 14,
|
|
||||||
color: "#9ca3af",
|
|
||||||
marginTop: 4,
|
|
||||||
},
|
|
||||||
footer: {
|
footer: {
|
||||||
height: 100,
|
height: 100,
|
||||||
},
|
},
|
||||||
fabContainer: {
|
fabContainer: {
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
right: 20,
|
right: 20,
|
||||||
bottom: 110, // Adjusted for tab bar height
|
bottom: 90,
|
||||||
},
|
},
|
||||||
fab: {
|
fab: {
|
||||||
width: 64,
|
width: 64,
|
||||||
height: 64,
|
height: 64,
|
||||||
borderRadius: 32,
|
borderRadius: 22,
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
...theme.shadows.glow,
|
shadowOffset: { width: 0, height: 6 },
|
||||||
|
shadowOpacity: 0.35,
|
||||||
|
shadowRadius: 12,
|
||||||
|
elevation: 8,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -1,4 +1,4 @@
|
|||||||
import React, { useState, useCallback, useRef } from "react";
|
import React, { useState, useCallback } from "react";
|
||||||
import {
|
import {
|
||||||
View,
|
View,
|
||||||
Text,
|
Text,
|
||||||
@ -9,11 +9,15 @@ import {
|
|||||||
ActivityIndicator,
|
ActivityIndicator,
|
||||||
Alert,
|
Alert,
|
||||||
} from "react-native";
|
} from "react-native";
|
||||||
import { LinearGradient } from "expo-linear-gradient";
|
|
||||||
import { Ionicons } from "@expo/vector-icons";
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import { useUser } from "@clerk/clerk-expo";
|
import { useUser } from "@clerk/clerk-expo";
|
||||||
import { useFocusEffect } from "expo-router";
|
import { useFocusEffect } from "expo-router";
|
||||||
import { theme } from "../../styles/theme";
|
import { useTheme } from "../../contexts/ThemeContext";
|
||||||
|
import { MinimalCard } from "../../components/MinimalCard";
|
||||||
|
import { SectionHeader } from "../../components/SectionHeader";
|
||||||
|
import { MinimalButton } from "../../components/MinimalButton";
|
||||||
|
import { Badge } from "../../components/Badge";
|
||||||
|
import { IconContainer } from "../../components/IconContainer";
|
||||||
import { useRecommendations } from "../../contexts/RecommendationsContext";
|
import { useRecommendations } from "../../contexts/RecommendationsContext";
|
||||||
import { useNotifications } from "../../contexts/NotificationsContext";
|
import { useNotifications } from "../../contexts/NotificationsContext";
|
||||||
import { NotificationsModal } from "../../components/NotificationsModal";
|
import { NotificationsModal } from "../../components/NotificationsModal";
|
||||||
@ -22,6 +26,7 @@ import log from "../../utils/logger";
|
|||||||
|
|
||||||
export default function RecommendationsScreen() {
|
export default function RecommendationsScreen() {
|
||||||
const { user } = useUser();
|
const { user } = useUser();
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
const {
|
const {
|
||||||
recommendations: allRecommendations,
|
recommendations: allRecommendations,
|
||||||
loading,
|
loading,
|
||||||
@ -43,7 +48,7 @@ export default function RecommendationsScreen() {
|
|||||||
setNotificationsVisible(false);
|
setNotificationsVisible(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Filter to show only approved recommendations for regular users
|
// Filter to show only approved recommendations
|
||||||
const recommendations = allRecommendations.filter(
|
const recommendations = allRecommendations.filter(
|
||||||
(rec) => rec.status === "approved",
|
(rec) => rec.status === "approved",
|
||||||
);
|
);
|
||||||
@ -101,14 +106,14 @@ export default function RecommendationsScreen() {
|
|||||||
|
|
||||||
if (loading && recommendations.length === 0) {
|
if (loading && recommendations.length === 0) {
|
||||||
return (
|
return (
|
||||||
<View style={styles.centered}>
|
<View style={[styles.centered, { backgroundColor: colors.background }]}>
|
||||||
<ActivityIndicator size="large" color={theme.colors.primary} />
|
<ActivityIndicator size="large" color={colors.primary} />
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={[styles.container, { backgroundColor: colors.background }]}>
|
||||||
<NotificationsModal
|
<NotificationsModal
|
||||||
visible={notificationsVisible}
|
visible={notificationsVisible}
|
||||||
onClose={handleCloseNotifications}
|
onClose={handleCloseNotifications}
|
||||||
@ -119,98 +124,125 @@ export default function RecommendationsScreen() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={onRefresh}
|
onRefresh={onRefresh}
|
||||||
tintColor={theme.colors.primary}
|
tintColor={colors.primary}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<LinearGradient
|
<View style={styles.header}>
|
||||||
colors={theme.gradients.primary}
|
<View style={{ flex: 1 }}>
|
||||||
start={{ x: 0, y: 0 }}
|
<Text
|
||||||
end={{ x: 1, y: 1 }}
|
style={[
|
||||||
style={styles.header}
|
typography.h1,
|
||||||
>
|
{ color: colors.textPrimary, fontSize: 32 },
|
||||||
<View>
|
]}
|
||||||
<Text style={styles.headerTitle}>AI Recommendations</Text>
|
>
|
||||||
<Text style={styles.headerSubtitle}>
|
Recommendations
|
||||||
Personalized fitness & nutrition plans
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{ color: colors.textSecondary, marginTop: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{recommendations.length === 0
|
||||||
|
? "Let's create your perfect plan!"
|
||||||
|
: `${recommendations.length} plan${recommendations.length !== 1 ? "s" : ""} ready for you!`}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
<TouchableOpacity
|
<TouchableOpacity
|
||||||
style={styles.iconContainer}
|
style={styles.notificationButton}
|
||||||
onPress={handleOpenNotifications}
|
onPress={handleOpenNotifications}
|
||||||
activeOpacity={0.7}
|
activeOpacity={0.7}
|
||||||
>
|
>
|
||||||
<Ionicons name="sparkles" size={32} color="#fff" />
|
<IconContainer
|
||||||
{unreadCount > 0 && (
|
variant="colored"
|
||||||
<View style={styles.badge}>
|
backgroundColor={colors.accent}
|
||||||
<Text style={styles.badgeText}>{unreadCount}</Text>
|
size="lg"
|
||||||
</View>
|
>
|
||||||
)}
|
<Ionicons name="sparkles" size={24} color={colors.white} />
|
||||||
|
{unreadCount > 0 && (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.notificationBadge,
|
||||||
|
{ backgroundColor: colors.danger },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Text style={styles.notificationBadgeText}>
|
||||||
|
{unreadCount}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
</IconContainer>
|
||||||
</TouchableOpacity>
|
</TouchableOpacity>
|
||||||
</LinearGradient>
|
</View>
|
||||||
|
|
||||||
{/* Generate Button */}
|
{/* Generate Button */}
|
||||||
<View style={styles.actionContainer}>
|
<View style={styles.section}>
|
||||||
<TouchableOpacity
|
<MinimalButton
|
||||||
|
title="Generate New Plan"
|
||||||
onPress={handleGenerateRecommendation}
|
onPress={handleGenerateRecommendation}
|
||||||
|
variant="primary"
|
||||||
|
size="lg"
|
||||||
|
fullWidth
|
||||||
|
loading={generating}
|
||||||
disabled={generating}
|
disabled={generating}
|
||||||
activeOpacity={0.8}
|
textStyle={{ fontSize: 16 }}
|
||||||
>
|
/>
|
||||||
<LinearGradient
|
|
||||||
colors={theme.gradients.purple}
|
|
||||||
start={{ x: 0, y: 0 }}
|
|
||||||
end={{ x: 1, y: 0 }}
|
|
||||||
style={[styles.generateButton, theme.shadows.medium]}
|
|
||||||
>
|
|
||||||
{generating ? (
|
|
||||||
<ActivityIndicator size="small" color="#fff" />
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<Ionicons
|
|
||||||
name="bulb"
|
|
||||||
size={24}
|
|
||||||
color="#fff"
|
|
||||||
style={{ marginRight: 12 }}
|
|
||||||
/>
|
|
||||||
<Text style={styles.generateButtonText}>
|
|
||||||
Generate New Plan
|
|
||||||
</Text>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</LinearGradient>
|
|
||||||
</TouchableOpacity>
|
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
{/* Recommendations List */}
|
{/* Recommendations List */}
|
||||||
<View style={styles.section}>
|
<View style={styles.section}>
|
||||||
|
<SectionHeader
|
||||||
|
title="💡 Your Plans"
|
||||||
|
subtitle={
|
||||||
|
recommendations.length > 0
|
||||||
|
? `${recommendations.length} active plan${recommendations.length !== 1 ? "s" : ""}`
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
|
||||||
{recommendations.length === 0 ? (
|
{recommendations.length === 0 ? (
|
||||||
<View style={styles.emptyState}>
|
<MinimalCard variant="default" style={{ borderRadius: 20 }}>
|
||||||
<LinearGradient
|
<View style={styles.emptyState}>
|
||||||
colors={["rgba(139, 92, 246, 0.1)", "rgba(139, 92, 246, 0.05)"]}
|
<Text style={{ fontSize: 64 }}>🤖</Text>
|
||||||
style={styles.emptyCard}
|
<Text
|
||||||
>
|
style={[
|
||||||
<Ionicons
|
typography.bodyEmphasis,
|
||||||
name="sparkles-outline"
|
{ color: colors.textPrimary, marginTop: 16 },
|
||||||
size={64}
|
]}
|
||||||
color={theme.colors.purple}
|
>
|
||||||
/>
|
No Recommendations Yet
|
||||||
<Text style={styles.emptyTitle}>No Recommendations Yet</Text>
|
|
||||||
<Text style={styles.emptyText}>
|
|
||||||
Tap "Generate New Plan" to get personalized AI-powered fitness
|
|
||||||
and nutrition recommendations based on your profile and goals.
|
|
||||||
</Text>
|
</Text>
|
||||||
</LinearGradient>
|
<Text
|
||||||
</View>
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{
|
||||||
|
color: colors.textSecondary,
|
||||||
|
marginTop: 8,
|
||||||
|
textAlign: "center",
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
Tap "Generate New Plan" to get personalized AI-powered fitness
|
||||||
|
and nutrition recommendations! 🎯
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</MinimalCard>
|
||||||
) : (
|
) : (
|
||||||
recommendations.map((recommendation) => (
|
<View style={styles.recommendationsList}>
|
||||||
<RecommendationCard
|
{recommendations.map((recommendation) => (
|
||||||
key={recommendation.id}
|
<RecommendationCard
|
||||||
recommendation={recommendation}
|
key={recommendation.id}
|
||||||
/>
|
recommendation={recommendation}
|
||||||
))
|
/>
|
||||||
|
))}
|
||||||
|
</View>
|
||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
|
{/* Bottom Spacer */}
|
||||||
|
<View style={{ height: 100 }} />
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
@ -221,250 +253,195 @@ interface RecommendationCardProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function RecommendationCard({ recommendation }: RecommendationCardProps) {
|
function RecommendationCard({ recommendation }: RecommendationCardProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
const [expanded, setExpanded] = useState(false);
|
const [expanded, setExpanded] = useState(false);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.card}>
|
<MinimalCard variant="bordered" style={{ borderRadius: 20 }}>
|
||||||
<LinearGradient
|
{/* Header */}
|
||||||
colors={["rgba(255, 255, 255, 1)", "rgba(249, 250, 251, 1)"]}
|
<TouchableOpacity
|
||||||
style={[styles.cardContent, theme.shadows.medium]}
|
onPress={() => setExpanded(!expanded)}
|
||||||
|
activeOpacity={0.7}
|
||||||
|
style={styles.cardHeader}
|
||||||
>
|
>
|
||||||
{/* Header */}
|
<View style={styles.cardHeaderLeft}>
|
||||||
<View style={styles.cardHeader}>
|
<IconContainer
|
||||||
<View style={styles.cardHeaderLeft}>
|
variant="colored"
|
||||||
<LinearGradient
|
backgroundColor={`${colors.success}20`}
|
||||||
colors={theme.gradients.success}
|
>
|
||||||
style={styles.cardIcon}
|
<Ionicons
|
||||||
>
|
name="checkmark-circle"
|
||||||
<Ionicons name="checkmark-circle" size={20} color="#fff" />
|
size={20}
|
||||||
</LinearGradient>
|
color={colors.success}
|
||||||
<View>
|
/>
|
||||||
<Text style={styles.cardTitle}>AI Fitness Plan</Text>
|
</IconContainer>
|
||||||
<Text style={styles.cardDate}>
|
<View style={{ marginLeft: 12 }}>
|
||||||
{new Date(recommendation.generatedAt).toLocaleDateString()}
|
<Text style={[typography.h3, { color: colors.textPrimary }]}>
|
||||||
|
✨ AI Fitness Plan
|
||||||
|
</Text>
|
||||||
|
<Text style={[typography.caption, { color: colors.textTertiary }]}>
|
||||||
|
{new Date(recommendation.generatedAt).toLocaleDateString()}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
<Ionicons
|
||||||
|
name={expanded ? "chevron-up" : "chevron-down"}
|
||||||
|
size={24}
|
||||||
|
color={colors.textTertiary}
|
||||||
|
/>
|
||||||
|
</TouchableOpacity>
|
||||||
|
|
||||||
|
{/* Summary */}
|
||||||
|
<View style={styles.cardSummary}>
|
||||||
|
<Text
|
||||||
|
style={[typography.body, { color: colors.textSecondary }]}
|
||||||
|
numberOfLines={expanded ? undefined : 3}
|
||||||
|
>
|
||||||
|
{recommendation.recommendationText}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
|
||||||
|
{/* Expanded Content */}
|
||||||
|
{expanded && (
|
||||||
|
<View style={styles.expandedContent}>
|
||||||
|
{/* Activity Plan */}
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.planSection,
|
||||||
|
{ backgroundColor: colors.surfaceElevated },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<View style={styles.planHeader}>
|
||||||
|
<Ionicons name="barbell" size={20} color={colors.primary} />
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.h3,
|
||||||
|
{ color: colors.textPrimary, marginLeft: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
💪 Activity Plan
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{ color: colors.textSecondary, marginTop: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{recommendation.activityPlan}
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
<TouchableOpacity onPress={() => setExpanded(!expanded)}>
|
|
||||||
<Ionicons
|
|
||||||
name={expanded ? "chevron-up" : "chevron-down"}
|
|
||||||
size={24}
|
|
||||||
color={theme.colors.gray400}
|
|
||||||
/>
|
|
||||||
</TouchableOpacity>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
{/* Summary */}
|
{/* Diet Plan */}
|
||||||
<View style={styles.cardSummary}>
|
<View
|
||||||
<Text
|
style={[
|
||||||
style={styles.summaryText}
|
styles.planSection,
|
||||||
numberOfLines={expanded ? undefined : 3}
|
{ backgroundColor: colors.surfaceElevated },
|
||||||
|
]}
|
||||||
>
|
>
|
||||||
{recommendation.recommendationText}
|
<View style={styles.planHeader}>
|
||||||
</Text>
|
<Ionicons name="restaurant" size={20} color={colors.success} />
|
||||||
</View>
|
<Text
|
||||||
|
style={[
|
||||||
{/* Expanded Content */}
|
typography.h3,
|
||||||
{expanded && (
|
{ color: colors.textPrimary, marginLeft: 8 },
|
||||||
<View style={styles.expandedContent}>
|
]}
|
||||||
{/* Activity Plan */}
|
>
|
||||||
<View style={styles.planSection}>
|
🍽️ Diet Plan
|
||||||
<View style={styles.planHeader}>
|
</Text>
|
||||||
<Ionicons
|
|
||||||
name="barbell"
|
|
||||||
size={20}
|
|
||||||
color={theme.colors.primary}
|
|
||||||
/>
|
|
||||||
<Text style={styles.planTitle}>Activity Plan</Text>
|
|
||||||
</View>
|
|
||||||
<Text style={styles.planText}>{recommendation.activityPlan}</Text>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
{/* Diet Plan */}
|
|
||||||
<View style={styles.planSection}>
|
|
||||||
<View style={styles.planHeader}>
|
|
||||||
<Ionicons
|
|
||||||
name="restaurant"
|
|
||||||
size={20}
|
|
||||||
color={theme.colors.success}
|
|
||||||
/>
|
|
||||||
<Text style={styles.planTitle}>Diet Plan</Text>
|
|
||||||
</View>
|
|
||||||
<Text style={styles.planText}>{recommendation.dietPlan}</Text>
|
|
||||||
</View>
|
</View>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{ color: colors.textSecondary, marginTop: 8 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{recommendation.dietPlan}
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
)}
|
</View>
|
||||||
</LinearGradient>
|
)}
|
||||||
</View>
|
</MinimalCard>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
backgroundColor: theme.colors.background,
|
|
||||||
},
|
},
|
||||||
centered: {
|
centered: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
backgroundColor: theme.colors.background,
|
|
||||||
},
|
},
|
||||||
scrollContent: {
|
scrollContent: {
|
||||||
paddingBottom: 100,
|
paddingBottom: 20,
|
||||||
},
|
},
|
||||||
header: {
|
header: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
alignItems: "center",
|
alignItems: "flex-start",
|
||||||
padding: 24,
|
paddingHorizontal: 24,
|
||||||
paddingTop: 60,
|
paddingTop: 60,
|
||||||
paddingBottom: 24,
|
paddingBottom: 24,
|
||||||
marginBottom: 20,
|
|
||||||
borderBottomLeftRadius: theme.borderRadius.xl,
|
|
||||||
borderBottomRightRadius: theme.borderRadius.xl,
|
|
||||||
},
|
},
|
||||||
headerTitle: {
|
notificationButton: {
|
||||||
fontSize: theme.typography.fontSize["3xl"],
|
position: "relative",
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.white,
|
|
||||||
},
|
},
|
||||||
headerSubtitle: {
|
notificationBadge: {
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
color: "rgba(255, 255, 255, 0.9)",
|
|
||||||
marginTop: 4,
|
|
||||||
},
|
|
||||||
iconContainer: {
|
|
||||||
backgroundColor: "rgba(255, 255, 255, 0.2)",
|
|
||||||
width: 64,
|
|
||||||
height: 64,
|
|
||||||
borderRadius: 32,
|
|
||||||
justifyContent: "center",
|
|
||||||
alignItems: "center",
|
|
||||||
},
|
|
||||||
badge: {
|
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
top: -4,
|
top: -4,
|
||||||
right: -4,
|
right: -4,
|
||||||
backgroundColor: theme.colors.danger,
|
minWidth: 18,
|
||||||
borderRadius: 10,
|
height: 18,
|
||||||
width: 20,
|
borderRadius: 9,
|
||||||
height: 20,
|
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
|
paddingHorizontal: 4,
|
||||||
},
|
},
|
||||||
badgeText: {
|
notificationBadgeText: {
|
||||||
color: "#fff",
|
color: "#fff",
|
||||||
fontSize: 12,
|
fontSize: 11,
|
||||||
fontWeight: "bold",
|
fontWeight: "700",
|
||||||
},
|
|
||||||
actionContainer: {
|
|
||||||
paddingHorizontal: 20,
|
|
||||||
marginBottom: 20,
|
|
||||||
},
|
|
||||||
generateButton: {
|
|
||||||
flexDirection: "row",
|
|
||||||
alignItems: "center",
|
|
||||||
justifyContent: "center",
|
|
||||||
paddingVertical: 16,
|
|
||||||
paddingHorizontal: 24,
|
|
||||||
borderRadius: theme.borderRadius.xl,
|
|
||||||
},
|
|
||||||
generateButtonText: {
|
|
||||||
fontSize: theme.typography.fontSize.lg,
|
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.white,
|
|
||||||
},
|
},
|
||||||
section: {
|
section: {
|
||||||
paddingHorizontal: 20,
|
paddingHorizontal: 24,
|
||||||
|
marginBottom: 24,
|
||||||
},
|
},
|
||||||
emptyState: {
|
emptyState: {
|
||||||
paddingVertical: 40,
|
|
||||||
},
|
|
||||||
emptyCard: {
|
|
||||||
borderRadius: theme.borderRadius["2xl"],
|
|
||||||
padding: 32,
|
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
|
paddingVertical: 40,
|
||||||
|
paddingHorizontal: 20,
|
||||||
},
|
},
|
||||||
emptyTitle: {
|
recommendationsList: {
|
||||||
fontSize: theme.typography.fontSize.xl,
|
gap: 12,
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.gray700,
|
|
||||||
marginTop: 16,
|
|
||||||
marginBottom: 8,
|
|
||||||
},
|
|
||||||
emptyText: {
|
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
color: theme.colors.gray500,
|
|
||||||
textAlign: "center",
|
|
||||||
lineHeight: 24,
|
|
||||||
},
|
|
||||||
card: {
|
|
||||||
marginBottom: 16,
|
|
||||||
},
|
|
||||||
cardContent: {
|
|
||||||
borderRadius: theme.borderRadius["2xl"],
|
|
||||||
padding: 20,
|
|
||||||
},
|
},
|
||||||
cardHeader: {
|
cardHeader: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
marginBottom: 16,
|
|
||||||
},
|
},
|
||||||
cardHeaderLeft: {
|
cardHeaderLeft: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
gap: 12,
|
flex: 1,
|
||||||
},
|
|
||||||
cardIcon: {
|
|
||||||
width: 40,
|
|
||||||
height: 40,
|
|
||||||
borderRadius: 20,
|
|
||||||
justifyContent: "center",
|
|
||||||
alignItems: "center",
|
|
||||||
},
|
|
||||||
cardTitle: {
|
|
||||||
fontSize: theme.typography.fontSize.lg,
|
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.gray800,
|
|
||||||
},
|
|
||||||
cardDate: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
color: theme.colors.gray500,
|
|
||||||
marginTop: 2,
|
|
||||||
},
|
},
|
||||||
cardSummary: {
|
cardSummary: {
|
||||||
marginBottom: 12,
|
marginTop: 12,
|
||||||
},
|
paddingTop: 12,
|
||||||
summaryText: {
|
borderTopWidth: 1,
|
||||||
fontSize: theme.typography.fontSize.base,
|
borderTopColor: "rgba(0, 0, 0, 0.05)",
|
||||||
color: theme.colors.gray700,
|
|
||||||
lineHeight: 24,
|
|
||||||
},
|
},
|
||||||
expandedContent: {
|
expandedContent: {
|
||||||
marginTop: 12,
|
marginTop: 16,
|
||||||
paddingTop: 16,
|
gap: 12,
|
||||||
borderTopWidth: 1,
|
|
||||||
borderTopColor: theme.colors.gray200,
|
|
||||||
},
|
},
|
||||||
planSection: {
|
planSection: {
|
||||||
marginBottom: 16,
|
padding: 16,
|
||||||
|
borderRadius: 16,
|
||||||
},
|
},
|
||||||
planHeader: {
|
planHeader: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
gap: 8,
|
|
||||||
marginBottom: 8,
|
|
||||||
},
|
|
||||||
planTitle: {
|
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.gray800,
|
|
||||||
},
|
|
||||||
planText: {
|
|
||||||
fontSize: theme.typography.fontSize.base,
|
|
||||||
color: theme.colors.gray600,
|
|
||||||
lineHeight: 22,
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@ -3,7 +3,9 @@ import { Stack } from "expo-router";
|
|||||||
import * as SecureStore from "expo-secure-store";
|
import * as SecureStore from "expo-secure-store";
|
||||||
import { View, Text } from "react-native";
|
import { View, Text } from "react-native";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
import { SafeAreaProvider } from "react-native-safe-area-context";
|
||||||
import { validateEnv } from "../utils/env";
|
import { validateEnv } from "../utils/env";
|
||||||
|
import { ThemeProvider } from "../contexts/ThemeContext";
|
||||||
import { StatisticsProvider } from "../contexts/StatisticsContext";
|
import { StatisticsProvider } from "../contexts/StatisticsContext";
|
||||||
import { FitnessGoalsProvider } from "../contexts/FitnessGoalsContext";
|
import { FitnessGoalsProvider } from "../contexts/FitnessGoalsContext";
|
||||||
import { RecommendationsProvider } from "../contexts/RecommendationsContext";
|
import { RecommendationsProvider } from "../contexts/RecommendationsContext";
|
||||||
@ -169,18 +171,22 @@ export default function RootLayout() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}>
|
<SafeAreaProvider>
|
||||||
<ClerkLoaded>
|
<ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}>
|
||||||
<NotificationsProvider>
|
<ClerkLoaded>
|
||||||
<StatisticsProvider>
|
<ThemeProvider>
|
||||||
<FitnessGoalsProvider>
|
<NotificationsProvider>
|
||||||
<RecommendationsProvider>
|
<StatisticsProvider>
|
||||||
<AppContent />
|
<FitnessGoalsProvider>
|
||||||
</RecommendationsProvider>
|
<RecommendationsProvider>
|
||||||
</FitnessGoalsProvider>
|
<AppContent />
|
||||||
</StatisticsProvider>
|
</RecommendationsProvider>
|
||||||
</NotificationsProvider>
|
</FitnessGoalsProvider>
|
||||||
</ClerkLoaded>
|
</StatisticsProvider>
|
||||||
</ClerkProvider>
|
</NotificationsProvider>
|
||||||
|
</ThemeProvider>
|
||||||
|
</ClerkLoaded>
|
||||||
|
</ClerkProvider>
|
||||||
|
</SafeAreaProvider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,4 +1,4 @@
|
|||||||
import React, { useState, useEffect } from 'react';
|
import React, { useState, useEffect } from "react";
|
||||||
import {
|
import {
|
||||||
View,
|
View,
|
||||||
Text,
|
Text,
|
||||||
@ -9,13 +9,13 @@ import {
|
|||||||
Alert,
|
Alert,
|
||||||
TextInput,
|
TextInput,
|
||||||
Platform,
|
Platform,
|
||||||
} from 'react-native';
|
} from "react-native";
|
||||||
import { useRouter, Stack } from 'expo-router';
|
import { useRouter, Stack } from "expo-router";
|
||||||
import { useAuth } from '@clerk/clerk-expo';
|
import { useAuth } from "@clerk/clerk-expo";
|
||||||
import { Ionicons } from '@expo/vector-icons';
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import { LinearGradient } from 'expo-linear-gradient';
|
import { LinearGradient } from "expo-linear-gradient";
|
||||||
import { theme } from '../styles/theme';
|
import { theme } from "../styles/theme";
|
||||||
import { API_BASE_URL } from '../config/api';
|
import { API_BASE_URL } from "../config/api";
|
||||||
|
|
||||||
interface FitnessProfileData {
|
interface FitnessProfileData {
|
||||||
height?: number;
|
height?: number;
|
||||||
@ -30,25 +30,71 @@ interface FitnessProfileData {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const GENDER_OPTIONS = [
|
const GENDER_OPTIONS = [
|
||||||
{ label: 'Male', value: 'male', icon: 'male' },
|
{ label: "Male", value: "male", icon: "male" },
|
||||||
{ label: 'Female', value: 'female', icon: 'female' },
|
{ label: "Female", value: "female", icon: "female" },
|
||||||
{ label: 'Other', value: 'other', icon: 'transgender' },
|
{ label: "Other", value: "other", icon: "transgender" },
|
||||||
|
{
|
||||||
|
label: "Prefer not to say",
|
||||||
|
value: "prefer_not_to_say",
|
||||||
|
icon: "help-circle",
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const FITNESS_GOAL_OPTIONS = [
|
const FITNESS_GOAL_OPTIONS = [
|
||||||
{ label: 'Weight Loss', value: 'weight_loss', icon: 'trending-down', color: theme.colors.danger },
|
{
|
||||||
{ label: 'Muscle Gain', value: 'muscle_gain', icon: 'barbell', color: theme.colors.primary },
|
label: "Weight Loss",
|
||||||
{ label: 'Endurance', value: 'endurance', icon: 'bicycle', color: theme.colors.success },
|
value: "weight_loss",
|
||||||
{ label: 'Flexibility', value: 'flexibility', icon: 'body', color: theme.colors.purple },
|
icon: "trending-down",
|
||||||
{ label: 'General Fitness', value: 'general_fitness', icon: 'fitness', color: theme.colors.warning },
|
color: theme.colors.danger,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Muscle Gain",
|
||||||
|
value: "muscle_gain",
|
||||||
|
icon: "barbell",
|
||||||
|
color: theme.colors.primary,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Endurance",
|
||||||
|
value: "endurance",
|
||||||
|
icon: "bicycle",
|
||||||
|
color: theme.colors.success,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Flexibility",
|
||||||
|
value: "flexibility",
|
||||||
|
icon: "body",
|
||||||
|
color: theme.colors.purple,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "General Fitness",
|
||||||
|
value: "general_fitness",
|
||||||
|
icon: "fitness",
|
||||||
|
color: theme.colors.warning,
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const ACTIVITY_LEVEL_OPTIONS = [
|
const ACTIVITY_LEVEL_OPTIONS = [
|
||||||
{ label: 'Sedentary', value: 'sedentary', description: 'Little to no exercise' },
|
{
|
||||||
{ label: 'Light', value: 'light', description: '1-3 days/week' },
|
label: "Sedentary",
|
||||||
{ label: 'Moderate', value: 'moderate', description: '3-5 days/week' },
|
value: "sedentary",
|
||||||
{ label: 'Active', value: 'active', description: '6-7 days/week' },
|
description: "Little to no exercise",
|
||||||
{ label: 'Very Active', value: 'very_active', description: 'Intense daily training' },
|
},
|
||||||
|
{
|
||||||
|
label: "Lightly Active",
|
||||||
|
value: "lightly_active",
|
||||||
|
description: "1-3 days/week",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Moderately Active",
|
||||||
|
value: "moderately_active",
|
||||||
|
description: "3-5 days/week",
|
||||||
|
},
|
||||||
|
{ label: "Very Active", value: "very_active", description: "6-7 days/week" },
|
||||||
|
{
|
||||||
|
label: "Extremely Active",
|
||||||
|
value: "extremely_active",
|
||||||
|
description: "Intense daily training",
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export default function FitnessProfileScreen() {
|
export default function FitnessProfileScreen() {
|
||||||
@ -66,34 +112,43 @@ export default function FitnessProfileScreen() {
|
|||||||
try {
|
try {
|
||||||
setFetchingProfile(true);
|
setFetchingProfile(true);
|
||||||
const token = await getToken();
|
const token = await getToken();
|
||||||
const response = await fetch(`${API_BASE_URL}/api/profile/fitness?userId=${userId}`, {
|
const response = await fetch(
|
||||||
headers: {
|
`${API_BASE_URL}/api/profile/fitness?userId=${userId}`,
|
||||||
Authorization: `Bearer ${token}`,
|
{
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${token}`,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
);
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
if (data.profile) {
|
if (data.profile) {
|
||||||
|
// Normalize old activity level values to new schema
|
||||||
|
let activityLevel = data.profile.activityLevel || "";
|
||||||
|
if (activityLevel === "light") activityLevel = "lightly_active";
|
||||||
|
if (activityLevel === "moderate") activityLevel = "moderately_active";
|
||||||
|
if (activityLevel === "active") activityLevel = "very_active";
|
||||||
|
|
||||||
setProfileData({
|
setProfileData({
|
||||||
height: data.profile.height,
|
height: data.profile.height,
|
||||||
weight: data.profile.weight,
|
weight: data.profile.weight,
|
||||||
age: data.profile.age,
|
age: data.profile.age,
|
||||||
gender: data.profile.gender || '',
|
gender: data.profile.gender || "",
|
||||||
fitnessGoal: Array.isArray(data.profile.fitnessGoals)
|
fitnessGoal: Array.isArray(data.profile.fitnessGoals)
|
||||||
? data.profile.fitnessGoals[0]
|
? data.profile.fitnessGoals[0]
|
||||||
: (typeof data.profile.fitnessGoals === 'string'
|
: typeof data.profile.fitnessGoals === "string"
|
||||||
? JSON.parse(data.profile.fitnessGoals)[0]
|
? JSON.parse(data.profile.fitnessGoals)[0]
|
||||||
: ''),
|
: "",
|
||||||
activityLevel: data.profile.activityLevel || '',
|
activityLevel: activityLevel,
|
||||||
medicalConditions: data.profile.medicalConditions || '',
|
medicalConditions: data.profile.medicalConditions || "",
|
||||||
allergies: data.profile.allergies || '',
|
allergies: data.profile.allergies || "",
|
||||||
injuries: data.profile.injuries || '',
|
injuries: data.profile.injuries || "",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Error fetching profile:', error);
|
console.error("Error fetching profile:", error);
|
||||||
} finally {
|
} finally {
|
||||||
setFetchingProfile(false);
|
setFetchingProfile(false);
|
||||||
}
|
}
|
||||||
@ -105,39 +160,40 @@ export default function FitnessProfileScreen() {
|
|||||||
const token = await getToken();
|
const token = await getToken();
|
||||||
|
|
||||||
// Prepare data with userId and convert fitnessGoal to fitnessGoals array
|
// Prepare data with userId and convert fitnessGoal to fitnessGoals array
|
||||||
|
// Convert empty strings to undefined for optional enum fields
|
||||||
const dataToSave = {
|
const dataToSave = {
|
||||||
userId: userId,
|
userId: userId,
|
||||||
height: profileData.height,
|
height: profileData.height,
|
||||||
weight: profileData.weight,
|
weight: profileData.weight,
|
||||||
age: profileData.age,
|
age: profileData.age,
|
||||||
gender: profileData.gender,
|
gender: profileData.gender || undefined,
|
||||||
fitnessGoals: profileData.fitnessGoal ? [profileData.fitnessGoal] : [],
|
fitnessGoals: profileData.fitnessGoal ? [profileData.fitnessGoal] : [],
|
||||||
activityLevel: profileData.activityLevel,
|
activityLevel: profileData.activityLevel || undefined,
|
||||||
medicalConditions: profileData.medicalConditions,
|
medicalConditions: profileData.medicalConditions,
|
||||||
allergies: profileData.allergies,
|
allergies: profileData.allergies,
|
||||||
injuries: profileData.injuries,
|
injuries: profileData.injuries,
|
||||||
};
|
};
|
||||||
|
|
||||||
const response = await fetch(`${API_BASE_URL}/api/profile/fitness`, {
|
const response = await fetch(`${API_BASE_URL}/api/profile/fitness`, {
|
||||||
method: 'POST',
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json',
|
"Content-Type": "application/json",
|
||||||
Authorization: `Bearer ${token}`,
|
Authorization: `Bearer ${token}`,
|
||||||
},
|
},
|
||||||
body: JSON.stringify(dataToSave),
|
body: JSON.stringify(dataToSave),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
Alert.alert('Success', 'Fitness profile saved successfully!', [
|
Alert.alert("Success", "Fitness profile saved successfully!", [
|
||||||
{ text: 'OK', onPress: () => router.back() },
|
{ text: "OK", onPress: () => router.back() },
|
||||||
]);
|
]);
|
||||||
} else {
|
} else {
|
||||||
const error = await response.json();
|
const error = await response.json();
|
||||||
Alert.alert('Error', error.error || 'Failed to save profile');
|
Alert.alert("Error", error.error || "Failed to save profile");
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Error saving profile:', error);
|
console.error("Error saving profile:", error);
|
||||||
Alert.alert('Error', 'Failed to save fitness profile');
|
Alert.alert("Error", "Failed to save fitness profile");
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
@ -161,7 +217,10 @@ export default function FitnessProfileScreen() {
|
|||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<LinearGradient colors={theme.gradients.primary} style={styles.header}>
|
<LinearGradient colors={theme.gradients.primary} style={styles.header}>
|
||||||
<TouchableOpacity style={styles.backButton} onPress={() => router.back()}>
|
<TouchableOpacity
|
||||||
|
style={styles.backButton}
|
||||||
|
onPress={() => router.back()}
|
||||||
|
>
|
||||||
<Ionicons name="arrow-back" size={24} color="#fff" />
|
<Ionicons name="arrow-back" size={24} color="#fff" />
|
||||||
</TouchableOpacity>
|
</TouchableOpacity>
|
||||||
<Text style={styles.headerTitle}>Fitness Profile</Text>
|
<Text style={styles.headerTitle}>Fitness Profile</Text>
|
||||||
@ -181,12 +240,19 @@ export default function FitnessProfileScreen() {
|
|||||||
<View style={styles.inputGroup}>
|
<View style={styles.inputGroup}>
|
||||||
<Text style={styles.label}>Height (cm)</Text>
|
<Text style={styles.label}>Height (cm)</Text>
|
||||||
<View style={styles.inputContainer}>
|
<View style={styles.inputContainer}>
|
||||||
<Ionicons name="resize-outline" size={20} color={theme.colors.gray400} />
|
<Ionicons
|
||||||
|
name="resize-outline"
|
||||||
|
size={20}
|
||||||
|
color={theme.colors.gray400}
|
||||||
|
/>
|
||||||
<TextInput
|
<TextInput
|
||||||
style={styles.input}
|
style={styles.input}
|
||||||
value={profileData.height?.toString() || ''}
|
value={profileData.height?.toString() || ""}
|
||||||
onChangeText={(text) =>
|
onChangeText={(text) =>
|
||||||
updateField('height', text ? parseFloat(text) : undefined)
|
updateField(
|
||||||
|
"height",
|
||||||
|
text ? parseFloat(text) : undefined,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
keyboardType="decimal-pad"
|
keyboardType="decimal-pad"
|
||||||
placeholder="175"
|
placeholder="175"
|
||||||
@ -197,12 +263,19 @@ export default function FitnessProfileScreen() {
|
|||||||
<View style={styles.inputGroup}>
|
<View style={styles.inputGroup}>
|
||||||
<Text style={styles.label}>Weight (kg)</Text>
|
<Text style={styles.label}>Weight (kg)</Text>
|
||||||
<View style={styles.inputContainer}>
|
<View style={styles.inputContainer}>
|
||||||
<Ionicons name="scale-outline" size={20} color={theme.colors.gray400} />
|
<Ionicons
|
||||||
|
name="scale-outline"
|
||||||
|
size={20}
|
||||||
|
color={theme.colors.gray400}
|
||||||
|
/>
|
||||||
<TextInput
|
<TextInput
|
||||||
style={styles.input}
|
style={styles.input}
|
||||||
value={profileData.weight?.toString() || ''}
|
value={profileData.weight?.toString() || ""}
|
||||||
onChangeText={(text) =>
|
onChangeText={(text) =>
|
||||||
updateField('weight', text ? parseFloat(text) : undefined)
|
updateField(
|
||||||
|
"weight",
|
||||||
|
text ? parseFloat(text) : undefined,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
keyboardType="decimal-pad"
|
keyboardType="decimal-pad"
|
||||||
placeholder="70"
|
placeholder="70"
|
||||||
@ -214,12 +287,16 @@ export default function FitnessProfileScreen() {
|
|||||||
<View style={styles.inputGroup}>
|
<View style={styles.inputGroup}>
|
||||||
<Text style={styles.label}>Age</Text>
|
<Text style={styles.label}>Age</Text>
|
||||||
<View style={styles.inputContainer}>
|
<View style={styles.inputContainer}>
|
||||||
<Ionicons name="calendar-outline" size={20} color={theme.colors.gray400} />
|
<Ionicons
|
||||||
|
name="calendar-outline"
|
||||||
|
size={20}
|
||||||
|
color={theme.colors.gray400}
|
||||||
|
/>
|
||||||
<TextInput
|
<TextInput
|
||||||
style={styles.input}
|
style={styles.input}
|
||||||
value={profileData.age?.toString() || ''}
|
value={profileData.age?.toString() || ""}
|
||||||
onChangeText={(text) =>
|
onChangeText={(text) =>
|
||||||
updateField('age', text ? parseInt(text, 10) : undefined)
|
updateField("age", text ? parseInt(text, 10) : undefined)
|
||||||
}
|
}
|
||||||
keyboardType="number-pad"
|
keyboardType="number-pad"
|
||||||
placeholder="25"
|
placeholder="25"
|
||||||
@ -239,9 +316,10 @@ export default function FitnessProfileScreen() {
|
|||||||
key={option.value}
|
key={option.value}
|
||||||
style={[
|
style={[
|
||||||
styles.optionCard,
|
styles.optionCard,
|
||||||
profileData.gender === option.value && styles.optionCardActive,
|
profileData.gender === option.value &&
|
||||||
|
styles.optionCardActive,
|
||||||
]}
|
]}
|
||||||
onPress={() => updateField('gender', option.value)}
|
onPress={() => updateField("gender", option.value)}
|
||||||
>
|
>
|
||||||
<Ionicons
|
<Ionicons
|
||||||
name={option.icon as any}
|
name={option.icon as any}
|
||||||
@ -255,7 +333,8 @@ export default function FitnessProfileScreen() {
|
|||||||
<Text
|
<Text
|
||||||
style={[
|
style={[
|
||||||
styles.optionLabel,
|
styles.optionLabel,
|
||||||
profileData.gender === option.value && styles.optionLabelActive,
|
profileData.gender === option.value &&
|
||||||
|
styles.optionLabelActive,
|
||||||
]}
|
]}
|
||||||
>
|
>
|
||||||
{option.label}
|
{option.label}
|
||||||
@ -273,17 +352,32 @@ export default function FitnessProfileScreen() {
|
|||||||
<React.Fragment key={option.value}>
|
<React.Fragment key={option.value}>
|
||||||
<TouchableOpacity
|
<TouchableOpacity
|
||||||
style={styles.listItem}
|
style={styles.listItem}
|
||||||
onPress={() => updateField('fitnessGoal', option.value)}
|
onPress={() => updateField("fitnessGoal", option.value)}
|
||||||
>
|
>
|
||||||
<View style={[styles.iconCircle, { backgroundColor: `${option.color}20` }]}>
|
<View
|
||||||
<Ionicons name={option.icon as any} size={20} color={option.color} />
|
style={[
|
||||||
|
styles.iconCircle,
|
||||||
|
{ backgroundColor: `${option.color}20` },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Ionicons
|
||||||
|
name={option.icon as any}
|
||||||
|
size={20}
|
||||||
|
color={option.color}
|
||||||
|
/>
|
||||||
</View>
|
</View>
|
||||||
<Text style={styles.listItemText}>{option.label}</Text>
|
<Text style={styles.listItemText}>{option.label}</Text>
|
||||||
{profileData.fitnessGoal === option.value && (
|
{profileData.fitnessGoal === option.value && (
|
||||||
<Ionicons name="checkmark-circle" size={24} color={theme.colors.primary} />
|
<Ionicons
|
||||||
|
name="checkmark-circle"
|
||||||
|
size={24}
|
||||||
|
color={theme.colors.primary}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</TouchableOpacity>
|
</TouchableOpacity>
|
||||||
{index < FITNESS_GOAL_OPTIONS.length - 1 && <View style={styles.divider} />}
|
{index < FITNESS_GOAL_OPTIONS.length - 1 && (
|
||||||
|
<View style={styles.divider} />
|
||||||
|
)}
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
))}
|
))}
|
||||||
</View>
|
</View>
|
||||||
@ -297,17 +391,25 @@ export default function FitnessProfileScreen() {
|
|||||||
<React.Fragment key={option.value}>
|
<React.Fragment key={option.value}>
|
||||||
<TouchableOpacity
|
<TouchableOpacity
|
||||||
style={styles.listItem}
|
style={styles.listItem}
|
||||||
onPress={() => updateField('activityLevel', option.value)}
|
onPress={() => updateField("activityLevel", option.value)}
|
||||||
>
|
>
|
||||||
<View style={{ flex: 1 }}>
|
<View style={{ flex: 1 }}>
|
||||||
<Text style={styles.listItemText}>{option.label}</Text>
|
<Text style={styles.listItemText}>{option.label}</Text>
|
||||||
<Text style={styles.listItemDescription}>{option.description}</Text>
|
<Text style={styles.listItemDescription}>
|
||||||
|
{option.description}
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
{profileData.activityLevel === option.value && (
|
{profileData.activityLevel === option.value && (
|
||||||
<Ionicons name="checkmark-circle" size={24} color={theme.colors.primary} />
|
<Ionicons
|
||||||
|
name="checkmark-circle"
|
||||||
|
size={24}
|
||||||
|
color={theme.colors.primary}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</TouchableOpacity>
|
</TouchableOpacity>
|
||||||
{index < ACTIVITY_LEVEL_OPTIONS.length - 1 && <View style={styles.divider} />}
|
{index < ACTIVITY_LEVEL_OPTIONS.length - 1 && (
|
||||||
|
<View style={styles.divider} />
|
||||||
|
)}
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
))}
|
))}
|
||||||
</View>
|
</View>
|
||||||
@ -315,14 +417,18 @@ export default function FitnessProfileScreen() {
|
|||||||
|
|
||||||
{/* Health Information */}
|
{/* Health Information */}
|
||||||
<View style={styles.section}>
|
<View style={styles.section}>
|
||||||
<Text style={styles.sectionTitle}>Health Information (Optional)</Text>
|
<Text style={styles.sectionTitle}>
|
||||||
|
Health Information (Optional)
|
||||||
|
</Text>
|
||||||
<View style={styles.card}>
|
<View style={styles.card}>
|
||||||
<View style={styles.inputGroup}>
|
<View style={styles.inputGroup}>
|
||||||
<Text style={styles.label}>Medical Conditions</Text>
|
<Text style={styles.label}>Medical Conditions</Text>
|
||||||
<TextInput
|
<TextInput
|
||||||
style={[styles.textArea]}
|
style={[styles.textArea]}
|
||||||
value={profileData.medicalConditions || ''}
|
value={profileData.medicalConditions || ""}
|
||||||
onChangeText={(text) => updateField('medicalConditions', text)}
|
onChangeText={(text) =>
|
||||||
|
updateField("medicalConditions", text)
|
||||||
|
}
|
||||||
placeholder="e.g., Asthma, diabetes..."
|
placeholder="e.g., Asthma, diabetes..."
|
||||||
placeholderTextColor={theme.colors.gray400}
|
placeholderTextColor={theme.colors.gray400}
|
||||||
multiline
|
multiline
|
||||||
@ -334,8 +440,8 @@ export default function FitnessProfileScreen() {
|
|||||||
<Text style={styles.label}>Allergies</Text>
|
<Text style={styles.label}>Allergies</Text>
|
||||||
<TextInput
|
<TextInput
|
||||||
style={[styles.textArea]}
|
style={[styles.textArea]}
|
||||||
value={profileData.allergies || ''}
|
value={profileData.allergies || ""}
|
||||||
onChangeText={(text) => updateField('allergies', text)}
|
onChangeText={(text) => updateField("allergies", text)}
|
||||||
placeholder="e.g., Peanuts, latex..."
|
placeholder="e.g., Peanuts, latex..."
|
||||||
placeholderTextColor={theme.colors.gray400}
|
placeholderTextColor={theme.colors.gray400}
|
||||||
multiline
|
multiline
|
||||||
@ -347,8 +453,8 @@ export default function FitnessProfileScreen() {
|
|||||||
<Text style={styles.label}>Injuries</Text>
|
<Text style={styles.label}>Injuries</Text>
|
||||||
<TextInput
|
<TextInput
|
||||||
style={[styles.textArea]}
|
style={[styles.textArea]}
|
||||||
value={profileData.injuries || ''}
|
value={profileData.injuries || ""}
|
||||||
onChangeText={(text) => updateField('injuries', text)}
|
onChangeText={(text) => updateField("injuries", text)}
|
||||||
placeholder="e.g., Previous knee injury..."
|
placeholder="e.g., Previous knee injury..."
|
||||||
placeholderTextColor={theme.colors.gray400}
|
placeholderTextColor={theme.colors.gray400}
|
||||||
multiline
|
multiline
|
||||||
@ -367,7 +473,10 @@ export default function FitnessProfileScreen() {
|
|||||||
onPress={handleSave}
|
onPress={handleSave}
|
||||||
disabled={loading}
|
disabled={loading}
|
||||||
>
|
>
|
||||||
<LinearGradient colors={theme.gradients.primary} style={styles.saveButtonGradient}>
|
<LinearGradient
|
||||||
|
colors={theme.gradients.primary}
|
||||||
|
style={styles.saveButtonGradient}
|
||||||
|
>
|
||||||
{loading ? (
|
{loading ? (
|
||||||
<ActivityIndicator color="#fff" />
|
<ActivityIndicator color="#fff" />
|
||||||
) : (
|
) : (
|
||||||
@ -391,15 +500,15 @@ const styles = StyleSheet.create({
|
|||||||
},
|
},
|
||||||
loadingContainer: {
|
loadingContainer: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
justifyContent: 'center',
|
justifyContent: "center",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
backgroundColor: theme.colors.background,
|
backgroundColor: theme.colors.background,
|
||||||
},
|
},
|
||||||
header: {
|
header: {
|
||||||
flexDirection: 'row',
|
flexDirection: "row",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
justifyContent: 'space-between',
|
justifyContent: "space-between",
|
||||||
paddingTop: Platform.OS === 'ios' ? 60 : 40,
|
paddingTop: Platform.OS === "ios" ? 60 : 40,
|
||||||
paddingBottom: 20,
|
paddingBottom: 20,
|
||||||
paddingHorizontal: 20,
|
paddingHorizontal: 20,
|
||||||
},
|
},
|
||||||
@ -407,14 +516,14 @@ const styles = StyleSheet.create({
|
|||||||
width: 40,
|
width: 40,
|
||||||
height: 40,
|
height: 40,
|
||||||
borderRadius: 20,
|
borderRadius: 20,
|
||||||
backgroundColor: 'rgba(255, 255, 255, 0.2)',
|
backgroundColor: "rgba(255, 255, 255, 0.2)",
|
||||||
justifyContent: 'center',
|
justifyContent: "center",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
headerTitle: {
|
headerTitle: {
|
||||||
fontSize: theme.typography.fontSize['2xl'],
|
fontSize: theme.typography.fontSize["2xl"],
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
fontWeight: theme.typography.fontWeight.bold,
|
||||||
color: '#fff',
|
color: "#fff",
|
||||||
},
|
},
|
||||||
scrollView: {
|
scrollView: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
@ -433,7 +542,7 @@ const styles = StyleSheet.create({
|
|||||||
marginBottom: 12,
|
marginBottom: 12,
|
||||||
},
|
},
|
||||||
card: {
|
card: {
|
||||||
backgroundColor: '#fff',
|
backgroundColor: "#fff",
|
||||||
borderRadius: theme.borderRadius.xl,
|
borderRadius: theme.borderRadius.xl,
|
||||||
padding: 16,
|
padding: 16,
|
||||||
...theme.shadows.subtle,
|
...theme.shadows.subtle,
|
||||||
@ -441,7 +550,7 @@ const styles = StyleSheet.create({
|
|||||||
borderColor: theme.colors.gray100,
|
borderColor: theme.colors.gray100,
|
||||||
},
|
},
|
||||||
row: {
|
row: {
|
||||||
flexDirection: 'row',
|
flexDirection: "row",
|
||||||
gap: 12,
|
gap: 12,
|
||||||
marginBottom: 16,
|
marginBottom: 16,
|
||||||
},
|
},
|
||||||
@ -456,8 +565,8 @@ const styles = StyleSheet.create({
|
|||||||
marginBottom: 8,
|
marginBottom: 8,
|
||||||
},
|
},
|
||||||
inputContainer: {
|
inputContainer: {
|
||||||
flexDirection: 'row',
|
flexDirection: "row",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
backgroundColor: theme.colors.gray50,
|
backgroundColor: theme.colors.gray50,
|
||||||
borderRadius: theme.borderRadius.lg,
|
borderRadius: theme.borderRadius.lg,
|
||||||
borderWidth: 1,
|
borderWidth: 1,
|
||||||
@ -482,15 +591,15 @@ const styles = StyleSheet.create({
|
|||||||
minHeight: 80,
|
minHeight: 80,
|
||||||
},
|
},
|
||||||
optionsRow: {
|
optionsRow: {
|
||||||
flexDirection: 'row',
|
flexDirection: "row",
|
||||||
gap: 12,
|
gap: 12,
|
||||||
},
|
},
|
||||||
optionCard: {
|
optionCard: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
backgroundColor: '#fff',
|
backgroundColor: "#fff",
|
||||||
borderRadius: theme.borderRadius.lg,
|
borderRadius: theme.borderRadius.lg,
|
||||||
padding: 16,
|
padding: 16,
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
gap: 8,
|
gap: 8,
|
||||||
borderWidth: 2,
|
borderWidth: 2,
|
||||||
borderColor: theme.colors.gray200,
|
borderColor: theme.colors.gray200,
|
||||||
@ -510,8 +619,8 @@ const styles = StyleSheet.create({
|
|||||||
fontWeight: theme.typography.fontWeight.bold,
|
fontWeight: theme.typography.fontWeight.bold,
|
||||||
},
|
},
|
||||||
listItem: {
|
listItem: {
|
||||||
flexDirection: 'row',
|
flexDirection: "row",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
paddingVertical: 12,
|
paddingVertical: 12,
|
||||||
gap: 12,
|
gap: 12,
|
||||||
},
|
},
|
||||||
@ -519,8 +628,8 @@ const styles = StyleSheet.create({
|
|||||||
width: 40,
|
width: 40,
|
||||||
height: 40,
|
height: 40,
|
||||||
borderRadius: 20,
|
borderRadius: 20,
|
||||||
justifyContent: 'center',
|
justifyContent: "center",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
listItemText: {
|
listItemText: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
@ -538,25 +647,25 @@ const styles = StyleSheet.create({
|
|||||||
backgroundColor: theme.colors.gray100,
|
backgroundColor: theme.colors.gray100,
|
||||||
},
|
},
|
||||||
footer: {
|
footer: {
|
||||||
position: 'absolute',
|
position: "absolute",
|
||||||
bottom: 0,
|
bottom: 0,
|
||||||
left: 0,
|
left: 0,
|
||||||
right: 0,
|
right: 0,
|
||||||
padding: 20,
|
padding: 20,
|
||||||
paddingBottom: Platform.OS === 'ios' ? 40 : 20,
|
paddingBottom: Platform.OS === "ios" ? 40 : 20,
|
||||||
backgroundColor: '#fff',
|
backgroundColor: "#fff",
|
||||||
borderTopWidth: 1,
|
borderTopWidth: 1,
|
||||||
borderTopColor: theme.colors.gray100,
|
borderTopColor: theme.colors.gray100,
|
||||||
...theme.shadows.medium,
|
...theme.shadows.medium,
|
||||||
},
|
},
|
||||||
saveButton: {
|
saveButton: {
|
||||||
borderRadius: theme.borderRadius.lg,
|
borderRadius: theme.borderRadius.lg,
|
||||||
overflow: 'hidden',
|
overflow: "hidden",
|
||||||
},
|
},
|
||||||
saveButtonGradient: {
|
saveButtonGradient: {
|
||||||
flexDirection: 'row',
|
flexDirection: "row",
|
||||||
alignItems: 'center',
|
alignItems: "center",
|
||||||
justifyContent: 'center',
|
justifyContent: "center",
|
||||||
paddingVertical: 16,
|
paddingVertical: 16,
|
||||||
gap: 8,
|
gap: 8,
|
||||||
},
|
},
|
||||||
@ -566,6 +675,6 @@ const styles = StyleSheet.create({
|
|||||||
saveButtonText: {
|
saveButtonText: {
|
||||||
fontSize: theme.typography.fontSize.base,
|
fontSize: theme.typography.fontSize.base,
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
fontWeight: theme.typography.fontWeight.bold,
|
||||||
color: '#fff',
|
color: "#fff",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
128
apps/mobile/src/components/ActivityRing.tsx
Normal file
128
apps/mobile/src/components/ActivityRing.tsx
Normal file
@ -0,0 +1,128 @@
|
|||||||
|
import React, { useEffect, useRef } from "react";
|
||||||
|
import { View, Animated, StyleSheet, Text } from "react-native";
|
||||||
|
import Svg, { Circle } from "react-native-svg";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
|
const AnimatedCircle = Animated.createAnimatedComponent(Circle);
|
||||||
|
|
||||||
|
interface ActivityRingProps {
|
||||||
|
size?: number;
|
||||||
|
strokeWidth?: number;
|
||||||
|
progress: number;
|
||||||
|
current: number;
|
||||||
|
goal: number;
|
||||||
|
label: string;
|
||||||
|
color: string;
|
||||||
|
icon?: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ActivityRing({
|
||||||
|
size = 100,
|
||||||
|
strokeWidth = 10,
|
||||||
|
progress,
|
||||||
|
current,
|
||||||
|
goal,
|
||||||
|
label,
|
||||||
|
color,
|
||||||
|
icon,
|
||||||
|
}: ActivityRingProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
|
const animatedValue = useRef(new Animated.Value(0)).current;
|
||||||
|
const radius = (size - strokeWidth) / 2;
|
||||||
|
const circumference = radius * 2 * Math.PI;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
Animated.timing(animatedValue, {
|
||||||
|
toValue: Math.min(progress, 100),
|
||||||
|
duration: 1200,
|
||||||
|
useNativeDriver: true,
|
||||||
|
}).start();
|
||||||
|
}, [progress]);
|
||||||
|
|
||||||
|
const strokeDashoffset = animatedValue.interpolate({
|
||||||
|
inputRange: [0, 100],
|
||||||
|
outputRange: [circumference, 0],
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View style={styles.container}>
|
||||||
|
<View
|
||||||
|
style={{
|
||||||
|
width: size,
|
||||||
|
height: size,
|
||||||
|
justifyContent: "center",
|
||||||
|
alignItems: "center",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Svg width={size} height={size} viewBox={`0 0 ${size} ${size}`}>
|
||||||
|
<Circle
|
||||||
|
cx={size / 2}
|
||||||
|
cy={size / 2}
|
||||||
|
r={radius}
|
||||||
|
stroke={colors.surfaceElevated}
|
||||||
|
strokeWidth={strokeWidth}
|
||||||
|
fill="transparent"
|
||||||
|
/>
|
||||||
|
<AnimatedCircle
|
||||||
|
cx={size / 2}
|
||||||
|
cy={size / 2}
|
||||||
|
r={radius}
|
||||||
|
stroke={color}
|
||||||
|
strokeWidth={strokeWidth}
|
||||||
|
fill="transparent"
|
||||||
|
strokeDasharray={circumference}
|
||||||
|
strokeDashoffset={strokeDashoffset}
|
||||||
|
strokeLinecap="round"
|
||||||
|
rotation="-90"
|
||||||
|
origin={`${size / 2}, ${size / 2}`}
|
||||||
|
/>
|
||||||
|
</Svg>
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
StyleSheet.absoluteFillObject,
|
||||||
|
{ justifyContent: "center", alignItems: "center" },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{icon ? (
|
||||||
|
<View style={styles.iconContainer}>{icon}</View>
|
||||||
|
) : (
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.statLarge,
|
||||||
|
{ color: colors.textPrimary, fontSize: size * 0.28 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{Math.round(current)}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.label,
|
||||||
|
{ color: colors.textTertiary, marginTop: 8, textAlign: "center" },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textSecondary, marginTop: 2 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
/ {goal.toLocaleString()}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
container: {
|
||||||
|
alignItems: "center",
|
||||||
|
},
|
||||||
|
iconContainer: {
|
||||||
|
justifyContent: "center",
|
||||||
|
alignItems: "center",
|
||||||
|
},
|
||||||
|
});
|
||||||
117
apps/mobile/src/components/Badge.tsx
Normal file
117
apps/mobile/src/components/Badge.tsx
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { View, Text, StyleSheet, ViewStyle, StyleProp } from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
import { fontSize, fontWeight } from "../styles/typography";
|
||||||
|
|
||||||
|
type BadgeVariant =
|
||||||
|
| "neutral"
|
||||||
|
| "success"
|
||||||
|
| "warning"
|
||||||
|
| "danger"
|
||||||
|
| "info"
|
||||||
|
| "primary";
|
||||||
|
type BadgeSize = "sm" | "md" | "lg";
|
||||||
|
|
||||||
|
interface BadgeProps {
|
||||||
|
label: string;
|
||||||
|
variant?: BadgeVariant;
|
||||||
|
size?: BadgeSize;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
emoji?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Badge({
|
||||||
|
label,
|
||||||
|
variant = "neutral",
|
||||||
|
size = "md",
|
||||||
|
style,
|
||||||
|
emoji,
|
||||||
|
}: BadgeProps) {
|
||||||
|
const { colors } = useTheme();
|
||||||
|
|
||||||
|
const sizeStyles = {
|
||||||
|
sm: {
|
||||||
|
paddingVertical: 6,
|
||||||
|
paddingHorizontal: 12,
|
||||||
|
fontSize: fontSize.xs,
|
||||||
|
},
|
||||||
|
md: {
|
||||||
|
paddingVertical: 8,
|
||||||
|
paddingHorizontal: 14,
|
||||||
|
fontSize: fontSize.sm,
|
||||||
|
},
|
||||||
|
lg: {
|
||||||
|
paddingVertical: 10,
|
||||||
|
paddingHorizontal: 18,
|
||||||
|
fontSize: fontSize.base,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const variantStyles: Record<
|
||||||
|
BadgeVariant,
|
||||||
|
{ backgroundColor: string; color: string }
|
||||||
|
> = {
|
||||||
|
neutral: {
|
||||||
|
backgroundColor: colors.surfaceElevated,
|
||||||
|
color: colors.textSecondary,
|
||||||
|
},
|
||||||
|
success: {
|
||||||
|
backgroundColor: colors.success,
|
||||||
|
color: colors.white,
|
||||||
|
},
|
||||||
|
warning: {
|
||||||
|
backgroundColor: colors.warning,
|
||||||
|
color: colors.black,
|
||||||
|
},
|
||||||
|
danger: {
|
||||||
|
backgroundColor: colors.danger,
|
||||||
|
color: colors.white,
|
||||||
|
},
|
||||||
|
info: {
|
||||||
|
backgroundColor: colors.info,
|
||||||
|
color: colors.white,
|
||||||
|
},
|
||||||
|
primary: {
|
||||||
|
backgroundColor: colors.primary,
|
||||||
|
color: colors.white,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.badge,
|
||||||
|
{
|
||||||
|
backgroundColor: variantStyles[variant].backgroundColor,
|
||||||
|
paddingVertical: sizeStyles[size].paddingVertical,
|
||||||
|
paddingHorizontal: sizeStyles[size].paddingHorizontal,
|
||||||
|
},
|
||||||
|
style,
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
styles.label,
|
||||||
|
{
|
||||||
|
color: variantStyles[variant].color,
|
||||||
|
fontSize: sizeStyles[size].fontSize,
|
||||||
|
fontWeight: fontWeight.bold,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{emoji && `${emoji} `}
|
||||||
|
{label}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
badge: {
|
||||||
|
borderRadius: 12,
|
||||||
|
alignSelf: "flex-start",
|
||||||
|
},
|
||||||
|
label: {
|
||||||
|
textAlign: "center",
|
||||||
|
},
|
||||||
|
});
|
||||||
@ -1,143 +1,151 @@
|
|||||||
import React from 'react';
|
import React from "react";
|
||||||
import { View, StyleSheet, TouchableOpacity, Platform, Dimensions } from 'react-native';
|
import { View, StyleSheet, TouchableOpacity, Text } from "react-native";
|
||||||
import { BottomTabBarProps } from '@react-navigation/bottom-tabs';
|
import { BottomTabBarProps } from "@react-navigation/bottom-tabs";
|
||||||
import { Ionicons } from '@expo/vector-icons';
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import { LinearGradient } from 'expo-linear-gradient';
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { theme } from '../styles/theme';
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
import { Animated } from 'react-native';
|
|
||||||
|
|
||||||
const { width } = Dimensions.get('window');
|
export function CustomTabBar({
|
||||||
|
state,
|
||||||
|
descriptors,
|
||||||
|
navigation,
|
||||||
|
}: BottomTabBarProps) {
|
||||||
|
const { colors } = useTheme();
|
||||||
|
const insets = useSafeAreaInsets();
|
||||||
|
|
||||||
export function CustomTabBar({ state, descriptors, navigation }: BottomTabBarProps) {
|
return (
|
||||||
return (
|
<View
|
||||||
<View style={styles.container}>
|
style={[
|
||||||
<LinearGradient
|
styles.container,
|
||||||
colors={['rgba(255, 255, 255, 0.9)', 'rgba(255, 255, 255, 0.7)']}
|
{
|
||||||
style={[styles.tabBar, theme.shadows.medium]}
|
backgroundColor: colors.surface,
|
||||||
start={{ x: 0, y: 0 }}
|
borderTopColor: colors.border,
|
||||||
end={{ x: 0, y: 1 }}
|
paddingBottom: insets.bottom,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{state.routes.map((route, index) => {
|
||||||
|
const { options } = descriptors[route.key];
|
||||||
|
const isFocused = state.index === index;
|
||||||
|
|
||||||
|
const onPress = () => {
|
||||||
|
const event = navigation.emit({
|
||||||
|
type: "tabPress",
|
||||||
|
target: route.key,
|
||||||
|
canPreventDefault: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!isFocused && !event.defaultPrevented) {
|
||||||
|
navigation.navigate(route.name);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getIconName = (
|
||||||
|
routeName: string,
|
||||||
|
focused: boolean,
|
||||||
|
): keyof typeof Ionicons.glyphMap => {
|
||||||
|
switch (routeName) {
|
||||||
|
case "index":
|
||||||
|
return focused ? "home" : "home-outline";
|
||||||
|
case "goals":
|
||||||
|
return focused ? "trophy" : "trophy-outline";
|
||||||
|
case "attendance":
|
||||||
|
return focused ? "calendar" : "calendar-outline";
|
||||||
|
case "recommendations":
|
||||||
|
return focused ? "sparkles" : "sparkles-outline";
|
||||||
|
case "profile":
|
||||||
|
return focused ? "person" : "person-outline";
|
||||||
|
default:
|
||||||
|
return "ellipse-outline";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLabel = (routeName: string) => {
|
||||||
|
switch (routeName) {
|
||||||
|
case "index":
|
||||||
|
return "Home";
|
||||||
|
case "goals":
|
||||||
|
return "Goals";
|
||||||
|
case "attendance":
|
||||||
|
return "Attendance";
|
||||||
|
case "recommendations":
|
||||||
|
return "Plans";
|
||||||
|
case "profile":
|
||||||
|
return "Profile";
|
||||||
|
default:
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<TouchableOpacity
|
||||||
|
key={index}
|
||||||
|
accessibilityRole="button"
|
||||||
|
accessibilityState={isFocused ? { selected: true } : {}}
|
||||||
|
accessibilityLabel={options.tabBarAccessibilityLabel}
|
||||||
|
testID={(options as any).tabBarTestID}
|
||||||
|
onPress={onPress}
|
||||||
|
style={styles.tabItem}
|
||||||
|
activeOpacity={0.7}
|
||||||
|
>
|
||||||
|
<View style={styles.iconWrapper}>
|
||||||
|
<Ionicons
|
||||||
|
name={getIconName(route.name, isFocused)}
|
||||||
|
size={26}
|
||||||
|
color={isFocused ? colors.primary : colors.textTertiary}
|
||||||
|
/>
|
||||||
|
{isFocused && (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.indicator,
|
||||||
|
{ backgroundColor: colors.primary },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
styles.label,
|
||||||
|
{
|
||||||
|
color: isFocused ? colors.primary : colors.textTertiary,
|
||||||
|
fontWeight: isFocused ? "700" : "500",
|
||||||
|
},
|
||||||
|
]}
|
||||||
>
|
>
|
||||||
{state.routes.map((route, index) => {
|
{getLabel(route.name)}
|
||||||
const { options } = descriptors[route.key];
|
</Text>
|
||||||
const isFocused = state.index === index;
|
</TouchableOpacity>
|
||||||
|
);
|
||||||
const onPress = () => {
|
})}
|
||||||
const event = navigation.emit({
|
</View>
|
||||||
type: 'tabPress',
|
);
|
||||||
target: route.key,
|
|
||||||
canPreventDefault: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!isFocused && !event.defaultPrevented) {
|
|
||||||
navigation.navigate(route.name);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const getIconName = (routeName: string, focused: boolean): keyof typeof Ionicons.glyphMap => {
|
|
||||||
switch (routeName) {
|
|
||||||
case 'index':
|
|
||||||
return focused ? 'home' : 'home-outline';
|
|
||||||
case 'goals':
|
|
||||||
return focused ? 'trophy' : 'trophy-outline';
|
|
||||||
case 'attendance':
|
|
||||||
return focused ? 'calendar' : 'calendar-outline';
|
|
||||||
case 'recommendations':
|
|
||||||
return focused ? 'sparkles' : 'sparkles-outline';
|
|
||||||
case 'profile':
|
|
||||||
return focused ? 'person' : 'person-outline';
|
|
||||||
default:
|
|
||||||
return 'ellipse-outline';
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Animation for scale
|
|
||||||
const scaleValue = React.useRef(new Animated.Value(1)).current;
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
Animated.spring(scaleValue, {
|
|
||||||
toValue: isFocused ? 1.2 : 1,
|
|
||||||
useNativeDriver: true,
|
|
||||||
friction: 8,
|
|
||||||
}).start();
|
|
||||||
}, [isFocused]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<TouchableOpacity
|
|
||||||
key={index}
|
|
||||||
accessibilityRole="button"
|
|
||||||
accessibilityState={isFocused ? { selected: true } : {}}
|
|
||||||
accessibilityLabel={options.tabBarAccessibilityLabel}
|
|
||||||
testID={(options as any).tabBarTestID}
|
|
||||||
onPress={onPress}
|
|
||||||
style={styles.tabItem}
|
|
||||||
activeOpacity={0.7}
|
|
||||||
>
|
|
||||||
<Animated.View style={{ transform: [{ scale: scaleValue }] }}>
|
|
||||||
{isFocused ? (
|
|
||||||
<LinearGradient
|
|
||||||
colors={theme.gradients.primary}
|
|
||||||
style={styles.iconContainer}
|
|
||||||
>
|
|
||||||
<Ionicons
|
|
||||||
name={getIconName(route.name, true)}
|
|
||||||
size={20}
|
|
||||||
color="#fff"
|
|
||||||
/>
|
|
||||||
</LinearGradient>
|
|
||||||
) : (
|
|
||||||
<Ionicons
|
|
||||||
name={getIconName(route.name, false)}
|
|
||||||
size={24}
|
|
||||||
color={theme.colors.gray500}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Animated.View>
|
|
||||||
</TouchableOpacity>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</LinearGradient>
|
|
||||||
</View>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
position: 'absolute',
|
flexDirection: "row",
|
||||||
bottom: 0,
|
height: 70,
|
||||||
left: 0,
|
borderTopWidth: 1,
|
||||||
right: 0,
|
paddingTop: 8,
|
||||||
alignItems: 'center',
|
},
|
||||||
paddingBottom: Platform.OS === 'ios' ? 30 : 20,
|
tabItem: {
|
||||||
pointerEvents: 'box-none',
|
flex: 1,
|
||||||
},
|
alignItems: "center",
|
||||||
tabBar: {
|
justifyContent: "center",
|
||||||
flexDirection: 'row',
|
height: "100%",
|
||||||
backgroundColor: 'rgba(255, 255, 255, 0.8)',
|
},
|
||||||
borderRadius: 35,
|
iconWrapper: {
|
||||||
height: 70,
|
alignItems: "center",
|
||||||
width: width - 40,
|
justifyContent: "center",
|
||||||
justifyContent: 'space-around',
|
},
|
||||||
alignItems: 'center',
|
indicator: {
|
||||||
borderWidth: 1,
|
width: 20,
|
||||||
borderColor: 'rgba(255, 255, 255, 0.5)',
|
height: 4,
|
||||||
paddingHorizontal: 10,
|
borderRadius: 2,
|
||||||
},
|
marginTop: 4,
|
||||||
tabItem: {
|
},
|
||||||
flex: 1,
|
label: {
|
||||||
alignItems: 'center',
|
fontSize: 11,
|
||||||
justifyContent: 'center',
|
marginTop: 4,
|
||||||
height: '100%',
|
},
|
||||||
},
|
|
||||||
iconContainer: {
|
|
||||||
width: 40,
|
|
||||||
height: 40,
|
|
||||||
borderRadius: 20,
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'center',
|
|
||||||
shadowColor: theme.colors.primary,
|
|
||||||
shadowOffset: { width: 0, height: 4 },
|
|
||||||
shadowOpacity: 0.3,
|
|
||||||
shadowRadius: 8,
|
|
||||||
elevation: 4,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|||||||
@ -1,426 +1,526 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from "react";
|
||||||
import {
|
import {
|
||||||
View,
|
View,
|
||||||
Text,
|
Text,
|
||||||
StyleSheet,
|
StyleSheet,
|
||||||
Modal,
|
Modal,
|
||||||
TextInput,
|
TextInput,
|
||||||
TouchableOpacity,
|
TouchableOpacity,
|
||||||
ScrollView,
|
ScrollView,
|
||||||
Platform,
|
Platform,
|
||||||
Alert,
|
Alert,
|
||||||
} from 'react-native';
|
} from "react-native";
|
||||||
import { Ionicons } from '@expo/vector-icons';
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import DateTimePicker from '@react-native-community/datetimepicker';
|
import DateTimePicker from "@react-native-community/datetimepicker";
|
||||||
import type { CreateGoalData } from '../services/fitnessGoals';
|
import type { CreateGoalData } from "../services/fitnessGoals";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
import { MinimalButton } from "./MinimalButton";
|
||||||
|
|
||||||
interface GoalCreationModalProps {
|
interface GoalCreationModalProps {
|
||||||
visible: boolean;
|
visible: boolean;
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
onSubmit: (goalData: CreateGoalData) => Promise<void>;
|
onSubmit: (goalData: CreateGoalData) => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const GOAL_TYPES = [
|
const GOAL_TYPES = [
|
||||||
{ value: 'weight_target', label: 'Weight Target' },
|
{ value: "weight_target", label: "Weight Target" },
|
||||||
{ value: 'strength_milestone', label: 'Strength Milestone' },
|
{ value: "strength_milestone", label: "Strength Milestone" },
|
||||||
{ value: 'endurance_target', label: 'Endurance Target' },
|
{ value: "endurance_target", label: "Endurance Target" },
|
||||||
{ value: 'flexibility_goal', label: 'Flexibility Goal' },
|
{ value: "flexibility_goal", label: "Flexibility Goal" },
|
||||||
{ value: 'habit_building', label: 'Habit Building' },
|
{ value: "habit_building", label: "Habit Building" },
|
||||||
{ value: 'custom', label: 'Custom Goal' },
|
{ value: "custom", label: "Custom Goal" },
|
||||||
] as const;
|
] as const;
|
||||||
|
|
||||||
const PRIORITIES = [
|
const PRIORITIES = [
|
||||||
{ value: 'low', label: 'Low', color: '#10b981' },
|
{ value: "low", label: "Low" },
|
||||||
{ value: 'medium', label: 'Medium', color: '#f59e0b' },
|
{ value: "medium", label: "Medium" },
|
||||||
{ value: 'high', label: 'High', color: '#ef4444' },
|
{ value: "high", label: "High" },
|
||||||
] as const;
|
] as const;
|
||||||
|
|
||||||
export function GoalCreationModal({ visible, onClose, onSubmit }: GoalCreationModalProps) {
|
export function GoalCreationModal({
|
||||||
const [goalType, setGoalType] = useState<CreateGoalData['goalType']>('weight_target');
|
visible,
|
||||||
const [title, setTitle] = useState('');
|
onClose,
|
||||||
const [description, setDescription] = useState('');
|
onSubmit,
|
||||||
const [targetValue, setTargetValue] = useState('');
|
}: GoalCreationModalProps) {
|
||||||
const [currentValue, setCurrentValue] = useState('');
|
const { colors, typography } = useTheme();
|
||||||
const [unit, setUnit] = useState('');
|
const [goalType, setGoalType] =
|
||||||
const [priority, setPriority] = useState<'low' | 'medium' | 'high'>('medium');
|
useState<CreateGoalData["goalType"]>("weight_target");
|
||||||
const [targetDate, setTargetDate] = useState<Date | undefined>();
|
const [title, setTitle] = useState("");
|
||||||
const [showDatePicker, setShowDatePicker] = useState(false);
|
const [description, setDescription] = useState("");
|
||||||
const [submitting, setSubmitting] = useState(false);
|
const [targetValue, setTargetValue] = useState("");
|
||||||
|
const [currentValue, setCurrentValue] = useState("");
|
||||||
|
const [unit, setUnit] = useState("");
|
||||||
|
const [priority, setPriority] = useState<"low" | "medium" | "high">("medium");
|
||||||
|
const [targetDate, setTargetDate] = useState<Date | undefined>();
|
||||||
|
const [showDatePicker, setShowDatePicker] = useState(false);
|
||||||
|
const [submitting, setSubmitting] = useState(false);
|
||||||
|
|
||||||
const resetForm = () => {
|
const resetForm = () => {
|
||||||
setGoalType('weight_target');
|
setGoalType("weight_target");
|
||||||
setTitle('');
|
setTitle("");
|
||||||
setDescription('');
|
setDescription("");
|
||||||
setTargetValue('');
|
setTargetValue("");
|
||||||
setCurrentValue('');
|
setCurrentValue("");
|
||||||
setUnit('');
|
setUnit("");
|
||||||
setPriority('medium');
|
setPriority("medium");
|
||||||
setTargetDate(undefined);
|
setTargetDate(undefined);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleSubmit = async () => {
|
const handleSubmit = async () => {
|
||||||
if (!title.trim()) {
|
if (!title.trim()) {
|
||||||
Alert.alert('Error', 'Please enter a goal title');
|
Alert.alert("Error", "Please enter a goal title");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setSubmitting(true);
|
setSubmitting(true);
|
||||||
try {
|
try {
|
||||||
const goalData: CreateGoalData = {
|
const goalData: CreateGoalData = {
|
||||||
goalType,
|
goalType,
|
||||||
title: title.trim(),
|
title: title.trim(),
|
||||||
description: description.trim() || undefined,
|
description: description.trim() || undefined,
|
||||||
targetValue: targetValue ? parseFloat(targetValue) : undefined,
|
targetValue: targetValue ? parseFloat(targetValue) : undefined,
|
||||||
currentValue: currentValue ? parseFloat(currentValue) : undefined,
|
currentValue: currentValue ? parseFloat(currentValue) : undefined,
|
||||||
unit: unit.trim() || undefined,
|
unit: unit.trim() || undefined,
|
||||||
targetDate: targetDate?.toISOString(),
|
targetDate: targetDate?.toISOString(),
|
||||||
priority,
|
priority,
|
||||||
};
|
};
|
||||||
|
|
||||||
await onSubmit(goalData);
|
await onSubmit(goalData);
|
||||||
resetForm();
|
resetForm();
|
||||||
onClose();
|
onClose();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Error creating goal:', error);
|
console.error("Error creating goal:", error);
|
||||||
Alert.alert('Error', 'Failed to create goal. Please try again.');
|
Alert.alert("Error", "Failed to create goal. Please try again.");
|
||||||
} finally {
|
} finally {
|
||||||
setSubmitting(false);
|
setSubmitting(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClose = () => {
|
const handleClose = () => {
|
||||||
resetForm();
|
resetForm();
|
||||||
onClose();
|
onClose();
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
const getPriorityColor = (p: "low" | "medium" | "high") => {
|
||||||
<Modal
|
switch (p) {
|
||||||
visible={visible}
|
case "high":
|
||||||
animationType="slide"
|
return colors.danger;
|
||||||
presentationStyle="pageSheet"
|
case "medium":
|
||||||
onRequestClose={handleClose}
|
return colors.warning;
|
||||||
|
case "low":
|
||||||
|
return colors.success;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
visible={visible}
|
||||||
|
animationType="slide"
|
||||||
|
presentationStyle="pageSheet"
|
||||||
|
onRequestClose={handleClose}
|
||||||
|
>
|
||||||
|
<View style={[styles.container, { backgroundColor: colors.background }]}>
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.header,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderBottomColor: colors.border,
|
||||||
|
},
|
||||||
|
]}
|
||||||
>
|
>
|
||||||
<View style={styles.container}>
|
<Text style={[typography.h2, { color: colors.textPrimary }]}>
|
||||||
<View style={styles.header}>
|
Create Fitness Goal
|
||||||
<Text style={styles.headerTitle}>Create Fitness Goal</Text>
|
</Text>
|
||||||
<TouchableOpacity onPress={handleClose} style={styles.closeButton}>
|
<TouchableOpacity onPress={handleClose} style={styles.closeButton}>
|
||||||
<Ionicons name="close" size={28} color="#111827" />
|
<Ionicons name="close" size={28} color={colors.textPrimary} />
|
||||||
</TouchableOpacity>
|
</TouchableOpacity>
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<ScrollView style={styles.content} showsVerticalScrollIndicator={false}>
|
<ScrollView style={styles.content} showsVerticalScrollIndicator={false}>
|
||||||
{/* Goal Type */}
|
{/* Goal Type */}
|
||||||
<View style={styles.field}>
|
<View style={styles.field}>
|
||||||
<Text style={styles.label}>Goal Type *</Text>
|
<Text
|
||||||
<ScrollView horizontal showsHorizontalScrollIndicator={false} style={styles.typeScroll}>
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
{GOAL_TYPES.map((type) => (
|
>
|
||||||
<TouchableOpacity
|
Goal Type *
|
||||||
key={type.value}
|
</Text>
|
||||||
style={[
|
<ScrollView
|
||||||
styles.typeButton,
|
horizontal
|
||||||
goalType === type.value && styles.typeButtonActive,
|
showsHorizontalScrollIndicator={false}
|
||||||
]}
|
style={styles.typeScroll}
|
||||||
onPress={() => setGoalType(type.value)}
|
>
|
||||||
>
|
{GOAL_TYPES.map((type) => (
|
||||||
<Text
|
<TouchableOpacity
|
||||||
style={[
|
key={type.value}
|
||||||
styles.typeButtonText,
|
style={[
|
||||||
goalType === type.value && styles.typeButtonTextActive,
|
styles.typeButton,
|
||||||
]}
|
{
|
||||||
>
|
borderColor: colors.border,
|
||||||
{type.label}
|
backgroundColor: colors.surface,
|
||||||
</Text>
|
},
|
||||||
</TouchableOpacity>
|
goalType === type.value && {
|
||||||
))}
|
backgroundColor: colors.primary,
|
||||||
</ScrollView>
|
borderColor: colors.primary,
|
||||||
</View>
|
},
|
||||||
|
]}
|
||||||
|
onPress={() => setGoalType(type.value)}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textSecondary },
|
||||||
|
goalType === type.value && {
|
||||||
|
color: colors.white,
|
||||||
|
fontWeight: "600",
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{type.label}
|
||||||
|
</Text>
|
||||||
|
</TouchableOpacity>
|
||||||
|
))}
|
||||||
|
</ScrollView>
|
||||||
|
</View>
|
||||||
|
|
||||||
{/* Title */}
|
{/* Title */}
|
||||||
<View style={styles.field}>
|
<View style={styles.field}>
|
||||||
<Text style={styles.label}>Title *</Text>
|
<Text
|
||||||
<TextInput
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
style={styles.input}
|
>
|
||||||
value={title}
|
Title *
|
||||||
onChangeText={setTitle}
|
</Text>
|
||||||
placeholder="e.g., Lose 5kg"
|
<TextInput
|
||||||
placeholderTextColor="#9ca3af"
|
style={[
|
||||||
/>
|
styles.input,
|
||||||
</View>
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderColor: colors.border,
|
||||||
|
color: colors.textPrimary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
value={title}
|
||||||
|
onChangeText={setTitle}
|
||||||
|
placeholder="e.g., Lose 5kg"
|
||||||
|
placeholderTextColor={colors.textTertiary}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
|
||||||
{/* Description */}
|
{/* Description */}
|
||||||
<View style={styles.field}>
|
<View style={styles.field}>
|
||||||
<Text style={styles.label}>Description</Text>
|
<Text
|
||||||
<TextInput
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
style={[styles.input, styles.textArea]}
|
>
|
||||||
value={description}
|
Description
|
||||||
onChangeText={setDescription}
|
</Text>
|
||||||
placeholder="Optional description"
|
<TextInput
|
||||||
placeholderTextColor="#9ca3af"
|
style={[
|
||||||
multiline
|
styles.input,
|
||||||
numberOfLines={3}
|
styles.textArea,
|
||||||
/>
|
{
|
||||||
</View>
|
backgroundColor: colors.surface,
|
||||||
|
borderColor: colors.border,
|
||||||
|
color: colors.textPrimary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
value={description}
|
||||||
|
onChangeText={setDescription}
|
||||||
|
placeholder="Optional description"
|
||||||
|
placeholderTextColor={colors.textTertiary}
|
||||||
|
multiline
|
||||||
|
numberOfLines={3}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
|
||||||
{/* Target Value & Unit */}
|
{/* Target Value & Unit */}
|
||||||
<View style={styles.row}>
|
<View style={styles.row}>
|
||||||
<View style={[styles.field, styles.flex1]}>
|
<View style={[styles.field, styles.flex1]}>
|
||||||
<Text style={styles.label}>Target Value</Text>
|
<Text
|
||||||
<TextInput
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
style={styles.input}
|
>
|
||||||
value={targetValue}
|
Target Value
|
||||||
onChangeText={setTargetValue}
|
</Text>
|
||||||
placeholder="e.g., 70"
|
<TextInput
|
||||||
placeholderTextColor="#9ca3af"
|
style={[
|
||||||
keyboardType="numeric"
|
styles.input,
|
||||||
/>
|
{
|
||||||
</View>
|
backgroundColor: colors.surface,
|
||||||
<View style={[styles.field, styles.flex1]}>
|
borderColor: colors.border,
|
||||||
<Text style={styles.label}>Unit</Text>
|
color: colors.textPrimary,
|
||||||
<TextInput
|
},
|
||||||
style={styles.input}
|
]}
|
||||||
value={unit}
|
value={targetValue}
|
||||||
onChangeText={setUnit}
|
onChangeText={setTargetValue}
|
||||||
placeholder="e.g., kg"
|
placeholder="e.g., 70"
|
||||||
placeholderTextColor="#9ca3af"
|
placeholderTextColor={colors.textTertiary}
|
||||||
/>
|
keyboardType="numeric"
|
||||||
</View>
|
/>
|
||||||
</View>
|
|
||||||
|
|
||||||
{/* Current Value */}
|
|
||||||
<View style={styles.field}>
|
|
||||||
<Text style={styles.label}>Current Value</Text>
|
|
||||||
<TextInput
|
|
||||||
style={styles.input}
|
|
||||||
value={currentValue}
|
|
||||||
onChangeText={setCurrentValue}
|
|
||||||
placeholder="Starting value (optional)"
|
|
||||||
placeholderTextColor="#9ca3af"
|
|
||||||
keyboardType="numeric"
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
{/* Target Date */}
|
|
||||||
<View style={styles.field}>
|
|
||||||
<Text style={styles.label}>Target Date</Text>
|
|
||||||
<TouchableOpacity
|
|
||||||
style={styles.dateButton}
|
|
||||||
onPress={() => setShowDatePicker(!showDatePicker)}
|
|
||||||
>
|
|
||||||
<Text style={targetDate ? styles.dateText : styles.datePlaceholder}>
|
|
||||||
{targetDate ? targetDate.toLocaleDateString() : 'Select target date'}
|
|
||||||
</Text>
|
|
||||||
<Ionicons name={showDatePicker ? "chevron-up" : "calendar-outline"} size={20} color="#6b7280" />
|
|
||||||
</TouchableOpacity>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
{showDatePicker && (
|
|
||||||
<View style={Platform.OS === 'ios' ? styles.datePickerContainer : undefined}>
|
|
||||||
<DateTimePicker
|
|
||||||
value={targetDate || new Date()}
|
|
||||||
mode="date"
|
|
||||||
display={Platform.OS === 'ios' ? 'inline' : 'default'}
|
|
||||||
onChange={(event, selectedDate) => {
|
|
||||||
setShowDatePicker(Platform.OS === 'ios');
|
|
||||||
if (selectedDate) {
|
|
||||||
setTargetDate(selectedDate);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
minimumDate={new Date()}
|
|
||||||
themeVariant="light"
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Priority */}
|
|
||||||
<View style={styles.field}>
|
|
||||||
<Text style={styles.label}>Priority</Text>
|
|
||||||
<View style={styles.priorityContainer}>
|
|
||||||
{PRIORITIES.map((p) => (
|
|
||||||
<TouchableOpacity
|
|
||||||
key={p.value}
|
|
||||||
style={[
|
|
||||||
styles.priorityButton,
|
|
||||||
priority === p.value && { backgroundColor: p.color },
|
|
||||||
]}
|
|
||||||
onPress={() => setPriority(p.value)}
|
|
||||||
>
|
|
||||||
<Text
|
|
||||||
style={[
|
|
||||||
styles.priorityButtonText,
|
|
||||||
priority === p.value && styles.priorityButtonTextActive,
|
|
||||||
]}
|
|
||||||
>
|
|
||||||
{p.label}
|
|
||||||
</Text>
|
|
||||||
</TouchableOpacity>
|
|
||||||
))}
|
|
||||||
</View>
|
|
||||||
</View>
|
|
||||||
</ScrollView>
|
|
||||||
|
|
||||||
<View style={styles.footer}>
|
|
||||||
<TouchableOpacity
|
|
||||||
style={[styles.submitButton, submitting && styles.submitButtonDisabled]}
|
|
||||||
onPress={handleSubmit}
|
|
||||||
disabled={submitting}
|
|
||||||
>
|
|
||||||
<Text style={styles.submitButtonText}>
|
|
||||||
{submitting ? 'Creating...' : 'Create Goal'}
|
|
||||||
</Text>
|
|
||||||
</TouchableOpacity>
|
|
||||||
</View>
|
|
||||||
</View>
|
</View>
|
||||||
</Modal>
|
<View style={[styles.field, styles.flex1]}>
|
||||||
);
|
<Text
|
||||||
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
Unit
|
||||||
|
</Text>
|
||||||
|
<TextInput
|
||||||
|
style={[
|
||||||
|
styles.input,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderColor: colors.border,
|
||||||
|
color: colors.textPrimary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
value={unit}
|
||||||
|
onChangeText={setUnit}
|
||||||
|
placeholder="e.g., kg"
|
||||||
|
placeholderTextColor={colors.textTertiary}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
|
||||||
|
{/* Current Value */}
|
||||||
|
<View style={styles.field}>
|
||||||
|
<Text
|
||||||
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
Current Value
|
||||||
|
</Text>
|
||||||
|
<TextInput
|
||||||
|
style={[
|
||||||
|
styles.input,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderColor: colors.border,
|
||||||
|
color: colors.textPrimary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
value={currentValue}
|
||||||
|
onChangeText={setCurrentValue}
|
||||||
|
placeholder="Starting value (optional)"
|
||||||
|
placeholderTextColor={colors.textTertiary}
|
||||||
|
keyboardType="numeric"
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
|
||||||
|
{/* Target Date */}
|
||||||
|
<View style={styles.field}>
|
||||||
|
<Text
|
||||||
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
Target Date
|
||||||
|
</Text>
|
||||||
|
<TouchableOpacity
|
||||||
|
style={[
|
||||||
|
styles.dateButton,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderColor: colors.border,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
onPress={() => setShowDatePicker(!showDatePicker)}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{
|
||||||
|
color: targetDate
|
||||||
|
? colors.textPrimary
|
||||||
|
: colors.textTertiary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{targetDate
|
||||||
|
? targetDate.toLocaleDateString()
|
||||||
|
: "Select target date"}
|
||||||
|
</Text>
|
||||||
|
<Ionicons
|
||||||
|
name={showDatePicker ? "chevron-up" : "calendar-outline"}
|
||||||
|
size={20}
|
||||||
|
color={colors.textSecondary}
|
||||||
|
/>
|
||||||
|
</TouchableOpacity>
|
||||||
|
</View>
|
||||||
|
|
||||||
|
{showDatePicker && (
|
||||||
|
<View
|
||||||
|
style={
|
||||||
|
Platform.OS === "ios"
|
||||||
|
? [
|
||||||
|
styles.datePickerContainer,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderColor: colors.border,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<DateTimePicker
|
||||||
|
value={targetDate || new Date()}
|
||||||
|
mode="date"
|
||||||
|
display={Platform.OS === "ios" ? "inline" : "default"}
|
||||||
|
onChange={(event, selectedDate) => {
|
||||||
|
setShowDatePicker(Platform.OS === "ios");
|
||||||
|
if (selectedDate) {
|
||||||
|
setTargetDate(selectedDate);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
minimumDate={new Date()}
|
||||||
|
themeVariant="light"
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Priority */}
|
||||||
|
<View style={styles.field}>
|
||||||
|
<Text
|
||||||
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
Priority
|
||||||
|
</Text>
|
||||||
|
<View style={styles.priorityContainer}>
|
||||||
|
{PRIORITIES.map((p) => (
|
||||||
|
<TouchableOpacity
|
||||||
|
key={p.value}
|
||||||
|
style={[
|
||||||
|
styles.priorityButton,
|
||||||
|
{
|
||||||
|
borderColor: colors.border,
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
},
|
||||||
|
priority === p.value && {
|
||||||
|
backgroundColor: getPriorityColor(p.value),
|
||||||
|
borderColor: getPriorityColor(p.value),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
onPress={() => setPriority(p.value)}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.body,
|
||||||
|
{ color: colors.textSecondary },
|
||||||
|
priority === p.value && {
|
||||||
|
color: colors.white,
|
||||||
|
fontWeight: "600",
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{p.label}
|
||||||
|
</Text>
|
||||||
|
</TouchableOpacity>
|
||||||
|
))}
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
</ScrollView>
|
||||||
|
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.footer,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
borderTopColor: colors.border,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<MinimalButton
|
||||||
|
variant="primary"
|
||||||
|
title={submitting ? "Creating..." : "Create Goal"}
|
||||||
|
onPress={handleSubmit}
|
||||||
|
disabled={submitting}
|
||||||
|
loading={submitting}
|
||||||
|
style={styles.submitButton}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
backgroundColor: '#f9fafb',
|
},
|
||||||
},
|
header: {
|
||||||
header: {
|
flexDirection: "row",
|
||||||
flexDirection: 'row',
|
justifyContent: "space-between",
|
||||||
justifyContent: 'space-between',
|
alignItems: "center",
|
||||||
alignItems: 'center',
|
padding: 20,
|
||||||
padding: 20,
|
paddingTop: Platform.OS === "ios" ? 60 : 20,
|
||||||
paddingTop: Platform.OS === 'ios' ? 60 : 20,
|
borderBottomWidth: 1,
|
||||||
backgroundColor: '#fff',
|
},
|
||||||
borderBottomWidth: 1,
|
closeButton: {
|
||||||
borderBottomColor: '#e5e7eb',
|
padding: 4,
|
||||||
},
|
},
|
||||||
headerTitle: {
|
content: {
|
||||||
fontSize: 20,
|
flex: 1,
|
||||||
fontWeight: '600',
|
padding: 20,
|
||||||
color: '#111827',
|
},
|
||||||
},
|
field: {
|
||||||
closeButton: {
|
marginBottom: 20,
|
||||||
padding: 4,
|
},
|
||||||
},
|
input: {
|
||||||
content: {
|
borderWidth: 1,
|
||||||
flex: 1,
|
borderRadius: 8,
|
||||||
padding: 20,
|
padding: 12,
|
||||||
},
|
fontSize: 16,
|
||||||
field: {
|
marginTop: 8,
|
||||||
marginBottom: 20,
|
},
|
||||||
},
|
textArea: {
|
||||||
label: {
|
height: 80,
|
||||||
fontSize: 14,
|
textAlignVertical: "top",
|
||||||
fontWeight: '600',
|
},
|
||||||
color: '#374151',
|
row: {
|
||||||
marginBottom: 8,
|
flexDirection: "row",
|
||||||
},
|
gap: 12,
|
||||||
input: {
|
},
|
||||||
backgroundColor: '#fff',
|
flex1: {
|
||||||
borderWidth: 1,
|
flex: 1,
|
||||||
borderColor: '#d1d5db',
|
},
|
||||||
borderRadius: 8,
|
typeScroll: {
|
||||||
padding: 12,
|
flexGrow: 0,
|
||||||
fontSize: 16,
|
marginTop: 8,
|
||||||
color: '#111827',
|
},
|
||||||
},
|
typeButton: {
|
||||||
textArea: {
|
paddingHorizontal: 16,
|
||||||
height: 80,
|
paddingVertical: 8,
|
||||||
textAlignVertical: 'top',
|
borderRadius: 8,
|
||||||
},
|
borderWidth: 1,
|
||||||
row: {
|
marginRight: 8,
|
||||||
flexDirection: 'row',
|
},
|
||||||
gap: 12,
|
dateButton: {
|
||||||
},
|
flexDirection: "row",
|
||||||
flex1: {
|
justifyContent: "space-between",
|
||||||
flex: 1,
|
alignItems: "center",
|
||||||
},
|
borderWidth: 1,
|
||||||
typeScroll: {
|
borderRadius: 8,
|
||||||
flexGrow: 0,
|
padding: 12,
|
||||||
},
|
marginTop: 8,
|
||||||
typeButton: {
|
},
|
||||||
paddingHorizontal: 16,
|
priorityContainer: {
|
||||||
paddingVertical: 8,
|
flexDirection: "row",
|
||||||
borderRadius: 8,
|
gap: 12,
|
||||||
borderWidth: 1,
|
marginTop: 8,
|
||||||
borderColor: '#d1d5db',
|
},
|
||||||
backgroundColor: '#fff',
|
priorityButton: {
|
||||||
marginRight: 8,
|
flex: 1,
|
||||||
},
|
paddingVertical: 12,
|
||||||
typeButtonActive: {
|
borderRadius: 8,
|
||||||
backgroundColor: '#2563eb',
|
borderWidth: 1,
|
||||||
borderColor: '#2563eb',
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
typeButtonText: {
|
footer: {
|
||||||
fontSize: 14,
|
padding: 20,
|
||||||
color: '#374151',
|
paddingBottom: Platform.OS === "ios" ? 40 : 20,
|
||||||
},
|
borderTopWidth: 1,
|
||||||
typeButtonTextActive: {
|
},
|
||||||
color: '#fff',
|
submitButton: {
|
||||||
fontWeight: '600',
|
width: "100%",
|
||||||
},
|
},
|
||||||
dateButton: {
|
datePickerContainer: {
|
||||||
flexDirection: 'row',
|
borderRadius: 8,
|
||||||
justifyContent: 'space-between',
|
borderWidth: 1,
|
||||||
alignItems: 'center',
|
marginTop: 8,
|
||||||
backgroundColor: '#fff',
|
overflow: "hidden",
|
||||||
borderWidth: 1,
|
},
|
||||||
borderColor: '#d1d5db',
|
|
||||||
borderRadius: 8,
|
|
||||||
padding: 12,
|
|
||||||
},
|
|
||||||
dateText: {
|
|
||||||
fontSize: 16,
|
|
||||||
color: '#111827',
|
|
||||||
},
|
|
||||||
datePlaceholder: {
|
|
||||||
fontSize: 16,
|
|
||||||
color: '#9ca3af',
|
|
||||||
},
|
|
||||||
priorityContainer: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
gap: 12,
|
|
||||||
},
|
|
||||||
priorityButton: {
|
|
||||||
flex: 1,
|
|
||||||
paddingVertical: 12,
|
|
||||||
borderRadius: 8,
|
|
||||||
borderWidth: 1,
|
|
||||||
borderColor: '#d1d5db',
|
|
||||||
backgroundColor: '#fff',
|
|
||||||
alignItems: 'center',
|
|
||||||
},
|
|
||||||
priorityButtonText: {
|
|
||||||
fontSize: 14,
|
|
||||||
color: '#374151',
|
|
||||||
fontWeight: '500',
|
|
||||||
},
|
|
||||||
priorityButtonTextActive: {
|
|
||||||
color: '#fff',
|
|
||||||
fontWeight: '600',
|
|
||||||
},
|
|
||||||
footer: {
|
|
||||||
padding: 20,
|
|
||||||
paddingBottom: Platform.OS === 'ios' ? 40 : 20,
|
|
||||||
backgroundColor: '#fff',
|
|
||||||
borderTopWidth: 1,
|
|
||||||
borderTopColor: '#e5e7eb',
|
|
||||||
},
|
|
||||||
submitButton: {
|
|
||||||
backgroundColor: '#2563eb',
|
|
||||||
borderRadius: 8,
|
|
||||||
padding: 16,
|
|
||||||
alignItems: 'center',
|
|
||||||
},
|
|
||||||
submitButtonDisabled: {
|
|
||||||
opacity: 0.6,
|
|
||||||
},
|
|
||||||
submitButtonText: {
|
|
||||||
fontSize: 16,
|
|
||||||
fontWeight: '600',
|
|
||||||
color: '#fff',
|
|
||||||
},
|
|
||||||
datePickerContainer: {
|
|
||||||
backgroundColor: '#fff',
|
|
||||||
borderRadius: 8,
|
|
||||||
borderWidth: 1,
|
|
||||||
borderColor: '#d1d5db',
|
|
||||||
marginTop: 8,
|
|
||||||
overflow: 'hidden',
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|||||||
@ -1,293 +1,333 @@
|
|||||||
import React from 'react';
|
import React, { useEffect, useRef } from "react";
|
||||||
import { View, Text, StyleSheet, TouchableOpacity, Alert } from 'react-native';
|
import {
|
||||||
import { Ionicons } from '@expo/vector-icons';
|
View,
|
||||||
import { LinearGradient } from 'expo-linear-gradient';
|
Text,
|
||||||
import type { FitnessGoal } from '../services/fitnessGoals';
|
StyleSheet,
|
||||||
import { theme } from '../styles/theme';
|
TouchableOpacity,
|
||||||
|
Alert,
|
||||||
|
Animated,
|
||||||
|
} from "react-native";
|
||||||
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
|
import type { FitnessGoal } from "../services/fitnessGoals";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
import { MinimalCard } from "./MinimalCard";
|
||||||
|
import { Badge } from "./Badge";
|
||||||
|
import { ProgressBar } from "./ProgressBar";
|
||||||
|
import { IconContainer } from "./IconContainer";
|
||||||
|
|
||||||
interface GoalProgressCardProps {
|
interface GoalProgressCardProps {
|
||||||
goal: FitnessGoal;
|
goal: FitnessGoal;
|
||||||
onPress?: () => void;
|
onPress?: () => void;
|
||||||
onComplete?: () => void;
|
onComplete?: () => void;
|
||||||
onDelete?: () => void;
|
onDelete?: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function GoalProgressCard({ goal, onPress, onComplete, onDelete }: GoalProgressCardProps) {
|
export function GoalProgressCard({
|
||||||
const isCompleted = goal.status === 'completed';
|
goal,
|
||||||
const progress = goal.progress || 0;
|
onPress,
|
||||||
|
onComplete,
|
||||||
|
onDelete,
|
||||||
|
}: GoalProgressCardProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
|
const isCompleted = goal.status === "completed";
|
||||||
|
const progress = (goal.progress || 0) / 100; // Convert to 0-1 scale
|
||||||
|
const scaleAnim = useRef(new Animated.Value(1)).current;
|
||||||
|
|
||||||
// Calculate days remaining
|
// Celebration animation when goal is completed
|
||||||
const daysRemaining = goal.targetDate
|
useEffect(() => {
|
||||||
? Math.ceil((new Date(goal.targetDate).getTime() - Date.now()) / (1000 * 60 * 60 * 24))
|
if (isCompleted) {
|
||||||
: null;
|
Animated.sequence([
|
||||||
|
Animated.spring(scaleAnim, {
|
||||||
|
toValue: 1.05,
|
||||||
|
friction: 3,
|
||||||
|
tension: 40,
|
||||||
|
useNativeDriver: true,
|
||||||
|
}),
|
||||||
|
Animated.spring(scaleAnim, {
|
||||||
|
toValue: 1,
|
||||||
|
friction: 3,
|
||||||
|
tension: 40,
|
||||||
|
useNativeDriver: true,
|
||||||
|
}),
|
||||||
|
]).start();
|
||||||
|
}
|
||||||
|
}, [isCompleted]);
|
||||||
|
|
||||||
const getGoalTypeIcon = (type: string) => {
|
const handleComplete = () => {
|
||||||
switch (type) {
|
if (onComplete) {
|
||||||
case 'weight_target': return 'scale-outline';
|
// Trigger celebration animation
|
||||||
case 'strength_milestone': return 'barbell-outline';
|
Animated.sequence([
|
||||||
case 'endurance_target': return 'bicycle-outline';
|
Animated.spring(scaleAnim, {
|
||||||
case 'flexibility_goal': return 'body-outline';
|
toValue: 1.1,
|
||||||
case 'habit_building': return 'calendar-outline';
|
friction: 3,
|
||||||
default: return 'flag-outline';
|
tension: 40,
|
||||||
}
|
useNativeDriver: true,
|
||||||
};
|
}),
|
||||||
|
Animated.spring(scaleAnim, {
|
||||||
|
toValue: 1,
|
||||||
|
friction: 3,
|
||||||
|
tension: 40,
|
||||||
|
useNativeDriver: true,
|
||||||
|
}),
|
||||||
|
]).start(() => {
|
||||||
|
onComplete();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const getPriorityGradient = (priority: string): readonly [string, string] => {
|
// Calculate days remaining
|
||||||
switch (priority) {
|
const daysRemaining = goal.targetDate
|
||||||
case 'high': return theme.gradients.danger;
|
? Math.ceil(
|
||||||
case 'medium': return theme.gradients.warning;
|
(new Date(goal.targetDate).getTime() - Date.now()) /
|
||||||
case 'low': return theme.gradients.success;
|
(1000 * 60 * 60 * 24),
|
||||||
default: return theme.gradients.primary;
|
)
|
||||||
}
|
: null;
|
||||||
};
|
|
||||||
|
|
||||||
const handleDelete = () => {
|
const getGoalTypeIcon = (type: string) => {
|
||||||
Alert.alert(
|
switch (type) {
|
||||||
'Delete Goal',
|
case "weight_target":
|
||||||
'Are you sure you want to delete this goal?',
|
return "scale-outline";
|
||||||
[
|
case "strength_milestone":
|
||||||
{ text: 'Cancel', style: 'cancel' },
|
return "barbell-outline";
|
||||||
{ text: 'Delete', style: 'destructive', onPress: onDelete },
|
case "endurance_target":
|
||||||
]
|
return "bicycle-outline";
|
||||||
);
|
case "flexibility_goal":
|
||||||
};
|
return "body-outline";
|
||||||
|
case "habit_building":
|
||||||
|
return "calendar-outline";
|
||||||
|
default:
|
||||||
|
return "flag-outline";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
const getPriorityColor = (priority: string) => {
|
||||||
<TouchableOpacity
|
switch (priority) {
|
||||||
onPress={onPress}
|
case "high":
|
||||||
activeOpacity={0.7}
|
return colors.danger;
|
||||||
|
case "medium":
|
||||||
|
return colors.warning;
|
||||||
|
case "low":
|
||||||
|
return colors.success;
|
||||||
|
default:
|
||||||
|
return colors.primary;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDelete = () => {
|
||||||
|
Alert.alert("Delete Goal", "Are you sure you want to delete this goal?", [
|
||||||
|
{ text: "Cancel", style: "cancel" },
|
||||||
|
{ text: "Delete", style: "destructive", onPress: onDelete },
|
||||||
|
]);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Animated.View style={{ transform: [{ scale: scaleAnim }] }}>
|
||||||
|
<TouchableOpacity onPress={onPress} activeOpacity={0.85}>
|
||||||
|
<MinimalCard
|
||||||
|
variant="elevated"
|
||||||
|
style={[
|
||||||
|
styles.card,
|
||||||
|
isCompleted && {
|
||||||
|
backgroundColor: colors.surfaceElevated,
|
||||||
|
opacity: 0.8,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
padding={20}
|
||||||
>
|
>
|
||||||
<LinearGradient
|
{/* Header */}
|
||||||
colors={isCompleted
|
<View style={styles.header}>
|
||||||
? ['rgba(16, 185, 129, 0.05)', 'rgba(5, 150, 105, 0.02)'] as const
|
<View style={styles.titleRow}>
|
||||||
: ['rgba(255, 255, 255, 1)', 'rgba(249, 250, 251, 1)'] as const
|
<IconContainer
|
||||||
|
variant="colored"
|
||||||
|
backgroundColor={
|
||||||
|
isCompleted ? colors.success : getPriorityColor(goal.priority)
|
||||||
}
|
}
|
||||||
style={[
|
>
|
||||||
styles.card,
|
<Ionicons
|
||||||
theme.shadows.medium,
|
name={getGoalTypeIcon(goal.goalType) as any}
|
||||||
isCompleted && styles.cardCompleted
|
size={20}
|
||||||
]}
|
color={colors.white}
|
||||||
>
|
|
||||||
{/* Priority Accent Bar */}
|
|
||||||
<LinearGradient
|
|
||||||
colors={getPriorityGradient(goal.priority)}
|
|
||||||
start={{ x: 0, y: 0 }}
|
|
||||||
end={{ x: 0, y: 1 }}
|
|
||||||
style={styles.priorityAccent}
|
|
||||||
/>
|
/>
|
||||||
|
</IconContainer>
|
||||||
|
|
||||||
<View style={styles.header}>
|
<View style={styles.titleContainer}>
|
||||||
<View style={styles.titleRow}>
|
<Text
|
||||||
<LinearGradient
|
style={[
|
||||||
colors={isCompleted ? theme.gradients.success : getPriorityGradient(goal.priority)}
|
typography.h3,
|
||||||
style={styles.iconContainer}
|
{ color: colors.textPrimary },
|
||||||
>
|
isCompleted && {
|
||||||
<Ionicons
|
color: colors.textSecondary,
|
||||||
name={getGoalTypeIcon(goal.goalType) as any}
|
textDecorationLine: "line-through",
|
||||||
size={20}
|
},
|
||||||
color="#fff"
|
]}
|
||||||
/>
|
>
|
||||||
</LinearGradient>
|
{goal.title}
|
||||||
<View style={styles.titleContainer}>
|
</Text>
|
||||||
<Text style={[styles.title, isCompleted && styles.titleCompleted]}>
|
{goal.description && (
|
||||||
{goal.title}
|
<Text
|
||||||
</Text>
|
style={[
|
||||||
{goal.description && (
|
typography.caption,
|
||||||
<Text style={styles.description} numberOfLines={2}>
|
{ color: colors.textTertiary, marginTop: 2 },
|
||||||
{goal.description}
|
]}
|
||||||
</Text>
|
numberOfLines={2}
|
||||||
)}
|
>
|
||||||
</View>
|
{goal.description}
|
||||||
</View>
|
</Text>
|
||||||
|
|
||||||
<View style={styles.actions}>
|
|
||||||
{!isCompleted && onComplete && (
|
|
||||||
<TouchableOpacity onPress={onComplete} style={styles.actionButton}>
|
|
||||||
<Ionicons name="checkmark-circle-outline" size={24} color={theme.colors.success} />
|
|
||||||
</TouchableOpacity>
|
|
||||||
)}
|
|
||||||
{onDelete && (
|
|
||||||
<TouchableOpacity onPress={handleDelete} style={styles.actionButton}>
|
|
||||||
<Ionicons name="trash-outline" size={22} color={theme.colors.danger} />
|
|
||||||
</TouchableOpacity>
|
|
||||||
)}
|
|
||||||
</View>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
{goal.targetValue && (
|
|
||||||
<View style={styles.progressSection}>
|
|
||||||
<View style={styles.progressInfo}>
|
|
||||||
<Text style={styles.progressText}>
|
|
||||||
{goal.currentValue || 0} / {goal.targetValue} {goal.unit || ''}
|
|
||||||
</Text>
|
|
||||||
<Text style={[styles.progressPercentage, isCompleted && { color: theme.colors.success }]}>
|
|
||||||
{progress.toFixed(0)}%
|
|
||||||
</Text>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
<View style={styles.progressBarContainer}>
|
|
||||||
<LinearGradient
|
|
||||||
colors={isCompleted ? theme.gradients.success : getPriorityGradient(goal.priority)}
|
|
||||||
start={{ x: 0, y: 0 }}
|
|
||||||
end={{ x: 1, y: 0 }}
|
|
||||||
style={[
|
|
||||||
styles.progressBar,
|
|
||||||
{ width: `${Math.min(progress, 100)}%` }
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
</View>
|
|
||||||
)}
|
)}
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
|
||||||
<View style={styles.footer}>
|
{/* Action Buttons */}
|
||||||
<LinearGradient
|
<View style={styles.actions}>
|
||||||
colors={getPriorityGradient(goal.priority)}
|
{!isCompleted && onComplete && (
|
||||||
style={styles.priorityBadge}
|
<TouchableOpacity
|
||||||
>
|
onPress={handleComplete}
|
||||||
<Text style={styles.priorityText}>{goal.priority.toUpperCase()}</Text>
|
style={styles.actionButton}
|
||||||
</LinearGradient>
|
>
|
||||||
|
<Ionicons
|
||||||
|
name="checkmark-circle-outline"
|
||||||
|
size={24}
|
||||||
|
color={colors.success}
|
||||||
|
/>
|
||||||
|
</TouchableOpacity>
|
||||||
|
)}
|
||||||
|
{onDelete && (
|
||||||
|
<TouchableOpacity
|
||||||
|
onPress={handleDelete}
|
||||||
|
style={styles.actionButton}
|
||||||
|
>
|
||||||
|
<Ionicons
|
||||||
|
name="trash-outline"
|
||||||
|
size={22}
|
||||||
|
color={colors.danger}
|
||||||
|
/>
|
||||||
|
</TouchableOpacity>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
|
||||||
{daysRemaining !== null && !isCompleted && (
|
{/* Progress Section */}
|
||||||
<Text style={[styles.daysRemaining, daysRemaining < 0 && styles.overdue]}>
|
{goal.targetValue && (
|
||||||
{daysRemaining < 0
|
<View style={styles.progressSection}>
|
||||||
? `${Math.abs(daysRemaining)} days overdue`
|
<View style={styles.progressInfo}>
|
||||||
: `${daysRemaining} days remaining`
|
<Text
|
||||||
}
|
style={[typography.caption, { color: colors.textSecondary }]}
|
||||||
</Text>
|
>
|
||||||
)}
|
{goal.currentValue || 0} / {goal.targetValue}{" "}
|
||||||
|
{goal.unit || ""}
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.bodyEmphasis,
|
||||||
|
{
|
||||||
|
color: isCompleted ? colors.success : colors.primary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{(progress * 100).toFixed(0)}%
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
|
||||||
{isCompleted && goal.completedDate && (
|
<ProgressBar
|
||||||
<Text style={styles.completedDate}>
|
progress={progress}
|
||||||
Completed {new Date(goal.completedDate).toLocaleDateString()}
|
color={
|
||||||
</Text>
|
isCompleted ? colors.success : getPriorityColor(goal.priority)
|
||||||
)}
|
}
|
||||||
</View>
|
/>
|
||||||
</LinearGradient>
|
</View>
|
||||||
</TouchableOpacity>
|
)}
|
||||||
);
|
|
||||||
|
{/* Footer */}
|
||||||
|
<View style={styles.footer}>
|
||||||
|
{isCompleted ? (
|
||||||
|
<Badge variant="success" label="COMPLETED" />
|
||||||
|
) : (
|
||||||
|
<View>
|
||||||
|
{goal.priority === "high" && (
|
||||||
|
<Badge variant="danger" label={goal.priority.toUpperCase()} />
|
||||||
|
)}
|
||||||
|
{goal.priority === "medium" && (
|
||||||
|
<Badge
|
||||||
|
variant="warning"
|
||||||
|
label={goal.priority.toUpperCase()}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{goal.priority === "low" && (
|
||||||
|
<Badge
|
||||||
|
variant="success"
|
||||||
|
label={goal.priority.toUpperCase()}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{daysRemaining !== null && !isCompleted && (
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{
|
||||||
|
color:
|
||||||
|
daysRemaining < 0 ? colors.danger : colors.textTertiary,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{daysRemaining < 0
|
||||||
|
? `${Math.abs(daysRemaining)} days overdue`
|
||||||
|
: `${daysRemaining} days remaining`}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{isCompleted && goal.completedDate && (
|
||||||
|
<Text style={[typography.caption, { color: colors.success }]}>
|
||||||
|
Completed {new Date(goal.completedDate).toLocaleDateString()}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
</MinimalCard>
|
||||||
|
</TouchableOpacity>
|
||||||
|
</Animated.View>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
card: {
|
card: {
|
||||||
borderRadius: theme.borderRadius.xl,
|
marginBottom: 16,
|
||||||
padding: 16,
|
borderRadius: 20,
|
||||||
marginBottom: 12,
|
},
|
||||||
borderWidth: 1,
|
header: {
|
||||||
borderColor: 'rgba(59, 130, 246, 0.1)',
|
flexDirection: "row",
|
||||||
overflow: 'hidden',
|
justifyContent: "space-between",
|
||||||
},
|
alignItems: "flex-start",
|
||||||
cardCompleted: {
|
marginBottom: 16,
|
||||||
borderColor: 'rgba(16, 185, 129, 0.2)',
|
},
|
||||||
},
|
titleRow: {
|
||||||
priorityAccent: {
|
flexDirection: "row",
|
||||||
position: 'absolute',
|
alignItems: "flex-start",
|
||||||
left: 0,
|
flex: 1,
|
||||||
top: 0,
|
},
|
||||||
bottom: 0,
|
titleContainer: {
|
||||||
width: 4,
|
flex: 1,
|
||||||
},
|
marginLeft: 14,
|
||||||
header: {
|
},
|
||||||
flexDirection: 'row',
|
actions: {
|
||||||
justifyContent: 'space-between',
|
flexDirection: "row",
|
||||||
alignItems: 'flex-start',
|
gap: 12,
|
||||||
marginBottom: 12,
|
},
|
||||||
marginLeft: 8,
|
actionButton: {
|
||||||
},
|
padding: 6,
|
||||||
titleRow: {
|
},
|
||||||
flexDirection: 'row',
|
progressSection: {
|
||||||
alignItems: 'flex-start',
|
marginBottom: 16,
|
||||||
flex: 1,
|
},
|
||||||
},
|
progressInfo: {
|
||||||
iconContainer: {
|
flexDirection: "row",
|
||||||
width: 40,
|
justifyContent: "space-between",
|
||||||
height: 40,
|
alignItems: "center",
|
||||||
borderRadius: 20,
|
marginBottom: 10,
|
||||||
justifyContent: 'center',
|
},
|
||||||
alignItems: 'center',
|
footer: {
|
||||||
marginRight: 12,
|
flexDirection: "row",
|
||||||
},
|
alignItems: "center",
|
||||||
titleContainer: {
|
justifyContent: "space-between",
|
||||||
flex: 1,
|
},
|
||||||
},
|
|
||||||
title: {
|
|
||||||
fontSize: theme.typography.fontSize.lg,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
color: theme.colors.gray900,
|
|
||||||
marginBottom: 4,
|
|
||||||
},
|
|
||||||
titleCompleted: {
|
|
||||||
color: theme.colors.gray600,
|
|
||||||
textDecorationLine: 'line-through',
|
|
||||||
},
|
|
||||||
description: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
color: theme.colors.gray600,
|
|
||||||
lineHeight: 18,
|
|
||||||
},
|
|
||||||
actions: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
gap: 8,
|
|
||||||
},
|
|
||||||
actionButton: {
|
|
||||||
padding: 4,
|
|
||||||
},
|
|
||||||
progressSection: {
|
|
||||||
marginBottom: 12,
|
|
||||||
marginLeft: 8,
|
|
||||||
},
|
|
||||||
progressInfo: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
marginBottom: 8,
|
|
||||||
},
|
|
||||||
progressText: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
fontWeight: theme.typography.fontWeight.medium,
|
|
||||||
color: theme.colors.gray700,
|
|
||||||
},
|
|
||||||
progressPercentage: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
color: theme.colors.primary,
|
|
||||||
},
|
|
||||||
progressBarContainer: {
|
|
||||||
height: 8,
|
|
||||||
backgroundColor: theme.colors.gray200,
|
|
||||||
borderRadius: 4,
|
|
||||||
overflow: 'hidden',
|
|
||||||
},
|
|
||||||
progressBar: {
|
|
||||||
height: '100%',
|
|
||||||
borderRadius: 4,
|
|
||||||
},
|
|
||||||
footer: {
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
marginLeft: 8,
|
|
||||||
},
|
|
||||||
priorityBadge: {
|
|
||||||
paddingHorizontal: 10,
|
|
||||||
paddingVertical: 5,
|
|
||||||
borderRadius: theme.borderRadius.md,
|
|
||||||
},
|
|
||||||
priorityText: {
|
|
||||||
fontSize: theme.typography.fontSize.xs,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
color: theme.colors.white,
|
|
||||||
},
|
|
||||||
daysRemaining: {
|
|
||||||
fontSize: theme.typography.fontSize.xs,
|
|
||||||
color: theme.colors.gray600,
|
|
||||||
fontWeight: theme.typography.fontWeight.medium,
|
|
||||||
},
|
|
||||||
overdue: {
|
|
||||||
color: theme.colors.danger,
|
|
||||||
fontWeight: theme.typography.fontWeight.semibold,
|
|
||||||
},
|
|
||||||
completedDate: {
|
|
||||||
fontSize: theme.typography.fontSize.xs,
|
|
||||||
color: theme.colors.success,
|
|
||||||
fontWeight: theme.typography.fontWeight.medium,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|||||||
@ -1,7 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { View, Text, StyleSheet, Dimensions } from "react-native";
|
import { View, Text, StyleSheet, Dimensions } from "react-native";
|
||||||
import { PieChart } from "react-native-chart-kit";
|
import { PieChart } from "react-native-chart-kit";
|
||||||
import { theme } from "../styles/theme";
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
interface GoalTypeData {
|
interface GoalTypeData {
|
||||||
goalType: string;
|
goalType: string;
|
||||||
@ -17,37 +17,40 @@ export function GoalTypeBreakdownChart({
|
|||||||
data,
|
data,
|
||||||
title = "Goals by Type",
|
title = "Goals by Type",
|
||||||
}: GoalTypeBreakdownChartProps) {
|
}: GoalTypeBreakdownChartProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
const screenWidth = Dimensions.get("window").width;
|
const screenWidth = Dimensions.get("window").width;
|
||||||
|
|
||||||
// Color palette for different goal types
|
const chartColors = [
|
||||||
const colors = [
|
colors.primary,
|
||||||
"#3b82f6", // Blue
|
colors.success,
|
||||||
"#10b981", // Green
|
colors.warning,
|
||||||
"#f59e0b", // Orange
|
colors.accent,
|
||||||
"#8b5cf6", // Purple
|
colors.secondary,
|
||||||
"#ec4899", // Pink
|
colors.info,
|
||||||
"#06b6d4", // Cyan
|
|
||||||
];
|
];
|
||||||
|
|
||||||
// Prepare chart data
|
|
||||||
const chartData = data.map((item, index) => ({
|
const chartData = data.map((item, index) => ({
|
||||||
name: item.goalType,
|
name: item.goalType
|
||||||
|
.replace(/_/g, " ")
|
||||||
|
.replace(/\b\w/g, (l) => l.toUpperCase()),
|
||||||
count: item.count,
|
count: item.count,
|
||||||
color: colors[index % colors.length],
|
color: chartColors[index % chartColors.length],
|
||||||
legendFontColor: theme.colors.gray600,
|
legendFontColor: colors.textSecondary,
|
||||||
legendFontSize: 12,
|
legendFontSize: 12,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const chartConfig = {
|
const totalGoals = data.reduce((sum, item) => sum + item.count, 0);
|
||||||
color: (opacity = 1) => `rgba(59, 130, 246, ${opacity})`,
|
|
||||||
};
|
|
||||||
|
|
||||||
if (data.length === 0) {
|
if (data.length === 0) {
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
<Text style={styles.title}>{title}</Text>
|
<Text style={[typography.h4, { color: colors.textPrimary }]}>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
<View style={styles.emptyState}>
|
<View style={styles.emptyState}>
|
||||||
<Text style={styles.emptyText}>No goals yet</Text>
|
<Text style={[typography.body, { color: colors.textTertiary }]}>
|
||||||
|
No goals yet
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
@ -55,46 +58,123 @@ export function GoalTypeBreakdownChart({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
<Text style={styles.title}>{title}</Text>
|
<Text
|
||||||
|
style={[typography.h4, { color: colors.textPrimary, marginBottom: 16 }]}
|
||||||
|
>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
{/* Summary Stats */}
|
||||||
|
<View style={styles.summaryRow}>
|
||||||
|
<View style={styles.summaryItem}>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.statLarge,
|
||||||
|
{ color: colors.primary, fontSize: 32 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{totalGoals}
|
||||||
|
</Text>
|
||||||
|
<Text style={[typography.caption, { color: colors.textTertiary }]}>
|
||||||
|
Total Goals
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
<View style={styles.summaryItem}>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.statLarge,
|
||||||
|
{ color: colors.success, fontSize: 32 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{data.length}
|
||||||
|
</Text>
|
||||||
|
<Text style={[typography.caption, { color: colors.textTertiary }]}>
|
||||||
|
Types
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
|
||||||
<View style={styles.chartContainer}>
|
<View style={styles.chartContainer}>
|
||||||
<PieChart
|
<PieChart
|
||||||
data={chartData}
|
data={chartData}
|
||||||
width={screenWidth - 60}
|
width={screenWidth - 80}
|
||||||
height={200}
|
height={160}
|
||||||
chartConfig={chartConfig}
|
chartConfig={{
|
||||||
|
color: (opacity = 1) => `rgba(0, 0, 0, ${opacity})`,
|
||||||
|
}}
|
||||||
accessor="count"
|
accessor="count"
|
||||||
backgroundColor="transparent"
|
backgroundColor="transparent"
|
||||||
paddingLeft="15"
|
paddingLeft="15"
|
||||||
absolute
|
absolute
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
|
{/* Legend */}
|
||||||
|
<View style={styles.legendGrid}>
|
||||||
|
{data.map((item, index) => (
|
||||||
|
<View key={item.goalType} style={styles.legendItem}>
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.legendDot,
|
||||||
|
{ backgroundColor: chartColors[index % chartColors.length] },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textSecondary, flex: 1 },
|
||||||
|
]}
|
||||||
|
numberOfLines={1}
|
||||||
|
>
|
||||||
|
{item.goalType
|
||||||
|
.replace(/_/g, " ")
|
||||||
|
.replace(/\b\w/g, (l) => l.toUpperCase())}
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[typography.bodyEmphasis, { color: colors.textPrimary }]}
|
||||||
|
>
|
||||||
|
{item.count}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
))}
|
||||||
|
</View>
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
backgroundColor: theme.colors.white,
|
borderRadius: 16,
|
||||||
borderRadius: theme.borderRadius.xl,
|
padding: 4,
|
||||||
padding: 16,
|
|
||||||
marginBottom: 16,
|
|
||||||
...theme.shadows.medium,
|
|
||||||
},
|
},
|
||||||
title: {
|
summaryRow: {
|
||||||
fontSize: theme.typography.fontSize.lg,
|
flexDirection: "row",
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
justifyContent: "space-around",
|
||||||
color: theme.colors.gray700,
|
marginBottom: 16,
|
||||||
marginBottom: 12,
|
paddingVertical: 12,
|
||||||
|
},
|
||||||
|
summaryItem: {
|
||||||
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
chartContainer: {
|
chartContainer: {
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
|
marginBottom: 16,
|
||||||
|
},
|
||||||
|
legendGrid: {
|
||||||
|
gap: 10,
|
||||||
|
},
|
||||||
|
legendItem: {
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "center",
|
||||||
|
gap: 10,
|
||||||
|
},
|
||||||
|
legendDot: {
|
||||||
|
width: 12,
|
||||||
|
height: 12,
|
||||||
|
borderRadius: 6,
|
||||||
},
|
},
|
||||||
emptyState: {
|
emptyState: {
|
||||||
paddingVertical: 40,
|
paddingVertical: 32,
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
},
|
},
|
||||||
emptyText: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
color: theme.colors.gray400,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|||||||
77
apps/mobile/src/components/IconContainer.tsx
Normal file
77
apps/mobile/src/components/IconContainer.tsx
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { View, StyleSheet, ViewStyle, StyleProp } from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
|
type IconContainerVariant = "plain" | "subtle" | "colored";
|
||||||
|
type IconContainerSize = "sm" | "md" | "lg" | "xl";
|
||||||
|
|
||||||
|
interface IconContainerProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
variant?: IconContainerVariant;
|
||||||
|
size?: IconContainerSize;
|
||||||
|
backgroundColor?: string;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function IconContainer({
|
||||||
|
children,
|
||||||
|
variant = "subtle",
|
||||||
|
size = "md",
|
||||||
|
backgroundColor,
|
||||||
|
style,
|
||||||
|
}: IconContainerProps) {
|
||||||
|
const { colors } = useTheme();
|
||||||
|
|
||||||
|
const sizeStyles: Record<IconContainerSize, ViewStyle> = {
|
||||||
|
sm: {
|
||||||
|
width: 36,
|
||||||
|
height: 36,
|
||||||
|
borderRadius: 10,
|
||||||
|
},
|
||||||
|
md: {
|
||||||
|
width: 48,
|
||||||
|
height: 48,
|
||||||
|
borderRadius: 14,
|
||||||
|
},
|
||||||
|
lg: {
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
borderRadius: 16,
|
||||||
|
},
|
||||||
|
xl: {
|
||||||
|
width: 64,
|
||||||
|
height: 64,
|
||||||
|
borderRadius: 18,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const getBackgroundColor = (): string | undefined => {
|
||||||
|
if (variant === "colored" && backgroundColor) {
|
||||||
|
return backgroundColor;
|
||||||
|
}
|
||||||
|
if (variant === "subtle") {
|
||||||
|
return colors.surfaceElevated;
|
||||||
|
}
|
||||||
|
return "transparent";
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.container,
|
||||||
|
sizeStyles[size],
|
||||||
|
{ backgroundColor: getBackgroundColor() },
|
||||||
|
style,
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
container: {
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
},
|
||||||
|
});
|
||||||
95
apps/mobile/src/components/ListItem.tsx
Normal file
95
apps/mobile/src/components/ListItem.tsx
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
import React from "react";
|
||||||
|
import {
|
||||||
|
View,
|
||||||
|
Text,
|
||||||
|
TouchableOpacity,
|
||||||
|
StyleSheet,
|
||||||
|
ViewStyle,
|
||||||
|
StyleProp,
|
||||||
|
} from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
|
interface ListItemProps {
|
||||||
|
title: string;
|
||||||
|
subtitle?: string;
|
||||||
|
leftIcon?: React.ReactNode;
|
||||||
|
rightElement?: React.ReactNode;
|
||||||
|
onPress?: () => void;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ListItem - Clean list item with optional icon and trailing element
|
||||||
|
*
|
||||||
|
* Usage:
|
||||||
|
* - Settings lists
|
||||||
|
* - Navigation items
|
||||||
|
* - Selectable options
|
||||||
|
*
|
||||||
|
* Features:
|
||||||
|
* - Optional left icon (wrap in IconContainer)
|
||||||
|
* - Optional subtitle for additional context
|
||||||
|
* - Optional right element (arrow, badge, switch, etc.)
|
||||||
|
* - Optional onPress for touchable items
|
||||||
|
*/
|
||||||
|
export function ListItem({
|
||||||
|
title,
|
||||||
|
subtitle,
|
||||||
|
leftIcon,
|
||||||
|
rightElement,
|
||||||
|
onPress,
|
||||||
|
style,
|
||||||
|
}: ListItemProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
|
|
||||||
|
const content = (
|
||||||
|
<View style={[styles.container, style]}>
|
||||||
|
{leftIcon && <View style={styles.leftIcon}>{leftIcon}</View>}
|
||||||
|
<View style={styles.textContainer}>
|
||||||
|
<Text style={[typography.h3, { color: colors.textPrimary }]}>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
{subtitle && (
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textTertiary, marginTop: 2 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{subtitle}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
{rightElement && <View style={styles.rightElement}>{rightElement}</View>}
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (onPress) {
|
||||||
|
return (
|
||||||
|
<TouchableOpacity onPress={onPress} activeOpacity={0.7}>
|
||||||
|
{content}
|
||||||
|
</TouchableOpacity>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
container: {
|
||||||
|
flexDirection: "row",
|
||||||
|
alignItems: "center",
|
||||||
|
paddingVertical: 12,
|
||||||
|
paddingHorizontal: 16,
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
},
|
||||||
|
leftIcon: {
|
||||||
|
marginRight: 12,
|
||||||
|
},
|
||||||
|
textContainer: {
|
||||||
|
flex: 1,
|
||||||
|
},
|
||||||
|
rightElement: {
|
||||||
|
marginLeft: 12,
|
||||||
|
},
|
||||||
|
});
|
||||||
152
apps/mobile/src/components/MinimalButton.tsx
Normal file
152
apps/mobile/src/components/MinimalButton.tsx
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
import React from "react";
|
||||||
|
import {
|
||||||
|
TouchableOpacity,
|
||||||
|
Text,
|
||||||
|
StyleSheet,
|
||||||
|
ActivityIndicator,
|
||||||
|
ViewStyle,
|
||||||
|
TextStyle,
|
||||||
|
StyleProp,
|
||||||
|
} from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
import { fontSize, fontWeight } from "../styles/typography";
|
||||||
|
|
||||||
|
type ButtonVariant =
|
||||||
|
| "primary"
|
||||||
|
| "secondary"
|
||||||
|
| "tertiary"
|
||||||
|
| "danger"
|
||||||
|
| "success";
|
||||||
|
type ButtonSize = "sm" | "md" | "lg" | "xl";
|
||||||
|
|
||||||
|
interface MinimalButtonProps {
|
||||||
|
title: string;
|
||||||
|
onPress: () => void;
|
||||||
|
variant?: ButtonVariant;
|
||||||
|
size?: ButtonSize;
|
||||||
|
loading?: boolean;
|
||||||
|
disabled?: boolean;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
textStyle?: StyleProp<TextStyle>;
|
||||||
|
fullWidth?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function MinimalButton({
|
||||||
|
title,
|
||||||
|
onPress,
|
||||||
|
variant = "primary",
|
||||||
|
size = "md",
|
||||||
|
loading = false,
|
||||||
|
disabled = false,
|
||||||
|
style,
|
||||||
|
textStyle,
|
||||||
|
fullWidth = false,
|
||||||
|
}: MinimalButtonProps) {
|
||||||
|
const { colors } = useTheme();
|
||||||
|
|
||||||
|
const isDisabled = disabled || loading;
|
||||||
|
|
||||||
|
const getButtonStyle = (): ViewStyle => {
|
||||||
|
const baseStyle: ViewStyle = {
|
||||||
|
borderRadius: 14,
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
opacity: isDisabled ? 0.5 : 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
const sizeStyles: Record<
|
||||||
|
ButtonSize,
|
||||||
|
{ paddingVertical: number; paddingHorizontal: number }
|
||||||
|
> = {
|
||||||
|
sm: { paddingVertical: 12, paddingHorizontal: 20 },
|
||||||
|
md: { paddingVertical: 16, paddingHorizontal: 28 },
|
||||||
|
lg: { paddingVertical: 18, paddingHorizontal: 36 },
|
||||||
|
xl: { paddingVertical: 20, paddingHorizontal: 44 },
|
||||||
|
};
|
||||||
|
|
||||||
|
const variantStyles: Record<ButtonVariant, ViewStyle> = {
|
||||||
|
primary: {
|
||||||
|
backgroundColor: colors.primary,
|
||||||
|
shadowColor: colors.primary,
|
||||||
|
shadowOffset: { width: 0, height: 4 },
|
||||||
|
shadowOpacity: 0.3,
|
||||||
|
shadowRadius: 8,
|
||||||
|
elevation: 4,
|
||||||
|
},
|
||||||
|
secondary: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
borderWidth: 2,
|
||||||
|
borderColor: colors.primary,
|
||||||
|
},
|
||||||
|
tertiary: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
},
|
||||||
|
danger: {
|
||||||
|
backgroundColor: colors.danger,
|
||||||
|
shadowColor: colors.danger,
|
||||||
|
shadowOffset: { width: 0, height: 4 },
|
||||||
|
shadowOpacity: 0.3,
|
||||||
|
shadowRadius: 8,
|
||||||
|
elevation: 4,
|
||||||
|
},
|
||||||
|
success: {
|
||||||
|
backgroundColor: colors.success,
|
||||||
|
shadowColor: colors.success,
|
||||||
|
shadowOffset: { width: 0, height: 4 },
|
||||||
|
shadowOpacity: 0.3,
|
||||||
|
shadowRadius: 8,
|
||||||
|
elevation: 4,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
...baseStyle,
|
||||||
|
...sizeStyles[size],
|
||||||
|
...variantStyles[variant],
|
||||||
|
...(fullWidth && { width: "100%" }),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const getTextStyle = (): TextStyle => {
|
||||||
|
const baseTextStyle: TextStyle = {
|
||||||
|
fontSize: size === "sm" ? fontSize.sm : fontSize.md,
|
||||||
|
fontWeight: fontWeight.bold,
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
};
|
||||||
|
|
||||||
|
const variantTextStyles: Record<ButtonVariant, TextStyle> = {
|
||||||
|
primary: { color: colors.white },
|
||||||
|
secondary: { color: colors.primary },
|
||||||
|
tertiary: { color: colors.primary },
|
||||||
|
danger: { color: colors.white },
|
||||||
|
success: { color: colors.white },
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
...baseTextStyle,
|
||||||
|
...variantTextStyles[variant],
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<TouchableOpacity
|
||||||
|
style={[getButtonStyle(), style]}
|
||||||
|
onPress={onPress}
|
||||||
|
disabled={isDisabled}
|
||||||
|
activeOpacity={0.85}
|
||||||
|
>
|
||||||
|
{loading ? (
|
||||||
|
<ActivityIndicator
|
||||||
|
size="small"
|
||||||
|
color={
|
||||||
|
variant === "secondary" || variant === "tertiary"
|
||||||
|
? colors.primary
|
||||||
|
: colors.white
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Text style={[getTextStyle(), textStyle]}>{title}</Text>
|
||||||
|
)}
|
||||||
|
</TouchableOpacity>
|
||||||
|
);
|
||||||
|
}
|
||||||
84
apps/mobile/src/components/MinimalCard.tsx
Normal file
84
apps/mobile/src/components/MinimalCard.tsx
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
import React from "react";
|
||||||
|
import {
|
||||||
|
View,
|
||||||
|
StyleSheet,
|
||||||
|
TouchableOpacity,
|
||||||
|
ViewStyle,
|
||||||
|
StyleProp,
|
||||||
|
} from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
|
type CardVariant = "default" | "elevated" | "bordered" | "gradient";
|
||||||
|
|
||||||
|
interface MinimalCardProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
variant?: CardVariant;
|
||||||
|
onPress?: () => void;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
padding?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function MinimalCard({
|
||||||
|
children,
|
||||||
|
variant = "default",
|
||||||
|
onPress,
|
||||||
|
style,
|
||||||
|
padding = 20,
|
||||||
|
}: MinimalCardProps) {
|
||||||
|
const { colors } = useTheme();
|
||||||
|
|
||||||
|
const cardStyles = [
|
||||||
|
styles.base,
|
||||||
|
{
|
||||||
|
backgroundColor: colors.surface,
|
||||||
|
padding: padding,
|
||||||
|
},
|
||||||
|
variant === "default" && styles.default,
|
||||||
|
variant === "elevated" && {
|
||||||
|
...styles.elevated,
|
||||||
|
backgroundColor: colors.surfaceElevated,
|
||||||
|
},
|
||||||
|
variant === "bordered" && {
|
||||||
|
borderWidth: 1,
|
||||||
|
borderColor: colors.border,
|
||||||
|
},
|
||||||
|
variant === "gradient" && {
|
||||||
|
backgroundColor: colors.surfaceElevated,
|
||||||
|
},
|
||||||
|
style,
|
||||||
|
];
|
||||||
|
|
||||||
|
if (onPress) {
|
||||||
|
return (
|
||||||
|
<TouchableOpacity
|
||||||
|
style={cardStyles}
|
||||||
|
onPress={onPress}
|
||||||
|
activeOpacity={0.85}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</TouchableOpacity>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return <View style={cardStyles}>{children}</View>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
base: {
|
||||||
|
borderRadius: 20,
|
||||||
|
},
|
||||||
|
default: {
|
||||||
|
shadowColor: "#000",
|
||||||
|
shadowOffset: { width: 0, height: 2 },
|
||||||
|
shadowOpacity: 0.08,
|
||||||
|
shadowRadius: 8,
|
||||||
|
elevation: 3,
|
||||||
|
},
|
||||||
|
elevated: {
|
||||||
|
shadowColor: "#000",
|
||||||
|
shadowOffset: { width: 0, height: 4 },
|
||||||
|
shadowOpacity: 0.12,
|
||||||
|
shadowRadius: 16,
|
||||||
|
elevation: 6,
|
||||||
|
},
|
||||||
|
});
|
||||||
67
apps/mobile/src/components/ProgressBar.tsx
Normal file
67
apps/mobile/src/components/ProgressBar.tsx
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { View, StyleSheet, ViewStyle, StyleProp } from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
|
interface ProgressBarProps {
|
||||||
|
progress: number;
|
||||||
|
color?: string;
|
||||||
|
backgroundColor?: string;
|
||||||
|
height?: number;
|
||||||
|
borderRadius?: number;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
animated?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProgressBar({
|
||||||
|
progress,
|
||||||
|
color,
|
||||||
|
backgroundColor,
|
||||||
|
height = 10,
|
||||||
|
borderRadius = 999,
|
||||||
|
style,
|
||||||
|
}: ProgressBarProps) {
|
||||||
|
const { colors } = useTheme();
|
||||||
|
|
||||||
|
const clampedProgress = Math.min(Math.max(progress, 0), 1);
|
||||||
|
|
||||||
|
const trackColor = backgroundColor || colors.surfaceElevated;
|
||||||
|
const fillColor = color || colors.primary;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.track,
|
||||||
|
{
|
||||||
|
height,
|
||||||
|
borderRadius,
|
||||||
|
backgroundColor: trackColor,
|
||||||
|
},
|
||||||
|
style,
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
styles.fill,
|
||||||
|
{
|
||||||
|
width: `${clampedProgress * 100}%`,
|
||||||
|
height,
|
||||||
|
borderRadius,
|
||||||
|
backgroundColor: fillColor,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
track: {
|
||||||
|
width: "100%",
|
||||||
|
overflow: "hidden",
|
||||||
|
},
|
||||||
|
fill: {
|
||||||
|
position: "absolute",
|
||||||
|
left: 0,
|
||||||
|
top: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
67
apps/mobile/src/components/SectionHeader.tsx
Normal file
67
apps/mobile/src/components/SectionHeader.tsx
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
import React from "react";
|
||||||
|
import {
|
||||||
|
View,
|
||||||
|
Text,
|
||||||
|
TouchableOpacity,
|
||||||
|
StyleSheet,
|
||||||
|
ViewStyle,
|
||||||
|
StyleProp,
|
||||||
|
} from "react-native";
|
||||||
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
|
|
||||||
|
interface SectionHeaderProps {
|
||||||
|
title: string;
|
||||||
|
subtitle?: string;
|
||||||
|
actionLabel?: string;
|
||||||
|
onActionPress?: () => void;
|
||||||
|
style?: StyleProp<ViewStyle>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SectionHeader({
|
||||||
|
title,
|
||||||
|
subtitle,
|
||||||
|
actionLabel,
|
||||||
|
onActionPress,
|
||||||
|
style,
|
||||||
|
}: SectionHeaderProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View style={[styles.container, style]}>
|
||||||
|
<View style={styles.textContainer}>
|
||||||
|
<Text style={[typography.h2, { color: colors.textPrimary }]}>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
{subtitle && (
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.caption,
|
||||||
|
{ color: colors.textTertiary, marginTop: 4 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{subtitle}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
{actionLabel && onActionPress && (
|
||||||
|
<TouchableOpacity onPress={onActionPress} activeOpacity={0.7}>
|
||||||
|
<Text style={[typography.bodyEmphasis, { color: colors.primary }]}>
|
||||||
|
{actionLabel}
|
||||||
|
</Text>
|
||||||
|
</TouchableOpacity>
|
||||||
|
)}
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
container: {
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "space-between",
|
||||||
|
alignItems: "center",
|
||||||
|
marginBottom: 16,
|
||||||
|
},
|
||||||
|
textContainer: {
|
||||||
|
flex: 1,
|
||||||
|
},
|
||||||
|
});
|
||||||
@ -1,7 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { View, Text, StyleSheet, Dimensions } from "react-native";
|
import { View, Text, StyleSheet, Dimensions } from "react-native";
|
||||||
import { LineChart } from "react-native-chart-kit";
|
import { LineChart } from "react-native-chart-kit";
|
||||||
import { theme } from "../styles/theme";
|
import { useTheme } from "../contexts/ThemeContext";
|
||||||
import type { WeeklyTrendData } from "../api/types";
|
import type { WeeklyTrendData } from "../api/types";
|
||||||
|
|
||||||
interface WeeklyProgressChartProps {
|
interface WeeklyProgressChartProps {
|
||||||
@ -13,28 +13,26 @@ export function WeeklyProgressChart({
|
|||||||
weeklyData,
|
weeklyData,
|
||||||
title = "Weekly Progress",
|
title = "Weekly Progress",
|
||||||
}: WeeklyProgressChartProps) {
|
}: WeeklyProgressChartProps) {
|
||||||
|
const { colors, typography } = useTheme();
|
||||||
const screenWidth = Dimensions.get("window").width;
|
const screenWidth = Dimensions.get("window").width;
|
||||||
|
|
||||||
// Prepare chart data
|
const labels = weeklyData.map((week: WeeklyTrendData) => week.weekLabel);
|
||||||
const labels = weeklyData.map((week) => week.weekLabel);
|
const checkInsData = weeklyData.map((week: WeeklyTrendData) => week.checkIns);
|
||||||
const checkInsData = weeklyData.map((week) => week.checkIns);
|
const goalsCompletedData = weeklyData.map(
|
||||||
const goalsCompletedData = weeklyData.map((week) => week.goalsCompleted);
|
(week: WeeklyTrendData) => week.goalsCompleted,
|
||||||
const avgProgressData = weeklyData.map((week) => week.averageProgress);
|
);
|
||||||
|
|
||||||
const chartConfig = {
|
const chartConfig = {
|
||||||
backgroundColor: theme.colors.white,
|
backgroundColor: colors.surface,
|
||||||
backgroundGradientFrom: theme.colors.white,
|
backgroundGradientFrom: colors.surface,
|
||||||
backgroundGradientTo: theme.colors.white,
|
backgroundGradientTo: colors.surface,
|
||||||
decimalPlaces: 0,
|
decimalPlaces: 0,
|
||||||
color: (opacity = 1) => `rgba(59, 130, 246, ${opacity})`,
|
color: (opacity = 1) => `rgba(0, 102, 255, ${opacity})`,
|
||||||
labelColor: (opacity = 1) => `rgba(107, 114, 128, ${opacity})`,
|
labelColor: (opacity = 1) => colors.textTertiary,
|
||||||
style: {
|
|
||||||
borderRadius: theme.borderRadius.lg,
|
|
||||||
},
|
|
||||||
propsForDots: {
|
propsForDots: {
|
||||||
r: "4",
|
r: "5",
|
||||||
strokeWidth: "2",
|
strokeWidth: "2",
|
||||||
stroke: theme.colors.primary,
|
stroke: colors.primary,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -43,31 +41,40 @@ export function WeeklyProgressChart({
|
|||||||
datasets: [
|
datasets: [
|
||||||
{
|
{
|
||||||
data: checkInsData,
|
data: checkInsData,
|
||||||
color: (opacity = 1) => `rgba(59, 130, 246, ${opacity})`, // Blue for check-ins
|
color: () => colors.primary,
|
||||||
strokeWidth: 2,
|
strokeWidth: 3,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
data: goalsCompletedData,
|
data: goalsCompletedData,
|
||||||
color: (opacity = 1) => `rgba(16, 185, 129, ${opacity})`, // Green for goals
|
color: () => colors.success,
|
||||||
strokeWidth: 2,
|
strokeWidth: 3,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
legend: ["Check-ins", "Goals Completed"],
|
legend: ["Check-ins", "Goals"],
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={styles.container}>
|
<View style={styles.container}>
|
||||||
<Text style={styles.title}>{title}</Text>
|
{title && (
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
typography.h4,
|
||||||
|
{ color: colors.textPrimary, marginBottom: 16 },
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
<View style={styles.chartContainer}>
|
<View style={styles.chartContainer}>
|
||||||
<LineChart
|
<LineChart
|
||||||
data={data}
|
data={data}
|
||||||
width={screenWidth - 60}
|
width={screenWidth - 80}
|
||||||
height={220}
|
height={180}
|
||||||
chartConfig={chartConfig}
|
chartConfig={chartConfig}
|
||||||
bezier
|
bezier
|
||||||
style={styles.chart}
|
style={styles.chart}
|
||||||
withInnerLines={true}
|
withInnerLines={true}
|
||||||
withOuterLines={true}
|
withOuterLines={false}
|
||||||
withVerticalLabels={true}
|
withVerticalLabels={true}
|
||||||
withHorizontalLabels={true}
|
withHorizontalLabels={true}
|
||||||
fromZero={true}
|
fromZero={true}
|
||||||
@ -75,12 +82,20 @@ export function WeeklyProgressChart({
|
|||||||
</View>
|
</View>
|
||||||
<View style={styles.legend}>
|
<View style={styles.legend}>
|
||||||
<View style={styles.legendItem}>
|
<View style={styles.legendItem}>
|
||||||
<View style={[styles.legendDot, { backgroundColor: "#3b82f6" }]} />
|
<View
|
||||||
<Text style={styles.legendText}>Check-ins</Text>
|
style={[styles.legendDot, { backgroundColor: colors.primary }]}
|
||||||
|
/>
|
||||||
|
<Text style={[typography.caption, { color: colors.textSecondary }]}>
|
||||||
|
Check-ins
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
<View style={styles.legendItem}>
|
<View style={styles.legendItem}>
|
||||||
<View style={[styles.legendDot, { backgroundColor: "#10b981" }]} />
|
<View
|
||||||
<Text style={styles.legendText}>Goals Completed</Text>
|
style={[styles.legendDot, { backgroundColor: colors.success }]}
|
||||||
|
/>
|
||||||
|
<Text style={[typography.caption, { color: colors.textSecondary }]}>
|
||||||
|
Goals
|
||||||
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
@ -89,17 +104,8 @@ export function WeeklyProgressChart({
|
|||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
container: {
|
container: {
|
||||||
backgroundColor: theme.colors.white,
|
borderRadius: 16,
|
||||||
borderRadius: theme.borderRadius.xl,
|
padding: 4,
|
||||||
padding: 16,
|
|
||||||
marginBottom: 16,
|
|
||||||
...theme.shadows.medium,
|
|
||||||
},
|
|
||||||
title: {
|
|
||||||
fontSize: theme.typography.fontSize.lg,
|
|
||||||
fontWeight: theme.typography.fontWeight.bold,
|
|
||||||
color: theme.colors.gray700,
|
|
||||||
marginBottom: 12,
|
|
||||||
},
|
},
|
||||||
chartContainer: {
|
chartContainer: {
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
@ -107,26 +113,22 @@ const styles = StyleSheet.create({
|
|||||||
},
|
},
|
||||||
chart: {
|
chart: {
|
||||||
marginVertical: 8,
|
marginVertical: 8,
|
||||||
borderRadius: theme.borderRadius.lg,
|
borderRadius: 12,
|
||||||
},
|
},
|
||||||
legend: {
|
legend: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
gap: 20,
|
gap: 24,
|
||||||
paddingTop: 8,
|
paddingTop: 8,
|
||||||
},
|
},
|
||||||
legendItem: {
|
legendItem: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
gap: 6,
|
gap: 8,
|
||||||
},
|
},
|
||||||
legendDot: {
|
legendDot: {
|
||||||
width: 10,
|
width: 10,
|
||||||
height: 10,
|
height: 10,
|
||||||
borderRadius: 5,
|
borderRadius: 5,
|
||||||
},
|
},
|
||||||
legendText: {
|
|
||||||
fontSize: theme.typography.fontSize.sm,
|
|
||||||
color: theme.colors.gray500,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|||||||
@ -35,6 +35,7 @@ export const API_ENDPOINTS = {
|
|||||||
USERS: {
|
USERS: {
|
||||||
LIST: "/api/users",
|
LIST: "/api/users",
|
||||||
STATISTICS: "/api/users/statistics",
|
STATISTICS: "/api/users/statistics",
|
||||||
|
GYM: "/api/users/gym",
|
||||||
},
|
},
|
||||||
GYMS: "/api/gyms",
|
GYMS: "/api/gyms",
|
||||||
ATTENDANCE: {
|
ATTENDANCE: {
|
||||||
|
|||||||
129
apps/mobile/src/contexts/ThemeContext.tsx
Normal file
129
apps/mobile/src/contexts/ThemeContext.tsx
Normal file
@ -0,0 +1,129 @@
|
|||||||
|
import React, {
|
||||||
|
createContext,
|
||||||
|
useContext,
|
||||||
|
useState,
|
||||||
|
useEffect,
|
||||||
|
ReactNode,
|
||||||
|
} from "react";
|
||||||
|
import { useColorScheme } from "react-native";
|
||||||
|
import AsyncStorage from "@react-native-async-storage/async-storage";
|
||||||
|
import { ColorScheme, lightColors, darkColors } from "../styles/colors";
|
||||||
|
import {
|
||||||
|
TypographyPresets,
|
||||||
|
createTypographyPresets,
|
||||||
|
} from "../styles/typography";
|
||||||
|
|
||||||
|
type ThemeMode = "light" | "dark" | "system";
|
||||||
|
type ActiveTheme = "light" | "dark";
|
||||||
|
|
||||||
|
interface ThemeContextType {
|
||||||
|
// Current active theme
|
||||||
|
theme: ActiveTheme;
|
||||||
|
|
||||||
|
// User's theme preference
|
||||||
|
themeMode: ThemeMode;
|
||||||
|
|
||||||
|
// Active color scheme
|
||||||
|
colors: ColorScheme;
|
||||||
|
|
||||||
|
// Typography presets
|
||||||
|
typography: TypographyPresets;
|
||||||
|
|
||||||
|
// Theme actions
|
||||||
|
setTheme: (mode: ThemeMode) => void;
|
||||||
|
toggleTheme: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ThemeContext = createContext<ThemeContextType | undefined>(undefined);
|
||||||
|
|
||||||
|
const THEME_STORAGE_KEY = "@fitai:theme";
|
||||||
|
|
||||||
|
interface ThemeProviderProps {
|
||||||
|
children: ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ThemeProvider({ children }: ThemeProviderProps) {
|
||||||
|
const systemColorScheme = useColorScheme();
|
||||||
|
const [themeMode, setThemeMode] = useState<ThemeMode>("system");
|
||||||
|
const [isLoading, setIsLoading] = useState(true);
|
||||||
|
|
||||||
|
// Determine active theme based on mode and system preference
|
||||||
|
const getActiveTheme = (): ActiveTheme => {
|
||||||
|
if (themeMode === "system") {
|
||||||
|
return systemColorScheme === "dark" ? "dark" : "light";
|
||||||
|
}
|
||||||
|
return themeMode;
|
||||||
|
};
|
||||||
|
|
||||||
|
const activeTheme = getActiveTheme();
|
||||||
|
const colors = activeTheme === "dark" ? darkColors : lightColors;
|
||||||
|
const typography = createTypographyPresets(
|
||||||
|
colors.textPrimary,
|
||||||
|
colors.textSecondary,
|
||||||
|
colors.textTertiary,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Load saved theme preference on mount
|
||||||
|
useEffect(() => {
|
||||||
|
const loadTheme = async () => {
|
||||||
|
try {
|
||||||
|
const savedTheme = await AsyncStorage.getItem(THEME_STORAGE_KEY);
|
||||||
|
if (savedTheme && ["light", "dark", "system"].includes(savedTheme)) {
|
||||||
|
setThemeMode(savedTheme as ThemeMode);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to load theme preference:", error);
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
loadTheme();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Save theme preference when it changes
|
||||||
|
const setTheme = async (mode: ThemeMode) => {
|
||||||
|
try {
|
||||||
|
setThemeMode(mode);
|
||||||
|
await AsyncStorage.setItem(THEME_STORAGE_KEY, mode);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to save theme preference:", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Toggle between light and dark (sets explicit mode, not system)
|
||||||
|
const toggleTheme = () => {
|
||||||
|
const newMode = activeTheme === "dark" ? "light" : "dark";
|
||||||
|
setTheme(newMode);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Don't render children until theme is loaded
|
||||||
|
if (isLoading) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const value: ThemeContextType = {
|
||||||
|
theme: activeTheme,
|
||||||
|
themeMode,
|
||||||
|
colors,
|
||||||
|
typography,
|
||||||
|
setTheme,
|
||||||
|
toggleTheme,
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ThemeContext.Provider value={value}>{children}</ThemeContext.Provider>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hook to access theme context
|
||||||
|
* @throws Error if used outside ThemeProvider
|
||||||
|
*/
|
||||||
|
export function useTheme() {
|
||||||
|
const context = useContext(ThemeContext);
|
||||||
|
if (context === undefined) {
|
||||||
|
throw new Error("useTheme must be used within a ThemeProvider");
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
}
|
||||||
167
apps/mobile/src/styles/colors.ts
Normal file
167
apps/mobile/src/styles/colors.ts
Normal file
@ -0,0 +1,167 @@
|
|||||||
|
/**
|
||||||
|
* FitAI Color System - BOLD MODERN
|
||||||
|
* Electric Blue palette with high-energy fitness app aesthetics
|
||||||
|
*/
|
||||||
|
|
||||||
|
export interface ColorScheme {
|
||||||
|
// Primary Colors
|
||||||
|
primary: string;
|
||||||
|
primaryDark: string;
|
||||||
|
primaryLight: string;
|
||||||
|
|
||||||
|
// Accent Colors
|
||||||
|
accent: string;
|
||||||
|
secondary: string;
|
||||||
|
terracotta: string;
|
||||||
|
sand: string;
|
||||||
|
|
||||||
|
// Status Colors
|
||||||
|
success: string;
|
||||||
|
warning: string;
|
||||||
|
danger: string;
|
||||||
|
info: string;
|
||||||
|
|
||||||
|
// Activity Ring Colors
|
||||||
|
calories: string;
|
||||||
|
water: string;
|
||||||
|
workouts: string;
|
||||||
|
|
||||||
|
// Neutrals
|
||||||
|
background: string;
|
||||||
|
surface: string;
|
||||||
|
surfaceElevated: string;
|
||||||
|
|
||||||
|
// Text
|
||||||
|
textPrimary: string;
|
||||||
|
textSecondary: string;
|
||||||
|
textTertiary: string;
|
||||||
|
|
||||||
|
// Borders
|
||||||
|
border: string;
|
||||||
|
borderLight: string;
|
||||||
|
|
||||||
|
// Overlays
|
||||||
|
overlay: string;
|
||||||
|
overlayLight: string;
|
||||||
|
|
||||||
|
// Gradients (as arrays)
|
||||||
|
primaryGradient: string[];
|
||||||
|
cardGradient: string[];
|
||||||
|
|
||||||
|
// Legacy compatibility
|
||||||
|
white: string;
|
||||||
|
black: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Light Mode - Bold & Energetic
|
||||||
|
*/
|
||||||
|
export const lightColors: ColorScheme = {
|
||||||
|
// Primary Colors - Electric Blue
|
||||||
|
primary: "#0066FF",
|
||||||
|
primaryDark: "#0052CC",
|
||||||
|
primaryLight: "#3385FF",
|
||||||
|
|
||||||
|
// Accent Colors
|
||||||
|
accent: "#7B2CBF", // Purple
|
||||||
|
secondary: "#FF3B7A", // Hot Pink
|
||||||
|
terracotta: "#FF6B35", // Neon Orange
|
||||||
|
sand: "#FFD60A", // Electric Yellow
|
||||||
|
|
||||||
|
// Status Colors - Vibrant
|
||||||
|
success: "#00D26A",
|
||||||
|
warning: "#FFB800",
|
||||||
|
danger: "#FF3B3B",
|
||||||
|
info: "#00B8D9",
|
||||||
|
|
||||||
|
// Activity Ring Colors
|
||||||
|
calories: "#FF6B35", // Orange for calories
|
||||||
|
water: "#00B8D9", // Cyan for water
|
||||||
|
workouts: "#0066FF", // Blue for workouts
|
||||||
|
|
||||||
|
// Neutrals - Bold dark on light
|
||||||
|
background: "#F5F5F7",
|
||||||
|
surface: "#FFFFFF",
|
||||||
|
surfaceElevated: "#FFFFFF",
|
||||||
|
|
||||||
|
// Text - High contrast dark
|
||||||
|
textPrimary: "#1A1A1A",
|
||||||
|
textSecondary: "#4A4A4A",
|
||||||
|
textTertiary: "#8E8E93",
|
||||||
|
|
||||||
|
// Borders
|
||||||
|
border: "#E5E5EA",
|
||||||
|
borderLight: "#F0F0F5",
|
||||||
|
|
||||||
|
// Overlays
|
||||||
|
overlay: "rgba(0, 0, 0, 0.5)",
|
||||||
|
overlayLight: "rgba(0, 0, 0, 0.03)",
|
||||||
|
|
||||||
|
// Gradients
|
||||||
|
primaryGradient: ["#0066FF", "#0052CC"],
|
||||||
|
cardGradient: ["#FFFFFF", "#F8F8FA"],
|
||||||
|
|
||||||
|
// Legacy
|
||||||
|
white: "#FFFFFF",
|
||||||
|
black: "#1A1A1A",
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dark Mode - Premium & Immersive
|
||||||
|
*/
|
||||||
|
export const darkColors: ColorScheme = {
|
||||||
|
// Primary Colors - Electric Blue (brighter on dark)
|
||||||
|
primary: "#0A84FF",
|
||||||
|
primaryDark: "#0066FF",
|
||||||
|
primaryLight: "#5AC8FA",
|
||||||
|
|
||||||
|
// Accent Colors
|
||||||
|
accent: "#BF5AF2", // Purple
|
||||||
|
secondary: "#FF375F", // Hot Pink
|
||||||
|
terracotta: "#FF9500", // Orange
|
||||||
|
sand: "#FFD60A", // Yellow
|
||||||
|
|
||||||
|
// Status Colors
|
||||||
|
success: "#30D158",
|
||||||
|
warning: "#FFD60A",
|
||||||
|
danger: "#FF453A",
|
||||||
|
info: "#64D2FF",
|
||||||
|
|
||||||
|
// Activity Ring Colors (even brighter for dark mode)
|
||||||
|
calories: "#FF9500",
|
||||||
|
water: "#64D2FF",
|
||||||
|
workouts: "#0A84FF",
|
||||||
|
|
||||||
|
// Neutrals - Dark backgrounds
|
||||||
|
background: "#000000",
|
||||||
|
surface: "#1C1C1E",
|
||||||
|
surfaceElevated: "#2C2C2E",
|
||||||
|
|
||||||
|
// Text - Bright on dark
|
||||||
|
textPrimary: "#FFFFFF",
|
||||||
|
textSecondary: "#EBEBF5",
|
||||||
|
textTertiary: "#8E8E93",
|
||||||
|
|
||||||
|
// Borders
|
||||||
|
border: "#38383A",
|
||||||
|
borderLight: "#48484A",
|
||||||
|
|
||||||
|
// Overlays
|
||||||
|
overlay: "rgba(0, 0, 0, 0.6)",
|
||||||
|
overlayLight: "rgba(255, 255, 255, 0.05)",
|
||||||
|
|
||||||
|
// Gradients
|
||||||
|
primaryGradient: ["#0A84FF", "#0066FF"],
|
||||||
|
cardGradient: ["#1C1C1E", "#2C2C2E"],
|
||||||
|
|
||||||
|
// Legacy
|
||||||
|
white: "#FFFFFF",
|
||||||
|
black: "#000000",
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get color scheme based on theme mode
|
||||||
|
*/
|
||||||
|
export const getColors = (mode: "light" | "dark"): ColorScheme => {
|
||||||
|
return mode === "dark" ? darkColors : lightColors;
|
||||||
|
};
|
||||||
@ -1,191 +1,159 @@
|
|||||||
/**
|
/**
|
||||||
* Modern Design System Theme
|
* FitAI Design System Theme
|
||||||
* Centralized theme configuration with gradients, colors, shadows, and spacing
|
* Minimalist theme with light/dark mode support
|
||||||
|
* @deprecated Use useTheme() hook instead for dynamic theming
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { lightColors, darkColors } from "./colors";
|
||||||
|
import { typography as typographySystem } from "./typography";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Legacy theme object for backward compatibility
|
||||||
|
* New components should use useTheme() hook instead
|
||||||
|
*/
|
||||||
export const theme = {
|
export const theme = {
|
||||||
// Color Palette
|
// Color Palette (light mode - for legacy components)
|
||||||
colors: {
|
colors: {
|
||||||
// Primary colors
|
...lightColors,
|
||||||
primary: '#3b82f6',
|
|
||||||
primaryDark: '#2563eb',
|
|
||||||
primaryLight: '#60a5fa',
|
|
||||||
secondary: '#8b5cf6',
|
|
||||||
|
|
||||||
// Accent colors
|
// Legacy color mappings (deprecated - use new colors instead)
|
||||||
purple: '#8b5cf6',
|
secondary: "#8b5cf6", // Old purple - deprecated
|
||||||
purpleDark: '#7c3aed',
|
purple: "#A9B4A0", // Mapped to accent
|
||||||
pink: '#ec4899',
|
purpleDark: "#5A7A6E",
|
||||||
|
pink: "#C1876B", // Mapped to terracotta
|
||||||
|
successLight: "#8DB76A",
|
||||||
|
gray50: "#F2EFE9",
|
||||||
|
gray100: "#E8E4DF",
|
||||||
|
gray200: "#E8E4DF",
|
||||||
|
gray300: "#B8BFB5",
|
||||||
|
gray400: "#8B9A8F",
|
||||||
|
gray500: "#5C6B61",
|
||||||
|
gray600: "#5C6B61",
|
||||||
|
gray700: "#2C3731",
|
||||||
|
gray800: "#2C3731",
|
||||||
|
gray900: "#2C3731",
|
||||||
|
},
|
||||||
|
|
||||||
// Success
|
// Typography (updated to new system)
|
||||||
success: '#10b981',
|
typography: typographySystem,
|
||||||
successDark: '#059669',
|
|
||||||
successLight: '#34d399',
|
|
||||||
|
|
||||||
// Warning
|
// Spacing Scale (updated for minimalism)
|
||||||
warning: '#f59e0b',
|
spacing: {
|
||||||
warningDark: '#d97706',
|
xs: 4,
|
||||||
|
sm: 8,
|
||||||
|
md: 12,
|
||||||
|
lg: 16,
|
||||||
|
xl: 24, // Increased from 20
|
||||||
|
"2xl": 32, // Increased from 24
|
||||||
|
"3xl": 40, // Increased from 32
|
||||||
|
"4xl": 48, // New
|
||||||
|
"5xl": 64, // New
|
||||||
|
},
|
||||||
|
|
||||||
// Danger
|
// Border Radius (reduced for minimalism)
|
||||||
danger: '#ef4444',
|
borderRadius: {
|
||||||
dangerDark: '#dc2626',
|
sm: 4,
|
||||||
|
md: 6,
|
||||||
|
lg: 10,
|
||||||
|
xl: 12, // Reduced from 16
|
||||||
|
"2xl": 16, // Reduced from 20
|
||||||
|
"3xl": 20, // Reduced from 24
|
||||||
|
full: 9999,
|
||||||
|
},
|
||||||
|
|
||||||
// Neutrals
|
// Shadow System (simplified)
|
||||||
white: '#ffffff',
|
shadows: {
|
||||||
black: '#000000',
|
subtle: {
|
||||||
gray50: '#f9fafb',
|
shadowColor: "#000",
|
||||||
gray100: '#f3f4f6',
|
shadowOffset: { width: 0, height: 1 },
|
||||||
gray200: '#e5e7eb',
|
shadowOpacity: 0.05,
|
||||||
gray300: '#d1d5db',
|
shadowRadius: 3,
|
||||||
gray400: '#9ca3af',
|
elevation: 1,
|
||||||
gray500: '#6b7280',
|
|
||||||
gray600: '#4b5563',
|
|
||||||
gray700: '#374151',
|
|
||||||
gray800: '#1f2937',
|
|
||||||
gray900: '#111827',
|
|
||||||
|
|
||||||
// Backgrounds
|
|
||||||
background: '#f5f5f5',
|
|
||||||
backgroundDark: '#0f172a',
|
|
||||||
surface: '#ffffff',
|
|
||||||
surfaceDark: '#1e293b',
|
|
||||||
},
|
},
|
||||||
|
medium: {
|
||||||
// Gradient Definitions
|
shadowColor: "#000",
|
||||||
gradients: {
|
shadowOffset: { width: 0, height: 2 },
|
||||||
primary: ['#3b82f6', '#8b5cf6'] as const,
|
shadowOpacity: 0.08,
|
||||||
primaryVertical: ['#3b82f6', '#2563eb'] as const,
|
shadowRadius: 8,
|
||||||
success: ['#10b981', '#059669'] as const,
|
elevation: 2,
|
||||||
warning: ['#f59e0b', '#d97706'] as const,
|
|
||||||
danger: ['#ef4444', '#ec4899'] as const,
|
|
||||||
purple: ['#8b5cf6', '#7c3aed'] as const,
|
|
||||||
ocean: ['#06b6d4', '#3b82f6'] as const,
|
|
||||||
sunset: ['#f59e0b', '#ef4444'] as const,
|
|
||||||
forest: ['#10b981', '#059669'] as const,
|
|
||||||
lavender: ['#a78bfa', '#ec4899'] as const,
|
|
||||||
dark: ['#1e293b', '#0f172a'] as const,
|
|
||||||
},
|
},
|
||||||
|
strong: {
|
||||||
// Shadow System
|
shadowColor: "#000",
|
||||||
shadows: {
|
shadowOffset: { width: 0, height: 4 },
|
||||||
subtle: {
|
shadowOpacity: 0.12,
|
||||||
shadowColor: '#000',
|
shadowRadius: 16,
|
||||||
shadowOffset: { width: 0, height: 1 },
|
elevation: 4,
|
||||||
shadowOpacity: 0.05,
|
|
||||||
shadowRadius: 2,
|
|
||||||
elevation: 2,
|
|
||||||
},
|
|
||||||
medium: {
|
|
||||||
shadowColor: '#000',
|
|
||||||
shadowOffset: { width: 0, height: 2 },
|
|
||||||
shadowOpacity: 0.1,
|
|
||||||
shadowRadius: 4,
|
|
||||||
elevation: 3,
|
|
||||||
},
|
|
||||||
strong: {
|
|
||||||
shadowColor: '#000',
|
|
||||||
shadowOffset: { width: 0, height: 4 },
|
|
||||||
shadowOpacity: 0.15,
|
|
||||||
shadowRadius: 8,
|
|
||||||
elevation: 5,
|
|
||||||
},
|
|
||||||
glow: {
|
|
||||||
shadowColor: '#3b82f6',
|
|
||||||
shadowOffset: { width: 0, height: 4 },
|
|
||||||
shadowOpacity: 0.3,
|
|
||||||
shadowRadius: 12,
|
|
||||||
elevation: 8,
|
|
||||||
},
|
|
||||||
glowDanger: {
|
|
||||||
shadowColor: '#ef4444',
|
|
||||||
shadowOffset: { width: 0, height: 4 },
|
|
||||||
shadowOpacity: 0.3,
|
|
||||||
shadowRadius: 12,
|
|
||||||
elevation: 8,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
// Legacy glow shadows (deprecated - avoid in new designs)
|
||||||
// Typography
|
glow: {
|
||||||
typography: {
|
shadowColor: "#6B9080",
|
||||||
// Font sizes
|
shadowOffset: { width: 0, height: 4 },
|
||||||
fontSize: {
|
shadowOpacity: 0.2,
|
||||||
xs: 12,
|
shadowRadius: 12,
|
||||||
sm: 14,
|
elevation: 6,
|
||||||
base: 16,
|
|
||||||
lg: 18,
|
|
||||||
xl: 20,
|
|
||||||
'2xl': 24,
|
|
||||||
'3xl': 28,
|
|
||||||
'4xl': 32,
|
|
||||||
'5xl': 36,
|
|
||||||
},
|
|
||||||
// Font weights
|
|
||||||
fontWeight: {
|
|
||||||
normal: '400' as const,
|
|
||||||
medium: '500' as const,
|
|
||||||
semibold: '600' as const,
|
|
||||||
bold: '700' as const,
|
|
||||||
extrabold: '800' as const,
|
|
||||||
},
|
|
||||||
// Line heights
|
|
||||||
lineHeight: {
|
|
||||||
tight: 1.2,
|
|
||||||
normal: 1.5,
|
|
||||||
relaxed: 1.75,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
glowDanger: {
|
||||||
// Spacing Scale
|
shadowColor: "#B66B6B",
|
||||||
spacing: {
|
shadowOffset: { width: 0, height: 4 },
|
||||||
xs: 4,
|
shadowOpacity: 0.2,
|
||||||
sm: 8,
|
shadowRadius: 12,
|
||||||
md: 12,
|
elevation: 6,
|
||||||
lg: 16,
|
|
||||||
xl: 20,
|
|
||||||
'2xl': 24,
|
|
||||||
'3xl': 32,
|
|
||||||
'4xl': 40,
|
|
||||||
'5xl': 48,
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
// Border Radius
|
// Animation Timing (simplified)
|
||||||
borderRadius: {
|
animation: {
|
||||||
sm: 4,
|
duration: {
|
||||||
md: 8,
|
fast: 200,
|
||||||
lg: 12,
|
normal: 300,
|
||||||
xl: 16,
|
slow: 500,
|
||||||
'2xl': 20,
|
|
||||||
'3xl': 24,
|
|
||||||
full: 9999,
|
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
|
||||||
// Animation Timing
|
// Gradients (kept for legacy compatibility - should be avoided in new designs)
|
||||||
animation: {
|
gradients: {
|
||||||
duration: {
|
primary: ["#6B9080", "#8AAE9E"] as const,
|
||||||
fast: 150,
|
success: ["#7BA05B", "#8DB76A"] as const,
|
||||||
normal: 250,
|
warning: ["#D4A574", "#E0B886"] as const,
|
||||||
slow: 350,
|
danger: ["#B66B6B", "#C87D7D"] as const,
|
||||||
},
|
// Legacy gradients (deprecated)
|
||||||
easing: {
|
purple: ["#8b5cf6", "#7c3aed"] as const,
|
||||||
easeIn: 'ease-in',
|
ocean: ["#06b6d4", "#3b82f6"] as const,
|
||||||
easeOut: 'ease-out',
|
sunset: ["#f59e0b", "#ef4444"] as const,
|
||||||
easeInOut: 'ease-in-out',
|
forest: ["#10b981", "#059669"] as const,
|
||||||
},
|
lavender: ["#a78bfa", "#ec4899"] as const,
|
||||||
},
|
dark: ["#1e293b", "#0f172a"] as const,
|
||||||
|
primaryVertical: ["#6B9080", "#5A7A6E"] as const,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
// Glassmorphism
|
/**
|
||||||
glass: {
|
* Dark theme object
|
||||||
light: {
|
* @deprecated Use useTheme() hook instead
|
||||||
backgroundColor: 'rgba(255, 255, 255, 0.7)',
|
*/
|
||||||
borderWidth: 1,
|
export const darkTheme = {
|
||||||
borderColor: 'rgba(255, 255, 255, 0.3)',
|
...theme,
|
||||||
},
|
colors: {
|
||||||
dark: {
|
...darkColors,
|
||||||
backgroundColor: 'rgba(0, 0, 0, 0.3)',
|
|
||||||
borderWidth: 1,
|
// Legacy color mappings
|
||||||
borderColor: 'rgba(255, 255, 255, 0.1)',
|
secondary: "#8b5cf6", // Old purple - deprecated
|
||||||
},
|
purple: "#B5C2B0", // Mapped to dark accent
|
||||||
},
|
purpleDark: "#6B9080",
|
||||||
|
pink: "#D4A285", // Mapped to dark terracotta
|
||||||
|
successLight: "#8DB76A",
|
||||||
|
gray50: "#2F3432",
|
||||||
|
gray100: "#3A3F3C",
|
||||||
|
gray200: "#3A3F3C",
|
||||||
|
gray300: "#7A8379",
|
||||||
|
gray400: "#7A8379",
|
||||||
|
gray500: "#B8BFB5",
|
||||||
|
gray600: "#B8BFB5",
|
||||||
|
gray700: "#E8E6E1",
|
||||||
|
gray800: "#E8E6E1",
|
||||||
|
gray900: "#E8E6E1",
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Theme = typeof theme;
|
export type Theme = typeof theme;
|
||||||
|
export type DarkTheme = typeof darkTheme;
|
||||||
|
|||||||
174
apps/mobile/src/styles/typography.ts
Normal file
174
apps/mobile/src/styles/typography.ts
Normal file
@ -0,0 +1,174 @@
|
|||||||
|
/**
|
||||||
|
* FitAI Typography System - BOLD MODERN
|
||||||
|
* High-impact typography with clear hierarchy using system fonts
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { TextStyle } from "react-native";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Font Sizes - Larger for bold impact
|
||||||
|
*/
|
||||||
|
export const fontSize = {
|
||||||
|
xs: 12,
|
||||||
|
sm: 14,
|
||||||
|
base: 16,
|
||||||
|
md: 18, // Body emphasis
|
||||||
|
lg: 22,
|
||||||
|
xl: 26,
|
||||||
|
"2xl": 32,
|
||||||
|
"3xl": 40,
|
||||||
|
"4xl": 52,
|
||||||
|
"5xl": 64,
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Font Weights - Emphasize bold
|
||||||
|
*/
|
||||||
|
export const fontWeight = {
|
||||||
|
regular: "400" as TextStyle["fontWeight"],
|
||||||
|
medium: "500" as TextStyle["fontWeight"],
|
||||||
|
semibold: "600" as TextStyle["fontWeight"],
|
||||||
|
bold: "700" as TextStyle["fontWeight"],
|
||||||
|
extrabold: "800" as TextStyle["fontWeight"],
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Line Heights
|
||||||
|
*/
|
||||||
|
export const lineHeight = {
|
||||||
|
tight: 1.15,
|
||||||
|
normal: 1.4,
|
||||||
|
relaxed: 1.6,
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Letter Spacing
|
||||||
|
*/
|
||||||
|
export const letterSpacing = {
|
||||||
|
tight: -1,
|
||||||
|
normal: 0,
|
||||||
|
wide: 0.5,
|
||||||
|
wider: 1.5,
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Typography Presets
|
||||||
|
* Ready-to-use text styles for common use cases
|
||||||
|
*/
|
||||||
|
export interface TypographyPresets {
|
||||||
|
h1: TextStyle;
|
||||||
|
h2: TextStyle;
|
||||||
|
h3: TextStyle;
|
||||||
|
h4: TextStyle;
|
||||||
|
body: TextStyle;
|
||||||
|
bodyEmphasis: TextStyle;
|
||||||
|
label: TextStyle;
|
||||||
|
stat: TextStyle;
|
||||||
|
statLarge: TextStyle;
|
||||||
|
caption: TextStyle;
|
||||||
|
button: TextStyle;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const createTypographyPresets = (
|
||||||
|
textPrimary: string,
|
||||||
|
textSecondary: string,
|
||||||
|
textTertiary: string,
|
||||||
|
): TypographyPresets => ({
|
||||||
|
// Display Text (Screen Titles) - Extra Bold
|
||||||
|
h1: {
|
||||||
|
fontSize: fontSize["4xl"],
|
||||||
|
fontWeight: fontWeight.extrabold,
|
||||||
|
letterSpacing: letterSpacing.tight,
|
||||||
|
lineHeight: fontSize["4xl"] * lineHeight.tight,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Section Headers - Bold
|
||||||
|
h2: {
|
||||||
|
fontSize: fontSize["2xl"],
|
||||||
|
fontWeight: fontWeight.bold,
|
||||||
|
letterSpacing: -0.5,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Card Titles - Semibold
|
||||||
|
h3: {
|
||||||
|
fontSize: fontSize.lg,
|
||||||
|
fontWeight: fontWeight.semibold,
|
||||||
|
letterSpacing: -0.3,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Small Headers
|
||||||
|
h4: {
|
||||||
|
fontSize: fontSize.md,
|
||||||
|
fontWeight: fontWeight.semibold,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Body Text
|
||||||
|
body: {
|
||||||
|
fontSize: fontSize.base,
|
||||||
|
fontWeight: fontWeight.regular,
|
||||||
|
lineHeight: fontSize.base * lineHeight.normal,
|
||||||
|
color: textSecondary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Emphasized Body
|
||||||
|
bodyEmphasis: {
|
||||||
|
fontSize: fontSize.md,
|
||||||
|
fontWeight: fontWeight.medium,
|
||||||
|
lineHeight: fontSize.md * lineHeight.normal,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Labels (uppercase, spaced)
|
||||||
|
label: {
|
||||||
|
fontSize: fontSize.xs,
|
||||||
|
fontWeight: fontWeight.semibold,
|
||||||
|
letterSpacing: letterSpacing.wider,
|
||||||
|
textTransform: "uppercase",
|
||||||
|
color: textTertiary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Stats/Numbers - Bold Large
|
||||||
|
stat: {
|
||||||
|
fontSize: fontSize["3xl"],
|
||||||
|
fontWeight: fontWeight.bold,
|
||||||
|
letterSpacing: -1,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Large Stats (Hero numbers)
|
||||||
|
statLarge: {
|
||||||
|
fontSize: fontSize["5xl"],
|
||||||
|
fontWeight: fontWeight.extrabold,
|
||||||
|
letterSpacing: -2,
|
||||||
|
lineHeight: fontSize["5xl"] * lineHeight.tight,
|
||||||
|
color: textPrimary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Caption/Small text
|
||||||
|
caption: {
|
||||||
|
fontSize: fontSize.xs,
|
||||||
|
fontWeight: fontWeight.regular,
|
||||||
|
color: textTertiary,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Button Text
|
||||||
|
button: {
|
||||||
|
fontSize: fontSize.base,
|
||||||
|
fontWeight: fontWeight.bold,
|
||||||
|
letterSpacing: 0.5,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Typography utility object
|
||||||
|
*/
|
||||||
|
export const typography = {
|
||||||
|
fontSize,
|
||||||
|
fontWeight,
|
||||||
|
lineHeight,
|
||||||
|
letterSpacing,
|
||||||
|
};
|
||||||
220
apps/mobile/src/utils/activityFeed.ts
Normal file
220
apps/mobile/src/utils/activityFeed.ts
Normal file
@ -0,0 +1,220 @@
|
|||||||
|
/**
|
||||||
|
* Activity Feed Utilities
|
||||||
|
*
|
||||||
|
* Combines multiple data sources (check-ins, goals, nutrition) into a unified activity feed
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type { AttendanceStatistics } from "../api/types";
|
||||||
|
import type { FitnessGoal } from "../services/fitnessGoals";
|
||||||
|
|
||||||
|
export enum ActivityType {
|
||||||
|
GYM_CHECKIN = "gym_checkin",
|
||||||
|
GOAL_COMPLETED = "goal_completed",
|
||||||
|
GOAL_CREATED = "goal_created",
|
||||||
|
NUTRITION_MILESTONE = "nutrition_milestone",
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ActivityItem {
|
||||||
|
id: string;
|
||||||
|
type: ActivityType;
|
||||||
|
title: string;
|
||||||
|
timestamp: Date;
|
||||||
|
duration?: number; // in minutes
|
||||||
|
metadata?: Record<string, any>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert attendance check-ins to activity items
|
||||||
|
*/
|
||||||
|
export function checkInsToActivities(
|
||||||
|
attendance: AttendanceStatistics,
|
||||||
|
): ActivityItem[] {
|
||||||
|
return attendance.recentCheckIns.map((checkIn) => {
|
||||||
|
const checkInDate = new Date(checkIn.checkInTime);
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: `checkin-${checkIn.id}`,
|
||||||
|
type: ActivityType.GYM_CHECKIN,
|
||||||
|
title: "Gym Check-in",
|
||||||
|
timestamp: checkInDate,
|
||||||
|
duration: checkIn.duration || undefined,
|
||||||
|
metadata: {
|
||||||
|
checkOutTime: checkIn.checkOutTime,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert completed goals to activity items
|
||||||
|
*/
|
||||||
|
export function completedGoalsToActivities(
|
||||||
|
goals: FitnessGoal[],
|
||||||
|
): ActivityItem[] {
|
||||||
|
return goals
|
||||||
|
.filter((goal) => goal.status === "completed" && goal.completedDate)
|
||||||
|
.map((goal) => {
|
||||||
|
const completedDate = new Date(goal.completedDate!);
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: `goal-completed-${goal.id}`,
|
||||||
|
type: ActivityType.GOAL_COMPLETED,
|
||||||
|
title: goal.title,
|
||||||
|
timestamp: completedDate,
|
||||||
|
metadata: {
|
||||||
|
goalType: goal.goalType,
|
||||||
|
priority: goal.priority,
|
||||||
|
targetValue: goal.targetValue,
|
||||||
|
unit: goal.unit,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert recently created goals to activity items
|
||||||
|
*/
|
||||||
|
export function newGoalsToActivities(
|
||||||
|
goals: FitnessGoal[],
|
||||||
|
daysBack: number = 7,
|
||||||
|
): ActivityItem[] {
|
||||||
|
const cutoffDate = new Date();
|
||||||
|
cutoffDate.setDate(cutoffDate.getDate() - daysBack);
|
||||||
|
|
||||||
|
return goals
|
||||||
|
.filter((goal) => {
|
||||||
|
const createdDate = new Date(goal.createdAt);
|
||||||
|
return goal.status === "active" && createdDate > cutoffDate;
|
||||||
|
})
|
||||||
|
.map((goal) => {
|
||||||
|
const createdDate = new Date(goal.createdAt);
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: `goal-created-${goal.id}`,
|
||||||
|
type: ActivityType.GOAL_CREATED,
|
||||||
|
title: goal.title,
|
||||||
|
timestamp: createdDate,
|
||||||
|
metadata: {
|
||||||
|
goalType: goal.goalType,
|
||||||
|
priority: goal.priority,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Combine and sort all activities by timestamp (most recent first)
|
||||||
|
*/
|
||||||
|
export function combineActivities(
|
||||||
|
...activityGroups: ActivityItem[][]
|
||||||
|
): ActivityItem[] {
|
||||||
|
const allActivities = activityGroups.flat();
|
||||||
|
|
||||||
|
return allActivities.sort(
|
||||||
|
(a, b) => b.timestamp.getTime() - a.timestamp.getTime(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get recent activities limited to a specific count
|
||||||
|
*/
|
||||||
|
export function getRecentActivities(
|
||||||
|
activities: ActivityItem[],
|
||||||
|
limit: number = 5,
|
||||||
|
): ActivityItem[] {
|
||||||
|
return activities.slice(0, limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Format activity timestamp for display
|
||||||
|
*/
|
||||||
|
export function formatActivityTime(timestamp: Date): string {
|
||||||
|
const now = new Date();
|
||||||
|
const diffMs = now.getTime() - timestamp.getTime();
|
||||||
|
const diffMins = Math.floor(diffMs / 60000);
|
||||||
|
const diffHours = Math.floor(diffMs / 3600000);
|
||||||
|
const diffDays = Math.floor(diffMs / 86400000);
|
||||||
|
|
||||||
|
// Today
|
||||||
|
if (diffDays === 0) {
|
||||||
|
if (diffMins < 60) {
|
||||||
|
return diffMins <= 1 ? "Just now" : `${diffMins}m ago`;
|
||||||
|
}
|
||||||
|
return `Today, ${timestamp.toLocaleTimeString([], {
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
})}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Yesterday
|
||||||
|
if (diffDays === 1) {
|
||||||
|
return `Yesterday, ${timestamp.toLocaleTimeString([], {
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
})}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This week (within 7 days)
|
||||||
|
if (diffDays < 7) {
|
||||||
|
return `${diffDays}d ago`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Older
|
||||||
|
return timestamp.toLocaleDateString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Format duration in minutes to human-readable string
|
||||||
|
*/
|
||||||
|
export function formatDuration(minutes?: number): string | undefined {
|
||||||
|
if (!minutes) return undefined;
|
||||||
|
|
||||||
|
if (minutes < 60) {
|
||||||
|
return `${minutes}m`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const hours = Math.floor(minutes / 60);
|
||||||
|
const mins = minutes % 60;
|
||||||
|
|
||||||
|
if (mins === 0) {
|
||||||
|
return `${hours}h`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${hours}h ${mins}m`;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get icon name for activity type
|
||||||
|
*/
|
||||||
|
export function getActivityIcon(type: ActivityType): string {
|
||||||
|
switch (type) {
|
||||||
|
case ActivityType.GYM_CHECKIN:
|
||||||
|
return "barbell";
|
||||||
|
case ActivityType.GOAL_COMPLETED:
|
||||||
|
return "trophy";
|
||||||
|
case ActivityType.GOAL_CREATED:
|
||||||
|
return "flag";
|
||||||
|
case ActivityType.NUTRITION_MILESTONE:
|
||||||
|
return "restaurant";
|
||||||
|
default:
|
||||||
|
return "checkmark-circle";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get activity title prefix/emoji
|
||||||
|
*/
|
||||||
|
export function getActivityEmoji(type: ActivityType): string {
|
||||||
|
switch (type) {
|
||||||
|
case ActivityType.GYM_CHECKIN:
|
||||||
|
return "💪";
|
||||||
|
case ActivityType.GOAL_COMPLETED:
|
||||||
|
return "🏆";
|
||||||
|
case ActivityType.GOAL_CREATED:
|
||||||
|
return "🎯";
|
||||||
|
case ActivityType.NUTRITION_MILESTONE:
|
||||||
|
return "🍽️";
|
||||||
|
default:
|
||||||
|
return "✅";
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue
Block a user