Merge branch 'main' into CAKE-345-batch-sending

# Conflicts:
#	lib/bitcoin/electrum_wallet.dart
wownero
OleksandrSobol 3 years ago
commit 3abb78febf

6
.gitignore vendored

@ -100,4 +100,8 @@ ios/Flutter/.last_build_id
/**/#**# /**/#**#
**/google-services.json **/google-services.json
**/GoogleService-Info.plist **/GoogleService-Info.plist
\#*\#
.\#*

@ -366,7 +366,7 @@
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES; CLANG_ENABLE_MODULES = YES;
CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements;
CURRENT_PROJECT_VERSION = 51; CURRENT_PROJECT_VERSION = 53;
DEVELOPMENT_TEAM = 32J6BB6VUS; DEVELOPMENT_TEAM = 32J6BB6VUS;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
EXCLUDED_SOURCE_FILE_NAMES = ""; EXCLUDED_SOURCE_FILE_NAMES = "";
@ -510,7 +510,7 @@
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES; CLANG_ENABLE_MODULES = YES;
CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements;
CURRENT_PROJECT_VERSION = 51; CURRENT_PROJECT_VERSION = 53;
DEVELOPMENT_TEAM = 32J6BB6VUS; DEVELOPMENT_TEAM = 32J6BB6VUS;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
EXCLUDED_SOURCE_FILE_NAMES = ""; EXCLUDED_SOURCE_FILE_NAMES = "";
@ -546,7 +546,7 @@
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
CLANG_ENABLE_MODULES = YES; CLANG_ENABLE_MODULES = YES;
CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements;
CURRENT_PROJECT_VERSION = 51; CURRENT_PROJECT_VERSION = 53;
DEVELOPMENT_TEAM = 32J6BB6VUS; DEVELOPMENT_TEAM = 32J6BB6VUS;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
EXCLUDED_SOURCE_FILE_NAMES = ""; EXCLUDED_SOURCE_FILE_NAMES = "";

@ -1,4 +1,4 @@
class BitcoinTransactionNoInputsException implements Exception { class BitcoinTransactionNoInputsException implements Exception {
@override @override
String toString() => 'No inputs for the transaction.'; String toString() => 'Not enough inputs available';
} }

@ -188,6 +188,7 @@ class ExchangePage extends BasePage {
exchangeViewModel.isDepositAddressEnabled, exchangeViewModel.isDepositAddressEnabled,
isAmountEstimated: false, isAmountEstimated: false,
hasRefundAddress: true, hasRefundAddress: true,
isMoneroWallet: exchangeViewModel.isMoneroWallet,
currencies: CryptoCurrency.all, currencies: CryptoCurrency.all,
onCurrencySelected: (currency) { onCurrencySelected: (currency) {
// FIXME: need to move it into view model // FIXME: need to move it into view model
@ -260,6 +261,7 @@ class ExchangePage extends BasePage {
exchangeViewModel exchangeViewModel
.isReceiveAddressEnabled, .isReceiveAddressEnabled,
isAmountEstimated: true, isAmountEstimated: true,
isMoneroWallet: exchangeViewModel.isMoneroWallet,
currencies: currencies:
exchangeViewModel.receiveCurrencies, exchangeViewModel.receiveCurrencies,
onCurrencySelected: (currency) => onCurrencySelected: (currency) =>

@ -136,6 +136,7 @@ class ExchangeTemplatePage extends BasePage {
.isDepositAddressEnabled, .isDepositAddressEnabled,
isAmountEstimated: false, isAmountEstimated: false,
hasRefundAddress: true, hasRefundAddress: true,
isMoneroWallet: exchangeViewModel.isMoneroWallet,
currencies: CryptoCurrency.all, currencies: CryptoCurrency.all,
onCurrencySelected: (currency) => onCurrencySelected: (currency) =>
exchangeViewModel.changeDepositCurrency( exchangeViewModel.changeDepositCurrency(
@ -175,6 +176,7 @@ class ExchangeTemplatePage extends BasePage {
initialIsAddressEditable: initialIsAddressEditable:
exchangeViewModel.isReceiveAddressEnabled, exchangeViewModel.isReceiveAddressEnabled,
isAmountEstimated: true, isAmountEstimated: true,
isMoneroWallet: exchangeViewModel.isMoneroWallet,
currencies: exchangeViewModel.receiveCurrencies, currencies: exchangeViewModel.receiveCurrencies,
onCurrencySelected: (currency) => onCurrencySelected: (currency) =>
exchangeViewModel.changeReceiveCurrency( exchangeViewModel.changeReceiveCurrency(

@ -1,3 +1,5 @@
import 'package:cake_wallet/entities/contact_base.dart';
import 'package:cake_wallet/routes.dart';
import 'package:cake_wallet/utils/show_bar.dart'; import 'package:cake_wallet/utils/show_bar.dart';
import 'package:cake_wallet/utils/show_pop_up.dart'; import 'package:cake_wallet/utils/show_pop_up.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
@ -19,6 +21,7 @@ class ExchangeCard extends StatefulWidget {
this.initialIsAddressEditable, this.initialIsAddressEditable,
this.isAmountEstimated, this.isAmountEstimated,
this.hasRefundAddress = false, this.hasRefundAddress = false,
this.isMoneroWallet = false,
this.currencies, this.currencies,
this.onCurrencySelected, this.onCurrencySelected,
this.imageArrow, this.imageArrow,
@ -44,6 +47,7 @@ class ExchangeCard extends StatefulWidget {
final bool initialIsAddressEditable; final bool initialIsAddressEditable;
final bool isAmountEstimated; final bool isAmountEstimated;
final bool hasRefundAddress; final bool hasRefundAddress;
final bool isMoneroWallet;
final Image imageArrow; final Image imageArrow;
final Color currencyButtonColor; final Color currencyButtonColor;
final Color addressButtonsColor; final Color addressButtonsColor;
@ -72,6 +76,7 @@ class ExchangeCardState extends State<ExchangeCard> {
bool _isAmountEditable; bool _isAmountEditable;
bool _isAddressEditable; bool _isAddressEditable;
bool _isAmountEstimated; bool _isAmountEstimated;
bool _isMoneroWallet;
@override @override
void initState() { void initState() {
@ -81,6 +86,7 @@ class ExchangeCardState extends State<ExchangeCard> {
_walletName = widget.initialWalletName; _walletName = widget.initialWalletName;
_selectedCurrency = widget.initialCurrency; _selectedCurrency = widget.initialCurrency;
_isAmountEstimated = widget.isAmountEstimated; _isAmountEstimated = widget.isAmountEstimated;
_isMoneroWallet = widget.isMoneroWallet;
addressController.text = widget.initialAddress; addressController.text = widget.initialAddress;
super.initState(); super.initState();
} }
@ -322,34 +328,93 @@ class ExchangeCardState extends State<ExchangeCard> {
: Padding( : Padding(
padding: EdgeInsets.only(top: 10), padding: EdgeInsets.only(top: 10),
child: Builder( child: Builder(
builder: (context) => GestureDetector( builder: (context) => Stack(
onTap: () { children: <Widget> [
Clipboard.setData( BaseTextFormField(
ClipboardData(text: addressController.text)); controller: addressController,
showBar<void>( readOnly: true,
context, S.of(context).copied_to_clipboard); borderColor: Colors.transparent,
}, suffixIcon: SizedBox(
child: Row( width: _isMoneroWallet ? 80 : 36
mainAxisSize: MainAxisSize.max, ),
children: <Widget>[ textStyle: TextStyle(
Expanded( fontSize: 16,
child: Text( fontWeight: FontWeight.w600,
addressController.text, color: Colors.white),
maxLines: 1, validator: widget.addressTextFieldValidator
overflow: TextOverflow.ellipsis, ),
style: TextStyle( Positioned(
fontSize: 16, top: 2,
fontWeight: FontWeight.w600, right: 0,
color: Colors.white), child: SizedBox(
), width: _isMoneroWallet ? 80 : 36,
), child: Row(
Padding( children: <Widget>[
padding: EdgeInsets.only(left: 16), if (_isMoneroWallet) Padding(
child: copyImage, padding: EdgeInsets.only(left: 10),
child: Container(
width: 34,
height: 34,
padding: EdgeInsets.only(top: 0),
child: InkWell(
onTap: () async {
final contact = await Navigator
.of(context, rootNavigator: true)
.pushNamed(
Routes.pickerAddressBook);
if (contact is ContactBase &&
contact.address != null) {
setState(() =>
addressController.text =
contact.address);
}
},
child: Container(
padding: EdgeInsets.all(8),
decoration: BoxDecoration(
color: widget
.addressButtonsColor,
borderRadius: BorderRadius
.all(Radius.circular(6))),
child: Image.asset(
'assets/images/open_book.png',
color: Theme.of(context)
.primaryTextTheme
.display1
.decorationColor,
)),
)),
),
Padding(
padding: EdgeInsets.only(left: 2),
child: Container(
width: 34,
height: 34,
padding: EdgeInsets.only(top: 0),
child: InkWell(
onTap: () {
Clipboard.setData(
ClipboardData(
text: addressController.text));
showBar<void>(
context, S.of(context)
.copied_to_clipboard);
},
child: Container(
padding: EdgeInsets
.fromLTRB(8, 8, 0, 8),
color: Colors.transparent,
child: copyImage),
))
)
]
) )
], )
), )
)), ]
)
),
), ),
]), ]),
); );

@ -86,9 +86,11 @@ class UnspentCoinsListFormState extends State<UnspentCoinsListForm> {
.pushNamed(Routes.unspentCoinsDetails, .pushNamed(Routes.unspentCoinsDetails,
arguments: [item, unspentCoinsListViewModel]), arguments: [item, unspentCoinsListViewModel]),
child: UnspentCoinsListItem( child: UnspentCoinsListItem(
address: item.address, note: item.note,
amount: item.amount, amount: item.amount,
address: item.address,
isSending: item.isSending, isSending: item.isSending,
isFrozen: item.isFrozen,
onCheckBoxTap: item.isFrozen onCheckBoxTap: item.isFrozen
? null ? null
: () async { : () async {

@ -2,12 +2,15 @@ import 'package:auto_size_text/auto_size_text.dart';
import 'package:cake_wallet/palette.dart'; import 'package:cake_wallet/palette.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:cake_wallet/generated/i18n.dart';
class UnspentCoinsListItem extends StatelessWidget { class UnspentCoinsListItem extends StatelessWidget {
UnspentCoinsListItem({ UnspentCoinsListItem({
@required this.address, @required this.note,
@required this.amount, @required this.amount,
@required this.address,
@required this.isSending, @required this.isSending,
@required this.isFrozen,
@required this.onCheckBoxTap, @required this.onCheckBoxTap,
}); });
@ -16,14 +19,17 @@ class UnspentCoinsListItem extends StatelessWidget {
static const selectedItemColor = Palette.paleCornflowerBlue; static const selectedItemColor = Palette.paleCornflowerBlue;
static const unselectedItemColor = Palette.moderateLavender; static const unselectedItemColor = Palette.moderateLavender;
final String address; final String note;
final String amount; final String amount;
final String address;
final bool isSending; final bool isSending;
final bool isFrozen;
final Function() onCheckBoxTap; final Function() onCheckBoxTap;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final itemColor = isSending? selectedItemColor : unselectedItemColor; final itemColor = isSending? selectedItemColor : unselectedItemColor;
final _note = (note?.isNotEmpty ?? false) ? note : address;
return Container( return Container(
height: 62, height: 62,
@ -51,7 +57,7 @@ class UnspentCoinsListItem extends StatelessWidget {
width: 1.0), width: 1.0),
borderRadius: BorderRadius.all( borderRadius: BorderRadius.all(
Radius.circular(8.0)), Radius.circular(8.0)),
color: Theme.of(context).backgroundColor), color: itemColor),
child: isSending child: isSending
? Icon( ? Icon(
Icons.check, Icons.check,
@ -67,22 +73,48 @@ class UnspentCoinsListItem extends StatelessWidget {
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
AutoSizeText( Row(
amount, mainAxisSize: MainAxisSize.max,
style: TextStyle( mainAxisAlignment: MainAxisAlignment.start,
color: amountColor, crossAxisAlignment: CrossAxisAlignment.center,
fontSize: 16, children: [
fontWeight: FontWeight.w600 Expanded(
), child: AutoSizeText(
maxLines: 1, amount,
style: TextStyle(
color: amountColor,
fontSize: 16,
fontWeight: FontWeight.w600
),
maxLines: 1,
),
),
if (isFrozen) Container(
height: 17,
padding: EdgeInsets.only(left: 6, right: 6),
decoration: BoxDecoration(
borderRadius: BorderRadius.all(Radius.circular(8.5)),
color: Colors.white),
alignment: Alignment.center,
child: Text(
S.of(context).frozen,
style: TextStyle(
color: amountColor,
fontSize: 7,
fontWeight: FontWeight.w600
),
)
)
],
), ),
AutoSizeText( Text(
address, _note,
style: TextStyle( style: TextStyle(
color: addressColor, color: addressColor,
fontSize: 12, fontSize: 12,
), ),
maxLines: 1, maxLines: 1,
overflow: TextOverflow.ellipsis
) )
] ]
) )

@ -9,6 +9,7 @@ import 'package:cake_wallet/exchange/exchange_provider.dart';
import 'package:cake_wallet/exchange/limits.dart'; import 'package:cake_wallet/exchange/limits.dart';
import 'package:cake_wallet/exchange/trade.dart'; import 'package:cake_wallet/exchange/trade.dart';
import 'package:cake_wallet/exchange/limits_state.dart'; import 'package:cake_wallet/exchange/limits_state.dart';
import 'package:cake_wallet/monero/monero_wallet.dart';
import 'package:cake_wallet/store/dashboard/trades_store.dart'; import 'package:cake_wallet/store/dashboard/trades_store.dart';
import 'package:cake_wallet/store/settings_store.dart'; import 'package:cake_wallet/store/settings_store.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
@ -125,6 +126,8 @@ abstract class ExchangeViewModelBase with Store {
bool get hasAllAmount => bool get hasAllAmount =>
wallet.type == WalletType.bitcoin && depositCurrency == wallet.currency; wallet.type == WalletType.bitcoin && depositCurrency == wallet.currency;
bool get isMoneroWallet => wallet is MoneroWallet;
List<CryptoCurrency> receiveCurrencies; List<CryptoCurrency> receiveCurrencies;
Limits limits; Limits limits;

@ -11,7 +11,7 @@ description: Cake Wallet.
# In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion. # In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion.
# Read more about iOS versioning at # Read more about iOS versioning at
# https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html
version: 4.2.3+53 version: 4.2.4+57
environment: environment:
sdk: ">=2.7.0 <3.0.0" sdk: ">=2.7.0 <3.0.0"

@ -488,6 +488,7 @@
"unspent_coins_title" : "Nicht ausgegebene Münzen", "unspent_coins_title" : "Nicht ausgegebene Münzen",
"unspent_coins_details_title" : "Details zu nicht ausgegebenen Münzen", "unspent_coins_details_title" : "Details zu nicht ausgegebenen Münzen",
"freeze" : "Einfrieren", "freeze" : "Einfrieren",
"frozen" : "Gefroren",
"coin_control" : "Münzkontrolle (optional)", "coin_control" : "Münzkontrolle (optional)",
"address_detected" : "Adresse erkannt", "address_detected" : "Adresse erkannt",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Unspent coins", "unspent_coins_title" : "Unspent coins",
"unspent_coins_details_title" : "Unspent coins details", "unspent_coins_details_title" : "Unspent coins details",
"freeze" : "Freeze", "freeze" : "Freeze",
"frozen" : "Frozen",
"coin_control" : "Coin control (optional)", "coin_control" : "Coin control (optional)",
"address_detected" : "Address detected", "address_detected" : "Address detected",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Monedas no gastadas", "unspent_coins_title" : "Monedas no gastadas",
"unspent_coins_details_title" : "Detalles de monedas no gastadas", "unspent_coins_details_title" : "Detalles de monedas no gastadas",
"freeze" : "Congelar", "freeze" : "Congelar",
"frozen" : "Congelada",
"coin_control" : "Control de monedas (opcional)", "coin_control" : "Control de monedas (opcional)",
"address_detected" : "Dirección detectada", "address_detected" : "Dirección detectada",

@ -488,6 +488,7 @@
"unspent_coins_title" : "खर्च न किए गए सिक्के", "unspent_coins_title" : "खर्च न किए गए सिक्के",
"unspent_coins_details_title" : "अव्ययित सिक्कों का विवरण", "unspent_coins_details_title" : "अव्ययित सिक्कों का विवरण",
"freeze" : "फ्रीज", "freeze" : "फ्रीज",
"frozen" : "जमा हुआ",
"coin_control" : "सिक्का नियंत्रण (वैकल्पिक)", "coin_control" : "सिक्का नियंत्रण (वैकल्पिक)",
"address_detected" : "पता लग गया", "address_detected" : "पता लग गया",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Nepotrošeni novčići", "unspent_coins_title" : "Nepotrošeni novčići",
"unspent_coins_details_title" : "Nepotrošeni detalji o novčićima", "unspent_coins_details_title" : "Nepotrošeni detalji o novčićima",
"freeze" : "Zamrznuti", "freeze" : "Zamrznuti",
"frozen" : "Smrznuto",
"coin_control" : "Kontrola novca (nije obavezno)", "coin_control" : "Kontrola novca (nije obavezno)",
"address_detected" : "Adresa je otkrivena", "address_detected" : "Adresa je otkrivena",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Monete non spese", "unspent_coins_title" : "Monete non spese",
"unspent_coins_details_title" : "Dettagli sulle monete non spese", "unspent_coins_details_title" : "Dettagli sulle monete non spese",
"freeze" : "Congelare", "freeze" : "Congelare",
"frozen" : "Congelato",
"coin_control" : "Controllo monete (opzionale)", "coin_control" : "Controllo monete (opzionale)",
"address_detected" : "Indirizzo rilevato", "address_detected" : "Indirizzo rilevato",

@ -488,6 +488,7 @@
"unspent_coins_title" : "未使用のコイン", "unspent_coins_title" : "未使用のコイン",
"unspent_coins_details_title" : "未使用のコインの詳細", "unspent_coins_details_title" : "未使用のコインの詳細",
"freeze" : "氷結", "freeze" : "氷結",
"frozen" : "凍った",
"coin_control" : "コインコントロール(オプション)", "coin_control" : "コインコントロール(オプション)",
"address_detected" : "アドレスが検出されました", "address_detected" : "アドレスが検出されました",

@ -488,6 +488,7 @@
"unspent_coins_title" : "사용하지 않은 동전", "unspent_coins_title" : "사용하지 않은 동전",
"unspent_coins_details_title" : "사용하지 않은 동전 세부 정보", "unspent_coins_details_title" : "사용하지 않은 동전 세부 정보",
"freeze" : "얼다", "freeze" : "얼다",
"frozen" : "겨울 왕국",
"coin_control" : "코인 제어 (옵션)", "coin_control" : "코인 제어 (옵션)",
"address_detected" : "주소 감지", "address_detected" : "주소 감지",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Ongebruikte munten", "unspent_coins_title" : "Ongebruikte munten",
"unspent_coins_details_title" : "Details van niet-uitgegeven munten", "unspent_coins_details_title" : "Details van niet-uitgegeven munten",
"freeze" : "Bevriezen", "freeze" : "Bevriezen",
"frozen" : "Bevroren",
"coin_control" : "Muntcontrole (optioneel)", "coin_control" : "Muntcontrole (optioneel)",
"address_detected" : "Adres gedetecteerd", "address_detected" : "Adres gedetecteerd",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Niewydane monety", "unspent_coins_title" : "Niewydane monety",
"unspent_coins_details_title" : "Szczegóły niewydanych monet", "unspent_coins_details_title" : "Szczegóły niewydanych monet",
"freeze" : "Zamrażać", "freeze" : "Zamrażać",
"frozen" : "Mrożony",
"coin_control" : "Kontrola monet (opcjonalnie)", "coin_control" : "Kontrola monet (opcjonalnie)",
"address_detected" : "Wykryto adres", "address_detected" : "Wykryto adres",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Moedas não gastas", "unspent_coins_title" : "Moedas não gastas",
"unspent_coins_details_title" : "Detalhes de moedas não gastas", "unspent_coins_details_title" : "Detalhes de moedas não gastas",
"freeze" : "Congelar", "freeze" : "Congelar",
"frozen" : "Congeladas",
"coin_control" : "Controle de moedas (opcional)", "coin_control" : "Controle de moedas (opcional)",
"address_detected" : "Endereço detectado", "address_detected" : "Endereço detectado",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Неизрасходованные монеты", "unspent_coins_title" : "Неизрасходованные монеты",
"unspent_coins_details_title" : "Сведения о неизрасходованных монетах", "unspent_coins_details_title" : "Сведения о неизрасходованных монетах",
"freeze" : "Заморозить", "freeze" : "Заморозить",
"frozen" : "Заморожено",
"coin_control" : "Контроль монет (необязательно)", "coin_control" : "Контроль монет (необязательно)",
"address_detected" : "Обнаружен адрес", "address_detected" : "Обнаружен адрес",

@ -488,6 +488,7 @@
"unspent_coins_title" : "Невитрачені монети", "unspent_coins_title" : "Невитрачені монети",
"unspent_coins_details_title" : "Відомості про невитрачені монети", "unspent_coins_details_title" : "Відомості про невитрачені монети",
"freeze" : "Заморозити", "freeze" : "Заморозити",
"frozen" : "Заморожено",
"coin_control" : "Контроль монет (необов’язково)", "coin_control" : "Контроль монет (необов’язково)",
"address_detected" : "Виявлено адресу", "address_detected" : "Виявлено адресу",

@ -488,6 +488,7 @@
"unspent_coins_title" : "未使用的硬幣", "unspent_coins_title" : "未使用的硬幣",
"unspent_coins_details_title" : "未使用代幣詳情", "unspent_coins_details_title" : "未使用代幣詳情",
"freeze" : "凍結", "freeze" : "凍結",
"frozen" : "凍結的",
"coin_control" : "硬幣控制(可選)", "coin_control" : "硬幣控制(可選)",
"address_detected" : "檢測到地址", "address_detected" : "檢測到地址",

Loading…
Cancel
Save