Libplasma

ComboBox.qml
1/*
2 SPDX-FileCopyrightText: 2016 Marco Martin <mart@kde.org>
3
4 SPDX-License-Identifier: LGPL-2.0-or-later
5*/
6
7pragma ComponentBehavior: Bound
8
9import QtQuick
10import QtQuick.Templates as T
11import QtQuick.Controls as Controls
12import org.kde.ksvg as KSvg
13//NOTE: importing PlasmaCore is necessary in order to make KSvg load the current Plasma Theme
14import org.kde.plasma.core as PlasmaCore
15import org.kde.kirigami as Kirigami
16import "private" as Private
17import "mobiletextselection" as MobileTextSelection
18
19T.ComboBox {
20 id: control
21
22 property real __indicatorMargin: control.indicator && control.indicator.visible && control.indicator.width > 0 ? control.spacing + indicator.width : 0
23
24 implicitWidth: Math.max(implicitBackgroundWidth + leftInset + rightInset,
25 implicitContentWidth + leftPadding + rightPadding)
26 implicitHeight: Math.max(implicitBackgroundHeight + topInset + bottomInset,
27 implicitContentHeight + topPadding + bottomPadding,
28 implicitIndicatorHeight + topPadding + bottomPadding)
29
30 baselineOffset: contentItem.y + contentItem.baselineOffset
31
32 hoverEnabled: true
33
34 topPadding: surfaceNormal.margins.top
35 leftPadding: surfaceNormal.margins.left + (!control.mirrored ? 0 : __indicatorMargin)
36 rightPadding: surfaceNormal.margins.right + (control.mirrored ? 0 : __indicatorMargin)
37 bottomPadding: surfaceNormal.margins.bottom
38 spacing: Kirigami.Units.smallSpacing
39
40 delegate: ItemDelegate {
41 required property var model
42 required property int index
43
44 width: ListView.view.width
45 text: model[control.textRole]
46 highlighted: control.highlightedIndex == index
47 property bool separatorVisible: false
48 }
49
50 indicator: KSvg.SvgItem {
51 implicitWidth: Kirigami.Units.iconSizes.small
52 implicitHeight: implicitWidth
53 anchors {
54 right: parent.right
55 rightMargin: surfaceNormal.margins.right
56 verticalCenter: parent.verticalCenter
57 }
58 svg: KSvg.Svg {
59 imagePath: "widgets/arrows"
60 colorSet: KSvg.Svg.Button
61 }
62 elementId: "down-arrow"
63 }
64
65 contentItem: T.TextField {
66 id: textField
67 implicitWidth: Math.ceil(contentWidth) + leftPadding + rightPadding
68 implicitHeight: Math.ceil(contentHeight) + topPadding + bottomPadding
69 padding: 0
70 text: control.editable ? control.editText : control.displayText
71
72 // We don't want a disabled text color when the control is not editable
73 Kirigami.Theme.textColor: control.Kirigami.Theme.textColor
74 enabled: control.editable
75 autoScroll: control.editable
76
77 readOnly: control.down || !control.editable
78 inputMethodHints: control.inputMethodHints
79 validator: control.validator
80 color: Kirigami.Theme.textColor
81 selectionColor: Kirigami.Theme.highlightColor
82 selectedTextColor: Kirigami.Theme.highlightedTextColor
83
84 selectByMouse: !Kirigami.Settings.tabletMode
85 cursorDelegate: Kirigami.Settings.tabletMode ? mobileCursor : null
86
87 font: control.font
88 horizontalAlignment: Text.AlignLeft
89 verticalAlignment: Text.AlignVCenter
90 opacity: control.enabled ? 1 : 0.3
91 onFocusChanged: {
92 if (focus) {
93 MobileTextSelection.MobileTextActionsToolBar.controlRoot = textField;
94 }
95 }
96
97 onTextChanged: MobileTextSelection.MobileTextActionsToolBar.shouldBeVisible = false;
98 onPressed: event => MobileTextSelection.MobileTextActionsToolBar.shouldBeVisible = true;
99
100 onPressAndHold: event => {
101 if (!Kirigami.Settings.tabletMode) {
102 return;
103 }
104 forceActiveFocus();
105 cursorPosition = positionAt(event.x, event.y);
106 selectWord();
107 }
108 }
109
110 Component {
111 id: mobileCursor
112 MobileTextSelection.MobileCursor {
113 target: textField
114 }
115 }
116
117 MobileTextSelection.MobileCursor {
118 target: textField
119 selectionStartHandle: true
120 property var rect: textField.positionToRectangle(textField.selectionStart)
121 //FIXME: this magic values seem to be always valid, for every font,every dpi, every scaling
122 x: rect.x + 5
123 y: rect.y + 6
124 }
125
126 background: KSvg.FrameSvgItem {
127 id: surfaceNormal
128
129 anchors.fill: parent
130
131 imagePath: control.editable ? "widgets/lineedit" : "widgets/button"
132 prefix: control.editable
133 ? "base"
134 : (control.down ? "pressed" : "normal")
135
136 Private.ButtonShadow {
137 anchors.fill: parent
138 showShadow: !control.editable && !control.down
139 }
140
141 Private.TextFieldFocus {
142 visible: control.editable
143 z: -1
144 state: control.activeFocus ? "focus" : (control.enabled && control.hovered ? "hover" : "hidden")
145 anchors.fill: parent
146 }
147
148 Private.ButtonFocus {
149 anchors.fill: parent
150 showFocus: control.activeFocus && !control.down
151 }
152
153 Private.ButtonHover {
154 anchors.fill: parent
155 showHover: control.enabled && control.hovered && !control.down
156 }
157
158 MouseArea {
159 anchors {
160 fill: parent
161 leftMargin: control.leftPadding
162 rightMargin: control.rightPadding
163 }
164 acceptedButtons: Qt.NoButton
165 onWheel: wheel => {
166 if (wheel.pixelDelta.y < 0 || wheel.angleDelta.y < 0) {
167 control.currentIndex = Math.min(control.currentIndex + 1, delegateModel.count -1);
168 } else {
169 control.currentIndex = Math.max(control.currentIndex - 1, 0);
170 }
171 control.activated(control.currentIndex);
172 }
173 }
174 }
175
176 popup: T.Popup {
177 x: control.mirrored ? control.width - width : 0
178 y: control.height
179 width: Math.max(control.width, 150)
180 implicitHeight: contentItem.implicitHeight
181 topMargin: 6
182 bottomMargin: 6
183
184 contentItem: ListView {
185 id: listView
186 clip: true
187 implicitHeight: contentHeight
188 model: control.popup.visible ? control.delegateModel : null
189 currentIndex: control.highlightedIndex
190 highlightRangeMode: ListView.ApplyRange
191 highlightMoveDuration: 0
192 // HACK: When the ComboBox is not inside a top-level Window, it's Popup does not inherit
193 // the LayoutMirroring options. This is a workaround to fix this by enforcing
194 // the LayoutMirroring options properly.
195 // QTBUG: https://bugreports.qt.io/browse/QTBUG-66446
196 LayoutMirroring.enabled: Qt.application.layoutDirection === Qt.RightToLeft
197 LayoutMirroring.childrenInherit: true
198 T.ScrollBar.vertical: Controls.ScrollBar { }
199 }
200 background: Kirigami.ShadowedRectangle {
201 anchors {
202 fill: parent
203 margins: -1
204 }
205 radius: 2
206 Kirigami.Theme.colorSet: Kirigami.Theme.View
207 Kirigami.Theme.inherit: false
208 color: Kirigami.Theme.backgroundColor
209 border {
210 color: Qt.rgba(Kirigami.Theme.textColor.r, Kirigami.Theme.textColor.g, Kirigami.Theme.textColor.b, 0.3)
211 width: 1
212 }
213 shadow {
214 size: 4
215 xOffset: 2
216 yOffset: 2
217 color: Qt.rgba(0, 0, 0, 0.3)
218 }
219 }
220 }
221}
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:08:30 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.