From 37cc9423d739a501be45018554e3c9d0e104bb81 Mon Sep 17 00:00:00 2001 From: Nydragon Date: Fri, 27 Sep 2024 19:20:19 +0200 Subject: [PATCH] feat: add current playing song display --- flake.lock | 14 +++---- src/windows/audioman/AudioEntry.qml | 19 ++------- src/windows/audioman/AudioManager.qml | 54 +++++++++++++++++++++---- src/windows/audioman/OutputElem.qml | 12 ++++++ src/windows/audioman/OutputSelector.qml | 39 ++++++++++++++++++ 5 files changed, 108 insertions(+), 30 deletions(-) create mode 100644 src/windows/audioman/OutputElem.qml create mode 100644 src/windows/audioman/OutputSelector.qml diff --git a/flake.lock b/flake.lock index d702992..c6ba93f 100644 --- a/flake.lock +++ b/flake.lock @@ -2,11 +2,11 @@ "nodes": { "nixpkgs": { "locked": { - "lastModified": 1726755586, - "narHash": "sha256-PmUr/2GQGvFTIJ6/Tvsins7Q43KTMvMFhvG6oaYK+Wk=", + "lastModified": 1727122398, + "narHash": "sha256-o8VBeCWHBxGd4kVMceIayf5GApqTavJbTa44Xcg5Rrk=", "owner": "nixos", "repo": "nixpkgs", - "rev": "c04d5652cfa9742b1d519688f65d1bbccea9eb7e", + "rev": "30439d93eb8b19861ccbe3e581abf97bdc91b093", "type": "github" }, "original": { @@ -23,11 +23,11 @@ ] }, "locked": { - "lastModified": 1726641881, - "narHash": "sha256-ee2YIeFbYhm/gGs26grUW37BxqqDoFNElW1Vy5PIuHA=", + "lastModified": 1727391151, + "narHash": "sha256-PIEJfHzze4MWwaS7Py+az0IL7A+bZd4pFTE/uAy1b70=", "ref": "refs/heads/master", - "rev": "7f9762be5368ca33b84e7b2b3e23a626d432436d", - "revCount": 348, + "rev": "3ed39b2a798419a168e5c79a2db9f7ee20de70fa", + "revCount": 352, "type": "git", "url": "https://git.outfoxxed.me/outfoxxed/quickshell" }, diff --git a/src/windows/audioman/AudioEntry.qml b/src/windows/audioman/AudioEntry.qml index 03cf52f..ea53ae8 100644 --- a/src/windows/audioman/AudioEntry.qml +++ b/src/windows/audioman/AudioEntry.qml @@ -41,6 +41,7 @@ RowLayout { const app = root.node.isStream ? `[${root.node.properties["application.name"]}] ` : ""; return app + (root.node.properties["media.name"] ?? root.node.description); } + color: "white" // Cede space to other elements -> don't have stupidly long names destroying the layout Layout.maximumWidth: 0 } @@ -64,27 +65,13 @@ RowLayout { } 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" - - onClicked: makeDefault.running = true - - Process { - id: makeDefault - command: ["pactl", "set-default-sink", root.node?.name] - running: false - } - } } + RowLayout { Label { Layout.preferredWidth: 50 text: `${Math.floor(root.node.audio.volume * 100)}%` + color: "white" } Slider { diff --git a/src/windows/audioman/AudioManager.qml b/src/windows/audioman/AudioManager.qml index f58b527..f2a1d8c 100644 --- a/src/windows/audioman/AudioManager.qml +++ b/src/windows/audioman/AudioManager.qml @@ -3,41 +3,81 @@ import Quickshell.Services.Pipewire import QtQuick import QtQuick.Layouts import QtQuick.Controls +import Quickshell.Services.Mpris import "root:base" +import QtQuick.Effects PopupWindow { + id: audioman anchor { rect.x: lbar.width * 1.2 rect.y: lbar.width * 0.2 window: lbar } + //gravity: Edges.Bottom | Edges.Right + color: "transparent" width: 500 - height: 300 + height: 600 visible: false + property var player: Mpris.players.values.find(p => p.playbackState == MprisPlaybackState.Playing) ?? Mpris.players.values[0] + BRectangle { - anchors.fill: parent + id: bn + + //color: "transparent" + + Image { + id: background + //anchors.margins: parent.border.width + anchors.fill: parent + source: audioman.player?.trackArtUrl + Layout.alignment: Qt.AlignHCenter + visible: true + } + + MultiEffect { + autoPaddingEnabled: false + source: background + anchors.fill: background + blurEnabled: true + blurMax: 64 + blurMultiplier: 2 + blur: 1 + brightness: -0.15 + } ScrollView { + id: test anchors.fill: parent contentWidth: availableWidth ColumnLayout { + id: p // BUG: We access nodes before they are initialized anchors.fill: parent anchors.margins: 10 - Repeater { - model: Pipewire.nodes.values.filter(e => e.isSink) + ColumnLayout { + Layout.alignment: Qt.AlignHCenter + Label { + text: audioman.player.trackTitle + Layout.alignment: Qt.AlignHCenter + color: "white" + } - AudioEntry { - required property PwNode modelData - node: modelData + Image { + source: audioman.player.trackArtUrl + sourceSize.width: 300 + sourceSize.height: 300 + Layout.alignment: Qt.AlignHCenter } } + OutputSelector {} + Rectangle { height: 2 color: "black" diff --git a/src/windows/audioman/OutputElem.qml b/src/windows/audioman/OutputElem.qml new file mode 100644 index 0000000..34dd005 --- /dev/null +++ b/src/windows/audioman/OutputElem.qml @@ -0,0 +1,12 @@ +import QtQuick +import Quickshell.Services.Pipewire + +Item { + id: root + required property PwNode node + property string name: root.node.properties["media.name"] ?? root.node.description + + PwObjectTracker { + objects: [root.node] + } +} diff --git a/src/windows/audioman/OutputSelector.qml b/src/windows/audioman/OutputSelector.qml new file mode 100644 index 0000000..c7c236b --- /dev/null +++ b/src/windows/audioman/OutputSelector.qml @@ -0,0 +1,39 @@ +import QtQuick.Layouts +import QtQuick.Controls +import Quickshell.Io +import Quickshell.Services.Pipewire + +ColumnLayout { + ComboBox { + id: select + property var creator: Qt.createComponent("OutputElem.qml") + property var currentNode: select.model[select.currentIndex] + + Layout.fillWidth: true + textRole: "name" + model: Pipewire.nodes.values.filter(e => e.isSink && !e.isStream).map(m => { + return creator.createObject(select, { + node: m + }); + }) + + onActivated: i => { + Pipewire.preferredDefaultAudioSink = model[i].node; + //makeDefault.running = true; + } + + currentIndex: select.model.findIndex(e => e?.node?.id == Pipewire.defaultAudioSink?.id) + + Process { + id: makeDefault + command: ["pactl", "set-default-sink", select.model[select.currentIndex]?.node?.name] + running: false + } + } + + Slider { + Layout.fillWidth: true + value: select.model[select.currentIndex]?.node.audio.volume ?? 0 + onValueChanged: select.model[select.currentIndex].node.audio.volume = value + } +}