mirror of
https://github.com/noctalia-dev/noctalia-shell.git
synced 2026-05-11 17:08:27 +08:00
218 lines
5.6 KiB
QML
218 lines
5.6 KiB
QML
import QtQuick
|
|
import QtQuick.Controls
|
|
import QtQuick.Layouts
|
|
import Quickshell
|
|
import Quickshell.Wayland
|
|
import qs.Commons
|
|
import qs.Services
|
|
import qs.Widgets
|
|
|
|
Loader {
|
|
id: windowLoader
|
|
active: false
|
|
|
|
readonly property real currentVolume: AudioService.volume
|
|
readonly property bool isMuted: AudioService.muted
|
|
readonly property real scaling: ScalingService.getScreenScale(Quickshell.screens[0])
|
|
|
|
// Used to avoid showing OSD on Quickshell startup
|
|
property bool firstVolumeReceived: false
|
|
property bool firstMuteReceived: false
|
|
|
|
function getIcon() {
|
|
if (AudioService.muted) {
|
|
return "volume-mute"
|
|
}
|
|
return (AudioService.volume <= Number.EPSILON) ? "volume-zero" : (AudioService.volume <= 0.5) ? "volume-low" : "volume-high"
|
|
}
|
|
|
|
sourceComponent: PanelWindow {
|
|
id: panel
|
|
|
|
screen: Quickshell.screens[0] // Use primary screen
|
|
|
|
anchors {
|
|
top: true
|
|
}
|
|
|
|
implicitWidth: 320 * windowLoader.scaling
|
|
implicitHeight: osdItem.height
|
|
|
|
// Set margins based on bar position
|
|
margins.top: {
|
|
switch (Settings.data.bar.position) {
|
|
case "top":
|
|
return (Style.barHeight + Style.marginS) * windowLoader.scaling + (Settings.data.bar.floating ? Settings.data.bar.marginVertical * Style.marginXL * windowLoader.scaling : 0)
|
|
default:
|
|
return Style.marginL * windowLoader.scaling
|
|
}
|
|
}
|
|
|
|
color: Color.transparent
|
|
|
|
WlrLayershell.keyboardFocus: WlrKeyboardFocus.None
|
|
exclusionMode: PanelWindow.ExclusionMode.Ignore
|
|
|
|
Rectangle {
|
|
id: osdItem
|
|
|
|
width: parent.width
|
|
height: Math.round(contentLayout.implicitHeight + Style.marginL * 2 * windowLoader.scaling)
|
|
radius: Style.radiusL * windowLoader.scaling
|
|
color: Color.mSurface
|
|
border.color: Color.mOutline
|
|
border.width: Math.max(2, Style.borderM * windowLoader.scaling)
|
|
visible: false
|
|
opacity: 0
|
|
scale: 0.7
|
|
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
|
|
Behavior on opacity {
|
|
NumberAnimation {
|
|
duration: Style.animationFast
|
|
easing.type: Easing.OutCubic
|
|
}
|
|
}
|
|
|
|
Behavior on scale {
|
|
NumberAnimation {
|
|
duration: Style.animationFast
|
|
easing.type: Easing.OutCubic
|
|
}
|
|
}
|
|
|
|
Timer {
|
|
id: hideTimer
|
|
interval: 2000
|
|
onTriggered: osdItem.hide()
|
|
}
|
|
|
|
RowLayout {
|
|
id: contentLayout
|
|
anchors.fill: parent
|
|
anchors.margins: Style.marginM * windowLoader.scaling
|
|
spacing: Style.marginM * windowLoader.scaling
|
|
|
|
NIcon {
|
|
icon: windowLoader.getIcon()
|
|
color: windowLoader.isMuted ? Color.mError : Color.mOnSurface
|
|
font.pointSize: Style.fontSizeXL * windowLoader.scaling
|
|
Layout.alignment: Qt.AlignVCenter
|
|
}
|
|
|
|
RowLayout {
|
|
Layout.fillWidth: true
|
|
Layout.alignment: Qt.AlignVCenter
|
|
spacing: Style.marginXS * windowLoader.scaling
|
|
|
|
Rectangle {
|
|
Layout.fillWidth: true
|
|
Layout.preferredHeight: Math.round(6 * windowLoader.scaling)
|
|
radius: Math.round(3 * windowLoader.scaling)
|
|
color: Color.mSurfaceVariant
|
|
|
|
Rectangle {
|
|
anchors.left: parent.left
|
|
anchors.top: parent.top
|
|
anchors.bottom: parent.bottom
|
|
width: parent.width * (windowLoader.isMuted ? 0 : Math.min(1.0, windowLoader.currentVolume))
|
|
radius: parent.radius
|
|
color: windowLoader.isMuted ? Color.mError : Color.mPrimary
|
|
|
|
Behavior on width {
|
|
NumberAnimation {
|
|
duration: Style.animationFast
|
|
easing.type: Easing.OutCubic
|
|
}
|
|
}
|
|
|
|
Behavior on color {
|
|
ColorAnimation {
|
|
duration: Style.animationFast
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
NText {
|
|
text: windowLoader.isMuted ? "0%" : Math.round(windowLoader.currentVolume * 100) + "%"
|
|
color: Color.mOnSurfaceVariant
|
|
font.pointSize: Style.fontSizeS * windowLoader.scaling
|
|
Layout.alignment: Qt.AlignVCenter
|
|
Layout.minimumWidth: Math.round(32 * windowLoader.scaling)
|
|
}
|
|
}
|
|
}
|
|
|
|
function show() {
|
|
hideTimer.stop()
|
|
osdItem.visible = true
|
|
osdItem.opacity = 1
|
|
osdItem.scale = 1.0
|
|
hideTimer.start()
|
|
}
|
|
|
|
function hide() {
|
|
hideTimer.stop()
|
|
osdItem.opacity = 0
|
|
osdItem.scale = 0.7
|
|
|
|
Qt.callLater(function () {
|
|
osdItem.visible = false
|
|
windowLoader.active = false
|
|
})
|
|
}
|
|
}
|
|
|
|
function showOSD() {
|
|
osdItem.show()
|
|
}
|
|
}
|
|
|
|
// Monitor volume changes
|
|
Connections {
|
|
target: AudioService
|
|
|
|
function onVolumeChanged() {
|
|
if (!firstVolumeReceived) {
|
|
// Ignore the first volume change on startup
|
|
firstVolumeReceived = true
|
|
} else {
|
|
showOSD()
|
|
}
|
|
}
|
|
|
|
function onMutedChanged() {
|
|
if (!firstMuteReceived) {
|
|
// Ignore the first mute state change on startup
|
|
firstMuteReceived = true
|
|
} else {
|
|
showOSD()
|
|
}
|
|
}
|
|
}
|
|
|
|
// Signal to coordinate with other OSDs
|
|
signal osdShowing
|
|
|
|
function showOSD() {
|
|
// Check if OSD is enabled in settings
|
|
if (!Settings.data.general.showOSD) {
|
|
return
|
|
}
|
|
|
|
osdShowing() // Notify other OSDs to hide
|
|
windowLoader.active = true
|
|
if (windowLoader.item) {
|
|
windowLoader.item.showOSD()
|
|
}
|
|
}
|
|
|
|
function hideOSD() {
|
|
if (windowLoader.item) {
|
|
windowLoader.item.osdItem.hideImmediately()
|
|
}
|
|
}
|
|
}
|