GODT-1316: Set default TextArea and TextField behavior

This commit is contained in:
Alexander Bilyak
2021-11-03 13:47:44 +00:00
committed by Jakub
parent 07b7fa7364
commit b5b477a3ce
7 changed files with 490 additions and 356 deletions

View File

@ -37,7 +37,6 @@ SettingsView {
id: description id: description
property int _minLength: 150 property int _minLength: 150
property int _maxLength: 800 property int _maxLength: 800
property bool _inputOK: description.text.length>=description._minLength && description.text.length<=description._maxLength
label: qsTr("Description") label: qsTr("Description")
colorScheme: root.colorScheme colorScheme: root.colorScheme
@ -46,71 +45,59 @@ SettingsView {
hint: description.text.length + "/" + _maxLength hint: description.text.length + "/" + _maxLength
placeholderText: qsTr("Tell us what went wrong or isn't working (min. %1 characters).").arg(_minLength) placeholderText: qsTr("Tell us what went wrong or isn't working (min. %1 characters).").arg(_minLength)
onEditingFinished: { validator: function(text) {
if (!description._inputOK) { if (description.text.length < description._minLength) {
description.error = true return qsTr("Enter a problem description (min. %1 characters).").arg(_minLength)
if (description.text.length <= description._minLength) {
description.assistiveText = qsTr("Enter a problem description (min. %1 characters).").arg(_minLength)
} else {
description.assistiveText = qsTr("Enter a problem description (max. %1 characters).").arg(_maxLength)
} }
} else {
description.error = false if (description.text.length > description._maxLength) {
description.assistiveText = "" return qsTr("Enter a problem description (max. %1 characters).").arg(_maxLength)
} }
return
} }
onTextChanged: { onTextChanged: {
description.error = false // Rise max length error imidiatly while typing
description.assistiveText = "" if (description.text.length > description._maxLength) {
validate()
} }
} }
KeyNavigation.priority: KeyNavigation.BeforeItem
KeyNavigation.tab: address
}
TextField { TextField {
id: address id: address
property bool _inputOK: root.isValidEmail(address.text)
label: qsTr("Your contact email") label: qsTr("Your contact email")
colorScheme: root.colorScheme colorScheme: root.colorScheme
Layout.fillWidth: true Layout.fillWidth: true
placeholderText: qsTr("e.g. jane.doe@protonmail.com") placeholderText: qsTr("e.g. jane.doe@protonmail.com")
onEditingFinished: { validator: function(str) {
if (!address._inputOK) { if (!isValidEmail(str)) {
address.error = true return qsTr("Enter valid email address")
address.assistiveText = qsTr("Enter valid email address")
} else {
address.assistiveText = ""
address.error = false
} }
} return
onTextChanged: {
address.error = false
address.assistiveText = ""
} }
} }
TextField { TextField {
id: emailClient id: emailClient
property bool _inputOK: emailClient.text.length > 0
label: qsTr("Your email client (including version)") label: qsTr("Your email client (including version)")
colorScheme: root.colorScheme colorScheme: root.colorScheme
Layout.fillWidth: true Layout.fillWidth: true
placeholderText: qsTr("e.g. Apple Mail 14.0") placeholderText: qsTr("e.g. Apple Mail 14.0")
onEditingFinished: { validator: function(str) {
if (!emailClient._inputOK) { if (str.length === 0) {
emailClient.assistiveText = qsTr("Enter an email client name and version") return qsTr("Enter an email client name and version")
emailClient.error = true
} else {
emailClient.assistiveText = ""
emailClient.error = false
} }
} return
onTextChanged: {
emailClient.error = false
emailClient.assistiveText = ""
} }
} }
@ -148,25 +135,26 @@ SettingsView {
id: sendButton id: sendButton
text: qsTr("Send") text: qsTr("Send")
colorScheme: root.colorScheme colorScheme: root.colorScheme
onClicked: root.submit()
enabled: description._inputOK && address._inputOK && emailClient._inputOK onClicked: {
description.validate()
address.validate()
emailClient.validate()
if (description.error || address.error || emailClient.error) {
return
}
submit()
}
Connections {target: root.backend; onReportBugFinished: sendButton.loading = false } Connections {target: root.backend; onReportBugFinished: sendButton.loading = false }
} }
function setDefaultValue() { function setDefaultValue() {
description.text = "" description.text = ""
description.error = false
description.assistiveText = ""
address.text = root.selectedAddress address.text = root.selectedAddress
address.error = false
address.assistiveText = ""
emailClient.text = root.backend.currentEmailClient emailClient.text = root.backend.currentEmailClient
emailClient.error = false
emailClient.assistiveText = ""
includeLogs.checked = true includeLogs.checked = true
} }
@ -185,8 +173,6 @@ SettingsView {
) )
} }
Component.onCompleted: root.setDefaultValue()
onVisibleChanged: { onVisibleChanged: {
root.setDefaultValue() root.setDefaultValue()
} }

View File

@ -25,10 +25,9 @@ import Proton 4.0
SettingsView { SettingsView {
id: root id: root
property bool _valuesOK: !imapField.error && !smtpField.error
property bool _valuesChanged: ( property bool _valuesChanged: (
imapField.text*1 != root.backend.portIMAP || imapField.text*1 !== root.backend.portIMAP ||
smtpField.text*1 != root.backend.portSMTP smtpField.text*1 !== root.backend.portSMTP
) )
Label { Label {
@ -55,14 +54,14 @@ SettingsView {
colorScheme: root.colorScheme colorScheme: root.colorScheme
label: qsTr("IMAP port") label: qsTr("IMAP port")
Layout.preferredWidth: 160 Layout.preferredWidth: 160
onEditingFinished: root.validate(imapField) validator: root.validate
} }
TextField { TextField {
id: smtpField id: smtpField
colorScheme: root.colorScheme colorScheme: root.colorScheme
label: qsTr("SMTP port") label: qsTr("SMTP port")
Layout.preferredWidth: 160 Layout.preferredWidth: 160
onEditingFinished: root.validate(smtpField) validator: root.validate
} }
} }
@ -79,10 +78,34 @@ SettingsView {
id: submitButton id: submitButton
colorScheme: root.colorScheme colorScheme: root.colorScheme
text: qsTr("Save and restart") text: qsTr("Save and restart")
enabled: root._valuesOK && root._valuesChanged enabled: root._valuesChanged
onClicked: { onClicked: {
// removing error here because we may have set it manually (port occupied)
imapField.error = false
smtpField.error = false
// checking errors seperatly because we want to display "same port" error only once
imapField.validate()
if (imapField.error) {
return
}
smtpField.validate()
if (smtpField.error) {
return
}
submitButton.loading = true submitButton.loading = true
root.submit()
// check both ports before returning an error
var err = false
err |= !isPortFree(imapField)
err |= !isPortFree(smtpField)
if (err) {
submitButton.loading = false
return
}
root.backend.changePorts(imapField.text, smtpField.text)
} }
} }
@ -104,48 +127,32 @@ SettingsView {
root.setDefaultValues() root.setDefaultValues()
} }
function validate(field) { function validate(port) {
var num = field.text*1 var num = port*1
if (! (num > 1 && num < 65536) ) { if (! (num > 1 && num < 65536) ) {
field.error = true return qsTr("Invalid port number")
field.assistiveText = qsTr("Invalid port number")
return
} }
if (imapField.text == smtpField.text) { if (imapField.text == smtpField.text) {
field.error = true return qsTr("Port numbers must be different")
field.assistiveText = qsTr("Port numbers must be different") }
return return
} }
field.error = false
field.assistiveText = ""
}
function isPortFree(field) { function isPortFree(field) {
field.error = false
field.assistiveText = ""
var num = field.text*1 var num = field.text*1
if (num == root.backend.portIMAP) return true if (num === root.backend.portIMAP) return true
if (num == root.backend.portSMTP) return true if (num === root.backend.portSMTP) return true
if (!root.backend.isPortFree(num)) { if (!root.backend.isPortFree(num)) {
field.error = true field.error = true
field.assistiveText = qsTr("Port occupied") field.errorString = qsTr("Port occupied")
submitButton.loading = false
return false return false
} }
return true return true
} }
function submit(){
submitButton.loading = true
if (!isPortFree(imapField)) return
if (!isPortFree(smtpField)) return
root.backend.changePorts(imapField.text, smtpField.text)
}
function setDefaultValues(){ function setDefaultValues(){
imapField.text = backend.portIMAP imapField.text = backend.portIMAP
smtpField.text = backend.portSMTP smtpField.text = backend.portSMTP

View File

@ -20,10 +20,11 @@ import QtQuick 2.12
import QtQuick.Controls 2.12 import QtQuick.Controls 2.12
import QtQuick.Controls.impl 2.12 import QtQuick.Controls.impl 2.12
import QtQuick.Templates 2.12 as T import QtQuick.Templates 2.12 as T
import QtQuick.Layouts 1.12
import "." as Proton import "." as Proton
Item { FocusScope {
id: root id: root
property ColorScheme colorScheme property ColorScheme colorScheme
@ -84,61 +85,58 @@ Item {
property alias textFormat: control.textFormat property alias textFormat: control.textFormat
property alias textMargin: control.textMargin property alias textMargin: control.textMargin
property alias topPadding: control.topPadding property alias topPadding: control.topPadding
// We are using our own type of validators. It should be a function
// returning an error string in case of error and undefined if no error
property var validator
property alias verticalAlignment: control.verticalAlignment property alias verticalAlignment: control.verticalAlignment
property alias wrapMode: control.wrapMode property alias wrapMode: control.wrapMode
implicitWidth: background.width implicitWidth: children[0].implicitWidth
implicitHeight: control.implicitHeight + Math.max( implicitHeight: children[0].implicitHeight
label.implicitHeight + label.anchors.topMargin + label.anchors.bottomMargin,
hint.implicitHeight + hint.anchors.topMargin + hint.anchors.bottomMargin
) + assistiveText.implicitHeight
property alias label: label.text property alias label: label.text
property alias hint: hint.text property alias hint: hint.text
property alias assistiveText: assistiveText.text property string assistiveText
property string errorString
property bool error: false property bool error: false
signal editingFinished() signal editingFinished()
// Backgroud is moved away from within control as it will be clipped with scrollview function append(text) { return control.append(text) }
Rectangle { function clear() { return control.clear() }
id: background function copy() { return control.copy() }
function cut() { return control.cut() }
function deselect() { return control.deselect() }
function getFormattedText(start, end) { return control.getFormattedText(start, end) }
function getText(start, end) { return control.getText(start, end) }
function insert(position, text) { return control.insert(position, text) }
function isRightToLeft(start, end) { return control.isRightToLeft(start, end) }
function linkAt(x, y) { return control.linkAt(x, y) }
function moveCursorSelection(position, mode) { return control.moveCursorSelection(position, mode) }
function paste() { return control.paste() }
function positionAt(x, y) { return control.positionAt(x, y) }
function positionToRectangle(position) { return control.positionToRectangle(position) }
function redo() { return control.redo() }
function remove(start, end) { return control.remove(start, end) }
function select(start, end) { return control.select(start, end) }
function selectAll() { return control.selectAll() }
function selectWord() { return control.selectWord() }
function undo() { return control.undo() }
anchors.fill: controlView ColumnLayout {
anchors.fill: parent
spacing: 0
radius: 4 RowLayout {
visible: true Layout.fillWidth: true
color: root.colorScheme.background_norm spacing: 0
border.color: {
if (!control.enabled) {
return root.colorScheme.field_disabled
}
if (control.activeFocus) {
return root.colorScheme.interaction_norm
}
if (root.error) {
return root.colorScheme.signal_danger
}
if (control.hovered) {
return root.colorScheme.field_hover
}
return root.colorScheme.field_norm
}
border.width: 1
}
Proton.Label { Proton.Label {
colorScheme: root.colorScheme colorScheme: root.colorScheme
id: label id: label
anchors.top: root.top Layout.fillWidth: true
anchors.left: root.left
anchors.bottomMargin: 4
color: root.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled color: root.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled
@ -149,57 +147,19 @@ Item {
colorScheme: root.colorScheme colorScheme: root.colorScheme
id: hint id: hint
anchors.right: root.right Layout.fillWidth: true
anchors.bottom: controlView.top
anchors.bottomMargin: 5
color: root.enabled ? root.colorScheme.text_weak : root.colorScheme.text_disabled color: root.enabled ? root.colorScheme.text_weak : root.colorScheme.text_disabled
horizontalAlignment: Text.AlignRight
type: Proton.Label.LabelType.Caption type: Proton.Label.LabelType.Caption
} }
ColorImage {
id: errorIcon
visible: root.error
anchors.left: parent.left
anchors.top: assistiveText.top
anchors.bottom: assistiveText.bottom
source: "../icons/ic-exclamation-circle-filled.svg"
sourceSize.height: height
color: root.colorScheme.signal_danger
}
Proton.Label {
colorScheme: root.colorScheme
id: assistiveText
anchors.left: root.error ? errorIcon.right : parent.left
anchors.leftMargin: root.error ? 5 : 0
anchors.bottom: root.bottom
anchors.topMargin: 4
color: {
if (!root.enabled) {
return root.colorScheme.text_disabled
}
if (root.error) {
return root.colorScheme.signal_danger
}
return root.colorScheme.text_weak
}
type: root.error ? Proton.Label.LabelType.Caption_semibold : Proton.Label.LabelType.Caption
} }
ScrollView { ScrollView {
id: controlView id: controlView
anchors.top: label.bottom Layout.fillHeight: true
anchors.left: root.left Layout.fillWidth: true
anchors.right: root.right
anchors.bottom: assistiveText.top
clip: true clip: true
@ -217,17 +177,38 @@ Item {
placeholder.implicitHeight + topPadding + bottomPadding placeholder.implicitHeight + topPadding + bottomPadding
) )
padding: 8 topPadding: 8
bottomPadding: 8
leftPadding: 12 leftPadding: 12
rightPadding: 12
font.family: Style.font_family
font.weight: Style.fontWeight_400
font.pixelSize: Style.body_font_size
font.letterSpacing: Style.body_letter_spacing
color: control.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled color: control.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled
placeholderTextColor: control.enabled ? root.colorScheme.text_hint : root.colorScheme.text_disabled placeholderTextColor: control.enabled ? root.colorScheme.text_hint : root.colorScheme.text_disabled
selectionColor: control.palette.highlight selectionColor: control.palette.highlight
selectedTextColor: control.palette.highlightedText selectedTextColor: control.palette.highlightedText
onEditingFinished: root.editingFinished() onEditingFinished: root.editingFinished()
wrapMode: TextInput.Wrap
// enforcing default focus here within component
focus: root.focus
KeyNavigation.priority: root.KeyNavigation.priority
KeyNavigation.backtab: root.KeyNavigation.backtab
KeyNavigation.tab: root.KeyNavigation.tab
KeyNavigation.up: root.KeyNavigation.up
KeyNavigation.down: root.KeyNavigation.down
KeyNavigation.left: root.KeyNavigation.left
KeyNavigation.right: root.KeyNavigation.right
selectByMouse: true
cursorDelegate: Rectangle { cursorDelegate: Rectangle {
id: cursor id: cursor
width: 1 width: 1
@ -269,6 +250,104 @@ Item {
elide: Text.ElideRight elide: Text.ElideRight
renderType: control.renderType renderType: control.renderType
} }
background: Rectangle {
anchors.fill: parent
radius: 4
visible: true
color: root.colorScheme.background_norm
border.color: {
if (!control.enabled) {
return root.colorScheme.field_disabled
} }
if (control.activeFocus) {
return root.colorScheme.interaction_norm
}
if (root.error) {
return root.colorScheme.signal_danger
}
if (control.hovered) {
return root.colorScheme.field_hover
}
return root.colorScheme.field_norm
}
border.width: 1
}
}
}
RowLayout {
Layout.fillWidth: true
spacing: 0
ColorImage {
id: errorIcon
Layout.rightMargin: 4
visible: root.error && (assistiveText.text.length > 0)
source: "../icons/ic-exclamation-circle-filled.svg"
color: root.colorScheme.signal_danger
height: assistiveText.height
sourceSize.height: assistiveText.height
}
Proton.Label {
colorScheme: root.colorScheme
id: assistiveText
Layout.fillWidth: true
text: root.error ? root.errorString : root.assistiveText
color: {
if (!root.enabled) {
return root.colorScheme.text_disabled
}
if (root.error) {
return root.colorScheme.signal_danger
}
return root.colorScheme.text_weak
}
type: root.error ? Proton.Label.LabelType.Caption_semibold : Proton.Label.LabelType.Caption
}
}
}
property bool validateOnEditingFinished: true
onEditingFinished: {
if (!validateOnEditingFinished) {
return
}
validate()
}
function validate() {
if (validator === undefined) {
return
}
var error = validator(text)
if (error) {
root.error = true
root.errorString = error
} else {
root.error = false
root.errorString = ""
}
}
onTextChanged: {
root.error = false
root.errorString = ""
} }
} }

View File

@ -24,7 +24,7 @@ import QtQuick.Layouts 1.12
import "." as Proton import "." as Proton
Item { FocusScope {
id: root id: root
property ColorScheme colorScheme property ColorScheme colorScheme
@ -82,7 +82,9 @@ Item {
property alias selectionEnd: control.selectionEnd property alias selectionEnd: control.selectionEnd
property alias selectionStart: control.selectionStart property alias selectionStart: control.selectionStart
property alias text: control.text property alias text: control.text
property alias validator: control.validator // We are using our own type of validators. It should be a function
// returning an error string in case of error and undefined if no error
property var validator
property alias verticalAlignment: control.verticalAlignment property alias verticalAlignment: control.verticalAlignment
property alias wrapMode: control.wrapMode property alias wrapMode: control.wrapMode
@ -91,7 +93,8 @@ Item {
property alias label: label.text property alias label: label.text
property alias hint: hint.text property alias hint: hint.text
property alias assistiveText: assistiveText.text property string assistiveText
property string errorString
property int echoMode: TextInput.Normal property int echoMode: TextInput.Normal
@ -200,18 +203,38 @@ Item {
contentHeight + topPadding + bottomPadding, contentHeight + topPadding + bottomPadding,
placeholder.implicitHeight + topPadding + bottomPadding) placeholder.implicitHeight + topPadding + bottomPadding)
padding: 8 topPadding: 8
bottomPadding: 8
leftPadding: 12 leftPadding: 12
rightPadding: 12
font.family: Style.font_family
font.weight: Style.fontWeight_400
font.pixelSize: Style.body_font_size
font.letterSpacing: Style.body_letter_spacing
color: control.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled color: control.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled
placeholderTextColor: control.enabled ? root.colorScheme.text_hint : root.colorScheme.text_disabled
selectionColor: control.palette.highlight selectionColor: control.palette.highlight
selectedTextColor: control.palette.highlightedText selectedTextColor: control.palette.highlightedText
placeholderTextColor: control.enabled ? root.colorScheme.text_hint : root.colorScheme.text_disabled
verticalAlignment: TextInput.AlignVCenter verticalAlignment: TextInput.AlignVCenter
echoMode: eyeButton.checked ? TextInput.Normal : root.echoMode echoMode: eyeButton.checked ? TextInput.Normal : root.echoMode
// enforcing default focus here within component
focus: true
KeyNavigation.priority: root.KeyNavigation.priority
KeyNavigation.backtab: root.KeyNavigation.backtab
KeyNavigation.tab: root.KeyNavigation.tab
KeyNavigation.up: root.KeyNavigation.up
KeyNavigation.down: root.KeyNavigation.down
KeyNavigation.left: root.KeyNavigation.left
KeyNavigation.right: root.KeyNavigation.right
selectByMouse: true
cursorDelegate: Rectangle { cursorDelegate: Rectangle {
id: cursor id: cursor
width: 1 width: 1
@ -292,6 +315,8 @@ Item {
ColorImage { ColorImage {
id: errorIcon id: errorIcon
Layout.rightMargin: 4
visible: root.error && (assistiveText.text.length > 0) visible: root.error && (assistiveText.text.length > 0)
source: "../icons/ic-exclamation-circle-filled.svg" source: "../icons/ic-exclamation-circle-filled.svg"
color: root.colorScheme.signal_danger color: root.colorScheme.signal_danger
@ -305,7 +330,8 @@ Item {
Layout.fillHeight: true Layout.fillHeight: true
Layout.fillWidth: true Layout.fillWidth: true
Layout.leftMargin: 4
text: root.error ? root.errorString : root.assistiveText
color: { color: {
if (!root.enabled) { if (!root.enabled) {
@ -323,4 +349,33 @@ Item {
} }
} }
} }
property bool validateOnEditingFinished: true
onEditingFinished: {
if (!validateOnEditingFinished) {
return
}
validate()
}
function validate() {
if (validator === undefined) {
return
}
var error = validator(text)
if (error) {
root.error = true
root.errorString = error
} else {
root.error = false
root.errorString = ""
}
}
onTextChanged: {
root.error = false
root.errorString = ""
}
} }

View File

@ -99,7 +99,7 @@ Item {
twoFactorPasswordTextField.enabled = true twoFactorPasswordTextField.enabled = true
twoFactorPasswordTextField.error = true twoFactorPasswordTextField.error = true
twoFactorPasswordTextField.assistiveText = qsTr("Your code is incorrect") twoFactorPasswordTextField.errorString = qsTr("Your code is incorrect")
} }
onLogin2FAErrorAbort: { onLogin2FAErrorAbort: {
console.assert(stackLayout.currentIndex == 1, "Unexpected login2FAErrorAbort") console.assert(stackLayout.currentIndex == 1, "Unexpected login2FAErrorAbort")
@ -118,7 +118,7 @@ Item {
secondPasswordTextField.enabled = true secondPasswordTextField.enabled = true
secondPasswordTextField.error = true secondPasswordTextField.error = true
secondPasswordTextField.assistiveText = qsTr("Your mailbox password is incorrect") secondPasswordTextField.errorString = qsTr("Your mailbox password is incorrect")
} }
onLogin2PasswordErrorAbort: { onLogin2PasswordErrorAbort: {
console.assert(stackLayout.currentIndex == 2, "Unexpected login2PasswordErrorAbort") console.assert(stackLayout.currentIndex == 2, "Unexpected login2PasswordErrorAbort")
@ -142,11 +142,11 @@ Item {
usernameTextField.enabled = true usernameTextField.enabled = true
usernameTextField.error = false usernameTextField.error = false
usernameTextField.assistiveText = "" usernameTextField.errorString = ""
passwordTextField.enabled = true passwordTextField.enabled = true
passwordTextField.error = false passwordTextField.error = false
passwordTextField.assistiveText = "" passwordTextField.errorString = ""
passwordTextField.text = "" passwordTextField.text = ""
} }
@ -202,18 +202,22 @@ Item {
Layout.fillWidth: true Layout.fillWidth: true
Layout.topMargin: 24 Layout.topMargin: 24
onTextEdited: { // TODO: repeating? onTextChanged: {
// remove "invalid username / password error"
if (error || errorLabel.text.length > 0) { if (error || errorLabel.text.length > 0) {
errorLabel.text = "" errorLabel.text = ""
usernameTextField.error = false usernameTextField.error = false
usernameTextField.assistiveText = ""
passwordTextField.error = false passwordTextField.error = false
passwordTextField.assistiveText = ""
} }
} }
validator: function(str) {
if (str.length === 0) {
return qsTr("Enter username or email")
}
return
}
onAccepted: passwordTextField.forceActiveFocus() onAccepted: passwordTextField.forceActiveFocus()
} }
@ -226,18 +230,22 @@ Item {
Layout.topMargin: 8 Layout.topMargin: 8
echoMode: TextInput.Password echoMode: TextInput.Password
onTextEdited: { onTextChanged: {
// remove "invalid username / password error"
if (error || errorLabel.text.length > 0) { if (error || errorLabel.text.length > 0) {
errorLabel.text = "" errorLabel.text = ""
usernameTextField.error = false usernameTextField.error = false
usernameTextField.assistiveText = ""
passwordTextField.error = false passwordTextField.error = false
passwordTextField.assistiveText = ""
} }
} }
validator: function(str) {
if (str.length === 0) {
return qsTr("Enter password")
}
return
}
onAccepted: signInButton.checkAndSignIn() onAccepted: signInButton.checkAndSignIn()
} }
@ -253,27 +261,10 @@ Item {
onClicked: checkAndSignIn() onClicked: checkAndSignIn()
function checkAndSignIn() { function checkAndSignIn() {
var err = false usernameTextField.validate()
passwordTextField.validate()
if (usernameTextField.text.length == 0) { if (usernameTextField.error || passwordTextField.error) {
usernameTextField.error = true
usernameTextField.assistiveText = qsTr("Enter username or email")
err = true
} else {
usernameTextField.error = false
usernameTextField.assistiveText = qsTr("")
}
if (passwordTextField.text.length == 0) {
passwordTextField.error = true
passwordTextField.assistiveText = qsTr("Enter password")
err = true
} else {
passwordTextField.error = false
passwordTextField.assistiveText = qsTr("")
}
if (err) {
return return
} }
@ -310,8 +301,8 @@ Item {
twoFactorPasswordTextField.enabled = true twoFactorPasswordTextField.enabled = true
twoFactorPasswordTextField.error = false twoFactorPasswordTextField.error = false
twoFactorPasswordTextField.assistiveText = "" twoFactorPasswordTextField.errorString = ""
twoFactorPasswordTextField.text="" twoFactorPasswordTextField.text = ""
} }
spacing: 0 spacing: 0
@ -343,11 +334,11 @@ Item {
Layout.fillWidth: true Layout.fillWidth: true
Layout.topMargin: 32 Layout.topMargin: 32
onTextEdited: { validator: function(str) {
if (error) { if (str.length === 0) {
twoFactorPasswordTextField.error = false return qsTr("Enter username or email")
twoFactorPasswordTextField.assistiveText = ""
} }
return
} }
} }
@ -360,18 +351,9 @@ Item {
Layout.topMargin: 24 Layout.topMargin: 24
onClicked: { onClicked: {
var err = false twoFactorPasswordTextField.validate()
if (twoFactorPasswordTextField.text.length == 0) { if (twoFactorPasswordTextField.error) {
twoFactorPasswordTextField.error = true
twoFactorPasswordTextField.assistiveText = qsTr("Enter username or email")
err = true
} else {
twoFactorPasswordTextField.error = false
twoFactorPasswordTextField.assistiveText = qsTr("")
}
if (err) {
return return
} }
@ -393,7 +375,7 @@ Item {
secondPasswordTextField.enabled = true secondPasswordTextField.enabled = true
secondPasswordTextField.error = false secondPasswordTextField.error = false
secondPasswordTextField.assistiveText = "" secondPasswordTextField.errorString = ""
secondPasswordTextField.text = "" secondPasswordTextField.text = ""
} }
@ -416,11 +398,11 @@ Item {
Layout.topMargin: 8 + implicitHeight + 24 + subTitle.implicitHeight Layout.topMargin: 8 + implicitHeight + 24 + subTitle.implicitHeight
echoMode: TextInput.Password echoMode: TextInput.Password
onTextEdited: { validator: function(str) {
if (error) { if (str.length === 0) {
secondPasswordTextField.error = false return qsTr("Enter username or email")
secondPasswordTextField.assistiveText = ""
} }
return
} }
} }
@ -433,18 +415,9 @@ Item {
Layout.topMargin: 24 Layout.topMargin: 24
onClicked: { onClicked: {
var err = false secondPasswordTextField.validate()
if (secondPasswordTextField.text.length == 0) { if (secondPasswordTextField.error) {
secondPasswordTextField.error = true
secondPasswordTextField.assistiveText = qsTr("Enter username or email")
err = true
} else {
secondPasswordTextField.error = false
secondPasswordTextField.assistiveText = qsTr("")
}
if (err) {
return return
} }

View File

@ -22,14 +22,11 @@ import QtQuick.Controls 2.12
import "../Proton" import "../Proton"
RowLayout { ColumnLayout {
id: root id: root
property ColorScheme colorScheme property ColorScheme colorScheme
ColumnLayout { spacing: 10
Layout.fillWidth: true
spacing: parent.spacing
TextArea { TextArea {
colorScheme: root.colorScheme colorScheme: root.colorScheme
@ -40,6 +37,8 @@ RowLayout {
label: "Label" label: "Label"
hint: "Hint" hint: "Hint"
assistiveText: "Assistive text" assistiveText: "Assistive text"
wrapMode: TextInput.Wrap
} }
TextArea { TextArea {
@ -52,6 +51,8 @@ RowLayout {
label: "Label" label: "Label"
hint: "Hint" hint: "Hint"
assistiveText: "Assistive text" assistiveText: "Assistive text"
wrapMode: TextInput.Wrap
} }
@ -66,7 +67,9 @@ RowLayout {
placeholderText: "Placeholder" placeholderText: "Placeholder"
label: "Label" label: "Label"
hint: "Hint" hint: "Hint"
assistiveText: "Error message" errorString: "Error message"
wrapMode: TextInput.Wrap
} }
@ -82,6 +85,29 @@ RowLayout {
label: "Label" label: "Label"
hint: "Hint" hint: "Hint"
assistiveText: "Assistive text" assistiveText: "Assistive text"
wrapMode: TextInput.Wrap
}
TextArea {
colorScheme: root.colorScheme
Layout.fillWidth: true
Layout.preferredHeight: 100
placeholderText: "Type 42 here"
label: "42 Validator"
hint: "Accepts only \"42\""
assistiveText: "Type sometihng here, preferably 42"
wrapMode: TextInput.Wrap
validator: function(str) {
if (str === "42") {
return
}
return "Not 42"
} }
} }
} }

View File

@ -63,7 +63,7 @@ RowLayout {
placeholderText: "Placeholder" placeholderText: "Placeholder"
label: "Label" label: "Label"
hint: "Hint" hint: "Hint"
assistiveText: "Error message" errorString: "Error message"
} }
@ -119,7 +119,7 @@ RowLayout {
placeholderText: "Password" placeholderText: "Password"
label: "Label" label: "Label"
hint: "Hint" hint: "Hint"
assistiveText: "Error message" errorString: "Error message"
} }
TextField { TextField {
@ -146,10 +146,18 @@ RowLayout {
colorScheme: root.colorScheme colorScheme: root.colorScheme
Layout.fillWidth: true Layout.fillWidth: true
placeholderText: "Placeholder" placeholderText: "Type 42 here"
label: "Label" label: "42 Validator"
hint: "Hint" hint: "Accepts only \"42\""
assistiveText: "Assistive text" assistiveText: "Type sometihng here, preferably 42"
validator: function(str) {
if (str === "42") {
return
}
return "Not 42"
}
} }
TextField { TextField {