2018-01-07 07:20:45 +02:00
// Copyright (c) 2014-2018, The Monero Project
2016-11-12 15:28:13 +02:00
//
2015-04-01 11:56:05 +03:00
// All rights reserved.
2016-11-12 15:28:13 +02:00
//
2015-04-01 11:56:05 +03:00
// Redistribution and use in source and binary forms, with or without modification, are
// permitted provided that the following conditions are met:
2016-11-12 15:28:13 +02:00
//
2015-04-01 11:56:05 +03:00
// 1. Redistributions of source code must retain the above copyright notice, this list of
// conditions and the following disclaimer.
2016-11-12 15:28:13 +02:00
//
2015-04-01 11:56:05 +03:00
// 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.
2016-11-12 15:28:13 +02:00
//
2015-04-01 11:56:05 +03:00
// 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.
2016-11-12 15:28:13 +02:00
//
2015-04-01 11:56:05 +03:00
// 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.
2014-07-07 20:08:30 +03:00
import QtQuick 2.2
import QtQuick . Window 2.0
import QtQuick . Controls 1.1
import QtQuick . Controls . Styles 1.1
2016-06-28 22:37:14 +03:00
import QtQuick . Dialogs 1.2
2016-06-15 16:34:55 +03:00
import Qt . labs . settings 1.0
2016-10-02 21:40:40 +03:00
import moneroComponents . Wallet 1.0
import moneroComponents . PendingTransaction 1.0
2018-03-05 18:19:45 +02:00
import moneroComponents . NetworkType 1.0
2016-06-15 13:25:45 +03:00
2016-07-13 15:24:40 +03:00
2014-07-07 20:08:30 +03:00
import "components"
2014-08-19 15:58:02 +03:00
import "wizard"
2018-12-07 09:48:20 +02:00
import "js/Utils.js" as Utils
2018-04-21 22:58:00 +03:00
import "js/Windows.js" as Windows
2014-07-07 20:08:30 +03:00
ApplicationWindow {
id: appWindow
2017-07-13 18:39:51 +03:00
title: "Monero"
2016-08-17 15:14:43 +03:00
2014-07-09 17:18:48 +03:00
property var currentItem
2018-12-15 06:02:01 +02:00
property bool hideBalanceForced: false
2014-07-07 20:08:30 +03:00
property bool whatIsEnable: false
2014-07-13 15:27:50 +03:00
property bool ctrlPressed: false
2016-06-20 22:08:09 +03:00
property bool rightPanelExpanded: false
2014-07-16 15:40:09 +03:00
property bool osx: false
2016-06-15 16:34:55 +03:00
property alias persistentSettings : persistentSettings
2016-08-18 21:55:34 +03:00
property var currentWallet ;
2016-06-28 22:37:14 +03:00
property var transaction ;
2016-11-06 01:19:28 +02:00
property var transactionDescription ;
2017-11-17 03:16:35 +02:00
property var walletPassword
2016-10-05 23:34:47 +03:00
property bool isNewWallet: false
2016-10-08 02:37:47 +03:00
property int restoreHeight: 0
2016-11-02 01:17:58 +02:00
property bool daemonSynced: false
2018-03-20 20:19:30 +02:00
property bool walletSynced: false
2017-08-07 17:20:37 +03:00
property int maxWindowHeight: ( isAndroid || isIOS ) ? screenHeight : ( screenHeight < 900 ) ? 720 : 800 ;
2016-11-25 22:09:32 +02:00
property bool daemonRunning: false
2016-12-15 15:09:37 +02:00
property alias toolTip: toolTip
2017-01-14 00:21:58 +02:00
property string walletName
property bool viewOnly: false
property bool foundNewBlock: false
property int timeToUnlock: 0
2017-03-27 20:39:47 +03:00
property bool qrScannerEnabled: ( typeof builtWithScanner != "undefined" ) && builtWithScanner
2017-02-24 22:51:37 +02:00
property int blocksToSync: 1
2017-08-07 17:20:37 +03:00
property var isMobile: ( appWindow . width > 700 && ! isAndroid ) ? false : true
2018-12-16 06:10:12 +02:00
property bool isMining: false
2017-03-27 20:39:47 +03:00
property var cameraUi
2017-08-07 18:03:34 +03:00
property bool remoteNodeConnected: false
2017-08-08 12:29:02 +03:00
property bool androidCloseTapped: false ;
2018-12-13 20:02:02 +02:00
property int userLastActive ; // epoch
2017-08-07 18:03:34 +03:00
// Default daemon addresses
2018-03-27 21:41:15 +03:00
readonly property string localDaemonAddress : persistentSettings . nettype == NetworkType . MAINNET ? "localhost:18081" : persistentSettings . nettype == NetworkType . TESTNET ? "localhost:28081" : "localhost:38081"
2017-08-07 18:03:34 +03:00
property string currentDaemonAddress ;
2017-11-03 19:08:59 +02:00
property bool startLocalNodeCancelled: false
2018-04-05 10:21:43 +03:00
property int estimatedBlockchainSize: 50 // GB
2016-10-08 02:37:47 +03:00
2016-10-07 00:47:28 +03:00
// true if wallet ever synchronized
property bool walletInitialized : false
2016-07-13 15:24:40 +03:00
2014-07-13 15:27:50 +03:00
function altKeyReleased ( ) { ctrlPressed = false ; }
2016-06-15 16:34:55 +03:00
2014-07-13 15:27:50 +03:00
function showPageRequest ( page ) {
middlePanel . state = page
leftPanel . selectItem ( page )
}
2016-06-15 16:34:55 +03:00
2014-07-13 15:27:50 +03:00
function sequencePressed ( obj , seq ) {
2017-12-08 17:33:56 +02:00
if ( seq === undefined || ! leftPanel . enabled )
2014-07-09 17:18:48 +03:00
return
2014-07-13 15:27:50 +03:00
if ( seq === "Ctrl" ) {
ctrlPressed = true
return
}
2017-01-17 23:59:40 +02:00
// Dashboard is not implemented
// if(seq === "Ctrl+") middlePanel.state = "Dashboard"
if ( seq === "Ctrl+S" ) middlePanel . state = "Transfer"
2016-12-18 15:59:24 +02:00
else if ( seq === "Ctrl+R" ) middlePanel . state = "Receive"
else if ( seq === "Ctrl+K" ) middlePanel . state = "TxKey"
2014-07-13 15:27:50 +03:00
else if ( seq === "Ctrl+H" ) middlePanel . state = "History"
else if ( seq === "Ctrl+B" ) middlePanel . state = "AddressBook"
else if ( seq === "Ctrl+M" ) middlePanel . state = "Mining"
2016-12-18 15:59:24 +02:00
else if ( seq === "Ctrl+I" ) middlePanel . state = "Sign"
2018-07-03 16:26:37 +03:00
else if ( seq === "Ctrl+G" ) middlePanel . state = "SharedRingDB"
2016-12-18 15:59:24 +02:00
else if ( seq === "Ctrl+E" ) middlePanel . state = "Settings"
2018-09-27 17:01:14 +03:00
else if ( seq === "Ctrl+Y" ) leftPanel . keysClicked ( )
2017-01-17 23:59:40 +02:00
else if ( seq === "Ctrl+D" ) middlePanel . state = "Advanced"
2014-07-16 15:40:09 +03:00
else if ( seq === "Ctrl+Tab" || seq === "Alt+Tab" ) {
2016-12-18 16:24:48 +02:00
/ *
2014-07-13 15:27:50 +03:00
if ( middlePanel . state === "Dashboard" ) middlePanel . state = "Transfer"
2016-12-18 15:59:24 +02:00
else if ( middlePanel . state === "Transfer" ) middlePanel . state = "Receive"
else if ( middlePanel . state === "Receive" ) middlePanel . state = "TxKey"
2018-03-07 02:01:53 +02:00
else if ( middlePanel . state === "TxKey" ) middlePanel . state = "SharedRingDB"
else if ( middlePanel . state === "SharedRingDB" ) middlePanel . state = "History"
2014-07-13 15:27:50 +03:00
else if ( middlePanel . state === "History" ) middlePanel . state = "AddressBook"
else if ( middlePanel . state === "AddressBook" ) middlePanel . state = "Mining"
2016-12-18 15:59:24 +02:00
else if ( middlePanel . state === "Mining" ) middlePanel . state = "Sign"
else if ( middlePanel . state === "Sign" ) middlePanel . state = "Settings"
2014-07-13 15:27:50 +03:00
else if ( middlePanel . state === "Settings" ) middlePanel . state = "Dashboard"
2016-12-18 16:24:48 +02:00
* /
if ( middlePanel . state === "Settings" ) middlePanel . state = "Transfer"
2018-05-08 20:00:30 +03:00
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 = "Mining"
else if ( middlePanel . state === "Mining" ) middlePanel . state = "TxKey"
2018-03-07 02:01:53 +02:00
else if ( middlePanel . state === "TxKey" ) middlePanel . state = "SharedRingDB"
2018-05-08 20:00:30 +03:00
else if ( middlePanel . state === "SharedRingDB" ) middlePanel . state = "Sign"
2016-12-18 16:24:48 +02:00
else if ( middlePanel . state === "Sign" ) middlePanel . state = "Settings"
2014-07-16 15:40:09 +03:00
} else if ( seq === "Ctrl+Shift+Backtab" || seq === "Alt+Shift+Backtab" ) {
2016-12-18 16:24:48 +02:00
/ *
2014-07-13 15:27:50 +03:00
if ( middlePanel . state === "Dashboard" ) middlePanel . state = "Settings"
2016-12-18 16:24:48 +02:00
if ( middlePanel . state === "Settings" ) middlePanel . state = "Sign"
2016-12-18 15:59:24 +02:00
else if ( middlePanel . state === "Sign" ) middlePanel . state = "Mining"
2014-07-13 15:27:50 +03:00
else if ( middlePanel . state === "Mining" ) middlePanel . state = "AddressBook"
else if ( middlePanel . state === "AddressBook" ) middlePanel . state = "History"
2018-03-07 02:01:53 +02:00
else if ( middlePanel . state === "History" ) middlePanel . state = "SharedRingDB"
else if ( middlePanel . state === "SharedRingDB" ) middlePanel . state = "TxKey"
2016-12-18 15:59:24 +02:00
else if ( middlePanel . state === "TxKey" ) middlePanel . state = "Receive"
else if ( middlePanel . state === "Receive" ) middlePanel . state = "Transfer"
2014-07-13 15:27:50 +03:00
else if ( middlePanel . state === "Transfer" ) middlePanel . state = "Dashboard"
2016-12-18 16:24:48 +02:00
* /
if ( middlePanel . state === "Settings" ) middlePanel . state = "Sign"
2018-05-11 23:30:40 +03:00
else if ( middlePanel . state === "Sign" ) middlePanel . state = "SharedRingDB"
2018-03-07 02:01:53 +02:00
else if ( middlePanel . state === "SharedRingDB" ) middlePanel . state = "TxKey"
2018-05-11 23:30:40 +03:00
else if ( middlePanel . state === "TxKey" ) middlePanel . state = "Mining"
else if ( middlePanel . state === "Mining" ) 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"
2016-12-18 16:24:48 +02:00
else if ( middlePanel . state === "Transfer" ) middlePanel . state = "Settings"
2014-07-13 15:27:50 +03:00
}
2017-12-11 13:15:16 +02:00
if ( middlePanel . state !== "Advanced" ) updateBalance ( ) ;
2014-07-09 19:03:37 +03:00
leftPanel . selectItem ( middlePanel . state )
2014-07-09 17:18:48 +03:00
}
2014-07-13 15:27:50 +03:00
function sequenceReleased ( obj , seq ) {
if ( seq === "Ctrl" )
ctrlPressed = false
}
2016-12-04 14:13:59 +02:00
function mousePressed ( obj , mouseX , mouseY ) { }
function mouseReleased ( obj , mouseX , mouseY ) { }
2014-07-07 20:08:30 +03:00
2017-02-23 20:47:58 +02:00
function loadPage ( page ) {
middlePanel . state = page ;
leftPanel . selectItem ( page ) ;
}
2016-10-29 17:48:10 +03:00
function openWalletFromFile ( ) {
persistentSettings . restore_height = 0
2016-11-23 20:32:30 +02:00
restoreHeight = 0 ;
2016-10-29 17:48:10 +03:00
persistentSettings . is_recovering = false
2017-11-17 03:16:35 +02:00
walletPassword = ""
2016-10-29 17:48:10 +03:00
fileDialog . open ( ) ;
}
2016-06-15 16:34:55 +03:00
function initialize ( ) {
2016-07-13 15:24:40 +03:00
console . log ( "initializing.." )
2016-11-07 12:27:53 +02:00
2016-12-31 12:56:08 +02:00
// Use stored log level
2017-02-05 14:38:43 +02:00
if ( persistentSettings . logLevel == 5 )
walletManager . setLogCategories ( persistentSettings . logCategories )
else
walletManager . setLogLevel ( persistentSettings . logLevel )
2016-12-31 12:56:08 +02:00
2016-07-19 23:45:12 +03:00
// setup language
var locale = persistentSettings . locale
if ( locale !== "" ) {
translationManager . setLanguage ( locale . split ( "_" ) [ 0 ] ) ;
}
2017-03-09 17:46:03 +02:00
// Reload transfer page with translations enabled
middlePanel . transferView . onPageCompleted ( ) ;
2016-12-04 14:13:59 +02:00
// If currentWallet exists, we're just switching daemon - close/reopen wallet
if ( typeof currentWallet !== "undefined" && currentWallet !== null ) {
console . log ( "Daemon change - closing " + currentWallet )
2016-12-15 14:18:04 +02:00
closeWallet ( ) ;
2016-12-14 14:11:20 +02:00
currentWallet = undefined
2017-08-07 18:03:34 +03:00
} else if ( ! walletInitialized ) {
2016-10-29 17:48:10 +03:00
// set page to transfer if not changing daemon
middlePanel . state = "Transfer" ;
leftPanel . selectItem ( middlePanel . state )
2016-10-05 01:18:50 +03:00
}
2016-07-19 23:45:12 +03:00
2017-08-07 18:03:34 +03:00
// Local daemon settings
walletManager . setDaemonAddress ( localDaemonAddress )
2018-12-13 20:02:02 +02:00
// enable user inactivity timer
userInActivityTimer . running = true ;
2017-08-07 18:03:34 +03:00
2016-08-19 14:44:44 +03:00
// wallet already opened with wizard, we just need to initialize it
2017-08-06 18:11:12 +03:00
if ( typeof wizard . m_wallet !== 'undefined' ) {
2016-10-05 01:18:50 +03:00
console . log ( "using wizard wallet" )
2016-10-08 02:37:47 +03:00
//Set restoreHeight
2018-06-04 11:05:29 +03:00
if ( persistentSettings . restore_height == 0 && persistentSettings . is_recovering_from_device && walletManager . localDaemonSynced ( ) ) {
persistentSettings . restore_height = walletManager . blockchainHeight ( ) - 1 ;
}
2016-10-10 17:37:03 +03:00
if ( persistentSettings . restore_height > 0 ) {
2016-11-23 20:32:30 +02:00
// We store restore height in own variable for performance reasons.
2016-10-10 17:37:03 +03:00
restoreHeight = persistentSettings . restore_height
2016-10-08 02:37:47 +03:00
}
2017-08-06 18:11:12 +03:00
connectWallet ( wizard . m_wallet )
2016-10-08 02:37:47 +03:00
2016-10-05 23:34:47 +03:00
isNewWallet = true
2016-10-05 01:18:50 +03:00
// We don't need the wizard wallet any more - delete to avoid conflict with daemon adress change
2017-08-06 18:11:12 +03:00
delete wizard . m_wallet
2016-06-15 16:34:55 +03:00
} else {
2016-08-16 23:21:46 +03:00
var wallet_path = walletPath ( ) ;
2017-04-03 19:51:55 +03:00
if ( isIOS )
wallet_path = moneroAccountsDir + wallet_path ;
2018-01-16 19:00:53 +02:00
// console.log("opening wallet at: ", wallet_path, "with password: ", appWindow.walletPassword);
2018-03-27 21:41:15 +03:00
console . log ( "opening wallet at: " , wallet_path , ", network type: " , persistentSettings . nettype == NetworkType . MAINNET ? "mainnet" : persistentSettings . nettype == NetworkType . TESTNET ? "testnet" : "stagenet" ) ;
2017-11-17 03:16:35 +02:00
walletManager . openWalletAsync ( wallet_path , walletPassword ,
2018-07-06 12:39:58 +03:00
persistentSettings . nettype , persistentSettings . kdfRounds ) ;
2016-06-15 16:34:55 +03:00
}
2016-10-29 17:48:10 +03:00
2017-12-07 18:01:16 +02:00
// Hide titlebar based on persistentSettings.customDecorations
titleBar . visible = persistentSettings . customDecorations ;
2016-06-17 16:35:07 +03:00
}
2017-12-07 18:01:16 +02:00
2016-12-15 14:18:04 +02:00
function closeWallet ( ) {
2016-06-17 16:35:07 +03:00
2016-12-15 14:18:04 +02:00
// Disconnect all listeners
2016-12-16 01:12:27 +02:00
if ( typeof currentWallet !== "undefined" && currentWallet !== null ) {
currentWallet . refreshed . disconnect ( onWalletRefresh )
currentWallet . updated . disconnect ( onWalletUpdate )
currentWallet . newBlock . disconnect ( onWalletNewBlock )
currentWallet . moneySpent . disconnect ( onWalletMoneySent )
currentWallet . moneyReceived . disconnect ( onWalletMoneyReceived )
2017-01-12 20:57:30 +02:00
currentWallet . unconfirmedMoneyReceived . disconnect ( onWalletUnconfirmedMoneyReceived )
2016-12-16 01:12:27 +02:00
currentWallet . transactionCreated . disconnect ( onTransactionCreated )
currentWallet . connectionStatusChanged . disconnect ( onWalletConnectionStatusChanged )
middlePanel . paymentClicked . disconnect ( handlePayment ) ;
middlePanel . sweepUnmixableClicked . disconnect ( handleSweepUnmixable ) ;
2017-11-20 09:24:29 +02:00
middlePanel . getProofClicked . disconnect ( handleGetProof ) ;
middlePanel . checkProofClicked . disconnect ( handleCheckProof ) ;
2016-12-16 01:12:27 +02:00
}
2017-08-08 12:40:54 +03:00
2016-12-15 14:18:04 +02:00
currentWallet = undefined ;
2017-08-08 12:40:54 +03:00
walletManager . closeWallet ( ) ;
2016-12-15 14:18:04 +02:00
}
function connectWallet ( wallet ) {
currentWallet = wallet
2018-03-21 04:07:38 +02:00
// TODO:
// When the wallet variable is undefined, it yields a zero balance.
// This can scare users, restart the GUI (as a quick fix).
//
// To reproduce, follow these steps:
// 1) Open the GUI, load up a wallet that has a balance
// 2) Settings -> close wallet
// 3) Create a new wallet
// 4) Settings -> close wallet
// 5) Open the wallet from step 1
if ( ! wallet || wallet === undefined || wallet . path === undefined ) {
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString ;
informationPopup . text = qsTr ( "Couldn't open wallet: " ) + 'please restart GUI.' ;
informationPopup . icon = StandardIcon . Critical
informationPopup . open ( )
informationPopup . onCloseCallback = function ( ) {
appWindow . close ( ) ;
}
}
2017-01-30 11:37:14 +02:00
walletName = usefulName ( wallet . path )
2016-12-31 15:47:40 +02:00
updateSyncing ( false )
2016-12-15 14:18:04 +02:00
2017-01-12 21:53:27 +02:00
viewOnly = currentWallet . viewOnly ;
2017-07-31 16:11:01 +03:00
// New wallets saves the testnet flag in keys file.
2018-03-05 18:19:45 +02:00
if ( persistentSettings . nettype != currentWallet . nettype ) {
console . log ( "Using network type from keys file" )
persistentSettings . nettype = currentWallet . nettype ;
2017-07-31 16:11:01 +03:00
}
2016-12-04 14:13:59 +02:00
// connect handlers
2016-08-19 14:44:44 +03:00
currentWallet . refreshed . connect ( onWalletRefresh )
currentWallet . updated . connect ( onWalletUpdate )
2016-09-26 22:55:25 +03:00
currentWallet . newBlock . connect ( onWalletNewBlock )
2016-10-07 00:47:28 +03:00
currentWallet . moneySpent . connect ( onWalletMoneySent )
currentWallet . moneyReceived . connect ( onWalletMoneyReceived )
2017-01-12 20:57:30 +02:00
currentWallet . unconfirmedMoneyReceived . connect ( onWalletUnconfirmedMoneyReceived )
2016-11-08 19:05:33 +02:00
currentWallet . transactionCreated . connect ( onTransactionCreated )
2016-11-26 16:50:04 +02:00
currentWallet . connectionStatusChanged . connect ( onWalletConnectionStatusChanged )
2016-12-04 14:13:59 +02:00
middlePanel . paymentClicked . connect ( handlePayment ) ;
middlePanel . sweepUnmixableClicked . connect ( handleSweepUnmixable ) ;
2017-11-20 09:24:29 +02:00
middlePanel . getProofClicked . connect ( handleGetProof ) ;
middlePanel . checkProofClicked . connect ( handleCheckProof ) ;
2016-11-07 00:40:26 +02:00
2017-08-07 18:03:34 +03:00
2016-10-10 22:40:58 +03:00
console . log ( "Recovering from seed: " , persistentSettings . is_recovering )
console . log ( "restore Height" , persistentSettings . restore_height )
2017-02-25 23:16:58 +02:00
// Use saved daemon rpc login settings
2017-08-07 18:03:34 +03:00
currentWallet . setDaemonLogin ( persistentSettings . daemonUsername , persistentSettings . daemonPassword )
if ( persistentSettings . useRemoteNode )
currentDaemonAddress = persistentSettings . remoteNodeAddress
else
currentDaemonAddress = localDaemonAddress
2017-02-25 23:16:58 +02:00
2017-08-07 18:03:34 +03:00
console . log ( "initializing with daemon address: " , currentDaemonAddress )
2018-06-04 11:05:29 +03:00
currentWallet . initAsync ( currentDaemonAddress , 0 , persistentSettings . is_recovering , persistentSettings . is_recovering_from_device , persistentSettings . restore_height ) ;
// save wallet keys in case wallet settings have been changed in the init
currentWallet . setPassword ( walletPassword ) ;
2016-08-19 14:44:44 +03:00
}
2016-08-16 23:21:46 +03:00
function walletPath ( ) {
2016-10-29 17:48:10 +03:00
var wallet_path = persistentSettings . wallet_path
2016-08-16 23:21:46 +03:00
return wallet_path ;
}
2017-01-03 22:54:40 +02:00
function usefulName ( path ) {
// arbitrary "short enough" limit
if ( path . length < 32 )
return path
return path . replace ( /.*[\/\\]/ , '' ) . replace ( /\.keys$/ , '' )
}
2017-12-11 13:15:16 +02:00
function updateBalance ( ) {
2017-07-04 06:34:09 +03:00
if ( ! currentWallet )
return ;
2018-12-12 10:33:24 +02:00
var balance_unlocked = qsTr ( "HIDDEN" ) ;
var balance = qsTr ( "HIDDEN" ) ;
2018-12-15 06:02:01 +02:00
if ( ! hideBalanceForced && ! persistentSettings . hideBalance ) {
2018-12-12 10:33:24 +02:00
balance_unlocked = walletManager . displayAmount ( currentWallet . unlockedBalance ( currentWallet . currentSubaddressAccount ) ) ;
balance = walletManager . displayAmount ( currentWallet . balance ( currentWallet . currentSubaddressAccount ) ) ;
}
middlePanel . unlockedBalanceText = balance_unlocked ;
leftPanel . unlockedBalanceText = balance_unlocked ;
middlePanel . balanceText = balance ;
leftPanel . balanceText = balance ;
2017-12-11 13:15:16 +02:00
}
2017-01-31 11:34:18 +02:00
function onWalletConnectionStatusChanged ( status ) {
2017-02-05 14:49:25 +02:00
console . log ( "Wallet connection status changed " + status )
2016-11-26 16:50:04 +02:00
middlePanel . updateStatus ( ) ;
2017-01-31 11:34:18 +02:00
leftPanel . networkStatus . connected = status
2017-02-05 14:49:25 +02:00
2018-06-20 23:26:45 +03:00
// update local daemon status.
if ( ! isMobile && walletManager . isDaemonLocal ( appWindow . persistentSettings . daemon_address ) )
daemonRunning = status ;
2017-03-23 23:54:35 +02:00
// Update fee multiplier dropdown on transfer page
middlePanel . transferView . updatePriorityDropdown ( ) ;
2017-02-05 14:49:25 +02:00
// If wallet isnt connected and no daemon is running - Ask
2018-03-05 18:19:45 +02:00
if ( ! isMobile && walletManager . isDaemonLocal ( appWindow . persistentSettings . daemon_address ) && ! walletInitialized && status === Wallet . ConnectionStatus_Disconnected && ! daemonManager . running ( persistentSettings . nettype ) ) {
2017-02-05 14:49:25 +02:00
daemonManagerDialog . open ( ) ;
}
// initialize transaction history once wallet is initialized first time;
if ( ! walletInitialized ) {
2017-07-04 06:34:09 +03:00
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount )
2017-02-05 14:49:25 +02:00
walletInitialized = true
}
2017-01-31 11:34:18 +02:00
}
2016-11-26 16:50:04 +02:00
2016-08-18 21:55:34 +03:00
function onWalletOpened ( wallet ) {
2017-01-03 22:54:40 +02:00
walletName = usefulName ( wallet . path )
2016-08-18 21:55:34 +03:00
console . log ( ">>> wallet opened: " + wallet )
if ( wallet . status !== Wallet . Status_Ok ) {
2017-11-17 03:16:35 +02:00
passwordDialog . onAcceptedCallback = function ( ) {
walletPassword = passwordDialog . password ;
appWindow . initialize ( ) ;
}
passwordDialog . onRejectedCallback = function ( ) {
walletPassword = "" ;
//appWindow.enableUI(false)
rootItem . state = "wizard" ;
}
// opening with password but password doesn't match
console . error ( "Error opening wallet with password: " , wallet . errorString ) ;
2018-10-24 06:50:56 +03:00
passwordDialog . showError ( qsTr ( "Couldn't open wallet: " ) + wallet . errorString ) ;
2017-11-17 03:16:35 +02:00
console . log ( "closing wallet async : " + wallet . address )
closeWallet ( ) ;
2016-08-18 21:55:34 +03:00
return ;
}
// wallet opened successfully, subscribing for wallet updates
2016-08-19 14:44:44 +03:00
connectWallet ( wallet )
2016-08-18 21:55:34 +03:00
}
function onWalletClosed ( walletAddress ) {
console . log ( ">>> wallet closed: " + walletAddress )
}
2016-08-16 23:21:46 +03:00
2016-06-17 16:35:07 +03:00
function onWalletUpdate ( ) {
2016-07-13 15:24:40 +03:00
console . log ( ">>> wallet updated" )
2017-12-11 13:15:16 +02:00
updateBalance ( ) ;
2017-03-05 01:05:31 +02:00
// Update history if new block found since last update
if ( foundNewBlock ) {
2017-01-14 00:21:58 +02:00
foundNewBlock = false ;
console . log ( "New block found - updating history" )
2017-07-04 06:34:09 +03:00
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount )
2017-01-14 00:21:58 +02:00
timeToUnlock = currentWallet . history . minutesToUnlock
2017-08-07 18:03:34 +03:00
leftPanel . minutesToUnlockTxt = ( timeToUnlock > 0 ) ? ( timeToUnlock == 20 ) ? qsTr ( "Unlocked balance (waiting for block)" ) : qsTr ( "Unlocked balance (~%1 min)" ) . arg ( timeToUnlock ) : qsTr ( "Unlocked balance" ) ;
2017-01-14 00:21:58 +02:00
}
2016-07-13 15:24:40 +03:00
}
2017-08-07 18:03:34 +03:00
function connectRemoteNode ( ) {
console . log ( "connecting remote node" ) ;
persistentSettings . useRemoteNode = true ;
2018-04-07 08:10:31 +03:00
currentDaemonAddress = persistentSettings . remoteNodeAddress ;
currentWallet . initAsync ( currentDaemonAddress ) ;
walletManager . setDaemonAddress ( currentDaemonAddress ) ;
2017-08-07 18:03:34 +03:00
remoteNodeConnected = true ;
}
function disconnectRemoteNode ( ) {
console . log ( "disconnecting remote node" ) ;
persistentSettings . useRemoteNode = false ;
currentDaemonAddress = localDaemonAddress
currentWallet . initAsync ( currentDaemonAddress ) ;
2018-04-07 08:10:31 +03:00
walletManager . setDaemonAddress ( currentDaemonAddress ) ;
2017-08-07 18:03:34 +03:00
remoteNodeConnected = false ;
}
2016-07-13 15:24:40 +03:00
function onWalletRefresh ( ) {
console . log ( ">>> wallet refreshed" )
2016-11-02 01:17:58 +02:00
2016-11-26 18:01:29 +02:00
// Daemon connected
2017-02-23 20:47:58 +02:00
leftPanel . networkStatus . connected = currentWallet . connected ( )
2016-11-26 18:01:29 +02:00
2018-03-20 20:19:30 +02:00
// Wallet height
var bcHeight = currentWallet . blockChainHeight ( ) ;
2016-11-02 01:17:58 +02:00
// Check daemon status
2016-10-04 00:29:30 +03:00
var dCurrentBlock = currentWallet . daemonBlockChainHeight ( ) ;
var dTargetBlock = currentWallet . daemonBlockChainTargetHeight ( ) ;
2016-11-02 01:17:58 +02:00
// Daemon fully synced
// TODO: implement onDaemonSynced or similar in wallet API and don't start refresh thread before daemon is synced
2017-02-24 19:07:46 +02:00
// targetBlock = currentBlock = 1 before network connection is established.
daemonSynced = dCurrentBlock >= dTargetBlock && dTargetBlock != 1
2018-03-20 20:19:30 +02:00
walletSynced = bcHeight >= dTargetBlock
// Update progress bars
if ( ! daemonSynced ) {
leftPanel . daemonProgressBar . updateProgress ( dCurrentBlock , dTargetBlock , dTargetBlock - dCurrentBlock ) ;
2018-03-15 17:35:47 +02:00
leftPanel . progressBar . updateProgress ( 0 , dTargetBlock , dTargetBlock , qsTr ( "Waiting for daemon to sync" ) ) ;
2018-03-20 20:19:30 +02:00
} 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" ) )
}
2017-01-31 11:34:18 +02:00
// Update wallet sync progress
2017-02-23 20:47:58 +02:00
updateSyncing ( ( currentWallet . connected ( ) !== Wallet . ConnectionStatus_Disconnected ) && ! daemonSynced )
2017-01-31 11:34:18 +02:00
// Update transfer page status
2016-11-27 14:35:27 +02:00
middlePanel . updateStatus ( ) ;
2016-11-02 01:17:58 +02:00
2016-11-10 13:11:25 +02:00
// Refresh is succesfull if blockchain height > 1
2018-03-20 20:19:30 +02:00
if ( bcHeight > 1 ) {
2016-11-10 13:11:25 +02:00
// Save new wallet after first refresh
// Wallet is nomrmally saved to disk on app exit. This prevents rescan from block 0 after app crash
if ( isNewWallet ) {
console . log ( "Saving wallet after first refresh" ) ;
2016-11-10 13:23:43 +02:00
currentWallet . store ( )
2016-11-10 13:11:25 +02:00
isNewWallet = false
2017-07-04 06:34:09 +03:00
// Update History
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount ) ;
2016-11-10 13:11:25 +02:00
}
2016-10-28 01:24:02 +03:00
// recovering from seed is finished after first refresh
if ( persistentSettings . is_recovering ) {
persistentSettings . is_recovering = false
}
}
2017-08-08 13:23:01 +03:00
// Update history on every refresh if it's empty
if ( currentWallet . history . count == 0 )
2017-07-04 06:34:09 +03:00
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount )
2017-08-08 13:23:01 +03:00
2016-07-14 13:09:39 +03:00
onWalletUpdate ( ) ;
2016-06-15 16:34:55 +03:00
}
2016-12-21 15:30:15 +02:00
function startDaemon ( flags ) {
2017-02-25 21:25:16 +02:00
// Pause refresh while starting daemon
currentWallet . pauseRefresh ( ) ;
2016-11-26 12:31:27 +02:00
appWindow . showProcessingSplash ( qsTr ( "Waiting for daemon to start..." ) )
2018-03-05 18:19:45 +02:00
daemonManager . start ( flags , persistentSettings . nettype , persistentSettings . blockchainDataDir , persistentSettings . bootstrapNodeAddress ) ;
2016-12-21 15:30:15 +02:00
persistentSettings . daemonFlags = flags
2016-11-26 12:31:27 +02:00
}
2016-11-26 18:01:29 +02:00
2016-11-26 12:31:27 +02:00
function stopDaemon ( ) {
appWindow . showProcessingSplash ( qsTr ( "Waiting for daemon to stop..." ) )
2018-03-05 18:19:45 +02:00
daemonManager . stop ( persistentSettings . nettype ) ;
2016-11-26 12:31:27 +02:00
}
2016-11-25 22:09:32 +02:00
function onDaemonStarted ( ) {
console . log ( "daemon started" ) ;
daemonRunning = true ;
2017-02-23 20:47:58 +02:00
hideProcessingSplash ( ) ;
2017-02-25 15:57:39 +02:00
currentWallet . connected ( true ) ;
2017-02-25 21:25:16 +02:00
// resume refresh
currentWallet . startRefresh ( ) ;
2016-11-25 22:09:32 +02:00
}
function onDaemonStopped ( ) {
console . log ( "daemon stopped" ) ;
2016-12-14 14:48:12 +02:00
hideProcessingSplash ( ) ;
2016-11-25 22:09:32 +02:00
daemonRunning = false ;
2017-02-23 20:47:58 +02:00
currentWallet . connected ( true ) ;
2016-11-25 22:09:32 +02:00
}
2017-02-25 21:25:16 +02:00
function onDaemonStartFailure ( ) {
console . log ( "daemon start failed" ) ;
hideProcessingSplash ( ) ;
// resume refresh
currentWallet . startRefresh ( ) ;
daemonRunning = false ;
informationPopup . title = qsTr ( "Daemon failed to start" ) + translationManager . emptyString ;
informationPopup . text = 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 ( ) ;
}
2017-01-31 11:34:18 +02:00
function onWalletNewBlock ( blockHeight , targetHeight ) {
// Update progress bar
2017-02-24 22:51:37 +02:00
var remaining = targetHeight - blockHeight ;
if ( blocksToSync < remaining ) {
blocksToSync = remaining ;
}
leftPanel . progressBar . updateProgress ( blockHeight , targetHeight , blocksToSync ) ;
2018-03-20 20:19:30 +02:00
// If wallet is syncing, daemon is already synced
leftPanel . daemonProgressBar . updateProgress ( 1 , 1 , 0 , qsTr ( "Daemon is synchronized" ) ) ;
2017-01-31 11:34:18 +02:00
foundNewBlock = true ;
2016-09-26 22:55:25 +03:00
}
2016-10-07 00:47:28 +03:00
function onWalletMoneyReceived ( txId , amount ) {
// refresh transaction history here
currentWallet . refresh ( )
2017-08-08 13:23:01 +03:00
console . log ( "Confirmed money found" )
// history refresh is handled by walletUpdated
2017-07-04 06:34:09 +03:00
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount ) // this will refresh model
currentWallet . subaddress . refresh ( currentWallet . currentSubaddressAccount )
2016-10-07 00:47:28 +03:00
}
2017-01-12 20:57:30 +02:00
function onWalletUnconfirmedMoneyReceived ( txId , amount ) {
// refresh history
2017-01-14 00:21:58 +02:00
console . log ( "unconfirmed money found" )
2017-07-04 06:34:09 +03:00
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount )
2017-01-12 20:57:30 +02:00
}
2016-10-07 00:47:28 +03:00
function onWalletMoneySent ( txId , amount ) {
// refresh transaction history here
2018-03-24 00:53:29 +02:00
console . log ( "monero sent found" )
2016-10-07 00:47:28 +03:00
currentWallet . refresh ( )
2017-07-04 06:34:09 +03:00
currentWallet . history . refresh ( currentWallet . currentSubaddressAccount ) // this will refresh model
2016-10-07 00:47:28 +03:00
}
2016-06-07 16:26:25 +03:00
function walletsFound ( ) {
2016-10-30 18:58:12 +02:00
if ( persistentSettings . wallet_path . length > 0 ) {
2017-04-03 19:51:55 +03:00
if ( isIOS )
return walletManager . walletExists ( moneroAccountsDir + persistentSettings . wallet_path ) ;
else
return walletManager . walletExists ( persistentSettings . wallet_path ) ;
2016-06-07 16:26:25 +03:00
}
2016-11-13 15:14:26 +02:00
return false ;
2016-06-07 16:26:25 +03:00
}
2016-11-08 19:05:33 +02:00
function onTransactionCreated ( pendingTransaction , address , paymentId , mixinCount ) {
console . log ( "Transaction created" ) ;
hideProcessingSplash ( ) ;
transaction = pendingTransaction ;
// validate address;
if ( transaction . status !== PendingTransaction . Status_Ok ) {
console . error ( "Can't create transaction: " , transaction . errorString ) ;
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString ;
2017-02-23 20:47:58 +02:00
if ( currentWallet . connected ( ) == Wallet . ConnectionStatus_WrongVersion )
2016-11-08 19:05:33 +02:00
informationPopup . text = qsTr ( "Can't create transaction: Wrong daemon version: " ) + transaction . errorString
else
informationPopup . text = qsTr ( "Can't create transaction: " ) + transaction . errorString
informationPopup . icon = StandardIcon . Critical
informationPopup . onCloseCallback = null
informationPopup . open ( ) ;
// deleting transaction object, we don't want memleaks
currentWallet . disposeTransaction ( transaction ) ;
2016-11-11 23:54:17 +02:00
} else if ( transaction . txCount == 0 ) {
2017-05-03 16:29:23 +03:00
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString
2016-11-11 23:54:17 +02:00
informationPopup . text = qsTr ( "No unmixable outputs to sweep" ) + translationManager . emptyString
informationPopup . icon = StandardIcon . Information
informationPopup . onCloseCallback = null
informationPopup . open ( )
// deleting transaction object, we don't want memleaks
currentWallet . disposeTransaction ( transaction ) ;
2016-11-08 19:05:33 +02:00
} else {
console . log ( "Transaction created, amount: " + walletManager . displayAmount ( transaction . amount )
+ ", fee: " + walletManager . displayAmount ( transaction . fee ) ) ;
// here we show confirmation popup;
2018-03-20 01:58:49 +02:00
transactionConfirmationPopup . title = qsTr ( "Please confirm transaction:\n" ) + translationManager . emptyString ;
2018-03-27 22:12:10 +03:00
transactionConfirmationPopup . text = "" ;
2018-03-31 18:26:32 +03:00
transactionConfirmationPopup . text += ( address === "" ? "" : ( qsTr ( "Address: " ) + address ) ) ;
transactionConfirmationPopup . text += ( paymentId === "" ? "" : ( qsTr ( "\nPayment ID: " ) + paymentId ) ) ;
transactionConfirmationPopup . text += qsTr ( "\n\nAmount: " ) + walletManager . displayAmount ( transaction . amount ) ;
transactionConfirmationPopup . text += qsTr ( "\nFee: " ) + walletManager . displayAmount ( transaction . fee ) ;
transactionConfirmationPopup . text += qsTr ( "\nRingsize: " ) + ( mixinCount + 1 ) ;
transactionConfirmationPopup . text += qsTr ( "\n\nNumber of transactions: " ) + transaction . txCount
2018-04-03 06:29:23 +03:00
transactionConfirmationPopup . text += ( transactionDescription === "" ? "" : ( qsTr ( "\nDescription: " ) + transactionDescription ) )
2018-03-20 01:58:49 +02:00
for ( var i = 0 ; i < transaction . subaddrIndices . length ; ++ i ) {
transactionConfirmationPopup . text += qsTr ( "\nSpending address index: " ) + transaction . subaddrIndices [ i ] ;
}
transactionConfirmationPopup . text += translationManager . emptyString ;
2016-11-08 19:05:33 +02:00
transactionConfirmationPopup . icon = StandardIcon . Question
transactionConfirmationPopup . open ( )
}
}
2016-08-17 15:14:43 +03:00
2016-06-28 22:37:14 +03:00
// called on "transfer"
2017-01-12 21:53:27 +02:00
function handlePayment ( address , paymentId , amount , mixinCount , priority , description , createFile ) {
2016-06-27 15:45:48 +03:00
console . log ( "Creating transaction: " )
console . log ( "\taddress: " , address ,
", payment_id: " , paymentId ,
", amount: " , amount ,
", mixins: " , mixinCount ,
2016-11-06 01:19:28 +02:00
", priority: " , priority ,
", description: " , description ) ;
2016-06-27 15:45:48 +03:00
2016-11-08 19:05:33 +02:00
showProcessingSplash ( "Creating transaction" ) ;
2016-11-12 15:28:13 +02:00
2016-11-08 19:05:33 +02:00
transactionDescription = description ;
2016-11-12 15:28:13 +02:00
2016-08-23 16:07:52 +03:00
// validate amount;
2016-11-09 15:00:43 +02:00
if ( amount !== "(all)" ) {
var amountxmr = walletManager . amountFromString ( amount ) ;
console . log ( "integer amount: " , amountxmr ) ;
console . log ( "integer unlocked" , currentWallet . unlockedBalance )
if ( amountxmr <= 0 ) {
2017-03-21 00:13:03 +02:00
hideProcessingSplash ( )
2016-11-09 15:00:43 +02:00
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString ;
informationPopup . text = qsTr ( "Amount is wrong: expected number from %1 to %2" )
. arg ( walletManager . displayAmount ( 0 ) )
. arg ( walletManager . maximumAllowedAmountAsSting ( ) )
+ translationManager . emptyString
2016-08-23 16:07:52 +03:00
2016-11-09 15:00:43 +02:00
informationPopup . icon = StandardIcon . Critical
informationPopup . onCloseCallback = null
informationPopup . open ( )
return ;
} else if ( amountxmr > currentWallet . unlockedBalance ) {
2017-03-21 00:13:03 +02:00
hideProcessingSplash ( )
2016-11-09 15:00:43 +02:00
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString ;
2017-03-21 03:28:43 +02:00
informationPopup . text = qsTr ( "Insufficient funds. Unlocked balance: %1" )
2016-11-09 15:00:43 +02:00
. arg ( walletManager . displayAmount ( currentWallet . unlockedBalance ) )
+ translationManager . emptyString
2016-10-12 00:02:21 +03:00
2016-11-09 15:00:43 +02:00
informationPopup . icon = StandardIcon . Critical
informationPopup . onCloseCallback = null
informationPopup . open ( )
return ;
}
2016-08-23 16:07:52 +03:00
}
2016-11-09 15:00:43 +02:00
if ( amount === "(all)" )
2016-11-18 21:17:13 +02:00
currentWallet . createTransactionAllAsync ( address , paymentId , mixinCount , priority ) ;
2016-11-09 15:00:43 +02:00
else
currentWallet . createTransactionAsync ( address , paymentId , amountxmr , mixinCount , priority ) ;
2016-06-28 22:37:14 +03:00
}
2017-01-12 21:53:27 +02:00
//Choose where to save transaction
FileDialog {
id: saveTxDialog
title: "Please choose a location"
folder: "file://" + moneroAccountsDir
selectExisting: false ;
onAccepted: {
handleTransactionConfirmed ( )
}
onRejected: {
// do nothing
}
}
2016-11-08 22:23:50 +02:00
function handleSweepUnmixable ( ) {
console . log ( "Creating transaction: " )
transaction = currentWallet . createSweepUnmixableTransaction ( ) ;
if ( transaction . status !== PendingTransaction . Status_Ok ) {
console . error ( "Can't create transaction: " , transaction . errorString ) ;
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString ;
informationPopup . text = qsTr ( "Can't create transaction: " ) + transaction . errorString
informationPopup . icon = StandardIcon . Critical
informationPopup . onCloseCallback = null
informationPopup . open ( ) ;
// deleting transaction object, we don't want memleaks
currentWallet . disposeTransaction ( transaction ) ;
} else if ( transaction . txCount == 0 ) {
2017-05-03 16:29:23 +03:00
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString
2016-11-08 22:23:50 +02:00
informationPopup . text = qsTr ( "No unmixable outputs to sweep" ) + translationManager . emptyString
informationPopup . icon = StandardIcon . Information
informationPopup . onCloseCallback = null
informationPopup . open ( )
// 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 show confirmation popup;
transactionConfirmationPopup . title = qsTr ( "Confirmation" ) + translationManager . emptyString
transactionConfirmationPopup . text = qsTr ( "Please confirm transaction:\n" )
+ qsTr ( "\n\nAmount: " ) + walletManager . displayAmount ( transaction . amount )
+ qsTr ( "\nFee: " ) + walletManager . displayAmount ( transaction . fee )
+ translationManager . emptyString
transactionConfirmationPopup . icon = StandardIcon . Question
transactionConfirmationPopup . open ( )
// committing transaction
}
}
2016-06-28 22:37:14 +03:00
// called after user confirms transaction
2017-01-12 21:53:27 +02:00
function handleTransactionConfirmed ( fileName ) {
2016-11-06 01:19:28 +02:00
// grab transaction.txid before commit, since it clears it.
// we actually need to copy it, because QML will incredibly
// call the function multiple times when the variable is used
// after commit, where it returns another result...
// Of course, this loop is also calling the function multiple
// times, but at least with the same result.
var txid = [ ] , txid_org = transaction . txid , txid_text = ""
for ( var i = 0 ; i < txid_org . length ; ++ i )
txid [ i ] = txid_org [ i ]
2017-01-12 21:53:27 +02:00
// View only wallet - we save the tx
if ( viewOnly && saveTxDialog . fileUrl ) {
// No file specified - abort
if ( ! saveTxDialog . fileUrl ) {
currentWallet . disposeTransaction ( transaction )
return ;
}
var path = walletManager . urlToLocalPath ( saveTxDialog . fileUrl )
// Store to file
transaction . setFilename ( path ) ;
}
2016-06-28 22:37:14 +03:00
if ( ! transaction . commit ( ) ) {
console . log ( "Error committing transaction: " + transaction . errorString ) ;
2016-07-20 22:28:11 +03:00
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString
2016-06-28 22:37:14 +03:00
informationPopup . text = qsTr ( "Couldn't send the money: " ) + transaction . errorString
informationPopup . icon = StandardIcon . Critical
} else {
2016-07-20 22:28:11 +03:00
informationPopup . title = qsTr ( "Information" ) + translationManager . emptyString
2016-11-06 01:19:28 +02:00
for ( var i = 0 ; i < txid . length ; ++ i ) {
if ( txid_text . length > 0 )
txid_text += ", "
txid_text += txid [ i ]
}
2018-03-24 00:53:29 +02:00
informationPopup . text = ( viewOnly ) ? qsTr ( "Transaction saved to file: %1" ) . arg ( path ) : qsTr ( "Monero sent successfully: %1 transaction(s) " ) . arg ( txid . length ) + txid_text + translationManager . emptyString
2016-06-28 22:37:14 +03:00
informationPopup . icon = StandardIcon . Information
2016-11-06 01:19:28 +02:00
if ( transactionDescription . length > 0 ) {
for ( var i = 0 ; i < txid . length ; ++ i )
currentWallet . setUserNote ( txid [ i ] , transactionDescription ) ;
}
2017-02-04 15:44:30 +02:00
// Clear tx fields
middlePanel . transferView . clearFields ( )
2016-06-16 17:13:46 +03:00
}
2016-08-17 15:14:43 +03:00
informationPopup . onCloseCallback = null
2016-06-28 22:37:14 +03:00
informationPopup . open ( )
2016-08-23 16:07:52 +03:00
currentWallet . refresh ( )
currentWallet . disposeTransaction ( transaction )
2016-11-19 18:33:30 +02:00
currentWallet . store ( ) ;
2016-06-16 17:13:46 +03:00
}
2017-11-20 09:24:29 +02:00
// called on "getProof"
function handleGetProof ( txid , address , message ) {
2017-09-12 11:42:00 +03:00
console . log ( "Getting payment proof: " )
console . log ( "\ttxid: " , txid ,
", address: " , address ,
", message: " , message ) ;
2016-11-05 16:58:49 +02:00
2017-11-20 09:12:36 +02:00
var result ;
if ( address . length > 0 )
result = currentWallet . getTxProof ( txid , address , message ) ;
2018-02-20 05:10:19 +02:00
if ( ! result || result . indexOf ( "error|" ) === 0 )
2017-11-20 09:12:36 +02:00
result = currentWallet . getSpendProof ( txid , message ) ;
2017-09-12 11:42:00 +03:00
informationPopup . title = qsTr ( "Payment proof" ) + translationManager . emptyString ;
2018-02-20 05:10:19 +02:00
if ( result . indexOf ( "error|" ) === 0 ) {
2017-09-12 11:42:00 +03:00
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 ;
2016-11-05 16:58:49 +02:00
}
2017-09-12 11:42:00 +03:00
informationPopup . onCloseCallback = null
informationPopup . open ( )
}
2017-11-20 09:24:29 +02:00
// called on "checkProof"
function handleCheckProof ( txid , address , message , signature ) {
2017-09-12 11:42:00 +03:00
console . log ( "Checking payment proof: " )
console . log ( "\ttxid: " , txid ,
", address: " , address ,
", message: " , message ,
", signature: " , signature ) ;
2017-11-20 09:12:36 +02:00
var result ;
if ( address . length > 0 )
result = currentWallet . checkTxProof ( txid , address , message , signature ) ;
else
result = currentWallet . checkSpendProof ( txid , message , signature ) ;
2017-09-12 11:42:00 +03:00
var results = result . split ( "|" ) ;
2017-11-20 09:12:36 +02:00
if ( address . length > 0 && results . length == 5 && results [ 0 ] === "true" ) {
var good = results [ 1 ] === "true" ;
2017-09-12 11:42:00 +03:00
var received = results [ 2 ] ;
2017-11-20 09:12:36 +02:00
var in_pool = results [ 3 ] === "true" ;
2017-09-12 11:42:00 +03:00
var confirmations = results [ 4 ] ;
informationPopup . title = qsTr ( "Payment proof check" ) + translationManager . emptyString ;
2016-11-05 16:58:49 +02:00
informationPopup . icon = StandardIcon . Information
2017-09-12 11:42:00 +03:00
if ( ! good ) {
informationPopup . text = qsTr ( "Bad signature" ) ;
informationPopup . icon = StandardIcon . Critical ;
} else if ( received > 0 ) {
2016-11-05 16:58:49 +02:00
received = received / 1 e12
2017-09-12 11:42:00 +03:00
if ( in_pool ) {
2016-11-05 16:58:49 +02:00
informationPopup . text = qsTr ( "This address received %1 monero, but the transaction is not yet mined" ) . arg ( received ) ;
}
else {
2017-01-15 05:52:04 +02:00
informationPopup . text = qsTr ( "This address received %1 monero, with %2 confirmation(s)." ) . arg ( received ) . arg ( confirmations ) ;
2016-11-05 16:58:49 +02:00
}
}
else {
informationPopup . text = qsTr ( "This address received nothing" ) ;
}
}
2017-11-20 09:12:36 +02:00
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" ) ;
}
2016-11-05 16:58:49 +02:00
else {
informationPopup . title = qsTr ( "Error" ) + translationManager . emptyString ;
2017-09-12 11:42:00 +03:00
informationPopup . text = currentWallet . errorString ;
2016-11-05 16:58:49 +02:00
informationPopup . icon = StandardIcon . Critical
}
2017-09-12 11:42:00 +03:00
informationPopup . onCloseCallback = null
2016-11-05 16:58:49 +02:00
informationPopup . open ( )
}
2016-12-31 15:47:40 +02:00
function updateSyncing ( syncing ) {
var text = ( syncing ? qsTr ( "Balance (syncing)" ) : qsTr ( "Balance" ) ) + translationManager . emptyString
leftPanel . balanceLabelText = text
middlePanel . balanceLabelText = text
}
2016-11-05 16:58:49 +02:00
2016-08-17 15:14:43 +03:00
// blocks UI if wallet can't be opened or no connection to the daemon
function enableUI ( enable ) {
middlePanel . enabled = enable ;
leftPanel . enabled = enable ;
rightPanel . enabled = enable ;
}
2016-08-23 11:55:51 +03:00
function showProcessingSplash ( message ) {
console . log ( "Displaying processing splash" )
if ( typeof message != 'undefined' ) {
2016-11-08 19:05:33 +02:00
splash . messageText = message
splash . heightProgressText = ""
2016-08-23 11:55:51 +03:00
}
splash . show ( )
}
function hideProcessingSplash ( ) {
console . log ( "Hiding processing splash" )
2016-09-01 23:55:30 +03:00
splash . close ( )
2016-08-23 11:55:51 +03:00
}
2016-10-29 17:48:10 +03:00
// close wallet and show wizard
function showWizard ( ) {
2018-12-17 06:13:07 +02:00
clearMoneroCardLabelText ( ) ;
2016-10-29 17:59:31 +03:00
walletInitialized = false ;
2016-12-15 14:18:04 +02:00
closeWallet ( ) ;
2016-12-16 01:12:27 +02:00
currentWallet = undefined ;
2016-10-29 17:48:10 +03:00
wizard . restart ( ) ;
rootItem . state = "wizard"
2017-03-04 01:11:26 +02:00
// reset balance
leftPanel . balanceText = leftPanel . unlockedBalanceText = walletManager . displayAmount ( 0 ) ;
2018-12-13 20:02:02 +02:00
// disable inactivity timer
userInActivityTimer . running = false ;
2016-10-29 17:48:10 +03:00
}
2017-04-03 19:51:55 +03:00
function hideMenu ( ) {
goToBasicAnimation . start ( ) ;
console . log ( appWindow . width )
}
function showMenu ( ) {
goToProAnimation . start ( ) ;
console . log ( appWindow . width )
}
2016-08-17 15:14:43 +03:00
objectName: "appWindow"
2014-07-07 20:08:30 +03:00
visible: true
2017-08-07 17:20:37 +03:00
// width: screenWidth //rightPanelExpanded ? 1269 : 1269 - 300
2017-04-03 19:51:55 +03:00
// height: 900 //300//maxWindowHeight;
2014-07-07 20:08:30 +03:00
color: "#FFFFFF"
2018-04-21 22:58:00 +03:00
flags: persistentSettings . customDecorations ? Windows.flagsCustomDecorations : Windows . flags
2014-07-19 17:07:40 +03:00
onWidthChanged: x -= 0
Component.onCompleted: {
x = ( Screen . width - width ) / 2
2016-11-21 23:14:49 +02:00
y = ( Screen . height - maxWindowHeight ) / 2
2016-06-07 16:26:25 +03:00
//
2016-08-18 21:55:34 +03:00
walletManager . walletOpened . connect ( onWalletOpened ) ;
walletManager . walletClosed . connect ( onWalletClosed ) ;
2017-08-06 01:10:59 +03:00
walletManager . checkUpdatesComplete . connect ( onWalletCheckUpdatesComplete ) ;
2016-08-18 21:55:34 +03:00
2017-04-03 19:51:55 +03:00
if ( typeof daemonManager != "undefined" ) {
daemonManager . daemonStarted . connect ( onDaemonStarted ) ;
daemonManager . daemonStartFailure . connect ( onDaemonStartFailure ) ;
daemonManager . daemonStopped . connect ( onDaemonStopped ) ;
}
2016-11-25 22:09:32 +02:00
2017-03-01 23:03:50 +02:00
// Connect app exit to qml window exit handling
mainApp . closing . connect ( appWindow . close ) ;
2017-03-27 20:39:47 +03:00
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" ) ;
2016-10-30 18:58:12 +02:00
if ( ! walletsFound ( ) ) {
2016-10-29 17:48:10 +03:00
rootItem . state = "wizard"
} else {
rootItem . state = "normal"
2017-11-17 03:16:35 +02:00
passwordDialog . onAcceptedCallback = function ( ) {
walletPassword = passwordDialog . password ;
2016-10-29 17:48:10 +03:00
initialize ( persistentSettings ) ;
2017-11-17 03:16:35 +02:00
}
passwordDialog . onRejectedCallback = function ( ) {
rootItem . state = "wizard"
}
passwordDialog . open ( usefulName ( walletPath ( ) ) )
2016-06-15 16:34:55 +03:00
}
2016-10-29 17:48:10 +03:00
2017-02-19 12:38:03 +02:00
checkUpdates ( ) ;
2016-06-15 16:34:55 +03:00
}
2016-06-21 11:58:06 +03:00
onRightPanelExpandedChanged: {
if ( rightPanelExpanded ) {
rightPanel . updateTweets ( )
}
}
2016-08-17 15:14:43 +03:00
2016-06-15 16:34:55 +03:00
Settings {
id: persistentSettings
property string language
2016-07-19 23:45:12 +03:00
property string locale
2016-06-15 16:34:55 +03:00
property string account_name
property string wallet_path
2016-11-12 13:11:24 +02:00
property bool auto_donations_enabled : false
2016-06-15 16:34:55 +03:00
property int auto_donations_amount : 50
2016-11-12 13:11:24 +02:00
property bool allow_background_mining : false
2017-03-02 16:44:37 +02:00
property bool miningIgnoreBattery : true
2018-03-05 18:19:45 +02:00
property var nettype: NetworkType . MAINNET
2018-03-27 22:34:30 +03:00
property string daemon_address: nettype == NetworkType . TESTNET ? "localhost:28081" : nettype == NetworkType . STAGENET ? "localhost:38081" : "localhost:18081"
2016-06-26 18:04:45 +03:00
property string payment_id
2016-10-10 22:40:58 +03:00
property int restore_height : 0
property bool is_recovering : false
2018-06-04 11:05:29 +03:00
property bool is_recovering_from_device : false
2016-12-09 23:23:43 +02:00
property bool customDecorations : true
2016-12-21 15:30:15 +02:00
property string daemonFlags
2016-12-31 12:56:08 +02:00
property int logLevel: 0
2017-02-05 14:38:43 +02:00
property string logCategories: ""
2017-02-25 23:16:58 +02:00
property string daemonUsername: ""
property string daemonPassword: ""
2017-02-27 23:05:28 +02:00
property bool transferShowAdvanced: false
2018-12-06 21:56:23 +02:00
property bool receiveShowAdvanced: false
2017-05-04 15:06:29 +03:00
property string blockchainDataDir: ""
2017-08-07 18:03:34 +03:00
property bool useRemoteNode: false
property string remoteNodeAddress: ""
2018-01-22 11:43:39 +02:00
property string bootstrapNodeAddress: ""
2018-03-16 13:55:56 +02:00
property bool segregatePreForkOutputs: true
property bool keyReuseMitigation2: true
property int segregationHeight: 0
2018-07-06 12:39:58 +03:00
property int kdfRounds: 1
2018-12-12 10:33:24 +02:00
property bool hideBalance: false
2018-12-13 20:02:02 +02:00
property bool lockOnUserInActivity: true
property int lockOnUserInActivityInterval: 10 // minutes
2014-07-19 17:07:40 +03:00
}
2014-07-07 20:08:30 +03:00
2016-06-28 22:37:14 +03:00
// Information dialog
2016-11-19 16:15:11 +02:00
StandardDialog {
2016-08-17 15:14:43 +03:00
// dynamically change onclose handler
property var onCloseCallback
2016-06-28 22:37:14 +03:00
id: informationPopup
2017-08-08 14:35:24 +03:00
anchors.fill: parent
z: parent . z + 1
2016-11-19 16:15:11 +02:00
cancelVisible: false
2016-08-17 15:14:43 +03:00
onAccepted: {
if ( onCloseCallback ) {
onCloseCallback ( )
}
}
2016-06-28 22:37:14 +03:00
}
// Confrirmation aka question dialog
2016-11-19 16:15:11 +02:00
StandardDialog {
2017-11-02 20:51:53 +02:00
z: parent . z + 1
2016-06-28 22:37:14 +03:00
id: transactionConfirmationPopup
onAccepted: {
2016-11-19 16:15:11 +02:00
close ( ) ;
2017-11-17 03:16:35 +02:00
passwordDialog . onAcceptedCallback = function ( ) {
if ( walletPassword === passwordDialog . password ) {
2017-09-19 01:57:06 +03:00
// Save transaction to file if view only wallet
if ( viewOnly ) {
saveTxDialog . open ( ) ;
} else {
handleTransactionConfirmed ( )
}
} else {
2018-10-24 06:50:56 +03:00
passwordDialog . showError ( qsTr ( "Wrong password" ) ) ;
2017-09-19 01:57:06 +03:00
}
2017-10-29 06:00:18 +02:00
}
2017-11-17 03:16:35 +02:00
passwordDialog . onRejectedCallback = null ;
passwordDialog . open ( )
2017-03-02 16:44:37 +02:00
}
2017-01-12 21:53:27 +02:00
}
StandardDialog {
2017-11-02 20:51:53 +02:00
z: parent . z + 1
2017-01-12 21:53:27 +02:00
id: confirmationDialog
2017-11-02 20:51:53 +02:00
anchors.fill: parent
2017-01-12 21:53:27 +02:00
property var onAcceptedCallback
property var onRejectedCallback
onAccepted: {
if ( onAcceptedCallback )
onAcceptedCallback ( )
}
onRejected: {
if ( onRejectedCallback )
onRejectedCallback ( ) ;
2016-06-28 22:37:14 +03:00
}
}
2017-01-12 21:53:27 +02:00
2016-10-29 17:48:10 +03:00
//Open Wallet from file
FileDialog {
id: fileDialog
title: "Please choose a file"
folder: "file://" + moneroAccountsDir
2016-10-30 18:58:12 +02:00
nameFilters: [ "Wallet files (*.keys)" ]
2017-11-02 08:52:38 +02:00
sidebarVisible: false
2016-10-29 17:48:10 +03:00
onAccepted: {
persistentSettings . wallet_path = walletManager . urlToLocalPath ( fileDialog . fileUrl )
2017-11-02 08:52:38 +02:00
if ( isIOS )
persistentSettings . wallet_path = persistentSettings . wallet_path . replace ( moneroAccountsDir , "" )
console . log ( "ÖPPPPNA" )
console . log ( moneroAccountsDir )
console . log ( fileDialog . fileUrl )
console . log ( persistentSettings . wallet_path )
2017-11-17 03:16:35 +02:00
passwordDialog . onAcceptedCallback = function ( ) {
walletPassword = passwordDialog . password ;
initialize ( ) ;
}
passwordDialog . onRejectedCallback = function ( ) {
console . log ( "Canceled" )
rootItem . state = "wizard" ;
}
passwordDialog . open ( usefulName ( walletPath ( ) ) ) ;
2016-10-29 17:48:10 +03:00
}
onRejected: {
console . log ( "Canceled" )
rootItem . state = "wizard" ;
}
}
2017-08-08 12:30:08 +03:00
// Choose blockchain folder
FileDialog {
id: blockchainFileDialog
title: "Please choose a folder"
selectFolder: true
folder: "file://" + persistentSettings . blockchainDataDir
onAccepted: {
var dataDir = walletManager . urlToLocalPath ( blockchainFileDialog . fileUrl )
var validator = daemonManager . validateDataDir ( dataDir ) ;
if ( ! validator . valid ) {
confirmationDialog . title = qsTr ( "Warning" ) + translationManager . emptyString ;
confirmationDialog . text = "" ;
if ( validator . readOnly )
confirmationDialog . text += qsTr ( "Error: Filesystem is read only" ) + "\n\n"
if ( validator . storageAvailable < 20 )
2018-04-05 10:21:43 +03:00
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"
2017-08-08 12:30:08 +03:00
else
2018-04-05 10:21:43 +03:00
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"
2017-08-08 12:30:08 +03:00
if ( ! validator . lmdbExists )
confirmationDialog . text += qsTr ( "Note: lmdb folder not found. A new folder will be created." ) + "\n\n"
confirmationDialog . icon = StandardIcon . Question
confirmationDialog . cancelText = qsTr ( "Cancel" )
// Continue
confirmationDialog . onAcceptedCallback = function ( ) {
persistentSettings . blockchainDataDir = dataDir
}
// Cancel
confirmationDialog . onRejectedCallback = function ( ) {
} ;
confirmationDialog . open ( )
} else {
persistentSettings . blockchainDataDir = dataDir
}
delete validator ;
}
onRejected: {
console . log ( "data dir selection canceled" )
}
}
2016-08-16 23:21:46 +03:00
PasswordDialog {
id: passwordDialog
2017-08-08 14:35:24 +03:00
visible: false
z: parent . z + 1
anchors.fill: parent
2017-09-19 01:57:06 +03:00
property var onAcceptedCallback
2017-10-29 06:00:18 +02:00
property var onRejectedCallback
2017-09-19 01:57:06 +03:00
onAccepted: {
2017-10-29 06:00:18 +02:00
if ( onAcceptedCallback )
2017-09-19 01:57:06 +03:00
onAcceptedCallback ( ) ;
}
2017-10-29 06:00:18 +02:00
onRejected: {
if ( onRejectedCallback )
onRejectedCallback ( ) ;
}
2017-09-19 01:57:06 +03:00
}
2017-12-11 13:15:16 +02:00
2017-09-22 17:25:25 +03:00
NewPasswordDialog {
id: newPasswordDialog
z: parent . z + 1
visible: false
anchors.fill: parent
onAccepted: {
if ( currentWallet . setPassword ( newPasswordDialog . password ) ) {
appWindow . walletPassword = newPasswordDialog . 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 ( ) ;
}
onRejected: {
}
}
2017-07-04 06:34:09 +03:00
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 ( )
}
}
2016-11-05 11:54:11 +02:00
DaemonManagerDialog {
id: daemonManagerDialog
2017-02-23 20:47:58 +02:00
onRejected: {
loadPage ( "Settings" ) ;
2017-11-03 19:08:59 +02:00
startLocalNodeCancelled = true
2017-02-23 20:47:58 +02:00
}
2016-11-05 11:54:11 +02:00
}
2016-08-23 11:55:51 +03:00
ProcessingSplash {
id: splash
2016-10-09 21:49:56 +03:00
width: appWindow . width / 1.5
2016-08-23 11:55:51 +03:00
height: appWindow . height / 2
2017-08-20 23:23:22 +03:00
x: ( appWindow . width - width ) / 2
y: ( appWindow . height - height ) / 2
2016-09-26 22:55:25 +03:00
messageText: qsTr ( "Please wait..." )
2016-07-13 15:24:40 +03:00
}
2014-07-07 20:08:30 +03:00
Item {
id: rootItem
anchors.fill: parent
2014-07-19 17:07:40 +03:00
clip: true
2014-07-07 20:08:30 +03:00
2014-08-19 15:58:02 +03:00
state: "wizard"
states: [
State {
name: "wizard"
PropertyChanges { target: leftPanel ; visible: false }
PropertyChanges { target: rightPanel ; visible: false }
PropertyChanges { target: middlePanel ; visible: false }
2017-05-06 18:49:43 +03:00
PropertyChanges { target: titleBar ; basicButtonVisible: false }
2014-08-19 15:58:02 +03:00
PropertyChanges { target: wizard ; visible: true }
2017-08-07 17:20:37 +03:00
PropertyChanges { target: appWindow ; width: ( screenWidth < 930 || isAndroid || isIOS ) ? screenWidth : 930 ; }
2017-05-06 18:49:43 +03:00
PropertyChanges { target: appWindow ; height: maxWindowHeight ; }
2017-08-08 14:35:24 +03:00
PropertyChanges { target: resizeArea ; visible: true }
2018-12-17 05:27:27 +02:00
PropertyChanges { target: titleBar ; showMaximizeButton: true }
2017-04-03 19:51:55 +03:00
// PropertyChanges { target: frameArea; blocked: true }
2018-12-17 05:27:27 +02:00
PropertyChanges { target: titleBar ; visible: true }
2016-12-09 23:23:43 +02:00
PropertyChanges { target: titleBar ; y: 0 }
2018-12-17 05:27:27 +02:00
PropertyChanges { target: titleBar ; showMoneroLogo: false }
PropertyChanges { target: titleBar ; titleBarGradientImageOpacity: 0.2 }
PropertyChanges { target: titleBar ; small: true }
2017-08-08 14:35:24 +03:00
PropertyChanges { target: mobileHeader ; visible: false }
2014-08-22 12:03:10 +03:00
} , State {
2014-08-19 15:58:02 +03:00
name: "normal"
2017-04-03 19:51:55 +03:00
PropertyChanges { target: leftPanel ; visible: ( isMobile ) ? false : true }
2014-08-19 15:58:02 +03:00
PropertyChanges { target: rightPanel ; visible: true }
PropertyChanges { target: middlePanel ; visible: true }
2017-05-06 18:49:43 +03:00
PropertyChanges { target: titleBar ; basicButtonVisible: true }
2014-08-19 15:58:02 +03:00
PropertyChanges { target: wizard ; visible: false }
2017-08-07 17:20:37 +03:00
PropertyChanges { target: appWindow ; width: ( screenWidth < 969 || isAndroid || isIOS ) ? screenWidth : 969 } //rightPanelExpanded ? 1269 : 1269 - 300;
2017-05-06 18:49:43 +03:00
PropertyChanges { target: appWindow ; height: maxWindowHeight ; }
2014-08-21 13:09:52 +03:00
PropertyChanges { target: resizeArea ; visible: true }
2018-04-21 22:59:31 +03:00
PropertyChanges { target: titleBar ; showMaximizeButton: true }
2017-05-06 18:49:43 +03:00
// PropertyChanges { target: frameArea; blocked: true }
PropertyChanges { target: titleBar ; visible: true }
2017-04-03 19:51:55 +03:00
// PropertyChanges { target: titleBar; y: 0 }
2016-12-09 23:23:43 +02:00
PropertyChanges { target: titleBar ; title: qsTr ( "Monero" ) + translationManager . emptyString }
2017-08-07 17:20:37 +03:00
PropertyChanges { target: mobileHeader ; visible: isMobile ? true : false }
2014-08-19 15:58:02 +03:00
}
]
2017-04-03 19:51:55 +03:00
MobileHeader {
id: mobileHeader
visible: isMobile
anchors.left: parent . left
anchors.right: parent . right
2017-08-07 17:20:37 +03:00
height: visible ? 65 * scaleRatio : 0
2018-07-03 19:56:23 +03:00
MouseArea {
enabled: persistentSettings . customDecorations
property var previousPosition
anchors.fill: parent
propagateComposedEvents: true
onPressed: previousPosition = globalCursor . getPosition ( )
onPositionChanged: {
if ( pressedButtons == Qt . LeftButton ) {
var pos = globalCursor . getPosition ( )
var dx = pos . x - previousPosition . x
var dy = pos . y - previousPosition . y
appWindow . x += dx
appWindow . y += dy
previousPosition = pos
}
}
}
2017-04-03 19:51:55 +03:00
}
2014-07-07 20:08:30 +03:00
LeftPanel {
id: leftPanel
2017-04-03 19:51:55 +03:00
anchors.top: mobileHeader . bottom
2014-07-07 20:08:30 +03:00
anchors.left: parent . left
anchors.bottom: parent . bottom
2018-04-05 11:32:43 +03:00
onDashboardClicked: {
middlePanel . state = "Dashboard" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onTransferClicked: {
middlePanel . state = "Transfer" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onReceiveClicked: {
middlePanel . state = "Receive" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onTxkeyClicked: {
middlePanel . state = "TxKey" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onSharedringdbClicked: {
middlePanel . state = "SharedRingDB" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onHistoryClicked: {
middlePanel . state = "History" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onAddressBookClicked: {
middlePanel . state = "AddressBook" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onMiningClicked: {
middlePanel . state = "Mining" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onSignClicked: {
middlePanel . state = "Sign" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
}
onSettingsClicked: {
middlePanel . state = "Settings" ;
2018-04-05 12:52:09 +03:00
middlePanel . flickable . contentY = 0 ;
2018-04-05 11:32:43 +03:00
if ( isMobile ) {
hideMenu ( ) ;
}
updateBalance ( ) ;
2018-04-05 12:52:09 +03:00
}
2018-04-05 11:32:43 +03:00
2018-07-18 17:51:56 +03:00
onKeysClicked: Utils . showSeedPage ( ) ;
2014-07-07 20:08:30 +03:00
}
RightPanel {
id: rightPanel
anchors.right: parent . right
anchors.bottom: parent . bottom
2014-07-16 11:24:59 +03:00
width: appWindow . rightPanelExpanded ? 300 : 0
visible: appWindow . rightPanelExpanded
2014-07-07 20:08:30 +03:00
}
2016-07-14 13:40:27 +03:00
2014-07-07 20:08:30 +03:00
MiddlePanel {
id: middlePanel
2017-04-03 19:51:55 +03:00
anchors.top: mobileHeader . bottom
2014-07-07 20:08:30 +03:00
anchors.bottom: parent . bottom
2016-10-09 00:50:35 +03:00
anchors.left: leftPanel . visible ? leftPanel.right : parent . left
2017-04-03 19:51:55 +03:00
anchors.right: parent . right
2016-06-26 18:04:45 +03:00
state: "Transfer"
2014-07-07 20:08:30 +03:00
}
TipItem {
id: tipItem
2016-07-20 22:28:11 +03:00
text: qsTr ( "send to the same destination" ) + translationManager . emptyString
2014-07-07 20:08:30 +03:00
visible: false
}
2014-07-15 17:03:39 +03:00
2014-07-19 17:07:40 +03:00
SequentialAnimation {
id: goToBasicAnimation
2017-04-03 19:51:55 +03:00
// PropertyAction {
// target: appWindow
// properties: "visibility"
// value: Window.Windowed
// }
// PropertyAction {
// target: titleBar
// properties: "maximizeButtonVisible"
// value: false
// }
// PropertyAction {
// target: frameArea
// properties: "blocked"
// value: true
// }
2014-08-22 12:03:10 +03:00
PropertyAction {
target: resizeArea
properties: "visible"
2017-04-03 19:51:55 +03:00
value: true
2014-07-19 17:07:40 +03:00
}
2017-04-03 19:51:55 +03:00
// PropertyAction {
// target: appWindow
// properties: "height"
// value: 30
// }
// PropertyAction {
// target: appWindow
// properties: "width"
// value: 326
// }
2014-07-19 17:07:40 +03:00
PropertyAction {
2016-10-09 00:50:35 +03:00
targets: [ leftPanel , rightPanel ]
2014-07-19 17:07:40 +03:00
properties: "visible"
value: false
}
PropertyAction {
2016-10-09 00:50:35 +03:00
target: middlePanel
properties: "basicMode"
2014-07-19 17:07:40 +03:00
value: true
}
2016-10-09 00:50:35 +03:00
2017-04-03 19:51:55 +03:00
// PropertyAction {
// target: appWindow
// properties: "height"
// value: middlePanel.height
// }
2014-07-19 17:07:40 +03:00
onStopped: {
2016-10-09 00:50:35 +03:00
// middlePanel.visible = false
2014-07-19 17:07:40 +03:00
rightPanel . visible = false
leftPanel . visible = false
}
}
SequentialAnimation {
id: goToProAnimation
2017-04-03 19:51:55 +03:00
// PropertyAction {
// target: appWindow
// properties: "height"
// value: 30
// }
2014-07-19 17:07:40 +03:00
PropertyAction {
2016-10-09 00:50:35 +03:00
target: middlePanel
properties: "basicMode"
2014-07-19 17:07:40 +03:00
value: false
}
PropertyAction {
2014-08-22 12:03:10 +03:00
targets: [ leftPanel , middlePanel , rightPanel , resizeArea ]
2014-07-19 17:07:40 +03:00
properties: "visible"
value: true
}
2017-04-03 19:51:55 +03:00
// PropertyAction {
// target: appWindow
// properties: "width"
// value: rightPanelExpanded ? 1269 : 1269 - 300
// }
// PropertyAction {
// target: appWindow
// properties: "height"
// value: maxWindowHeight
// }
// PropertyAction {
// target: frameArea
// properties: "blocked"
// value: false
// }
// PropertyAction {
// target: titleBar
// properties: "maximizeButtonVisible"
// value: true
// }
2014-07-19 17:07:40 +03:00
}
2014-08-19 15:58:02 +03:00
WizardMain {
id: wizard
anchors.fill: parent
2016-06-15 16:34:55 +03:00
onUseMoneroClicked: {
rootItem . state = "normal" // TODO: listen for this state change in appWindow;
appWindow . initialize ( ) ;
}
2016-10-29 17:48:10 +03:00
onOpenWalletFromFileClicked: {
rootItem . state = "normal" // TODO: listen for this state change in appWindow;
appWindow . openWalletFromFile ( ) ;
}
2014-08-19 15:58:02 +03:00
}
2017-01-30 11:37:14 +02:00
property int minWidth: 326
2017-04-03 19:51:55 +03:00
property int minHeight: 400
2014-07-23 13:39:35 +03:00
MouseArea {
2014-08-21 13:09:52 +03:00
id: resizeArea
2018-12-19 23:10:54 +02:00
enabled: persistentSettings . customDecorations
2014-07-23 13:39:35 +03:00
hoverEnabled: true
anchors.right: parent . right
anchors.bottom: parent . bottom
2014-07-23 14:27:27 +03:00
height: 30
width: 30
2014-07-23 13:39:35 +03:00
Rectangle {
anchors.fill: parent
2014-07-23 14:59:26 +03:00
color: parent . containsMouse || parent . pressed ? "#111111" : "transparent"
2014-07-23 13:39:35 +03:00
}
Image {
anchors.centerIn: parent
2018-12-19 23:10:54 +02:00
visible: persistentSettings . customDecorations
2014-07-23 13:39:35 +03:00
source: parent . containsMouse || parent . pressed ? "images/resizeHovered.png" :
"images/resize.png"
}
2016-02-03 17:37:10 +02:00
property var previousPosition
2016-06-16 17:13:46 +03:00
2014-07-23 13:39:35 +03:00
onPressed: {
2016-02-03 17:37:10 +02:00
previousPosition = globalCursor . getPosition ( )
2014-07-23 13:39:35 +03:00
}
onPositionChanged: {
if ( ! pressed ) return
2016-02-03 17:37:10 +02:00
var pos = globalCursor . getPosition ( )
//var delta = previousPosition - pos
var dx = previousPosition . x - pos . x
var dy = previousPosition . y - pos . y
2014-07-23 13:39:35 +03:00
2017-01-30 11:37:14 +02:00
if ( appWindow . width - dx > parent . minWidth )
2014-07-23 13:39:35 +03:00
appWindow . width -= dx
2017-01-30 11:37:14 +02:00
else appWindow . width = parent . minWidth
2014-07-23 13:39:35 +03:00
2016-11-23 22:17:01 +02:00
if ( appWindow . height - dy > parent . minHeight )
2014-07-23 13:39:35 +03:00
appWindow . height -= dy
2016-11-23 22:17:01 +02:00
else appWindow . height = parent . minHeight
2016-02-03 17:37:10 +02:00
previousPosition = pos
2014-07-23 13:39:35 +03:00
}
}
2014-07-22 17:55:25 +03:00
2014-07-15 17:03:39 +03:00
TitleBar {
id: titleBar
2016-11-19 16:22:56 +02:00
x: 0
y: 0
2018-04-21 22:59:31 +03:00
anchors.left: parent . left
anchors.right: parent . right
showMinimizeButton: true
showMaximizeButton: true
showWhatIsButton: false
showMoneroLogo: true
onCloseClicked: appWindow . close ( ) ;
onMaximizeClicked: {
2018-07-28 15:06:05 +03:00
appWindow . visibility = appWindow . visibility !== Window . Maximized ? Window.Maximized :
2018-04-21 22:59:31 +03:00
Window . Windowed
}
onMinimizeClicked: appWindow . visibility = Window . Minimized
2014-07-19 17:07:40 +03:00
onGoToBasicVersion: {
2016-10-09 00:50:35 +03:00
if ( yes ) {
// basicPanel.currentView = middlePanel.currentView
goToBasicAnimation . start ( )
} else {
// middlePanel.currentView = basicPanel.currentView
goToProAnimation . start ( )
}
2014-07-19 17:07:40 +03:00
}
2016-12-09 23:23:43 +02:00
2017-05-06 18:49:43 +03:00
MouseArea {
enabled: persistentSettings . customDecorations
property var previousPosition
anchors.fill: parent
propagateComposedEvents: true
onPressed: previousPosition = globalCursor . getPosition ( )
onPositionChanged: {
if ( pressedButtons == Qt . LeftButton ) {
var pos = globalCursor . getPosition ( )
var dx = pos . x - previousPosition . x
var dy = pos . y - previousPosition . y
appWindow . x += dx
appWindow . y += dy
previousPosition = pos
}
}
}
2014-07-15 17:03:39 +03:00
}
2016-12-15 15:09:37 +02:00
// 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: "../images/tip.png"
}
Text {
id: content
anchors.horizontalCenter: parent . horizontalCenter
y: 6
lineHeight: 0.7
font.family: "Arial"
2017-08-07 17:20:37 +03:00
font.pixelSize: 12 * scaleRatio
2016-12-15 15:09:37 +02:00
color: "#FFFFFF"
}
}
2017-02-19 12:38:03 +02:00
Notifier {
2017-04-03 19:51:55 +03:00
visible: false
2017-02-19 12:38:03 +02:00
id: notifier
}
2014-07-07 20:08:30 +03:00
}
2017-03-01 23:03:50 +02:00
2017-08-08 12:29:02 +03:00
// TODO: Make the callback dynamic
Timer {
id: statusMessageTimer
interval: 5 ;
running: false ;
repeat: false
onTriggered: resetAndroidClose ( )
triggeredOnStart: false
}
2018-12-13 20:02:02 +02:00
Timer {
id: userInActivityTimer
interval: 2000 ; running: false ; repeat: true
onTriggered: checkInUserActivity ( )
}
2017-08-08 12:29:02 +03:00
Rectangle {
id: statusMessage
2017-11-02 08:51:15 +02:00
z: 99
2017-08-08 12:29:02 +03:00
visible: false
property alias text: statusMessageText . text
anchors.bottom: parent . bottom
2017-11-02 08:51:15 +02:00
width: statusMessageText . contentWidth + 20 * scaleRatio
2017-08-08 12:29:02 +03:00
anchors.horizontalCenter: parent . horizontalCenter
color: "black"
height: 40 * scaleRatio
Text {
id: statusMessageText
anchors.fill: parent
anchors.margins: 10 * scaleRatio
font.pixelSize: 14 * scaleRatio
color: "white"
}
}
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
}
2016-10-10 22:42:25 +03:00
onClosing: {
2017-08-08 12:29:02 +03:00
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 ;
}
}
2017-03-01 23:03:50 +02:00
// If daemon is running - prompt user before exiting
2018-03-05 18:19:45 +02:00
if ( typeof daemonManager != "undefined" && daemonManager . running ( persistentSettings . nettype ) ) {
2017-03-01 23:03:50 +02:00
// Show confirmation dialog
confirmationDialog . title = qsTr ( "Daemon is running" ) + translationManager . emptyString ;
confirmationDialog . text = qsTr ( "Daemon will still be running in background when GUI is closed." ) ;
confirmationDialog . icon = StandardIcon . Question
confirmationDialog . cancelText = qsTr ( "Stop daemon" )
confirmationDialog . onAcceptedCallback = function ( ) {
closeAccepted ( ) ;
}
confirmationDialog . onRejectedCallback = function ( ) {
2018-03-05 18:19:45 +02:00
daemonManager . stop ( persistentSettings . nettype ) ;
2017-03-01 23:03:50 +02:00
closeAccepted ( ) ;
} ;
confirmationDialog . open ( )
} else {
closeAccepted ( ) ;
}
}
function closeAccepted ( ) {
2017-08-08 12:29:02 +03:00
console . log ( "close accepted" ) ;
2016-12-15 14:18:04 +02:00
// Close wallet non async on exit
2017-02-25 15:57:39 +02:00
daemonManager . exit ( ) ;
2016-12-04 14:13:59 +02:00
walletManager . closeWallet ( ) ;
2017-03-01 23:03:50 +02:00
Qt . quit ( ) ;
2016-10-10 22:42:25 +03:00
}
2017-02-19 12:38:03 +02:00
2017-08-06 01:10:59 +03:00
function onWalletCheckUpdatesComplete ( update ) {
2017-02-19 12:38:03 +02:00
if ( update === "" )
return
print ( "Update found: " + update )
var parts = update . split ( "|" )
if ( parts . length == 4 ) {
var version = parts [ 0 ]
var hash = parts [ 1 ]
var user_url = parts [ 2 ]
var auto_url = parts [ 3 ]
var msg = qsTr ( "New version of monero-wallet-gui is available: %1<br>%2" ) . arg ( version ) . arg ( user_url ) + translationManager . emptyString
notifier . show ( msg )
}
else {
print ( "Failed to parse update spec" )
}
}
2017-08-06 01:10:59 +03:00
function checkUpdates ( ) {
walletManager . checkUpdatesAsync ( "monero-gui" , "gui" )
}
2017-02-19 12:38:03 +02:00
Timer {
id: updatesTimer
interval: 3600 * 1000 ; running: true ; repeat: true
onTriggered: checkUpdates ( )
}
2017-08-07 13:22:49 +03:00
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
}
2018-04-21 23:21:35 +03:00
}
2017-08-07 13:22:49 +03:00
2018-12-17 06:13:07 +02:00
// reset label text. othewise potential privacy leak showing unlock time when switching wallets
function clearMoneroCardLabelText ( ) {
leftPanel . minutesToUnlockTxt = qsTr ( "Unlocked balance" )
leftPanel . balanceLabelText = qsTr ( "Balance" )
}
2018-12-13 20:02:02 +02:00
function userActivity ( ) {
// register user activity
var epoch = Math . floor ( ( new Date ) . getTime ( ) / 1000 ) ;
appWindow . userLastActive = epoch ;
}
function checkInUserActivity ( ) {
if ( ! persistentSettings . lockOnUserInActivity ) return ;
// prompt password after X seconds of inactivity
var epoch = Math . floor ( ( new Date ) . getTime ( ) / 1000 ) ;
var inactivity = epoch - appWindow . userLastActive ;
if ( inactivity < ( persistentSettings . lockOnUserInActivityInterval * 60 ) ) return ;
passwordDialog . onAcceptedCallback = function ( ) {
if ( walletPassword === passwordDialog . password ) {
passwordDialog . close ( ) ;
} else {
passwordDialog . showError ( qsTr ( "Wrong password" ) ) ;
}
}
passwordDialog . onRejectedCallback = function ( ) { appWindow . showWizard ( ) ; }
passwordDialog . open ( ) ;
}
2018-04-21 23:21:35 +03:00
// Daemon console
DaemonConsole {
id: daemonConsolePopup
height: 500
width: 800
title: qsTr ( "Daemon log" ) + translationManager . emptyString
onAccepted: {
close ( ) ;
}
}
2017-08-07 13:22:49 +03:00
2018-04-21 23:21:35 +03:00
// background gradient
Rectangle {
id: inactiveOverlay
visible: false
anchors.fill: parent
color: "black"
opacity: 0.8
2017-08-07 13:22:49 +03:00
}
2014-07-07 20:08:30 +03:00
}