diff --git a/.github/workflows/pr_test_build.yml b/.github/workflows/pr_test_build.yml index 4cbd0cd36..c735f8161 100644 --- a/.github/workflows/pr_test_build.yml +++ b/.github/workflows/pr_test_build.yml @@ -99,6 +99,7 @@ jobs: run: | cd /opt/android/cake_wallet touch lib/.secrets.g.dart + touch cw_ethereum/lib/.secrets.g.dart echo "const salt = '${{ secrets.SALT }}';" > lib/.secrets.g.dart echo "const keychainSalt = '${{ secrets.KEY_CHAIN_SALT }}';" >> lib/.secrets.g.dart echo "const key = '${{ secrets.KEY }}';" >> lib/.secrets.g.dart diff --git a/PRIVACY.md b/PRIVACY.md index d740dcba8..88f180c5e 100644 --- a/PRIVACY.md +++ b/PRIVACY.md @@ -1,6 +1,6 @@ Privacy Policy -Last modified: July 21, 2022 +Last modified: August 9, 2023 Introduction ============ @@ -13,7 +13,7 @@ Introduction - On this App. - In email, text, and other electronic messages between you and this App. It does not apply to information collected by: - - Us offline or through any other means, including on any other App operated by Company or any third party (including our affiliates and subsidiaries)]; or + - Us offline or through any other means, including on any other App operated by Company or any third party (including our affiliates and subsidiaries); or - Any third party (including our affiliates and subsidiaries), including through any application or content (including advertising) that may link to or be accessible from or on the App. Please read this policy carefully to understand our policies and practices regarding your information and how we will treat it. If you do not agree with our policies and practices, you have the choice to not use the App. By accessing or using this App, you agree to this privacy policy. This policy may change from time to time. Your continued use of this App after we make changes is deemed to be acceptance of those changes, so please check the policy periodically for updates. @@ -25,7 +25,7 @@ Definitions - "Node" means a server on a supported cryptocurrency network which transmits data to your App for processing and synchronization, and to which your Device transmits transactions which you would like to submit to the supported cryptocurrency networks. This includes full nodes, Electrum servers, and lightning network nodes. - "Cake Labs Nodes" refers to the set of cryptocurrency nodes operated and maintained by Cake Labs LLC. - "Service" refers to the App. - - "Third-party Service" refers to any service integrated into the App. This includes but is not limited to ChangeNOW, Wyre, MoonPay, and BlockBuy. + - "Third-party Service" refers to any service integrated into the App. This includes but is not limited to ChangeNOW, Onramper, and MoonPay. - "Usage Data" refers to data collected automatically about your usage of an App. - "You" means the individual, group, corporation, or any other entity accessing or using the Service. @@ -40,26 +40,29 @@ Information We Never Receive Nor Collect Information We May Receive But Do Not Retain -------------------------------------------- - We receive but do NOT store information from and about users of our App, including: + We may receive but do NOT store information from and about users of our App, including: - The device IP address, the block height to which your wallet is synchronized, and any transactions or channels which you use our Node to submit to supported cryptocurrency networks. We receive this information: - - Automatically as you use the App. + - Automatically as you use the App, unless you turn certain features off in your App privacy settings. - This data is provided by connecting to the Nodes and price API maintained by Cake Labs. You have the right to choose not to provide synchronization data to Cake Labs by choosing a different Node. We provide a list of Nodes in the app that include our own and third party Nodes, or you can use your own Node (which we recommend). + This data is provided by connecting to the Nodes and price API maintained by Cake Labs. You have the right to choose not to provide synchronization data to Cake Labs by choosing a different Node. We provide a list of Nodes in the app that include our own and third party Nodes, or you can use your own Node (which we recommend). You have the right to choose not to connect to our Fiat API service by disabling this Fiat API in App privacy settings. - Personal Data sent through the Cake Labs Nodes is limited to your device's IP address, the block height to which your wallet is synchronized, and any transactions or channels which you use our Node to submit to the supported cryptocurrency networks. Personal Data received by Cake Labs in this manner is not stored for any length of time, and thus Cake Labs is both unwilling to and incapable of sharing this data, or using it for any purpose beyond ensuring your appropriate connection to our Nodes. + Personal Data that may be sent through the Cake Labs Nodes is limited to your device's IP address, the block height to which your wallet is synchronized, and any transactions or channels which you use our Node to submit to the supported cryptocurrency networks. Personal Data received by Cake Labs in this manner is not stored for any length of time, and thus Cake Labs is incapable of sharing this data and will not use it for any purpose beyond ensuring your appropriate connection to our Nodes. If you decide to use a Node offered by any third party, some of which we include in our Apps, said third party will receive this Personal Data instead of Cake Labs. We take no responsibility for the actions of any third-party Node offered within the Application. We recommend connecting to your own Node to limit third party sharing of your Personal Information. + If you use our Fiat API service, you will share your IP address and the cryptocurrency and fiat currency exchange pair for which your wallet requests a spot price quote. You can disable this Fiat API in App privacy settings. + Information We May Collect About You and How We Collect It ---------------------------------------------------------- We collect several types of information from and about users of our App, including information: - - By which you may be personally identified, such as name, e-mail address, or and a/any other identifier by which you may be contacted online or offline ("personal information" or "Personal Data”), ONLY when you provide it to us; + - By which you may be personally identified, such as name, e-mail address, or and a/any other identifier by which you may be contacted online or offline ("personal information" or "Personal Data”); + - Device data and error log data; We collect this information: - - Directly from you when you provide it to us. + - Directly from you ONLY when you provide it to us. - Personal information is received by Cake Labs ONLY in the event that you choose to provide it to us by voluntarily contacting Cake Labs regarding support, questions or suggestions. + Personal information is received by Cake Labs ONLY in the event that you choose to provide it to us by voluntarily contacting Cake Labs regarding support, questions or suggestions. You may optionally send us Error reports to help us improve the App. These Error reports contain error logs and basic device data. You can review and make modifications to these Error reports before sending them to us, or you may choose not to send them to us at all. How We Use Your Information --------------------------- @@ -112,7 +115,9 @@ Data Security Links to Other Websites ----------------------- - The App may contain links to other websites that are not operated by us. If you click on a third-party link, you will be directed to that third party's site. We strongly advise you to review the Privacy Policy of every site you visit. We have no control over and assume no responsibility for the content, privacy policies or practices of any third-party sites or services. + The App may contain links to other websites that are not operated by us. If you click on a Third-Party Service link, you will be directed to that third party's site. We strongly advise you to review the Privacy Policy of every site you visit. We have no control over and assume no responsibility for the content, privacy policies or practices of any third-party sites or services. + + The App includes several optional Third-Party Services, which may not be available to all users. If you use Third-Party Services, you must agree to their respective Privacy Policies. Changes to Our Privacy Policy ----------------------------- diff --git a/README.md b/README.md index 317ad91b7..7b739f980 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -# Cake Wallet for Android and iOS +# Cake Wallet for Mobile and Desktop ## Open Source Multi-Currency Wallet @@ -7,6 +7,7 @@ * Website: https://cakewallet.com * App Store (iOS / MacOS): https://cakewallet.com/ios * Google Play: https://cakewallet.com/gp +* F-Droid: https://fdroid.cakelabs.com * APK: https://github.com/cake-tech/cake_wallet/releases * Linux: https://github.com/cake-tech/cake_wallet/releases @@ -17,9 +18,8 @@ * Completely noncustodial. *Your keys, your coins.* * Built-in exchange for dozens of pairs * Easily pay cryptocurrency invoices with fixed rate exchanges -* Buy cryptocurrency (BTC/LTC/XMR) with credit/debit/bank +* Buy cryptocurrency (BTC/LTC/XMR/ETH) with credit/debit/bank * Sell cryptocurrency by bank transfer -* Purchase gift cards at a discount using only an email with [Cake Pay](https://cakepay.com), available in-app * Scan QR codes for easy cryptocurrency transfers * Create several wallets * Select your own custom nodes/servers @@ -32,6 +32,7 @@ * Convenient exchange and sending templates for recurring payments * Create donation links and invoices in the receive screen * Robust privacy settings (eg: Tor-only connections) +* Robust security settings (eg: Cake 2FA) ### Monero Specific Features @@ -40,13 +41,19 @@ * Specify restore height for faster syncing * Specify multiple recipients for batch sending * Optionally set Monero nodes as trusted for faster syncing +* Specify a proxy for Monero nodes, compatible with Tor and i2p ### Bitcoin Specific Features * Bitcoin coin control (specify specific outputs to spend) * Automatically generate new addresses * Specify multiple recipients for batch sending -* Sell BTC for USD + +### Ethereum Specific Features + +* Store ETH and all ERc-20 tokens +* Add custom tokens by contract address +* Enable or disable Etherscan for transaction history ### Litecoin Specific Features @@ -69,6 +76,7 @@ * Website: https://monero.com * App Store (iOS): https://apps.apple.com/app/id1601990386 * Google Play: https://play.google.com/store/apps/details?id=com.monero.app +* F-Droid: https://fdroid.cakelabs.com * APK: https://github.com/cake-tech/cake_wallet/releases # Support @@ -123,7 +131,7 @@ Edit the applicable `strings_XX.arb` file in `res/values/` and open a pull reque 2. Edit the strings in this file, replacing XXX below with the translation for each string. -`"welcome" : "Welcome to",` -> `"welcome" : "XXX",` +`"welcome": "Welcome to",` -> `"welcome": "XXX",` 3. For strings where there is a variable, denoted by a $ symbol and braces, such as ${status}, the string in braces should not be translated. For example, when editing line 106: diff --git a/assets/text/Monerocom_Release_Notes.txt b/assets/text/Monerocom_Release_Notes.txt index 403bb9f08..62f649704 100644 --- a/assets/text/Monerocom_Release_Notes.txt +++ b/assets/text/Monerocom_Release_Notes.txt @@ -1,8 +1,2 @@ -Monero background syncing! See https://guides.cakewallet.com/docs/monero/#background-syncing -Cake 2FA access control settings! See https://guides.cakewallet.com/docs/advanced-features/authentication/#cake-2fa-presets-and-access-control-settings -Support Monero node proxy -UI improvements when sending to Address Book entry -Allow renaming Monero account names -Send templates now support multiple recipients (try using to make Monero change) -Onramper improvements -Scan node QR codes (for Umbrel) \ No newline at end of file +Bug fixes +Fiat Onramp improvements \ No newline at end of file diff --git a/assets/text/Release_Notes.txt b/assets/text/Release_Notes.txt index cedec7b7f..62f649704 100644 --- a/assets/text/Release_Notes.txt +++ b/assets/text/Release_Notes.txt @@ -1,9 +1,2 @@ -Ethereum! Store ETH and ERC-20 tokens -Monero background syncing! See https://guides.cakewallet.com/docs/monero/#background-syncing -Cake 2FA access control settings! See https://guides.cakewallet.com/docs/advanced-features/authentication/#cake-2fa-presets-and-access-control-settings -Support Monero node proxy -UI improvements when sending to Address Book entry -Allow renaming Monero/Haven account names -Send templates now support multiple recipients (try using to make Monero change) -Onramper improvements -Scan node QR codes (for Umbrel) \ No newline at end of file +Bug fixes +Fiat Onramp improvements \ No newline at end of file diff --git a/cw_core/lib/erc20_token.dart b/cw_core/lib/erc20_token.dart index 2e205e484..db5b6db5b 100644 --- a/cw_core/lib/erc20_token.dart +++ b/cw_core/lib/erc20_token.dart @@ -57,7 +57,8 @@ class Erc20Token extends CryptoCurrency with HiveObjectMixin { static const boxName = 'Erc20Tokens'; @override - bool operator ==(other) => other is Erc20Token && other.contractAddress == contractAddress; + bool operator ==(other) => (other is Erc20Token && other.contractAddress == contractAddress) || + (other is CryptoCurrency && other.title == title); @override int get hashCode => contractAddress.hashCode; diff --git a/cw_ethereum/lib/default_erc20_tokens.dart b/cw_ethereum/lib/default_erc20_tokens.dart index 241e301ce..8c38e2e64 100644 --- a/cw_ethereum/lib/default_erc20_tokens.dart +++ b/cw_ethereum/lib/default_erc20_tokens.dart @@ -283,6 +283,13 @@ class DefaultErc20Tokens { decimal: 18, enabled: false, ), + Erc20Token( + name: "PayPal USD", + symbol: "PYUSD", + contractAddress: "0x6c3ea9036406852006290770bedfcaba0e23a0e8", + decimal: 6, + enabled: false, + ), ]; List get initialErc20Tokens => _defaultTokens.map((token) { diff --git a/cw_ethereum/lib/ethereum_wallet.dart b/cw_ethereum/lib/ethereum_wallet.dart index cce2687c2..fb3be4d44 100644 --- a/cw_ethereum/lib/ethereum_wallet.dart +++ b/cw_ethereum/lib/ethereum_wallet.dart @@ -165,15 +165,19 @@ abstract class EthereumWalletBase final _credentials = credentials as EthereumTransactionCredentials; final outputs = _credentials.outputs; final hasMultiDestination = outputs.length > 1; - final _erc20Balance = balance[_credentials.currency]!; + + final CryptoCurrency transactionCurrency = + balance.keys.firstWhere((element) => element.title == _credentials.currency.title); + + final _erc20Balance = balance[transactionCurrency]!; BigInt totalAmount = BigInt.zero; - int exponent = - _credentials.currency is Erc20Token ? (_credentials.currency as Erc20Token).decimal : 18; + int exponent = transactionCurrency is Erc20Token ? transactionCurrency.decimal : 18; num amountToEthereumMultiplier = pow(10, exponent); + // so far this can not be made with Ethereum as Ethereum does not support multiple recipients if (hasMultiDestination) { if (outputs.any((item) => item.sendAll || (item.formattedCryptoAmount ?? 0) <= 0)) { - throw EthereumTransactionCreationException(_credentials.currency); + throw EthereumTransactionCreationException(transactionCurrency); } final totalOriginalAmount = EthereumFormatter.parseEthereumAmountToDouble( @@ -181,12 +185,19 @@ abstract class EthereumWalletBase totalAmount = BigInt.from(totalOriginalAmount * amountToEthereumMultiplier); if (_erc20Balance.balance < totalAmount) { - throw EthereumTransactionCreationException(_credentials.currency); + throw EthereumTransactionCreationException(transactionCurrency); } } else { final output = outputs.first; - final BigInt allAmount = - _erc20Balance.balance - BigInt.from(calculateEstimatedFee(_credentials.priority!, null)); + // since the fees are taken from Ethereum + // then no need to subtract the fees from the amount if send all + final BigInt allAmount; + if (transactionCurrency is Erc20Token) { + allAmount = _erc20Balance.balance; + } else { + allAmount = _erc20Balance.balance - + BigInt.from(calculateEstimatedFee(_credentials.priority!, null)); + } final totalOriginalAmount = EthereumFormatter.parseEthereumAmountToDouble(output.formattedCryptoAmount ?? 0); totalAmount = output.sendAll @@ -194,21 +205,22 @@ abstract class EthereumWalletBase : BigInt.from(totalOriginalAmount * amountToEthereumMultiplier); if (_erc20Balance.balance < totalAmount) { - throw EthereumTransactionCreationException(_credentials.currency); + throw EthereumTransactionCreationException(transactionCurrency); } } final pendingEthereumTransaction = await _client.signTransaction( privateKey: _privateKey, - toAddress: _credentials.outputs.first.address, + toAddress: _credentials.outputs.first.isParsedAddress + ? _credentials.outputs.first.extractedAddress! + : _credentials.outputs.first.address, amount: totalAmount.toString(), gas: _estimatedGas!, priority: _credentials.priority!, - currency: _credentials.currency, + currency: transactionCurrency, exponent: exponent, - contractAddress: _credentials.currency is Erc20Token - ? (_credentials.currency as Erc20Token).contractAddress - : null, + contractAddress: + transactionCurrency is Erc20Token ? transactionCurrency.contractAddress : null, ); return pendingEthereumTransaction; diff --git a/cw_ethereum/lib/pending_ethereum_transaction.dart b/cw_ethereum/lib/pending_ethereum_transaction.dart index 23dfa3b87..35b0123cc 100644 --- a/cw_ethereum/lib/pending_ethereum_transaction.dart +++ b/cw_ethereum/lib/pending_ethereum_transaction.dart @@ -20,13 +20,19 @@ class PendingEthereumTransaction with PendingTransaction { }); @override - String get amountFormatted => (BigInt.parse(amount) / BigInt.from(pow(10, exponent))).toString(); + String get amountFormatted { + final _amount = BigInt.parse(amount) / BigInt.from(pow(10, exponent)); + return _amount.toStringAsFixed(min(15, _amount.toString().length)); + } @override Future commit() async => await sendTransaction(); @override - String get feeFormatted => (fee / BigInt.from(pow(10, 18))).toString(); + String get feeFormatted { + final _fee = fee / BigInt.from(pow(10, 18)); + return _fee.toStringAsFixed(min(15, _fee.toString().length)); + } @override String get hex => bytesToHex(signedTransaction, include0x: true); diff --git a/lib/entities/main_actions.dart b/lib/entities/main_actions.dart index ba71f7098..912269d8e 100644 --- a/lib/entities/main_actions.dart +++ b/lib/entities/main_actions.dart @@ -1,6 +1,5 @@ import 'package:cake_wallet/buy/moonpay/moonpay_buy_provider.dart'; import 'package:cake_wallet/buy/onramper/onramper_buy_provider.dart'; -import 'package:cake_wallet/buy/payfura/payfura_buy_provider.dart'; import 'package:cake_wallet/di.dart'; import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/routes.dart'; @@ -48,6 +47,7 @@ class MainActions { case WalletType.bitcoin: case WalletType.litecoin: case WalletType.ethereum: + case WalletType.monero: if (viewModel.isEnabledBuyAction) { final uri = getIt.get().requestUrl(); if (DeviceInfo.instance.isMobile) { @@ -58,16 +58,6 @@ class MainActions { } } break; - case WalletType.monero: - if (viewModel.isEnabledBuyAction) { - if (DeviceInfo.instance.isMobile) { - Navigator.of(context).pushNamed(Routes.payfuraPage); - } else { - final uri = getIt.get().requestUrl(); - await launchUrl(uri); - } - } - break; default: await showPopUp( context: context, diff --git a/lib/entities/parse_address_from_domain.dart b/lib/entities/parse_address_from_domain.dart index 8ac9bb51f..c0eab6d65 100644 --- a/lib/entities/parse_address_from_domain.dart +++ b/lib/entities/parse_address_from_domain.dart @@ -19,14 +19,19 @@ class AddressResolver { 'crypto', 'zil', 'x', - 'coin', 'wallet', 'bitcoin', '888', 'nft', 'dao', 'blockchain', - 'polygon' + 'polygon', + 'klever', + 'hi', + 'kresus', + 'anime', + 'manga', + 'binanceus' ]; static String? extractAddressByType({required String raw, required CryptoCurrency type}) { diff --git a/lib/src/screens/dashboard/dashboard_page.dart b/lib/src/screens/dashboard/dashboard_page.dart index 56b778e08..ff9f85784 100644 --- a/lib/src/screens/dashboard/dashboard_page.dart +++ b/lib/src/screens/dashboard/dashboard_page.dart @@ -43,30 +43,31 @@ class DashboardPage extends StatelessWidget { @override Widget build(BuildContext context) { - return Scaffold(body: LayoutBuilder( - builder: (context, constraints) { - if (DeviceInfo.instance.isDesktop) { - if (constraints.maxWidth > ResponsiveLayoutUtil.kDesktopMaxDashBoardWidthConstraint) { - return getIt.get(); - } else { + return Scaffold( + body: LayoutBuilder( + builder: (context, constraints) { + if (DeviceInfo.instance.isDesktop) { + if (constraints.maxWidth > ResponsiveLayoutUtil.kDesktopMaxDashBoardWidthConstraint) { + return getIt.get(); + } else { + return _DashboardPageView( + balancePage: balancePage, + dashboardViewModel: dashboardViewModel, + addressListViewModel: addressListViewModel, + ); + } + } else if (ResponsiveLayoutUtil.instance.shouldRenderMobileUI()) { return _DashboardPageView( balancePage: balancePage, dashboardViewModel: dashboardViewModel, addressListViewModel: addressListViewModel, ); + } else { + return getIt.get(); } - } else if (ResponsiveLayoutUtil.instance.shouldRenderMobileUI()) { - return _DashboardPageView( - balancePage: balancePage, - dashboardViewModel: dashboardViewModel, - addressListViewModel: addressListViewModel, - ); - } else { - return getIt.get(); - } - - }, - )); + }, + ), + ); } } @@ -89,13 +90,19 @@ class _DashboardPageView extends BasePage { @override Widget Function(BuildContext, Widget) get rootWrapper => (BuildContext context, Widget scaffold) => Container( - decoration: BoxDecoration( - gradient: LinearGradient(colors: [ - Theme.of(context).colorScheme.secondary, - Theme.of(context).scaffoldBackgroundColor, - Theme.of(context).primaryColor, - ], begin: Alignment.topRight, end: Alignment.bottomLeft)), - child: scaffold); + decoration: BoxDecoration( + gradient: LinearGradient( + colors: [ + Theme.of(context).colorScheme.secondary, + Theme.of(context).scaffoldBackgroundColor, + Theme.of(context).primaryColor, + ], + begin: Alignment.topRight, + end: Alignment.bottomLeft, + ), + ), + child: scaffold, + ); @override bool get resizeToAvoidBottomInset => false; @@ -106,29 +113,30 @@ class _DashboardPageView extends BasePage { @override Widget middle(BuildContext context) { return SyncIndicator( - dashboardViewModel: dashboardViewModel, - onTap: () => Navigator.of(context, rootNavigator: true).pushNamed(Routes.connectionSync)); + dashboardViewModel: dashboardViewModel, + onTap: () => Navigator.of(context, rootNavigator: true).pushNamed(Routes.connectionSync), + ); } @override Widget trailing(BuildContext context) { - final menuButton = Image.asset('assets/images/menu.png', - color: Theme.of(context) - .accentTextTheme! - .displayMedium! - .backgroundColor); + final menuButton = Image.asset( + 'assets/images/menu.png', + color: Theme.of(context).accentTextTheme.displayMedium!.backgroundColor, + ); return Container( - alignment: Alignment.centerRight, - width: 40, - child: TextButton( - // FIX-ME: Style - //highlightColor: Colors.transparent, - //splashColor: Colors.transparent, - //padding: EdgeInsets.all(0), - onPressed: () => onOpenEndDrawer(), - child: Semantics( - label: S.of(context).wallet_menu, child: menuButton))); + alignment: Alignment.centerRight, + width: 40, + child: TextButton( + // FIX-ME: Style + //highlightColor: Colors.transparent, + //splashColor: Colors.transparent, + //padding: EdgeInsets.all(0), + onPressed: () => onOpenEndDrawer(), + child: Semantics(label: S.of(context).wallet_menu, child: menuButton), + ), + ); } final DashboardViewModel dashboardViewModel; @@ -142,56 +150,65 @@ class _DashboardPageView extends BasePage { Widget body(BuildContext context) { final controller = PageController(initialPage: initialPage); - reaction((_) => dashboardViewModel.shouldShowMarketPlaceInDashboard, (bool value) { - if (!dashboardViewModel.shouldShowMarketPlaceInDashboard) { - controller.jumpToPage(0); - } - pages.clear(); - _isEffectsInstalled = false; - _setEffects(context); + reaction( + (_) => dashboardViewModel.shouldShowMarketPlaceInDashboard, + (bool value) { + if (!dashboardViewModel.shouldShowMarketPlaceInDashboard) { + controller.jumpToPage(0); + } + pages.clear(); + _isEffectsInstalled = false; + _setEffects(context); - if (value) { - controller.jumpToPage(1); - } else { - controller.jumpToPage(0); - } - }); + if (value) { + controller.jumpToPage(1); + } else { + controller.jumpToPage(0); + } + }, + ); _setEffects(context); return SafeArea( - minimum: EdgeInsets.only(bottom: 24), - child: Column( - mainAxisSize: MainAxisSize.max, - children: [ - Expanded( - child: Observer(builder: (context) { - return PageView.builder( + minimum: EdgeInsets.only(bottom: 24), + child: Column( + mainAxisSize: MainAxisSize.max, + children: [ + Expanded( + child: Observer( + builder: (context) { + return PageView.builder( controller: controller, itemCount: pages.length, - itemBuilder: (context, index) => pages[index]); - })), - Padding( - padding: EdgeInsets.only(bottom: 24, top: 10), - child: Observer(builder: (context) { - return ExcludeSemantics( - child: SmoothPageIndicator( - controller: controller, - count: pages.length, - effect: ColorTransitionEffect( - spacing: 6.0, - radius: 6.0, - dotWidth: 6.0, - dotHeight: 6.0, - dotColor: Theme.of(context).indicatorColor, - activeDotColor: Theme.of(context) - .accentTextTheme! - .headlineMedium! - .backgroundColor!), + itemBuilder: (context, index) => pages[index], + ); + }, + ), + ), + Padding( + padding: EdgeInsets.only(bottom: 24, top: 10), + child: Observer( + builder: (context) { + return ExcludeSemantics( + child: SmoothPageIndicator( + controller: controller, + count: pages.length, + effect: ColorTransitionEffect( + spacing: 6.0, + radius: 6.0, + dotWidth: 6.0, + dotHeight: 6.0, + dotColor: Theme.of(context).indicatorColor, + activeDotColor: + Theme.of(context).accentTextTheme.headlineMedium!.backgroundColor!, ), - ); - } - )), - Observer(builder: (_) { + ), + ); + }, + ), + ), + Observer( + builder: (_) { return ClipRect( child: Container( margin: const EdgeInsets.only(left: 16, right: 16), @@ -204,10 +221,7 @@ class _DashboardPageView extends BasePage { : Colors.transparent, width: 1, ), - color: Theme.of(context) - .textTheme! - .titleLarge! - .backgroundColor!, + color: Theme.of(context).textTheme.titleLarge!.backgroundColor!, ), child: Container( padding: EdgeInsets.only(left: 32, right: 32), @@ -215,48 +229,48 @@ class _DashboardPageView extends BasePage { mainAxisAlignment: MainAxisAlignment.spaceBetween, children: MainActions.all .where((element) => element.canShow?.call(dashboardViewModel) ?? true) - .map((action) => Semantics( - button: true, - enabled: (action.isEnabled - ?.call(dashboardViewModel) ?? - true), - child: ActionButton( - image: Image.asset(action.image, - height: 24, - width: 24, - color: action.isEnabled?.call( - dashboardViewModel) ?? - true - ? Theme.of(context) - .accentTextTheme! - .displayMedium! - .backgroundColor! - : Theme.of(context) - .accentTextTheme! - .displaySmall! - .backgroundColor!), - title: action.name(context), - onClick: () async => await action.onTap( - context, dashboardViewModel), - textColor: action.isEnabled - ?.call(dashboardViewModel) ?? - true - ? null + .map( + (action) => Semantics( + button: true, + enabled: (action.isEnabled?.call(dashboardViewModel) ?? true), + child: ActionButton( + image: Image.asset( + action.image, + height: 24, + width: 24, + color: action.isEnabled?.call(dashboardViewModel) ?? true + ? Theme.of(context) + .accentTextTheme + .displayMedium! + .backgroundColor! : Theme.of(context) - .accentTextTheme! + .accentTextTheme .displaySmall! .backgroundColor!, ), - )) + title: action.name(context), + onClick: () async => + await action.onTap(context, dashboardViewModel), + textColor: action.isEnabled?.call(dashboardViewModel) ?? true + ? null + : Theme.of(context) + .accentTextTheme + .displaySmall! + .backgroundColor!, + ), + ), + ) .toList(), ), ), ), ), ); - }), - ], - )); + }, + ), + ], + ), + ); } void _setEffects(BuildContext context) async { @@ -264,7 +278,8 @@ class _DashboardPageView extends BasePage { return; } if (dashboardViewModel.shouldShowMarketPlaceInDashboard) { - pages.add(Semantics( + pages.add( + Semantics( label: S.of(context).market_place, child: MarketPlacePage( dashboardViewModel: dashboardViewModel, @@ -274,73 +289,92 @@ class _DashboardPageView extends BasePage { ); } pages.add(Semantics(label: S.of(context).balance_page, child: balancePage)); - pages.add(Semantics( + pages.add( + Semantics( label: S.of(context).settings_transactions, - child: TransactionsPage(dashboardViewModel: dashboardViewModel))); + child: TransactionsPage(dashboardViewModel: dashboardViewModel), + ), + ); _isEffectsInstalled = true; - autorun((_) async { - if (!dashboardViewModel.isOutdatedElectrumWallet) { - return; - } + autorun( + (_) async { + if (!dashboardViewModel.isOutdatedElectrumWallet) { + return; + } - await Future.delayed(Duration(seconds: 1)); - if (context.mounted) { - await showPopUp( - context: context, - builder: (BuildContext context) { - return AlertWithOneAction( + await Future.delayed(Duration(seconds: 1)); + if (context.mounted) { + await showPopUp( + context: context, + builder: (BuildContext context) { + return AlertWithOneAction( alertTitle: S.of(context).pre_seed_title, alertContent: S.of(context).outdated_electrum_wallet_description, buttonText: S.of(context).understand, - buttonAction: () => Navigator.of(context).pop()); - }); - } - }); + buttonAction: () => Navigator.of(context).pop(), + ); + }, + ); + } + }, + ); _showReleaseNotesPopup(context); var needToPresentYat = false; var isInactive = false; - _onInactiveSub = rootKey.currentState!.isInactive.listen((inactive) { - isInactive = inactive; + _onInactiveSub = rootKey.currentState!.isInactive.listen( + (inactive) { + isInactive = inactive; - if (needToPresentYat) { - Future.delayed(Duration(milliseconds: 500)).then((_) { - showPopUp( - context: navigatorKey.currentContext!, - builder: (_) => YatEmojiId(dashboardViewModel.yatStore.emoji)); - needToPresentYat = false; - }); - } - }); + if (needToPresentYat) { + Future.delayed(Duration(milliseconds: 500)).then( + (_) { + showPopUp( + context: navigatorKey.currentContext!, + builder: (_) => YatEmojiId(dashboardViewModel.yatStore.emoji), + ); + needToPresentYat = false; + }, + ); + } + }, + ); - dashboardViewModel.yatStore.emojiIncommingStream.listen((String emoji) { - if (!_isEffectsInstalled || emoji.isEmpty) { - return; - } + dashboardViewModel.yatStore.emojiIncommingStream.listen( + (String emoji) { + if (!_isEffectsInstalled || emoji.isEmpty) { + return; + } - needToPresentYat = true; - }); + needToPresentYat = true; + }, + ); } void _showReleaseNotesPopup(BuildContext context) async { final sharedPrefs = await SharedPreferences.getInstance(); final currentAppVersion = - VersionComparator.getExtendedVersionNumber(dashboardViewModel.settingsStore.appVersion); + VersionComparator.getExtendedVersionNumber(dashboardViewModel.settingsStore.appVersion); final lastSeenAppVersion = sharedPrefs.getInt(PreferencesKey.lastSeenAppVersion); final isNewInstall = sharedPrefs.getBool(PreferencesKey.isNewInstall); if (currentAppVersion != lastSeenAppVersion && !isNewInstall!) { - Future.delayed(Duration(seconds: 1), () { - showPopUp( - context: context, - builder: (BuildContext context) { - return ReleaseNotesScreen( - title: 'Version ${dashboardViewModel.settingsStore.appVersion}'); - }); - }); + Future.delayed( + Duration(seconds: 1), + () { + showPopUp( + context: context, + builder: (BuildContext context) { + return ReleaseNotesScreen( + title: 'Version ${dashboardViewModel.settingsStore.appVersion}', + ); + }, + ); + }, + ); sharedPrefs.setInt(PreferencesKey.lastSeenAppVersion, currentAppVersion); } else if (isNewInstall!) { diff --git a/lib/utils/exception_handler.dart b/lib/utils/exception_handler.dart index 6a610120b..a661aaf76 100644 --- a/lib/utils/exception_handler.dart +++ b/lib/utils/exception_handler.dart @@ -145,7 +145,10 @@ class ExceptionHandler { "Connection closed before full header was received", "Connection terminated during handshake", "PERMISSION_NOT_GRANTED", - "Failed host lookup: ", + "Failed host lookup:", + "CERTIFICATE_VERIFY_FAILED", + "Handshake error in client", + "Error while launching http", ]; static Future _addDeviceInfo(File file) async { diff --git a/lib/view_model/exchange/exchange_trade_view_model.dart b/lib/view_model/exchange/exchange_trade_view_model.dart index 254982738..de10b9221 100644 --- a/lib/view_model/exchange/exchange_trade_view_model.dart +++ b/lib/view_model/exchange/exchange_trade_view_model.dart @@ -29,7 +29,9 @@ abstract class ExchangeTradeViewModelBase with Store { required this.sendViewModel}) : trade = tradesStore.trade!, isSendable = tradesStore.trade!.from == wallet.currency || - tradesStore.trade!.provider == ExchangeProviderDescription.xmrto, + tradesStore.trade!.provider == ExchangeProviderDescription.xmrto || + (wallet.currency == CryptoCurrency.eth && + tradesStore.trade!.from.tag == CryptoCurrency.eth.title), items = ObservableList() { switch (trade.provider) { case ExchangeProviderDescription.xmrto: @@ -103,6 +105,7 @@ abstract class ExchangeTradeViewModelBase with Store { final output = sendViewModel.outputs.first; output.address = trade.inputAddress ?? ''; output.setCryptoAmount(trade.amount); + sendViewModel.selectedCryptoCurrency = trade.from; await sendViewModel.createTransaction(); } diff --git a/lib/view_model/restore/restore_from_qr_vm.dart b/lib/view_model/restore/restore_from_qr_vm.dart index 7efb92e69..996b3b3fb 100644 --- a/lib/view_model/restore/restore_from_qr_vm.dart +++ b/lib/view_model/restore/restore_from_qr_vm.dart @@ -1,4 +1,5 @@ import 'package:cake_wallet/bitcoin/bitcoin.dart'; +import 'package:cake_wallet/ethereum/ethereum.dart'; import 'package:cake_wallet/view_model/restore/restore_mode.dart'; import 'package:cake_wallet/view_model/restore/restore_wallet.dart'; import 'package:hive/hive.dart'; @@ -80,6 +81,9 @@ abstract class WalletRestorationFromQRVMBase extends WalletCreationVM with Store case WalletType.litecoin: return bitcoin!.createBitcoinRestoreWalletFromSeedCredentials( name: name, mnemonic: restoreWallet.mnemonicSeed ?? '', password: password); + case WalletType.ethereum: + return ethereum!.createEthereumRestoreWalletFromSeedCredentials( + name: name, mnemonic: restoreWallet.mnemonicSeed ?? '', password: password); default: throw Exception('Unexpected type: ${type.toString()}'); } diff --git a/screenshot-android.jpg b/screenshot-android.jpg deleted file mode 100644 index 9ec562026..000000000 Binary files a/screenshot-android.jpg and /dev/null differ diff --git a/screenshot-ios.jpg b/screenshot-ios.jpg deleted file mode 100644 index 7efa31595..000000000 Binary files a/screenshot-ios.jpg and /dev/null differ diff --git a/scripts/android/app_env.sh b/scripts/android/app_env.sh index 75f3bbd06..741eb8780 100644 --- a/scripts/android/app_env.sh +++ b/scripts/android/app_env.sh @@ -14,14 +14,14 @@ TYPES=($MONERO_COM $CAKEWALLET $HAVEN) APP_ANDROID_TYPE=$1 MONERO_COM_NAME="Monero.com" -MONERO_COM_VERSION="1.5.0" -MONERO_COM_BUILD_NUMBER=54 +MONERO_COM_VERSION="1.5.1" +MONERO_COM_BUILD_NUMBER=55 MONERO_COM_BUNDLE_ID="com.monero.app" MONERO_COM_PACKAGE="com.monero.app" CAKEWALLET_NAME="Cake Wallet" -CAKEWALLET_VERSION="4.8.0" -CAKEWALLET_BUILD_NUMBER=167 +CAKEWALLET_VERSION="4.8.1" +CAKEWALLET_BUILD_NUMBER=168 CAKEWALLET_BUNDLE_ID="com.cakewallet.cake_wallet" CAKEWALLET_PACKAGE="com.cakewallet.cake_wallet" diff --git a/scripts/ios/app_env.sh b/scripts/ios/app_env.sh index 68d3f69a8..401755b40 100644 --- a/scripts/ios/app_env.sh +++ b/scripts/ios/app_env.sh @@ -13,13 +13,13 @@ TYPES=($MONERO_COM $CAKEWALLET $HAVEN) APP_IOS_TYPE=$1 MONERO_COM_NAME="Monero.com" -MONERO_COM_VERSION="1.5.0" -MONERO_COM_BUILD_NUMBER=52 +MONERO_COM_VERSION="1.5.1" +MONERO_COM_BUILD_NUMBER=53 MONERO_COM_BUNDLE_ID="com.cakewallet.monero" CAKEWALLET_NAME="Cake Wallet" -CAKEWALLET_VERSION="4.8.0" -CAKEWALLET_BUILD_NUMBER=175 +CAKEWALLET_VERSION="4.8.1" +CAKEWALLET_BUILD_NUMBER=176 CAKEWALLET_BUNDLE_ID="com.fotolockr.cakewallet" HAVEN_NAME="Haven" diff --git a/scripts/macos/app_env.sh b/scripts/macos/app_env.sh index 387bd39c5..1aed1f4fa 100755 --- a/scripts/macos/app_env.sh +++ b/scripts/macos/app_env.sh @@ -15,8 +15,8 @@ if [ -n "$1" ]; then fi CAKEWALLET_NAME="Cake Wallet" -CAKEWALLET_VERSION="1.1.0" -CAKEWALLET_BUILD_NUMBER=28 +CAKEWALLET_VERSION="1.1.1" +CAKEWALLET_BUILD_NUMBER=29 CAKEWALLET_BUNDLE_ID="com.fotolockr.cakewallet" if ! [[ " ${TYPES[*]} " =~ " ${APP_MACOS_TYPE} " ]]; then