-
Liam Coursodon authored
Change-Id: If8f92dfdba20005fe8430bfbe880167061301b95
Liam Coursodon authoredChange-Id: If8f92dfdba20005fe8430bfbe880167061301b95
Code owners
Assign users and groups as approvers for specific file changes. Learn more.
MaterialTextField.qml 6.97 KiB
/*
* Copyright (C) 2022-2023 Savoir-faire Linux Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
import QtQuick
import QtQuick.Controls
import Qt5Compat.GraphicalEffects
import net.jami.Constants 1.1
TextField {
id: root
// We need to remove focus when another widget takes activeFocus,
// except the context menu.
property bool isActive: activeFocus || contextMenu.active
property bool isSettings: false
property bool isSwarmDetail: false
property bool dontShowFocusState: !readOnly
onActiveFocusChanged: {
if (!activeFocus && !contextMenu.active) {
root.focus = false;
}
}
signal keyPressed
signal rejected
property bool inputIsValid: true
property string prefixIconSrc
property alias prefixIconColor: prefixIcon.color
property string suffixIconSrc
property alias suffixIconColor: suffixIcon.color
property string suffixBisIconSrc
property alias suffixBisIconColor: suffixBisIcon.color
property alias icon: container.data
property color accent: isActive || hovered ? prefixIconColor : JamiTheme.buttonTintedBlue
property color baseColor: JamiTheme.primaryForegroundColor
property color textColor: JamiTheme.textColor
color: textColor
placeholderTextColor: !isActive ? JamiTheme.transparentColor : root.color
property alias infoTipText: infoTip.text
property alias infoTipLineText: infoTipLine.text
//https://doc.qt.io/qt-6/qml-qtquick-textinput.html#maximumLength-prop
property var maxCharacters: undefined
maximumLength: maxCharacters ? maxCharacters : 32767
wrapMode: "NoWrap"
font.pixelSize: JamiTheme.materialLineEditPixelSize
font.kerning: true
selectByMouse: true
mouseSelectionMode: TextInput.SelectCharacters
leftPadding: readOnly || prefixIconSrc === '' || (isSwarmDetail && !root.isActive) ? 0 : 32
rightPadding: {
var total = 2;
if (!readOnly) {
if (suffixIconSrc !== "")
total = +30;
if (suffixBisIconSrc !== "")
total = +30;
}
return total;
}
topPadding: 2
Keys.onPressed: function (event) {
if (event.key === Qt.Key_Enter || event.key === Qt.Key_Return) {
if (inputIsValid && acceptableInput) {
root.accepted();
}
event.accepted = true;
} else if (event.key === Qt.Key_Escape) {
root.focus = false;
root.rejected();
} else {
root.keyPressed();
}
}
// Context menu.
LineEditContextMenu {
id: contextMenu
lineEditObj: root
selectOnly: readOnly
}
onReleased: function (event) {
if (event.button === Qt.RightButton)
contextMenu.openMenuAt(event);
}
// The centered placeholder that appears in the design specs.
Label {
id: overBaseLineLabel
font.pixelSize: root.font.pixelSize
anchors.baseline: root.baseline
anchors.horizontalCenter: !isSwarmDetail ? root.horizontalCenter : undefined
text: root.placeholderText
color: isSwarmDetail ? root.color : root.baseColor
visible: !root.isActive && !readOnly && root.text.toString() === ""
}
Rectangle {
id: baselineLine
width: parent.width
height: visible ? 1 : 0
anchors.top: root.baseline
anchors.topMargin: 10
color: isSwarmDetail ? textColor : root.accent
visible: {
if (!readOnly) {
if (isSwarmDetail && root.hovered || root.isActive) {
return true;
}
if (isSwarmDetail) {
return false;
}
return true;
}
return false;
}
}
component TextFieldIcon: ResponsiveImage {
property real size: 18
width: visible ? size : 0
height: size
opacity: root.isActive && !readOnly && source.toString() !== ''
visible: opacity
HoverHandler {
cursorShape: Qt.ArrowCursor
}
Behavior on opacity {
NumberAnimation {
duration: JamiTheme.longFadeDuration / 2
}
}
}
TextFieldIcon {
id: prefixIcon
anchors.left: parent.left
anchors.verticalCenter: root.verticalCenter
anchors.verticalCenterOffset: -root.bottomPadding / 2
color: prefixIconColor
source: prefixIconSrc
}
Label {
id: underBaseLineLabel
font.pixelSize: JamiTheme.materialLineEditSelectedPixelSize
anchors.top: baselineLine.bottom
anchors.topMargin: 2
text: root.placeholderText
color: root.textColor
// Show the alternate placeholder while the user types.
visible: root.isActive && !readOnly && root.text.toString() !== "" && !root.isSettings && !root.isSwarmDetail
}
Item {
id: container
width: suffixIcon.width
height: suffixIcon.height
anchors.right: suffixBisIcon.left
anchors.rightMargin: suffixBisIconSrc !== '' ? 5 : root.isActive ? 0 : 20
anchors.verticalCenter: root.verticalCenter
anchors.verticalCenterOffset: -root.bottomPadding / 2
visible: !readOnly
TextFieldIcon {
id: suffixIcon
size: 20
color: suffixIconColor
source: suffixIconSrc
MaterialToolTip {
id: infoTip
textColor: JamiTheme.blackColor
backGroundColor: JamiTheme.whiteColor
visible: parent.hovered && infoTipText.toString() !== ""
delay: Qt.styleHints.mousePressAndHoldInterval
}
}
}
TextFieldIcon {
id: suffixBisIcon
size: 20
anchors.right: parent.right
anchors.verticalCenter: root.verticalCenter
anchors.verticalCenterOffset: -root.bottomPadding / 2
color: suffixBisIconColor
source: suffixBisIconSrc
TapHandler {
cursorShape: Qt.ArrowCursor
onTapped: {
modalTextEditRoot.icoClicked();
}
}
}
MaterialToolTip {
id: infoTipLine
visible: parent.hovered && infoTipLineText.toString() !== "" && !readOnly
delay: Qt.styleHints.mousePressAndHoldInterval
y: implicitHeight
}
background: null
}