mirror of
https://github.com/nisrulz/flutter-examples.git
synced 2025-11-08 20:50:04 +00:00
Added: expanse planner app
This commit is contained in:
65
expense_planner/lib/widgets/chart.dart
Normal file
65
expense_planner/lib/widgets/chart.dart
Normal file
@@ -0,0 +1,65 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
import './chart_bar.dart';
|
||||
import '../models/transaction.dart';
|
||||
|
||||
class Chart extends StatelessWidget {
|
||||
final List<Transaction> recentTransactions;
|
||||
|
||||
Chart(this.recentTransactions);
|
||||
|
||||
List<Map<String, Object>> get groupedTransactionValues {
|
||||
return List.generate(7, (index) {
|
||||
final weekDay = DateTime.now().subtract(
|
||||
Duration(days: index),
|
||||
);
|
||||
var totalSum = 0.0;
|
||||
|
||||
for (var i = 0; i < recentTransactions.length; i++) {
|
||||
if (recentTransactions[i].date.day == weekDay.day &&
|
||||
recentTransactions[i].date.month == weekDay.month &&
|
||||
recentTransactions[i].date.year == weekDay.year) {
|
||||
totalSum += recentTransactions[i].amount;
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
'day': DateFormat.E().format(weekDay).substring(0, 1),
|
||||
'amount': totalSum,
|
||||
};
|
||||
}).reversed.toList();
|
||||
}
|
||||
|
||||
double get totalSpending {
|
||||
return groupedTransactionValues.fold(0.0, (sum, item) {
|
||||
return sum + item['amount'];
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Card(
|
||||
elevation: 6,
|
||||
margin: EdgeInsets.all(20),
|
||||
child: Padding(
|
||||
padding: EdgeInsets.all(10),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceAround,
|
||||
children: groupedTransactionValues.map((data) {
|
||||
return Flexible(
|
||||
fit: FlexFit.tight,
|
||||
child: ChartBar(
|
||||
data['day'],
|
||||
data['amount'],
|
||||
totalSpending == 0.0
|
||||
? 0.0
|
||||
: (data['amount'] as double) / totalSpending,
|
||||
),
|
||||
);
|
||||
}).toList(),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
54
expense_planner/lib/widgets/chart_bar.dart
Normal file
54
expense_planner/lib/widgets/chart_bar.dart
Normal file
@@ -0,0 +1,54 @@
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class ChartBar extends StatelessWidget {
|
||||
final String label;
|
||||
final double spendingAmount;
|
||||
final double spendingPctOfTotal;
|
||||
|
||||
ChartBar(this.label, this.spendingAmount, this.spendingPctOfTotal);
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Column(
|
||||
children: <Widget>[
|
||||
Container(
|
||||
height: 20,
|
||||
child: FittedBox(
|
||||
child: Text('\$${spendingAmount.toStringAsFixed(0)}'),
|
||||
),
|
||||
),
|
||||
SizedBox(
|
||||
height: 4,
|
||||
),
|
||||
Container(
|
||||
height: 60,
|
||||
width: 10,
|
||||
child: Stack(
|
||||
children: <Widget>[
|
||||
Container(
|
||||
decoration: BoxDecoration(
|
||||
border: Border.all(color: Colors.grey, width: 1.0),
|
||||
color: Color.fromRGBO(220, 220, 220, 1),
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
),
|
||||
FractionallySizedBox(
|
||||
heightFactor: spendingPctOfTotal,
|
||||
child: Container(
|
||||
decoration: BoxDecoration(
|
||||
color: Theme.of(context).primaryColor,
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
SizedBox(
|
||||
height: 4,
|
||||
),
|
||||
Text(label),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
||||
114
expense_planner/lib/widgets/new_transaction.dart
Normal file
114
expense_planner/lib/widgets/new_transaction.dart
Normal file
@@ -0,0 +1,114 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
class NewTransaction extends StatefulWidget {
|
||||
final Function addTx;
|
||||
|
||||
NewTransaction(this.addTx);
|
||||
|
||||
@override
|
||||
_NewTransactionState createState() => _NewTransactionState();
|
||||
}
|
||||
|
||||
class _NewTransactionState extends State<NewTransaction> {
|
||||
final _titleController = TextEditingController();
|
||||
final _amountController = TextEditingController();
|
||||
DateTime _selectedDate;
|
||||
|
||||
void _submitData() {
|
||||
if (_amountController.text.isEmpty) {
|
||||
return;
|
||||
}
|
||||
final enteredTitle = _titleController.text;
|
||||
final enteredAmount = double.parse(_amountController.text);
|
||||
|
||||
if (enteredTitle.isEmpty || enteredAmount <= 0 || _selectedDate == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
widget.addTx(
|
||||
enteredTitle,
|
||||
enteredAmount,
|
||||
_selectedDate,
|
||||
);
|
||||
|
||||
Navigator.of(context).pop();
|
||||
}
|
||||
|
||||
void _presentDatePicker() {
|
||||
showDatePicker(
|
||||
context: context,
|
||||
initialDate: DateTime.now(),
|
||||
firstDate: DateTime(2019),
|
||||
lastDate: DateTime.now(),
|
||||
).then((pickedDate) {
|
||||
if (pickedDate == null) {
|
||||
return;
|
||||
}
|
||||
setState(() {
|
||||
_selectedDate = pickedDate;
|
||||
});
|
||||
});
|
||||
print('...');
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Card(
|
||||
elevation: 5,
|
||||
child: Container(
|
||||
padding: EdgeInsets.all(10),
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.end,
|
||||
children: <Widget>[
|
||||
TextField(
|
||||
decoration: InputDecoration(labelText: 'Title'),
|
||||
controller: _titleController,
|
||||
onSubmitted: (_) => _submitData(),
|
||||
// onChanged: (val) {
|
||||
// titleInput = val;
|
||||
// },
|
||||
),
|
||||
TextField(
|
||||
decoration: InputDecoration(labelText: 'Amount'),
|
||||
controller: _amountController,
|
||||
keyboardType: TextInputType.number,
|
||||
onSubmitted: (_) => _submitData(),
|
||||
// onChanged: (val) => amountInput = val,
|
||||
),
|
||||
Container(
|
||||
height: 70,
|
||||
child: Row(
|
||||
children: <Widget>[
|
||||
Expanded(
|
||||
child: Text(
|
||||
_selectedDate == null
|
||||
? 'No Date Chosen!'
|
||||
: 'Picked Date: ${DateFormat.yMd().format(_selectedDate)}',
|
||||
),
|
||||
),
|
||||
FlatButton(
|
||||
textColor: Theme.of(context).primaryColor,
|
||||
child: Text(
|
||||
'Choose Date',
|
||||
style: TextStyle(
|
||||
fontWeight: FontWeight.bold,
|
||||
),
|
||||
),
|
||||
onPressed: _presentDatePicker,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
RaisedButton(
|
||||
child: Text('Add Transaction'),
|
||||
color: Theme.of(context).primaryColor,
|
||||
textColor: Theme.of(context).textTheme.button.color,
|
||||
onPressed: _submitData,
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
71
expense_planner/lib/widgets/transaction_list.dart
Normal file
71
expense_planner/lib/widgets/transaction_list.dart
Normal file
@@ -0,0 +1,71 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
import '../models/transaction.dart';
|
||||
|
||||
class TransactionList extends StatelessWidget {
|
||||
final List<Transaction> transactions;
|
||||
final Function deleteTx;
|
||||
|
||||
TransactionList(this.transactions, this.deleteTx);
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Container(
|
||||
height: 450,
|
||||
child: transactions.isEmpty
|
||||
? Column(
|
||||
children: <Widget>[
|
||||
Text(
|
||||
'No transactions added yet!',
|
||||
style: Theme.of(context).textTheme.title,
|
||||
),
|
||||
SizedBox(
|
||||
height: 20,
|
||||
),
|
||||
Container(
|
||||
height: 200,
|
||||
child: Image.asset(
|
||||
'assets/images/waiting.png',
|
||||
fit: BoxFit.cover,
|
||||
)),
|
||||
],
|
||||
)
|
||||
: ListView.builder(
|
||||
itemBuilder: (ctx, index) {
|
||||
return Card(
|
||||
elevation: 5,
|
||||
margin: EdgeInsets.symmetric(
|
||||
vertical: 8,
|
||||
horizontal: 5,
|
||||
),
|
||||
child: ListTile(
|
||||
leading: CircleAvatar(
|
||||
radius: 30,
|
||||
child: Padding(
|
||||
padding: EdgeInsets.all(6),
|
||||
child: FittedBox(
|
||||
child: Text('\$${transactions[index].amount}'),
|
||||
),
|
||||
),
|
||||
),
|
||||
title: Text(
|
||||
transactions[index].title,
|
||||
style: Theme.of(context).textTheme.title,
|
||||
),
|
||||
subtitle: Text(
|
||||
DateFormat.yMMMd().format(transactions[index].date),
|
||||
),
|
||||
trailing: IconButton(
|
||||
icon: Icon(Icons.delete),
|
||||
color: Theme.of(context).errorColor,
|
||||
onPressed: () => deleteTx(transactions[index].id),
|
||||
),
|
||||
),
|
||||
);
|
||||
},
|
||||
itemCount: transactions.length,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
50
expense_planner/lib/widgets/user_transactions.dart
Normal file
50
expense_planner/lib/widgets/user_transactions.dart
Normal file
@@ -0,0 +1,50 @@
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
import './new_transaction.dart';
|
||||
import './transaction_list.dart';
|
||||
import '../models/transaction.dart';
|
||||
|
||||
class UserTransactions extends StatefulWidget {
|
||||
@override
|
||||
_UserTransactionsState createState() => _UserTransactionsState();
|
||||
}
|
||||
|
||||
class _UserTransactionsState extends State<UserTransactions> {
|
||||
final List<Transaction> _userTransactions = [
|
||||
Transaction(
|
||||
id: 't1',
|
||||
title: 'New Shoes',
|
||||
amount: 69.99,
|
||||
date: DateTime.now(),
|
||||
),
|
||||
Transaction(
|
||||
id: 't2',
|
||||
title: 'Weekly Groceries',
|
||||
amount: 16.53,
|
||||
date: DateTime.now(),
|
||||
),
|
||||
];
|
||||
|
||||
void _addNewTransaction(String txTitle, double txAmount) {
|
||||
final newTx = Transaction(
|
||||
title: txTitle,
|
||||
amount: txAmount,
|
||||
date: DateTime.now(),
|
||||
id: DateTime.now().toString(),
|
||||
);
|
||||
|
||||
setState(() {
|
||||
_userTransactions.add(newTx);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Column(
|
||||
children: <Widget>[
|
||||
NewTransaction(_addNewTransaction),
|
||||
TransactionList(_userTransactions),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user