mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-04-02 20:49:10 +00:00
Merge branch 'mweb_enhancements_4' into mweb-bg-sync-2
This commit is contained in:
commit
2914e4d6ea
77 changed files with 1078 additions and 403 deletions
PRIVACY.mdREADME.mdSECURITY.mdbuild-guide-linux.mdpubspec_base.yaml
cw_bitcoin
lib
electrum.dartelectrum_balance.dartelectrum_transaction_info.dartelectrum_wallet.dartlitecoin_wallet.dartlitecoin_wallet_addresses.dartlitecoin_wallet_service.dartpending_bitcoin_transaction.dart
pubspec.yamlcw_bitcoin_cash
cw_core/lib
cw_haven/lib
cw_mweb/lib
lib
core
di.dartentities
exchange/provider
exolix_exchange_provider.dartsimpleswap_exchange_provider.dartstealth_ex_exchange_provider.dartthorchain_exchange.provider.dart
router.dartroutes.dartsrc
screens
dashboard
settings
wallet_connect/widgets/modals
widgets
store
utils
view_model
res/values
strings_ar.arbstrings_bg.arbstrings_cs.arbstrings_de.arbstrings_en.arbstrings_es.arbstrings_fr.arbstrings_ha.arbstrings_hi.arbstrings_hr.arbstrings_hy.arbstrings_id.arbstrings_it.arbstrings_ja.arbstrings_ko.arbstrings_my.arbstrings_nl.arbstrings_pl.arbstrings_pt.arbstrings_ru.arbstrings_th.arbstrings_tl.arbstrings_tr.arbstrings_uk.arbstrings_ur.arbstrings_vi.arbstrings_yo.arbstrings_zh.arb
|
@ -5,7 +5,7 @@ Last modified: January 24, 2024
|
|||
Introduction
|
||||
============
|
||||
|
||||
Cake Labs LLC ("Cake Labs", "Company", or "We") respect your privacy and are committed to protecting it through our compliance with this policy.
|
||||
Cake Labs LLC ("Cake Labs", "Company", or "We") respects your privacy and are committed to protecting it through our compliance with this policy.
|
||||
|
||||
This policy describes the types of information we may collect from you or that you may provide when you use the App (our "App") and our practices for collecting, using, maintaining, protecting, and disclosing that information.
|
||||
|
||||
|
@ -13,7 +13,7 @@ Introduction
|
|||
- On this App.
|
||||
- In email, text, and other electronic messages between you and this App.
|
||||
It does not apply to information collected by:
|
||||
- Us offline or through any other means, including on any other App operated by Company or any third party (including our affiliates and subsidiaries); or
|
||||
- Us offline or through any other means, including on any other App operated by the Company or any third party (including our affiliates and subsidiaries); or
|
||||
- Any third party (including our affiliates and subsidiaries), including through any application or content (including advertising) that may link to or be accessible from or on the App.
|
||||
Please read this policy carefully to understand our policies and practices regarding your information and how we will treat it. If you do not agree with our policies and practices, you have the choice to not use the App. By accessing or using this App, you agree to this privacy policy. This policy may change from time to time. Your continued use of this App after we make changes is deemed to be acceptance of those changes, so please check the policy periodically for updates.
|
||||
|
||||
|
|
|
@ -18,7 +18,7 @@
|
|||
|
||||
# Cake Wallet
|
||||
|
||||
[Cake Wallet](https://cakewallet.com) is an open source, non-custodial, and private multi-currency crypto wallet for Android, iOS, macOS, and Linux.
|
||||
[Cake Wallet](https://cakewallet.com) is an open-source, non-custodial, and private multi-currency crypto wallet for Android, iOS, macOS, and Linux.
|
||||
|
||||
Cake Wallet includes support for several cryptocurrencies, including:
|
||||
* Monero (XMR)
|
||||
|
@ -44,7 +44,7 @@ Cake Wallet includes support for several cryptocurrencies, including:
|
|||
* Create several wallets
|
||||
* Select your own custom nodes/servers
|
||||
* Address book
|
||||
* Backup to external location or iCloud
|
||||
* Backup to an external location or iCloud
|
||||
* Send to OpenAlias, Unstoppable Domains, Yats, and FIO Crypto Handles
|
||||
* Set desired network fee level
|
||||
* Store local transaction notes
|
||||
|
@ -161,7 +161,7 @@ The only parts to be translated, if needed, are the values m and s after the var
|
|||
|
||||
4. Add the language to `lib/entities/language_service.dart` under both `supportedLocales` and `localeCountryCode`. Use the name of the language in the local language and in English in parentheses after for `supportedLocales`. Use the [ISO 3166-1 alpha-3 code](https://en.wikipedia.org/wiki/ISO_3166-1_alpha-3) for `localeCountryCode`. You must choose one country, so choose the country with the most native speakers of this language or is otherwise best associated with this language.
|
||||
|
||||
5. Add a relevant flag to `assets/images/flags/XXXX.png`, replacing XXXX with the 3 letters localeCountryCode. The image must be 42x26 pixels with a 3 pixels of transparent margin on all 4 sides. You can resize the flag with [paint.net](https://www.getpaint.net/) to 36x20 pixels, expand the canvas to 42x26 pixels with the flag anchored in the middle, and then manually delete the 3 pixels on each side to make transparent. Or you can use another program like Photoshop.
|
||||
5. Add a relevant flag to `assets/images/flags/XXXX.png`, replacing XXXX with the 3 letters localeCountryCode. The image must be 42x26 pixels with 3 pixels of transparent margin on all 4 sides. You can resize the flag with [paint.net](https://www.getpaint.net/) to 36x20 pixels, expand the canvas to 42x26 pixels with the flag anchored in the middle, and then manually delete the 3 pixels on each side to make it transparent. Or you can use another program like Photoshop.
|
||||
|
||||
6. Add the new language code to `tool/utils/translation/translation_constants.dart`
|
||||
|
||||
|
|
|
@ -9,4 +9,4 @@ If you need to report a vulnerability, please either:
|
|||
|
||||
## Supported Versions
|
||||
|
||||
As we don't maintain prevoius versions of the app, only the latest release for each platform is supported and any updates will bump the version number.
|
||||
As we don't maintain previous versions of the app, only the latest release for each platform is supported and any updates will bump the version number.
|
||||
|
|
|
@ -55,7 +55,7 @@ Need to install flutter. For this please check section [How to install flutter o
|
|||
|
||||
### 3. Verify Installations
|
||||
|
||||
Verify that the Flutter have been correctly installed on your system with the following command:
|
||||
Verify that the Flutter has been correctly installed on your system with the following command:
|
||||
|
||||
`$ flutter doctor`
|
||||
|
||||
|
@ -163,7 +163,7 @@ And then export bundle:
|
|||
|
||||
`$ flatpak build-bundle export cake_wallet.flatpak com.cakewallet.CakeWallet`
|
||||
|
||||
Result file: `cake_wallet.flatpak` should be generated in current directory.
|
||||
Result file: `cake_wallet.flatpak` should be generated in the current directory.
|
||||
|
||||
For install generated flatpak file use:
|
||||
|
||||
|
|
|
@ -124,6 +124,7 @@ class ElectrumClient {
|
|||
final errorMsg = error.toString();
|
||||
print(errorMsg);
|
||||
unterminatedString = '';
|
||||
socket = null;
|
||||
},
|
||||
onDone: () {
|
||||
print("SOCKET CLOSED!!!!!");
|
||||
|
@ -132,6 +133,7 @@ class ElectrumClient {
|
|||
if (host == socket?.address.host || socket == null) {
|
||||
_setConnectionStatus(ConnectionStatus.disconnected);
|
||||
socket?.destroy();
|
||||
socket = null;
|
||||
}
|
||||
} catch (e) {
|
||||
print("onDone: $e");
|
||||
|
|
|
@ -24,9 +24,12 @@ class ElectrumBalance extends Balance {
|
|||
final decoded = json.decode(jsonSource) as Map;
|
||||
|
||||
return ElectrumBalance(
|
||||
confirmed: decoded['confirmed'] as int? ?? 0,
|
||||
unconfirmed: decoded['unconfirmed'] as int? ?? 0,
|
||||
frozen: decoded['frozen'] as int? ?? 0);
|
||||
confirmed: decoded['confirmed'] as int? ?? 0,
|
||||
unconfirmed: decoded['unconfirmed'] as int? ?? 0,
|
||||
frozen: decoded['frozen'] as int? ?? 0,
|
||||
secondConfirmed: decoded['secondConfirmed'] as int? ?? 0,
|
||||
secondUnconfirmed: decoded['secondUnconfirmed'] as int? ?? 0,
|
||||
);
|
||||
}
|
||||
|
||||
int confirmed;
|
||||
|
@ -36,8 +39,7 @@ class ElectrumBalance extends Balance {
|
|||
int secondUnconfirmed = 0;
|
||||
|
||||
@override
|
||||
String get formattedAvailableBalance =>
|
||||
bitcoinAmountToString(amount: confirmed - frozen);
|
||||
String get formattedAvailableBalance => bitcoinAmountToString(amount: confirmed - frozen);
|
||||
|
||||
@override
|
||||
String get formattedAdditionalBalance => bitcoinAmountToString(amount: unconfirmed);
|
||||
|
|
|
@ -41,6 +41,7 @@ class ElectrumTransactionInfo extends TransactionInfo {
|
|||
String? to,
|
||||
this.unspents,
|
||||
this.isReceivedSilentPayment = false,
|
||||
Map<String, dynamic>? additionalInfo,
|
||||
}) {
|
||||
this.id = id;
|
||||
this.height = height;
|
||||
|
@ -54,6 +55,7 @@ class ElectrumTransactionInfo extends TransactionInfo {
|
|||
this.isReplaced = isReplaced;
|
||||
this.confirmations = confirmations;
|
||||
this.to = to;
|
||||
this.additionalInfo = additionalInfo ?? {};
|
||||
}
|
||||
|
||||
factory ElectrumTransactionInfo.fromElectrumVerbose(Map<String, Object> obj, WalletType type,
|
||||
|
@ -212,6 +214,7 @@ class ElectrumTransactionInfo extends TransactionInfo {
|
|||
BitcoinSilentPaymentsUnspent.fromJSON(null, unspent as Map<String, dynamic>))
|
||||
.toList(),
|
||||
isReceivedSilentPayment: data['isReceivedSilentPayment'] as bool? ?? false,
|
||||
additionalInfo: data['additionalInfo'] as Map<String, dynamic>?,
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -246,7 +249,8 @@ class ElectrumTransactionInfo extends TransactionInfo {
|
|||
isReplaced: isReplaced ?? false,
|
||||
inputAddresses: inputAddresses,
|
||||
outputAddresses: outputAddresses,
|
||||
confirmations: info.confirmations);
|
||||
confirmations: info.confirmations,
|
||||
additionalInfo: additionalInfo);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
|
@ -265,10 +269,11 @@ class ElectrumTransactionInfo extends TransactionInfo {
|
|||
m['inputAddresses'] = inputAddresses;
|
||||
m['outputAddresses'] = outputAddresses;
|
||||
m['isReceivedSilentPayment'] = isReceivedSilentPayment;
|
||||
m['additionalInfo'] = additionalInfo;
|
||||
return m;
|
||||
}
|
||||
|
||||
String toString() {
|
||||
return 'ElectrumTransactionInfo(id: $id, height: $height, amount: $amount, fee: $fee, direction: $direction, date: $date, isPending: $isPending, isReplaced: $isReplaced, confirmations: $confirmations, to: $to, unspent: $unspents, inputAddresses: $inputAddresses, outputAddresses: $outputAddresses)';
|
||||
return 'ElectrumTransactionInfo(id: $id, height: $height, amount: $amount, fee: $fee, direction: $direction, date: $date, isPending: $isPending, isReplaced: $isReplaced, confirmations: $confirmations, to: $to, unspent: $unspents, inputAddresses: $inputAddresses, outputAddresses: $outputAddresses, additionalInfo: $additionalInfo)';
|
||||
}
|
||||
}
|
||||
|
|
|
@ -5,6 +5,7 @@ import 'dart:isolate';
|
|||
|
||||
import 'package:bitcoin_base/bitcoin_base.dart';
|
||||
import 'package:cw_bitcoin/bitcoin_wallet.dart';
|
||||
import 'package:cw_bitcoin/litecoin_wallet.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
import 'package:blockchain_utils/blockchain_utils.dart';
|
||||
import 'package:collection/collection.dart';
|
||||
|
@ -52,10 +53,9 @@ part 'electrum_wallet.g.dart';
|
|||
|
||||
class ElectrumWallet = ElectrumWalletBase with _$ElectrumWallet;
|
||||
|
||||
abstract class ElectrumWalletBase extends WalletBase<
|
||||
ElectrumBalance,
|
||||
ElectrumTransactionHistory,
|
||||
ElectrumTransactionInfo> with Store, WalletKeysFile {
|
||||
abstract class ElectrumWalletBase
|
||||
extends WalletBase<ElectrumBalance, ElectrumTransactionHistory, ElectrumTransactionInfo>
|
||||
with Store, WalletKeysFile {
|
||||
ElectrumWalletBase({
|
||||
required String password,
|
||||
required WalletInfo walletInfo,
|
||||
|
@ -71,8 +71,8 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
ElectrumBalance? initialBalance,
|
||||
CryptoCurrency? currency,
|
||||
this.alwaysScan,
|
||||
}) : accountHD = getAccountHDWallet(
|
||||
currency, network, seedBytes, xpub, walletInfo.derivationInfo),
|
||||
}) : accountHD =
|
||||
getAccountHDWallet(currency, network, seedBytes, xpub, walletInfo.derivationInfo),
|
||||
syncStatus = NotConnectedSyncStatus(),
|
||||
_password = password,
|
||||
_feeRates = <int>[],
|
||||
|
@ -107,12 +107,8 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
sharedPrefs.complete(SharedPreferences.getInstance());
|
||||
}
|
||||
|
||||
static Bip32Slip10Secp256k1 getAccountHDWallet(
|
||||
CryptoCurrency? currency,
|
||||
BasedUtxoNetwork network,
|
||||
Uint8List? seedBytes,
|
||||
String? xpub,
|
||||
DerivationInfo? derivationInfo) {
|
||||
static Bip32Slip10Secp256k1 getAccountHDWallet(CryptoCurrency? currency, BasedUtxoNetwork network,
|
||||
Uint8List? seedBytes, String? xpub, DerivationInfo? derivationInfo) {
|
||||
if (seedBytes == null && xpub == null) {
|
||||
throw Exception(
|
||||
"To create a Wallet you need either a seed or an xpub. This should not happen");
|
||||
|
@ -123,9 +119,8 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
case CryptoCurrency.btc:
|
||||
case CryptoCurrency.ltc:
|
||||
case CryptoCurrency.tbtc:
|
||||
return Bip32Slip10Secp256k1.fromSeed(seedBytes, getKeyNetVersion(network))
|
||||
.derivePath(_hardenedDerivationPath(
|
||||
derivationInfo?.derivationPath ?? electrum_path))
|
||||
return Bip32Slip10Secp256k1.fromSeed(seedBytes, getKeyNetVersion(network)).derivePath(
|
||||
_hardenedDerivationPath(derivationInfo?.derivationPath ?? electrum_path))
|
||||
as Bip32Slip10Secp256k1;
|
||||
case CryptoCurrency.bch:
|
||||
return bitcoinCashHDWallet(seedBytes);
|
||||
|
@ -134,13 +129,11 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
}
|
||||
}
|
||||
|
||||
return Bip32Slip10Secp256k1.fromExtendedKey(
|
||||
xpub!, getKeyNetVersion(network));
|
||||
return Bip32Slip10Secp256k1.fromExtendedKey(xpub!, getKeyNetVersion(network));
|
||||
}
|
||||
|
||||
static Bip32Slip10Secp256k1 bitcoinCashHDWallet(Uint8List seedBytes) =>
|
||||
Bip32Slip10Secp256k1.fromSeed(seedBytes).derivePath("m/44'/145'/0'")
|
||||
as Bip32Slip10Secp256k1;
|
||||
Bip32Slip10Secp256k1.fromSeed(seedBytes).derivePath("m/44'/145'/0'") as Bip32Slip10Secp256k1;
|
||||
|
||||
static int estimatedTransactionSize(int inputsCount, int outputsCounts) =>
|
||||
inputsCount * 68 + outputsCounts * 34 + 10;
|
||||
|
@ -250,7 +243,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
}
|
||||
|
||||
if (tip > walletInfo.restoreHeight) {
|
||||
_setListeners(walletInfo.restoreHeight, chainTipParam: _currentChainTip);
|
||||
_setListeners(walletInfo.restoreHeight, chainTipParam: currentChainTip);
|
||||
}
|
||||
} else {
|
||||
alwaysScan = false;
|
||||
|
@ -265,23 +258,23 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
}
|
||||
}
|
||||
|
||||
int? _currentChainTip;
|
||||
int? currentChainTip;
|
||||
|
||||
Future<int> getCurrentChainTip() async {
|
||||
if ((_currentChainTip ?? 0) > 0) {
|
||||
return _currentChainTip!;
|
||||
if ((currentChainTip ?? 0) > 0) {
|
||||
return currentChainTip!;
|
||||
}
|
||||
_currentChainTip = await electrumClient.getCurrentBlockChainTip() ?? 0;
|
||||
currentChainTip = await electrumClient.getCurrentBlockChainTip() ?? 0;
|
||||
|
||||
return _currentChainTip!;
|
||||
return currentChainTip!;
|
||||
}
|
||||
|
||||
Future<int> getUpdatedChainTip() async {
|
||||
final newTip = await electrumClient.getCurrentBlockChainTip();
|
||||
if (newTip != null && newTip > (_currentChainTip ?? 0)) {
|
||||
_currentChainTip = newTip;
|
||||
if (newTip != null && newTip > (currentChainTip ?? 0)) {
|
||||
currentChainTip = newTip;
|
||||
}
|
||||
return _currentChainTip ?? 0;
|
||||
return currentChainTip ?? 0;
|
||||
}
|
||||
|
||||
@override
|
||||
|
@ -357,7 +350,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
isSingleScan: doSingleScan ?? false,
|
||||
));
|
||||
|
||||
_receiveStream?.cancel();
|
||||
await _receiveStream?.cancel();
|
||||
_receiveStream = receivePort.listen((var message) async {
|
||||
if (message is Map<String, ElectrumTransactionInfo>) {
|
||||
for (final map in message.entries) {
|
||||
|
@ -641,7 +634,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
bool spendsUnconfirmedTX = false;
|
||||
|
||||
int leftAmount = credentialsAmount;
|
||||
final availableInputs = unspentCoins.where((utx) {
|
||||
var availableInputs = unspentCoins.where((utx) {
|
||||
if (!utx.isSending || utx.isFrozen) {
|
||||
return false;
|
||||
}
|
||||
|
@ -657,6 +650,9 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
}).toList();
|
||||
final unconfirmedCoins = availableInputs.where((utx) => utx.confirmations == 0).toList();
|
||||
|
||||
// sort the unconfirmed coins so that mweb coins are first:
|
||||
availableInputs.sort((a, b) => a.bitcoinAddressRecord.type == SegwitAddresType.mweb ? -1 : 1);
|
||||
|
||||
for (int i = 0; i < availableInputs.length; i++) {
|
||||
final utx = availableInputs[i];
|
||||
if (!spendsUnconfirmedTX) spendsUnconfirmedTX = utx.confirmations == 0;
|
||||
|
@ -675,9 +671,8 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
ECPrivate? privkey;
|
||||
bool? isSilentPayment = false;
|
||||
|
||||
final hd = utx.bitcoinAddressRecord.isHidden
|
||||
? walletAddresses.sideHd
|
||||
: walletAddresses.mainHd;
|
||||
final hd =
|
||||
utx.bitcoinAddressRecord.isHidden ? walletAddresses.sideHd : walletAddresses.mainHd;
|
||||
|
||||
if (utx.bitcoinAddressRecord is BitcoinSilentPaymentAddressRecord) {
|
||||
final unspentAddress = utx.bitcoinAddressRecord as BitcoinSilentPaymentAddressRecord;
|
||||
|
@ -1256,8 +1251,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
}
|
||||
}
|
||||
|
||||
void setLedgerConnection(ledger.LedgerConnection connection) =>
|
||||
throw UnimplementedError();
|
||||
void setLedgerConnection(ledger.LedgerConnection connection) => throw UnimplementedError();
|
||||
|
||||
Future<BtcTransaction> buildHardwareWalletTransaction({
|
||||
required List<BitcoinBaseOutput> outputs,
|
||||
|
@ -1616,9 +1610,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
|
||||
final btcAddress = RegexUtils.addressTypeFromStr(addressRecord.address, network);
|
||||
final privkey = generateECPrivate(
|
||||
hd: addressRecord.isHidden
|
||||
? walletAddresses.sideHd
|
||||
: walletAddresses.mainHd,
|
||||
hd: addressRecord.isHidden ? walletAddresses.sideHd : walletAddresses.mainHd,
|
||||
index: addressRecord.index,
|
||||
network: network);
|
||||
|
||||
|
@ -1800,8 +1792,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
|
||||
if (height != null) {
|
||||
if (time == null && height > 0) {
|
||||
time = (getDateByBitcoinHeight(height).millisecondsSinceEpoch / 1000)
|
||||
.round();
|
||||
time = (getDateByBitcoinHeight(height).millisecondsSinceEpoch / 1000).round();
|
||||
}
|
||||
|
||||
if (confirmations == null) {
|
||||
|
@ -1870,6 +1861,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
.map((type) => fetchTransactionsForAddressType(historiesWithDetails, type)));
|
||||
} else if (type == WalletType.litecoin) {
|
||||
await Future.wait(LITECOIN_ADDRESS_TYPES
|
||||
.where((type) => type != SegwitAddresType.mweb)
|
||||
.map((type) => fetchTransactionsForAddressType(historiesWithDetails, type)));
|
||||
}
|
||||
|
||||
|
@ -1981,6 +1973,20 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
|
||||
// Got a new transaction fetched, add it to the transaction history
|
||||
// instead of waiting all to finish, and next time it will be faster
|
||||
|
||||
if (this is LitecoinWallet) {
|
||||
// if we have a peg out transaction with the same value
|
||||
// that matches this received transaction, mark it as being from a peg out:
|
||||
for (final tx2 in transactionHistory.transactions.values) {
|
||||
final heightDiff = ((tx2.height ?? 0) - (tx.height ?? 0)).abs();
|
||||
// this isn't a perfect matching algorithm since we don't have the right input/output information from these transaction models (the addresses are in different formats), but this should be more than good enough for now as it's extremely unlikely a user receives the EXACT same amount from 2 different sources and one of them is a peg out and the other isn't WITHIN 5 blocks of each other
|
||||
if (tx2.additionalInfo["isPegOut"] == true &&
|
||||
tx2.amount == tx.amount &&
|
||||
heightDiff <= 5) {
|
||||
tx.additionalInfo["fromPegOut"] = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
transactionHistory.addOne(tx);
|
||||
await transactionHistory.save();
|
||||
}
|
||||
|
@ -2007,18 +2013,28 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
if (_isTransactionUpdating) {
|
||||
return;
|
||||
}
|
||||
await getCurrentChainTip();
|
||||
currentChainTip = await getUpdatedChainTip();
|
||||
|
||||
bool updated = false;
|
||||
transactionHistory.transactions.values.forEach((tx) {
|
||||
if (tx.unspents != null &&
|
||||
tx.unspents!.isNotEmpty &&
|
||||
tx.height != null &&
|
||||
tx.height! > 0 &&
|
||||
(_currentChainTip ?? 0) > 0) {
|
||||
tx.confirmations = _currentChainTip! - tx.height! + 1;
|
||||
if ((tx.height ?? 0) > 0 && (currentChainTip ?? 0) > 0) {
|
||||
var confirmations = currentChainTip! - tx.height! + 1;
|
||||
if (confirmations < 0) {
|
||||
// if our chain tip is outdated then it could lead to negative confirmations so this is just a failsafe:
|
||||
confirmations = 0;
|
||||
}
|
||||
if (confirmations != tx.confirmations) {
|
||||
updated = true;
|
||||
tx.confirmations = confirmations;
|
||||
transactionHistory.addOne(tx);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
if (updated) {
|
||||
await transactionHistory.save();
|
||||
}
|
||||
|
||||
_isTransactionUpdating = true;
|
||||
await fetchTransactions();
|
||||
walletAddresses.updateReceiveAddresses();
|
||||
|
@ -2066,6 +2082,8 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
library: this.runtimeType.toString(),
|
||||
));
|
||||
}
|
||||
}, onError: (e, s) {
|
||||
print("sub_listen error: $e $s");
|
||||
});
|
||||
}));
|
||||
}
|
||||
|
@ -2115,6 +2133,13 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
|
||||
final balances = await Future.wait(balanceFutures);
|
||||
|
||||
if (balances.isNotEmpty && balances.first['confirmed'] == null) {
|
||||
// if we got null balance responses from the server, set our connection status to lost and return our last known balance:
|
||||
print("got null balance responses from the server, setting connection status to lost");
|
||||
syncStatus = LostConnectionSyncStatus();
|
||||
return balance[currency] ?? ElectrumBalance(confirmed: 0, unconfirmed: 0, frozen: 0);
|
||||
}
|
||||
|
||||
for (var i = 0; i < balances.length; i++) {
|
||||
final addressRecord = addresses[i];
|
||||
final balance = balances[i];
|
||||
|
@ -2220,10 +2245,10 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
Future<void> _setInitialHeight() async {
|
||||
if (_chainTipUpdateSubject != null) return;
|
||||
|
||||
_currentChainTip = await getUpdatedChainTip();
|
||||
currentChainTip = await getUpdatedChainTip();
|
||||
|
||||
if ((_currentChainTip == null || _currentChainTip! == 0) && walletInfo.restoreHeight == 0) {
|
||||
await walletInfo.updateRestoreHeight(_currentChainTip!);
|
||||
if ((currentChainTip == null || currentChainTip! == 0) && walletInfo.restoreHeight == 0) {
|
||||
await walletInfo.updateRestoreHeight(currentChainTip!);
|
||||
}
|
||||
|
||||
_chainTipUpdateSubject = electrumClient.chainTipSubscribe();
|
||||
|
@ -2232,7 +2257,7 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
final height = int.tryParse(event['height'].toString());
|
||||
|
||||
if (height != null) {
|
||||
_currentChainTip = height;
|
||||
currentChainTip = height;
|
||||
|
||||
if (alwaysScan == true && syncStatus is SyncedSyncStatus) {
|
||||
_setListeners(walletInfo.restoreHeight);
|
||||
|
@ -2246,7 +2271,6 @@ abstract class ElectrumWalletBase extends WalletBase<
|
|||
|
||||
@action
|
||||
void _onConnectionStatusChange(ConnectionStatus status) {
|
||||
|
||||
switch (status) {
|
||||
case ConnectionStatus.connected:
|
||||
if (syncStatus is NotConnectedSyncStatus ||
|
||||
|
|
|
@ -9,6 +9,7 @@ import 'package:crypto/crypto.dart';
|
|||
import 'package:cw_bitcoin/bitcoin_transaction_credentials.dart';
|
||||
import 'package:cw_core/cake_hive.dart';
|
||||
import 'package:cw_core/mweb_utxo.dart';
|
||||
import 'package:cw_core/node.dart';
|
||||
import 'package:cw_mweb/mwebd.pbgrpc.dart';
|
||||
import 'package:fixnum/fixnum.dart';
|
||||
import 'package:bip39/bip39.dart' as bip39;
|
||||
|
@ -47,6 +48,7 @@ import 'package:cw_mweb/cw_mweb.dart';
|
|||
import 'package:bitcoin_base/src/crypto/keypair/sign_utils.dart';
|
||||
import 'package:pointycastle/ecc/api.dart';
|
||||
import 'package:pointycastle/ecc/curves/secp256k1.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
part 'litecoin_wallet.g.dart';
|
||||
|
||||
|
@ -85,8 +87,8 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
alwaysScan: alwaysScan,
|
||||
) {
|
||||
if (seedBytes != null) {
|
||||
mwebHd = Bip32Slip10Secp256k1.fromSeed(seedBytes).derivePath(
|
||||
"m/1000'") as Bip32Slip10Secp256k1;
|
||||
mwebHd =
|
||||
Bip32Slip10Secp256k1.fromSeed(seedBytes).derivePath("m/1000'") as Bip32Slip10Secp256k1;
|
||||
mwebEnabled = alwaysScan ?? false;
|
||||
} else {
|
||||
mwebHd = null;
|
||||
|
@ -287,6 +289,16 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
await (walletAddresses as LitecoinWalletAddresses).ensureMwebAddressUpToIndexExists(1020);
|
||||
}
|
||||
|
||||
@action
|
||||
@override
|
||||
Future<void> connectToNode({required Node node}) async {
|
||||
await super.connectToNode(node: node);
|
||||
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final mwebNodeUri = prefs.getString("mwebNodeUri") ?? "ltc-electrum.cakewallet.com:9333";
|
||||
await CwMweb.setNodeUriOverride(mwebNodeUri);
|
||||
}
|
||||
|
||||
@action
|
||||
@override
|
||||
Future<void> startSync() async {
|
||||
|
@ -349,6 +361,9 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
return;
|
||||
}
|
||||
|
||||
// update the current chain tip so that confirmation calculations are accurate:
|
||||
currentChainTip = nodeHeight;
|
||||
|
||||
final resp = await CwMweb.status(StatusRequest());
|
||||
|
||||
try {
|
||||
|
@ -361,22 +376,47 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
} else if (resp.mwebUtxosHeight < nodeHeight) {
|
||||
mwebSyncStatus = SyncingSyncStatus(1, 0.999);
|
||||
} else {
|
||||
bool confirmationsUpdated = false;
|
||||
if (resp.mwebUtxosHeight > walletInfo.restoreHeight) {
|
||||
await walletInfo.updateRestoreHeight(resp.mwebUtxosHeight);
|
||||
await checkMwebUtxosSpent();
|
||||
// update the confirmations for each transaction:
|
||||
for (final transaction in transactionHistory.transactions.values) {
|
||||
if (transaction.isPending) continue;
|
||||
int txHeight = transaction.height ?? resp.mwebUtxosHeight;
|
||||
final confirmations = (resp.mwebUtxosHeight - txHeight) + 1;
|
||||
if (transaction.confirmations == confirmations) continue;
|
||||
if (transaction.confirmations == 0) {
|
||||
updateBalance();
|
||||
for (final tx in transactionHistory.transactions.values) {
|
||||
if (tx.height == null || tx.height == 0) {
|
||||
// update with first confirmation on next block since it hasn't been confirmed yet:
|
||||
tx.height = resp.mwebUtxosHeight;
|
||||
continue;
|
||||
}
|
||||
transaction.confirmations = confirmations;
|
||||
transactionHistory.addOne(transaction);
|
||||
|
||||
final confirmations = (resp.mwebUtxosHeight - tx.height!) + 1;
|
||||
|
||||
// if the confirmations haven't changed, skip updating:
|
||||
if (tx.confirmations == confirmations) continue;
|
||||
|
||||
// print("updating confs ${tx.id} from ${tx.confirmations} -> $confirmations");
|
||||
|
||||
// if an outgoing tx is now confirmed, delete the utxo from the box (delete the unspent coin):
|
||||
if (confirmations >= 2 &&
|
||||
tx.direction == TransactionDirection.outgoing &&
|
||||
tx.unspents != null) {
|
||||
for (var coin in tx.unspents!) {
|
||||
final utxo = mwebUtxosBox.get(coin.address);
|
||||
if (utxo != null) {
|
||||
print("deleting utxo ${coin.address} @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@");
|
||||
await mwebUtxosBox.delete(coin.address);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
tx.confirmations = confirmations;
|
||||
tx.isPending = false;
|
||||
transactionHistory.addOne(tx);
|
||||
confirmationsUpdated = true;
|
||||
}
|
||||
if (confirmationsUpdated) {
|
||||
await transactionHistory.save();
|
||||
await updateTransactions();
|
||||
}
|
||||
await transactionHistory.save();
|
||||
}
|
||||
|
||||
// prevent unnecessary reaction triggers:
|
||||
|
@ -501,13 +541,12 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
outputAddresses: [utxo.outputId],
|
||||
isReplaced: false,
|
||||
);
|
||||
}
|
||||
|
||||
// don't update the confirmations if the tx is updated by electrum:
|
||||
if (tx.confirmations == 0 || utxo.height != 0) {
|
||||
tx.height = utxo.height;
|
||||
tx.isPending = utxo.height == 0;
|
||||
tx.confirmations = confirmations;
|
||||
} else {
|
||||
if (tx.confirmations != confirmations || tx.height != utxo.height) {
|
||||
tx.height = utxo.height;
|
||||
tx.confirmations = confirmations;
|
||||
tx.isPending = utxo.height == 0;
|
||||
}
|
||||
}
|
||||
|
||||
bool isNew = transactionHistory.transactions[tx.id] == null;
|
||||
|
@ -557,56 +596,88 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
if (responseStream == null) {
|
||||
throw Exception("failed to get utxos stream!");
|
||||
}
|
||||
_utxoStream = responseStream.listen((Utxo sUtxo) async {
|
||||
// we're processing utxos, so our balance could still be innacurate:
|
||||
if (mwebSyncStatus is! SyncronizingSyncStatus && mwebSyncStatus is! SyncingSyncStatus) {
|
||||
mwebSyncStatus = SyncronizingSyncStatus();
|
||||
processingUtxos = true;
|
||||
_processingTimer?.cancel();
|
||||
_processingTimer = Timer.periodic(const Duration(seconds: 2), (timer) async {
|
||||
processingUtxos = false;
|
||||
timer.cancel();
|
||||
});
|
||||
}
|
||||
|
||||
final utxo = MwebUtxo(
|
||||
address: sUtxo.address,
|
||||
blockTime: sUtxo.blockTime,
|
||||
height: sUtxo.height,
|
||||
outputId: sUtxo.outputId,
|
||||
value: sUtxo.value.toInt(),
|
||||
);
|
||||
|
||||
if (mwebUtxosBox.containsKey(utxo.outputId)) {
|
||||
// we've already stored this utxo, skip it:
|
||||
// but do update the utxo height if it's somehow different:
|
||||
final existingUtxo = mwebUtxosBox.get(utxo.outputId);
|
||||
if (existingUtxo!.height != utxo.height) {
|
||||
print(
|
||||
"updating utxo height for $utxo.outputId: ${existingUtxo.height} -> ${utxo.height}");
|
||||
existingUtxo.height = utxo.height;
|
||||
await mwebUtxosBox.put(utxo.outputId, existingUtxo);
|
||||
_utxoStream = responseStream.listen(
|
||||
(Utxo sUtxo) async {
|
||||
// we're processing utxos, so our balance could still be innacurate:
|
||||
if (mwebSyncStatus is! SyncronizingSyncStatus && mwebSyncStatus is! SyncingSyncStatus) {
|
||||
mwebSyncStatus = SyncronizingSyncStatus();
|
||||
processingUtxos = true;
|
||||
_processingTimer?.cancel();
|
||||
_processingTimer = Timer.periodic(const Duration(seconds: 2), (timer) async {
|
||||
processingUtxos = false;
|
||||
timer.cancel();
|
||||
});
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
await updateUnspent();
|
||||
await updateBalance();
|
||||
final utxo = MwebUtxo(
|
||||
address: sUtxo.address,
|
||||
blockTime: sUtxo.blockTime,
|
||||
height: sUtxo.height,
|
||||
outputId: sUtxo.outputId,
|
||||
value: sUtxo.value.toInt(),
|
||||
);
|
||||
|
||||
final mwebAddrs = (walletAddresses as LitecoinWalletAddresses).mwebAddrs;
|
||||
if (mwebUtxosBox.containsKey(utxo.outputId)) {
|
||||
// we've already stored this utxo, skip it:
|
||||
// but do update the utxo height if it's somehow different:
|
||||
final existingUtxo = mwebUtxosBox.get(utxo.outputId);
|
||||
if (existingUtxo!.height != utxo.height) {
|
||||
print(
|
||||
"updating utxo height for $utxo.outputId: ${existingUtxo.height} -> ${utxo.height}");
|
||||
existingUtxo.height = utxo.height;
|
||||
await mwebUtxosBox.put(utxo.outputId, existingUtxo);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// don't process utxos with addresses that are not in the mwebAddrs list:
|
||||
if (utxo.address.isNotEmpty && !mwebAddrs.contains(utxo.address)) {
|
||||
return;
|
||||
}
|
||||
await updateUnspent();
|
||||
await updateBalance();
|
||||
|
||||
await mwebUtxosBox.put(utxo.outputId, utxo);
|
||||
final mwebAddrs = (walletAddresses as LitecoinWalletAddresses).mwebAddrs;
|
||||
|
||||
await handleIncoming(utxo);
|
||||
});
|
||||
// don't process utxos with addresses that are not in the mwebAddrs list:
|
||||
if (utxo.address.isNotEmpty && !mwebAddrs.contains(utxo.address)) {
|
||||
return;
|
||||
}
|
||||
|
||||
await mwebUtxosBox.put(utxo.outputId, utxo);
|
||||
|
||||
await handleIncoming(utxo);
|
||||
},
|
||||
onError: (error) {
|
||||
print("error in utxo stream: $error");
|
||||
mwebSyncStatus = FailedSyncStatus(error: error.toString());
|
||||
},
|
||||
cancelOnError: true,
|
||||
);
|
||||
}
|
||||
|
||||
Future<void> deleteSpentUtxos() async {
|
||||
print("deleteSpentUtxos() called!");
|
||||
final chainHeight = await electrumClient.getCurrentBlockChainTip();
|
||||
final status = await CwMweb.status(StatusRequest());
|
||||
if (chainHeight == null || status.blockHeaderHeight != chainHeight) return;
|
||||
if (status.mwebUtxosHeight != chainHeight) return; // we aren't synced
|
||||
|
||||
// delete any spent utxos with >= 2 confirmations:
|
||||
final spentOutputIds = mwebUtxosBox.values
|
||||
.where((utxo) => utxo.spent && (chainHeight - utxo.height) >= 2)
|
||||
.map((utxo) => utxo.outputId)
|
||||
.toList();
|
||||
|
||||
if (spentOutputIds.isEmpty) return;
|
||||
|
||||
final resp = await CwMweb.spent(SpentRequest(outputId: spentOutputIds));
|
||||
final spent = resp.outputId;
|
||||
if (spent.isEmpty) return;
|
||||
|
||||
for (final outputId in spent) {
|
||||
await mwebUtxosBox.delete(outputId);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> checkMwebUtxosSpent() async {
|
||||
print("checkMwebUtxosSpent() called!");
|
||||
if (!mwebEnabled) {
|
||||
return;
|
||||
}
|
||||
|
@ -620,15 +691,17 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
updatedAny = await isConfirmed(tx) || updatedAny;
|
||||
}
|
||||
|
||||
await deleteSpentUtxos();
|
||||
|
||||
// get output ids of all the mweb utxos that have > 0 height:
|
||||
final outputIds =
|
||||
mwebUtxosBox.values.where((utxo) => utxo.height > 0).map((utxo) => utxo.outputId).toList();
|
||||
final outputIds = mwebUtxosBox.values
|
||||
.where((utxo) => utxo.height > 0 && !utxo.spent)
|
||||
.map((utxo) => utxo.outputId)
|
||||
.toList();
|
||||
|
||||
final resp = await CwMweb.spent(SpentRequest(outputId: outputIds));
|
||||
final spent = resp.outputId;
|
||||
if (spent.isEmpty) {
|
||||
return;
|
||||
}
|
||||
if (spent.isEmpty) return;
|
||||
|
||||
final status = await CwMweb.status(StatusRequest());
|
||||
final height = await electrumClient.getCurrentBlockChainTip();
|
||||
|
@ -739,7 +812,7 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
mwebUtxosBox.keys.forEach((dynamic oId) {
|
||||
final String outputId = oId as String;
|
||||
final utxo = mwebUtxosBox.get(outputId);
|
||||
if (utxo == null) {
|
||||
if (utxo == null || utxo.spent) {
|
||||
return;
|
||||
}
|
||||
if (utxo.address.isEmpty) {
|
||||
|
@ -789,15 +862,23 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
int unconfirmedMweb = 0;
|
||||
try {
|
||||
mwebUtxosBox.values.forEach((utxo) {
|
||||
if (utxo.height > 0) {
|
||||
bool isConfirmed = utxo.height > 0;
|
||||
|
||||
print(
|
||||
"utxo: ${isConfirmed ? "confirmed" : "unconfirmed"} ${utxo.spent ? "spent" : "unspent"} ${utxo.outputId} ${utxo.height} ${utxo.value}");
|
||||
|
||||
if (isConfirmed) {
|
||||
confirmedMweb += utxo.value.toInt();
|
||||
} else {
|
||||
}
|
||||
|
||||
if (isConfirmed && utxo.spent) {
|
||||
unconfirmedMweb -= utxo.value.toInt();
|
||||
}
|
||||
|
||||
if (!isConfirmed && !utxo.spent) {
|
||||
unconfirmedMweb += utxo.value.toInt();
|
||||
}
|
||||
});
|
||||
if (unconfirmedMweb > 0) {
|
||||
unconfirmedMweb = -1 * (confirmedMweb - unconfirmedMweb);
|
||||
}
|
||||
} catch (_) {}
|
||||
|
||||
for (var addressRecord in walletAddresses.allAddresses) {
|
||||
|
@ -829,7 +910,6 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
// update the txCount for each address using the tx history, since we can't rely on mwebd
|
||||
// to have an accurate count, we should just keep it in sync with what we know from the tx history:
|
||||
for (final tx in transactionHistory.transactions.values) {
|
||||
// if (tx.isPending) continue;
|
||||
if (tx.inputAddresses == null || tx.outputAddresses == null) {
|
||||
continue;
|
||||
}
|
||||
|
@ -908,7 +988,26 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
// https://github.com/ltcmweb/mwebd?tab=readme-ov-file#fee-estimation
|
||||
final preOutputSum =
|
||||
outputs.fold<BigInt>(BigInt.zero, (acc, output) => acc + output.toOutput.amount);
|
||||
final fee = utxos.sumOfUtxosValue() - preOutputSum;
|
||||
var fee = utxos.sumOfUtxosValue() - preOutputSum;
|
||||
|
||||
// determines if the fee is correct:
|
||||
BigInt _sumOutputAmounts(List<TxOutput> outputs) {
|
||||
BigInt sum = BigInt.zero;
|
||||
for (final e in outputs) {
|
||||
sum += e.amount;
|
||||
}
|
||||
return sum;
|
||||
}
|
||||
|
||||
final sum1 = _sumOutputAmounts(outputs.map((e) => e.toOutput).toList()) + fee;
|
||||
final sum2 = utxos.sumOfUtxosValue();
|
||||
if (sum1 != sum2) {
|
||||
print("@@@@@ WE HAD TO ADJUST THE FEE! @@@@@@@@");
|
||||
final diff = sum2 - sum1;
|
||||
// add the difference to the fee (abs value):
|
||||
fee += diff.abs();
|
||||
}
|
||||
|
||||
final txb =
|
||||
BitcoinTransactionBuilder(utxos: utxos, outputs: outputs, fee: fee, network: network);
|
||||
final resp = await CwMweb.create(CreateRequest(
|
||||
|
@ -949,8 +1048,7 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
|
||||
if (!mwebEnabled) {
|
||||
tx.changeAddressOverride =
|
||||
(await (walletAddresses as LitecoinWalletAddresses)
|
||||
.getChangeAddress(isPegIn: false))
|
||||
(await (walletAddresses as LitecoinWalletAddresses).getChangeAddress(isPegIn: false))
|
||||
.address;
|
||||
return tx;
|
||||
}
|
||||
|
@ -969,15 +1067,25 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
|
||||
bool hasMwebInput = false;
|
||||
bool hasMwebOutput = false;
|
||||
bool hasRegularOutput = false;
|
||||
|
||||
for (final output in transactionCredentials.outputs) {
|
||||
if (output.extractedAddress?.toLowerCase().contains("mweb") ?? false) {
|
||||
final address = output.address.toLowerCase();
|
||||
final extractedAddress = output.extractedAddress?.toLowerCase();
|
||||
|
||||
if (address.contains("mweb")) {
|
||||
hasMwebOutput = true;
|
||||
break;
|
||||
}
|
||||
if (output.address.toLowerCase().contains("mweb")) {
|
||||
hasMwebOutput = true;
|
||||
break;
|
||||
if (!address.contains("mweb")) {
|
||||
hasRegularOutput = true;
|
||||
}
|
||||
if (extractedAddress != null && extractedAddress.isNotEmpty) {
|
||||
if (extractedAddress.contains("mweb")) {
|
||||
hasMwebOutput = true;
|
||||
}
|
||||
if (!extractedAddress.contains("mweb")) {
|
||||
hasRegularOutput = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -989,11 +1097,11 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
}
|
||||
|
||||
bool isPegIn = !hasMwebInput && hasMwebOutput;
|
||||
bool isPegOut = hasMwebInput && hasRegularOutput;
|
||||
bool isRegular = !hasMwebInput && !hasMwebOutput;
|
||||
tx.changeAddressOverride =
|
||||
(await (walletAddresses as LitecoinWalletAddresses)
|
||||
.getChangeAddress(isPegIn: isPegIn || isRegular))
|
||||
.address;
|
||||
tx.changeAddressOverride = (await (walletAddresses as LitecoinWalletAddresses)
|
||||
.getChangeAddress(isPegIn: isPegIn || isRegular))
|
||||
.address;
|
||||
if (!hasMwebInput && !hasMwebOutput) {
|
||||
tx.isMweb = false;
|
||||
return tx;
|
||||
|
@ -1046,8 +1154,11 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
final addresses = <String>{};
|
||||
transaction.inputAddresses?.forEach((id) async {
|
||||
final utxo = mwebUtxosBox.get(id);
|
||||
await mwebUtxosBox.delete(id); // gets deleted in checkMwebUtxosSpent
|
||||
// await mwebUtxosBox.delete(id); // gets deleted in checkMwebUtxosSpent
|
||||
if (utxo == null) return;
|
||||
// mark utxo as spent so we add it to the unconfirmed balance (as negative):
|
||||
utxo.spent = true;
|
||||
await mwebUtxosBox.put(id, utxo);
|
||||
final addressRecord = walletAddresses.allAddresses
|
||||
.firstWhere((addressRecord) => addressRecord.address == utxo.address);
|
||||
if (!addresses.contains(utxo.address)) {
|
||||
|
@ -1056,7 +1167,9 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
addressRecord.balance -= utxo.value.toInt();
|
||||
});
|
||||
transaction.inputAddresses?.addAll(addresses);
|
||||
|
||||
print("isPegIn: $isPegIn, isPegOut: $isPegOut");
|
||||
transaction.additionalInfo["isPegIn"] = isPegIn;
|
||||
transaction.additionalInfo["isPegOut"] = isPegOut;
|
||||
transactionHistory.addOne(transaction);
|
||||
await updateUnspent();
|
||||
await updateBalance();
|
||||
|
@ -1240,8 +1353,8 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
@override
|
||||
void setLedgerConnection(LedgerConnection connection) {
|
||||
_ledgerConnection = connection;
|
||||
_litecoinLedgerApp =
|
||||
LitecoinLedgerApp(_ledgerConnection!, derivationPath: walletInfo.derivationInfo!.derivationPath!);
|
||||
_litecoinLedgerApp = LitecoinLedgerApp(_ledgerConnection!,
|
||||
derivationPath: walletInfo.derivationInfo!.derivationPath!);
|
||||
}
|
||||
|
||||
@override
|
||||
|
@ -1277,19 +1390,17 @@ abstract class LitecoinWalletBase extends ElectrumWallet with Store {
|
|||
if (maybeChangePath != null) changePath ??= maybeChangePath.derivationPath;
|
||||
}
|
||||
|
||||
|
||||
final rawHex = await _litecoinLedgerApp!.createTransaction(
|
||||
inputs: readyInputs,
|
||||
outputs: outputs
|
||||
.map((e) => TransactionOutput.fromBigInt(
|
||||
(e as BitcoinOutput).value, Uint8List.fromList(e.address.toScriptPubKey().toBytes())))
|
||||
.toList(),
|
||||
changePath: changePath,
|
||||
sigHashType: 0x01,
|
||||
additionals: ["bech32"],
|
||||
isSegWit: true,
|
||||
useTrustedInputForSegwit: true
|
||||
);
|
||||
inputs: readyInputs,
|
||||
outputs: outputs
|
||||
.map((e) => TransactionOutput.fromBigInt((e as BitcoinOutput).value,
|
||||
Uint8List.fromList(e.address.toScriptPubKey().toBytes())))
|
||||
.toList(),
|
||||
changePath: changePath,
|
||||
sigHashType: 0x01,
|
||||
additionals: ["bech32"],
|
||||
isSegWit: true,
|
||||
useTrustedInputForSegwit: true);
|
||||
|
||||
return BtcTransaction.fromRaw(rawHex);
|
||||
}
|
||||
|
|
|
@ -16,11 +16,9 @@ import 'package:mobx/mobx.dart';
|
|||
|
||||
part 'litecoin_wallet_addresses.g.dart';
|
||||
|
||||
class LitecoinWalletAddresses = LitecoinWalletAddressesBase
|
||||
with _$LitecoinWalletAddresses;
|
||||
class LitecoinWalletAddresses = LitecoinWalletAddressesBase with _$LitecoinWalletAddresses;
|
||||
|
||||
abstract class LitecoinWalletAddressesBase extends ElectrumWalletAddresses
|
||||
with Store {
|
||||
abstract class LitecoinWalletAddressesBase extends ElectrumWalletAddresses with Store {
|
||||
LitecoinWalletAddressesBase(
|
||||
WalletInfo walletInfo, {
|
||||
required super.mainHd,
|
||||
|
@ -46,8 +44,7 @@ abstract class LitecoinWalletAddressesBase extends ElectrumWalletAddresses
|
|||
List<String> mwebAddrs = [];
|
||||
bool generating = false;
|
||||
|
||||
List<int> get scanSecret =>
|
||||
mwebHd!.childKey(Bip32KeyIndex(0x80000000)).privateKey.privKey.raw;
|
||||
List<int> get scanSecret => mwebHd!.childKey(Bip32KeyIndex(0x80000000)).privateKey.privKey.raw;
|
||||
List<int> get spendPubkey =>
|
||||
mwebHd!.childKey(Bip32KeyIndex(0x80000001)).publicKey.pubKey.compressed;
|
||||
|
||||
|
@ -203,4 +200,12 @@ abstract class LitecoinWalletAddressesBase extends ElectrumWalletAddresses
|
|||
|
||||
return super.getChangeAddress();
|
||||
}
|
||||
|
||||
@override
|
||||
String get addressForExchange {
|
||||
// don't use mweb addresses for exchange refund address:
|
||||
final addresses = receiveAddresses
|
||||
.where((element) => element.type == SegwitAddresType.p2wpkh && !element.isUsed);
|
||||
return addresses.first.address;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -112,6 +112,7 @@ class LitecoinWalletService extends WalletService<
|
|||
File neturinoDb = File('$appDirPath/neutrino.db');
|
||||
File blockHeaders = File('$appDirPath/block_headers.bin');
|
||||
File regFilterHeaders = File('$appDirPath/reg_filter_headers.bin');
|
||||
File mwebdLogs = File('$appDirPath/logs/debug.log');
|
||||
if (neturinoDb.existsSync()) {
|
||||
neturinoDb.deleteSync();
|
||||
}
|
||||
|
@ -121,6 +122,9 @@ class LitecoinWalletService extends WalletService<
|
|||
if (regFilterHeaders.existsSync()) {
|
||||
regFilterHeaders.deleteSync();
|
||||
}
|
||||
if (mwebdLogs.existsSync()) {
|
||||
mwebdLogs.deleteSync();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -118,8 +118,7 @@ class PendingBitcoinTransaction with PendingTransaction {
|
|||
|
||||
Future<void> _ltcCommit() async {
|
||||
try {
|
||||
final stub = await CwMweb.stub();
|
||||
final resp = await stub.broadcast(BroadcastRequest(rawTx: BytesUtils.fromHexString(hex)));
|
||||
final resp = await CwMweb.broadcast(BroadcastRequest(rawTx: BytesUtils.fromHexString(hex)));
|
||||
idOverride = resp.txid;
|
||||
} on GrpcError catch (e) {
|
||||
throw BitcoinTransactionCommitFailed(errorMessage: e.message);
|
||||
|
|
|
@ -64,7 +64,7 @@ dependency_overrides:
|
|||
bitcoin_base:
|
||||
git:
|
||||
url: https://github.com/cake-tech/bitcoin_base
|
||||
ref: cake-update-v8
|
||||
ref: cake-update-v9
|
||||
pointycastle: 3.7.4
|
||||
ffi: 2.1.0
|
||||
|
||||
|
|
|
@ -42,7 +42,7 @@ dependency_overrides:
|
|||
bitcoin_base:
|
||||
git:
|
||||
url: https://github.com/cake-tech/bitcoin_base
|
||||
ref: cake-update-v8
|
||||
ref: cake-update-v9
|
||||
|
||||
# For information on the generic Dart part of this file, see the
|
||||
# following page: https://dart.dev/tools/pub/pubspec
|
||||
|
|
|
@ -11,6 +11,7 @@ class MwebUtxo extends HiveObject {
|
|||
required this.address,
|
||||
required this.outputId,
|
||||
required this.blockTime,
|
||||
this.spent = false,
|
||||
});
|
||||
|
||||
static const typeId = MWEB_UTXO_TYPE_ID;
|
||||
|
@ -30,4 +31,7 @@ class MwebUtxo extends HiveObject {
|
|||
|
||||
@HiveField(4)
|
||||
int blockTime;
|
||||
|
||||
@HiveField(5, defaultValue: false)
|
||||
bool spent;
|
||||
}
|
||||
|
|
|
@ -79,6 +79,9 @@ class Node extends HiveObject with Keyable {
|
|||
@HiveField(9)
|
||||
bool? supportsSilentPayments;
|
||||
|
||||
@HiveField(10)
|
||||
bool? supportsMweb;
|
||||
|
||||
bool get isSSL => useSSL ?? false;
|
||||
|
||||
bool get useSocksProxy => socksProxyAddress == null ? false : socksProxyAddress!.isNotEmpty;
|
||||
|
|
|
@ -25,6 +25,5 @@ abstract class TransactionInfo extends Object with Keyable {
|
|||
@override
|
||||
dynamic get keyIndex => id;
|
||||
|
||||
late Map<String, dynamic> additionalInfo;
|
||||
Map<String, dynamic> additionalInfo = {};
|
||||
}
|
||||
|
||||
|
|
|
@ -23,6 +23,8 @@ abstract class HavenWalletAddressesBase extends WalletAddressesWithAccount<Accou
|
|||
@observable
|
||||
String address;
|
||||
|
||||
String get primaryAddress => address;
|
||||
|
||||
// @override
|
||||
@observable
|
||||
Account? account;
|
||||
|
|
|
@ -13,8 +13,18 @@ class CwMweb {
|
|||
static RpcClient? _rpcClient;
|
||||
static ClientChannel? _clientChannel;
|
||||
static int? _port;
|
||||
static const TIMEOUT_DURATION = Duration(seconds: 5);
|
||||
static const TIMEOUT_DURATION = Duration(seconds: 15);
|
||||
static Timer? logTimer;
|
||||
static String? nodeUriOverride;
|
||||
|
||||
|
||||
static Future<void> setNodeUriOverride(String uri) async {
|
||||
nodeUriOverride = uri;
|
||||
if (_rpcClient != null) {
|
||||
await stop();
|
||||
// will be re-started automatically when the next rpc call is made
|
||||
}
|
||||
}
|
||||
|
||||
static void readFileWithTimer(String filePath) {
|
||||
final file = File(filePath);
|
||||
|
@ -47,7 +57,7 @@ class CwMweb {
|
|||
String debugLogPath = "${appDir.path}/logs/debug.log";
|
||||
readFileWithTimer(debugLogPath);
|
||||
|
||||
_port = await CwMwebPlatform.instance.start(appDir.path, ltcNodeUri);
|
||||
_port = await CwMwebPlatform.instance.start(appDir.path, nodeUriOverride ?? ltcNodeUri);
|
||||
if (_port == null || _port == 0) {
|
||||
throw Exception("Failed to start server");
|
||||
}
|
||||
|
@ -197,4 +207,18 @@ class CwMweb {
|
|||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
static Future<BroadcastResponse> broadcast(BroadcastRequest request) async {
|
||||
log("mweb.broadcast() called");
|
||||
try {
|
||||
_rpcClient = await stub();
|
||||
return await _rpcClient!.broadcast(request, options: CallOptions(timeout: TIMEOUT_DURATION));
|
||||
} on GrpcError catch (e) {
|
||||
log('Caught grpc error: ${e.message}');
|
||||
throw "error from broadcast mweb: $e";
|
||||
} catch (e) {
|
||||
log("Error getting create: $e");
|
||||
rethrow;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -106,8 +106,7 @@ class AddressValidator extends TextValidator {
|
|||
case CryptoCurrency.wow:
|
||||
pattern = '[0-9a-zA-Z]+';
|
||||
case CryptoCurrency.bch:
|
||||
pattern =
|
||||
'(?!bitcoincash:)[0-9a-zA-Z]*|(?!bitcoincash:)q|p[0-9a-zA-Z]{41}|(?!bitcoincash:)q|p[0-9a-zA-Z]{42}|bitcoincash:q|p[0-9a-zA-Z]{41}|bitcoincash:q|p[0-9a-zA-Z]{42}';
|
||||
pattern = '^(bitcoincash:)?(q|p)[0-9a-zA-Z]{41,42}';
|
||||
case CryptoCurrency.bnb:
|
||||
pattern = '[0-9a-zA-Z]+';
|
||||
case CryptoCurrency.hbar:
|
||||
|
|
|
@ -267,6 +267,8 @@ abstract class Web3WalletServiceBase with Store {
|
|||
|
||||
final keyForWallet = getKeyForStoringTopicsForWallet();
|
||||
|
||||
if (keyForWallet.isEmpty) return;
|
||||
|
||||
final currentTopicsForWallet = getPairingTopicsForWallet(keyForWallet);
|
||||
|
||||
final filteredPairings =
|
||||
|
@ -360,6 +362,10 @@ abstract class Web3WalletServiceBase with Store {
|
|||
String getKeyForStoringTopicsForWallet() {
|
||||
List<ChainKeyModel> chainKeys = walletKeyService.getKeysForChain(appStore.wallet!);
|
||||
|
||||
if (chainKeys.isEmpty) {
|
||||
return '';
|
||||
}
|
||||
|
||||
final keyForPairingTopic =
|
||||
PreferencesKey.walletConnectPairingTopicsListForWallet(chainKeys.first.publicKey);
|
||||
|
||||
|
@ -386,6 +392,8 @@ abstract class Web3WalletServiceBase with Store {
|
|||
// Get key specific to the current wallet
|
||||
final key = getKeyForStoringTopicsForWallet();
|
||||
|
||||
if (key.isEmpty) return;
|
||||
|
||||
// Get all pairing topics attached to this key
|
||||
final pairingTopicsForWallet = getPairingTopicsForWallet(key);
|
||||
|
||||
|
|
|
@ -35,6 +35,8 @@ import 'package:cake_wallet/entities/parse_address_from_domain.dart';
|
|||
import 'package:cake_wallet/entities/wallet_edit_page_arguments.dart';
|
||||
import 'package:cake_wallet/entities/wallet_manager.dart';
|
||||
import 'package:cake_wallet/src/screens/receive/address_list_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/mweb_logs_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/mweb_node_page.dart';
|
||||
import 'package:cake_wallet/view_model/link_view_model.dart';
|
||||
import 'package:cake_wallet/tron/tron.dart';
|
||||
import 'package:cake_wallet/src/screens/transaction_details/rbf_details_page.dart';
|
||||
|
@ -941,6 +943,10 @@ Future<void> setup({
|
|||
|
||||
getIt.registerFactory(() => MwebSettingsPage(getIt.get<MwebSettingsViewModel>()));
|
||||
|
||||
getIt.registerFactory(() => MwebLogsPage(getIt.get<MwebSettingsViewModel>()));
|
||||
|
||||
getIt.registerFactory(() => MwebNodePage(getIt.get<MwebSettingsViewModel>()));
|
||||
|
||||
getIt.registerFactory(() => OtherSettingsPage(getIt.get<OtherSettingsViewModel>()));
|
||||
|
||||
getIt.registerFactory(() => NanoChangeRepPage(
|
||||
|
|
|
@ -114,7 +114,7 @@ class FiatCurrency extends EnumerableItem<String> with Serializable<String> impl
|
|||
FiatCurrency.tur.raw: FiatCurrency.tur,
|
||||
};
|
||||
|
||||
static FiatCurrency deserialize({required String raw}) => _all[raw]!;
|
||||
static FiatCurrency deserialize({required String raw}) => _all[raw] ?? FiatCurrency.usd;
|
||||
|
||||
@override
|
||||
bool operator ==(Object other) => other is FiatCurrency && other.raw == raw;
|
||||
|
|
|
@ -52,6 +52,7 @@ class PreferencesKey {
|
|||
static const mwebEnabled = 'mwebEnabled';
|
||||
static const hasEnabledMwebBefore = 'hasEnabledMwebBefore';
|
||||
static const mwebAlwaysScan = 'mwebAlwaysScan';
|
||||
static const mwebNodeUri = 'mwebNodeUri';
|
||||
static const shouldShowReceiveWarning = 'should_show_receive_warning';
|
||||
static const shouldShowYatPopup = 'should_show_yat_popup';
|
||||
static const shouldShowRepWarning = 'should_show_rep_warning';
|
||||
|
|
|
@ -141,8 +141,8 @@ class ExolixExchangeProvider extends ExchangeProvider {
|
|||
'coinTo': _normalizeCurrency(request.toCurrency),
|
||||
'networkFrom': _networkFor(request.fromCurrency),
|
||||
'networkTo': _networkFor(request.toCurrency),
|
||||
'withdrawalAddress': request.toAddress,
|
||||
'refundAddress': request.refundAddress,
|
||||
'withdrawalAddress': _normalizeAddress(request.toAddress),
|
||||
'refundAddress': _normalizeAddress(request.refundAddress),
|
||||
'rateType': _getRateType(isFixedRateMode),
|
||||
'apiToken': apiKey,
|
||||
};
|
||||
|
@ -275,4 +275,7 @@ class ExolixExchangeProvider extends ExchangeProvider {
|
|||
return tag;
|
||||
}
|
||||
}
|
||||
|
||||
String _normalizeAddress(String address) =>
|
||||
address.startsWith('bitcoincash:') ? address.replaceFirst('bitcoincash:', '') : address;
|
||||
}
|
||||
|
|
|
@ -129,8 +129,8 @@ class SimpleSwapExchangeProvider extends ExchangeProvider {
|
|||
"currency_to": _normalizeCurrency(request.toCurrency),
|
||||
"amount": request.fromAmount,
|
||||
"fixed": isFixedRateMode,
|
||||
"user_refund_address": request.refundAddress,
|
||||
"address_to": request.toAddress
|
||||
"user_refund_address": _normalizeAddress(request.refundAddress),
|
||||
"address_to": _normalizeAddress(request.toAddress)
|
||||
};
|
||||
final uri = Uri.https(apiAuthority, createExchangePath, params);
|
||||
|
||||
|
@ -243,4 +243,7 @@ class SimpleSwapExchangeProvider extends ExchangeProvider {
|
|||
return currency.title.toLowerCase();
|
||||
}
|
||||
}
|
||||
|
||||
String _normalizeAddress(String address) =>
|
||||
address.startsWith('bitcoincash:') ? address.replaceFirst('bitcoincash:', '') : address;
|
||||
}
|
||||
|
|
|
@ -129,8 +129,8 @@ class StealthExExchangeProvider extends ExchangeProvider {
|
|||
if (isFixedRateMode) 'rate_id': rateId,
|
||||
'amount':
|
||||
isFixedRateMode ? double.parse(request.toAmount) : double.parse(request.fromAmount),
|
||||
'address': request.toAddress,
|
||||
'refund_address': request.refundAddress,
|
||||
'address': _normalizeAddress(request.toAddress),
|
||||
'refund_address': _normalizeAddress(request.refundAddress),
|
||||
'additional_fee_percent': _additionalFeePercent,
|
||||
};
|
||||
|
||||
|
@ -296,4 +296,7 @@ class StealthExExchangeProvider extends ExchangeProvider {
|
|||
|
||||
return currency.tag!.toLowerCase();
|
||||
}
|
||||
|
||||
String _normalizeAddress(String address) =>
|
||||
address.startsWith('bitcoincash:') ? address.replaceFirst('bitcoincash:', '') : address;
|
||||
}
|
||||
|
|
|
@ -116,9 +116,7 @@ class ThorChainExchangeProvider extends ExchangeProvider {
|
|||
required bool isFixedRateMode,
|
||||
required bool isSendAll,
|
||||
}) async {
|
||||
String formattedToAddress = request.toAddress.startsWith('bitcoincash:')
|
||||
? request.toAddress.replaceFirst('bitcoincash:', '')
|
||||
: request.toAddress;
|
||||
|
||||
|
||||
final formattedFromAmount = double.parse(request.fromAmount);
|
||||
|
||||
|
@ -126,11 +124,11 @@ class ThorChainExchangeProvider extends ExchangeProvider {
|
|||
'from_asset': _normalizeCurrency(request.fromCurrency),
|
||||
'to_asset': _normalizeCurrency(request.toCurrency),
|
||||
'amount': _doubleToThorChainString(formattedFromAmount),
|
||||
'destination': formattedToAddress,
|
||||
'destination': _normalizeAddress(request.toAddress),
|
||||
'affiliate': _affiliateName,
|
||||
'affiliate_bps': _affiliateBps,
|
||||
'refund_address':
|
||||
isRefundAddressSupported.contains(request.fromCurrency) ? request.refundAddress : '',
|
||||
isRefundAddressSupported.contains(request.fromCurrency) ? _normalizeAddress(request.refundAddress) : '',
|
||||
};
|
||||
|
||||
final responseJSON = await _getSwapQuote(params);
|
||||
|
@ -288,4 +286,7 @@ class ThorChainExchangeProvider extends ExchangeProvider {
|
|||
|
||||
return currentState;
|
||||
}
|
||||
|
||||
String _normalizeAddress(String address) =>
|
||||
address.startsWith('bitcoincash:') ? address.replaceFirst('bitcoincash:', '') : address;
|
||||
}
|
||||
|
|
|
@ -72,6 +72,8 @@ import 'package:cake_wallet/src/screens/settings/desktop_settings/desktop_settin
|
|||
import 'package:cake_wallet/src/screens/settings/display_settings_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/domain_lookups_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/manage_nodes_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/mweb_logs_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/mweb_node_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/mweb_settings.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/other_settings_page.dart';
|
||||
import 'package:cake_wallet/src/screens/settings/privacy_page.dart';
|
||||
|
@ -461,6 +463,14 @@ Route<dynamic> createRoute(RouteSettings settings) {
|
|||
return CupertinoPageRoute<void>(
|
||||
fullscreenDialog: true, builder: (_) => getIt.get<MwebSettingsPage>());
|
||||
|
||||
case Routes.mwebLogs:
|
||||
return CupertinoPageRoute<void>(
|
||||
fullscreenDialog: true, builder: (_) => getIt.get<MwebLogsPage>());
|
||||
|
||||
case Routes.mwebNode:
|
||||
return CupertinoPageRoute<void>(
|
||||
fullscreenDialog: true, builder: (_) => getIt.get<MwebNodePage>());
|
||||
|
||||
case Routes.connectionSync:
|
||||
return CupertinoPageRoute<void>(
|
||||
fullscreenDialog: true, builder: (_) => getIt.get<ConnectionSyncPage>());
|
||||
|
|
|
@ -74,6 +74,8 @@ class Routes {
|
|||
static const webViewPage = '/web_view_page';
|
||||
static const silentPaymentsSettings = '/silent_payments_settings';
|
||||
static const mwebSettings = '/mweb_settings';
|
||||
static const mwebLogs = '/mweb_logs';
|
||||
static const mwebNode = '/mweb_node';
|
||||
static const connectionSync = '/connection_sync_page';
|
||||
static const securityBackupPage = '/security_and_backup_page';
|
||||
static const privacyPage = '/privacy_page';
|
||||
|
|
|
@ -10,7 +10,6 @@ import 'package:cake_wallet/src/screens/dashboard/desktop_widgets/dropdown_item_
|
|||
import 'package:cake_wallet/src/screens/wallet_unlock/wallet_unlock_arguments.dart';
|
||||
import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart';
|
||||
import 'package:cake_wallet/store/settings_store.dart';
|
||||
import 'package:cake_wallet/themes/extensions/cake_text_theme.dart';
|
||||
import 'package:cake_wallet/themes/extensions/menu_theme.dart';
|
||||
import 'package:cake_wallet/utils/show_bar.dart';
|
||||
import 'package:cake_wallet/utils/show_pop_up.dart';
|
||||
|
@ -100,6 +99,11 @@ class _DesktopWalletSelectionDropDownState extends State<DesktopWalletSelectionD
|
|||
),
|
||||
];
|
||||
|
||||
final selectedItem = dropDownItems.firstWhere(
|
||||
(element) => element.isSelected,
|
||||
orElse: () => dropDownItems.first,
|
||||
);
|
||||
|
||||
return DropdownButton<DesktopDropdownItem>(
|
||||
items: dropDownItems
|
||||
.map(
|
||||
|
@ -115,7 +119,7 @@ class _DesktopWalletSelectionDropDownState extends State<DesktopWalletSelectionD
|
|||
dropdownColor: themeData.extension<CakeMenuTheme>()!.backgroundColor,
|
||||
style: TextStyle(color: themeData.extension<CakeTextTheme>()!.titleColor),
|
||||
selectedItemBuilder: (context) => dropDownItems.map((item) => item.child).toList(),
|
||||
value: dropDownItems.firstWhere((element) => element.isSelected),
|
||||
value: selectedItem,
|
||||
underline: const SizedBox(),
|
||||
focusColor: Colors.transparent,
|
||||
borderRadius: BorderRadius.circular(15.0),
|
||||
|
|
|
@ -886,17 +886,37 @@ class BalanceRowWidget extends StatelessWidget {
|
|||
Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
'${secondAvailableBalanceLabel}',
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(
|
||||
fontSize: 12,
|
||||
fontFamily: 'Lato',
|
||||
fontWeight: FontWeight.w400,
|
||||
color: Theme.of(context)
|
||||
.extension<BalancePageTheme>()!
|
||||
.labelTextColor,
|
||||
height: 1,
|
||||
GestureDetector(
|
||||
behavior: HitTestBehavior.opaque,
|
||||
onTap: () => launchUrl(
|
||||
Uri.parse(
|
||||
"https://guides.cakewallet.com/docs/cryptos/litecoin/#mweb"),
|
||||
mode: LaunchMode.externalApplication,
|
||||
),
|
||||
child: Row(
|
||||
children: [
|
||||
Text(
|
||||
'${secondAvailableBalanceLabel}',
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(
|
||||
fontSize: 12,
|
||||
fontFamily: 'Lato',
|
||||
fontWeight: FontWeight.w400,
|
||||
color: Theme.of(context)
|
||||
.extension<BalancePageTheme>()!
|
||||
.labelTextColor,
|
||||
height: 1,
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.symmetric(horizontal: 4),
|
||||
child: Icon(Icons.help_outline,
|
||||
size: 16,
|
||||
color: Theme.of(context)
|
||||
.extension<BalancePageTheme>()!
|
||||
.labelTextColor),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
SizedBox(height: 8),
|
||||
|
|
127
lib/src/screens/settings/mweb_logs_page.dart
Normal file
127
lib/src/screens/settings/mweb_logs_page.dart
Normal file
|
@ -0,0 +1,127 @@
|
|||
import 'dart:io';
|
||||
import 'package:cake_wallet/src/screens/base_page.dart';
|
||||
import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart';
|
||||
import 'package:cake_wallet/src/widgets/primary_button.dart';
|
||||
import 'package:cake_wallet/generated/i18n.dart';
|
||||
import 'package:cake_wallet/utils/exception_handler.dart';
|
||||
import 'package:cake_wallet/utils/share_util.dart';
|
||||
import 'package:cake_wallet/utils/show_pop_up.dart';
|
||||
import 'package:cake_wallet/view_model/settings/mweb_settings_view_model.dart';
|
||||
import 'package:cw_core/root_dir.dart';
|
||||
import 'package:file_picker/file_picker.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:path_provider/path_provider.dart';
|
||||
|
||||
class MwebLogsPage extends BasePage {
|
||||
MwebLogsPage(this.mwebSettingsViewModelBase);
|
||||
|
||||
final MwebSettingsViewModelBase mwebSettingsViewModelBase;
|
||||
|
||||
@override
|
||||
String get title => S.current.litecoin_mweb_logs;
|
||||
|
||||
@override
|
||||
Widget body(BuildContext context) {
|
||||
return Stack(
|
||||
fit: StackFit.expand,
|
||||
children: [
|
||||
FutureBuilder<String>(
|
||||
future: mwebSettingsViewModelBase.getAbbreviatedLogs(),
|
||||
builder: (context, snapshot) {
|
||||
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||
return Center(child: CircularProgressIndicator());
|
||||
} else if (snapshot.hasError || !snapshot.hasData || snapshot.data!.isEmpty) {
|
||||
return Center(child: Text('No logs found'));
|
||||
} else {
|
||||
return SingleChildScrollView(
|
||||
child: Padding(
|
||||
padding: EdgeInsets.all(16.0),
|
||||
child: Text(
|
||||
snapshot.data!,
|
||||
style: TextStyle(fontFamily: 'Monospace'),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
},
|
||||
),
|
||||
Positioned(
|
||||
child: LoadingPrimaryButton(
|
||||
onPressed: () => onExportLogs(context),
|
||||
text: S.of(context).export_logs,
|
||||
color: Theme.of(context).primaryColor,
|
||||
textColor: Colors.white,
|
||||
),
|
||||
bottom: 24,
|
||||
left: 24,
|
||||
right: 24,
|
||||
)
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
void onExportLogs(BuildContext context) {
|
||||
if (Platform.isAndroid) {
|
||||
onExportAndroid(context);
|
||||
} else if (Platform.isIOS) {
|
||||
share(context);
|
||||
} else {
|
||||
_saveFile();
|
||||
}
|
||||
}
|
||||
|
||||
void onExportAndroid(BuildContext context) {
|
||||
showPopUp<void>(
|
||||
context: context,
|
||||
builder: (dialogContext) {
|
||||
return AlertWithTwoActions(
|
||||
alertTitle: S.of(context).export_backup,
|
||||
alertContent: S.of(context).select_destination,
|
||||
rightButtonText: S.of(context).save_to_downloads,
|
||||
leftButtonText: S.of(context).share,
|
||||
actionRightButton: () async {
|
||||
const downloadDirPath = "/storage/emulated/0/Download";
|
||||
final filePath = downloadDirPath + "/debug.log";
|
||||
await mwebSettingsViewModelBase.saveLogsLocally(filePath);
|
||||
Navigator.of(dialogContext).pop();
|
||||
},
|
||||
actionLeftButton: () async {
|
||||
Navigator.of(dialogContext).pop();
|
||||
try {
|
||||
await share(context);
|
||||
} catch (e, s) {
|
||||
ExceptionHandler.onError(FlutterErrorDetails(
|
||||
exception: e,
|
||||
stack: s,
|
||||
library: "Export Logs",
|
||||
));
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> share(BuildContext context) async {
|
||||
final filePath = (await getAppDir()).path + "/debug.log";
|
||||
bool success = await mwebSettingsViewModelBase.saveLogsLocally(filePath);
|
||||
if (!success) return;
|
||||
await ShareUtil.shareFile(filePath: filePath, fileName: "debug.log", context: context);
|
||||
await mwebSettingsViewModelBase.removeLogsLocally(filePath);
|
||||
}
|
||||
|
||||
Future<void> _saveFile() async {
|
||||
String? outputFile = await FilePicker.platform
|
||||
.saveFile(dialogTitle: 'Save Your File to desired location', fileName: "debug.log");
|
||||
|
||||
try {
|
||||
final filePath = (await getApplicationSupportDirectory()).path + "/debug.log";
|
||||
File debugLogFile = File(filePath);
|
||||
await debugLogFile.copy(outputFile!);
|
||||
} catch (exception, stackTrace) {
|
||||
ExceptionHandler.onError(FlutterErrorDetails(
|
||||
exception: exception,
|
||||
stack: stackTrace,
|
||||
library: "Export Logs",
|
||||
));
|
||||
}
|
||||
}
|
||||
}
|
56
lib/src/screens/settings/mweb_node_page.dart
Normal file
56
lib/src/screens/settings/mweb_node_page.dart
Normal file
|
@ -0,0 +1,56 @@
|
|||
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/primary_button.dart';
|
||||
import 'package:cake_wallet/generated/i18n.dart';
|
||||
import 'package:cake_wallet/view_model/settings/mweb_settings_view_model.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_mobx/flutter_mobx.dart';
|
||||
|
||||
class MwebNodePage extends BasePage {
|
||||
MwebNodePage(this.mwebSettingsViewModelBase)
|
||||
: _nodeUriController = TextEditingController(text: mwebSettingsViewModelBase.mwebNodeUri),
|
||||
super();
|
||||
|
||||
final MwebSettingsViewModelBase mwebSettingsViewModelBase;
|
||||
final TextEditingController _nodeUriController;
|
||||
|
||||
@override
|
||||
String get title => S.current.litecoin_mweb_node;
|
||||
|
||||
@override
|
||||
Widget body(BuildContext context) {
|
||||
return Stack(
|
||||
fit: StackFit.expand,
|
||||
children: [
|
||||
Container(
|
||||
padding: EdgeInsets.symmetric(horizontal: 24),
|
||||
child: Row(
|
||||
children: <Widget>[
|
||||
Expanded(
|
||||
child: BaseTextFormField(controller: _nodeUriController),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
Positioned(
|
||||
child: Observer(
|
||||
builder: (_) => LoadingPrimaryButton(
|
||||
onPressed: () => save(context),
|
||||
text: S.of(context).save,
|
||||
color: Theme.of(context).primaryColor,
|
||||
textColor: Colors.white,
|
||||
),
|
||||
),
|
||||
bottom: 24,
|
||||
left: 24,
|
||||
right: 24,
|
||||
)
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
void save(BuildContext context) {
|
||||
mwebSettingsViewModelBase.setMwebNodeUri(_nodeUriController.text);
|
||||
Navigator.pop(context);
|
||||
}
|
||||
}
|
|
@ -31,7 +31,7 @@ class MwebSettingsPage extends BasePage {
|
|||
},
|
||||
),
|
||||
SettingsSwitcherCell(
|
||||
title: S.current.litecoin_mweb_always_scan,
|
||||
title: S.current.litecoin_mweb_enable,
|
||||
value: _mwebSettingsViewModel.mwebEnabled,
|
||||
onValueChange: (_, bool value) {
|
||||
_mwebSettingsViewModel.setMwebEnabled(value);
|
||||
|
@ -41,6 +41,14 @@ class MwebSettingsPage extends BasePage {
|
|||
title: S.current.litecoin_mweb_scanning,
|
||||
handler: (BuildContext context) => Navigator.of(context).pushNamed(Routes.rescan),
|
||||
),
|
||||
SettingsCellWithArrow(
|
||||
title: S.current.litecoin_mweb_logs,
|
||||
handler: (BuildContext context) => Navigator.of(context).pushNamed(Routes.mwebLogs),
|
||||
),
|
||||
SettingsCellWithArrow(
|
||||
title: S.current.litecoin_mweb_node,
|
||||
handler: (BuildContext context) => Navigator.of(context).pushNamed(Routes.mwebNode),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
|
|
|
@ -52,7 +52,9 @@ class BottomSheetListenerState extends State<BottomSheetListener> {
|
|||
);
|
||||
},
|
||||
);
|
||||
item.completer.complete(value);
|
||||
if (!item.completer.isCompleted) {
|
||||
item.completer.complete(value);
|
||||
}
|
||||
widget.bottomSheetService.resetCurrentSheet();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -25,6 +25,7 @@ class CakeImageWidget extends StatelessWidget {
|
|||
imageUrl!,
|
||||
height: height,
|
||||
width: width,
|
||||
errorBuilder: (_, __, ___) => Icon(Icons.error),
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -33,6 +34,7 @@ class CakeImageWidget extends StatelessWidget {
|
|||
imageUrl!,
|
||||
height: height,
|
||||
width: width,
|
||||
placeholderBuilder: (_) => Icon(Icons.error),
|
||||
);
|
||||
}
|
||||
|
||||
|
|
|
@ -46,6 +46,7 @@ class _ServicesUpdatesWidgetState extends State<ServicesUpdatesWidget> {
|
|||
"assets/images/notification_icon.svg",
|
||||
color: Theme.of(context).extension<DashboardPageTheme>()!.pageTitleTextColor,
|
||||
width: 30,
|
||||
placeholderBuilder: (_) => Icon(Icons.error),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
@ -136,6 +137,7 @@ class _ServicesUpdatesWidgetState extends State<ServicesUpdatesWidget> {
|
|||
"assets/images/notification_icon.svg",
|
||||
color: Theme.of(context).extension<DashboardPageTheme>()!.pageTitleTextColor,
|
||||
width: 30,
|
||||
placeholderBuilder: (_) => Icon(Icons.error),
|
||||
),
|
||||
if (state.hasData && state.data!.hasUpdates && !wasOpened)
|
||||
Container(
|
||||
|
|
|
@ -91,8 +91,9 @@ abstract class TransactionFilterStoreBase with Store {
|
|||
(displayOutgoing && item.transaction.direction == TransactionDirection.outgoing) ||
|
||||
(displayIncoming &&
|
||||
item.transaction.direction == TransactionDirection.incoming &&
|
||||
!bitcoin!.txIsReceivedSilentPayment(item.transaction)) ||
|
||||
(displaySilentPayments && bitcoin!.txIsReceivedSilentPayment(item.transaction));
|
||||
!(bitcoin?.txIsReceivedSilentPayment(item.transaction) ?? false)) ||
|
||||
(displaySilentPayments &&
|
||||
(bitcoin?.txIsReceivedSilentPayment(item.transaction) ?? false));
|
||||
} else if (item is AnonpayTransactionListItem) {
|
||||
allowed = displayIncoming;
|
||||
}
|
||||
|
|
|
@ -119,6 +119,7 @@ abstract class SettingsStoreBase with Store {
|
|||
required this.mwebCardDisplay,
|
||||
required this.mwebEnabled,
|
||||
required this.hasEnabledMwebBefore,
|
||||
required this.mwebNodeUri,
|
||||
TransactionPriority? initialBitcoinTransactionPriority,
|
||||
TransactionPriority? initialMoneroTransactionPriority,
|
||||
TransactionPriority? initialWowneroTransactionPriority,
|
||||
|
@ -346,8 +347,8 @@ abstract class SettingsStoreBase with Store {
|
|||
|
||||
reaction(
|
||||
(_) => bitcoinSeedType,
|
||||
(BitcoinSeedType bitcoinSeedType) => sharedPreferences.setInt(
|
||||
PreferencesKey.bitcoinSeedType, bitcoinSeedType.raw));
|
||||
(BitcoinSeedType bitcoinSeedType) =>
|
||||
sharedPreferences.setInt(PreferencesKey.bitcoinSeedType, bitcoinSeedType.raw));
|
||||
|
||||
reaction(
|
||||
(_) => nanoSeedType,
|
||||
|
@ -434,8 +435,10 @@ abstract class SettingsStoreBase with Store {
|
|||
reaction((_) => useTronGrid,
|
||||
(bool useTronGrid) => _sharedPreferences.setBool(PreferencesKey.useTronGrid, useTronGrid));
|
||||
|
||||
reaction((_) => useMempoolFeeAPI,
|
||||
(bool useMempoolFeeAPI) => _sharedPreferences.setBool(PreferencesKey.useMempoolFeeAPI, useMempoolFeeAPI));
|
||||
reaction(
|
||||
(_) => useMempoolFeeAPI,
|
||||
(bool useMempoolFeeAPI) =>
|
||||
_sharedPreferences.setBool(PreferencesKey.useMempoolFeeAPI, useMempoolFeeAPI));
|
||||
|
||||
reaction((_) => defaultNanoRep,
|
||||
(String nanoRep) => _sharedPreferences.setString(PreferencesKey.defaultNanoRep, nanoRep));
|
||||
|
@ -583,6 +586,11 @@ abstract class SettingsStoreBase with Store {
|
|||
(bool hasEnabledMwebBefore) =>
|
||||
_sharedPreferences.setBool(PreferencesKey.hasEnabledMwebBefore, hasEnabledMwebBefore));
|
||||
|
||||
reaction(
|
||||
(_) => mwebNodeUri,
|
||||
(String mwebNodeUri) =>
|
||||
_sharedPreferences.setString(PreferencesKey.mwebNodeUri, mwebNodeUri));
|
||||
|
||||
this.nodes.observe((change) {
|
||||
if (change.newValue != null && change.key != null) {
|
||||
_saveCurrentNode(change.newValue!, change.key!);
|
||||
|
@ -811,6 +819,9 @@ abstract class SettingsStoreBase with Store {
|
|||
@observable
|
||||
bool hasEnabledMwebBefore;
|
||||
|
||||
@observable
|
||||
String mwebNodeUri;
|
||||
|
||||
final SecureStorage _secureStorage;
|
||||
final SharedPreferences _sharedPreferences;
|
||||
final BackgroundTasks _backgroundTasks;
|
||||
|
@ -973,6 +984,8 @@ abstract class SettingsStoreBase with Store {
|
|||
final mwebEnabled = sharedPreferences.getBool(PreferencesKey.mwebEnabled) ?? false;
|
||||
final hasEnabledMwebBefore =
|
||||
sharedPreferences.getBool(PreferencesKey.hasEnabledMwebBefore) ?? false;
|
||||
final mwebNodeUri = sharedPreferences.getString(PreferencesKey.mwebNodeUri) ??
|
||||
"ltc-electrum.cakewallet.com:9333";
|
||||
|
||||
// If no value
|
||||
if (pinLength == null || pinLength == 0) {
|
||||
|
@ -1243,6 +1256,7 @@ abstract class SettingsStoreBase with Store {
|
|||
mwebAlwaysScan: mwebAlwaysScan,
|
||||
mwebCardDisplay: mwebCardDisplay,
|
||||
mwebEnabled: mwebEnabled,
|
||||
mwebNodeUri: mwebNodeUri,
|
||||
hasEnabledMwebBefore: hasEnabledMwebBefore,
|
||||
initialMoneroTransactionPriority: moneroTransactionPriority,
|
||||
initialWowneroTransactionPriority: wowneroTransactionPriority,
|
||||
|
@ -1669,7 +1683,8 @@ abstract class SettingsStoreBase with Store {
|
|||
deviceName = windowsInfo.productName;
|
||||
} catch (e) {
|
||||
print(e);
|
||||
print('likely digitalProductId is null wait till https://github.com/fluttercommunity/plus_plugins/pull/3188 is merged');
|
||||
print(
|
||||
'likely digitalProductId is null wait till https://github.com/fluttercommunity/plus_plugins/pull/3188 is merged');
|
||||
deviceName = "Windows Device";
|
||||
}
|
||||
}
|
||||
|
|
|
@ -62,6 +62,14 @@ class ExceptionHandler {
|
|||
|
||||
await _addDeviceInfo(_file!);
|
||||
|
||||
// Check if a mail client is available
|
||||
final bool canSend = await FlutterMailer.canSendMail();
|
||||
|
||||
if (Platform.isIOS && !canSend) {
|
||||
debugPrint('Mail app is not available');
|
||||
return;
|
||||
}
|
||||
|
||||
final MailOptions mailOptions = MailOptions(
|
||||
subject: 'Mobile App Issue',
|
||||
recipients: ['support@cakewallet.com'],
|
||||
|
|
|
@ -53,8 +53,18 @@ class ImageUtil {
|
|||
);
|
||||
} else {
|
||||
return isSvg
|
||||
? SvgPicture.asset(imagePath, height: _height, width: _width)
|
||||
: Image.asset(imagePath, height: _height, width: _width);
|
||||
? SvgPicture.asset(
|
||||
imagePath,
|
||||
height: _height,
|
||||
width: _width,
|
||||
placeholderBuilder: (_) => Icon(Icons.error),
|
||||
)
|
||||
: Image.asset(
|
||||
imagePath,
|
||||
height: _height,
|
||||
width: _width,
|
||||
errorBuilder: (_, __, ___) => Icon(Icons.error),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -99,18 +99,23 @@ abstract class ContactListViewModelBase with Store {
|
|||
|
||||
Future<void> delete(ContactRecord contact) async => contact.original.delete();
|
||||
|
||||
@computed
|
||||
List<ContactRecord> get contactsToShow =>
|
||||
contacts.where((element) => _isValidForCurrency(element)).toList();
|
||||
ObservableList<ContactRecord> get contactsToShow =>
|
||||
ObservableList.of(contacts.where((element) => _isValidForCurrency(element)));
|
||||
|
||||
@computed
|
||||
List<WalletContact> get walletContactsToShow =>
|
||||
walletContacts.where((element) => _isValidForCurrency(element)).toList();
|
||||
|
||||
bool _isValidForCurrency(ContactBase element) {
|
||||
if (element.name.contains('Silent Payments')) return false;
|
||||
if (element.name.contains('MWEB')) return false;
|
||||
|
||||
return _currency == null ||
|
||||
element.type == _currency ||
|
||||
element.type.title == _currency!.tag ||
|
||||
element.type.tag == _currency!.tag;
|
||||
(element.type.tag != null &&
|
||||
_currency?.tag != null &&
|
||||
element.type.tag == _currency?.tag) ||
|
||||
_currency?.toString() == element.type.tag ||
|
||||
_currency?.tag == element.type.toString();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -381,7 +381,7 @@ abstract class BalanceViewModelBase with Store {
|
|||
|
||||
bool _hasSecondAdditionalBalanceForWalletType(WalletType type) {
|
||||
if (wallet.type == WalletType.litecoin) {
|
||||
if ((wallet.balance[CryptoCurrency.ltc]?.secondAdditional ?? 0) > 0) {
|
||||
if ((wallet.balance[CryptoCurrency.ltc]?.secondAdditional ?? 0) != 0) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -90,11 +90,12 @@ abstract class DashboardViewModelBase with Store {
|
|||
value: () => transactionFilterStore.displayOutgoing,
|
||||
caption: S.current.outgoing,
|
||||
onChanged: transactionFilterStore.toggleOutgoing),
|
||||
FilterItem(
|
||||
value: () => transactionFilterStore.displaySilentPayments,
|
||||
caption: S.current.silent_payments,
|
||||
onChanged: transactionFilterStore.toggleSilentPayments,
|
||||
),
|
||||
if (appStore.wallet!.type == WalletType.bitcoin)
|
||||
FilterItem(
|
||||
value: () => transactionFilterStore.displaySilentPayments,
|
||||
caption: S.current.silent_payments,
|
||||
onChanged: transactionFilterStore.toggleSilentPayments,
|
||||
),
|
||||
// FilterItem(
|
||||
// value: () => false,
|
||||
// caption: S.current.transactions_by_date,
|
||||
|
@ -435,7 +436,10 @@ abstract class DashboardViewModelBase with Store {
|
|||
}
|
||||
|
||||
@computed
|
||||
bool get hasMweb => wallet.type == WalletType.litecoin && (Platform.isIOS || Platform.isAndroid) && !wallet.isHardwareWallet;
|
||||
bool get hasMweb =>
|
||||
wallet.type == WalletType.litecoin &&
|
||||
(Platform.isIOS || Platform.isAndroid) &&
|
||||
!wallet.isHardwareWallet;
|
||||
|
||||
@computed
|
||||
bool get showMwebCard => hasMweb && settingsStore.mwebCardDisplay && !mwebEnabled;
|
||||
|
|
|
@ -56,25 +56,53 @@ class TransactionListItem extends ActionListItem with Keyable {
|
|||
}
|
||||
|
||||
String get formattedPendingStatus {
|
||||
if (balanceViewModel.wallet.type == WalletType.monero ||
|
||||
balanceViewModel.wallet.type == WalletType.haven) {
|
||||
if (transaction.confirmations >= 0 && transaction.confirmations < 10) {
|
||||
return ' (${transaction.confirmations}/10)';
|
||||
}
|
||||
} else if (balanceViewModel.wallet.type == WalletType.wownero) {
|
||||
if (transaction.confirmations >= 0 && transaction.confirmations < 3) {
|
||||
return ' (${transaction.confirmations}/3)';
|
||||
}
|
||||
switch (balanceViewModel.wallet.type) {
|
||||
case WalletType.monero:
|
||||
case WalletType.haven:
|
||||
if (transaction.confirmations >= 0 && transaction.confirmations < 10) {
|
||||
return ' (${transaction.confirmations}/10)';
|
||||
}
|
||||
break;
|
||||
case WalletType.wownero:
|
||||
if (transaction.confirmations >= 0 && transaction.confirmations < 3) {
|
||||
return ' (${transaction.confirmations}/3)';
|
||||
}
|
||||
break;
|
||||
case WalletType.litecoin:
|
||||
bool isPegIn = (transaction.additionalInfo["isPegIn"] as bool?) ?? false;
|
||||
bool isPegOut = (transaction.additionalInfo["isPegOut"] as bool?) ?? false;
|
||||
bool fromPegOut = (transaction.additionalInfo["fromPegOut"] as bool?) ?? false;
|
||||
String str = '';
|
||||
if (transaction.confirmations <= 0) {
|
||||
str = S.current.pending;
|
||||
}
|
||||
if ((isPegOut || fromPegOut) && transaction.confirmations >= 0 && transaction.confirmations < 6) {
|
||||
str = " (${transaction.confirmations}/6)";
|
||||
}
|
||||
if (isPegIn) {
|
||||
str += " (Peg In)";
|
||||
}
|
||||
if (isPegOut) {
|
||||
str += " (Peg Out)";
|
||||
}
|
||||
return str;
|
||||
default:
|
||||
return '';
|
||||
}
|
||||
|
||||
return '';
|
||||
}
|
||||
|
||||
String get formattedStatus {
|
||||
if (balanceViewModel.wallet.type == WalletType.monero ||
|
||||
balanceViewModel.wallet.type == WalletType.wownero ||
|
||||
balanceViewModel.wallet.type == WalletType.haven) {
|
||||
if ([
|
||||
WalletType.monero,
|
||||
WalletType.haven,
|
||||
WalletType.wownero,
|
||||
WalletType.litecoin,
|
||||
].contains(balanceViewModel.wallet.type)) {
|
||||
return formattedPendingStatus;
|
||||
}
|
||||
|
||||
return transaction.isPending ? S.current.pending : '';
|
||||
}
|
||||
|
||||
|
|
|
@ -400,7 +400,8 @@ abstract class SendViewModelBase extends WalletChangeListenerViewModel with Stor
|
|||
final updatedOutputs = bitcoin!.updateOutputs(pendingTransaction!, outputs);
|
||||
|
||||
if (outputs.length == updatedOutputs.length) {
|
||||
outputs = ObservableList.of(updatedOutputs);
|
||||
outputs.clear();
|
||||
outputs.addAll(updatedOutputs);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -1,7 +1,12 @@
|
|||
import 'dart:io';
|
||||
|
||||
import 'package:cake_wallet/bitcoin/bitcoin.dart';
|
||||
import 'package:cake_wallet/store/settings_store.dart';
|
||||
import 'package:cake_wallet/utils/exception_handler.dart';
|
||||
import 'package:cw_core/wallet_base.dart';
|
||||
import 'package:flutter/widgets.dart';
|
||||
import 'package:mobx/mobx.dart';
|
||||
import 'package:path_provider/path_provider.dart';
|
||||
|
||||
part 'mweb_settings_view_model.g.dart';
|
||||
|
||||
|
@ -22,15 +27,60 @@ abstract class MwebSettingsViewModelBase with Store {
|
|||
@observable
|
||||
late bool mwebEnabled;
|
||||
|
||||
@computed
|
||||
String get mwebNodeUri => _settingsStore.mwebNodeUri;
|
||||
|
||||
@action
|
||||
void setMwebCardDisplay(bool value) {
|
||||
_settingsStore.mwebCardDisplay = value;
|
||||
}
|
||||
|
||||
@action
|
||||
void setMwebNodeUri(String value) {
|
||||
_settingsStore.mwebNodeUri = value;
|
||||
}
|
||||
|
||||
@action
|
||||
void setMwebEnabled(bool value) {
|
||||
mwebEnabled = value;
|
||||
bitcoin!.setMwebEnabled(_wallet, value);
|
||||
_settingsStore.mwebAlwaysScan = value;
|
||||
}
|
||||
|
||||
Future<bool> saveLogsLocally(String filePath) async {
|
||||
try {
|
||||
final appSupportPath = (await getApplicationSupportDirectory()).path;
|
||||
final logsFile = File("$appSupportPath/logs/debug.log");
|
||||
if (!logsFile.existsSync()) {
|
||||
throw Exception('Logs file does not exist');
|
||||
}
|
||||
await logsFile.copy(filePath);
|
||||
return true;
|
||||
} catch (e, s) {
|
||||
ExceptionHandler.onError(FlutterErrorDetails(
|
||||
exception: e,
|
||||
stack: s,
|
||||
library: "Export Logs",
|
||||
));
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
Future<String> getAbbreviatedLogs() async {
|
||||
final appSupportPath = (await getApplicationSupportDirectory()).path;
|
||||
final logsFile = File("$appSupportPath/logs/debug.log");
|
||||
if (!logsFile.existsSync()) {
|
||||
return "";
|
||||
}
|
||||
final logs = logsFile.readAsStringSync();
|
||||
// return last 10000 characters:
|
||||
return logs.substring(logs.length > 10000 ? logs.length - 10000 : 0);
|
||||
}
|
||||
|
||||
Future<void> removeLogsLocally(String filePath) async {
|
||||
final logsFile = File(filePath);
|
||||
if (logsFile.existsSync()) {
|
||||
await logsFile.delete();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -136,7 +136,7 @@ dependency_overrides:
|
|||
bitcoin_base:
|
||||
git:
|
||||
url: https://github.com/cake-tech/bitcoin_base
|
||||
ref: cake-update-v8
|
||||
ref: cake-update-v9
|
||||
ffi: 2.1.0
|
||||
|
||||
flutter_icons:
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "ﻲﻓ ﻪﺘﻴﺣﻼﺻ ﻲﻬﺘﻨﺗ",
|
||||
"expiry_and_validity": "انتهاء الصلاحية والصلاحية",
|
||||
"export_backup": "تصدير نسخة احتياطية",
|
||||
"export_logs": "سجلات التصدير",
|
||||
"extra_id": "معرف إضافي:",
|
||||
"extracted_address_content": "سوف ترسل الأموال إلى\n${recipient_name}",
|
||||
"failed_authentication": "${state_error} فشل المصادقة.",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB هو بروتوكول جديد يجلب معاملات أسرع وأرخص وأكثر خصوصية إلى Litecoin",
|
||||
"litecoin_mweb_dismiss": "رفض",
|
||||
"litecoin_mweb_display_card": "عرض بطاقة mweb",
|
||||
"litecoin_mweb_enable": "تمكين MWEB",
|
||||
"litecoin_mweb_enable_later": "يمكنك اختيار تمكين MWEB مرة أخرى ضمن إعدادات العرض.",
|
||||
"litecoin_mweb_logs": "سجلات MWEB",
|
||||
"litecoin_mweb_node": "عقدة MWEB",
|
||||
"litecoin_mweb_pegin": "ربط في",
|
||||
"litecoin_mweb_pegout": "ربط",
|
||||
"litecoin_mweb_scanning": "MWEB المسح الضوئي",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Изтича на",
|
||||
"expiry_and_validity": "Изтичане и валидност",
|
||||
"export_backup": "Експортиране на резервно копие",
|
||||
"export_logs": "Експортни дневници",
|
||||
"extra_id": "Допълнително ID:",
|
||||
"extracted_address_content": "Ще изпратите средства на \n${recipient_name}",
|
||||
"failed_authentication": "Неуспешно удостоверяване. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWeb е нов протокол, който носи по -бърз, по -евтин и повече частни транзакции на Litecoin",
|
||||
"litecoin_mweb_dismiss": "Уволнение",
|
||||
"litecoin_mweb_display_card": "Показване на MWEB карта",
|
||||
"litecoin_mweb_enable": "Активирайте MWeb",
|
||||
"litecoin_mweb_enable_later": "Можете да изберете да активирате MWEB отново под настройките на дисплея.",
|
||||
"litecoin_mweb_logs": "MWeb logs",
|
||||
"litecoin_mweb_node": "MWEB възел",
|
||||
"litecoin_mweb_pegin": "PEG в",
|
||||
"litecoin_mweb_pegout": "PEG OUT",
|
||||
"litecoin_mweb_scanning": "Сканиране на MWEB",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Vyprší dne",
|
||||
"expiry_and_validity": "Vypršení a platnost",
|
||||
"export_backup": "Exportovat zálohu",
|
||||
"export_logs": "Vývozní protokoly",
|
||||
"extra_id": "Extra ID:",
|
||||
"extracted_address_content": "Prostředky budete posílat na\n${recipient_name}",
|
||||
"failed_authentication": "Ověřování selhalo. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB je nový protokol, který do Litecoin přináší rychlejší, levnější a více soukromých transakcí",
|
||||
"litecoin_mweb_dismiss": "Propustit",
|
||||
"litecoin_mweb_display_card": "Zobrazit kartu MWeb",
|
||||
"litecoin_mweb_enable": "Povolit mWeb",
|
||||
"litecoin_mweb_enable_later": "V nastavení zobrazení můžete vybrat znovu povolit MWeb.",
|
||||
"litecoin_mweb_logs": "Protokoly mWeb",
|
||||
"litecoin_mweb_node": "Uzel mWeb",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Zkrachovat",
|
||||
"litecoin_mweb_scanning": "Skenování mWeb",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Läuft aus am",
|
||||
"expiry_and_validity": "Ablauf und Gültigkeit",
|
||||
"export_backup": "Sicherung exportieren",
|
||||
"export_logs": "Exportprotokolle",
|
||||
"extra_id": "Extra ID:",
|
||||
"extracted_address_content": "Sie senden Geld an\n${recipient_name}",
|
||||
"failed_authentication": "Authentifizierung fehlgeschlagen. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWWB ist ein neues Protokoll, das schnellere, billigere und privatere Transaktionen zu Litecoin bringt",
|
||||
"litecoin_mweb_dismiss": "Zurückweisen",
|
||||
"litecoin_mweb_display_card": "MWEB-Karte anzeigen",
|
||||
"litecoin_mweb_enable": "Aktivieren Sie MWeb",
|
||||
"litecoin_mweb_enable_later": "Sie können MWEB unter Anzeigeeinstellungen erneut aktivieren.",
|
||||
"litecoin_mweb_logs": "MWEB -Protokolle",
|
||||
"litecoin_mweb_node": "MWEB -Knoten",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Abstecken",
|
||||
"litecoin_mweb_scanning": "MWEB Scanning",
|
||||
|
@ -941,4 +945,4 @@
|
|||
"you_will_get": "Konvertieren zu",
|
||||
"you_will_send": "Konvertieren von",
|
||||
"yy": "YY"
|
||||
}
|
||||
}
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Expires on",
|
||||
"expiry_and_validity": "Expiry and Validity",
|
||||
"export_backup": "Export backup",
|
||||
"export_logs": "Export logs",
|
||||
"extra_id": "Extra ID:",
|
||||
"extracted_address_content": "You will be sending funds to\n${recipient_name}",
|
||||
"failed_authentication": "Failed authentication. ${state_error}",
|
||||
|
@ -373,7 +374,10 @@
|
|||
"litecoin_mweb_description": "MWEB is a new protocol that brings faster, cheaper, and more private transactions to Litecoin",
|
||||
"litecoin_mweb_dismiss": "Dismiss",
|
||||
"litecoin_mweb_display_card": "Show MWEB card",
|
||||
"litecoin_mweb_enable": "Enable MWEB",
|
||||
"litecoin_mweb_enable_later": "You can choose to enable MWEB again under Display Settings.",
|
||||
"litecoin_mweb_logs": "MWEB Logs",
|
||||
"litecoin_mweb_node": "MWEB Node",
|
||||
"litecoin_mweb_pegin": "Peg In",
|
||||
"litecoin_mweb_pegout": "Peg Out",
|
||||
"litecoin_mweb_scanning": "MWEB Scanning",
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
{
|
||||
"about_cake_pay": "Cake Pay le permite comprar fácilmente tarjetas de regalo con activos virtuales, gastables instantáneamente en más de 150 000 comerciantes en los Estados Unidos.",
|
||||
"about_cake_pay": "Cake Pay te permite comprar fácilmente tarjetas de regalo con activos virtuales, gastables instantáneamente en más de 150,000 comerciantes en los Estados Unidos.",
|
||||
"account": "Cuenta",
|
||||
"accounts": "Cuentas",
|
||||
"accounts_subaddresses": "Cuentas y subdirecciones",
|
||||
|
@ -21,13 +21,13 @@
|
|||
"add_token_disclaimer_check": "He confirmado la dirección del contrato del token y la información utilizando una fuente confiable. Agregar información maliciosa o incorrecta puede resultar en una pérdida de fondos.",
|
||||
"add_token_warning": "No edite ni agregue tokens según las instrucciones de los estafadores.\n¡Confirme siempre las direcciones de los tokens con fuentes acreditadas!",
|
||||
"add_value": "Añadir valor",
|
||||
"address": "DIRECCIÓN",
|
||||
"address": "Dirección",
|
||||
"address_book": "Libreta de direcciones",
|
||||
"address_book_menu": "Libreta de direcciones",
|
||||
"address_detected": "Dirección detectada",
|
||||
"address_from_domain": "Esta dirección es de ${domain} en Unstoppable Domains",
|
||||
"address_from_yat": "Esta dirección es de ${emoji} en Yat",
|
||||
"address_label": "Address label",
|
||||
"address_label": "Etiqueta de dirección",
|
||||
"address_remove_contact": "Remover contacto",
|
||||
"address_remove_content": "¿Estás seguro de que quieres eliminar el contacto seleccionado?",
|
||||
"addresses": "Direcciones",
|
||||
|
@ -37,12 +37,12 @@
|
|||
"agree_and_continue": "Aceptar y continuar",
|
||||
"agree_to": "Al crear una cuenta, aceptas ",
|
||||
"alert_notice": "Aviso",
|
||||
"all": "TODOS",
|
||||
"all": "Todos",
|
||||
"all_trades": "Todos los oficios",
|
||||
"all_transactions": "Todas las transacciones",
|
||||
"alphabetical": "Alfabético",
|
||||
"already_have_account": "¿Ya tienes una cuenta?",
|
||||
"always": "siempre",
|
||||
"always": "Siempre",
|
||||
"amount": "Cantidad: ",
|
||||
"amount_is_below_minimum_limit": "Su saldo después de las tarifas sería menor que la cantidad mínima necesaria para el intercambio (${min})",
|
||||
"amount_is_estimate": "El monto recibido es un estimado",
|
||||
|
@ -54,17 +54,17 @@
|
|||
"arrive_in_this_address": "${currency} ${tag}llegará a esta dirección",
|
||||
"ascending": "Ascendente",
|
||||
"ask_each_time": "Pregunta cada vez",
|
||||
"auth_store_ban_timeout": "prohibición de tiempo de espera",
|
||||
"auth_store_ban_timeout": "Prohibición de tiempo de espera",
|
||||
"auth_store_banned_for": "Prohibido para ",
|
||||
"auth_store_banned_minutes": " minutos",
|
||||
"auth_store_incorrect_password": "Contraseña PIN",
|
||||
"authenticated": "Autenticados",
|
||||
"authentication": "Autenticación",
|
||||
"auto_generate_addresses": "Auto Generar direcciones",
|
||||
"auto_generate_addresses": "Auto-generar nuevas direcciones",
|
||||
"auto_generate_subaddresses": "Generar subdirecciones automáticamente",
|
||||
"automatic": "Automático",
|
||||
"available_balance": "Balance disponible",
|
||||
"available_balance_description": "Su saldo disponible es la cantidad de fondos que puede gastar. Los fondos que se muestran aquí se pueden gastar inmediatamente.",
|
||||
"available_balance_description": "Tu saldo disponible es la cantidad de fondos que puedes gastar. Los fondos que se muestran aquí, se pueden gastar inmediatamente.",
|
||||
"avg_savings": "Ahorro promedio",
|
||||
"awaitDAppProcessing": "Espere a que la dApp termine de procesarse.",
|
||||
"awaiting_payment_confirmation": "Esperando confirmación de pago",
|
||||
|
@ -79,7 +79,7 @@
|
|||
"biometric_auth_reason": "Escanee su huella digital para autenticar",
|
||||
"bitcoin_dark_theme": "Tema oscuro de Bitcoin",
|
||||
"bitcoin_light_theme": "Tema claro de Bitcoin",
|
||||
"bitcoin_payments_require_1_confirmation": "Los pagos de Bitcoin requieren 1 confirmación, que puede demorar 20 minutos o más. ¡Gracias por su paciencia! Se le enviará un correo electrónico cuando se confirme el pago.",
|
||||
"bitcoin_payments_require_1_confirmation": "Los pagos de Bitcoin requieren 1 confirmación, que puede demorar 20 minutos o más. ¡Gracias por tu paciencia! Se te enviará un correo electrónico cuando se confirme el pago.",
|
||||
"block_remaining": "1 bloqueo restante",
|
||||
"Blocks_remaining": "${status} Bloques restantes",
|
||||
"bluetooth": "Bluetooth",
|
||||
|
@ -94,25 +94,25 @@
|
|||
"by_cake_pay": "por Cake Pay",
|
||||
"cake_2fa_preset": "Pastel 2FA preestablecido",
|
||||
"cake_dark_theme": "Tema oscuro",
|
||||
"cake_pay_account_note": "Regístrese con solo una dirección de correo electrónico para ver y comprar tarjetas. ¡Algunas incluso están disponibles con descuento!",
|
||||
"cake_pay_learn_more": "¡Compre y canjee tarjetas de regalo al instante en la aplicación!\nDeslice el dedo de izquierda a derecha para obtener más información.",
|
||||
"cake_pay_save_order": "La tarjeta debe enviarse a su correo electrónico dentro de 1 día hábil \n Guardar su ID de pedido:",
|
||||
"cake_pay_subtitle": "Compre tarjetas prepagas y tarjetas de regalo en todo el mundo",
|
||||
"cake_pay_web_cards_subtitle": "Compre tarjetas de prepago y tarjetas de regalo en todo el mundo",
|
||||
"cake_pay_account_note": "Regístrate con solo una dirección de correo electrónico para ver y comprar tarjetas. ¡Algunas incluso están disponibles con descuento!",
|
||||
"cake_pay_learn_more": "¡Compra y canjea tarjetas de regalo al instante en la aplicación!\nDesliza el dedo de izquierda a derecha para obtener más información.",
|
||||
"cake_pay_save_order": "La tarjeta debe enviarse a tu correo electrónico dentro de 1 día hábil \n Guardar su ID de pedido:",
|
||||
"cake_pay_subtitle": "Compra tarjetas prepagadas y tarjetas de regalo en todo el mundo",
|
||||
"cake_pay_web_cards_subtitle": "Compra tarjetas de prepago y tarjetas de regalo en todo el mundo",
|
||||
"cake_pay_web_cards_title": "Tarjetas Web Cake Pay",
|
||||
"cake_wallet": "Cake Wallet",
|
||||
"cakepay_prepaid_card": "Tarjeta de Débito Prepago CakePay",
|
||||
"camera_consent": "Su cámara será utilizada para capturar una imagen con fines de identificación por ${provider}. Consulte su Política de privacidad para obtener más detalles.",
|
||||
"camera_permission_is_required": "Se requiere permiso de la cámara.\nHabilítelo desde la configuración de la aplicación.",
|
||||
"camera_consent": "Su cámara será utilizada para capturar una imagen con fines de identificación por ${provider}. Consulta tu Política de privacidad para obtener más detalles.",
|
||||
"camera_permission_is_required": "Se requiere permiso de la cámara.\nHabilítalo desde la configuración de la aplicación.",
|
||||
"cancel": "Cancelar",
|
||||
"card_address": "Dirección:",
|
||||
"cardholder_agreement": "Acuerdo del titular de la tarjeta",
|
||||
"cards": "Cartas",
|
||||
"chains": "Cadenas",
|
||||
"change": "Cambio",
|
||||
"change_backup_password_alert": "Sus archivos de respaldo anteriores no estarán disponibles para importar con la nueva contraseña de respaldo. La nueva contraseña de respaldo se utilizará solo para los nuevos archivos de respaldo. ¿Está seguro de que desea cambiar la contraseña de respaldo?",
|
||||
"change_backup_password_alert": "Tus archivos de respaldo anteriores no estarán disponibles para importar con la nueva contraseña de respaldo. La nueva contraseña de respaldo se utilizará solo para los nuevos archivos de respaldo. ¿Está seguro de que desea cambiar la contraseña de respaldo?",
|
||||
"change_currency": "Cambiar moneda",
|
||||
"change_current_node": "¿Está seguro de cambiar el nodo actual a ${node}?",
|
||||
"change_current_node": "¿Estás seguro de cambiar el nodo actual a ${node}?",
|
||||
"change_current_node_title": "Cambiar el nodo actual",
|
||||
"change_exchange_provider": "Cambiar proveedor de intercambio",
|
||||
"change_language": "Cambiar idioma",
|
||||
|
@ -125,38 +125,38 @@
|
|||
"change_wallet_alert_title": "Cambiar billetera actual",
|
||||
"choose_account": "Elegir cuenta",
|
||||
"choose_address": "\n\nPor favor elija la dirección:",
|
||||
"choose_card_value": "Elija un valor de tarjeta",
|
||||
"choose_derivation": "Elija la derivación de la billetera",
|
||||
"choose_from_available_options": "Elija entre las opciones disponibles:",
|
||||
"choose_card_value": "Elige un valor de tarjeta",
|
||||
"choose_derivation": "Elige la derivación de la billetera",
|
||||
"choose_from_available_options": "Elige entre las opciones disponibles:",
|
||||
"choose_one": "Elige uno",
|
||||
"choose_relay": "Por favor elija un relé para usar",
|
||||
"choose_wallet_currency": "Por favor, elija la moneda de la billetera:",
|
||||
"choose_wallet_group": "Elija el grupo de billetera",
|
||||
"choose_relay": "Por favor elige un relay para usar",
|
||||
"choose_wallet_currency": "Por favor, elige la moneda de la billetera:",
|
||||
"choose_wallet_group": "Elige el grupo de billetera",
|
||||
"clear": "Claro",
|
||||
"clearnet_link": "enlace Clearnet",
|
||||
"close": "Cerca",
|
||||
"coin_control": "Control de monedas (opcional)",
|
||||
"cold_or_recover_wallet": "Agregue una billetera fría o recupere una billetera de papel",
|
||||
"cold_or_recover_wallet": "Agrega una billetera fría o recupera una billetera de papel",
|
||||
"color_theme": "Tema de color",
|
||||
"commit_transaction_amount_fee": "Confirmar transacción\nCantidad: ${amount}\nCuota: ${fee}",
|
||||
"confirm": "Confirmar",
|
||||
"confirm_delete_template": "Esta acción eliminará esta plantilla. ¿Desea continuar?",
|
||||
"confirm_delete_wallet": "Esta acción eliminará esta billetera. ¿Desea continuar?",
|
||||
"confirm_delete_template": "Esta acción eliminará esta plantilla. ¿Deseas continuar?",
|
||||
"confirm_delete_wallet": "Esta acción eliminará esta billetera. ¿Deseas continuar?",
|
||||
"confirm_fee_deduction": "Confirmar la deducción de la tarifa",
|
||||
"confirm_fee_deduction_content": "¿Acepta deducir la tarifa de la producción?",
|
||||
"confirm_fee_deduction_content": "¿Aceptas deducir la tarifa de la producción?",
|
||||
"confirm_passphrase": "Confirmar la contraseña",
|
||||
"confirm_sending": "Confirmar envío",
|
||||
"confirm_silent_payments_switch_node": "Su nodo actual no admite pagos silenciosos \\ nCake cambiará a un nodo compatible, solo para escanear",
|
||||
"confirm_silent_payments_switch_node": "Tu nodo actual no admite pagos silenciosos \\ nCake cambiará a un nodo compatible, solo para escanear",
|
||||
"confirmations": "Confirmaciones",
|
||||
"confirmed": "Saldo confirmado",
|
||||
"confirmed_tx": "Confirmado",
|
||||
"congratulations": "Felicidades!",
|
||||
"connect_an_existing_yat": "Conectar un Yat existente",
|
||||
"connect_yats": "Conectar Yats",
|
||||
"connect_your_hardware_wallet": "Conecte su billetera de hardware con Bluetooth o USB",
|
||||
"connect_your_hardware_wallet_ios": "Conecte su billetera de hardware con Bluetooth",
|
||||
"connect_your_hardware_wallet": "Conecta tu billetera de hardware con Bluetooth o USB",
|
||||
"connect_your_hardware_wallet_ios": "Conecta tu billetera de hardware con Bluetooth",
|
||||
"connection_sync": "Conexión y sincronización",
|
||||
"connectWalletPrompt": "Conecte su billetera con WalletConnect para realizar transacciones",
|
||||
"connectWalletPrompt": "Conecte tu billetera con WalletConnect para realizar transacciones",
|
||||
"contact": "Contacto",
|
||||
"contact_list_contacts": "Contactos",
|
||||
"contact_list_wallets": "Mis billeteras",
|
||||
|
@ -187,7 +187,7 @@
|
|||
"custom_drag": "Custom (mantenía y arrastre)",
|
||||
"custom_redeem_amount": "Cantidad de canje personalizada",
|
||||
"custom_value": "Valor personalizado",
|
||||
"dark_theme": "Oscura",
|
||||
"dark_theme": "Oscuro",
|
||||
"debit_card": "Tarjeta de Débito",
|
||||
"debit_card_terms": "El almacenamiento y el uso de su número de tarjeta de pago (y las credenciales correspondientes a su número de tarjeta de pago) en esta billetera digital están sujetos a los Términos y condiciones del acuerdo del titular de la tarjeta aplicable con el emisor de la tarjeta de pago, en vigor desde tiempo al tiempo.",
|
||||
"decimal_places_error": "Demasiados lugares decimales",
|
||||
|
@ -197,21 +197,21 @@
|
|||
"delete": "Borrar",
|
||||
"delete_account": "Eliminar cuenta",
|
||||
"delete_wallet": "Eliminar billetera",
|
||||
"delete_wallet_confirm_message": "¿Está seguro de que desea eliminar la billetera ${wallet_name}?",
|
||||
"deleteConnectionConfirmationPrompt": "¿Está seguro de que desea eliminar la conexión a",
|
||||
"delete_wallet_confirm_message": "¿Estás seguro de que deseas eliminar la billetera ${wallet_name}?",
|
||||
"deleteConnectionConfirmationPrompt": "¿Estás seguro de que deseas eliminar la conexión a",
|
||||
"denominations": "Denominaciones",
|
||||
"derivationpath": "Ruta de derivación",
|
||||
"descending": "Descendente",
|
||||
"description": "Descripción",
|
||||
"destination_tag": "Etiqueta de destino:",
|
||||
"dfx_option_description": "Compre criptografía con EUR y CHF. Para clientes minoristas y corporativos en Europa",
|
||||
"dfx_option_description": "Compre cripto con EUR y CHF. Para clientes minoristas y corporativos en Europa",
|
||||
"didnt_get_code": "¿No recibiste el código?",
|
||||
"digit_pin": "-dígito PIN",
|
||||
"digital_and_physical_card": " tarjeta de débito prepago digital y física",
|
||||
"disable": "Desactivar",
|
||||
"disable_bulletin": "Desactivar el boletín de estado del servicio",
|
||||
"disable_buy": "Desactivar acción de compra",
|
||||
"disable_cake_2fa": "Desactivar pastel 2FA",
|
||||
"disable_cake_2fa": "Desactivar 2FA",
|
||||
"disable_exchange": "Deshabilitar intercambio",
|
||||
"disable_fee_api_warning": "Al apagar esto, las tasas de tarifas pueden ser inexactas en algunos casos, por lo que puede terminar pagando en exceso o pagando menos las tarifas por sus transacciones",
|
||||
"disable_fiat": "Deshabilitar fiat",
|
||||
|
@ -224,7 +224,7 @@
|
|||
"displayable": "Visualizable",
|
||||
"do_not_have_enough_gas_asset": "No tienes suficiente ${currency} para realizar una transacción con las condiciones actuales de la red blockchain. Necesita más ${currency} para pagar las tarifas de la red blockchain, incluso si envía un activo diferente.",
|
||||
"do_not_send": "no enviar",
|
||||
"do_not_share_warning_text": "No comparta estos con nadie más, incluido el soporte.\n\n¡Sus fondos pueden ser y serán robados!",
|
||||
"do_not_share_warning_text": "No compartas estos con nadie más, incluido el soporte.\n\n¡Tus fondos pueden ser y serán robados!",
|
||||
"do_not_show_me": "no me muestres esto otra vez",
|
||||
"domain_looks_up": "Búsquedas de dominio",
|
||||
"donation_link_details": "Detalles del enlace de donación",
|
||||
|
@ -238,21 +238,21 @@
|
|||
"enable": "Permitir",
|
||||
"enable_mempool_api": "API de Mempool para tarifas y fechas precisas",
|
||||
"enable_replace_by_fee": "Habilitar reemplazar por tarea",
|
||||
"enable_silent_payments_scanning": "Comience a escanear pagos silenciosos, hasta que se alcance la punta",
|
||||
"enable_silent_payments_scanning": "Comienza a escanear pagos silenciosos, hasta que se alcance la altura actual",
|
||||
"enabled": "Activado",
|
||||
"enter_amount": "Ingrese la cantidad",
|
||||
"enter_backup_password": "Ingrese la contraseña de respaldo aquí",
|
||||
"enter_amount": "Ingresa la cantidad",
|
||||
"enter_backup_password": "Ingresa la contraseña de respaldo aquí",
|
||||
"enter_code": "Ingresar código",
|
||||
"enter_seed_phrase": "Ingrese su frase de semillas",
|
||||
"enter_totp_code": "Ingrese el código TOTP.",
|
||||
"enter_wallet_password": "Ingrese la contraseña de la billetera",
|
||||
"enter_seed_phrase": "Ingresa su frase de semillas",
|
||||
"enter_totp_code": "Ingresa el código TOTP.",
|
||||
"enter_wallet_password": "Ingresa la contraseña de la billetera",
|
||||
"enter_your_note": "Ingresa tu nota…",
|
||||
"enter_your_pin": "Introduce tu PIN",
|
||||
"enter_your_pin_again": "Ingrese su PIN nuevamente",
|
||||
"enterTokenID": "Ingrese el ID del token",
|
||||
"enterWalletConnectURI": "Ingrese el URI de WalletConnect",
|
||||
"enter_your_pin_again": "Ingresa su PIN nuevamente",
|
||||
"enterTokenID": "Ingresa el ID del token",
|
||||
"enterWalletConnectURI": "Ingresa el URI de WalletConnect",
|
||||
"error": "Error",
|
||||
"error_dialog_content": "Vaya, tenemos un error.\n\nEnvíe el informe de bloqueo a nuestro equipo de soporte para mejorar la aplicación.",
|
||||
"error_dialog_content": "Vaya, tenemos un error.\n\nEnvía el informe de error a nuestro equipo de soporte para mejorar la aplicación.",
|
||||
"error_text_account_name": "El nombre de la cuenta solo puede contener letras, números \ny debe tener entre 1 y 15 caracteres de longitud",
|
||||
"error_text_address": "La dirección de la billetera debe corresponder al tipo \nde criptomoneda",
|
||||
"error_text_amount": "La cantidad solo puede contener números",
|
||||
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Expira el",
|
||||
"expiry_and_validity": "Vencimiento y validez",
|
||||
"export_backup": "Exportar copia de seguridad",
|
||||
"export_logs": "Registros de exportación",
|
||||
"extra_id": "ID adicional:",
|
||||
"extracted_address_content": "Enviará fondos a\n${recipient_name}",
|
||||
"failed_authentication": "Autenticación fallida. ${state_error}",
|
||||
|
@ -304,14 +305,14 @@
|
|||
"fee_rate": "Tarifa",
|
||||
"fetching": "Cargando",
|
||||
"fiat_api": "Fiat API",
|
||||
"fiat_balance": "Equilibrio Fiat",
|
||||
"fiat_balance": "Balance fiat",
|
||||
"field_required": "Este campo es obligatorio",
|
||||
"fill_code": "Por favor complete el código de verificación proporcionado a su correo electrónico",
|
||||
"fill_code": "Por favor completa el código de verificación proporcionado en tu correo electrónico",
|
||||
"filter_by": "Filtrado por",
|
||||
"first_wallet_text": "Impresionante billetera para Monero, Bitcoin, Ethereum, Litecoin, y Haven",
|
||||
"fixed_pair_not_supported": "Este par fijo no es compatible con los servicios de intercambio seleccionados",
|
||||
"fixed_rate": "Tipo de interés fijo",
|
||||
"fixed_rate_alert": "Podrá ingresar la cantidad recibida cuando el modo de tarifa fija esté marcado. ¿Quieres cambiar al modo de tarifa fija?",
|
||||
"fixed_rate_alert": "Podrás ingresar la cantidad recibida cuando el modo de tarifa fija esté marcado. ¿Quieres cambiar al modo de tarifa fija?",
|
||||
"forgot_password": "Olvidé mi contraseña",
|
||||
"freeze": "Congelar",
|
||||
"frequently_asked_questions": "Preguntas frecuentes",
|
||||
|
@ -333,7 +334,7 @@
|
|||
"gross_balance": "Saldo bruto",
|
||||
"group_by_type": "Grupo por tipo",
|
||||
"haven_app": "Haven by Cake Wallet",
|
||||
"haven_app_wallet_text": "Awesome wallet for Haven",
|
||||
"haven_app_wallet_text": "Increíble billetera para Haven",
|
||||
"help": "ayuda",
|
||||
"hidden_balance": "Balance oculto",
|
||||
"hide_details": "Ocultar detalles",
|
||||
|
@ -349,9 +350,9 @@
|
|||
"incoming": "Entrante",
|
||||
"incorrect_seed": "El texto ingresado no es válido.",
|
||||
"inputs": "Entradas",
|
||||
"insufficient_lamport_for_tx": "No tiene suficiente SOL para cubrir la transacción y su tarifa de transacción. Por favor, agregue más SOL a su billetera o reduzca la cantidad de sol que está enviando.",
|
||||
"insufficient_lamports": "No tiene suficiente SOL para cubrir la transacción y su tarifa de transacción. Necesita al menos ${solValueNeeded} sol. Por favor, agregue más sol a su billetera o reduzca la cantidad de sol que está enviando",
|
||||
"insufficientFundsForRentError": "No tiene suficiente SOL para cubrir la tarifa de transacción y alquilar para la cuenta. Por favor, agregue más sol a su billetera o reduzca la cantidad de sol que está enviando",
|
||||
"insufficient_lamport_for_tx": "No tienes suficiente SOL para cubrir la transacción y su tarifa de transacción. Por favor, agrega más SOL a su billetera o reduce la cantidad de sol que está enviando.",
|
||||
"insufficient_lamports": "No tienes suficiente SOL para cubrir la transacción y su tarifa de transacción. Necesita al menos ${solValueNeeded} sol. Por favor, agrega más sol a su billetera o reduzca la cantidad de sol que está enviando",
|
||||
"insufficientFundsForRentError": "No tienes suficiente SOL para cubrir la tarifa de transacción y alquilar para la cuenta. Por favor, agrega más sol a su billetera o reduce la cantidad de sol que está enviando",
|
||||
"introducing_cake_pay": "¡Presentamos Cake Pay!",
|
||||
"invalid_input": "Entrada inválida",
|
||||
"invalid_password": "Contraseña invalida",
|
||||
|
@ -359,19 +360,22 @@
|
|||
"is_percentage": "es",
|
||||
"last_30_days": "Últimos 30 días",
|
||||
"learn_more": "Aprende más",
|
||||
"ledger_connection_error": "No se pudo conectar con su libro mayor. Inténtalo de nuevo.",
|
||||
"ledger_error_device_locked": "El libro mayor está bloqueado",
|
||||
"ledger_connection_error": "No se pudo conectar con ledger. Inténtalo de nuevo.",
|
||||
"ledger_error_device_locked": "Ledger está bloqueado",
|
||||
"ledger_error_tx_rejected_by_user": "Transacción rechazada en el dispositivo",
|
||||
"ledger_error_wrong_app": "Por favor, asegúrese de abrir la aplicación correcta en su libro mayor.",
|
||||
"ledger_please_enable_bluetooth": "Habilite Bluetooth para detectar su libro mayor",
|
||||
"light_theme": "Ligera",
|
||||
"ledger_error_wrong_app": "Por favor, asegúrate de abrir la aplicación correcta en su libro mayor.",
|
||||
"ledger_please_enable_bluetooth": "Habilita tu Bluetooth para detectar tu ledger",
|
||||
"light_theme": "Ligero",
|
||||
"litecoin_enable_mweb_sync": "Habilitar el escaneo mweb",
|
||||
"litecoin_mweb": "Mweb",
|
||||
"litecoin_mweb_always_scan": "Establecer mweb siempre escaneo",
|
||||
"litecoin_mweb_description": "Mweb es un nuevo protocolo que trae transacciones más rápidas, más baratas y más privadas a Litecoin",
|
||||
"litecoin_mweb_dismiss": "Despedir",
|
||||
"litecoin_mweb_display_card": "Mostrar tarjeta MWEB",
|
||||
"litecoin_mweb_enable": "Habilitar mweb",
|
||||
"litecoin_mweb_enable_later": "Puede elegir habilitar MWEB nuevamente en la configuración de visualización.",
|
||||
"litecoin_mweb_logs": "Registros de mweb",
|
||||
"litecoin_mweb_node": "Nodo mweb",
|
||||
"litecoin_mweb_pegin": "Convertir",
|
||||
"litecoin_mweb_pegout": "Recuperar",
|
||||
"litecoin_mweb_scanning": "Escaneo mweb",
|
||||
|
@ -401,9 +405,9 @@
|
|||
"min_value": "Min: ${value} ${currency}",
|
||||
"minutes_to_pin_code": "${minute} minutos",
|
||||
"mm": "mm",
|
||||
"modify_2fa": "Modificar torta 2FA",
|
||||
"monero_com": "Monero.com by Cake Wallet",
|
||||
"monero_com_wallet_text": "Awesome wallet for Monero",
|
||||
"modify_2fa": "Modificar 2FA",
|
||||
"monero_com": "Monero.com por Cake Wallet",
|
||||
"monero_com_wallet_text": "Increíble billetera para Monero",
|
||||
"monero_dark_theme": "Tema oscuro de Monero",
|
||||
"monero_light_theme": "Tema ligero de Monero",
|
||||
"moonpay_alert_text": "El valor de la cantidad debe ser mayor o igual a ${minAmount} ${fiatCurrency}",
|
||||
|
@ -412,12 +416,12 @@
|
|||
"mweb_unconfirmed": "Mweb no confirmado",
|
||||
"name": "Nombre",
|
||||
"nano_current_rep": "Representante actual",
|
||||
"nano_gpt_thanks_message": "¡Gracias por usar nanogpt! ¡Recuerde regresar al navegador después de que se complete su transacción!",
|
||||
"nano_gpt_thanks_message": "¡Gracias por usar nanogpt! ¡Recuerda regresar al navegador después de que se complete su transacción!",
|
||||
"nano_pick_new_rep": "Elija un nuevo representante",
|
||||
"nanogpt_subtitle": "Todos los modelos más nuevos (GPT-4, Claude). \\nSin suscripción, pague con cripto.",
|
||||
"narrow": "Angosto",
|
||||
"new_first_wallet_text": "Mantenga fácilmente su criptomoneda segura",
|
||||
"new_node_testing": "Prueba de nuevos nodos",
|
||||
"new_first_wallet_text": "Mantén fácilmente tu criptomoneda segura",
|
||||
"new_node_testing": "Prueba nuevos nodos",
|
||||
"new_subaddress_create": "Crear",
|
||||
"new_subaddress_label_name": "Nombre de etiqueta",
|
||||
"new_subaddress_title": "Nueva direccion",
|
||||
|
@ -426,10 +430,10 @@
|
|||
"newConnection": "Nueva conexión",
|
||||
"no_cards_found": "No se encuentran cartas",
|
||||
"no_id_needed": "¡No se necesita identificación!",
|
||||
"no_id_required": "No se requiere identificación. Recargue y gaste en cualquier lugar",
|
||||
"no_relay_on_domain": "No hay una retransmisión para el dominio del usuario o la retransmisión no está disponible. Elija un relé para usar.",
|
||||
"no_relays": "Sin relevos",
|
||||
"no_relays_message": "Encontramos un registro Nostr NIP-05 para este usuario, pero no contiene ningún relé. Indique al destinatario que agregue retransmisiones a su registro Nostr.",
|
||||
"no_id_required": "No se requiere identificación. Recarga y gaste en cualquier lugar",
|
||||
"no_relay_on_domain": "No hay una retransmisión para el dominio del usuario o la retransmisión no está disponible. Elige un relay para usar.",
|
||||
"no_relays": "Sin relays",
|
||||
"no_relays_message": "Encontramos un registro Nostr NIP-05 para este usuario, pero no contiene ningún relay. Indica al destinatario que agregue retransmisiones a su registro Nostr.",
|
||||
"node_address": "Dirección de nodo",
|
||||
"node_connection_failed": "La conexión falló",
|
||||
"node_connection_successful": "La conexión fue exitosa",
|
||||
|
@ -449,15 +453,15 @@
|
|||
"offline": "fuera de línea",
|
||||
"ok": "OK",
|
||||
"old_fee": "Tarifa antigua",
|
||||
"onion_link": "Enlace de cebolla",
|
||||
"onion_link": "Enlace de cebolla (Tor)",
|
||||
"online": "En línea",
|
||||
"onramper_option_description": "Compre rápidamente cripto con muchos métodos de pago. Disponible en la mayoría de los países. Los diferenciales y las tarifas varían.",
|
||||
"onramper_option_description": "Compra rápidamente cripto con muchos métodos de pago. Disponible en la mayoría de los países. Los diferenciales y las tarifas varían.",
|
||||
"open_gift_card": "Abrir tarjeta de regalo",
|
||||
"optional_description": "Descripción opcional",
|
||||
"optional_email_hint": "Correo electrónico de notificación del beneficiario opcional",
|
||||
"optional_name": "Nombre del destinatario opcional",
|
||||
"optionally_order_card": "Opcionalmente pide una tarjeta física.",
|
||||
"orbot_running_alert": "Asegúrese de que Orbot se esté ejecutando antes de conectarse a este nodo.",
|
||||
"orbot_running_alert": "Asegúrate de que Orbot se esté ejecutando antes de conectarte a este nodo.",
|
||||
"order_by": "Ordenar",
|
||||
"order_id": "Identificación del pedido",
|
||||
"order_physical_card": "Pedir tarjeta física",
|
||||
|
@ -466,7 +470,7 @@
|
|||
"outdated_electrum_wallet_receive_warning": "Si esta billetera tiene una semilla de 12 palabras y se creó en Cake, NO deposite Bitcoin en esta billetera. Cualquier BTC transferido a esta billetera se puede perder. Cree una nueva billetera de 24 palabras (toque el menú en la parte superior derecha, seleccione Monederos, elija Crear nueva billetera, luego seleccione Bitcoin) e INMEDIATAMENTE mueva su BTC allí. Las nuevas carteras BTC (24 palabras) de Cake son seguras",
|
||||
"outgoing": "Saliente",
|
||||
"outputs": "Salidas",
|
||||
"overwrite_amount": "Overwrite amount",
|
||||
"overwrite_amount": "Sobreescribir monto",
|
||||
"pairingInvalidEvent": "Evento de emparejamiento no válido",
|
||||
"passphrase": "Passfrase (opcional)",
|
||||
"passphrases_doesnt_match": "Las frases de contrato no coinciden, intente nuevamente",
|
||||
|
@ -482,18 +486,18 @@
|
|||
"pin_is_incorrect": "PIN es incorrecto",
|
||||
"pin_number": "Número PIN",
|
||||
"placeholder_contacts": "Tus contactos se mostrarán aquí",
|
||||
"placeholder_transactions": "Sus transacciones se mostrarán aquí",
|
||||
"please_fill_totp": "Complete el código de 8 dígitos presente en su otro dispositivo",
|
||||
"please_make_selection": "Seleccione a continuación para crear o recuperar su billetera.",
|
||||
"please_reference_document": "Consulte los documentos a continuación para obtener más información.",
|
||||
"please_select": "Por favor seleccione:",
|
||||
"please_select_backup_file": "Seleccione el archivo de respaldo e ingrese la contraseña de respaldo.",
|
||||
"placeholder_transactions": "Tus transacciones se mostrarán aquí",
|
||||
"please_fill_totp": "Completa el código de 8 dígitos presente en su otro dispositivo",
|
||||
"please_make_selection": "Selecciona a continuación para crear o recuperar su billetera.",
|
||||
"please_reference_document": "Consulta los documentos a continuación para obtener más información.",
|
||||
"please_select": "Por favor selecciona:",
|
||||
"please_select_backup_file": "Selecciona el archivo de respaldo e ingrese la contraseña de respaldo.",
|
||||
"please_try_to_connect_to_another_node": "Intenta conectarte a otro nodo",
|
||||
"please_wait": "Espere por favor",
|
||||
"please_wait": "Espera por favor",
|
||||
"polygonscan_history": "Historial de PolygonScan",
|
||||
"powered_by": "Energizado por ${title}",
|
||||
"powered_by": "Posible gracias a ${title}",
|
||||
"pre_seed_button_text": "Entiendo. Muéstrame mi semilla",
|
||||
"pre_seed_description": "En la página siguiente verá una serie de ${words} palabras. Esta es su semilla única y privada y es la ÚNICA forma de recuperar su billetera en caso de pérdida o mal funcionamiento. Es SU responsabilidad escribirlo y guardarlo en un lugar seguro fuera de la aplicación Cake Wallet.",
|
||||
"pre_seed_description": "En la página siguiente verás una serie de ${words} palabras. Esta es su semilla única y privada y es la ÚNICA forma de recuperar tu billetera en caso de pérdida o mal funcionamiento. Es TU responsabilidad escribirla y guardarla en un lugar seguro fuera de la aplicación Cake Wallet.",
|
||||
"pre_seed_title": "IMPORTANTE",
|
||||
"prepaid_cards": "Tajetas prepagadas",
|
||||
"prevent_screenshots": "Evitar capturas de pantalla y grabación de pantalla",
|
||||
|
@ -502,17 +506,17 @@
|
|||
"privacy_settings": "Configuración de privacidad",
|
||||
"private_key": "Clave privada",
|
||||
"proceed_after_one_minute": "Si la pantalla no continúa después de 1 minuto, revisa tu correo electrónico.",
|
||||
"proceed_on_device": "Continúe con su dispositivo",
|
||||
"proceed_on_device_description": "Siga las instrucciones solicitadas en su billetera de hardware",
|
||||
"proceed_on_device": "Continúa con tu dispositivo",
|
||||
"proceed_on_device_description": "Sigue las instrucciones solicitadas en su billetera de hardware",
|
||||
"profile": "Perfil",
|
||||
"provider_error": "${provider} error",
|
||||
"public_key": "Clave pública",
|
||||
"purchase_gift_card": "Comprar tarjeta de regalo",
|
||||
"purple_dark_theme": "Tema morado oscuro",
|
||||
"qr_fullscreen": "Toque para abrir el código QR en pantalla completa",
|
||||
"qr_payment_amount": "This QR code contains a payment amount. Do you want to overwrite the current value?",
|
||||
"qr_payment_amount": "Este código QR contiene un monto de pago. ¿Quieres sobreescribirlo?",
|
||||
"quantity": "Cantidad",
|
||||
"question_to_disable_2fa": "¿Está seguro de que desea deshabilitar Cake 2FA? Ya no se necesitará un código 2FA para acceder a la billetera y a ciertas funciones.",
|
||||
"question_to_disable_2fa": "¿Estás seguro de que desea deshabilitar Cake 2FA? Ya no se necesitará un código 2FA para acceder a la billetera y a ciertas funciones.",
|
||||
"receivable_balance": "Saldo de cuentas por cobrar",
|
||||
"receive": "Recibir",
|
||||
"receive_amount": "Cantidad",
|
||||
|
@ -529,12 +533,12 @@
|
|||
"remaining": "restante",
|
||||
"remove": "Retirar",
|
||||
"remove_node": "Eliminar nodo",
|
||||
"remove_node_message": "¿Está seguro de que desea eliminar el nodo seleccionado?",
|
||||
"rename": "Rebautizar",
|
||||
"remove_node_message": "¿Estás seguro de que desea eliminar el nodo seleccionado?",
|
||||
"rename": "Renombrar",
|
||||
"rep_warning": "Advertencia representativa",
|
||||
"rep_warning_sub": "Su representante no parece estar en buena posición. Toque aquí para seleccionar uno nuevo",
|
||||
"repeat_wallet_password": "Repita la contraseña de billetera",
|
||||
"repeated_password_is_incorrect": "La contraseña repetida es incorrecta. Repita la contraseña de la billetera nuevamente.",
|
||||
"rep_warning_sub": "Tu representante no parece estar en buena posición. Toca aquí para seleccionar uno nuevo",
|
||||
"repeat_wallet_password": "Repite la contraseña de billetera",
|
||||
"repeated_password_is_incorrect": "La contraseña repetida es incorrecta. Repite la contraseña de la billetera nuevamente.",
|
||||
"require_for_adding_contacts": "Requerido para agregar contactos",
|
||||
"require_for_all_security_and_backup_settings": "Requerido para todas las configuraciones de seguridad y copia de seguridad",
|
||||
"require_for_assessing_wallet": "Requerido para acceder a la billetera",
|
||||
|
@ -566,17 +570,17 @@
|
|||
"restore_recover": "Recuperar",
|
||||
"restore_restore_wallet": "Recuperar Cartera",
|
||||
"restore_seed_keys_restore": "Restauración de semillas / llaves",
|
||||
"restore_spend_key_private": "Spend clave (privado)",
|
||||
"restore_spend_key_private": "Llave de gasto (privada)",
|
||||
"restore_title_from_backup": "Restaurar desde un archivo de respaldo",
|
||||
"restore_title_from_hardware_wallet": "Restaurar desde la billetera de hardware",
|
||||
"restore_title_from_keys": "De las claves",
|
||||
"restore_title_from_seed": "De la semilla",
|
||||
"restore_title_from_seed_keys": "Restaurar desde semilla/claves",
|
||||
"restore_view_key_private": "View clave (privado)",
|
||||
"restore_title_from_hardware_wallet": "Restaurar desde una cartera fría",
|
||||
"restore_title_from_keys": "Usando las claves",
|
||||
"restore_title_from_seed": "Usando la semilla",
|
||||
"restore_title_from_seed_keys": "Restaurar usando semilla/claves",
|
||||
"restore_view_key_private": "Llave de vista (privado)",
|
||||
"restore_wallet": "Restaurar billetera",
|
||||
"restore_wallet_name": "Nombre de la billetera",
|
||||
"restore_wallet_restore_description": "Restaurar billetera",
|
||||
"robinhood_option_description": "Compre y transfiera instantáneamente utilizando su tarjeta de débito, cuenta bancaria o saldo de Robinhood. Solo EE. UU.",
|
||||
"robinhood_option_description": "Compra y transfiere instantáneamente utilizando su tarjeta de débito, cuenta bancaria o saldo de Robinhood. Solo EE. UU.",
|
||||
"router_no_route": "No hay ruta definida para ${name}",
|
||||
"save": "Salvar",
|
||||
"save_backup_password": "Asegúrese de haber guardado su contraseña de respaldo. No podrá importar sus archivos de respaldo sin él.",
|
||||
|
@ -585,7 +589,7 @@
|
|||
"saved_the_trade_id": "He salvado comercial ID",
|
||||
"scan_one_block": "Escanear un bloque",
|
||||
"scan_qr_code": "Escanear código QR",
|
||||
"scan_qr_code_to_get_address": "Escanee el código QR para obtener la dirección",
|
||||
"scan_qr_code_to_get_address": "Escanea el código QR para obtener la dirección",
|
||||
"scan_qr_on_device": "Escanea este código QR en otro dispositivo",
|
||||
"search": "Búsqueda",
|
||||
"search_add_token": "Buscar/Agregar token",
|
||||
|
@ -612,25 +616,25 @@
|
|||
"seed_language_german": "Alemán",
|
||||
"seed_language_italian": "Italiana/Italiano",
|
||||
"seed_language_japanese": "Japonés",
|
||||
"seed_language_korean": "coreano",
|
||||
"seed_language_korean": "Coreano",
|
||||
"seed_language_next": "Próximo",
|
||||
"seed_language_portuguese": "Portugués",
|
||||
"seed_language_russian": "Ruso",
|
||||
"seed_language_spanish": "Español",
|
||||
"seed_phrase_length": "Longitud de la frase inicial",
|
||||
"seed_reminder": "Anótelos en caso de que pierda o borre su teléfono",
|
||||
"seed_reminder": "Anótalos en caso de que pierdas o borres tu aplicación",
|
||||
"seed_share": "Compartir semillas",
|
||||
"seed_title": "Semilla",
|
||||
"seedtype": "Type de semillas",
|
||||
"seedtype_alert_content": "Compartir semillas con otras billeteras solo es posible con Bip39 Seed Type.",
|
||||
"seedtype_alert_title": "Alerta de type de semillas",
|
||||
"seedtype_legacy": "Legado (25 palabras)",
|
||||
"seedtype_polyseed": "Polieta (16 palabras)",
|
||||
"seedtype": "Tipos de semillas",
|
||||
"seedtype_alert_content": "Compartir semillas con otras billeteras solo es posible con semillas bip39 - un tipo específico de semilla.",
|
||||
"seedtype_alert_title": "Alerta de tipo de semillas",
|
||||
"seedtype_legacy": "Semilla clásica-legacy (25 palabras)",
|
||||
"seedtype_polyseed": "Poli-semilla (16 palabras)",
|
||||
"seedtype_wownero": "Wownero (14 palabras)",
|
||||
"select_backup_file": "Seleccionar archivo de respaldo",
|
||||
"select_buy_provider_notice": "Seleccione un proveedor de compra arriba. Puede omitir esta pantalla configurando su proveedor de compra predeterminado en la configuración de la aplicación.",
|
||||
"select_destination": "Seleccione el destino del archivo de copia de seguridad.",
|
||||
"select_sell_provider_notice": "Seleccione un proveedor de venta arriba. Puede omitir esta pantalla configurando su proveedor de venta predeterminado en la configuración de la aplicación.",
|
||||
"select_buy_provider_notice": "Selecciona un proveedor de compra arriba. Puede omitir esta pantalla configurando su proveedor de compra predeterminado en la configuración de la aplicación.",
|
||||
"select_destination": "Selecciona el destino del archivo de copia de seguridad.",
|
||||
"select_sell_provider_notice": "Selecciona un proveedor de venta arriba. Puede omitir esta pantalla configurando su proveedor de venta predeterminado en la configuración de la aplicación.",
|
||||
"sell": "Vender",
|
||||
"sell_alert_content": "Actualmente solo admitimos la venta de Bitcoin, Ethereum y Litecoin. Cree o cambie a su billetera Bitcoin, Ethereum o Litecoin.",
|
||||
"sell_monero_com_alert_content": "Aún no se admite la venta de Monero",
|
||||
|
@ -676,16 +680,16 @@
|
|||
"settings_only_transactions": "Solo transacciones",
|
||||
"settings_personal": "Personal",
|
||||
"settings_save_recipient_address": "Guardar dirección del destinatario",
|
||||
"settings_support": "Apoyo",
|
||||
"settings_support": "Ayuda",
|
||||
"settings_terms_and_conditions": "Términos y Condiciones",
|
||||
"settings_title": "Configuraciones",
|
||||
"settings_trades": "Comercia",
|
||||
"settings_transactions": "Transacciones",
|
||||
"settings_wallets": "Carteras",
|
||||
"setup_2fa": "Configurar pastel 2FA",
|
||||
"setup_2fa_text": "Cake 2FA funciona utilizando TOTP como segundo factor de autenticación.\n\nEl TOTP de Cake 2FA requiere SHA-512 y soporte de 8 dígitos; esto proporciona una mayor seguridad. Puede encontrar más información y aplicaciones compatibles en la guía.",
|
||||
"setup_2fa": "Configurar 2FA",
|
||||
"setup_2fa_text": "Cake 2FA funciona utilizando TOTP como segundo factor de autenticación.\n\nEl TOTP de Cake 2FA requiere SHA-512 y soporte de 8 dígitos; esto proporciona una mayor seguridad. Puedes encontrar más información y aplicaciones compatibles en la guía.",
|
||||
"setup_pin": "PIN de configuración",
|
||||
"setup_successful": "Su PIN se ha configurado correctamente!",
|
||||
"setup_successful": "Tu PIN se ha configurado correctamente!",
|
||||
"setup_totp_recommended": "Configurar TOTP",
|
||||
"setup_warning_2fa_text": "Deberá restaurar su billetera a partir de la semilla mnemotécnica.\n\nEl soporte de Cake no podrá ayudarlo si pierde el acceso a su 2FA o a sus semillas mnemotécnicas.\nCake 2FA es una segunda autenticación para ciertas acciones en la billetera. Antes de usar Cake 2FA, recomendamos leer la guía.NO es tan seguro como el almacenamiento en frío.\n\nSi pierde el acceso a su aplicación 2FA o a sus claves TOTP, perderá el acceso a esta billetera. ",
|
||||
"setup_your_debit_card": "Configura tu tarjeta de débito",
|
||||
|
@ -704,23 +708,23 @@
|
|||
"signature": "Firma",
|
||||
"signature_invalid_error": "La firma no es válida para el mensaje dado",
|
||||
"signTransaction": "Firmar transacción",
|
||||
"signup_for_card_accept_terms": "Regístrese para obtener la tarjeta y acepte los términos.",
|
||||
"signup_for_card_accept_terms": "Regístrate para obtener la tarjeta y acepte los términos.",
|
||||
"silent_payment": "Pago silencioso",
|
||||
"silent_payments": "Pagos silenciosos",
|
||||
"silent_payments_always_scan": "Establecer pagos silenciosos siempre escaneando",
|
||||
"silent_payments_disclaimer": "Las nuevas direcciones no son nuevas identidades. Es una reutilización de una identidad existente con una etiqueta diferente.",
|
||||
"silent_payments_display_card": "Mostrar tarjeta de pagos silenciosos",
|
||||
"silent_payments_scan_from_date": "Escanear desde la fecha",
|
||||
"silent_payments_scan_from_date_or_blockheight": "Ingrese la altura del bloque que desea comenzar a escanear para pagos silenciosos entrantes, o use la fecha en su lugar. Puede elegir si la billetera continúa escaneando cada bloque, o verifica solo la altura especificada.",
|
||||
"silent_payments_scan_from_height": "Escanear desde la altura del bloque",
|
||||
"silent_payments_scanned_tip": "Escaneado hasta la punta! (${tip})",
|
||||
"silent_payments_scan_from_date_or_blockheight": "Ingresa la altura de bloque que desea comenzar a escanear para pagos silenciosos entrantes, o usa la fecha en su lugar. Puedes elegir si la billetera continúa escaneando cada bloque, o verifica solo la altura especificada.",
|
||||
"silent_payments_scan_from_height": "Escanear desde la altura de bloque específico",
|
||||
"silent_payments_scanned_tip": "Escaneado hasta la altura actual! (${tip})",
|
||||
"silent_payments_scanning": "Escaneo de pagos silenciosos",
|
||||
"silent_payments_settings": "Configuración de pagos silenciosos",
|
||||
"single_seed_wallets_group": "Billeteras de semillas individuales",
|
||||
"slidable": "deslizable",
|
||||
"sort_by": "Ordenar por",
|
||||
"spend_key_private": "Spend clave (privado)",
|
||||
"spend_key_public": "Spend clave (público)",
|
||||
"spend_key_private": "Llave de gasto (privada)",
|
||||
"spend_key_public": "Llave de gasto (pública)",
|
||||
"status": "Estado: ",
|
||||
"string_default": "Por defecto",
|
||||
"subaddress_title": "Lista de subdirecciones",
|
||||
|
@ -729,14 +733,14 @@
|
|||
"successful": "Exitoso",
|
||||
"support_description_guides": "Documentación y apoyo para problemas comunes",
|
||||
"support_description_live_chat": "¡GRATIS y RÁPIDO! Los representantes de apoyo capacitado están disponibles para ayudar",
|
||||
"support_description_other_links": "Únase a nuestras comunidades o comuníquese con nosotros nuestros socios a través de otros métodos",
|
||||
"support_description_other_links": "Únete a nuestras comunidades o comunícate con nosotros nuestros socios a través de otros métodos",
|
||||
"support_title_guides": "Guías de billetera para pastel",
|
||||
"support_title_live_chat": "Soporte vital",
|
||||
"support_title_live_chat": "Soporte en tiempo real",
|
||||
"support_title_other_links": "Otros enlaces de soporte",
|
||||
"sweeping_wallet": "Billetera de barrido",
|
||||
"sweeping_wallet": "Barrer billetera (gastar todos los fondos disponibles)",
|
||||
"sweeping_wallet_alert": "Esto no debería llevar mucho tiempo. NO DEJES ESTA PANTALLA O SE PUEDEN PERDER LOS FONDOS BARRIDOS",
|
||||
"switchToETHWallet": "Cambie a una billetera Ethereum e inténtelo nuevamente.",
|
||||
"switchToEVMCompatibleWallet": "Cambie a una billetera compatible con EVM e inténtelo nuevamente (Ethereum, Polygon)",
|
||||
"switchToETHWallet": "Cambia a una billetera Ethereum e inténtelo nuevamente.",
|
||||
"switchToEVMCompatibleWallet": "Cambia a una billetera compatible con EVM e inténtelo nuevamente (Ethereum, Polygon)",
|
||||
"symbol": "Símbolo",
|
||||
"sync_all_wallets": "Sincronizar todas las billeteras",
|
||||
"sync_status_attempting_scan": "Intento de escaneo",
|
||||
|
@ -759,7 +763,7 @@
|
|||
"third_intro_content": "Los Yats también viven fuera de Cake Wallet. Cualquier dirección de billetera en la tierra se puede reemplazar con un Yat!",
|
||||
"third_intro_title": "Yat juega muy bien con otras",
|
||||
"thorchain_contract_address_not_supported": "Thorchain no admite enviar a una dirección de contrato",
|
||||
"thorchain_taproot_address_not_supported": "El proveedor de Thorchain no admite las direcciones de Taproot. Cambie la dirección o seleccione un proveedor diferente.",
|
||||
"thorchain_taproot_address_not_supported": "El proveedor de Thorchain no admite las direcciones de Taproot. Cambia la dirección o selecciona un proveedor diferente.",
|
||||
"time": "${minutes}m ${seconds}s",
|
||||
"tip": "Consejo:",
|
||||
"today": "Hoy",
|
||||
|
@ -772,8 +776,8 @@
|
|||
"tor_only": "solo Tor",
|
||||
"total": "Total",
|
||||
"total_saving": "Ahorro Total",
|
||||
"totp_2fa_failure": "Código incorrecto. Intente con un código diferente o genere una nueva clave secreta. Use una aplicación 2FA compatible que admita códigos de 8 dígitos y SHA512.",
|
||||
"totp_2fa_success": "¡Éxito! Cake 2FA habilitado para esta billetera. Recuerde guardar su semilla mnemotécnica en caso de que pierda el acceso a la billetera.",
|
||||
"totp_2fa_failure": "Código incorrecto. Intente con un código diferente o genere una nueva clave secreta. Usa una aplicación 2FA compatible que admita códigos de 8 dígitos y SHA512.",
|
||||
"totp_2fa_success": "¡Éxito! Cake 2FA habilitado para esta billetera. Recuerda guardar tu semilla mnemotécnica en caso de que pierdas el acceso a la billetera.",
|
||||
"totp_auth_url": "URL de autenticación TOTP",
|
||||
"totp_code": "Código TOTP",
|
||||
"totp_secret_code": "Código secreto TOTP",
|
||||
|
@ -831,16 +835,16 @@
|
|||
"trongrid_history": "Historia trongrid",
|
||||
"trusted": "de confianza",
|
||||
"tx_commit_exception_no_dust_on_change": "La transacción se rechaza con esta cantidad. Con estas monedas puede enviar ${min} sin cambios o ${max} que devuelve el cambio.",
|
||||
"tx_commit_failed": "La confirmación de transacción falló. Póngase en contacto con el soporte.",
|
||||
"tx_commit_failed_no_peers": "La transacción no se transmitió, intente nuevamente en un segundo más o menos",
|
||||
"tx_commit_failed": "La confirmación de transacción falló. Ponte en contacto con el soporte.",
|
||||
"tx_commit_failed_no_peers": "La transacción no se transmitió, intenta nuevamente en un segundo más o menos",
|
||||
"tx_invalid_input": "Está utilizando el tipo de entrada incorrecto para este tipo de pago",
|
||||
"tx_no_dust_exception": "La transacción se rechaza enviando una cantidad demasiado pequeña. Intente aumentar la cantidad.",
|
||||
"tx_not_enough_inputs_exception": "No hay suficientes entradas disponibles. Seleccione más bajo control de monedas",
|
||||
"tx_not_enough_inputs_exception": "No hay suficientes entradas disponibles. Selecciona más bajo control de monedas",
|
||||
"tx_rejected_bip68_final": "La transacción tiene entradas no confirmadas y no ha podido reemplazar por tarifa.",
|
||||
"tx_rejected_dust_change": "Transacción rechazada por reglas de red, bajo cambio de cambio (polvo). Intente enviar todo o reducir la cantidad.",
|
||||
"tx_rejected_dust_output": "Transacción rechazada por reglas de red, baja cantidad de salida (polvo). Aumente la cantidad.",
|
||||
"tx_rejected_dust_change": "Transacción rechazada por reglas de red, bajo cambio de cambio (polvo). Intenta enviar todo o reducir la cantidad.",
|
||||
"tx_rejected_dust_output": "Transacción rechazada por reglas de red, baja cantidad de salida (polvo). Aumenta la cantidad.",
|
||||
"tx_rejected_dust_output_send_all": "Transacción rechazada por reglas de red, baja cantidad de salida (polvo). Verifique el saldo de monedas seleccionadas bajo control de monedas.",
|
||||
"tx_rejected_vout_negative": "No es suficiente saldo para pagar las tarifas de esta transacción. Verifique el saldo de monedas bajo control de monedas.",
|
||||
"tx_rejected_vout_negative": "No es suficiente saldo para pagar las tarifas de esta transacción. Verifica el saldo de monedas bajo control de monedas.",
|
||||
"tx_wrong_balance_exception": "No tiene suficiente ${currency} para enviar esta cantidad.",
|
||||
"tx_wrong_balance_with_amount_exception": "No tiene suficiente ${currency} para enviar la cantidad total de ${amount}",
|
||||
"tx_zero_fee_exception": "No se puede enviar transacciones con 0 tarifa. Intente aumentar la tasa o verificar su conexión para las últimas estimaciones.",
|
||||
|
@ -849,7 +853,7 @@
|
|||
"unconfirmed": "Saldo no confirmado",
|
||||
"understand": "Entiendo",
|
||||
"unlock": "desbloquear",
|
||||
"unmatched_currencies": "La moneda de su billetera actual no coincide con la del QR escaneado",
|
||||
"unmatched_currencies": "La moneda de tu billetera actual no coincide con la del QR escaneado",
|
||||
"unspent_change": "Cambiar",
|
||||
"unspent_coins_details_title": "Detalles de monedas no gastadas",
|
||||
"unspent_coins_title": "Monedas no gastadas",
|
||||
|
@ -858,11 +862,11 @@
|
|||
"upto": "hasta ${value}",
|
||||
"usb": "USB",
|
||||
"use": "Utilizar a ",
|
||||
"use_card_info_three": "Utilice la tarjeta digital en línea o con métodos de pago sin contacto.",
|
||||
"use_card_info_three": "Utiliza la tarjeta digital en línea o con métodos de pago sin contacto.",
|
||||
"use_card_info_two": "Los fondos se convierten a USD cuando se mantienen en la cuenta prepaga, no en monedas digitales.",
|
||||
"use_ssl": "Utilice SSL",
|
||||
"use_ssl": "Utiliza SSL",
|
||||
"use_suggested": "Usar sugerido",
|
||||
"use_testnet": "Use TestNet",
|
||||
"use_testnet": "Usar TestNet",
|
||||
"value": "Valor",
|
||||
"value_type": "Tipo de valor",
|
||||
"variable_pair_not_supported": "Este par de variables no es compatible con los intercambios seleccionados",
|
||||
|
@ -871,17 +875,17 @@
|
|||
"verify_with_2fa": "Verificar con Cake 2FA",
|
||||
"version": "Versión ${currentVersion}",
|
||||
"view_all": "Ver todo",
|
||||
"view_in_block_explorer": "View in Block Explorer",
|
||||
"view_key_private": "View clave (privado)",
|
||||
"view_key_public": "View clave (público)",
|
||||
"view_transaction_on": "View Transaction on ",
|
||||
"view_in_block_explorer": "Ver en explorador de bloques",
|
||||
"view_key_private": "Llave de vista (privada)",
|
||||
"view_key_public": "Llave de vista (pública)",
|
||||
"view_transaction_on": "Ver transacción en ",
|
||||
"voting_weight": "Peso de votación",
|
||||
"waitFewSecondForTxUpdate": "Espere unos segundos para que la transacción se refleje en el historial de transacciones.",
|
||||
"waitFewSecondForTxUpdate": "Espera unos segundos para que la transacción se refleje en el historial de transacciones.",
|
||||
"wallet_group": "Grupo de billetera",
|
||||
"wallet_group_description_four": "para crear una billetera con una semilla completamente nueva.",
|
||||
"wallet_group_description_four": "Para crear una billetera con una semilla completamente nueva.",
|
||||
"wallet_group_description_one": "En la billetera de pastel, puedes crear un",
|
||||
"wallet_group_description_three": "Para ver las billeteras disponibles y/o la pantalla de grupos de billeteras. O elegir",
|
||||
"wallet_group_description_two": "seleccionando una billetera existente para compartir una semilla con. Cada grupo de billetera puede contener una sola billetera de cada tipo de moneda. \n\n puede seleccionar",
|
||||
"wallet_group_description_two": "Seleccionando una billetera existente para compartir una semilla con. Cada grupo de billetera puede contener una sola billetera de cada tipo de moneda. \n\n puedes seleccionar",
|
||||
"wallet_group_empty_state_text_one": "Parece que no tienes ningún grupo de billetera compatible !\n\n toque",
|
||||
"wallet_group_empty_state_text_two": "a continuación para hacer uno nuevo.",
|
||||
"wallet_keys": "Billetera semilla/claves",
|
||||
|
@ -889,7 +893,7 @@
|
|||
"wallet_list_edit_group_name": "Editar nombre de grupo",
|
||||
"wallet_list_edit_wallet": "Editar billetera",
|
||||
"wallet_list_failed_to_load": "No se pudo cargar ${wallet_name} la billetera. ${error}",
|
||||
"wallet_list_failed_to_remove": "Error al elimina ${wallet_name} billetera. ${error}",
|
||||
"wallet_list_failed_to_remove": "Error al eliminar ${wallet_name} billetera. ${error}",
|
||||
"wallet_list_load_wallet": "Billetera de carga",
|
||||
"wallet_list_loading_wallet": "Billetera ${wallet_name} de carga",
|
||||
"wallet_list_removing_wallet": "Retirar ${wallet_name} billetera",
|
||||
|
@ -898,8 +902,8 @@
|
|||
"wallet_list_wallet_name": "Nombre de la billetera",
|
||||
"wallet_menu": "Menú de billetera",
|
||||
"wallet_name": "Nombre de la billetera",
|
||||
"wallet_name_exists": "Wallet con ese nombre ya ha existido",
|
||||
"wallet_password_is_empty": "La contraseña de billetera está vacía. La contraseña de la billetera no debe estar vacía",
|
||||
"wallet_name_exists": "Cartera con ese nombre ya existe, escoge otro",
|
||||
"wallet_password_is_empty": "La contraseña de billetera está vacía. La contraseña de la billetera no puede estar vacía",
|
||||
"wallet_recovery_height": "Altura de recuperación",
|
||||
"wallet_restoration_store_incorrect_seed_length": "Longitud de semilla incorrecta",
|
||||
"wallet_seed": "Semilla de billetera",
|
||||
|
@ -913,30 +917,30 @@
|
|||
"what_is_silent_payments": "¿Qué son los pagos silenciosos?",
|
||||
"widgets_address": "Dirección",
|
||||
"widgets_or": "o",
|
||||
"widgets_restore_from_blockheight": "Restaurar desde blockheight",
|
||||
"widgets_restore_from_blockheight": "Restaurar desde altura de bloque",
|
||||
"widgets_restore_from_date": "Restaurar desde fecha",
|
||||
"widgets_seed": "Semilla",
|
||||
"wouoldLikeToConnect": "quisiera conectar",
|
||||
"write_down_backup_password": "Escriba su contraseña de respaldo, que se utiliza para la importación de sus archivos de respaldo.",
|
||||
"xlm_extra_info": "No olvide especificar el ID de nota al enviar la transacción XLM para el intercambio",
|
||||
"write_down_backup_password": "Escribe su contraseña de respaldo, que se utiliza para la importación de sus archivos de respaldo.",
|
||||
"xlm_extra_info": "No olvides especificar el ID de nota al enviar la transacción XLM para el intercambio",
|
||||
"xmr_available_balance": "Saldo disponible",
|
||||
"xmr_full_balance": "Balance total",
|
||||
"xmr_hidden": "Oculto",
|
||||
"xmr_to_error": "Error de XMR.TO",
|
||||
"xmr_to_error_description": "Monto invalido. Límite máximo de 8 dígitos después del punto decimal",
|
||||
"xrp_extra_info": "No olvide especificar la etiqueta de destino al enviar la transacción XRP para el intercambio",
|
||||
"xrp_extra_info": "No olvides especificar la etiqueta de destino al enviar la transacción XRP para el intercambio",
|
||||
"yat": "Yat",
|
||||
"yat_address": "Dirección de Yat",
|
||||
"yat_alert_content": "Los usuarios de Cake Wallet ahora pueden enviar y recibir todas sus monedas favoritas con un nombre de usuario único basado en emoji.",
|
||||
"yat_alert_title": "Envíe y reciba criptomonedas más fácilmente con Yat",
|
||||
"yat_alert_title": "Envía y recibe criptomonedas más fácilmente con Yat",
|
||||
"yat_error": "Error de Yat",
|
||||
"yat_error_content": "No hay direcciones vinculadas con este Yat. Prueba con otro Yat",
|
||||
"yat_popup_content": "Ahora puede enviar y recibir criptografía en Cake Wallet con su Yat, un nombre de usuario corto basado en emoji. Administre Yats en cualquier momento en la pantalla de configuración",
|
||||
"yat_popup_title": "La dirección de su billetera se puede emojificar.",
|
||||
"yat_popup_content": "Ahora puede enviar y recibir criptografía en Cake Wallet con su Yat, un nombre de usuario corto basado en emoji. Administra Yats en cualquier momento en la pantalla de configuración",
|
||||
"yat_popup_title": "La dirección de tu billetera se puede emojificar.",
|
||||
"yesterday": "Ayer",
|
||||
"you_now_have_debit_card": "Ahora tiene una tarjeta de débito",
|
||||
"you_now_have_debit_card": "Ahora tienes una tarjeta de débito",
|
||||
"you_pay": "Tú pagas",
|
||||
"you_will_get": "Convertir a",
|
||||
"you_will_send": "Convertir de",
|
||||
"yy": "YY"
|
||||
}
|
||||
}
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Expire le",
|
||||
"expiry_and_validity": "Expiration et validité",
|
||||
"export_backup": "Exporter la sauvegarde",
|
||||
"export_logs": "Journaux d'exportation",
|
||||
"extra_id": "ID supplémentaire :",
|
||||
"extracted_address_content": "Vous allez envoyer des fonds à\n${recipient_name}",
|
||||
"failed_authentication": "Échec d'authentification. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB est un nouveau protocole qui apporte des transactions plus rapides, moins chères et plus privées à Litecoin",
|
||||
"litecoin_mweb_dismiss": "Rejeter",
|
||||
"litecoin_mweb_display_card": "Afficher la carte MWeb",
|
||||
"litecoin_mweb_enable": "Activer Mweb",
|
||||
"litecoin_mweb_enable_later": "Vous pouvez choisir d'activer à nouveau MWEB sous Paramètres d'affichage.",
|
||||
"litecoin_mweb_logs": "Journaux MWEB",
|
||||
"litecoin_mweb_node": "Node MWEB",
|
||||
"litecoin_mweb_pegin": "Entraver",
|
||||
"litecoin_mweb_pegout": "Crever",
|
||||
"litecoin_mweb_scanning": "Scann mweb",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Yana ƙarewa",
|
||||
"expiry_and_validity": "Karewa da inganci",
|
||||
"export_backup": "Ajiyayyen fitarwa",
|
||||
"export_logs": "Injin fitarwa",
|
||||
"extra_id": "Karin ID:",
|
||||
"extracted_address_content": "Za ku aika da kudade zuwa\n${recipient_name}",
|
||||
"failed_authentication": "Binne wajen shiga. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "Mweb shine sabon tsarin yarjejeniya da ya kawo da sauri, mai rahusa, da kuma ma'amaloli masu zaman kansu zuwa Litecoin",
|
||||
"litecoin_mweb_dismiss": "Tuɓe \\ sallama",
|
||||
"litecoin_mweb_display_card": "Nuna katin Mweb",
|
||||
"litecoin_mweb_enable": "Kunna Mweb",
|
||||
"litecoin_mweb_enable_later": "Kuna iya zaɓar kunna Mweb kuma a ƙarƙashin saitunan nuni.",
|
||||
"litecoin_mweb_logs": "Jagoran Mweb",
|
||||
"litecoin_mweb_node": "Mweb Node",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Peg fita",
|
||||
"litecoin_mweb_scanning": "Mweb scanning",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "पर समय सीमा समाप्त",
|
||||
"expiry_and_validity": "समाप्ति और वैधता",
|
||||
"export_backup": "निर्यात बैकअप",
|
||||
"export_logs": "निर्यात लॉग",
|
||||
"extra_id": "अतिरिक्त आईडी:",
|
||||
"extracted_address_content": "आपको धनराशि भेजी जाएगी\n${recipient_name}",
|
||||
"failed_authentication": "प्रमाणीकरण विफल. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB एक नया प्रोटोकॉल है जो लिटकोइन के लिए तेजी से, सस्ता और अधिक निजी लेनदेन लाता है",
|
||||
"litecoin_mweb_dismiss": "नकार देना",
|
||||
"litecoin_mweb_display_card": "MWEB कार्ड दिखाएं",
|
||||
"litecoin_mweb_enable": "MWEB सक्षम करें",
|
||||
"litecoin_mweb_enable_later": "आप प्रदर्शन सेटिंग्स के तहत फिर से MWEB को सक्षम करने के लिए चुन सकते हैं।",
|
||||
"litecoin_mweb_logs": "MWEB लॉग",
|
||||
"litecoin_mweb_node": "MWEB नोड",
|
||||
"litecoin_mweb_pegin": "खूंटी",
|
||||
"litecoin_mweb_pegout": "मरना",
|
||||
"litecoin_mweb_scanning": "MWEB स्कैनिंग",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Istječe",
|
||||
"expiry_and_validity": "Istek i valjanost",
|
||||
"export_backup": "Izvezi sigurnosnu kopiju",
|
||||
"export_logs": "Izvozni trupci",
|
||||
"extra_id": "Dodatni ID:",
|
||||
"extracted_address_content": "Poslat ćete sredstva primatelju\n${recipient_name}",
|
||||
"failed_authentication": "Autentifikacija neuspješna. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB je novi protokol koji u Litecoin donosi brže, jeftinije i privatnije transakcije",
|
||||
"litecoin_mweb_dismiss": "Odbaciti",
|
||||
"litecoin_mweb_display_card": "Prikaži MWeb karticu",
|
||||
"litecoin_mweb_enable": "Omogući MWeb",
|
||||
"litecoin_mweb_enable_later": "Možete odabrati da MWEB ponovo omogućite pod postavkama zaslona.",
|
||||
"litecoin_mweb_logs": "MWEB trupci",
|
||||
"litecoin_mweb_node": "MWEB čvor",
|
||||
"litecoin_mweb_pegin": "Uvući se",
|
||||
"litecoin_mweb_pegout": "Odapeti",
|
||||
"litecoin_mweb_scanning": "MWEB skeniranje",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Վավերականությունը լրանում է",
|
||||
"expiry_and_validity": "Վավերականություն և լրացում",
|
||||
"export_backup": "Արտահանել կրկնօրինակը",
|
||||
"export_logs": "Արտահանման տեղեկամատյաններ",
|
||||
"extra_id": "Լրացուցիչ ID",
|
||||
"extracted_address_content": "Դուք կուղարկեք գումար ${recipient_name}",
|
||||
"failed_authentication": "Վավերացումը ձախողվեց. ${state_error}",
|
||||
|
@ -367,7 +368,10 @@
|
|||
"light_theme": "Լուսավոր",
|
||||
"litecoin_mweb_description": "Mweb- ը նոր արձանագրություն է, որը բերում է ավելի արագ, ավելի էժան եւ ավելի մասնավոր գործարքներ դեպի LITECOIN",
|
||||
"litecoin_mweb_dismiss": "Հեռացնել",
|
||||
"litecoin_mweb_enable": "Միացնել Mweb- ը",
|
||||
"litecoin_mweb_enable_later": "Կարող եք ընտրել Mweb- ը կրկին միացնել ցուցադրման պարամետրերը:",
|
||||
"litecoin_mweb_logs": "Mweb տեղեկամատյաններ",
|
||||
"litecoin_mweb_node": "Mweb հանգույց",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Հափշտակել",
|
||||
"live_fee_rates": "Ապակի վարձավճարներ API- ի միջոցով",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Kadaluarsa pada",
|
||||
"expiry_and_validity": "Kedaluwarsa dan validitas",
|
||||
"export_backup": "Ekspor cadangan",
|
||||
"export_logs": "Log ekspor",
|
||||
"extra_id": "ID tambahan:",
|
||||
"extracted_address_content": "Anda akan mengirim dana ke\n${recipient_name}",
|
||||
"failed_authentication": "Otentikasi gagal. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB adalah protokol baru yang membawa transaksi yang lebih cepat, lebih murah, dan lebih pribadi ke Litecoin",
|
||||
"litecoin_mweb_dismiss": "Membubarkan",
|
||||
"litecoin_mweb_display_card": "Tunjukkan kartu mWeb",
|
||||
"litecoin_mweb_enable": "Aktifkan MWEB",
|
||||
"litecoin_mweb_enable_later": "Anda dapat memilih untuk mengaktifkan MWEB lagi di bawah pengaturan tampilan.",
|
||||
"litecoin_mweb_logs": "Log MWeb",
|
||||
"litecoin_mweb_node": "Node MWEB",
|
||||
"litecoin_mweb_pegin": "Pasak masuk",
|
||||
"litecoin_mweb_pegout": "Mati",
|
||||
"litecoin_mweb_scanning": "Pemindaian MWEB",
|
||||
|
|
|
@ -296,6 +296,7 @@
|
|||
"expiresOn": "Scade il",
|
||||
"expiry_and_validity": "Scadenza e validità",
|
||||
"export_backup": "Esporta backup",
|
||||
"export_logs": "Registri di esportazione",
|
||||
"extra_id": "Extra ID:",
|
||||
"extracted_address_content": "Invierai i tuoi fondi a\n${recipient_name}",
|
||||
"failed_authentication": "Autenticazione fallita. ${state_error}",
|
||||
|
@ -372,7 +373,10 @@
|
|||
"litecoin_mweb_description": "MWeb è un nuovo protocollo che porta transazioni più veloci, più economiche e più private a Litecoin",
|
||||
"litecoin_mweb_dismiss": "Congedare",
|
||||
"litecoin_mweb_display_card": "Mostra la scheda MWeb",
|
||||
"litecoin_mweb_enable": "Abilita mWeb",
|
||||
"litecoin_mweb_enable_later": "È possibile scegliere di abilitare nuovamente MWeb nelle impostazioni di visualizzazione.",
|
||||
"litecoin_mweb_logs": "Registri mWeb",
|
||||
"litecoin_mweb_node": "Nodo MWeb",
|
||||
"litecoin_mweb_pegin": "Piolo in",
|
||||
"litecoin_mweb_pegout": "PEG OUT",
|
||||
"litecoin_mweb_scanning": "Scansione MWeb",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "有効期限は次のとおりです",
|
||||
"expiry_and_validity": "有効期限と有効性",
|
||||
"export_backup": "バックアップのエクスポート",
|
||||
"export_logs": "ログをエクスポートします",
|
||||
"extra_id": "追加ID:",
|
||||
"extracted_address_content": "に送金します\n${recipient_name}",
|
||||
"failed_authentication": "認証失敗. ${state_error}",
|
||||
|
@ -372,7 +373,10 @@
|
|||
"litecoin_mweb_description": "MWEBは、Litecoinにより速く、より安価で、よりプライベートなトランザクションをもたらす新しいプロトコルです",
|
||||
"litecoin_mweb_dismiss": "却下する",
|
||||
"litecoin_mweb_display_card": "MWEBカードを表示します",
|
||||
"litecoin_mweb_enable": "MWEBを有効にします",
|
||||
"litecoin_mweb_enable_later": "表示設定の下で、MWEBを再度有効にすることを選択できます。",
|
||||
"litecoin_mweb_logs": "MWEBログ",
|
||||
"litecoin_mweb_node": "MWEBノード",
|
||||
"litecoin_mweb_pegin": "ペグイン",
|
||||
"litecoin_mweb_pegout": "ペグアウト",
|
||||
"litecoin_mweb_scanning": "MWEBスキャン",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "만료 날짜",
|
||||
"expiry_and_validity": "만료와 타당성",
|
||||
"export_backup": "백업 내보내기",
|
||||
"export_logs": "내보내기 로그",
|
||||
"extra_id": "추가 ID:",
|
||||
"extracted_address_content": "당신은에 자금을 보낼 것입니다\n${recipient_name}",
|
||||
"failed_authentication": "인증 실패. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB는 Litecoin에 더 빠르고 저렴하며 개인 거래를 제공하는 새로운 프로토콜입니다.",
|
||||
"litecoin_mweb_dismiss": "해고하다",
|
||||
"litecoin_mweb_display_card": "mweb 카드를 보여주십시오",
|
||||
"litecoin_mweb_enable": "mweb 활성화",
|
||||
"litecoin_mweb_enable_later": "디스플레이 설정에서 MWEB를 다시 활성화하도록 선택할 수 있습니다.",
|
||||
"litecoin_mweb_logs": "mweb 로그",
|
||||
"litecoin_mweb_node": "mweb 노드",
|
||||
"litecoin_mweb_pegin": "페그를 입력하십시오",
|
||||
"litecoin_mweb_pegout": "죽다",
|
||||
"litecoin_mweb_scanning": "mweb 스캔",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "သက်တမ်းကုန်သည်။",
|
||||
"expiry_and_validity": "သက်တမ်းကုန်ဆုံးခြင်းနှင့်တရားဝင်မှု",
|
||||
"export_backup": "အရန်ကူးထုတ်ရန်",
|
||||
"export_logs": "ပို့ကုန်မှတ်တမ်းများ",
|
||||
"extra_id": "အပို ID-",
|
||||
"extracted_address_content": "သင်သည် \n${recipient_name} သို့ ရန်ပုံငွေများ ပေးပို့ပါမည်",
|
||||
"failed_authentication": "အထောက်အထားစိစစ်ခြင်း မအောင်မြင်ပါ။. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "Mweb သည် Protocol အသစ်ဖြစ်ပြီး LitCoin သို့ပိုမိုဈေးချိုသာသော, စျေးသက်သက်သာသာသုံးခြင်းနှင့်ပိုမိုများပြားသောပုဂ္ဂလိကငွေပို့ဆောင်မှုများကိုဖြစ်ပေါ်စေသည်",
|
||||
"litecoin_mweb_dismiss": "ထုတ်ပစ်",
|
||||
"litecoin_mweb_display_card": "MweB ကဒ်ကိုပြပါ",
|
||||
"litecoin_mweb_enable": "mweb enable",
|
||||
"litecoin_mweb_enable_later": "သင် MweB ကို display settings အောက်ရှိ ထပ်မံ. ခွင့်ပြုရန်ရွေးချယ်နိုင်သည်။",
|
||||
"litecoin_mweb_logs": "Mweb မှတ်တမ်းများ",
|
||||
"litecoin_mweb_node": "mweb node ကို",
|
||||
"litecoin_mweb_pegin": "တံစို့",
|
||||
"litecoin_mweb_pegout": "တံစို့",
|
||||
"litecoin_mweb_scanning": "mweb scanning",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Verloopt op",
|
||||
"expiry_and_validity": "Vervallen en geldigheid",
|
||||
"export_backup": "Back-up exporteren",
|
||||
"export_logs": "Exporteer logboeken",
|
||||
"extra_id": "Extra ID:",
|
||||
"extracted_address_content": "U stuurt geld naar\n${recipient_name}",
|
||||
"failed_authentication": "Mislukte authenticatie. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB is een nieuw protocol dat snellere, goedkopere en meer privé -transacties naar Litecoin brengt",
|
||||
"litecoin_mweb_dismiss": "Afwijzen",
|
||||
"litecoin_mweb_display_card": "Toon MWEB -kaart",
|
||||
"litecoin_mweb_enable": "MWEB inschakelen",
|
||||
"litecoin_mweb_enable_later": "U kunt ervoor kiezen om MWeb opnieuw in te schakelen onder weergave -instellingen.",
|
||||
"litecoin_mweb_logs": "MWEB -logboeken",
|
||||
"litecoin_mweb_node": "MWEB -knooppunt",
|
||||
"litecoin_mweb_pegin": "Vastmaken",
|
||||
"litecoin_mweb_pegout": "Uithakken",
|
||||
"litecoin_mweb_scanning": "MWEB -scanning",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Upływa w dniu",
|
||||
"expiry_and_validity": "Wygaśnięcie i ważność",
|
||||
"export_backup": "Eksportuj kopię zapasową",
|
||||
"export_logs": "Dzienniki eksportu",
|
||||
"extra_id": "Dodatkowy ID:",
|
||||
"extracted_address_content": "Wysyłasz środki na\n${recipient_name}",
|
||||
"failed_authentication": "Nieudane uwierzytelnienie. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB to nowy protokół, który przynosi szybciej, tańsze i bardziej prywatne transakcje do Litecoin",
|
||||
"litecoin_mweb_dismiss": "Odrzucać",
|
||||
"litecoin_mweb_display_card": "Pokaż kartę MWEB",
|
||||
"litecoin_mweb_enable": "Włącz MWEB",
|
||||
"litecoin_mweb_enable_later": "Możesz ponownie włączyć MWEB w ustawieniach wyświetlania.",
|
||||
"litecoin_mweb_logs": "Dzienniki MWEB",
|
||||
"litecoin_mweb_node": "Węzeł MWEB",
|
||||
"litecoin_mweb_pegin": "Kołek",
|
||||
"litecoin_mweb_pegout": "Palikować",
|
||||
"litecoin_mweb_scanning": "Skanowanie MWEB",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Expira em",
|
||||
"expiry_and_validity": "Expiração e validade",
|
||||
"export_backup": "Backup de exportação",
|
||||
"export_logs": "Exportar logs",
|
||||
"extra_id": "ID extra:",
|
||||
"extracted_address_content": "Você enviará fundos para\n${recipient_name}",
|
||||
"failed_authentication": "Falha na autenticação. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB é um novo protocolo que traz transações mais rápidas, baratas e mais privadas para o Litecoin",
|
||||
"litecoin_mweb_dismiss": "Liberar",
|
||||
"litecoin_mweb_display_card": "Mostre o cartão MWEB",
|
||||
"litecoin_mweb_enable": "Ativar Mweb",
|
||||
"litecoin_mweb_enable_later": "Você pode optar por ativar o MWEB novamente em Configurações de exibição.",
|
||||
"litecoin_mweb_logs": "Logs MWeb",
|
||||
"litecoin_mweb_node": "Nó MWeb",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Peg fora",
|
||||
"litecoin_mweb_scanning": "MWEB Scanning",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Годен до",
|
||||
"expiry_and_validity": "Истечение и достоверность",
|
||||
"export_backup": "Экспорт резервной копии",
|
||||
"export_logs": "Экспортные журналы",
|
||||
"extra_id": "Дополнительный ID:",
|
||||
"extracted_address_content": "Вы будете отправлять средства\n${recipient_name}",
|
||||
"failed_authentication": "Ошибка аутентификации. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB - это новый протокол, который приносит быстрее, дешевле и более частные транзакции в Litecoin",
|
||||
"litecoin_mweb_dismiss": "Увольнять",
|
||||
"litecoin_mweb_display_card": "Показать карту MWEB",
|
||||
"litecoin_mweb_enable": "Включить MWEB",
|
||||
"litecoin_mweb_enable_later": "Вы можете снова включить MWEB в настройках отображения.",
|
||||
"litecoin_mweb_logs": "MWEB журналы",
|
||||
"litecoin_mweb_node": "Узел MWEB",
|
||||
"litecoin_mweb_pegin": "Внедрять",
|
||||
"litecoin_mweb_pegout": "Выкрикивать",
|
||||
"litecoin_mweb_scanning": "MWEB сканирование",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "หมดอายุวันที่",
|
||||
"expiry_and_validity": "หมดอายุและถูกต้อง",
|
||||
"export_backup": "ส่งออกข้อมูลสำรอง",
|
||||
"export_logs": "บันทึกการส่งออก",
|
||||
"extra_id": "ไอดีเพิ่มเติม:",
|
||||
"extracted_address_content": "คุณกำลังจะส่งเงินไปยัง\n${recipient_name}",
|
||||
"failed_authentication": "การยืนยันสิทธิ์ล้มเหลว ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB เป็นโปรโตคอลใหม่ที่นำการทำธุรกรรมที่เร็วกว่าราคาถูกกว่าและเป็นส่วนตัวมากขึ้นไปยัง Litecoin",
|
||||
"litecoin_mweb_dismiss": "อนุญาตให้ออกไป",
|
||||
"litecoin_mweb_display_card": "แสดงการ์ด mweb",
|
||||
"litecoin_mweb_enable": "เปิดใช้งาน mweb",
|
||||
"litecoin_mweb_enable_later": "คุณสามารถเลือกเปิดใช้งาน MWEB อีกครั้งภายใต้การตั้งค่าการแสดงผล",
|
||||
"litecoin_mweb_logs": "บันทึก MWEB",
|
||||
"litecoin_mweb_node": "โหนด MWEB",
|
||||
"litecoin_mweb_pegin": "หมุด",
|
||||
"litecoin_mweb_pegout": "ตรึง",
|
||||
"litecoin_mweb_scanning": "การสแกน MWEB",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Mag-e-expire sa",
|
||||
"expiry_and_validity": "Pag-expire at Bisa",
|
||||
"export_backup": "I-export ang backup",
|
||||
"export_logs": "Mga log ng pag -export",
|
||||
"extra_id": "Dagdag na ID:",
|
||||
"extracted_address_content": "Magpapadala ka ng pondo sa\n${recipient_name}",
|
||||
"failed_authentication": "Nabigo ang pagpapatunay. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "Ang MWeb ay isang bagong protocol na nagdadala ng mas mabilis, mas mura, at mas maraming pribadong mga transaksyon sa Litecoin",
|
||||
"litecoin_mweb_dismiss": "Tanggalin",
|
||||
"litecoin_mweb_display_card": "Ipakita ang MWEB Card",
|
||||
"litecoin_mweb_enable": "Paganahin ang MWeb",
|
||||
"litecoin_mweb_enable_later": "Maaari kang pumili upang paganahin muli ang MWeb sa ilalim ng mga setting ng pagpapakita.",
|
||||
"litecoin_mweb_logs": "MWEB log",
|
||||
"litecoin_mweb_node": "Mweb node",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Peg out",
|
||||
"litecoin_mweb_scanning": "Pag -scan ng Mweb",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Tarihinde sona eriyor",
|
||||
"expiry_and_validity": "Sona erme ve geçerlilik",
|
||||
"export_backup": "Yedeği dışa aktar",
|
||||
"export_logs": "Dışa aktarma günlükleri",
|
||||
"extra_id": "Ekstra ID:",
|
||||
"extracted_address_content": "Parayı buraya gönderceksin:\n${recipient_name}",
|
||||
"failed_authentication": "Doğrulama başarısız oldu. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB, Litecoin'e daha hızlı, daha ucuz ve daha fazla özel işlem getiren yeni bir protokoldür",
|
||||
"litecoin_mweb_dismiss": "Azletmek",
|
||||
"litecoin_mweb_display_card": "MWEB kartını göster",
|
||||
"litecoin_mweb_enable": "MWEB'i etkinleştir",
|
||||
"litecoin_mweb_enable_later": "Ekran ayarlarının altında MWEB'yi tekrar etkinleştirmeyi seçebilirsiniz.",
|
||||
"litecoin_mweb_logs": "MWEB günlükleri",
|
||||
"litecoin_mweb_node": "MWEB düğümü",
|
||||
"litecoin_mweb_pegin": "Takılmak",
|
||||
"litecoin_mweb_pegout": "Çiğnemek",
|
||||
"litecoin_mweb_scanning": "MWEB taraması",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "Термін дії закінчується",
|
||||
"expiry_and_validity": "Закінчення та обгрунтованість",
|
||||
"export_backup": "Експортувати резервну копію",
|
||||
"export_logs": "Експортні журнали",
|
||||
"extra_id": "Додатковий ID:",
|
||||
"extracted_address_content": "Ви будете відправляти кошти\n${recipient_name}",
|
||||
"failed_authentication": "Помилка аутентифікації. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB - це новий протокол, який приносить швидкі, дешевші та більш приватні транзакції Litecoin",
|
||||
"litecoin_mweb_dismiss": "Звільнити",
|
||||
"litecoin_mweb_display_card": "Показати карту MWeb",
|
||||
"litecoin_mweb_enable": "Увімкнути mweb",
|
||||
"litecoin_mweb_enable_later": "Ви можете знову ввімкнути MWEB в налаштуваннях дисплея.",
|
||||
"litecoin_mweb_logs": "Журнали MWeb",
|
||||
"litecoin_mweb_node": "Вузол MWeb",
|
||||
"litecoin_mweb_pegin": "Подякувати",
|
||||
"litecoin_mweb_pegout": "Подякувати",
|
||||
"litecoin_mweb_scanning": "Сканування Mweb",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "ﺩﺎﻌﯿﻣ ﯽﻣﺎﺘﺘﺧﺍ",
|
||||
"expiry_and_validity": "میعاد ختم اور صداقت",
|
||||
"export_backup": "بیک اپ برآمد کریں۔",
|
||||
"export_logs": "نوشتہ جات برآمد کریں",
|
||||
"extra_id": "اضافی ID:",
|
||||
"extracted_address_content": "آپ فنڈز بھیج رہے ہوں گے\n${recipient_name}",
|
||||
"failed_authentication": "ناکام تصدیق۔ ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB ایک نیا پروٹوکول ہے جو لیٹیکوئن میں تیز ، سستا اور زیادہ نجی لین دین لاتا ہے",
|
||||
"litecoin_mweb_dismiss": "خارج",
|
||||
"litecoin_mweb_display_card": "MWEB کارڈ دکھائیں",
|
||||
"litecoin_mweb_enable": "MWEB کو فعال کریں",
|
||||
"litecoin_mweb_enable_later": "آپ ڈسپلے کی ترتیبات کے تحت MWEB کو دوبارہ فعال کرنے کا انتخاب کرسکتے ہیں۔",
|
||||
"litecoin_mweb_logs": "MWEB لاگز",
|
||||
"litecoin_mweb_node": "MWEB نوڈ",
|
||||
"litecoin_mweb_pegin": "پیگ میں",
|
||||
"litecoin_mweb_pegout": "پیگ آؤٹ",
|
||||
"litecoin_mweb_scanning": "MWEB اسکیننگ",
|
||||
|
|
|
@ -296,6 +296,7 @@
|
|||
"expiresOn": "Hết hạn vào",
|
||||
"expiry_and_validity": "Hạn và hiệu lực",
|
||||
"export_backup": "Xuất sao lưu",
|
||||
"export_logs": "Nhật ký xuất khẩu",
|
||||
"extra_id": "ID bổ sung:",
|
||||
"extracted_address_content": "Bạn sẽ gửi tiền cho\n${recipient_name}",
|
||||
"failed_authentication": "Xác thực không thành công. ${state_error}",
|
||||
|
@ -368,7 +369,10 @@
|
|||
"light_theme": "Chủ đề sáng",
|
||||
"litecoin_mweb_description": "MWEB là một giao thức mới mang lại các giao dịch nhanh hơn, rẻ hơn và riêng tư hơn cho Litecoin",
|
||||
"litecoin_mweb_dismiss": "Miễn nhiệm",
|
||||
"litecoin_mweb_enable": "Bật MWEB",
|
||||
"litecoin_mweb_enable_later": "Bạn có thể chọn bật lại MWEB trong cài đặt hiển thị.",
|
||||
"litecoin_mweb_logs": "Nhật ký MWEB",
|
||||
"litecoin_mweb_node": "Nút MWEB",
|
||||
"litecoin_mweb_pegin": "Chốt vào",
|
||||
"litecoin_mweb_pegout": "Chốt ra",
|
||||
"live_fee_rates": "Tỷ lệ phí hiện tại qua API",
|
||||
|
|
|
@ -296,6 +296,7 @@
|
|||
"expiresOn": "Ipari lori",
|
||||
"expiry_and_validity": "Ipari ati idaniloju",
|
||||
"export_backup": "Sún ẹ̀dà nípamọ́ síta",
|
||||
"export_logs": "Wọle si okeere",
|
||||
"extra_id": "Àmì ìdánimọ̀ tó fikún:",
|
||||
"extracted_address_content": "Ẹ máa máa fi owó ránṣẹ́ sí\n${recipient_name}",
|
||||
"failed_authentication": "Ìfẹ̀rílàdí pipòfo. ${state_error}",
|
||||
|
@ -372,7 +373,10 @@
|
|||
"litecoin_mweb_description": "Mweb jẹ ilana ilana tuntun ti o mu iyara wa yiyara, din owo, ati awọn iṣowo ikọkọ diẹ sii si Livcoin",
|
||||
"litecoin_mweb_dismiss": "Tuka",
|
||||
"litecoin_mweb_display_card": "Fihan kaadi Mweb",
|
||||
"litecoin_mweb_enable": "Mu mweb",
|
||||
"litecoin_mweb_enable_later": "O le yan lati ṣiṣẹ Mweb lẹẹkansi labẹ awọn eto ifihan.",
|
||||
"litecoin_mweb_logs": "MTweb logs",
|
||||
"litecoin_mweb_node": "Alweb joko",
|
||||
"litecoin_mweb_pegin": "Peg in",
|
||||
"litecoin_mweb_pegout": "Peg jade",
|
||||
"litecoin_mweb_scanning": "Mweb scanning",
|
||||
|
|
|
@ -295,6 +295,7 @@
|
|||
"expiresOn": "到期",
|
||||
"expiry_and_validity": "到期和有效性",
|
||||
"export_backup": "导出备份",
|
||||
"export_logs": "导出日志",
|
||||
"extra_id": "额外ID:",
|
||||
"extracted_address_content": "您将汇款至\n${recipient_name}",
|
||||
"failed_authentication": "身份验证失败. ${state_error}",
|
||||
|
@ -371,7 +372,10 @@
|
|||
"litecoin_mweb_description": "MWEB是一项新协议,它将更快,更便宜和更多的私人交易带给Litecoin",
|
||||
"litecoin_mweb_dismiss": "解雇",
|
||||
"litecoin_mweb_display_card": "显示MWEB卡",
|
||||
"litecoin_mweb_enable": "启用MWEB",
|
||||
"litecoin_mweb_enable_later": "您可以选择在显示设置下再次启用MWEB。",
|
||||
"litecoin_mweb_logs": "MWEB日志",
|
||||
"litecoin_mweb_node": "MWEB节点",
|
||||
"litecoin_mweb_pegin": "钉进",
|
||||
"litecoin_mweb_pegout": "昏倒",
|
||||
"litecoin_mweb_scanning": "MWEB扫描",
|
||||
|
|
Loading…
Reference in a new issue