feat: composition page finished, can search

This commit is contained in:
Florian Griffon 2024-11-05 23:39:48 +01:00
parent c0ec11098d
commit 64595a1755

View File

@ -1,4 +1,5 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_contacts/flutter_contacts.dart';
class CompositionPage extends StatefulWidget { class CompositionPage extends StatefulWidget {
const CompositionPage({super.key}); const CompositionPage({super.key});
@ -9,22 +10,40 @@ class CompositionPage extends StatefulWidget {
class _CompositionPageState extends State<CompositionPage> { class _CompositionPageState extends State<CompositionPage> {
String dialedNumber = ""; String dialedNumber = "";
final List<String> contacts = [ List<Contact> _allContacts = [];
'Alice Johnson', List<Contact> _filteredContacts = [];
'Bob Smith',
'Carol White', @override
'David Brown', void initState() {
'Eve Black', super.initState();
'Frank Grey', _fetchContacts();
'Grace Green', }
'Heidi Gold',
'Ivan Silver', Future<void> _fetchContacts() async {
'Judy Blue' if (await FlutterContacts.requestPermission()) {
]; _allContacts = await FlutterContacts.getContacts(withProperties: true);
_filteredContacts = _allContacts;
setState(() {});
}
}
void _filterContacts() {
setState(() {
_filteredContacts = _allContacts.where((contact) {
final phoneMatch = contact.phones.any((phone) =>
phone.number.replaceAll(RegExp(r'\D'), '').contains(dialedNumber));
final nameMatch = contact.displayName
.toLowerCase()
.contains(dialedNumber.toLowerCase());
return phoneMatch || nameMatch;
}).toList();
});
}
void _onNumberPress(String number) { void _onNumberPress(String number) {
setState(() { setState(() {
dialedNumber += number; dialedNumber += number;
_filterContacts();
}); });
} }
@ -32,6 +51,7 @@ class _CompositionPageState extends State<CompositionPage> {
setState(() { setState(() {
if (dialedNumber.isNotEmpty) { if (dialedNumber.isNotEmpty) {
dialedNumber = dialedNumber.substring(0, dialedNumber.length - 1); dialedNumber = dialedNumber.substring(0, dialedNumber.length - 1);
_filterContacts();
} }
}); });
} }
@ -39,141 +59,179 @@ class _CompositionPageState extends State<CompositionPage> {
void _onClearPress() { void _onClearPress() {
setState(() { setState(() {
dialedNumber = ""; dialedNumber = "";
_filteredContacts = _allContacts;
}); });
} }
List<String> _getFilteredContacts() { // Placeholder function for adding contact
return contacts void addContact(String number) {
.where((contact) => // This function is empty for now
contact.toLowerCase().contains(dialedNumber.toLowerCase()))
.toList();
} }
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Scaffold( return Scaffold(
backgroundColor: Colors.black, backgroundColor: Colors.black,
body: Column( body: Stack(
children: [ children: [
// Top half: Display contacts matching dialed number Column(
Expanded( children: [
flex: 2, // Top half: Display contacts matching dialed number
child: Container( Expanded(
padding: const EdgeInsets.all(16.0), flex: 2,
color: Colors.black, child:
child: Column( Container(
crossAxisAlignment: CrossAxisAlignment.start, padding: const EdgeInsets.only(top: 42.0, left: 16.0, right: 16.0, bottom: 16.0),
children: [ color: Colors.black,
Expanded( child: Column(
child: ListView( crossAxisAlignment: CrossAxisAlignment.start,
children: _getFilteredContacts().map((contact) {
return ListTile(
title: Text(contact,
style: const TextStyle(color: Colors.white)),
onTap: () {
// Handle contact selection if needed
},
);
}).toList(),
),
),
],
),
),
),
// Bottom half: Dialpad and Dialed number display with erase button
Expanded(
flex: 2,
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 16.0),
child: Column(
mainAxisAlignment: MainAxisAlignment.start,
children: [
// Display dialed number with erase button
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
Expanded( Expanded(
child: Align( child: ListView(
alignment: Alignment children: _filteredContacts.isNotEmpty
.center, // Aligns text to the center of the screen ? _filteredContacts.map((contact) {
child: Text( return ListTile(
dialedNumber, title: Text(
style: const TextStyle( contact.displayName,
fontSize: 24, color: Colors.white), style: const TextStyle(color: Colors.white),
overflow: TextOverflow.ellipsis, ),
), subtitle: contact.phones.isNotEmpty
? Text(
contact.phones.first.number,
style: const TextStyle(color: Colors.grey),
)
: null,
trailing: Row(
mainAxisSize: MainAxisSize.min,
children: [
// Call button
IconButton(
icon: Icon(Icons.phone, color: Colors.green[300], size: 20),
onPressed: () {
print('Calling ${contact.displayName}');
},
),
// Text button
IconButton(
icon: Icon(Icons.message, color: Colors.blue[300], size: 20),
onPressed: () {
print('Texting ${contact.displayName}');
},
),
],
),
onTap: () {
// Handle contact selection if needed
},
);
}).toList()
: [Center(child: Text('No contacts found', style: TextStyle(color: Colors.white)))],
), ),
), ),
IconButton(
onPressed: _onClearPress,
icon: const Icon(Icons.backspace, color: Colors.white),
),
], ],
), ),
const SizedBox(height: 10), ),
),
// Wrapping the dialpad in a SingleChildScrollView to prevent overflow // Bottom half: Dialpad and Dialed number display with erase button
Expanded( Expanded(
child: SingleChildScrollView( flex: 2,
child: Column( child: Container(
mainAxisAlignment: MainAxisAlignment.spaceEvenly, padding: const EdgeInsets.symmetric(horizontal: 16.0),
child: Column(
mainAxisAlignment: MainAxisAlignment.start,
children: [
// Display dialed number with erase button
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
// First Row (1, 2, 3) Expanded(
Row( child: Align(
mainAxisAlignment: MainAxisAlignment.spaceEvenly, alignment: Alignment.center,
children: [ child: Text(
_buildDialButton('1'), dialedNumber,
_buildDialButton('2'), style: const TextStyle(fontSize: 24, color: Colors.white),
_buildDialButton('3'), overflow: TextOverflow.ellipsis,
], ),
),
), ),
// Second Row (4, 5, 6) IconButton(
Row( onPressed: _onClearPress,
mainAxisAlignment: MainAxisAlignment.spaceEvenly, icon: const Icon(Icons.backspace, color: Colors.white),
children: [
_buildDialButton('4'),
_buildDialButton('5'),
_buildDialButton('6'),
],
),
// Third Row (7, 8, 9)
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
_buildDialButton('7'),
_buildDialButton('8'),
_buildDialButton('9'),
],
),
// Fourth Row (*, 0, #)
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
_buildDialButton('*'),
_buildDialButton('0'),
_buildDialButton('#'),
],
), ),
], ],
), ),
), const SizedBox(height: 10),
),
],
),
),
),
// Bottom action: Call button with padding // Dialpad
Padding( Expanded(
padding: const EdgeInsets.only(bottom: 20.0), child: SingleChildScrollView(
child: FloatingActionButton( child: Column(
backgroundColor: Colors.green, mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
_buildDialButton('1'),
_buildDialButton('2'),
_buildDialButton('3'),
],
),
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
_buildDialButton('4'),
_buildDialButton('5'),
_buildDialButton('6'),
],
),
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
_buildDialButton('7'),
_buildDialButton('8'),
_buildDialButton('9'),
],
),
Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
_buildDialButton('*'),
_buildDialButton('0'),
_buildDialButton('#'),
],
),
],
),
),
),
],
),
),
),
// Add Contact Button with empty function call
Padding(
padding: const EdgeInsets.only(bottom: 20.0),
child: FloatingActionButton(
backgroundColor: Colors.blue,
onPressed: () {
addContact(dialedNumber);
},
child: const Icon(Icons.person_add, color: Colors.white),
),
),
],
),
// Top Row with Back Arrow
Positioned(
top: 40.0,
left: 16.0,
child: IconButton(
icon: const Icon(Icons.arrow_back, color: Colors.white),
onPressed: () { onPressed: () {
// Handle call action Navigator.pop(context);
}, },
child: const Icon(Icons.phone, color: Colors.white),
), ),
), ),
], ],
@ -187,8 +245,7 @@ class _CompositionPageState extends State<CompositionPage> {
style: ElevatedButton.styleFrom( style: ElevatedButton.styleFrom(
backgroundColor: Colors.black, backgroundColor: Colors.black,
shape: const CircleBorder(), shape: const CircleBorder(),
padding: padding: const EdgeInsets.all(16),
const EdgeInsets.all(16), // Adjusted padding to prevent overflow
), ),
child: Text( child: Text(
number, number,
@ -199,21 +256,4 @@ class _CompositionPageState extends State<CompositionPage> {
), ),
); );
} }
Widget _buildDeleteButton() {
return ElevatedButton(
onPressed: _onDeletePress,
style: ElevatedButton.styleFrom(
backgroundColor: Colors.black,
shape: const CircleBorder(),
padding:
const EdgeInsets.all(16), // Adjusted padding to prevent overflow
),
child: const Icon(
Icons.backspace,
color: Colors.white,
size: 24,
),
);
}
} }