summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
author2 * r + 2 * t <61896496+soramanew@users.noreply.github.com>2025-09-12 16:51:38 +1000
committer2 * r + 2 * t <61896496+soramanew@users.noreply.github.com>2025-09-12 16:51:38 +1000
commit83d3b4ebd514e8c2bccc92c191bebd72792bb1d4 (patch)
treee307c1f18f5980096acefe3993acb9691939d9c6
parentinteractions: fix interaction area (diff)
downloadcaelestia-shell-83d3b4ebd514e8c2bccc92c191bebd72792bb1d4.tar.gz
caelestia-shell-83d3b4ebd514e8c2bccc92c191bebd72792bb1d4.tar.bz2
caelestia-shell-83d3b4ebd514e8c2bccc92c191bebd72792bb1d4.zip
utilities: add keep awake and quick toggles
-rw-r--r--config/Config.qml2
-rw-r--r--config/UtilitiesConfig.qml11
-rw-r--r--modules/drawers/Panels.qml2
-rw-r--r--modules/utilities/Content.qml30
-rw-r--r--modules/utilities/Wrapper.qml15
-rw-r--r--modules/utilities/cards/IdleInhibit.qml122
-rw-r--r--modules/utilities/cards/Toggles.qml144
-rw-r--r--services/IdleInhibitor.qml7
8 files changed, 306 insertions, 27 deletions
diff --git a/config/Config.qml b/config/Config.qml
index b5218be..b23a4e2 100644
--- a/config/Config.qml
+++ b/config/Config.qml
@@ -20,6 +20,7 @@ Singleton {
property alias session: adapter.session
property alias winfo: adapter.winfo
property alias lock: adapter.lock
+ property alias utilities: adapter.utilities
property alias services: adapter.services
property alias paths: adapter.paths
@@ -44,6 +45,7 @@ Singleton {
property SessionConfig session: SessionConfig {}
property WInfoConfig winfo: WInfoConfig {}
property LockConfig lock: LockConfig {}
+ property UtilitiesConfig utilities: UtilitiesConfig {}
property ServiceConfig services: ServiceConfig {}
property UserPaths paths: UserPaths {}
}
diff --git a/config/UtilitiesConfig.qml b/config/UtilitiesConfig.qml
new file mode 100644
index 0000000..ca1ea0b
--- /dev/null
+++ b/config/UtilitiesConfig.qml
@@ -0,0 +1,11 @@
+import Quickshell.Io
+
+JsonObject {
+ property bool enabled: true
+
+ property Sizes sizes: Sizes {}
+
+ component Sizes: JsonObject {
+ property int width: 400
+ }
+}
diff --git a/modules/drawers/Panels.qml b/modules/drawers/Panels.qml
index 5d42d34..4fd8bdf 100644
--- a/modules/drawers/Panels.qml
+++ b/modules/drawers/Panels.qml
@@ -99,7 +99,7 @@ Item {
Utilities.Wrapper {
id: utilities
- visibility: root.visibilities.utilities
+ visibilities: root.visibilities
anchors.bottom: parent.bottom
anchors.right: parent.right
diff --git a/modules/utilities/Content.qml b/modules/utilities/Content.qml
index 49fdaa4..bcac9b8 100644
--- a/modules/utilities/Content.qml
+++ b/modules/utilities/Content.qml
@@ -1,31 +1,17 @@
-import qs.components
-import qs.services
+import "cards"
import qs.config
-import Quickshell
-import Quickshell.Widgets
-import QtQuick
+import QtQuick.Layouts
-Item {
+ColumnLayout {
id: root
- anchors.top: parent.top
- anchors.bottom: parent.bottom
- anchors.right: parent.right
+ required property var visibilities
- // implicitWidth: 300
- // implicitHeight: 100
+ spacing: Appearance.spacing.normal
- // Rectangle {
- // anchors.fill: parent
- // }
+ IdleInhibit {}
- Behavior on implicitHeight {
- Anim {}
- }
-
- component Anim: NumberAnimation {
- duration: Appearance.anim.durations.expressiveDefaultSpatial
- easing.type: Easing.BezierSpline
- easing.bezierCurve: Appearance.anim.curves.expressiveDefaultSpatial
+ Toggles {
+ visibilities: root.visibilities
}
}
diff --git a/modules/utilities/Wrapper.qml b/modules/utilities/Wrapper.qml
index 1da4c67..a5637a4 100644
--- a/modules/utilities/Wrapper.qml
+++ b/modules/utilities/Wrapper.qml
@@ -5,18 +5,18 @@ import QtQuick
Item {
id: root
- required property bool visibility
+ required property var visibilities
visible: height > 0
implicitHeight: 0
- implicitWidth: content.implicitWidth
+ implicitWidth: Config.utilities.sizes.width
states: State {
name: "visible"
- when: root.visibility
+ when: root.visibilities.utilities
PropertyChanges {
- root.implicitHeight: content.implicitHeight
+ root.implicitHeight: content.implicitHeight + Appearance.padding.large * 2
}
}
@@ -46,5 +46,12 @@ Item {
Content {
id: content
+
+ anchors.top: parent.top
+ anchors.left: parent.left
+ anchors.right: parent.right
+ anchors.margins: Appearance.padding.large
+
+ visibilities: root.visibilities
}
}
diff --git a/modules/utilities/cards/IdleInhibit.qml b/modules/utilities/cards/IdleInhibit.qml
new file mode 100644
index 0000000..87f6211
--- /dev/null
+++ b/modules/utilities/cards/IdleInhibit.qml
@@ -0,0 +1,122 @@
+import qs.components
+import qs.components.controls
+import qs.services
+import qs.config
+import QtQuick
+import QtQuick.Layouts
+
+StyledRect {
+ id: root
+
+ Layout.fillWidth: true
+ implicitHeight: layout.implicitHeight + (IdleInhibitor.enabled ? activeChip.implicitHeight + activeChip.anchors.topMargin : 0) + Appearance.padding.large * 2
+
+ radius: Appearance.rounding.normal
+ color: Colours.palette.m3surfaceContainer
+ clip: true
+
+ RowLayout {
+ id: layout
+
+ anchors.top: parent.top
+ anchors.left: parent.left
+ anchors.right: parent.right
+ anchors.margins: Appearance.padding.large
+ spacing: Appearance.spacing.normal
+
+ StyledRect {
+ implicitWidth: implicitHeight
+ implicitHeight: icon.implicitHeight + Appearance.padding.smaller * 2
+
+ radius: Appearance.rounding.full
+ color: IdleInhibitor.enabled ? Colours.palette.m3secondary : Colours.palette.m3secondaryContainer
+
+ MaterialIcon {
+ id: icon
+
+ anchors.centerIn: parent
+ text: "coffee"
+ color: IdleInhibitor.enabled ? Colours.palette.m3onSecondary : Colours.palette.m3onSecondaryContainer
+ font.pointSize: Appearance.font.size.large
+ }
+ }
+
+ ColumnLayout {
+ Layout.fillWidth: true
+ spacing: 0
+
+ StyledText {
+ Layout.fillWidth: true
+ text: qsTr("Keep Awake")
+ font.pointSize: Appearance.font.size.normal
+ elide: Text.ElideRight
+ }
+
+ StyledText {
+ Layout.fillWidth: true
+ text: IdleInhibitor.enabled ? qsTr("Preventing sleep mode") : qsTr("Normal power management")
+ color: Colours.palette.m3onSurfaceVariant
+ font.pointSize: Appearance.font.size.small
+ elide: Text.ElideRight
+ }
+ }
+
+ StyledSwitch {
+ checked: IdleInhibitor.enabled
+ onToggled: IdleInhibitor.enabled = checked
+ }
+ }
+
+ Loader {
+ id: activeChip
+
+ anchors.bottom: parent.bottom
+ anchors.left: parent.left
+ anchors.topMargin: Appearance.spacing.larger
+ anchors.bottomMargin: IdleInhibitor.enabled ? Appearance.padding.large : -implicitHeight
+ anchors.leftMargin: Appearance.padding.large
+
+ opacity: IdleInhibitor.enabled ? 1 : 0
+ scale: IdleInhibitor.enabled ? 1 : 0.5
+
+ Component.onCompleted: active = Qt.binding(() => opacity > 0)
+
+ sourceComponent: StyledRect {
+ implicitWidth: activeText.implicitWidth + Appearance.padding.normal * 2
+ implicitHeight: activeText.implicitHeight + Appearance.padding.small * 2
+
+ radius: Appearance.rounding.full
+ color: Colours.palette.m3primary
+
+ StyledText {
+ id: activeText
+
+ anchors.centerIn: parent
+ text: qsTr("Active since %1").arg(Qt.formatTime(IdleInhibitor.enabledSince, Config.services.useTwelveHourClock ? "hh:mm a" : "hh:mm"))
+ color: Colours.palette.m3onPrimary
+ font.pointSize: Math.round(Appearance.font.size.small * 0.9)
+ }
+ }
+
+ Behavior on anchors.bottomMargin {
+ Anim {
+ duration: Appearance.anim.durations.expressiveDefaultSpatial
+ easing.bezierCurve: Appearance.anim.curves.expressiveDefaultSpatial
+ }
+ }
+
+ Behavior on opacity {
+ Anim {
+ duration: Appearance.anim.durations.small
+ }
+ }
+
+ Behavior on scale {
+ Anim {}
+ }
+ }
+
+ Behavior on implicitHeight {
+ Anim {}
+ }
+}
diff --git a/modules/utilities/cards/Toggles.qml b/modules/utilities/cards/Toggles.qml
new file mode 100644
index 0000000..8e4dfb5
--- /dev/null
+++ b/modules/utilities/cards/Toggles.qml
@@ -0,0 +1,144 @@
+import qs.components
+import qs.services
+import qs.config
+import qs.modules.controlcenter
+import Quickshell
+import Quickshell.Bluetooth
+import QtQuick
+import QtQuick.Layouts
+
+StyledRect {
+ id: root
+
+ required property var visibilities
+
+ Layout.fillWidth: true
+ implicitHeight: layout.implicitHeight + Appearance.padding.large * 2
+
+ radius: Appearance.rounding.normal
+ color: Colours.palette.m3surfaceContainer
+
+ GridLayout {
+ id: layout
+
+ anchors.fill: parent
+ anchors.margins: Appearance.padding.large
+
+ columns: 2
+ rowSpacing: Appearance.spacing.normal
+ columnSpacing: Appearance.spacing.normal
+ uniformCellWidths: true
+
+ StyledText {
+ Layout.columnSpan: 2
+ text: qsTr("Quick Toggles")
+ font.pointSize: Appearance.font.size.normal
+ }
+
+ Toggle {
+ icon: "wifi"
+ text: qsTr("WiFi")
+ checked: Network.wifiEnabled
+
+ function onClicked(): void {
+ Network.toggleWifi();
+ }
+ }
+
+ Toggle {
+ icon: "bluetooth"
+ text: qsTr("Bluetooth")
+ checked: Bluetooth.defaultAdapter?.enabled ?? false
+
+ function onClicked(): void {
+ const adapter = Bluetooth.defaultAdapter;
+ if (adapter)
+ adapter.enabled = !adapter.enabled;
+ }
+ }
+
+ Toggle {
+ icon: "mic"
+ text: qsTr("Microphone")
+ checked: !Audio.sourceMuted
+
+ function onClicked(): void {
+ const audio = Audio.source?.audio;
+ if (audio)
+ audio.muted = !audio.muted;
+ }
+ }
+
+ Toggle {
+ icon: "settings"
+ text: qsTr("Settings")
+ toggle: false
+
+ function onClicked(): void {
+ root.visibilities.utilities = false;
+ WindowFactory.create(null, {
+ screen: QsWindow.window?.screen ?? null
+ });
+ }
+ }
+ }
+
+ component Toggle: StyledRect {
+ id: toggle
+
+ required property string icon
+ required property string text
+ property bool checked
+ property bool toggle
+ property bool internalChecked
+
+ function onClicked(): void {
+ }
+
+ onCheckedChanged: internalChecked = checked
+
+ radius: internalChecked ? Appearance.rounding.small : implicitHeight / 2
+ color: internalChecked ? Colours.palette.m3primary : Colours.palette.m3surfaceContainerHigh
+
+ Layout.fillWidth: true
+ implicitWidth: label.implicitWidth + Appearance.padding.larger * 2
+ implicitHeight: label.implicitHeight + Appearance.padding.smaller * 2
+
+ StateLayer {
+ color: toggle.internalChecked ? Colours.palette.m3onPrimary : Colours.palette.m3onSurface
+
+ function onClicked(): void {
+ if (toggle.toggle)
+ toggle.internalChecked = !toggle.internalChecked;
+ toggle.onClicked();
+ }
+ }
+
+ RowLayout {
+ id: label
+
+ anchors.centerIn: parent
+ spacing: Appearance.spacing.small
+
+ MaterialIcon {
+ text: toggle.icon
+ color: toggle.internalChecked ? Colours.palette.m3onPrimary : Colours.palette.m3onSurface
+ fill: toggle.internalChecked ? 1 : 0
+
+ Behavior on fill {
+ Anim {}
+ }
+ }
+
+ StyledText {
+ text: toggle.text
+ color: toggle.internalChecked ? Colours.palette.m3onPrimary : Colours.palette.m3onSurface
+ font.pointSize: Appearance.font.size.small
+ }
+ }
+
+ Behavior on radius {
+ Anim {}
+ }
+ }
+}
diff --git a/services/IdleInhibitor.qml b/services/IdleInhibitor.qml
index 0e1c835..73951e9 100644
--- a/services/IdleInhibitor.qml
+++ b/services/IdleInhibitor.qml
@@ -7,11 +7,18 @@ Singleton {
id: root
property alias enabled: props.enabled
+ readonly property alias enabledSince: props.enabledSince
+
+ onEnabledChanged: {
+ if (enabled)
+ props.enabledSince = new Date();
+ }
PersistentProperties {
id: props
property bool enabled
+ property date enabledSince
reloadableId: "idleInhibitor"
}