fix(@desktop/settings): [re-add] ensure new password is confirmed

Fixes #4672
This commit is contained in:
Jonathan Rainville 2022-02-10 10:44:40 -05:00
parent 3b43d2f9ea
commit cd68691ed8
2 changed files with 106 additions and 57 deletions

View File

@ -12,49 +12,42 @@ import StatusQ.Popups 0.1
import StatusQ.Controls 0.1 import StatusQ.Controls 0.1
StatusModal { StatusModal {
id: popup id: root
property var privacyStore
signal passwordChanged()
width: 480 width: 480
height: 510 height: 510
closePolicy: Popup.NoAutoClose closePolicy: Popup.NoAutoClose
header.title: qsTr("Change password") header.title: qsTr("Change password")
onOpened: { onOpened: root.reset()
reset();
function lengthValidator(text) {
return text.length >= 6 ? "" : qsTr("At least 6 characters")
} }
property var privacyStore
property var successPopup
property string indicationText: ""
property bool passwordInputValid
property bool currPasswordInputValid
property string currPasswordValidationError: ""
function reset() { function reset() {
passwordInput.text = ""; currentPasswordInput.state = "init"
currentPasswordInput.text = ""; passwordInput.state = "init"
currentPasswordInput.forceActiveFocus(Qt.MouseFocusReason); confirmPasswordInput.state = "init"
popup.indicationText = "At least 6 characters. Your password protects your keys. You need it to unlock Status and transact."; currentPasswordInput.forceActiveFocus(Qt.MouseFocusReason)
popup.currPasswordValidationError = "";
passwordInput.validationError = "";
popup.passwordInputValid = false;
popup.currPasswordInputValid = false;
} }
function onChangePasswordResponse(success) { function onChangePasswordResponse(success) {
if (success) { if (success) {
popup.successPopup.open(); passwordChanged()
submitBtn.enabled = false; submitBtn.enabled = false;
} else { } else {
reset(); currentPasswordInput.state = "incorrect"
passwordInput.validationError = " "; currentPasswordInput.forceActiveFocus(Qt.MouseFocusReason)
popup.currPasswordValidationError = qsTr("Incorrect password");
} }
submitBtn.loading = false; submitBtn.loading = false;
} }
Connections { Connections {
target: popup.privacyStore.privacyModule target: root.privacyStore.privacyModule
onPasswordChanged: onChangePasswordResponse(success) onPasswordChanged: onChangePasswordResponse(success)
} }
@ -62,44 +55,108 @@ StatusModal {
id: contentItem id: contentItem
anchors.fill: parent anchors.fill: parent
anchors { anchors {
topMargin: (Style.current.xlPadding + popup.topPadding) topMargin: Style.current.xlPadding + root.topPadding
leftMargin: Style.current.xlPadding leftMargin: Style.current.xlPadding
rightMargin: Style.current.xlPadding rightMargin: Style.current.xlPadding
bottomMargin: (Style.current.xlPadding + popup.bottomPadding) bottomMargin: Style.current.xlPadding + root.bottomPadding
} }
spacing: Style.current.padding spacing: Style.current.padding
//TODO replace with StatusInput as soon as it supports password // TODO replace with StatusInput as soon as it supports password
Input { Input {
id: currentPasswordInput id: currentPasswordInput
readonly property bool ready: state == "typing" && validationError == ""
anchors.left: undefined anchors.left: undefined
anchors.right: undefined anchors.right: undefined
Layout.fillWidth: true Layout.fillWidth: true
placeholderText: ""
label: qsTr("Current password") label: qsTr("Current password")
textField.echoMode: TextInput.Password textField.echoMode: TextInput.Password
onTextChanged: { keepHeight: true
popup.currPasswordInputValid = (currentPasswordInput.text.length >= 6); placeholderText: ""
state: "init"
onTextChanged: if (text != "" && state != "typing") state = "typing"
onStateChanged: if (state == "init") resetInternal()
states: [
State {
name: "init"
},
State {
name: "typing"
PropertyChanges { target: currentPasswordInput; validationError: root.lengthValidator(text) }
},
State {
name: "incorrect"
PropertyChanges { target: currentPasswordInput; validationError: qsTr("Incorrect password") }
} }
]
} }
//TODO replace with StatusInput as soon as it supports password // TODO replace with StatusInput as soon as it supports password
Input { Input {
id: passwordInput id: passwordInput
readonly property bool ready: state == "typing" && validationError == ""
anchors.left: undefined anchors.left: undefined
anchors.right: undefined anchors.right: undefined
Layout.fillWidth: true Layout.fillWidth: true
placeholderText: "" label: qsTr("New password")
label: qsTrId("new-password...")
textField.echoMode: TextInput.Password textField.echoMode: TextInput.Password
onTextChanged: { keepHeight: true
popup.passwordInputValid = ((passwordInput.text !== "") && (passwordInput.text.length >= 6)); placeholderText: ""
//setting validationError so that input becomes red state: "init"
passwordInput.validationError = (!popup.passwordInputValid) ? " " : "";
popup.indicationText = (!popup.passwordInputValid ? "<font color=\"#FF2D55\">" : "") onTextChanged: if (text != "" && state != "typing") state = "typing"
+ "At least 6 characters." + (!popup.passwordInputValid ? "</font>" : "") onStateChanged: if (state == "init") resetInternal()
+ "Your password protects your keys. You need it to unlock Status and transact."
states: [
State {
name: "init"
},
State {
name: "typing"
PropertyChanges { target: passwordInput; validationError: root.lengthValidator(text) }
} }
]
}
// TODO replace with StatusInput as soon as it supports password
Input {
id: confirmPasswordInput
readonly property bool ready: state == "typing" && validationError == ""
anchors.left: undefined
anchors.right: undefined
Layout.fillWidth: true
label: qsTr("Confirm new password")
textField.echoMode: TextInput.Password
keepHeight: true
placeholderText: ""
state: "init"
onTextChanged: if (text != "" && state != "typing") state = "typing"
onStateChanged: if (state == "init") resetInternal()
states: [
State {
name: "init"
},
State {
name: "typing"
PropertyChanges {
target: confirmPasswordInput;
validationError: confirmPasswordInput.text != passwordInput.text ? qsTr("Password does not match") : ""
}
}
]
} }
Item { Item {
@ -107,18 +164,7 @@ StatusModal {
} }
StyledText { StyledText {
id: validationError text: qsTr("Your password protects your keys. You need it to unlock Status and transact.")
Layout.preferredWidth: parent.width
Layout.alignment: Qt.AlignHCenter
horizontalAlignment: Text.AlignHCenter
visible: (text !== "")
font.pixelSize: Style.current.tertiaryTextFontSize
color: Style.current.danger
text: popup.currPasswordValidationError
}
StyledText {
text: qsTr(indicationText)
wrapMode: Text.WordWrap wrapMode: Text.WordWrap
Layout.preferredWidth: 340 Layout.preferredWidth: 340
Layout.alignment: Qt.AlignHCenter Layout.alignment: Qt.AlignHCenter
@ -131,14 +177,16 @@ StatusModal {
rightButtons: [ rightButtons: [
StatusButton { StatusButton {
id: submitBtn id: submitBtn
text: qsTr("Change password") text: qsTr("Change password")
enabled: (popup.passwordInputValid && popup.currPasswordInputValid && !submitBtn.loading) enabled: !submitBtn.loading && currentPasswordInput.ready &&
passwordInput.ready && confirmPasswordInput.ready
property Timer sim: Timer { property Timer sim: Timer {
id: pause id: pause
interval: 20 interval: 20
onTriggered: { onTriggered: {
popup.privacyStore.changePassword(currentPasswordInput.text, passwordInput.text) root.privacyStore.changePassword(currentPasswordInput.text, passwordInput.text)
} }
} }

View File

@ -119,7 +119,8 @@ Item {
id: changePasswordModal id: changePasswordModal
privacyStore: root.privacyStore privacyStore: root.privacyStore
anchors.centerIn: parent anchors.centerIn: parent
successPopup: successPopup
onPasswordChanged: successPopup.open()
} }
ChangePasswordSuccessModal { ChangePasswordSuccessModal {