diff --git a/main.qml b/main.qml index 15bfa68473..33e989579e 100644 --- a/main.qml +++ b/main.qml @@ -1,2391 +1,2386 @@ -// 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 ? 55 : 150 // 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, persistentSettings.blockchainDataDir, 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 - - if (currentWallet) - 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, persistentSettings.blockchainDataDir, 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(); - middlePanel.advancedView.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: isAndroid - 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, persistentSettings.blockchainDataDir, 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() - } -} +// 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 ? 55 : 150 // GB + property alias viewState: rootItem.state + property string prevSplashText; + property bool splashDisplayedBeforeButtonRequest; + property bool themeTransition: false + + // fiat price conversion + property real fiatPrice: 0 + + // {provider name: {ticker: price_api_url}} + // API response schema depends on the provider + property var fiatCurrencies: ["usd", "eur", "aed", "ars", "aud", "bdt", "bhd", "brl", "cad", "chf", "clp", "cny", "czk", "gbp", "hkd", + "huf", "idr", "ils", "inr", "jpy", "krw", "kwd", "lkr", "mmk", "mxn", "myr", "ngn", "nok", "nzd", "php", + "pkr", "pln", "rub", "sar", "sek", "sgd", "thb", "try", "twd", "uah", "vef", "vnd", "zar", "xau"]; + property var fiatPriceAPIs: fiatCurrencies.reduce(function(obj, x) { + const key = `xmr${x}`; // e.g. xmrusd + if (x === "usd" || x === "eur") { + // Kraken only supports XMRUSD and XMREUR + obj["kraken"][key] = `https://api.kraken.com/0/public/Ticker?pair=XMR${x}`; + } + obj["coingecko"][key] = `https://api.coingecko.com/api/v3/simple/price?ids=monero&vs_currencies=${x}`; + obj["cryptocompare"][key] = `https://min-api.cryptocompare.com/data/price?fsym=XMR&tsyms=${x}`; + return obj; + }, {"kraken": {}, "coingecko": {}, "cryptocompare": {}}) + // if the user is using Kraken, the following is used if the user wants non USD/EUR + property string fiatPriceBackupProvider: "coingecko" + // 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, persistentSettings.blockchainDataDir, 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 + + if (currentWallet) + 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, persistentSettings.blockchainDataDir, 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(); + middlePanel.advancedView.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; + } + // i.e. xmr[a-z]+ -> XXMRZ[A-Z]+ + var key = `XXMRZ${currency.substring(3).toUpperCase()}`; + var ticker = resp.result[key]["c"][0]; + return ticker; + } else if(url.startsWith("https://api.coingecko.com/api/v3/")){ + // i.e. xmr[a-z]+ -> [a-z]+ + var key = currency.substring(3); + 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/")){ + // i.e. xmr[a-z]+ -> [A-Z]+ + var key = currency.substring(3).toUpperCase(); + 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 + "\"is not supported by provider \"" + userProvider + "\""); + } + + var url = provider[userCurrency]; + network.getJSON(url, fiatApiJsonReceived); + } + + function fiatApiCurrencySymbol() { + return persistentSettings.fiatPriceCurrency.substring(3).toUpperCase(); + } + + 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: isAndroid + 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, persistentSettings.blockchainDataDir, 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() + } +} diff --git a/pages/settings/SettingsLayout.qml b/pages/settings/SettingsLayout.qml index 38ce823ea8..dbd91b5f02 100644 --- a/pages/settings/SettingsLayout.qml +++ b/pages/settings/SettingsLayout.qml @@ -1,323 +1,337 @@ -// Copyright (c) 2014-2018, 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 QtQuick 2.9 -import QtQuick.Layouts 1.1 -import QtQuick.Controls 2.0 -import QtQuick.Dialogs 1.2 - -import "../../js/Utils.js" as Utils -import "../../js/Windows.js" as Windows -import "../../components" as MoneroComponents - -Rectangle { - color: "transparent" - Layout.fillWidth: true - property alias layoutHeight: settingsUI.height - - ColumnLayout { - id: settingsUI - property int itemHeight: 60 - Layout.fillWidth: true - anchors.left: parent.left - anchors.top: parent.top - anchors.right: parent.right - anchors.margins: 20 - anchors.topMargin: 0 - spacing: 6 - - MoneroComponents.CheckBox { - id: customDecorationsCheckBox - checked: persistentSettings.customDecorations - onClicked: Windows.setCustomWindowDecorations(checked) - text: qsTr("Custom decorations") + translationManager.emptyString - } - - MoneroComponents.CheckBox { - id: checkForUpdatesCheckBox - enabled: !disableCheckUpdatesFlag - checked: persistentSettings.checkForUpdates && !disableCheckUpdatesFlag - onClicked: persistentSettings.checkForUpdates = !persistentSettings.checkForUpdates - text: qsTr("Check for updates periodically") + translationManager.emptyString - } - - MoneroComponents.CheckBox { - checked: persistentSettings.displayWalletNameInTitleBar - onClicked: persistentSettings.displayWalletNameInTitleBar = !persistentSettings.displayWalletNameInTitleBar - text: qsTr("Display wallet name in title bar") + translationManager.emptyString - } - - MoneroComponents.CheckBox { - id: hideBalanceCheckBox - checked: persistentSettings.hideBalance - onClicked: { - persistentSettings.hideBalance = !persistentSettings.hideBalance - appWindow.updateBalance(); - } - text: qsTr("Hide balance") + translationManager.emptyString - } - - MoneroComponents.CheckBox { - id: themeCheckbox - checked: !MoneroComponents.Style.blackTheme - text: qsTr("Light theme") + translationManager.emptyString - toggleOnClick: false - onClicked: { - MoneroComponents.Style.blackTheme = !MoneroComponents.Style.blackTheme; - } - } - - MoneroComponents.CheckBox { - checked: persistentSettings.askPasswordBeforeSending - text: qsTr("Ask for password before sending a transaction") + translationManager.emptyString - toggleOnClick: false - onClicked: { - if (persistentSettings.askPasswordBeforeSending) { - passwordDialog.onAcceptedCallback = function() { - if (appWindow.walletPassword === passwordDialog.password){ - persistentSettings.askPasswordBeforeSending = false; - } else { - passwordDialog.showError(qsTr("Wrong password")); - } - } - passwordDialog.onRejectedCallback = null; - passwordDialog.open() - } else { - persistentSettings.askPasswordBeforeSending = true; - } - } - } - - MoneroComponents.CheckBox { - checked: persistentSettings.autosave - onClicked: persistentSettings.autosave = !persistentSettings.autosave - text: qsTr("Autosave") + translationManager.emptyString - } - - MoneroComponents.Slider { - Layout.fillWidth: true - Layout.leftMargin: 35 - Layout.topMargin: 6 - visible: persistentSettings.autosave - from: 1 - stepSize: 1 - to: 60 - value: persistentSettings.autosaveMinutes - text: "%1 %2 %3".arg(qsTr("Every")).arg(value).arg(qsTr("minute(s)")) + translationManager.emptyString - onMoved: persistentSettings.autosaveMinutes = value - } - - MoneroComponents.CheckBox { - id: userInActivityCheckbox - checked: persistentSettings.lockOnUserInActivity - onClicked: persistentSettings.lockOnUserInActivity = !persistentSettings.lockOnUserInActivity - text: qsTr("Lock wallet on inactivity") + translationManager.emptyString - } - - MoneroComponents.Slider { - visible: userInActivityCheckbox.checked - Layout.fillWidth: true - Layout.topMargin: 6 - Layout.leftMargin: 35 - from: 1 - stepSize: 1 - to: 60 - value: persistentSettings.lockOnUserInActivityInterval - text: { - var minutes = value > 1 ? qsTr("minutes") : qsTr("minute"); - return qsTr("After ") + value + " " + minutes + translationManager.emptyString; - } - onMoved: persistentSettings.lockOnUserInActivityInterval = value - } - - MoneroComponents.CheckBox { - checked: persistentSettings.askStopLocalNode - onClicked: persistentSettings.askStopLocalNode = !persistentSettings.askStopLocalNode - text: qsTr("Ask to stop local node during program exit") + translationManager.emptyString - } - - //! Manage pricing - RowLayout { - MoneroComponents.CheckBox { - id: enableConvertCurrency - text: qsTr("Enable displaying balance in other currencies") + translationManager.emptyString - checked: persistentSettings.fiatPriceEnabled - onCheckedChanged: { - if (!checked) { - console.log("Disabled price conversion"); - persistentSettings.fiatPriceEnabled = false; - } - } - } - } - - GridLayout { - visible: enableConvertCurrency.checked - columns: 2 - Layout.fillWidth: true - Layout.leftMargin: 36 - columnSpacing: 32 - - MoneroComponents.StandardDropdown { - id: fiatPriceProviderDropDown - Layout.maximumWidth: 200 - labelText: qsTr("Price source") + translationManager.emptyString - labelFontSize: 14 - dataModel: fiatPriceProvidersModel - onChanged: { - var obj = dataModel.get(currentIndex); - persistentSettings.fiatPriceProvider = obj.data; - - if(persistentSettings.fiatPriceEnabled) - appWindow.fiatApiRefresh(); - } - } - - MoneroComponents.StandardDropdown { - id: fiatPriceCurrencyDropdown - Layout.maximumWidth: 100 - labelText: qsTr("Currency") + translationManager.emptyString - labelFontSize: 14 - currentIndex: persistentSettings.fiatPriceCurrency === "xmrusd" ? 0 : 1 - dataModel: fiatPriceCurrencyModel - onChanged: { - var obj = dataModel.get(currentIndex); - persistentSettings.fiatPriceCurrency = obj.data; - - if(persistentSettings.fiatPriceEnabled) - appWindow.fiatApiRefresh(); - } - } - - z: parent.z + 1 - } - - ColumnLayout { - // Feature needs to be double enabled for security purposes (miss-clicks) - visible: enableConvertCurrency.checked && !persistentSettings.fiatPriceEnabled - spacing: 0 - Layout.topMargin: 5 - Layout.leftMargin: 36 - - MoneroComponents.WarningBox { - text: qsTr("Enabling price conversion exposes your IP address to the selected price source.") + translationManager.emptyString; - } - - MoneroComponents.StandardButton { - Layout.topMargin: 10 - Layout.bottomMargin: 10 - small: true - text: qsTr("Confirm and enable") + translationManager.emptyString - - onClicked: { - console.log("Enabled price conversion"); - persistentSettings.fiatPriceEnabled = true; - } - } - } - - MoneroComponents.CheckBox { - id: proxyCheckbox - Layout.topMargin: 6 - enabled: !socksProxyFlagSet - checked: socksProxyFlagSet ? socksProxyFlag : persistentSettings.proxyEnabled - onClicked: { - persistentSettings.proxyEnabled = !persistentSettings.proxyEnabled; - } - text: qsTr("Socks5 proxy (%1%2)") - .arg(appWindow.walletMode >= 2 ? qsTr("remote node connections, ") : "") - .arg(qsTr("updates downloading, fetching price sources")) + translationManager.emptyString - } - - MoneroComponents.RemoteNodeEdit { - id: proxyEdit - enabled: proxyCheckbox.enabled - Layout.leftMargin: 36 - Layout.topMargin: 6 - Layout.minimumWidth: 100 - placeholderFontSize: 15 - visible: proxyCheckbox.checked - - daemonAddrLabelText: qsTr("IP address") + translationManager.emptyString - daemonPortLabelText: qsTr("Port") + translationManager.emptyString - - initialAddress: socksProxyFlagSet ? socksProxyFlag : persistentSettings.proxyAddress - onEditingFinished: { - persistentSettings.proxyAddress = proxyEdit.getAddress(); - } - } - - MoneroComponents.StandardButton { - visible: !persistentSettings.customDecorations - Layout.topMargin: 10 - small: true - text: qsTr("Change language") + translationManager.emptyString - - onClicked: { - appWindow.toggleLanguageView(); - } - } - } - - ListModel { - id: fiatPriceProvidersModel - } - - ListModel { - id: fiatPriceCurrencyModel - ListElement { - data: "xmrusd" - column1: "USD" - } - ListElement { - data: "xmreur" - column1: "EUR" - } - } - - Component.onCompleted: { - // Dynamically fill fiatPrice dropdown based on `appWindow.fiatPriceAPIs` - var apis = appWindow.fiatPriceAPIs; - fiatPriceProvidersModel.clear(); - - var i = 0; - for (var api in apis){ - if (!apis.hasOwnProperty(api)) - continue; - - fiatPriceProvidersModel.append({"column1": Utils.capitalize(api), "data": api}); - - if(api === persistentSettings.fiatPriceProvider) - fiatPriceProviderDropDown.currentIndex = i; - i += 1; - } - - console.log('SettingsLayout loaded'); - } -} - +// Copyright (c) 2014-2018, 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 QtQuick 2.9 +import QtQuick.Layouts 1.1 +import QtQuick.Controls 2.0 +import QtQuick.Dialogs 1.2 + +import "../../js/Utils.js" as Utils +import "../../js/Windows.js" as Windows +import "../../components" as MoneroComponents + +Rectangle { + color: "transparent" + Layout.fillWidth: true + property alias layoutHeight: settingsUI.height + + ColumnLayout { + id: settingsUI + property int itemHeight: 60 + Layout.fillWidth: true + anchors.left: parent.left + anchors.top: parent.top + anchors.right: parent.right + anchors.margins: 20 + anchors.topMargin: 0 + spacing: 6 + + MoneroComponents.CheckBox { + id: customDecorationsCheckBox + checked: persistentSettings.customDecorations + onClicked: Windows.setCustomWindowDecorations(checked) + text: qsTr("Custom decorations") + translationManager.emptyString + } + + MoneroComponents.CheckBox { + id: checkForUpdatesCheckBox + enabled: !disableCheckUpdatesFlag + checked: persistentSettings.checkForUpdates && !disableCheckUpdatesFlag + onClicked: persistentSettings.checkForUpdates = !persistentSettings.checkForUpdates + text: qsTr("Check for updates periodically") + translationManager.emptyString + } + + MoneroComponents.CheckBox { + checked: persistentSettings.displayWalletNameInTitleBar + onClicked: persistentSettings.displayWalletNameInTitleBar = !persistentSettings.displayWalletNameInTitleBar + text: qsTr("Display wallet name in title bar") + translationManager.emptyString + } + + MoneroComponents.CheckBox { + id: hideBalanceCheckBox + checked: persistentSettings.hideBalance + onClicked: { + persistentSettings.hideBalance = !persistentSettings.hideBalance + appWindow.updateBalance(); + } + text: qsTr("Hide balance") + translationManager.emptyString + } + + MoneroComponents.CheckBox { + id: themeCheckbox + checked: !MoneroComponents.Style.blackTheme + text: qsTr("Light theme") + translationManager.emptyString + toggleOnClick: false + onClicked: { + MoneroComponents.Style.blackTheme = !MoneroComponents.Style.blackTheme; + } + } + + MoneroComponents.CheckBox { + checked: persistentSettings.askPasswordBeforeSending + text: qsTr("Ask for password before sending a transaction") + translationManager.emptyString + toggleOnClick: false + onClicked: { + if (persistentSettings.askPasswordBeforeSending) { + passwordDialog.onAcceptedCallback = function() { + if (appWindow.walletPassword === passwordDialog.password){ + persistentSettings.askPasswordBeforeSending = false; + } else { + passwordDialog.showError(qsTr("Wrong password")); + } + } + passwordDialog.onRejectedCallback = null; + passwordDialog.open() + } else { + persistentSettings.askPasswordBeforeSending = true; + } + } + } + + MoneroComponents.CheckBox { + checked: persistentSettings.autosave + onClicked: persistentSettings.autosave = !persistentSettings.autosave + text: qsTr("Autosave") + translationManager.emptyString + } + + MoneroComponents.Slider { + Layout.fillWidth: true + Layout.leftMargin: 35 + Layout.topMargin: 6 + visible: persistentSettings.autosave + from: 1 + stepSize: 1 + to: 60 + value: persistentSettings.autosaveMinutes + text: "%1 %2 %3".arg(qsTr("Every")).arg(value).arg(qsTr("minute(s)")) + translationManager.emptyString + onMoved: persistentSettings.autosaveMinutes = value + } + + MoneroComponents.CheckBox { + id: userInActivityCheckbox + checked: persistentSettings.lockOnUserInActivity + onClicked: persistentSettings.lockOnUserInActivity = !persistentSettings.lockOnUserInActivity + text: qsTr("Lock wallet on inactivity") + translationManager.emptyString + } + + MoneroComponents.Slider { + visible: userInActivityCheckbox.checked + Layout.fillWidth: true + Layout.topMargin: 6 + Layout.leftMargin: 35 + from: 1 + stepSize: 1 + to: 60 + value: persistentSettings.lockOnUserInActivityInterval + text: { + var minutes = value > 1 ? qsTr("minutes") : qsTr("minute"); + return qsTr("After ") + value + " " + minutes + translationManager.emptyString; + } + onMoved: persistentSettings.lockOnUserInActivityInterval = value + } + + MoneroComponents.CheckBox { + checked: persistentSettings.askStopLocalNode + onClicked: persistentSettings.askStopLocalNode = !persistentSettings.askStopLocalNode + text: qsTr("Ask to stop local node during program exit") + translationManager.emptyString + } + + //! Manage pricing + RowLayout { + MoneroComponents.CheckBox { + id: enableConvertCurrency + text: qsTr("Enable displaying balance in other currencies") + translationManager.emptyString + checked: persistentSettings.fiatPriceEnabled + onCheckedChanged: { + if (!checked) { + console.log("Disabled price conversion"); + persistentSettings.fiatPriceEnabled = false; + } + } + } + } + + GridLayout { + visible: enableConvertCurrency.checked + columns: 2 + Layout.fillWidth: true + Layout.leftMargin: 36 + columnSpacing: 32 + + MoneroComponents.StandardDropdown { + id: fiatPriceProviderDropDown + Layout.maximumWidth: 200 + labelText: qsTr("Price source") + translationManager.emptyString + labelFontSize: 14 + dataModel: fiatPriceProvidersModel + onChanged: { + var newProvider = dataModel.get(currentIndex).data; + var providerCurrencies = appWindow.fiatPriceAPIs[newProvider]; + // ONLY when the user changes the provider should the currency list also update + // This way, since Kraken is the default, users can see ALL supported currencies + fiatPriceCurrencyModel.clear(); + appWindow.fiatCurrencies.forEach(el => { + if (`xmr${el}` in providerCurrencies) fiatPriceCurrencyModel.append({ data: `xmr${el}`, column1: el.toUpperCase()}) + }); + // if fiatPriceCurrency is not supported by the new provider, use first available currency + if (!(persistentSettings.fiatPriceCurrency in providerCurrencies)) { + persistentSettings.fiatPriceCurrency = Object.keys(providerCurrencies)[0]; + fiatPriceCurrencyDropdown.currentIndex = 0; + } + persistentSettings.fiatPriceProvider = newProvider; + // refresh price after validating that the fiat currency is provided by the provider + if(persistentSettings.fiatPriceEnabled) + appWindow.fiatApiRefresh(); + } + } + + MoneroComponents.StandardDropdown { + id: fiatPriceCurrencyDropdown + Layout.maximumWidth: 100 + labelText: qsTr("Currency") + translationManager.emptyString + labelFontSize: 14 + currentIndex: appWindow.fiatCurrencies.indexOf(persistentSettings.fiatPriceCurrency.substring(3)) + dataModel: fiatPriceCurrencyModel + onChanged: { + var newCurrency = dataModel.get(currentIndex).data; + if (!(newCurrency in appWindow.fiatPriceAPIs[persistentSettings.fiatPriceProvider])) { + // this occurs if a fiat currency other than EUR/USD is selected and provider is Kraken + // so use appWindow.fiatPriceBackupProvider instead + let backupIdx = Object.keys(appWindow.fiatPriceAPIs).indexOf(appWindow.fiatPriceBackupProvider); + fiatPriceProviderDropDown.currentIndex = backupIdx; + persistentSettings.fiatPriceProvider = appWindow.fiatPriceBackupProvider; + } + persistentSettings.fiatPriceCurrency = newCurrency; + if(persistentSettings.fiatPriceEnabled) + appWindow.fiatApiRefresh(); + } + } + + z: parent.z + 1 + } + + ColumnLayout { + // Feature needs to be double enabled for security purposes (miss-clicks) + visible: enableConvertCurrency.checked && !persistentSettings.fiatPriceEnabled + spacing: 0 + Layout.topMargin: 5 + Layout.leftMargin: 36 + + MoneroComponents.WarningBox { + text: qsTr("Enabling price conversion exposes your IP address to the selected price source.") + translationManager.emptyString; + } + + MoneroComponents.StandardButton { + Layout.topMargin: 10 + Layout.bottomMargin: 10 + small: true + text: qsTr("Confirm and enable") + translationManager.emptyString + + onClicked: { + console.log("Enabled price conversion"); + persistentSettings.fiatPriceEnabled = true; + } + } + } + + MoneroComponents.CheckBox { + id: proxyCheckbox + Layout.topMargin: 6 + enabled: !socksProxyFlagSet + checked: socksProxyFlagSet ? socksProxyFlag : persistentSettings.proxyEnabled + onClicked: { + persistentSettings.proxyEnabled = !persistentSettings.proxyEnabled; + } + text: qsTr("Socks5 proxy (%1%2)") + .arg(appWindow.walletMode >= 2 ? qsTr("remote node connections, ") : "") + .arg(qsTr("updates downloading, fetching price sources")) + translationManager.emptyString + } + + MoneroComponents.RemoteNodeEdit { + id: proxyEdit + enabled: proxyCheckbox.enabled + Layout.leftMargin: 36 + Layout.topMargin: 6 + Layout.minimumWidth: 100 + placeholderFontSize: 15 + visible: proxyCheckbox.checked + + daemonAddrLabelText: qsTr("IP address") + translationManager.emptyString + daemonPortLabelText: qsTr("Port") + translationManager.emptyString + + initialAddress: socksProxyFlagSet ? socksProxyFlag : persistentSettings.proxyAddress + onEditingFinished: { + persistentSettings.proxyAddress = proxyEdit.getAddress(); + } + } + + MoneroComponents.StandardButton { + visible: !persistentSettings.customDecorations + Layout.topMargin: 10 + small: true + text: qsTr("Change language") + translationManager.emptyString + + onClicked: { + appWindow.toggleLanguageView(); + } + } + } + + ListModel { + id: fiatPriceProvidersModel + } + + ListModel { + id: fiatPriceCurrencyModel + Component.onCompleted: { + // populate with fiat currencies + appWindow.fiatCurrencies.forEach(el => { + fiatPriceCurrencyModel.append({ data: `xmr${el}`, column1: el.toUpperCase()}); + }); + } + } + + Component.onCompleted: { + // Dynamically fill fiatPrice dropdowns based on `appWindow.fiatPriceAPIs` + var apis = appWindow.fiatPriceAPIs; + fiatPriceProvidersModel.clear(); + var i = 0; + for (var apiProvider in apis){ + if (!apis.hasOwnProperty(apiProvider)) + continue; + + fiatPriceProvidersModel.append({data: apiProvider, column1: Utils.capitalize(apiProvider)}); + + if(apiProvider === persistentSettings.fiatPriceProvider) + fiatPriceProviderDropDown.currentIndex = i; + i += 1; + } + + console.log('SettingsLayout loaded'); + } +}