From 80fb71378749c7b7ef4eb8cc82cc575058d68174 Mon Sep 17 00:00:00 2001 From: David Edmundson Date: Mon, 18 Jul 2022 23:13:47 +0100 Subject: [PATCH] Support keyboard navigation between windows across desktops If a DesktopView doesn't handle a keyboard navigation event it propagates to main which then focusses the next desktop view and the relevant window. Empty desktops can also be selected. BUG: 456068 --- src/effects/desktopgrid/qml/DesktopView.qml | 4 ++ src/effects/desktopgrid/qml/main.qml | 63 +++++++++++++++++++++ 2 files changed, 67 insertions(+) diff --git a/src/effects/desktopgrid/qml/DesktopView.qml b/src/effects/desktopgrid/qml/DesktopView.qml index 7c41238344..87e2508be2 100644 --- a/src/effects/desktopgrid/qml/DesktopView.qml +++ b/src/effects/desktopgrid/qml/DesktopView.qml @@ -21,6 +21,10 @@ FocusScope { property real panelOpacity: 1 focus: true + function selectLastItem(direction) { + heap.selectLastItem(direction); + } + DropArea { anchors.fill: parent onEntered: { diff --git a/src/effects/desktopgrid/qml/main.qml b/src/effects/desktopgrid/qml/main.qml index 9e00ea40df..f843e68249 100644 --- a/src/effects/desktopgrid/qml/main.qml +++ b/src/effects/desktopgrid/qml/main.qml @@ -38,6 +38,53 @@ Rectangle { container.effect.deactivate(container.effect.animationDuration); } + function selectNext(direction) { + let currentIndex = 0 + for (let i = 0; i < gridRepeater.count; i++) { + if (gridRepeater.itemAt(i).focus) { + currentIndex = i; + break; + } + } + let x = currentIndex % grid.columns; + let y = Math.floor(currentIndex / grid.columns); + + // the direction we move in is the opposite of the window to select + // i.e pressing left should select the rightmost window on the desktop + // to the left + let invertedDirection; + switch(direction) { + case WindowHeap.Direction.Up: + y--; + invertedDirection = WindowHeap.Direction.Down; + break; + case WindowHeap.Direction.Down: + y++ + invertedDirection = WindowHeap.Direction.Up; + break; + case WindowHeap.Direction.Left: + x--; + invertedDirection = WindowHeap.Direction.Right; + break; + case WindowHeap.Direction.Right: + x++; + invertedDirection = WindowHeap.Direction.Left; + break; + } + + if (x < 0 || x >= grid.columns) { + return false; + } + if (y < 0 || y >= grid.rows) { + return false; + } + let newIndex = y * grid.columns + x; + + gridRepeater.itemAt(newIndex).focus = true; + gridRepeater.itemAt(newIndex).selectLastItem(invertedDirection); + return true; + } + Keys.onPressed: { if (event.key == Qt.Key_Escape) { effect.deactivate(effect.animationDuration); @@ -51,6 +98,21 @@ Rectangle { } else if (event.key >= Qt.Key_0 && event.key <= Qt.Key_9) { const desktopId = event.key == Qt.Key_0 ? 10 : (event.key - Qt.Key_0); switchTo(desktopId); + } else if (event.key == Qt.Key_Up) { + event.accepted = selectNext(WindowHeap.Direction.Up); + } else if (event.key == Qt.Key_Down) { + event.accepted = selectNext(WindowHeap.Direction.Down); + } else if (event.key == Qt.Key_Left) { + event.accepted = selectNext(WindowHeap.Direction.Left); + } else if (event.key == Qt.Key_Right) { + event.accepted = selectNext(WindowHeap.Direction.Right); + } else if (event.key == Qt.Key_Return || event.key == Qt.Key_Space) { + for (let i = 0; i < gridRepeater.count; i++) { + if (gridRepeater.itemAt(i).focus) { + switchTo(gridRepeater.itemAt(i).desktop.x11DesktopNumber) + break; + } + } } } Keys.priority: Keys.AfterItem @@ -145,6 +207,7 @@ Rectangle { } ] Repeater { + id: gridRepeater model: desktopModel DesktopView { id: thumbnail