diff --git a/client/app/contexts/LocationContext.tsx b/client/app/contexts/LocationContext.tsx index 4b1f38e70..00d7659f1 100644 --- a/client/app/contexts/LocationContext.tsx +++ b/client/app/contexts/LocationContext.tsx @@ -1,30 +1,19 @@ import { LOCATION_REFRESH_RATE } from "@env"; -import * as Location from "expo-location"; import React, { createContext, useContext, useEffect, useState } from "react"; +import { getLocation, checkLocationPermission } from "@app/services/LocationService"; +import { LocationContextProps, LocationType } from "@app/types/Location"; -interface LocationContextProps { - longitude: number; - latitude: number; - isLocationEnabled: boolean; -} -interface LocationType { - longitude: number; - latitude: number; -} +// LocationContext Creation const LocationContext = createContext(null); -const getLocation = async () => { - return await Location.getCurrentPositionAsync({ - accuracy: Location.Accuracy.Balanced, - }); // Change accuracy while testing. Could become .env variable. -}; - +// Custom Hook for Consuming Location Context export const useLocation = () => { return useContext(LocationContext); }; +// LocationProvider Component to Provide Location Context export const LocationProvider = ({ children, }: { @@ -37,43 +26,39 @@ export const LocationProvider = ({ const [isLocationEnabled, setIsLocationEnabled] = useState(false); useEffect(() => { - // TODO: Refactor this useEffect into a different file (service?) outside of the context, as it is not part of the purpose of a context. - (async () => { - // Request location permissions, if not granted, return - const { status } = await Location.requestForegroundPermissionsAsync(); - if (status !== "granted") { - console.log("Permission to access location was denied"); - return; - } + let interval: NodeJS.Timeout; + + const startLocationTracking = async () => { + const hasPermission = await checkLocationPermission(); // Use permission service + if (!hasPermission) return; setIsLocationEnabled(true); - const interval = setInterval(async () => { - // FIXME: This loop does not stop after refreshing app. Must completely close out and restart app when LOCATION_REFRESH_RATE is changed. - try { - const locationData = await getLocation(); - if ( - locationData.coords.latitude !== location.latitude || - locationData.coords.longitude !== location.longitude - ) { - setLocation({ - latitude: locationData.coords.latitude, - longitude: locationData.coords.longitude, - }); + // Set up the interval once after permission is granted + interval = setInterval(async () => { + const locationData = await getLocation(); + if (locationData && locationData.coords) { + const { latitude, longitude } = locationData.coords; + if (latitude !== location.latitude || longitude !== location.longitude) { + setLocation({ latitude, longitude }); } else { console.log("Location has not changed"); } - } catch (error) { - console.error("Error fetching location:", error); } - }, Number(LOCATION_REFRESH_RATE)); // Fetch location every 3 seconds + }, Number(LOCATION_REFRESH_RATE)); + }; + + startLocationTracking(); - // Cleanup function to clear interval when component unmounts - return () => clearInterval(interval); - })(); + // Cleanup function to clear interval when component unmounts + return () => { + if (interval) { + clearInterval(interval); + console.log("[LOG]: Cleaning up location useEffect"); + } + }; + }, []); - return () => console.log("[LOG]: Cleaning up location useEffect"); - }, []); return ( { return useContext(SettingsContext); }; -const loadSettings = async () => { - try { - const themeSetting = await AsyncStorage.getItem("theme"); - if (themeSetting !== null) { - return { - theme: themeSetting, - }; - } else { - await AsyncStorage.setItem("theme", "light"); - return { - theme: "light", - }; - } - } catch (err) { - console.error(err); - } -}; - export const SettingsProvider = ({ children, }: { children: React.ReactNode; }) => { - const [theme, setTheme] = useState("light"); + const [theme, setTheme] = useState("light"); // Initial settings load useEffect(() => { - const loadInitialSettings = async () => { - const settings = await loadSettings(); - if (settings) { - setTheme(settings.theme); - } + const initializeTheme = async () => { + const savedTheme = await loadTheme(); + setTheme(savedTheme); }; - loadInitialSettings(); + initializeTheme(); }, []); - // Setting toggler - const reloadSettings = async () => { - const settings = await loadSettings(); - if (settings) { - setTheme(settings.theme); - } - }; - + // Toggle theme and update AsyncStorage const toggleTheme = async () => { - console.log("Toggling theme"); - try { - const settings = await loadSettings(); - if (settings && settings.theme === "light") { - await AsyncStorage.setItem("theme", "dark"); - } else { - await AsyncStorage.setItem("theme", "light"); - } - } catch (err) { - console.error(err); - } - - await reloadSettings(); + const newTheme = theme === "light" ? "dark" : "light"; + setTheme(newTheme); + await saveTheme(newTheme); // Save the new theme }; return ( diff --git a/client/app/contexts/SocketContext.tsx b/client/app/contexts/SocketContext.tsx index f5c9f5b47..fd39e575a 100644 --- a/client/app/contexts/SocketContext.tsx +++ b/client/app/contexts/SocketContext.tsx @@ -1,9 +1,8 @@ -import { EXPO_IP } from "@env"; import React, { createContext, useContext, useEffect, useState } from "react"; -import { io, Socket } from "socket.io-client"; - +import { Socket } from "socket.io-client"; import { useLocation } from "./LocationContext"; -import { AuthStore } from "../services/AuthStore"; +import { initializeSocket, getToken, sendLocationUpdate } from "@app/services/SocketService"; + const SocketContext = createContext(null); @@ -17,35 +16,25 @@ export const SocketProvider = ({ children }: { children: React.ReactNode }) => { const locationContext = useLocation(); useEffect(() => { - const getToken = async () => { - const token = await AuthStore.getRawState().userAuthInfo?.getIdToken(); - console.log("Token:", token); - return token; - }; - - const initializeSocket = async () => { + const initialize = async () => { const token = await getToken(); - const socketIo = io(`http://${EXPO_IP}:8080`, { - auth: { - token, - }, - }); - - socketIo.connect(); - setSocket(socketIo); - setMounted(true); + if (token) { + const socketIo = await initializeSocket(token); + setSocket(socketIo); + setMounted(true); + } }; if (!mounted) { - initializeSocket(); + initialize(); } + return () => { socket?.disconnect(); - console.log("[LOG]: Cleaning up intializeSocket useEffect"); + console.log("[LOG]: Cleaning up initializeSocket useEffect"); }; - }, []); + }, [mounted]); - // Listen to the socket state and run once the socket is set! useEffect(() => { if (!socket) return; @@ -62,31 +51,19 @@ export const SocketProvider = ({ children }: { children: React.ReactNode }) => { }, [socket]); useEffect(() => { - // TODO: Refactor this useEffect into a different file (service?) outside of the context, as it is not part of the purpose of a context. if ( socket && + locationContext && locationContext?.latitude !== 9999 && locationContext?.longitude !== 9999 ) { - console.log( - "Sending location update to server:", - locationContext?.latitude, - locationContext?.longitude, - ); - socket.emit( - "updateLocation", - { - lat: locationContext?.latitude, - lon: locationContext?.longitude, - }, - (ack: string) => { - console.log("Location update ack:", ack); - }, - ); + sendLocationUpdate(socket, locationContext.latitude, locationContext.longitude); } - }, [locationContext?.latitude, locationContext?.longitude]); + }, [locationContext?.latitude, locationContext?.longitude, socket]); return ( - {children} + + {children} + ); }; diff --git a/client/app/contexts/UserContext.tsx b/client/app/contexts/UserContext.tsx index 9dcaf92b2..26bef57e0 100644 --- a/client/app/contexts/UserContext.tsx +++ b/client/app/contexts/UserContext.tsx @@ -1,7 +1,6 @@ -import React, { createContext, useContext, useEffect, useState } from "react"; - +import React, { createContext, useContext, useState } from "react"; import { UserType } from "../types/User"; -import { generateName } from "@app/utils/scripts"; +import { initializeUser } from "@app/services/UserService"; const UserContext = createContext(null); @@ -10,17 +9,7 @@ export const useUser = () => { }; export const UserProvider = ({ children }: { children: React.ReactNode }) => { - const [user, setUser] = useState({ - displayName: "DefaultDisplayName", - userIcon: { - imagePath: "DefaultImagePath", - colorHex: "#fff", - }, - }); - - useEffect(() => { - user.displayName = generateName() - }, []) + const [user, setUser] = useState(initializeUser); return {children}; }; diff --git a/client/app/services/LocationService.ts b/client/app/services/LocationService.ts new file mode 100644 index 000000000..be23242f6 --- /dev/null +++ b/client/app/services/LocationService.ts @@ -0,0 +1,23 @@ +import * as Location from "expo-location"; + +// Location Service to Handle Location Fetching +export const getLocation = async (): Promise => { + try { + return await Location.getCurrentPositionAsync({ + accuracy: Location.Accuracy.Balanced, + }); + } catch (error) { + console.error("Error fetching location:", error); + return null; + } +}; + +// Permission Service to Handle Location Permissions +export const checkLocationPermission = async (): Promise => { + const { status } = await Location.requestForegroundPermissionsAsync(); + if (status !== "granted") { + console.log("Permission to access location was denied"); + return false; + } + return true; +}; diff --git a/client/app/services/SocketService.ts b/client/app/services/SocketService.ts new file mode 100644 index 000000000..60bc46065 --- /dev/null +++ b/client/app/services/SocketService.ts @@ -0,0 +1,41 @@ +import { io, Socket } from "socket.io-client"; +import { AuthStore } from "../services/AuthStore"; +import { EXPO_IP } from "@env"; + +export const initializeSocket = async (token: string): Promise => { + const socketIo = io(`http://${EXPO_IP}:8080`, { + auth: { + token, + }, + }); + + socketIo.connect(); + return socketIo; +}; + + +export const getToken = async (): Promise => { + const token = await AuthStore.getRawState().userAuthInfo?.getIdToken(); + console.log("Token:", token); + return token ?? null; +}; + + + +export const sendLocationUpdate = ( + socket: Socket, + latitude: number, + longitude: number +) => { + console.log("Sending location update to server:", latitude, longitude); + socket.emit( + "updateLocation", + { + lat: latitude, + lon: longitude, + }, + (ack: string) => { + console.log("Location update ack:", ack); + } + ); +}; \ No newline at end of file diff --git a/client/app/services/UserService.ts b/client/app/services/UserService.ts new file mode 100644 index 000000000..4976d8c7e --- /dev/null +++ b/client/app/services/UserService.ts @@ -0,0 +1,13 @@ +import { UserType } from "../types/User"; +import { generateName } from "@app/utils/scripts"; + +// Function to initialize default user +export const initializeUser = (): UserType => { + return { + displayName: generateName(), + userIcon: { + imagePath: "DefaultImagePath", + colorHex: "#fff", + }, + }; +}; diff --git a/client/app/types/Location.ts b/client/app/types/Location.ts new file mode 100644 index 000000000..fb384602d --- /dev/null +++ b/client/app/types/Location.ts @@ -0,0 +1,10 @@ +export interface LocationContextProps { + longitude: number; + latitude: number; + isLocationEnabled: boolean; + } + +export interface LocationType { + longitude: number; + latitude: number; + } \ No newline at end of file diff --git a/client/app/utils/storageUtils.ts b/client/app/utils/storageUtils.ts new file mode 100644 index 000000000..6c8e62a3c --- /dev/null +++ b/client/app/utils/storageUtils.ts @@ -0,0 +1,26 @@ +import AsyncStorage from "@react-native-async-storage/async-storage"; + +// Utility to load theme from AsyncStorage +export const loadTheme = async (): Promise => { + try { + const themeSetting = await AsyncStorage.getItem("theme"); + if (themeSetting !== null) { + return themeSetting; + } else { + await AsyncStorage.setItem("theme", "light"); + return "light"; + } + } catch (error) { + console.error("Failed to load theme", error); + return "light"; + } +}; + +// Utility to save theme to AsyncStorage +export const saveTheme = async (theme: string): Promise => { + try { + await AsyncStorage.setItem("theme", theme); + } catch (error) { + console.error("Failed to save theme", error); + } +};