import 'package:cake_wallet/bitcoin/bitcoin.dart'; import 'package:cake_wallet/di.dart'; import 'package:cake_wallet/nano/nano.dart'; import 'package:cake_wallet/ethereum/ethereum.dart'; import 'package:cw_core/nano_account_info_response.dart'; import 'package:hive/hive.dart'; import 'package:mobx/mobx.dart'; import 'package:cake_wallet/store/app_store.dart'; import 'package:cw_core/wallet_base.dart'; import 'package:cake_wallet/core/generate_wallet_password.dart'; import 'package:cake_wallet/core/wallet_creation_service.dart'; import 'package:cw_core/wallet_credentials.dart'; import 'package:cw_core/wallet_type.dart'; import 'package:cw_core/wallet_info.dart'; import 'package:cake_wallet/view_model/wallet_creation_vm.dart'; import 'package:cake_wallet/monero/monero.dart'; import 'package:cake_wallet/haven/haven.dart'; import 'package:cake_wallet/view_model/restore/restore_mode.dart'; part 'wallet_restore_view_model.g.dart'; class WalletRestoreViewModel = WalletRestoreViewModelBase with _$WalletRestoreViewModel; abstract class WalletRestoreViewModelBase extends WalletCreationVM with Store { WalletRestoreViewModelBase(AppStore appStore, WalletCreationService walletCreationService, Box walletInfoSource, {required WalletType type}) : hasSeedLanguageSelector = type == WalletType.monero || type == WalletType.haven, hasBlockchainHeightLanguageSelector = type == WalletType.monero || type == WalletType.haven, hasRestoreFromPrivateKey = type == WalletType.ethereum || type == WalletType.nano || type == WalletType.banano, isButtonEnabled = false, mode = WalletRestoreMode.seed, super(appStore, walletInfoSource, walletCreationService, type: type, isRecovery: true) { switch (type) { case WalletType.monero: case WalletType.haven: case WalletType.ethereum: availableModes = WalletRestoreMode.values; break; case WalletType.nano: case WalletType.banano: availableModes = [WalletRestoreMode.seed, WalletRestoreMode.keys]; break; default: availableModes = [WalletRestoreMode.seed]; break; } isButtonEnabled = !hasSeedLanguageSelector && !hasBlockchainHeightLanguageSelector; walletCreationService.changeWalletType(type: type); } static const moneroSeedMnemonicLength = 25; static const electrumSeedMnemonicLength = 24; static const electrumShortSeedMnemonicLength = 12; late List availableModes; final bool hasSeedLanguageSelector; final bool hasBlockchainHeightLanguageSelector; final bool hasRestoreFromPrivateKey; @observable WalletRestoreMode mode; @observable bool isButtonEnabled; @override WalletCredentials getCredentials(dynamic options) { final password = generateWalletPassword(); final height = options['height'] as int? ?? 0; name = options['name'] as String; DerivationInfo? derivationInfo = options["derivationInfo"] as DerivationInfo?; if (mode == WalletRestoreMode.seed) { final seed = options['seed'] as String; switch (type) { case WalletType.monero: return monero!.createMoneroRestoreWalletFromSeedCredentials( name: name, height: height, mnemonic: seed, password: password); case WalletType.bitcoin: return bitcoin!.createBitcoinRestoreWalletFromSeedCredentials( name: name, mnemonic: seed, password: password, derivationType: derivationInfo?.derivationType, derivationPath: derivationInfo?.derivationPath, ); case WalletType.litecoin: return bitcoin!.createBitcoinRestoreWalletFromSeedCredentials( name: name, mnemonic: seed, password: password); case WalletType.haven: return haven!.createHavenRestoreWalletFromSeedCredentials( name: name, height: height, mnemonic: seed, password: password); case WalletType.ethereum: return ethereum!.createEthereumRestoreWalletFromSeedCredentials( name: name, mnemonic: seed, password: password); case WalletType.nano: return nano!.createNanoRestoreWalletFromSeedCredentials( name: name, mnemonic: seed, password: password, derivationType: derivationInfo?.derivationType, ); default: break; } } if (mode == WalletRestoreMode.keys) { final viewKey = options['viewKey'] as String?; final spendKey = options['spendKey'] as String?; final address = options['address'] as String?; switch (type) { case WalletType.monero: return monero!.createMoneroRestoreWalletFromKeysCredentials( name: name, height: height, spendKey: spendKey!, viewKey: viewKey!, address: address!, password: password, language: 'English', ); case WalletType.haven: return haven!.createHavenRestoreWalletFromKeysCredentials( name: name, height: height, spendKey: spendKey!, viewKey: viewKey!, address: address!, password: password, language: 'English', ); case WalletType.ethereum: return ethereum!.createEthereumRestoreWalletFromPrivateKey( name: name, privateKey: options['private_key'] as String, password: password, ); case WalletType.nano: return nano!.createNanoRestoreWalletFromKeysCredentials( name: name, password: password, seedKey: options['private_key'] as String, derivationType: options["derivationType"] as DerivationType, ); default: break; } } throw Exception('Unexpected type: ${type.toString()}'); } Future> getDerivationInfo(dynamic credentials) async { var list = []; var walletType = credentials["walletType"] as WalletType; var appStore = getIt.get(); var node = appStore.settingsStore.getCurrentNode(walletType); switch (walletType) { case WalletType.bitcoin: String? mnemonic = credentials['seed'] as String?; return bitcoin!.getDerivationsFromMnemonic(mnemonic: mnemonic!, node: node); case WalletType.nano: String? mnemonic = credentials['seed'] as String?; String? seedKey = credentials['private_key'] as String?; AccountInfoResponse? bip39Info = await nanoUtil!.getInfoFromSeedOrMnemonic( DerivationType.bip39, mnemonic: mnemonic, seedKey: seedKey, node: node); AccountInfoResponse? standardInfo = await nanoUtil!.getInfoFromSeedOrMnemonic( DerivationType.nano, mnemonic: mnemonic, seedKey: seedKey, node: node, ); if (standardInfo?.balance != null) { list.add(DerivationInfo( derivationType: DerivationType.nano, balance: nanoUtil!.getRawAsUsableString(standardInfo!.balance, nanoUtil!.rawPerNano), address: standardInfo.address!, height: standardInfo.confirmationHeight, )); } if (bip39Info?.balance != null) { list.add(DerivationInfo( derivationType: DerivationType.bip39, balance: nanoUtil!.getRawAsUsableString(bip39Info!.balance, nanoUtil!.rawPerNano), address: bip39Info.address!, height: bip39Info.confirmationHeight, )); } break; default: break; } return list; } Future> getDerivationTypes(dynamic options) async { final seedKey = options['private_key'] as String?; final mnemonic = options['seed'] as String?; WalletType walletType = options['walletType'] as WalletType; var appStore = getIt.get(); var node = appStore.settingsStore.getCurrentNode(walletType); switch (type) { case WalletType.bitcoin: return bitcoin!.compareDerivationMethods(mnemonic: mnemonic!, node: node); case WalletType.nano: return nanoUtil!.compareDerivationMethods( mnemonic: mnemonic, privateKey: seedKey, node: node, ); case WalletType.litecoin: return [DerivationType.electrum2]; default: break; } // throw Exception('Unexpected type: ${type.toString()}'); return [DerivationType.def]; } DerivationInfo getDefaultDerivation() { switch (type) { case WalletType.nano: return DerivationInfo( derivationType: DerivationType.nano, ); case WalletType.bitcoin: case WalletType.litecoin: default: return DerivationInfo( derivationType: DerivationType.electrum2, derivationPath: "m/0'/1", ); } } @override Future process(WalletCredentials credentials) async { if (mode == WalletRestoreMode.keys) { return walletCreationService.restoreFromKeys(credentials); } return walletCreationService.restoreFromSeed(credentials); } }