status-desktop/ui/app/AppLayouts/Profile/popups/ChangePasswordModal.qml

203 lines
6.1 KiB
QML
Raw Normal View History

2021-08-11 09:55:59 +00:00
import QtQuick 2.13
import QtQuick.Controls 2.13
import QtQuick.Layouts 1.12
import utils 1.0
import shared 1.0
import shared.panels 1.0
import shared.controls 1.0
2021-08-11 09:55:59 +00:00
import StatusQ.Popups 0.1
import StatusQ.Controls 0.1
StatusModal {
id: root
property var privacyStore
signal passwordChanged()
width: 480
2021-08-11 09:55:59 +00:00
height: 510
closePolicy: Popup.NoAutoClose
header.title: qsTr("Change password")
2021-08-11 09:55:59 +00:00
onOpened: root.reset()
function lengthValidator(text) {
return text.length >= 6 ? "" : qsTr("At least 6 characters")
}
function reset() {
currentPasswordInput.state = "init"
passwordInput.state = "init"
confirmPasswordInput.state = "init"
currentPasswordInput.forceActiveFocus(Qt.MouseFocusReason)
}
function onChangePasswordResponse(success) {
if (success) {
passwordChanged()
submitBtn.enabled = false;
} else {
currentPasswordInput.state = "incorrect"
currentPasswordInput.forceActiveFocus(Qt.MouseFocusReason)
}
submitBtn.loading = false;
}
2021-08-11 09:55:59 +00:00
Connections {
target: root.privacyStore.privacyModule
onPasswordChanged: onChangePasswordResponse(success)
2021-08-11 09:55:59 +00:00
}
contentItem: ColumnLayout {
id: contentItem
2021-08-11 09:55:59 +00:00
anchors.fill: parent
anchors {
topMargin: Style.current.xlPadding + root.topPadding
leftMargin: Style.current.xlPadding
rightMargin: Style.current.xlPadding
bottomMargin: Style.current.xlPadding + root.bottomPadding
}
spacing: Style.current.padding
2021-08-11 09:55:59 +00:00
// TODO replace with StatusInput as soon as it supports password
2021-08-11 09:55:59 +00:00
Input {
id: currentPasswordInput
readonly property bool ready: state == "typing" && validationError == ""
2021-08-11 09:55:59 +00:00
anchors.left: undefined
anchors.right: undefined
Layout.fillWidth: true
label: qsTr("Current password")
2021-08-11 09:55:59 +00:00
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: currentPasswordInput; validationError: root.lengthValidator(text) }
},
State {
name: "incorrect"
PropertyChanges { target: currentPasswordInput; validationError: qsTr("Incorrect password") }
}
]
2021-08-11 09:55:59 +00:00
}
// TODO replace with StatusInput as soon as it supports password
2021-08-11 09:55:59 +00:00
Input {
id: passwordInput
readonly property bool ready: state == "typing" && validationError == ""
2021-08-11 09:55:59 +00:00
anchors.left: undefined
anchors.right: undefined
Layout.fillWidth: true
label: qsTr("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: 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")
2021-08-11 09:55:59 +00:00
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") : ""
}
}
]
2021-08-11 09:55:59 +00:00
}
Item {
Layout.fillHeight: true
2021-08-11 09:55:59 +00:00
}
StyledText {
text: qsTr("Your password protects your keys. You need it to unlock Status and transact.")
2021-08-11 09:55:59 +00:00
wrapMode: Text.WordWrap
Layout.preferredWidth: 340
Layout.alignment: Qt.AlignHCenter
horizontalAlignment: Text.AlignHCenter
color: Style.current.secondaryText
2021-08-12 12:09:57 +00:00
font.pixelSize: Style.current.tertiaryTextFontSize
2021-08-11 09:55:59 +00:00
}
}
rightButtons: [
2021-08-11 09:55:59 +00:00
StatusButton {
id: submitBtn
2021-08-11 09:55:59 +00:00
text: qsTr("Change password")
enabled: !submitBtn.loading && currentPasswordInput.ready &&
passwordInput.ready && confirmPasswordInput.ready
property Timer sim: Timer {
id: pause
interval: 20
onTriggered: {
root.privacyStore.changePassword(currentPasswordInput.text, passwordInput.text)
}
}
2021-08-11 09:55:59 +00:00
onClicked: {
submitBtn.loading = true;
//changePassword operation blocks the UI so loading = true; will never
//have any affect until changePassword is done. Getting around it with a
//small pause (timer) in order to get the desired behavior
pause.start();
}
2021-08-11 09:55:59 +00:00
}
]
2021-08-11 09:55:59 +00:00
}