From e5f0a50011559d5a3e5ef76f0741b938b4a548fd Mon Sep 17 00:00:00 2001 From: sneurlax Date: Fri, 28 Jun 2024 21:18:18 -0500 Subject: [PATCH 01/19] add kNanoSwapRpcApiKey to windows prebuild --- scripts/prebuild.ps1 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/scripts/prebuild.ps1 b/scripts/prebuild.ps1 index 80a6991b7..aa5492834 100644 --- a/scripts/prebuild.ps1 +++ b/scripts/prebuild.ps1 @@ -2,7 +2,7 @@ $KEYS = "..\lib\external_api_keys.dart" if (-not (Test-Path $KEYS)) { Write-Host "prebuild.ps1: creating template lib/external_api_keys.dart file" - "const kChangeNowApiKey = '';" + "`nconst kSimpleSwapApiKey = '';" | Out-File $KEYS -Encoding UTF8 + "const kChangeNowApiKey = '';" + "`nconst kSimpleSwapApiKey = '';" + "const kNanoSwapRpcApiKey = '';" | Out-File $KEYS -Encoding UTF8 } # Create template wallet test parameter files if they don't already exist From d015632c151e239cc4326c5268cb2e1677c07537 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Sat, 29 Jun 2024 19:04:52 -0500 Subject: [PATCH 02/19] add kNanoSwapRpcApiKey to general prebuild and a wild `n appeard on windows --- scripts/prebuild.ps1 | 2 +- scripts/prebuild.sh | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/scripts/prebuild.ps1 b/scripts/prebuild.ps1 index aa5492834..2e98aaf14 100644 --- a/scripts/prebuild.ps1 +++ b/scripts/prebuild.ps1 @@ -2,7 +2,7 @@ $KEYS = "..\lib\external_api_keys.dart" if (-not (Test-Path $KEYS)) { Write-Host "prebuild.ps1: creating template lib/external_api_keys.dart file" - "const kChangeNowApiKey = '';" + "`nconst kSimpleSwapApiKey = '';" + "const kNanoSwapRpcApiKey = '';" | Out-File $KEYS -Encoding UTF8 + "const kChangeNowApiKey = '';" + "`nconst kSimpleSwapApiKey = '';" + "`nconst kNanoSwapRpcApiKey = '';" | Out-File $KEYS -Encoding UTF8 } # Create template wallet test parameter files if they don't already exist diff --git a/scripts/prebuild.sh b/scripts/prebuild.sh index 77d65b253..1337651d2 100755 --- a/scripts/prebuild.sh +++ b/scripts/prebuild.sh @@ -4,7 +4,7 @@ KEYS=../lib/external_api_keys.dart if ! test -f "$KEYS"; then echo 'prebuild.sh: creating template lib/external_api_keys.dart file' - printf 'const kChangeNowApiKey = "";\nconst kSimpleSwapApiKey = "";\n' > $KEYS + printf 'const kChangeNowApiKey = "";\nconst kSimpleSwapApiKey = "";\nconst kNanoSwapRpcApiKey = "";\n' > $KEYS fi # Create template wallet test parameter files if they don't already exist From 5eb65ba13e80dd08779c8f5c6b1c5002c0ced5f3 Mon Sep 17 00:00:00 2001 From: julian Date: Thu, 4 Jul 2024 14:54:20 -0600 Subject: [PATCH 03/19] desktop receive copy button fix --- .../my_stack_view/wallet_view/sub_widgets/desktop_receive.dart | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/desktop_receive.dart b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/desktop_receive.dart index 50c76ab7e..09de81b26 100644 --- a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/desktop_receive.dart +++ b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/desktop_receive.dart @@ -338,7 +338,7 @@ class _DesktopReceiveState extends ConsumerState { onTap: () { clipboard.setData( ClipboardData( - text: ref.watch(pWalletReceivingAddress(walletId)), + text: address, ), ); showFloatingFlushBar( From e4c13a9cf24604c049f5adcff009ead582216adb Mon Sep 17 00:00:00 2001 From: sneurlax Date: Thu, 4 Jul 2024 16:00:09 -0500 Subject: [PATCH 04/19] add kNanswapApiKey --- scripts/prebuild.ps1 | 2 +- scripts/prebuild.sh | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/scripts/prebuild.ps1 b/scripts/prebuild.ps1 index 2e98aaf14..04b68bc35 100644 --- a/scripts/prebuild.ps1 +++ b/scripts/prebuild.ps1 @@ -2,7 +2,7 @@ $KEYS = "..\lib\external_api_keys.dart" if (-not (Test-Path $KEYS)) { Write-Host "prebuild.ps1: creating template lib/external_api_keys.dart file" - "const kChangeNowApiKey = '';" + "`nconst kSimpleSwapApiKey = '';" + "`nconst kNanoSwapRpcApiKey = '';" | Out-File $KEYS -Encoding UTF8 + "const kChangeNowApiKey = '';" + "`nconst kSimpleSwapApiKey = '';" + "`nconst kNanswapApiKey = '';" + "`nconst kNanoSwapRpcApiKey = '';" | Out-File $KEYS -Encoding UTF8 } # Create template wallet test parameter files if they don't already exist diff --git a/scripts/prebuild.sh b/scripts/prebuild.sh index 1337651d2..6c50fbefd 100755 --- a/scripts/prebuild.sh +++ b/scripts/prebuild.sh @@ -4,7 +4,7 @@ KEYS=../lib/external_api_keys.dart if ! test -f "$KEYS"; then echo 'prebuild.sh: creating template lib/external_api_keys.dart file' - printf 'const kChangeNowApiKey = "";\nconst kSimpleSwapApiKey = "";\nconst kNanoSwapRpcApiKey = "";\n' > $KEYS + printf 'const kChangeNowApiKey = "";\nconst kSimpleSwapApiKey = "";\nconst kNanswapApiKey = "";\nconst kNanoSwapRpcApiKey = "";\n' > $KEYS fi # Create template wallet test parameter files if they don't already exist From eebe1df050176ec706f6f683cba8de1bee5a5bfd Mon Sep 17 00:00:00 2001 From: julian Date: Thu, 4 Jul 2024 15:17:05 -0600 Subject: [PATCH 05/19] xmr/wow keys --- lib/models/keys/cw_key_data.dart | 21 ++ lib/models/keys/key_data_interface.dart | 3 + lib/models/keys/xpriv_data.dart | 17 ++ .../wallet_backup_views/cn_wallet_keys.dart | 199 ++++++++++++++++++ .../wallet_backup_view.dart | 58 +++-- .../wallet_backup_views/wallet_xprivs.dart | 4 +- .../wallet_settings_view.dart | 13 +- .../firo_rescan_recovery_error_dialog.dart | 10 +- .../unlock_wallet_keys_desktop.dart | 23 +- .../wallet_keys_desktop_popup.dart | 31 ++- lib/route_generator.dart | 25 +-- lib/wallets/wallet/impl/monero_wallet.dart | 20 ++ lib/wallets/wallet/impl/wownero_wallet.dart | 20 ++ .../cw_based_interface.dart | 3 + .../extended_keys_interface.dart | 6 +- 15 files changed, 392 insertions(+), 61 deletions(-) create mode 100644 lib/models/keys/cw_key_data.dart create mode 100644 lib/models/keys/key_data_interface.dart create mode 100644 lib/models/keys/xpriv_data.dart create mode 100644 lib/pages/settings_views/wallet_settings_view/wallet_backup_views/cn_wallet_keys.dart diff --git a/lib/models/keys/cw_key_data.dart b/lib/models/keys/cw_key_data.dart new file mode 100644 index 000000000..c20c7938c --- /dev/null +++ b/lib/models/keys/cw_key_data.dart @@ -0,0 +1,21 @@ +import 'key_data_interface.dart'; + +class CWKeyData with KeyDataInterface { + CWKeyData({ + required this.walletId, + required String? privateSpendKey, + required String? privateViewKey, + required String? publicSpendKey, + required String? publicViewKey, + }) : keys = List.unmodifiable([ + (label: "Public View Key", key: publicViewKey), + (label: "Private View Key", key: privateViewKey), + (label: "Public Spend Key", key: publicSpendKey), + (label: "Private Spend Key", key: privateSpendKey), + ]); + + @override + final String walletId; + + final List<({String label, String key})> keys; +} diff --git a/lib/models/keys/key_data_interface.dart b/lib/models/keys/key_data_interface.dart new file mode 100644 index 000000000..1b4572f5c --- /dev/null +++ b/lib/models/keys/key_data_interface.dart @@ -0,0 +1,3 @@ +mixin KeyDataInterface { + String get walletId; +} diff --git a/lib/models/keys/xpriv_data.dart b/lib/models/keys/xpriv_data.dart new file mode 100644 index 000000000..cf0959b5d --- /dev/null +++ b/lib/models/keys/xpriv_data.dart @@ -0,0 +1,17 @@ +import '../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; +import 'key_data_interface.dart'; + +class XPrivData with KeyDataInterface { + XPrivData({ + required this.walletId, + required this.fingerprint, + required List xprivs, + }) : xprivs = List.unmodifiable(xprivs); + + @override + final String walletId; + + final String fingerprint; + + final List xprivs; +} diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/cn_wallet_keys.dart b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/cn_wallet_keys.dart new file mode 100644 index 000000000..14cae2ee0 --- /dev/null +++ b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/cn_wallet_keys.dart @@ -0,0 +1,199 @@ +import 'dart:async'; + +import 'package:dropdown_button2/dropdown_button2.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_svg/flutter_svg.dart'; + +import '../../../../models/keys/cw_key_data.dart'; +import '../../../../notifications/show_flush_bar.dart'; +import '../../../../themes/stack_colors.dart'; +import '../../../../utilities/assets.dart'; +import '../../../../utilities/clipboard_interface.dart'; +import '../../../../utilities/constants.dart'; +import '../../../../utilities/text_styles.dart'; +import '../../../../utilities/util.dart'; +import '../../../../widgets/desktop/primary_button.dart'; +import '../../../../widgets/detail_item.dart'; +import '../../../../widgets/qr.dart'; +import '../../../../widgets/rounded_white_container.dart'; + +class CNWalletKeys extends StatefulWidget { + const CNWalletKeys({ + super.key, + required this.cwKeyData, + required this.walletId, + this.clipboardInterface = const ClipboardWrapper(), + }); + + final CWKeyData cwKeyData; + final String walletId; + final ClipboardInterface clipboardInterface; + + @override + State createState() => _CNWalletKeysState(); +} + +class _CNWalletKeysState extends State { + late String _currentDropDownValue; + + String _current(String key) => + widget.cwKeyData.keys.firstWhere((e) => e.label == key).key; + + Future _copy() async { + await widget.clipboardInterface.setData( + ClipboardData(text: _current(_currentDropDownValue)), + ); + if (mounted) { + unawaited( + showFloatingFlushBar( + type: FlushBarType.info, + message: "Copied to clipboard", + iconAsset: Assets.svg.copy, + context: context, + ), + ); + } + } + + @override + void initState() { + _currentDropDownValue = widget.cwKeyData.keys.first.label; + super.initState(); + } + + @override + Widget build(BuildContext context) { + return Padding( + padding: Util.isDesktop + ? const EdgeInsets.symmetric(horizontal: 20) + : EdgeInsets.zero, + child: Column( + mainAxisSize: Util.isDesktop ? MainAxisSize.min : MainAxisSize.max, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + SizedBox( + height: Util.isDesktop ? 12 : 16, + ), + DetailItemBase( + horizontal: true, + borderColor: Util.isDesktop + ? Theme.of(context).extension()!.textFieldDefaultBG + : null, + title: Text( + "Selected key", + style: STextStyles.itemSubtitle(context), + ), + detail: SizedBox( + width: Util.isDesktop ? 200 : 170, + child: DropdownButtonHideUnderline( + child: DropdownButton2( + value: _currentDropDownValue, + items: [ + ...widget.cwKeyData.keys.map( + (e) => DropdownMenuItem( + value: e.label, + child: Text( + e.label, + style: STextStyles.w500_14(context), + ), + ), + ), + ], + onChanged: (value) { + if (value is String) { + setState(() { + _currentDropDownValue = value; + }); + } + }, + isExpanded: true, + buttonStyleData: ButtonStyleData( + decoration: BoxDecoration( + color: Theme.of(context) + .extension()! + .textFieldDefaultBG, + borderRadius: BorderRadius.circular( + Constants.size.circularBorderRadius, + ), + ), + ), + iconStyleData: IconStyleData( + icon: Padding( + padding: const EdgeInsets.only(right: 10), + child: SvgPicture.asset( + Assets.svg.chevronDown, + width: 12, + height: 6, + color: Theme.of(context) + .extension()! + .textFieldActiveSearchIconRight, + ), + ), + ), + dropdownStyleData: DropdownStyleData( + offset: const Offset(0, -10), + elevation: 0, + decoration: BoxDecoration( + color: Theme.of(context) + .extension()! + .textFieldDefaultBG, + borderRadius: BorderRadius.circular( + Constants.size.circularBorderRadius, + ), + ), + ), + menuItemStyleData: const MenuItemStyleData( + padding: EdgeInsets.symmetric( + horizontal: 16, + vertical: 8, + ), + ), + ), + ), + ), + ), + SizedBox( + height: Util.isDesktop ? 12 : 16, + ), + QR( + data: _current(_currentDropDownValue), + size: + Util.isDesktop ? 256 : MediaQuery.of(context).size.width / 1.5, + ), + SizedBox( + height: Util.isDesktop ? 12 : 16, + ), + RoundedWhiteContainer( + borderColor: Util.isDesktop + ? Theme.of(context).extension()!.textFieldDefaultBG + : null, + child: SelectableText( + _current(_currentDropDownValue), + style: STextStyles.w500_14(context), + ), + ), + SizedBox( + height: Util.isDesktop ? 12 : 16, + ), + if (!Util.isDesktop) const Spacer(), + Row( + children: [ + if (Util.isDesktop) const Spacer(), + if (Util.isDesktop) + const SizedBox( + width: 16, + ), + Expanded( + child: PrimaryButton( + label: "Copy", + onPressed: _copy, + ), + ), + ], + ), + ], + ), + ); + } +} diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart index 9c1488fcc..a673e5a7d 100644 --- a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart +++ b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart @@ -16,6 +16,9 @@ import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/flutter_svg.dart'; import '../../../../app_config.dart'; +import '../../../../models/keys/cw_key_data.dart'; +import '../../../../models/keys/key_data_interface.dart'; +import '../../../../models/keys/xpriv_data.dart'; import '../../../../notifications/show_flush_bar.dart'; import '../../../../themes/stack_colors.dart'; import '../../../../utilities/address_utils.dart'; @@ -25,7 +28,6 @@ import '../../../../utilities/constants.dart'; import '../../../../utilities/text_styles.dart'; import '../../../../utilities/util.dart'; import '../../../../wallets/isar/providers/wallet_info_provider.dart'; -import '../../../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; import '../../../../widgets/background.dart'; import '../../../../widgets/custom_buttons/app_bar_icon_button.dart'; import '../../../../widgets/custom_buttons/blue_text_button.dart'; @@ -36,6 +38,7 @@ import '../../../../widgets/rounded_white_container.dart'; import '../../../../widgets/stack_dialog.dart'; import '../../../add_wallet_views/new_wallet_recovery_phrase_view/sub_widgets/mnemonic_table.dart'; import '../../../wallet_view/transaction_views/transaction_details_view.dart'; +import 'cn_wallet_keys.dart'; import 'wallet_xprivs.dart'; class WalletBackupView extends ConsumerWidget { @@ -45,7 +48,7 @@ class WalletBackupView extends ConsumerWidget { required this.mnemonic, this.frostWalletData, this.clipboardInterface = const ClipboardWrapper(), - this.xprivData, + this.keyData, }); static const String routeName = "/walletBackup"; @@ -59,7 +62,7 @@ class WalletBackupView extends ConsumerWidget { ({String config, String keys})? prevGen, })? frostWalletData; final ClipboardInterface clipboardInterface; - final ({List xprivs, String fingerprint})? xprivData; + final KeyDataInterface? keyData; @override Widget build(BuildContext context, WidgetRef ref) { @@ -81,24 +84,30 @@ class WalletBackupView extends ConsumerWidget { style: STextStyles.navBarTitle(context), ), actions: [ - if (xprivData != null) + if (keyData != null) Padding( padding: const EdgeInsets.all(10), child: CustomTextButton( - text: "xpriv(s)", + text: switch (keyData.runtimeType) { + XPrivData() => "xpriv(s)", + CWKeyData() => "keys", + Type() => throw UnimplementedError( + "Don't forget to add your KeyDataInterface here!", + ), + }, onTap: () { Navigator.pushNamed( context, - MobileXPrivsView.routeName, + MobileKeyDataView.routeName, arguments: ( walletId: walletId, - xprivData: xprivData!, + keyData: keyData!, ), ); }, ), ), - if (!frost && xprivData == null) + if (!frost && keyData == null) Padding( padding: const EdgeInsets.all(10), child: AspectRatio( @@ -433,19 +442,19 @@ class _FrostKeys extends StatelessWidget { } } -class MobileXPrivsView extends StatelessWidget { - const MobileXPrivsView({ +class MobileKeyDataView extends StatelessWidget { + const MobileKeyDataView({ super.key, required this.walletId, this.clipboardInterface = const ClipboardWrapper(), - required this.xprivData, + required this.keyData, }); static const String routeName = "/mobileXPrivView"; final String walletId; final ClipboardInterface clipboardInterface; - final ({List xprivs, String fingerprint}) xprivData; + final KeyDataInterface keyData; @override Widget build(BuildContext context) { @@ -459,7 +468,13 @@ class MobileXPrivsView extends StatelessWidget { }, ), title: Text( - "Wallet xpriv(s)", + "Wallet ${switch (keyData.runtimeType) { + XPrivData() => "xpriv(s)", + CWKeyData() => "keys", + Type() => throw UnimplementedError( + "Don't forget to add your KeyDataInterface here!", + ), + }}", style: STextStyles.navBarTitle(context), ), ), @@ -475,10 +490,19 @@ class MobileXPrivsView extends StatelessWidget { mainAxisSize: MainAxisSize.min, children: [ Expanded( - child: WalletXPrivs( - walletId: walletId, - xprivData: xprivData, - ), + child: switch (keyData.runtimeType) { + XPrivData() => WalletXPrivs( + walletId: walletId, + xprivData: keyData as XPrivData, + ), + CWKeyData() => CNWalletKeys( + walletId: walletId, + cwKeyData: keyData as CWKeyData, + ), + Type() => throw UnimplementedError( + "Don't forget to add your KeyDataInterface here!", + ), + }, ), const SizedBox( height: 16, diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_xprivs.dart b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_xprivs.dart index 133dd95a1..594db33a7 100644 --- a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_xprivs.dart +++ b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_xprivs.dart @@ -16,6 +16,7 @@ import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/svg.dart'; +import '../../../../models/keys/xpriv_data.dart'; import '../../../../notifications/show_flush_bar.dart'; import '../../../../themes/stack_colors.dart'; import '../../../../utilities/assets.dart'; @@ -23,7 +24,6 @@ import '../../../../utilities/clipboard_interface.dart'; import '../../../../utilities/constants.dart'; import '../../../../utilities/text_styles.dart'; import '../../../../utilities/util.dart'; -import '../../../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; import '../../../../widgets/desktop/primary_button.dart'; import '../../../../widgets/detail_item.dart'; import '../../../../widgets/qr.dart'; @@ -37,7 +37,7 @@ class WalletXPrivs extends ConsumerStatefulWidget { this.clipboardInterface = const ClipboardWrapper(), }); - final ({List xprivs, String fingerprint}) xprivData; + final XPrivData xprivData; final String walletId; final ClipboardInterface clipboardInterface; diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_settings_view.dart b/lib/pages/settings_views/wallet_settings_view/wallet_settings_view.dart index a9fdd52f0..6985bb00c 100644 --- a/lib/pages/settings_views/wallet_settings_view/wallet_settings_view.dart +++ b/lib/pages/settings_views/wallet_settings_view/wallet_settings_view.dart @@ -18,6 +18,7 @@ import 'package:tuple/tuple.dart'; import '../../../db/hive/db.dart'; import '../../../db/sqlite/firo_cache.dart'; import '../../../models/epicbox_config_model.dart'; +import '../../../models/keys/key_data_interface.dart'; import '../../../notifications/show_flush_bar.dart'; import '../../../providers/global/wallets_provider.dart'; import '../../../providers/ui/transaction_filter_provider.dart'; @@ -35,6 +36,7 @@ import '../../../wallets/crypto_currency/intermediate/frost_currency.dart'; import '../../../wallets/crypto_currency/intermediate/nano_currency.dart'; import '../../../wallets/wallet/impl/bitcoin_frost_wallet.dart'; import '../../../wallets/wallet/impl/epiccash_wallet.dart'; +import '../../../wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart'; import '../../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; import '../../../wallets/wallet/wallet_mixin_interfaces/mnemonic_interface.dart'; import '../../../widgets/background.dart'; @@ -261,10 +263,6 @@ class _WalletSettingsViewState extends ConsumerState { // TODO: [prio=med] take wallets that don't have a mnemonic into account - ({ - List xprivs, - String fingerprint - })? xprivData; List? mnemonic; ({ String myName, @@ -306,8 +304,11 @@ class _WalletSettingsViewState extends ConsumerState { await wallet.getMnemonicAsWords(); } + KeyDataInterface? keyData; if (wallet is ExtendedKeysInterface) { - xprivData = await wallet.getXPrivs(); + keyData = await wallet.getXPrivs(); + } else if (wallet is CwBasedInterface) { + keyData = await wallet.getKeys(); } if (context.mounted) { @@ -323,7 +324,7 @@ class _WalletSettingsViewState extends ConsumerState { mnemonic: mnemonic ?? [], frostWalletData: frostWalletData, - xprivData: xprivData, + keyData: keyData, ), showBackButton: true, routeOnSuccess: diff --git a/lib/pages/special/firo_rescan_recovery_error_dialog.dart b/lib/pages/special/firo_rescan_recovery_error_dialog.dart index 6409646c6..92d226d4b 100644 --- a/lib/pages/special/firo_rescan_recovery_error_dialog.dart +++ b/lib/pages/special/firo_rescan_recovery_error_dialog.dart @@ -2,6 +2,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/flutter_svg.dart'; +import '../../models/keys/key_data_interface.dart'; import '../../pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/desktop_delete_wallet_dialog.dart'; import '../../pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/unlock_wallet_keys_desktop.dart'; import '../../providers/global/wallets_provider.dart'; @@ -11,6 +12,7 @@ import '../../utilities/assets.dart'; import '../../utilities/text_styles.dart'; import '../../utilities/util.dart'; import '../../wallets/isar/providers/wallet_info_provider.dart'; +import '../../wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart'; import '../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; import '../../wallets/wallet/wallet_mixin_interfaces/mnemonic_interface.dart'; import '../../widgets/background.dart'; @@ -265,9 +267,11 @@ class _FiroRescanRecoveryErrorViewState if (wallet is MnemonicInterface) { final mnemonic = await wallet.getMnemonicAsWords(); - ({List xprivs, String fingerprint})? xprivData; + KeyDataInterface? keyData; if (wallet is ExtendedKeysInterface) { - xprivData = await wallet.getXPrivs(); + keyData = await wallet.getXPrivs(); + } else if (wallet is CwBasedInterface) { + keyData = await wallet.getKeys(); } if (context.mounted) { @@ -280,7 +284,7 @@ class _FiroRescanRecoveryErrorViewState routeOnSuccessArguments: ( walletId: widget.walletId, mnemonic: mnemonic, - xprivData: xprivData, + keyData: keyData, ), showBackButton: true, routeOnSuccess: WalletBackupView.routeName, diff --git a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/unlock_wallet_keys_desktop.dart b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/unlock_wallet_keys_desktop.dart index 6500b669e..e332c4543 100644 --- a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/unlock_wallet_keys_desktop.dart +++ b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/unlock_wallet_keys_desktop.dart @@ -14,6 +14,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/svg.dart'; +import '../../../../models/keys/key_data_interface.dart'; import '../../../../notifications/show_flush_bar.dart'; import '../../../../providers/desktop/storage_crypto_handler_provider.dart'; import '../../../../providers/providers.dart'; @@ -22,6 +23,7 @@ import '../../../../utilities/assets.dart'; import '../../../../utilities/constants.dart'; import '../../../../utilities/text_styles.dart'; import '../../../../wallets/wallet/impl/bitcoin_frost_wallet.dart'; +import '../../../../wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart'; import '../../../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; import '../../../../wallets/wallet/wallet_mixin_interfaces/mnemonic_interface.dart'; import '../../../../widgets/desktop/desktop_dialog.dart'; @@ -85,7 +87,6 @@ class _UnlockWalletKeysDesktopState final wallet = ref.read(pWallets).getWallet(widget.walletId); ({String keys, String config})? frostData; List? words; - ({List xprivs, String fingerprint})? xprivData; // TODO: [prio=low] handle wallets that don't have a mnemonic // All wallets currently are mnemonic based @@ -102,8 +103,11 @@ class _UnlockWalletKeysDesktopState words = await wallet.getMnemonicAsWords(); } + KeyDataInterface? keyData; if (wallet is ExtendedKeysInterface) { - xprivData = await wallet.getXPrivs(); + keyData = await wallet.getXPrivs(); + } else if (wallet is CwBasedInterface) { + keyData = await wallet.getKeys(); } if (mounted) { @@ -113,7 +117,7 @@ class _UnlockWalletKeysDesktopState mnemonic: words ?? [], walletId: widget.walletId, frostData: frostData, - xprivData: xprivData, + keyData: keyData, ), ); } @@ -327,10 +331,6 @@ class _UnlockWalletKeysDesktopState ({String keys, String config})? frostData; List? words; - ({ - List xprivs, - String fingerprint - })? xprivData; final wallet = ref.read(pWallets).getWallet(widget.walletId); @@ -350,11 +350,14 @@ class _UnlockWalletKeysDesktopState words = await wallet.getMnemonicAsWords(); } + KeyDataInterface? keyData; if (wallet is ExtendedKeysInterface) { - xprivData = await wallet.getXPrivs(); + keyData = await wallet.getXPrivs(); + } else if (wallet is CwBasedInterface) { + keyData = await wallet.getKeys(); } - if (mounted) { + if (context.mounted) { await Navigator.of(context) .pushReplacementNamed( WalletKeysDesktopPopup.routeName, @@ -362,7 +365,7 @@ class _UnlockWalletKeysDesktopState mnemonic: words ?? [], walletId: widget.walletId, frostData: frostData, - xprivData: xprivData, + keyData: keyData, ), ); } diff --git a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart index a3ec985da..75c851989 100644 --- a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart +++ b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart @@ -14,8 +14,12 @@ import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; +import '../../../../models/keys/cw_key_data.dart'; +import '../../../../models/keys/key_data_interface.dart'; +import '../../../../models/keys/xpriv_data.dart'; import '../../../../notifications/show_flush_bar.dart'; import '../../../../pages/add_wallet_views/new_wallet_recovery_phrase_view/sub_widgets/mnemonic_table.dart'; +import '../../../../pages/settings_views/wallet_settings_view/wallet_backup_views/cn_wallet_keys.dart'; import '../../../../pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_xprivs.dart'; import '../../../../pages/wallet_view/transaction_views/transaction_details_view.dart'; import '../../../../themes/stack_colors.dart'; @@ -23,7 +27,6 @@ import '../../../../utilities/address_utils.dart'; import '../../../../utilities/assets.dart'; import '../../../../utilities/clipboard_interface.dart'; import '../../../../utilities/text_styles.dart'; -import '../../../../wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart'; import '../../../../widgets/custom_tab_view.dart'; import '../../../../widgets/desktop/desktop_dialog.dart'; import '../../../../widgets/desktop/desktop_dialog_close_button.dart'; @@ -39,14 +42,14 @@ class WalletKeysDesktopPopup extends ConsumerWidget { required this.walletId, this.frostData, this.clipboardInterface = const ClipboardWrapper(), - this.xprivData, + this.keyData, }); final List words; final String walletId; final ({String keys, String config})? frostData; final ClipboardInterface clipboardInterface; - final ({List xprivs, String fingerprint})? xprivData; + final KeyDataInterface? keyData; static const String routeName = "walletKeysDesktopPopup"; @@ -176,9 +179,13 @@ class WalletKeysDesktopPopup extends ConsumerWidget { ), ], ) - : xprivData != null + : keyData != null ? CustomTabView( - titles: const ["Mnemonic", "XPriv(s)"], + titles: [ + "Mnemonic", + if (keyData is XPrivData) "XPriv(s)", + if (keyData is CWKeyData) "Keys", + ], children: [ Padding( padding: const EdgeInsets.only(top: 16), @@ -186,10 +193,16 @@ class WalletKeysDesktopPopup extends ConsumerWidget { words: words, ), ), - WalletXPrivs( - xprivData: xprivData!, - walletId: walletId, - ), + if (keyData is XPrivData) + WalletXPrivs( + xprivData: keyData as XPrivData, + walletId: walletId, + ), + if (keyData is CWKeyData) + CNWalletKeys( + cwKeyData: keyData as CWKeyData, + walletId: walletId, + ), ], ) : _Mnemonic( diff --git a/lib/route_generator.dart b/lib/route_generator.dart index fe7b36872..b449525b6 100644 --- a/lib/route_generator.dart +++ b/lib/route_generator.dart @@ -22,6 +22,7 @@ import 'models/isar/models/blockchain_data/v2/transaction_v2.dart'; import 'models/isar/models/contact_entry.dart'; import 'models/isar/models/isar_models.dart'; import 'models/isar/ordinal.dart'; +import 'models/keys/key_data_interface.dart'; import 'models/paynym/paynym_account_lite.dart'; import 'models/send_view_auto_fill_data.dart'; import 'pages/add_wallet_views/add_token_view/add_custom_token_view.dart'; @@ -1280,14 +1281,14 @@ class RouteGenerator { } else if (args is ({ String walletId, List mnemonic, - ({List xprivs, String fingerprint})? xprivData, + KeyDataInterface? keyData, })) { return getRoute( shouldUseMaterialRoute: useMaterialPageRoute, builder: (_) => WalletBackupView( walletId: args.walletId, mnemonic: args.mnemonic, - xprivData: args.xprivData, + keyData: args.keyData, ), settings: RouteSettings( name: settings.name, @@ -1296,7 +1297,7 @@ class RouteGenerator { } else if (args is ({ String walletId, List mnemonic, - ({List xprivs, String fingerprint})? xprivData, + KeyDataInterface? keyData, ({ String myName, String config, @@ -1310,7 +1311,7 @@ class RouteGenerator { walletId: args.walletId, mnemonic: args.mnemonic, frostWalletData: args.frostWalletData, - xprivData: args.xprivData, + keyData: args.keyData, ), settings: RouteSettings( name: settings.name, @@ -1319,16 +1320,16 @@ class RouteGenerator { } return _routeError("${settings.name} invalid args: ${args.toString()}"); - case MobileXPrivsView.routeName: + case MobileKeyDataView.routeName: if (args is ({ String walletId, - ({List xprivs, String fingerprint}) xprivData, + KeyDataInterface keyData, })) { return getRoute( shouldUseMaterialRoute: useMaterialPageRoute, - builder: (_) => MobileXPrivsView( + builder: (_) => MobileKeyDataView( walletId: args.walletId, - xprivData: args.xprivData, + keyData: args.keyData, ), settings: RouteSettings( name: settings.name, @@ -2369,14 +2370,14 @@ class RouteGenerator { List mnemonic, String walletId, ({String keys, String config})? frostData, - ({List xprivs, String fingerprint})? xprivData, + KeyDataInterface? keyData, })) { return FadePageRoute( WalletKeysDesktopPopup( words: args.mnemonic, walletId: args.walletId, frostData: args.frostData, - xprivData: args.xprivData, + keyData: args.keyData, ), RouteSettings( name: settings.name, @@ -2385,13 +2386,13 @@ class RouteGenerator { } else if (args is ({ List mnemonic, String walletId, - ({List xprivs, String fingerprint})? xprivData, + KeyDataInterface? keyData, })) { return FadePageRoute( WalletKeysDesktopPopup( words: args.mnemonic, walletId: args.walletId, - xprivData: args.xprivData, + keyData: args.keyData, ), RouteSettings( name: settings.name, diff --git a/lib/wallets/wallet/impl/monero_wallet.dart b/lib/wallets/wallet/impl/monero_wallet.dart index 9bcb4c3d7..9c4848b8c 100644 --- a/lib/wallets/wallet/impl/monero_wallet.dart +++ b/lib/wallets/wallet/impl/monero_wallet.dart @@ -25,6 +25,7 @@ import 'package:tuple/tuple.dart'; import '../../../db/hive/db.dart'; import '../../../models/isar/models/blockchain_data/address.dart'; import '../../../models/isar/models/blockchain_data/transaction.dart'; +import '../../../models/keys/cw_key_data.dart'; import '../../../services/event_bus/events/global/tor_connection_status_changed_event.dart'; import '../../../services/event_bus/events/global/tor_status_changed_event.dart'; import '../../../services/event_bus/global_event_bus.dart'; @@ -235,6 +236,25 @@ class MoneroWallet extends CryptonoteWallet with CwBasedInterface { return; } + @override + Future getKeys() async { + final base = (CwBasedInterface.cwWalletBase as MoneroWalletBase?); + + if (base == null || + base.walletInfo.name != walletId || + CwBasedInterface.exitMutex.isLocked) { + return null; + } + + return CWKeyData( + walletId: walletId, + publicViewKey: base.keys.publicViewKey, + privateViewKey: base.keys.privateViewKey, + publicSpendKey: base.keys.publicSpendKey, + privateSpendKey: base.keys.privateSpendKey, + ); + } + @override Future updateTransactions() async { final base = (CwBasedInterface.cwWalletBase as MoneroWalletBase?); diff --git a/lib/wallets/wallet/impl/wownero_wallet.dart b/lib/wallets/wallet/impl/wownero_wallet.dart index 69f42a57d..a3ef7d04b 100644 --- a/lib/wallets/wallet/impl/wownero_wallet.dart +++ b/lib/wallets/wallet/impl/wownero_wallet.dart @@ -28,6 +28,7 @@ import 'package:tuple/tuple.dart'; import '../../../db/hive/db.dart'; import '../../../models/isar/models/blockchain_data/address.dart'; import '../../../models/isar/models/blockchain_data/transaction.dart'; +import '../../../models/keys/cw_key_data.dart'; import '../../../services/event_bus/events/global/tor_connection_status_changed_event.dart'; import '../../../services/event_bus/events/global/tor_status_changed_event.dart'; import '../../../services/event_bus/global_event_bus.dart'; @@ -214,6 +215,25 @@ class WowneroWallet extends CryptonoteWallet with CwBasedInterface { return; } + @override + Future getKeys() async { + final base = (CwBasedInterface.cwWalletBase as WowneroWalletBase?); + + if (base == null || + base.walletInfo.name != walletId || + CwBasedInterface.exitMutex.isLocked) { + return null; + } + + return CWKeyData( + walletId: walletId, + publicViewKey: base.keys.publicViewKey, + privateViewKey: base.keys.privateViewKey, + publicSpendKey: base.keys.publicSpendKey, + privateSpendKey: base.keys.privateSpendKey, + ); + } + @override Future updateTransactions() async { final base = (CwBasedInterface.cwWalletBase as WowneroWalletBase?); diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart index 069571ead..de1f7a699 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart @@ -14,6 +14,7 @@ import 'package:mutex/mutex.dart'; import '../../../models/balance.dart'; import '../../../models/isar/models/blockchain_data/address.dart'; +import '../../../models/keys/cw_key_data.dart'; import '../../../models/paymint/fee_object_model.dart'; import '../../../services/event_bus/events/global/blocks_remaining_event.dart'; import '../../../services/event_bus/events/global/refresh_percent_changed_event.dart'; @@ -195,6 +196,8 @@ mixin CwBasedInterface on CryptonoteWallet Address addressFor({required int index, int account = 0}); + Future getKeys(); + // ============ Private ====================================================== Future _refreshTxDataHelper() async { if (_txRefreshLock) return; diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart index 2e2836964..1606b1295 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/extended_keys_interface.dart @@ -1,3 +1,4 @@ +import '../../../models/keys/xpriv_data.dart'; import '../../crypto_currency/interfaces/electrumx_currency_interface.dart'; import 'electrumx_interface.dart'; @@ -42,7 +43,7 @@ mixin ExtendedKeysInterface ); } - Future<({List xprivs, String fingerprint})> getXPrivs() async { + Future getXPrivs() async { final paths = cryptoCurrency.supportedDerivationPathTypes.map( (e) => ( path: e, @@ -71,7 +72,8 @@ mixin ExtendedKeysInterface ); }); - return ( + return XPrivData( + walletId: walletId, fingerprint: fingerprint, xprivs: [ ( From 810981dd40c454af4f6ea1b0c8ac737ab1ead0ea Mon Sep 17 00:00:00 2001 From: julian Date: Thu, 4 Jul 2024 15:27:38 -0600 Subject: [PATCH 06/19] fix mobile keys switch --- .../wallet_backup_view.dart | 20 +++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart index a673e5a7d..fbfd75e89 100644 --- a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart +++ b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart @@ -89,10 +89,10 @@ class WalletBackupView extends ConsumerWidget { padding: const EdgeInsets.all(10), child: CustomTextButton( text: switch (keyData.runtimeType) { - XPrivData() => "xpriv(s)", - CWKeyData() => "keys", - Type() => throw UnimplementedError( - "Don't forget to add your KeyDataInterface here!", + const (XPrivData) => "xpriv(s)", + const (CWKeyData) => "keys", + _ => throw UnimplementedError( + "Don't forget to add your KeyDataInterface here! ${keyData.runtimeType}", ), }, onTap: () { @@ -469,9 +469,9 @@ class MobileKeyDataView extends StatelessWidget { ), title: Text( "Wallet ${switch (keyData.runtimeType) { - XPrivData() => "xpriv(s)", - CWKeyData() => "keys", - Type() => throw UnimplementedError( + const (XPrivData) => "xpriv(s)", + const (CWKeyData) => "keys", + _ => throw UnimplementedError( "Don't forget to add your KeyDataInterface here!", ), }}", @@ -491,15 +491,15 @@ class MobileKeyDataView extends StatelessWidget { children: [ Expanded( child: switch (keyData.runtimeType) { - XPrivData() => WalletXPrivs( + const (XPrivData) => WalletXPrivs( walletId: walletId, xprivData: keyData as XPrivData, ), - CWKeyData() => CNWalletKeys( + const (CWKeyData) => CNWalletKeys( walletId: walletId, cwKeyData: keyData as CWKeyData, ), - Type() => throw UnimplementedError( + _ => throw UnimplementedError( "Don't forget to add your KeyDataInterface here!", ), }, From 3f5ebee2eeaff96aaaedf349eaf59521a132bae5 Mon Sep 17 00:00:00 2001 From: julian Date: Thu, 4 Jul 2024 15:44:45 -0600 Subject: [PATCH 07/19] probable fix for some transaction creation bug --- lib/wallets/models/tx_data.dart | 48 +++++++++++++++++++++++---------- 1 file changed, 34 insertions(+), 14 deletions(-) diff --git a/lib/wallets/models/tx_data.dart b/lib/wallets/models/tx_data.dart index 774d85e06..99a28a138 100644 --- a/lib/wallets/models/tx_data.dart +++ b/lib/wallets/models/tx_data.dart @@ -133,21 +133,41 @@ class TxData { .reduce((total, amount) => total += amount) : null; - Amount? get amountWithoutChange => - recipients != null && recipients!.isNotEmpty - ? recipients! - .where((e) => !e.isChange) - .map((e) => e.amount) - .reduce((total, amount) => total += amount) - : null; + Amount? get amountWithoutChange { + if (recipients != null && recipients!.isNotEmpty) { + if (recipients!.where((e) => !e.isChange).isEmpty) { + return Amount( + rawValue: BigInt.zero, + fractionDigits: recipients!.first.amount.fractionDigits, + ); + } else { + return recipients! + .where((e) => !e.isChange) + .map((e) => e.amount) + .reduce((total, amount) => total += amount); + } + } else { + return null; + } + } - Amount? get amountSparkWithoutChange => - sparkRecipients != null && sparkRecipients!.isNotEmpty - ? sparkRecipients! - .where((e) => !e.isChange) - .map((e) => e.amount) - .reduce((total, amount) => total += amount) - : null; + Amount? get amountSparkWithoutChange { + if (sparkRecipients != null && sparkRecipients!.isNotEmpty) { + if (sparkRecipients!.where((e) => !e.isChange).isEmpty) { + return Amount( + rawValue: BigInt.zero, + fractionDigits: sparkRecipients!.first.amount.fractionDigits, + ); + } else { + return sparkRecipients! + .where((e) => !e.isChange) + .map((e) => e.amount) + .reduce((total, amount) => total += amount); + } + } else { + return null; + } + } int? get estimatedSatsPerVByte => fee != null && vSize != null ? (fee!.raw ~/ BigInt.from(vSize!)).toInt() From 60690349eae18021234fdbb3f38c347955ac9d73 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Fri, 5 Jul 2024 11:47:59 -0500 Subject: [PATCH 08/19] add campfire svgs to campfire light theme bundle --- .../default_themes/campfire/light.zip | Bin 898738 -> 935424 bytes 1 file changed, 0 insertions(+), 0 deletions(-) diff --git a/asset_sources/default_themes/campfire/light.zip b/asset_sources/default_themes/campfire/light.zip index 5dd67bcb1410f71296acf8fc720de04326f15087..898d97dc3865e372230f4fbb97fa14c74e86ce1f 100644 GIT binary patch delta 75974 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�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

amHY2@5rnzC~`*#J!X& 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 JzB&v1^j|YrWwrnS From 9ac00e2ed0a1f8f8e7d0d0954ae7d495089ead65 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Tue, 2 Jul 2024 23:08:37 -0500 Subject: [PATCH 09/19] add reuseAddress pref --- lib/utilities/prefs.dart | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/lib/utilities/prefs.dart b/lib/utilities/prefs.dart index 78a461ec7..585bdc1fc 100644 --- a/lib/utilities/prefs.dart +++ b/lib/utilities/prefs.dart @@ -1131,4 +1131,30 @@ class Prefs extends ChangeNotifier { ) as bool? ?? false; } + + // Reuse addresses (ie., don't generate new addresses by default). + + bool _reuseAddress = false; + + bool get reuseAddress => _reuseAddress; + + set reuseAddress(bool reuseAddress) { + if (_reuseAddress != reuseAddress) { + DB.instance.put( + boxName: DB.boxNamePrefs, + key: "reuseAddress", + value: reuseAddress, + ); + _reuseAddress = reuseAddress; + notifyListeners(); + } + } + + Future _getReuseAddress() async { + return await DB.instance.get( + boxName: DB.boxNamePrefs, + key: "reuseAddress", + ) as bool? ?? + true; + } } From e88e0fb5b737e32c301aa3e3ad459d49b1d63224 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Tue, 2 Jul 2024 23:40:59 -0500 Subject: [PATCH 10/19] add reuseAddress pref to advanced settings menu now with warning dialog! but why do you not toggle off when cancel is clicked? --- .../advanced_settings_view.dart | 214 ++++++++++++++++++ .../advanced_settings/advanced_settings.dart | 210 +++++++++++++++++ 2 files changed, 424 insertions(+) diff --git a/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart b/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart index 3d5413798..99b724076 100644 --- a/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart +++ b/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart @@ -16,11 +16,17 @@ import '../../../../providers/global/prefs_provider.dart'; import '../../../../themes/stack_colors.dart'; import '../../../../utilities/constants.dart'; import '../../../../utilities/text_styles.dart'; +import '../../../../utilities/util.dart'; import '../../../../widgets/background.dart'; import '../../../../widgets/choose_coin_view.dart'; import '../../../../widgets/custom_buttons/app_bar_icon_button.dart'; import '../../../../widgets/custom_buttons/draggable_switch_button.dart'; +import '../../../../widgets/desktop/desktop_dialog.dart'; +import '../../../../widgets/desktop/desktop_dialog_close_button.dart'; +import '../../../../widgets/desktop/primary_button.dart'; +import '../../../../widgets/desktop/secondary_button.dart'; import '../../../../widgets/rounded_white_container.dart'; +import '../../../../widgets/stack_dialog.dart'; import '../../../stack_privacy_calls.dart'; import 'debug_view.dart'; import 'manage_coin_units/manage_coin_units_view.dart'; @@ -182,6 +188,214 @@ class AdvancedSettingsView extends StatelessWidget { }, ), ), + // reuseAddress pref. + const SizedBox( + height: 8, + ), + RoundedWhiteContainer( + child: Consumer( + builder: (_, ref, __) { + final reuseAddress = + ref.watch(prefsChangeNotifierProvider.select( + (value) => value.reuseAddress, + )); + return RawMaterialButton( + // splashColor: Theme.of(context).extension()!.highlight, + materialTapTargetSize: MaterialTapTargetSize.shrinkWrap, + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular( + Constants.size.circularBorderRadius, + ), + ), + onPressed: null, + child: Padding( + padding: const EdgeInsets.symmetric(vertical: 8), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + "Reuse address", + style: STextStyles.titleBold12(context), + textAlign: TextAlign.left, + ), + SizedBox( + height: 20, + width: 40, + child: DraggableSwitchButton( + isOn: reuseAddress, + onValueChanged: (newValue) { + if (newValue) { + showDialog( + context: context, + builder: (context) { + final isDesktop = Util.isDesktop; + return isDesktop + ? DesktopDialog( + maxWidth: 576, + child: Column( + mainAxisSize: + MainAxisSize.min, + children: [ + Row( + mainAxisAlignment: + MainAxisAlignment + .spaceBetween, + children: [ + Padding( + padding: + const EdgeInsets + .only( + left: 32), + child: Text( + "Warning!", + style: STextStyles + .desktopH3( + context), + ), + ), + const DesktopDialogCloseButton(), + ], + ), + Padding( + padding: + const EdgeInsets.only( + top: 8, + left: 32, + right: 32, + bottom: 32, + ), + child: Column( + mainAxisSize: + MainAxisSize.min, + children: [ + Text( + "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", + style: STextStyles + .desktopTextSmall( + context), + ), + const SizedBox( + height: 43, + ), + Row( + children: [ + Expanded( + child: + SecondaryButton( + buttonHeight: + ButtonHeight + .l, + onPressed: + () { + Navigator.of( + context) + .pop( + false); + }, + label: + "Cancel", + ), + ), + const SizedBox( + width: 16, + ), + Expanded( + child: + PrimaryButton( + buttonHeight: + ButtonHeight + .l, + onPressed: + () { + ref.read(prefsChangeNotifierProvider).reuseAddress = + newValue; + Navigator.of( + context) + .pop( + true); + }, + label: + "Continue", + ), + ), + ], + ), + ], + ), + ), + ], + ), + ) + : WillPopScope( + onWillPop: () async { + return true; + }, + child: StackDialog( + title: "Warning!", + message: + "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", + leftButton: TextButton( + style: Theme.of(context) + .extension< + StackColors>()! + .getSecondaryEnabledButtonStyle( + context), + child: Text( + "Cancel", + style: STextStyles + .itemSubtitle12( + context), + ), + onPressed: () { + Navigator.of(context) + .pop(false); + }, + ), + rightButton: TextButton( + style: Theme.of(context) + .extension< + StackColors>()! + .getPrimaryEnabledButtonStyle( + context), + child: Text( + "Continue", + style: STextStyles.button( + context), + ), + onPressed: () { + Navigator.of(context) + .pop(true); + }, + ), + ), + ); + }, + ).then((confirmed) { + if (confirmed == true) { + ref + .read(prefsChangeNotifierProvider) + .reuseAddress = true; + } else { + ref + .read(prefsChangeNotifierProvider) + .reuseAddress = false; + } + }); + } else { + ref + .read(prefsChangeNotifierProvider) + .reuseAddress = newValue; + } + }, + ), + ), + ], + ), + ), + ); + }, + ), + ), const SizedBox( height: 8, ), diff --git a/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart b/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart index 029c5e294..9fe5ee8d0 100644 --- a/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart +++ b/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart @@ -17,9 +17,14 @@ import '../../../../providers/global/prefs_provider.dart'; import '../../../../themes/stack_colors.dart'; import '../../../../utilities/assets.dart'; import '../../../../utilities/text_styles.dart'; +import '../../../../utilities/util.dart'; import '../../../../widgets/custom_buttons/draggable_switch_button.dart'; +import '../../../../widgets/desktop/desktop_dialog.dart'; +import '../../../../widgets/desktop/desktop_dialog_close_button.dart'; import '../../../../widgets/desktop/primary_button.dart'; +import '../../../../widgets/desktop/secondary_button.dart'; import '../../../../widgets/rounded_white_container.dart'; +import '../../../../widgets/stack_dialog.dart'; import 'debug_info_dialog.dart'; import 'desktop_manage_block_explorers_dialog.dart'; import 'stack_privacy_dialog.dart'; @@ -37,6 +42,11 @@ class _AdvancedSettings extends ConsumerState { @override Widget build(BuildContext context) { debugPrint("BUILD: $runtimeType"); + + final reuseAddress = ref.watch(prefsChangeNotifierProvider.select( + (value) => value.reuseAddress, + )); + return SingleChildScrollView( child: Column( children: [ @@ -161,6 +171,206 @@ class _AdvancedSettings extends ConsumerState { ], ), ), + // reuseAddress pref. + const Padding( + padding: EdgeInsets.all(10.0), + child: Divider( + thickness: 0.5, + ), + ), + Padding( + padding: const EdgeInsets.all(10), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + "Reuse addresses", + style: STextStyles.desktopTextExtraSmall(context) + .copyWith( + color: Theme.of(context) + .extension()! + .textDark, + ), + textAlign: TextAlign.left, + ), + SizedBox( + height: 20, + width: 40, + child: DraggableSwitchButton( + isOn: reuseAddress, + onValueChanged: (newValue) { + if (newValue) { + showDialog( + context: context, + builder: (context) { + final isDesktop = Util.isDesktop; + return isDesktop + ? DesktopDialog( + maxWidth: 576, + child: Column( + mainAxisSize: + MainAxisSize.min, + children: [ + Row( + mainAxisAlignment: + MainAxisAlignment + .spaceBetween, + children: [ + Padding( + padding: + const EdgeInsets + .only( + left: 32), + child: Text( + "Warning!", + style: STextStyles + .desktopH3( + context), + ), + ), + const DesktopDialogCloseButton(), + ], + ), + Padding( + padding: + const EdgeInsets.only( + top: 8, + left: 32, + right: 32, + bottom: 32, + ), + child: Column( + mainAxisSize: + MainAxisSize.min, + children: [ + Text( + "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", + style: STextStyles + .desktopTextSmall( + context), + ), + const SizedBox( + height: 43, + ), + Row( + children: [ + Expanded( + child: + SecondaryButton( + buttonHeight: + ButtonHeight + .l, + onPressed: + () { + Navigator.of( + context) + .pop(); + }, + label: + "Cancel", + ), + ), + const SizedBox( + width: 16, + ), + Expanded( + child: + PrimaryButton( + buttonHeight: + ButtonHeight + .l, + onPressed: + () { + ref.read(prefsChangeNotifierProvider).reuseAddress = + newValue; + Navigator.of( + context) + .pop(); + }, + label: + "Continue", + ), + ), + ], + ), + ], + ), + ), + ], + ), + ) + : WillPopScope( + onWillPop: () async { + return true; + }, + child: StackDialog( + title: "Warning!", + message: + "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", + leftButton: TextButton( + style: Theme.of(context) + .extension< + StackColors>()! + .getSecondaryEnabledButtonStyle( + context), + child: Text( + "Cancel", + style: STextStyles + .itemSubtitle12( + context), + ), + onPressed: () { + Navigator.of(context) + .pop(); + }, + ), + rightButton: TextButton( + style: Theme.of(context) + .extension< + StackColors>()! + .getPrimaryEnabledButtonStyle( + context), + child: Text( + "Continue", + style: STextStyles.button( + context), + ), + onPressed: () { + ref + .read( + prefsChangeNotifierProvider) + .reuseAddress = newValue; + Navigator.of(context) + .pop(); + }, + ), + ), + ); + }, + ).then((confirmed) { + if (confirmed == true) { + ref + .read(prefsChangeNotifierProvider) + .reuseAddress = true; + } else { + ref + .read(prefsChangeNotifierProvider) + .reuseAddress = false; + } + }); + } else { + ref + .read( + prefsChangeNotifierProvider, + ) + .reuseAddress = newValue; + } + }, + ), + ), + ], + ), + ), const Padding( padding: EdgeInsets.all(10.0), child: Divider( From 27536fe6426551adc42e7799b3d5bc541ddfd6d1 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Wed, 3 Jul 2024 10:16:26 -0500 Subject: [PATCH 11/19] override needsGenerate is reuseAddress pref set --- .../wallet/wallet_mixin_interfaces/electrumx_interface.dart | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart index 035ae649f..83752c3a7 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart @@ -1333,6 +1333,11 @@ mixin ElectrumXInterface needsGenerate = txCount > 0 || currentReceiving.derivationIndex < 0; } + // If the reuseAddress flag is set, don't generate a new address. + if (prefs.reuseAddress) { + return; + } + if (needsGenerate) { await generateNewReceivingAddress(); From 8fb49ef0292ead5503f395b26f32313855dddeb7 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Wed, 3 Jul 2024 14:50:52 -0500 Subject: [PATCH 12/19] move reuseAddress pref from global prefs to walletInfo --- .../advanced_settings_view.dart | 214 ------------------ .../more_features/more_features_dialog.dart | 177 +++++++++++++++ .../advanced_settings/advanced_settings.dart | 210 ----------------- lib/utilities/prefs.dart | 26 --- lib/wallets/isar/models/wallet_info.dart | 1 + .../electrumx_interface.dart | 15 +- 6 files changed, 190 insertions(+), 453 deletions(-) diff --git a/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart b/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart index 99b724076..3d5413798 100644 --- a/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart +++ b/lib/pages/settings_views/global_settings_view/advanced_views/advanced_settings_view.dart @@ -16,17 +16,11 @@ import '../../../../providers/global/prefs_provider.dart'; import '../../../../themes/stack_colors.dart'; import '../../../../utilities/constants.dart'; import '../../../../utilities/text_styles.dart'; -import '../../../../utilities/util.dart'; import '../../../../widgets/background.dart'; import '../../../../widgets/choose_coin_view.dart'; import '../../../../widgets/custom_buttons/app_bar_icon_button.dart'; import '../../../../widgets/custom_buttons/draggable_switch_button.dart'; -import '../../../../widgets/desktop/desktop_dialog.dart'; -import '../../../../widgets/desktop/desktop_dialog_close_button.dart'; -import '../../../../widgets/desktop/primary_button.dart'; -import '../../../../widgets/desktop/secondary_button.dart'; import '../../../../widgets/rounded_white_container.dart'; -import '../../../../widgets/stack_dialog.dart'; import '../../../stack_privacy_calls.dart'; import 'debug_view.dart'; import 'manage_coin_units/manage_coin_units_view.dart'; @@ -188,214 +182,6 @@ class AdvancedSettingsView extends StatelessWidget { }, ), ), - // reuseAddress pref. - const SizedBox( - height: 8, - ), - RoundedWhiteContainer( - child: Consumer( - builder: (_, ref, __) { - final reuseAddress = - ref.watch(prefsChangeNotifierProvider.select( - (value) => value.reuseAddress, - )); - return RawMaterialButton( - // splashColor: Theme.of(context).extension()!.highlight, - materialTapTargetSize: MaterialTapTargetSize.shrinkWrap, - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular( - Constants.size.circularBorderRadius, - ), - ), - onPressed: null, - child: Padding( - padding: const EdgeInsets.symmetric(vertical: 8), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text( - "Reuse address", - style: STextStyles.titleBold12(context), - textAlign: TextAlign.left, - ), - SizedBox( - height: 20, - width: 40, - child: DraggableSwitchButton( - isOn: reuseAddress, - onValueChanged: (newValue) { - if (newValue) { - showDialog( - context: context, - builder: (context) { - final isDesktop = Util.isDesktop; - return isDesktop - ? DesktopDialog( - maxWidth: 576, - child: Column( - mainAxisSize: - MainAxisSize.min, - children: [ - Row( - mainAxisAlignment: - MainAxisAlignment - .spaceBetween, - children: [ - Padding( - padding: - const EdgeInsets - .only( - left: 32), - child: Text( - "Warning!", - style: STextStyles - .desktopH3( - context), - ), - ), - const DesktopDialogCloseButton(), - ], - ), - Padding( - padding: - const EdgeInsets.only( - top: 8, - left: 32, - right: 32, - bottom: 32, - ), - child: Column( - mainAxisSize: - MainAxisSize.min, - children: [ - Text( - "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", - style: STextStyles - .desktopTextSmall( - context), - ), - const SizedBox( - height: 43, - ), - Row( - children: [ - Expanded( - child: - SecondaryButton( - buttonHeight: - ButtonHeight - .l, - onPressed: - () { - Navigator.of( - context) - .pop( - false); - }, - label: - "Cancel", - ), - ), - const SizedBox( - width: 16, - ), - Expanded( - child: - PrimaryButton( - buttonHeight: - ButtonHeight - .l, - onPressed: - () { - ref.read(prefsChangeNotifierProvider).reuseAddress = - newValue; - Navigator.of( - context) - .pop( - true); - }, - label: - "Continue", - ), - ), - ], - ), - ], - ), - ), - ], - ), - ) - : WillPopScope( - onWillPop: () async { - return true; - }, - child: StackDialog( - title: "Warning!", - message: - "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", - leftButton: TextButton( - style: Theme.of(context) - .extension< - StackColors>()! - .getSecondaryEnabledButtonStyle( - context), - child: Text( - "Cancel", - style: STextStyles - .itemSubtitle12( - context), - ), - onPressed: () { - Navigator.of(context) - .pop(false); - }, - ), - rightButton: TextButton( - style: Theme.of(context) - .extension< - StackColors>()! - .getPrimaryEnabledButtonStyle( - context), - child: Text( - "Continue", - style: STextStyles.button( - context), - ), - onPressed: () { - Navigator.of(context) - .pop(true); - }, - ), - ), - ); - }, - ).then((confirmed) { - if (confirmed == true) { - ref - .read(prefsChangeNotifierProvider) - .reuseAddress = true; - } else { - ref - .read(prefsChangeNotifierProvider) - .reuseAddress = false; - } - }); - } else { - ref - .read(prefsChangeNotifierProvider) - .reuseAddress = newValue; - } - }, - ), - ), - ], - ), - ), - ); - }, - ), - ), const SizedBox( height: 8, ), diff --git a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart index 76fa097a4..352b9e378 100644 --- a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart +++ b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart @@ -19,6 +19,7 @@ import '../../../../../providers/global/wallets_provider.dart'; import '../../../../../themes/stack_colors.dart'; import '../../../../../utilities/assets.dart'; import '../../../../../utilities/text_styles.dart'; +import '../../../../../utilities/util.dart'; import '../../../../../wallets/crypto_currency/crypto_currency.dart'; import '../../../../../wallets/isar/models/wallet_info.dart'; import '../../../../../wallets/isar/providers/wallet_info_provider.dart'; @@ -32,7 +33,10 @@ import '../../../../../wallets/wallet/wallet_mixin_interfaces/spark_interface.da import '../../../../../widgets/custom_buttons/draggable_switch_button.dart'; import '../../../../../widgets/desktop/desktop_dialog.dart'; import '../../../../../widgets/desktop/desktop_dialog_close_button.dart'; +import '../../../../../widgets/desktop/primary_button.dart'; +import '../../../../../widgets/desktop/secondary_button.dart'; import '../../../../../widgets/rounded_container.dart'; +import '../../../../../widgets/stack_dialog.dart'; class MoreFeaturesDialog extends ConsumerStatefulWidget { const MoreFeaturesDialog({ @@ -102,6 +106,147 @@ class _MoreFeaturesDialogState extends ConsumerState { } } + bool _switchReuseAddressToggledLock = false; // Mutex. + Future _switchReuseAddressToggled(bool newValue) async { + if (newValue) { + await showDialog( + context: context, + builder: (context) { + final isDesktop = Util.isDesktop; + return isDesktop + ? DesktopDialog( + maxWidth: 576, + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Padding( + padding: const EdgeInsets.only(left: 32), + child: Text( + "Warning!", + style: STextStyles.desktopH3(context), + ), + ), + const DesktopDialogCloseButton(), + ], + ), + Padding( + padding: const EdgeInsets.only( + top: 8, + left: 32, + right: 32, + bottom: 32, + ), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Text( + "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", + style: STextStyles.desktopTextSmall(context), + ), + const SizedBox( + height: 43, + ), + Row( + children: [ + Expanded( + child: SecondaryButton( + buttonHeight: ButtonHeight.l, + onPressed: () { + Navigator.of(context).pop(false); + }, + label: "Cancel", + ), + ), + const SizedBox( + width: 16, + ), + Expanded( + child: PrimaryButton( + buttonHeight: ButtonHeight.l, + onPressed: () { + Navigator.of(context).pop(true); + }, + label: "Continue", + ), + ), + ], + ), + ], + ), + ), + ], + ), + ) + : StackDialog( + title: "Warning!", + message: + "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", + leftButton: TextButton( + style: Theme.of(context) + .extension()! + .getSecondaryEnabledButtonStyle(context), + child: Text( + "Cancel", + style: STextStyles.itemSubtitle12(context), + ), + onPressed: () { + Navigator.of(context).pop(false); + }, + ), + rightButton: TextButton( + style: Theme.of(context) + .extension()! + .getPrimaryEnabledButtonStyle(context), + child: Text( + "Continue", + style: STextStyles.button(context), + ), + onPressed: () { + Navigator.of(context).pop(true); + }, + ), + ); + }, + ).then((confirmed) async { + if (_switchReuseAddressToggledLock) { + return; + } + _switchReuseAddressToggledLock = true; // Lock mutex. + + try { + if (confirmed == true) { + await ref.read(pWalletInfo(widget.walletId)).updateOtherData( + newEntries: { + WalletInfoKeys.reuseAddress: true, + }, + isar: ref.read(mainDBProvider).isar, + ); + } else { + await ref.read(pWalletInfo(widget.walletId)).updateOtherData( + newEntries: { + WalletInfoKeys.reuseAddress: false, + }, + isar: ref.read(mainDBProvider).isar, + ); + } + } finally { + // ensure _switchReuseAddressToggledLock is set to false no matter what. + _switchReuseAddressToggledLock = false; + } + }); + } else { + await ref.read(pWalletInfo(widget.walletId)).updateOtherData( + newEntries: { + WalletInfoKeys.reuseAddress: false, + }, + isar: ref.read(mainDBProvider).isar, + ); + } + } + @override Widget build(BuildContext context) { final wallet = ref.watch( @@ -253,6 +398,38 @@ class _MoreFeaturesDialogState extends ConsumerState { ], ), ), + // reuseAddress preference. + _MoreFeaturesItemBase( + child: Row( + children: [ + const SizedBox(width: 3), + SizedBox( + height: 20, + width: 40, + child: DraggableSwitchButton( + isOn: ref.watch( + pWalletInfo(widget.walletId) + .select((value) => value.otherData), + )[WalletInfoKeys.reuseAddress] as bool? ?? + false, + onValueChanged: _switchReuseAddressToggled, + ), + ), + const SizedBox( + width: 16, + ), + Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + "Reuse addresses by default", + style: STextStyles.w600_20(context), + ), + ], + ), + ], + ), + ), const SizedBox( height: 28, ), diff --git a/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart b/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart index 9fe5ee8d0..029c5e294 100644 --- a/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart +++ b/lib/pages_desktop_specific/settings/settings_menu/advanced_settings/advanced_settings.dart @@ -17,14 +17,9 @@ import '../../../../providers/global/prefs_provider.dart'; import '../../../../themes/stack_colors.dart'; import '../../../../utilities/assets.dart'; import '../../../../utilities/text_styles.dart'; -import '../../../../utilities/util.dart'; import '../../../../widgets/custom_buttons/draggable_switch_button.dart'; -import '../../../../widgets/desktop/desktop_dialog.dart'; -import '../../../../widgets/desktop/desktop_dialog_close_button.dart'; import '../../../../widgets/desktop/primary_button.dart'; -import '../../../../widgets/desktop/secondary_button.dart'; import '../../../../widgets/rounded_white_container.dart'; -import '../../../../widgets/stack_dialog.dart'; import 'debug_info_dialog.dart'; import 'desktop_manage_block_explorers_dialog.dart'; import 'stack_privacy_dialog.dart'; @@ -42,11 +37,6 @@ class _AdvancedSettings extends ConsumerState { @override Widget build(BuildContext context) { debugPrint("BUILD: $runtimeType"); - - final reuseAddress = ref.watch(prefsChangeNotifierProvider.select( - (value) => value.reuseAddress, - )); - return SingleChildScrollView( child: Column( children: [ @@ -171,206 +161,6 @@ class _AdvancedSettings extends ConsumerState { ], ), ), - // reuseAddress pref. - const Padding( - padding: EdgeInsets.all(10.0), - child: Divider( - thickness: 0.5, - ), - ), - Padding( - padding: const EdgeInsets.all(10), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text( - "Reuse addresses", - style: STextStyles.desktopTextExtraSmall(context) - .copyWith( - color: Theme.of(context) - .extension()! - .textDark, - ), - textAlign: TextAlign.left, - ), - SizedBox( - height: 20, - width: 40, - child: DraggableSwitchButton( - isOn: reuseAddress, - onValueChanged: (newValue) { - if (newValue) { - showDialog( - context: context, - builder: (context) { - final isDesktop = Util.isDesktop; - return isDesktop - ? DesktopDialog( - maxWidth: 576, - child: Column( - mainAxisSize: - MainAxisSize.min, - children: [ - Row( - mainAxisAlignment: - MainAxisAlignment - .spaceBetween, - children: [ - Padding( - padding: - const EdgeInsets - .only( - left: 32), - child: Text( - "Warning!", - style: STextStyles - .desktopH3( - context), - ), - ), - const DesktopDialogCloseButton(), - ], - ), - Padding( - padding: - const EdgeInsets.only( - top: 8, - left: 32, - right: 32, - bottom: 32, - ), - child: Column( - mainAxisSize: - MainAxisSize.min, - children: [ - Text( - "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", - style: STextStyles - .desktopTextSmall( - context), - ), - const SizedBox( - height: 43, - ), - Row( - children: [ - Expanded( - child: - SecondaryButton( - buttonHeight: - ButtonHeight - .l, - onPressed: - () { - Navigator.of( - context) - .pop(); - }, - label: - "Cancel", - ), - ), - const SizedBox( - width: 16, - ), - Expanded( - child: - PrimaryButton( - buttonHeight: - ButtonHeight - .l, - onPressed: - () { - ref.read(prefsChangeNotifierProvider).reuseAddress = - newValue; - Navigator.of( - context) - .pop(); - }, - label: - "Continue", - ), - ), - ], - ), - ], - ), - ), - ], - ), - ) - : WillPopScope( - onWillPop: () async { - return true; - }, - child: StackDialog( - title: "Warning!", - message: - "Reusing addresses reduces your privacy and security. Are you sure you want to reuse addresses by default?", - leftButton: TextButton( - style: Theme.of(context) - .extension< - StackColors>()! - .getSecondaryEnabledButtonStyle( - context), - child: Text( - "Cancel", - style: STextStyles - .itemSubtitle12( - context), - ), - onPressed: () { - Navigator.of(context) - .pop(); - }, - ), - rightButton: TextButton( - style: Theme.of(context) - .extension< - StackColors>()! - .getPrimaryEnabledButtonStyle( - context), - child: Text( - "Continue", - style: STextStyles.button( - context), - ), - onPressed: () { - ref - .read( - prefsChangeNotifierProvider) - .reuseAddress = newValue; - Navigator.of(context) - .pop(); - }, - ), - ), - ); - }, - ).then((confirmed) { - if (confirmed == true) { - ref - .read(prefsChangeNotifierProvider) - .reuseAddress = true; - } else { - ref - .read(prefsChangeNotifierProvider) - .reuseAddress = false; - } - }); - } else { - ref - .read( - prefsChangeNotifierProvider, - ) - .reuseAddress = newValue; - } - }, - ), - ), - ], - ), - ), const Padding( padding: EdgeInsets.all(10.0), child: Divider( diff --git a/lib/utilities/prefs.dart b/lib/utilities/prefs.dart index 585bdc1fc..78a461ec7 100644 --- a/lib/utilities/prefs.dart +++ b/lib/utilities/prefs.dart @@ -1131,30 +1131,4 @@ class Prefs extends ChangeNotifier { ) as bool? ?? false; } - - // Reuse addresses (ie., don't generate new addresses by default). - - bool _reuseAddress = false; - - bool get reuseAddress => _reuseAddress; - - set reuseAddress(bool reuseAddress) { - if (_reuseAddress != reuseAddress) { - DB.instance.put( - boxName: DB.boxNamePrefs, - key: "reuseAddress", - value: reuseAddress, - ); - _reuseAddress = reuseAddress; - notifyListeners(); - } - } - - Future _getReuseAddress() async { - return await DB.instance.get( - boxName: DB.boxNamePrefs, - key: "reuseAddress", - ) as bool? ?? - true; - } } diff --git a/lib/wallets/isar/models/wallet_info.dart b/lib/wallets/isar/models/wallet_info.dart index 6f6e12a42..6a7a9b54c 100644 --- a/lib/wallets/isar/models/wallet_info.dart +++ b/lib/wallets/isar/models/wallet_info.dart @@ -511,4 +511,5 @@ abstract class WalletInfoKeys { static const String firoSparkCacheSetTimestampCache = "firoSparkCacheSetTimestampCacheKey"; static const String enableOptInRbf = "enableOptInRbfKey"; + static const String reuseAddress = "reuseAddressKey"; } diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart index 83752c3a7..621209910 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart @@ -1,4 +1,5 @@ import 'dart:async'; +import 'dart:convert'; import 'dart:math'; import 'dart:typed_data'; @@ -23,6 +24,7 @@ import '../../../utilities/logger.dart'; import '../../../utilities/paynym_is_api.dart'; import '../../crypto_currency/coins/firo.dart'; import '../../crypto_currency/interfaces/electrumx_currency_interface.dart'; +import '../../isar/models/wallet_info.dart'; import '../../models/tx_data.dart'; import '../impl/bitcoin_wallet.dart'; import '../impl/firo_wallet.dart'; @@ -1333,9 +1335,16 @@ mixin ElectrumXInterface needsGenerate = txCount > 0 || currentReceiving.derivationIndex < 0; } - // If the reuseAddress flag is set, don't generate a new address. - if (prefs.reuseAddress) { - return; + // If reuseAddress is set, don't generate an address by default. + final WalletInfo info = mainDB.isar.walletInfo + .where() + .walletIdEqualTo(walletId) + .findFirstSync()!; + if (info.otherDataJsonString != null) { + final otherData = jsonDecode(info.otherDataJsonString!); + if (otherData[WalletInfoKeys.reuseAddress] as bool? ?? false) { + return; + } } if (needsGenerate) { From 4f4057236653a0eae5c6ee9c1dcf25a6ce6a62ab Mon Sep 17 00:00:00 2001 From: julian Date: Thu, 4 Jul 2024 10:34:45 -0600 Subject: [PATCH 13/19] small tweaks --- .../more_features/more_features_dialog.dart | 2 +- .../electrumx_interface.dart | 13 +++---------- 2 files changed, 4 insertions(+), 11 deletions(-) diff --git a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart index 352b9e378..a474fb7bb 100644 --- a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart +++ b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/more_features/more_features_dialog.dart @@ -422,7 +422,7 @@ class _MoreFeaturesDialogState extends ConsumerState { crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( - "Reuse addresses by default", + "Reuse receiving address by default", style: STextStyles.w600_20(context), ), ], diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart index 621209910..44bebf17a 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart @@ -1,5 +1,4 @@ import 'dart:async'; -import 'dart:convert'; import 'dart:math'; import 'dart:typed_data'; @@ -1335,16 +1334,10 @@ mixin ElectrumXInterface needsGenerate = txCount > 0 || currentReceiving.derivationIndex < 0; } + // TODO this in the wrong place // If reuseAddress is set, don't generate an address by default. - final WalletInfo info = mainDB.isar.walletInfo - .where() - .walletIdEqualTo(walletId) - .findFirstSync()!; - if (info.otherDataJsonString != null) { - final otherData = jsonDecode(info.otherDataJsonString!); - if (otherData[WalletInfoKeys.reuseAddress] as bool? ?? false) { - return; - } + if (info.otherData[WalletInfoKeys.reuseAddress] as bool? ?? false) { + return; } if (needsGenerate) { From bb7f0ff46f0f3e53b123e4cbf0d42f2525a07b98 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Thu, 4 Jul 2024 15:05:25 -0500 Subject: [PATCH 14/19] avoid checkReceivingAddressForTransactions altogether when reuse is set --- lib/wallets/wallet/impl/firo_wallet.dart | 4 ++- lib/wallets/wallet/wallet.dart | 6 +++-- .../cw_based_interface.dart | 22 +++++++++++++--- .../electrumx_interface.dart | 26 ++++++++++++------- 4 files changed, 43 insertions(+), 15 deletions(-) diff --git a/lib/wallets/wallet/impl/firo_wallet.dart b/lib/wallets/wallet/impl/firo_wallet.dart index 33ae3fb24..f9fcb11b5 100644 --- a/lib/wallets/wallet/impl/firo_wallet.dart +++ b/lib/wallets/wallet/impl/firo_wallet.dart @@ -787,7 +787,9 @@ class FiroWallet extends Bip39HDWallet for (final tuple in receiveResults) { if (tuple.addresses.isEmpty) { - await checkReceivingAddressForTransactions(); + if (info.otherData[WalletInfoKeys.reuseAddress] != true) { + await checkReceivingAddressForTransactions(); + } } else { highestReceivingIndexWithHistory = max( tuple.index, diff --git a/lib/wallets/wallet/wallet.dart b/lib/wallets/wallet/wallet.dart index 1bca3c858..46540eef7 100644 --- a/lib/wallets/wallet/wallet.dart +++ b/lib/wallets/wallet/wallet.dart @@ -522,8 +522,10 @@ abstract class Wallet { // TODO: [prio=low] handle this differently. Extra modification of this file for coin specific functionality should be avoided. if (this is MultiAddressInterface) { - await (this as MultiAddressInterface) - .checkReceivingAddressForTransactions(); + if (info.otherData[WalletInfoKeys.reuseAddress] != true) { + await (this as MultiAddressInterface) + .checkReceivingAddressForTransactions(); + } } GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.2, walletId)); diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart index de1f7a699..661b10bd3 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/cw_based_interface.dart @@ -25,6 +25,7 @@ import '../../../utilities/amount/amount.dart'; import '../../../utilities/logger.dart'; import '../../../utilities/stack_file_system.dart'; import '../../crypto_currency/intermediate/cryptonote_currency.dart'; +import '../../isar/models/wallet_info.dart'; import '../intermediate/cryptonote_wallet.dart'; import 'multi_address_interface.dart'; @@ -286,7 +287,9 @@ mixin CwBasedInterface on CryptonoteWallet await updateTransactions(); await updateBalance(); - await checkReceivingAddressForTransactions(); + if (info.otherData[WalletInfoKeys.reuseAddress] != true) { + await checkReceivingAddressForTransactions(); + } if (cwWalletBase?.syncStatus is SyncedSyncStatus) { refreshMutex.release(); @@ -342,6 +345,17 @@ mixin CwBasedInterface on CryptonoteWallet @override Future checkReceivingAddressForTransactions() async { + if (info.otherData[WalletInfoKeys.reuseAddress] == true) { + try { + throw Exception(); + } catch (_, s) { + Logging.instance.log( + "checkReceivingAddressForTransactions called but reuse address flag set: $s", + level: LogLevel.Error, + ); + } + } + try { int highestIndex = -1; final entries = cwWalletBase?.transactionHistory?.transactions?.entries; @@ -380,8 +394,10 @@ mixin CwBasedInterface on CryptonoteWallet // we need to update the address await mainDB.updateAddress(existing, newReceivingAddress); } - // keep checking until address with no tx history is set as current - await checkReceivingAddressForTransactions(); + if (info.otherData[WalletInfoKeys.reuseAddress] != true) { + // keep checking until address with no tx history is set as current + await checkReceivingAddressForTransactions(); + } } } on SocketException catch (se, s) { Logging.instance.log( diff --git a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart index 44bebf17a..c31f7cd4a 100644 --- a/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart +++ b/lib/wallets/wallet/wallet_mixin_interfaces/electrumx_interface.dart @@ -1316,6 +1316,17 @@ mixin ElectrumXInterface @override Future checkReceivingAddressForTransactions() async { + if (info.otherData[WalletInfoKeys.reuseAddress] == true) { + try { + throw Exception(); + } catch (_, s) { + Logging.instance.log( + "checkReceivingAddressForTransactions called but reuse address flag set: $s", + level: LogLevel.Error, + ); + } + } + try { final currentReceiving = await getCurrentReceivingAddress(); @@ -1334,18 +1345,15 @@ mixin ElectrumXInterface needsGenerate = txCount > 0 || currentReceiving.derivationIndex < 0; } - // TODO this in the wrong place - // If reuseAddress is set, don't generate an address by default. - if (info.otherData[WalletInfoKeys.reuseAddress] as bool? ?? false) { - return; - } - if (needsGenerate) { await generateNewReceivingAddress(); - // TODO: get rid of this? Could cause problems (long loading/infinite loop or something) - // keep checking until address with no tx history is set as current - await checkReceivingAddressForTransactions(); + // TODO: [prio=low] Make sure we scan all addresses but only show one. + if (info.otherData[WalletInfoKeys.reuseAddress] != true) { + // TODO: get rid of this? Could cause problems (long loading/infinite loop or something) + // keep checking until address with no tx history is set as current + await checkReceivingAddressForTransactions(); + } } } catch (e, s) { Logging.instance.log( From 23f07334a6329c927fcd03dcfcebe13e9f46e5ff Mon Sep 17 00:00:00 2001 From: julian Date: Fri, 5 Jul 2024 11:13:49 -0600 Subject: [PATCH 15/19] consistent copy and qr buttons in back screen across platforms --- .../wallet_backup_view.dart | 82 +++++++------------ .../wallet_keys_desktop_popup.dart | 32 ++++---- 2 files changed, 47 insertions(+), 67 deletions(-) diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart index fbfd75e89..0c91eda12 100644 --- a/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart +++ b/lib/pages/settings_views/wallet_settings_view/wallet_backup_views/wallet_backup_view.dart @@ -13,7 +13,6 @@ import 'dart:async'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:flutter_svg/flutter_svg.dart'; import '../../../../app_config.dart'; import '../../../../models/keys/cw_key_data.dart'; @@ -32,6 +31,8 @@ import '../../../../widgets/background.dart'; import '../../../../widgets/custom_buttons/app_bar_icon_button.dart'; import '../../../../widgets/custom_buttons/blue_text_button.dart'; import '../../../../widgets/custom_buttons/simple_copy_button.dart'; +import '../../../../widgets/desktop/primary_button.dart'; +import '../../../../widgets/desktop/secondary_button.dart'; import '../../../../widgets/detail_item.dart'; import '../../../../widgets/qr.dart'; import '../../../../widgets/rounded_white_container.dart'; @@ -47,7 +48,6 @@ class WalletBackupView extends ConsumerWidget { required this.walletId, required this.mnemonic, this.frostWalletData, - this.clipboardInterface = const ClipboardWrapper(), this.keyData, }); @@ -61,7 +61,6 @@ class WalletBackupView extends ConsumerWidget { String keys, ({String config, String keys})? prevGen, })? frostWalletData; - final ClipboardInterface clipboardInterface; final KeyDataInterface? keyData; @override @@ -107,40 +106,6 @@ class WalletBackupView extends ConsumerWidget { }, ), ), - if (!frost && keyData == null) - Padding( - padding: const EdgeInsets.all(10), - child: AspectRatio( - aspectRatio: 1, - child: AppBarIconButton( - color: - Theme.of(context).extension()!.background, - shadows: const [], - icon: SvgPicture.asset( - Assets.svg.copy, - width: 20, - height: 20, - color: Theme.of(context) - .extension()! - .topNavIconPrimary, - ), - onPressed: () async { - await clipboardInterface - .setData(ClipboardData(text: mnemonic.join(" "))); - if (context.mounted) { - unawaited( - showFloatingFlushBar( - type: FlushBarType.info, - message: "Copied to clipboard", - iconAsset: Assets.svg.copy, - context: context, - ), - ); - } - }, - ), - ), - ), ], ), body: Padding( @@ -161,19 +126,22 @@ class WalletBackupView extends ConsumerWidget { } class _Mnemonic extends ConsumerWidget { - const _Mnemonic({super.key, required this.walletId, required this.mnemonic}); + const _Mnemonic({ + super.key, + required this.walletId, + required this.mnemonic, + this.clipboardInterface = const ClipboardWrapper(), + }); final String walletId; final List mnemonic; + final ClipboardInterface clipboardInterface; @override Widget build(BuildContext context, WidgetRef ref) { return Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ - const SizedBox( - height: 4, - ), Text( ref.watch(pWalletName(walletId)), textAlign: TextAlign.center, @@ -221,10 +189,28 @@ class _Mnemonic extends ConsumerWidget { const SizedBox( height: 12, ), - TextButton( - style: Theme.of(context) - .extension()! - .getPrimaryEnabledButtonStyle(context), + SecondaryButton( + label: "Copy", + onPressed: () async { + await clipboardInterface + .setData(ClipboardData(text: mnemonic.join(" "))); + if (context.mounted) { + unawaited( + showFloatingFlushBar( + type: FlushBarType.info, + message: "Copied to clipboard", + iconAsset: Assets.svg.copy, + context: context, + ), + ); + } + }, + ), + const SizedBox( + height: 12, + ), + PrimaryButton( + label: "Show QR Code", onPressed: () { final String data = AddressUtils.encodeQRSeedData(mnemonic); @@ -293,10 +279,6 @@ class _Mnemonic extends ConsumerWidget { }, ); }, - child: Text( - "Show QR Code", - style: STextStyles.button(context), - ), ), ], ); @@ -310,8 +292,6 @@ class _FrostKeys extends StatelessWidget { this.frostWalletData, }); - static const String routeName = "/walletBackup"; - final String walletId; final ({ String myName, diff --git a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart index 75c851989..f1bc2d9f6 100644 --- a/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart +++ b/lib/pages_desktop_specific/my_stack_view/wallet_view/sub_widgets/wallet_keys_desktop_popup.dart @@ -277,22 +277,6 @@ class _Mnemonic extends StatelessWidget { children: [ Expanded( child: SecondaryButton( - label: "Show QR code", - onPressed: () { - // TODO: address utils - final String value = AddressUtils.encodeQRSeedData(words); - Navigator.of(context).pushNamed( - QRCodeDesktopPopupContent.routeName, - arguments: value, - ); - }, - ), - ), - const SizedBox( - width: 16, - ), - Expanded( - child: PrimaryButton( label: "Copy", onPressed: () async { await clipboardInterface.setData( @@ -311,6 +295,22 @@ class _Mnemonic extends StatelessWidget { }, ), ), + const SizedBox( + width: 16, + ), + Expanded( + child: PrimaryButton( + label: "Show QR code", + onPressed: () { + // TODO: address utils + final String value = AddressUtils.encodeQRSeedData(words); + Navigator.of(context).pushNamed( + QRCodeDesktopPopupContent.routeName, + arguments: value, + ); + }, + ), + ), ], ), ), From df233dc311ec3c7b9cd2d203def218231ca95b60 Mon Sep 17 00:00:00 2001 From: Julian Date: Fri, 5 Jul 2024 11:46:52 -0600 Subject: [PATCH 16/19] use latest coinlib (and fix sol dep) --- scripts/app_config/templates/pubspec.template | 23 +++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/scripts/app_config/templates/pubspec.template b/scripts/app_config/templates/pubspec.template index e7eb439f7..a784dbabd 100644 --- a/scripts/app_config/templates/pubspec.template +++ b/scripts/app_config/templates/pubspec.template @@ -171,7 +171,12 @@ dependencies: convert: ^3.1.1 flutter_hooks: ^0.20.3 meta: ^1.9.1 - coinlib_flutter: ^2.0.0 +# coinlib_flutter: ^2.1.0-rc.1 + coinlib_flutter: + git: + url: https://github.com/peercoin/coinlib.git + ref: b88e68e2e10ffe54d802deeed6b9e83da7721a1f + path: coinlib_flutter electrum_adapter: git: url: https://github.com/cypherstack/electrum_adapter.git @@ -180,7 +185,7 @@ dependencies: solana: git: # TODO [prio=low]: Revert to official package once Tor support is merged upstream. url: https://github.com/cypherstack/espresso-cash-public.git - ref: a83e375678eb22fe544dc125d29bbec0fb833882 + ref: 706be5f166d31736c443cca4cd311b7fcfc2a9bf path: packages/solana calendar_date_picker2: ^1.0.2 sqlite3: ^2.4.3 @@ -210,6 +215,20 @@ flutter_native_splash: android_disable_fullscreen: true dependency_overrides: + + # coin lib git for testing while waiting for publishing + coinlib: + git: + url: https://github.com/peercoin/coinlib.git + ref: b88e68e2e10ffe54d802deeed6b9e83da7721a1f + path: coinlib + + coinlib_flutter: + git: + url: https://github.com/peercoin/coinlib.git + ref: b88e68e2e10ffe54d802deeed6b9e83da7721a1f + path: coinlib_flutter + # adding here due to pure laziness tor_ffi_plugin: git: From ba8215a3d3bed2a2b4f46a0fa67d45720f1131e6 Mon Sep 17 00:00:00 2001 From: sneurlax Date: Wed, 3 Jul 2024 17:54:51 -0500 Subject: [PATCH 17/19] use wakelock on fusion progress view / dialog revert desktop changes --- lib/pages/cashfusion/fusion_progress_view.dart | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/lib/pages/cashfusion/fusion_progress_view.dart b/lib/pages/cashfusion/fusion_progress_view.dart index fd2921c7f..e4d787903 100644 --- a/lib/pages/cashfusion/fusion_progress_view.dart +++ b/lib/pages/cashfusion/fusion_progress_view.dart @@ -12,6 +12,7 @@ import 'dart:async'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:wakelock/wakelock.dart'; import '../../pages_desktop_specific/cashfusion/sub_widgets/fusion_progress.dart'; import '../../providers/cash_fusion/fusion_progress_ui_state_provider.dart'; @@ -84,6 +85,8 @@ class _FusionProgressViewState extends ConsumerState { message: "Stopping fusion", ); + await Wakelock.disable(); + return true; } else { return false; @@ -96,6 +99,12 @@ class _FusionProgressViewState extends ConsumerState { super.initState(); } + @override + void dispose() { + Wakelock.disable(); + super.dispose(); + } + @override Widget build(BuildContext context) { final bool _succeeded = @@ -108,6 +117,8 @@ class _FusionProgressViewState extends ConsumerState { .watch(fusionProgressUIStateProvider(widget.walletId)) .fusionRoundsCompleted; + Wakelock.enable(); + return WillPopScope( onWillPop: () async { return await _requestAndProcessCancel(); From 977207ffb3f81406d1d87cb03ff741131e25acba Mon Sep 17 00:00:00 2001 From: sneurlax Date: Wed, 3 Jul 2024 18:01:32 -0500 Subject: [PATCH 18/19] enable wakelock on windows and macos --- .../cashfusion/sub_widgets/fusion_dialog.dart | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/lib/pages_desktop_specific/cashfusion/sub_widgets/fusion_dialog.dart b/lib/pages_desktop_specific/cashfusion/sub_widgets/fusion_dialog.dart index 51502bf81..09bce3a3b 100644 --- a/lib/pages_desktop_specific/cashfusion/sub_widgets/fusion_dialog.dart +++ b/lib/pages_desktop_specific/cashfusion/sub_widgets/fusion_dialog.dart @@ -1,7 +1,9 @@ import 'dart:async'; +import 'dart:io'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:wakelock/wakelock.dart'; import '../../../providers/cash_fusion/fusion_progress_ui_state_provider.dart'; import '../../../providers/global/prefs_provider.dart'; @@ -137,6 +139,8 @@ class _FusionDialogViewState extends ConsumerState { message: "Stopping fusion", ); + await Wakelock.disable(); + return true; } else { return false; @@ -150,6 +154,12 @@ class _FusionDialogViewState extends ConsumerState { super.initState(); } + @override + dispose() { + Wakelock.disable(); + super.dispose(); + } + @override Widget build(BuildContext context) { final bool _succeeded = @@ -162,6 +172,10 @@ class _FusionDialogViewState extends ConsumerState { .watch(fusionProgressUIStateProvider(widget.walletId)) .fusionRoundsCompleted; + if (!Platform.isLinux) { + Wakelock.enable(); + } + return DesktopDialog( maxHeight: 600, child: SingleChildScrollView( From 830fdab0c332bf88e75a329c584b85c3218b6b96 Mon Sep 17 00:00:00 2001 From: julian Date: Fri, 5 Jul 2024 13:25:26 -0600 Subject: [PATCH 19/19] fix frost send for people who use multisig using a single signer --- lib/pages/send_view/frost_ms/send_steps/frost_send_step_3.dart | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/pages/send_view/frost_ms/send_steps/frost_send_step_3.dart b/lib/pages/send_view/frost_ms/send_steps/frost_send_step_3.dart index de0ade7df..c96f15233 100644 --- a/lib/pages/send_view/frost_ms/send_steps/frost_send_step_3.dart +++ b/lib/pages/send_view/frost_ms/send_steps/frost_send_step_3.dart @@ -175,7 +175,7 @@ class _FrostSendStep3State extends ConsumerState { PrimaryButton( label: "Generate transaction", enabled: _userVerifyContinue && - !fieldIsEmptyFlags.reduce((v, e) => v |= e), + !fieldIsEmptyFlags.fold(false, (v, e) => v |= e), onPressed: () async { // collect Share strings final sharesCollected = controllers.map((e) => e.text).toList();