Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Backend refactor [WIP] #310

Merged
merged 21 commits into from
Jan 15, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
67 changes: 67 additions & 0 deletions api.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
Websocket API Protocol

=== Objects ===

"""
ObjectName {
requiredProp: type
optionalProp: type?
}
"""

Location {
lat: float,
lon: float,
geohash: string? (generated if not provided)
}

Message {
author: string,
content: {
text?: string,
attachment?: string,
},
location: Location,
replyTo?: string,
reactions: {
[key: string]: int,
}
}

UserProfile {
displayName: string,
profilePicture: int,
}

===

=== Client -> Server Methods ===

"""
methodName(
arguments / inputs...
) ack -> ackResponse
"""

// Must be called at least once before calling any other methods
updateLocation(
location: Location
ack: func?
) ack -> "success"

sendMessage(
message: Message
ack: func?
) ack -> "success"

getNearbyUsers(
callback: func(nearbyUserUids: { [uid: string]: UserProfile }) -> _,
) callback -> map of nearby user uids to user profiles

// Call after the client has already updated their profile document in firestore
notifyUpdateProfile(
ack: func?
) ack -> "success"


===
5 changes: 2 additions & 3 deletions client/app.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,9 +12,7 @@
"resizeMode": "contain",
"backgroundColor": "34D1BF"
},
"assetBundlePatterns": [
"**/*"
],
"assetBundlePatterns": ["**/*"],
"ios": {
"supportsTablet": true
},
Expand All @@ -27,6 +25,7 @@
"web": {
"favicon": "./assets/favicon.png"
},
"newArchEnabled": true,
"plugins": [
"expo-router",
[
Expand Down
50 changes: 0 additions & 50 deletions client/app/components/auth/AuthButtons.tsx
Original file line number Diff line number Diff line change
@@ -1,65 +1,15 @@
import { useFonts } from "expo-font";

Check warning on line 1 in client/app/components/auth/AuthButtons.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'useFonts' is defined but never used
import React, { useState } from "react";

Check warning on line 2 in client/app/components/auth/AuthButtons.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'useState' is defined but never used
import {
StyleSheet,
Text,

Check warning on line 5 in client/app/components/auth/AuthButtons.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'Text' is defined but never used
TouchableOpacity,
Dimensions,
Alert,

Check warning on line 8 in client/app/components/auth/AuthButtons.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'Alert' is defined but never used
Image,
ImageSourcePropType,
} from "react-native";

// Migrated to SettingsScree.tsx

// import { appSignOut } from "../../services/AuthStore";
// interface SignOutButtonProps {}
// export const SignOutButton: React.FC<SignOutButtonProps> = () => {
// const [loading, setLoading] = useState(false);

// const handleSignOut = async () => {
// Alert.alert(
// "Confirm Sign Out",
// "Are you sure you want to sign out?",
// [
// {
// text: "Cancel",
// style: "cancel",
// },
// {
// text: "Sign Out",
// onPress: async () => {
// setLoading(true);
// const response = await appSignOut();
// setLoading(false);

// if (response?.user === null) {
// console.log("Sign out successful");
// } else if (response?.error) {
// console.log(response.error);
// Alert.alert(
// "Sign Out Failed",
// "An error occurred during sign out. Please try again.",
// );
// }
// },
// },
// ],
// { cancelable: false },
// );
// };

// return (
// <TouchableOpacity
// style={styles.sign_out_button}
// onPress={handleSignOut}
// disabled={loading}>
// <Text style={styles.button_text}>Sign Out</Text>
// </TouchableOpacity>

// );
// };

export const ExternalAuthButton: React.FC<{
onPress?: () => void;
companyName: string;
Expand Down
6 changes: 3 additions & 3 deletions client/app/components/chat/ChatScreenFooter.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,9 @@
StyleSheet,
Dimensions,
Platform,
TouchableOpacity,

Check warning on line 8 in client/app/components/chat/ChatScreenFooter.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'TouchableOpacity' is defined but never used
} from "react-native";
import { Smile, Image } from "react-native-feather";

Check warning on line 10 in client/app/components/chat/ChatScreenFooter.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'Smile' is defined but never used

Check warning on line 10 in client/app/components/chat/ChatScreenFooter.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'Image' is defined but never used

import { ChatSendButton } from "../common/CustomButtons";

Expand All @@ -26,12 +26,12 @@
return (
<View style={styles.container}>
<View style={styles.iconContainer}>
<TouchableOpacity>
{/* <TouchableOpacity>
<Image color="black" strokeWidth={1.8} style={styles.icons} />
</TouchableOpacity>
<TouchableOpacity>
<Smile color="black" strokeWidth={1.8} style={styles.icons} />
</TouchableOpacity>
</TouchableOpacity> */}
</View>
<TextInput
placeholder="Say Something..."
Expand All @@ -57,7 +57,7 @@
borderWidth: 1,
borderRadius: Dimensions.get("window").width * 0.058,
marginHorizontal: Dimensions.get("window").width * 0.005,
marginBottom: Platform.OS === "ios" ? 0 : 5,
marginBottom: 3,
minHeight: Dimensions.get("window").width * 0.113,
maxHeight: Dimensions.get("window").width * 0.3,
},
Expand Down
25 changes: 16 additions & 9 deletions client/app/components/chat/MessageChannel.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,10 @@ import { FlatList } from "react-native";
import Message from "./ChatMessage";
import { MessageChannelProps } from "../../types/Props";

const MessageChannel: React.FC<MessageChannelProps> = ({ messages }) => {
const MessageChannel: React.FC<MessageChannelProps> = ({
nearbyUsers,
messages,
}) => {
const reverseMessages = [...messages].reverse();

return (
Expand All @@ -13,14 +16,18 @@ const MessageChannel: React.FC<MessageChannelProps> = ({ messages }) => {
width: "100%",
}}
data={reverseMessages}
keyExtractor={(item) => item.msgId}
renderItem={({ item }) => (
<Message
messageContent={item.msgContent}
author={item.author.displayName} // TODO: call server to get author name from UID. Or should this stored with MessageType?
time={item.timestamp}
/>
)}
renderItem={({ item }) => {
const user = nearbyUsers[item.author];
// console.log(nearbyUsers);
if (user === undefined) return null;
return (
<Message
messageContent={item.content.text!}
author={user.displayName}
time={item.timestamp}
/>
);
}}
inverted // This will render items from the bottom
onLayout={() => {}} // This will make sure the list is scrolled to the bottom on first render
/>
Expand Down
31 changes: 20 additions & 11 deletions client/app/components/chat/NearbyHeader.tsx
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { UserProfile } from "@app/types/User";
import React from "react";
import {
View,
Expand All @@ -7,20 +8,27 @@
Image,
TouchableOpacity,
} from "react-native";
import { ChevronLeft } from "react-native-feather";

Check warning on line 11 in client/app/components/chat/NearbyHeader.tsx

View workflow job for this annotation

GitHub Actions / lint (21.x)

'ChevronLeft' is defined but never used

export const NearbyHeader: React.FC = () => {
interface NearbyHeaderProps {
onClick: () => void;
nearbyUsers: { [uid: string]: UserProfile };
}

export const NearbyHeader: React.FC<NearbyHeaderProps> = ({ onClick, nearbyUsers }) => {
return (
<View style={styles.nearbyContainer}>
<ChevronLeft color="white" strokeWidth={1.4} width={40} height={40} />
<Text style={styles.nearbyText}>Nearby</Text>
<View style={styles.iconContainer}>
<Image
style={styles.peopleIcon}
source={require("../../../assets/icons/misc/nearby_icon.png")}
/>
<Text style={styles.countText}>{5}</Text>
</View>
<TouchableOpacity onPress={onClick}>
<View style={styles.iconContainer}>
<Image
style={styles.peopleIcon}
source={require("../../../assets/icons/misc/nearby_icon.png")}
/>

<Text style={styles.countText}>{Object.keys(nearbyUsers).length}</Text>
</View>
</TouchableOpacity>
</View>
);
};
Expand All @@ -38,9 +46,10 @@
shadowOpacity: 0.3,
shadowRadius: 2,
paddingVertical: 15,
paddingRight: 25,
paddingLeft: 10,
paddingRight: "5%",
paddingLeft: "10%",
gap: 10,
zIndex: 1,
},
nearbyText: {
fontFamily: "Quicksand",
Expand Down
121 changes: 121 additions & 0 deletions client/app/components/chat/NearbyUserDrawer.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
import React, { useState } from "react";
import { View, Text, StyleSheet, FlatList, Dimensions, Image } from "react-native";
import Animated, {
useSharedValue,
useAnimatedStyle,
withTiming,
Easing,
} from "react-native-reanimated";
import NearbyHeader from "./NearbyHeader";
import { Pressable } from "react-native";
import { UserProfile } from "@app/types/User";

const SCREEN_WIDTH = Dimensions.get("window").width;

interface NearbyUserDrawerProps {
nearbyUsers: { [uid: string]: UserProfile };
}

const NearbyUserDrawer: React.FC<NearbyUserDrawerProps> = ({ nearbyUsers }) => {
const [isOpen, setIsOpen] = useState(false);
const translateX = useSharedValue(SCREEN_WIDTH);

const toggleDrawer = () => {
translateX.value = isOpen
? withTiming(SCREEN_WIDTH, {
duration: 300, // Duration in milliseconds
easing: Easing.out(Easing.ease), // Smooth easing out
})
: withTiming(0, {
duration: 300,
easing: Easing.out(Easing.ease), // Smooth easing out
});
setIsOpen(!isOpen);
};

const animatedStyle = useAnimatedStyle(() => ({
transform: [{ translateX: translateX.value }],
}));

return (
<View style={styles.container}>
<NearbyHeader onClick={toggleDrawer} nearbyUsers={nearbyUsers} />
{/* Overlay and Drawer */}
{isOpen && <Pressable style={styles.overlay} onPress={toggleDrawer} />}
<Animated.View style={[styles.drawer, animatedStyle]}>
<Text style={styles.headerText}>Nearby Users</Text>
<FlatList
data={Object.entries(nearbyUsers)}
keyExtractor={(item, index) => item[0]}
renderItem={({ item }) => {
return (
<View style= {{
width: "100%",
display: "flex",
flexDirection: "row",
}}>
<Image
style={{
height: Dimensions.get("window").height * 0.055,
width: Dimensions.get("window").height * 0.055,
borderRadius: 100,
marginRight: 12,
}}
source={require("../../../assets/icons/user/fake_pfp.jpg")}
/>
<Text style={styles.name}>{item[1].displayName}</Text>
</View>
);
}}
/>
</Animated.View>
</View>
);
};

const styles = StyleSheet.create({
container: {
display: "flex",
height: Dimensions.get("screen").height,
flexDirection: "column",
alignItems: "center",
position: "absolute",
},
overlay: {
position: "absolute",
top: 0,
left: 0,
width: "100%",
height: "100%",
backgroundColor: "rgba(0, 0, 0, 0.15)",
zIndex: 1,
},
drawer: {
position: "absolute",
right: 0,
top: 0,
width: SCREEN_WIDTH * 0.6,
height: "100%",
backgroundColor: "#fff",
borderLeftWidth: 1,
borderLeftColor: "#ddd",
padding: 30,
shadowColor: "#000",
shadowOffset: { width: -2, height: 0 },
shadowOpacity: 0.3,
shadowRadius: 4,
zIndex: 2,
gap: 20,
},
headerText: {
fontFamily: "Quicksand",
fontSize: 24,
fontWeight: "bold",
},
name: {
fontSize: 18,
marginVertical: 10,
},
});

export default NearbyUserDrawer;
Loading
Loading