2024-10-18 19:45:35 +00:00
|
|
|
import 'package:dialer/widgets/username_color_generator.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_contacts/flutter_contacts.dart';
|
|
|
|
import '../contact_state.dart';
|
|
|
|
|
|
|
|
class AlphabetScrollPage extends StatefulWidget {
|
|
|
|
final List<Contact> contacts;
|
|
|
|
final double scrollOffset;
|
|
|
|
|
2024-10-26 12:47:45 +00:00
|
|
|
const AlphabetScrollPage({super.key, required this.contacts, required this.scrollOffset});
|
2024-10-18 19:45:35 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
_AlphabetScrollPageState createState() => _AlphabetScrollPageState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _AlphabetScrollPageState extends State<AlphabetScrollPage> {
|
|
|
|
late ScrollController _scrollController;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
_scrollController = ScrollController(initialScrollOffset: widget.scrollOffset);
|
|
|
|
_scrollController.addListener(_onScroll);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _onScroll() {
|
|
|
|
final contactState = ContactState.of(context);
|
|
|
|
contactState.setScrollOffset(_scrollController.offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
Map<String, List<Contact>> alphabetizedContacts = {};
|
|
|
|
for (var contact in widget.contacts) {
|
|
|
|
String firstLetter = contact.displayName.isNotEmpty ? contact.displayName[0].toUpperCase() : '#';
|
|
|
|
if (!alphabetizedContacts.containsKey(firstLetter)) {
|
|
|
|
alphabetizedContacts[firstLetter] = [];
|
|
|
|
}
|
|
|
|
alphabetizedContacts[firstLetter]!.add(contact);
|
|
|
|
}
|
|
|
|
|
|
|
|
List<String> alphabetKeys = alphabetizedContacts.keys.toList()..sort();
|
|
|
|
|
|
|
|
return ListView.builder(
|
|
|
|
controller: _scrollController,
|
|
|
|
itemCount: alphabetKeys.length,
|
|
|
|
itemBuilder: (context, index) {
|
|
|
|
String letter = alphabetKeys[index];
|
|
|
|
List<Contact> contacts = alphabetizedContacts[letter]!;
|
|
|
|
return Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
2024-10-18 20:13:47 +00:00
|
|
|
Padding(
|
2024-10-18 19:45:35 +00:00
|
|
|
padding: const EdgeInsets.symmetric(vertical: 8.0, horizontal: 16.0),
|
|
|
|
child: Text(
|
|
|
|
letter,
|
|
|
|
style: TextStyle(
|
2024-10-18 20:13:47 +00:00
|
|
|
fontSize: 28,
|
2024-10-18 19:45:35 +00:00
|
|
|
fontWeight: FontWeight.bold,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
...contacts.map((contact) {
|
|
|
|
String phoneNumber = contact.phones.isNotEmpty ? contact.phones.first.number : 'No phone number';
|
|
|
|
Color avatarColor = generateColorFromName(contact.displayName);
|
|
|
|
return ListTile(
|
|
|
|
leading: (contact.thumbnail != null && contact.thumbnail!.isNotEmpty)
|
|
|
|
? CircleAvatar(
|
|
|
|
backgroundImage: MemoryImage(contact.thumbnail!),
|
|
|
|
)
|
|
|
|
: CircleAvatar(
|
|
|
|
backgroundColor: avatarColor,
|
|
|
|
child: Text(
|
|
|
|
contact.displayName.isNotEmpty ? contact.displayName[0].toUpperCase() : '?',
|
|
|
|
style: TextStyle(color: Colors.white),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
title: Text(contact.displayName),
|
|
|
|
subtitle: Text(phoneNumber),
|
|
|
|
onTap: () {
|
|
|
|
// Handle contact tap
|
|
|
|
},
|
|
|
|
);
|
2024-10-18 20:13:47 +00:00
|
|
|
}),
|
2024-10-18 19:45:35 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
_scrollController.dispose();
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
}
|