refactor: improve contact fetching and initialization logic for better performance and clarity
This commit is contained in:
parent
cef7a27e88
commit
62d48dc084
@ -35,10 +35,20 @@ class _ContactStateState extends State<ContactState> {
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
fetchContacts(); // Fetch all contacts by default
|
||||
_initializeContacts(); // Rename to make it clear this is initialization
|
||||
FlutterContacts.addListener(_onContactChange);
|
||||
}
|
||||
|
||||
// Private method to initialize contacts without setState during build
|
||||
Future<void> _initializeContacts() async {
|
||||
try {
|
||||
List<Contact> contacts = await _contactService.fetchContacts();
|
||||
_processContactsInitial(contacts);
|
||||
} catch (e) {
|
||||
debugPrint('Error fetching contacts: $e');
|
||||
}
|
||||
}
|
||||
|
||||
void _onContactChange() async {
|
||||
await fetchContacts();
|
||||
}
|
||||
@ -49,14 +59,16 @@ class _ContactStateState extends State<ContactState> {
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
// Fetch all contacts
|
||||
// Fetch all contacts - public method that can be called after build
|
||||
Future<void> fetchContacts() async {
|
||||
if (!mounted) return;
|
||||
|
||||
setState(() => _loading = true);
|
||||
try {
|
||||
List<Contact> contacts = await _contactService.fetchContacts();
|
||||
_processContacts(contacts);
|
||||
if (mounted) {
|
||||
_processContacts(contacts);
|
||||
}
|
||||
} catch (e) {
|
||||
debugPrint('Error fetching contacts: $e');
|
||||
} finally {
|
||||
@ -85,6 +97,34 @@ class _ContactStateState extends State<ContactState> {
|
||||
}
|
||||
}
|
||||
|
||||
// Process contacts without setState for initial loading
|
||||
void _processContactsInitial(List<Contact> contacts) {
|
||||
if (!mounted) return;
|
||||
|
||||
_selfContact = contacts.firstWhere(
|
||||
(contact) => contact.displayName.toLowerCase() == "user",
|
||||
orElse: () => Contact(),
|
||||
);
|
||||
|
||||
if (_selfContact!.phones.isEmpty) {
|
||||
_selfContact = null;
|
||||
}
|
||||
|
||||
contacts = contacts.where((contact) => contact.phones.isNotEmpty).toList();
|
||||
contacts.sort((a, b) => a.displayName.compareTo(b.displayName));
|
||||
|
||||
_allContacts = contacts;
|
||||
_favoriteContacts = contacts.where((contact) => contact.isStarred).toList();
|
||||
_loading = false;
|
||||
|
||||
// Force a rebuild after initialization is complete
|
||||
if (mounted) {
|
||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||
setState(() {});
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
void _processContacts(List<Contact> contacts) {
|
||||
if (!mounted) return;
|
||||
|
||||
@ -94,7 +134,6 @@ class _ContactStateState extends State<ContactState> {
|
||||
);
|
||||
|
||||
if (_selfContact!.phones.isEmpty) {
|
||||
debugPrint("Self contact has no phone numbers");
|
||||
_selfContact = null;
|
||||
}
|
||||
|
||||
@ -103,8 +142,7 @@ class _ContactStateState extends State<ContactState> {
|
||||
|
||||
setState(() {
|
||||
_allContacts = contacts;
|
||||
_favoriteContacts =
|
||||
contacts.where((contact) => contact.isStarred).toList();
|
||||
_favoriteContacts = contacts.where((contact) => contact.isStarred).toList();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -24,22 +24,41 @@ class _MyHomePageState extends State<MyHomePage>
|
||||
List<Contact> _contactSuggestions = [];
|
||||
final ObfuscateService _obfuscateService = ObfuscateService();
|
||||
final TextEditingController _searchController = TextEditingController();
|
||||
bool _isInitialized = false;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
// Set the TabController length to 4
|
||||
_tabController = TabController(length: 4, vsync: this, initialIndex: 1);
|
||||
_tabController.addListener(_handleTabIndex);
|
||||
_fetchContacts();
|
||||
}
|
||||
|
||||
@override
|
||||
void didChangeDependencies() {
|
||||
super.didChangeDependencies();
|
||||
if (!_isInitialized) {
|
||||
// Use a post-frame callback to avoid setState during build
|
||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||
_fetchContacts();
|
||||
});
|
||||
_isInitialized = true;
|
||||
}
|
||||
}
|
||||
|
||||
void _fetchContacts() async {
|
||||
final contactState = ContactState.of(context);
|
||||
// Wait for initial load to finish if it hasn't already
|
||||
if (contactState.loading) {
|
||||
await Future.delayed(const Duration(milliseconds: 100));
|
||||
}
|
||||
// Then explicitly fetch contacts (which will call setState safely)
|
||||
await contactState.fetchContacts();
|
||||
setState(() {
|
||||
_allContacts = contactState.contacts;
|
||||
});
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
_allContacts = contactState.contacts;
|
||||
_contactSuggestions = _allContacts;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
void _clearSearch() {
|
||||
|
Loading…
Reference in New Issue
Block a user