nysh/src/windows/audioman/AudioEntry.qml

88 lines
2.7 KiB
QML

import QtQuick
import QtQuick.Layouts
import QtQuick.Controls
import Quickshell.Services.Pipewire
RowLayout {
id: root
required property PwNode node
// bind the node so we can read its properties
PwObjectTracker {
objects: [root.node]
}
Image {
source: {
const getFallback = () => node.isStream ? "root:/../assets/folder-music.svg" : "root:/../assets/audio-volume-high.svg";
root.node.properties["application.icon-name"] ? `image://icon/${root.node.properties["application.icon-name"]}` : getFallback();
}
fillMode: Image.PreserveAspectFit
Layout.fillHeight: true
Layout.fillWidth: true
Layout.maximumWidth: 40
Layout.maximumHeight: parent.height
sourceSize.width: 40
sourceSize.height: 40
}
ColumnLayout {
RowLayout {
spacing: 6
Layout.preferredHeight: 30
Label {
id: name
text: {
const app = root.node.isStream ? `[${root.node.properties["application.name"]}] ` : "";
return app + (root.node.properties["media.name"] ?? root.node.description);
}
// Cede space to other elements -> don't have stupidly long names detroying the layout
Layout.maximumWidth: 0
}
Item {
// Padding to move the buttons to the right
Layout.fillWidth: true
}
Button {
visible: root.node.isSink
width: 10
checkable: true
Image {
source: node.audio.muted ? "root:/../assets/audio-volume-muted.svg" : "root:/../assets/audio-volume-high.svg"
height: parent.height * (2 / 3)
anchors.centerIn: parent
fillMode: Image.PreserveAspectFit
}
onClicked: root.node.audio.muted = !root.node.audio.muted
}
Button {
property bool isDefault: root.node.id === Pipewire.defaultAudioSink.id
checked: isDefault
checkable: false
visible: root.node.isSink
text: isDefault ? "default" : "not default"
}
}
RowLayout {
Label {
Layout.preferredWidth: 50
text: `${Math.floor(root.node.audio.volume * 100)}%`
}
Slider {
Layout.fillWidth: true
value: root.node.audio.volume
onValueChanged: root.node.audio.volume = value
}
}
}
}