From cbe62f8e4146ea566c33580527caba8415e8b82c Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 10:04:01 +0330 Subject: [PATCH 01/16] feat: Added findAccountsLastIndex to account use case --- lib/features/common/account/account_use_case.dart | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/lib/features/common/account/account_use_case.dart b/lib/features/common/account/account_use_case.dart index a700f976..6e976730 100644 --- a/lib/features/common/account/account_use_case.dart +++ b/lib/features/common/account/account_use_case.dart @@ -44,6 +44,17 @@ class AccountUseCase extends ReactiveUseCase { update(account, item); } + int findAccountsLastIndex() { + int lastIndex = 0; + for (Account account in accounts.value.reversed) { + if (!account.isCustom) { + lastIndex = int.parse(account.name); + break; + } + } + return lastIndex; + } + void resetXsdConversionRate(double value) { _accountCacheRepository.setXsdConversionRate(value); update(xsdConversionRate, value); From afb6572c439f1571d8e69b75cef385c6827dc984 Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 10:04:24 +0330 Subject: [PATCH 02/16] feat: Added private key validation --- lib/common/utils/validation.dart | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/lib/common/utils/validation.dart b/lib/common/utils/validation.dart index 0b099670..f765e2c5 100644 --- a/lib/common/utils/validation.dart +++ b/lib/common/utils/validation.dart @@ -53,6 +53,17 @@ class Validation { return null; } + static String? checkEthereumPrivateKey(BuildContext context, String value) { + String ethereumPrivateKeyPattern = r'^[0-9a-fA-F]{64}$'; + + if (!RegExp(ethereumPrivateKeyPattern, caseSensitive: false) + .hasMatch(value)) { + return FlutterI18n.translate(context, 'invalid_format'); + } + + return null; + } + static String? checkMnsValidation(BuildContext context, String value) { if (!((value.endsWith('.mxc') || value.endsWith('.MXC')) && value.length > 4)) { From 0279c24cabf10005d45d32d6edfd1bf524196178 Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 10:06:53 +0330 Subject: [PATCH 03/16] impv: Add new account settings page --- .../presentation/settings_page_presenter.dart | 20 +++++-------------- 1 file changed, 5 insertions(+), 15 deletions(-) diff --git a/lib/features/settings/presentation/settings_page_presenter.dart b/lib/features/settings/presentation/settings_page_presenter.dart index a9c6f66e..7def881b 100644 --- a/lib/features/settings/presentation/settings_page_presenter.dart +++ b/lib/features/settings/presentation/settings_page_presenter.dart @@ -48,31 +48,21 @@ class SettingsPresenter extends CompletePresenter { notify(() => state.isLoading = true); try { - final index = findAccountLastIndex(state.accounts); - // final index = state.accounts.length; + final index = _accountUserCase.findAccountsLastIndex(); + final newAccount = await _authUseCase.addNewAccount(index); - // final newAccount = await _authUseCase.addCustomAccount('index' ,'6373f6b31ccb382ea61f02a89c28d88972bdc8a45ea0d817826c097188832b3c'); _accountUserCase.addAccount(newAccount); loadCache(); notify(() => state.isLoading = false); - navigator?.pop(); + navigator?.popUntil((route) { + return route.settings.name?.contains('SettingsPage') ?? false; + }); } catch (e, s) { addError(e, s); } } - int findAccountLastIndex(List accounts) { - int lastIndex = 0; - for (Account account in accounts.reversed) { - if (!account.isCustom) { - lastIndex = int.parse(account.name); - break; - } - } - return lastIndex; - } - void changeAccount(Account item) { _accountUserCase.changeAccount(item); _authUseCase.changeAccount(item); From ecdbedd41d2ca28cad0bfe3e35545f6a21579c65 Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 10:10:22 +0330 Subject: [PATCH 04/16] feat: Added imported chip to account item & limit accounts list size --- assets/flutter_i18n/en.json | 14 ++++++++-- .../subfeatures/accounts/account_item.dart | 22 ++++++++++----- .../subfeatures/accounts/portrait.dart | 2 +- .../accounts/show_accounts_dialog.dart | 27 +++++++++++-------- 4 files changed, 45 insertions(+), 20 deletions(-) diff --git a/assets/flutter_i18n/en.json b/assets/flutter_i18n/en.json index a571d04a..7a259164 100644 --- a/assets/flutter_i18n/en.json +++ b/assets/flutter_i18n/en.json @@ -280,7 +280,7 @@ "checking_balance": "Checking balance...", "no_balance": "You have no balance", "no_balance_tip": "You currently have no MXC in your wallet.\nYou can either receive tokens or choose to claim a username later.", - "signature_request": "Signature request", + "signature_request": "Signature request", "sign": "Sign", "insufficient_balance": "Insufficient balance", "unregistered_mns_notice": "Oops! Sending tokens to a username as elusive as a unicorn in a forest. Check if they're registered!", @@ -288,5 +288,15 @@ "send_&_receive": "Send & Receive", "contract": "Contract", "decimals": "Decimals", - "add_token_success_message" : "Hooray! The token has been successfully added to your AXS wallet! 🎉" + "add_token_success_message": "Hooray! The token has been successfully added to your AXS wallet! 🎉", + "import_account": "Import account", + "add_account": "Add account", + "private_key": "Private key", + "import_notice": "Imported accounts won’t be associated with your AXS wallet Secret Recovery Phrase.", + "imported": "Imported", + "show_private_key": "Show private key", + "show_private_key_notice": "Warning: Never disclose this key. Anyone with your private keys can steal any assets held in your account.", + "confirm_show_private_key_notice_1": "Your Private Key provides full access to your wallet and funds.", + "confirm_show_private_key_notice_2": "Do not share this with anyone. MetaMask Support will not request this, but phishers might.", + "confirm_show_private_key_title": "Keep your private key safe" } \ No newline at end of file diff --git a/lib/features/settings/subfeatures/accounts/account_item.dart b/lib/features/settings/subfeatures/accounts/account_item.dart index b135c802..841087d5 100644 --- a/lib/features/settings/subfeatures/accounts/account_item.dart +++ b/lib/features/settings/subfeatures/accounts/account_item.dart @@ -7,17 +7,20 @@ import 'package:mxc_ui/mxc_ui.dart'; import 'portrait.dart'; class AccountItem extends StatelessWidget { - const AccountItem({ - super.key, - required this.account, - this.isSelected = false, - this.onSelect, - }); + const AccountItem( + {super.key, + required this.account, + this.isSelected = false, + this.onSelect, + required this.isCustom}); final Account account; final bool isSelected; final VoidCallback? onSelect; + /// Imported + final bool isCustom; + @override Widget build(BuildContext context) { return InkWell( @@ -35,12 +38,19 @@ class AccountItem extends StatelessWidget { '${FlutterI18n.translate(context, 'account')} ${account.name}', style: FontTheme.of(context).body2.secondary(), ), + const SizedBox(height: Sizes.space2XSmall), Text( account.mns ?? Formatter.formatWalletAddress(account.address, nCharacters: 10), style: FontTheme.of(context).body1.primary(), ), + const SizedBox(height: Sizes.space2XSmall), + if (isCustom) + MxcChipButton( + key: const Key('importedChip'), + onTap: () {}, + title: FlutterI18n.translate(context, 'imported')) ], ), const Spacer(), diff --git a/lib/features/settings/subfeatures/accounts/portrait.dart b/lib/features/settings/subfeatures/accounts/portrait.dart index 6dc4047d..7314e461 100644 --- a/lib/features/settings/subfeatures/accounts/portrait.dart +++ b/lib/features/settings/subfeatures/accounts/portrait.dart @@ -13,7 +13,7 @@ class Portrait extends StatelessWidget { @override Widget build(BuildContext context) { return CircleAvatar( - radius: 12, + radius: 16, child: SvgPicture.string( Jdenticon.toSvg(name), fit: BoxFit.contain, diff --git a/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart b/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart index fb490bff..95db2eb7 100644 --- a/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart +++ b/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart @@ -11,6 +11,7 @@ void showAccountsDialog({ required List accounts, bool isLoading = false, VoidCallback? onAdd, + VoidCallback? onImport, required Function(Account) onSelect, }) { showModalBottomSheet( @@ -41,17 +42,21 @@ void showAccountsDialog({ ), ), ), - ListView.builder( - padding: EdgeInsets.zero, - itemCount: accounts.length, - shrinkWrap: true, - itemBuilder: (ctx, index) { - return AccountItem( - account: accounts[index], - isSelected: currentAccount.address == accounts[index].address, - onSelect: () => onSelect(accounts[index]), - ); - }, + ConstrainedBox( + constraints: const BoxConstraints(maxHeight: 400), + child: ListView.builder( + padding: EdgeInsets.zero, + itemCount: accounts.length, + shrinkWrap: true, + itemBuilder: (ctx, index) { + return AccountItem( + account: accounts[index], + isSelected: currentAccount.address == accounts[index].address, + onSelect: () => onSelect(accounts[index]), + isCustom: accounts[index].isCustom, + ); + }, + ), ), const SizedBox(height: Sizes.spaceXSmall), MxcButton.primary( From 8901eb1eab992555919e6b36f75b6dc9ec03e0ce Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 10:13:06 +0330 Subject: [PATCH 05/16] feat: Added add accounts dialog --- .../account_managment_panel.dart | 12 +++- .../subfeatures/accounts/row_item.dart | 60 +++++++++++++++++++ .../accounts/show_add_accounts_dialog.dart | 50 ++++++++++++++++ 3 files changed, 121 insertions(+), 1 deletion(-) create mode 100644 lib/features/settings/subfeatures/accounts/row_item.dart create mode 100644 lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart diff --git a/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart b/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart index ed75986d..00fd07f4 100644 --- a/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart +++ b/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart @@ -2,11 +2,13 @@ import 'package:datadashwallet/common/common.dart'; import 'package:datadashwallet/core/core.dart'; import 'package:datadashwallet/features/settings/settings.dart'; import 'package:datadashwallet/features/settings/subfeatures/accounts/show_accounts_dialog.dart'; +import 'package:datadashwallet/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_page.dart'; import 'package:flutter/material.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:mxc_ui/mxc_ui.dart'; +import '../../../subfeatures/accounts/show_add_accounts_dialog.dart'; import 'copyable_item.dart'; class AccountManagementPanel extends HookConsumerWidget { @@ -34,7 +36,15 @@ class AccountManagementPanel extends HookConsumerWidget { currentAccount: state.account!, accounts: state.accounts, isLoading: state.isLoading, - onAdd: () => presenter.addNewAccount(), + onAdd: () => showAddAccountsDialog( + context: context, + isLoading: state.isLoading, + onAdd: presenter.addNewAccount, + onImport: () => Navigator.of(context).push( + route.featureDialog( + const ImportAccountPage(), + ), + ),), onSelect: (item) => presenter.changeAccount(item)), child: Row( children: [ diff --git a/lib/features/settings/subfeatures/accounts/row_item.dart b/lib/features/settings/subfeatures/accounts/row_item.dart new file mode 100644 index 00000000..514fd48c --- /dev/null +++ b/lib/features/settings/subfeatures/accounts/row_item.dart @@ -0,0 +1,60 @@ +import 'package:flutter/material.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:mxc_ui/mxc_ui.dart'; + +class RowItem extends HookConsumerWidget { + const RowItem( + this.title, + this.icon, + this.onTap, { + super.key, + }); + + final String title; + final IconData icon; + final void Function() onTap; + + @override + Widget build(BuildContext context, WidgetRef ref) { + return Container( + margin: const EdgeInsets.only(top: Sizes.spaceNormal), + child: InkWell( + onTap: () => onTap(), + child: Padding( + padding: const EdgeInsets.symmetric(vertical: Sizes.spaceSmall), + child: Row( + children: [ + Icon( + icon, + size: 24, + color: ColorsTheme.of(context).iconPrimary, + ), + const SizedBox( + width: 24, + ), + Text( + title, + style: FontTheme.of(context).body2.primary(), + ), + const Spacer(), + const SizedBox( + width: 16, + ), + // trailingIcon != null + // ? Icon( + // MxcIcons.external_link, + // size: 24, + // color: ColorsTheme.of(context).iconPrimary, + // ) + // : Icon( + // Icons.arrow_forward_ios, + // size: 16, + // color: ColorsTheme.of(context).iconWhite32, + // ) + ], + ), + ), + ), + ); + } +} diff --git a/lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart b/lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart new file mode 100644 index 00000000..80f7ff4b --- /dev/null +++ b/lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart @@ -0,0 +1,50 @@ +import 'package:datadashwallet/features/settings/subfeatures/accounts/row_item.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_i18n/flutter_i18n.dart'; +import 'package:mxc_ui/mxc_ui.dart'; + +import 'account_item.dart'; + +void showAddAccountsDialog({ + required BuildContext context, + bool isLoading = false, + required VoidCallback onAdd, + required VoidCallback onImport, +}) { + showModalBottomSheet( + context: context, + useRootNavigator: true, + isScrollControlled: true, + useSafeArea: true, + backgroundColor: Colors.transparent, + builder: (BuildContext context) => Container( + padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 44), + decoration: BoxDecoration( + color: ColorsTheme.of(context).screenBackground, + borderRadius: const BorderRadius.only( + topLeft: Radius.circular(20), + topRight: Radius.circular(20), + ), + ), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + MxcAppBarEvenly.title( + titleText: FlutterI18n.translate(context, 'add_account'), + action: Container( + alignment: Alignment.centerRight, + child: InkWell( + child: const Icon(Icons.close), + onTap: () => Navigator.of(context).pop(false), + ), + ), + ), + RowItem(FlutterI18n.translate(context, 'add_new_account'), + Icons.add_rounded, onAdd), + RowItem(FlutterI18n.translate(context, 'import_account'), + Icons.file_download_outlined, onImport), + ], + ), + ), + ); +} From a476ad25e38b88893c26eb4abdb0f1ffc37270b0 Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 10:14:04 +0330 Subject: [PATCH 06/16] feat: Added import account page --- .../account_details/import_account_page.dart | 90 +++++++++++++++++++ .../import_account_presenter.dart | 51 +++++++++++ .../account_details/import_account_state.dart | 10 +++ 3 files changed, 151 insertions(+) create mode 100644 lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_page.dart create mode 100644 lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_presenter.dart create mode 100644 lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_state.dart diff --git a/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_page.dart b/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_page.dart new file mode 100644 index 00000000..c3c9750b --- /dev/null +++ b/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_page.dart @@ -0,0 +1,90 @@ +import 'package:datadashwallet/common/common.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_i18n/flutter_i18n.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:mxc_ui/mxc_ui.dart'; + +import 'import_account_presenter.dart'; + +class ImportAccountPage extends HookConsumerWidget { + const ImportAccountPage({ + super.key, + }); + + @override + Widget build(BuildContext context, WidgetRef ref) { + final formKey = useMemoized(() => GlobalKey()); + String translate(String text) => FlutterI18n.translate(context, text); + final presenter = ref.read(importAccountContainer.actions); + final state = ref.watch(importAccountContainer.state); + return MxcPage.layer( + presenter: presenter, + crossAxisAlignment: CrossAxisAlignment.start, + layout: LayoutType.scrollable, + children: [ + MxcAppBarEvenly.text( + titleText: translate('import_account'), + actionText: translate('save'), + onActionTap: () { + if (!formKey.currentState!.validate()) return; + presenter.onSave(); + }, + isActionTap: state.ableToSave, + ), + Form( + key: formKey, + child: Column( + children: [ + MxcTextField( + key: const ValueKey('privateKeyTextField'), + label: translate('private_key'), + hint: translate('private_key'), + controller: presenter.privateKeyController, + action: TextInputAction.next, + validator: (value) { + final res = Validation.notEmpty( + context, + value, + translate('x_not_empty') + .replaceFirst('{0}', translate('private_key'))); + if (res != null) return res; + return Validation.checkEthereumPrivateKey( + context, value ?? ''); + }, + onChanged: (value) { + presenter.changeAbleToSave( + formKey.currentState!.validate() ? true : false); + }, + ), + // MxcTextField( + // key: const ValueKey('accountNameTextField'), + // label: translate('account_name'), + // hint: translate('account_name'), + // controller: presenter.rpcUrlController, + // action: TextInputAction.next, + // validator: (value) { + // final res = Validation.notEmpty( + // context, + // value, + // translate('x_not_empty') + // .replaceFirst('{0}', translate('account_name'))); + // if (res != null) return res; + // return Validation.checkHttps(context, value!, + // errorText: translate('invalid_url_format_notice')); + // }, + // onChanged: (value) { + // presenter.changeAbleToSave( + // formKey.currentState!.validate() ? true : false); + // presenter.onRpcUrlChange(value); + // }, + // onFocused: (focused) => + // focused ? null : formKey.currentState!.validate(), + // ), + ], + ), + ), + ], + ); + } +} diff --git a/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_presenter.dart b/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_presenter.dart new file mode 100644 index 00000000..7d7001e2 --- /dev/null +++ b/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_presenter.dart @@ -0,0 +1,51 @@ +import 'package:datadashwallet/core/core.dart'; +import 'package:datadashwallet/features/settings/domain/webview_use_case.dart'; +import 'package:flutter/material.dart'; +import 'package:mxc_ui/mxc_ui.dart'; + +import 'import_account_state.dart'; + +final importAccountContainer = + PresenterContainer( + () => ImportAccountPresenter()); + +class ImportAccountPresenter extends CompletePresenter { + ImportAccountPresenter() : super(ImportAccountState()); + + late final _accountUserCase = ref.read(accountUseCaseProvider); + late final _authUseCase = ref.read(authUseCaseProvider); + late final _webviewUseCase = WebviewUseCase(); + + final TextEditingController privateKeyController = TextEditingController(); + + void onSave() async { + loading = true; + try { + final index = _accountUserCase.findAccountsLastIndex(); + final privateKey = privateKeyController.text; + + final newAccount = + await _authUseCase.addCustomAccount('${index + 1}', privateKey); + _accountUserCase.addAccount(newAccount); + loadCache(); + + notify(() => state.isLoading = false); + BottomFlowDialog.of(context!).close(); + navigator?.popUntil((route) { + return route.settings.name?.contains('SettingsPage') ?? false; + }); + } catch (error, stackTrace) { + addError(error, stackTrace); + } finally { + loading = false; + } + } + + void changeAbleToSave(bool value) { + notify(() => state.ableToSave = value); + } + + void loadCache() { + _webviewUseCase.clearCache(); + } +} diff --git a/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_state.dart b/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_state.dart new file mode 100644 index 00000000..6052c416 --- /dev/null +++ b/lib/features/settings/subfeatures/accounts/subfeatures/account_details/import_account_state.dart @@ -0,0 +1,10 @@ +import 'package:equatable/equatable.dart'; +import 'package:mxc_logic/mxc_logic.dart'; + +class ImportAccountState with EquatableMixin { + bool ableToSave = false; + bool isLoading = false; + + @override + List get props => [ableToSave, isLoading]; +} From d5073793f8950dcf8ba11413d94f5d0f906dbf88 Mon Sep 17 00:00:00 2001 From: reasje Date: Mon, 16 Oct 2023 13:22:43 +0330 Subject: [PATCH 07/16] feat: Added remove account functionality --- .../common/account/account_cache_repository.dart | 2 +- lib/features/common/account/account_use_case.dart | 9 +++++++++ 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/lib/features/common/account/account_cache_repository.dart b/lib/features/common/account/account_cache_repository.dart index 70878aea..7f9267fb 100644 --- a/lib/features/common/account/account_cache_repository.dart +++ b/lib/features/common/account/account_cache_repository.dart @@ -56,7 +56,7 @@ class AccountCacheRepository extends GlobalCacheRepository { void addAccount(Account item) => accounts.value = [...accounts.value, item]; void removeAccount(Account item) => accounts.value = - accounts.value.where((e) => e.name != item.name).toList(); + accounts.value.where((e) => e.address != item.address).toList(); void updateAccount(Account item) => accounts.value = accounts.value.map((e) { if (item.address == account.value!.address) { account.value = item; diff --git a/lib/features/common/account/account_use_case.dart b/lib/features/common/account/account_use_case.dart index 6e976730..35611e3f 100644 --- a/lib/features/common/account/account_use_case.dart +++ b/lib/features/common/account/account_use_case.dart @@ -40,6 +40,15 @@ class AccountUseCase extends ReactiveUseCase { getAccountsNames(); } + /// Deletes the given account, If the account is selected will select the index 0 account + /// This is only used to delete the imported accounts. + void removeAccount(Account item) async { + _accountCacheRepository.removeAccount(item); + final items = _accountCacheRepository.accountItems; + if (item.address == account.value!.address) update(account, items[0]); + update(accounts, items); + } + void changeAccount(Account item) { update(account, item); } From 63ebe97e1bcc7e195ac33c5244c53a839573d58f Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 13:25:19 +0330 Subject: [PATCH 08/16] fix: Removed add accounts for better UX --- .../accounts/show_add_accounts_dialog.dart | 50 ------------------- 1 file changed, 50 deletions(-) delete mode 100644 lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart diff --git a/lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart b/lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart deleted file mode 100644 index 80f7ff4b..00000000 --- a/lib/features/settings/subfeatures/accounts/show_add_accounts_dialog.dart +++ /dev/null @@ -1,50 +0,0 @@ -import 'package:datadashwallet/features/settings/subfeatures/accounts/row_item.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_i18n/flutter_i18n.dart'; -import 'package:mxc_ui/mxc_ui.dart'; - -import 'account_item.dart'; - -void showAddAccountsDialog({ - required BuildContext context, - bool isLoading = false, - required VoidCallback onAdd, - required VoidCallback onImport, -}) { - showModalBottomSheet( - context: context, - useRootNavigator: true, - isScrollControlled: true, - useSafeArea: true, - backgroundColor: Colors.transparent, - builder: (BuildContext context) => Container( - padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 44), - decoration: BoxDecoration( - color: ColorsTheme.of(context).screenBackground, - borderRadius: const BorderRadius.only( - topLeft: Radius.circular(20), - topRight: Radius.circular(20), - ), - ), - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - MxcAppBarEvenly.title( - titleText: FlutterI18n.translate(context, 'add_account'), - action: Container( - alignment: Alignment.centerRight, - child: InkWell( - child: const Icon(Icons.close), - onTap: () => Navigator.of(context).pop(false), - ), - ), - ), - RowItem(FlutterI18n.translate(context, 'add_new_account'), - Icons.add_rounded, onAdd), - RowItem(FlutterI18n.translate(context, 'import_account'), - Icons.file_download_outlined, onImport), - ], - ), - ), - ); -} From a189f4f0a1be88cd7056d2ed2837fd59810d722b Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:27:05 +0330 Subject: [PATCH 09/16] feat: Added private key validation & moved pub key validation to validation --- lib/common/utils/validation.dart | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/lib/common/utils/validation.dart b/lib/common/utils/validation.dart index f765e2c5..ef9f1db0 100644 --- a/lib/common/utils/validation.dart +++ b/lib/common/utils/validation.dart @@ -1,6 +1,8 @@ import 'package:datadashwallet/common/config.dart'; import 'package:flutter/material.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; +import 'package:web3dart/web3dart.dart'; + class Validation { static String? notEmpty(BuildContext context, String? value, @@ -129,4 +131,23 @@ class Validation { return regex.hasMatch(value); } + + + static bool isAddress(String address) { + try { + EthereumAddress.fromHex(address); + return true; + } catch (e) { + return false; + } + } + + static bool isPrivateKey(String privateKey) { + try { + EthPrivateKey.fromHex(privateKey); + return true; + } catch (e) { + return false; + } + } } From b7049adb9abad1d7f8be91d17c3f0c3a37d6bf9f Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:35:15 +0330 Subject: [PATCH 10/16] feat: Updated single line info for custom icons & functions --- .../list/single_line_info_item.dart | 32 ++++++++++++------- 1 file changed, 21 insertions(+), 11 deletions(-) diff --git a/lib/common/components/list/single_line_info_item.dart b/lib/common/components/list/single_line_info_item.dart index 47a55441..f3bf98f1 100644 --- a/lib/common/components/list/single_line_info_item.dart +++ b/lib/common/components/list/single_line_info_item.dart @@ -1,23 +1,29 @@ -import 'package:datadashwallet/features/dapps/subfeatures/open_dapp/open_dapp_presenter.dart'; +import 'package:datadashwallet/common/common.dart'; +import 'package:datadashwallet/core/src/providers/providers_use_cases.dart'; import 'package:flutter/material.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:mxc_ui/mxc_ui.dart'; class SingleLineInfoItem extends HookConsumerWidget { - const SingleLineInfoItem({ - super.key, - required this.title, - required this.value, - this.hint, - }); + const SingleLineInfoItem( + {super.key, + required this.title, + required this.value, + this.hint, + this.valueActionIcon}); + final String title; final String value; final String? hint; + final Widget? valueActionIcon; + @override Widget build(BuildContext context, WidgetRef ref) { - final presenter = ref.read(openDAppPageContainer.actions); - final isAddress = presenter.isAddress(value); + late final _chainConfigurationUseCase = + ref.read(chainConfigurationUseCaseProvider); + + final isAddress = Validation.isAddress(value); return Padding( padding: const EdgeInsets.symmetric(vertical: Sizes.spaceXSmall), child: Row( @@ -34,7 +40,9 @@ class SingleLineInfoItem extends HookConsumerWidget { ), Expanded( child: InkWell( - onTap: isAddress ? () => presenter.launchAddress(value) : null, + onTap: isAddress + ? () => _chainConfigurationUseCase.launchAddress(value) + : null, child: Row( mainAxisAlignment: MainAxisAlignment.end, children: [ @@ -46,7 +54,9 @@ class SingleLineInfoItem extends HookConsumerWidget { textAlign: TextAlign.end, ), ), - if (isAddress) ...[ + if (valueActionIcon != null) + valueActionIcon! + else if (isAddress) ...[ const SizedBox(width: 8), Icon( MxcIcons.external_link, From 7ed6fc45a337a4a3416f66fcea2a63942cda37a6 Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:36:59 +0330 Subject: [PATCH 11/16] feat: Added passcode athentication page for verifying user --- .../open_dapp/widgets/add_asset_info.dart | 4 +- .../passcode_authenticate_user_page.dart | 67 +++++++++++++++++++ .../passcode_authenticate_user_presenter.dart | 41 ++++++++++++ .../passcode_authenticate_user_state.dart | 11 +++ 4 files changed, 122 insertions(+), 1 deletion(-) create mode 100644 lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_page.dart create mode 100644 lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_presenter.dart create mode 100644 lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_state.dart diff --git a/lib/features/dapps/subfeatures/open_dapp/widgets/add_asset_info.dart b/lib/features/dapps/subfeatures/open_dapp/widgets/add_asset_info.dart index 2c784dbd..250a966d 100644 --- a/lib/features/dapps/subfeatures/open_dapp/widgets/add_asset_info.dart +++ b/lib/features/dapps/subfeatures/open_dapp/widgets/add_asset_info.dart @@ -65,7 +65,9 @@ class AddAssetInfo extends ConsumerWidget { title: FlutterI18n.translate(context, 'contract'), value: contractAddress ?? '')); infoList.add(SingleLineInfoItem( - title: FlutterI18n.translate(context, 'symbol'), value: symbol ?? '')); + title: FlutterI18n.translate(context, 'symbol'), + value: symbol ?? '', + )); infoList.add(SingleLineInfoItem( title: FlutterI18n.translate(context, 'decimals'), value: decimalsString)); diff --git a/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_page.dart b/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_page.dart new file mode 100644 index 00000000..d8a3ce30 --- /dev/null +++ b/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_page.dart @@ -0,0 +1,67 @@ +import 'package:datadashwallet/features/security/security.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_i18n/flutter_i18n.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:mxc_ui/mxc_ui.dart'; + +import 'passcode_authenticate_user_presenter.dart'; +import 'passcode_authenticate_user_state.dart'; + +class PasscodeAuthenticateUserPage extends PasscodeBasePage { + const PasscodeAuthenticateUserPage( + {Key? key, this.change = false, this.dismissedDest}) + : super(key: key); + + final bool change; + final String? dismissedDest; + + @override + String title(BuildContext context, WidgetRef ref) => + FlutterI18n.translate(context, 'view_private_key'); + + @override + String hint(BuildContext context, WidgetRef ref) => + FlutterI18n.translate(context, 'enter_your_passcode'); + + @override + String secondHint(BuildContext context, WidgetRef ref) => + FlutterI18n.translate(context, 'view_private_key_notice'); + + @override + String? dismissedPage() => dismissedDest; + + @override + ProviderBase get presenter => + passcodeAuthenticateUserContainer.actions; + + @override + ProviderBase get state => + passcodeAuthenticateUserContainer.state; + + @override + Widget buildErrorMessage(BuildContext context, WidgetRef ref) => SizedBox( + height: 90, + width: 280, + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + crossAxisAlignment: CrossAxisAlignment.stretch, + children: [ + if (ref.watch(state).errorText != null) ...[ + Text( + ref.watch(state).errorText!, + textAlign: TextAlign.center, + style: FontTheme.of(context).subtitle2.error(), + ), + if (ref.watch(state).wrongInputCounter == 2) ...[ + const SizedBox(height: 6), + Text( + FlutterI18n.translate(context, 'app_will_be_locked_alert'), + textAlign: TextAlign.center, + style: FontTheme.of(context).subtitle1.error(), + ), + ], + ] + ], + ), + ); +} diff --git a/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_presenter.dart b/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_presenter.dart new file mode 100644 index 00000000..fc71b572 --- /dev/null +++ b/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_presenter.dart @@ -0,0 +1,41 @@ +import 'package:datadashwallet/core/core.dart'; +import 'package:datadashwallet/features/security/security.dart'; +import 'passcode_authenticate_user_state.dart'; + +final passcodeAuthenticateUserContainer = PresenterContainer< + PasscodeAuthenticateUserPresenter, + PasscodeAuthenticateUserState>( + () => PasscodeAuthenticateUserPresenter()); + +class PasscodeAuthenticateUserPresenter + extends PasscodeBasePagePresenter { + PasscodeAuthenticateUserPresenter() + : super(PasscodeAuthenticateUserState()); + + late final PasscodeUseCase _passcodeUseCase = + ref.read(passcodeUseCaseProvider); + + @override + void onAllNumbersEntered(String? dismissedPage) async { + if (state.enteredNumbers.join('') != _passcodeUseCase.passcode.value) { + if (state.wrongInputCounter < 2) { + state.errorText = translate('incorrect_passcode')!; + state.wrongInputCounter++; + } else { + state.errorText = null; + state.wrongInputCounter = 0; + ref.read(passcodeUseCaseProvider).penaltyLock(); + } + state.enteredNumbers = []; + notify(); + return; + } + + Future.delayed( + passcodeTransitionDuration, + () => notify(() => state.enteredNumbers = []), + ); + + navigator?.pop(true); + } +} diff --git a/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_state.dart b/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_state.dart new file mode 100644 index 00000000..e1c5a093 --- /dev/null +++ b/lib/features/security/presentation/passcode_authenticate/passcode_authenticate_user_state.dart @@ -0,0 +1,11 @@ +import 'package:datadashwallet/features/security/security.dart'; + +class PasscodeAuthenticateUserState extends PasscodeBasePageState { + int wrongInputCounter = 0; + + @override + List get props => [ + ...super.props, + wrongInputCounter, + ]; +} From 157638db744cdc7ef3a1400a15fe1642ac96f573 Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:37:37 +0330 Subject: [PATCH 12/16] feat: Added second hint for passcode pages --- .../presentation/passcode_base/passcode_base_page.dart | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/lib/features/security/presentation/passcode_base/passcode_base_page.dart b/lib/features/security/presentation/passcode_base/passcode_base_page.dart index a1db5786..13242570 100644 --- a/lib/features/security/presentation/passcode_base/passcode_base_page.dart +++ b/lib/features/security/presentation/passcode_base/passcode_base_page.dart @@ -22,6 +22,8 @@ abstract class PasscodeBasePage extends HookConsumerWidget { String hint(BuildContext context, WidgetRef ref); + String? secondHint(BuildContext context, WidgetRef ref) => null; + String description(BuildContext context, WidgetRef ref) => ''; String? dismissedPage() => null; @@ -178,6 +180,13 @@ abstract class PasscodeBasePage extends HookConsumerWidget { hint(context, ref), style: FontTheme.of(context).body1.white(), ), + if (secondHint(context, ref) != null) ...[ + const SizedBox(height: 16), + Text( + secondHint(context, ref)!, + style: FontTheme.of(context).body1.white(), + ), + ], const SizedBox(height: 64), SizedBox( height: 57.5, From b236cf8ac7449ab21b245a604b0e2b09ee3dbe17 Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:40:01 +0330 Subject: [PATCH 13/16] feat: Added remove account functionality --- assets/flutter_i18n/en.json | 10 +++--- .../common/account/account_use_case.dart | 5 ++- .../open_dapp/open_dapp_presenter.dart | 7 +--- .../presentation/settings_page_presenter.dart | 34 +++++++++++++++++-- .../account_managment/copyable_item.dart | 3 -- .../subfeatures/accounts/account_item.dart | 15 +++++--- 6 files changed, 53 insertions(+), 21 deletions(-) diff --git a/assets/flutter_i18n/en.json b/assets/flutter_i18n/en.json index 7a259164..334f3b21 100644 --- a/assets/flutter_i18n/en.json +++ b/assets/flutter_i18n/en.json @@ -294,9 +294,9 @@ "private_key": "Private key", "import_notice": "Imported accounts won’t be associated with your AXS wallet Secret Recovery Phrase.", "imported": "Imported", - "show_private_key": "Show private key", - "show_private_key_notice": "Warning: Never disclose this key. Anyone with your private keys can steal any assets held in your account.", - "confirm_show_private_key_notice_1": "Your Private Key provides full access to your wallet and funds.", - "confirm_show_private_key_notice_2": "Do not share this with anyone. MetaMask Support will not request this, but phishers might.", - "confirm_show_private_key_title": "Keep your private key safe" + "view_private_key": "View private key", + "view_private_key_notice": "Warning: Never disclose this key. Anyone with your private keys can steal any assets held in your account.", + "removing_account": "Removing account", + "removing_account_warning": "Are you sure you want to remove this account?", + "remove": "Remove" } \ No newline at end of file diff --git a/lib/features/common/account/account_use_case.dart b/lib/features/common/account/account_use_case.dart index 35611e3f..3b6dae30 100644 --- a/lib/features/common/account/account_use_case.dart +++ b/lib/features/common/account/account_use_case.dart @@ -45,9 +45,12 @@ class AccountUseCase extends ReactiveUseCase { void removeAccount(Account item) async { _accountCacheRepository.removeAccount(item); final items = _accountCacheRepository.accountItems; - if (item.address == account.value!.address) update(account, items[0]); update(accounts, items); } + + bool isAccountSelected(Account item) { + return (item.address == account.value!.address); + } void changeAccount(Account item) { update(account, item); diff --git a/lib/features/dapps/subfeatures/open_dapp/open_dapp_presenter.dart b/lib/features/dapps/subfeatures/open_dapp/open_dapp_presenter.dart index f863657e..41a75575 100644 --- a/lib/features/dapps/subfeatures/open_dapp/open_dapp_presenter.dart +++ b/lib/features/dapps/subfeatures/open_dapp/open_dapp_presenter.dart @@ -350,12 +350,7 @@ class OpenDAppPresenter extends CompletePresenter { } bool isAddress(String address) { - try { - EthereumAddress.fromHex(address); - return true; - } catch (e) { - return false; - } + return Validation.isAddress(address); } void addAsset(int id, Map data, diff --git a/lib/features/settings/presentation/settings_page_presenter.dart b/lib/features/settings/presentation/settings_page_presenter.dart index 7def881b..2c96bf35 100644 --- a/lib/features/settings/presentation/settings_page_presenter.dart +++ b/lib/features/settings/presentation/settings_page_presenter.dart @@ -1,6 +1,9 @@ import 'package:clipboard/clipboard.dart'; +import 'package:datadashwallet/common/components/snack_bar.dart'; +import 'package:datadashwallet/common/dialogs/alert_dialog.dart'; import 'package:datadashwallet/core/core.dart'; import 'package:datadashwallet/features/settings/settings.dart'; +import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:mxc_logic/mxc_logic.dart'; import 'package:package_info_plus/package_info_plus.dart'; import 'settings_page_state.dart'; @@ -33,6 +36,9 @@ class SettingsPresenter extends CompletePresenter { void copyToClipboard(String text) async { FlutterClipboard.copy(text).then((value) => null); + + showSnackBar( + context: context!, content: FlutterI18n.translate(context!, 'copied')); } void getAppVersion() async { @@ -63,12 +69,36 @@ class SettingsPresenter extends CompletePresenter { } } - void changeAccount(Account item) { + void changeAccount(Account item, {bool shouldPop = true}) { _accountUserCase.changeAccount(item); _authUseCase.changeAccount(item); loadCache(); - navigator?.pop(); + if (shouldPop) navigator?.pop(); + } + + void removeAccount(Account item) async { + try { + final result = await showAlertDialog( + context: context!, + title: translate('removing_account')!, + content: translate('removing_account_warning')!, + ok: translate('remove')!, + ); + + if (result != null && result) { + _accountUserCase.removeAccount(item); + + final isSelected = _accountUserCase.isAccountSelected(item); + if (isSelected) { + changeAccount(state.accounts[0], shouldPop: false); + } + + navigator?.pop(); + } + } catch (e, s) { + addError(e, s); + } } void loadCache() { diff --git a/lib/features/settings/presentation/widgets/account_managment/copyable_item.dart b/lib/features/settings/presentation/widgets/account_managment/copyable_item.dart index 1ac44a16..dc0cd530 100644 --- a/lib/features/settings/presentation/widgets/account_managment/copyable_item.dart +++ b/lib/features/settings/presentation/widgets/account_managment/copyable_item.dart @@ -19,9 +19,6 @@ class CopyableItem extends HookConsumerWidget { return InkWell( onTap: () { presenter.copyToClipboard(copyableText); - showSnackBar( - context: context, - content: FlutterI18n.translate(context, 'copied')); }, child: Row( mainAxisAlignment: MainAxisAlignment.center, diff --git a/lib/features/settings/subfeatures/accounts/account_item.dart b/lib/features/settings/subfeatures/accounts/account_item.dart index 841087d5..7f9b1d77 100644 --- a/lib/features/settings/subfeatures/accounts/account_item.dart +++ b/lib/features/settings/subfeatures/accounts/account_item.dart @@ -12,11 +12,13 @@ class AccountItem extends StatelessWidget { required this.account, this.isSelected = false, this.onSelect, - required this.isCustom}); + required this.isCustom, + this.onRemove}); final Account account; final bool isSelected; final VoidCallback? onSelect; + final Function(Account)? onRemove; /// Imported final bool isCustom; @@ -54,9 +56,14 @@ class AccountItem extends StatelessWidget { ], ), const Spacer(), - if (isSelected) ...[ - const Icon(Icons.check_rounded), - ] + if (isSelected) const Icon(Icons.check_rounded), + if (isCustom) + IconButton( + key: const Key('removeImportedAccountButton'), + icon: Icon(Icons.delete, + size: 24, color: ColorsTheme.of(context).iconPrimary), + onPressed:() => onRemove!(account), + ) ], ), ), From c16446a103b9a1e12be0b38972fbb073c70d3759 Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:40:32 +0330 Subject: [PATCH 14/16] feat: Added view private key button to qr code page --- .../qr_code/show_qa_code/qr_code_page.dart | 32 ++++++++++++++++--- 1 file changed, 27 insertions(+), 5 deletions(-) diff --git a/lib/features/settings/subfeatures/qr_code/show_qa_code/qr_code_page.dart b/lib/features/settings/subfeatures/qr_code/show_qa_code/qr_code_page.dart index 0f3f7c23..b2a7c649 100644 --- a/lib/features/settings/subfeatures/qr_code/show_qa_code/qr_code_page.dart +++ b/lib/features/settings/subfeatures/qr_code/show_qa_code/qr_code_page.dart @@ -1,6 +1,8 @@ import 'package:datadashwallet/common/common.dart'; import 'package:datadashwallet/core/core.dart'; +import 'package:datadashwallet/features/security/presentation/passcode_authenticate/passcode_authenticate_user_page.dart'; import 'package:datadashwallet/features/settings/presentation/widgets/account_managment/copyable_item.dart'; +import 'package:datadashwallet/features/settings/subfeatures/accounts/show_view_private_key_dialog.dart'; import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; @@ -8,20 +10,21 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:mxc_ui/mxc_ui.dart'; import 'package:qr_flutter/qr_flutter.dart'; +import '../../../presentation/settings_page_presenter.dart'; import '../qr_scanner/qr_scanner_page.dart'; class QrCodePage extends HookConsumerWidget { - const QrCodePage({ - Key? key, - this.name, - this.address, - }) : super(key: key); + const QrCodePage( + {Key? key, this.name, this.address, required this.privateKey}) + : super(key: key); final String? name; final String? address; + final String privateKey; @override Widget build(BuildContext context, WidgetRef ref) { + final presenter = ref.read(settingsContainer.actions); String translate(String text) => FlutterI18n.translate(context, text); return MxcPage( @@ -68,6 +71,25 @@ class QrCodePage extends HookConsumerWidget { ], ), ), + MxcButton.secondary( + key: const ValueKey('viewPrivateKeyButton'), + title: FlutterI18n.translate(context, 'view_private_key'), + onTap: () => Navigator.of(context) + .push( + route.featureDialog( + const PasscodeAuthenticateUserPage( + dismissedDest: 'QrCodePage', + )), + ) + .then((value) { + if (value == true) { + showViewPrivateKeyDialog( + context: context, + privateKey: privateKey, + onCopy: presenter.copyToClipboard); + } + }), + ), const SizedBox(height: Sizes.space5XLarge), MxcButton.primary( key: const ValueKey('scanQrCodeButton'), From 3af7afde051169d5ab93b65cb311bdc6e87f9aa5 Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:40:56 +0330 Subject: [PATCH 15/16] feat: Added show private key dialog --- .../show_view_private_key_dialog.dart | 63 +++++++++++++++++++ 1 file changed, 63 insertions(+) create mode 100644 lib/features/settings/subfeatures/accounts/show_view_private_key_dialog.dart diff --git a/lib/features/settings/subfeatures/accounts/show_view_private_key_dialog.dart b/lib/features/settings/subfeatures/accounts/show_view_private_key_dialog.dart new file mode 100644 index 00000000..78a25247 --- /dev/null +++ b/lib/features/settings/subfeatures/accounts/show_view_private_key_dialog.dart @@ -0,0 +1,63 @@ +import 'package:datadashwallet/common/common.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_i18n/flutter_i18n.dart'; +import 'package:mxc_ui/mxc_ui.dart'; + +void showViewPrivateKeyDialog( + {required BuildContext context, + required String privateKey, + required Function(String) onCopy}) { + showModalBottomSheet( + context: context, + useRootNavigator: true, + isScrollControlled: true, + useSafeArea: true, + backgroundColor: Colors.transparent, + builder: (BuildContext context) => Container( + padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 44), + decoration: BoxDecoration( + color: ColorsTheme.of(context).screenBackground, + borderRadius: const BorderRadius.only( + topLeft: Radius.circular(20), + topRight: Radius.circular(20), + ), + ), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + MxcAppBarEvenly.title( + titleText: FlutterI18n.translate(context, 'private_key'), + action: Container( + alignment: Alignment.centerRight, + child: InkWell( + child: const Icon(Icons.close), + onTap: () => Navigator.of(context).pop(false), + ), + ), + ), + SingleLineInfoItem( + title: 'private_key', + value: privateKey, + valueActionIcon: IconButton( + icon: Icon( + MxcIcons.copy, + size: 20, + color: ColorsTheme.of(context).iconGrey1, + ), + onPressed: () { + onCopy(privateKey); + Navigator.of(context).pop(); + }), + ), + const SizedBox(height: Sizes.spaceXSmall), + MxcButton.primary( + key: const ValueKey('doneButton'), + title: FlutterI18n.translate(context, 'done'), + onTap: () => Navigator.of(context).pop(false), + size: AxsButtonSize.xl, + ), + ], + ), + ), + ); +} From 91bdf2b1a6bd2bc4002bd27f89e5c3390dd10635 Mon Sep 17 00:00:00 2001 From: reasje Date: Tue, 17 Oct 2023 16:41:53 +0330 Subject: [PATCH 16/16] feat: Updated show accounts dialog & added import button --- .../account_managment_panel.dart | 27 ++-- .../accounts/show_accounts_dialog.dart | 120 ++++++++++-------- 2 files changed, 81 insertions(+), 66 deletions(-) diff --git a/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart b/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart index 00fd07f4..a65dab97 100644 --- a/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart +++ b/lib/features/settings/presentation/widgets/account_managment/account_managment_panel.dart @@ -8,7 +8,6 @@ import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:mxc_ui/mxc_ui.dart'; -import '../../../subfeatures/accounts/show_add_accounts_dialog.dart'; import 'copyable_item.dart'; class AccountManagementPanel extends HookConsumerWidget { @@ -32,20 +31,19 @@ class AccountManagementPanel extends HookConsumerWidget { children: [ InkWell( onTap: () => showAccountsDialog( - context: context, - currentAccount: state.account!, - accounts: state.accounts, - isLoading: state.isLoading, - onAdd: () => showAddAccountsDialog( - context: context, - isLoading: state.isLoading, - onAdd: presenter.addNewAccount, - onImport: () => Navigator.of(context).push( - route.featureDialog( - const ImportAccountPage(), + context: context, + currentAccount: state.account!, + accounts: state.accounts, + isLoading: state.isLoading, + onImport: () => Navigator.of(context).push( + route.featureDialog( + const ImportAccountPage(), + ), ), - ),), - onSelect: (item) => presenter.changeAccount(item)), + onAdd: () => presenter.addNewAccount(), + onSelect: (item) => presenter.changeAccount(item), + onRemove: (item) => presenter.removeAccount(item), + ), child: Row( children: [ Portrait( @@ -95,6 +93,7 @@ class AccountManagementPanel extends HookConsumerWidget { onTap: () => Navigator.of(context).push(route(QrCodePage( name: account.mns, address: account.address, + privateKey: state.account?.privateKey ?? '', ))), child: Container( padding: const EdgeInsets.all(Sizes.spaceXSmall), diff --git a/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart b/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart index 95db2eb7..c321eb4e 100644 --- a/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart +++ b/lib/features/settings/subfeatures/accounts/show_accounts_dialog.dart @@ -1,72 +1,88 @@ import 'package:flutter/material.dart'; +import 'package:flutter/rendering.dart'; import 'package:flutter_i18n/flutter_i18n.dart'; import 'package:mxc_logic/mxc_logic.dart'; import 'package:mxc_ui/mxc_ui.dart'; import 'account_item.dart'; -void showAccountsDialog({ - required BuildContext context, - required Account currentAccount, - required List accounts, - bool isLoading = false, - VoidCallback? onAdd, - VoidCallback? onImport, - required Function(Account) onSelect, -}) { +void showAccountsDialog( + {required BuildContext context, + required Account currentAccount, + required List accounts, + bool isLoading = false, + VoidCallback? onAdd, + VoidCallback? onImport, + required Function(Account) onSelect, + required Function(Account) onRemove}) { showModalBottomSheet( context: context, useRootNavigator: true, isScrollControlled: true, useSafeArea: true, backgroundColor: Colors.transparent, - builder: (BuildContext context) => Container( - padding: const EdgeInsets.only(left: 16, right: 16, top: 0, bottom: 44), - decoration: BoxDecoration( - color: ColorsTheme.of(context).screenBackground, - borderRadius: const BorderRadius.only( - topLeft: Radius.circular(20), - topRight: Radius.circular(20), - ), + builder: (BuildContext context) => ConstrainedBox( + constraints: BoxConstraints( + maxHeight: MediaQuery.of(context).size.height * 0.95, ), - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - MxcAppBarEvenly.title( - titleText: FlutterI18n.translate(context, 'accounts'), - action: Container( - alignment: Alignment.centerRight, - child: InkWell( - child: const Icon(Icons.close), - onTap: () => Navigator.of(context).pop(false), + child: Container( + padding: const EdgeInsets.only( + left: 16, right: 16, top: 0, bottom: Sizes.space3XLarge), + decoration: BoxDecoration( + color: ColorsTheme.of(context).screenBackground, + borderRadius: const BorderRadius.only( + topLeft: Radius.circular(20), + topRight: Radius.circular(20), + ), + ), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + MxcAppBarEvenly.title( + titleText: FlutterI18n.translate(context, 'accounts'), + action: Container( + alignment: Alignment.centerRight, + child: InkWell( + child: const Icon(Icons.close), + onTap: () => Navigator.of(context).pop(false), + ), ), ), - ), - ConstrainedBox( - constraints: const BoxConstraints(maxHeight: 400), - child: ListView.builder( - padding: EdgeInsets.zero, - itemCount: accounts.length, - shrinkWrap: true, - itemBuilder: (ctx, index) { - return AccountItem( - account: accounts[index], - isSelected: currentAccount.address == accounts[index].address, - onSelect: () => onSelect(accounts[index]), - isCustom: accounts[index].isCustom, - ); - }, + Flexible( + child: ListView.builder( + padding: EdgeInsets.zero, + itemCount: accounts.length, + shrinkWrap: true, + itemBuilder: (ctx, index) { + return AccountItem( + account: accounts[index], + isSelected: + currentAccount.address == accounts[index].address, + onSelect: () => onSelect(accounts[index]), + isCustom: accounts[index].isCustom, + onRemove: onRemove, + ); + }, + ), ), - ), - const SizedBox(height: Sizes.spaceXSmall), - MxcButton.primary( - key: const ValueKey('addAccountButton'), - title: FlutterI18n.translate( - context, isLoading ? 'adding_account' : 'add_new_account'), - onTap: onAdd, - size: AxsButtonSize.xl, - ), - ], + const SizedBox(height: Sizes.spaceXSmall), + MxcButton.primary( + key: const ValueKey('addAccountButton'), + title: FlutterI18n.translate( + context, isLoading ? 'adding_account' : 'add_new_account'), + onTap: onAdd, + size: AxsButtonSize.xl, + ), + const SizedBox(height: Sizes.spaceXSmall), + MxcButton.plainWhite( + key: const ValueKey('importAccountButton'), + title: FlutterI18n.translate(context, 'import_account'), + onTap: onImport, + size: AxsButtonSize.xl, + titleColor: ColorsTheme.of(context).primary, + ), + ], + ), ), ), );