2020-06-17 15:18:31 -04:00
|
|
|
import QtQuick 2.13
|
2020-07-30 15:18:54 +10:00
|
|
|
import QtQuick.Controls 2.13
|
2020-06-25 15:31:30 -04:00
|
|
|
import QtQuick.Dialogs 1.3
|
2020-06-08 14:35:48 -04:00
|
|
|
import "../../../../imports"
|
|
|
|
import "../../../../shared"
|
2021-01-12 11:58:53 +01:00
|
|
|
import "../../../../shared/status"
|
2020-06-08 14:35:48 -04:00
|
|
|
|
|
|
|
ModalPopup {
|
|
|
|
id: popup
|
2021-04-14 16:50:09 -04:00
|
|
|
height: 615
|
2020-06-08 14:35:48 -04:00
|
|
|
|
|
|
|
property int marginBetweenInputs: 38
|
2020-06-22 13:57:06 -04:00
|
|
|
property string passwordValidationError: ""
|
|
|
|
property string seedValidationError: ""
|
|
|
|
property string accountNameValidationError: ""
|
2020-06-25 15:31:30 -04:00
|
|
|
property bool loading: false
|
fix: loading of wallet history, display of tx datetime
Fixes: #2539.
Transaction history is now correctly being fetched from status-go as per mobile. Firstly, when accounts are added (ie as watch accounts), `wallet_checkRecentHistory` must be called first so that the status-go db is populated. After that, `wallet_getTransfersByAddress` can be called. On app load, when we run the `initBalance` logic, we are calling `wallet_getTransfersByAddress`, asking for the last 20 transactions with the `loadMore` parameter set to false. When the user navigates to the Wallet > History tab, they can then click “Load More” to fetch more transactions from status-go. Once the number of transactions returns false below the expected amount, the remaining transactions to fetch have been exhausted and the “Load More” button is disabled.
feat: add non-archival node warning to the UI to indicate to the user that they may not have complete results
feat: set active account to the added account
Once an account is added to the wallet, that newly added account is selected as the active account.
1. The “load more” button is active when new transactions that aren’t already displayed are returned from `wallet_getTransfersByAddress`. This is the only way to enable or disable the “Load more” button as status-go is not able to return information regarding whether or not there are more transactions to be fetched. The downside to this is that lets say the last page of transactions is returned, but there are no more pages left. These returned txs are not currently displayed, so the “load more” button will still be enabled. However, the next click of the button will return 0 results, thus disabling it. It’s effectively an extra click to get to the disabled state.
2. For more information on how the `toBlock` parameter operates for the `wallet_getTransfersForAddress` RPC call, see https://notes.status.im/XmENTrVRRaqhwE4gK0m8Mg?view.
2021-06-01 00:03:41 +10:00
|
|
|
property var onAfterAddAccount: function() {}
|
2020-06-22 13:57:06 -04:00
|
|
|
|
2020-09-11 13:34:52 +02:00
|
|
|
function reset() {
|
|
|
|
passwordInput.text = ""
|
|
|
|
accountNameInput.text = ""
|
2021-04-14 16:50:09 -04:00
|
|
|
seedPhraseTextArea.textArea.text = ""
|
2020-09-11 13:34:52 +02:00
|
|
|
}
|
|
|
|
|
2020-06-22 13:57:06 -04:00
|
|
|
function validate() {
|
|
|
|
if (passwordInput.text === "") {
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "You need to enter a password"
|
|
|
|
passwordValidationError = qsTrId("you-need-to-enter-a-password")
|
2020-12-04 00:26:05 +02:00
|
|
|
} else if (passwordInput.text.length < 6) {
|
2021-02-18 11:36:05 -05:00
|
|
|
//% "Password needs to be 6 characters or more"
|
|
|
|
passwordValidationError = qsTrId("password-needs-to-be-6-characters-or-more")
|
2020-06-22 13:57:06 -04:00
|
|
|
} else {
|
|
|
|
passwordValidationError = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if (accountNameInput.text === "") {
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "You need to enter an account name"
|
|
|
|
accountNameValidationError = qsTrId("you-need-to-enter-an-account-name")
|
2020-06-22 13:57:06 -04:00
|
|
|
} else {
|
|
|
|
accountNameValidationError = ""
|
|
|
|
}
|
|
|
|
|
2021-04-14 16:50:09 -04:00
|
|
|
if (seedPhraseTextArea.textArea.text === "") {
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "You need to enter a seed phrase"
|
|
|
|
seedValidationError = qsTrId("you-need-to-enter-a-seed-phrase")
|
2021-04-14 16:50:09 -04:00
|
|
|
} else if (!Utils.isMnemonic(seedPhraseTextArea.textArea.text)) {
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "Enter a valid mnemonic"
|
|
|
|
seedValidationError = qsTrId("enter-a-valid-mnemonic")
|
2020-06-22 13:57:06 -04:00
|
|
|
} else {
|
|
|
|
seedValidationError = ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return passwordValidationError === "" && seedValidationError === "" && accountNameValidationError === ""
|
|
|
|
}
|
2020-06-08 14:35:48 -04:00
|
|
|
|
|
|
|
onOpened: {
|
2021-04-14 16:50:09 -04:00
|
|
|
seedPhraseTextArea.textArea.text = "";
|
2020-11-27 11:13:35 -04:00
|
|
|
passwordInput.text = "";
|
|
|
|
accountNameInput.text = "";
|
2020-12-02 14:51:56 -04:00
|
|
|
passwordValidationError = "";
|
|
|
|
seedValidationError = "";
|
|
|
|
accountNameValidationError = "";
|
2021-03-16 15:19:48 -04:00
|
|
|
accountColorInput.selectedColor = Style.current.accountColors[Math.floor(Math.random() * Style.current.accountColors.length)]
|
2020-06-08 14:35:48 -04:00
|
|
|
passwordInput.forceActiveFocus(Qt.MouseFocusReason)
|
|
|
|
}
|
|
|
|
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "Add account with a seed phrase"
|
|
|
|
title: qsTrId("add-seed-account")
|
2020-06-09 12:01:19 +02:00
|
|
|
|
2020-06-08 14:35:48 -04:00
|
|
|
Input {
|
|
|
|
id: passwordInput
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "Enter your password…"
|
|
|
|
placeholderText: qsTrId("enter-your-password…")
|
|
|
|
//% "Password"
|
|
|
|
label: qsTrId("password")
|
2020-06-08 14:35:48 -04:00
|
|
|
textField.echoMode: TextInput.Password
|
2020-06-22 13:57:06 -04:00
|
|
|
validationError: popup.passwordValidationError
|
2020-06-08 14:35:48 -04:00
|
|
|
}
|
|
|
|
|
2021-04-14 16:50:09 -04:00
|
|
|
SeedPhraseTextArea {
|
|
|
|
id: seedPhraseTextArea
|
2020-06-08 14:35:48 -04:00
|
|
|
anchors.top: passwordInput.bottom
|
|
|
|
anchors.topMargin: marginBetweenInputs
|
2021-04-14 16:50:09 -04:00
|
|
|
width: parent.width
|
2020-10-03 15:53:46 +02:00
|
|
|
}
|
|
|
|
|
2020-06-08 14:35:48 -04:00
|
|
|
Input {
|
|
|
|
id: accountNameInput
|
2021-04-14 16:50:09 -04:00
|
|
|
anchors.top: seedPhraseTextArea.bottom
|
2020-06-08 14:35:48 -04:00
|
|
|
anchors.topMargin: marginBetweenInputs
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "Enter an account name..."
|
|
|
|
placeholderText: qsTrId("enter-an-account-name...")
|
|
|
|
//% "Account name"
|
|
|
|
label: qsTrId("account-name")
|
2020-06-22 13:57:06 -04:00
|
|
|
validationError: popup.accountNameValidationError
|
2020-06-08 14:35:48 -04:00
|
|
|
}
|
|
|
|
|
2021-01-20 17:24:21 +01:00
|
|
|
StatusWalletColorSelect {
|
2020-06-08 14:35:48 -04:00
|
|
|
id: accountColorInput
|
|
|
|
anchors.top: accountNameInput.bottom
|
|
|
|
anchors.topMargin: marginBetweenInputs
|
2020-07-30 15:18:54 +10:00
|
|
|
anchors.left: parent.left
|
|
|
|
anchors.right: parent.right
|
2020-06-08 14:35:48 -04:00
|
|
|
}
|
|
|
|
|
2021-01-12 11:58:53 +01:00
|
|
|
footer: StatusButton {
|
2020-06-08 14:35:48 -04:00
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.right: parent.right
|
2021-01-12 11:58:53 +01:00
|
|
|
text: loading ?
|
2020-07-06 16:39:55 -04:00
|
|
|
//% "Loading..."
|
|
|
|
qsTrId("loading") :
|
2021-02-18 11:36:05 -05:00
|
|
|
//% "Add account"
|
|
|
|
qsTrId("add-account")
|
2020-06-08 14:35:48 -04:00
|
|
|
|
2021-04-14 16:50:09 -04:00
|
|
|
enabled: !loading && passwordInput.text !== "" && accountNameInput.text !== "" && seedPhraseTextArea.correctWordCount
|
2020-06-25 15:31:30 -04:00
|
|
|
|
|
|
|
MessageDialog {
|
|
|
|
id: accountError
|
|
|
|
title: "Adding the account failed"
|
|
|
|
icon: StandardIcon.Critical
|
|
|
|
standardButtons: StandardButton.Ok
|
|
|
|
}
|
2020-06-08 14:35:48 -04:00
|
|
|
|
|
|
|
onClicked : {
|
2021-04-14 16:50:09 -04:00
|
|
|
// TODO the loading doesn't work because the function freezes the view. Might need to use threads
|
2020-06-25 15:31:30 -04:00
|
|
|
loading = true
|
2021-04-14 16:50:09 -04:00
|
|
|
if (!validate() || !seedPhraseTextArea.validateSeed()) {
|
2020-06-30 16:01:37 -04:00
|
|
|
errorSound.play()
|
2020-06-25 15:31:30 -04:00
|
|
|
return loading = false
|
|
|
|
}
|
|
|
|
|
2021-06-08 08:48:31 -04:00
|
|
|
const result = walletModel.accountsView.addAccountsFromSeed(seedPhraseTextArea.textArea.text, passwordInput.text, accountNameInput.text, accountColorInput.selectedColor)
|
2020-06-25 15:31:30 -04:00
|
|
|
loading = false
|
fix: prevent crash on generate account wrong password
Fixes #2448.
Currently, if a wrong password is entered when generating a wallet account, the app will crash due to attempting to decode a `GeneratedAccount ` from an rpc response containing only an error.
With this PR, we are detecting if an error is returned in the response, and if so, raising a StatusGoException. This exception is caught in the call chain, and translated in to a `StatusGoError` which is serialised and sent to the QML view, where it is parsed and displayed as an invalid password error in the input box.
refactor: remove string return values as error messages in wallet model
In the wallet model, we were passing back empty strings for no error, or an error as a string. This is not only confusing, but does not benefit from leaning on the compiler and strong types. One has to read the entire code to understand if a string result is returned when there is no error instead of implicitly being able to understand there is no return type.
To alleviate this, account creation fundtions that do not need to return a value have been changed to a void return type, and raise `StatusGoException` if there is an error encountered. This can be caught in the call chain and used as necessary (ie to pass to QML).
refactor: move invalid password string detection to Utils
Currently, we are reading returned view model values and checking to see if they include a known string from Status Go that means there was an invalid password used. This string was placed in the codebased in mulitple locations.
This PR moves the string check to a Utils function and updates all the references to use the function in Utils.
2021-05-13 14:41:48 +10:00
|
|
|
if (result) {
|
|
|
|
let resultJson = JSON.parse(result);
|
|
|
|
errorSound.play();
|
|
|
|
if (Utils.isInvalidPasswordMessage(resultJson.error)) {
|
|
|
|
//% "Wrong password"
|
|
|
|
popup.passwordValidationError = qsTrId("wrong-password")
|
|
|
|
} else {
|
|
|
|
accountError.text = resultJson.error
|
|
|
|
accountError.open()
|
|
|
|
}
|
|
|
|
return
|
2020-06-22 13:57:06 -04:00
|
|
|
}
|
fix: loading of wallet history, display of tx datetime
Fixes: #2539.
Transaction history is now correctly being fetched from status-go as per mobile. Firstly, when accounts are added (ie as watch accounts), `wallet_checkRecentHistory` must be called first so that the status-go db is populated. After that, `wallet_getTransfersByAddress` can be called. On app load, when we run the `initBalance` logic, we are calling `wallet_getTransfersByAddress`, asking for the last 20 transactions with the `loadMore` parameter set to false. When the user navigates to the Wallet > History tab, they can then click “Load More” to fetch more transactions from status-go. Once the number of transactions returns false below the expected amount, the remaining transactions to fetch have been exhausted and the “Load More” button is disabled.
feat: add non-archival node warning to the UI to indicate to the user that they may not have complete results
feat: set active account to the added account
Once an account is added to the wallet, that newly added account is selected as the active account.
1. The “load more” button is active when new transactions that aren’t already displayed are returned from `wallet_getTransfersByAddress`. This is the only way to enable or disable the “Load more” button as status-go is not able to return information regarding whether or not there are more transactions to be fetched. The downside to this is that lets say the last page of transactions is returned, but there are no more pages left. These returned txs are not currently displayed, so the “load more” button will still be enabled. However, the next click of the button will return 0 results, thus disabling it. It’s effectively an extra click to get to the disabled state.
2. For more information on how the `toBlock` parameter operates for the `wallet_getTransfersForAddress` RPC call, see https://notes.status.im/XmENTrVRRaqhwE4gK0m8Mg?view.
2021-06-01 00:03:41 +10:00
|
|
|
popup.onAfterAddAccount()
|
2020-09-11 13:34:52 +02:00
|
|
|
popup.reset()
|
2020-06-08 14:35:48 -04:00
|
|
|
popup.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*##^##
|
|
|
|
Designer {
|
|
|
|
D{i:0;formeditorColor:"#ffffff";height:500;width:400}
|
|
|
|
}
|
|
|
|
##^##*/
|