// 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.0 import QtQuick.Layouts 1.1 import QtQuick.Dialogs 1.2 import moneroComponents.PendingTransaction 1.0 import "../components" import moneroComponents.Wallet 1.0 Rectangle { id: root signal paymentClicked(string address, string paymentId, string amount, int mixinCount, int priority, string description) signal sweepUnmixableClicked() color: "#F0EEEE" property string startLinkText: " (" + qsTr("Start daemon") + ")" + translationManager.emptyString property bool showAdvanced: false function scaleValueToMixinCount(scaleValue) { var scaleToMixinCount = [4,5,6,7,8,9,10,11,12,14,16,18,21,25]; if (scaleValue < scaleToMixinCount.length) { return scaleToMixinCount[scaleValue]; } else { return 0; } } function isValidOpenAliasAddress(address) { address = address.trim() var dot = address.indexOf('.') if (dot < 0) return false // we can get an awful lot of valid domains, including non ASCII chars... accept anything return true } function oa_message(text) { oaPopup.title = qsTr("OpenAlias error") + translationManager.emptyString oaPopup.text = text oaPopup.icon = StandardIcon.Information oaPopup.onCloseCallback = null oaPopup.open() } function updateMixin() { var fillLevel = (isMobile) ? privacyLevelItemSmall.fillLevel : privacyLevelItem.fillLevel var mixin = scaleValueToMixinCount(fillLevel) console.log("PrivacyLevel changed:" + fillLevel) console.log("mixin count: " + mixin) privacyLabel.text = qsTr("Privacy level (ringsize %1)").arg(mixin+1) + translationManager.emptyString } function updateFromQrCode(address, payment_id, amount, tx_description, recipient_name) { console.log("updateFromQrCode") addressLine.text = address paymentIdLine.text = payment_id amountLine.text = amount descriptionLine.text = recipient_name + " " + tx_description cameraUi.qrcode_decoded.disconnect(updateFromQrCode) } function clearFields() { addressLine.text = "" paymentIdLine.text = "" amountLine.text = "" descriptionLine.text = "" } // Information dialog StandardDialog { // dynamically change onclose handler property var onCloseCallback id: oaPopup cancelVisible: false onAccepted: { if (onCloseCallback) { onCloseCallback() } } } ColumnLayout { id: pageRoot anchors.top: parent.top anchors.left: parent.left anchors.right: parent.right anchors.margins: 17 * scaleRatio spacing: 0 GridLayout { columns: (isMobile)? 1 : 2 Layout.fillWidth: true ColumnLayout { Layout.fillWidth: true Label { id: amountLabel text: qsTr("Amount") + translationManager.emptyString } RowLayout { Layout.fillWidth: true id: amountRow Layout.minimumWidth: 200 Item { visible: !isMobile width: 37 * scaleRatio height: 37 * scaleRatio Image { anchors.centerIn: parent source: "../images/moneroIcon.png" } } // Amount input LineEdit { Layout.fillWidth: true id: amountLine placeholderText: qsTr("") + translationManager.emptyString width:100 validator: DoubleValidator { bottom: 0.0 top: 18446744.073709551615 decimals: 12 notation: DoubleValidator.StandardNotation locale: "C" } } StandardButton { id: amountAllButton width: 60 * scaleRatio text: qsTr("All") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" enabled : true onClicked: amountLine.text = "(all)" } } } ColumnLayout { Layout.fillWidth: true Label { id: transactionPriority text: qsTr("Transaction priority") + translationManager.emptyString } // Note: workaround for translations in listElements // ListElement: cannot use script for property value, so // code like this wont work: // ListElement { column1: qsTr("LOW") + translationManager.emptyString ; column2: ""; priority: PendingTransaction.Priority_Low } // For translations to work, the strings need to be listed in // the file components/StandardDropdown.qml too. // Priorites after v5 ListModel { id: priorityModelV5 ListElement { column1: qsTr("Slow (x0.25 fee)") ; column2: ""; priority: 1} ListElement { column1: qsTr("Default (x1 fee)") ; column2: ""; priority: 2 } ListElement { column1: qsTr("Fast (x5 fee)") ; column2: ""; priority: 3 } ListElement { column1: qsTr("Fastest (x41.5 fee)") ; column2: ""; priority: 4 } } StandardDropdown { Layout.fillWidth: true id: priorityDropdown shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" } } // Make sure dropdown is on top z: parent.z + 1 } ColumnLayout { Layout.fillWidth: true Label { id: addressLabel textFormat: Text.RichText text: "" + qsTr("Address") + " ( " + qsTr("Paste in or select from Address book") + " )" + translationManager.emptyString onLinkActivated: appWindow.showPageRequest("AddressBook") Layout.fillWidth: true } // recipient address input RowLayout { id: addressLineRow Layout.fillWidth: true StandardButton { id: qrfinderButton text: qsTr("QR Code") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" visible : appWindow.qrScannerEnabled enabled : visible width: visible ? 60 * scaleRatio : 0 onClicked: { cameraUi.state = "Capture" cameraUi.qrcode_decoded.connect(updateFromQrCode) } } LineEdit { id: addressLine Layout.fillWidth: true anchors.topMargin: 5 * scaleRatio placeholderText: "4..." // validator: RegExpValidator { regExp: /[0-9A-Fa-f]{95}/g } } StandardButton { id: resolveButton width: 60 * scaleRatio text: qsTr("Resolve") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" enabled : isValidOpenAliasAddress(addressLine.text) onClicked: { var result = walletManager.resolveOpenAlias(addressLine.text) if (result) { var parts = result.split("|") if (parts.length == 2) { var address_ok = walletManager.addressValid(parts[1], appWindow.persistentSettings.testnet) if (parts[0] === "true") { if (address_ok) { addressLine.text = parts[1] addressLine.cursorPosition = 0 } else oa_message(qsTr("No valid address found at this OpenAlias address")) } else if (parts[0] === "false") { if (address_ok) { addressLine.text = parts[1] addressLine.cursorPosition = 0 oa_message(qsTr("Address found, but the DNSSEC signatures could not be verified, so this address may be spoofed")) } else { oa_message(qsTr("No valid address found at this OpenAlias address, but the DNSSEC signatures could not be verified, so this may be spoofed")) } } else { oa_message(qsTr("Internal error")) } } else { oa_message(qsTr("Internal error")) } } else { oa_message(qsTr("No address found")) } } } } Label { id: paymentIdLabel text: qsTr("Payment ID ( Optional )") + translationManager.emptyString } // payment id input LineEdit { id: paymentIdLine placeholderText: qsTr("16 or 64 hexadecimal characters") + translationManager.emptyString Layout.fillWidth: true } Label { text: qsTr("Description ( Optional )") + translationManager.emptyString } LineEdit { id: descriptionLine placeholderText: qsTr("Saved to local wallet history") + translationManager.emptyString Layout.fillWidth: true } StandardButton { id: sendButton Layout.bottomMargin: 17 * scaleRatio Layout.topMargin: 17 * scaleRatio text: qsTr("Send") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" enabled : !appWindow.viewOnly && pageRoot.checkInformation(amountLine.text, addressLine.text, paymentIdLine.text, appWindow.persistentSettings.testnet) onClicked: { console.log("Transfer: paymentClicked") var priority = priorityModelV5.get(priorityDropdown.currentIndex).priority console.log("priority: " + priority) console.log("amount: " + amountLine.text) addressLine.text = addressLine.text.trim() paymentIdLine.text = paymentIdLine.text.trim() root.paymentClicked(addressLine.text, paymentIdLine.text, amountLine.text, scaleValueToMixinCount(privacyLevelItem.fillLevel), priority, descriptionLine.text) } } } function checkInformation(amount, address, payment_id, testnet) { address = address.trim() payment_id = payment_id.trim() var amount_ok = amount.length > 0 var address_ok = walletManager.addressValid(address, testnet) var payment_id_ok = payment_id.length == 0 || walletManager.paymentIdValid(payment_id) var ipid = walletManager.paymentIdFromAddress(address, testnet) if (ipid.length > 0 && payment_id.length > 0) payment_id_ok = false addressLine.error = !address_ok amountLine.error = !amount_ok paymentIdLine.error = !payment_id_ok return amount_ok && address_ok && payment_id_ok } } // pageRoot Rectangle { id:desaturate color:"black" anchors.fill: parent opacity: 0.1 visible: (pageRoot.enabled)? 0 : 1; } ColumnLayout { anchors.top: pageRoot.bottom anchors.left: parent.left anchors.right: parent.right anchors.leftMargin: 17 * scaleRatio anchors.topMargin: 17 * scaleRatio anchors.bottomMargin: 17 * scaleRatio spacing: 10 * scaleRatio enabled: !viewOnly || pageRoot.enabled RowLayout { CheckBox { id: showAdvancedCheckbox checked: persistentSettings.transferShowAdvanced onClicked: { persistentSettings.transferShowAdvanced = !persistentSettings.transferShowAdvanced } text: qsTr("Show advanced options") + translationManager.emptyString checkedIcon: "../images/checkedVioletIcon.png" uncheckedIcon: "../images/uncheckedIcon.png" } } Rectangle { visible: persistentSettings.transferShowAdvanced Layout.fillWidth: true height: 1 color: "#DEDEDE" Layout.bottomMargin: 30 * scaleRatio } RowLayout { visible: persistentSettings.transferShowAdvanced anchors.left: parent.left anchors.right: parent.right Layout.fillWidth: true Label { id: privacyLabel fontSize: 14 text: "" } Label { id: costLabel fontSize: 14 text: qsTr("Transaction cost") + translationManager.emptyString anchors.right: parent.right } } PrivacyLevel { visible: persistentSettings.transferShowAdvanced && !isMobile id: privacyLevelItem anchors.left: parent.left anchors.right: parent.right anchors.rightMargin: 17 * scaleRatio onFillLevelChanged: updateMixin() } PrivacyLevelSmall { visible: persistentSettings.transferShowAdvanced && isMobile id: privacyLevelItemSmall anchors.left: parent.left anchors.right: parent.right anchors.rightMargin: 17 * scaleRatio onFillLevelChanged: updateMixin() } GridLayout { visible: persistentSettings.transferShowAdvanced Layout.topMargin: 50 * scaleRatio columns: (isMobile) ? 2 : 6 StandardButton { id: sweepUnmixableButton text: qsTr("Sweep Unmixable") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" enabled : pageRoot.enabled onClicked: { console.log("Transfer: sweepUnmixableClicked") root.sweepUnmixableClicked() } } StandardButton { id: saveTxButton text: qsTr("Create tx file") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" visible: appWindow.viewOnly enabled: pageRoot.checkInformation(amountLine.text, addressLine.text, paymentIdLine.text, appWindow.persistentSettings.testnet) onClicked: { console.log("Transfer: saveTx Clicked") var priority = priorityModelV5.get(priorityDropdown.currentIndex).priority console.log("priority: " + priority) console.log("amount: " + amountLine.text) addressLine.text = addressLine.text.trim() paymentIdLine.text = paymentIdLine.text.trim() root.paymentClicked(addressLine.text, paymentIdLine.text, amountLine.text, scaleValueToMixinCount(privacyLevelItem.fillLevel), priority, descriptionLine.text) } } StandardButton { id: signTxButton text: qsTr("Sign tx file") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" visible: !appWindow.viewOnly onClicked: { console.log("Transfer: sign tx clicked") signTxDialog.open(); } } StandardButton { id: submitTxButton text: qsTr("Submit tx file") + translationManager.emptyString shadowReleasedColor: "#FF4304" shadowPressedColor: "#B32D00" releasedColor: "#FF6C3C" pressedColor: "#FF4304" visible: appWindow.viewOnly enabled: pageRoot.enabled onClicked: { console.log("Transfer: submit tx clicked") submitTxDialog.open(); } } } } //SignTxDialog FileDialog { id: signTxDialog title: qsTr("Please choose a file") + translationManager.emptyString folder: "file://" +moneroAccountsDir nameFilters: [ "Unsigned transfers (*)"] onAccepted: { var path = walletManager.urlToLocalPath(fileUrl); // Load the unsigned tx from file var transaction = currentWallet.loadTxFile(path); if (transaction.status !== PendingTransaction.Status_Ok) { console.error("Can't load unsigned transaction: ", transaction.errorString); informationPopup.title = qsTr("Error") + translationManager.emptyString; informationPopup.text = qsTr("Can't load unsigned transaction: ") + transaction.errorString informationPopup.icon = StandardIcon.Critical informationPopup.onCloseCallback = null informationPopup.open(); // deleting transaction object, we don't want memleaks transaction.destroy(); } else { confirmationDialog.text = qsTr("\nNumber of transactions: ") + transaction.txCount for (var i = 0; i < transaction.txCount; ++i) { confirmationDialog.text += qsTr("\nTransaction #%1").arg(i+1) +qsTr("\nRecipient: ") + transaction.recipientAddress[i] + (transaction.paymentId[i] == "" ? "" : qsTr("\n\payment ID: ") + transaction.paymentId[i]) + qsTr("\nAmount: ") + walletManager.displayAmount(transaction.amount(i)) + qsTr("\nFee: ") + walletManager.displayAmount(transaction.fee(i)) + qsTr("\nRingsize: ") + transaction.mixin(i+1) // TODO: add descriptions to unsigned_tx_set? // + (transactionDescription === "" ? "" : (qsTr("\n\nDescription: ") + transactionDescription)) + translationManager.emptyString if (i > 0) { confirmationDialog.text += "\n\n" } } console.log(transaction.confirmationMessage); // Show confirmation dialog confirmationDialog.title = qsTr("Confirmation") + translationManager.emptyString confirmationDialog.icon = StandardIcon.Question confirmationDialog.onAcceptedCallback = function() { transaction.sign(path+"_signed"); transaction.destroy(); }; confirmationDialog.onRejectedCallback = transaction.destroy; confirmationDialog.open() } } onRejected: { // File dialog closed console.log("Canceled") } } //SignTxDialog FileDialog { id: submitTxDialog title: qsTr("Please choose a file") + translationManager.emptyString folder: "file://" +moneroAccountsDir nameFilters: [ "signed transfers (*)"] onAccepted: { if(!currentWallet.submitTxFile(walletManager.urlToLocalPath(fileUrl))){ informationPopup.title = qsTr("Error") + translationManager.emptyString; informationPopup.text = qsTr("Can't submit transaction: ") + currentWallet.errorString informationPopup.icon = StandardIcon.Critical informationPopup.onCloseCallback = null informationPopup.open(); } else { informationPopup.title = qsTr("Information") + translationManager.emptyString informationPopup.text = qsTr("Money sent successfully") + translationManager.emptyString informationPopup.icon = StandardIcon.Information informationPopup.onCloseCallback = null informationPopup.open(); } } onRejected: { console.log("Canceled") } } Rectangle { x: root.width/2 - width/2 y: root.height/2 - height/2 height:statusText.paintedHeight + 50 * scaleRatio width:statusText.paintedWidth + 40 * scaleRatio visible: statusText.text != "" opacity: 0.9 Text { id: statusText anchors.fill:parent horizontalAlignment: Text.AlignHCenter verticalAlignment: Text.AlignVCenter textFormat: Text.RichText onLinkActivated: { appWindow.startDaemon(appWindow.persistentSettings.daemonFlags); } } } Component.onCompleted: { //Disable password page until enabled by updateStatus pageRoot.enabled = false } // fires on every page load function onPageCompleted() { console.log("transfer page loaded") updateStatus(); updateMixin(); updatePriorityDropdown() } function updatePriorityDropdown() { priorityDropdown.dataModel = priorityModelV5; priorityDropdown.currentIndex = 1 priorityDropdown.update() } //TODO: Add daemon sync status //TODO: enable send page when we're connected and daemon is synced function updateStatus() { if(typeof currentWallet === "undefined") { statusText.text = qsTr("Wallet is not connected to daemon.") + "
" + root.startLinkText return; } if (currentWallet.viewOnly) { // statusText.text = qsTr("Wallet is view only.") //return; } pageRoot.enabled = false; switch (currentWallet.connected()) { case Wallet.ConnectionStatus_Disconnected: statusText.text = qsTr("Wallet is not connected to daemon.") + "
" + root.startLinkText break case Wallet.ConnectionStatus_WrongVersion: statusText.text = qsTr("Connected daemon is not compatible with GUI. \n" + "Please upgrade or connect to another daemon") break default: if(!appWindow.daemonSynced){ statusText.text = qsTr("Waiting on daemon synchronization to finish") } else { // everything OK, enable transfer page // Light wallet is always ready pageRoot.enabled = true; statusText.text = ""; } } } // Popuplate fields from addressbook. function sendTo(address, paymentId, description){ addressLine.text = address paymentIdLine.text = paymentId descriptionLine.text = description } }