summaryrefslogtreecommitdiff
path: root/app
diff options
context:
space:
mode:
authorSanto Cariotti <santo@dcariotti.me>2024-09-09 16:35:28 +0200
committerSanto Cariotti <santo@dcariotti.me>2024-09-09 16:35:28 +0200
commit558d10b39ceb6380272047da742c6d6b62830008 (patch)
tree5d3de5f46a34885ec351b1b13b774f346c5a5faf /app
parentd14ce814d083e2ee61f29a235f60d936a545ab68 (diff)
Add location update
Diffstat (limited to 'app')
-rw-r--r--app/(tabs)/index.tsx411
1 files changed, 253 insertions, 158 deletions
diff --git a/app/(tabs)/index.tsx b/app/(tabs)/index.tsx
index 237a032..b5bddcc 100644
--- a/app/(tabs)/index.tsx
+++ b/app/(tabs)/index.tsx
@@ -1,14 +1,38 @@
-import { Alert, Platform, Pressable, StyleSheet, Text, TextInput, View } from 'react-native';
-import ParallaxScrollView from '@/components/ParallaxScrollView';
-import { ThemedText } from '@/components/ThemedText';
-import { ThemedView } from '@/components/ThemedView';
-import React, { useState, useEffect, useRef } from 'react';
-import AsyncStorage from '@react-native-async-storage/async-storage';
-import MapView, { Marker } from 'react-native-maps';
-import * as Notifications from 'expo-notifications';
-import * as Device from 'expo-device';
-import Constants from 'expo-constants';
-import { Link } from 'expo-router';
+import {
+ Alert,
+ Platform,
+ Pressable,
+ StyleSheet,
+ Text,
+ TextInput,
+ View,
+} from "react-native";
+import ParallaxScrollView from "@/components/ParallaxScrollView";
+import { ThemedText } from "@/components/ThemedText";
+import { ThemedView } from "@/components/ThemedView";
+import React, { useState, useEffect, useRef } from "react";
+import AsyncStorage from "@react-native-async-storage/async-storage";
+import MapView, { LatLng, Marker } from "react-native-maps";
+import * as Notifications from "expo-notifications";
+import * as Device from "expo-device";
+import * as Location from "expo-location";
+import Constants from "expo-constants";
+import { Link } from "expo-router";
+import * as TaskManager from "expo-task-manager";
+
+const LOCATION_TASK_NAME = "background-location-task";
+
+TaskManager.defineTask(LOCATION_TASK_NAME, async ({ data, error }) => {
+ if (error) {
+ console.error(error);
+ return;
+ }
+ if (data) {
+ const { locations } = data;
+ console.log("Received new locations:", locations);
+ // Process the locations here
+ }
+});
interface NotificationData {
id: string;
@@ -23,36 +47,39 @@ Notifications.setNotificationHandler({
}),
});
-
function handleRegistrationError(errorMessage: string) {
Alert.alert("Error registering this device", errorMessage);
}
async function registerForPushNotificationsAsync() {
- if (Platform.OS === 'android') {
- Notifications.setNotificationChannelAsync('default', {
- name: 'default',
+ if (Platform.OS === "android") {
+ Notifications.setNotificationChannelAsync("default", {
+ name: "default",
importance: Notifications.AndroidImportance.MAX,
vibrationPattern: [0, 250, 250, 250],
- lightColor: '#007AFF',
+ lightColor: "#007AFF",
});
}
if (Device.isDevice) {
- const { status: existingStatus } = await Notifications.getPermissionsAsync();
+ const { status: existingStatus } =
+ await Notifications.getPermissionsAsync();
let finalStatus = existingStatus;
- if (existingStatus !== 'granted') {
+ if (existingStatus !== "granted") {
const { status } = await Notifications.requestPermissionsAsync();
finalStatus = status;
}
- if (finalStatus !== 'granted') {
- handleRegistrationError('Permission not granted to get push token for push notification!');
+ if (finalStatus !== "granted") {
+ handleRegistrationError(
+ "Permission not granted to get push token for push notification!",
+ );
return;
}
const projectId =
- Constants?.expoConfig?.extra?.eas?.projectId ?? Constants?.easConfig?.projectId;
+ Constants?.expoConfig?.extra?.eas?.projectId ??
+ Constants?.easConfig?.projectId;
if (!projectId) {
- handleRegistrationError('Project ID not found');
+ handleRegistrationError("Project ID not found");
}
try {
const pushTokenString = (
@@ -65,71 +92,84 @@ async function registerForPushNotificationsAsync() {
handleRegistrationError(`${e}`);
}
} else {
- handleRegistrationError('Must use physical device for push notifications');
+ handleRegistrationError("Must use physical device for push notifications");
}
}
export default function HomeScreen() {
- const [email, setEmail] = useState('');
- const [password, setPassword] = useState('');
- const [token, setToken] = useState('');
- const [userId, setUserId] = useState('');
- const [coordinates, setCoordinates] = useState({ latitude: 44.49738301084014, longitude: 11.356121722966094 });
- const [region, setRegion] = useState({ latitude: 44.49738301084014, longitude: 11.356121722966094, latitudeDelta: 0.03, longitudeDelta: 0.03 });
- const [notification, setNotification] = useState<NotificationData|null>(null);
+ const [email, setEmail] = useState("");
+ const [password, setPassword] = useState("");
+ const [token, setToken] = useState("");
+ const [userId, setUserId] = useState("");
+ const [coordinates, setCoordinates] = useState({
+ latitude: 0,
+ longitude: 0,
+ });
+ const [region, setRegion] = useState({
+ latitude: 0,
+ longitude: 0,
+ latitudeDelta: 0.03,
+ longitudeDelta: 0.03,
+ });
+ const [notification, setNotification] = useState<NotificationData | null>(
+ null,
+ );
const mapRef = useRef(null);
const storeToken = async (token: string) => {
- if (Platform.OS === 'web') {
- localStorage.setItem('token', token);
+ if (Platform.OS === "web") {
+ localStorage.setItem("token", token);
} else {
- await AsyncStorage.setItem('token', token);
+ await AsyncStorage.setItem("token", token);
}
};
const storeUserId = async (userId: string) => {
- if (Platform.OS === 'web') {
- localStorage.setItem('userId', userId);
+ if (Platform.OS === "web") {
+ localStorage.setItem("userId", userId);
} else {
- await AsyncStorage.setItem('userId', userId);
+ await AsyncStorage.setItem("userId", userId);
}
};
const handleLogin = async () => {
if (!email || !password) {
- Alert.alert('Error', 'Email and password are required.');
+ Alert.alert("Error", "Email and password are required.");
return;
}
try {
- const response = await fetch(`${process.env.EXPO_PUBLIC_API_URL}/graphql`, {
- method: 'POST',
- headers: {
- 'Content-Type': 'application/json',
- },
- body: JSON.stringify({
- query: `
- mutation Login($input: LoginCredentials!) {
- login(input: $input) {
- accessToken
- tokenType
+ const response = await fetch(
+ `${process.env.EXPO_PUBLIC_API_URL}/graphql`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify({
+ query: `
+ mutation Login($input: LoginCredentials!) {
+ login(input: $input) {
+ accessToken
+ tokenType
userId
- }
+ }
}`,
- variables: {
- input: {
- email: email,
- password,
+ variables: {
+ input: {
+ email: email,
+ password,
+ },
},
- },
- }),
- });
+ }),
+ },
+ );
const data = await response.json();
if (data.errors) {
const errorMessages = data.errors.map((error: any) => error.message);
- Alert.alert('Error', errorMessages.join('\n'));
+ Alert.alert("Error", errorMessages.join("\n"));
} else {
const { accessToken, userId } = data.data.login;
await storeToken(accessToken);
@@ -138,40 +178,40 @@ export default function HomeScreen() {
setUserId(String(userId));
registerForPushNotificationsAsync()
- .then(async notificationToken => {
+ .then(async (notificationToken) => {
if (!notificationToken) return;
const regex = /ExponentPushToken\[(.*?)\]/;
const match = notificationToken.match(regex);
if (match && match[1]) {
- notificationToken = match[1];
+ notificationToken = match[1];
}
- await fetch(`${process.env.EXPO_PUBLIC_API_URL}/graphql`, {
- method: 'POST',
- headers: {
- 'Authorization': `Bearer ${accessToken}`,
- 'Content-Type': 'application/json',
- },
- body: JSON.stringify({
- query: `
+ await fetch(`${process.env.EXPO_PUBLIC_API_URL}/graphql`, {
+ method: "POST",
+ headers: {
+ Authorization: `Bearer ${accessToken}`,
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify({
+ query: `
mutation RegisterDevice($input: RegisterNotificationToken!) {
registerDevice(input: $input) { id name email }
}
`,
- variables: {
- input: {
- token: notificationToken,
- },
+ variables: {
+ input: {
+ token: notificationToken,
},
- }),
- })
+ },
+ }),
+ });
})
.catch((error: any) => alert(`${error}`));
}
} catch (err) {
- console.error('Login Error:', err);
- Alert.alert('Error', 'An error occurred during login.');
+ console.error("Login Error:", err);
+ Alert.alert("Error", "An error occurred during login.");
}
};
@@ -180,53 +220,74 @@ export default function HomeScreen() {
};
const removeToken = async () => {
- if (Platform.OS === 'web') {
- localStorage.removeItem('token');
- localStorage.removeItem('userId');
+ if (Platform.OS === "web") {
+ localStorage.removeItem("token");
+ localStorage.removeItem("userId");
} else {
- await AsyncStorage.removeItem('token');
- await AsyncStorage.removeItem('userId');
+ await AsyncStorage.removeItem("token");
+ await AsyncStorage.removeItem("userId");
}
- setToken('');
- setUserId('');
+ setToken("");
+ setUserId("");
+ };
+
+ const formatDate = (timestamp: string) => {
+ const date = new Date(parseInt(timestamp) * 1000);
+ return `${date.toDateString()} ${date.getHours()}:${(date.getMinutes() < 10 ? "0" : "") + date.getMinutes()}`;
};
- const fetchMapData = async () => {
+ const updateLocation = async (coords: LatLng) => {
+ setCoordinates({
+ latitude: coords.latitude,
+ longitude: coords.longitude,
+ });
+
+ if (region.longitude == 0 && region.latitude == 0) {
+ setRegion({
+ latitude: coords.latitude,
+ longitude: coords.longitude,
+ latitudeDelta: 0.03,
+ longitudeDelta: 0.03,
+ });
+ }
+
+
if (!token || !userId) return;
try {
- const response = await fetch(`${process.env.EXPO_PUBLIC_API_URL}/graphql`, {
- method: 'POST',
- headers: {
- 'Authorization': `Bearer ${token}`,
- 'Content-Type': 'application/json',
+ const response = await fetch(
+ `${process.env.EXPO_PUBLIC_API_URL}/graphql`,
+ {
+ method: "POST",
+ headers: {
+ Authorization: `Bearer ${token}`,
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify({
+ query: `
+ mutation NewPosition($input: PositionInput!) {
+ newPosition(input: $input) {
+ id userId createdAt latitude longitude movingActivity
+ }
+ }
+ `,
+ variables: {
+ input: {
+ latitude: coords.latitude,
+ longitude: coords.longitude,
+ movingActivity: 'STILL',
+ },
+ },
+ }),
},
- body: JSON.stringify({
- query: `{ positions(userId: ${userId}) { id, userId, createdAt, latitude, longitude } }`,
- }),
- });
-
+ );
const data = await response.json();
-
- if (data.data.positions && data.data.positions.length > 0) {
- const position = data.data.positions[0];
- setCoordinates({ latitude: position.latitude, longitude: position.longitude });
- setRegion({
- latitude: position.latitude,
- longitude: position.longitude,
- latitudeDelta: 0.03,
- longitudeDelta: 0.03,
- });
- }
+ console.log(data)
} catch (err) {
- console.error('Fetch Map Data Error:', err);
+ console.error("Error on updating position");
}
- };
- const formatDate = (timestamp: string) => {
- const date = new Date(parseInt(timestamp) * 1000);
- return `${date.toDateString()} ${date.getHours()}:${(date.getMinutes() < 10 ? '0' : '') + date.getMinutes()}`;
- };
+ }
useEffect(() => {
const fetchNotifications = async () => {
@@ -236,15 +297,15 @@ export default function HomeScreen() {
const response = await fetch(
`${process.env.EXPO_PUBLIC_API_URL}/graphql`,
{
- method: 'POST',
+ method: "POST",
headers: {
Authorization: `Bearer ${token}`,
- 'Content-Type': 'application/json',
+ "Content-Type": "application/json",
},
body: JSON.stringify({
query: `{ notifications(seen: false) { id, createdAt } }`,
}),
- }
+ },
);
const data = await response.json();
@@ -253,45 +314,75 @@ export default function HomeScreen() {
setNotification(data.data.notifications[0]);
}
} catch (err) {
- console.error('Fetch notifications:', err);
+ console.error("Fetch notifications:", err);
}
};
if (token && userId) {
- const intervalId = setInterval(fetchNotifications, 2000);
+ const intervalId = setInterval(fetchNotifications, 10000);
return () => clearInterval(intervalId);
} else {
- setNotification('');
+ setNotification("");
}
}, [token, userId]);
-
useEffect(() => {
const retrieveToken = async () => {
- const storedToken = Platform.OS === 'web' ? localStorage.getItem('token') : await AsyncStorage.getItem('token');
- setToken(storedToken || '');
- const storedUserId = Platform.OS === 'web' ? localStorage.getItem('userId') : await AsyncStorage.getItem('userId');
- setUserId(storedUserId || '');
+ const storedToken =
+ Platform.OS === "web"
+ ? localStorage.getItem("token")
+ : await AsyncStorage.getItem("token");
+ setToken(storedToken || "");
+ const storedUserId =
+ Platform.OS === "web"
+ ? localStorage.getItem("userId")
+ : await AsyncStorage.getItem("userId");
+ setUserId(storedUserId || "");
};
retrieveToken();
}, []);
useEffect(() => {
- if (token && userId) {
- const intervalId = setInterval(fetchMapData, 100000);
-
- return () => clearInterval(intervalId);
- }
- }, [token, userId]);
-
- useEffect(() => {
if (mapRef.current && region) {
mapRef.current.animateToRegion(region, 1000);
}
}, [region]);
+ useEffect(() => {
+ const startBackgroundLocationTracking = async () => {
+ try {
+ const { status } = await Location.requestForegroundPermissionsAsync();
+ if (status === "granted") {
+ setInterval(async () => {
+ const location = await Location.getCurrentPositionAsync({});
+ updateLocation(location.coords);
+ }, 2000);
+
+ await Location.startLocationUpdatesAsync(LOCATION_TASK_NAME, {
+ accuracy: Location.Accuracy.Balanced,
+ distanceInterval: 0,
+ deferredUpdatesInterval: 1000,
+ showsBackgroundLocationIndicator: true,
+ foregroundService: {
+ notificationTitle: "CAS4 Location Tracking",
+ notificationBody:
+ "Your location is being tracked in the background.",
+ notificationColor: "#FFFFFF",
+ },
+ });
+ } else {
+ Alert.alert("Background location permission not granted");
+ }
+ } catch (error) {
+ console.error("Error starting background location updates:", error);
+ }
+ };
+
+ startBackgroundLocationTracking();
+ }, []);
+
return (
<ParallaxScrollView token={token} userId={userId}>
{token && userId ? (
@@ -300,29 +391,31 @@ export default function HomeScreen() {
<View>
<Link
href={`/notifications/${notification.id}`}
- style={{ width: '100%' }}
+ style={{ width: "100%" }}
>
- <View style={styles.notificationBox}>
- <Text style={styles.notificationBoxText}>Oh no, you are (or have been) in an alerted area in {formatDate(notification.createdAt)}!</Text>
- <Text style={styles.notificationBoxText}>Click this banner to know more!</Text>
- </View>
+ <View style={styles.notificationBox}>
+ <Text style={styles.notificationBoxText}>
+ Oh no, you are (or have been) in an alerted area in{" "}
+ {formatDate(notification.createdAt)}!
+ </Text>
+ <Text style={styles.notificationBoxText}>
+ Click this banner to know more!
+ </Text>
+ </View>
</Link>
</View>
) : (
- <>
- </>
+ <></>
)}
<ThemedView>
<Pressable onPress={handleLogout} style={styles.formButton}>
- <Text style={{ color: 'white', textAlign: 'center' }}>Logout</Text>
+ <Text style={{ color: "white", textAlign: "center" }}>
+ Logout
+ </Text>
</Pressable>
</ThemedView>
<View>
- <MapView
- ref={mapRef}
- initialRegion={region}
- style={styles.map}
- >
+ <MapView ref={mapRef} initialRegion={region} style={styles.map}>
<Marker coordinate={coordinates} title="Me" />
</MapView>
</View>
@@ -352,7 +445,9 @@ export default function HomeScreen() {
</View>
<View style={styles.buttonContainer}>
<Pressable onPress={handleLogin} style={styles.formButton}>
- <Text style={{ color: 'white', textAlign: 'center' }}>Login</Text>
+ <Text style={{ color: "white", textAlign: "center" }}>
+ Login
+ </Text>
</Pressable>
</View>
</ThemedView>
@@ -364,8 +459,8 @@ export default function HomeScreen() {
const styles = StyleSheet.create({
titleContainer: {
- flexDirection: 'row',
- alignItems: 'center',
+ flexDirection: "row",
+ alignItems: "center",
marginBottom: 20,
},
text: {
@@ -376,13 +471,13 @@ const styles = StyleSheet.create({
paddingHorizontal: 20,
},
formInput: {
- width: '100%',
+ width: "100%",
paddingVertical: 12,
paddingHorizontal: 16,
borderRadius: 8,
borderWidth: 1,
- borderColor: '#ccc',
- backgroundColor: '#f9f9f9',
+ borderColor: "#ccc",
+ backgroundColor: "#f9f9f9",
marginBottom: 20,
},
buttonContainer: {
@@ -391,12 +486,12 @@ const styles = StyleSheet.create({
formButton: {
paddingVertical: 12,
paddingHorizontal: 24,
- backgroundColor: '#007AFF',
+ backgroundColor: "#007AFF",
fontSize: 16,
- fontWeight: '600',
- textAlign: 'center',
+ fontWeight: "600",
+ textAlign: "center",
borderRadius: 8,
- color: 'white',
+ color: "white",
},
map: {
height: 400,
@@ -404,15 +499,15 @@ const styles = StyleSheet.create({
notificationBox: {
padding: 40,
borderRadius: 8,
- shadowColor: '#000',
+ shadowColor: "#000",
shadowOffset: { width: 0, height: 2 },
shadowOpacity: 0.1,
shadowRadius: 4,
- backgroundColor: '#EA2027',
+ backgroundColor: "#EA2027",
},
notificationBoxText: {
- color: '#fff',
- textAlign: 'center',
- fontWeight: 'bold'
- }
+ color: "#fff",
+ textAlign: "center",
+ fontWeight: "bold",
+ },
});