2024-10-23 15:38:31 +00:00
|
|
|
import 'dart:async';
|
2024-05-13 20:59:11 +00:00
|
|
|
import 'dart:io';
|
|
|
|
|
2024-05-05 01:44:50 +00:00
|
|
|
import 'package:cake_wallet/bitcoin/bitcoin.dart';
|
|
|
|
import 'package:cake_wallet/ethereum/ethereum.dart';
|
|
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
2024-12-13 23:32:36 +00:00
|
|
|
import 'package:cake_wallet/main.dart';
|
2024-11-12 03:26:09 +00:00
|
|
|
import 'package:cake_wallet/monero/monero.dart';
|
2024-05-05 01:44:50 +00:00
|
|
|
import 'package:cake_wallet/polygon/polygon.dart';
|
2024-12-13 23:32:36 +00:00
|
|
|
import 'package:cake_wallet/routes.dart';
|
|
|
|
import 'package:cake_wallet/src/screens/connect_device/connect_device_page.dart';
|
2024-05-08 14:26:57 +00:00
|
|
|
import 'package:cake_wallet/utils/device_info.dart';
|
2024-05-13 20:59:11 +00:00
|
|
|
import 'package:cake_wallet/wallet_type_utils.dart';
|
|
|
|
import 'package:cw_core/hardware/device_connection_type.dart';
|
2024-12-09 18:23:59 +00:00
|
|
|
import 'package:cw_core/utils/print_verbose.dart';
|
2024-05-05 01:44:50 +00:00
|
|
|
import 'package:cw_core/wallet_base.dart';
|
|
|
|
import 'package:cw_core/wallet_type.dart';
|
2024-12-13 23:32:36 +00:00
|
|
|
import 'package:flutter/widgets.dart';
|
2024-10-23 15:38:31 +00:00
|
|
|
|
|
|
|
import 'package:ledger_flutter_plus/ledger_flutter_plus.dart' as sdk;
|
|
|
|
import 'package:mobx/mobx.dart';
|
2024-05-05 01:44:50 +00:00
|
|
|
import 'package:permission_handler/permission_handler.dart';
|
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
part 'ledger_view_model.g.dart';
|
|
|
|
|
|
|
|
class LedgerViewModel = LedgerViewModelBase with _$LedgerViewModel;
|
|
|
|
|
|
|
|
abstract class LedgerViewModelBase with Store {
|
|
|
|
// late final Ledger ledger;
|
|
|
|
late final sdk.LedgerInterface ledgerPlusBLE;
|
|
|
|
late final sdk.LedgerInterface ledgerPlusUSB;
|
2024-05-05 01:44:50 +00:00
|
|
|
|
2024-05-13 20:59:11 +00:00
|
|
|
bool get _doesSupportHardwareWallets {
|
|
|
|
if (!DeviceInfo.instance.isMobile) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isMoneroOnly) {
|
2024-10-23 15:38:31 +00:00
|
|
|
return DeviceConnectionType.supportedConnectionTypes(
|
|
|
|
WalletType.monero, Platform.isIOS)
|
2024-05-13 20:59:11 +00:00
|
|
|
.isNotEmpty;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
LedgerViewModelBase() {
|
2024-05-13 20:59:11 +00:00
|
|
|
if (_doesSupportHardwareWallets) {
|
2024-10-23 15:38:31 +00:00
|
|
|
reaction((_) => bleIsEnabled, (_) {
|
|
|
|
if (bleIsEnabled) _initBLE();
|
|
|
|
});
|
|
|
|
updateBleState();
|
|
|
|
|
|
|
|
if (!Platform.isIOS) {
|
|
|
|
ledgerPlusUSB = sdk.LedgerInterface.usb();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@observable
|
|
|
|
bool bleIsEnabled = false;
|
|
|
|
|
|
|
|
bool _bleIsInitialized = false;
|
|
|
|
Future<void> _initBLE() async {
|
|
|
|
if (bleIsEnabled && !_bleIsInitialized) {
|
2024-12-13 23:32:36 +00:00
|
|
|
ledgerPlusBLE = sdk.LedgerInterface.ble(
|
|
|
|
onPermissionRequest: (_) async {
|
|
|
|
Map<Permission, PermissionStatus> statuses = await [
|
|
|
|
Permission.bluetoothScan,
|
|
|
|
Permission.bluetoothConnect,
|
|
|
|
Permission.bluetoothAdvertise,
|
|
|
|
].request();
|
|
|
|
|
|
|
|
return statuses.values.where((status) => status.isDenied).isEmpty;
|
|
|
|
},
|
|
|
|
bleOptions:
|
|
|
|
sdk.BluetoothOptions(maxScanDuration: Duration(minutes: 5)));
|
2024-10-23 15:38:31 +00:00
|
|
|
_bleIsInitialized = true;
|
2024-05-08 14:26:57 +00:00
|
|
|
}
|
|
|
|
}
|
2024-05-05 01:44:50 +00:00
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
Future<void> updateBleState() async {
|
|
|
|
final bleState = await sdk.UniversalBle.getBluetoothAvailabilityState();
|
|
|
|
|
|
|
|
final newState = bleState == sdk.AvailabilityState.poweredOn;
|
|
|
|
|
|
|
|
if (newState != bleIsEnabled) bleIsEnabled = newState;
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream<sdk.LedgerDevice> scanForBleDevices() => ledgerPlusBLE.scan();
|
|
|
|
|
|
|
|
Stream<sdk.LedgerDevice> scanForUsbDevices() => ledgerPlusUSB.scan();
|
|
|
|
|
2024-12-13 23:32:36 +00:00
|
|
|
Future<void> stopScanning() async {
|
|
|
|
await ledgerPlusBLE.stopScanning();
|
|
|
|
if (!Platform.isIOS) {
|
|
|
|
await ledgerPlusUSB.stopScanning();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
Future<void> connectLedger(sdk.LedgerDevice device, WalletType type) async {
|
2024-12-17 18:57:57 +00:00
|
|
|
_isConnecting = true;
|
|
|
|
_connectingWalletType = type;
|
2024-10-23 15:38:31 +00:00
|
|
|
if (isConnected) {
|
|
|
|
try {
|
2024-12-13 23:32:36 +00:00
|
|
|
await _connection!.disconnect().catchError((_) {});
|
2024-10-23 15:38:31 +00:00
|
|
|
} catch (_) {}
|
|
|
|
}
|
2024-12-17 18:57:57 +00:00
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
final ledger = device.connectionType == sdk.ConnectionType.ble
|
|
|
|
? ledgerPlusBLE
|
|
|
|
: ledgerPlusUSB;
|
|
|
|
|
2024-12-17 18:57:57 +00:00
|
|
|
if (_connectionChangeSubscription == null) {
|
|
|
|
_connectionChangeSubscription = ledger.deviceStateChanges
|
|
|
|
.listen(_connectionChangeListener);
|
2024-10-23 15:38:31 +00:00
|
|
|
}
|
2024-05-05 01:44:50 +00:00
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
_connection = await ledger.connect(device);
|
2024-12-17 18:57:57 +00:00
|
|
|
_isConnecting = false;
|
2024-05-05 01:44:50 +00:00
|
|
|
}
|
|
|
|
|
2024-12-17 18:57:57 +00:00
|
|
|
StreamSubscription<sdk.BleConnectionState>? _connectionChangeSubscription;
|
2024-10-23 15:38:31 +00:00
|
|
|
sdk.LedgerConnection? _connection;
|
2024-12-17 18:57:57 +00:00
|
|
|
bool _isConnecting = true;
|
|
|
|
WalletType? _connectingWalletType;
|
|
|
|
|
|
|
|
void _connectionChangeListener(
|
|
|
|
sdk.BleConnectionState event, ) {
|
|
|
|
printV('Ledger Device State Changed: $event');
|
|
|
|
if (event == sdk.BleConnectionState.disconnected && !_isConnecting) {
|
|
|
|
_connection = null;
|
|
|
|
if (_connectingWalletType == WalletType.monero) {
|
|
|
|
monero!.resetLedgerConnection();
|
|
|
|
|
|
|
|
Navigator.of(navigatorKey.currentContext!).pushNamed(
|
|
|
|
Routes.connectDevices,
|
|
|
|
arguments: ConnectDevicePageParams(
|
|
|
|
walletType: WalletType.monero,
|
|
|
|
allowChangeWallet: true,
|
|
|
|
isReconnect: true,
|
|
|
|
onConnectDevice: (context, ledgerVM) async {
|
|
|
|
Navigator.of(context).pop();
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-05-05 01:44:50 +00:00
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
bool get isConnected => _connection != null && !(_connection!.isDisconnected);
|
2024-05-05 01:44:50 +00:00
|
|
|
|
2024-10-23 15:38:31 +00:00
|
|
|
sdk.LedgerConnection get connection => _connection!;
|
2024-05-05 01:44:50 +00:00
|
|
|
|
|
|
|
void setLedger(WalletBase wallet) {
|
|
|
|
switch (wallet.type) {
|
2024-11-12 03:26:09 +00:00
|
|
|
case WalletType.monero:
|
|
|
|
return monero!.setLedgerConnection(wallet, connection);
|
2024-05-05 01:44:50 +00:00
|
|
|
case WalletType.bitcoin:
|
2024-10-23 15:38:31 +00:00
|
|
|
case WalletType.litecoin:
|
|
|
|
return bitcoin!.setLedgerConnection(wallet, connection);
|
2024-05-05 01:44:50 +00:00
|
|
|
case WalletType.ethereum:
|
2024-10-23 15:38:31 +00:00
|
|
|
return ethereum!.setLedgerConnection(wallet, connection);
|
2024-05-05 01:44:50 +00:00
|
|
|
case WalletType.polygon:
|
2024-10-23 15:38:31 +00:00
|
|
|
return polygon!.setLedgerConnection(wallet, connection);
|
2024-05-05 01:44:50 +00:00
|
|
|
default:
|
|
|
|
throw Exception('Unexpected wallet type: ${wallet.type}');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String? interpretErrorCode(String errorCode) {
|
|
|
|
switch (errorCode) {
|
|
|
|
case "6985":
|
|
|
|
return S.current.ledger_error_tx_rejected_by_user;
|
|
|
|
case "5515":
|
|
|
|
return S.current.ledger_error_device_locked;
|
|
|
|
case "6d02": // UNKNOWN_APDU
|
|
|
|
case "6511":
|
|
|
|
case "6e00":
|
|
|
|
return S.current.ledger_error_wrong_app;
|
|
|
|
default:
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|