forked from Shinonome/dots-hyprland
session menu
This commit is contained in:
+1
@@ -1,4 +1,5 @@
|
|||||||
import "root:/modules/common"
|
import "root:/modules/common"
|
||||||
|
import "root:/modules/common/widgets/"
|
||||||
import QtQuick
|
import QtQuick
|
||||||
import QtQuick.Controls
|
import QtQuick.Controls
|
||||||
import QtQuick.Layouts
|
import QtQuick.Layouts
|
||||||
@@ -164,6 +164,7 @@ Singleton {
|
|||||||
property int larger: 19
|
property int larger: 19
|
||||||
property int huge: 22
|
property int huge: 22
|
||||||
property int hugeass: 23
|
property int hugeass: 23
|
||||||
|
property int title: 28
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -176,7 +177,7 @@ Singleton {
|
|||||||
property QtObject elementDecelFast: QtObject {
|
property QtObject elementDecelFast: QtObject {
|
||||||
property int duration: 140
|
property int duration: 140
|
||||||
property int type: Easing.OutCirc
|
property int type: Easing.OutCirc
|
||||||
property int velocity: 750
|
property int velocity: 850
|
||||||
}
|
}
|
||||||
property QtObject menuDecel: QtObject {
|
property QtObject menuDecel: QtObject {
|
||||||
property int duration: 350
|
property int duration: 350
|
||||||
|
|||||||
@@ -30,8 +30,8 @@ Item {
|
|||||||
|
|
||||||
RowLayout { // Icon on the left, stuff on the right
|
RowLayout { // Icon on the left, stuff on the right
|
||||||
id: valueRow
|
id: valueRow
|
||||||
spacing: 5
|
|
||||||
Layout.margins: 10
|
Layout.margins: 10
|
||||||
|
spacing: 10
|
||||||
|
|
||||||
MaterialSymbol { // Icon
|
MaterialSymbol { // Icon
|
||||||
Layout.alignment: Qt.AlignVCenter
|
Layout.alignment: Qt.AlignVCenter
|
||||||
@@ -47,8 +47,8 @@ Item {
|
|||||||
spacing: 5
|
spacing: 5
|
||||||
|
|
||||||
RowLayout { // Name fill left, value on the right end
|
RowLayout { // Name fill left, value on the right end
|
||||||
Layout.leftMargin: valueBarHeight / 2 // Align text with progressbar radius curve's left end
|
Layout.leftMargin: valueProgressBar.height / 2 // Align text with progressbar radius curve's left end
|
||||||
Layout.rightMargin: valueBarHeight / 2 // Align text with progressbar radius curve's left end
|
Layout.rightMargin: valueProgressBar.height / 2 // Align text with progressbar radius curve's left end
|
||||||
|
|
||||||
StyledText {
|
StyledText {
|
||||||
color: Appearance.colors.colOnLayer0
|
color: Appearance.colors.colOnLayer0
|
||||||
|
|||||||
@@ -0,0 +1,290 @@
|
|||||||
|
import "root:/modules/common"
|
||||||
|
import "root:/modules/common/widgets"
|
||||||
|
import QtQuick
|
||||||
|
import QtQuick.Controls
|
||||||
|
import QtQuick.Layouts
|
||||||
|
import Quickshell
|
||||||
|
import Quickshell.Hyprland
|
||||||
|
import Quickshell.Io
|
||||||
|
import Quickshell.Wayland
|
||||||
|
import Quickshell.Widgets
|
||||||
|
|
||||||
|
Scope {
|
||||||
|
id: root
|
||||||
|
readonly property Toplevel activeWindow: ToplevelManager.activeToplevel
|
||||||
|
|
||||||
|
Variants {
|
||||||
|
id: sessionVariants
|
||||||
|
model: Quickshell.screens
|
||||||
|
|
||||||
|
PanelWindow { // Session menu
|
||||||
|
id: sessionRoot
|
||||||
|
visible: false
|
||||||
|
|
||||||
|
property var modelData
|
||||||
|
property string subtitle
|
||||||
|
|
||||||
|
screen: modelData
|
||||||
|
exclusionMode: ExclusionMode.Ignore
|
||||||
|
WlrLayershell.namespace: "quickshell:session"
|
||||||
|
WlrLayershell.layer: WlrLayer.Overlay
|
||||||
|
WlrLayershell.keyboardFocus: WlrKeyboardFocus.Exclusive
|
||||||
|
color: Appearance.transparentize(Appearance.m3colors.m3background, 0.4)
|
||||||
|
|
||||||
|
anchors {
|
||||||
|
top: true
|
||||||
|
left: true
|
||||||
|
right: true
|
||||||
|
}
|
||||||
|
|
||||||
|
width: modelData.width
|
||||||
|
height: modelData.height
|
||||||
|
|
||||||
|
HyprlandFocusGrab {
|
||||||
|
id: grab
|
||||||
|
windows: [ sessionRoot ]
|
||||||
|
active: false
|
||||||
|
onCleared: () => {
|
||||||
|
if (!active) sessionRoot.visible = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
target: sessionRoot
|
||||||
|
function onVisibleChanged() {
|
||||||
|
delayedGrabTimer.start()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Timer {
|
||||||
|
id: delayedGrabTimer
|
||||||
|
interval: ConfigOptions.hacks.arbitraryRaceConditionDelay
|
||||||
|
repeat: false
|
||||||
|
onTriggered: {
|
||||||
|
grab.active = sessionRoot.visible
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MouseArea {
|
||||||
|
id: sessionMouseArea
|
||||||
|
anchors.fill: parent
|
||||||
|
onClicked: {
|
||||||
|
sessionRoot.visible = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout { // Content column
|
||||||
|
anchors.centerIn: parent
|
||||||
|
spacing: 15
|
||||||
|
|
||||||
|
Keys.onPressed: (event) => {
|
||||||
|
if (event.key === Qt.Key_Escape) {
|
||||||
|
sessionRoot.visible = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
spacing: 0
|
||||||
|
StyledText { // Title
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
font.family: Appearance.font.family.title
|
||||||
|
font.pixelSize: Appearance.font.pixelSize.title
|
||||||
|
font.weight: Font.DemiBold
|
||||||
|
text: "Session"
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText { // Small instruction
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
font.family: Appearance.font.family.title
|
||||||
|
font.pixelSize: Appearance.font.pixelSize.normal
|
||||||
|
text: "Arrow keys to navigate, Enter to select\nEsc or click anywhere to cancel"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RowLayout { // First row of buttons
|
||||||
|
spacing: 15
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionLock
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "lock"
|
||||||
|
buttonText: "Lock"
|
||||||
|
onClicked: { lock.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.right: sessionSleep
|
||||||
|
KeyNavigation.down: sessionHibernate
|
||||||
|
}
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionSleep
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "dark_mode"
|
||||||
|
buttonText: "Sleep"
|
||||||
|
onClicked: { sleep.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.left: sessionLock
|
||||||
|
KeyNavigation.right: sessionLogout
|
||||||
|
KeyNavigation.down: sessionShutdown
|
||||||
|
}
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionLogout
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "logout"
|
||||||
|
buttonText: "Logout"
|
||||||
|
onClicked: { logout.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.left: sessionSleep
|
||||||
|
KeyNavigation.right: sessionTaskManager
|
||||||
|
KeyNavigation.down: sessionReboot
|
||||||
|
}
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionTaskManager
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "browse_activity"
|
||||||
|
buttonText: "Task Manager"
|
||||||
|
onClicked: { taskManager.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.left: sessionLogout
|
||||||
|
KeyNavigation.down: sessionFirmwareReboot
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RowLayout { // Second row of buttons
|
||||||
|
spacing: 15
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionHibernate
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "downloading"
|
||||||
|
buttonText: "Hibernate"
|
||||||
|
onClicked: { hibernate.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.up: sessionLock
|
||||||
|
KeyNavigation.right: sessionShutdown
|
||||||
|
}
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionShutdown
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "power_settings_new"
|
||||||
|
buttonText: "Shutdown"
|
||||||
|
onClicked: { shutdown.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.left: sessionHibernate
|
||||||
|
KeyNavigation.right: sessionReboot
|
||||||
|
KeyNavigation.up: sessionSleep
|
||||||
|
}
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionReboot
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "restart_alt"
|
||||||
|
buttonText: "Reboot"
|
||||||
|
onClicked: { reboot.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.left: sessionShutdown
|
||||||
|
KeyNavigation.right: sessionFirmwareReboot
|
||||||
|
KeyNavigation.up: sessionLogout
|
||||||
|
}
|
||||||
|
SessionActionButton {
|
||||||
|
id: sessionFirmwareReboot
|
||||||
|
focus: sessionRoot.visible
|
||||||
|
buttonIcon: "reset_wrench"
|
||||||
|
buttonText: "Reboot to firmware settings"
|
||||||
|
onClicked: { firmwareReboot.running = true; sessionRoot.visible = false }
|
||||||
|
onFocusChanged: { if (focus) sessionRoot.subtitle = buttonText }
|
||||||
|
KeyNavigation.up: sessionTaskManager
|
||||||
|
KeyNavigation.left: sessionReboot
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
Layout.alignment: Qt.AlignHCenter
|
||||||
|
radius: Appearance.rounding.normal
|
||||||
|
implicitHeight: sessionSubtitle.implicitHeight + 10 * 2
|
||||||
|
implicitWidth: sessionSubtitle.implicitWidth + 10 * 2
|
||||||
|
color: Appearance.colors.colTooltip
|
||||||
|
clip: true
|
||||||
|
|
||||||
|
Behavior on implicitWidth {
|
||||||
|
SmoothedAnimation {
|
||||||
|
velocity: Appearance.animation.elementDecelFast.velocity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledText {
|
||||||
|
id: sessionSubtitle
|
||||||
|
anchors.centerIn: parent
|
||||||
|
color: Appearance.colors.colOnTooltip
|
||||||
|
text: sessionRoot.subtitle
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Process {
|
||||||
|
id: lock
|
||||||
|
command: ["bash", "-c", "loginctl lock-session"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: sleep
|
||||||
|
command: ["bash", "-c", "systemctl suspend || loginctl suspend"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: logout
|
||||||
|
command: ["bash", "-c", "loginctl terminate-session $XDG_SESSION_ID"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: hibernate
|
||||||
|
command: ["bash", "-c", "systemctl hibernate || loginctl hibernate"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: shutdown
|
||||||
|
command: ["bash", "-c", "systemctl poweroff || loginctl poweroff"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: reboot
|
||||||
|
command: ["bash", "-c", "systemctl reboot || loginctl reboot"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: firmwareReboot
|
||||||
|
command: ["bash", "-c", "systemctl reboot --firmware-setup || loginctl reboot --firmware-setup"]
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: taskManager
|
||||||
|
command: ["bash", "-c", "gnome-system-monitor & disown"]
|
||||||
|
}
|
||||||
|
|
||||||
|
IpcHandler {
|
||||||
|
target: "session"
|
||||||
|
|
||||||
|
function toggle(): void {
|
||||||
|
for (let i = 0; i < sessionVariants.instances.length; i++) {
|
||||||
|
let panelWindow = sessionVariants.instances[i];
|
||||||
|
if (panelWindow.modelData.name == Hyprland.focusedMonitor.name) {
|
||||||
|
panelWindow.visible = !panelWindow.visible;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function close(): void {
|
||||||
|
for (let i = 0; i < sessionVariants.instances.length; i++) {
|
||||||
|
let panelWindow = sessionVariants.instances[i];
|
||||||
|
if (panelWindow.modelData.name == Hyprland.focusedMonitor.name) {
|
||||||
|
panelWindow.visible = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function open(): void {
|
||||||
|
for (let i = 0; i < sessionVariants.instances.length; i++) {
|
||||||
|
let panelWindow = sessionVariants.instances[i];
|
||||||
|
if (panelWindow.modelData.name == Hyprland.focusedMonitor.name) {
|
||||||
|
panelWindow.visible = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,65 @@
|
|||||||
|
import "root:/modules/common"
|
||||||
|
import "root:/modules/common/widgets/"
|
||||||
|
import QtQuick
|
||||||
|
import QtQuick.Controls
|
||||||
|
import QtQuick.Layouts
|
||||||
|
import Quickshell
|
||||||
|
import Quickshell.Io
|
||||||
|
|
||||||
|
Button {
|
||||||
|
id: button
|
||||||
|
|
||||||
|
property string buttonIcon
|
||||||
|
property string buttonText
|
||||||
|
property bool keyboardDown: false
|
||||||
|
|
||||||
|
implicitHeight: 120
|
||||||
|
implicitWidth: 120
|
||||||
|
|
||||||
|
PointingHandInteraction {}
|
||||||
|
Keys.onPressed: (event) => {
|
||||||
|
if (event.key === Qt.Key_Return || event.key === Qt.Key_Enter) {
|
||||||
|
keyboardDown = true
|
||||||
|
button.clicked()
|
||||||
|
event.accepted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Keys.onReleased: (event) => {
|
||||||
|
if (event.key === Qt.Key_Return || event.key === Qt.Key_Enter) {
|
||||||
|
keyboardDown = false
|
||||||
|
event.accepted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onClicked: {
|
||||||
|
console.log("Button clicked:", buttonText)
|
||||||
|
}
|
||||||
|
|
||||||
|
background: Rectangle {
|
||||||
|
anchors.fill: parent
|
||||||
|
radius: Appearance.rounding.full
|
||||||
|
color: (button.down || button.keyboardDown) ? Appearance.colors.colLayer2Active : ((button.hovered || button.focus) ? Appearance.colors.colLayer2Hover : Appearance.colors.colLayer2)
|
||||||
|
|
||||||
|
Behavior on color {
|
||||||
|
ColorAnimation {
|
||||||
|
duration: Appearance.animation.elementDecel.duration
|
||||||
|
easing.type: Appearance.animation.elementDecel.type
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
contentItem: MaterialSymbol {
|
||||||
|
id: icon
|
||||||
|
anchors.fill: parent
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
text: buttonIcon
|
||||||
|
font.pixelSize: 40
|
||||||
|
}
|
||||||
|
|
||||||
|
StyledToolTip {
|
||||||
|
content: buttonText
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -78,7 +78,6 @@ Scope {
|
|||||||
Keys.onPressed: (event) => {
|
Keys.onPressed: (event) => {
|
||||||
if (event.key === Qt.Key_Escape) {
|
if (event.key === Qt.Key_Escape) {
|
||||||
sidebarRoot.visible = false;
|
sidebarRoot.visible = false;
|
||||||
event.accepted = true; // Prevent further propagation of the event
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -108,10 +107,23 @@ Scope {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
Layout.fillHeight: true
|
Layout.fillWidth: true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QuickToggleButton {
|
||||||
|
toggled: false
|
||||||
|
buttonIcon: "power_settings_new"
|
||||||
|
onClicked: {
|
||||||
|
openSessionMenu.running = true
|
||||||
|
}
|
||||||
|
Process {
|
||||||
|
id: openSessionMenu
|
||||||
|
command: ["qs", "ipc", "call", "session", "open"]
|
||||||
|
}
|
||||||
|
StyledToolTip {
|
||||||
|
content: "Session"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
import "./modules/bar/"
|
import "./modules/bar/"
|
||||||
import "./modules/onScreenDisplay/"
|
import "./modules/onScreenDisplay/"
|
||||||
import "./modules/screenCorners/"
|
import "./modules/screenCorners/"
|
||||||
|
import "./modules/session/"
|
||||||
import "./modules/sidebarRight/"
|
import "./modules/sidebarRight/"
|
||||||
import QtQuick
|
import QtQuick
|
||||||
import QtQuick.Controls
|
import QtQuick.Controls
|
||||||
@@ -12,10 +13,11 @@ import Quickshell
|
|||||||
|
|
||||||
ShellRoot {
|
ShellRoot {
|
||||||
Bar {}
|
Bar {}
|
||||||
SidebarRight {}
|
|
||||||
ScreenCorners {}
|
|
||||||
ReloadPopup {}
|
|
||||||
OnScreenDisplayBrightness {}
|
OnScreenDisplayBrightness {}
|
||||||
OnScreenDisplayVolume {}
|
OnScreenDisplayVolume {}
|
||||||
|
ReloadPopup {}
|
||||||
|
ScreenCorners {}
|
||||||
|
Session {}
|
||||||
|
SidebarRight {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user