refactor: introduce LoadingIndicatorWidget and streamline contact page structure
This commit is contained in:
parent
8cb206a640
commit
f89c5440fc
@ -0,0 +1,10 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
class LoadingIndicatorWidget extends StatelessWidget {
|
||||||
|
const LoadingIndicatorWidget({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return const Center(child: CircularProgressIndicator());
|
||||||
|
}
|
||||||
|
}
|
@ -1,14 +1,8 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_contacts/flutter_contacts.dart';
|
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
|
||||||
import '../contacts/contact_state.dart';
|
import '../contacts/contact_state.dart';
|
||||||
|
import '../contacts/widgets/alphabet_scroll_page.dart';
|
||||||
|
import '../../common/widgets/loading_indicator.dart';
|
||||||
import '../../../domain/services/obfuscate_service.dart';
|
import '../../../domain/services/obfuscate_service.dart';
|
||||||
import '../../common/widgets/obfuscated_avatar.dart';
|
|
||||||
import '../../../core/utils/color_utils.dart';
|
|
||||||
import '../../../domain/services/call_service.dart';
|
|
||||||
import '../../../domain/services/block_service.dart';
|
|
||||||
import 'widgets/contact_modal.dart';
|
|
||||||
import 'widgets/alphabet_scroll_page.dart';
|
|
||||||
|
|
||||||
class ContactPage extends StatefulWidget {
|
class ContactPage extends StatefulWidget {
|
||||||
const ContactPage({super.key});
|
const ContactPage({super.key});
|
||||||
@ -19,198 +13,17 @@ class ContactPage extends StatefulWidget {
|
|||||||
|
|
||||||
class _ContactPageState extends State<ContactPage> {
|
class _ContactPageState extends State<ContactPage> {
|
||||||
final ObfuscateService _obfuscateService = ObfuscateService();
|
final ObfuscateService _obfuscateService = ObfuscateService();
|
||||||
final CallService _callService = CallService();
|
|
||||||
int? _expandedIndex;
|
|
||||||
|
|
||||||
Future<void> _refreshContacts() async {
|
|
||||||
final contactState = ContactState.of(context);
|
|
||||||
try {
|
|
||||||
await contactState.fetchContacts();
|
|
||||||
} catch (e) {
|
|
||||||
debugPrint('Error refreshing contacts: $e');
|
|
||||||
if (mounted) {
|
|
||||||
ScaffoldMessenger.of(context)
|
|
||||||
.showSnackBar(SnackBar(content: Text('Failed to refresh contacts')));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void _toggleFavorite(Contact contact) async {
|
|
||||||
try {
|
|
||||||
if (await FlutterContacts.requestPermission()) {
|
|
||||||
// Avoid full contact fetch by using minimal properties
|
|
||||||
Contact? fullContact = await FlutterContacts.getContact(contact.id,
|
|
||||||
withProperties: true,
|
|
||||||
withAccounts: false, // Don't need accounts for favorite toggle
|
|
||||||
withPhoto: false, // Don't need photo for favorite toggle
|
|
||||||
withThumbnail: false // Don't need thumbnail for favorite toggle
|
|
||||||
);
|
|
||||||
|
|
||||||
if (fullContact != null) {
|
|
||||||
fullContact.isStarred = !fullContact.isStarred;
|
|
||||||
await FlutterContacts.updateContact(fullContact);
|
|
||||||
await _refreshContacts();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
debugPrint("Could not fetch contact details");
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
debugPrint("Error updating favorite status: $e");
|
|
||||||
if (mounted) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text('Failed to update favorite status')));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final contactState = ContactState.of(context);
|
final contactState = ContactState.of(context);
|
||||||
|
|
||||||
if (contactState.loading) {
|
|
||||||
return const Center(child: CircularProgressIndicator());
|
|
||||||
}
|
|
||||||
|
|
||||||
final contacts = contactState.contacts;
|
|
||||||
|
|
||||||
if (contacts.isEmpty) {
|
|
||||||
return const Center(
|
|
||||||
child: Text(
|
|
||||||
'No contacts found.\nAdd contacts to get started.',
|
|
||||||
style: TextStyle(color: Colors.white60),
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
backgroundColor: Colors.black,
|
body: contactState.loading
|
||||||
body: AlphabetScrollPage(
|
? const Center(child: CircularProgressIndicator())
|
||||||
scrollOffset: contactState.scrollOffset,
|
: AlphabetScrollPage(
|
||||||
contacts: contacts,
|
scrollOffset: contactState.scrollOffset,
|
||||||
),
|
contacts: contactState.contacts,
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _getCallIcon(Contact contact) {
|
|
||||||
if (!contact.phones.isNotEmpty) return const SizedBox.shrink();
|
|
||||||
return Icon(
|
|
||||||
contact.isStarred ? Icons.star : Icons.star_border,
|
|
||||||
color: contact.isStarred ? Colors.amber : Colors.grey,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _launchSms(Contact contact) async {
|
|
||||||
if (!contact.phones.isNotEmpty) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
const SnackBar(content: Text('No phone number available')),
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final Uri smsUri = Uri(
|
|
||||||
scheme: 'sms',
|
|
||||||
path: contact.phones.first.number,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (await canLaunchUrl(smsUri)) {
|
|
||||||
await launchUrl(smsUri);
|
|
||||||
} else {
|
|
||||||
if (!mounted) return;
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
const SnackBar(content: Text('Could not launch SMS')),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void _editContact(Contact contact) async {
|
|
||||||
if (await FlutterContacts.requestPermission()) {
|
|
||||||
final updatedContact = await FlutterContacts.openExternalEdit(contact.id);
|
|
||||||
if (updatedContact != null) {
|
|
||||||
await _refreshContacts();
|
|
||||||
setState(() => _expandedIndex = null);
|
|
||||||
if (!mounted) return;
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text('${contact.displayName} updated successfully!')),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void _toggleBlock(Contact contact, bool isBlocked) async {
|
|
||||||
if (!contact.phones.isNotEmpty) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
const SnackBar(content: Text('No phone number to block')),
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final phoneNumber = contact.phones.first.number;
|
|
||||||
if (isBlocked) {
|
|
||||||
await BlockService().unblockNumber(phoneNumber);
|
|
||||||
if (!mounted) return;
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text('$phoneNumber unblocked')),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
await BlockService().blockNumber(phoneNumber);
|
|
||||||
if (!mounted) return;
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text('$phoneNumber blocked')),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
|
|
||||||
void _showContactModal(Contact contact) {
|
|
||||||
showModalBottomSheet(
|
|
||||||
context: context,
|
|
||||||
isScrollControlled: true,
|
|
||||||
backgroundColor: Colors.transparent,
|
|
||||||
builder: (context) => ContactModal(
|
|
||||||
contact: contact,
|
|
||||||
onEdit: () async {
|
|
||||||
if (await FlutterContacts.requestPermission()) {
|
|
||||||
final updatedContact = await FlutterContacts.openExternalEdit(contact.id);
|
|
||||||
if (updatedContact != null) {
|
|
||||||
await _refreshContacts();
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(content: Text('${contact.displayName} updated successfully!')),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
onToggleFavorite: () => _toggleFavorite(contact),
|
|
||||||
isFavorite: contact.isStarred,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildActionButton({
|
|
||||||
required IconData icon,
|
|
||||||
required String label,
|
|
||||||
required VoidCallback onPressed,
|
|
||||||
}) {
|
|
||||||
return InkWell(
|
|
||||||
onTap: onPressed,
|
|
||||||
child: Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(vertical: 8.0, horizontal: 12.0),
|
|
||||||
child: Column(
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
|
||||||
children: [
|
|
||||||
Icon(icon, color: Colors.white),
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
Text(
|
|
||||||
label,
|
|
||||||
style: const TextStyle(
|
|
||||||
color: Colors.white,
|
|
||||||
fontSize: 12,
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
import 'package:dialer/services/obfuscate_service.dart';
|
|
||||||
import 'package:dialer/widgets/username_color_generator.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_contacts/flutter_contacts.dart';
|
import 'package:flutter_contacts/flutter_contacts.dart';
|
||||||
|
import '../../../../domain/services/obfuscate_service.dart';
|
||||||
|
import '../../../../core/utils/color_utils.dart';
|
||||||
|
import '../../../common/widgets/obfuscated_avatar.dart';
|
||||||
import '../contact_state.dart';
|
import '../contact_state.dart';
|
||||||
import 'add_contact_button.dart';
|
import 'add_contact_button.dart';
|
||||||
import 'contact_modal.dart';
|
import 'contact_modal.dart';
|
||||||
@ -23,7 +24,6 @@ class AlphabetScrollPage extends StatefulWidget {
|
|||||||
|
|
||||||
class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
||||||
late ScrollController _scrollController;
|
late ScrollController _scrollController;
|
||||||
|
|
||||||
final ObfuscateService _obfuscateService = ObfuscateService();
|
final ObfuscateService _obfuscateService = ObfuscateService();
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -43,7 +43,7 @@ class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|||||||
try {
|
try {
|
||||||
await contactState.fetchContacts();
|
await contactState.fetchContacts();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print('Error refreshing contacts: $e');
|
debugPrint('Error refreshing contacts: $e');
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
SnackBar(content: Text('Failed to refresh contacts')),
|
SnackBar(content: Text('Failed to refresh contacts')),
|
||||||
);
|
);
|
||||||
@ -55,9 +55,9 @@ class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|||||||
if (await FlutterContacts.requestPermission()) {
|
if (await FlutterContacts.requestPermission()) {
|
||||||
Contact? fullContact = await FlutterContacts.getContact(contact.id,
|
Contact? fullContact = await FlutterContacts.getContact(contact.id,
|
||||||
withProperties: true,
|
withProperties: true,
|
||||||
withAccounts: true,
|
withAccounts: false,
|
||||||
withPhoto: true,
|
withPhoto: false,
|
||||||
withThumbnail: true);
|
withThumbnail: false);
|
||||||
|
|
||||||
if (fullContact != null) {
|
if (fullContact != null) {
|
||||||
fullContact.isStarred = !fullContact.isStarred;
|
fullContact.isStarred = !fullContact.isStarred;
|
||||||
@ -65,10 +65,10 @@ class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|||||||
}
|
}
|
||||||
await _refreshContacts();
|
await _refreshContacts();
|
||||||
} else {
|
} else {
|
||||||
print("Could not fetch contact details");
|
debugPrint("Could not fetch contact details");
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
print("Error updating favorite status: $e");
|
debugPrint("Error updating favorite status: $e");
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
SnackBar(content: Text('Failed to update contact favorite status')),
|
SnackBar(content: Text('Failed to update contact favorite status')),
|
||||||
);
|
);
|
||||||
@ -139,7 +139,7 @@ class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|||||||
? _obfuscateService.obfuscateData(contact.phones.first.number)
|
? _obfuscateService.obfuscateData(contact.phones.first.number)
|
||||||
: 'No phone number';
|
: 'No phone number';
|
||||||
Color avatarColor =
|
Color avatarColor =
|
||||||
generateColorFromName(contact.displayName);
|
generateColorFromName(contact.displayName);
|
||||||
return ListTile(
|
return ListTile(
|
||||||
leading: ObfuscatedAvatar(
|
leading: ObfuscatedAvatar(
|
||||||
imageBytes: contact.thumbnail,
|
imageBytes: contact.thumbnail,
|
||||||
@ -166,8 +166,8 @@ class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|||||||
onEdit: () async {
|
onEdit: () async {
|
||||||
if (await FlutterContacts.requestPermission()) {
|
if (await FlutterContacts.requestPermission()) {
|
||||||
final updatedContact =
|
final updatedContact =
|
||||||
await FlutterContacts.openExternalEdit(
|
await FlutterContacts.openExternalEdit(
|
||||||
contact.id);
|
contact.id);
|
||||||
if (updatedContact != null) {
|
if (updatedContact != null) {
|
||||||
await _refreshContacts();
|
await _refreshContacts();
|
||||||
Navigator.of(context).pop();
|
Navigator.of(context).pop();
|
||||||
@ -183,7 +183,7 @@ class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|||||||
.showSnackBar(
|
.showSnackBar(
|
||||||
SnackBar(
|
SnackBar(
|
||||||
content:
|
content:
|
||||||
Text('Edit canceled or failed.'),
|
Text('Edit canceled or failed.'),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user