feat: display multiple mpris players

This commit is contained in:
Nydragon 2024-11-16 03:16:26 +01:00
parent 5e5f265de6
commit 0ce0c58162
Signed by: nydragon
SSH key fingerprint: SHA256:WcjW5NJPQ8Dx4uQDmoIlVPLWE27Od3fxoe0IUvuoPHE

View file

@ -1,3 +1,5 @@
pragma ComponentBehavior: Bound
import QtQuick.Effects import QtQuick.Effects
import QtQuick.Controls import QtQuick.Controls
import QtQuick.Layouts import QtQuick.Layouts
@ -11,123 +13,141 @@ BRectangle {
id: mprisSmall id: mprisSmall
Layout.fillWidth: true Layout.fillWidth: true
Layout.preferredHeight: 200 Layout.preferredHeight: 200
radius: 15 radius: 15
clip: true
border.color: "transparent"
visible: list.count
visible: Player.current ?? false ListView {
id: list
BlurredImage {
source: Player.current?.trackArtUrl ?? ""
anchors.fill: parent anchors.fill: parent
radius: parent.radius model: Mpris.players
} orientation: Qt.Horizontal
snapMode: ListView.SnapOneItem
spacing: 10
delegate: Item {
id: card
required property var modelData
property MprisPlayer player: modelData
width: mprisSmall.width
height: mprisSmall.height
RowLayout { BlurredImage {
anchors.fill: parent source: card.player?.trackArtUrl ?? ""
clip: true anchors.fill: parent
BRoundedImage { radius: 15
id: im
color: "transparent"
visible: false
source: Player.current?.trackArtUrl ?? ""
radius: 15
}
MultiEffect {
id: effect
source: im
autoPaddingEnabled: true
shadowBlur: 1.0
shadowColor: 'black'
shadowEnabled: true
Layout.margins: 20
Layout.preferredWidth: parent.height - (Layout.margins * 2)
Layout.preferredHeight: parent.height - (Layout.margins * 2)
Layout.maximumWidth: {
const mWidth = parent.width - (Layout.margins * 2);
return mWidth > 0 ? mWidth : 0;
}
Layout.fillHeight: true
}
ColumnLayout {
Layout.maximumWidth: parent.width / 2
Layout.fillWidth: true
clip: true
Text {
text: Player.current?.trackTitle ?? "Unknown Track"
color: "white"
Layout.alignment: Qt.AlignCenter
Layout.maximumWidth: parent.width
elide: Text.ElideRight
}
Text {
text: Player.current?.trackAlbum ?? "Unknown Album"
color: "white"
Layout.alignment: Qt.AlignCenter
Layout.maximumWidth: parent.width
elide: Text.ElideRight
}
Text {
text: Player.current?.trackAlbumArtist ?? "Unknown Artist"
color: "white"
Layout.alignment: Qt.AlignCenter
Layout.maximumWidth: parent.width
elide: Text.ElideRight
} }
RowLayout { RowLayout {
Layout.alignment: Qt.AlignCenter anchors.fill: parent
BIconButton { clip: true
source: Quickshell.iconPath("media-seek-backward") BRoundedImage {
onClicked: Player.current.previous() id: im
size: 20 color: "transparent"
} visible: false
BIconButton { source: card.player?.trackArtUrl ?? ""
source: Quickshell.iconPath(Player.isPlaying ? "media-playback-pause" : "media-playback-start") radius: 15
onClicked: Player.current.togglePlaying()
size: 20
} }
BIconButton { MultiEffect {
source: Quickshell.iconPath("media-seek-forward") id: effect
onClicked: Player.current.next() source: im
size: 20 autoPaddingEnabled: true
} shadowBlur: 1.0
} shadowColor: 'black'
shadowEnabled: true
Slider { Layout.margins: 20
id: slider Layout.preferredWidth: parent.height - (Layout.margins * 2)
Layout.fillWidth: true Layout.preferredHeight: parent.height - (Layout.margins * 2)
Layout.minimumWidth: 10 Layout.maximumWidth: {
Layout.minimumHeight: 3 const mWidth = parent.width - (Layout.margins * 2);
from: 0 return mWidth > 0 ? mWidth : 0;
to: Player.current?.length ?? 0 }
value: Player.current?.position ?? 0 Layout.fillHeight: true
enabled: (Player.current?.canSeek && Player.current?.positionSupported) ?? false
onMoved: {
if (Player.current)
Player.current.position = value;
} }
Component.onCompleted: { ColumnLayout {
const con = () => mprisSmall.player?.positionChanged.connect(() => { Layout.maximumWidth: parent.width / 2
slider.value = Player.current?.position; Layout.fillWidth: true
}); clip: true
con();
Player.currentChanged.connect(() => {
con();
});
}
FrameAnimation { Text {
// only emit the signal when the position is actually changing. text: card.player?.trackTitle ?? "Unknown Track"
running: Player.current?.playbackState == MprisPlaybackState.Playing color: "white"
// emit the positionChanged signal every frame. Layout.alignment: Qt.AlignCenter
onTriggered: Player.current?.positionChanged() Layout.maximumWidth: parent.width
elide: Text.ElideRight
}
Text {
text: card.player?.trackAlbum ?? "Unknown Album"
color: "white"
Layout.alignment: Qt.AlignCenter
Layout.maximumWidth: parent.width
elide: Text.ElideRight
}
Text {
text: card.player?.trackAlbumArtist ?? "Unknown Artist"
color: "white"
Layout.alignment: Qt.AlignCenter
Layout.maximumWidth: parent.width
elide: Text.ElideRight
}
RowLayout {
Layout.alignment: Qt.AlignCenter
BIconButton {
source: Quickshell.iconPath("media-seek-backward")
onClicked: card.player?.previous()
size: 20
}
BIconButton {
source: Quickshell.iconPath(card.player.playbackState === MprisPlaybackState.Playing ? "media-playback-pause" : "media-playback-start")
onClicked: card.player?.togglePlaying()
size: 20
}
BIconButton {
source: Quickshell.iconPath("media-seek-forward")
onClicked: card.player?.next()
size: 20
}
}
Slider {
id: slider
Layout.fillWidth: true
Layout.minimumWidth: 10
Layout.minimumHeight: 3
from: 0
to: card.player?.length ?? 0
value: card.player?.position ?? 0
enabled: (card.player?.canSeek && card.player?.positionSupported) ?? false
onMoved: {
if (card.player)
card.player.position = value;
}
Component.onCompleted: {
const con = () => card.player?.positionChanged.connect(() => {
slider.value = card.player?.position;
});
con();
Player.currentChanged.connect(() => {
con();
});
}
FrameAnimation {
// only emit the signal when the position is actually changing.
running: card.player?.playbackState == MprisPlaybackState.Playing
// emit the positionChanged signal every frame.
onTriggered: card.player?.positionChanged()
}
}
} }
} }
} }