2024-10-16 16:50:26 -05:00
|
|
|
import React, { useEffect, useState } from 'react';
|
2024-10-30 15:03:53 -05:00
|
|
|
import { io } from 'socket.io-client';
|
2024-10-16 16:50:26 -05:00
|
|
|
import { Image, StyleSheet, AppState } from 'react-native';
|
|
|
|
import { ThemedText, ThemedView } from '@/components/theme/Theme';
|
|
|
|
import { getUser, getRelationship, getPartner, saveRelationshipData } from '@/components/services/SecureStore';
|
|
|
|
import RequestRelationship from '@/components/home/RequestRelationship';
|
|
|
|
import TextButton from '@/components/theme/buttons/TextButton';
|
|
|
|
import { checkRelationshipStatus, updateRelationshipStatus } from '@/constants/APIs';
|
|
|
|
import type { User, Relationship, RelationshipData } from '@/constants/Types';
|
|
|
|
|
|
|
|
type RelationshipProps = {
|
|
|
|
pfpUrl: string | null;
|
|
|
|
};
|
|
|
|
|
2024-10-18 12:39:54 -05:00
|
|
|
const RelationshipView: React.FC<RelationshipProps> = ({ pfpUrl }) => {
|
2024-10-16 16:50:26 -05:00
|
|
|
const [status, setStatus] = useState<RelationshipData | null>(null);
|
|
|
|
const [user, setUser] = useState<User | null>(null);
|
2024-10-30 15:03:53 -05:00
|
|
|
const [relationship, setRelationship] = useState<Relationship | null>(null);
|
2024-10-16 16:50:26 -05:00
|
|
|
const [showRequestRelationship, setShowRequestRelationship] = useState(false);
|
|
|
|
const [loading, setLoading] = useState(true);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
fetchRelationshipStatus();
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (pfpUrl && user) {
|
|
|
|
setUser(prevUser =>
|
|
|
|
prevUser ? {...prevUser, pfpUrl: pfpUrl} : null);
|
|
|
|
}
|
|
|
|
}, [pfpUrl]);
|
|
|
|
|
2024-10-30 15:03:53 -05:00
|
|
|
useEffect(() => {
|
|
|
|
if (!user || !relationship) {
|
|
|
|
console.log('User or relationship not found');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const socket = io(process.env.EXPO_PUBLIC_WEBSOCKET_URL as string, {
|
|
|
|
transports: ['websocket'],
|
|
|
|
});
|
|
|
|
socket.on('connect', () => {
|
|
|
|
console.log('Connected to WebSocket server');
|
|
|
|
socket.emit('join', { relationshipId: relationship.id });
|
|
|
|
});
|
|
|
|
socket.on('connect_error', (error) => {
|
|
|
|
console.error('Error connecting to WebSocket server:', error);
|
|
|
|
});
|
|
|
|
socket.on('relationship_status_change', async (relationshipStatus) => {
|
2024-10-16 16:50:26 -05:00
|
|
|
handleCheckRelationshipStatus();
|
2024-10-30 15:03:53 -05:00
|
|
|
});
|
2024-10-16 16:50:26 -05:00
|
|
|
return () => {
|
2024-10-30 15:03:53 -05:00
|
|
|
socket.disconnect();
|
2024-10-16 16:50:26 -05:00
|
|
|
};
|
2024-10-30 15:03:53 -05:00
|
|
|
}, [relationship]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleRequestSent = (relationshipData: RelationshipData) => {
|
|
|
|
setStatus(relationshipData);
|
|
|
|
setShowRequestRelationship(false);
|
2024-10-16 16:50:26 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
const fetchRelationshipStatus = async () => {
|
|
|
|
setLoading(true);
|
|
|
|
try {
|
|
|
|
const userFromStore: User = await getUser() as User;
|
|
|
|
if (!userFromStore) throw new Error('User not found in store.');
|
|
|
|
setUser(userFromStore);
|
|
|
|
const relationshipFromStore: Relationship = await getRelationship() as Relationship;
|
2024-10-30 15:03:53 -05:00
|
|
|
setRelationship(relationshipFromStore);
|
2024-10-16 16:50:26 -05:00
|
|
|
const partnerFromStore: User = await getPartner() as User;
|
|
|
|
if (!relationshipFromStore || !partnerFromStore)
|
|
|
|
throw new Error('Relationship not found in store.');
|
|
|
|
setStatus({
|
|
|
|
relationship: relationshipFromStore,
|
|
|
|
partner: partnerFromStore,
|
|
|
|
});
|
|
|
|
await handleCheckRelationshipStatus();
|
|
|
|
} catch (error) {
|
|
|
|
console.log('Error fetching relationship status:', error);
|
|
|
|
} finally {
|
|
|
|
setLoading(false);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleCheckRelationshipStatus = async () => {
|
|
|
|
try {
|
|
|
|
const userFromStore: User = await getUser() as User;
|
|
|
|
if (!userFromStore) throw new Error('User not found in store.');
|
|
|
|
const relationshipData: RelationshipData = await checkRelationshipStatus(userFromStore.id);
|
|
|
|
setStatus(relationshipData);
|
|
|
|
await saveRelationshipData(relationshipData);
|
|
|
|
} catch (error) {
|
|
|
|
console.log('No relationship found or error checking relationship status:', error);
|
|
|
|
setStatus(null);
|
|
|
|
} finally {
|
|
|
|
setLoading(false);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleUpdateRelationshipStatus = async (newStatus: 'accepted' | 'rejected') => {
|
|
|
|
if (!status || !status.relationship || !user || !user.id) {
|
|
|
|
console.error('No relationship found.');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
if (newStatus === 'accepted') {
|
|
|
|
const updatedRelationshipData: RelationshipData =
|
2024-10-30 15:03:53 -05:00
|
|
|
await updateRelationshipStatus(user.id, newStatus) as RelationshipData;
|
2024-10-16 16:50:26 -05:00
|
|
|
setStatus(updatedRelationshipData);
|
|
|
|
await saveRelationshipData(updatedRelationshipData);
|
|
|
|
} else {
|
|
|
|
await updateRelationshipStatus(user.id, newStatus);
|
|
|
|
console.log('Rejected relationship. Relationship deleted.');
|
|
|
|
setStatus(null);
|
|
|
|
}
|
|
|
|
} catch (error) {
|
2024-10-21 11:04:42 -05:00
|
|
|
console.log('Error updating relationship status:', error);
|
2024-10-16 16:50:26 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
const handleAcceptRequest = () => handleUpdateRelationshipStatus('accepted');
|
|
|
|
const handleRejectRequest = () => handleUpdateRelationshipStatus('rejected');
|
|
|
|
|
|
|
|
if (loading) {
|
|
|
|
return (
|
|
|
|
<ThemedText>
|
|
|
|
Loading...
|
|
|
|
</ThemedText>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const renderRelationshipContent = () => {
|
|
|
|
if (!status || !status.relationship) {
|
|
|
|
// Case 1: Not in a relationship
|
|
|
|
return showRequestRelationship ? (
|
2024-10-21 12:32:53 -05:00
|
|
|
<ThemedView>
|
|
|
|
<RequestRelationship onRequestSent={handleRequestSent} />
|
|
|
|
<TextButton
|
|
|
|
width={220} height={60} text='Stop searching' fontSize={18}
|
|
|
|
onPress={() => setShowRequestRelationship(false)}
|
|
|
|
/>
|
|
|
|
</ThemedView>
|
2024-10-16 16:50:26 -05:00
|
|
|
) : (
|
|
|
|
<TextButton
|
|
|
|
width={220} height={60} text='Request Relationship' fontSize={18}
|
|
|
|
onPress={() => setShowRequestRelationship(true)}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
} else if (!status.relationship.isAccepted && user?.id === status.relationship.requestorId) {
|
|
|
|
// Case 2: Pending relationship & our user requested it
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<ThemedText style={styles.title}>
|
|
|
|
Pending Relationship
|
|
|
|
</ThemedText>
|
|
|
|
<ThemedView style={styles.profileContainer}>
|
|
|
|
<ThemedView style={styles.profileWrapper}>
|
|
|
|
<Image
|
|
|
|
source={{
|
2024-10-18 14:27:27 -05:00
|
|
|
uri: `${process.env.EXPO_PUBLIC_API_URL}` +
|
2024-10-16 16:50:26 -05:00
|
|
|
`${status.partner.pfpUrl}`
|
|
|
|
}}
|
|
|
|
style={styles.profilePicture}
|
|
|
|
/>
|
|
|
|
<ThemedText style={styles.name}>
|
|
|
|
{status.partner.fullName}
|
|
|
|
</ThemedText>
|
|
|
|
</ThemedView>
|
|
|
|
</ThemedView>
|
|
|
|
<ThemedView style={styles.buttonContainer}>
|
|
|
|
<TextButton
|
2024-10-21 12:32:53 -05:00
|
|
|
width={150} height={40}
|
|
|
|
text='Cancel Request' fontSize={18}
|
2024-10-16 16:50:26 -05:00
|
|
|
onPress={handleRejectRequest}
|
|
|
|
/>
|
|
|
|
</ThemedView>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
} else if (!status.relationship.isAccepted && user?.id !== status.relationship.requestorId) {
|
|
|
|
// Case 3: Pending relationship & someone else requested it
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<ThemedText style={styles.title}>
|
|
|
|
Pending Relationship
|
|
|
|
</ThemedText>
|
|
|
|
<ThemedView style={styles.profileContainer}>
|
|
|
|
<ThemedView style={styles.profileWrapper}>
|
|
|
|
<Image
|
|
|
|
source={{
|
2024-10-18 14:27:27 -05:00
|
|
|
uri: `${process.env.EXPO_PUBLIC_API_URL}` +
|
2024-10-16 16:50:26 -05:00
|
|
|
`${status.partner.pfpUrl}`
|
|
|
|
}}
|
|
|
|
style={styles.profilePicture}
|
|
|
|
/>
|
|
|
|
<ThemedText style={styles.name}>
|
|
|
|
{status.partner.fullName}
|
|
|
|
</ThemedText>
|
|
|
|
</ThemedView>
|
|
|
|
</ThemedView>
|
|
|
|
<ThemedView style={styles.buttonContainer}>
|
|
|
|
<TextButton
|
|
|
|
width={100} height={40}
|
|
|
|
text='Accept' fontSize={18}
|
|
|
|
onPress={handleAcceptRequest}
|
|
|
|
/>
|
|
|
|
<TextButton
|
|
|
|
width={100} height={40}
|
|
|
|
text='Reject' fontSize={18}
|
|
|
|
onPress={handleRejectRequest}
|
|
|
|
/>
|
|
|
|
</ThemedView>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
|
|
|
|
} else if (status.relationship.isAccepted) {
|
|
|
|
// Case 4: In an accepted relationship
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<ThemedText style={styles.title}>
|
|
|
|
{status.relationship.title}
|
|
|
|
</ThemedText>
|
|
|
|
<ThemedView style={styles.profileContainer}>
|
|
|
|
{user && (
|
|
|
|
<ThemedView style={styles.profileWrapper}>
|
|
|
|
<Image
|
2024-10-18 14:27:27 -05:00
|
|
|
source={{ uri: `${process.env.EXPO_PUBLIC_API_URL}${user.pfpUrl}` }}
|
2024-10-16 16:50:26 -05:00
|
|
|
style={styles.profilePicture}
|
|
|
|
/>
|
|
|
|
<ThemedText style={styles.name}>
|
|
|
|
{user.fullName.split(' ')[0]}
|
|
|
|
</ThemedText>
|
|
|
|
</ThemedView>
|
|
|
|
)}
|
|
|
|
{status.partner && (
|
|
|
|
<ThemedView style={styles.profileWrapper}>
|
|
|
|
<Image
|
2024-10-18 14:27:27 -05:00
|
|
|
source={{ uri: `${process.env.EXPO_PUBLIC_API_URL}` +
|
2024-10-16 16:50:26 -05:00
|
|
|
`${status.partner.pfpUrl}` }}
|
|
|
|
style={styles.profilePicture}
|
|
|
|
/>
|
|
|
|
<ThemedText style={styles.name}>
|
|
|
|
{status.partner.fullName.split(' ')[0]}
|
|
|
|
</ThemedText>
|
|
|
|
</ThemedView>
|
|
|
|
)}
|
|
|
|
</ThemedView>
|
2024-10-30 15:03:53 -05:00
|
|
|
<TextButton
|
|
|
|
width={140} height={36}
|
|
|
|
text='Leave Relationship'
|
|
|
|
fontSize={16}
|
|
|
|
onPress={() => handleRejectRequest()}
|
|
|
|
/>
|
2024-10-16 16:50:26 -05:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<ThemedView style={styles.container}>
|
|
|
|
{renderRelationshipContent()}
|
|
|
|
</ThemedView>
|
|
|
|
);
|
|
|
|
};
|
2024-10-18 12:39:54 -05:00
|
|
|
export default RelationshipView;
|
2024-10-16 16:50:26 -05:00
|
|
|
|
|
|
|
const styles = StyleSheet.create({
|
|
|
|
container: {
|
|
|
|
flex: 1,
|
|
|
|
alignItems: 'center',
|
|
|
|
},
|
|
|
|
title: {
|
|
|
|
fontSize: 24,
|
|
|
|
fontWeight: 'bold',
|
|
|
|
marginBottom: 20,
|
|
|
|
},
|
|
|
|
profileContainer: {
|
|
|
|
flexDirection: 'row',
|
|
|
|
justifyContent: 'center',
|
|
|
|
alignItems: 'center',
|
|
|
|
},
|
|
|
|
profileWrapper: {
|
|
|
|
alignItems: 'center',
|
|
|
|
marginHorizontal: 10,
|
|
|
|
},
|
|
|
|
profilePicture: {
|
|
|
|
width: 100,
|
|
|
|
height: 100,
|
|
|
|
borderRadius: 50,
|
|
|
|
marginBottom: 10,
|
|
|
|
},
|
|
|
|
name: {
|
|
|
|
fontSize: 12,
|
|
|
|
fontWeight: 'bold',
|
|
|
|
},
|
|
|
|
lastChecked: {
|
|
|
|
fontSize: 12,
|
|
|
|
marginBottom: 10,
|
|
|
|
},
|
|
|
|
buttonContainer: {
|
2024-10-30 15:03:53 -05:00
|
|
|
alignItems: 'center',
|
2024-10-16 16:50:26 -05:00
|
|
|
justifyContent: 'space-around',
|
|
|
|
width: '100%',
|
|
|
|
marginTop: 20,
|
|
|
|
},
|
|
|
|
});
|