Compare commits

...

4 Commits

Author SHA1 Message Date
4c2e97b66d Merge branch 'screen1' 2026-03-31 17:05:43 +02:00
21afb085e3 scan food 2026-03-31 16:55:55 +02:00
ca64a100b6 integrate barcode scan with openfoodfacts and meal type selection 2026-03-31 16:36:18 +02:00
3c3dfb6cd6 hydration calories persistance 2026-03-31 16:17:08 +02:00
7 changed files with 1083 additions and 472 deletions

Binary file not shown.

View File

@ -0,0 +1,156 @@
import { NextResponse } from "next/server";
import { auth } from "@clerk/nextjs/server";
import { getUserMembershipContext } from "@/lib/membership/access";
import log from "@/lib/logger";
interface OpenFoodFactsProduct {
product_name?: string;
product_name_en?: string;
brands?: string;
image_url?: string;
image_front_url?: string;
serving_size?: string;
nutriments?: {
[key: string]: number | string | undefined;
"energy-kcal_serving"?: number;
"energy-kcal_100g"?: number;
proteins_serving?: number;
proteins_100g?: number;
carbohydrates_serving?: number;
carbohydrates_100g?: number;
fat_serving?: number;
fat_100g?: number;
};
}
interface OpenFoodFactsResponse {
status: number;
code: string;
product?: OpenFoodFactsProduct;
}
function normalizeBarcode(rawCode: string): string {
return rawCode.replace(/\D/g, "").trim();
}
function isSupportedBarcode(code: string): boolean {
return [8, 12, 13].includes(code.length);
}
function getNumber(value: unknown): number | undefined {
if (typeof value === "number" && Number.isFinite(value)) return value;
if (typeof value === "string") {
const parsed = Number(value);
if (Number.isFinite(parsed)) return parsed;
}
return undefined;
}
function buildProductPayload(code: string, product: OpenFoodFactsProduct) {
const caloriesPerServing =
getNumber(product.nutriments?.["energy-kcal_serving"]) ??
getNumber(product.nutriments?.["energy-kcal_100g"]) ??
0;
const protein =
getNumber(product.nutriments?.proteins_serving) ??
getNumber(product.nutriments?.proteins_100g);
const carbs =
getNumber(product.nutriments?.carbohydrates_serving) ??
getNumber(product.nutriments?.carbohydrates_100g);
const fat =
getNumber(product.nutriments?.fat_serving) ??
getNumber(product.nutriments?.fat_100g);
return {
barcode: code,
name: product.product_name || product.product_name_en || "Unknown Product",
brand: product.brands || null,
imageUrl: product.image_url || product.image_front_url || null,
servingSize: product.serving_size || "1 serving",
caloriesPerServing: Math.max(0, Math.round(caloriesPerServing)),
macros: {
protein: protein ?? null,
carbs: carbs ?? null,
fat: fat ?? null,
},
source: "openfoodfacts" as const,
};
}
export async function GET(
_request: Request,
{ params }: { params: Promise<{ code: string }> },
) {
try {
const { userId } = await auth();
if (!userId) {
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
}
const { features, membershipType } = await getUserMembershipContext(userId);
if (!features.nutritionTracking) {
return NextResponse.json(
{
error:
"Barcode food scan is available on Premium and VIP memberships",
membershipType,
},
{ status: 403 },
);
}
const { code: rawCode } = await params;
const code = normalizeBarcode(rawCode);
if (!isSupportedBarcode(code)) {
return NextResponse.json(
{ error: "Invalid barcode. Use EAN-8, UPC-A, or EAN-13 formats." },
{ status: 400 },
);
}
const response = await fetch(
`https://world.openfoodfacts.org/api/v2/product/${code}.json`,
{
headers: {
"User-Agent": "FitAI/1.0 (fitai.app)",
},
cache: "no-store",
},
);
if (!response.ok) {
log.warn("OpenFoodFacts lookup failed", {
status: response.status,
barcode: code,
});
return NextResponse.json(
{ error: "Food lookup service unavailable. Please try again." },
{ status: 503 },
);
}
const payload = (await response.json()) as OpenFoodFactsResponse;
if (payload.status !== 1 || !payload.product) {
return NextResponse.json(
{ error: "Product not found in OpenFoodFacts" },
{ status: 404 },
);
}
return NextResponse.json({
success: true,
data: buildProductPayload(code, payload.product),
meta: {
timestamp: new Date().toISOString(),
},
});
} catch (error) {
log.error("Failed barcode food lookup", error);
return NextResponse.json(
{ error: "Failed to lookup food barcode" },
{ status: 500 },
);
}
}

View File

@ -0,0 +1,35 @@
import { apiClient, withAuth } from "./client";
import { API_ENDPOINTS } from "../config/api";
export interface ScannedFoodProduct {
barcode: string;
name: string;
brand: string | null;
imageUrl: string | null;
servingSize: string;
caloriesPerServing: number;
macros: {
protein: number | null;
carbs: number | null;
fat: number | null;
};
source: "openfoodfacts";
}
interface FoodLookupResponse {
success: boolean;
data: ScannedFoodProduct;
}
export async function lookupFoodByBarcode(
barcode: string,
token: string | null,
): Promise<ScannedFoodProduct> {
const normalized = barcode.replace(/\D/g, "");
const response = await apiClient.get<FoodLookupResponse>(
API_ENDPOINTS.FOOD.LOOKUP_BARCODE(normalized),
withAuth(token),
);
return response.data.data;
}

View File

@ -15,4 +15,5 @@ export * from "./hydration";
export * from "./client"; export * from "./client";
export * from "./helpers"; export * from "./helpers";
export * from "./membership"; export * from "./membership";
export * from "./food";
export * from "./gyms"; export * from "./gyms";

View File

@ -8,6 +8,7 @@ import {
Animated, Animated,
TouchableOpacity, TouchableOpacity,
Alert, Alert,
AppState,
} from "react-native"; } from "react-native";
import { useUser } from "@clerk/clerk-expo"; import { useUser } from "@clerk/clerk-expo";
import { useState, useCallback, useEffect, useRef, useMemo } from "react"; import { useState, useCallback, useEffect, useRef, useMemo } from "react";
@ -43,6 +44,22 @@ const CALORIE_GOAL = 2000;
const WATER_GOAL = 2000; const WATER_GOAL = 2000;
const WORKOUT_GOAL = 3; const WORKOUT_GOAL = 3;
const MOTIVATION_KEY_PREFIX = "home-motivation"; const MOTIVATION_KEY_PREFIX = "home-motivation";
const HOME_METRICS_KEY_PREFIX = "home-metrics";
const getLocalDateKey = () => {
const now = new Date();
const year = now.getFullYear();
const month = String(now.getMonth() + 1).padStart(2, "0");
const day = String(now.getDate()).padStart(2, "0");
return `${year}-${month}-${day}`;
};
const getMillisecondsUntilNextMidnight = () => {
const now = new Date();
const nextMidnight = new Date(now);
nextMidnight.setHours(24, 0, 0, 0);
return Math.max(1000, nextMidnight.getTime() - now.getTime());
};
const getRandomMotivation = () => { const getRandomMotivation = () => {
const messages = [ const messages = [
@ -74,12 +91,102 @@ export default function HomeScreen() {
const caloriesBounce = useRef(new Animated.Value(1)).current; const caloriesBounce = useRef(new Animated.Value(1)).current;
const waterBounce = useRef(new Animated.Value(1)).current; const waterBounce = useRef(new Animated.Value(1)).current;
const caloriesRef = useRef(0);
const waterRef = useRef(0);
const currentDateRef = useRef(getLocalDateKey());
const midnightResetTimerRef = useRef<ReturnType<typeof setTimeout> | null>(
null,
);
useEffect(() => {
caloriesRef.current = calories;
}, [calories]);
useEffect(() => {
waterRef.current = waterIntake;
}, [waterIntake]);
const getMetricsStorageKey = useCallback(
() => `${HOME_METRICS_KEY_PREFIX}_${user?.id || "guest"}`,
[user?.id],
);
const persistDailyMetrics = useCallback(
async (nextCalories: number, nextWaterIntake: number, dateKey?: string) => {
if (!user?.id) return;
const targetDate = dateKey || currentDateRef.current;
await AsyncStorage.setItem(
getMetricsStorageKey(),
JSON.stringify({
date: targetDate,
calories: nextCalories,
waterIntake: nextWaterIntake,
}),
);
},
[getMetricsStorageKey, user?.id],
);
const reconcileDailyMetrics = useCallback(async () => {
const today = getLocalDateKey();
currentDateRef.current = today;
if (!user?.id) {
setCalories(0);
setWaterIntake(0);
return;
}
const stored = await AsyncStorage.getItem(getMetricsStorageKey());
if (!stored) {
setCalories(0);
setWaterIntake(0);
await persistDailyMetrics(0, 0, today);
return;
}
try {
const parsed = JSON.parse(stored) as {
date?: string;
calories?: number;
waterIntake?: number;
};
if (parsed.date === today) {
const nextCalories = Number(parsed.calories) || 0;
const nextWater = Number(parsed.waterIntake) || 0;
setCalories(nextCalories);
setWaterIntake(nextWater);
} else {
setCalories(0);
setWaterIntake(0);
await persistDailyMetrics(0, 0, today);
}
} catch {
setCalories(0);
setWaterIntake(0);
await persistDailyMetrics(0, 0, today);
}
}, [getMetricsStorageKey, persistDailyMetrics, user?.id]);
const scheduleMidnightReset = useCallback(() => {
if (midnightResetTimerRef.current) {
clearTimeout(midnightResetTimerRef.current);
}
midnightResetTimerRef.current = setTimeout(() => {
void reconcileDailyMetrics();
scheduleMidnightReset();
}, getMillisecondsUntilNextMidnight() + 50);
}, [reconcileDailyMetrics]);
useFocusEffect( useFocusEffect(
useCallback(() => { useCallback(() => {
void reconcileDailyMetrics();
refetchStatistics(); refetchStatistics();
refetchGoals(); refetchGoals();
}, [refetchStatistics, refetchGoals]), }, [reconcileDailyMetrics, refetchStatistics, refetchGoals]),
); );
const onRefresh = useCallback(async () => { const onRefresh = useCallback(async () => {
@ -100,7 +207,11 @@ export default function HomeScreen() {
name: string; name: string;
calories: number; calories: number;
}) => { }) => {
setCalories((prev) => prev + meal.calories); setCalories((prev) => {
const next = prev + meal.calories;
void persistDailyMetrics(next, waterRef.current);
return next;
});
setTrackMealModalVisible(false); setTrackMealModalVisible(false);
Animated.sequence([ Animated.sequence([
Animated.timing(caloriesBounce, { Animated.timing(caloriesBounce, {
@ -117,7 +228,11 @@ export default function HomeScreen() {
}; };
const handleAddWater = (amount: number) => { const handleAddWater = (amount: number) => {
setWaterIntake((prev) => prev + amount); setWaterIntake((prev) => {
const next = prev + amount;
void persistDailyMetrics(caloriesRef.current, next);
return next;
});
setAddWaterModalVisible(false); setAddWaterModalVisible(false);
Animated.sequence([ Animated.sequence([
Animated.timing(waterBounce, { Animated.timing(waterBounce, {
@ -133,20 +248,23 @@ export default function HomeScreen() {
]).start(); ]).start();
}; };
const handleResetCalories = () => setCalories(0); const handleResetCalories = () => {
const handleResetWater = () => setWaterIntake(0); setCalories(0);
const handleAddScannedFood = (scannedCalories: number) => { void persistDailyMetrics(0, waterRef.current);
setCalories((prev) => prev + scannedCalories);
setScanFoodModalVisible(false);
}; };
const resetAllCounters = async () => { const handleResetWater = () => {
setCalories(0);
setWaterIntake(0); setWaterIntake(0);
const today = new Date().toDateString(); void persistDailyMetrics(caloriesRef.current, 0);
await AsyncStorage.setItem("lastResetDate", today); };
await AsyncStorage.removeItem(`calories_${today}`);
await AsyncStorage.removeItem(`water_${today}`); const handleAddScannedFood = (scannedCalories: number) => {
setCalories((prev) => {
const next = prev + scannedCalories;
void persistDailyMetrics(next, waterRef.current);
return next;
});
setScanFoodModalVisible(false);
}; };
useEffect(() => { useEffect(() => {
@ -183,42 +301,28 @@ export default function HomeScreen() {
}, [user?.id]); }, [user?.id]);
useEffect(() => { useEffect(() => {
const loadPersistedData = async () => { void reconcileDailyMetrics();
const today = new Date().toDateString(); }, [reconcileDailyMetrics]);
const storedCalories = await AsyncStorage.getItem(`calories_${today}`);
const storedWater = await AsyncStorage.getItem(`water_${today}`);
if (storedCalories) setCalories(parseInt(storedCalories, 10));
if (storedWater) setWaterIntake(parseInt(storedWater, 10));
};
loadPersistedData();
}, []);
useEffect(() => { useEffect(() => {
const persistCalories = async () => { const appStateSubscription = AppState.addEventListener(
const today = new Date().toDateString(); "change",
await AsyncStorage.setItem(`calories_${today}`, calories.toString()); (state) => {
}; if (state === "active") {
persistCalories(); void reconcileDailyMetrics();
}, [calories]); }
},
);
useEffect(() => { scheduleMidnightReset();
const persistWater = async () => {
const today = new Date().toDateString();
await AsyncStorage.setItem(`water_${today}`, waterIntake.toString());
};
persistWater();
}, [waterIntake]);
useEffect(() => { return () => {
const checkAndResetIfNeeded = async () => { appStateSubscription.remove();
const lastResetDate = await AsyncStorage.getItem("lastResetDate"); if (midnightResetTimerRef.current) {
const today = new Date().toDateString(); clearTimeout(midnightResetTimerRef.current);
if (lastResetDate !== today) {
await resetAllCounters();
} }
}; };
checkAndResetIfNeeded(); }, [reconcileDailyMetrics, scheduleMidnightReset]);
}, []);
const checkInsThisWeek = statistics?.attendance.checkInsThisWeek || 0; const checkInsThisWeek = statistics?.attendance.checkInsThisWeek || 0;
const currentStreak = statistics?.attendance.currentStreak || 0; const currentStreak = statistics?.attendance.currentStreak || 0;

View File

@ -1,9 +1,24 @@
import React, { useState, useEffect } from 'react'; import React, { useEffect, useRef, useState } from "react";
import { View, Text, StyleSheet, Modal, TouchableOpacity, TextInput, Alert } from 'react-native'; import {
import { CameraView, useCameraPermissions } from 'expo-camera'; ActivityIndicator,
import { Ionicons } from '@expo/vector-icons'; Alert,
import { LinearGradient } from 'expo-linear-gradient'; KeyboardAvoidingView,
import { theme } from '../styles/theme'; Modal,
Platform,
ScrollView,
StyleSheet,
Text,
TextInput,
TouchableOpacity,
View,
} from "react-native";
import { CameraView, useCameraPermissions } from "expo-camera";
import { useAuth } from "@clerk/clerk-expo";
import { Ionicons } from "@expo/vector-icons";
import { LinearGradient } from "expo-linear-gradient";
import { addMealEntry } from "../api/nutrition";
import { lookupFoodByBarcode, type ScannedFoodProduct } from "../api/food";
import { theme } from "../styles/theme";
interface ScanFoodModalProps { interface ScanFoodModalProps {
visible: boolean; visible: boolean;
@ -11,62 +26,133 @@ interface ScanFoodModalProps {
onAddFood: (calories: number) => void; onAddFood: (calories: number) => void;
} }
// Mock food database const MEAL_TYPES = ["breakfast", "lunch", "dinner", "snack"] as const;
const FOOD_DATABASE: { [key: string]: { name: string; calories: number; servingSize: string } } = { type MealType = (typeof MEAL_TYPES)[number];
'0123456789': { name: 'Apple', calories: 95, servingSize: '1 medium' },
'9876543210': { name: 'Banana', calories: 105, servingSize: '1 medium' },
'5555555555': { name: 'Protein Bar', calories: 200, servingSize: '1 bar' },
'1111111111': { name: 'Greek Yogurt', calories: 150, servingSize: '1 cup' },
};
export function ScanFoodModal({ visible, onClose, onAddFood }: ScanFoodModalProps) { export function ScanFoodModal({
visible,
onClose,
onAddFood,
}: ScanFoodModalProps) {
const [permission, requestPermission] = useCameraPermissions(); const [permission, requestPermission] = useCameraPermissions();
const { getToken } = useAuth();
const [scanned, setScanned] = useState(false); const [scanned, setScanned] = useState(false);
const [foodData, setFoodData] = useState<{ name: string; calories: number; servingSize: string } | null>(null); const [isLookingUp, setIsLookingUp] = useState(false);
const [servings, setServings] = useState('1'); const [foodData, setFoodData] = useState<ScannedFoodProduct | null>(null);
const [notFound, setNotFound] = useState(false);
const [manualFoodName, setManualFoodName] = useState("Manual food entry");
const [manualCalories, setManualCalories] = useState("");
const [servings, setServings] = useState("1");
const [mealType, setMealType] = useState<MealType>("snack");
const lastScannedRef = useRef<{ code: string; ts: number } | null>(null);
useEffect(() => { useEffect(() => {
if (visible) { if (!visible) return;
setScanned(false); setScanned(false);
setIsLookingUp(false);
setFoodData(null); setFoodData(null);
setServings('1'); setNotFound(false);
} setManualFoodName("Manual food entry");
setManualCalories("");
setServings("1");
setMealType("snack");
}, [visible]); }, [visible]);
const handleBarCodeScanned = ({ data }: { data: string }) => { const handleReset = () => {
setScanned(false);
setIsLookingUp(false);
setFoodData(null);
setNotFound(false);
setManualFoodName("Manual food entry");
setManualCalories("");
setServings("1");
setMealType("snack");
};
const handleBarCodeScanned = async ({ data }: { data: string }) => {
if (scanned) return; if (scanned) return;
const normalized = data.replace(/\D/g, "");
const now = Date.now();
if (
lastScannedRef.current &&
lastScannedRef.current.code === normalized &&
now - lastScannedRef.current.ts < 2500
) {
return;
}
lastScannedRef.current = { code: normalized, ts: now };
setScanned(true); setScanned(true);
setIsLookingUp(true);
setNotFound(false);
setFoodData(null);
// Look up food in database try {
const food = FOOD_DATABASE[data]; const token = await getToken();
const product = await lookupFoodByBarcode(normalized, token);
if (food) { setFoodData(product);
setFoodData(food); } catch (error: any) {
const status = error?.response?.status;
if (status === 404) {
setNotFound(true);
} else if (status === 403) {
Alert.alert(
"Premium Feature",
"Barcode food scanning is available on Premium and VIP memberships.",
);
onClose();
} else { } else {
// Mock data for unknown barcodes Alert.alert("Scan Failed", "Unable to lookup this barcode right now.");
setFoodData({ setScanned(false);
name: 'Unknown Food', }
calories: 150, } finally {
servingSize: '1 serving' setIsLookingUp(false);
});
} }
}; };
const handleConfirm = () => { const getTotalCalories = () => {
if (foodData) { if (foodData) {
const totalCalories = foodData.calories * parseFloat(servings || '1'); return Math.round(
foodData.caloriesPerServing * parseFloat(servings || "1"),
);
}
const manual = parseInt(manualCalories || "0", 10);
return Number.isFinite(manual) && manual > 0 ? manual : 0;
};
const handleConfirm = async () => {
const totalCalories = getTotalCalories();
if (totalCalories <= 0) {
Alert.alert("Invalid calories", "Please enter a valid calorie amount.");
return;
}
try {
const token = await getToken();
await addMealEntry(
{
mealType,
foodName: foodData?.name || manualFoodName,
calories: totalCalories,
protein: foodData?.macros.protein ?? undefined,
carbs: foodData?.macros.carbs ?? undefined,
fats: foodData?.macros.fat ?? undefined,
},
token,
);
onAddFood(totalCalories); onAddFood(totalCalories);
onClose(); onClose();
} catch {
Alert.alert(
"Could not save",
"Failed to add scanned food to nutrition diary.",
);
} }
}; };
const handleRescan = () => {
setScanned(false);
setFoodData(null);
setServings('1');
};
if (!permission) { if (!permission) {
return null; return null;
} }
@ -76,17 +162,28 @@ export function ScanFoodModal({ visible, onClose, onAddFood }: ScanFoodModalProp
<Modal visible={visible} transparent animationType="slide"> <Modal visible={visible} transparent animationType="slide">
<View style={styles.permissionContainer}> <View style={styles.permissionContainer}>
<View style={styles.permissionContent}> <View style={styles.permissionContent}>
<Ionicons name="camera-outline" size={64} color={theme.colors.primary} /> <Ionicons
<Text style={styles.permissionTitle}>Camera Permission Required</Text> name="camera-outline"
size={64}
color={theme.colors.primary}
/>
<Text style={styles.permissionTitle}>
Camera Permission Required
</Text>
<Text style={styles.permissionText}> <Text style={styles.permissionText}>
We need access to your camera to scan food barcodes. We need access to your camera to scan food barcodes.
</Text> </Text>
<TouchableOpacity onPress={requestPermission} style={styles.permissionButtonContainer}> <TouchableOpacity
onPress={requestPermission}
style={styles.permissionButtonContainer}
>
<LinearGradient <LinearGradient
colors={theme.gradients.primary} colors={theme.gradients.primary}
style={styles.permissionButton} style={styles.permissionButton}
> >
<Text style={styles.permissionButtonText}>Grant Permission</Text> <Text style={styles.permissionButtonText}>
Grant Permission
</Text>
</LinearGradient> </LinearGradient>
</TouchableOpacity> </TouchableOpacity>
<TouchableOpacity onPress={onClose} style={styles.cancelButton}> <TouchableOpacity onPress={onClose} style={styles.cancelButton}>
@ -98,10 +195,12 @@ export function ScanFoodModal({ visible, onClose, onAddFood }: ScanFoodModalProp
); );
} }
const showScanner = !foodData && !notFound;
return ( return (
<Modal visible={visible} animationType="slide"> <Modal visible={visible} animationType="slide">
<View style={styles.container}> <View style={styles.container}>
{!foodData ? ( {showScanner ? (
<> <>
<View style={styles.header}> <View style={styles.header}>
<TouchableOpacity onPress={onClose} style={styles.closeButton}> <TouchableOpacity onPress={onClose} style={styles.closeButton}>
@ -116,51 +215,108 @@ export function ScanFoodModal({ visible, onClose, onAddFood }: ScanFoodModalProp
facing="back" facing="back"
onBarcodeScanned={scanned ? undefined : handleBarCodeScanned} onBarcodeScanned={scanned ? undefined : handleBarCodeScanned}
barcodeScannerSettings={{ barcodeScannerSettings={{
barcodeTypes: ['ean13', 'ean8', 'upc_a', 'upc_e', 'code128', 'code39'], barcodeTypes: [
"ean13",
"ean8",
"upc_a",
"upc_e",
"code128",
"code39",
],
}} }}
> >
<View style={styles.scanOverlay}> <View style={styles.scanOverlay}>
<View style={styles.scanFrame} /> <View style={styles.scanFrame} />
<Text style={styles.scanText}>Position barcode within frame</Text> {isLookingUp ? (
<View style={styles.lookupContainer}>
<ActivityIndicator color="#fff" />
<Text style={styles.scanText}>Looking up product...</Text>
</View>
) : (
<Text style={styles.scanText}>
Position barcode within frame
</Text>
)}
</View> </View>
</CameraView> </CameraView>
</> </>
) : ( ) : (
<View style={styles.resultContainer}> <KeyboardAvoidingView
behavior={Platform.OS === "ios" ? "padding" : "height"}
style={styles.resultContainer}
>
<View style={styles.resultSheet}>
<View style={styles.resultHeader}> <View style={styles.resultHeader}>
<TouchableOpacity onPress={onClose} style={styles.closeButton}> <TouchableOpacity
<Ionicons name="close" size={28} color={theme.colors.gray900} /> onPress={handleReset}
style={styles.closeButton}
>
<Ionicons
name="scan"
size={24}
color={theme.colors.gray700}
/>
</TouchableOpacity> </TouchableOpacity>
<Text style={styles.resultTitle}>Food Details</Text> <Text style={styles.resultTitle}>
<View style={{ width: 28 }} /> {notFound ? "Barcode Not Found" : "Food Details"}
</Text>
<View style={{ width: 24 }} />
</View> </View>
<ScrollView
style={styles.resultScroll}
contentContainerStyle={styles.resultScrollContent}
showsVerticalScrollIndicator={false}
>
<View style={styles.foodCard}> <View style={styles.foodCard}>
{!notFound && foodData ? (
<>
<View style={styles.foodIconContainer}> <View style={styles.foodIconContainer}>
<LinearGradient <LinearGradient
colors={theme.gradients.success} colors={theme.gradients.success}
style={styles.foodIcon} style={styles.foodIcon}
> >
<Ionicons name="restaurant" size={32} color="#fff" /> <Ionicons name="restaurant" size={28} color="#fff" />
</LinearGradient> </LinearGradient>
</View> </View>
<Text style={styles.foodName}>{foodData.name}</Text> <Text style={styles.foodName}>{foodData.name}</Text>
<Text style={styles.servingSize}>{foodData.servingSize}</Text> <Text style={styles.servingMeta}>
{[foodData.brand, foodData.servingSize]
.filter(Boolean)
.join(" • ")}
</Text>
<View style={styles.caloriesBadge}> <View style={styles.caloriesBadge}>
<Text style={styles.caloriesValue}>{foodData.calories}</Text> <Text style={styles.caloriesValue}>
<Text style={styles.caloriesLabel}>kcal per serving</Text> {foodData.caloriesPerServing}
</Text>
<Text style={styles.caloriesLabel}>
kcal per serving
</Text>
</View> </View>
<View style={styles.servingsContainer}> <View style={styles.servingsContainer}>
<Text style={styles.label}>Number of Servings</Text> <Text style={styles.label}>Servings</Text>
<View style={styles.servingsInput}> <View style={styles.servingsInput}>
<TouchableOpacity <TouchableOpacity
onPress={() => setServings(String(Math.max(0.5, parseFloat(servings || '1') - 0.5)))} onPress={() =>
setServings(
String(
Math.max(
0.5,
parseFloat(servings || "1") - 0.5,
),
),
)
}
style={styles.servingsButton} style={styles.servingsButton}
> >
<Ionicons name="remove" size={20} color={theme.colors.primary} /> <Ionicons
name="remove"
size={20}
color={theme.colors.primary}
/>
</TouchableOpacity> </TouchableOpacity>
<TextInput <TextInput
style={styles.servingsValue} style={styles.servingsValue}
@ -169,37 +325,106 @@ export function ScanFoodModal({ visible, onClose, onAddFood }: ScanFoodModalProp
keyboardType="decimal-pad" keyboardType="decimal-pad"
/> />
<TouchableOpacity <TouchableOpacity
onPress={() => setServings(String(parseFloat(servings || '1') + 0.5))} onPress={() =>
setServings(
String(parseFloat(servings || "1") + 0.5),
)
}
style={styles.servingsButton} style={styles.servingsButton}
> >
<Ionicons name="add" size={20} color={theme.colors.primary} /> <Ionicons
name="add"
size={20}
color={theme.colors.primary}
/>
</TouchableOpacity> </TouchableOpacity>
</View> </View>
</View> </View>
</>
<View style={styles.totalCalories}> ) : (
<Text style={styles.totalLabel}>Total Calories</Text> <>
<Text style={styles.totalValue}> <Text style={styles.servingMeta}>
{Math.round(foodData.calories * parseFloat(servings || '1'))} kcal We could not find this product in OpenFoodFacts.
</Text> </Text>
<Text style={styles.label}>Food Name</Text>
<TextInput
style={styles.input}
value={manualFoodName}
onChangeText={setManualFoodName}
placeholder="Enter food name"
placeholderTextColor={theme.colors.gray400}
/>
<Text style={styles.label}>Calories</Text>
<View style={styles.caloriesInputContainer}>
<TextInput
style={[styles.input, styles.caloriesInput]}
value={manualCalories}
onChangeText={setManualCalories}
keyboardType="number-pad"
placeholder="0"
placeholderTextColor={theme.colors.gray400}
/>
<Text style={styles.caloriesUnit}>kcal</Text>
</View> </View>
</>
)}
<View style={styles.mealTypeContainer}>
<Text style={styles.label}>Meal Type</Text>
<View style={styles.mealTypeRow}>
{MEAL_TYPES.map((type) => {
const active = mealType === type;
return (
<TouchableOpacity
key={type}
onPress={() => setMealType(type)}
style={[
styles.mealTypeChip,
active && styles.mealTypeChipActive,
]}
>
<Text
style={[
styles.mealTypeText,
active && styles.mealTypeTextActive,
]}
>
{type.charAt(0).toUpperCase() + type.slice(1)}
</Text>
</TouchableOpacity>
);
})}
</View>
</View>
</View>
</ScrollView>
<View style={styles.footerSummary}>
<Text style={styles.totalLabel}>Total Calories</Text>
<Text style={styles.totalValue}>{getTotalCalories()} kcal</Text>
</View> </View>
<View style={styles.buttonRow}> <View style={styles.buttonRow}>
<TouchableOpacity onPress={handleRescan} style={styles.rescanButton}> <TouchableOpacity onPress={onClose} style={styles.rejectButton}>
<Text style={styles.rescanButtonText}>Scan Again</Text> <Text style={styles.rejectButtonText}>Reject</Text>
</TouchableOpacity> </TouchableOpacity>
<TouchableOpacity onPress={handleConfirm} style={styles.confirmButtonContainer}> <TouchableOpacity
onPress={handleConfirm}
style={styles.confirmButtonContainer}
>
<LinearGradient <LinearGradient
colors={theme.gradients.success} colors={theme.gradients.success}
style={styles.confirmButton} style={styles.confirmButton}
> >
<Text style={styles.confirmButtonText}>Add to Diary</Text> <Text style={styles.confirmButtonText}>Add to Count</Text>
</LinearGradient> </LinearGradient>
</TouchableOpacity> </TouchableOpacity>
</View> </View>
</View> </View>
</KeyboardAvoidingView>
)} )}
</View> </View>
</Modal> </Modal>
@ -209,106 +434,129 @@ export function ScanFoodModal({ visible, onClose, onAddFood }: ScanFoodModalProp
const styles = StyleSheet.create({ const styles = StyleSheet.create({
container: { container: {
flex: 1, flex: 1,
backgroundColor: '#000', backgroundColor: "#000",
}, },
header: { header: {
flexDirection: 'row', flexDirection: "row",
justifyContent: 'space-between', justifyContent: "space-between",
alignItems: 'center', alignItems: "center",
padding: 20, padding: 20,
paddingTop: 60, paddingTop: 60,
backgroundColor: 'rgba(0, 0, 0, 0.8)', backgroundColor: "rgba(0, 0, 0, 0.8)",
}, },
closeButton: { closeButton: {
padding: 4, padding: 4,
}, },
title: { title: {
fontSize: 18, fontSize: 18,
fontWeight: '700', fontWeight: "700",
color: '#fff', color: "#fff",
}, },
camera: { camera: {
flex: 1, flex: 1,
}, },
scanOverlay: { scanOverlay: {
flex: 1, flex: 1,
justifyContent: 'center', justifyContent: "center",
alignItems: 'center', alignItems: "center",
backgroundColor: 'rgba(0, 0, 0, 0.3)', backgroundColor: "rgba(0, 0, 0, 0.3)",
}, },
scanFrame: { scanFrame: {
width: 250, width: 250,
height: 250, height: 250,
borderWidth: 2, borderWidth: 2,
borderColor: '#fff', borderColor: "#fff",
borderRadius: 20, borderRadius: 20,
backgroundColor: 'transparent', backgroundColor: "transparent",
}, },
scanText: { scanText: {
color: '#fff', color: "#fff",
fontSize: 16, fontSize: 16,
fontWeight: '600', fontWeight: "600",
marginTop: 16,
textAlign: "center",
},
lookupContainer: {
alignItems: "center",
marginTop: 24, marginTop: 24,
textAlign: 'center',
}, },
resultContainer: { resultContainer: {
flex: 1, flex: 1,
backgroundColor: "rgba(0, 0, 0, 0.35)",
justifyContent: "flex-end",
},
resultSheet: {
backgroundColor: theme.colors.background, backgroundColor: theme.colors.background,
borderTopLeftRadius: 24,
borderTopRightRadius: 24,
maxHeight: "80%",
}, },
resultHeader: { resultHeader: {
flexDirection: 'row', flexDirection: "row",
justifyContent: 'space-between', justifyContent: "space-between",
alignItems: 'center', alignItems: "center",
padding: 20, paddingHorizontal: 20,
paddingTop: 60, paddingVertical: 14,
backgroundColor: '#fff', backgroundColor: "#fff",
borderTopLeftRadius: 24,
borderTopRightRadius: 24,
borderBottomWidth: 1,
borderBottomColor: theme.colors.gray100,
},
resultScroll: {
flexGrow: 0,
},
resultScrollContent: {
paddingBottom: 12,
}, },
resultTitle: { resultTitle: {
fontSize: 18, fontSize: 18,
fontWeight: '700', fontWeight: "700",
color: theme.colors.gray900, color: theme.colors.gray900,
}, },
foodCard: { foodCard: {
margin: 20, marginHorizontal: 16,
padding: 24, marginTop: 12,
backgroundColor: '#fff', padding: 16,
borderRadius: 24, backgroundColor: "#fff",
alignItems: 'center', borderRadius: 16,
alignItems: "center",
...theme.shadows.medium, ...theme.shadows.medium,
}, },
foodIconContainer: { foodIconContainer: {
marginBottom: 16, marginBottom: 10,
}, },
foodIcon: { foodIcon: {
width: 80, width: 64,
height: 80, height: 64,
borderRadius: 40, borderRadius: 32,
justifyContent: 'center', justifyContent: "center",
alignItems: 'center', alignItems: "center",
}, },
foodName: { foodName: {
fontSize: 24, fontSize: 20,
fontWeight: '700', fontWeight: "700",
color: theme.colors.gray900, color: theme.colors.gray900,
marginBottom: 8, marginBottom: 4,
textAlign: 'center', textAlign: "center",
}, },
servingSize: { servingMeta: {
fontSize: 16, fontSize: 14,
color: theme.colors.gray600, color: theme.colors.gray600,
marginBottom: 24, marginBottom: 12,
textAlign: "center",
}, },
caloriesBadge: { caloriesBadge: {
backgroundColor: theme.colors.gray50, backgroundColor: theme.colors.gray50,
paddingHorizontal: 24, paddingHorizontal: 18,
paddingVertical: 16, paddingVertical: 10,
borderRadius: 16, borderRadius: 12,
alignItems: 'center', alignItems: "center",
marginBottom: 32, marginBottom: 14,
}, },
caloriesValue: { caloriesValue: {
fontSize: 32, fontSize: 26,
fontWeight: '700', fontWeight: "700",
color: theme.colors.primary, color: theme.colors.primary,
}, },
caloriesLabel: { caloriesLabel: {
@ -317,19 +565,49 @@ const styles = StyleSheet.create({
marginTop: 4, marginTop: 4,
}, },
servingsContainer: { servingsContainer: {
width: '100%', width: "100%",
marginBottom: 24, marginBottom: 14,
},
mealTypeContainer: {
width: "100%",
marginBottom: 6,
},
mealTypeRow: {
flexDirection: "row",
flexWrap: "wrap",
gap: 8,
},
mealTypeChip: {
paddingHorizontal: 14,
paddingVertical: 8,
borderRadius: 18,
backgroundColor: theme.colors.gray100,
borderWidth: 1,
borderColor: theme.colors.gray200,
},
mealTypeChipActive: {
backgroundColor: theme.colors.primary,
borderColor: theme.colors.primary,
},
mealTypeText: {
fontSize: 13,
fontWeight: "600",
color: theme.colors.gray700,
},
mealTypeTextActive: {
color: "#fff",
}, },
label: { label: {
fontSize: 16, fontSize: 16,
fontWeight: '600', fontWeight: "600",
color: theme.colors.gray700, color: theme.colors.gray700,
marginBottom: 12, marginBottom: 12,
alignSelf: "flex-start",
}, },
servingsInput: { servingsInput: {
flexDirection: 'row', flexDirection: "row",
alignItems: 'center', alignItems: "center",
justifyContent: 'center', justifyContent: "center",
gap: 16, gap: 16,
}, },
servingsButton: { servingsButton: {
@ -337,50 +615,84 @@ const styles = StyleSheet.create({
height: 40, height: 40,
borderRadius: 20, borderRadius: 20,
backgroundColor: theme.colors.gray100, backgroundColor: theme.colors.gray100,
justifyContent: 'center', justifyContent: "center",
alignItems: 'center', alignItems: "center",
}, },
servingsValue: { servingsValue: {
fontSize: 24, fontSize: 24,
fontWeight: '700', fontWeight: "700",
color: theme.colors.gray900, color: theme.colors.gray900,
textAlign: 'center', textAlign: "center",
minWidth: 60, minWidth: 60,
}, },
input: {
width: "100%",
backgroundColor: theme.colors.gray50,
borderWidth: 1,
borderColor: theme.colors.gray200,
borderRadius: 16,
padding: 16,
fontSize: 16,
color: theme.colors.gray900,
marginBottom: 16,
},
caloriesInputContainer: {
width: "100%",
flexDirection: "row",
alignItems: "center",
},
caloriesInput: {
flex: 1,
marginBottom: 0,
marginRight: 12,
},
caloriesUnit: {
fontSize: 16,
fontWeight: "600",
color: theme.colors.gray500,
},
totalCalories: { totalCalories: {
width: '100%', display: "none",
flexDirection: 'row', },
justifyContent: 'space-between', footerSummary: {
alignItems: 'center', flexDirection: "row",
paddingTop: 24, justifyContent: "space-between",
alignItems: "center",
paddingHorizontal: 20,
paddingTop: 10,
paddingBottom: 10,
borderTopWidth: 1, borderTopWidth: 1,
borderTopColor: theme.colors.gray200, borderTopColor: theme.colors.gray200,
backgroundColor: "#fff",
}, },
totalLabel: { totalLabel: {
fontSize: 16, fontSize: 16,
fontWeight: '600', fontWeight: "600",
color: theme.colors.gray700, color: theme.colors.gray700,
}, },
totalValue: { totalValue: {
fontSize: 20, fontSize: 20,
fontWeight: '700', fontWeight: "700",
color: theme.colors.gray900, color: theme.colors.gray900,
}, },
buttonRow: { buttonRow: {
flexDirection: 'row', flexDirection: "row",
padding: 20, paddingHorizontal: 20,
paddingTop: 8,
paddingBottom: 16,
gap: 12, gap: 12,
backgroundColor: "#fff",
}, },
rescanButton: { rejectButton: {
flex: 1, flex: 1,
paddingVertical: 16, paddingVertical: 16,
borderRadius: 20, borderRadius: 20,
backgroundColor: theme.colors.gray100, backgroundColor: theme.colors.gray100,
alignItems: 'center', alignItems: "center",
}, },
rescanButtonText: { rejectButtonText: {
fontSize: 16, fontSize: 16,
fontWeight: '700', fontWeight: "700",
color: theme.colors.gray700, color: theme.colors.gray700,
}, },
confirmButtonContainer: { confirmButtonContainer: {
@ -389,30 +701,30 @@ const styles = StyleSheet.create({
confirmButton: { confirmButton: {
paddingVertical: 16, paddingVertical: 16,
borderRadius: 20, borderRadius: 20,
alignItems: 'center', alignItems: "center",
}, },
confirmButtonText: { confirmButtonText: {
fontSize: 16, fontSize: 16,
fontWeight: '700', fontWeight: "700",
color: '#fff', color: "#fff",
}, },
permissionContainer: { permissionContainer: {
flex: 1, flex: 1,
justifyContent: 'center', justifyContent: "center",
alignItems: 'center', alignItems: "center",
backgroundColor: 'rgba(0, 0, 0, 0.5)', backgroundColor: "rgba(0, 0, 0, 0.5)",
}, },
permissionContent: { permissionContent: {
backgroundColor: '#fff', backgroundColor: "#fff",
borderRadius: 24, borderRadius: 24,
padding: 32, padding: 32,
margin: 20, margin: 20,
alignItems: 'center', alignItems: "center",
...theme.shadows.strong, ...theme.shadows.strong,
}, },
permissionTitle: { permissionTitle: {
fontSize: 20, fontSize: 20,
fontWeight: '700', fontWeight: "700",
color: theme.colors.gray900, color: theme.colors.gray900,
marginTop: 16, marginTop: 16,
marginBottom: 8, marginBottom: 8,
@ -420,29 +732,29 @@ const styles = StyleSheet.create({
permissionText: { permissionText: {
fontSize: 16, fontSize: 16,
color: theme.colors.gray600, color: theme.colors.gray600,
textAlign: 'center', textAlign: "center",
marginBottom: 24, marginBottom: 24,
}, },
permissionButtonContainer: { permissionButtonContainer: {
width: '100%', width: "100%",
marginBottom: 12, marginBottom: 12,
}, },
permissionButton: { permissionButton: {
paddingVertical: 16, paddingVertical: 16,
borderRadius: 20, borderRadius: 20,
alignItems: 'center', alignItems: "center",
}, },
permissionButtonText: { permissionButtonText: {
fontSize: 16, fontSize: 16,
fontWeight: '700', fontWeight: "700",
color: '#fff', color: "#fff",
}, },
cancelButton: { cancelButton: {
paddingVertical: 12, paddingVertical: 12,
}, },
cancelButtonText: { cancelButtonText: {
fontSize: 16, fontSize: 16,
fontWeight: '600', fontWeight: "600",
color: theme.colors.gray600, color: theme.colors.gray600,
}, },
}); });

View File

@ -47,6 +47,9 @@ export const API_ENDPOINTS = {
MEMBERSHIP: { MEMBERSHIP: {
FEATURES: "/api/membership/features", FEATURES: "/api/membership/features",
}, },
FOOD: {
LOOKUP_BARCODE: (code: string) => `/api/food/barcode/${code}`,
},
NUTRITION: { NUTRITION: {
BASE: "/api/nutrition", BASE: "/api/nutrition",
MEALS: "/api/nutrition/meals", MEALS: "/api/nutrition/meals",