import 'package:isar/isar.dart'; import 'package:stackwallet/models/isar/models/blockchain_data/address.dart'; import 'package:stackwallet/services/event_bus/events/global/updated_in_background_event.dart'; import 'package:stackwallet/services/event_bus/global_event_bus.dart'; import 'package:stackwallet/services/node_service.dart'; import 'package:stackwallet/utilities/amount/amount.dart'; import 'package:stackwallet/utilities/enums/coin_enum.dart'; import 'package:stackwallet/wallets/crypto_currency/coins/bitcoin.dart'; import 'package:stackwallet/wallets/wallet/intermediate/bip39_hd_wallet.dart'; import 'package:stackwallet/wallets/wallet/mixins/electrumx_mixin.dart'; import 'package:tuple/tuple.dart'; class BitcoinWallet extends Bip39HDWallet with ElectrumXMixin { @override int get isarTransactionVersion => 2; BitcoinWallet( super.cryptoCurrency, { required NodeService nodeService, }) { // TODO: [prio=low] ensure this hack isn't needed assert(cryptoCurrency is Bitcoin); this.nodeService = nodeService; } // =========================================================================== Future> _fetchAllOwnAddresses() async { final allAddresses = await mainDB .getAddresses(walletId) .filter() .not() .group( (q) => q .typeEqualTo(AddressType.nonWallet) .or() .subTypeEqualTo(AddressSubType.nonWallet), ) .findAll(); return allAddresses; } // =========================================================================== @override Future refresh() { // TODO: implement refresh throw UnimplementedError(); } @override Future updateTransactions() async { final currentChainHeight = await fetchChainHeight(); final data = await fetchTransactionsV1( addresses: await _fetchAllOwnAddresses(), currentChainHeight: currentChainHeight, ); await mainDB.addNewTransactionData( data .map( (e) => Tuple2( e.transaction, e.address, ), ) .toList(), walletId, ); // TODO: [prio=med] get rid of this and watch isar instead // quick hack to notify manager to call notifyListeners if // transactions changed if (data.isNotEmpty) { GlobalEventBus.instance.fire( UpdatedInBackgroundEvent( "Transactions updated/added for: $walletId ${info.name}", walletId, ), ); } } @override Future updateUTXOs() { // TODO: implement updateUTXOs throw UnimplementedError(); } @override Future pingCheck() async { try { final result = await electrumX.ping(); return result; } catch (_) { return false; } } @override Future updateChainHeight() async { final height = await fetchChainHeight(); await info.updateCachedChainHeight( newHeight: height, isar: mainDB.isar, ); } @override Amount roughFeeEstimate(int inputCount, int outputCount, int feeRatePerKB) { return Amount( rawValue: BigInt.from( ((42 + (272 * inputCount) + (128 * outputCount)) / 4).ceil() * (feeRatePerKB / 1000).ceil()), fractionDigits: info.coin.decimals, ); } }