2024-05-04 20:44:50 -05:00
|
|
|
import 'dart:async';
|
|
|
|
import 'dart:io';
|
|
|
|
|
|
|
|
import 'package:cake_wallet/generated/i18n.dart';
|
|
|
|
import 'package:cake_wallet/src/screens/base_page.dart';
|
2024-05-21 05:26:49 +03:00
|
|
|
import 'package:cake_wallet/src/screens/connect_device/debug_device_page.dart';
|
2024-05-04 20:44:50 -05:00
|
|
|
import 'package:cake_wallet/src/screens/connect_device/widgets/device_tile.dart';
|
|
|
|
import 'package:cake_wallet/themes/extensions/cake_text_theme.dart';
|
|
|
|
import 'package:cake_wallet/utils/responsive_layout_util.dart';
|
|
|
|
import 'package:cake_wallet/view_model/hardware_wallet/ledger_view_model.dart';
|
|
|
|
import 'package:cw_core/wallet_type.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:ledger_flutter/ledger_flutter.dart';
|
|
|
|
import 'package:permission_handler/permission_handler.dart';
|
|
|
|
|
|
|
|
typedef OnConnectDevice = void Function(BuildContext, LedgerViewModel);
|
|
|
|
|
|
|
|
class ConnectDevicePageParams {
|
|
|
|
final WalletType walletType;
|
|
|
|
final OnConnectDevice onConnectDevice;
|
|
|
|
|
|
|
|
ConnectDevicePageParams({required this.walletType, required this.onConnectDevice});
|
|
|
|
}
|
|
|
|
|
|
|
|
class ConnectDevicePage extends BasePage {
|
|
|
|
final WalletType walletType;
|
|
|
|
final OnConnectDevice onConnectDevice;
|
|
|
|
final LedgerViewModel ledgerVM;
|
|
|
|
|
|
|
|
ConnectDevicePage(ConnectDevicePageParams params, this.ledgerVM)
|
|
|
|
: walletType = params.walletType,
|
|
|
|
onConnectDevice = params.onConnectDevice;
|
|
|
|
|
|
|
|
@override
|
|
|
|
String get title => S.current.restore_title_from_hardware_wallet;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget body(BuildContext context) => ConnectDevicePageBody(walletType, onConnectDevice, ledgerVM);
|
|
|
|
}
|
|
|
|
|
|
|
|
class ConnectDevicePageBody extends StatefulWidget {
|
|
|
|
final WalletType walletType;
|
|
|
|
final OnConnectDevice onConnectDevice;
|
|
|
|
final LedgerViewModel ledgerVM;
|
|
|
|
|
|
|
|
const ConnectDevicePageBody(this.walletType, this.onConnectDevice, this.ledgerVM);
|
|
|
|
|
|
|
|
@override
|
|
|
|
ConnectDevicePageBodyState createState() => ConnectDevicePageBodyState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class ConnectDevicePageBodyState extends State<ConnectDevicePageBody> {
|
|
|
|
final imageLedger = 'assets/images/ledger_nano.png';
|
|
|
|
|
|
|
|
final ledger = Ledger(
|
|
|
|
options: LedgerOptions(
|
|
|
|
scanMode: ScanMode.balanced,
|
|
|
|
maxScanDuration: const Duration(minutes: 5),
|
|
|
|
),
|
|
|
|
onPermissionRequest: (_) async {
|
|
|
|
Map<Permission, PermissionStatus> statuses = await [
|
|
|
|
Permission.bluetoothScan,
|
|
|
|
Permission.bluetoothConnect,
|
|
|
|
Permission.bluetoothAdvertise,
|
|
|
|
].request();
|
|
|
|
|
|
|
|
return statuses.values.where((status) => status.isDenied).isEmpty;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
var bleIsEnabled = true;
|
|
|
|
var bleDevices = <LedgerDevice>[];
|
|
|
|
var usbDevices = <LedgerDevice>[];
|
|
|
|
|
|
|
|
late Timer? _usbRefreshTimer = null;
|
|
|
|
late Timer? _bleRefreshTimer = null;
|
|
|
|
late StreamSubscription<LedgerDevice>? _bleRefresh = null;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
2024-05-17 08:15:19 -05:00
|
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
|
|
_bleRefreshTimer = Timer.periodic(Duration(seconds: 1), (_) => _refreshBleDevices());
|
|
|
|
|
|
|
|
if (Platform.isAndroid) {
|
|
|
|
_usbRefreshTimer = Timer.periodic(Duration(seconds: 1), (_) => _refreshUsbDevices());
|
|
|
|
}
|
|
|
|
});
|
2024-05-04 20:44:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
_bleRefreshTimer?.cancel();
|
|
|
|
_usbRefreshTimer?.cancel();
|
|
|
|
_bleRefresh?.cancel();
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _refreshUsbDevices() async {
|
|
|
|
final dev = await ledger.listUsbDevices();
|
|
|
|
if (usbDevices.length != dev.length) setState(() => usbDevices = dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _refreshBleDevices() async {
|
2024-05-17 08:15:19 -05:00
|
|
|
try {
|
|
|
|
_bleRefresh = ledger.scan().listen((device) => setState(() => bleDevices.add(device)))
|
|
|
|
..onError((e) {
|
2024-05-21 05:26:49 +03:00
|
|
|
throw e.toString();
|
2024-05-17 08:15:19 -05:00
|
|
|
});
|
|
|
|
setState(() => bleIsEnabled = true);
|
2024-05-04 20:44:50 -05:00
|
|
|
_bleRefreshTimer?.cancel();
|
|
|
|
_bleRefreshTimer = null;
|
2024-05-17 08:15:19 -05:00
|
|
|
} catch (e) {
|
|
|
|
setState(() => bleIsEnabled = false);
|
2024-05-04 20:44:50 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _connectToDevice(LedgerDevice device) async {
|
|
|
|
await widget.ledgerVM.connectLedger(device);
|
|
|
|
widget.onConnectDevice(context, widget.ledgerVM);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Center(
|
|
|
|
child: Container(
|
|
|
|
width: ResponsiveLayoutUtilBase.kDesktopMaxWidthConstraint,
|
|
|
|
height: double.infinity,
|
|
|
|
padding: EdgeInsets.symmetric(vertical: 24, horizontal: 24),
|
|
|
|
child: SingleChildScrollView(
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
Padding(
|
|
|
|
padding: EdgeInsets.only(left: 20, right: 20, bottom: 20),
|
|
|
|
child: Text(
|
|
|
|
Platform.isIOS
|
|
|
|
? S.of(context).connect_your_hardware_wallet_ios
|
|
|
|
: S.of(context).connect_your_hardware_wallet,
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 16,
|
|
|
|
fontWeight: FontWeight.w500,
|
|
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor),
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
),
|
|
|
|
),
|
2024-05-17 08:15:19 -05:00
|
|
|
// DeviceTile(
|
|
|
|
// onPressed: () => Navigator.of(context).push(
|
|
|
|
// MaterialPageRoute<void>(
|
|
|
|
// builder: (BuildContext context) => DebugDevicePage(),
|
|
|
|
// ),
|
|
|
|
// ),
|
|
|
|
// title: "Debug Ledger",
|
|
|
|
// leading: imageLedger,
|
|
|
|
// ),
|
2024-05-04 20:44:50 -05:00
|
|
|
if (!bleIsEnabled)
|
|
|
|
Padding(
|
|
|
|
padding: EdgeInsets.only(left: 20, right: 20, bottom: 20),
|
|
|
|
child: Text(
|
|
|
|
S.of(context).ledger_please_enable_bluetooth,
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 16,
|
|
|
|
fontWeight: FontWeight.w500,
|
|
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor),
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
if (bleDevices.length > 0) ...[
|
|
|
|
Padding(
|
|
|
|
padding: EdgeInsets.only(left: 20, right: 20, bottom: 20),
|
|
|
|
child: Container(
|
|
|
|
width: double.infinity,
|
|
|
|
child: Text(
|
|
|
|
S.of(context).bluetooth,
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 14,
|
|
|
|
fontWeight: FontWeight.w400,
|
|
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
...bleDevices
|
|
|
|
.map(
|
|
|
|
(device) => Padding(
|
|
|
|
padding: EdgeInsets.only(bottom: 20),
|
|
|
|
child: DeviceTile(
|
|
|
|
onPressed: () => _connectToDevice(device),
|
|
|
|
title: device.name,
|
|
|
|
leading: imageLedger,
|
|
|
|
connectionType: device.connectionType,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.toList()
|
|
|
|
],
|
|
|
|
if (usbDevices.length > 0) ...[
|
|
|
|
Padding(
|
|
|
|
padding: EdgeInsets.only(left: 20, right: 20, bottom: 20),
|
|
|
|
child: Container(
|
|
|
|
width: double.infinity,
|
|
|
|
child: Text(
|
|
|
|
S.of(context).usb,
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 14,
|
|
|
|
fontWeight: FontWeight.w400,
|
|
|
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
...usbDevices
|
|
|
|
.map(
|
|
|
|
(device) => Padding(
|
|
|
|
padding: EdgeInsets.only(bottom: 20),
|
|
|
|
child: DeviceTile(
|
|
|
|
onPressed: () => _connectToDevice(device),
|
|
|
|
title: device.name,
|
|
|
|
leading: imageLedger,
|
|
|
|
connectionType: device.connectionType,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.toList(),
|
|
|
|
]
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|