import 'package:cake_wallet/bitcoin/bitcoin.dart'; import 'package:cake_wallet/core/execution_state.dart'; import 'package:cake_wallet/core/wallet_creation_service.dart'; import 'package:cake_wallet/di.dart'; import 'package:cake_wallet/entities/background_tasks.dart'; import 'package:cake_wallet/entities/generate_name.dart'; import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/nano/nano.dart'; import 'package:cake_wallet/store/app_store.dart'; import 'package:cake_wallet/store/settings_store.dart'; import 'package:cake_wallet/view_model/restore/restore_mode.dart'; import 'package:cake_wallet/view_model/restore/restore_wallet.dart'; import 'package:cake_wallet/view_model/seed_settings_view_model.dart'; import 'package:cw_core/pathForWallet.dart'; import 'package:cw_core/wallet_base.dart'; import 'package:cw_core/wallet_credentials.dart'; import 'package:cw_core/wallet_info.dart'; import 'package:cw_core/wallet_type.dart'; import 'package:hive/hive.dart'; import 'package:mobx/mobx.dart'; import 'package:polyseed/polyseed.dart'; part 'wallet_creation_vm.g.dart'; class WalletCreationVM = WalletCreationVMBase with _$WalletCreationVM; abstract class WalletCreationVMBase with Store { WalletCreationVMBase(this._appStore, this._walletInfoSource, this.walletCreationService, this.seedSettingsViewModel, {required this.type, required this.isRecovery}) : state = InitialExecutionState(), name = ''; @observable bool _useTestnet = false; @computed bool get useTestnet => _useTestnet; @observable String name; @observable ExecutionState state; @observable String? walletPassword; @observable String? repeatedWalletPassword; bool get hasWalletPassword => SettingsStoreBase.walletPasswordDirectInput; WalletType type; final bool isRecovery; final WalletCreationService walletCreationService; final Box _walletInfoSource; final AppStore _appStore; final SeedSettingsViewModel seedSettingsViewModel; bool isPolyseed(String seed) => (type == WalletType.monero || type == WalletType.wownero) && (Polyseed.isValidSeed(seed) || (seed.split(" ").length == 14)); bool nameExists(String name) => walletCreationService.exists(name); bool typeExists(WalletType type) => walletCreationService.typeExists(type); Future create({dynamic options, RestoredWallet? restoreWallet}) async { final type = restoreWallet?.type ?? this.type; try { state = IsExecutingState(); if (name.isEmpty) { name = await generateName(); } if (hasWalletPassword && (walletPassword?.isEmpty ?? true)) { throw Exception(S.current.wallet_password_is_empty); } if (hasWalletPassword && walletPassword != repeatedWalletPassword) { throw Exception(S.current.repeated_password_is_incorrect); } walletCreationService.checkIfExists(name); final dirPath = await pathForWalletDir(name: name, type: type); final path = await pathForWallet(name: name, type: type); final credentials = restoreWallet != null ? await getWalletCredentialsFromQRCredentials(restoreWallet) : getCredentials(options); final walletInfo = WalletInfo.external( id: WalletBase.idFor(name, type), name: name, type: type, isRecovery: isRecovery, restoreHeight: credentials.height ?? 0, date: DateTime.now(), path: path, dirPath: dirPath, address: '', showIntroCakePayCard: (!walletCreationService.typeExists(type)) && type != WalletType.haven, derivationInfo: credentials.derivationInfo ?? getDefaultCreateDerivation(), hardwareWalletType: credentials.hardwareWalletType, parentAddress: credentials.parentAddress, ); credentials.walletInfo = walletInfo; final wallet = restoreWallet != null ? await processFromRestoredWallet(credentials, restoreWallet) : await process(credentials); walletInfo.address = wallet.walletAddresses.address; await _walletInfoSource.add(walletInfo); await _appStore.changeCurrentWallet(wallet); getIt.get().registerBackgroundService(); _appStore.authenticationStore.allowed(); state = ExecutedSuccessfullyState(); } catch (e, s) { print("error: $e"); print("stack: $s"); state = FailureState(e.toString()); } } DerivationInfo? getDefaultCreateDerivation() { final useBip39ForBitcoin = seedSettingsViewModel.bitcoinSeedType.type == DerivationType.bip39; final useBip39ForNano = seedSettingsViewModel.nanoSeedType.type == DerivationType.bip39; switch (type) { case WalletType.nano: if (useBip39ForNano) { return DerivationInfo(derivationType: DerivationType.bip39); } return DerivationInfo(derivationType: DerivationType.nano); case WalletType.bitcoin: if (useBip39ForBitcoin) { return DerivationInfo( derivationType: DerivationType.bip39, derivationPath: "m/84'/0'/0'", description: "Standard BIP84 native segwit", scriptType: "p2wpkh", ); } return bitcoin!.getElectrumDerivations()[DerivationType.electrum]!.first; case WalletType.litecoin: if (useBip39ForBitcoin) { return DerivationInfo( derivationType: DerivationType.bip39, derivationPath: "m/84'/2'/0'", description: "Default Litecoin", scriptType: "p2wpkh", ); } return bitcoin!.getElectrumDerivations()[DerivationType.electrum]!.first; default: return null; } } DerivationInfo? getCommonRestoreDerivation() { final useElectrum = seedSettingsViewModel.bitcoinSeedType.type == DerivationType.electrum; final useNanoStandard = seedSettingsViewModel.nanoSeedType.type == DerivationType.nano; switch (this.type) { case WalletType.nano: if (useNanoStandard) { return DerivationInfo(derivationType: DerivationType.nano); } return DerivationInfo(derivationType: DerivationType.bip39); case WalletType.bitcoin: if (useElectrum) { return bitcoin!.getElectrumDerivations()[DerivationType.electrum]!.first; } return DerivationInfo( derivationType: DerivationType.bip39, derivationPath: "m/84'/0'/0'/0", description: "Standard BIP84 native segwit", scriptType: "p2wpkh", ); case WalletType.litecoin: if (useElectrum) { return bitcoin!.getElectrumDerivations()[DerivationType.electrum]!.first; } return DerivationInfo( derivationType: DerivationType.bip39, derivationPath: "m/84'/2'/0'/0", description: "Standard BIP84 native segwit (litecoin)", scriptType: "p2wpkh", ); default: return null; } } Future> getDerivationInfoFromQRCredentials( RestoredWallet restoreWallet) async { var list = []; final walletType = restoreWallet.type; var appStore = getIt.get(); var node = appStore.settingsStore.getCurrentNode(walletType); switch (walletType) { case WalletType.bitcoin: case WalletType.litecoin: final derivationList = await bitcoin!.getDerivationsFromMnemonic( mnemonic: restoreWallet.mnemonicSeed!, node: node, passphrase: restoreWallet.passphrase, ); if (derivationList.firstOrNull?.transactionsCount == 0 && derivationList.length > 1) return []; return derivationList; case WalletType.nano: return nanoUtil!.getDerivationsFromMnemonic( mnemonic: restoreWallet.mnemonicSeed!, node: node, ); default: break; } return list; } WalletCredentials getCredentials(dynamic options) => throw UnimplementedError(); Future process(WalletCredentials credentials) => throw UnimplementedError(); Future getWalletCredentialsFromQRCredentials( RestoredWallet restoreWallet) async => throw UnimplementedError(); Future processFromRestoredWallet( WalletCredentials credentials, RestoredWallet restoreWallet) => throw UnimplementedError(); @action void toggleUseTestnet(bool? value) { _useTestnet = value ?? !_useTestnet; } }