diff --git a/assets/images/2.0x/close.png b/assets/images/2.0x/close.png new file mode 100644 index 000000000..aa65228c8 Binary files /dev/null and b/assets/images/2.0x/close.png differ diff --git a/assets/images/2.0x/copy_address.png b/assets/images/2.0x/copy_address.png new file mode 100644 index 000000000..0efaacf43 Binary files /dev/null and b/assets/images/2.0x/copy_address.png differ diff --git a/assets/images/2.0x/share.png b/assets/images/2.0x/share.png index dc4610066..56f2bf7d1 100644 Binary files a/assets/images/2.0x/share.png and b/assets/images/2.0x/share.png differ diff --git a/assets/images/3.0x/close.png b/assets/images/3.0x/close.png new file mode 100644 index 000000000..5c480a926 Binary files /dev/null and b/assets/images/3.0x/close.png differ diff --git a/assets/images/3.0x/copy_address.png b/assets/images/3.0x/copy_address.png new file mode 100644 index 000000000..f7c198cf3 Binary files /dev/null and b/assets/images/3.0x/copy_address.png differ diff --git a/assets/images/3.0x/share.png b/assets/images/3.0x/share.png index 8958ae5ed..00b325970 100644 Binary files a/assets/images/3.0x/share.png and b/assets/images/3.0x/share.png differ diff --git a/assets/images/close.png b/assets/images/close.png new file mode 100644 index 000000000..eaa8fedde Binary files /dev/null and b/assets/images/close.png differ diff --git a/assets/images/copy_address.png b/assets/images/copy_address.png new file mode 100644 index 000000000..282fec6d5 Binary files /dev/null and b/assets/images/copy_address.png differ diff --git a/assets/images/share.png b/assets/images/share.png index da95e38ba..458d4e373 100644 Binary files a/assets/images/share.png and b/assets/images/share.png differ diff --git a/lib/di.dart b/lib/di.dart index fff785d99..40ae92396 100644 --- a/lib/di.dart +++ b/lib/di.dart @@ -65,6 +65,7 @@ import 'package:cake_wallet/store/templates/send_template_store.dart'; import 'package:cake_wallet/store/templates/exchange_template_store.dart'; import 'package:cake_wallet/src/domain/common/template.dart'; import 'package:cake_wallet/src/domain/exchange/exchange_template.dart'; +import 'package:cake_wallet/store/dashboard/page_view_store.dart'; final getIt = GetIt.instance; @@ -120,6 +121,7 @@ Future setup( TradeFilterStore(wallet: getIt.get().wallet)); getIt.registerSingleton(TransactionFilterStore()); getIt.registerSingleton(FiatConvertationStore()); + getIt.registerSingleton(PageViewStore()); getIt.registerSingleton( SendTemplateStore(templateSource: templates)); getIt.registerSingleton( @@ -167,7 +169,8 @@ Future setup( appStore: getIt.get(), tradesStore: getIt.get(), tradeFilterStore: getIt.get(), - transactionFilterStore: getIt.get() + transactionFilterStore: getIt.get(), + pageViewStore: getIt.get() )); getIt.registerFactory(() => AuthService( @@ -197,7 +200,9 @@ Future setup( closable: false)); getIt.registerFactory( - () => DashboardPage(walletViewModel: getIt.get())); + () => DashboardPage( + walletViewModel: getIt.get(), + addressListViewModel: getIt.get())); getIt.registerFactory(() => ReceivePage( addressListViewModel: getIt.get())); diff --git a/lib/generated/i18n.dart b/lib/generated/i18n.dart index 7dcdbd306..4c9a59474 100644 --- a/lib/generated/i18n.dart +++ b/lib/generated/i18n.dart @@ -165,6 +165,7 @@ class S implements WidgetsLocalizations { String get restore_wallet_restore_description => "Wallet restore description"; String get save => "Save"; String get saved_the_trade_id => "I've saved the trade ID"; + String get scan_qr_code => "Scan the QR code to get the address"; String get seed_choose => "Choose seed language"; String get seed_language_chinese => "Chinese"; String get seed_language_choose => "Please choose seed language:"; @@ -629,6 +630,8 @@ class $de extends S { @override String get xmr_available_balance => "XMR verfügbares Guthaben"; @override + String get scan_qr_code => "Scannen Sie den QR-Code, um die Adresse zu erhalten"; + @override String get trade_state_paid => "Bezahlt"; @override String get node_new => "Neuer Knoten"; @@ -761,6 +764,8 @@ class $de extends S { @override String get error_text_fiat => "Der Wert des Betrags darf den verfügbaren Kontostand nicht überschreiten.\nDie Anzahl der Nachkommastellen muss kleiner oder gleich 2 sein"; @override + String get addresses => "Adressen"; + @override String get transactions_by_date => "Transaktionen nach Datum"; @override String get restore_from_seed_placeholder => "Bitte geben Sie hier Ihren Code ein"; @@ -1257,6 +1262,8 @@ class $hi extends S { @override String get xmr_available_balance => "XMR उपलब्ध शेष राशि"; @override + String get scan_qr_code => "पता प्राप्त करने के लिए QR कोड स्कैन करें"; + @override String get trade_state_paid => "भुगतान किया है"; @override String get node_new => "नया नोड"; @@ -1389,6 +1396,8 @@ class $hi extends S { @override String get error_text_fiat => "राशि का मूल्य उपलब्ध शेष राशि से अधिक नहीं हो सकता.\nअंश अंकों की संख्या कम या 2 के बराबर होनी चाहिए"; @override + String get addresses => "पतों"; + @override String get transactions_by_date => "तारीख से लेन-देन"; @override String get restore_from_seed_placeholder => "कृपया अपना कोड वाक्यांश यहां दर्ज करें या पेस्ट करें"; @@ -1885,6 +1894,8 @@ class $ru extends S { @override String get xmr_available_balance => "Доступный XMR баланс"; @override + String get scan_qr_code => "Отсканируйте QR-код для получения адреса"; + @override String get trade_state_paid => "Оплаченная"; @override String get node_new => "Новая нода"; @@ -2017,6 +2028,8 @@ class $ru extends S { @override String get error_text_fiat => "Значение суммы не может превышать доступный баланс.\nКоличество цифр после запятой должно быть меньше или равно 2"; @override + String get addresses => "Адреса"; + @override String get transactions_by_date => "Сортировать по дате"; @override String get restore_from_seed_placeholder => "Введите или вставьте мнемоническую фразу вашего кошелька"; @@ -2513,6 +2526,8 @@ class $ko extends S { @override String get xmr_available_balance => "XMR 사용 가능한 잔액"; @override + String get scan_qr_code => "QR 코드를 스캔하여 주소를 얻습니다."; + @override String get trade_state_paid => "유료"; @override String get node_new => "새로운 노드"; @@ -2645,6 +2660,8 @@ class $ko extends S { @override String get error_text_fiat => "금액은 사용 가능한 잔액을 초과 할 수 없습니다.\n소수 자릿수는 2보다 작거나 같아야합니다"; @override + String get addresses => "구애"; + @override String get transactions_by_date => "날짜 별 거래"; @override String get restore_from_seed_placeholder => "여기에 코드 문구를 입력하거나 붙여 넣으십시오."; @@ -3141,6 +3158,8 @@ class $pt extends S { @override String get xmr_available_balance => "Saldo XMR disponível"; @override + String get scan_qr_code => "Digitalize o código QR para obter o endereço"; + @override String get trade_state_paid => "Paga"; @override String get node_new => "Novo nó"; @@ -3273,6 +3292,8 @@ class $pt extends S { @override String get error_text_fiat => "O valor do valor não pode exceder o saldo disponível.\nO número de dígitos decimais deve ser menor ou igual a 2"; @override + String get addresses => "Endereços"; + @override String get transactions_by_date => "Transações por data"; @override String get restore_from_seed_placeholder => "Digite ou cole sua frase de código aqui"; @@ -3769,6 +3790,8 @@ class $uk extends S { @override String get xmr_available_balance => "Доступний XMR баланс"; @override + String get scan_qr_code => "Скануйте QR-код для одержання адреси"; + @override String get trade_state_paid => "Оплачена"; @override String get node_new => "Новий вузол"; @@ -3901,6 +3924,8 @@ class $uk extends S { @override String get error_text_fiat => "Значення суми не може перевищувати доступний баланс.\nКількість цифр після коми повинно бути меншим або дорівнювати 2"; @override + String get addresses => "Адреси"; + @override String get transactions_by_date => "Сортувати по даті"; @override String get restore_from_seed_placeholder => "Введіть або вставте мнемонічну фразу вашого гаманця"; @@ -4397,6 +4422,8 @@ class $ja extends S { @override String get xmr_available_balance => "XMR利用可能残高"; @override + String get scan_qr_code => "QRコードをスキャンして住所を取得します"; + @override String get trade_state_paid => "有料"; @override String get node_new => "新しいノード"; @@ -4529,6 +4556,8 @@ class $ja extends S { @override String get error_text_fiat => "金額は利用可能な残高を超えることはできません.\n小数桁の数は2以下でなければなりません"; @override + String get addresses => "住所"; + @override String get transactions_by_date => "日付ごとの取引"; @override String get restore_from_seed_placeholder => "ここにコードフレーズを入力または貼り付けてください"; @@ -5029,6 +5058,8 @@ class $pl extends S { @override String get xmr_available_balance => "XMR Dostępne saldo"; @override + String get scan_qr_code => "Zeskanuj kod QR, aby uzyskać adres"; + @override String get trade_state_paid => "Płatny"; @override String get node_new => "Nowy węzeł"; @@ -5161,6 +5192,8 @@ class $pl extends S { @override String get error_text_fiat => "Wartość kwoty nie może przekroczyć dostępnego salda.\nLiczba cyfr ułamkowych musi być mniejsza lub równa 2"; @override + String get addresses => "Adresy"; + @override String get transactions_by_date => "Transakcje według daty"; @override String get restore_from_seed_placeholder => "Wpisz lub wklej tutaj swoją frazę kodową"; @@ -5657,6 +5690,8 @@ class $es extends S { @override String get xmr_available_balance => "XMR Available Balance"; @override + String get scan_qr_code => "Escanee el código QR para obtener la dirección"; + @override String get trade_state_paid => "Pagado"; @override String get node_new => "Nuevo nodo"; @@ -5789,6 +5824,8 @@ class $es extends S { @override String get error_text_fiat => "El valor de la cantidad no puede exceder el saldo disponible.\nEl número de dígitos de fracción debe ser menor o igual a 2"; @override + String get addresses => "Direcciones"; + @override String get transactions_by_date => "Transacciones por fecha"; @override String get restore_from_seed_placeholder => "Ingrese o pegue su frase de código aquí"; @@ -6285,6 +6322,8 @@ class $nl extends S { @override String get xmr_available_balance => "XMR Beschikbaar saldo"; @override + String get scan_qr_code => "Scan de QR-code om het adres te krijgen"; + @override String get trade_state_paid => "Betaald"; @override String get node_new => "Nieuw knooppunt"; @@ -6417,6 +6456,8 @@ class $nl extends S { @override String get error_text_fiat => "Waarde van bedrag kan het beschikbare saldo niet overschrijden.\nHet aantal breukcijfers moet kleiner zijn dan of gelijk zijn aan 2"; @override + String get addresses => "Adressen"; + @override String get transactions_by_date => "Transacties op datum"; @override String get restore_from_seed_placeholder => "Voer hier uw codefrase in of plak deze"; @@ -6913,6 +6954,8 @@ class $zh extends S { @override String get xmr_available_balance => "XMR 可用余额 "; @override + String get scan_qr_code => "掃描二維碼獲取地址"; + @override String get trade_state_paid => "已付费"; @override String get node_new => "新节点"; @@ -7045,6 +7088,8 @@ class $zh extends S { @override String get error_text_fiat => "金额不能超过可用余额.\n小数位数必须小于或等于2"; @override + String get addresses => "地址"; + @override String get transactions_by_date => "按日期交易"; @override String get restore_from_seed_placeholder => "请在此处输入或粘贴您的代码短语"; diff --git a/lib/palette.dart b/lib/palette.dart index f5f3becc1..0a562718b 100644 --- a/lib/palette.dart +++ b/lib/palette.dart @@ -19,7 +19,7 @@ class Palette { } class PaletteDark { - static const Color distantBlue = Color.fromRGBO(70, 85, 133, 1.0); // mainBackgroundColor + //static const Color distantBlue = Color.fromRGBO(70, 85, 133, 1.0); // mainBackgroundColor static const Color lightDistantBlue = Color.fromRGBO(81, 96, 147, 1.0); // borderCardColor static const Color gray = Color.fromRGBO(140, 153, 201, 1.0); // walletCardText static const Color violetBlue = Color.fromRGBO(51, 63, 104, 1.0); // walletCardAddressField @@ -35,11 +35,13 @@ class PaletteDark { static const Color backgroundColor = Color.fromRGBO(25, 35, 60, 1.0); static const Color nightBlue = Color.fromRGBO(35, 47, 79, 1.0); static const Color wildNightBlue = Color.fromRGBO(39, 53, 96, 1.0); + static const Color distantNightBlue = Color.fromRGBO(46, 57, 96, 1.0); static const Color cyanBlue = Color.fromRGBO(99, 113, 150, 1.0); static const Color darkCyanBlue = Color.fromRGBO(91, 112, 146, 1.0); static const Color orangeYellow = Color.fromRGBO(243, 166, 50, 1.0); static const Color brightGreen = Color.fromRGBO(88, 243, 50, 1.0); static const Color oceanBlue = Color.fromRGBO(27, 39, 71, 1.0); + static const Color lightOceanBlue = Color.fromRGBO(32, 45, 80, 1.0); static const Color lightNightBlue = Color.fromRGBO(39, 52, 89, 1.0); static const Color wildBlue = Color.fromRGBO(165, 176, 205, 1.0); static const Color buttonNightBlue = Color.fromRGBO(46, 57, 96, 1.0); @@ -52,6 +54,13 @@ class PaletteDark { static const Color moderateBlue = Color.fromRGBO(60, 73, 118, 1.0); static const Color deepPurpleBlue = Color.fromRGBO(19, 29, 56, 1.0); static const Color lightOceanBlue = Color.fromRGBO(30, 42, 73, 1.0); + static const Color lightBlueGrey = Color.fromRGBO(125, 141, 183, 1.0); + static const Color wildBlueGrey = Color.fromRGBO(125, 137, 182, 1.0); + static const Color darkGrey = Color.fromRGBO(118, 131, 169, 1.0); + static const Color dividerColor = Color.fromRGBO(48, 59, 95, 1.0); + static const Color violetBlue = Color.fromRGBO(59, 72, 119, 1.0); + static const Color deepPurpleBlue = Color.fromRGBO(19, 29, 56, 1.0); + static const Color distantBlue = Color.fromRGBO(72, 85, 131, 1.0); // FIXME: Rename. static const Color eee = Color.fromRGBO(236, 239, 245, 1.0); diff --git a/lib/src/screens/dashboard/dashboard_page.dart b/lib/src/screens/dashboard/dashboard_page.dart index 5b7939c92..7623891f4 100644 --- a/lib/src/screens/dashboard/dashboard_page.dart +++ b/lib/src/screens/dashboard/dashboard_page.dart @@ -9,13 +9,18 @@ import 'package:cake_wallet/palette.dart'; import 'package:dots_indicator/dots_indicator.dart'; import 'package:cake_wallet/src/screens/dashboard/widgets/action_button.dart'; import 'package:cake_wallet/src/screens/dashboard/widgets/balance_page.dart'; +import 'package:cake_wallet/src/screens/dashboard/widgets/address_page.dart'; import 'package:cake_wallet/src/screens/dashboard/widgets/transactions_page.dart'; import 'package:flutter_mobx/flutter_mobx.dart'; import 'package:mobx/mobx.dart'; import 'package:cake_wallet/src/screens/dashboard/widgets/sync_indicator.dart'; +import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_view_model.dart'; class DashboardPage extends BasePage { - DashboardPage({@required this.walletViewModel}); + DashboardPage({ + @required this.walletViewModel, + @required this.addressListViewModel, + }); @override Color get backgroundLightColor => PaletteDark.backgroundColor; @@ -23,6 +28,9 @@ class DashboardPage extends BasePage { @override Color get backgroundDarkColor => PaletteDark.backgroundColor; + @override + bool get resizeToAvoidBottomPadding => false; + @override Widget get endDrawer => MenuWidget( name: walletViewModel.name, @@ -53,13 +61,14 @@ class DashboardPage extends BasePage { } final DashboardViewModel walletViewModel; + final WalletAddressListViewModel addressListViewModel; final sendImage = Image.asset('assets/images/upload.png', height: 22.24, width: 24, color: Colors.white); final exchangeImage = Image.asset('assets/images/transfer.png', height: 24.27, width: 22.25, color: Colors.white); final receiveImage = Image.asset('assets/images/download.png', height: 22.24, width: 24, color: Colors.white); - final controller = PageController(initialPage: 0); + final controller = PageController(initialPage: 1); var pages = []; bool _isEffectsInstalled = false; @@ -146,11 +155,12 @@ class DashboardPage extends BasePage { return; } + pages.add(AddressPage(addressListViewModel: addressListViewModel)); pages.add(BalancePage(dashboardViewModel: walletViewModel)); pages.add(TransactionsPage(dashboardViewModel: walletViewModel)); controller.addListener(() { - walletViewModel.currentPage = controller.page; + walletViewModel.pageViewStore.setCurrentPage(controller.page); }); reaction((_) => walletViewModel.currentPage, (double currentPage) { diff --git a/lib/src/screens/dashboard/widgets/address_page.dart b/lib/src/screens/dashboard/widgets/address_page.dart new file mode 100644 index 000000000..35134d51b --- /dev/null +++ b/lib/src/screens/dashboard/widgets/address_page.dart @@ -0,0 +1,59 @@ +import 'package:flutter/material.dart'; +import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_view_model.dart'; +import 'package:cake_wallet/palette.dart'; +import 'package:cake_wallet/src/screens/receive/widgets/qr_widget.dart'; +import 'package:cake_wallet/routes.dart'; +import 'package:cake_wallet/generated/i18n.dart'; + +class AddressPage extends StatelessWidget { + AddressPage({@required this.addressListViewModel}); + + final WalletAddressListViewModel addressListViewModel; + + @override + Widget build(BuildContext context) { + return Container( + padding: EdgeInsets.fromLTRB(24, 24, 24, 32), + child: Column( + children: [ + Expanded( + child: Center( + child: QRWidget(addressListViewModel: addressListViewModel), + ) + ), + GestureDetector( + onTap: () => Navigator.of(context).pushNamed(Routes.receive), + child: Container( + height: 50, + padding: EdgeInsets.only(left: 24, right: 12), + alignment: Alignment.center, + decoration: BoxDecoration( + borderRadius: BorderRadius.all(Radius.circular(25)), + color: PaletteDark.nightBlue + ), + child: Row( + mainAxisSize: MainAxisSize.max, + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + S.of(context).addresses, + style: TextStyle( + fontSize: 14, + fontWeight: FontWeight.w500, + color: Colors.white + ), + ), + Icon( + Icons.arrow_forward_ios, + size: 14, + color: Colors.white, + ) + ], + ), + ), + ) + ], + ), + ); + } +} \ No newline at end of file diff --git a/lib/src/screens/monero_accounts/monero_account_edit_or_create_page.dart b/lib/src/screens/monero_accounts/monero_account_edit_or_create_page.dart index 96ab7fba7..7cd1b200a 100644 --- a/lib/src/screens/monero_accounts/monero_account_edit_or_create_page.dart +++ b/lib/src/screens/monero_accounts/monero_account_edit_or_create_page.dart @@ -8,6 +8,7 @@ import 'package:cake_wallet/view_model/monero_account_list/monero_account_edit_o import 'package:cake_wallet/src/widgets/primary_button.dart'; import 'package:cake_wallet/src/screens/base_page.dart'; import 'package:cake_wallet/src/widgets/base_text_form_field.dart'; +import 'package:cake_wallet/palette.dart'; class MoneroAccountEditOrCreatePage extends BasePage { MoneroAccountEditOrCreatePage({@required this.moneroAccountCreationViewModel}) @@ -23,6 +24,12 @@ class MoneroAccountEditOrCreatePage extends BasePage { @override String get title => S.current.account; + @override + Color get backgroundLightColor => PaletteDark.backgroundColor; + + @override + Color get backgroundDarkColor => PaletteDark.backgroundColor; + final GlobalKey _formKey; final TextEditingController _textController; @@ -31,7 +38,6 @@ class MoneroAccountEditOrCreatePage extends BasePage { Form( key: _formKey, child: Container( - color: Theme.of(context).backgroundColor, padding: EdgeInsets.all(24.0), child: Column( children: [ @@ -39,6 +45,7 @@ class MoneroAccountEditOrCreatePage extends BasePage { child: Center( child: BaseTextFormField( controller: _textController, + textColor: Colors.white, hintText: S.of(context).account, validator: MoneroLabelValidator(), ))), diff --git a/lib/src/screens/monero_accounts/monero_account_list_page.dart b/lib/src/screens/monero_accounts/monero_account_list_page.dart index 74792e4ff..c59e566d1 100644 --- a/lib/src/screens/monero_accounts/monero_account_list_page.dart +++ b/lib/src/screens/monero_accounts/monero_account_list_page.dart @@ -7,133 +7,157 @@ import 'package:cake_wallet/routes.dart'; import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/view_model/monero_account_list/monero_account_list_view_model.dart'; import 'package:cake_wallet/src/screens/monero_accounts/widgets/account_tile.dart'; +import 'package:cake_wallet/src/widgets/alert_background.dart'; +import 'package:cake_wallet/src/widgets/cake_scrollbar.dart'; +import 'package:cake_wallet/src/widgets/alert_close_button.dart'; -class MoneroAccountListPage extends StatefulWidget { - MoneroAccountListPage({@required this.accountListViewModel}); +class MoneroAccountListPage extends StatelessWidget { + MoneroAccountListPage({@required this.accountListViewModel}) { + backgroundHeight = 194; + thumbHeight = 72; + isAlwaysShowScrollThumb = false; + controller = ScrollController(); + + controller.addListener(() { + final scrollOffsetFromTop = controller.hasClients + ? (controller.offset / controller.position.maxScrollExtent * (backgroundHeight - thumbHeight)) + : 0.0; + accountListViewModel.setScrollOffsetFromTop(scrollOffsetFromTop); + }); + } final MoneroAccountListViewModel accountListViewModel; + final closeIcon = Image.asset('assets/images/close.png'); - @override - MoneroAccountListPageForm createState() => - MoneroAccountListPageForm(accountListViewModel); -} - -class MoneroAccountListPageForm extends State { - MoneroAccountListPageForm(this.accountListViewModel); - - final MoneroAccountListViewModel accountListViewModel; + ScrollController controller; + double backgroundHeight; + double thumbHeight; + bool isAlwaysShowScrollThumb; @override Widget build(BuildContext context) { - return GestureDetector( - onTap: () => Navigator.of(context).pop(), - child: Container( - color: Colors.transparent, - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 3.0, sigmaY: 3.0), - child: Container( - decoration: BoxDecoration( - color: PaletteDark.darkNightBlue.withOpacity(0.75)), - child: Center( - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - Container( - padding: EdgeInsets.only(left: 24, right: 24), - child: Text( - S.of(context).choose_account, - textAlign: TextAlign.center, - style: TextStyle( - fontSize: 18, - fontWeight: FontWeight.bold, - decoration: TextDecoration.none, - color: Colors.white), - ), + return AlertBackground( + child: Stack( + alignment: Alignment.center, + children: [ + Column( + mainAxisSize: MainAxisSize.min, + children: [ + Container( + padding: EdgeInsets.only(left: 24, right: 24), + child: Text( + S.of(context).choose_account, + textAlign: TextAlign.center, + style: TextStyle( + fontSize: 18, + fontWeight: FontWeight.bold, + decoration: TextDecoration.none, + color: Colors.white ), - Padding( - padding: EdgeInsets.only(left: 24, right: 24, top: 24), - child: GestureDetector( - onTap: () => null, - child: ClipRRect( - borderRadius: BorderRadius.all(Radius.circular(14)), - child: Container( - height: 296, - color: Theme.of(context) - .accentTextTheme - .title - .backgroundColor, - child: Column( - children: [ - Expanded(child: Observer(builder: (_) { - final accounts = - widget.accountListViewModel.accounts; + ), + ), + Padding( + padding: EdgeInsets.only(left: 24, right: 24, top: 24), + child: GestureDetector( + onTap: () => null, + child: ClipRRect( + borderRadius: BorderRadius.all(Radius.circular(14)), + child: Container( + height: 296, + color: PaletteDark.deepPurpleBlue, + child: Column( + children: [ + Expanded( + child: Observer( + builder: (_) { + final accounts = accountListViewModel.accounts; + isAlwaysShowScrollThumb = accounts == null + ? false + : accounts.length > 3; - return ListView.separated( - separatorBuilder: (context, index) => Divider( - color: Theme.of(context).dividerColor, - height: 1), - itemCount: accounts.length ?? 0, - itemBuilder: (context, index) { - final account = accounts[index]; - - return AccountTile( - isCurrent: account.isSelected, - accountName: account.label, - onTap: () { - if (account.isSelected) { - return; - } - - widget.accountListViewModel - .select(account); - Navigator.of(context).pop(); - }); - }, - ); - })), - GestureDetector( - onTap: () async => await Navigator.of(context) - .pushNamed(Routes.accountCreation), - child: Container( - height: 62, - color: Colors.white, - padding: EdgeInsets.only(left: 24, right: 24), - child: Center( - child: Row( - mainAxisSize: MainAxisSize.min, + return Stack( + alignment: Alignment.center, children: [ - Icon( - Icons.add, - color: PaletteDark.darkNightBlue, - ), - Padding( - padding: EdgeInsets.only(left: 5), - child: Text( - S.of(context).create_new_account, - style: TextStyle( - fontSize: 15, - fontWeight: FontWeight.w600, - color: PaletteDark.darkNightBlue, - decoration: TextDecoration.none, - ), + ListView.separated( + controller: controller, + separatorBuilder: (context, index) => + Container( + height: 1, + color: PaletteDark.dividerColor, ), + itemCount: accounts.length ?? 0, + itemBuilder: (context, index) { + final account = accounts[index]; + + return AccountTile( + isCurrent: account.isSelected, + accountName: account.label, + onTap: () { + if (account.isSelected) { + return; + } + + accountListViewModel + .select(account); + Navigator.of(context).pop(); + }); + }, + ), + isAlwaysShowScrollThumb + ? CakeScrollbar( + backgroundHeight: backgroundHeight, + thumbHeight: thumbHeight, + fromTop: accountListViewModel + .scrollOffsetFromTop ) + : Offstage(), ], - ), - ), - ), + ); + } ) - ], ), - ), + GestureDetector( + onTap: () async => await Navigator.of(context) + .pushNamed(Routes.accountCreation), + child: Container( + height: 62, + color: Colors.white, + padding: EdgeInsets.only(left: 24, right: 24), + child: Center( + child: Row( + mainAxisSize: MainAxisSize.min, + children: [ + Icon( + Icons.add, + color: PaletteDark.darkNightBlue, + ), + Padding( + padding: EdgeInsets.only(left: 5), + child: Text( + S.of(context).create_new_account, + style: TextStyle( + fontSize: 15, + fontWeight: FontWeight.w600, + color: PaletteDark.darkNightBlue, + decoration: TextDecoration.none, + ), + ), + ) + ], + ), + ), + ), + ) + ], ), ), - ) - ], - ), - ), + ), + ), + ) + ], ), - ), + AlertCloseButton(image: closeIcon) + ], ), ); } diff --git a/lib/src/screens/monero_accounts/widgets/account_tile.dart b/lib/src/screens/monero_accounts/widgets/account_tile.dart index 84b965e03..a9f131f1c 100644 --- a/lib/src/screens/monero_accounts/widgets/account_tile.dart +++ b/lib/src/screens/monero_accounts/widgets/account_tile.dart @@ -1,4 +1,5 @@ import 'package:flutter/material.dart'; +import 'package:cake_wallet/palette.dart'; class AccountTile extends StatelessWidget { AccountTile({ @@ -13,8 +14,8 @@ class AccountTile extends StatelessWidget { @override Widget build(BuildContext context) { - final color = isCurrent ? Theme.of(context).accentTextTheme.subtitle.decorationColor : Colors.transparent; - final textColor = isCurrent ? Colors.blue : Theme.of(context).primaryTextTheme.title.color; + final color = isCurrent ? PaletteDark.lightOceanBlue : Colors.transparent; + final textColor = isCurrent ? Colors.blue : Colors.white; return GestureDetector( onTap: onTap, diff --git a/lib/src/screens/receive/receive_page.dart b/lib/src/screens/receive/receive_page.dart index 47608b59c..21e8e3b5e 100644 --- a/lib/src/screens/receive/receive_page.dart +++ b/lib/src/screens/receive/receive_page.dart @@ -1,199 +1,72 @@ import 'package:flutter/material.dart'; import 'package:flutter/cupertino.dart'; -import 'package:flutter/services.dart'; import 'package:flutter_mobx/flutter_mobx.dart'; import 'package:esys_flutter_share/esys_flutter_share.dart'; import 'package:cake_wallet/routes.dart'; import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/di.dart'; import 'package:cake_wallet/src/screens/base_page.dart'; -import 'package:cake_wallet/src/screens/receive/widgets/qr_image.dart'; import 'package:cake_wallet/src/screens/monero_accounts/monero_account_list_page.dart'; import 'package:cake_wallet/src/screens/receive/widgets/header_tile.dart'; -import 'package:cake_wallet/src/widgets/base_text_form_field.dart'; -import 'package:cake_wallet/core/amount_validator.dart'; import 'package:cake_wallet/src/screens/receive/widgets/address_cell.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_account_list_header.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_header.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_item.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_view_model.dart'; +import 'package:cake_wallet/src/screens/receive/widgets/qr_widget.dart'; +import 'package:cake_wallet/palette.dart'; class ReceivePage extends BasePage { - ReceivePage({this.addressListViewModel}) - : amountController = TextEditingController(), - _formKey = GlobalKey() { - amountController.addListener(() => addressListViewModel.amount = - _formKey.currentState.validate() ? amountController.text : ''); - } + ReceivePage({this.addressListViewModel}); final WalletAddressListViewModel addressListViewModel; - final TextEditingController amountController; - final GlobalKey _formKey; @override - Color get backgroundLightColor => Colors.transparent; + String get title => S.current.receive; @override - Color get backgroundDarkColor => Colors.transparent; + Color get backgroundLightColor => PaletteDark.backgroundColor; @override - Widget Function(BuildContext, Widget) get rootWrapper => - (BuildContext context, Widget scaffold) => Container( - decoration: BoxDecoration( - gradient: LinearGradient(colors: [ - Theme.of(context).scaffoldBackgroundColor, - Theme.of(context).primaryColor - ], begin: Alignment.topLeft, end: Alignment.bottomRight)), - child: scaffold); - - @override - Widget middle(BuildContext context) => Text( - S.of(context).receive, - style: TextStyle( - fontSize: 18.0, - fontWeight: FontWeight.bold, - color: Theme.of(context).primaryTextTheme.title.color), - ); + Color get backgroundDarkColor => PaletteDark.backgroundColor; @override Widget trailing(BuildContext context) { final shareImage = Image.asset('assets/images/share.png', - color: Theme.of(context).primaryTextTheme.title.color); + color: Colors.white); return SizedBox( height: 20.0, - width: 14.0, + width: 20.0, child: ButtonTheme( minWidth: double.minPositive, child: FlatButton( - highlightColor: Colors.transparent, - splashColor: Colors.transparent, - padding: EdgeInsets.all(0), - onPressed: () => Share.text(S.current.share_address, - addressListViewModel.address.address, 'text/plain'), - child: shareImage), + highlightColor: Colors.transparent, + splashColor: Colors.transparent, + padding: EdgeInsets.all(0), + onPressed: () => Share.text(S.current.share_address, + addressListViewModel.address.address, 'text/plain'), + child: shareImage), ), ); } @override Widget body(BuildContext context) { - final copyImage = Image.asset('assets/images/copy_content.png', - color: Theme.of(context).primaryTextTheme.title.color); - return SingleChildScrollView( child: Column( children: [ - SizedBox(height: 25), - Row(children: [ - Spacer(flex: 4), - Observer( - builder: (_) => Flexible( - flex: 6, - child: Center( - child: AspectRatio( - aspectRatio: 1.0, - child: QrImage( - data: addressListViewModel.uri.toString(), - backgroundColor: Colors.transparent, - foregroundColor: Theme.of(context) - .primaryTextTheme - .display4 - .color, - ))))), - Spacer(flex: 4) - ]), Padding( - padding: EdgeInsets.fromLTRB(24, 40, 24, 0), - child: Row( - children: [ - Expanded( - child: Form( - key: _formKey, - child: BaseTextFormField( - controller: amountController, - keyboardType: - TextInputType.numberWithOptions(decimal: true), - inputFormatters: [ - BlacklistingTextInputFormatter( - RegExp('[\\-|\\ |\\,]')) - ], - textAlign: TextAlign.center, - hintText: S.of(context).receive_amount, - borderColor: Theme.of(context) - .primaryTextTheme - .headline5 - .color - .withOpacity(0.4), - validator: AmountValidator(isAutovalidate: true), - autovalidate: true, - placeholderTextStyle: TextStyle( - color: Theme.of(context) - .primaryTextTheme - .headline5 - .color, - fontSize: 20, - fontWeight: FontWeight.w600)))) - ], + padding: EdgeInsets.fromLTRB(24, 80, 24, 40), + child: QRWidget( + addressListViewModel: addressListViewModel, + isAmountFieldShow: true, ), ), - Padding( - padding: EdgeInsets.only(left: 24, right: 24, bottom: 40, top: 40), - child: Builder( - builder: (context) => Observer( - builder: (context) => GestureDetector( - onTap: () { - Clipboard.setData(ClipboardData( - text: addressListViewModel.address.address)); - Scaffold.of(context).showSnackBar(SnackBar( - content: Text( - S.of(context).copied_to_clipboard, - style: TextStyle(color: Colors.white), - ), - backgroundColor: Colors.green, - duration: Duration(milliseconds: 500), - )); - }, - child: Container( - height: 48, - padding: EdgeInsets.only(left: 24, right: 24), - decoration: BoxDecoration( - borderRadius: - BorderRadius.all(Radius.circular(24)), - color: Theme.of(context) - .primaryTextTheme - .overline - .color), - child: Row( - mainAxisSize: MainAxisSize.max, - children: [ - Expanded( - child: Text( - addressListViewModel.address.address, - maxLines: 1, - overflow: TextOverflow.ellipsis, - style: TextStyle( - fontSize: 18, - fontWeight: FontWeight.w500, - color: Theme.of(context) - .primaryTextTheme - .title - .color), - ), - ), - Padding( - padding: EdgeInsets.only(left: 12), - child: copyImage, - ) - ], - ), - ), - ))), - ), Observer( builder: (_) => ListView.separated( separatorBuilder: (context, _) => - Divider(height: 1, color: Theme.of(context).dividerColor), + Container(height: 1, color: PaletteDark.dividerColor), shrinkWrap: true, physics: NeverScrollableScrollPhysics(), itemCount: addressListViewModel.items.length, @@ -210,8 +83,7 @@ class ReceivePage extends BasePage { icon: Icon( Icons.arrow_forward_ios, size: 14, - color: - Theme.of(context).primaryTextTheme.title.color, + color: Colors.white, )); } @@ -223,28 +95,40 @@ class ReceivePage extends BasePage { icon: Icon( Icons.add, size: 20, - color: - Theme.of(context).primaryTextTheme.title.color, + color: Colors.white, )); } if (item is WalletAddressListItem) { cell = Observer( - builder: (_) => AddressCell.fromItem(item, - isCurrent: item.address == - addressListViewModel.address.address, - onTap: (_) => addressListViewModel.address = item, - onEdit: () => Navigator.of(context).pushNamed( - Routes.newSubaddress, - arguments: item))); + builder: (_) { + final isCurrent = item.address == + addressListViewModel.address.address; + final backgroundColor = isCurrent + ? PaletteDark.lightOceanBlue + : PaletteDark.nightBlue; + final textColor = isCurrent + ? Colors.blue + : Colors.white; + + return AddressCell.fromItem(item, + isCurrent: isCurrent, + backgroundColor: backgroundColor, + textColor: textColor, + onTap: (_) => addressListViewModel.address = item, + onEdit: () => Navigator.of(context).pushNamed( + Routes.newSubaddress, + arguments: item)); + } + ); } return index != 0 ? cell : ClipRRect( borderRadius: BorderRadius.only( - topLeft: Radius.circular(24), - topRight: Radius.circular(24)), + topLeft: Radius.circular(30), + topRight: Radius.circular(30)), child: cell, ); })), diff --git a/lib/src/screens/receive/widgets/address_cell.dart b/lib/src/screens/receive/widgets/address_cell.dart index 5e725583b..153a3f51e 100644 --- a/lib/src/screens/receive/widgets/address_cell.dart +++ b/lib/src/screens/receive/widgets/address_cell.dart @@ -6,12 +6,16 @@ import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_i class AddressCell extends StatelessWidget { factory AddressCell.fromItem(WalletAddressListItem item, {@required bool isCurrent, + @required Color backgroundColor, + @required Color textColor, Function(String) onTap, Function() onEdit}) => AddressCell( address: item.address, name: item.name, isCurrent: isCurrent, + backgroundColor: backgroundColor, + textColor: textColor, onTap: onTap, onEdit: onEdit); @@ -19,12 +23,16 @@ class AddressCell extends StatelessWidget { {@required this.address, @required this.name, @required this.isCurrent, + @required this.backgroundColor, + @required this.textColor, this.onTap, this.onEdit}); final String address; final String name; final bool isCurrent; + final Color backgroundColor; + final Color textColor; final Function(String) onTap; final Function() onEdit; @@ -32,23 +40,19 @@ class AddressCell extends StatelessWidget { @override Widget build(BuildContext context) { - const currentTextColor = Colors.blue; // FIXME: Why it's defined here ? - final currentColor = - Theme.of(context).accentTextTheme.subtitle.decorationColor; - final notCurrentColor = Theme.of(context).backgroundColor; - final notCurrentTextColor = - Theme.of(context).primaryTextTheme.caption.color; final Widget cell = InkWell( onTap: () => onTap(address), child: Container( - color: isCurrent ? currentColor : notCurrentColor, + color: backgroundColor, padding: EdgeInsets.only(left: 24, right: 24, top: 28, bottom: 28), child: Text( name ?? address, + maxLines: 1, + overflow: TextOverflow.ellipsis, style: TextStyle( - fontSize: name?.isNotEmpty ?? false ? 18 : 10, + fontSize: 18, fontWeight: FontWeight.w600, - color: isCurrent ? currentTextColor : notCurrentTextColor, + color: textColor, ), ), )); @@ -62,7 +66,7 @@ class AddressCell extends StatelessWidget { secondaryActions: [ IconSlideAction( caption: S.of(context).edit, - color: Theme.of(context).primaryTextTheme.overline.color, + color: Colors.blue, icon: Icons.edit, onTap: () => onEdit?.call()) ]); diff --git a/lib/src/screens/receive/widgets/header_tile.dart b/lib/src/screens/receive/widgets/header_tile.dart index 84ed2b526..effb7349e 100644 --- a/lib/src/screens/receive/widgets/header_tile.dart +++ b/lib/src/screens/receive/widgets/header_tile.dart @@ -23,7 +23,7 @@ class HeaderTile extends StatelessWidget { top: 24, bottom: 24 ), - color: Theme.of(context).backgroundColor, + color: PaletteDark.nightBlue, child: Row( mainAxisSize: MainAxisSize.max, mainAxisAlignment: MainAxisAlignment.spaceBetween, @@ -33,7 +33,7 @@ class HeaderTile extends StatelessWidget { style: TextStyle( fontSize: 18, fontWeight: FontWeight.w500, - color: Theme.of(context).primaryTextTheme.title.color + color: Colors.white ), ), Container( @@ -41,7 +41,7 @@ class HeaderTile extends StatelessWidget { width: 32, decoration: BoxDecoration( shape: BoxShape.circle, - color: Theme.of(context).primaryTextTheme.overline.color + color: PaletteDark.distantNightBlue ), child: icon, ) diff --git a/lib/src/screens/receive/widgets/qr_widget.dart b/lib/src/screens/receive/widgets/qr_widget.dart new file mode 100644 index 000000000..c8fd78b5f --- /dev/null +++ b/lib/src/screens/receive/widgets/qr_widget.dart @@ -0,0 +1,137 @@ +import 'package:flutter/material.dart'; +import 'package:flutter/cupertino.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_mobx/flutter_mobx.dart'; +import 'package:cake_wallet/generated/i18n.dart'; +import 'package:cake_wallet/src/screens/receive/widgets/qr_image.dart'; +import 'package:cake_wallet/src/widgets/base_text_form_field.dart'; +import 'package:cake_wallet/core/amount_validator.dart'; +import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_view_model.dart'; +import 'package:cake_wallet/palette.dart'; + +class QRWidget extends StatelessWidget { + QRWidget({ + @required this.addressListViewModel, + this.isAmountFieldShow = false + }) : amountController = TextEditingController(), + _formKey = GlobalKey() { + amountController.addListener(() => addressListViewModel.amount = + _formKey.currentState.validate() ? amountController.text : ''); + } + + final WalletAddressListViewModel addressListViewModel; + final bool isAmountFieldShow; + final TextEditingController amountController; + final GlobalKey _formKey; + + @override + Widget build(BuildContext context) { + final copyImage = Image.asset('assets/images/copy_address.png', + color: PaletteDark.lightBlueGrey); + final addressTopOffset = isAmountFieldShow ? 60.0 : 40.0; + + return Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Row(children: [ + Spacer(flex: 3), + Observer( + builder: (_) => Flexible( + flex: 5, + child: Center( + child: AspectRatio( + aspectRatio: 1.0, + child: QrImage( + data: addressListViewModel.uri.toString(), + backgroundColor: Colors.transparent, + foregroundColor: PaletteDark.lightBlueGrey, + ))))), + Spacer(flex: 3) + ]), + Padding( + padding: EdgeInsets.only(top: 20), + child: Text( + S.of(context).scan_qr_code, + style: TextStyle( + fontSize: 12, + fontWeight: FontWeight.w500, + color: PaletteDark.cyanBlue + ), + ), + ), + isAmountFieldShow + ? Padding( + padding: EdgeInsets.only(top: 40), + child: Row( + children: [ + Expanded( + child: Form( + key: _formKey, + child: BaseTextFormField( + controller: amountController, + keyboardType: + TextInputType.numberWithOptions(decimal: true), + inputFormatters: [ + BlacklistingTextInputFormatter( + RegExp('[\\-|\\ |\\,]')) + ], + textAlign: TextAlign.center, + hintText: S.of(context).receive_amount, + textColor: Colors.white, + borderColor: PaletteDark.darkGrey, + validator: AmountValidator( + type: addressListViewModel.type + ), + autovalidate: true, + placeholderTextStyle: TextStyle( + color: PaletteDark.cyanBlue, + fontSize: 18, + fontWeight: FontWeight.w500)))) + ], + ), + ) + : Offstage(), + Padding( + padding: EdgeInsets.only(top: addressTopOffset), + child: Builder( + builder: (context) => Observer( + builder: (context) => GestureDetector( + onTap: () { + Clipboard.setData(ClipboardData( + text: addressListViewModel.address.address)); + Scaffold.of(context).showSnackBar(SnackBar( + content: Text( + S.of(context).copied_to_clipboard, + style: TextStyle(color: Colors.white), + ), + backgroundColor: Colors.green, + duration: Duration(milliseconds: 500), + )); + }, + child: Row( + mainAxisSize: MainAxisSize.max, + children: [ + Expanded( + child: Text( + addressListViewModel.address.address, + maxLines: 1, + overflow: TextOverflow.ellipsis, + style: TextStyle( + fontSize: 16, + fontWeight: FontWeight.w500, + color: Colors.white), + ), + ), + Padding( + padding: EdgeInsets.only(left: 12), + child: copyImage, + ) + ], + ), + ))), + ) + ], + ); + } +} \ No newline at end of file diff --git a/lib/src/screens/subaddress/address_edit_or_create_page.dart b/lib/src/screens/subaddress/address_edit_or_create_page.dart index 84c492196..77eacab88 100644 --- a/lib/src/screens/subaddress/address_edit_or_create_page.dart +++ b/lib/src/screens/subaddress/address_edit_or_create_page.dart @@ -8,6 +8,7 @@ import 'package:cake_wallet/core/address_label_validator.dart'; import 'package:cake_wallet/src/widgets/primary_button.dart'; import 'package:cake_wallet/src/widgets/base_text_form_field.dart'; import 'package:cake_wallet/src/screens/base_page.dart'; +import 'package:cake_wallet/palette.dart'; class AddressEditOrCreatePage extends BasePage { AddressEditOrCreatePage({@required this.addressEditOrCreateViewModel}) @@ -28,6 +29,12 @@ class AddressEditOrCreatePage extends BasePage { @override String get title => S.current.new_subaddress_title; + @override + Color get backgroundLightColor => PaletteDark.backgroundColor; + + @override + Color get backgroundDarkColor => PaletteDark.backgroundColor; + @override Widget body(BuildContext context) { reaction((_) => addressEditOrCreateViewModel.state, @@ -48,6 +55,7 @@ class AddressEditOrCreatePage extends BasePage { child: Center( child: BaseTextFormField( controller: _labelController, + textColor: Colors.white, hintText: S.of(context).new_subaddress_label_name, validator: AddressLabelValidator()))), Observer( diff --git a/lib/src/widgets/alert_background.dart b/lib/src/widgets/alert_background.dart new file mode 100644 index 000000000..db73e7c0d --- /dev/null +++ b/lib/src/widgets/alert_background.dart @@ -0,0 +1,26 @@ +import 'dart:ui'; +import 'package:flutter/cupertino.dart'; +import 'package:flutter/material.dart'; +import 'package:cake_wallet/palette.dart'; + +class AlertBackground extends StatelessWidget { + AlertBackground({@required this.child}); + + final Widget child; + + @override + Widget build(BuildContext context) { + return Container( + height: double.infinity, + width: double.infinity, + color: Colors.transparent, + child: BackdropFilter( + filter: ImageFilter.blur(sigmaX: 3.0, sigmaY: 3.0), + child: Container( + decoration: BoxDecoration(color: PaletteDark.darkNightBlue.withOpacity(0.75)), + child: child, + ), + ), + ); + } +} \ No newline at end of file diff --git a/lib/src/widgets/alert_close_button.dart b/lib/src/widgets/alert_close_button.dart new file mode 100644 index 000000000..0f21be5df --- /dev/null +++ b/lib/src/widgets/alert_close_button.dart @@ -0,0 +1,28 @@ +import 'package:flutter/material.dart'; + +class AlertCloseButton extends StatelessWidget { + AlertCloseButton({@required this.image}); + + final Image image; + + @override + Widget build(BuildContext context) { + return Positioned( + bottom: 24, + child: GestureDetector( + onTap: () => Navigator.of(context).pop(), + child: Container( + height: 42, + width: 42, + decoration: BoxDecoration( + color: Colors.white, + shape: BoxShape.circle + ), + child: Center( + child: image, + ), + ), + ) + ); + } +} \ No newline at end of file diff --git a/lib/src/widgets/cake_scrollbar.dart b/lib/src/widgets/cake_scrollbar.dart new file mode 100644 index 000000000..6258e5f0b --- /dev/null +++ b/lib/src/widgets/cake_scrollbar.dart @@ -0,0 +1,45 @@ +import 'package:flutter/material.dart'; +import 'package:cake_wallet/palette.dart'; + +class CakeScrollbar extends StatelessWidget { + CakeScrollbar({ + @required this.backgroundHeight, + @required this.thumbHeight, + @required this.fromTop + }); + + final double backgroundHeight; + final double thumbHeight; + final double fromTop; + + @override + Widget build(BuildContext context) { + return Positioned( + right: 6, + child: Container( + height: backgroundHeight, + width: 6, + decoration: BoxDecoration( + color: PaletteDark.violetBlue, + borderRadius: BorderRadius.all(Radius.circular(3)) + ), + child: Stack( + children: [ + AnimatedPositioned( + duration: Duration(milliseconds: 0), + top: fromTop, + child: Container( + height: thumbHeight, + width: 6.0, + decoration: BoxDecoration( + color: PaletteDark.wildBlueGrey, + borderRadius: BorderRadius.all(Radius.circular(3)) + ), + ), + ) + ], + ), + ) + ); + } +} \ No newline at end of file diff --git a/lib/store/dashboard/page_view_store.dart b/lib/store/dashboard/page_view_store.dart new file mode 100644 index 000000000..30b938519 --- /dev/null +++ b/lib/store/dashboard/page_view_store.dart @@ -0,0 +1,19 @@ +import 'package:mobx/mobx.dart'; + +part 'page_view_store.g.dart'; + +class PageViewStore = PageViewStoreBase with _$PageViewStore; + +abstract class PageViewStoreBase with Store { + PageViewStoreBase() { + setCurrentPage(1); + } + + @observable + double currentPage; + + @action + void setCurrentPage(double currentPage) { + this.currentPage = currentPage; + } +} \ No newline at end of file diff --git a/lib/view_model/dashboard/dashboard_view_model.dart b/lib/view_model/dashboard/dashboard_view_model.dart index fac416af8..64530b655 100644 --- a/lib/view_model/dashboard/dashboard_view_model.dart +++ b/lib/view_model/dashboard/dashboard_view_model.dart @@ -22,6 +22,7 @@ import 'package:cake_wallet/store/dashboard/trades_store.dart'; import 'package:cake_wallet/store/dashboard/trade_filter_store.dart'; import 'package:cake_wallet/store/dashboard/transaction_filter_store.dart'; import 'package:cake_wallet/view_model/dashboard/formatted_item_list.dart'; +import 'package:cake_wallet/store/dashboard/page_view_store.dart'; part 'dashboard_view_model.g.dart'; @@ -33,7 +34,8 @@ abstract class DashboardViewModelBase with Store { this.appStore, this.tradesStore, this.tradeFilterStore, - this.transactionFilterStore}) { + this.transactionFilterStore, + this.pageViewStore}) { name = appStore.wallet?.name; wallet ??= appStore.wallet; @@ -53,8 +55,6 @@ abstract class DashboardViewModelBase with Store { if (_wallet is MoneroWallet) { subname = _wallet.account?.label; } - - currentPage = 0; } @observable @@ -63,15 +63,15 @@ abstract class DashboardViewModelBase with Store { @observable String name; - @observable - double currentPage; - @observable ObservableList transactions; @observable String subname; + @computed + double get currentPage => pageViewStore.currentPage; + @computed String get address => wallet.address; @@ -130,6 +130,8 @@ abstract class DashboardViewModelBase with Store { TransactionFilterStore transactionFilterStore; + PageViewStore pageViewStore; + ReactionDisposer _reaction; void _onWalletChange(WalletBase wallet) { diff --git a/lib/view_model/monero_account_list/monero_account_list_view_model.dart b/lib/view_model/monero_account_list/monero_account_list_view_model.dart index e7ad6e4e9..c03fee8a3 100644 --- a/lib/view_model/monero_account_list/monero_account_list_view_model.dart +++ b/lib/view_model/monero_account_list/monero_account_list_view_model.dart @@ -9,7 +9,15 @@ class MoneroAccountListViewModel = MoneroAccountListViewModelBase with _$MoneroAccountListViewModel; abstract class MoneroAccountListViewModelBase with Store { - MoneroAccountListViewModelBase(this._moneroWallet); + MoneroAccountListViewModelBase(this._moneroWallet) : scrollOffsetFromTop = 0; + + @observable + double scrollOffsetFromTop; + + @action + void setScrollOffsetFromTop(double scrollOffsetFromTop) { + this.scrollOffsetFromTop = scrollOffsetFromTop; + } @computed List get accounts => _moneroWallet.accountList.accounts diff --git a/lib/view_model/wallet_address_list/wallet_address_list_view_model.dart b/lib/view_model/wallet_address_list/wallet_address_list_view_model.dart index e36dc18cb..4f2e3662c 100644 --- a/lib/view_model/wallet_address_list/wallet_address_list_view_model.dart +++ b/lib/view_model/wallet_address_list/wallet_address_list_view_model.dart @@ -7,6 +7,7 @@ import 'package:cake_wallet/utils/list_item.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_account_list_header.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_header.dart'; import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_item.dart'; +import 'package:cake_wallet/src/domain/common/wallet_type.dart'; part 'wallet_address_list_view_model.g.dart'; @@ -62,6 +63,9 @@ abstract class WalletAddressListViewModelBase with Store { @observable String amount; + @computed + WalletType get type => _wallet.type; + @computed WalletAddressListItem get address => WalletAddressListItem(address: _wallet.address); diff --git a/res/values/strings_de.arb b/res/values/strings_de.arb index 0724a9577..cc97e8a72 100644 --- a/res/values/strings_de.arb +++ b/res/values/strings_de.arb @@ -138,6 +138,8 @@ "share_address" : "Adresse teilen ", "receive_amount" : "Menge", "subaddresses" : "Unteradressen", + "addresses" : "Adressen", + "scan_qr_code" : "Scannen Sie den QR-Code, um die Adresse zu erhalten", "rename" : "Umbenennen", "choose_account" : "Konto auswählen", "create_new_account" : "Neues Konto erstellen", diff --git a/res/values/strings_en.arb b/res/values/strings_en.arb index ff0c481c0..362e493bf 100644 --- a/res/values/strings_en.arb +++ b/res/values/strings_en.arb @@ -139,6 +139,7 @@ "receive_amount" : "Amount", "subaddresses" : "Subaddresses", "addresses" : "Addresses", + "scan_qr_code" : "Scan the QR code to get the address", "rename" : "Rename", "choose_account" : "Choose account", "create_new_account" : "Create new account", diff --git a/res/values/strings_es.arb b/res/values/strings_es.arb index 9f0ea478a..cfaeb7a42 100644 --- a/res/values/strings_es.arb +++ b/res/values/strings_es.arb @@ -138,6 +138,8 @@ "share_address" : "Compartir dirección", "receive_amount" : "Cantidad", "subaddresses" : "Subdirecciones", + "addresses" : "Direcciones", + "scan_qr_code" : "Escanee el código QR para obtener la dirección", "rename" : "Rebautizar", "choose_account" : "Elegir cuenta", "create_new_account" : "Crear una nueva cuenta", diff --git a/res/values/strings_hi.arb b/res/values/strings_hi.arb index 1a915a1d0..7e43ebbc9 100644 --- a/res/values/strings_hi.arb +++ b/res/values/strings_hi.arb @@ -138,6 +138,8 @@ "share_address" : "पता साझा करें", "receive_amount" : "रकम", "subaddresses" : "उप पते", + "addresses" : "पतों", + "scan_qr_code" : "पता प्राप्त करने के लिए QR कोड स्कैन करें", "rename" : "नाम बदलें", "choose_account" : "खाता चुनें", "create_new_account" : "नया खाता बनाएँ", diff --git a/res/values/strings_ja.arb b/res/values/strings_ja.arb index 5f45c0949..173317288 100644 --- a/res/values/strings_ja.arb +++ b/res/values/strings_ja.arb @@ -138,6 +138,8 @@ "share_address" : "住所を共有する", "receive_amount" : "量", "subaddresses" : "サブアドレス", + "addresses" : "住所", + "scan_qr_code" : "QRコードをスキャンして住所を取得します", "rename" : "リネーム", "choose_account" : "アカウントを選択", "create_new_account" : "新しいアカウントを作成する", diff --git a/res/values/strings_ko.arb b/res/values/strings_ko.arb index 74430e1a2..3c817739c 100644 --- a/res/values/strings_ko.arb +++ b/res/values/strings_ko.arb @@ -138,6 +138,8 @@ "share_address" : "주소 공유", "receive_amount" : "양", "subaddresses" : "하위 주소", + "addresses" : "구애", + "scan_qr_code" : "QR 코드를 스캔하여 주소를 얻습니다.", "rename" : "이름 바꾸기", "choose_account" : "계정을 선택하십시오", "create_new_account" : "새 계정을 만들", diff --git a/res/values/strings_nl.arb b/res/values/strings_nl.arb index 78e3b77f6..3e2e4cb0c 100644 --- a/res/values/strings_nl.arb +++ b/res/values/strings_nl.arb @@ -139,6 +139,8 @@ "receive_amount" : "Bedrag", "subaddresses" : "Subadressen", "rename" : "Hernoemen", + "addresses" : "Adressen", + "scan_qr_code" : "Scan de QR-code om het adres te krijgen", "choose_account" : "Kies account", "create_new_account" : "Creëer een nieuw account", "accounts_subaddresses" : "Accounts en subadressen", diff --git a/res/values/strings_pl.arb b/res/values/strings_pl.arb index 0647fd32c..e992deb50 100644 --- a/res/values/strings_pl.arb +++ b/res/values/strings_pl.arb @@ -138,6 +138,8 @@ "share_address" : "Udostępnij adres", "receive_amount" : "Ilość", "subaddresses" : "Podadresy", + "addresses" : "Adresy", + "scan_qr_code" : "Zeskanuj kod QR, aby uzyskać adres", "rename" : "Przemianować", "choose_account" : "Wybierz konto", "create_new_account" : "Stwórz nowe konto", diff --git a/res/values/strings_pt.arb b/res/values/strings_pt.arb index 04b12862c..59919ecbf 100644 --- a/res/values/strings_pt.arb +++ b/res/values/strings_pt.arb @@ -138,6 +138,8 @@ "share_address" : "Compartilhar endereço", "receive_amount" : "Quantia", "subaddresses" : "Sub-endereços", + "addresses" : "Endereços", + "scan_qr_code" : "Digitalize o código QR para obter o endereço", "rename" : "Renomear", "choose_account" : "Escolha uma conta", "create_new_account" : "Criar nova conta", diff --git a/res/values/strings_ru.arb b/res/values/strings_ru.arb index 441ff68a0..81a2c76b8 100644 --- a/res/values/strings_ru.arb +++ b/res/values/strings_ru.arb @@ -138,6 +138,8 @@ "share_address" : "Поделиться адресом", "receive_amount" : "Сумма", "subaddresses" : "Субадреса", + "addresses" : "Адреса", + "scan_qr_code" : "Отсканируйте QR-код для получения адреса", "rename" : "Переименовать", "choose_account" : "Выберите аккаунт", "create_new_account" : "Создать новый аккаунт", diff --git a/res/values/strings_uk.arb b/res/values/strings_uk.arb index 648bd8e70..31268c9d3 100644 --- a/res/values/strings_uk.arb +++ b/res/values/strings_uk.arb @@ -138,6 +138,8 @@ "share_address" : "Поділитися адресою", "receive_amount" : "Сума", "subaddresses" : "Субадреси", + "addresses" : "Адреси", + "scan_qr_code" : "Скануйте QR-код для одержання адреси", "rename" : "Перейменувати", "choose_account" : "Оберіть акаунт", "create_new_account" : "Створити новий акаунт", diff --git a/res/values/strings_zh.arb b/res/values/strings_zh.arb index bf0523f96..0450aa214 100644 --- a/res/values/strings_zh.arb +++ b/res/values/strings_zh.arb @@ -138,6 +138,8 @@ "share_address" : "分享地址", "receive_amount" : "量", "subaddresses" : "子地址", + "addresses" : "地址", + "scan_qr_code" : "掃描二維碼獲取地址", "rename" : "改名", "choose_account" : "選擇帳號", "create_new_account" : "建立新帳戶",