MauiKit Terminal

Terminal.qml
1
2import QtQuick
3import QtQuick.Controls
4import QtQuick.Layouts
5
6import org.mauikit.controls as Maui
7import org.mauikit.terminal as Term
8import "private" as Private
9
10/**
11 * @brief A terminal emulator console with built-in support for touch and keyboard inputs, and many other features.
12 *
13 * @image html terminal.png "Demo of the Terminal control"
14 *
15 * @section features Features
16 *
17 * This control has a pack of built-in functionality making it ready for embedding it into any other application.
18 * A quick overview includes support for touch gestures for navigation, scrollbar, search and find, contextual menu actions and keyboard shortcuts.
19 *
20 * @note For creating your own implementation you can look into using the exposed classes `QMLTermWidget` and `QMLTermSession`.
21 *
22 * @code
23 * import QtQuick
24 * import QtQuick.Controls
25 *
26 * import org.mauikit.controls as Maui
27 *
28 * import org.mauikit.terminal as Term
29 *
30 * Maui.ApplicationWindow
31 * {
32 * id: root
33 *
34 * Maui.Page
35 * {
36 * Maui.Controls.showCSD: true
37 * anchors.fill: parent
38 *
39 * Term.Terminal
40 * {
41 * anchors.fill: parent
42 * }
43 * }
44 * }
45 * @endcode
46 */
47Maui.Page
48{
49 id: control
50
51 Maui.Theme.colorSet: Maui.Theme.Header
52 Maui.Theme.inherit: false
53
54 title: ksession.title
55 showTitle: false
56
57 headBar.visible: false
58
59 focus: true
60 clip: true
61
62 /**
63 * @see Konsole::TerminalDisplay::readOnly
64 */
65 property alias readOnly : kterminal.readOnly
67 /**
68 * @brief The resolution size of the emulated terminal display
69 */
70 readonly property size virtualResolution: Qt.size(kterminal.width, kterminal.height)
71
72 /**
73 * @brief Alias to the terminal display object
74 * @property Konsole::TerminalDisplay Terminal::kterminal
75 */
76 readonly property alias kterminal: kterminal
77
78 /**
79 * @brief Alias to the emulated terminal session
80 * @property KSession Terminal::session
81 */
82 readonly property alias session: ksession
83
84 /**
85 * @brief Alias to the TextField used for handling the search queries. It is exposed to allow appending action entries, for example.
86 * @property TextField Terminal::findBar
87 */
88 readonly property alias findBar : findBar
90 /**
91 * @brief The content of the contextual menu. A set of default actions is already added, to append more actions use this property.
92 * @code
93 * menu: [
94 * Action { text: "Action1" },
95 * MenuItem {text: "Action2"}
96 * ]
97 * @endcode
98 * @property list<QtObject> Terminal::menu
99 */
100 property alias menu : terminalMenu.contentData
101
102 /**
103 * @see TerminalDisplay::terminalSize
104 */
105 readonly property alias terminalSize: kterminal.terminalSize
106
107 /**
108 * @see TerminalDisplay::fontMetricts
109 */
110 readonly property alias fontMetrics: kterminal.fontMetrics
111
112 /**
113 * @brief Emitted when a drag&drop action has been performed and the drop contains valid URLs
114 * @param urls The list of URLs dropped
115 */
116 signal urlsDropped(var urls)
117
118 /**
119 * @brief Emitted when a keyboard shortcut has been triggered and it is not one of the default ones that are recognized
120 * @param event Object with the event information
121 */
122 signal keyPressed(var event)
123
124 /**
125 * @brief Emitted when the terminal control area has been clicked
126 */
127 signal clicked()
128
129 //Actions
130 Action
131 {
132 id: _copyAction
133 text: i18nd("mauikitterminal", "Copy")
134 icon.name: "edit-copy"
135 onTriggered:
136 {
137 kterminal.copyClipboard();
138 control.forceActiveFocus()
139 }
140 // shortcut: "Ctrl+Shift+C"
141 }
142
143 Action
144 {
145 id: _pasteAction
146 text: i18nd("mauikitterminal", "Paste")
147 icon.name: "edit-paste"
148 onTriggered:
149 {
150 kterminal.pasteClipboard()
151 control.forceActiveFocus()
152 }
153 // shortcut: "Ctrl+Shift+V"
154 }
155
156 Action
157 {
158 id: _findAction
159 text: i18nd("mauikitterminal", "Find")
160 icon.name: "edit-find"
161 checkable: true
162 // shortcut: "Ctrl+Shift+F"
163 onTriggered: toggleSearchBar()
164 }
165
167 {
168 id: terminalMenu
169 Maui.Controls.component: !kterminal.isTextSelected ? null : _headerComponent
170
171 Component
172 {
173 id: _headerComponent
174 Pane
175 {
176 width: ListView.view.width
177 implicitHeight: Math.min(80, contentHeight) + topPadding + bottomPadding
178 clip: true
179
180 background: Rectangle
181 {
182 color: kterminal.backgroundColor
183 radius: Maui.Style.radiusV
184 }
185
186 contentItem: Label
187 {
188 text: kterminal.isTextSelected ? kterminal.selectedText() : ""
189 color :kterminal.foregroundColor
190 wrapMode: Text.WordWrap
191 elide: Text.ElideRight
192 font: kterminal.font
193 }
194 }
195 }
196
197 MenuItem
198 {
199 action: _copyAction
200 enabled: kterminal.isTextSelected
201 }
202
203 MenuItem
204 {
205 action: _pasteAction
206 }
207
208 MenuSeparator {}
209
210 MenuItem
211 {
212 action: _findAction
213 }
214 }
215
216 footBar.visible: false
217 footBar.forceCenterMiddleContent: false
218 footBar.rightContent: Maui.ToolButtonMenu
219 {
220 icon.name: "overflow-menu"
221
222 MenuItem
223 {
224 id: _findCaseSensitively
225 checkable: true
226 text: i18nd("mauikittexteditor","Case Sensitive")
227 }
228
229 MenuItem
230 {
231 id: _findWholeWords
232 checkable: true
233 text: i18nd("mauikittexteditor","Whole Words Only")
234 }
235 }
236
237 footBar.middleContent: Maui.SearchField
238 {
239 id: findBar
240 Layout.fillWidth: true
241 Layout.alignment: Qt.AlignHCenter
242 Layout.maximumWidth: 500
243 placeholderText: i18nd("mauikitterminal", "Find...")
244 onAccepted: ksession.search(text, ksession.previousLineSearch, ksession.previousColumnSearch, false)
245 onVisibleChanged:
246 {
247 if(visible)
248 {
249 findBar.forceActiveFocus()
250 }else
251 {
252 control.forceActiveFocus()
253 }
254 }
255
256 actions: [
257
258 Action
259 {
260 icon.name: "go-up"
261 // text: i18n("Previous")
262 onTriggered: ksession.search(findBar.text, ksession.previousLineSearch, ksession.previousColumnSearch, false)
263 }
264 ]
265
266 Keys.enabled: true
267 Keys.onPressed: (event) =>
268 {
269 if ((event.key == Qt.Key_F) && (event.modifiers & Qt.ControlModifier) && (event.modifiers & Qt.ShiftModifier))
270 {
271 control.toggleSearchBar()
272 }
273 }
274 }
275
276 Term.QMLTermWidget
277 {
278 id: kterminal
279 anchors.fill: parent
280 // terminalUsesMouse: true
281
282 enableBold: true
283 fullCursorHeight: true
284
285 property int totalLines: kterminal.scrollbarMaximum - kterminal.scrollbarMinimum + kterminal.lines
286 // onKeyPressedSignal: console.log(e.key)
287
288 font: Maui.Style.monospacedFont
289
290 backgroundOpacity: 1
291
292 onTerminalUsesMouseChanged: console.log(terminalUsesMouse);
293
294 session: Term.QMLTermSession
295 {
296 id: ksession
297 initialWorkingDirectory: "$HOME"
298 onFinished:
299 {
300 console.log("Terminal finished")
301 }
302
303 // Disable search until implemented correctly
304 property int previousColumnSearch : 0
305 property int previousLineSearch: 0
306
307 onMatchFound:
308 {
309 previousColumnSearch = startColumn
310 previousLineSearch = startLine
311
312
313 _scrollBarLoader.item.highlightLine = startLine
314
315 kterminal.matchFound(startColumn, startLine, endColumn, endLine)
316 console.log("found at: %1 %2 %3 %4".arg(startColumn).arg(startLine).arg(endColumn).arg(endLine));
317 }
318
319 onNoMatchFound:
320 {
321 previousColumnSearch = 0
322 previousLineSearch = 0
323 _scrollBarLoader.item.highlightLine = -1
324
325 kterminal.noMatchFound();
326 console.log("not found");
327 }
328 }
329
330 customColorScheme
331 {
332 backgroundColor: Maui.Theme.backgroundColor
333 foregroundColor: Maui.Theme.textColor
334 color2: Maui.Theme.disabledTextColor
335 color3: Maui.Theme.negativeBackgroundColor
336 color4: Maui.Theme.positiveBackgroundColor
337 color5: Maui.Theme.neutralBackgroundColor
338 color6: Maui.Theme.linkColor
339 color7: Maui.Theme.visitedLinkColor
340 color8: Maui.Theme.highlightColor
341 color9: Maui.Theme.highlightedTextColor
342 }
343
344 Keys.enabled: true
345
346 Keys.onPressed: (event) =>
347 {
348 if ((event.key === Qt.Key_A) && (event.modifiers & Qt.ControlModifier) && (event.modifiers & Qt.ShiftModifier))
349 {
350 kterminal.selectAll()
351 event.accepted = true
352 return
353 }
354
355 if ((event.key === Qt.Key_C) && (event.modifiers & Qt.ControlModifier) && (event.modifiers & Qt.ShiftModifier))
356 {
357 _copyAction.trigger()
358 event.accepted = true
359 return
360 }
361
362 if ((event.key === Qt.Key_V) && (event.modifiers & Qt.ControlModifier) && (event.modifiers & Qt.ShiftModifier))
363 {
364 _pasteAction.trigger()
365 event.accepted = true
366 return
367 }
368
369
370 if ((event.key == Qt.Key_F) && (event.modifiers & Qt.ControlModifier) && (event.modifiers & Qt.ShiftModifier))
371 {
372 control.toggleSearchBar()
373 return
374 }
375
376 control.keyPressed(event)
377 }
378
379 Loader
380 {
381 asynchronous: true
382 anchors.fill: parent
383
384 sourceComponent: Private.TerminalInputArea
385 {
386 // enabled: terminalPage.state != "SELECTION"
387
388 // FIXME: should anchor to the bottom of the window to cater for the case when the OSK is up
389
390 // This is the minimum wheel event registered by the plugin (with the current settings).
391 property real wheelValue: 40
392
393 // This is needed to fake a "flickable" scrolling.
394 swipeDelta: kterminal.fontMetrics.height
395
396 // Mouse actions
397 onMouseMoveDetected:(x, y, button, buttons, modifiers) => kterminal.simulateMouseMove(x, y, button, buttons, modifiers);
398 onDoubleClickDetected:(x, y, button, buttons, modifiers) => kterminal.simulateMouseDoubleClick(x, y, button, buttons, modifiers);
399 onMousePressDetected:(x, y, button, buttons, modifiers) =>
400 {
401 kterminal.forceActiveFocus();
402 kterminal.simulateMousePress(x, y, button, buttons, modifiers);
403 control.clicked()
404 }
405 onMouseReleaseDetected: (x, y, button, buttons, modifiers) => kterminal.simulateMouseRelease(x, y, button, buttons, modifiers);
406 onMouseWheelDetected: (x, y, buttons, modifiers, angleDelta) => kterminal.simulateWheel(x, y, buttons, modifiers, angleDelta);
407
408 // Touch actions
409 onTouchPress: (x, y) =>
410 {
411 // kterminal.forceActiveFocus()
412 // control.clicked()
413 }
414
415 onTouchClick: (x, y) =>
416 {
417 kterminal.forceActiveFocus()
418 // kterminal.simulateKeyPress(Qt.Key_Tab, Qt.NoModifier, true, 0, "");
419 control.clicked()
420 }
421
422 onTouchPressAndHold: (x, y) =>
423 {
424 alternateAction(x, y);
425 }
426
427 // Swipe actions
428 onSwipeYDetected: (steps) => {
429 if (steps > 0) {
430 simulateSwipeDown(steps);
431 } else {
432 simulateSwipeUp(-steps);
433 }
434 }
435
436 onSwipeXDetected: (steps) => {
437 if (steps > 0) {
438 simulateSwipeRight(steps);
439 } else {
440 simulateSwipeLeft(-steps);
441 }
442 }
443
444 onTwoFingerSwipeYDetected: (steps) => {
445 if (steps > 0) {
446 simulateDualSwipeDown(steps);
447 } else {
448 simulateDualSwipeUp(-steps);
449 }
450 }
451
452 function simulateSwipeUp(steps) {
453 while(steps > 0) {
454 kterminal.simulateKeyPress(Qt.Key_Up, Qt.NoModifier, true, 0, "");
455 steps--;
456 }
457 }
458 function simulateSwipeDown(steps) {
459 while(steps > 0) {
460 kterminal.simulateKeyPress(Qt.Key_Down, Qt.NoModifier, true, 0, "");
461 steps--;
462 }
463 }
464 function simulateSwipeLeft(steps) {
465 while(steps > 0) {
466 kterminal.simulateKeyPress(Qt.Key_Left, Qt.NoModifier, true, 0, "");
467 steps--;
468 }
469 }
470 function simulateSwipeRight(steps) {
471 while(steps > 0) {
472 kterminal.simulateKeyPress(Qt.Key_Right, Qt.NoModifier, true, 0, "");
473 steps--;
474 }
475 }
476 function simulateDualSwipeUp(steps) {
477 while(steps > 0) {
478 kterminal.simulateWheel(width * 0.5, height * 0.5, Qt.NoButton, Qt.NoModifier, Qt.point(0, -wheelValue));
479 steps--;
480 }
481 }
482 function simulateDualSwipeDown(steps) {
483 while(steps > 0) {
484 kterminal.simulateWheel(width * 0.5, height * 0.5, Qt.NoButton, Qt.NoModifier, Qt.point(0, wheelValue));
485 steps--;
486 }
487 }
488
489 // Semantic actions
490 onAlternateAction: (x, y) => {
491 // Force the hiddenButton in the event position.
492 //hiddenButton.x = x;
493 //hiddenButton.y = y;
494 terminalMenu.show()
495
496 }
497 }
498 }
499
500 Loader
501 {
502 id: _scrollBarLoader
503 asynchronous: true
504 anchors.fill: parent
505
506 sourceComponent: Private.TerminalScrollBar
507 {
508 terminal: kterminal
509 }
510 }
511
512 Maui.FloatingButton
513 {
514 visible: Maui.Handy.isMobile
515 anchors.right: parent.right
516 anchors.bottom: parent.bottom
517 anchors.margins: Maui.Style.space.big
518 icon.name: "input-keyboard-virtual"
519 text: i18n("Toggle Virtual Keyboard")
520 onClicked:
521 {
522 if (Qt.inputMethod.visible)
523 {
524 Qt.inputMethod.hide();
525 } else
526 {
527 control.forceActiveFocus();
528 Qt.inputMethod.show();
529 }
530 }
531 }
532 }
533
534 opacity: _dropArea.containsDrag ? 0.5 : 1
535
536 DropArea
537 {
538 id: _dropArea
539 anchors.fill: parent
540 onDropped: (drop) =>
541 {
542 if(drop.hasUrls)
543 control.urlsDropped(drop.urls)
544 }
545 }
546
547 Component.onCompleted:
548 {
549 ksession.startShellProgram();
550 forceActiveFocus()
551 }
552
553 /**
554 * @brief Force to focus the terminal display for entering input
555 */
556 function forceActiveFocus()
557 {
558 kterminal.forceActiveFocus()
559 }
560
561 /**
562 * Toggle the search text field bar
563 */
564 function toggleSearchBar()
565 {
566 footBar.visible = !footBar.visible
567 }
568}
bool visible
alias readOnly
Definition Terminal.qml:55
alias kterminal
Alias to the terminal display object.
Definition Terminal.qml:66
alias findBar
Alias to the TextField used for handling the search queries.
Definition Terminal.qml:78
void keyPressed(var event)
Emitted when a keyboard shortcut has been triggered and it is not one of the default ones that are re...
alias session
Alias to the emulated terminal session.
Definition Terminal.qml:72
alias menu
The content of the contextual menu.
Definition Terminal.qml:89
alias terminalSize
Definition Terminal.qml:94
void urlsDropped(var urls)
Emitted when a drag&drop action has been performed and the drop contains valid URLs.
size virtualResolution
The resolution size of the emulated terminal display.
Definition Terminal.qml:60
void clicked()
Emitted when the terminal control area has been clicked.
alias fontMetrics
Definition Terminal.qml:99
QString i18nd(const char *domain, const char *text, const TYPE &arg...)
QString i18n(const char *text, const TYPE &arg...)
AKONADI_CALENDAR_EXPORT KCalendarCore::Event::Ptr event(const Akonadi::Item &item)
KIOWIDGETS_EXPORT DropJob * drop(const QDropEvent *dropEvent, const QUrl &destUrl, DropJobFlags dropjobFlags, JobFlags flags=DefaultFlags)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:53:15 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.