mirror of
https://github.com/monero-project/monero-gui
synced 2024-12-01 08:11:54 +01:00
325 lines
11 KiB
QML
325 lines
11 KiB
QML
// 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.2
|
|
import moneroComponents.TranslationManager 1.0
|
|
import QtQuick.Dialogs 1.2
|
|
import QtQuick.Layouts 1.2
|
|
import "../components"
|
|
import 'utils.js' as Utils
|
|
|
|
// Reusable component for mnaging wallet (account name, path, private key)
|
|
ColumnLayout {
|
|
id: page
|
|
Layout.leftMargin: wizardLeftMargin
|
|
Layout.rightMargin: wizardRightMargin
|
|
property alias titleText: titleText.text
|
|
property alias accountNameText: accountName.text
|
|
property alias walletPath: fileUrlInput.text
|
|
property alias wordsTextItem : memoTextItem
|
|
property alias restoreHeight : restoreHeightItem.text
|
|
property alias restoreHeightVisible: restoreHeightItem.visible
|
|
property alias walletName : accountName.text
|
|
property alias progressDotsModel : progressDots.model
|
|
property alias recoverFromKeysAddress: addressLine.text;
|
|
property alias recoverFromKeysViewKey: viewKeyLine.text;
|
|
property alias recoverFromKeysSpendKey: spendKeyLine.text;
|
|
// recover mode or create new wallet
|
|
property bool recoverMode: false
|
|
// Recover form seed or keys
|
|
property bool recoverFromSeedMode: true
|
|
property int rowSpacing: 10
|
|
|
|
function checkFields(){
|
|
var addressOK = walletManager.addressValid(addressLine.text, persistentSettings.testnet)
|
|
var viewKeyOK = walletManager.keyValid(viewKeyLine.text, addressLine.text, true, persistentSettings.testnet)
|
|
// Spendkey is optional
|
|
var spendKeyOK = (spendKeyLine.text.length > 0)? walletManager.keyValid(spendKeyLine.text, addressLine.text, false, persistentSettings.testnet) : true
|
|
|
|
addressLine.error = !addressOK && addressLine.text.length != 0
|
|
viewKeyLine.error = !viewKeyOK && viewKeyLine.text.length != 0
|
|
spendKeyLine.error = !spendKeyOK && spendKeyLine.text.length != 0
|
|
|
|
return addressOK && viewKeyOK && spendKeyOK
|
|
}
|
|
|
|
function checkNextButton(){
|
|
wizard.nextButton.enabled = false
|
|
console.log("check next", recoverFromSeed.visible)
|
|
if(recoverMode && !recoverFromSeedMode) {
|
|
console.log("checking key fields")
|
|
wizard.nextButton.enabled = checkFields();
|
|
} else if (recoverMode && recoverFromSeedMode) {
|
|
wizard.nextButton.enabled = checkSeed()
|
|
} else
|
|
wizard.nextButton.enabled = true;
|
|
}
|
|
|
|
function checkSeed() {
|
|
console.log("Checking seed")
|
|
var wordsArray = Utils.lineBreaksToSpaces(uiItem.wordsTextItem.memoText).split(" ");
|
|
return wordsArray.length === 25 || wordsArray.length === 24
|
|
}
|
|
|
|
function updateFromQrCode(address, payment_id, amount, tx_description, recipient_name, extra_parameters) {
|
|
// Switch to recover from keys
|
|
recoverFromSeedMode = false
|
|
spendKeyLine.text = ""
|
|
viewKeyLine.text = ""
|
|
restoreHeightItem.text = ""
|
|
|
|
|
|
if(typeof extra_parameters.secret_view_key != "undefined") {
|
|
viewKeyLine.text = extra_parameters.secret_view_key
|
|
}
|
|
if(typeof extra_parameters.secret_spend_key != "undefined") {
|
|
spendKeyLine.text = extra_parameters.secret_spend_key
|
|
}
|
|
if(typeof extra_parameters.restore_height != "undefined") {
|
|
restoreHeightItem.text = extra_parameters.restore_height
|
|
}
|
|
addressLine.text = address
|
|
|
|
cameraUi.qrcode_decoded.disconnect(updateFromQrCode)
|
|
|
|
// Check if keys are correct
|
|
checkNextButton();
|
|
}
|
|
|
|
RowLayout {
|
|
id: dotsRow
|
|
Layout.alignment: Qt.AlignRight
|
|
spacing: 6
|
|
|
|
ListModel {
|
|
id: dotsModel
|
|
ListElement { dotColor: "#FFE00A" }
|
|
ListElement { dotColor: "#DBDBDB" }
|
|
ListElement { dotColor: "#DBDBDB" }
|
|
ListElement { dotColor: "#DBDBDB" }
|
|
}
|
|
|
|
Repeater {
|
|
id: progressDots
|
|
model: dotsModel
|
|
delegate: Rectangle {
|
|
width: 12; height: 12
|
|
radius: 6
|
|
color: dotColor
|
|
}
|
|
}
|
|
}
|
|
|
|
RowLayout {
|
|
id: headerColumn
|
|
Layout.fillWidth: true
|
|
Text {
|
|
Layout.fillWidth: true
|
|
horizontalAlignment: Text.AlignHCenter
|
|
id: titleText
|
|
font.family: "Arial"
|
|
font.pixelSize: 28 * scaleRatio
|
|
wrapMode: Text.Wrap
|
|
color: "#3F3F3F"
|
|
}
|
|
}
|
|
|
|
ColumnLayout {
|
|
Layout.bottomMargin: rowSpacing
|
|
|
|
Label {
|
|
Layout.topMargin: 20 * scaleRatio
|
|
fontSize: 14 * scaleRatio
|
|
text: qsTr("Wallet name")
|
|
+ translationManager.emptyString
|
|
}
|
|
|
|
LineEdit {
|
|
id: accountName
|
|
Layout.fillWidth: true
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
text: defaultAccountName
|
|
onTextUpdated: checkNextButton()
|
|
}
|
|
}
|
|
|
|
GridLayout{
|
|
columns: (isMobile)? 2 : 4
|
|
visible: recoverMode
|
|
|
|
StandardButton {
|
|
id: recoverFromSeedButton
|
|
text: qsTr("Restore from seed") + translationManager.emptyString
|
|
shadowReleasedColor: "#FF4304"
|
|
shadowPressedColor: "#B32D00"
|
|
releasedColor: "#FF6C3C"
|
|
pressedColor: "#FF4304"
|
|
enabled: recoverFromKeys.visible
|
|
onClicked: {
|
|
recoverFromSeedMode = true;
|
|
checkNextButton();
|
|
}
|
|
}
|
|
|
|
StandardButton {
|
|
id: recoverFromKeysButton
|
|
text: qsTr("Restore from keys") + translationManager.emptyString
|
|
shadowReleasedColor: "#FF4304"
|
|
shadowPressedColor: "#B32D00"
|
|
releasedColor: "#FF6C3C"
|
|
pressedColor: "#FF4304"
|
|
enabled: recoverFromSeed.visible
|
|
onClicked: {
|
|
recoverFromSeedMode = false;
|
|
checkNextButton();
|
|
}
|
|
}
|
|
|
|
StandardButton {
|
|
id: qrfinderButton
|
|
text: qsTr("From QR Code") + translationManager.emptyString
|
|
shadowReleasedColor: "#FF4304"
|
|
shadowPressedColor: "#B32D00"
|
|
releasedColor: "#FF6C3C"
|
|
pressedColor: "#FF4304"
|
|
visible : true //appWindow.qrScannerEnabled
|
|
enabled : visible
|
|
onClicked: {
|
|
cameraUi.state = "Capture"
|
|
cameraUi.qrcode_decoded.connect(updateFromQrCode)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// Recover from seed
|
|
RowLayout {
|
|
id: recoverFromSeed
|
|
visible: !recoverMode || ( recoverMode && recoverFromSeedMode)
|
|
WizardMemoTextInput {
|
|
id : memoTextItem
|
|
Layout.fillWidth: true
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
}
|
|
}
|
|
|
|
// Recover from keys
|
|
GridLayout {
|
|
Layout.bottomMargin: page.rowSpacing
|
|
rowSpacing: page.rowSpacing
|
|
id: recoverFromKeys
|
|
visible: recoverMode && !recoverFromSeedMode
|
|
columns: 1
|
|
LineEdit {
|
|
Layout.fillWidth: true
|
|
id: addressLine
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
placeholderText: qsTr("Account address (public)") + translationManager.emptyString
|
|
onTextUpdated: checkNextButton()
|
|
}
|
|
LineEdit {
|
|
Layout.fillWidth: true
|
|
id: viewKeyLine
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
placeholderText: qsTr("View key (private)") + translationManager.emptyString
|
|
onTextUpdated: checkNextButton()
|
|
|
|
}
|
|
LineEdit {
|
|
Layout.fillWidth: true
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
id: spendKeyLine
|
|
placeholderText: qsTr("Spend key (private)") + translationManager.emptyString
|
|
onTextUpdated: checkNextButton()
|
|
}
|
|
}
|
|
|
|
// Restore Height
|
|
RowLayout {
|
|
LineEdit {
|
|
id: restoreHeightItem
|
|
Layout.fillWidth: true
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
placeholderText: qsTr("Restore height (optional)") + translationManager.emptyString
|
|
validator: IntValidator {
|
|
bottom:0
|
|
}
|
|
}
|
|
}
|
|
|
|
// Wallet store location
|
|
ColumnLayout {
|
|
Label {
|
|
Layout.fillWidth: true
|
|
Layout.topMargin: 20 * scaleRatio
|
|
fontSize: 14
|
|
text: qsTr("Your wallet is stored in") + ": " + fileUrlInput.text;
|
|
}
|
|
|
|
LineEdit {
|
|
Layout.fillWidth: true
|
|
Layout.maximumWidth: 600 * scaleRatio
|
|
Layout.minimumWidth: 200 * scaleRatio
|
|
id: fileUrlInput
|
|
text: moneroAccountsDir + "/"
|
|
|
|
// workaround for the bug "filechooser only opens once"
|
|
MouseArea {
|
|
anchors.fill: parent
|
|
onClicked: {
|
|
mouse.accepted = false
|
|
fileDialog.folder = walletManager.localPathToUrl(fileUrlInput.text)
|
|
fileDialog.open()
|
|
fileUrlInput.focus = true
|
|
}
|
|
}
|
|
}
|
|
|
|
FileDialog {
|
|
id: fileDialog
|
|
selectMultiple: false
|
|
selectFolder: true
|
|
title: qsTr("Please choose a directory") + translationManager.emptyString
|
|
onAccepted: {
|
|
fileUrlInput.text = walletManager.urlToLocalPath(fileDialog.folder)
|
|
fileDialog.visible = false
|
|
}
|
|
onRejected: {
|
|
fileDialog.visible = false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|