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
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
fetchContacts(); // Fetch all contacts by default
|
_initializeContacts(); // Rename to make it clear this is initialization
|
||||||
FlutterContacts.addListener(_onContactChange);
|
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 {
|
void _onContactChange() async {
|
||||||
await fetchContacts();
|
await fetchContacts();
|
||||||
}
|
}
|
||||||
@ -49,14 +59,16 @@ class _ContactStateState extends State<ContactState> {
|
|||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fetch all contacts
|
// Fetch all contacts - public method that can be called after build
|
||||||
Future<void> fetchContacts() async {
|
Future<void> fetchContacts() async {
|
||||||
if (!mounted) return;
|
if (!mounted) return;
|
||||||
|
|
||||||
setState(() => _loading = true);
|
setState(() => _loading = true);
|
||||||
try {
|
try {
|
||||||
List<Contact> contacts = await _contactService.fetchContacts();
|
List<Contact> contacts = await _contactService.fetchContacts();
|
||||||
_processContacts(contacts);
|
if (mounted) {
|
||||||
|
_processContacts(contacts);
|
||||||
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
debugPrint('Error fetching contacts: $e');
|
debugPrint('Error fetching contacts: $e');
|
||||||
} finally {
|
} 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) {
|
void _processContacts(List<Contact> contacts) {
|
||||||
if (!mounted) return;
|
if (!mounted) return;
|
||||||
|
|
||||||
@ -94,7 +134,6 @@ class _ContactStateState extends State<ContactState> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (_selfContact!.phones.isEmpty) {
|
if (_selfContact!.phones.isEmpty) {
|
||||||
debugPrint("Self contact has no phone numbers");
|
|
||||||
_selfContact = null;
|
_selfContact = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -103,8 +142,7 @@ class _ContactStateState extends State<ContactState> {
|
|||||||
|
|
||||||
setState(() {
|
setState(() {
|
||||||
_allContacts = contacts;
|
_allContacts = contacts;
|
||||||
_favoriteContacts =
|
_favoriteContacts = contacts.where((contact) => contact.isStarred).toList();
|
||||||
contacts.where((contact) => contact.isStarred).toList();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,22 +24,41 @@ class _MyHomePageState extends State<MyHomePage>
|
|||||||
List<Contact> _contactSuggestions = [];
|
List<Contact> _contactSuggestions = [];
|
||||||
final ObfuscateService _obfuscateService = ObfuscateService();
|
final ObfuscateService _obfuscateService = ObfuscateService();
|
||||||
final TextEditingController _searchController = TextEditingController();
|
final TextEditingController _searchController = TextEditingController();
|
||||||
|
bool _isInitialized = false;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
// Set the TabController length to 4
|
|
||||||
_tabController = TabController(length: 4, vsync: this, initialIndex: 1);
|
_tabController = TabController(length: 4, vsync: this, initialIndex: 1);
|
||||||
_tabController.addListener(_handleTabIndex);
|
_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 {
|
void _fetchContacts() async {
|
||||||
final contactState = ContactState.of(context);
|
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();
|
await contactState.fetchContacts();
|
||||||
setState(() {
|
if (mounted) {
|
||||||
_allContacts = contactState.contacts;
|
setState(() {
|
||||||
});
|
_allContacts = contactState.contacts;
|
||||||
|
_contactSuggestions = _allContacts;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _clearSearch() {
|
void _clearSearch() {
|
||||||
|
Loading…
Reference in New Issue
Block a user