2022-03-01 16:59:38 +01:00
import QtQuick 2.14
import QtQuick . Controls 2.14
import QtQuick . Layouts 1.12
import shared . panels 1.0
import shared . controls 1.0
2022-03-08 00:59:38 +02:00
import shared . stores 1.0
2022-03-01 16:59:38 +01:00
import utils 1.0
import StatusQ . Controls 0.1
import StatusQ . Core 0.1
import StatusQ . Core . Theme 0.1
import StatusQ . Components 0.1
Column {
id: root
property bool ready: newPswInput . text . length >= root . minPswLen && newPswInput . text === confirmPswInput . text && errorTxt . text === ""
2022-04-12 12:04:21 +02:00
property int minPswLen: 10
2022-03-01 16:59:38 +01:00
property bool createNewPsw: true
property string title: qsTr ( "Create a password" )
property bool titleVisible: true
property string introText: qsTr ( "Create a password to unlock Status on this device & sign transactions." )
property string recoverText: qsTr ( "You will not be able to recover this password if it is lost." )
2022-05-12 12:08:28 +03:00
property string strengthenText: qsTr ( "Minimum %1 characters. To strengthen your password consider including:" ) . arg ( minPswLen )
2022-07-20 14:34:44 +02:00
property var passwordStrengthScoreFunction: function ( ) { }
2022-03-22 10:29:59 +01:00
2022-03-01 16:59:38 +01:00
readonly property int zBehind: 1
readonly property int zFront: 100
property alias currentPswText: currentPswInput . text
property alias newPswText: newPswInput . text
property alias confirmationPswText: confirmPswInput . text
property alias errorMsgText: errorTxt . text
2022-05-06 14:47:46 +02:00
signal returnPressed ( )
2022-03-01 16:59:38 +01:00
function forceNewPswInputFocus ( ) { newPswInput . forceActiveFocus ( Qt . MouseFocusReason ) }
function reset ( ) {
newPswInput . text = ""
currentPswInput . text = ""
confirmPswInput . text = ""
errorTxt . text = ""
strengthInditactor . strength = StatusPasswordStrengthIndicator . Strength . None
// Update focus:
if ( root . createNewPsw )
newPswInput . forceActiveFocus ( Qt . MouseFocusReason )
else
currentPswInput . forceActiveFocus ( Qt . MouseFocusReason )
}
2022-05-11 10:10:31 +02:00
function checkPasswordMatches ( onlyIfConfirmPasswordHasFocus = true ) {
2022-05-11 13:33:07 +02:00
if ( confirmPswInput . textField . text . length === 0 ) {
errorTxt . text = ""
return
}
2022-05-11 10:10:31 +02:00
if ( onlyIfConfirmPasswordHasFocus && ! confirmPswInput . textField . focus ) {
return
}
2022-03-01 16:59:38 +01:00
if ( newPswInput . text . length >= root . minPswLen ) {
errorTxt . text = ""
if ( confirmPswInput . text !== newPswInput . text ) {
errorTxt . text = qsTr ( "Passwords don't match" )
}
}
}
QtObject {
id: d
property bool containsLower: false
property bool containsUpper: false
property bool containsNumbers: false
property bool containsSymbols: false
2022-03-22 10:29:59 +01:00
readonly property var validator: RegExpValidator { regExp: /^[!-~]{0,64}$/ } // That incudes NOT extended ASCII printable characters less space and a maximum of 64 characters allowed
2022-03-01 16:59:38 +01:00
// Password strength categorization / validation
function lowerCaseValidator ( text ) { return ( /[a-z]/ . test ( text ) ) }
function upperCaseValidator ( text ) { return ( /[A-Z]/ . test ( text ) ) }
function numbersValidator ( text ) { return ( /\d/ . test ( text ) ) }
// That incudes NOT extended ASCII printable symbols less space:
function symbolsValidator ( text ) { return ( /[!-\/:-@[-`{-~]/ . test ( text ) ) }
2022-03-22 10:29:59 +01:00
// Used to convert strength from a given score to a specific category
function convertStrength ( score ) {
var strength = StatusPasswordStrengthIndicator . Strength . None
switch ( score ) {
2022-05-23 15:36:23 +03:00
case 0 : strength = StatusPasswordStrengthIndicator . Strength . VeryWeak ; break
case 1 : strength = StatusPasswordStrengthIndicator . Strength . Weak ; break
case 2 : strength = StatusPasswordStrengthIndicator . Strength . SoSo ; break
case 3 : strength = StatusPasswordStrengthIndicator . Strength . Good ; break
case 4 : strength = StatusPasswordStrengthIndicator . Strength . Great ; break
2022-03-22 10:29:59 +01:00
}
if ( strength > 4 )
strength = StatusPasswordStrengthIndicator . Strength . Great
return strength
2022-03-01 16:59:38 +01:00
}
// Password validation / error message selection:
function passwordValidation ( ) {
errorTxt . text = ""
// 3 rules to validate:
// * Password is in pwnd passwords database
if ( isInPwndDatabase ( ) )
errorTxt . text = qsTr ( "This password has been pwned and shouldn't be used" )
// * Common password
else if ( isCommonPassword ( ) )
errorTxt . text = qsTr ( "This password is a common word and shouldn't be used" )
// * Password too short
else if ( isTooShort ( ) )
2022-04-12 12:04:21 +02:00
errorTxt . text = qsTr ( "Password must be at least %1 characters long" ) . arg ( root . minPswLen )
2022-03-01 16:59:38 +01:00
}
function isInPwndDatabase ( ) {
// "TODO - Nice To Have: Pwnd password validation NOT implemented yet! "
return false
}
function isCommonPassword ( ) {
// "TODO - Nice To Have: Common password validation NOT implemented yet! "
return false
}
2022-05-23 15:36:23 +03:00
function isTooShort ( ) { return newPswInput . text . length < root . minPswLen }
2022-03-01 16:59:38 +01:00
}
spacing: 3 * Style . current . padding / 2
z: root . zFront
width: 416
// View visual content:
StatusBaseText {
id: title
anchors.horizontalCenter: parent . horizontalCenter
visible: root . titleVisible
text: root . title
font.pixelSize: 22
font.bold: true
color: Theme . palette . directColor1
}
Column {
StatusBaseText {
id: introTxt
anchors.horizontalCenter: parent . horizontalCenter
text: root . introText
font.pixelSize: 12
color: Theme . palette . baseColor1
}
StatusBaseText {
id: recoverTxt
anchors.horizontalCenter: parent . horizontalCenter
text: root . recoverText
font.pixelSize: 12
color: Theme . palette . dangerColor1
}
}
// TODO replace with StatusInput as soon as it supports password
Input {
id: currentPswInput
property bool showPassword
z: root . zFront
visible: ! root . createNewPsw
width: parent . width
placeholderText: qsTr ( "Current password" )
textField.echoMode: showPassword ? TextInput.Normal : TextInput . Password
textField.validator: d . validator
keepHeight: true
textField.rightPadding: showHideCurrentIcon . width + showHideCurrentIcon . anchors . rightMargin + Style . current . padding / 2
2022-05-06 14:47:46 +02:00
Keys.onReturnPressed: { root . returnPressed ( ) }
2022-03-01 16:59:38 +01:00
StatusFlatRoundButton {
id: showHideCurrentIcon
visible: currentPswInput . text !== ""
anchors.verticalCenter: parent . verticalCenter
anchors.right: parent . right
anchors.rightMargin: 16
width: 24
height: 24
icon.name: currentPswInput . showPassword ? "hide" : "show"
icon.color: Theme . palette . baseColor1
onClicked: currentPswInput . showPassword = ! currentPswInput . showPassword
}
}
Column {
spacing: Style . current . padding / 2
z: root . zFront
width: parent . width
// TODO replace with StatusInput as soon as it supports password
Input {
id: newPswInput
property bool showPassword
width: parent . width
placeholderText: qsTr ( "New password" )
textField.echoMode: showPassword ? TextInput.Normal : TextInput . Password
textField.validator: d . validator
keepHeight: true
textField.rightPadding: showHideNewIcon . width + showHideNewIcon . anchors . rightMargin + Style . current . padding / 2
onTextChanged: {
// Update password checkers
d . containsLower = d . lowerCaseValidator ( text )
d . containsUpper = d . upperCaseValidator ( text )
d . containsNumbers = d . numbersValidator ( text )
d . containsSymbols = d . symbolsValidator ( text )
// Update strength indicator:
2022-07-20 14:34:44 +02:00
strengthInditactor . strength = d . convertStrength ( root . passwordStrengthScoreFunction ( newPswInput . text ) )
2022-05-23 15:36:23 +03:00
if ( textField . text . length === confirmPswInput . text . length ) {
root . checkPasswordMatches ( false )
}
2022-03-01 16:59:38 +01:00
}
2022-05-06 14:47:46 +02:00
Keys.onReturnPressed: { root . returnPressed ( ) }
2022-03-01 16:59:38 +01:00
StatusFlatRoundButton {
id: showHideNewIcon
visible: newPswInput . text !== ""
anchors.verticalCenter: parent . verticalCenter
anchors.right: parent . right
anchors.rightMargin: 16
width: 24
height: 24
icon.name: newPswInput . showPassword ? "hide" : "show"
icon.color: Theme . palette . baseColor1
onClicked: newPswInput . showPassword = ! newPswInput . showPassword
}
}
StatusPasswordStrengthIndicator {
id: strengthInditactor
width: parent . width
anchors.horizontalCenter: parent . horizontalCenter
2022-05-12 12:08:28 +03:00
value: Math . min ( root . minPswLen , newPswInput . text . length )
2022-03-01 16:59:38 +01:00
from: 0
to: root . minPswLen
labelVeryWeak: qsTr ( "Very weak" )
labelWeak: qsTr ( "Weak" )
labelSoso: qsTr ( "So-so" )
labelGood: qsTr ( "Good" )
labelGreat: qsTr ( "Great" )
}
}
StatusBaseText {
id: strengthenTxt
anchors.horizontalCenter: parent . horizontalCenter
text: root . strengthenText
font.pixelSize: 12
color: Theme . palette . baseColor1
}
Row {
spacing: Style . current . padding
anchors.horizontalCenter: parent . horizontalCenter
StatusBaseText {
id: lowerCaseTxt
text: "• " + qsTr ( "Lower case" )
font.pixelSize: 12
color: d . containsLower ? Theme.palette.successColor1 : Theme . palette . baseColor1
}
StatusBaseText {
id: upperCaseTxt
text: "• " + qsTr ( "Upper case" )
font.pixelSize: 12
color: d . containsUpper ? Theme.palette.successColor1 : Theme . palette . baseColor1
}
StatusBaseText {
id: numbersTxt
text: "• " + qsTr ( "Numbers" )
font.pixelSize: 12
color: d . containsNumbers ? Theme.palette.successColor1 : Theme . palette . baseColor1
}
StatusBaseText {
id: symbolsTxt
text: "• " + qsTr ( "Symbols" )
font.pixelSize: 12
color: d . containsSymbols ? Theme.palette.successColor1 : Theme . palette . baseColor1
}
}
// TODO replace with StatusInput as soon as it supports password
Input {
id: confirmPswInput
property bool showPassword
z: root . zFront
width: parent . width
placeholderText: qsTr ( "Confirm password" )
textField.echoMode: showPassword ? TextInput.Normal : TextInput . Password
textField.validator: d . validator
keepHeight: true
textField.rightPadding: showHideConfirmIcon . width + showHideConfirmIcon . anchors . rightMargin + Style . current . padding / 2
2022-05-11 13:33:07 +02:00
onTextChanged: {
2022-05-18 17:30:00 +03:00
d . passwordValidation ( ) ;
if ( textField . text . length === newPswInput . text . length ) {
2022-05-11 13:33:07 +02:00
root . checkPasswordMatches ( )
}
}
2022-03-01 16:59:38 +01:00
textField.onFocusChanged: {
// When clicking into the confirmation input, validate if new password:
2022-05-11 10:10:31 +02:00
if ( textField . focus ) {
d . passwordValidation ( )
}
2022-03-01 16:59:38 +01:00
// When leaving the confirmation input because of the button or other input component is focused, check if password matches
2022-05-11 10:10:31 +02:00
else {
root . checkPasswordMatches ( false )
}
2022-03-01 16:59:38 +01:00
}
2022-05-06 14:47:46 +02:00
Keys.onReturnPressed: { root . returnPressed ( ) }
2022-03-01 16:59:38 +01:00
StatusFlatRoundButton {
id: showHideConfirmIcon
visible: confirmPswInput . text !== ""
anchors.verticalCenter: parent . verticalCenter
anchors.right: parent . right
anchors.rightMargin: 16
width: 24
height: 24
icon.name: confirmPswInput . showPassword ? "hide" : "show"
icon.color: Theme . palette . baseColor1
onClicked: confirmPswInput . showPassword = ! confirmPswInput . showPassword
}
}
StatusBaseText {
id: errorTxt
anchors.horizontalCenter: parent . horizontalCenter
font.pixelSize: 12
color: Theme . palette . dangerColor1
onTextChanged: {
if ( text === "" ) filler . visible = true
else filler . visible = false
}
}
// Just a column filler to keep the component height althought errorTxt.text is ""
Item {
id: filler
width: root . width
visible: true
height: errorTxt . height
}
}