2024-02-23 22:41:26 +00:00
|
|
|
import 'package:bitcoin_base/bitcoin_base.dart';
|
2021-12-24 12:52:08 +00:00
|
|
|
import 'package:cw_bitcoin/bitcoin_mnemonic.dart';
|
2022-10-12 17:09:57 +00:00
|
|
|
import 'package:cw_core/crypto_currency.dart';
|
2021-12-24 12:52:08 +00:00
|
|
|
import 'package:cw_core/unspent_coins_info.dart';
|
|
|
|
import 'package:hive/hive.dart';
|
|
|
|
import 'package:mobx/mobx.dart';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:bitcoin_flutter/bitcoin_flutter.dart' as bitcoin;
|
|
|
|
import 'package:cw_bitcoin/electrum_wallet_snapshot.dart';
|
|
|
|
import 'package:cw_bitcoin/electrum_wallet.dart';
|
|
|
|
import 'package:cw_core/wallet_info.dart';
|
|
|
|
import 'package:cw_bitcoin/bitcoin_address_record.dart';
|
|
|
|
import 'package:cw_bitcoin/electrum_balance.dart';
|
|
|
|
import 'package:cw_bitcoin/bitcoin_wallet_addresses.dart';
|
|
|
|
|
|
|
|
part 'bitcoin_wallet.g.dart';
|
|
|
|
|
|
|
|
class BitcoinWallet = BitcoinWalletBase with _$BitcoinWallet;
|
|
|
|
|
|
|
|
abstract class BitcoinWalletBase extends ElectrumWallet with Store {
|
2024-02-20 22:19:25 +00:00
|
|
|
BitcoinWalletBase({
|
|
|
|
required String mnemonic,
|
|
|
|
required String password,
|
|
|
|
required WalletInfo walletInfo,
|
|
|
|
required Box<UnspentCoinsInfo> unspentCoinsInfo,
|
|
|
|
required Uint8List seedBytes,
|
|
|
|
String? addressPageType,
|
|
|
|
BasedUtxoNetwork? networkParam,
|
|
|
|
List<BitcoinAddressRecord>? initialAddresses,
|
|
|
|
ElectrumBalance? initialBalance,
|
|
|
|
Map<String, int>? initialRegularAddressIndex,
|
|
|
|
Map<String, int>? initialChangeAddressIndex,
|
|
|
|
List<BitcoinAddressRecord>? initialSilentAddresses,
|
|
|
|
int initialSilentAddressIndex = 0,
|
|
|
|
SilentPaymentOwner? silentAddress,
|
|
|
|
}) : super(
|
2021-12-24 12:52:08 +00:00
|
|
|
mnemonic: mnemonic,
|
|
|
|
password: password,
|
|
|
|
walletInfo: walletInfo,
|
|
|
|
unspentCoinsInfo: unspentCoinsInfo,
|
2024-02-23 22:41:26 +00:00
|
|
|
networkType: networkParam == null
|
|
|
|
? bitcoin.bitcoin
|
|
|
|
: networkParam == BitcoinNetwork.mainnet
|
|
|
|
? bitcoin.bitcoin
|
|
|
|
: bitcoin.testnet,
|
2021-12-24 12:52:08 +00:00
|
|
|
initialAddresses: initialAddresses,
|
2022-10-12 17:09:57 +00:00
|
|
|
initialBalance: initialBalance,
|
|
|
|
seedBytes: seedBytes,
|
|
|
|
currency: CryptoCurrency.btc) {
|
2021-12-24 12:52:08 +00:00
|
|
|
walletAddresses = BitcoinWalletAddresses(
|
2024-02-20 22:19:25 +00:00
|
|
|
walletInfo,
|
|
|
|
initialAddresses: initialAddresses,
|
|
|
|
initialRegularAddressIndex: initialRegularAddressIndex,
|
|
|
|
initialChangeAddressIndex: initialChangeAddressIndex,
|
|
|
|
initialSilentAddresses: initialSilentAddresses,
|
|
|
|
initialSilentAddressIndex: initialSilentAddressIndex,
|
|
|
|
silentAddress: silentAddress,
|
|
|
|
mainHd: hd,
|
|
|
|
sideHd: bitcoin.HDWallet.fromSeed(seedBytes, network: networkType).derivePath("m/0'/1"),
|
|
|
|
network: networkParam ?? network,
|
|
|
|
);
|
2024-01-23 05:15:24 +00:00
|
|
|
autorun((_) {
|
|
|
|
this.walletAddresses.isEnabledAutoGenerateSubaddress = this.isEnabledAutoGenerateSubaddress;
|
|
|
|
});
|
2021-12-24 12:52:08 +00:00
|
|
|
}
|
|
|
|
|
2022-10-12 17:09:57 +00:00
|
|
|
static Future<BitcoinWallet> create({
|
|
|
|
required String mnemonic,
|
|
|
|
required String password,
|
|
|
|
required WalletInfo walletInfo,
|
|
|
|
required Box<UnspentCoinsInfo> unspentCoinsInfo,
|
2024-02-23 22:41:26 +00:00
|
|
|
String? addressPageType,
|
|
|
|
BasedUtxoNetwork? network,
|
2022-10-12 17:09:57 +00:00
|
|
|
List<BitcoinAddressRecord>? initialAddresses,
|
2024-02-20 22:19:25 +00:00
|
|
|
List<BitcoinAddressRecord>? initialSilentAddresses,
|
2022-10-12 17:09:57 +00:00
|
|
|
ElectrumBalance? initialBalance,
|
2024-02-20 22:19:25 +00:00
|
|
|
Map<String, int>? initialRegularAddressIndex,
|
|
|
|
Map<String, int>? initialChangeAddressIndex,
|
|
|
|
int initialSilentAddressIndex = 0,
|
2022-10-12 17:09:57 +00:00
|
|
|
}) async {
|
2024-02-23 22:41:26 +00:00
|
|
|
final seedBytes = await mnemonicToSeedBytes(mnemonic);
|
2022-10-12 17:09:57 +00:00
|
|
|
return BitcoinWallet(
|
2024-02-20 22:19:25 +00:00
|
|
|
mnemonic: mnemonic,
|
|
|
|
password: password,
|
|
|
|
walletInfo: walletInfo,
|
|
|
|
unspentCoinsInfo: unspentCoinsInfo,
|
|
|
|
initialAddresses: initialAddresses,
|
|
|
|
initialSilentAddresses: initialSilentAddresses,
|
|
|
|
initialSilentAddressIndex: initialSilentAddressIndex,
|
2024-02-23 22:41:26 +00:00
|
|
|
silentAddress: await SilentPaymentOwner.fromPrivateKeys(
|
|
|
|
scanPrivkey: ECPrivate.fromHex(bitcoin.HDWallet.fromSeed(
|
|
|
|
seedBytes,
|
|
|
|
network: network == BitcoinNetwork.testnet ? bitcoin.testnet : bitcoin.bitcoin,
|
|
|
|
).derivePath(SCAN_PATH).privKey!),
|
|
|
|
spendPrivkey: ECPrivate.fromHex(bitcoin.HDWallet.fromSeed(
|
|
|
|
seedBytes,
|
|
|
|
network: network == BitcoinNetwork.testnet ? bitcoin.testnet : bitcoin.bitcoin,
|
|
|
|
).derivePath(SPEND_PATH).privKey!),
|
2024-02-20 22:19:25 +00:00
|
|
|
hrp: network == BitcoinNetwork.testnet ? 'tsp' : 'sp'),
|
|
|
|
initialBalance: initialBalance,
|
2024-02-23 22:41:26 +00:00
|
|
|
seedBytes: seedBytes,
|
2024-02-20 22:19:25 +00:00
|
|
|
initialRegularAddressIndex: initialRegularAddressIndex,
|
|
|
|
initialChangeAddressIndex: initialChangeAddressIndex,
|
|
|
|
addressPageType: addressPageType,
|
|
|
|
networkParam: network,
|
|
|
|
);
|
2022-10-12 17:09:57 +00:00
|
|
|
}
|
|
|
|
|
2021-12-24 12:52:08 +00:00
|
|
|
static Future<BitcoinWallet> open({
|
2022-10-12 17:09:57 +00:00
|
|
|
required String name,
|
|
|
|
required WalletInfo walletInfo,
|
|
|
|
required Box<UnspentCoinsInfo> unspentCoinsInfo,
|
|
|
|
required String password,
|
2021-12-24 12:52:08 +00:00
|
|
|
}) async {
|
2024-02-23 22:41:26 +00:00
|
|
|
final snp = await ElectrumWalletSnapshot.load(name, walletInfo.type, password,
|
|
|
|
walletInfo.network != null ? BasedUtxoNetwork.fromName(walletInfo.network!) : null);
|
|
|
|
|
|
|
|
final seedBytes = await mnemonicToSeedBytes(snp.mnemonic);
|
2021-12-24 12:52:08 +00:00
|
|
|
return BitcoinWallet(
|
2024-02-20 22:19:25 +00:00
|
|
|
mnemonic: snp.mnemonic,
|
|
|
|
password: password,
|
|
|
|
walletInfo: walletInfo,
|
|
|
|
unspentCoinsInfo: unspentCoinsInfo,
|
|
|
|
initialAddresses: snp.addresses,
|
|
|
|
initialSilentAddresses: snp.silentAddresses,
|
|
|
|
initialSilentAddressIndex: snp.silentAddressIndex,
|
2024-02-23 22:41:26 +00:00
|
|
|
silentAddress: await SilentPaymentOwner.fromPrivateKeys(
|
|
|
|
scanPrivkey: ECPrivate.fromHex(bitcoin.HDWallet.fromSeed(
|
|
|
|
seedBytes,
|
|
|
|
network: snp.network == BitcoinNetwork.testnet ? bitcoin.testnet : bitcoin.bitcoin,
|
|
|
|
).derivePath(SCAN_PATH).privKey!),
|
|
|
|
spendPrivkey: ECPrivate.fromHex(bitcoin.HDWallet.fromSeed(
|
|
|
|
seedBytes,
|
|
|
|
network: snp.network == BitcoinNetwork.testnet ? bitcoin.testnet : bitcoin.bitcoin,
|
|
|
|
).derivePath(SPEND_PATH).privKey!),
|
2024-02-20 22:19:25 +00:00
|
|
|
hrp: snp.network == BitcoinNetwork.testnet ? 'tsp' : 'sp'),
|
|
|
|
initialBalance: snp.balance,
|
2024-02-23 22:41:26 +00:00
|
|
|
seedBytes: seedBytes,
|
2024-02-20 22:19:25 +00:00
|
|
|
initialRegularAddressIndex: snp.regularAddressIndex,
|
|
|
|
initialChangeAddressIndex: snp.changeAddressIndex,
|
|
|
|
addressPageType: snp.addressPageType,
|
|
|
|
networkParam: snp.network,
|
|
|
|
);
|
2021-12-24 12:52:08 +00:00
|
|
|
}
|
2024-02-20 22:19:25 +00:00
|
|
|
}
|