import 'dart:convert'; import 'package:cw_bitcoin/encryption_file_utils.dart'; import 'package:cw_core/pathForWallet.dart'; import 'package:cw_core/wallet_info.dart'; import 'package:mobx/mobx.dart'; import 'package:cw_core/transaction_history.dart'; import 'package:cw_bitcoin/electrum_transaction_info.dart'; part 'electrum_transaction_history.g.dart'; const transactionsHistoryFileName = 'transactions.json'; class ElectrumTransactionHistory = ElectrumTransactionHistoryBase with _$ElectrumTransactionHistory; abstract class ElectrumTransactionHistoryBase extends TransactionHistoryBase with Store { ElectrumTransactionHistoryBase( {required this.walletInfo, required String password, required this.encryptionFileUtils}) : _password = password, _height = 0 { transactions = ObservableMap(); } final WalletInfo walletInfo; final EncryptionFileUtils encryptionFileUtils; String _password; int _height; Future init() async => await _load(); @override void addOne(ElectrumTransactionInfo transaction) => transactions[transaction.id] = transaction; @override void addMany(Map transactions) => transactions.forEach((_, tx) => _updateOrInsert(tx)); @override Future save() async { try { final dirPath = await pathForWalletDir(name: walletInfo.name, type: walletInfo.type); final path = '$dirPath/$transactionsHistoryFileName'; final data = json.encode({'height': _height, 'transactions': transactions}); await encryptionFileUtils.write(path: path, password: _password, data: data); } catch (e) { print('Error while save bitcoin transaction history: ${e.toString()}'); } } Future changePassword(String password) async { _password = password; await save(); } Future> _read() async { final dirPath = await pathForWalletDir(name: walletInfo.name, type: walletInfo.type); final path = '$dirPath/$transactionsHistoryFileName'; final content = await encryptionFileUtils.read(path: path, password: _password); return json.decode(content) as Map; } Future _load() async { try { final content = await _read(); final txs = content['transactions'] as Map? ?? {}; txs.entries.forEach((entry) { final val = entry.value; if (val is Map) { final tx = ElectrumTransactionInfo.fromJson(val, walletInfo.type); _updateOrInsert(tx); } }); _height = content['height'] as int; } catch (e) { print(e); } } void _updateOrInsert(ElectrumTransactionInfo transaction) { if (transactions[transaction.id] == null) { transactions[transaction.id] = transaction; } else { final originalTx = transactions[transaction.id]; originalTx?.confirmations = transaction.confirmations; originalTx?.amount = transaction.amount; originalTx?.height = transaction.height; originalTx?.date ??= transaction.date; originalTx?.isPending = transaction.isPending; originalTx?.direction = transaction.direction; } } }