Merge pull request #90 from cake-tech/CAKE-19-new-receive-screen
Cake 19 new receive screen
BIN
assets/images/2.0x/close.png
Normal file
After Width: | Height: | Size: 378 B |
BIN
assets/images/2.0x/copy_address.png
Normal file
After Width: | Height: | Size: 802 B |
Before Width: | Height: | Size: 319 B After Width: | Height: | Size: 798 B |
BIN
assets/images/3.0x/close.png
Normal file
After Width: | Height: | Size: 461 B |
BIN
assets/images/3.0x/copy_address.png
Normal file
After Width: | Height: | Size: 1.1 KiB |
Before Width: | Height: | Size: 498 B After Width: | Height: | Size: 1.1 KiB |
BIN
assets/images/close.png
Normal file
After Width: | Height: | Size: 272 B |
BIN
assets/images/copy_address.png
Normal file
After Width: | Height: | Size: 493 B |
Before Width: | Height: | Size: 277 B After Width: | Height: | Size: 392 B |
|
@ -57,6 +57,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/trade_filter_store.dart';
|
||||||
import 'package:cake_wallet/store/dashboard/transaction_filter_store.dart';
|
import 'package:cake_wallet/store/dashboard/transaction_filter_store.dart';
|
||||||
import 'package:cake_wallet/store/dashboard/fiat_convertation_store.dart';
|
import 'package:cake_wallet/store/dashboard/fiat_convertation_store.dart';
|
||||||
|
import 'package:cake_wallet/store/dashboard/page_view_store.dart';
|
||||||
|
|
||||||
final getIt = GetIt.instance;
|
final getIt = GetIt.instance;
|
||||||
|
|
||||||
|
@ -110,6 +111,7 @@ Future setup(
|
||||||
TradeFilterStore(wallet: getIt.get<AppStore>().wallet));
|
TradeFilterStore(wallet: getIt.get<AppStore>().wallet));
|
||||||
getIt.registerSingleton<TransactionFilterStore>(TransactionFilterStore());
|
getIt.registerSingleton<TransactionFilterStore>(TransactionFilterStore());
|
||||||
getIt.registerSingleton<FiatConvertationStore>(FiatConvertationStore());
|
getIt.registerSingleton<FiatConvertationStore>(FiatConvertationStore());
|
||||||
|
getIt.registerSingleton<PageViewStore>(PageViewStore());
|
||||||
|
|
||||||
getIt.registerFactory<KeyService>(
|
getIt.registerFactory<KeyService>(
|
||||||
() => KeyService(getIt.get<FlutterSecureStorage>()));
|
() => KeyService(getIt.get<FlutterSecureStorage>()));
|
||||||
|
@ -153,7 +155,8 @@ Future setup(
|
||||||
appStore: getIt.get<AppStore>(),
|
appStore: getIt.get<AppStore>(),
|
||||||
tradesStore: getIt.get<TradesStore>(),
|
tradesStore: getIt.get<TradesStore>(),
|
||||||
tradeFilterStore: getIt.get<TradeFilterStore>(),
|
tradeFilterStore: getIt.get<TradeFilterStore>(),
|
||||||
transactionFilterStore: getIt.get<TransactionFilterStore>()
|
transactionFilterStore: getIt.get<TransactionFilterStore>(),
|
||||||
|
pageViewStore: getIt.get<PageViewStore>()
|
||||||
));
|
));
|
||||||
|
|
||||||
getIt.registerFactory<AuthService>(() => AuthService(
|
getIt.registerFactory<AuthService>(() => AuthService(
|
||||||
|
@ -183,7 +186,9 @@ Future setup(
|
||||||
closable: false));
|
closable: false));
|
||||||
|
|
||||||
getIt.registerFactory<DashboardPage>(
|
getIt.registerFactory<DashboardPage>(
|
||||||
() => DashboardPage(walletViewModel: getIt.get<DashboardViewModel>()));
|
() => DashboardPage(
|
||||||
|
walletViewModel: getIt.get<DashboardViewModel>(),
|
||||||
|
addressListViewModel: getIt.get<WalletAddressListViewModel>()));
|
||||||
|
|
||||||
getIt.registerFactory<ReceivePage>(() => ReceivePage(
|
getIt.registerFactory<ReceivePage>(() => ReceivePage(
|
||||||
addressListViewModel: getIt.get<WalletAddressListViewModel>()));
|
addressListViewModel: getIt.get<WalletAddressListViewModel>()));
|
||||||
|
|
|
@ -162,6 +162,7 @@ class S implements WidgetsLocalizations {
|
||||||
String get restore_wallet_restore_description => "Wallet restore description";
|
String get restore_wallet_restore_description => "Wallet restore description";
|
||||||
String get save => "Save";
|
String get save => "Save";
|
||||||
String get saved_the_trade_id => "I've saved the trade ID";
|
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_choose => "Choose seed language";
|
||||||
String get seed_language_chinese => "Chinese";
|
String get seed_language_chinese => "Chinese";
|
||||||
String get seed_language_choose => "Please choose seed language:";
|
String get seed_language_choose => "Please choose seed language:";
|
||||||
|
@ -623,6 +624,8 @@ class $de extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR verfügbares Guthaben";
|
String get xmr_available_balance => "XMR verfügbares Guthaben";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Scannen Sie den QR-Code, um die Adresse zu erhalten";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Bezahlt";
|
String get trade_state_paid => "Bezahlt";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Neuer Knoten";
|
String get node_new => "Neuer Knoten";
|
||||||
|
@ -749,6 +752,8 @@ class $de extends S {
|
||||||
@override
|
@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";
|
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
|
@override
|
||||||
|
String get addresses => "Adressen";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Transaktionen nach Datum";
|
String get transactions_by_date => "Transaktionen nach Datum";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Bitte geben Sie hier Ihren Code ein";
|
String get restore_from_seed_placeholder => "Bitte geben Sie hier Ihren Code ein";
|
||||||
|
@ -1243,6 +1248,8 @@ class $hi extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR उपलब्ध शेष राशि";
|
String get xmr_available_balance => "XMR उपलब्ध शेष राशि";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "पता प्राप्त करने के लिए QR कोड स्कैन करें";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "भुगतान किया है";
|
String get trade_state_paid => "भुगतान किया है";
|
||||||
@override
|
@override
|
||||||
String get node_new => "नया नोड";
|
String get node_new => "नया नोड";
|
||||||
|
@ -1369,6 +1376,8 @@ class $hi extends S {
|
||||||
@override
|
@override
|
||||||
String get error_text_fiat => "राशि का मूल्य उपलब्ध शेष राशि से अधिक नहीं हो सकता.\nअंश अंकों की संख्या कम या 2 के बराबर होनी चाहिए";
|
String get error_text_fiat => "राशि का मूल्य उपलब्ध शेष राशि से अधिक नहीं हो सकता.\nअंश अंकों की संख्या कम या 2 के बराबर होनी चाहिए";
|
||||||
@override
|
@override
|
||||||
|
String get addresses => "पतों";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "तारीख से लेन-देन";
|
String get transactions_by_date => "तारीख से लेन-देन";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "कृपया अपना कोड वाक्यांश यहां दर्ज करें या पेस्ट करें";
|
String get restore_from_seed_placeholder => "कृपया अपना कोड वाक्यांश यहां दर्ज करें या पेस्ट करें";
|
||||||
|
@ -1863,6 +1872,8 @@ class $ru extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "Доступный XMR баланс";
|
String get xmr_available_balance => "Доступный XMR баланс";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Отсканируйте QR-код для получения адреса";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Оплаченная";
|
String get trade_state_paid => "Оплаченная";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Новая нода";
|
String get node_new => "Новая нода";
|
||||||
|
@ -1989,6 +2000,8 @@ class $ru extends S {
|
||||||
@override
|
@override
|
||||||
String get error_text_fiat => "Значение суммы не может превышать доступный баланс.\nКоличество цифр после запятой должно быть меньше или равно 2";
|
String get error_text_fiat => "Значение суммы не может превышать доступный баланс.\nКоличество цифр после запятой должно быть меньше или равно 2";
|
||||||
@override
|
@override
|
||||||
|
String get addresses => "Адреса";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Сортировать по дате";
|
String get transactions_by_date => "Сортировать по дате";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Введите или вставьте мнемоническую фразу вашего кошелька";
|
String get restore_from_seed_placeholder => "Введите или вставьте мнемоническую фразу вашего кошелька";
|
||||||
|
@ -2483,6 +2496,8 @@ class $ko extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR 사용 가능한 잔액";
|
String get xmr_available_balance => "XMR 사용 가능한 잔액";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "QR 코드를 스캔하여 주소를 얻습니다.";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "유료";
|
String get trade_state_paid => "유료";
|
||||||
@override
|
@override
|
||||||
String get node_new => "새로운 노드";
|
String get node_new => "새로운 노드";
|
||||||
|
@ -2609,6 +2624,8 @@ class $ko extends S {
|
||||||
@override
|
@override
|
||||||
String get error_text_fiat => "금액은 사용 가능한 잔액을 초과 할 수 없습니다.\n소수 자릿수는 2보다 작거나 같아야합니다";
|
String get error_text_fiat => "금액은 사용 가능한 잔액을 초과 할 수 없습니다.\n소수 자릿수는 2보다 작거나 같아야합니다";
|
||||||
@override
|
@override
|
||||||
|
String get addresses => "구애";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "날짜 별 거래";
|
String get transactions_by_date => "날짜 별 거래";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "여기에 코드 문구를 입력하거나 붙여 넣으십시오.";
|
String get restore_from_seed_placeholder => "여기에 코드 문구를 입력하거나 붙여 넣으십시오.";
|
||||||
|
@ -3103,6 +3120,8 @@ class $pt extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "Saldo XMR disponível";
|
String get xmr_available_balance => "Saldo XMR disponível";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Digitalize o código QR para obter o endereço";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Paga";
|
String get trade_state_paid => "Paga";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Novo nó";
|
String get node_new => "Novo nó";
|
||||||
|
@ -3229,6 +3248,8 @@ class $pt extends S {
|
||||||
@override
|
@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";
|
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
|
@override
|
||||||
|
String get addresses => "Endereços";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Transações por data";
|
String get transactions_by_date => "Transações por data";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Digite ou cole sua frase de código aqui";
|
String get restore_from_seed_placeholder => "Digite ou cole sua frase de código aqui";
|
||||||
|
@ -3723,6 +3744,8 @@ class $uk extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "Доступний XMR баланс";
|
String get xmr_available_balance => "Доступний XMR баланс";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Скануйте QR-код для одержання адреси";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Оплачена";
|
String get trade_state_paid => "Оплачена";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Новий вузол";
|
String get node_new => "Новий вузол";
|
||||||
|
@ -3849,6 +3872,8 @@ class $uk extends S {
|
||||||
@override
|
@override
|
||||||
String get error_text_fiat => "Значення суми не може перевищувати доступний баланс.\nКількість цифр після коми повинно бути меншим або дорівнювати 2";
|
String get error_text_fiat => "Значення суми не може перевищувати доступний баланс.\nКількість цифр після коми повинно бути меншим або дорівнювати 2";
|
||||||
@override
|
@override
|
||||||
|
String get addresses => "Адреси";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Сортувати по даті";
|
String get transactions_by_date => "Сортувати по даті";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Введіть або вставте мнемонічну фразу вашого гаманця";
|
String get restore_from_seed_placeholder => "Введіть або вставте мнемонічну фразу вашого гаманця";
|
||||||
|
@ -4343,6 +4368,8 @@ class $ja extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR利用可能残高";
|
String get xmr_available_balance => "XMR利用可能残高";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "QRコードをスキャンして住所を取得します";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "有料";
|
String get trade_state_paid => "有料";
|
||||||
@override
|
@override
|
||||||
String get node_new => "新しいノード";
|
String get node_new => "新しいノード";
|
||||||
|
@ -4469,6 +4496,8 @@ class $ja extends S {
|
||||||
@override
|
@override
|
||||||
String get error_text_fiat => "金額は利用可能な残高を超えることはできません.\n小数桁の数は2以下でなければなりません";
|
String get error_text_fiat => "金額は利用可能な残高を超えることはできません.\n小数桁の数は2以下でなければなりません";
|
||||||
@override
|
@override
|
||||||
|
String get addresses => "住所";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "日付ごとの取引";
|
String get transactions_by_date => "日付ごとの取引";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "ここにコードフレーズを入力または貼り付けてください";
|
String get restore_from_seed_placeholder => "ここにコードフレーズを入力または貼り付けてください";
|
||||||
|
@ -4967,6 +4996,8 @@ class $pl extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR Dostępne saldo";
|
String get xmr_available_balance => "XMR Dostępne saldo";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Zeskanuj kod QR, aby uzyskać adres";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Płatny";
|
String get trade_state_paid => "Płatny";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Nowy węzeł";
|
String get node_new => "Nowy węzeł";
|
||||||
|
@ -5093,6 +5124,8 @@ class $pl extends S {
|
||||||
@override
|
@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";
|
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
|
@override
|
||||||
|
String get addresses => "Adresy";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Transakcje według daty";
|
String get transactions_by_date => "Transakcje według daty";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Wpisz lub wklej tutaj swoją frazę kodową";
|
String get restore_from_seed_placeholder => "Wpisz lub wklej tutaj swoją frazę kodową";
|
||||||
|
@ -5587,6 +5620,8 @@ class $es extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR Available Balance";
|
String get xmr_available_balance => "XMR Available Balance";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Escanee el código QR para obtener la dirección";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Pagado";
|
String get trade_state_paid => "Pagado";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Nuevo nodo";
|
String get node_new => "Nuevo nodo";
|
||||||
|
@ -5713,6 +5748,8 @@ class $es extends S {
|
||||||
@override
|
@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";
|
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
|
@override
|
||||||
|
String get addresses => "Direcciones";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Transacciones por fecha";
|
String get transactions_by_date => "Transacciones por fecha";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Ingrese o pegue su frase de código aquí";
|
String get restore_from_seed_placeholder => "Ingrese o pegue su frase de código aquí";
|
||||||
|
@ -6207,6 +6244,8 @@ class $nl extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR Beschikbaar saldo";
|
String get xmr_available_balance => "XMR Beschikbaar saldo";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "Scan de QR-code om het adres te krijgen";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "Betaald";
|
String get trade_state_paid => "Betaald";
|
||||||
@override
|
@override
|
||||||
String get node_new => "Nieuw knooppunt";
|
String get node_new => "Nieuw knooppunt";
|
||||||
|
@ -6333,6 +6372,8 @@ class $nl extends S {
|
||||||
@override
|
@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";
|
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
|
@override
|
||||||
|
String get addresses => "Adressen";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "Transacties op datum";
|
String get transactions_by_date => "Transacties op datum";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "Voer hier uw codefrase in of plak deze";
|
String get restore_from_seed_placeholder => "Voer hier uw codefrase in of plak deze";
|
||||||
|
@ -6827,6 +6868,8 @@ class $zh extends S {
|
||||||
@override
|
@override
|
||||||
String get xmr_available_balance => "XMR 可用余额 ";
|
String get xmr_available_balance => "XMR 可用余额 ";
|
||||||
@override
|
@override
|
||||||
|
String get scan_qr_code => "掃描二維碼獲取地址";
|
||||||
|
@override
|
||||||
String get trade_state_paid => "已付费";
|
String get trade_state_paid => "已付费";
|
||||||
@override
|
@override
|
||||||
String get node_new => "新节点";
|
String get node_new => "新节点";
|
||||||
|
@ -6953,6 +6996,8 @@ class $zh extends S {
|
||||||
@override
|
@override
|
||||||
String get error_text_fiat => "金额不能超过可用余额.\n小数位数必须小于或等于2";
|
String get error_text_fiat => "金额不能超过可用余额.\n小数位数必须小于或等于2";
|
||||||
@override
|
@override
|
||||||
|
String get addresses => "地址";
|
||||||
|
@override
|
||||||
String get transactions_by_date => "按日期交易";
|
String get transactions_by_date => "按日期交易";
|
||||||
@override
|
@override
|
||||||
String get restore_from_seed_placeholder => "请在此处输入或粘贴您的代码短语";
|
String get restore_from_seed_placeholder => "请在此处输入或粘贴您的代码短语";
|
||||||
|
|
|
@ -18,10 +18,9 @@ class Palette {
|
||||||
}
|
}
|
||||||
|
|
||||||
class PaletteDark {
|
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 lightDistantBlue = Color.fromRGBO(81, 96, 147, 1.0); // borderCardColor
|
||||||
static const Color gray = Color.fromRGBO(140, 153, 201, 1.0); // walletCardText
|
static const Color gray = Color.fromRGBO(140, 153, 201, 1.0); // walletCardText
|
||||||
static const Color violetBlue = Color.fromRGBO(51, 63, 104, 1.0); // walletCardAddressField
|
|
||||||
static const Color moderateBlue = Color.fromRGBO(63, 77, 122, 1.0); // walletCardSubAddressField
|
static const Color moderateBlue = Color.fromRGBO(63, 77, 122, 1.0); // walletCardSubAddressField
|
||||||
static const Color darkNightBlue = Color.fromRGBO(33, 43, 73, 1.0); // historyPanel
|
static const Color darkNightBlue = Color.fromRGBO(33, 43, 73, 1.0); // historyPanel
|
||||||
static const Color pigeonBlue = Color.fromRGBO(91, 112, 146, 1.0); // historyPanelText
|
static const Color pigeonBlue = Color.fromRGBO(91, 112, 146, 1.0); // historyPanelText
|
||||||
|
@ -34,13 +33,22 @@ class PaletteDark {
|
||||||
static const Color backgroundColor = Color.fromRGBO(25, 35, 60, 1.0);
|
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 nightBlue = Color.fromRGBO(35, 47, 79, 1.0);
|
||||||
static const Color wildNightBlue = Color.fromRGBO(39, 53, 96, 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 cyanBlue = Color.fromRGBO(99, 113, 150, 1.0);
|
||||||
static const Color darkCyanBlue = Color.fromRGBO(91, 112, 146, 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 orangeYellow = Color.fromRGBO(243, 166, 50, 1.0);
|
||||||
static const Color brightGreen = Color.fromRGBO(88, 243, 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 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 lightNightBlue = Color.fromRGBO(39, 52, 89, 1.0);
|
||||||
static const Color wildBlue = Color.fromRGBO(165, 176, 205, 1.0);
|
static const Color wildBlue = Color.fromRGBO(165, 176, 205, 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.
|
// FIXME: Rename.
|
||||||
static const Color eee = Color.fromRGBO(236, 239, 245, 1.0);
|
static const Color eee = Color.fromRGBO(236, 239, 245, 1.0);
|
||||||
|
|
|
@ -23,9 +23,9 @@ abstract class BasePage extends StatelessWidget {
|
||||||
|
|
||||||
Widget Function(BuildContext, Widget) get rootWrapper => null;
|
Widget Function(BuildContext, Widget) get rootWrapper => null;
|
||||||
|
|
||||||
final _backArrowImage = Image.asset('assets/images/back_arrow.png');
|
final _backArrowImage = Image.asset('assets/images/back_arrow.png', color: Colors.white);
|
||||||
final _backArrowImageDarkTheme =
|
final _backArrowImageDarkTheme =
|
||||||
Image.asset('assets/images/back_arrow_dark_theme.png');
|
Image.asset('assets/images/back_arrow_dark_theme.png', color: Colors.white);
|
||||||
final _closeButtonImage = Image.asset('assets/images/close_button.png');
|
final _closeButtonImage = Image.asset('assets/images/close_button.png');
|
||||||
final _closeButtonImageDarkTheme =
|
final _closeButtonImageDarkTheme =
|
||||||
Image.asset('assets/images/close_button_dark_theme.png');
|
Image.asset('assets/images/close_button_dark_theme.png');
|
||||||
|
@ -71,7 +71,9 @@ abstract class BasePage extends StatelessWidget {
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 18.0,
|
fontSize: 18.0,
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
color: Theme.of(context).primaryTextTheme.title.color),
|
//color: Theme.of(context).primaryTextTheme.title.color,
|
||||||
|
color: Colors.white,
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,13 +9,18 @@ import 'package:cake_wallet/palette.dart';
|
||||||
import 'package:dots_indicator/dots_indicator.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/action_button.dart';
|
||||||
import 'package:cake_wallet/src/screens/dashboard/widgets/balance_page.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:cake_wallet/src/screens/dashboard/widgets/transactions_page.dart';
|
||||||
import 'package:flutter_mobx/flutter_mobx.dart';
|
import 'package:flutter_mobx/flutter_mobx.dart';
|
||||||
import 'package:mobx/mobx.dart';
|
import 'package:mobx/mobx.dart';
|
||||||
import 'package:cake_wallet/src/screens/dashboard/widgets/sync_indicator.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 {
|
class DashboardPage extends BasePage {
|
||||||
DashboardPage({@required this.walletViewModel});
|
DashboardPage({
|
||||||
|
@required this.walletViewModel,
|
||||||
|
@required this.addressListViewModel,
|
||||||
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Color get backgroundLightColor => PaletteDark.backgroundColor;
|
Color get backgroundLightColor => PaletteDark.backgroundColor;
|
||||||
|
@ -23,6 +28,9 @@ class DashboardPage extends BasePage {
|
||||||
@override
|
@override
|
||||||
Color get backgroundDarkColor => PaletteDark.backgroundColor;
|
Color get backgroundDarkColor => PaletteDark.backgroundColor;
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool get resizeToAvoidBottomPadding => false;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget middle(BuildContext context) {
|
Widget middle(BuildContext context) {
|
||||||
return SyncIndicator(dashboardViewModel: walletViewModel);
|
return SyncIndicator(dashboardViewModel: walletViewModel);
|
||||||
|
@ -54,13 +62,14 @@ class DashboardPage extends BasePage {
|
||||||
}
|
}
|
||||||
|
|
||||||
final DashboardViewModel walletViewModel;
|
final DashboardViewModel walletViewModel;
|
||||||
|
final WalletAddressListViewModel addressListViewModel;
|
||||||
final sendImage = Image.asset('assets/images/upload.png',
|
final sendImage = Image.asset('assets/images/upload.png',
|
||||||
height: 22.24, width: 24, color: Colors.white);
|
height: 22.24, width: 24, color: Colors.white);
|
||||||
final exchangeImage = Image.asset('assets/images/transfer.png',
|
final exchangeImage = Image.asset('assets/images/transfer.png',
|
||||||
height: 24.27, width: 22.25, color: Colors.white);
|
height: 24.27, width: 22.25, color: Colors.white);
|
||||||
final receiveImage = Image.asset('assets/images/download.png',
|
final receiveImage = Image.asset('assets/images/download.png',
|
||||||
height: 22.24, width: 24, color: Colors.white);
|
height: 22.24, width: 24, color: Colors.white);
|
||||||
final controller = PageController(initialPage: 0);
|
final controller = PageController(initialPage: 1);
|
||||||
|
|
||||||
var pages = <Widget>[];
|
var pages = <Widget>[];
|
||||||
bool _isEffectsInstalled = false;
|
bool _isEffectsInstalled = false;
|
||||||
|
@ -147,11 +156,12 @@ class DashboardPage extends BasePage {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pages.add(AddressPage(addressListViewModel: addressListViewModel));
|
||||||
pages.add(BalancePage(dashboardViewModel: walletViewModel));
|
pages.add(BalancePage(dashboardViewModel: walletViewModel));
|
||||||
pages.add(TransactionsPage(dashboardViewModel: walletViewModel));
|
pages.add(TransactionsPage(dashboardViewModel: walletViewModel));
|
||||||
|
|
||||||
controller.addListener(() {
|
controller.addListener(() {
|
||||||
walletViewModel.currentPage = controller.page;
|
walletViewModel.pageViewStore.setCurrentPage(controller.page);
|
||||||
});
|
});
|
||||||
|
|
||||||
reaction((_) => walletViewModel.currentPage, (double currentPage) {
|
reaction((_) => walletViewModel.currentPage, (double currentPage) {
|
||||||
|
|
59
lib/src/screens/dashboard/widgets/address_page.dart
Normal file
|
@ -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: <Widget>[
|
||||||
|
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: <Widget>[
|
||||||
|
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,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -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/widgets/primary_button.dart';
|
||||||
import 'package:cake_wallet/src/screens/base_page.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/src/widgets/base_text_form_field.dart';
|
||||||
|
import 'package:cake_wallet/palette.dart';
|
||||||
|
|
||||||
class MoneroAccountEditOrCreatePage extends BasePage {
|
class MoneroAccountEditOrCreatePage extends BasePage {
|
||||||
MoneroAccountEditOrCreatePage({@required this.moneroAccountCreationViewModel})
|
MoneroAccountEditOrCreatePage({@required this.moneroAccountCreationViewModel})
|
||||||
|
@ -23,6 +24,12 @@ class MoneroAccountEditOrCreatePage extends BasePage {
|
||||||
@override
|
@override
|
||||||
String get title => S.current.account;
|
String get title => S.current.account;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Color get backgroundLightColor => PaletteDark.backgroundColor;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Color get backgroundDarkColor => PaletteDark.backgroundColor;
|
||||||
|
|
||||||
final GlobalKey<FormState> _formKey;
|
final GlobalKey<FormState> _formKey;
|
||||||
final TextEditingController _textController;
|
final TextEditingController _textController;
|
||||||
|
|
||||||
|
@ -31,7 +38,6 @@ class MoneroAccountEditOrCreatePage extends BasePage {
|
||||||
Form(
|
Form(
|
||||||
key: _formKey,
|
key: _formKey,
|
||||||
child: Container(
|
child: Container(
|
||||||
color: Theme.of(context).backgroundColor,
|
|
||||||
padding: EdgeInsets.all(24.0),
|
padding: EdgeInsets.all(24.0),
|
||||||
child: Column(
|
child: Column(
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
|
@ -39,6 +45,7 @@ class MoneroAccountEditOrCreatePage extends BasePage {
|
||||||
child: Center(
|
child: Center(
|
||||||
child: BaseTextFormField(
|
child: BaseTextFormField(
|
||||||
controller: _textController,
|
controller: _textController,
|
||||||
|
textColor: Colors.white,
|
||||||
hintText: S.of(context).account,
|
hintText: S.of(context).account,
|
||||||
validator: MoneroLabelValidator(),
|
validator: MoneroLabelValidator(),
|
||||||
))),
|
))),
|
||||||
|
|
|
@ -7,35 +7,40 @@ import 'package:cake_wallet/routes.dart';
|
||||||
import 'package:cake_wallet/generated/i18n.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/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/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 {
|
class MoneroAccountListPage extends StatelessWidget {
|
||||||
MoneroAccountListPage({@required this.accountListViewModel});
|
MoneroAccountListPage({@required this.accountListViewModel}) {
|
||||||
|
backgroundHeight = 194;
|
||||||
|
thumbHeight = 72;
|
||||||
|
isAlwaysShowScrollThumb = false;
|
||||||
|
controller = ScrollController();
|
||||||
|
|
||||||
final MoneroAccountListViewModel accountListViewModel;
|
controller.addListener(() {
|
||||||
|
final scrollOffsetFromTop = controller.hasClients
|
||||||
@override
|
? (controller.offset / controller.position.maxScrollExtent * (backgroundHeight - thumbHeight))
|
||||||
MoneroAccountListPageForm createState() =>
|
: 0.0;
|
||||||
MoneroAccountListPageForm(accountListViewModel);
|
accountListViewModel.setScrollOffsetFromTop(scrollOffsetFromTop);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
class MoneroAccountListPageForm extends State<MoneroAccountListPage> {
|
|
||||||
MoneroAccountListPageForm(this.accountListViewModel);
|
|
||||||
|
|
||||||
final MoneroAccountListViewModel accountListViewModel;
|
final MoneroAccountListViewModel accountListViewModel;
|
||||||
|
final closeIcon = Image.asset('assets/images/close.png');
|
||||||
|
|
||||||
|
ScrollController controller;
|
||||||
|
double backgroundHeight;
|
||||||
|
double thumbHeight;
|
||||||
|
bool isAlwaysShowScrollThumb;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return GestureDetector(
|
return AlertBackground(
|
||||||
onTap: () => Navigator.of(context).pop(),
|
child: Stack(
|
||||||
child: Container(
|
alignment: Alignment.center,
|
||||||
color: Colors.transparent,
|
children: <Widget>[
|
||||||
child: BackdropFilter(
|
Column(
|
||||||
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,
|
mainAxisSize: MainAxisSize.min,
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
Container(
|
Container(
|
||||||
|
@ -47,7 +52,8 @@ class MoneroAccountListPageForm extends State<MoneroAccountListPage> {
|
||||||
fontSize: 18,
|
fontSize: 18,
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
decoration: TextDecoration.none,
|
decoration: TextDecoration.none,
|
||||||
color: Colors.white),
|
color: Colors.white
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Padding(
|
Padding(
|
||||||
|
@ -58,20 +64,27 @@ class MoneroAccountListPageForm extends State<MoneroAccountListPage> {
|
||||||
borderRadius: BorderRadius.all(Radius.circular(14)),
|
borderRadius: BorderRadius.all(Radius.circular(14)),
|
||||||
child: Container(
|
child: Container(
|
||||||
height: 296,
|
height: 296,
|
||||||
color: Theme.of(context)
|
color: PaletteDark.deepPurpleBlue,
|
||||||
.accentTextTheme
|
|
||||||
.title
|
|
||||||
.backgroundColor,
|
|
||||||
child: Column(
|
child: Column(
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
Expanded(child: Observer(builder: (_) {
|
Expanded(
|
||||||
final accounts =
|
child: Observer(
|
||||||
widget.accountListViewModel.accounts;
|
builder: (_) {
|
||||||
|
final accounts = accountListViewModel.accounts;
|
||||||
|
isAlwaysShowScrollThumb = accounts == null
|
||||||
|
? false
|
||||||
|
: accounts.length > 3;
|
||||||
|
|
||||||
return ListView.separated(
|
return Stack(
|
||||||
separatorBuilder: (context, index) => Divider(
|
alignment: Alignment.center,
|
||||||
color: Theme.of(context).dividerColor,
|
children: <Widget>[
|
||||||
height: 1),
|
ListView.separated(
|
||||||
|
controller: controller,
|
||||||
|
separatorBuilder: (context, index) =>
|
||||||
|
Container(
|
||||||
|
height: 1,
|
||||||
|
color: PaletteDark.dividerColor,
|
||||||
|
),
|
||||||
itemCount: accounts.length ?? 0,
|
itemCount: accounts.length ?? 0,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
final account = accounts[index];
|
final account = accounts[index];
|
||||||
|
@ -84,13 +97,25 @@ class MoneroAccountListPageForm extends State<MoneroAccountListPage> {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
widget.accountListViewModel
|
accountListViewModel
|
||||||
.select(account);
|
.select(account);
|
||||||
Navigator.of(context).pop();
|
Navigator.of(context).pop();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
),
|
||||||
|
isAlwaysShowScrollThumb
|
||||||
|
? CakeScrollbar(
|
||||||
|
backgroundHeight: backgroundHeight,
|
||||||
|
thumbHeight: thumbHeight,
|
||||||
|
fromTop: accountListViewModel
|
||||||
|
.scrollOffsetFromTop
|
||||||
|
)
|
||||||
|
: Offstage(),
|
||||||
|
],
|
||||||
);
|
);
|
||||||
})),
|
}
|
||||||
|
)
|
||||||
|
),
|
||||||
GestureDetector(
|
GestureDetector(
|
||||||
onTap: () async => await Navigator.of(context)
|
onTap: () async => await Navigator.of(context)
|
||||||
.pushNamed(Routes.accountCreation),
|
.pushNamed(Routes.accountCreation),
|
||||||
|
@ -131,9 +156,8 @@ class MoneroAccountListPageForm extends State<MoneroAccountListPage> {
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
AlertCloseButton(image: closeIcon)
|
||||||
),
|
],
|
||||||
),
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:cake_wallet/palette.dart';
|
||||||
|
|
||||||
class AccountTile extends StatelessWidget {
|
class AccountTile extends StatelessWidget {
|
||||||
AccountTile({
|
AccountTile({
|
||||||
|
@ -13,8 +14,8 @@ class AccountTile extends StatelessWidget {
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final color = isCurrent ? Theme.of(context).accentTextTheme.subtitle.decorationColor : Colors.transparent;
|
final color = isCurrent ? PaletteDark.lightOceanBlue : Colors.transparent;
|
||||||
final textColor = isCurrent ? Colors.blue : Theme.of(context).primaryTextTheme.title.color;
|
final textColor = isCurrent ? Colors.blue : Colors.white;
|
||||||
|
|
||||||
return GestureDetector(
|
return GestureDetector(
|
||||||
onTap: onTap,
|
onTap: onTap,
|
||||||
|
|
|
@ -1,68 +1,43 @@
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/cupertino.dart';
|
import 'package:flutter/cupertino.dart';
|
||||||
import 'package:flutter/services.dart';
|
|
||||||
import 'package:flutter_mobx/flutter_mobx.dart';
|
import 'package:flutter_mobx/flutter_mobx.dart';
|
||||||
import 'package:esys_flutter_share/esys_flutter_share.dart';
|
import 'package:esys_flutter_share/esys_flutter_share.dart';
|
||||||
import 'package:cake_wallet/routes.dart';
|
import 'package:cake_wallet/routes.dart';
|
||||||
import 'package:cake_wallet/generated/i18n.dart';
|
import 'package:cake_wallet/generated/i18n.dart';
|
||||||
import 'package:cake_wallet/di.dart';
|
import 'package:cake_wallet/di.dart';
|
||||||
import 'package:cake_wallet/src/screens/base_page.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/monero_accounts/monero_account_list_page.dart';
|
||||||
import 'package:cake_wallet/src/screens/receive/widgets/header_tile.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/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_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_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_item.dart';
|
||||||
import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_view_model.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 {
|
class ReceivePage extends BasePage {
|
||||||
ReceivePage({this.addressListViewModel})
|
ReceivePage({this.addressListViewModel});
|
||||||
: amountController = TextEditingController(),
|
|
||||||
_formKey = GlobalKey<FormState>() {
|
|
||||||
amountController.addListener(() => addressListViewModel.amount =
|
|
||||||
_formKey.currentState.validate() ? amountController.text : '');
|
|
||||||
}
|
|
||||||
|
|
||||||
final WalletAddressListViewModel addressListViewModel;
|
final WalletAddressListViewModel addressListViewModel;
|
||||||
final TextEditingController amountController;
|
|
||||||
final GlobalKey<FormState> _formKey;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Color get backgroundLightColor => Colors.transparent;
|
String get title => S.current.receive;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Color get backgroundDarkColor => Colors.transparent;
|
Color get backgroundLightColor => PaletteDark.backgroundColor;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget Function(BuildContext, Widget) get rootWrapper =>
|
Color get backgroundDarkColor => PaletteDark.backgroundColor;
|
||||||
(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),
|
|
||||||
);
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget trailing(BuildContext context) {
|
Widget trailing(BuildContext context) {
|
||||||
final shareImage = Image.asset('assets/images/share.png',
|
final shareImage = Image.asset('assets/images/share.png',
|
||||||
color: Theme.of(context).primaryTextTheme.title.color);
|
color: Colors.white);
|
||||||
|
|
||||||
return SizedBox(
|
return SizedBox(
|
||||||
height: 20.0,
|
height: 20.0,
|
||||||
width: 14.0,
|
width: 20.0,
|
||||||
child: ButtonTheme(
|
child: ButtonTheme(
|
||||||
minWidth: double.minPositive,
|
minWidth: double.minPositive,
|
||||||
child: FlatButton(
|
child: FlatButton(
|
||||||
|
@ -78,122 +53,20 @@ class ReceivePage extends BasePage {
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget body(BuildContext context) {
|
Widget body(BuildContext context) {
|
||||||
final copyImage = Image.asset('assets/images/copy_content.png',
|
|
||||||
color: Theme.of(context).primaryTextTheme.title.color);
|
|
||||||
|
|
||||||
return SingleChildScrollView(
|
return SingleChildScrollView(
|
||||||
child: Column(
|
child: Column(
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
SizedBox(height: 25),
|
|
||||||
Row(children: <Widget>[
|
|
||||||
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(
|
||||||
padding: EdgeInsets.fromLTRB(24, 40, 24, 0),
|
padding: EdgeInsets.fromLTRB(24, 80, 24, 40),
|
||||||
child: Row(
|
child: QRWidget(
|
||||||
children: <Widget>[
|
addressListViewModel: addressListViewModel,
|
||||||
Expanded(
|
isAmountFieldShow: true,
|
||||||
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(),
|
|
||||||
autovalidate: true,
|
|
||||||
placeholderTextStyle: TextStyle(
|
|
||||||
color: Theme.of(context)
|
|
||||||
.primaryTextTheme
|
|
||||||
.headline5
|
|
||||||
.color,
|
|
||||||
fontSize: 20,
|
|
||||||
fontWeight: FontWeight.w600))))
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
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: <Widget>[
|
|
||||||
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(
|
Observer(
|
||||||
builder: (_) => ListView.separated(
|
builder: (_) => ListView.separated(
|
||||||
separatorBuilder: (context, _) =>
|
separatorBuilder: (context, _) =>
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
Container(height: 1, color: PaletteDark.dividerColor),
|
||||||
shrinkWrap: true,
|
shrinkWrap: true,
|
||||||
physics: NeverScrollableScrollPhysics(),
|
physics: NeverScrollableScrollPhysics(),
|
||||||
itemCount: addressListViewModel.items.length,
|
itemCount: addressListViewModel.items.length,
|
||||||
|
@ -210,8 +83,7 @@ class ReceivePage extends BasePage {
|
||||||
icon: Icon(
|
icon: Icon(
|
||||||
Icons.arrow_forward_ios,
|
Icons.arrow_forward_ios,
|
||||||
size: 14,
|
size: 14,
|
||||||
color:
|
color: Colors.white,
|
||||||
Theme.of(context).primaryTextTheme.title.color,
|
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -223,28 +95,40 @@ class ReceivePage extends BasePage {
|
||||||
icon: Icon(
|
icon: Icon(
|
||||||
Icons.add,
|
Icons.add,
|
||||||
size: 20,
|
size: 20,
|
||||||
color:
|
color: Colors.white,
|
||||||
Theme.of(context).primaryTextTheme.title.color,
|
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (item is WalletAddressListItem) {
|
if (item is WalletAddressListItem) {
|
||||||
cell = Observer(
|
cell = Observer(
|
||||||
builder: (_) => AddressCell.fromItem(item,
|
builder: (_) {
|
||||||
isCurrent: item.address ==
|
final isCurrent = item.address ==
|
||||||
addressListViewModel.address.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,
|
onTap: (_) => addressListViewModel.address = item,
|
||||||
onEdit: () => Navigator.of(context).pushNamed(
|
onEdit: () => Navigator.of(context).pushNamed(
|
||||||
Routes.newSubaddress,
|
Routes.newSubaddress,
|
||||||
arguments: item)));
|
arguments: item));
|
||||||
|
}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return index != 0
|
return index != 0
|
||||||
? cell
|
? cell
|
||||||
: ClipRRect(
|
: ClipRRect(
|
||||||
borderRadius: BorderRadius.only(
|
borderRadius: BorderRadius.only(
|
||||||
topLeft: Radius.circular(24),
|
topLeft: Radius.circular(30),
|
||||||
topRight: Radius.circular(24)),
|
topRight: Radius.circular(30)),
|
||||||
child: cell,
|
child: cell,
|
||||||
);
|
);
|
||||||
})),
|
})),
|
||||||
|
|
|
@ -6,12 +6,16 @@ import 'package:cake_wallet/view_model/wallet_address_list/wallet_address_list_i
|
||||||
class AddressCell extends StatelessWidget {
|
class AddressCell extends StatelessWidget {
|
||||||
factory AddressCell.fromItem(WalletAddressListItem item,
|
factory AddressCell.fromItem(WalletAddressListItem item,
|
||||||
{@required bool isCurrent,
|
{@required bool isCurrent,
|
||||||
|
@required Color backgroundColor,
|
||||||
|
@required Color textColor,
|
||||||
Function(String) onTap,
|
Function(String) onTap,
|
||||||
Function() onEdit}) =>
|
Function() onEdit}) =>
|
||||||
AddressCell(
|
AddressCell(
|
||||||
address: item.address,
|
address: item.address,
|
||||||
name: item.name,
|
name: item.name,
|
||||||
isCurrent: isCurrent,
|
isCurrent: isCurrent,
|
||||||
|
backgroundColor: backgroundColor,
|
||||||
|
textColor: textColor,
|
||||||
onTap: onTap,
|
onTap: onTap,
|
||||||
onEdit: onEdit);
|
onEdit: onEdit);
|
||||||
|
|
||||||
|
@ -19,12 +23,16 @@ class AddressCell extends StatelessWidget {
|
||||||
{@required this.address,
|
{@required this.address,
|
||||||
@required this.name,
|
@required this.name,
|
||||||
@required this.isCurrent,
|
@required this.isCurrent,
|
||||||
|
@required this.backgroundColor,
|
||||||
|
@required this.textColor,
|
||||||
this.onTap,
|
this.onTap,
|
||||||
this.onEdit});
|
this.onEdit});
|
||||||
|
|
||||||
final String address;
|
final String address;
|
||||||
final String name;
|
final String name;
|
||||||
final bool isCurrent;
|
final bool isCurrent;
|
||||||
|
final Color backgroundColor;
|
||||||
|
final Color textColor;
|
||||||
final Function(String) onTap;
|
final Function(String) onTap;
|
||||||
final Function() onEdit;
|
final Function() onEdit;
|
||||||
|
|
||||||
|
@ -32,23 +40,19 @@ class AddressCell extends StatelessWidget {
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
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(
|
final Widget cell = InkWell(
|
||||||
onTap: () => onTap(address),
|
onTap: () => onTap(address),
|
||||||
child: Container(
|
child: Container(
|
||||||
color: isCurrent ? currentColor : notCurrentColor,
|
color: backgroundColor,
|
||||||
padding: EdgeInsets.only(left: 24, right: 24, top: 28, bottom: 28),
|
padding: EdgeInsets.only(left: 24, right: 24, top: 28, bottom: 28),
|
||||||
child: Text(
|
child: Text(
|
||||||
name ?? address,
|
name ?? address,
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: name?.isNotEmpty ?? false ? 18 : 10,
|
fontSize: 18,
|
||||||
fontWeight: FontWeight.w600,
|
fontWeight: FontWeight.w600,
|
||||||
color: isCurrent ? currentTextColor : notCurrentTextColor,
|
color: textColor,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
));
|
));
|
||||||
|
@ -62,7 +66,7 @@ class AddressCell extends StatelessWidget {
|
||||||
secondaryActions: <Widget>[
|
secondaryActions: <Widget>[
|
||||||
IconSlideAction(
|
IconSlideAction(
|
||||||
caption: S.of(context).edit,
|
caption: S.of(context).edit,
|
||||||
color: Theme.of(context).primaryTextTheme.overline.color,
|
color: Colors.blue,
|
||||||
icon: Icons.edit,
|
icon: Icons.edit,
|
||||||
onTap: () => onEdit?.call())
|
onTap: () => onEdit?.call())
|
||||||
]);
|
]);
|
||||||
|
|
|
@ -23,7 +23,7 @@ class HeaderTile extends StatelessWidget {
|
||||||
top: 24,
|
top: 24,
|
||||||
bottom: 24
|
bottom: 24
|
||||||
),
|
),
|
||||||
color: Theme.of(context).backgroundColor,
|
color: PaletteDark.nightBlue,
|
||||||
child: Row(
|
child: Row(
|
||||||
mainAxisSize: MainAxisSize.max,
|
mainAxisSize: MainAxisSize.max,
|
||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
@ -33,7 +33,7 @@ class HeaderTile extends StatelessWidget {
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 18,
|
fontSize: 18,
|
||||||
fontWeight: FontWeight.w500,
|
fontWeight: FontWeight.w500,
|
||||||
color: Theme.of(context).primaryTextTheme.title.color
|
color: Colors.white
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Container(
|
Container(
|
||||||
|
@ -41,7 +41,7 @@ class HeaderTile extends StatelessWidget {
|
||||||
width: 32,
|
width: 32,
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
shape: BoxShape.circle,
|
shape: BoxShape.circle,
|
||||||
color: Theme.of(context).primaryTextTheme.overline.color
|
color: PaletteDark.distantNightBlue
|
||||||
),
|
),
|
||||||
child: icon,
|
child: icon,
|
||||||
)
|
)
|
||||||
|
|
137
lib/src/screens/receive/widgets/qr_widget.dart
Normal file
|
@ -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<FormState>() {
|
||||||
|
amountController.addListener(() => addressListViewModel.amount =
|
||||||
|
_formKey.currentState.validate() ? amountController.text : '');
|
||||||
|
}
|
||||||
|
|
||||||
|
final WalletAddressListViewModel addressListViewModel;
|
||||||
|
final bool isAmountFieldShow;
|
||||||
|
final TextEditingController amountController;
|
||||||
|
final GlobalKey<FormState> _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: <Widget>[
|
||||||
|
Row(children: <Widget>[
|
||||||
|
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: <Widget>[
|
||||||
|
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: <Widget>[
|
||||||
|
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,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
))),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
|
@ -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/primary_button.dart';
|
||||||
import 'package:cake_wallet/src/widgets/base_text_form_field.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/src/screens/base_page.dart';
|
||||||
|
import 'package:cake_wallet/palette.dart';
|
||||||
|
|
||||||
class AddressEditOrCreatePage extends BasePage {
|
class AddressEditOrCreatePage extends BasePage {
|
||||||
AddressEditOrCreatePage({@required this.addressEditOrCreateViewModel})
|
AddressEditOrCreatePage({@required this.addressEditOrCreateViewModel})
|
||||||
|
@ -28,6 +29,12 @@ class AddressEditOrCreatePage extends BasePage {
|
||||||
@override
|
@override
|
||||||
String get title => S.current.new_subaddress_title;
|
String get title => S.current.new_subaddress_title;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Color get backgroundLightColor => PaletteDark.backgroundColor;
|
||||||
|
|
||||||
|
@override
|
||||||
|
Color get backgroundDarkColor => PaletteDark.backgroundColor;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget body(BuildContext context) {
|
Widget body(BuildContext context) {
|
||||||
reaction((_) => addressEditOrCreateViewModel.state,
|
reaction((_) => addressEditOrCreateViewModel.state,
|
||||||
|
@ -48,6 +55,7 @@ class AddressEditOrCreatePage extends BasePage {
|
||||||
child: Center(
|
child: Center(
|
||||||
child: BaseTextFormField(
|
child: BaseTextFormField(
|
||||||
controller: _labelController,
|
controller: _labelController,
|
||||||
|
textColor: Colors.white,
|
||||||
hintText: S.of(context).new_subaddress_label_name,
|
hintText: S.of(context).new_subaddress_label_name,
|
||||||
validator: AddressLabelValidator()))),
|
validator: AddressLabelValidator()))),
|
||||||
Observer(
|
Observer(
|
||||||
|
|
26
lib/src/widgets/alert_background.dart
Normal file
|
@ -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,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
28
lib/src/widgets/alert_close_button.dart
Normal file
|
@ -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,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
45
lib/src/widgets/cake_scrollbar.dart
Normal file
|
@ -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: <Widget>[
|
||||||
|
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))
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
19
lib/store/dashboard/page_view_store.dart
Normal file
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
|
@ -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/trade_filter_store.dart';
|
||||||
import 'package:cake_wallet/store/dashboard/transaction_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/view_model/dashboard/formatted_item_list.dart';
|
||||||
|
import 'package:cake_wallet/store/dashboard/page_view_store.dart';
|
||||||
|
|
||||||
part 'dashboard_view_model.g.dart';
|
part 'dashboard_view_model.g.dart';
|
||||||
|
|
||||||
|
@ -33,7 +34,8 @@ abstract class DashboardViewModelBase with Store {
|
||||||
this.appStore,
|
this.appStore,
|
||||||
this.tradesStore,
|
this.tradesStore,
|
||||||
this.tradeFilterStore,
|
this.tradeFilterStore,
|
||||||
this.transactionFilterStore}) {
|
this.transactionFilterStore,
|
||||||
|
this.pageViewStore}) {
|
||||||
|
|
||||||
name = appStore.wallet?.name;
|
name = appStore.wallet?.name;
|
||||||
wallet ??= appStore.wallet;
|
wallet ??= appStore.wallet;
|
||||||
|
@ -53,8 +55,6 @@ abstract class DashboardViewModelBase with Store {
|
||||||
if (_wallet is MoneroWallet) {
|
if (_wallet is MoneroWallet) {
|
||||||
subname = _wallet.account?.label;
|
subname = _wallet.account?.label;
|
||||||
}
|
}
|
||||||
|
|
||||||
currentPage = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@observable
|
@observable
|
||||||
|
@ -63,15 +63,15 @@ abstract class DashboardViewModelBase with Store {
|
||||||
@observable
|
@observable
|
||||||
String name;
|
String name;
|
||||||
|
|
||||||
@observable
|
|
||||||
double currentPage;
|
|
||||||
|
|
||||||
@observable
|
@observable
|
||||||
ObservableList<TransactionListItem> transactions;
|
ObservableList<TransactionListItem> transactions;
|
||||||
|
|
||||||
@observable
|
@observable
|
||||||
String subname;
|
String subname;
|
||||||
|
|
||||||
|
@computed
|
||||||
|
double get currentPage => pageViewStore.currentPage;
|
||||||
|
|
||||||
@computed
|
@computed
|
||||||
String get address => wallet.address;
|
String get address => wallet.address;
|
||||||
|
|
||||||
|
@ -130,6 +130,8 @@ abstract class DashboardViewModelBase with Store {
|
||||||
|
|
||||||
TransactionFilterStore transactionFilterStore;
|
TransactionFilterStore transactionFilterStore;
|
||||||
|
|
||||||
|
PageViewStore pageViewStore;
|
||||||
|
|
||||||
ReactionDisposer _reaction;
|
ReactionDisposer _reaction;
|
||||||
|
|
||||||
void _onWalletChange(WalletBase wallet) {
|
void _onWalletChange(WalletBase wallet) {
|
||||||
|
|
|
@ -9,7 +9,15 @@ class MoneroAccountListViewModel = MoneroAccountListViewModelBase
|
||||||
with _$MoneroAccountListViewModel;
|
with _$MoneroAccountListViewModel;
|
||||||
|
|
||||||
abstract class MoneroAccountListViewModelBase with Store {
|
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
|
@computed
|
||||||
List<AccountListItem> get accounts => _moneroWallet.accountList.accounts
|
List<AccountListItem> get accounts => _moneroWallet.accountList.accounts
|
||||||
|
|
|
@ -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_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_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_item.dart';
|
||||||
|
import 'package:cake_wallet/src/domain/common/wallet_type.dart';
|
||||||
|
|
||||||
part 'wallet_address_list_view_model.g.dart';
|
part 'wallet_address_list_view_model.g.dart';
|
||||||
|
|
||||||
|
@ -62,6 +63,9 @@ abstract class WalletAddressListViewModelBase with Store {
|
||||||
@observable
|
@observable
|
||||||
String amount;
|
String amount;
|
||||||
|
|
||||||
|
@computed
|
||||||
|
WalletType get type => _wallet.type;
|
||||||
|
|
||||||
@computed
|
@computed
|
||||||
WalletAddressListItem get address => WalletAddressListItem(address: _wallet.address);
|
WalletAddressListItem get address => WalletAddressListItem(address: _wallet.address);
|
||||||
|
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "Adresse teilen ",
|
"share_address" : "Adresse teilen ",
|
||||||
"receive_amount" : "Menge",
|
"receive_amount" : "Menge",
|
||||||
"subaddresses" : "Unteradressen",
|
"subaddresses" : "Unteradressen",
|
||||||
|
"addresses" : "Adressen",
|
||||||
|
"scan_qr_code" : "Scannen Sie den QR-Code, um die Adresse zu erhalten",
|
||||||
"rename" : "Umbenennen",
|
"rename" : "Umbenennen",
|
||||||
"choose_account" : "Konto auswählen",
|
"choose_account" : "Konto auswählen",
|
||||||
"create_new_account" : "Neues Konto erstellen",
|
"create_new_account" : "Neues Konto erstellen",
|
||||||
|
|
|
@ -138,6 +138,7 @@
|
||||||
"receive_amount" : "Amount",
|
"receive_amount" : "Amount",
|
||||||
"subaddresses" : "Subaddresses",
|
"subaddresses" : "Subaddresses",
|
||||||
"addresses" : "Addresses",
|
"addresses" : "Addresses",
|
||||||
|
"scan_qr_code" : "Scan the QR code to get the address",
|
||||||
"rename" : "Rename",
|
"rename" : "Rename",
|
||||||
"choose_account" : "Choose account",
|
"choose_account" : "Choose account",
|
||||||
"create_new_account" : "Create new account",
|
"create_new_account" : "Create new account",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "Compartir dirección",
|
"share_address" : "Compartir dirección",
|
||||||
"receive_amount" : "Cantidad",
|
"receive_amount" : "Cantidad",
|
||||||
"subaddresses" : "Subdirecciones",
|
"subaddresses" : "Subdirecciones",
|
||||||
|
"addresses" : "Direcciones",
|
||||||
|
"scan_qr_code" : "Escanee el código QR para obtener la dirección",
|
||||||
"rename" : "Rebautizar",
|
"rename" : "Rebautizar",
|
||||||
"choose_account" : "Elegir cuenta",
|
"choose_account" : "Elegir cuenta",
|
||||||
"create_new_account" : "Crear una nueva cuenta",
|
"create_new_account" : "Crear una nueva cuenta",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "पता साझा करें",
|
"share_address" : "पता साझा करें",
|
||||||
"receive_amount" : "रकम",
|
"receive_amount" : "रकम",
|
||||||
"subaddresses" : "उप पते",
|
"subaddresses" : "उप पते",
|
||||||
|
"addresses" : "पतों",
|
||||||
|
"scan_qr_code" : "पता प्राप्त करने के लिए QR कोड स्कैन करें",
|
||||||
"rename" : "नाम बदलें",
|
"rename" : "नाम बदलें",
|
||||||
"choose_account" : "खाता चुनें",
|
"choose_account" : "खाता चुनें",
|
||||||
"create_new_account" : "नया खाता बनाएँ",
|
"create_new_account" : "नया खाता बनाएँ",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "住所を共有する",
|
"share_address" : "住所を共有する",
|
||||||
"receive_amount" : "量",
|
"receive_amount" : "量",
|
||||||
"subaddresses" : "サブアドレス",
|
"subaddresses" : "サブアドレス",
|
||||||
|
"addresses" : "住所",
|
||||||
|
"scan_qr_code" : "QRコードをスキャンして住所を取得します",
|
||||||
"rename" : "リネーム",
|
"rename" : "リネーム",
|
||||||
"choose_account" : "アカウントを選択",
|
"choose_account" : "アカウントを選択",
|
||||||
"create_new_account" : "新しいアカウントを作成する",
|
"create_new_account" : "新しいアカウントを作成する",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "주소 공유",
|
"share_address" : "주소 공유",
|
||||||
"receive_amount" : "양",
|
"receive_amount" : "양",
|
||||||
"subaddresses" : "하위 주소",
|
"subaddresses" : "하위 주소",
|
||||||
|
"addresses" : "구애",
|
||||||
|
"scan_qr_code" : "QR 코드를 스캔하여 주소를 얻습니다.",
|
||||||
"rename" : "이름 바꾸기",
|
"rename" : "이름 바꾸기",
|
||||||
"choose_account" : "계정을 선택하십시오",
|
"choose_account" : "계정을 선택하십시오",
|
||||||
"create_new_account" : "새 계정을 만들",
|
"create_new_account" : "새 계정을 만들",
|
||||||
|
|
|
@ -138,6 +138,8 @@
|
||||||
"receive_amount" : "Bedrag",
|
"receive_amount" : "Bedrag",
|
||||||
"subaddresses" : "Subadressen",
|
"subaddresses" : "Subadressen",
|
||||||
"rename" : "Hernoemen",
|
"rename" : "Hernoemen",
|
||||||
|
"addresses" : "Adressen",
|
||||||
|
"scan_qr_code" : "Scan de QR-code om het adres te krijgen",
|
||||||
"choose_account" : "Kies account",
|
"choose_account" : "Kies account",
|
||||||
"create_new_account" : "Creëer een nieuw account",
|
"create_new_account" : "Creëer een nieuw account",
|
||||||
"accounts_subaddresses" : "Accounts en subadressen",
|
"accounts_subaddresses" : "Accounts en subadressen",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "Udostępnij adres",
|
"share_address" : "Udostępnij adres",
|
||||||
"receive_amount" : "Ilość",
|
"receive_amount" : "Ilość",
|
||||||
"subaddresses" : "Podadresy",
|
"subaddresses" : "Podadresy",
|
||||||
|
"addresses" : "Adresy",
|
||||||
|
"scan_qr_code" : "Zeskanuj kod QR, aby uzyskać adres",
|
||||||
"rename" : "Przemianować",
|
"rename" : "Przemianować",
|
||||||
"choose_account" : "Wybierz konto",
|
"choose_account" : "Wybierz konto",
|
||||||
"create_new_account" : "Stwórz nowe konto",
|
"create_new_account" : "Stwórz nowe konto",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "Compartilhar endereço",
|
"share_address" : "Compartilhar endereço",
|
||||||
"receive_amount" : "Quantia",
|
"receive_amount" : "Quantia",
|
||||||
"subaddresses" : "Sub-endereços",
|
"subaddresses" : "Sub-endereços",
|
||||||
|
"addresses" : "Endereços",
|
||||||
|
"scan_qr_code" : "Digitalize o código QR para obter o endereço",
|
||||||
"rename" : "Renomear",
|
"rename" : "Renomear",
|
||||||
"choose_account" : "Escolha uma conta",
|
"choose_account" : "Escolha uma conta",
|
||||||
"create_new_account" : "Criar nova conta",
|
"create_new_account" : "Criar nova conta",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "Поделиться адресом",
|
"share_address" : "Поделиться адресом",
|
||||||
"receive_amount" : "Сумма",
|
"receive_amount" : "Сумма",
|
||||||
"subaddresses" : "Субадреса",
|
"subaddresses" : "Субадреса",
|
||||||
|
"addresses" : "Адреса",
|
||||||
|
"scan_qr_code" : "Отсканируйте QR-код для получения адреса",
|
||||||
"rename" : "Переименовать",
|
"rename" : "Переименовать",
|
||||||
"choose_account" : "Выберите аккаунт",
|
"choose_account" : "Выберите аккаунт",
|
||||||
"create_new_account" : "Создать новый аккаунт",
|
"create_new_account" : "Создать новый аккаунт",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "Поділитися адресою",
|
"share_address" : "Поділитися адресою",
|
||||||
"receive_amount" : "Сума",
|
"receive_amount" : "Сума",
|
||||||
"subaddresses" : "Субадреси",
|
"subaddresses" : "Субадреси",
|
||||||
|
"addresses" : "Адреси",
|
||||||
|
"scan_qr_code" : "Скануйте QR-код для одержання адреси",
|
||||||
"rename" : "Перейменувати",
|
"rename" : "Перейменувати",
|
||||||
"choose_account" : "Оберіть акаунт",
|
"choose_account" : "Оберіть акаунт",
|
||||||
"create_new_account" : "Створити новий акаунт",
|
"create_new_account" : "Створити новий акаунт",
|
||||||
|
|
|
@ -137,6 +137,8 @@
|
||||||
"share_address" : "分享地址",
|
"share_address" : "分享地址",
|
||||||
"receive_amount" : "量",
|
"receive_amount" : "量",
|
||||||
"subaddresses" : "子地址",
|
"subaddresses" : "子地址",
|
||||||
|
"addresses" : "地址",
|
||||||
|
"scan_qr_code" : "掃描二維碼獲取地址",
|
||||||
"rename" : "改名",
|
"rename" : "改名",
|
||||||
"choose_account" : "選擇帳號",
|
"choose_account" : "選擇帳號",
|
||||||
"create_new_account" : "建立新帳戶",
|
"create_new_account" : "建立新帳戶",
|
||||||
|
|