zwallet/lib/account.dart

1073 lines
35 KiB
Dart
Raw Normal View History

2021-06-26 07:30:12 -07:00
import 'dart:async';
2021-08-05 06:43:05 -07:00
import 'package:file_picker/file_picker.dart';
2021-06-26 07:30:12 -07:00
import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';
2021-07-10 22:20:53 -07:00
import 'package:flutter/services.dart';
2021-08-09 07:13:42 -07:00
import 'package:flutter_form_builder/flutter_form_builder.dart';
2021-06-26 07:30:12 -07:00
import 'package:flutter_mobx/flutter_mobx.dart';
2021-08-09 07:13:42 -07:00
import 'package:intl/intl.dart';
2021-06-26 07:30:12 -07:00
import 'package:local_auth/local_auth.dart';
import 'package:qr_flutter/qr_flutter.dart';
2021-07-07 08:40:05 -07:00
import 'package:warp/store.dart';
2021-06-26 07:30:12 -07:00
import 'package:warp_api/warp_api.dart';
2021-07-12 04:32:49 -07:00
import 'package:charts_flutter/flutter.dart' as charts;
2021-07-18 08:59:02 -07:00
import 'package:grouped_list/grouped_list.dart';
2021-06-26 07:30:12 -07:00
import 'about.dart';
import 'main.dart';
class AccountPage extends StatefulWidget {
@override
State<StatefulWidget> createState() => _AccountPageState();
}
class _AccountPageState extends State<AccountPage>
2021-07-10 22:20:53 -07:00
with
WidgetsBindingObserver,
AutomaticKeepAliveClientMixin,
SingleTickerProviderStateMixin {
2021-06-26 07:30:12 -07:00
Timer _timerSync;
2021-07-07 08:40:05 -07:00
int _progress = 0;
2021-07-07 21:22:54 -07:00
bool _useSnapAddress = false;
String _snapAddress = "";
2021-07-10 22:20:53 -07:00
TabController _tabController;
bool _accountTab = true;
2021-07-12 04:32:49 -07:00
bool _syncing = false;
2021-06-26 07:30:12 -07:00
@override
bool get wantKeepAlive => true;
@override
initState() {
2021-07-07 08:40:05 -07:00
super.initState();
2021-08-09 07:13:42 -07:00
_tabController = TabController(length: 6, vsync: this);
2021-07-10 22:20:53 -07:00
_tabController.addListener(() {
setState(() {
_accountTab = _tabController.index == 0;
});
});
2021-06-26 07:30:12 -07:00
Future.microtask(() async {
2021-07-07 08:40:05 -07:00
await accountManager.updateUnconfirmedBalance();
2021-07-12 04:32:49 -07:00
await accountManager.fetchAccountData();
2021-07-19 01:17:23 -07:00
await _setupTimer();
2021-06-26 07:30:12 -07:00
});
WidgetsBinding.instance.addObserver(this);
2021-07-07 08:40:05 -07:00
progressStream.listen((percent) {
setState(() {
_progress = percent;
});
});
2021-06-26 07:30:12 -07:00
}
@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);
2021-07-12 04:32:49 -07:00
if (!syncStatus.isSynced() && !_syncing) _trySync();
2021-06-26 07:30:12 -07:00
if (accountManager.active == null) return CircularProgressIndicator();
2021-07-12 04:32:49 -07:00
final theme = Theme.of(this.context);
2021-08-07 00:32:10 -07:00
final hasTAddr = accountManager.taddress.isNotEmpty;
2021-07-10 22:20:53 -07:00
return Scaffold(
2021-08-09 07:13:42 -07:00
appBar: AppBar(
title: Observer(
builder: (context) =>
2021-08-13 20:44:53 -07:00
Text("${coin.symbol} Wallet - ${accountManager.active.name}")),
2021-08-09 07:13:42 -07:00
bottom: TabBar(controller: _tabController, isScrollable: true, tabs: [
Tab(text: "Account"),
Tab(text: "Notes"),
Tab(text: "History"),
Tab(text: "Budget"),
Tab(text: "Trading P&L"),
Tab(text: "Contacts"),
2021-07-10 22:20:53 -07:00
]),
2021-08-09 07:13:42 -07:00
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();
2021-08-13 20:44:53 -07:00
final showTAddr = accountManager.showTAddr;
2021-08-09 07:13:42 -07:00
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)),
])),
2021-08-13 20:44:53 -07:00
Padding(padding: EdgeInsets.symmetric(vertical: 4)),
2021-08-09 07:13:42 -07:00
if (!showTAddr)
2021-08-13 20:44:53 -07:00
OutlinedButton(
2021-08-09 07:13:42 -07:00
child: Text('New Snap Address'),
2021-08-13 20:44:53 -07:00
style: OutlinedButton.styleFrom(
side: BorderSide(
width: 1, color: theme.primaryColor)),
2021-08-09 07:13:42 -07:00
onPressed: _onSnapAddress),
if (showTAddr)
2021-08-13 20:44:53 -07:00
OutlinedButton(
2021-08-09 07:13:42 -07:00
child: Text('Shield Transp. Balance'),
2021-08-13 20:44:53 -07:00
style: OutlinedButton.styleFrom(
side:
BorderSide(width: 1, color: theme.primaryColor)),
2021-08-09 07:13:42 -07:00
onPressed: _onShieldTAddr,
)
]);
}),
Observer(builder: (context) {
2021-08-13 20:44:53 -07:00
final balance = accountManager.showTAddr
2021-08-09 07:13:42 -07:00
? accountManager.tbalance
: accountManager.balance;
return Row(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.baseline,
textBaseline: TextBaseline.ideographic,
children: <Widget>[
2021-08-13 20:44:53 -07:00
Text('${coin.symbol} ${_getBalance_hi(balance)}',
2021-08-09 07:13:42 -07:00
style: theme.textTheme.headline2),
Text('${_getBalance_lo(balance)}'),
]);
}),
Observer(builder: (context) {
2021-08-13 20:44:53 -07:00
final balance = accountManager.showTAddr
2021-08-09 07:13:42 -07:00
? 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(
2021-08-13 20:44:53 -07:00
"1 ${coin.ticker} = ${fx.toStringAsFixed(2)} ${settings.currency}"),
2021-08-09 07:13:42 -07:00
]);
}),
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(),
PnLWidget(),
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.
);
2021-06-26 07:30:12 -07:00
}
2021-07-12 04:32:49 -07:00
void tabTo(int index) {
if (index != _tabController.index) _tabController.animateTo(index);
}
2021-08-07 00:32:10 -07:00
_address() => accountManager.showTAddr
2021-07-10 22:20:53 -07:00
? accountManager.taddress
: (_useSnapAddress ? _snapAddress : accountManager.active.address);
2021-06-26 07:30:12 -07:00
_sign(int b) {
return b < 0 ? '-' : '+';
}
2021-07-09 06:33:39 -07:00
_onQRTap() {
2021-08-07 00:32:10 -07:00
accountManager.toggleShowTAddr();
2021-07-09 06:33:39 -07:00
}
2021-07-10 22:20:53 -07:00
_onAddressCopy() {
Clipboard.setData(ClipboardData(text: _address()));
final snackBar = SnackBar(content: Text('Address copied to clipboard'));
rootScaffoldMessengerKey.currentState.showSnackBar(snackBar);
}
2021-07-09 06:33:39 -07:00
_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?'),
2021-08-13 20:44:53 -07:00
actions: confirmButtons(context, () 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);
})),
2021-07-09 06:33:39 -07:00
);
}
2021-06-26 07:30:12 -07:00
_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');
}
2021-07-19 01:17:23 -07:00
_setupTimer() async {
await _sync();
2021-06-26 07:30:12 -07:00
_timerSync = Timer.periodic(Duration(seconds: 15), (Timer t) {
_trySync();
});
}
2021-08-07 00:32:10 -07:00
double _fx() {
return priceStore.zecPrice;
}
2021-07-19 01:17:23 -07:00
_sync() async {
2021-07-12 04:32:49 -07:00
_syncing = true;
2021-07-19 01:17:23 -07:00
await syncStatus.update();
2021-07-09 22:44:34 -07:00
WarpApi.warpSync(settings.getTx, settings.anchorOffset, (int height) async {
2021-06-26 07:30:12 -07:00
setState(() {
2021-07-12 04:32:49 -07:00
if (height >= 0)
syncStatus.setSyncHeight(height);
else {
_syncing = false;
_trySync();
}
2021-06-26 07:30:12 -07:00
});
});
}
_trySync() async {
priceStore.fetchZecPrice();
if (syncStatus.syncedHeight < 0) return;
await syncStatus.update();
await accountManager.updateUnconfirmedBalance();
if (!syncStatus.isSynced()) {
2021-07-09 22:44:34 -07:00
final res =
await WarpApi.tryWarpSync(settings.getTx, settings.anchorOffset);
2021-06-26 07:30:12 -07:00
if (res == 1) {
// Reorg
final targetHeight = syncStatus.syncedHeight - 10;
WarpApi.rewindToHeight(targetHeight);
syncStatus.setSyncHeight(targetHeight);
} else if (res == 0) {
2021-07-19 01:17:23 -07:00
syncStatus.update();
2021-06-26 07:30:12 -07:00
}
}
2021-07-12 04:32:49 -07:00
await accountManager.fetchAccountData();
2021-07-07 08:40:05 -07:00
await accountManager.updateBalance();
2021-06-26 07:30:12 -07:00
await accountManager.updateUnconfirmedBalance();
2021-07-09 06:33:39 -07:00
accountManager.updateTBalance();
2021-06-26 07:30:12 -07:00
}
2021-07-07 21:22:54 -07:00
_onSnapAddress() {
final address = accountManager.newAddress();
setState(() {
_useSnapAddress = true;
_snapAddress = address;
});
Timer(Duration(seconds: 15), () {
setState(() {
_useSnapAddress = false;
});
});
}
2021-07-09 06:33:39 -07:00
2021-06-26 07:30:12 -07:00
_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;
2021-07-07 08:40:05 -07:00
case "Cold":
_cold();
break;
2021-08-02 22:58:02 -07:00
case "MultiPay":
_multiPay();
break;
2021-08-05 06:43:05 -07:00
case "Broadcast":
_broadcast();
break;
2021-07-09 06:33:39 -07:00
case "Settings":
_settings();
break;
2021-06-26 07:30:12 -07:00
case "About":
showAbout(this.context);
break;
}
}
_backup() async {
final localAuth = LocalAuthentication();
2021-07-30 05:36:46 -07:00
try {
final didAuthenticate = await localAuth.authenticate(
localizedReason: "Please authenticate to show account seed");
if (didAuthenticate) {
Navigator.of(context).pushNamed('/backup', arguments: true);
}
2021-08-09 07:13:42 -07:00
} on PlatformException catch (e) {
2021-07-30 05:36:46 -07:00
await showDialog(
2021-08-09 07:13:42 -07:00
context: context,
2021-07-30 05:36:46 -07:00
barrierDismissible: true,
builder: (context) => AlertDialog(
2021-08-09 07:13:42 -07:00
title: Text('No Authentication Method'),
content: Text(e.message)));
2021-06-26 07:30:12 -07:00
}
}
_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();
},
),
]),
);
}
2021-07-07 08:40:05 -07:00
_cold() {
showDialog(
2021-07-30 05:36:46 -07:00
context: context,
2021-07-07 08:40:05 -07:00
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.'),
2021-08-13 20:44:53 -07:00
actions:
confirmButtons(context, _convertToWatchOnly, okLabel: 'DELETE')
));
2021-07-07 08:40:05 -07:00
}
2021-08-02 22:58:02 -07:00
_multiPay() {
Navigator.of(context).pushNamed('/multipay');
}
2021-08-05 06:43:05 -07:00
_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);
}
}
2021-07-07 08:40:05 -07:00
_convertToWatchOnly() {
accountManager.convertToWatchOnly();
Navigator.of(context).pop();
}
2021-07-09 06:33:39 -07:00
_settings() {
Navigator.of(this.context).pushNamed('/settings');
}
2021-06-26 07:30:12 -07:00
}
class NoteWidget extends StatefulWidget {
2021-07-12 04:32:49 -07:00
void Function(int index) tabTo;
NoteWidget(this.tabTo);
2021-06-26 07:30:12 -07:00
@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(
2021-07-12 04:32:49 -07:00
padding: EdgeInsets.all(4),
2021-06-26 07:30:12 -07:00
scrollDirection: Axis.vertical,
2021-07-12 04:32:49 -07:00
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: [
2021-07-30 02:45:43 -07:00
DataColumn(
2021-08-09 07:13:42 -07:00
label: settings.showConfirmations
? Text('Confs')
: Text('Height'),
2021-07-30 02:45:43 -07:00
onSort: (_, __) {
setState(() {
settings.toggleShowConfirmations();
});
}),
2021-07-12 04:32:49 -07:00
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,
2021-07-30 02:45:43 -07:00
source: NotesDataSource(context, _onRowSelected),
2021-07-12 04:32:49 -07:00
))));
}
_onRowSelected(Note note) {
accountManager.excludeNote(note);
2021-07-10 22:20:53 -07:00
}
}
class NotesDataSource extends DataTableSource {
2021-07-12 04:32:49 -07:00
final BuildContext context;
final Function(Note) onRowSelected;
2021-07-10 22:20:53 -07:00
2021-07-12 04:32:49 -07:00
NotesDataSource(this.context, this.onRowSelected);
2021-07-10 22:20:53 -07:00
@override
DataRow getRow(int index) {
final note = accountManager.notes[index];
2021-07-18 08:59:02 -07:00
final theme = Theme.of(context);
2021-08-09 07:13:42 -07:00
final confsOrHeight = settings.showConfirmations
? syncStatus.latestHeight - note.height + 1
: note.height;
2021-07-12 04:32:49 -07:00
return DataRow.byIndex(
index: index,
selected: note.excluded,
2021-07-18 08:59:02 -07:00
color: MaterialStateColor.resolveWith((states) =>
states.contains(MaterialState.selected)
? theme.primaryColor.withOpacity(0.5)
: theme.backgroundColor),
2021-07-10 22:20:53 -07:00
cells: [
2021-07-30 02:45:43 -07:00
DataCell(Text("$confsOrHeight",
2021-07-10 22:20:53 -07:00
style: !_confirmed(note.height)
? Theme.of(this.context).textTheme.overline
: null)),
DataCell(Text("${note.timestamp}")),
DataCell(Text("${note.value.toStringAsFixed(8)}")),
],
2021-07-12 04:32:49 -07:00
onSelectChanged: (selected) => _noteSelected(note, selected),
2021-07-10 22:20:53 -07:00
);
}
@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;
2021-06-26 07:30:12 -07:00
}
2021-07-12 04:32:49 -07:00
void _noteSelected(Note note, bool selected) {
note.excluded = !note.excluded;
notifyListeners();
onRowSelected(note);
}
2021-06-26 07:30:12 -07:00
}
class HistoryWidget extends StatefulWidget {
2021-07-12 04:32:49 -07:00
void Function(int index) tabTo;
HistoryWidget(this.tabTo);
2021-06-26 07:30:12 -07:00
@override
2021-08-09 07:13:42 -07:00
State<StatefulWidget> createState() => HistoryState();
2021-06-26 07:30:12 -07:00
}
2021-08-09 07:13:42 -07:00
class HistoryState extends State<HistoryWidget>
2021-06-26 07:30:12 -07:00
with AutomaticKeepAliveClientMixin {
@override
bool get wantKeepAlive => true; //Set to true
@override
Widget build(BuildContext context) {
super.build(context);
return SingleChildScrollView(
2021-07-12 04:32:49 -07:00
padding: EdgeInsets.all(4),
2021-06-26 07:30:12 -07:00
scrollDirection: Axis.vertical,
2021-07-12 04:32:49 -07:00
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: [
2021-07-30 02:45:43 -07:00
DataColumn(
2021-08-09 07:13:42 -07:00
label: settings.showConfirmations
? Text('Confs')
: Text('Height'),
2021-07-30 02:45:43 -07:00
onSort: (_, __) {
setState(() {
settings.toggleShowConfirmations();
});
}),
2021-07-12 04:32:49 -07:00
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,
2021-07-30 02:45:43 -07:00
source: HistoryDataSource(context)))));
2021-06-26 07:30:12 -07:00
}
}
2021-07-10 22:20:53 -07:00
class HistoryDataSource extends DataTableSource {
BuildContext context;
HistoryDataSource(this.context);
@override
DataRow getRow(int index) {
final tx = accountManager.txs[index];
2021-08-09 07:13:42 -07:00
final confsOrHeight = settings.showConfirmations
? syncStatus.latestHeight - tx.height + 1
: tx.height;
2021-07-10 22:20:53 -07:00
return DataRow(
cells: [
2021-07-30 02:45:43 -07:00
DataCell(Text("$confsOrHeight")),
2021-07-10 22:20:53 -07:00
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;
}
2021-07-12 04:32:49 -07:00
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) {
2021-07-18 08:59:02 -07:00
final _ = accountManager.dataEpoch;
2021-07-12 04:32:49 -07:00
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,
)
];
}
}
2021-07-18 08:59:02 -07:00
2021-08-09 07:13:42 -07:00
class PnLWidget extends StatefulWidget {
@override
State<StatefulWidget> createState() => PnLState();
}
class PnLState extends State<PnLWidget> with AutomaticKeepAliveClientMixin {
@override
bool get wantKeepAlive => true; //Set to true
@override
Widget build(BuildContext context) {
return Column(children: [
FormBuilderRadioGroup(
orientation: OptionsOrientation.horizontal,
name: 'Pnl',
initialValue: accountManager.pnlSeriesIndex,
onChanged: (v) {
setState(() {
accountManager.setPnlSeriesIndex(v);
});
},
options: [
FormBuilderFieldOption(child: Text('Real'), value: 0),
FormBuilderFieldOption(child: Text('M/M'), value: 1),
FormBuilderFieldOption(child: Text('Total'), value: 2),
FormBuilderFieldOption(child: Text('Price'), value: 3),
FormBuilderFieldOption(child: Text('Qty'), value: 4),
FormBuilderFieldOption(child: Text('Table'), value: 5),
]),
Observer(builder: (context) {
final _ = accountManager.pnls;
return Expanded(
child: accountManager.pnlSeriesIndex != 5
? PnLChart(accountManager.pnlSeriesIndex)
: PnLTable());
})
]);
}
}
class PnLChart extends StatelessWidget {
final data = accountManager.pnls;
final seriesIndex;
PnLChart(this.seriesIndex);
@override
Widget build(BuildContext context) {
final axisColor = charts.ColorUtil.fromDartColor(
Theme.of(context).textTheme.headline5.color);
final seriesList = _createSeries(data, seriesIndex,
charts.ColorUtil.fromDartColor(Theme.of(context).primaryColor));
return new charts.TimeSeriesChart(
seriesList,
animate: false,
dateTimeFactory: const charts.LocalDateTimeFactory(),
primaryMeasureAxis: charts.NumericAxisSpec(
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 _seriesData(PnL pnl, int index) {
switch (index) {
case 0:
return pnl.realized;
case 1:
return pnl.unrealized;
case 2:
return pnl.realized + pnl.unrealized;
case 3:
return pnl.price;
case 4:
return pnl.amount;
}
}
static List<charts.Series<PnL, DateTime>> _createSeries(
List<PnL> data, int index, charts.Color lineColor) {
return [
new charts.Series<PnL, DateTime>(
id: 'P/L',
colorFn: (_, __) => lineColor,
domainFn: (PnL pnl, _) => pnl.timestamp,
measureFn: (PnL pnl, _) => _seriesData(pnl, index),
data: data,
),
];
}
}
class PnLTable extends StatelessWidget {
@override
Widget build(BuildContext context) {
return SingleChildScrollView(
child: PaginatedDataTable(
columns: [
DataColumn(label: Text('Date/Time')),
DataColumn(label: Text('Qty'), numeric: true),
DataColumn(label: Text('Price'), numeric: true),
DataColumn(label: Text('Realized'), numeric: true),
DataColumn(label: Text('M/M'), numeric: true),
DataColumn(label: Text('Total'), numeric: true),
],
columnSpacing: 16,
showCheckboxColumn: false,
availableRowsPerPage: [5, 10, 25, 100],
onRowsPerPageChanged: (int value) {
settings.setRowsPerPage(value);
},
showFirstLastButtons: true,
rowsPerPage: settings.rowsPerPage,
source: PnLDataSource(context)));
}
}
class PnLDataSource extends DataTableSource {
BuildContext context;
final dateFormat = DateFormat("MM-dd");
PnLDataSource(this.context);
@override
DataRow getRow(int index) {
final pnl = accountManager.pnls[index];
final ts = dateFormat.format(pnl.timestamp);
return DataRow(cells: [
DataCell(Text("$ts")),
DataCell(Text("${pnl.amount.toStringAsFixed(2)}")),
DataCell(Text("${pnl.price.toStringAsFixed(3)}")),
DataCell(Text("${pnl.realized.toStringAsFixed(3)}")),
DataCell(Text("${pnl.unrealized.toStringAsFixed(3)}")),
DataCell(Text("${(pnl.realized + pnl.unrealized).toStringAsFixed(3)}")),
]);
}
@override
bool get isRowCountApproximate => false;
@override
int get rowCount => accountManager.pnls.length;
@override
int get selectedRowCount => 0;
}
2021-07-18 08:59:02 -07:00
class ContactsWidget extends StatefulWidget {
@override
State<StatefulWidget> createState() => ContactsState();
}
2021-08-09 07:13:42 -07:00
class ContactsState extends State<ContactsWidget>
with AutomaticKeepAliveClientMixin {
@override
bool get wantKeepAlive => true; //Set to true
2021-07-18 08:59:02 -07:00
@override
Widget build(BuildContext context) {
return Observer(builder: (context) {
final _ = accountManager.dataEpoch;
final theme = Theme.of(context);
2021-08-09 07:13:42 -07:00
return Padding(
padding: EdgeInsets.all(12),
child: Column(children: [
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),
)),
]));
2021-07-18 08:59:02 -07:00
});
}
_onContact(Contact contact) {
Navigator.of(context).pushNamed('/send', arguments: contact);
}
}