XuqmGroup-RNChatDemo/src/screens/contact/ContactsScreen.tsx

123 行
5.0 KiB
TypeScript

import React, { useEffect, useState, useCallback } from 'react'
import {
View, Text, FlatList, StyleSheet, TouchableOpacity, SafeAreaView,
ActivityIndicator,
} from 'react-native'
import { useNavigation, useFocusEffect } from '@react-navigation/native'
import type { NativeStackNavigationProp } from '@react-navigation/native-stack'
import type { RootStackParams } from '../../navigation/types'
import { ImSDK } from '@xuqm/rn-sdk'
import { demoApi, type UserProfile } from '../../api/demo'
import { load, save, K } from '../../utils/storage'
type Nav = NativeStackNavigationProp<RootStackParams>
function ContactRow({ user, onPress }: { user: UserProfile; onPress(): void }) {
const letter = (user.nickname || user.userId).charAt(0).toUpperCase()
return (
<TouchableOpacity style={styles.row} onPress={onPress} activeOpacity={0.7}>
<View style={styles.avatar}><Text style={styles.avatarText}>{letter}</Text></View>
<View style={styles.body}>
<Text style={styles.name}>{user.nickname}</Text>
<Text style={styles.uid}>@{user.userId}</Text>
</View>
<Text style={styles.arrow}></Text>
</TouchableOpacity>
)
}
export default function ContactsScreen() {
const navigation = useNavigation<Nav>()
const [contacts, setContacts] = useState<UserProfile[]>([])
const [loading, setLoading] = useState(false)
const fetchContacts = useCallback(async () => {
setLoading(true)
try {
const friendIds = await ImSDK.listFriends()
const profiles: UserProfile[] = []
await Promise.all(
friendIds.map(async (id) => {
try {
const results = await demoApi.searchUsers(id)
const match = results.find(u => u.userId === id)
if (match) profiles.push(match)
} catch {/* skip individual failures */}
}),
)
setContacts(profiles)
await save(K.CONTACTS, profiles)
} catch {
// network failed — load from local cache
const cached = await load<UserProfile[]>(K.CONTACTS)
if (cached) setContacts(cached)
} finally {
setLoading(false)
}
}, [])
useFocusEffect(
useCallback(() => {
fetchContacts()
}, [fetchContacts]),
)
const openChat = (user: UserProfile) => {
navigation.navigate('SingleChat', { targetId: user.userId, targetName: user.nickname, targetAvatar: user.avatar })
}
return (
<SafeAreaView style={styles.root}>
<View style={styles.header}>
<Text style={styles.title}></Text>
<View style={styles.headerActions}>
<TouchableOpacity onPress={() => navigation.navigate('GroupList')} style={styles.headerBtn}>
<Text style={styles.headerBtnText}></Text>
</TouchableOpacity>
<TouchableOpacity onPress={() => navigation.navigate('UserSearch')} style={styles.headerBtn}>
<Text style={styles.headerBtnText}></Text>
</TouchableOpacity>
</View>
</View>
{loading && <ActivityIndicator style={styles.loadingIndicator} color="#07C160" />}
<FlatList
data={contacts}
keyExtractor={u => u.userId}
renderItem={({ item }) => <ContactRow user={item} onPress={() => openChat(item)} />}
ItemSeparatorComponent={() => <View style={styles.sep} />}
ListEmptyComponent={
!loading ? (
<View style={styles.empty}>
<Text style={styles.emptyText}></Text>
<TouchableOpacity onPress={() => navigation.navigate('UserSearch')}>
<Text style={styles.emptyLink}></Text>
</TouchableOpacity>
</View>
) : null
}
/>
</SafeAreaView>
)
}
const styles = StyleSheet.create({
root: { flex: 1, backgroundColor: '#f5f5f5' },
header: { flexDirection: 'row', justifyContent: 'space-between', alignItems: 'center', padding: 16, backgroundColor: '#fff', borderBottomWidth: StyleSheet.hairlineWidth, borderBottomColor: '#e0e0e0' },
title: { fontSize: 20, fontWeight: '700', color: '#111' },
headerActions: { flexDirection: 'row', gap: 12 },
headerBtn: { paddingHorizontal: 12, paddingVertical: 6, backgroundColor: '#07C160', borderRadius: 6 },
headerBtnText: { color: '#fff', fontSize: 13, fontWeight: '600' },
loadingIndicator: { marginVertical: 8 },
row: { flexDirection: 'row', alignItems: 'center', padding: 12, backgroundColor: '#fff' },
avatar: { width: 46, height: 46, borderRadius: 8, backgroundColor: '#07C160', alignItems: 'center', justifyContent: 'center', marginRight: 12 },
avatarText: { color: '#fff', fontSize: 18, fontWeight: '600' },
body: { flex: 1 },
name: { fontSize: 16, fontWeight: '500', color: '#111' },
uid: { fontSize: 13, color: '#888', marginTop: 2 },
arrow: { color: '#ccc', fontSize: 20 },
sep: { height: StyleSheet.hairlineWidth, backgroundColor: '#e0e0e0', marginLeft: 70 },
empty: { alignItems: 'center', paddingTop: 80 },
emptyText: { color: '#bbb', fontSize: 15 },
emptyLink: { color: '#07C160', fontSize: 14, marginTop: 12 },
})