914 lines
30 KiB
Dart
914 lines
30 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:file_picker/file_picker.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/rendering.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_mobx/flutter_mobx.dart';
|
|
import 'package:local_auth/local_auth.dart';
|
|
import 'package:qr_flutter/qr_flutter.dart';
|
|
import 'package:warp/store.dart';
|
|
import 'package:warp_api/warp_api.dart';
|
|
import 'package:charts_flutter/flutter.dart' as charts;
|
|
import 'package:grouped_list/grouped_list.dart';
|
|
|
|
import 'about.dart';
|
|
import 'main.dart';
|
|
|
|
class AccountPage extends StatefulWidget {
|
|
@override
|
|
State<StatefulWidget> createState() => _AccountPageState();
|
|
}
|
|
|
|
class _AccountPageState extends State<AccountPage>
|
|
with
|
|
WidgetsBindingObserver,
|
|
AutomaticKeepAliveClientMixin,
|
|
SingleTickerProviderStateMixin {
|
|
Timer _timerSync;
|
|
int _progress = 0;
|
|
bool _useSnapAddress = false;
|
|
String _snapAddress = "";
|
|
TabController _tabController;
|
|
bool _accountTab = true;
|
|
bool _syncing = false;
|
|
|
|
@override
|
|
bool get wantKeepAlive => true;
|
|
|
|
@override
|
|
initState() {
|
|
super.initState();
|
|
_tabController = TabController(length: 5, vsync: this);
|
|
_tabController.addListener(() {
|
|
setState(() {
|
|
_accountTab = _tabController.index == 0;
|
|
});
|
|
});
|
|
Future.microtask(() async {
|
|
await accountManager.updateUnconfirmedBalance();
|
|
await accountManager.fetchAccountData();
|
|
await _setupTimer();
|
|
});
|
|
WidgetsBinding.instance.addObserver(this);
|
|
progressStream.listen((percent) {
|
|
setState(() {
|
|
_progress = percent;
|
|
});
|
|
});
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_timerSync?.cancel();
|
|
WidgetsBinding.instance.removeObserver(this);
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
void didChangeAppLifecycleState(AppLifecycleState state) {
|
|
switch (state) {
|
|
case AppLifecycleState.detached:
|
|
case AppLifecycleState.inactive:
|
|
case AppLifecycleState.paused:
|
|
_timerSync?.cancel();
|
|
break;
|
|
case AppLifecycleState.resumed:
|
|
_setupTimer();
|
|
break;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
super.build(context);
|
|
if (!syncStatus.isSynced() && !_syncing) _trySync();
|
|
if (accountManager.active == null) return CircularProgressIndicator();
|
|
final theme = Theme.of(this.context);
|
|
final hasTAddr = accountManager.taddress.isNotEmpty;
|
|
final showTAddr = accountManager.showTAddr;
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
title: Observer(
|
|
builder: (context) =>
|
|
Text("\u24E9 Wallet - ${accountManager.active.name}")),
|
|
bottom: TabBar(controller: _tabController, isScrollable: true, tabs: [
|
|
Tab(text: "Account"),
|
|
Tab(text: "Notes"),
|
|
Tab(text: "History"),
|
|
Tab(text: "Budget"),
|
|
Tab(text: "Contacts"),
|
|
]),
|
|
actions: [
|
|
Observer(builder: (context) {
|
|
accountManager.canPay;
|
|
return PopupMenuButton<String>(
|
|
itemBuilder: (context) => [
|
|
PopupMenuItem(child: Text("Accounts"), value: "Accounts"),
|
|
PopupMenuItem(child: Text("Backup"), value: "Backup"),
|
|
PopupMenuItem(child: Text("Rescan"), value: "Rescan"),
|
|
if (accountManager.canPay)
|
|
PopupMenuItem(child: Text("Cold Storage"), value: "Cold"),
|
|
if (accountManager.canPay)
|
|
PopupMenuItem(child: Text('MultiPay'), value: "MultiPay"),
|
|
PopupMenuItem(child: Text('Broadcast'), value: "Broadcast"),
|
|
PopupMenuItem(child: Text('Settings'), value: "Settings"),
|
|
PopupMenuItem(child: Text("About"), value: "About"),
|
|
],
|
|
onSelected: _onMenu,
|
|
);
|
|
})
|
|
],
|
|
),
|
|
body: TabBarView(controller: _tabController, children: [
|
|
SingleChildScrollView(
|
|
padding: EdgeInsets.all(20),
|
|
child: Center(
|
|
child: Column(children: [
|
|
Observer(
|
|
builder: (context) => syncStatus.syncedHeight <= 0
|
|
? Text('Synching')
|
|
: syncStatus.isSynced()
|
|
? Text('${syncStatus.syncedHeight}',
|
|
style: theme.textTheme.caption)
|
|
: Text(
|
|
'${syncStatus.syncedHeight} / ${syncStatus.latestHeight}',
|
|
style: theme.textTheme.caption
|
|
.apply(color: theme.primaryColor))),
|
|
Padding(padding: EdgeInsets.symmetric(vertical: 8)),
|
|
Observer(builder: (context) {
|
|
final _ = accountManager.active.address;
|
|
final address = _address();
|
|
return Column(children: [
|
|
if (hasTAddr)
|
|
Text(showTAddr
|
|
? 'Tap QR Code for Shielded Address'
|
|
: 'Tap QR Code for Transparent Address'),
|
|
Padding(padding: EdgeInsets.symmetric(vertical: 4)),
|
|
GestureDetector(
|
|
onTap: hasTAddr ? _onQRTap : null,
|
|
child: QrImage(
|
|
data: address,
|
|
size: 200,
|
|
backgroundColor: Colors.white)),
|
|
Padding(padding: EdgeInsets.symmetric(vertical: 8)),
|
|
RichText(
|
|
text: TextSpan(children: [
|
|
TextSpan(
|
|
text: '$address ', style: theme.textTheme.bodyText2),
|
|
WidgetSpan(
|
|
child: GestureDetector(
|
|
child: Icon(Icons.content_copy),
|
|
onTap: _onAddressCopy)),
|
|
])),
|
|
if (!showTAddr)
|
|
TextButton(
|
|
child: Text('New Snap Address'),
|
|
onPressed: _onSnapAddress),
|
|
if (showTAddr)
|
|
TextButton(
|
|
child: Text('Shield Transp. Balance'),
|
|
onPressed: _onShieldTAddr,
|
|
)
|
|
]);
|
|
}),
|
|
Observer(builder: (context) {
|
|
final balance = showTAddr
|
|
? accountManager.tbalance
|
|
: accountManager.balance;
|
|
return Row(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
crossAxisAlignment: CrossAxisAlignment.baseline,
|
|
textBaseline: TextBaseline.ideographic,
|
|
children: <Widget>[
|
|
Text('\u24E9 ${_getBalance_hi(balance)}',
|
|
style: theme.textTheme.headline2),
|
|
Text('${_getBalance_lo(balance)}'),
|
|
]);
|
|
}),
|
|
Observer(builder: (context) {
|
|
final balance = showTAddr
|
|
? accountManager.tbalance
|
|
: accountManager.balance;
|
|
final fx = _fx();
|
|
final balanceFX = balance * fx / ZECUNIT;
|
|
return Column(children: [
|
|
if (fx != 0.0)
|
|
Text("${balanceFX.toStringAsFixed(2)} ${settings.currency}",
|
|
style: theme.textTheme.headline6),
|
|
if (fx != 0.0)
|
|
Text("1 ZEC = ${fx.toStringAsFixed(2)} ${settings.currency}"),
|
|
]);
|
|
}),
|
|
Padding(padding: EdgeInsets.symmetric(vertical: 8)),
|
|
Observer(
|
|
builder: (context) =>
|
|
(accountManager.unconfirmedBalance != 0)
|
|
? Row(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
crossAxisAlignment: CrossAxisAlignment.baseline,
|
|
textBaseline: TextBaseline.ideographic,
|
|
children: <Widget>[
|
|
Text(
|
|
'${_sign(accountManager.unconfirmedBalance)} ${_getBalance_hi(accountManager.unconfirmedBalance)}',
|
|
style: theme.textTheme.headline4
|
|
?.merge(_unconfirmedStyle())),
|
|
Text(
|
|
'${_getBalance_lo(accountManager.unconfirmedBalance)}',
|
|
style: _unconfirmedStyle()),
|
|
])
|
|
: Container()),
|
|
if (_progress > 0)
|
|
LinearProgressIndicator(value: _progress / 100.0),
|
|
]))),
|
|
NoteWidget(tabTo),
|
|
HistoryWidget(tabTo),
|
|
BudgetWidget(),
|
|
ContactsWidget(),
|
|
]),
|
|
floatingActionButton: _accountTab
|
|
? FloatingActionButton(
|
|
onPressed: _onSend,
|
|
tooltip: 'Send',
|
|
backgroundColor: Theme.of(context).accentColor.withOpacity(
|
|
accountManager.canPay ? 1.0 : 0.3),
|
|
child: Icon(Icons.send),
|
|
)
|
|
: Container(), // This trailing comma makes auto-formatting nicer for build methods.
|
|
);
|
|
}
|
|
|
|
void tabTo(int index) {
|
|
if (index != _tabController.index) _tabController.animateTo(index);
|
|
}
|
|
|
|
_address() => accountManager.showTAddr
|
|
? accountManager.taddress
|
|
: (_useSnapAddress ? _snapAddress : accountManager.active.address);
|
|
|
|
_sign(int b) {
|
|
return b < 0 ? '-' : '+';
|
|
}
|
|
|
|
_onQRTap() {
|
|
accountManager.toggleShowTAddr();
|
|
}
|
|
|
|
_onAddressCopy() {
|
|
Clipboard.setData(ClipboardData(text: _address()));
|
|
final snackBar = SnackBar(content: Text('Address copied to clipboard'));
|
|
rootScaffoldMessengerKey.currentState.showSnackBar(snackBar);
|
|
}
|
|
|
|
_onShieldTAddr() {
|
|
showDialog(
|
|
context: this.context,
|
|
barrierDismissible: false,
|
|
builder: (context) => AlertDialog(
|
|
title: Text('Shield Transparent Balance'),
|
|
content: Text(
|
|
'Do you want to transfer your entire transparent balance to your shielded address?'),
|
|
actions: [
|
|
TextButton(
|
|
child: Text('Cancel'),
|
|
onPressed: () {
|
|
Navigator.of(this.context).pop();
|
|
},
|
|
),
|
|
TextButton(
|
|
child: Text('OK'),
|
|
onPressed: () async {
|
|
Navigator.of(this.context).pop();
|
|
final snackBar1 =
|
|
SnackBar(content: Text('Shielding in progress...'));
|
|
rootScaffoldMessengerKey.currentState.showSnackBar(snackBar1);
|
|
final txid =
|
|
await WarpApi.shieldTAddr(accountManager.active.id);
|
|
final snackBar2 = SnackBar(content: Text(txid));
|
|
rootScaffoldMessengerKey.currentState.showSnackBar(snackBar2);
|
|
},
|
|
)
|
|
]),
|
|
);
|
|
}
|
|
|
|
_unconfirmedStyle() {
|
|
return accountManager.unconfirmedBalance > 0
|
|
? TextStyle(color: Colors.green)
|
|
: TextStyle(color: Colors.red);
|
|
}
|
|
|
|
_getBalance_hi(int b) {
|
|
return ((b.abs() ~/ 100000) / 1000.0).toStringAsFixed(3);
|
|
}
|
|
|
|
_getBalance_lo(b) {
|
|
return (b.abs() % 100000).toString().padLeft(5, '0');
|
|
}
|
|
|
|
_setupTimer() async {
|
|
await _sync();
|
|
_timerSync = Timer.periodic(Duration(seconds: 15), (Timer t) {
|
|
_trySync();
|
|
});
|
|
}
|
|
|
|
double _fx() {
|
|
return priceStore.zecPrice;
|
|
}
|
|
|
|
_sync() async {
|
|
_syncing = true;
|
|
await syncStatus.update();
|
|
WarpApi.warpSync(settings.getTx, settings.anchorOffset, (int height) async {
|
|
setState(() {
|
|
if (height >= 0)
|
|
syncStatus.setSyncHeight(height);
|
|
else {
|
|
_syncing = false;
|
|
_trySync();
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
_trySync() async {
|
|
priceStore.fetchZecPrice();
|
|
if (syncStatus.syncedHeight < 0) return;
|
|
await syncStatus.update();
|
|
await accountManager.updateUnconfirmedBalance();
|
|
if (!syncStatus.isSynced()) {
|
|
final res =
|
|
await WarpApi.tryWarpSync(settings.getTx, settings.anchorOffset);
|
|
if (res == 1) {
|
|
// Reorg
|
|
final targetHeight = syncStatus.syncedHeight - 10;
|
|
WarpApi.rewindToHeight(targetHeight);
|
|
syncStatus.setSyncHeight(targetHeight);
|
|
} else if (res == 0) {
|
|
syncStatus.update();
|
|
}
|
|
}
|
|
await accountManager.fetchAccountData();
|
|
await accountManager.updateBalance();
|
|
await accountManager.updateUnconfirmedBalance();
|
|
accountManager.updateTBalance();
|
|
}
|
|
|
|
_onSnapAddress() {
|
|
final address = accountManager.newAddress();
|
|
setState(() {
|
|
_useSnapAddress = true;
|
|
_snapAddress = address;
|
|
});
|
|
Timer(Duration(seconds: 15), () {
|
|
setState(() {
|
|
_useSnapAddress = false;
|
|
});
|
|
});
|
|
}
|
|
|
|
_onSend() {
|
|
Navigator.of(this.context).pushNamed('/send');
|
|
}
|
|
|
|
_onMenu(String choice) {
|
|
switch (choice) {
|
|
case "Accounts":
|
|
Navigator.of(this.context).pushNamed('/accounts');
|
|
break;
|
|
case "Backup":
|
|
_backup();
|
|
break;
|
|
case "Rescan":
|
|
_rescan();
|
|
break;
|
|
case "Cold":
|
|
_cold();
|
|
break;
|
|
case "MultiPay":
|
|
_multiPay();
|
|
break;
|
|
case "Broadcast":
|
|
_broadcast();
|
|
break;
|
|
case "Settings":
|
|
_settings();
|
|
break;
|
|
case "About":
|
|
showAbout(this.context);
|
|
break;
|
|
}
|
|
}
|
|
|
|
_backup() async {
|
|
final localAuth = LocalAuthentication();
|
|
try {
|
|
final didAuthenticate = await localAuth.authenticate(
|
|
localizedReason: "Please authenticate to show account seed");
|
|
if (didAuthenticate) {
|
|
Navigator.of(context).pushNamed('/backup', arguments: true);
|
|
}
|
|
}
|
|
on PlatformException catch (e) {
|
|
await showDialog(
|
|
context: context,
|
|
barrierDismissible: true,
|
|
builder: (context) => AlertDialog(
|
|
title: Text('No Authentication Method'),
|
|
content: Text(e.message)
|
|
));
|
|
}
|
|
}
|
|
|
|
_rescan() {
|
|
showDialog(
|
|
context: this.context,
|
|
barrierDismissible: false,
|
|
builder: (context) => AlertDialog(
|
|
title: Text('Rescan'),
|
|
content: Text('Rescan wallet from the first block?'),
|
|
actions: [
|
|
TextButton(
|
|
child: Text('Cancel'),
|
|
onPressed: () {
|
|
Navigator.of(this.context).pop();
|
|
},
|
|
),
|
|
TextButton(
|
|
child: Text('OK'),
|
|
onPressed: () {
|
|
Navigator.of(this.context).pop();
|
|
final snackBar = SnackBar(content: Text("Rescan Requested..."));
|
|
rootScaffoldMessengerKey.currentState.showSnackBar(snackBar);
|
|
syncStatus.setSyncHeight(0);
|
|
WarpApi.rewindToHeight(0);
|
|
_sync();
|
|
},
|
|
),
|
|
]),
|
|
);
|
|
}
|
|
|
|
_cold() {
|
|
showDialog(
|
|
context: context,
|
|
barrierDismissible: false,
|
|
builder: (context) => AlertDialog(
|
|
title: Text('Cold Storage'),
|
|
content: Text(
|
|
'Do you want to DELETE the secret key and convert this account to a watch-only account? '
|
|
'You will not be able to spend from this device anymore. This operation is NOT reversible.'),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () {
|
|
Navigator.of(context).pop();
|
|
},
|
|
child: Text('Cancel')),
|
|
TextButton(
|
|
onPressed: _convertToWatchOnly, child: Text('DELETE'))
|
|
]));
|
|
}
|
|
|
|
_multiPay() {
|
|
Navigator.of(context).pushNamed('/multipay');
|
|
}
|
|
|
|
_broadcast() async {
|
|
final result = await FilePicker.platform.pickFiles();
|
|
|
|
if (result != null) {
|
|
final res = WarpApi.broadcast(result.files.single.path);
|
|
final snackBar = SnackBar(content: Text(res));
|
|
rootScaffoldMessengerKey.currentState.showSnackBar(snackBar);
|
|
}
|
|
}
|
|
|
|
_convertToWatchOnly() {
|
|
accountManager.convertToWatchOnly();
|
|
Navigator.of(context).pop();
|
|
}
|
|
|
|
_settings() {
|
|
Navigator.of(this.context).pushNamed('/settings');
|
|
}
|
|
}
|
|
|
|
class NoteWidget extends StatefulWidget {
|
|
void Function(int index) tabTo;
|
|
|
|
NoteWidget(this.tabTo);
|
|
|
|
@override
|
|
State<StatefulWidget> createState() => _NoteState();
|
|
}
|
|
|
|
class _NoteState extends State<NoteWidget> with AutomaticKeepAliveClientMixin {
|
|
@override
|
|
bool get wantKeepAlive => true; //Set to true
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
super.build(context);
|
|
return SingleChildScrollView(
|
|
padding: EdgeInsets.all(4),
|
|
scrollDirection: Axis.vertical,
|
|
child: Observer(
|
|
builder: (context) => NotificationListener<OverscrollNotification>(
|
|
onNotification: (s) {
|
|
final os = s.overscroll;
|
|
if (os < 0) {
|
|
widget.tabTo(0);
|
|
return true;
|
|
}
|
|
if (os > 0) {
|
|
widget.tabTo(2);
|
|
return true;
|
|
}
|
|
return false;
|
|
},
|
|
child: PaginatedDataTable(
|
|
columns: [
|
|
DataColumn(
|
|
label: settings.showConfirmations ? Text('Confs') : Text('Height'),
|
|
onSort: (_, __) {
|
|
setState(() {
|
|
settings.toggleShowConfirmations();
|
|
});
|
|
}),
|
|
DataColumn(label: Text('Date/Time')),
|
|
DataColumn(
|
|
label: Text('Amount'),
|
|
numeric: true,
|
|
onSort: (_, __) {
|
|
setState(() {
|
|
accountManager.sortNoteAmount();
|
|
});
|
|
}),
|
|
],
|
|
header: Text('Select notes to EXCLUDE from payments',
|
|
style: Theme.of(context).textTheme.bodyText1),
|
|
columnSpacing: 16,
|
|
showCheckboxColumn: false,
|
|
availableRowsPerPage: [5, 10, 25, 100],
|
|
onRowsPerPageChanged: (int value) {
|
|
settings.setRowsPerPage(value);
|
|
},
|
|
showFirstLastButtons: true,
|
|
rowsPerPage: settings.rowsPerPage,
|
|
source: NotesDataSource(context, _onRowSelected),
|
|
))));
|
|
}
|
|
|
|
_onRowSelected(Note note) {
|
|
accountManager.excludeNote(note);
|
|
}
|
|
}
|
|
|
|
class NotesDataSource extends DataTableSource {
|
|
final BuildContext context;
|
|
final Function(Note) onRowSelected;
|
|
|
|
NotesDataSource(this.context, this.onRowSelected);
|
|
|
|
@override
|
|
DataRow getRow(int index) {
|
|
final note = accountManager.notes[index];
|
|
final theme = Theme.of(context);
|
|
final confsOrHeight = settings.showConfirmations ? syncStatus.latestHeight - note.height + 1: note.height;
|
|
return DataRow.byIndex(
|
|
index: index,
|
|
selected: note.excluded,
|
|
color: MaterialStateColor.resolveWith((states) =>
|
|
states.contains(MaterialState.selected)
|
|
? theme.primaryColor.withOpacity(0.5)
|
|
: theme.backgroundColor),
|
|
cells: [
|
|
DataCell(Text("$confsOrHeight",
|
|
style: !_confirmed(note.height)
|
|
? Theme.of(this.context).textTheme.overline
|
|
: null)),
|
|
DataCell(Text("${note.timestamp}")),
|
|
DataCell(Text("${note.value.toStringAsFixed(8)}")),
|
|
],
|
|
onSelectChanged: (selected) => _noteSelected(note, selected),
|
|
);
|
|
}
|
|
|
|
@override
|
|
bool get isRowCountApproximate => false;
|
|
|
|
@override
|
|
int get rowCount => accountManager.notes.length;
|
|
|
|
@override
|
|
int get selectedRowCount => 0;
|
|
|
|
bool _confirmed(int height) {
|
|
return syncStatus.latestHeight - height >= settings.anchorOffset;
|
|
}
|
|
|
|
void _noteSelected(Note note, bool selected) {
|
|
note.excluded = !note.excluded;
|
|
notifyListeners();
|
|
onRowSelected(note);
|
|
}
|
|
}
|
|
|
|
class HistoryWidget extends StatefulWidget {
|
|
void Function(int index) tabTo;
|
|
|
|
HistoryWidget(this.tabTo);
|
|
|
|
@override
|
|
State<StatefulWidget> createState() => _HistoryState();
|
|
}
|
|
|
|
class _HistoryState extends State<HistoryWidget>
|
|
with AutomaticKeepAliveClientMixin {
|
|
@override
|
|
bool get wantKeepAlive => true; //Set to true
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
super.build(context);
|
|
return SingleChildScrollView(
|
|
padding: EdgeInsets.all(4),
|
|
scrollDirection: Axis.vertical,
|
|
child: Observer(
|
|
builder: (context) => NotificationListener<OverscrollNotification>(
|
|
onNotification: (s) {
|
|
if (s.overscroll < 0) {
|
|
widget.tabTo(1);
|
|
return true;
|
|
}
|
|
if (s.overscroll > 0) {
|
|
widget.tabTo(3);
|
|
return true;
|
|
}
|
|
return false;
|
|
},
|
|
child: PaginatedDataTable(
|
|
columns: [
|
|
DataColumn(
|
|
label: settings.showConfirmations ? Text('Confs') : Text('Height'),
|
|
onSort: (_, __) {
|
|
setState(() {
|
|
settings.toggleShowConfirmations();
|
|
});
|
|
}),
|
|
DataColumn(label: Text('Date/Time')),
|
|
DataColumn(label: Text('TXID')),
|
|
DataColumn(
|
|
label: Text('Amount'),
|
|
numeric: true,
|
|
onSort: (_, __) {
|
|
setState(() {
|
|
accountManager.sortTxAmount();
|
|
});
|
|
}),
|
|
],
|
|
columnSpacing: 16,
|
|
showCheckboxColumn: false,
|
|
availableRowsPerPage: [5, 10, 25, 100],
|
|
onRowsPerPageChanged: (int value) {
|
|
settings.setRowsPerPage(value);
|
|
},
|
|
showFirstLastButtons: true,
|
|
rowsPerPage: settings.rowsPerPage,
|
|
source: HistoryDataSource(context)))));
|
|
}
|
|
}
|
|
|
|
class HistoryDataSource extends DataTableSource {
|
|
BuildContext context;
|
|
|
|
HistoryDataSource(this.context);
|
|
|
|
@override
|
|
DataRow getRow(int index) {
|
|
final tx = accountManager.txs[index];
|
|
final confsOrHeight = settings.showConfirmations ? syncStatus.latestHeight - tx.height + 1: tx.height;
|
|
return DataRow(
|
|
cells: [
|
|
DataCell(Text("$confsOrHeight")),
|
|
DataCell(Text("${tx.timestamp}")),
|
|
DataCell(Text("${tx.txid}")),
|
|
DataCell(Text("${tx.value.toStringAsFixed(8)}",
|
|
textAlign: TextAlign.left)),
|
|
],
|
|
onSelectChanged: (_) {
|
|
Navigator.of(this.context).pushNamed('/tx', arguments: tx);
|
|
});
|
|
}
|
|
|
|
@override
|
|
bool get isRowCountApproximate => false;
|
|
|
|
@override
|
|
int get rowCount => accountManager.txs.length;
|
|
|
|
@override
|
|
int get selectedRowCount => 0;
|
|
}
|
|
|
|
class BudgetWidget extends StatefulWidget {
|
|
@override
|
|
State<StatefulWidget> createState() => BudgetState();
|
|
}
|
|
|
|
class BudgetState extends State<BudgetWidget>
|
|
with AutomaticKeepAliveClientMixin {
|
|
@override
|
|
bool get wantKeepAlive => true; //Set to true
|
|
var _showAddress = true;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
super.build(context);
|
|
return Padding(
|
|
padding: EdgeInsets.all(4),
|
|
child: Observer(builder: (context) {
|
|
final _ = accountManager.dataEpoch;
|
|
return Column(
|
|
children: [
|
|
Expanded(
|
|
child: Card(
|
|
child: Column(children: [
|
|
Text('Largest Spendings by Address',
|
|
style: Theme.of(context).textTheme.headline6),
|
|
Expanded(
|
|
child: SpendingChart(accountManager.spendings, _showAddress,
|
|
_toggleAddress)),
|
|
Text('Tap Chart to Toggle between Address and Amount',
|
|
style: Theme.of(context).textTheme.caption)
|
|
]))),
|
|
Expanded(
|
|
child: Card(
|
|
child: Column(children: [
|
|
Text('Account Balance History',
|
|
style: Theme.of(context).textTheme.headline6),
|
|
Expanded(
|
|
child:
|
|
AccountBalanceTimeChart(accountManager.accountBalances))
|
|
]))),
|
|
],
|
|
);
|
|
}));
|
|
}
|
|
|
|
void _toggleAddress() {
|
|
setState(() {
|
|
_showAddress = !_showAddress;
|
|
});
|
|
}
|
|
}
|
|
|
|
class SpendingChart extends StatelessWidget {
|
|
final List<Spending> data;
|
|
final bool showAddress;
|
|
final void Function() onTap;
|
|
|
|
SpendingChart(this.data, this.showAddress, this.onTap);
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final seriesList = _createSeries(data, showAddress);
|
|
final color = charts.ColorUtil.fromDartColor(
|
|
Theme.of(context).textTheme.headline5.color);
|
|
if (seriesList[0].data.isEmpty)
|
|
return Column(
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
Text('No Spending in the Last 30 Days',
|
|
style: Theme.of(context).textTheme.headline5)
|
|
]);
|
|
return new charts.PieChart(seriesList,
|
|
animate: false,
|
|
selectionModels: [
|
|
charts.SelectionModelConfig(changedListener: (_) {
|
|
onTap();
|
|
})
|
|
],
|
|
defaultRenderer:
|
|
charts.ArcRendererConfig(arcWidth: 80, arcRendererDecorators: [
|
|
charts.ArcLabelDecorator(
|
|
outsideLabelStyleSpec:
|
|
charts.TextStyleSpec(color: color, fontSize: 12),
|
|
)
|
|
]));
|
|
}
|
|
|
|
static List<charts.Series<Spending, String>> _createSeries(
|
|
List<Spending> data, bool showAddress) {
|
|
final palette = settings.palette.makeShades(data.length + 5);
|
|
return [
|
|
new charts.Series<Spending, String>(
|
|
id: 'Largest Spending Last Month',
|
|
domainFn: (Spending sales, _) => sales.address,
|
|
measureFn: (Spending sales, _) => sales.amount,
|
|
colorFn: (_, index) => palette[index],
|
|
data: data,
|
|
labelAccessorFn: (Spending row, _) =>
|
|
showAddress ? row.address : row.amount.toString(),
|
|
)
|
|
];
|
|
}
|
|
}
|
|
|
|
class AccountBalanceTimeChart extends StatefulWidget {
|
|
List<AccountBalance> data;
|
|
|
|
AccountBalanceTimeChart(this.data);
|
|
|
|
@override
|
|
State<StatefulWidget> createState() => AccountBalanceTimeChartState();
|
|
}
|
|
|
|
class AccountBalanceTimeChartState extends State<AccountBalanceTimeChart> {
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final axisColor = charts.ColorUtil.fromDartColor(
|
|
Theme.of(context).textTheme.headline5.color);
|
|
final seriesList = _createSeries(widget.data,
|
|
charts.ColorUtil.fromDartColor(Theme.of(context).primaryColor));
|
|
return new charts.TimeSeriesChart(
|
|
seriesList,
|
|
animate: false,
|
|
dateTimeFactory: const charts.LocalDateTimeFactory(),
|
|
primaryMeasureAxis: charts.NumericAxisSpec(
|
|
viewport: charts.NumericExtents.fromValues(
|
|
widget.data.map((ab) => ab.balance).toList()),
|
|
tickProviderSpec: charts.BasicNumericTickProviderSpec(
|
|
zeroBound: true,
|
|
dataIsInWholeNumbers: false,
|
|
desiredTickCount: 5),
|
|
renderSpec: charts.GridlineRendererSpec(
|
|
labelStyle: charts.TextStyleSpec(color: axisColor))),
|
|
domainAxis: charts.DateTimeAxisSpec(
|
|
renderSpec: charts.SmallTickRendererSpec(
|
|
labelStyle: charts.TextStyleSpec(color: axisColor))),
|
|
behaviors: [
|
|
charts.SelectNearest(),
|
|
],
|
|
);
|
|
}
|
|
|
|
static List<charts.Series<AccountBalance, DateTime>> _createSeries(
|
|
List<AccountBalance> data, charts.Color lineColor) {
|
|
return [
|
|
new charts.Series<AccountBalance, DateTime>(
|
|
id: 'Balance',
|
|
colorFn: (_, __) => lineColor,
|
|
domainFn: (AccountBalance ab, _) => ab.time,
|
|
measureFn: (AccountBalance ab, _) => ab.balance,
|
|
data: data,
|
|
)
|
|
];
|
|
}
|
|
}
|
|
|
|
class ContactsWidget extends StatefulWidget {
|
|
@override
|
|
State<StatefulWidget> createState() => ContactsState();
|
|
}
|
|
|
|
class ContactsState extends State<ContactsWidget> {
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Observer(builder: (context) {
|
|
final _ = accountManager.dataEpoch;
|
|
final theme = Theme.of(context);
|
|
return Column(children: [
|
|
Padding(
|
|
padding: EdgeInsets.symmetric(vertical: 8),
|
|
child: Text(
|
|
'To make a contact, send them a memo with "Contact: Name"',
|
|
style: Theme.of(context).textTheme.caption)),
|
|
Expanded(
|
|
child: GroupedListView<Contact, String>(
|
|
elements: accountManager.contacts,
|
|
groupBy: (e) => e.name.isEmpty ? "" : e.name[0],
|
|
itemBuilder: (context, c) => ListTile(
|
|
title: Text(c.name),
|
|
subtitle: Text(c.address),
|
|
trailing: accountManager.canPay
|
|
? IconButton(
|
|
icon: Icon(Icons.chevron_right),
|
|
onPressed: () {
|
|
_onContact(c);
|
|
})
|
|
: null),
|
|
groupSeparatorBuilder: (String h) =>
|
|
Text(h, style: theme.textTheme.headline5),
|
|
)),
|
|
]);
|
|
});
|
|
}
|
|
|
|
_onContact(Contact contact) {
|
|
Navigator.of(context).pushNamed('/send', arguments: contact);
|
|
}
|
|
}
|