melawy-plasma-plasmoid-Menu.../AndromedaLauncher/contents/ui/AppGridView.qml

237 lines
9.6 KiB
QML

/*****************************************************************************
* Copyright (C) 2022 by Friedrich Schriewer <friedrich.schriewer@gmx.net> *
* *
* 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 2 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, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA . *
****************************************************************************/
import QtQuick
import org.kde.plasma.core 2.0 as PlasmaCore
import org.kde.plasma.extras 2.0 as PlasmaExtras
import org.kde.plasma.private.kicker 0.1 as Kicker
import QtQuick.Window 2.2
import org.kde.plasma.components 3.0 as PC3
import QtQuick.Layouts 1.15
import QtQuick.Controls 2.15
import org.kde.kirigami as Kirigami
GridView {
id: grid
property bool movedWithKeyboard: false
property bool movedWithWheel: false
property bool showScrollBar: true
readonly property int columns: plasmoid.configuration.numberColumns
focus: true
clip: true
currentIndex: count > 0 ? 0 : -1
interactive: height < contentHeight
reuseItems: true
boundsBehavior: Flickable.StopAtBounds
// default keyboard navigation doesn't allow focus reasons to be used
// and eats up/down key events when at the beginning or end of the list.
keyNavigationEnabled: false
keyNavigationWraps: false
highlightMoveDuration: 0
cellWidth: root.cellSizeWidth
cellHeight: root.cellSizeHeight
highlight: Highlight {
hideBg: plasmoid.configuration.enableGlow
}
delegate: FavoriteItem {
id: favitem
triggerModel: grid.model
}
move: normalTransition
moveDisplaced: normalTransition
Transition {
id: normalTransition
NumberAnimation {
duration: Kirigami.Units.shortDuration
properties: "x, y"
easing.type: Easing.OutCubic
}
}
PC3.ScrollBar.vertical: PC3.ScrollBar {
id: verticalScrollBar
parent: grid
z: 2
height: grid.height
anchors.right: parent.right
visible: grid.showScrollBar
}
Kirigami.WheelHandler {
id: wheelHandler
target: grid
filterMouseEvents: true
// `20 * Qt.styleHints.wheelScrollLines` is the default speed.
horizontalStepSize: 20 * Qt.styleHints.wheelScrollLines
verticalStepSize: 20 * Qt.styleHints.wheelScrollLines
onWheel: wheel => {
grid.movedWithWheel = true
grid.movedWithKeyboard = false
movedWithWheelTimer.restart()
}
}
Connections {
target: root
function onVisibleChanged() {
if (!root.visible) {
grid.currentIndex = 0
grid.positionViewAtBeginning()
}
}
}
// Used to block hover events temporarily after using keyboard navigation.
// If you have one hand on the touch pad or mouse and another hand on the keyboard,
// it's easy to accidentally reset the highlight/focus position to the mouse position.
Timer {
id: movedWithKeyboardTimer
interval: 200
onTriggered: grid.movedWithKeyboard = false
}
Timer {
id: movedWithWheelTimer
interval: 200
onTriggered: grid.movedWithWheel = false
}
function focusCurrentItem(event, focusReason) {
currentItem.forceActiveFocus(focusReason)
event.accepted = true
}
Keys.onPressed: event => {
const targetX = currentItem ? currentItem.x : contentX
let targetY = currentItem ? currentItem.y : contentY
let targetIndex = currentIndex
// supports mirroring
const atLeft = currentIndex % columns === (Qt.application.layoutDirection == Qt.RightToLeft ? columns - 1 : 0)
// at the beginning of a line
const isLeading = currentIndex % columns === 0
// at the top of a given column and in the top row
const atTop = currentIndex < columns
// supports mirroring
const atRight = currentIndex % columns === (Qt.application.layoutDirection == Qt.RightToLeft ? 0 : columns - 1)
// at the end of a line
const isTrailing = currentIndex % columns === columns - 1
// at bottom of a given column, not necessarily in the last row
let atBottom = currentIndex >= count - columns
// Implements the keyboard navigation described in https://www.w3.org/TR/wai-aria-practices-1.2/#grid
if (count > 1) {
switch (event.key) {
case Qt.Key_Left: if (!atLeft && !searchBar.textField.activeFocus) {
moveCurrentIndexLeft()
focusCurrentItem(event, Qt.BacktabFocusReason)
} break
case Qt.Key_H: if (!atLeft && !searchBar.textField.activeFocus && event.modifiers & Qt.ControlModifier) {
moveCurrentIndexLeft()
focusCurrentItem(event, Qt.BacktabFocusReason)
} break
case Qt.Key_Up: if (!atTop) {
moveCurrentIndexUp()
focusCurrentItem(event, Qt.BacktabFocusReason)
} break
case Qt.Key_K: if (!atTop && event.modifiers & Qt.ControlModifier) {
moveCurrentIndexUp()
focusCurrentItem(event, Qt.BacktabFocusReason)
} break
case Qt.Key_Right: if (!atRight && !searchBar.textField.activeFocus) {
moveCurrentIndexRight()
focusCurrentItem(event, Qt.TabFocusReason)
} break
case Qt.Key_L: if (!atRight && !searchBar.textField.activeFocus && event.modifiers & Qt.ControlModifier) {
moveCurrentIndexRight()
focusCurrentItem(event, Qt.TabFocusReason)
} break
case Qt.Key_Down: if (!atBottom) {
moveCurrentIndexDown()
focusCurrentItem(event, Qt.TabFocusReason)
} break
case Qt.Key_J: if (!atBottom && event.modifiers & Qt.ControlModifier) {
moveCurrentIndexDown()
focusCurrentItem(event, Qt.TabFocusReason)
} break
case Qt.Key_Home: if (event.modifiers === Qt.ControlModifier && currentIndex !== 0) {
currentIndex = 0
focusCurrentItem(event, Qt.BacktabFocusReason)
} else if (!isLeading) {
targetIndex -= currentIndex % columns
currentIndex = Math.max(targetIndex, 0)
focusCurrentItem(event, Qt.BacktabFocusReason)
} break
case Qt.Key_End: if (event.modifiers === Qt.ControlModifier && currentIndex !== count - 1) {
currentIndex = count - 1
focusCurrentItem(event, Qt.TabFocusReason)
} else if (!isTrailing) {
targetIndex += columns - 1 - (currentIndex % columns)
currentIndex = Math.min(targetIndex, count - 1)
focusCurrentItem(event, Qt.TabFocusReason)
} break
case Qt.Key_PageUp: if (!atTop) {
targetY = targetY - height + 1
targetIndex = indexAt(targetX, targetY)
// TODO: Find a more efficient, but accurate way to do this
while (targetIndex === -1) {
targetY += 1
targetIndex = indexAt(targetX, targetY)
}
currentIndex = Math.max(targetIndex, 0)
focusCurrentItem(event, Qt.BacktabFocusReason)
} break
case Qt.Key_PageDown: if (!atBottom) {
targetY = targetY + height - 1
targetIndex = indexAt(targetX, targetY)
// TODO: Find a more efficient, but accurate way to do this
while (targetIndex === -1) {
targetY -= 1
targetIndex = indexAt(targetX, targetY)
}
currentIndex = Math.min(targetIndex, count - 1)
focusCurrentItem(event, Qt.TabFocusReason)
} break
case Qt.Key_Return:
/* Fall through*/
case Qt.Key_Enter:
grid.currentItem.trigger();
grid.currentItem.forceActiveFocus(Qt.ShortcutFocusReason);
event.accepted = true;
break;
}
}
movedWithKeyboard = event.accepted
if (movedWithKeyboard) {
movedWithKeyboardTimer.restart()
}
}
}