mirror of
https://github.com/ProtonMail/proton-bridge.git
synced 2025-12-10 04:36:43 +00:00
366 lines
14 KiB
QML
366 lines
14 KiB
QML
// Copyright (c) 2023 Proton AG
|
|
// This file is part of Proton Mail Bridge.
|
|
// Proton Mail Bridge 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.
|
|
// Proton Mail Bridge 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 Proton Mail Bridge. If not, see <https://www.gnu.org/licenses/>.
|
|
import QtQml
|
|
import QtQuick
|
|
import QtQuick.Controls
|
|
import QtQuick.Controls.impl
|
|
import QtQuick.Templates as T
|
|
import QtQuick.Layouts
|
|
import "." as Proton
|
|
|
|
FocusScope {
|
|
id: root
|
|
|
|
property alias activeFocusOnPress: control.activeFocusOnPress
|
|
property string assistiveText
|
|
property alias background: control.background
|
|
property alias baseUrl: control.baseUrl
|
|
property alias bottomInset: control.bottomInset
|
|
property alias bottomPadding: control.bottomPadding
|
|
property alias canPaste: control.canPaste
|
|
property alias canRedo: control.canRedo
|
|
property alias canUndo: control.canUndo
|
|
property alias color: control.color
|
|
property ColorScheme colorScheme
|
|
property alias contentHeight: control.contentHeight
|
|
property alias contentWidth: control.contentWidth
|
|
property alias cursorDelegate: control.cursorDelegate
|
|
property alias cursorPosition: control.cursorPosition
|
|
property alias cursorRectangle: control.cursorRectangle
|
|
property alias cursorVisible: control.cursorVisible
|
|
property alias effectiveHorizontalAlignment: control.effectiveHorizontalAlignment
|
|
property bool error: false
|
|
property string errorString
|
|
//property alias flickable: control.flickable
|
|
property alias focusReason: control.focusReason
|
|
property alias font: control.font
|
|
property alias hint: hint.text
|
|
property alias horizontalAlignment: control.horizontalAlignment
|
|
property alias hoverEnabled: control.hoverEnabled
|
|
property alias hovered: control.hovered
|
|
property alias hoveredLink: control.hoveredLink
|
|
property alias implicitBackgroundHeight: control.implicitBackgroundHeight
|
|
property alias implicitBackgroundWidth: control.implicitBackgroundWidth
|
|
property alias inputMethodComposing: control.inputMethodComposing
|
|
property alias inputMethodHints: control.inputMethodHints
|
|
property alias label: label.text
|
|
property alias leftInset: control.leftInset
|
|
property alias leftPadding: control.leftPadding
|
|
property alias length: control.length
|
|
property alias lineCount: control.lineCount
|
|
property alias mouseSelectionMode: control.mouseSelectionMode
|
|
property alias overwriteMode: control.overwriteMode
|
|
property alias padding: control.padding
|
|
property alias palette: control.palette
|
|
property alias persistentSelection: control.persistentSelection
|
|
property alias placeholderText: control.placeholderText
|
|
property alias placeholderTextColor: control.placeholderTextColor
|
|
property alias preeditText: control.preeditText
|
|
property alias readOnly: control.readOnly
|
|
property alias renderType: control.renderType
|
|
property alias rightInset: control.rightInset
|
|
property alias rightPadding: control.rightPadding
|
|
property alias selectByKeyboard: control.selectByKeyboard
|
|
property alias selectByMouse: control.selectByMouse
|
|
property alias selectedText: control.selectedText
|
|
property alias selectedTextColor: control.selectedTextColor
|
|
property alias selectionColor: control.selectionColor
|
|
property alias selectionEnd: control.selectionEnd
|
|
property alias selectionStart: control.selectionStart
|
|
property alias tabStopDistance: control.tabStopDistance
|
|
property alias text: control.text
|
|
property alias textDocument: control.textDocument
|
|
property alias textFormat: control.textFormat
|
|
property alias textMargin: control.textMargin
|
|
property alias topInset: control.topInset
|
|
property alias topPadding: control.topPadding
|
|
property bool validateOnEditingFinished: true
|
|
// 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 wrapMode: control.wrapMode
|
|
|
|
signal editingFinished
|
|
|
|
function append(text) {
|
|
return control.append(text);
|
|
}
|
|
function clear() {
|
|
return control.clear();
|
|
}
|
|
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);
|
|
}
|
|
|
|
// Calculates the height of the component to make exactly lineNum visible in edit area
|
|
function heightForLinesVisible(lineNum) {
|
|
let totalHeight = 0;
|
|
totalHeight += headerLayout.height;
|
|
totalHeight += footerLayout.height;
|
|
totalHeight += control.topPadding + control.bottomPadding;
|
|
totalHeight += lineNum * fontMetrics.height;
|
|
return totalHeight;
|
|
}
|
|
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();
|
|
}
|
|
function validate() {
|
|
if (validator === undefined) {
|
|
return;
|
|
}
|
|
const error = validator(text);
|
|
if (error) {
|
|
root.error = true;
|
|
root.errorString = error;
|
|
} else {
|
|
root.error = false;
|
|
root.errorString = "";
|
|
}
|
|
}
|
|
|
|
implicitHeight: children[0].implicitHeight
|
|
implicitWidth: children[0].implicitWidth
|
|
|
|
onEditingFinished: {
|
|
if (!validateOnEditingFinished) {
|
|
return;
|
|
}
|
|
validate();
|
|
}
|
|
onTextChanged: {
|
|
root.error = false;
|
|
root.errorString = "";
|
|
}
|
|
|
|
FontMetrics {
|
|
id: fontMetrics
|
|
font: control.font
|
|
}
|
|
ColumnLayout {
|
|
anchors.fill: parent
|
|
spacing: 0
|
|
|
|
RowLayout {
|
|
id: headerLayout
|
|
Layout.fillWidth: true
|
|
spacing: 0
|
|
|
|
Proton.Label {
|
|
id: label
|
|
Layout.fillWidth: true
|
|
color: root.enabled ? root.colorScheme.text_norm : root.colorScheme.text_disabled
|
|
colorScheme: root.colorScheme
|
|
type: Proton.Label.LabelType.Body_semibold
|
|
}
|
|
Proton.Label {
|
|
id: hint
|
|
Layout.fillWidth: true
|
|
color: root.enabled ? root.colorScheme.text_weak : root.colorScheme.text_disabled
|
|
colorScheme: root.colorScheme
|
|
horizontalAlignment: Text.AlignRight
|
|
type: Proton.Label.LabelType.Caption
|
|
}
|
|
}
|
|
ScrollView {
|
|
id: controlView
|
|
Layout.fillHeight: true
|
|
Layout.fillWidth: true
|
|
clip: true
|
|
|
|
T.TextArea {
|
|
id: control
|
|
KeyNavigation.backtab: root.KeyNavigation.backtab
|
|
KeyNavigation.down: root.KeyNavigation.down
|
|
KeyNavigation.left: root.KeyNavigation.left
|
|
KeyNavigation.priority: root.KeyNavigation.priority
|
|
KeyNavigation.right: root.KeyNavigation.right
|
|
KeyNavigation.tab: root.KeyNavigation.tab
|
|
KeyNavigation.up: root.KeyNavigation.up
|
|
bottomPadding: 8
|
|
color: {
|
|
if (!control.enabled) {
|
|
return root.colorScheme.text_disabled;
|
|
}
|
|
if (control.readOnly) {
|
|
return root.colorScheme.text_hint;
|
|
}
|
|
return root.colorScheme.text_norm;
|
|
}
|
|
|
|
// enforcing default focus here within component
|
|
focus: root.focus
|
|
font.family: ProtonStyle.font_family
|
|
font.letterSpacing: ProtonStyle.body_letter_spacing
|
|
font.pixelSize: ProtonStyle.body_font_size
|
|
font.weight: ProtonStyle.fontWeight_400
|
|
implicitHeight: Math.max(contentHeight + topPadding + bottomPadding, implicitBackgroundHeight + topInset + bottomInset, placeholder.implicitHeight + topPadding + bottomPadding)
|
|
implicitWidth: Math.max(contentWidth + leftPadding + rightPadding, implicitBackgroundWidth + leftInset + rightInset, placeholder.implicitWidth + leftPadding + rightPadding)
|
|
leftPadding: 12
|
|
placeholderTextColor: control.enabled ? root.colorScheme.text_hint : root.colorScheme.text_disabled
|
|
rightPadding: 12
|
|
selectByMouse: true
|
|
selectedTextColor: control.palette.highlightedText
|
|
selectionColor: control.palette.highlight
|
|
topPadding: 8
|
|
wrapMode: TextInput.Wrap
|
|
|
|
background: Rectangle {
|
|
anchors.fill: parent
|
|
border.color: {
|
|
if (!control.enabled || control.readOnly) {
|
|
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
|
|
color: root.colorScheme.background_norm
|
|
radius: ProtonStyle.input_radius
|
|
visible: true
|
|
}
|
|
cursorDelegate: Rectangle {
|
|
id: cursor
|
|
color: root.colorScheme.interaction_norm
|
|
visible: control.activeFocus && !control.readOnly && control.selectionStart === control.selectionEnd
|
|
width: 1
|
|
|
|
Connections {
|
|
function onCursorPositionChanged() {
|
|
// keep a moving cursor visible
|
|
cursor.opacity = 1;
|
|
timer.restart();
|
|
}
|
|
|
|
target: control
|
|
}
|
|
Timer {
|
|
id: timer
|
|
interval: Qt.styleHints.cursorFlashTime / 2
|
|
repeat: true
|
|
running: control.activeFocus && !control.readOnly
|
|
|
|
// force the cursor visible when gaining focus
|
|
onRunningChanged: cursor.opacity = 1
|
|
onTriggered: cursor.opacity = !cursor.opacity ? 1 : 0
|
|
}
|
|
}
|
|
|
|
onEditingFinished: root.editingFinished()
|
|
|
|
PlaceholderText {
|
|
id: placeholder
|
|
color: control.placeholderTextColor
|
|
elide: Text.ElideRight
|
|
font: control.font
|
|
height: control.height - (control.topPadding + control.bottomPadding)
|
|
renderType: control.renderType
|
|
text: control.placeholderText
|
|
verticalAlignment: control.verticalAlignment
|
|
visible: !control.length && !control.preeditText && (!control.activeFocus || control.horizontalAlignment !== Qt.AlignHCenter)
|
|
width: control.width - (control.leftPadding + control.rightPadding)
|
|
x: control.leftPadding
|
|
y: control.topPadding
|
|
}
|
|
}
|
|
}
|
|
RowLayout {
|
|
id: footerLayout
|
|
Layout.fillWidth: true
|
|
spacing: 0
|
|
|
|
ColorImage {
|
|
id: errorIcon
|
|
Layout.rightMargin: 4
|
|
color: root.colorScheme.signal_danger
|
|
height: assistiveText.height
|
|
source: "/qml/icons/ic-exclamation-circle-filled.svg"
|
|
sourceSize.height: assistiveText.height
|
|
visible: root.error && (assistiveText.text.length > 0)
|
|
}
|
|
Proton.Label {
|
|
id: assistiveText
|
|
Layout.fillWidth: true
|
|
color: {
|
|
if (!root.enabled) {
|
|
return root.colorScheme.text_disabled;
|
|
}
|
|
if (root.error) {
|
|
return root.colorScheme.signal_danger;
|
|
}
|
|
return root.colorScheme.text_weak;
|
|
}
|
|
colorScheme: root.colorScheme
|
|
text: root.error ? root.errorString : root.assistiveText
|
|
type: root.error ? Proton.Label.LabelType.Caption_semibold : Proton.Label.LabelType.Caption
|
|
}
|
|
}
|
|
}
|
|
}
|