8 Commits

Author SHA1 Message Date
858ee5e28b Ok it might work 2026-04-25 12:08:14 +02:00
928db54b3e Jkdsjkdjw 2026-04-25 11:21:39 +02:00
72b45335dd What 2026-04-25 11:14:11 +02:00
6f238b4d80 Ok now releasegit add .git add .!
All checks were successful
Build APK / build (push) Successful in 12m22s
2026-04-24 22:51:45 +02:00
adfd3e770c jdksjk
All checks were successful
Build APK / build (push) Successful in 12m13s
2026-04-24 20:03:01 +02:00
5977bd790a Whoops
Some checks failed
Build APK / build (push) Failing after 13m3s
2026-04-24 19:01:50 +02:00
bb304c5bd9 Signature
Some checks failed
Build APK / build (push) Failing after 8m38s
2026-04-24 18:53:00 +02:00
806e1e7f5b CI/CD Works
All checks were successful
Build APK / build (push) Successful in 12m20s
2026-04-24 17:34:05 +02:00
16 changed files with 695 additions and 82 deletions

View File

@@ -0,0 +1,51 @@
name: Build APK
on:
push:
branches: [ master ]
jobs:
build:
runs-on: ubuntu-latest
defaults:
run:
shell: bash
working-directory: ./apk
steps:
- name: 📥 Checkout
uses: actions/checkout@v4
- name: 🟢 Setup Node
uses: actions/setup-node@v4
with:
node-version: 20
- name: 📦 Install dependencies
run: npm install
- name: ☕ Setup Java
uses: actions/setup-java@v4
with:
distribution: temurin
java-version: 17
- name: 📱 Setup Android SDK
uses: android-actions/setup-android@v3
- name: ⚙️ Expo prebuild
run: npx expo prebuild --clean --non-interactive
- name: 🔐 Decode Keystore
run: |
echo "${{ secrets.KEYSTORE_BASE64 }}" | base64 --decode > android/release.keystore
- name: 🏗️ Build APK (Release)
working-directory: ./apk/android
run: |
./gradlew assembleRelease
- name: 📤 Upload APK Artifact
uses: actions/upload-artifact@v3
with:
name: app-release-apk
path: apk/android/app/build/outputs/apk/release/app-release.apk

View File

@@ -9,7 +9,8 @@
"userInterfaceStyle": "automatic", "userInterfaceStyle": "automatic",
"newArchEnabled": true, "newArchEnabled": true,
"ios": { "ios": {
"supportsTablet": true "supportsTablet": true,
"bundleIdentifier": "com.arandano69.Birthday"
}, },
"android": { "android": {
"permissions": [ "permissions": [
@@ -22,7 +23,8 @@
"monochromeImage": "./assets/images/android-icon-monochrome.png" "monochromeImage": "./assets/images/android-icon-monochrome.png"
}, },
"edgeToEdgeEnabled": true, "edgeToEdgeEnabled": true,
"predictiveBackGestureEnabled": false "predictiveBackGestureEnabled": false,
"package": "com.arandano69.Birthday"
}, },
"web": { "web": {
"output": "static", "output": "static",
@@ -42,11 +44,28 @@
"backgroundColor": "#000000" "backgroundColor": "#000000"
} }
} }
],
[
"expo-build-properties",
{
"android": {
"gradleProperties": {
"org.gradle.jvmargs": "-Xmx2g -XX:MaxMetaspaceSize=512m",
"org.gradle.parallel": "false"
}
}
}
] ]
], ],
"experiments": { "experiments": {
"typedRoutes": true, "typedRoutes": true,
"reactCompiler": true "reactCompiler": true
},
"extra": {
"router": {},
"eas": {
"projectId": "f761fcbd-46f2-4387-8282-005e44223075"
}
} }
} }
} }

View File

@@ -0,0 +1,5 @@
import { Stack } from "expo-router";
export default function AuthLayout() {
return <Stack screenOptions={{ headerShown: false }} />;
}

149
apk/app/(auth)/login.tsx Normal file
View File

@@ -0,0 +1,149 @@
import React, { useState } from "react";
import { View, Text, TextInput, TouchableOpacity, StyleSheet } from "react-native";
import { useAuth } from "@/context/auth-context";
import { router } from "expo-router";
export default function AuthScreen() {
const [isLogin, setIsLogin] = useState(true);
return isLogin ? (
<LoginScreen onSwitch={() => setIsLogin(false)} />
) : (
<RegisterScreen onSwitch={() => setIsLogin(true)} />
);
}
function LoginScreen({ onSwitch }) {
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const { login } = useAuth();
const handleLogin = async () => {
const loginResult = await login(email, password);
if (loginResult) {
console.log("Login successful");
router.replace("/(tabs)");
}
};
return (
<View style={styles.container}>
<Text style={styles.title}>Welcome Back</Text>
<TextInput
style={styles.input}
placeholder="Email"
value={email}
onChangeText={setEmail}
keyboardType="email-address"
autoCapitalize="none"
/>
<TextInput
style={styles.input}
placeholder="Password"
value={password}
onChangeText={setPassword}
secureTextEntry
/>
<TouchableOpacity style={styles.button} onPress={handleLogin}>
<Text style={styles.buttonText}>Login</Text>
</TouchableOpacity>
<TouchableOpacity onPress={onSwitch}>
<Text style={styles.link}>Don&apos;t have an account? Register</Text>
</TouchableOpacity>
</View>
);
}
function RegisterScreen({ onSwitch }) {
const [name, setName] = useState("");
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const handleRegister = () => {
console.log("Register ->", { name, email, password });
};
return (
<View style={styles.container}>
<Text style={styles.title}>Create Account</Text>
<TextInput
style={styles.input}
placeholder="Full Name"
value={name}
onChangeText={setName}
/>
<TextInput
style={styles.input}
placeholder="Email"
value={email}
onChangeText={setEmail}
keyboardType="email-address"
autoCapitalize="none"
/>
<TextInput
style={styles.input}
placeholder="Password"
value={password}
onChangeText={setPassword}
secureTextEntry
/>
<TouchableOpacity style={styles.button} onPress={handleRegister}>
<Text style={styles.buttonText}>Register</Text>
</TouchableOpacity>
<TouchableOpacity onPress={onSwitch}>
<Text style={styles.link}>Already have an account? Login</Text>
</TouchableOpacity>
</View>
);
}
const styles = StyleSheet.create({
container: {
flex: 1,
justifyContent: "center",
padding: 20,
backgroundColor: "#f5f5f5",
},
title: {
fontSize: 28,
fontWeight: "bold",
marginBottom: 30,
textAlign: "center",
},
input: {
height: 50,
borderColor: "#ccc",
borderWidth: 1,
borderRadius: 10,
paddingHorizontal: 15,
marginBottom: 15,
backgroundColor: "#fff",
},
button: {
height: 50,
backgroundColor: "#4CAF50",
borderRadius: 10,
justifyContent: "center",
alignItems: "center",
marginTop: 10,
},
buttonText: {
color: "#fff",
fontSize: 16,
fontWeight: "bold",
},
link: {
marginTop: 15,
textAlign: "center",
color: "#007BFF",
},
});

View File

@@ -0,0 +1,25 @@
import { Tabs } from "expo-router";
import { Ionicons } from "@expo/vector-icons";
export default function TabsLayout() {
return (
<Tabs
screenOptions={{
headerShown: false,
}}
>
<Tabs.Screen name="index" options={{
title: "Birthdays",
tabBarIcon: ({ color, size }) => (
<Ionicons name="gift" size={size} color={color} />
),
}} />
<Tabs.Screen name="settings" options={{
title: "Settings",
tabBarIcon: ({ color, size }) => (
<Ionicons name="settings" size={size} color={color} />
),
}} />
</Tabs>
);
}

View File

@@ -14,7 +14,11 @@ export default function HomeScreen() {
<View style={styles.actionsContainer}> <View style={styles.actionsContainer}>
<Text style={styles.title}>Upcoming Birthdays</Text> <Text style={styles.title}>Upcoming Birthdays</Text>
<TouchableOpacity <TouchableOpacity
onPress={() => router.push("/add")} onPress={() => {
router.push("/add");
}
}
style={styles.addButton} style={styles.addButton}
> >
<Text style={styles.addButtonText}>+</Text> <Text style={styles.addButtonText}>+</Text>

148
apk/app/(tabs)/settings.tsx Normal file
View File

@@ -0,0 +1,148 @@
import React, { useState } from "react";
import { StyleSheet, Text, View, TouchableOpacity, Alert } from "react-native";
import { SafeAreaView } from "react-native-safe-area-context";
import { useRouter } from "expo-router";
import { useAuth } from "@/context/auth-context";
export default function SettingsScreen() {
const router = useRouter()
const { logout } = useAuth();
const [theme, setTheme] = useState("light"); // light | dark | system
const handleLogout = () => {
Alert.alert("Logout", "Are you sure you want to logout?", [
{ text: "Cancel", style: "cancel" },
{
text: "Logout",
style: "destructive",
onPress: () => {
// TODO: clear auth state here
router.replace("/login");
logout();
},
},
]);
};
const toggleTheme = (value) => {
setTheme(value);
// TODO: persist theme (AsyncStorage / context / zustand)
};
return (
<SafeAreaView style={styles.screen} edges={["top", "bottom"]}>
<View style={styles.content}>
<Text style={styles.title}>Settings</Text>
{/* Theme Selector */}
<View style={styles.section}>
<Text style={styles.sectionTitle}>Theme</Text>
<View style={styles.row}>
<ThemeButton
label="Light"
active={theme === "light"}
onPress={() => toggleTheme("light")}
/>
<ThemeButton
label="Dark"
active={theme === "dark"}
onPress={() => toggleTheme("dark")}
/>
<ThemeButton
label="System"
active={theme === "system"}
onPress={() => toggleTheme("system")}
/>
</View>
</View>
{/* Logout */}
<View style={styles.section}>
<Text style={styles.sectionTitle}>Account</Text>
<TouchableOpacity style={styles.logoutButton} onPress={handleLogout}>
<Text style={styles.logoutText}>Log Out</Text>
</TouchableOpacity>
</View>
</View>
</SafeAreaView>
);
}
function ThemeButton({ label, active, onPress }) {
return (
<TouchableOpacity
onPress={onPress}
style={[styles.themeButton, active && styles.themeButtonActive]}
>
<Text style={[styles.themeText, active && styles.themeTextActive]}>
{label}
</Text>
</TouchableOpacity>
);
}
const styles = StyleSheet.create({
screen: {
flex: 1,
backgroundColor: "#fff",
},
content: {
flex: 1,
paddingHorizontal: 16,
paddingTop: 12,
},
title: {
fontSize: 24,
fontWeight: "bold",
marginBottom: 20,
},
section: {
marginBottom: 24,
},
sectionTitle: {
fontSize: 16,
fontWeight: "600",
marginBottom: 10,
},
row: {
flexDirection: "row",
gap: 10,
},
themeButton: {
paddingVertical: 8,
paddingHorizontal: 12,
borderRadius: 8,
borderWidth: 1,
borderColor: "#ccc",
},
themeButtonActive: {
backgroundColor: "#111",
borderColor: "#111",
},
themeText: {
color: "#333",
paddingHorizontal: 5,
},
themeTextActive: {
color: "#fff",
},
logoutButton: {
padding: 14,
borderRadius: 10,
backgroundColor: "#ff3b30",
alignItems: "center",
},
logoutText: {
color: "#fff",
fontWeight: "600",
},
});

View File

@@ -1,31 +1,11 @@
import { import RootNavigation from "@/components/views/root-navigation";
DarkTheme, import { AuthProvider } from "@/context/auth-context";
DefaultTheme,
ThemeProvider,
} from "@react-navigation/native";
import { Stack } from "expo-router";
import { StatusBar } from "expo-status-bar";
import "react-native-reanimated";
import { BirthdaysProvider } from "@/context/birthdays-context";
import { useColorScheme } from "@/hooks/use-color-scheme";
export const unstable_settings = {
anchor: "(tabs)",
};
export default function RootLayout() { export default function RootLayout() {
const colorScheme = useColorScheme();
return ( return (
<BirthdaysProvider> <AuthProvider>
<ThemeProvider value={colorScheme === "dark" ? DarkTheme : DefaultTheme}> <RootNavigation />
<Stack> </AuthProvider>
<Stack.Screen name="index" options={{ headerShown: false }} />
<Stack.Screen name="add" options={{ headerShown: false }} />
</Stack>
<StatusBar style="auto" />
</ThemeProvider>
</BirthdaysProvider>
); );
} }

59
apk/build.sh Executable file
View File

@@ -0,0 +1,59 @@
#!/bin/bash
export ANDROID_HOME=$HOME/android
export ANDROID_SDK_ROOT=$ANDROID_HOME
export PATH=$ANDROID_HOME/emulator:$ANDROID_HOME/platform-tools:$ANDROID_HOME/tools:$ANDROID_HOME/tools/bin:$PATH
set -e # stop on error
echo "🚧 Starting APK build..."
# Step 1: Ensure native android folder exists
if [ ! -d "android" ]; then
echo "📦 Running Expo prebuild..."
npx expo prebuild
fi
# Step 2: Go to android folder
cd android
# Step 3: Check if keystore exists
KEYSTORE_FILE="app/my-release-key.keystore"
if [ ! -f "$KEYSTORE_FILE" ]; then
echo "🔑 Generating keystore..."
keytool -genkeypair -v \
-storetype PKCS12 \
-keystore $KEYSTORE_FILE \
-alias my-key-alias \
-keyalg RSA \
-keysize 2048 \
-validity 10000 \
-storepass password \
-keypass password \
-dname "CN=Your Name, OU=Dev, O=MyApp, L=City, S=State, C=US"
fi
# Step 4: Ensure gradle.properties has signing config
GRADLE_PROPS="gradle.properties"
if ! grep -q "MYAPP_UPLOAD_STORE_FILE" $GRADLE_PROPS; then
echo "⚙️ Adding signing config..."
cat <<EOF >> $GRADLE_PROPS
MYAPP_UPLOAD_STORE_FILE=my-release-key.keystore
MYAPP_UPLOAD_KEY_ALIAS=my-key-alias
MYAPP_UPLOAD_STORE_PASSWORD=password
MYAPP_UPLOAD_KEY_PASSWORD=password
EOF
fi
# Step 5: Build release APK
echo "🏗️ Building APK..."
./gradlew assembleRelease
# Step 6: Output path
APK_PATH="app/build/outputs/apk/release/app-release.apk"
echo "✅ Build complete!"
echo "📦 APK located at: android/$APK_PATH"

View File

@@ -52,34 +52,44 @@ export function BirthdayList() {
// Helper function to group and sort // Helper function to group and sort
function groupBirthdaysByDate(data: BirthdayEntry[]): SectionData[] { function groupBirthdaysByDate(data: BirthdayEntry[]): SectionData[] {
// Sort by date
const today = new Date(); const today = new Date();
const currentYear = today.getFullYear(); const currentYear = today.getFullYear();
const todayAtMidnight = new Date(today);
todayAtMidnight.setHours(0, 0, 0, 0);
const daysUntilNext = (month: number, day: number) => { const daysUntilNext = (month: number, day: number) => {
let target = new Date(currentYear, month, day); let target = new Date(currentYear, month, day);
if(target < today.setHours(0, 0, 0, 0)){ if (target < todayAtMidnight) {
target = new Date(currentYear + 1, month, day); target = new Date(currentYear + 1, month, day);
} }
const msPerDay = 1000 * 60 * 60 * 24; const msPerDay = 1000 * 60 * 60 * 24;
const diff = target - new Date(today.setHours(0,0,0,0)); const diff = target.getTime() - todayAtMidnight.getTime();
return Math.ceil(diff / msPerDay); return Math.ceil(diff / msPerDay);
}; };
const sorted = [...data].sort((a, b) => { const sorted = [...data].sort((a, b) => {
const dateA = parseBirthdayDate(a); const dateA = parseBirthdayDate(a.date);
const dateB = parseBirthdayDate(b); const dateB = parseBirthdayDate(b.date);
return daysUntilNext(dateA.getMonth(), dateA.getDate()) - daysUntilNext(dateB.getMonth(), dateB.getDate());
const nextBirthdayDiff =
daysUntilNext(dateA.getMonth(), dateA.getDate()) -
daysUntilNext(dateB.getMonth(), dateB.getDate());
if (nextBirthdayDiff !== 0) {
return nextBirthdayDiff;
}
return dateA.getFullYear() - dateB.getFullYear();
}); });
// Group by date
const grouped = sorted.reduce((acc, item) => { const grouped = sorted.reduce((acc, item) => {
const existing = acc.find((section) => section.title === item.date); const sectionKey = getMonthDayKey(item.date);
const existing = acc.find((section) => section.title === sectionKey);
if (existing) { if (existing) {
existing.data.push(item); existing.data.push(item);
} else { } else {
acc.push({ title: item.date, data: [item] }); acc.push({ title: sectionKey, data: [item] });
} }
return acc; return acc;
@@ -92,11 +102,20 @@ function parseBirthdayDate(date: string) {
return new Date(`${date}T00:00:00`); return new Date(`${date}T00:00:00`);
} }
function getMonthDayKey(date: string) {
const parsedDate = parseBirthdayDate(date);
const month = `${parsedDate.getMonth() + 1}`.padStart(2, "0");
const day = `${parsedDate.getDate()}`.padStart(2, "0");
return `${month}-${day}`;
}
function formatDisplayDate(date: string) { function formatDisplayDate(date: string) {
return parseBirthdayDate(date).toLocaleDateString(undefined, { const [month, day] = date.split("-").map(Number);
return new Date(2000, month - 1, day).toLocaleDateString(undefined, {
month: "long", month: "long",
day: "numeric", day: "numeric",
year: "numeric",
}); });
} }

View File

@@ -1,10 +1,8 @@
import { StyleSheet } from "react-native"; import { StyleSheet, View } from "react-native";
import Animated, { useAnimatedRef } from "react-native-reanimated"; import Animated, { useAnimatedRef } from "react-native-reanimated";
import { useColorScheme } from "@/hooks/use-color-scheme";
import { useThemeColor } from "@/hooks/use-theme-color"; import { useThemeColor } from "@/hooks/use-theme-color";
import { PropsWithChildren } from "react"; import { PropsWithChildren } from "react";
import { View } from "react-native";
import { useSafeAreaInsets } from "react-native-safe-area-context"; import { useSafeAreaInsets } from "react-native-safe-area-context";
type Props = PropsWithChildren<{ type Props = PropsWithChildren<{
@@ -14,7 +12,6 @@ type Props = PropsWithChildren<{
export default function ScrollView({ children, headerBackgroundColor }: Props) { export default function ScrollView({ children, headerBackgroundColor }: Props) {
const scrollRef = useAnimatedRef<Animated.ScrollView>(); const scrollRef = useAnimatedRef<Animated.ScrollView>();
const insets = useSafeAreaInsets(); const insets = useSafeAreaInsets();
const colorScheme = useColorScheme() ?? "light";
const backgroundColor = useThemeColor({}, "background"); const backgroundColor = useThemeColor({}, "background");
return ( return (

View File

@@ -0,0 +1,23 @@
import React from "react";
import { View, Text, ActivityIndicator, StyleSheet } from "react-native";
export default function LoadingScreen({ message = "Loading..." }) {
return (
<View style={styles.container}>
<ActivityIndicator size="large" color="#ffffff" />
<Text style={styles.text}>{message}</Text>
</View>
);
}
const styles = StyleSheet.create({
container: {
flex: 1,
alignItems: "center",
justifyContent: "center",
},
text: {
marginTop: 10,
fontSize: 16,
},
});

View File

@@ -0,0 +1,36 @@
import { BirthdaysProvider } from "@/context/birthdays-context";
import { useAuth } from "@/context/auth-context";
import { Stack, router, useRootNavigationState, useSegments } from "expo-router";
import React, { useEffect } from "react";
import LoadingScreen from "./loading-screen";
export default function RootLayout() {
const { user, isHydrated } = useAuth();
const navigationState = useRootNavigationState();
const segments = useSegments();
const [loaded, setLoaded] = React.useState(false);
useEffect(() => {
if (!isHydrated || !navigationState?.key) return;
const inAuthGroup = segments.length === 0;
if (!user) {
router.replace("/(auth)/login");
} else if (inAuthGroup) {
router.replace("/(tabs)");
}
setLoaded(true); // 👈 ALWAYS run this
}, [isHydrated, navigationState?.key, segments, user]);
if(!loaded) {
return (<LoadingScreen />); // or a loading spinner
}
return (
<BirthdaysProvider>
<Stack screenOptions={{ headerShown: false }} />
</BirthdaysProvider>
);
}

View File

@@ -0,0 +1,69 @@
import AsyncStorage from "@react-native-async-storage/async-storage";
import * as React from "react";
interface User {
email: string;
}
interface AuthContextValue {
user: User | null;
isHydrated: boolean;
login: (email: string, password: string) => Promise<boolean>;
logout: () => Promise<void>;
}
const authContext = React.createContext<AuthContextValue | undefined>(undefined);
const STORAGE_KEY = "user";
export function AuthProvider({ children }: { children: React.ReactNode }) {
const [user, setUser] = React.useState<User | null>(null);
const [isHydrated, setIsHydrated] = React.useState(false);
React.useEffect(() => {
const restoreUser = async () => {
try {
const storedUser = await AsyncStorage.getItem(STORAGE_KEY);
if (storedUser) {
setUser(JSON.parse(storedUser) as User);
}
} catch (error) {
console.log("Error restoring user", error);
} finally {
setIsHydrated(true);
}
};
restoreUser();
}, []);
const login = async (email: string, password: string) => {
const fakeUser = { email };
await AsyncStorage.setItem(STORAGE_KEY, JSON.stringify(fakeUser));
setUser(fakeUser);
console.log(fakeUser)
await new Promise((resolve) => setTimeout(resolve, 1000));
return true;
}
const logout = async () => {
await AsyncStorage.removeItem(STORAGE_KEY);
setUser(null);
}
return (
<authContext.Provider value={{ user, isHydrated, login, logout }}>
{children}
</authContext.Provider>
);
}
export function useAuth() {
const context = React.useContext(authContext);
if(!context) {
throw new Error("useAuth must be inside AuthProvider");
}
return context;
}

100
apk/package-lock.json generated
View File

@@ -15,6 +15,7 @@
"@react-navigation/elements": "^2.6.3", "@react-navigation/elements": "^2.6.3",
"@react-navigation/native": "^7.1.8", "@react-navigation/native": "^7.1.8",
"expo": "~54.0.33", "expo": "~54.0.33",
"expo-build-properties": "~1.0.10",
"expo-constants": "~18.0.13", "expo-constants": "~18.0.13",
"expo-font": "~14.0.11", "expo-font": "~14.0.11",
"expo-haptics": "~15.0.8", "expo-haptics": "~15.0.8",
@@ -3765,9 +3766,6 @@
"arm64" "arm64"
], ],
"dev": true, "dev": true,
"libc": [
"glibc"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3782,9 +3780,6 @@
"arm64" "arm64"
], ],
"dev": true, "dev": true,
"libc": [
"musl"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3799,9 +3794,6 @@
"ppc64" "ppc64"
], ],
"dev": true, "dev": true,
"libc": [
"glibc"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3816,9 +3808,6 @@
"riscv64" "riscv64"
], ],
"dev": true, "dev": true,
"libc": [
"glibc"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3833,9 +3822,6 @@
"riscv64" "riscv64"
], ],
"dev": true, "dev": true,
"libc": [
"musl"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3850,9 +3836,6 @@
"s390x" "s390x"
], ],
"dev": true, "dev": true,
"libc": [
"glibc"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3867,9 +3850,6 @@
"x64" "x64"
], ],
"dev": true, "dev": true,
"libc": [
"glibc"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -3884,9 +3864,6 @@
"x64" "x64"
], ],
"dev": true, "dev": true,
"libc": [
"musl"
],
"license": "MIT", "license": "MIT",
"optional": true, "optional": true,
"os": [ "os": [
@@ -6166,6 +6143,53 @@
"react-native": "*" "react-native": "*"
} }
}, },
"node_modules/expo-build-properties": {
"version": "1.0.10",
"resolved": "https://registry.npmjs.org/expo-build-properties/-/expo-build-properties-1.0.10.tgz",
"integrity": "sha512-mFCZbrbrv0AP5RB151tAoRzwRJelqM7bCJzCkxpu+owOyH+p/rFC/q7H5q8B9EpVWj8etaIuszR+gKwohpmu1Q==",
"license": "MIT",
"dependencies": {
"ajv": "^8.11.0",
"semver": "^7.6.0"
},
"peerDependencies": {
"expo": "*"
}
},
"node_modules/expo-build-properties/node_modules/ajv": {
"version": "8.18.0",
"resolved": "https://registry.npmjs.org/ajv/-/ajv-8.18.0.tgz",
"integrity": "sha512-PlXPeEWMXMZ7sPYOHqmDyCJzcfNrUr3fGNKtezX14ykXOEIvyK81d+qydx89KY5O71FKMPaQ2vBfBFI5NHR63A==",
"license": "MIT",
"dependencies": {
"fast-deep-equal": "^3.1.3",
"fast-uri": "^3.0.1",
"json-schema-traverse": "^1.0.0",
"require-from-string": "^2.0.2"
},
"funding": {
"type": "github",
"url": "https://github.com/sponsors/epoberezkin"
}
},
"node_modules/expo-build-properties/node_modules/json-schema-traverse": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz",
"integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==",
"license": "MIT"
},
"node_modules/expo-build-properties/node_modules/semver": {
"version": "7.7.4",
"resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz",
"integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==",
"license": "ISC",
"bin": {
"semver": "bin/semver.js"
},
"engines": {
"node": ">=10"
}
},
"node_modules/expo-constants": { "node_modules/expo-constants": {
"version": "18.0.13", "version": "18.0.13",
"resolved": "https://registry.npmjs.org/expo-constants/-/expo-constants-18.0.13.tgz", "resolved": "https://registry.npmjs.org/expo-constants/-/expo-constants-18.0.13.tgz",
@@ -6850,6 +6874,22 @@
"dev": true, "dev": true,
"license": "MIT" "license": "MIT"
}, },
"node_modules/fast-uri": {
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.1.0.tgz",
"integrity": "sha512-iPeeDKJSWf4IEOasVVrknXpaBV0IApz/gp7S2bb7Z4Lljbl2MGJRqInZiUrQwV16cpzw/D3S5j5Julj/gT52AA==",
"funding": [
{
"type": "github",
"url": "https://github.com/sponsors/fastify"
},
{
"type": "opencollective",
"url": "https://opencollective.com/fastify"
}
],
"license": "BSD-3-Clause"
},
"node_modules/fb-watchman": { "node_modules/fb-watchman": {
"version": "2.0.2", "version": "2.0.2",
"resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz",
@@ -8680,9 +8720,6 @@
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
"libc": [
"glibc"
],
"license": "MPL-2.0", "license": "MPL-2.0",
"optional": true, "optional": true,
"os": [ "os": [
@@ -8703,9 +8740,6 @@
"cpu": [ "cpu": [
"arm64" "arm64"
], ],
"libc": [
"musl"
],
"license": "MPL-2.0", "license": "MPL-2.0",
"optional": true, "optional": true,
"os": [ "os": [
@@ -8726,9 +8760,6 @@
"cpu": [ "cpu": [
"x64" "x64"
], ],
"libc": [
"glibc"
],
"license": "MPL-2.0", "license": "MPL-2.0",
"optional": true, "optional": true,
"os": [ "os": [
@@ -8749,9 +8780,6 @@
"cpu": [ "cpu": [
"x64" "x64"
], ],
"libc": [
"musl"
],
"license": "MPL-2.0", "license": "MPL-2.0",
"optional": true, "optional": true,
"os": [ "os": [

View File

@@ -5,8 +5,8 @@
"scripts": { "scripts": {
"start": "expo start", "start": "expo start",
"reset-project": "node ./scripts/reset-project.js", "reset-project": "node ./scripts/reset-project.js",
"android": "expo start --android", "android": "expo run:android",
"ios": "expo start --ios", "ios": "expo run:ios",
"web": "expo start --web", "web": "expo start --web",
"lint": "expo lint" "lint": "expo lint"
}, },
@@ -18,6 +18,7 @@
"@react-navigation/elements": "^2.6.3", "@react-navigation/elements": "^2.6.3",
"@react-navigation/native": "^7.1.8", "@react-navigation/native": "^7.1.8",
"expo": "~54.0.33", "expo": "~54.0.33",
"expo-build-properties": "~1.0.10",
"expo-constants": "~18.0.13", "expo-constants": "~18.0.13",
"expo-font": "~14.0.11", "expo-font": "~14.0.11",
"expo-haptics": "~15.0.8", "expo-haptics": "~15.0.8",