refactor credentials setup page to handle the focus navigation
This commit is contained in:
parent
39cbe6de28
commit
f331c11f51
1 changed files with 151 additions and 65 deletions
|
|
@ -20,99 +20,125 @@ PageType {
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
anchors.topMargin: 20
|
anchors.topMargin: 20
|
||||||
|
|
||||||
|
onFocusChanged: {
|
||||||
|
if (this.activeFocus) {
|
||||||
|
listView.positionViewAtBeginning()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
FlickableType {
|
ListView {
|
||||||
id: fl
|
id: listView
|
||||||
anchors.top: backButton.bottom
|
anchors.top: backButton.bottom
|
||||||
anchors.bottom: parent.bottom
|
anchors.bottom: parent.bottom
|
||||||
contentHeight: content.height
|
anchors.right: parent.right
|
||||||
|
anchors.left: parent.left
|
||||||
|
|
||||||
ColumnLayout {
|
property bool isFocusable: true
|
||||||
id: content
|
|
||||||
|
|
||||||
anchors.top: parent.top
|
Keys.onTabPressed: {
|
||||||
anchors.left: parent.left
|
FocusController.nextKeyTabItem()
|
||||||
anchors.right: parent.right
|
}
|
||||||
anchors.rightMargin: 16
|
|
||||||
anchors.leftMargin: 16
|
|
||||||
|
|
||||||
spacing: 16
|
Keys.onBacktabPressed: {
|
||||||
|
FocusController.previousKeyTabItem()
|
||||||
|
}
|
||||||
|
|
||||||
|
Keys.onUpPressed: {
|
||||||
|
FocusController.nextKeyUpItem()
|
||||||
|
}
|
||||||
|
|
||||||
|
Keys.onDownPressed: {
|
||||||
|
FocusController.nextKeyDownItem()
|
||||||
|
}
|
||||||
|
|
||||||
|
Keys.onLeftPressed: {
|
||||||
|
FocusController.nextKeyLeftItem()
|
||||||
|
}
|
||||||
|
|
||||||
|
Keys.onRightPressed: {
|
||||||
|
FocusController.nextKeyRightItem()
|
||||||
|
}
|
||||||
|
|
||||||
|
ScrollBar.vertical: ScrollBarType {}
|
||||||
|
|
||||||
|
header: ColumnLayout {
|
||||||
|
width: listView.width
|
||||||
|
|
||||||
HeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
Layout.leftMargin: 16
|
||||||
|
Layout.rightMargin: 16
|
||||||
|
Layout.bottomMargin: 16
|
||||||
|
|
||||||
headerText: qsTr("Configure your server")
|
headerText: qsTr("Configure your server")
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
model: inputFields
|
||||||
|
spacing: 16
|
||||||
|
clip: true
|
||||||
|
reuseItems: true
|
||||||
|
|
||||||
|
delegate: ColumnLayout {
|
||||||
|
property alias textField: _textField.textField
|
||||||
|
|
||||||
|
width: listView.width
|
||||||
|
|
||||||
TextFieldWithHeaderType {
|
TextFieldWithHeaderType {
|
||||||
id: hostname
|
id: _textField
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
headerText: qsTr("Server IP address [:port]")
|
Layout.leftMargin: 16
|
||||||
textFieldPlaceholderText: qsTr("255.255.255.255:22")
|
Layout.rightMargin: 16
|
||||||
|
|
||||||
parentFlickable: fl
|
property bool hidePassword: hideText
|
||||||
|
|
||||||
textField.onFocusChanged: {
|
headerText: title
|
||||||
textField.text = textField.text.replace(/^\s+|\s+$/g, '')
|
textField.echoMode: hideText ? TextInput.Password : TextInput.Normal
|
||||||
}
|
buttonImageSource: imageSource
|
||||||
}
|
textFieldPlaceholderText: placeholderText
|
||||||
|
textField.text: textFieldText
|
||||||
|
|
||||||
TextFieldWithHeaderType {
|
rightButtonClickedOnEnter: true
|
||||||
id: username
|
|
||||||
|
|
||||||
Layout.fillWidth: true
|
clickedFunc: function () {
|
||||||
headerText: qsTr("SSH Username")
|
clickedHandler
|
||||||
textFieldPlaceholderText: "root"
|
|
||||||
|
|
||||||
parentFlickable: fl
|
|
||||||
|
|
||||||
textField.onFocusChanged: {
|
|
||||||
textField.text = textField.text.replace(/^\s+|\s+$/g, '')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
TextFieldWithHeaderType {
|
|
||||||
id: secretData
|
|
||||||
|
|
||||||
property bool hidePassword: true
|
|
||||||
|
|
||||||
Layout.fillWidth: true
|
|
||||||
headerText: qsTr("Password or SSH private key")
|
|
||||||
textField.echoMode: hidePassword ? TextInput.Password : TextInput.Normal
|
|
||||||
buttonImageSource: textFieldText !== "" ? (hidePassword ? "qrc:/images/controls/eye.svg" : "qrc:/images/controls/eye-off.svg")
|
|
||||||
: ""
|
|
||||||
|
|
||||||
parentFlickable: fl
|
|
||||||
|
|
||||||
clickedFunc: function() {
|
|
||||||
hidePassword = !hidePassword
|
|
||||||
}
|
}
|
||||||
|
|
||||||
textField.onFocusChanged: {
|
textField.onFocusChanged: {
|
||||||
textField.text = textField.text.replace(/^\s+|\s+$/g, '')
|
var _currentIndex = listView.currentIndex
|
||||||
|
var _currentItem = listView.itemAtIndex(_currentIndex).children[0]
|
||||||
|
listView.model[_currentIndex].textFieldText = _currentItem.textFieldText.replace(/^\s+|\s+$/g, '')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
footer: ColumnLayout {
|
||||||
|
width: listView.width
|
||||||
|
|
||||||
BasicButtonType {
|
BasicButtonType {
|
||||||
id: continueButton
|
id: continueButton
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.topMargin: 24
|
Layout.topMargin: 32
|
||||||
|
Layout.leftMargin: 16
|
||||||
|
Layout.rightMargin: 16
|
||||||
|
|
||||||
text: qsTr("Continue")
|
text: qsTr("Continue")
|
||||||
|
|
||||||
parentFlickable: fl
|
|
||||||
|
|
||||||
clickedFunc: function() {
|
clickedFunc: function() {
|
||||||
forceActiveFocus()
|
if (!root.isCredentialsFilled()) {
|
||||||
if (!isCredentialsFilled()) {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
InstallController.setShouldCreateServer(true)
|
InstallController.setShouldCreateServer(true)
|
||||||
InstallController.setProcessedServerCredentials(hostname.textField.text, username.textField.text, secretData.textField.text)
|
var _hostname = listView.itemAtIndex(vars.hostnameIndex).children[0].textFieldText
|
||||||
|
var _username = listView.itemAtIndex(vars.usernameIndex).children[0].textFieldText
|
||||||
|
var _secretData = listView.itemAtIndex(vars.secretDataIndex).children[0].textFieldText
|
||||||
|
|
||||||
|
InstallController.setProcessedServerCredentials(_hostname, _username, _secretData)
|
||||||
|
|
||||||
PageController.showBusyIndicator(true)
|
PageController.showBusyIndicator(true)
|
||||||
var isConnectionOpened = InstallController.checkSshConnection()
|
var isConnectionOpened = InstallController.checkSshConnection()
|
||||||
|
|
@ -127,7 +153,10 @@ PageType {
|
||||||
|
|
||||||
LabelTextType {
|
LabelTextType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.topMargin: 12
|
Layout.topMargin: 24
|
||||||
|
Layout.leftMargin: 16
|
||||||
|
Layout.rightMargin: 16
|
||||||
|
Layout.bottomMargin: 16
|
||||||
|
|
||||||
text: qsTr("All data you enter will remain strictly confidential and will not be shared or disclosed to the Amnezia or any third parties")
|
text: qsTr("All data you enter will remain strictly confidential and will not be shared or disclosed to the Amnezia or any third parties")
|
||||||
}
|
}
|
||||||
|
|
@ -136,6 +165,8 @@ PageType {
|
||||||
id: siteLink
|
id: siteLink
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
Layout.leftMargin: 16
|
||||||
|
Layout.rightMargin: 16
|
||||||
Layout.bottomMargin: 16
|
Layout.bottomMargin: 16
|
||||||
|
|
||||||
headerText: qsTr("How to run your VPN server")
|
headerText: qsTr("How to run your VPN server")
|
||||||
|
|
@ -144,8 +175,6 @@ PageType {
|
||||||
rightImageSource: "qrc:/images/controls/chevron-right.svg"
|
rightImageSource: "qrc:/images/controls/chevron-right.svg"
|
||||||
leftImageSource: "qrc:/images/controls/help-circle.svg"
|
leftImageSource: "qrc:/images/controls/help-circle.svg"
|
||||||
|
|
||||||
parentFlickable: fl
|
|
||||||
|
|
||||||
onClicked: {
|
onClicked: {
|
||||||
Qt.openUrlExternally(LanguageModel.getCurrentSiteUrl() + "/starter-guide")
|
Qt.openUrlExternally(LanguageModel.getCurrentSiteUrl() + "/starter-guide")
|
||||||
}
|
}
|
||||||
|
|
@ -156,21 +185,78 @@ PageType {
|
||||||
function isCredentialsFilled() {
|
function isCredentialsFilled() {
|
||||||
var hasEmptyField = false
|
var hasEmptyField = false
|
||||||
|
|
||||||
if (hostname.textFieldText === "") {
|
var _hostname = listView.itemAtIndex(vars.hostnameIndex).children[0]
|
||||||
hostname.errorText = qsTr("Ip address cannot be empty")
|
if (_hostname.textFieldText === "") {
|
||||||
|
_hostname.errorText = qsTr("Ip address cannot be empty")
|
||||||
hasEmptyField = true
|
hasEmptyField = true
|
||||||
} else if (!hostname.textField.acceptableInput) {
|
} else if (!_hostname.textField.acceptableInput) {
|
||||||
hostname.errorText = qsTr("Enter the address in the format 255.255.255.255:88")
|
_hostname.errorText = qsTr("Enter the address in the format 255.255.255.255:88")
|
||||||
}
|
}
|
||||||
|
|
||||||
if (username.textFieldText === "") {
|
var _username = listView.itemAtIndex(vars.usernameIndex).children[0]
|
||||||
username.errorText = qsTr("Login cannot be empty")
|
if (_username.textFieldText === "") {
|
||||||
|
_username.errorText = qsTr("Login cannot be empty")
|
||||||
hasEmptyField = true
|
hasEmptyField = true
|
||||||
}
|
}
|
||||||
if (secretData.textFieldText === "") {
|
|
||||||
secretData.errorText = qsTr("Password/private key cannot be empty")
|
var _secretData = listView.itemAtIndex(vars.secretDataIndex).children[0]
|
||||||
|
if (_secretData.textFieldText === "") {
|
||||||
|
_secretData.errorText = qsTr("Password/private key cannot be empty")
|
||||||
hasEmptyField = true
|
hasEmptyField = true
|
||||||
}
|
}
|
||||||
|
|
||||||
return !hasEmptyField
|
return !hasEmptyField
|
||||||
}
|
}
|
||||||
|
|
||||||
|
property list<QtObject> inputFields: [
|
||||||
|
hostname,
|
||||||
|
username,
|
||||||
|
secretData
|
||||||
|
]
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
id: hostname
|
||||||
|
|
||||||
|
property string title: qsTr("Server IP address [:port]")
|
||||||
|
readonly property string placeholderText: qsTr("255.255.255.255:22")
|
||||||
|
property string textFieldText: ""
|
||||||
|
property bool hideText: false
|
||||||
|
property string imageSource: ""
|
||||||
|
readonly property var clickedHandler: function() {
|
||||||
|
console.debug(">>> Server IP address text field was clicked!!!")
|
||||||
|
clicked()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
id: username
|
||||||
|
|
||||||
|
property string title: qsTr("SSH Username")
|
||||||
|
readonly property string placeholderText: "root"
|
||||||
|
property string textFieldText: ""
|
||||||
|
property bool hideText: false
|
||||||
|
property string imageSource: ""
|
||||||
|
readonly property var clickedHandler: undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
id: secretData
|
||||||
|
|
||||||
|
property string title: qsTr("Password or SSH private key")
|
||||||
|
readonly property string placeholderText: ""
|
||||||
|
property string textFieldText: ""
|
||||||
|
property bool hideText: true
|
||||||
|
property string imageSource: textFieldText !== "" ? (hideText ? "qrc:/images/controls/eye.svg" : "qrc:/images/controls/eye-off.svg") : ""
|
||||||
|
readonly property var clickedHandler: function() {
|
||||||
|
hideText = !hideText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
id: vars
|
||||||
|
|
||||||
|
readonly property int hostnameIndex: 0
|
||||||
|
readonly property int usernameIndex: 1
|
||||||
|
readonly property int secretDataIndex: 2
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue