import 'dart:isolate'; import 'dart:typed_data'; import 'dart:math' as math; import 'package:json_annotation/json_annotation.dart'; import 'package:charts_flutter/flutter.dart' as charts show MaterialPalette; import 'package:flutter/material.dart'; import 'package:intl/intl.dart'; import 'package:mobx/mobx.dart'; import 'package:path/path.dart'; import 'package:sqflite/sqflite.dart'; import 'package:warp_api/warp_api.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:http/http.dart' as http; import 'dart:convert' as convert; import 'package:convert/convert.dart'; import 'package:flex_color_scheme/flex_color_scheme.dart'; import 'main.dart'; part 'store.g.dart'; class Settings = _Settings with _$Settings; abstract class _Settings with Store { @observable String ldUrl; @observable String ldUrlChoice; @observable int anchorOffset; @observable bool getTx; @observable int rowsPerPage; @observable String theme; @observable String themeBrightness; @observable ThemeData themeData = ThemeData.light(); @observable bool showConfirmations = false; @observable String currency = "USD"; @observable List currencies = ["USD"]; @observable String chartRange = '1Y'; var palette = charts.MaterialPalette.blue; @action Future restore() async { final prefs = await SharedPreferences.getInstance(); ldUrlChoice = prefs.getString('lightwalletd_choice') ?? "Lightwalletd"; ldUrl = prefs.getString('lightwalletd_custom') ?? ""; prefs.setString('lightwalletd_choice', ldUrlChoice); prefs.setString('lightwalletd_custom', ldUrl); anchorOffset = prefs.getInt('anchor_offset') ?? 3; getTx = prefs.getBool('get_txinfo') ?? true; rowsPerPage = prefs.getInt('rows_per_age') ?? 10; theme = prefs.getString('theme') ?? "zcash"; themeBrightness = prefs.getString('theme_brightness') ?? "dark"; showConfirmations = prefs.getBool('show_confirmations') ?? false; currency = prefs.getString('currency') ?? "USD"; chartRange = prefs.getString('chart_range') ?? "1Y"; _updateThemeData(); Future.microtask(_loadCurrencies); // lazily return true; } @action Future setURLChoice(String choice) async { ldUrlChoice = choice; final prefs = await SharedPreferences.getInstance(); prefs.setString('lightwalletd_choice', ldUrlChoice); updateLWD(); } @action Future setURL(String url) async { ldUrl = url; final prefs = await SharedPreferences.getInstance(); prefs.setString('lightwalletd_custom', ldUrl); updateLWD(); } @action Future setAnchorOffset(int offset) async { final prefs = await SharedPreferences.getInstance(); anchorOffset = offset; prefs.setInt('anchor_offset', offset); } @action Future setTheme(String thm) async { final prefs = await SharedPreferences.getInstance(); theme = thm; prefs.setString('theme', thm); _updateThemeData(); } @action Future setThemeBrightness(String brightness) async { final prefs = await SharedPreferences.getInstance(); themeBrightness = brightness; prefs.setString('theme_brightness', brightness); _updateThemeData(); } void _updateThemeData() { FlexScheme scheme; switch (theme) { case 'zcash': scheme = FlexScheme.mango; palette = charts.MaterialPalette.gray; break; case 'blue': scheme = FlexScheme.bahamaBlue; palette = charts.MaterialPalette.blue; break; case 'pink': scheme = FlexScheme.sakura; palette = charts.MaterialPalette.pink; break; case 'coffee': scheme = FlexScheme.espresso; palette = charts.MaterialPalette.gray; break; } switch (themeBrightness) { case 'light': themeData = FlexColorScheme.light(scheme: scheme).toTheme; break; case 'dark': themeData = FlexColorScheme.dark(scheme: scheme).toTheme; break; } } @action Future setChartRange(String v) async { final prefs = await SharedPreferences.getInstance(); chartRange = v; prefs.setString('chart_range', chartRange); accountManager.fetchPNL(); } String getLWD() { switch (ldUrlChoice) { case "custom": return ldUrl; default: return coin.lwd .firstWhere((lwd) => lwd.name == ldUrlChoice, orElse: () => coin.lwd.first) .url; } } void updateLWD() { WarpApi.updateLWD(getLWD()); } @action Future updateGetTx(bool v) async { final prefs = await SharedPreferences.getInstance(); getTx = v; prefs.setBool('get_txinfo', v); } @action Future setRowsPerPage(int v) async { final prefs = await SharedPreferences.getInstance(); rowsPerPage = v; prefs.setInt('rows_per_age', v); } @action Future toggleShowConfirmations() async { final prefs = await SharedPreferences.getInstance(); showConfirmations = !showConfirmations; prefs.setBool('show_confirmations', showConfirmations); } @action Future setCurrency(String newCurrency) async { final prefs = await SharedPreferences.getInstance(); currency = newCurrency; prefs.setString('currency', currency); await priceStore.fetchZecPrice(); await accountManager.fetchPNL(); } @action Future _loadCurrencies() async { final base = "api.coingecko.com"; final uri = Uri.https(base, '/api/v3/simple/supported_vs_currencies'); final rep = await http.get(uri); if (rep.statusCode == 200) { final _currencies = convert.jsonDecode(rep.body) as List; final c = _currencies.map((v) => (v as String).toUpperCase()).toList(); c.sort(); currencies = c; } } } class AccountManager = _AccountManager with _$AccountManager; abstract class _AccountManager with Store { Database db; @observable Account active; @observable bool canPay = false; @observable int balance = 0; @observable int unconfirmedBalance = 0; @observable String taddress = ""; @observable bool showTAddr = false; @observable int tbalance = 0; @observable List notes = []; @observable List txs = []; @observable int lastTxHeight = 0; @observable int dataEpoch = 0; @observable List spendings = []; @observable List accountBalances = []; @observable List pnls = []; @observable List accounts = []; @observable SortOrder noteSortOrder = SortOrder.Unsorted; @observable SortOrder txSortOrder = SortOrder.Unsorted; @observable int pnlSeriesIndex = 0; @observable List contacts = []; Future init() async { db = await getDatabase(); await resetToDefaultAccount(); } Future resetToDefaultAccount() async { await refresh(); if (accounts.isNotEmpty) { final prefs = await SharedPreferences.getInstance(); final account = prefs.getInt('account') ?? accounts[0].id; setActiveAccountId(account); } } refresh() async { accounts = await _list(); } @action Future setActiveAccount(Account account) async { if (account == null) return; final prefs = await SharedPreferences.getInstance(); prefs.setInt('account', account.id); final List res1 = await db.rawQuery( "SELECT address FROM taddrs WHERE account = ?1", [account.id]); taddress = res1.isNotEmpty ? res1[0]['address'] : ""; showTAddr = false; WarpApi.setMempoolAccount(account.id); final List res2 = await db.rawQuery( "SELECT sk FROM accounts WHERE id_account = ?1", [account.id]); canPay = res2.isNotEmpty && res2[0]['sk'] != null; active = account; print("Active account = ${account.id}"); await _fetchData(account.id); } @action Future setActiveAccountId(int idAccount) async { final account = accounts.firstWhere((account) => account.id == idAccount, orElse: () => accounts.isNotEmpty ? accounts[0] : null); await setActiveAccount(account); } String newAddress() { return WarpApi.newAddress(active.id); } Future getBackup() async { final List res = await db.rawQuery( "SELECT seed, sk, ivk FROM accounts WHERE id_account = ?1", [active.id]); if (res.isEmpty) return null; final row = res[0]; final seed = row['seed']; final sk = row['sk']; final ivk = row['ivk']; int type; if (seed != null) type = 0; else if (sk != null) type = 1; else if (ivk != null) type = 2; return Backup(type, seed, sk, ivk); } Future _getBalance(int accountId) async { final List res = await db.rawQuery( "SELECT SUM(value) AS value FROM received_notes WHERE account = ?1 AND (spent IS NULL OR spent = 0)", [accountId]); if (res.isEmpty) return 0; return res[0]['value'] ?? 0; } Future getBalanceSpendable(int height) async { final List res = await db.rawQuery( "SELECT SUM(value) AS value FROM received_notes WHERE account = ?1 AND spent IS NULL " "AND height <= ?2 AND (excluded IS NULL OR NOT excluded)", [active.id, height]); if (res.isEmpty) return 0; return res[0]['value'] ?? 0; } @action Future updateUnconfirmedBalance() async { unconfirmedBalance = await WarpApi.mempoolSync(); } isEmpty() async { final List res = await db.rawQuery("SELECT name FROM accounts", []); return res.isEmpty; } Future> _list() async { final List res = await db.rawQuery( "WITH notes AS (SELECT a.id_account, a.name, a.address, CASE WHEN r.spent IS NULL THEN r.value ELSE 0 END AS nv FROM accounts a LEFT JOIN received_notes r ON a.id_account = r.account) " "SELECT id_account, name, address, COALESCE(sum(nv), 0) AS balance FROM notes GROUP by id_account", []); return res .map((r) => Account(r['id_account'], r['name'], r['address'], r['balance'])) .toList(); } @action Future delete(int account) async { await db.rawDelete("DELETE FROM accounts WHERE id_account = ?1", [account]); await db.rawDelete("DELETE FROM taddrs WHERE account = ?1", [account]); } @action Future changeAccountName(String name) async { await db.execute("UPDATE accounts SET name = ?2 WHERE id_account = ?1", [active.id, name]); await refresh(); await setActiveAccountId(active.id); } @action Future updateBalance() async { if (active == null) return; balance = await _getBalance(active.id); } @action Future fetchAccountData() async { if (active == null) return; await _fetchData(active.id); } @action void toggleShowTAddr() { showTAddr = !showTAddr; } Future _fetchData(int accountId) async { await _updateBalance(accountId); final hasNewTx = await _fetchNotesAndHistory(accountId); int countNewPrices = await WarpApi.syncHistoricalPrices(settings.currency); if (hasNewTx) { await _fetchSpending(accountId); await _fetchAccountBalanceTimeSeries(accountId); await _fetchContacts(accountId); } if (countNewPrices > 0 || pnls.isEmpty || hasNewTx) await _fetchPNL(accountId); } final DateFormat noteDateFormat = DateFormat("yy-MM-dd HH:mm"); final DateFormat txDateFormat = DateFormat("MM-dd HH:mm"); Future _updateBalance(int accountId) async { final _balance = await _getBalance(accountId); if (_balance == balance) return; balance = _balance; dataEpoch = DateTime.now().millisecondsSinceEpoch; } Future _fetchNotesAndHistory(int accountId) async { final List res0 = await db.rawQuery( "SELECT MAX(height) as height FROM transactions WHERE account = ?1", [accountId]); if (res0.isEmpty) return false; final _lastTxHeight = res0[0]['height'] ?? 0; if (lastTxHeight == _lastTxHeight) return false; lastTxHeight = _lastTxHeight; final List res = await db.rawQuery( "SELECT n.id_note, n.height, n.value, t.timestamp, n.excluded, n.spent FROM received_notes n, transactions t " "WHERE n.account = ?1 AND (n.spent IS NULL OR n.spent = 0) " "AND n.tx = t.id_tx", [accountId]); notes = res.map((row) { final id = row['id_note']; final height = row['height']; final timestamp = noteDateFormat .format(DateTime.fromMillisecondsSinceEpoch(row['timestamp'] * 1000)); final excluded = (row['excluded'] ?? 0) != 0; final spent = row['spent'] == 0; return Note( id, height, timestamp, row['value'] / ZECUNIT, excluded, spent); }).toList(); final List res2 = await db.rawQuery( "SELECT id_tx, txid, height, timestamp, address, value, memo FROM transactions WHERE account = ?1", [accountId]); txs = res2.map((row) { Uint8List txid = row['txid']; final fullTxId = hex.encode(txid.reversed.toList()); final shortTxid = fullTxId.substring(0, 8); final timestamp = txDateFormat .format(DateTime.fromMillisecondsSinceEpoch(row['timestamp'] * 1000)); return Tx(row['id_tx'], row['height'], timestamp, shortTxid, fullTxId, row['value'] / ZECUNIT, row['address'], row['memo']); }).toList(); dataEpoch = DateTime.now().millisecondsSinceEpoch; return true; } @computed List get sortedNotes { var notes2 = [...notes]; return _sortNoteAmount(notes2, noteSortOrder); } @action Future sortNoteAmount() async { noteSortOrder = nextSortOrder(noteSortOrder); } List _sortNoteAmount(List notes, SortOrder order) { switch (order) { case SortOrder.Ascending: notes.sort((a, b) => a.value.compareTo(b.value)); break; case SortOrder.Descending: notes.sort((a, b) => -a.value.compareTo(b.value)); break; case SortOrder.Unsorted: notes.sort((a, b) => -a.height.compareTo(b.height)); break; } return notes; } @computed List get sortedTxs { var txs2 = [...txs]; return _sortTxAmount(txs2, txSortOrder); } @action Future sortTxAmount() async { txSortOrder = nextSortOrder(txSortOrder); } List _sortTxAmount(List txs, SortOrder order) { switch (order) { case SortOrder.Ascending: txs.sort((a, b) => a.value.compareTo(b.value)); break; case SortOrder.Descending: txs.sort((a, b) => -a.value.compareTo(b.value)); break; case SortOrder.Unsorted: txs.sort((a, b) => -a.height.compareTo(b.height)); break; } return txs; } Future _fetchSpending(int accountId) async { final cutoff = DateTime.now().add(Duration(days: -30)).millisecondsSinceEpoch / 1000; final List res = await db.rawQuery( "SELECT SUM(value) as v, address FROM transactions WHERE account = ?1 AND timestamp >= ?2 AND value < 0 GROUP BY address ORDER BY v ASC LIMIT 10", [accountId, cutoff]); spendings = res.map((row) { final address = row['address'] ?? ""; final value = -row['v'] / ZECUNIT; return Spending(addressLeftTrim(address), value); }).toList(); } Future _fetchAccountBalanceTimeSeries(int accountId) async { final now = DateTime.now(); final today = DateTime.utc(now.year, now.month, now.day); final end = today; final start = today.add(Duration(days: -30)); final cutoff = start.millisecondsSinceEpoch ~/ 1000; final List res = await db.rawQuery( "SELECT timestamp, value FROM transactions WHERE account = ?1 AND timestamp >= ?2 ORDER BY timestamp DESC", [accountId, cutoff]); List _accountBalances = []; var b = balance; _accountBalances.add(AccountBalance(DateTime.now(), b / ZECUNIT)); for (var row in res) { final timestamp = DateTime.fromMillisecondsSinceEpoch(row['timestamp'] * 1000); final value = row['value']; final ab = AccountBalance(timestamp, b / ZECUNIT); _accountBalances.add(ab); b -= value; } _accountBalances.add(AccountBalance(start, b / ZECUNIT)); _accountBalances = _accountBalances.reversed.toList(); accountBalances = sampleDaily( _accountBalances, start.millisecondsSinceEpoch, end.millisecondsSinceEpoch, (AccountBalance ab) => ab.time.millisecondsSinceEpoch ~/ DAY_MS, (AccountBalance ab) => ab.balance, (acc, v) => v, 0.0); } @action Future fetchPNL() async { if (active == null) return; await _fetchPNL(active.id); } Future _fetchPNL(int accountId) async { final now = DateTime.now(); final today = DateTime.utc(now.year, now.month, now.day); var days = 365; switch (settings.chartRange) { case '1M': days = 30; break; case '3M': days = 90; break; case '6M': days = 180; break; } final cutoff = today.add(Duration(days: -days)); final List res1 = await db.rawQuery( "SELECT timestamp, value FROM transactions WHERE timestamp >= ?2 AND account = ?1", [accountId, cutoff.millisecondsSinceEpoch ~/ 1000]); final List trades = []; for (var row in res1) { final dt = DateTime.fromMillisecondsSinceEpoch(row['timestamp'] * 1000); final qty = row['value'] / ZECUNIT; trades.add(Trade(dt, qty)); } final portfolioTimeSeries = sampleDaily( trades, cutoff.millisecondsSinceEpoch, today.millisecondsSinceEpoch, (t) => t.dt.millisecondsSinceEpoch ~/ DAY_MS, (t) => t, (acc, t) => acc + t.qty, 0.0); final List res2 = await db.rawQuery( "SELECT timestamp, price FROM historical_prices WHERE timestamp >= ?2 AND currency = ?1", [settings.currency, cutoff.millisecondsSinceEpoch ~/ 1000]); final List quotes = []; for (var row in res2) { final dt = DateTime.fromMillisecondsSinceEpoch(row['timestamp'] * 1000); final price = row['price']; quotes.add(Quote(dt, price)); } var prevBalance = 0.0; var cash = 0.0; var realized = 0.0; final List _pnls = []; for (var i = 0; i < quotes.length; i++) { final dt = quotes[i].dt; final price = quotes[i].price; final balance = portfolioTimeSeries[i].value; final qty = balance - prevBalance; final closeQty = qty * balance < 0 ? math.min(qty.abs(), prevBalance.abs()) * qty.sign : 0.0; final openQty = qty - closeQty; final avgPrice = prevBalance != 0 ? cash / prevBalance : 0.0; cash += openQty * price + closeQty * avgPrice; realized += closeQty * (avgPrice - price); final unrealized = price * balance - cash; final pnl = PnL(dt, price, balance, realized, unrealized); _pnls.add(pnl); prevBalance = balance; } pnls = _pnls; } @action Future convertToWatchOnly() async { await db.rawUpdate( "UPDATE accounts SET seed = NULL, sk = NULL WHERE id_account = ?1", [active.id]); canPay = false; } @action Future excludeNote(Note note) async { await db.execute( "UPDATE received_notes SET excluded = ?2 WHERE id_note = ?1", [note.id, note.excluded]); } void updateTBalance() { if (active == null) return; int balance = WarpApi.getTBalance(active.id); if (balance != tbalance) tbalance = balance; } Future _fetchContacts(int accountId) async { List res = await db.rawQuery( "SELECT name, address FROM contacts WHERE account = ?1 ORDER BY name", [accountId]); contacts = []; for (var c in res) { final contact = Contact(c['name'], c['address']); contacts.add(contact); } } @action void setPnlSeriesIndex(int index) { pnlSeriesIndex = index; } } class Account { final int id; final String name; final String address; final int balance; Account(this.id, this.name, this.address, this.balance); } class PriceStore = _PriceStore with _$PriceStore; abstract class _PriceStore with Store { @observable double zecPrice = 0.0; @action Future fetchZecPrice() async { final base = "api.coingecko.com"; final uri = Uri.https(base, '/api/v3/simple/price', {'ids': coin.currency, 'vs_currencies': settings.currency}); final rep = await http.get(uri); if (rep.statusCode == 200) { final json = convert.jsonDecode(rep.body) as Map; final p = json[coin.currency][settings.currency.toLowerCase()]; zecPrice = (p is double) ? p : (p as int).toDouble(); } else zecPrice = 0.0; } } class SyncStatus = _SyncStatus with _$SyncStatus; abstract class _SyncStatus with Store { Database _db; init() async { var databasesPath = await getDatabasesPath(); final path = join(databasesPath, 'zec.db'); _db = await openDatabase(path); await update(); } @observable int syncedHeight = -1; @observable int latestHeight = 0; bool isSynced() { return syncedHeight < 0 || syncedHeight == latestHeight; } @action setSyncHeight(int height) { syncedHeight = height; } @action Future update() async { latestHeight = await WarpApi.getLatestHeight(); final _syncedHeight = Sqflite.firstIntValue( await _db.rawQuery("SELECT MAX(height) FROM blocks")) ?? 0; if (_syncedHeight > 0) syncedHeight = _syncedHeight; return syncedHeight == latestHeight; } } class MultiPayStore = _MultiPayStore with _$MultiPayStore; abstract class _MultiPayStore with Store { @observable ObservableList recipients = ObservableList.of([]); @action void addRecipient(Recipient recipient) { recipients.add(recipient); } @action void removeRecipient(int index) { recipients.removeAt(index); } @action void clear() { recipients.clear(); } } var progressPort = ReceivePort(); var progressStream = progressPort.asBroadcastStream(); class Note { int id; int height; String timestamp; double value; bool excluded; bool spent; Note(this.id, this.height, this.timestamp, this.value, this.excluded, this.spent); } class Tx { int id; int height; String timestamp; String txid; String fullTxId; double value; String address; String memo; Tx(this.id, this.height, this.timestamp, this.txid, this.fullTxId, this.value, this.address, this.memo); } class Spending { final String address; final double amount; Spending(this.address, this.amount); } class AccountBalance { final DateTime time; final double balance; AccountBalance(this.time, this.balance); } class Backup { int type; final String seed; final String sk; final String ivk; Backup(this.type, this.seed, this.sk, this.ivk); String value() { switch (type) { case 0: return seed; case 1: return sk; case 2: return ivk; } return ""; } } class Contact { final String name; final String address; Contact(this.name, this.address); } String addressLeftTrim(String address) => "..." + address.substring(math.max(address.length - 6, 0)); enum SortOrder { Unsorted, Ascending, Descending, } SortOrder nextSortOrder(SortOrder order) => SortOrder.values[(order.index + 1) % 3]; @JsonSerializable() class Recipient { final String address; final int amount; final String memo; Recipient(this.address, this.amount, this.memo); factory Recipient.fromJson(Map json) => _$RecipientFromJson(json); Map toJson() => _$RecipientToJson(this); } class PnL { final DateTime timestamp; final double price; final double amount; final double realized; final double unrealized; PnL(this.timestamp, this.price, this.amount, this.realized, this.unrealized); @override String toString() { return "$timestamp $price $amount $realized $unrealized"; } } class TimeSeriesPoint { final int day; final V value; TimeSeriesPoint(this.day, this.value); } class Trade { final DateTime dt; final qty; Trade(this.dt, this.qty); } class Portfolio { final DateTime dt; final qty; Portfolio(this.dt, this.qty); } class Quote { final DateTime dt; final price; Quote(this.dt, this.price); }