mirror of
https://github.com/monero-project/monero-gui.git
synced 2025-01-18 08:44:46 +00:00
21831c9b24
Should solve issues with monerod taking a while to exit. Simple mode nodes aren't beneficial to the network anyway, so having no incoming connections in some cases should be fine.
2389 lines
90 KiB
QML
2389 lines
90 KiB
QML
// Copyright (c) 2014-2019, The Monero Project
|
||
//
|
||
// All rights reserved.
|
||
//
|
||
// Redistribution and use in source and binary forms, with or without modification, are
|
||
// permitted provided that the following conditions are met:
|
||
//
|
||
// 1. Redistributions of source code must retain the above copyright notice, this list of
|
||
// conditions and the following disclaimer.
|
||
//
|
||
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
||
// of conditions and the following disclaimer in the documentation and/or other
|
||
// materials provided with the distribution.
|
||
//
|
||
// 3. Neither the name of the copyright holder nor the names of its contributors may be
|
||
// used to endorse or promote products derived from this software without specific
|
||
// prior written permission.
|
||
//
|
||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
||
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
||
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
||
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
||
import QtQml.Models 2.2
|
||
import QtQuick 2.9
|
||
import QtQuick.Window 2.0
|
||
import QtQuick.Controls 1.1
|
||
import QtQuick.Controls.Styles 1.1
|
||
import QtQuick.Dialogs 1.2
|
||
import QtGraphicalEffects 1.0
|
||
|
||
import FontAwesome 1.0
|
||
|
||
import moneroComponents.Network 1.0
|
||
import moneroComponents.Wallet 1.0
|
||
import moneroComponents.WalletManager 1.0
|
||
import moneroComponents.PendingTransaction 1.0
|
||
import moneroComponents.NetworkType 1.0
|
||
import moneroComponents.Settings 1.0
|
||
import moneroComponents.P2PoolManager 1.0
|
||
|
||
import "components"
|
||
import "components" as MoneroComponents
|
||
import "components/effects" as MoneroEffects
|
||
import "pages/merchant" as MoneroMerchant
|
||
import "wizard"
|
||
import "js/Utils.js" as Utils
|
||
import "js/Windows.js" as Windows
|
||
import "version.js" as Version
|
||
|
||
ApplicationWindow {
|
||
id: appWindow
|
||
title: "Monero" +
|
||
(persistentSettings.displayWalletNameInTitleBar && walletName
|
||
? " - " + walletName
|
||
: "")
|
||
minimumWidth: 750
|
||
minimumHeight: 450
|
||
|
||
property var currentItem
|
||
property var previousActiveFocusItem
|
||
property bool hideBalanceForced: false
|
||
property bool ctrlPressed: false
|
||
property alias persistentSettings : persistentSettings
|
||
property string accountsDir: !persistentSettings.portable ? moneroAccountsDir : persistentSettings.portableFolderName + "/wallets"
|
||
property var currentWallet;
|
||
property bool disconnected: currentWallet ? currentWallet.disconnected : false
|
||
property var transaction;
|
||
property var walletPassword
|
||
property int restoreHeight:0
|
||
property bool daemonSynced: false
|
||
property bool walletSynced: false
|
||
property int maxWindowHeight: (isAndroid || isIOS)? screenAvailableHeight : (screenAvailableHeight < 900)? 720 : 800;
|
||
property bool daemonRunning: !persistentSettings.useRemoteNode && !disconnected
|
||
property int daemonStartStopInProgress: 0
|
||
property alias toolTip: toolTip
|
||
property string walletName
|
||
property bool viewOnly: false
|
||
property bool foundNewBlock: false
|
||
property bool qrScannerEnabled: (typeof builtWithScanner != "undefined") && builtWithScanner
|
||
property int blocksToSync: 1
|
||
property int firstBlockSeen
|
||
property bool isMining: false
|
||
property int walletMode: persistentSettings.walletMode
|
||
property var cameraUi
|
||
property bool androidCloseTapped: false;
|
||
property int userLastActive; // epoch
|
||
// Default daemon addresses
|
||
readonly property string localDaemonAddress : "localhost:" + getDefaultDaemonRpcPort(persistentSettings.nettype)
|
||
property string currentDaemonAddress;
|
||
property int disconnectedEpoch: 0
|
||
property int estimatedBlockchainSize: persistentSettings.pruneBlockchain ? 40 : 105 // GB
|
||
property alias viewState: rootItem.state
|
||
property string prevSplashText;
|
||
property bool splashDisplayedBeforeButtonRequest;
|
||
property bool themeTransition: false
|
||
|
||
// fiat price conversion
|
||
property real fiatPrice: 0
|
||
property var fiatPriceAPIs: {
|
||
return {
|
||
"kraken": {
|
||
"xmrusd": "https://api.kraken.com/0/public/Ticker?pair=XMRUSD",
|
||
"xmreur": "https://api.kraken.com/0/public/Ticker?pair=XMREUR"
|
||
},
|
||
"coingecko": {
|
||
"xmrusd": "https://api.coingecko.com/api/v3/simple/price?ids=monero&vs_currencies=usd",
|
||
"xmreur": "https://api.coingecko.com/api/v3/simple/price?ids=monero&vs_currencies=eur"
|
||
},
|
||
"cryptocompare": {
|
||
"xmrusd": "https://min-api.cryptocompare.com/data/price?fsym=XMR&tsyms=USD",
|
||
"xmreur": "https://min-api.cryptocompare.com/data/price?fsym=XMR&tsyms=EUR",
|
||
}
|
||
}
|
||
}
|
||
|
||
// true if wallet ever synchronized
|
||
property bool walletInitialized : false
|
||
|
||
// Current selected address / subaddress / (Receive/Account page)
|
||
property var current_address
|
||
property var current_address_label: "Primary"
|
||
property int current_subaddress_table_index: 0
|
||
|
||
function showPageRequest(page) {
|
||
middlePanel.state = page
|
||
leftPanel.selectItem(page)
|
||
}
|
||
|
||
function lock() {
|
||
passwordDialog.onRejectedCallback = function() { appWindow.showWizard(); }
|
||
passwordDialog.onAcceptedCallback = function() {
|
||
if(walletPassword === passwordDialog.password)
|
||
passwordDialog.close();
|
||
else
|
||
passwordDialog.showError(qsTr("Wrong password") + translationManager.emptyString);
|
||
}
|
||
passwordDialog.open(usefulName(persistentSettings.wallet_path));
|
||
}
|
||
|
||
function sequencePressed(obj, seq) {
|
||
if(seq === undefined || !leftPanel.enabled)
|
||
return
|
||
if(seq === "Ctrl") {
|
||
ctrlPressed = true
|
||
return
|
||
}
|
||
|
||
// lock wallet on demand
|
||
if(seq === "Ctrl+L" && !passwordDialog.visible) lock()
|
||
if(seq === "Ctrl+S") middlePanel.state = "Transfer"
|
||
else if(seq === "Ctrl+R") middlePanel.state = "Receive"
|
||
else if(seq === "Ctrl+H") middlePanel.state = "History"
|
||
else if(seq === "Ctrl+B") middlePanel.state = "AddressBook"
|
||
else if(seq === "Ctrl+E") middlePanel.state = "Settings"
|
||
else if(seq === "Ctrl+D") middlePanel.state = "Advanced"
|
||
else if(seq === "Ctrl+T") middlePanel.state = "Account"
|
||
else if(seq === "Ctrl+Tab" || seq === "Alt+Tab") {
|
||
/*
|
||
if(middlePanel.state === "Transfer") middlePanel.state = "Receive"
|
||
else if(middlePanel.state === "Receive") middlePanel.state = "TxKey"
|
||
else if(middlePanel.state === "TxKey") middlePanel.state = "SharedRingDB"
|
||
else if(middlePanel.state === "SharedRingDB") middlePanel.state = "History"
|
||
else if(middlePanel.state === "History") middlePanel.state = "AddressBook"
|
||
else if(middlePanel.state === "AddressBook") middlePanel.state = "Mining"
|
||
else if(middlePanel.state === "Mining") middlePanel.state = "Sign"
|
||
else if(middlePanel.state === "Sign") middlePanel.state = "Settings"
|
||
*/
|
||
if(middlePanel.state === "Settings") middlePanel.state = "Account"
|
||
else if(middlePanel.state === "Account") middlePanel.state = "Transfer"
|
||
else if(middlePanel.state === "Transfer") middlePanel.state = "AddressBook"
|
||
else if(middlePanel.state === "AddressBook") middlePanel.state = "Receive"
|
||
else if(middlePanel.state === "Receive") middlePanel.state = "History"
|
||
else if(middlePanel.state === "History") middlePanel.state = "Advanced"
|
||
else if(middlePanel.state === "Advanced") middlePanel.state = "Settings"
|
||
} else if(seq === "Ctrl+Shift+Backtab" || seq === "Alt+Shift+Backtab") {
|
||
/*
|
||
if(middlePanel.state === "Settings") middlePanel.state = "Sign"
|
||
else if(middlePanel.state === "Sign") middlePanel.state = "Mining"
|
||
else if(middlePanel.state === "Mining") middlePanel.state = "AddressBook"
|
||
else if(middlePanel.state === "AddressBook") middlePanel.state = "History"
|
||
else if(middlePanel.state === "History") middlePanel.state = "SharedRingDB"
|
||
else if(middlePanel.state === "SharedRingDB") middlePanel.state = "TxKey"
|
||
else if(middlePanel.state === "TxKey") middlePanel.state = "Receive"
|
||
else if(middlePanel.state === "Receive") middlePanel.state = "Transfer"
|
||
*/
|
||
if(middlePanel.state === "Settings") middlePanel.state = "Advanced"
|
||
else if(middlePanel.state === "Advanced") middlePanel.state = "History"
|
||
else if(middlePanel.state === "History") middlePanel.state = "Receive"
|
||
else if(middlePanel.state === "Receive") middlePanel.state = "AddressBook"
|
||
else if(middlePanel.state === "AddressBook") middlePanel.state = "Transfer"
|
||
else if(middlePanel.state === "Transfer") middlePanel.state = "Account"
|
||
else if(middlePanel.state === "Account") middlePanel.state = "Settings"
|
||
}
|
||
|
||
if (middlePanel.state !== "Advanced") updateBalance();
|
||
|
||
leftPanel.selectItem(middlePanel.state)
|
||
}
|
||
|
||
function sequenceReleased(obj, seq) {
|
||
if(seq === "Ctrl")
|
||
ctrlPressed = false
|
||
}
|
||
|
||
function mousePressed(obj, mouseX, mouseY) {}
|
||
function mouseReleased(obj, mouseX, mouseY) {}
|
||
|
||
function loadPage(page) {
|
||
middlePanel.state = page;
|
||
leftPanel.selectItem(page);
|
||
}
|
||
|
||
function openWallet(prevState) {
|
||
passwordDialog.onAcceptedCallback = function() {
|
||
walletPassword = passwordDialog.password;
|
||
initialize();
|
||
}
|
||
passwordDialog.onRejectedCallback = function() {
|
||
if (prevState) {
|
||
appWindow.viewState = prevState;
|
||
}
|
||
if (wizard.wizardState == "wizardOpenWallet1") {
|
||
wizard.wizardStateView.wizardOpenWallet1View.pageRoot.forceActiveFocus();
|
||
}
|
||
};
|
||
passwordDialog.open(usefulName(persistentSettings.wallet_path));
|
||
}
|
||
|
||
function initialize() {
|
||
console.log("initializing..")
|
||
|
||
// Use stored log level
|
||
if (persistentSettings.logLevel == 5)
|
||
walletManager.setLogCategories(persistentSettings.logCategories)
|
||
else
|
||
walletManager.setLogLevel(persistentSettings.logLevel)
|
||
|
||
// Reload transfer page with translations enabled
|
||
middlePanel.transferView.onPageCompleted();
|
||
|
||
// If currentWallet exists, we're just switching daemon - close/reopen wallet
|
||
if (typeof currentWallet !== "undefined" && currentWallet !== null) {
|
||
console.log("Daemon change - closing " + currentWallet)
|
||
closeWallet();
|
||
} else if (!walletInitialized) {
|
||
// set page to transfer if not changing daemon
|
||
middlePanel.state = "Transfer";
|
||
leftPanel.selectItem(middlePanel.state)
|
||
}
|
||
|
||
// Local daemon settings
|
||
walletManager.setDaemonAddressAsync(localDaemonAddress);
|
||
|
||
// enable timers
|
||
userInActivityTimer.running = true;
|
||
|
||
// wallet already opened with wizard, we just need to initialize it
|
||
var wallet_path = persistentSettings.wallet_path;
|
||
if(isIOS)
|
||
wallet_path = appWindow.accountsDir + wallet_path;
|
||
// console.log("opening wallet at: ", wallet_path, "with password: ", appWindow.walletPassword);
|
||
console.log("opening wallet at: ", wallet_path, ", network type: ", persistentSettings.nettype == NetworkType.MAINNET ? "mainnet" : persistentSettings.nettype == NetworkType.TESTNET ? "testnet" : "stagenet");
|
||
|
||
this.onWalletOpening();
|
||
walletManager.openWalletAsync(
|
||
wallet_path,
|
||
walletPassword,
|
||
persistentSettings.nettype,
|
||
persistentSettings.kdfRounds);
|
||
}
|
||
|
||
function closeWallet(callback) {
|
||
|
||
// Disconnect all listeners
|
||
if (typeof currentWallet === "undefined" || currentWallet === null) {
|
||
if (callback) {
|
||
callback();
|
||
}
|
||
return;
|
||
}
|
||
|
||
currentWallet.heightRefreshed.disconnect(onHeightRefreshed);
|
||
currentWallet.refreshed.disconnect(onWalletRefresh)
|
||
currentWallet.updated.disconnect(onWalletUpdate)
|
||
currentWallet.newBlock.disconnect(onWalletNewBlock)
|
||
currentWallet.moneySpent.disconnect(onWalletMoneySent)
|
||
currentWallet.moneyReceived.disconnect(onWalletMoneyReceived)
|
||
currentWallet.unconfirmedMoneyReceived.disconnect(onWalletUnconfirmedMoneyReceived)
|
||
currentWallet.transactionCreated.disconnect(onTransactionCreated)
|
||
currentWallet.connectionStatusChanged.disconnect(onWalletConnectionStatusChanged)
|
||
currentWallet.deviceButtonRequest.disconnect(onDeviceButtonRequest);
|
||
currentWallet.deviceButtonPressed.disconnect(onDeviceButtonPressed);
|
||
currentWallet.walletPassphraseNeeded.disconnect(onWalletPassphraseNeededWallet);
|
||
currentWallet.transactionCommitted.disconnect(onTransactionCommitted);
|
||
middlePanel.paymentClicked.disconnect(handlePayment);
|
||
middlePanel.sweepUnmixableClicked.disconnect(handleSweepUnmixable);
|
||
middlePanel.getProofClicked.disconnect(handleGetProof);
|
||
middlePanel.checkProofClicked.disconnect(handleCheckProof);
|
||
|
||
appWindow.walletName = "";
|
||
currentWallet = undefined;
|
||
|
||
appWindow.showProcessingSplash(qsTr("Closing wallet..."));
|
||
if (callback) {
|
||
walletManager.closeWalletAsync(function() {
|
||
hideProcessingSplash();
|
||
callback();
|
||
});
|
||
} else {
|
||
walletManager.closeWallet();
|
||
hideProcessingSplash();
|
||
}
|
||
}
|
||
|
||
function connectWallet(wallet) {
|
||
currentWallet = wallet
|
||
|
||
walletName = usefulName(wallet.path)
|
||
|
||
viewOnly = currentWallet.viewOnly;
|
||
|
||
// New wallets saves the testnet flag in keys file.
|
||
if(persistentSettings.nettype != currentWallet.nettype) {
|
||
console.log("Using network type from keys file")
|
||
persistentSettings.nettype = currentWallet.nettype;
|
||
}
|
||
|
||
// connect handlers
|
||
currentWallet.heightRefreshed.connect(onHeightRefreshed);
|
||
currentWallet.refreshed.connect(onWalletRefresh)
|
||
currentWallet.updated.connect(onWalletUpdate)
|
||
currentWallet.newBlock.connect(onWalletNewBlock)
|
||
currentWallet.moneySpent.connect(onWalletMoneySent)
|
||
currentWallet.moneyReceived.connect(onWalletMoneyReceived)
|
||
currentWallet.unconfirmedMoneyReceived.connect(onWalletUnconfirmedMoneyReceived)
|
||
currentWallet.transactionCreated.connect(onTransactionCreated)
|
||
currentWallet.connectionStatusChanged.connect(onWalletConnectionStatusChanged)
|
||
currentWallet.deviceButtonRequest.connect(onDeviceButtonRequest);
|
||
currentWallet.deviceButtonPressed.connect(onDeviceButtonPressed);
|
||
currentWallet.walletPassphraseNeeded.connect(onWalletPassphraseNeededWallet);
|
||
currentWallet.transactionCommitted.connect(onTransactionCommitted);
|
||
currentWallet.proxyAddress = Qt.binding(persistentSettings.getWalletProxyAddress);
|
||
middlePanel.paymentClicked.connect(handlePayment);
|
||
middlePanel.sweepUnmixableClicked.connect(handleSweepUnmixable);
|
||
middlePanel.getProofClicked.connect(handleGetProof);
|
||
middlePanel.checkProofClicked.connect(handleCheckProof);
|
||
|
||
persistentSettings.restore_height = currentWallet.walletCreationHeight;
|
||
|
||
console.log("Recovering from seed: ", persistentSettings.is_recovering)
|
||
console.log("restore Height", persistentSettings.restore_height)
|
||
|
||
if (persistentSettings.useRemoteNode) {
|
||
const remoteNode = remoteNodesModel.currentRemoteNode();
|
||
currentDaemonAddress = remoteNode.address;
|
||
currentWallet.setDaemonLogin(remoteNode.username, remoteNode.password);
|
||
} else {
|
||
currentDaemonAddress = localDaemonAddress;
|
||
}
|
||
|
||
console.log("initializing with daemon address: ", currentDaemonAddress)
|
||
currentWallet.initAsync(
|
||
currentDaemonAddress,
|
||
isTrustedDaemon(),
|
||
0,
|
||
persistentSettings.is_recovering,
|
||
persistentSettings.is_recovering_from_device,
|
||
persistentSettings.restore_height,
|
||
persistentSettings.getWalletProxyAddress());
|
||
|
||
// save wallet keys in case wallet settings have been changed in the init
|
||
currentWallet.setPassword(walletPassword);
|
||
}
|
||
|
||
function isTrustedDaemon() {
|
||
return !persistentSettings.useRemoteNode || remoteNodesModel.currentRemoteNode().trusted;
|
||
}
|
||
|
||
function usefulName(path) {
|
||
// arbitrary "short enough" limit
|
||
if (path.length < 32)
|
||
return path
|
||
return path.replace(/.*[\/\\]/, '').replace(/\.keys$/, '')
|
||
}
|
||
|
||
function getUnlockedBalance() {
|
||
if(!currentWallet){
|
||
return 0
|
||
}
|
||
return currentWallet.unlockedBalance()
|
||
}
|
||
|
||
function updateBalance() {
|
||
if (!currentWallet)
|
||
return;
|
||
|
||
var balance = "?.??";
|
||
var balanceU = "?.??";
|
||
if(!hideBalanceForced && !persistentSettings.hideBalance){
|
||
balance = walletManager.displayAmount(currentWallet.balance());
|
||
balanceU = walletManager.displayAmount(currentWallet.unlockedBalance());
|
||
}
|
||
|
||
if (persistentSettings.fiatPriceEnabled) {
|
||
appWindow.fiatApiUpdateBalance(balance);
|
||
}
|
||
|
||
leftPanel.minutesToUnlock = (balance !== balanceU) ? currentWallet.history.minutesToUnlock : "";
|
||
leftPanel.balanceString = balance
|
||
leftPanel.balanceUnlockedString = balanceU
|
||
if (middlePanel.state === "Account") {
|
||
middlePanel.accountView.balanceAllText = walletManager.displayAmount(appWindow.currentWallet.balanceAll()) + " XMR";
|
||
middlePanel.accountView.unlockedBalanceAllText = walletManager.displayAmount(appWindow.currentWallet.unlockedBalanceAll()) + " XMR";
|
||
}
|
||
}
|
||
|
||
function onUriHandler(uri){
|
||
if(uri.startsWith("monero://")){
|
||
var address = uri.substring("monero://".length);
|
||
|
||
var params = {}
|
||
if(address.length === 0) return;
|
||
var spl = address.split("?");
|
||
|
||
if(spl.length > 2) return;
|
||
if(spl.length >= 1) {
|
||
// parse additional params
|
||
address = spl[0];
|
||
|
||
if(spl.length === 2){
|
||
spl.shift();
|
||
var item = spl[0];
|
||
|
||
var _spl = item.split("&");
|
||
for (var param in _spl){
|
||
var _item = _spl[param];
|
||
if(!_item.indexOf("=") > 0) continue;
|
||
|
||
var __spl = _item.split("=");
|
||
if(__spl.length !== 2) continue;
|
||
|
||
params[__spl[0]] = __spl[1];
|
||
}
|
||
}
|
||
}
|
||
|
||
// Fill fields
|
||
middlePanel.transferView.sendTo(address, params["tx_payment_id"], params["tx_description"], params["tx_amount"]);
|
||
|
||
// Raise window
|
||
appWindow.raise();
|
||
appWindow.show();
|
||
}
|
||
}
|
||
|
||
function onWalletConnectionStatusChanged(status){
|
||
console.log("Wallet connection status changed " + status)
|
||
middlePanel.updateStatus();
|
||
leftPanel.networkStatus.connected = status
|
||
if (status == Wallet.ConnectionStatus_Disconnected) {
|
||
firstBlockSeen = 0;
|
||
}
|
||
|
||
// If wallet isnt connected, advanced wallet mode and no daemon is running - Ask
|
||
if (appWindow.walletMode >= 2 && !persistentSettings.useRemoteNode && !walletInitialized && disconnected) {
|
||
daemonManager.runningAsync(persistentSettings.nettype, function(running) {
|
||
if (!running) {
|
||
daemonManagerDialog.open();
|
||
}
|
||
});
|
||
}
|
||
// initialize transaction history once wallet is initialized first time;
|
||
if (!walletInitialized) {
|
||
currentWallet.history.refresh(currentWallet.currentSubaddressAccount)
|
||
walletInitialized = true
|
||
|
||
// check if daemon was already mining and add mining logo if true
|
||
if (!persistentSettings.useRemoteNode || persistentSettings.allowRemoteNodeMining) {
|
||
middlePanel.advancedView.miningView.update();
|
||
}
|
||
}
|
||
}
|
||
|
||
function onDeviceButtonRequest(code){
|
||
if (txConfirmationPopup.visible) {
|
||
txConfirmationPopup.bottomTextAnimation.running = true
|
||
if (!txConfirmationPopup.errorText.visible) {
|
||
txConfirmationPopup.bottomText.text = qsTr("Please confirm transaction on the device...") + translationManager.emptyString;
|
||
} else {
|
||
txConfirmationPopup.bottomText.text = qsTr("Please proceed to the device...") + translationManager.emptyString;
|
||
}
|
||
} else {
|
||
prevSplashText = splash.messageText;
|
||
splashDisplayedBeforeButtonRequest = splash.visible;
|
||
appWindow.showProcessingSplash(qsTr("Please proceed to the device..."));
|
||
}
|
||
}
|
||
|
||
function onDeviceButtonPressed(){
|
||
if (txConfirmationPopup.visible) {
|
||
txConfirmationPopup.bottomTextAnimation.running = false;
|
||
txConfirmationPopup.bottomText.text = qsTr("Signing transaction in the device...") + translationManager.emptyString;
|
||
} else {
|
||
if (splashDisplayedBeforeButtonRequest){
|
||
appWindow.showProcessingSplash(prevSplashText);
|
||
} else {
|
||
hideProcessingSplash();
|
||
}
|
||
}
|
||
}
|
||
|
||
function onWalletOpening(){
|
||
appWindow.showProcessingSplash(qsTr("Opening wallet ..."));
|
||
}
|
||
|
||
function onWalletOpened(wallet) {
|
||
hideProcessingSplash();
|
||
walletName = usefulName(wallet.path)
|
||
console.log(">>> wallet opened: " + wallet)
|
||
if (wallet.status !== Wallet.Status_Ok) {
|
||
// try to resolve common wallet cache errors automatically
|
||
switch (wallet.errorString) {
|
||
case "basic_string::_M_replace_aux":
|
||
case "std::bad_alloc":
|
||
walletManager.clearWalletCache(wallet.path);
|
||
walletPassword = passwordDialog.password;
|
||
appWindow.initialize();
|
||
console.error("Repairing wallet cache with error: ", wallet.errorString);
|
||
appWindow.showStatusMessage(qsTr("Repairing incompatible wallet cache. Resyncing wallet."),6);
|
||
return;
|
||
default:
|
||
// opening with password but password doesn't match
|
||
console.error("Error opening wallet with password: ", wallet.errorString);
|
||
passwordDialog.showError(qsTr("Couldn't open wallet: ") + wallet.errorString);
|
||
console.log("closing wallet async : " + wallet.address)
|
||
closeWallet();
|
||
return;
|
||
}
|
||
}
|
||
|
||
// wallet opened successfully, subscribing for wallet updates
|
||
connectWallet(wallet)
|
||
|
||
// Force switch normal view
|
||
rootItem.state = "normal";
|
||
|
||
// Process queued IPC command
|
||
if(typeof IPC !== "undefined" && IPC.queuedCmd().length > 0){
|
||
var queuedCmd = IPC.queuedCmd();
|
||
if(/^\w+:\/\/(.*)$/.test(queuedCmd)) appWindow.onUriHandler(queuedCmd); // uri
|
||
}
|
||
}
|
||
|
||
function onWalletPassphraseNeededManager(on_device){
|
||
onWalletPassphraseNeeded(walletManager, on_device)
|
||
}
|
||
|
||
function onWalletPassphraseNeededWallet(on_device){
|
||
onWalletPassphraseNeeded(currentWallet, on_device)
|
||
}
|
||
|
||
function onWalletPassphraseNeeded(handler, on_device){
|
||
hideProcessingSplash();
|
||
|
||
console.log(">>> wallet passphrase needed: ")
|
||
devicePassphraseDialog.onAcceptedCallback = function(passphrase) {
|
||
handler.onPassphraseEntered(passphrase, false, false);
|
||
appWindow.onWalletOpening();
|
||
}
|
||
devicePassphraseDialog.onWalletEntryCallback = function() {
|
||
handler.onPassphraseEntered("", true, false);
|
||
appWindow.onWalletOpening();
|
||
}
|
||
devicePassphraseDialog.onRejectedCallback = function() {
|
||
handler.onPassphraseEntered("", false, true);
|
||
appWindow.onWalletOpening();
|
||
}
|
||
|
||
devicePassphraseDialog.open(on_device)
|
||
}
|
||
|
||
function onWalletUpdate() {
|
||
console.log(">>> wallet updated")
|
||
updateBalance();
|
||
// Update history if new block found since last update
|
||
if(foundNewBlock) {
|
||
foundNewBlock = false;
|
||
console.log("New block found - updating history")
|
||
currentWallet.history.refresh(currentWallet.currentSubaddressAccount)
|
||
|
||
if(middlePanel.state == "History")
|
||
middlePanel.historyView.update();
|
||
}
|
||
}
|
||
|
||
function connectRemoteNode() {
|
||
console.log("connecting remote node");
|
||
|
||
const callback = function() {
|
||
persistentSettings.useRemoteNode = true;
|
||
const remoteNode = remoteNodesModel.currentRemoteNode();
|
||
currentDaemonAddress = remoteNode.address;
|
||
currentWallet.setDaemonLogin(remoteNode.username, remoteNode.password);
|
||
currentWallet.initAsync(
|
||
currentDaemonAddress,
|
||
isTrustedDaemon(),
|
||
0,
|
||
false,
|
||
false,
|
||
0,
|
||
persistentSettings.getWalletProxyAddress());
|
||
walletManager.setDaemonAddressAsync(currentDaemonAddress);
|
||
};
|
||
|
||
if (typeof daemonManager != "undefined" && daemonRunning) {
|
||
showDaemonIsRunningDialog(callback);
|
||
} else {
|
||
callback();
|
||
}
|
||
}
|
||
|
||
function disconnectRemoteNode() {
|
||
if (typeof currentWallet === "undefined" || currentWallet === null)
|
||
return;
|
||
|
||
console.log("disconnecting remote node");
|
||
persistentSettings.useRemoteNode = false;
|
||
currentDaemonAddress = localDaemonAddress
|
||
currentWallet.setDaemonLogin("", "");
|
||
currentWallet.initAsync(
|
||
currentDaemonAddress,
|
||
isTrustedDaemon(),
|
||
0,
|
||
false,
|
||
false,
|
||
0,
|
||
persistentSettings.getWalletProxyAddress());
|
||
walletManager.setDaemonAddressAsync(currentDaemonAddress);
|
||
firstBlockSeen = 0;
|
||
}
|
||
|
||
function onHeightRefreshed(bcHeight, dCurrentBlock, dTargetBlock) {
|
||
// Daemon fully synced
|
||
// TODO: implement onDaemonSynced or similar in wallet API and don't start refresh thread before daemon is synced
|
||
// targetBlock = currentBlock = 1 before network connection is established.
|
||
if (firstBlockSeen == 0 && dTargetBlock != 1) {
|
||
firstBlockSeen = dCurrentBlock;
|
||
}
|
||
daemonSynced = dCurrentBlock >= dTargetBlock && dTargetBlock != 1
|
||
walletSynced = bcHeight >= dTargetBlock
|
||
|
||
// Update progress bars
|
||
if(!daemonSynced) {
|
||
leftPanel.daemonProgressBar.updateProgress(dCurrentBlock,dTargetBlock, dTargetBlock-firstBlockSeen);
|
||
leftPanel.progressBar.updateProgress(0,dTargetBlock, dTargetBlock, qsTr("Waiting for daemon to sync"));
|
||
} else {
|
||
leftPanel.daemonProgressBar.updateProgress(dCurrentBlock,dTargetBlock, 0, qsTr("Daemon is synchronized (%1)").arg(dCurrentBlock.toFixed(0)));
|
||
if(walletSynced)
|
||
leftPanel.progressBar.updateProgress(bcHeight,dTargetBlock,dTargetBlock-bcHeight, qsTr("Wallet is synchronized"))
|
||
}
|
||
|
||
// Update wallet sync progress
|
||
leftPanel.isSyncing = !disconnected && !daemonSynced;
|
||
// Update transfer page status
|
||
middlePanel.updateStatus();
|
||
|
||
// Refresh is succesfull if blockchain height > 1
|
||
if (bcHeight > 1){
|
||
// recovering from seed is finished after first refresh
|
||
if(persistentSettings.is_recovering) {
|
||
persistentSettings.is_recovering = false
|
||
}
|
||
if (persistentSettings.is_recovering_from_device) {
|
||
persistentSettings.is_recovering_from_device = false;
|
||
}
|
||
}
|
||
|
||
// Update history on every refresh if it's empty
|
||
if(currentWallet.history.count == 0)
|
||
currentWallet.history.refresh(currentWallet.currentSubaddressAccount)
|
||
|
||
onWalletUpdate();
|
||
}
|
||
|
||
function onWalletRefresh() {
|
||
console.log(">>> wallet refreshed")
|
||
|
||
// Daemon connected
|
||
leftPanel.networkStatus.connected = currentWallet ? currentWallet.connected() : Wallet.ConnectionStatus_Disconnected
|
||
|
||
currentWallet.refreshHeightAsync();
|
||
}
|
||
|
||
function startDaemon(flags){
|
||
daemonStartStopInProgress = 1;
|
||
|
||
// Pause refresh while starting daemon
|
||
currentWallet.pauseRefresh();
|
||
|
||
const noSync = appWindow.walletMode === 0;
|
||
const bootstrapNodeAddress = persistentSettings.walletMode < 2 ? "auto" : persistentSettings.bootstrapNodeAddress
|
||
daemonManager.start(flags, persistentSettings.nettype, persistentSettings.blockchainDataDir, bootstrapNodeAddress, noSync, persistentSettings.pruneBlockchain);
|
||
}
|
||
|
||
function stopDaemon(callback, splash){
|
||
daemonStartStopInProgress = 2;
|
||
if (splash) {
|
||
appWindow.showProcessingSplash(qsTr("Waiting for daemon to stop..."));
|
||
}
|
||
p2poolManager.exit()
|
||
daemonManager.stopAsync(persistentSettings.nettype, function(result) {
|
||
daemonStartStopInProgress = 0;
|
||
if (splash) {
|
||
hideProcessingSplash();
|
||
}
|
||
callback(result);
|
||
});
|
||
}
|
||
|
||
function onDaemonStarted(){
|
||
console.log("daemon started");
|
||
daemonStartStopInProgress = 0;
|
||
if (currentWallet) {
|
||
currentWallet.connected(true);
|
||
// resume refresh
|
||
currentWallet.startRefresh();
|
||
}
|
||
// resume simplemode connection timer
|
||
appWindow.disconnectedEpoch = Utils.epoch();
|
||
}
|
||
function onDaemonStopped(){
|
||
if (currentWallet) {
|
||
currentWallet.connected(true);
|
||
}
|
||
}
|
||
|
||
function onDaemonStartFailure(error) {
|
||
console.log("daemon start failed");
|
||
daemonStartStopInProgress = 0;
|
||
// resume refresh
|
||
currentWallet.startRefresh();
|
||
informationPopup.title = qsTr("Daemon failed to start") + translationManager.emptyString;
|
||
informationPopup.text = error + ".\n\n" + qsTr("Please check your wallet and daemon log for errors. You can also try to start %1 manually.").arg((isWindows)? "monerod.exe" : "monerod")
|
||
informationPopup.icon = StandardIcon.Critical
|
||
informationPopup.onCloseCallback = null
|
||
informationPopup.open();
|
||
}
|
||
|
||
function onWalletNewBlock(blockHeight, targetHeight) {
|
||
// Update progress bar
|
||
var remaining = targetHeight - blockHeight;
|
||
if(blocksToSync < remaining) {
|
||
blocksToSync = remaining;
|
||
}
|
||
|
||
leftPanel.progressBar.updateProgress(blockHeight,targetHeight, blocksToSync);
|
||
|
||
// If wallet is syncing, daemon is already synced
|
||
leftPanel.daemonProgressBar.updateProgress(1,1,0,qsTr("Daemon is synchronized"));
|
||
|
||
foundNewBlock = true;
|
||
}
|
||
|
||
function onWalletMoneyReceived(txId, amount) {
|
||
// refresh transaction history here
|
||
console.log("Confirmed money found")
|
||
// history refresh is handled by walletUpdated
|
||
currentWallet.history.refresh(currentWallet.currentSubaddressAccount) // this will refresh model
|
||
currentWallet.subaddress.refresh(currentWallet.currentSubaddressAccount)
|
||
|
||
if(middlePanel.state == "History")
|
||
middlePanel.historyView.update();
|
||
}
|
||
|
||
function onWalletUnconfirmedMoneyReceived(txId, amount) {
|
||
// refresh history
|
||
console.log("unconfirmed money found")
|
||
currentWallet.history.refresh(currentWallet.currentSubaddressAccount);
|
||
|
||
if(middlePanel.state == "History")
|
||
middlePanel.historyView.update();
|
||
}
|
||
|
||
function onWalletMoneySent(txId, amount) {
|
||
// refresh transaction history here
|
||
console.log("monero sent found")
|
||
currentWallet.history.refresh(currentWallet.currentSubaddressAccount); // this will refresh model
|
||
|
||
if(middlePanel.state == "History")
|
||
middlePanel.historyView.update();
|
||
}
|
||
|
||
function walletsFound() {
|
||
if (persistentSettings.wallet_path.length > 0) {
|
||
if(isIOS)
|
||
return walletManager.walletExists(appWindow.accountsDir + persistentSettings.wallet_path);
|
||
else
|
||
return walletManager.walletExists(persistentSettings.wallet_path);
|
||
}
|
||
return false;
|
||
}
|
||
|
||
function onTransactionCreated(pendingTransaction, addresses, paymentId, mixinCount) {
|
||
console.log("Transaction created");
|
||
txConfirmationPopup.bottomText.text = "";
|
||
transaction = pendingTransaction;
|
||
// validate address;
|
||
if (transaction.status !== PendingTransaction.Status_Ok) {
|
||
console.error("Can't create transaction: ", transaction.errorString);
|
||
if (currentWallet.connected() == Wallet.ConnectionStatus_WrongVersion) {
|
||
txConfirmationPopup.errorText.text = qsTr("Can't create transaction: Wrong daemon version: ") + transaction.errorString
|
||
} else {
|
||
txConfirmationPopup.errorText.text = qsTr("Can't create transaction: ") + transaction.errorString
|
||
}
|
||
// deleting transaction object, we don't want memleaks
|
||
currentWallet.disposeTransaction(transaction);
|
||
|
||
} else if (transaction.txCount == 0) {
|
||
console.error("Can't create transaction: ", transaction.errorString);
|
||
txConfirmationPopup.errorText.text = qsTr("No unmixable outputs to sweep") + translationManager.emptyString
|
||
// deleting transaction object, we don't want memleaks
|
||
currentWallet.disposeTransaction(transaction);
|
||
} else {
|
||
console.log("Transaction created, amount: " + walletManager.displayAmount(transaction.amount)
|
||
+ ", fee: " + walletManager.displayAmount(transaction.fee));
|
||
|
||
// here we update txConfirmationPopup
|
||
txConfirmationPopup.transactionAmount = Utils.removeTrailingZeros(walletManager.displayAmount(transaction.amount));
|
||
txConfirmationPopup.transactionFee = Utils.removeTrailingZeros(walletManager.displayAmount(transaction.fee));
|
||
txConfirmationPopup.confirmButton.text = viewOnly ? qsTr("Save as file") : qsTr("Confirm") + translationManager.emptyString;
|
||
txConfirmationPopup.confirmButton.rightIcon = viewOnly ? "" : "qrc:///images/rightArrow.png"
|
||
}
|
||
}
|
||
|
||
function getDisplayAmountTotal(recipients) {
|
||
const amounts = recipients.map(function (recipient) {
|
||
return recipient.amount;
|
||
});
|
||
const total = walletManager.amountsSumFromStrings(amounts);
|
||
return Utils.removeTrailingZeros(walletManager.displayAmount(total));
|
||
}
|
||
|
||
// called on "transfer"
|
||
function handlePayment(recipients, paymentId, mixinCount, priority, description, createFile) {
|
||
console.log("Creating transaction: ")
|
||
console.log("\trecipients: ", recipients,
|
||
", payment_id: ", paymentId,
|
||
", mixins: ", mixinCount,
|
||
", priority: ", priority,
|
||
", description: ", description);
|
||
|
||
const recipientAll = recipients.find(function (recipient) {
|
||
return recipient.amount == "(all)";
|
||
});
|
||
if (recipientAll && recipients.length > 1) {
|
||
throw "Sending all requires one destination address";
|
||
}
|
||
|
||
txConfirmationPopup.bottomTextAnimation.running = false;
|
||
txConfirmationPopup.bottomText.text = qsTr("Creating transaction...") + translationManager.emptyString;
|
||
txConfirmationPopup.recipients = recipients;
|
||
txConfirmationPopup.transactionAmount = recipientAll ? "(all)" : getDisplayAmountTotal(recipients);
|
||
txConfirmationPopup.transactionPriority = priority;
|
||
txConfirmationPopup.transactionDescription = description;
|
||
txConfirmationPopup.open();
|
||
|
||
if (recipientAll) {
|
||
currentWallet.createTransactionAllAsync(recipientAll.address, paymentId, mixinCount, priority);
|
||
} else {
|
||
const addresses = recipients.map(function (recipient) {
|
||
return recipient.address;
|
||
});
|
||
const amountsxmr = recipients.map(function (recipient) {
|
||
return recipient.amount;
|
||
});
|
||
currentWallet.createTransactionAsync(addresses, paymentId, amountsxmr, mixinCount, priority);
|
||
}
|
||
}
|
||
|
||
//Choose where to save transaction
|
||
FileDialog {
|
||
id: saveTxDialog
|
||
title: "Please choose a location"
|
||
folder: "file://" + appWindow.accountsDir
|
||
selectExisting: false;
|
||
|
||
onAccepted: {
|
||
handleTransactionConfirmed()
|
||
}
|
||
onRejected: {
|
||
// do nothing
|
||
|
||
}
|
||
|
||
}
|
||
|
||
|
||
function handleSweepUnmixable() {
|
||
console.log("Creating transaction: ")
|
||
|
||
txConfirmationPopup.sweepUnmixable = true;
|
||
transaction = currentWallet.createSweepUnmixableTransaction();
|
||
if (transaction.status !== PendingTransaction.Status_Ok) {
|
||
console.error("Can't create transaction: ", transaction.errorString);
|
||
txConfirmationPopup.errorText.text = qsTr("Can't create transaction: ") + transaction.errorString + translationManager.emptyString
|
||
// deleting transaction object, we don't want memleaks
|
||
currentWallet.disposeTransaction(transaction);
|
||
|
||
} else if (transaction.txCount == 0) {
|
||
console.error("No unmixable outputs to sweep");
|
||
txConfirmationPopup.errorText.text = qsTr("No unmixable outputs to sweep") + translationManager.emptyString
|
||
// deleting transaction object, we don't want memleaks
|
||
currentWallet.disposeTransaction(transaction);
|
||
} else {
|
||
console.log("Transaction created, amount: " + walletManager.displayAmount(transaction.amount)
|
||
+ ", fee: " + walletManager.displayAmount(transaction.fee));
|
||
txConfirmationPopup.transactionAmount = Utils.removeTrailingZeros(walletManager.displayAmount(transaction.amount));
|
||
txConfirmationPopup.transactionFee = Utils.removeTrailingZeros(walletManager.displayAmount(transaction.fee));
|
||
// committing transaction
|
||
}
|
||
txConfirmationPopup.open();
|
||
}
|
||
|
||
// called after user confirms transaction
|
||
function handleTransactionConfirmed(fileName) {
|
||
// View only wallet - we save the tx
|
||
if(viewOnly){
|
||
// No file specified - abort
|
||
if(!saveTxDialog.fileUrl) {
|
||
currentWallet.disposeTransaction(transaction)
|
||
return;
|
||
}
|
||
|
||
var path = walletManager.urlToLocalPath(saveTxDialog.fileUrl)
|
||
|
||
// Store to file
|
||
transaction.setFilename(path);
|
||
}
|
||
appWindow.showProcessingSplash(qsTr("Sending transaction ..."));
|
||
currentWallet.commitTransactionAsync(transaction);
|
||
}
|
||
|
||
function onTransactionCommitted(success, transaction, txid) {
|
||
hideProcessingSplash();
|
||
if (!success) {
|
||
console.log("Error committing transaction: " + transaction.errorString);
|
||
informationPopup.title = qsTr("Error") + translationManager.emptyString
|
||
informationPopup.text = qsTr("Couldn't send the money: ") + transaction.errorString
|
||
informationPopup.icon = StandardIcon.Critical
|
||
informationPopup.onCloseCallback = null;
|
||
informationPopup.open();
|
||
} else {
|
||
if (txConfirmationPopup.transactionDescription.length > 0) {
|
||
for (var i = 0; i < txid.length; ++i)
|
||
currentWallet.setUserNote(txid[i], txConfirmationPopup.transactionDescription);
|
||
}
|
||
|
||
// Clear tx fields
|
||
middlePanel.transferView.clearFields()
|
||
txConfirmationPopup.clearFields()
|
||
successfulTxPopup.open(txid)
|
||
}
|
||
currentWallet.refresh()
|
||
currentWallet.disposeTransaction(transaction)
|
||
currentWallet.storeAsync(function(success) {
|
||
if (!success) {
|
||
appWindow.showStatusMessage(qsTr("Failed to store the wallet"), 3);
|
||
}
|
||
});
|
||
}
|
||
|
||
function doSearchInHistory(searchTerm) {
|
||
middlePanel.searchInHistory(searchTerm);
|
||
leftPanel.selectItem(middlePanel.state)
|
||
}
|
||
|
||
// called on "getProof"
|
||
function handleGetProof(txid, address, message, amount) {
|
||
if (amount !== null && amount.length > 0) {
|
||
var result = currentWallet.getReserveProof(false, currentWallet.currentSubaddressAccount, walletManager.amountFromString(amount), message)
|
||
txProofComputed(null, result)
|
||
} else {
|
||
console.log("Getting payment proof: ")
|
||
console.log("\ttxid: ", txid,
|
||
", address: ", address,
|
||
", message: ", message);
|
||
function spendProofFallback(txid, result){
|
||
if (!result || result.indexOf("error|") === 0) {
|
||
currentWallet.getSpendProofAsync(txid, message, txProofComputed);
|
||
} else {
|
||
txProofComputed(txid, result);
|
||
}
|
||
}
|
||
if (address.length > 0)
|
||
currentWallet.getTxProofAsync(txid, address, message, spendProofFallback);
|
||
else
|
||
spendProofFallback(txid, null);
|
||
}
|
||
informationPopup.open()
|
||
}
|
||
|
||
function txProofComputed(txid, result){
|
||
if (result.indexOf("error|") === 0) {
|
||
var errorString = result.split("|")[1];
|
||
informationPopup.text = qsTr("Couldn't generate a proof because of the following reason: \n") + errorString + translationManager.emptyString;
|
||
informationPopup.icon = StandardIcon.Critical;
|
||
} else {
|
||
informationPopup.text = result;
|
||
informationPopup.icon = StandardIcon.Critical;
|
||
}
|
||
}
|
||
|
||
// called on "checkProof"
|
||
function handleCheckProof(txid, address, message, signature) {
|
||
console.log("Checking payment proof: ")
|
||
console.log("\ttxid: ", txid,
|
||
", address: ", address,
|
||
", message: ", message,
|
||
", signature: ", signature);
|
||
|
||
var result;
|
||
var isReserveProof = signature.indexOf("ReserveProofV") === 0;
|
||
if (address.length > 0 && !isReserveProof) {
|
||
result = currentWallet.checkTxProof(txid, address, message, signature);
|
||
}
|
||
else if (isReserveProof) {
|
||
result = currentWallet.checkReserveProof(address, message, signature);
|
||
}
|
||
else {
|
||
result = currentWallet.checkSpendProof(txid, message, signature);
|
||
}
|
||
var results = result.split("|");
|
||
if (address.length > 0 && results.length == 5 && results[0] === "true" && !isReserveProof) {
|
||
var good = results[1] === "true";
|
||
var received = results[2];
|
||
var in_pool = results[3] === "true";
|
||
var confirmations = results[4];
|
||
|
||
informationPopup.title = qsTr("Payment proof check") + translationManager.emptyString;
|
||
informationPopup.icon = StandardIcon.Information
|
||
if (!good) {
|
||
informationPopup.text = qsTr("Bad signature");
|
||
informationPopup.icon = StandardIcon.Critical;
|
||
} else if (received > 0) {
|
||
if (in_pool) {
|
||
informationPopup.text = qsTr("This address received %1 monero, but the transaction is not yet mined").arg(walletManager.displayAmount(received));
|
||
}
|
||
else {
|
||
informationPopup.text = qsTr("This address received %1 monero, with %2 confirmation(s).").arg(walletManager.displayAmount(received)).arg(confirmations);
|
||
}
|
||
}
|
||
else {
|
||
informationPopup.text = qsTr("This address received nothing");
|
||
}
|
||
}
|
||
else if (results.length == 2 && results[0] === "true") {
|
||
var good = results[1] === "true";
|
||
informationPopup.title = qsTr("Payment proof check") + translationManager.emptyString;
|
||
informationPopup.icon = good ? StandardIcon.Information : StandardIcon.Critical;
|
||
informationPopup.text = good ? qsTr("Good signature") : qsTr("Bad signature");
|
||
}
|
||
else if (isReserveProof && results[0] === "true") {
|
||
var good = results[1] === "true";
|
||
informationPopup.title = qsTr("Reserve proof check") + translationManager.emptyString;
|
||
informationPopup.icon = good ? StandardIcon.Information : StandardIcon.Critical;
|
||
informationPopup.text = good ? qsTr("Good signature on %1 total and %2 spent.").arg(results[2]).arg(results[3]) : qsTr("Bad signature");
|
||
}
|
||
else {
|
||
informationPopup.title = qsTr("Error") + translationManager.emptyString;
|
||
informationPopup.text = currentWallet.errorString;
|
||
informationPopup.icon = StandardIcon.Critical
|
||
}
|
||
informationPopup.onCloseCallback = null
|
||
informationPopup.open()
|
||
}
|
||
|
||
function showProcessingSplash(message) {
|
||
console.log("Displaying processing splash")
|
||
if (typeof message != 'undefined') {
|
||
splash.messageText = message
|
||
}
|
||
|
||
leftPanel.enabled = false;
|
||
middlePanel.enabled = false;
|
||
titleBar.enabled = false;
|
||
splash.show();
|
||
}
|
||
|
||
function hideProcessingSplash() {
|
||
console.log("Hiding processing splash")
|
||
splash.close();
|
||
|
||
if (!passwordDialog.visible) {
|
||
leftPanel.enabled = true
|
||
middlePanel.enabled = true
|
||
titleBar.enabled = true
|
||
}
|
||
}
|
||
|
||
// close wallet and show wizard
|
||
function showWizard(){
|
||
walletInitialized = false;
|
||
closeWallet(function() {
|
||
wizard.restart();
|
||
wizard.wizardState = "wizardHome";
|
||
rootItem.state = "wizard"
|
||
// reset balance, clear spendable funds message
|
||
clearMoneroCardLabelText();
|
||
leftPanel.minutesToUnlock = "";
|
||
// reset fields
|
||
middlePanel.addressBookView.clearFields();
|
||
middlePanel.transferView.clearFields();
|
||
middlePanel.receiveView.clearFields();
|
||
middlePanel.historyView.clearFields();
|
||
// disable timers
|
||
userInActivityTimer.running = false;
|
||
});
|
||
}
|
||
|
||
objectName: "appWindow"
|
||
visible: true
|
||
width: screenAvailableWidth > 980
|
||
? 980
|
||
: Math.min(screenAvailableWidth, 800)
|
||
height: screenAvailableHeight > maxWindowHeight
|
||
? maxWindowHeight
|
||
: Math.min(screenAvailableHeight, 700)
|
||
color: MoneroComponents.Style.appWindowBackgroundColor
|
||
flags: persistentSettings.customDecorations ? Windows.flagsCustomDecorations : Windows.flags
|
||
|
||
Timer {
|
||
id: fiatPriceTimer
|
||
interval: 1000 * 60;
|
||
running: persistentSettings.fiatPriceEnabled && currentWallet !== undefined
|
||
repeat: true
|
||
onTriggered: appWindow.fiatApiRefresh()
|
||
triggeredOnStart: true
|
||
}
|
||
|
||
function fiatApiParseTicker(url, resp, currency){
|
||
// parse & validate incoming JSON
|
||
if(url.startsWith("https://api.kraken.com/0/")){
|
||
if(resp.hasOwnProperty("error") && resp.error.length > 0 || !resp.hasOwnProperty("result")){
|
||
appWindow.fiatApiError("Kraken API has error(s)");
|
||
return;
|
||
}
|
||
|
||
var key = currency === "xmreur" ? "XXMRZEUR" : "XXMRZUSD";
|
||
var ticker = resp.result[key]["c"][0];
|
||
return ticker;
|
||
} else if(url.startsWith("https://api.coingecko.com/api/v3/")){
|
||
var key = currency === "xmreur" ? "eur" : "usd";
|
||
if(!resp.hasOwnProperty("monero") || !resp["monero"].hasOwnProperty(key)){
|
||
appWindow.fiatApiError("Coingecko API has error(s)");
|
||
return;
|
||
}
|
||
return resp["monero"][key];
|
||
} else if(url.startsWith("https://min-api.cryptocompare.com/data/")){
|
||
var key = currency === "xmreur" ? "EUR" : "USD";
|
||
if(!resp.hasOwnProperty(key)){
|
||
appWindow.fiatApiError("cryptocompare API has error(s)");
|
||
return;
|
||
}
|
||
return resp[key];
|
||
}
|
||
}
|
||
|
||
function fiatApiGetCurrency(url) {
|
||
var apis = appWindow.fiatPriceAPIs;
|
||
for (var api in apis){
|
||
if (!apis.hasOwnProperty(api))
|
||
continue;
|
||
|
||
for (var cur in apis[api]){
|
||
if(!apis[api].hasOwnProperty(cur))
|
||
continue;
|
||
|
||
if (apis[api][cur] === url) {
|
||
return cur;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
function fiatApiJsonReceived(url, resp, error) {
|
||
if (error) {
|
||
appWindow.fiatApiError(error);
|
||
return;
|
||
}
|
||
|
||
try {
|
||
resp = JSON.parse(resp);
|
||
} catch (e) {
|
||
appWindow.fiatApiError("bad JSON: " + e);
|
||
return;
|
||
}
|
||
|
||
// handle incoming JSON, set ticker
|
||
var currency = appWindow.fiatApiGetCurrency(url);
|
||
if(typeof currency == "undefined"){
|
||
appWindow.fiatApiError("could not get currency");
|
||
return;
|
||
}
|
||
|
||
var ticker = appWindow.fiatApiParseTicker(url, resp, currency);
|
||
if(ticker <= 0){
|
||
appWindow.fiatApiError("could not get ticker");
|
||
return;
|
||
}
|
||
|
||
appWindow.fiatPrice = ticker;
|
||
|
||
appWindow.updateBalance();
|
||
}
|
||
|
||
function fiatApiRefresh(){
|
||
// trigger API call
|
||
if(!persistentSettings.fiatPriceEnabled)
|
||
return;
|
||
|
||
var userProvider = persistentSettings.fiatPriceProvider;
|
||
if(!appWindow.fiatPriceAPIs.hasOwnProperty(userProvider)){
|
||
appWindow.fiatApiError("provider \"" + userProvider + "\" not implemented");
|
||
return;
|
||
}
|
||
|
||
var provider = appWindow.fiatPriceAPIs[userProvider];
|
||
var userCurrency = persistentSettings.fiatPriceCurrency;
|
||
if(!provider.hasOwnProperty(userCurrency)){
|
||
appWindow.fiatApiError("currency \"" + userCurrency + "\" not implemented");
|
||
}
|
||
|
||
var url = provider[userCurrency];
|
||
network.getJSON(url, fiatApiJsonReceived);
|
||
}
|
||
|
||
function fiatApiCurrencySymbol() {
|
||
switch (persistentSettings.fiatPriceCurrency) {
|
||
case "xmrusd":
|
||
return "USD";
|
||
case "xmreur":
|
||
return "EUR";
|
||
default:
|
||
console.error("unsupported currency", persistentSettings.fiatPriceCurrency);
|
||
return "UNSUPPORTED";
|
||
}
|
||
}
|
||
|
||
function fiatApiConvertToFiat(amount) {
|
||
const ticker = appWindow.fiatPrice;
|
||
if(ticker <= 0){
|
||
fiatApiError("Invalid ticker value: " + ticker);
|
||
return "?.??";
|
||
}
|
||
return (amount * ticker).toFixed(2);
|
||
}
|
||
|
||
function fiatApiConvertToXMR(amount) {
|
||
const ticker = appWindow.fiatPrice;
|
||
if(ticker <= 0){
|
||
fiatApiError("Invalid ticker value: " + ticker);
|
||
return "?.??";
|
||
}
|
||
return (amount / ticker).toFixed(12);
|
||
}
|
||
|
||
function fiatApiUpdateBalance(balance){
|
||
// update balance card
|
||
var bFiat = "?.??"
|
||
if (!hideBalanceForced && !persistentSettings.hideBalance) {
|
||
bFiat = fiatApiConvertToFiat(balance);
|
||
}
|
||
leftPanel.balanceFiatString = bFiat;
|
||
}
|
||
|
||
function fiatApiError(msg){
|
||
console.log("fiatPriceError: " + msg);
|
||
}
|
||
|
||
Component.onCompleted: {
|
||
if (screenAvailableWidth > width) {
|
||
x = (screenAvailableWidth - width) / 2;
|
||
}
|
||
if (screenAvailableHeight > height) {
|
||
y = (screenAvailableHeight - height) / 2;
|
||
}
|
||
|
||
translationManager.setLanguage(persistentSettings.locale.split("_")[0]);
|
||
|
||
applyWalletMode(persistentSettings.walletMode);
|
||
|
||
//
|
||
walletManager.walletOpened.connect(onWalletOpened);
|
||
walletManager.deviceButtonRequest.connect(onDeviceButtonRequest);
|
||
walletManager.deviceButtonPressed.connect(onDeviceButtonPressed);
|
||
walletManager.checkUpdatesComplete.connect(onWalletCheckUpdatesComplete);
|
||
walletManager.walletPassphraseNeeded.connect(onWalletPassphraseNeededManager);
|
||
IPC.uriHandler.connect(onUriHandler);
|
||
|
||
if(typeof daemonManager != "undefined") {
|
||
daemonManager.daemonStarted.connect(onDaemonStarted);
|
||
daemonManager.daemonStartFailure.connect(onDaemonStartFailure);
|
||
daemonManager.daemonStopped.connect(onDaemonStopped);
|
||
}
|
||
|
||
// Connect app exit to qml window exit handling
|
||
mainApp.closing.connect(appWindow.close);
|
||
|
||
if( appWindow.qrScannerEnabled ){
|
||
console.log("qrScannerEnabled : load component QRCodeScanner");
|
||
var component = Qt.createComponent("components/QRCodeScanner.qml");
|
||
if (component.status == Component.Ready) {
|
||
console.log("Camera component ready");
|
||
cameraUi = component.createObject(appWindow);
|
||
} else {
|
||
console.log("component not READY !!!");
|
||
appWindow.qrScannerEnabled = false;
|
||
}
|
||
} else console.log("qrScannerEnabled disabled");
|
||
|
||
if(!walletsFound()) {
|
||
wizard.wizardState = "wizardLanguage";
|
||
rootItem.state = "wizard"
|
||
} else {
|
||
wizard.wizardState = "wizardHome";
|
||
rootItem.state = "normal"
|
||
logger.resetLogFilePath(persistentSettings.portable);
|
||
openWallet("wizard");
|
||
}
|
||
|
||
const desktopEntryEnabled = (typeof builtWithDesktopEntry != "undefined") && builtWithDesktopEntry;
|
||
if (persistentSettings.askDesktopShortcut && !persistentSettings.portable && desktopEntryEnabled) {
|
||
persistentSettings.askDesktopShortcut = false;
|
||
|
||
if (isTails) {
|
||
oshelper.createDesktopEntry();
|
||
} else if (isLinux) {
|
||
confirmationDialog.title = qsTr("Desktop entry") + translationManager.emptyString;
|
||
confirmationDialog.text = qsTr("Would you like to register Monero GUI Desktop entry?") + translationManager.emptyString;
|
||
confirmationDialog.icon = StandardIcon.Question;
|
||
confirmationDialog.cancelText = qsTr("No") + translationManager.emptyString;
|
||
confirmationDialog.okText = qsTr("Yes") + translationManager.emptyString;
|
||
confirmationDialog.onAcceptedCallback = function() {
|
||
oshelper.createDesktopEntry();
|
||
};
|
||
confirmationDialog.onRejectedCallback = null;
|
||
confirmationDialog.open();
|
||
}
|
||
}
|
||
|
||
remoteNodesModel.initialize();
|
||
}
|
||
|
||
MoneroSettings {
|
||
id: persistentSettings
|
||
fileName: {
|
||
if(isTails && tailsUsePersistence)
|
||
return homePath + "/Persistent/Monero/monero-core.conf";
|
||
return "";
|
||
}
|
||
|
||
property bool askDesktopShortcut: isLinux
|
||
property bool askStopLocalNode: true
|
||
property string language: 'English (US)'
|
||
property string language_wallet: 'English'
|
||
property string locale: 'en_US'
|
||
property string account_name
|
||
property string wallet_path
|
||
property bool allow_background_mining : false
|
||
property bool allow_p2pool_mining : false
|
||
property bool allowRemoteNodeMining : false
|
||
property bool miningIgnoreBattery : true
|
||
property int miningModeSelected: 0
|
||
property int chainDropdownSelected: 0
|
||
property var nettype: NetworkType.MAINNET
|
||
property int restore_height : 0
|
||
property bool is_trusted_daemon : false // TODO: drop after v0.17.2.0 release
|
||
property bool is_recovering : false
|
||
property bool is_recovering_from_device : false
|
||
property bool customDecorations : true
|
||
property string daemonFlags
|
||
property string p2poolFlags
|
||
property int logLevel: 0
|
||
property string logCategories: ""
|
||
property string daemonUsername: "" // TODO: drop after v0.17.2.0 release
|
||
property string daemonPassword: "" // TODO: drop after v0.17.2.0 release
|
||
property bool transferShowAdvanced: false
|
||
property bool receiveShowAdvanced: false
|
||
property bool historyShowAdvanced: false
|
||
property bool historyHumanDates: true
|
||
property string blockchainDataDir: ""
|
||
property bool useRemoteNode: false
|
||
property string remoteNodeAddress: "" // TODO: drop after v0.17.2.0 release
|
||
property string remoteNodesSerialized: JSON.stringify({
|
||
selected: 0,
|
||
nodes: remoteNodeAddress != ""
|
||
? [{
|
||
address: remoteNodeAddress,
|
||
username: daemonUsername,
|
||
password: daemonPassword,
|
||
trusted: is_trusted_daemon,
|
||
}]
|
||
: [],
|
||
})
|
||
property string bootstrapNodeAddress: ""
|
||
property bool segregatePreForkOutputs: true
|
||
property bool keyReuseMitigation2: true
|
||
property int segregationHeight: 0
|
||
property int kdfRounds: 1
|
||
property bool displayWalletNameInTitleBar: true
|
||
property bool hideBalance: false
|
||
property bool askPasswordBeforeSending: true
|
||
property bool lockOnUserInActivity: true
|
||
property int walletMode: 2
|
||
property int lockOnUserInActivityInterval: 10 // minutes
|
||
property bool blackTheme: MoneroComponents.Style.blackTheme
|
||
property bool checkForUpdates: true
|
||
property bool autosave: true
|
||
property int autosaveMinutes: 10
|
||
property bool pruneBlockchain: false
|
||
|
||
property bool fiatPriceEnabled: false
|
||
property bool fiatPriceToggle: false
|
||
property string fiatPriceProvider: "kraken"
|
||
property string fiatPriceCurrency: "xmrusd"
|
||
|
||
property string proxyAddress: "127.0.0.1:9050"
|
||
property bool proxyEnabled: isTails
|
||
function getProxyAddress() {
|
||
if ((socksProxyFlagSet && socksProxyFlag == "") || !proxyEnabled) {
|
||
return "";
|
||
}
|
||
var proxyAddressSetOrForced = socksProxyFlagSet ? socksProxyFlag : proxyAddress;
|
||
if (proxyAddressSetOrForced == "") {
|
||
return "127.0.0.1:0";
|
||
}
|
||
return proxyAddressSetOrForced;
|
||
}
|
||
function getWalletProxyAddress() {
|
||
if (!useRemoteNode) {
|
||
return "";
|
||
}
|
||
return getProxyAddress();
|
||
}
|
||
|
||
Component.onCompleted: {
|
||
MoneroComponents.Style.blackTheme = persistentSettings.blackTheme
|
||
}
|
||
}
|
||
|
||
ListModel {
|
||
id: remoteNodesModel
|
||
|
||
property int selected: 0
|
||
|
||
signal store()
|
||
|
||
function initialize() {
|
||
try {
|
||
const remoteNodes = JSON.parse(persistentSettings.remoteNodesSerialized);
|
||
for (var index = 0; index < remoteNodes.nodes.length; ++index) {
|
||
const remoteNode = remoteNodes.nodes[index];
|
||
remoteNodesModel.append(remoteNode);
|
||
}
|
||
selected = remoteNodes.selected % remoteNodesModel.count || 0;
|
||
} catch (e) {
|
||
console.error('failed to parse remoteNodesSerialized', e);
|
||
}
|
||
|
||
store.connect(function() {
|
||
var remoteNodes = [];
|
||
for (var index = 0; index < remoteNodesModel.count; ++index) {
|
||
remoteNodes.push(remoteNodesModel.get(index));
|
||
}
|
||
persistentSettings.remoteNodesSerialized = JSON.stringify({
|
||
selected: selected,
|
||
nodes: remoteNodes
|
||
});
|
||
});
|
||
}
|
||
|
||
function appendIfNotExists(newRemoteNode) {
|
||
for (var index = 0; index < remoteNodesModel.count; ++index) {
|
||
const remoteNode = remoteNodesModel.get(index);
|
||
if (remoteNode.address == newRemoteNode.address &&
|
||
remoteNode.username == newRemoteNode.username &&
|
||
remoteNode.password == newRemoteNode.password &&
|
||
remoteNode.trusted == newRemoteNode.trusted) {
|
||
return index;
|
||
}
|
||
}
|
||
remoteNodesModel.append(newRemoteNode);
|
||
return remoteNodesModel.count - 1;
|
||
}
|
||
|
||
function applyRemoteNode(index) {
|
||
selected = index;
|
||
const remoteNode = currentRemoteNode();
|
||
persistentSettings.useRemoteNode = true;
|
||
if (currentWallet) {
|
||
currentWallet.setDaemonLogin(remoteNode.username, remoteNode.password);
|
||
currentWallet.setTrustedDaemon(remoteNode.trusted);
|
||
appWindow.connectRemoteNode();
|
||
}
|
||
}
|
||
|
||
function currentRemoteNode() {
|
||
if (selected < remoteNodesModel.count) {
|
||
return remoteNodesModel.get(selected);
|
||
}
|
||
return {
|
||
address: "",
|
||
username: "",
|
||
password: "",
|
||
trusted: false,
|
||
};
|
||
}
|
||
|
||
function removeSelectNextIfNeeded(index) {
|
||
remoteNodesModel.remove(index);
|
||
if (selected == index) {
|
||
applyRemoteNode(selected % remoteNodesModel.count || 0);
|
||
} else if (selected > index) {
|
||
selected = selected - 1;
|
||
}
|
||
}
|
||
|
||
onCountChanged: store()
|
||
onDataChanged: store()
|
||
onSelectedChanged: store()
|
||
}
|
||
|
||
// Information dialog
|
||
StandardDialog {
|
||
// dynamically change onclose handler
|
||
property var onCloseCallback
|
||
id: informationPopup
|
||
anchors.fill: parent
|
||
z: parent.z + 1
|
||
cancelVisible: false
|
||
onAccepted: {
|
||
if (onCloseCallback) {
|
||
onCloseCallback()
|
||
}
|
||
}
|
||
}
|
||
|
||
// Transaction confirmation popup
|
||
TxConfirmationDialog {
|
||
// dynamically change onclose handler
|
||
id: txConfirmationPopup
|
||
z: parent.z + 1
|
||
onAccepted: {
|
||
var handleAccepted = function() {
|
||
// Save transaction to file if view only wallet
|
||
if (viewOnly) {
|
||
saveTxDialog.open();
|
||
} else {
|
||
handleTransactionConfirmed()
|
||
}
|
||
}
|
||
close();
|
||
passwordDialog.onAcceptedCallback = function() {
|
||
if(walletPassword === passwordDialog.password){
|
||
handleAccepted()
|
||
} else {
|
||
passwordDialog.showError(qsTr("Wrong password") + translationManager.emptyString);
|
||
}
|
||
}
|
||
passwordDialog.onRejectedCallback = null;
|
||
if(!persistentSettings.askPasswordBeforeSending) {
|
||
handleAccepted()
|
||
} else {
|
||
passwordDialog.open(
|
||
"",
|
||
"",
|
||
(appWindow.viewOnly ? qsTr("Save transaction file") : qsTr("Send transaction")) + translationManager.emptyString,
|
||
appWindow.viewOnly ? "" : FontAwesome.arrowCircleRight);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Transaction successfully sent popup
|
||
SuccessfulTxDialog {
|
||
id: successfulTxPopup
|
||
z: parent.z + 1
|
||
}
|
||
|
||
StandardDialog {
|
||
z: parent.z + 1
|
||
id: confirmationDialog
|
||
anchors.fill: parent
|
||
property var onAcceptedCallback
|
||
property var onRejectedCallback
|
||
onAccepted: {
|
||
if (onAcceptedCallback)
|
||
onAcceptedCallback()
|
||
}
|
||
onRejected: {
|
||
if (onRejectedCallback)
|
||
onRejectedCallback();
|
||
}
|
||
}
|
||
|
||
MoneroComponents.UpdateDialog {
|
||
id: updateDialog
|
||
|
||
allowed: !passwordDialog.visible && !inputDialog.visible && !splash.visible
|
||
x: (parent.width - width) / 2
|
||
y: (parent.height - height) / 2
|
||
}
|
||
|
||
MoneroComponents.RemoteNodeDialog {
|
||
id: remoteNodeDialog
|
||
}
|
||
|
||
// Choose blockchain folder
|
||
FileDialog {
|
||
id: blockchainFileDialog
|
||
property string directory: ""
|
||
signal changed();
|
||
|
||
title: "Please choose a folder"
|
||
selectFolder: true
|
||
folder: "file://" + persistentSettings.blockchainDataDir
|
||
|
||
onRejected: console.log("data dir selection canceled")
|
||
onAccepted: {
|
||
var dataDir = walletManager.urlToLocalPath(blockchainFileDialog.fileUrl)
|
||
var validator = daemonManager.validateDataDir(dataDir);
|
||
if(validator.valid) {
|
||
persistentSettings.blockchainDataDir = dataDir;
|
||
} else {
|
||
confirmationDialog.title = qsTr("Warning") + translationManager.emptyString;
|
||
confirmationDialog.text = "";
|
||
if(validator.readOnly)
|
||
confirmationDialog.text += qsTr("Error: Filesystem is read only") + "\n\n"
|
||
if(validator.storageAvailable < estimatedBlockchainSize)
|
||
confirmationDialog.text += qsTr("Warning: There's only %1 GB available on the device. Blockchain requires ~%2 GB of data.").arg(validator.storageAvailable).arg(estimatedBlockchainSize) + "\n\n"
|
||
else
|
||
confirmationDialog.text += qsTr("Note: There's %1 GB available on the device. Blockchain requires ~%2 GB of data.").arg(validator.storageAvailable).arg(estimatedBlockchainSize) + "\n\n"
|
||
if(!validator.lmdbExists)
|
||
confirmationDialog.text += qsTr("Note: lmdb folder not found. A new folder will be created.") + "\n\n"
|
||
|
||
confirmationDialog.icon = StandardIcon.Question
|
||
|
||
// Continue
|
||
confirmationDialog.onAcceptedCallback = function() {
|
||
persistentSettings.blockchainDataDir = dataDir
|
||
}
|
||
|
||
// Cancel
|
||
confirmationDialog.onRejectedCallback = function() { };
|
||
confirmationDialog.open()
|
||
}
|
||
|
||
blockchainFileDialog.directory = blockchainFileDialog.fileUrl;
|
||
delete validator;
|
||
}
|
||
}
|
||
|
||
PasswordDialog {
|
||
id: passwordDialog
|
||
visible: false
|
||
z: parent.z + 2
|
||
anchors.fill: parent
|
||
property var onAcceptedCallback
|
||
property var onRejectedCallback
|
||
onAccepted: {
|
||
if (onAcceptedCallback)
|
||
onAcceptedCallback();
|
||
}
|
||
onRejected: {
|
||
if (onRejectedCallback)
|
||
onRejectedCallback();
|
||
}
|
||
onAcceptedNewPassword: {
|
||
if (currentWallet.setPassword(passwordDialog.password)) {
|
||
appWindow.walletPassword = passwordDialog.password;
|
||
informationPopup.title = qsTr("Information") + translationManager.emptyString;
|
||
informationPopup.text = qsTr("Password changed successfully") + translationManager.emptyString;
|
||
informationPopup.icon = StandardIcon.Information;
|
||
} else {
|
||
informationPopup.title = qsTr("Error") + translationManager.emptyString;
|
||
informationPopup.text = qsTr("Error: ") + currentWallet.errorString;
|
||
informationPopup.icon = StandardIcon.Critical;
|
||
}
|
||
informationPopup.onCloseCallback = null;
|
||
informationPopup.open();
|
||
}
|
||
onRejectedNewPassword: {}
|
||
Keys.enabled: !passwordDialog.visible && informationPopup.visible
|
||
Keys.onEnterPressed: informationPopup.close()
|
||
Keys.onReturnPressed: informationPopup.close()
|
||
}
|
||
|
||
DevicePassphraseDialog {
|
||
id: devicePassphraseDialog
|
||
visible: false
|
||
z: parent.z + 1
|
||
anchors.fill: parent
|
||
}
|
||
|
||
InputDialog {
|
||
id: inputDialog
|
||
visible: false
|
||
z: parent.z + 1
|
||
anchors.fill: parent
|
||
property var onAcceptedCallback
|
||
property var onRejectedCallback
|
||
onAccepted: {
|
||
if (onAcceptedCallback)
|
||
onAcceptedCallback()
|
||
}
|
||
onRejected: {
|
||
if (onRejectedCallback)
|
||
onRejectedCallback()
|
||
}
|
||
}
|
||
|
||
DaemonManagerDialog {
|
||
id: daemonManagerDialog
|
||
onRejected: {
|
||
middlePanel.settingsView.settingsStateViewState = "Node";
|
||
loadPage("Settings");
|
||
}
|
||
|
||
}
|
||
|
||
ProcessingSplash {
|
||
id: splash
|
||
width: appWindow.width / 2
|
||
height: appWindow.height / 2.66
|
||
x: (appWindow.width - width) / 2
|
||
y: (appWindow.height - height) / 2
|
||
messageText: qsTr("Please wait...") + translationManager.emptyString
|
||
}
|
||
|
||
Item {
|
||
id: rootItem
|
||
anchors.fill: parent
|
||
clip: true
|
||
|
||
state: "wizard"
|
||
states: [
|
||
State {
|
||
name: "wizard"
|
||
PropertyChanges { target: middlePanel; visible: false }
|
||
PropertyChanges { target: wizard; visible: true }
|
||
PropertyChanges { target: resizeArea; visible: true }
|
||
PropertyChanges { target: titleBar; state: "essentials" }
|
||
}, State {
|
||
name: "normal"
|
||
PropertyChanges { target: middlePanel; visible: true }
|
||
PropertyChanges { target: wizard; visible: false }
|
||
PropertyChanges { target: resizeArea; visible: true }
|
||
PropertyChanges { target: titleBar; state: "default" }
|
||
}
|
||
]
|
||
|
||
Item {
|
||
id: blurredArea
|
||
anchors.fill: parent
|
||
|
||
LeftPanel {
|
||
id: leftPanel
|
||
anchors.top: parent.top
|
||
anchors.left: parent.left
|
||
anchors.bottom: parent.bottom
|
||
visible: rootItem.state == "normal" && middlePanel.state != "Merchant"
|
||
currentAccountIndex: currentWallet ? currentWallet.currentSubaddressAccount : 0
|
||
currentAccountLabel: {
|
||
if (currentWallet) {
|
||
return currentWallet.getSubaddressLabel(currentWallet.currentSubaddressAccount, 0);
|
||
}
|
||
return qsTr("Primary account") + translationManager.emptyString;
|
||
}
|
||
|
||
onTransferClicked: {
|
||
middlePanel.state = "Transfer";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
|
||
onReceiveClicked: {
|
||
middlePanel.state = "Receive";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
|
||
onHistoryClicked: {
|
||
middlePanel.state = "History";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
|
||
onAddressBookClicked: {
|
||
middlePanel.state = "AddressBook";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
|
||
onAdvancedClicked: {
|
||
middlePanel.state = "Advanced";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
|
||
onSettingsClicked: {
|
||
middlePanel.state = "Settings";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
|
||
onAccountClicked: {
|
||
middlePanel.state = "Account";
|
||
middlePanel.flickable.contentY = 0;
|
||
updateBalance();
|
||
}
|
||
}
|
||
|
||
MiddlePanel {
|
||
id: middlePanel
|
||
accountView.currentAccountIndex: currentWallet ? currentWallet.currentSubaddressAccount : 0
|
||
anchors.top: parent.top
|
||
anchors.bottom: parent.bottom
|
||
anchors.left: leftPanel.visible ? leftPanel.right : parent.left
|
||
anchors.right: parent.right
|
||
state: "Transfer"
|
||
}
|
||
|
||
WizardController {
|
||
id: wizard
|
||
anchors.fill: parent
|
||
onUseMoneroClicked: {
|
||
rootItem.state = "normal";
|
||
appWindow.openWallet("wizard");
|
||
}
|
||
}
|
||
}
|
||
|
||
FastBlur {
|
||
id: blur
|
||
anchors.fill: blurredArea
|
||
source: blurredArea
|
||
radius: 64
|
||
visible: passwordDialog.visible || inputDialog.visible || splash.visible || updateDialog.visible ||
|
||
devicePassphraseDialog.visible || txConfirmationPopup.visible || successfulTxPopup.visible ||
|
||
remoteNodeDialog.visible
|
||
}
|
||
|
||
|
||
property int minWidth: 326
|
||
property int minHeight: 400
|
||
MouseArea {
|
||
id: resizeArea
|
||
enabled: persistentSettings.customDecorations
|
||
hoverEnabled: true
|
||
cursorShape: persistentSettings.customDecorations ? Qt.PointingHandCursor : Qt.ArrowCursor
|
||
anchors.right: parent.right
|
||
anchors.bottom: parent.bottom
|
||
height: 34
|
||
width: 34
|
||
|
||
MoneroEffects.ImageMask {
|
||
anchors.centerIn: parent
|
||
visible: persistentSettings.customDecorations
|
||
image: "qrc:///images/resize.png"
|
||
color: MoneroComponents.Style.defaultFontColor
|
||
width: 12
|
||
height: 12
|
||
opacity: (parent.containsMouse || parent.pressed) ? 0.5 : 1.0
|
||
}
|
||
|
||
property var previousPosition
|
||
|
||
onPressed: {
|
||
previousPosition = globalCursor.getPosition()
|
||
}
|
||
|
||
onPositionChanged: {
|
||
if(!pressed) return
|
||
var pos = globalCursor.getPosition()
|
||
//var delta = previousPosition - pos
|
||
var dx = previousPosition.x - pos.x
|
||
var dy = previousPosition.y - pos.y
|
||
|
||
if(appWindow.width - dx > parent.minWidth)
|
||
appWindow.width -= dx
|
||
else appWindow.width = parent.minWidth
|
||
|
||
if(appWindow.height - dy > parent.minHeight)
|
||
appWindow.height -= dy
|
||
else appWindow.height = parent.minHeight
|
||
previousPosition = pos
|
||
}
|
||
}
|
||
|
||
TitleBar {
|
||
id: titleBar
|
||
visible: persistentSettings.customDecorations && middlePanel.state !== "Merchant"
|
||
walletName: persistentSettings.displayWalletNameInTitleBar && rootItem.state != "wizard" ? appWindow.walletName : ""
|
||
anchors.left: parent.left
|
||
anchors.right: parent.right
|
||
onCloseClicked: appWindow.close();
|
||
onLockWalletClicked: appWindow.lock();
|
||
onLanguageClicked: appWindow.toggleLanguageView();
|
||
onCloseWalletClicked: appWindow.showWizard();
|
||
onMaximizeClicked: appWindow.visibility = appWindow.visibility !== Window.Maximized ? Window.Maximized : Window.Windowed
|
||
onMinimizeClicked: appWindow.visibility = Window.Minimized
|
||
}
|
||
|
||
MoneroMerchant.MerchantTitlebar {
|
||
id: titleBarOrange
|
||
visible: persistentSettings.customDecorations && middlePanel.state === "Merchant"
|
||
anchors.left: parent.left
|
||
anchors.right: parent.right
|
||
onCloseClicked: appWindow.close();
|
||
onMaximizeClicked: appWindow.visibility = appWindow.visibility !== Window.Maximized ? Window.Maximized : Window.Windowed
|
||
onMinimizeClicked: appWindow.visibility = Window.Minimized
|
||
}
|
||
|
||
// new ToolTip
|
||
Rectangle {
|
||
id: toolTip
|
||
property alias text: content.text
|
||
width: content.width + 12
|
||
height: content.height + 17
|
||
color: "#FF6C3C"
|
||
//radius: 3
|
||
visible:false;
|
||
|
||
Image {
|
||
id: tip
|
||
anchors.top: parent.bottom
|
||
anchors.right: parent.right
|
||
anchors.rightMargin: 5
|
||
source: "qrc:///images/tip.png"
|
||
}
|
||
|
||
MoneroComponents.TextPlain {
|
||
id: content
|
||
anchors.horizontalCenter: parent.horizontalCenter
|
||
y: 6
|
||
lineHeight: 0.7
|
||
font.family: "Arial"
|
||
font.pixelSize: 12
|
||
color: "#FFFFFF"
|
||
}
|
||
}
|
||
}
|
||
|
||
function toggleLanguageView(){
|
||
languageSidebar.visible ? languageSidebar.close() : languageSidebar.open();
|
||
languageSidebar.selectCurrentLanguage()
|
||
resetLanguageFields()
|
||
}
|
||
|
||
Timer {
|
||
id: autosaveTimer
|
||
interval: persistentSettings.autosaveMinutes * 60 * 1000
|
||
repeat: true
|
||
running: persistentSettings.autosave
|
||
onTriggered: {
|
||
if (currentWallet && !currentWallet.refreshing) {
|
||
currentWallet.storeAsync(function(success) {
|
||
if (success) {
|
||
appWindow.showStatusMessage(qsTr("Autosaved the wallet"), 3);
|
||
} else {
|
||
appWindow.showStatusMessage(qsTr("Failed to autosave the wallet"), 3);
|
||
}
|
||
});
|
||
}
|
||
}
|
||
}
|
||
|
||
// TODO: Make the callback dynamic
|
||
Timer {
|
||
id: statusMessageTimer
|
||
interval: 5;
|
||
running: false;
|
||
repeat: false
|
||
onTriggered: resetAndroidClose()
|
||
triggeredOnStart: false
|
||
}
|
||
|
||
Timer {
|
||
id: userInActivityTimer
|
||
interval: 2000; running: false; repeat: true
|
||
onTriggered: checkInUserActivity()
|
||
}
|
||
|
||
Timer {
|
||
// enables theme transition animations after 500ms
|
||
id: appThemeTransition
|
||
running: true
|
||
repeat: false
|
||
interval: 500
|
||
onTriggered: appWindow.themeTransition = true;
|
||
}
|
||
|
||
function checkNoSyncFlag() {
|
||
if (!appWindow.daemonRunning) {
|
||
return true;
|
||
}
|
||
if (appWindow.walletMode == 0 && !daemonManager.noSync()) {
|
||
return false;
|
||
}
|
||
if (appWindow.walletMode == 1 && daemonManager.noSync()) {
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
function checkSimpleModeConnection(){
|
||
const disconnectedTimeoutSec = 30;
|
||
const firstCheckDelaySec = 2;
|
||
|
||
const firstRun = appWindow.disconnectedEpoch == 0;
|
||
if (firstRun) {
|
||
appWindow.disconnectedEpoch = Utils.epoch() + firstCheckDelaySec - disconnectedTimeoutSec;
|
||
} else if (!disconnected) {
|
||
appWindow.disconnectedEpoch = Utils.epoch();
|
||
}
|
||
|
||
const sinceLastConnect = Utils.epoch() - appWindow.disconnectedEpoch;
|
||
if (sinceLastConnect < disconnectedTimeoutSec && checkNoSyncFlag()) {
|
||
return;
|
||
}
|
||
|
||
const simpleModeFlags = "--enable-dns-blocklist --out-peers 16 --no-igd";
|
||
if (appWindow.daemonRunning) {
|
||
appWindow.stopDaemon(function() {
|
||
appWindow.startDaemon(simpleModeFlags)
|
||
});
|
||
} else {
|
||
appWindow.startDaemon(simpleModeFlags);
|
||
}
|
||
}
|
||
|
||
Timer {
|
||
// Simple mode connection check timer
|
||
id: simpleModeConnectionTimer
|
||
interval: 2000
|
||
running: appWindow.walletMode < 2 && currentWallet != undefined && daemonStartStopInProgress == 0
|
||
repeat: true
|
||
onTriggered: appWindow.checkSimpleModeConnection()
|
||
}
|
||
|
||
Rectangle {
|
||
id: statusMessage
|
||
z: 99
|
||
visible: false
|
||
property alias text: statusMessageText.text
|
||
anchors.bottom: parent.bottom
|
||
width: statusMessageText.contentWidth + 20
|
||
anchors.horizontalCenter: parent.horizontalCenter
|
||
color: MoneroComponents.Style.blackTheme ? "black" : "white"
|
||
height: 40
|
||
MoneroComponents.TextPlain {
|
||
id: statusMessageText
|
||
anchors.fill: parent
|
||
anchors.margins: 10
|
||
font.pixelSize: 14
|
||
color: MoneroComponents.Style.defaultFontColor
|
||
themeTransition: false
|
||
}
|
||
}
|
||
|
||
function resetAndroidClose() {
|
||
console.log("resetting android close");
|
||
androidCloseTapped = false;
|
||
statusMessage.visible = false
|
||
}
|
||
|
||
function showStatusMessage(msg,timeout) {
|
||
console.log("showing status message")
|
||
statusMessageTimer.interval = timeout * 1000;
|
||
statusMessageTimer.start()
|
||
statusMessageText.text = msg;
|
||
statusMessage.visible = true
|
||
}
|
||
|
||
function showDaemonIsRunningDialog(onClose) {
|
||
// Show confirmation dialog
|
||
confirmationDialog.title = qsTr("Local node is running") + translationManager.emptyString;
|
||
confirmationDialog.text = qsTr("Do you want to stop local node or keep it running in the background?") + translationManager.emptyString;
|
||
confirmationDialog.icon = StandardIcon.Question;
|
||
confirmationDialog.cancelText = qsTr("Force stop") + translationManager.emptyString;
|
||
confirmationDialog.okText = qsTr("Keep it running") + translationManager.emptyString;
|
||
confirmationDialog.onAcceptedCallback = function() {
|
||
onClose();
|
||
}
|
||
confirmationDialog.onRejectedCallback = function() {
|
||
stopDaemon(onClose);
|
||
};
|
||
confirmationDialog.open();
|
||
}
|
||
|
||
onClosing: {
|
||
close.accepted = false;
|
||
console.log("blocking close event");
|
||
if(isAndroid) {
|
||
console.log("blocking android exit");
|
||
if(qrScannerEnabled)
|
||
cameraUi.state = "Stopped"
|
||
|
||
if(!androidCloseTapped) {
|
||
androidCloseTapped = true;
|
||
appWindow.showStatusMessage(qsTr("Tap again to close..."),3)
|
||
|
||
// first close
|
||
return;
|
||
}
|
||
|
||
|
||
}
|
||
|
||
// If daemon is running - prompt user before exiting
|
||
if(daemonManager == undefined || persistentSettings.useRemoteNode) {
|
||
closeAccepted();
|
||
} else if (appWindow.walletMode == 0) {
|
||
stopDaemon(closeAccepted, true);
|
||
} else {
|
||
showProcessingSplash(qsTr("Checking local node status..."));
|
||
const handler = function(running) {
|
||
hideProcessingSplash();
|
||
if (running && persistentSettings.askStopLocalNode) {
|
||
showDaemonIsRunningDialog(closeAccepted);
|
||
} else {
|
||
closeAccepted();
|
||
}
|
||
};
|
||
|
||
if (currentWallet) {
|
||
handler(!currentWallet.disconnected);
|
||
} else {
|
||
daemonManager.runningAsync(persistentSettings.nettype, handler);
|
||
}
|
||
}
|
||
}
|
||
|
||
function closeAccepted(){
|
||
console.log("close accepted");
|
||
// Close wallet non async on exit
|
||
daemonManager.exit();
|
||
p2poolManager.exit();
|
||
closeWallet(Qt.quit);
|
||
}
|
||
|
||
function onWalletCheckUpdatesComplete(version, downloadUrl, hash, firstSigner, secondSigner) {
|
||
const alreadyAsked = updateDialog.url == downloadUrl && updateDialog.hash == hash;
|
||
if (!alreadyAsked)
|
||
{
|
||
updateDialog.show(version, isMac || isWindows || isLinux ? downloadUrl : "", hash);
|
||
}
|
||
}
|
||
|
||
function getBuildTag() {
|
||
if (isMac) {
|
||
return "mac-x64";
|
||
}
|
||
if (isWindows) {
|
||
return oshelper.installed ? "install-win-x64" : "win-x64";
|
||
}
|
||
if (isLinux) {
|
||
return "linux-x64";
|
||
}
|
||
return "source";
|
||
}
|
||
|
||
function checkUpdates() {
|
||
const version = Version.GUI_VERSION.match(/\d+\.\d+\.\d+\.\d+/);
|
||
if (version) {
|
||
walletManager.checkUpdatesAsync("monero-gui", "gui", getBuildTag(), version[0]);
|
||
} else {
|
||
console.error("failed to parse version number", Version.GUI_VERSION);
|
||
}
|
||
}
|
||
|
||
Timer {
|
||
id: updatesTimer
|
||
interval: 3600 * 1000
|
||
repeat: true
|
||
running: !disableCheckUpdatesFlag && persistentSettings.checkForUpdates
|
||
triggeredOnStart: true
|
||
onTriggered: checkUpdates()
|
||
}
|
||
|
||
function releaseFocus() {
|
||
// Workaround to release focus from textfield when scrolling (https://bugreports.qt.io/browse/QTBUG-34867)
|
||
if(isAndroid) {
|
||
console.log("releasing focus")
|
||
middlePanel.focus = true
|
||
middlePanel.focus = false
|
||
}
|
||
}
|
||
|
||
// reset label text. othewise potential privacy leak showing unlock time when switching wallets
|
||
function clearMoneroCardLabelText(){
|
||
leftPanel.balanceString = "?.??"
|
||
leftPanel.balanceFiatString = "?.??"
|
||
}
|
||
|
||
// some fields need an extra nudge when changing languages
|
||
function resetLanguageFields(){
|
||
clearMoneroCardLabelText()
|
||
if (currentWallet) {
|
||
onWalletRefresh();
|
||
}
|
||
}
|
||
|
||
function userActivity() {
|
||
// register user activity
|
||
appWindow.userLastActive = Utils.epoch();
|
||
}
|
||
|
||
function checkInUserActivity() {
|
||
if(rootItem.state !== "normal") return;
|
||
if(!persistentSettings.lockOnUserInActivity) return;
|
||
if(passwordDialog.visible) return;
|
||
var inputDialogVisible = inputDialog && inputDialog.visible
|
||
var successfulTxPopupVisible = successfulTxPopup && successfulTxPopup.visible
|
||
var informationPopupVisible = informationPopup && informationPopup.visible
|
||
|
||
// prompt password after X seconds of inactivity
|
||
var inactivity = Utils.epoch() - appWindow.userLastActive;
|
||
if(inactivity < (persistentSettings.lockOnUserInActivityInterval * 60)) return;
|
||
|
||
passwordDialog.onAcceptedCallback = function() {
|
||
if(walletPassword === passwordDialog.password){
|
||
passwordDialog.close();
|
||
if (inputDialogVisible) inputDialog.open(inputDialog.inputText)
|
||
if (successfulTxPopupVisible) successfulTxPopup.open(successfulTxPopup.transactionID)
|
||
if (informationPopupVisible) informationPopup.open()
|
||
} else {
|
||
passwordDialog.showError(qsTr("Wrong password"));
|
||
}
|
||
}
|
||
|
||
passwordDialog.onRejectedCallback = function() { appWindow.showWizard(); }
|
||
if (inputDialogVisible) inputDialog.close()
|
||
remoteNodeDialog.close();
|
||
informationPopup.close()
|
||
txConfirmationPopup.close()
|
||
txConfirmationPopup.clearFields()
|
||
txConfirmationPopup.rejected()
|
||
successfulTxPopup.close();
|
||
passwordDialog.open();
|
||
}
|
||
|
||
function getDefaultDaemonRpcPort(networkType) {
|
||
switch (parseInt(networkType)) {
|
||
case NetworkType.STAGENET:
|
||
return 38081;
|
||
case NetworkType.TESTNET:
|
||
return 28081;
|
||
default:
|
||
return 18081;
|
||
}
|
||
}
|
||
|
||
function changeWalletMode(mode){
|
||
appWindow.disconnectedEpoch = 0;
|
||
persistentSettings.walletMode = mode;
|
||
applyWalletMode(mode);
|
||
}
|
||
|
||
function applyWalletMode(mode){
|
||
if (mode < 2) {
|
||
persistentSettings.useRemoteNode = false;
|
||
|
||
if (middlePanel.settingsView.settingsStateViewState === "Node") {
|
||
middlePanel.settingsView.settingsStateViewState = "Wallet"
|
||
}
|
||
}
|
||
console.log("walletMode: " + (mode === 0 ? "simple": mode === 1 ? "simple (bootstrap)" : "Advanced"));
|
||
}
|
||
|
||
Rectangle {
|
||
id: inactiveOverlay
|
||
visible: blur.visible
|
||
anchors.fill: parent
|
||
anchors.topMargin: titleBar.height
|
||
color: MoneroComponents.Style.blackTheme ? "black" : "white"
|
||
opacity: isOpenGL ? 0.3 : inputDialog.visible || splash.visible ? 0.7 : 1.0
|
||
|
||
MoneroEffects.ColorTransition {
|
||
targetObj: parent
|
||
blackColor: "black"
|
||
whiteColor: "white"
|
||
}
|
||
|
||
MouseArea {
|
||
anchors.fill: parent
|
||
hoverEnabled: true
|
||
}
|
||
}
|
||
|
||
// borders on white theme + linux
|
||
Rectangle {
|
||
visible: isLinux && !MoneroComponents.Style.blackTheme && middlePanel.state !== "Merchant"
|
||
z: parent.z + 1
|
||
anchors.left: parent.left
|
||
anchors.top: parent.top
|
||
anchors.bottom: parent.bottom
|
||
width: 1
|
||
color: MoneroComponents.Style.appWindowBorderColor
|
||
|
||
MoneroEffects.ColorTransition {
|
||
targetObj: parent
|
||
blackColor: MoneroComponents.Style._b_appWindowBorderColor
|
||
whiteColor: MoneroComponents.Style._w_appWindowBorderColor
|
||
}
|
||
}
|
||
|
||
Rectangle {
|
||
visible: isLinux && !MoneroComponents.Style.blackTheme && middlePanel.state !== "Merchant"
|
||
z: parent.z + 1
|
||
anchors.right: parent.right
|
||
anchors.top: parent.top
|
||
anchors.bottom: parent.bottom
|
||
width: 1
|
||
color: MoneroComponents.Style.appWindowBorderColor
|
||
|
||
MoneroEffects.ColorTransition {
|
||
targetObj: parent
|
||
blackColor: MoneroComponents.Style._b_appWindowBorderColor
|
||
whiteColor: MoneroComponents.Style._w_appWindowBorderColor
|
||
}
|
||
}
|
||
|
||
Rectangle {
|
||
visible: isLinux && !MoneroComponents.Style.blackTheme && middlePanel.state !== "Merchant"
|
||
z: parent.z + 1
|
||
anchors.right: parent.right
|
||
anchors.top: parent.top
|
||
anchors.left: parent.left
|
||
height: 1
|
||
color: MoneroComponents.Style.appWindowBorderColor
|
||
|
||
MoneroEffects.ColorTransition {
|
||
targetObj: parent
|
||
blackColor: MoneroComponents.Style._b_appWindowBorderColor
|
||
whiteColor: MoneroComponents.Style._w_appWindowBorderColor
|
||
}
|
||
}
|
||
|
||
Rectangle {
|
||
visible: isLinux && !MoneroComponents.Style.blackTheme && middlePanel.state !== "Merchant"
|
||
z: parent.z + 1
|
||
anchors.right: parent.right
|
||
anchors.bottom: parent.bottom
|
||
anchors.left: parent.left
|
||
height: 1
|
||
color: MoneroComponents.Style.appWindowBorderColor
|
||
|
||
MoneroEffects.ColorTransition {
|
||
targetObj: parent
|
||
blackColor: MoneroComponents.Style._b_appWindowBorderColor
|
||
whiteColor: MoneroComponents.Style._w_appWindowBorderColor
|
||
}
|
||
}
|
||
|
||
MoneroComponents.LanguageSidebar {
|
||
id: languageSidebar
|
||
dragMargin: 0
|
||
onAboutToShow: previousActiveFocusItem = activeFocusItem;
|
||
onClosed: { if (previousActiveFocusItem) previousActiveFocusItem.forceActiveFocus() }
|
||
}
|
||
|
||
MoneroComponents.MenuBar { }
|
||
|
||
Network {
|
||
id: network
|
||
proxyAddress: persistentSettings.getProxyAddress()
|
||
}
|
||
|
||
WalletManager {
|
||
id: walletManager
|
||
proxyAddress: persistentSettings.getProxyAddress()
|
||
}
|
||
}
|