feat: add wip notifications to dash

This commit is contained in:
Nydragon 2024-11-15 03:08:33 +01:00
parent 4e9a6eaa4a
commit e1a3e47f49
Signed by: nydragon
SSH key fingerprint: SHA256:WcjW5NJPQ8Dx4uQDmoIlVPLWE27Od3fxoe0IUvuoPHE
11 changed files with 182 additions and 104 deletions

View file

@ -1,9 +1,11 @@
import Quickshell import Quickshell
import Quickshell.Services.Notifications
import QtQuick import QtQuick
import QtQuick.Layouts import QtQuick.Layouts
import QtQml import QtQml
import "base" import "base"
import "widgets/mpris" import "widgets/mpris"
import "widgets/notifications"
import "provider" import "provider"
PanelWindow { PanelWindow {
@ -47,23 +49,6 @@ PanelWindow {
Component.onCompleted: () => maxSize = homeWindow.screen.width * (2 / 7) Component.onCompleted: () => maxSize = homeWindow.screen.width * (2 / 7)
ListView {
width: parent.width
height: parent.height
model: Notifications.incoming
delegate: Rectangle {
required property var modelData
width: 100
height: 50
Text {
text: parent.modelData.appName
width: parent.width
height: parent.height
}
}
}
Behavior on width { Behavior on width {
PropertyAnimation { PropertyAnimation {
id: anim id: anim
@ -89,6 +74,50 @@ PanelWindow {
Layout.margins: 15 Layout.margins: 15
Layout.alignment: Qt.AlignBottom Layout.alignment: Qt.AlignBottom
ListView {
id: popupcol
Layout.fillHeight: true
Layout.fillWidth: true
Layout.preferredHeight: 1000
spacing: 10
width: parent.width
model: ListModel {
id: list
Component.onCompleted: () => {
Notifications._.notification.connect(e => {
list.append({
notif: e
});
});
}
}
delegate: NotificationToast {
id: toast
required property int index
width: ListView.view.width
Component.onCompleted: {
toast.notif.closed.connect(e => {
if (!toast)
return;
list.remove(toast.index, 1);
});
toast.close.connect(() => {
const notif = toast.notif;
list.remove(toast.index, 1);
notif.dismiss();
});
}
}
}
MprisSmall {} MprisSmall {}
} }
} }

View file

@ -7,6 +7,7 @@ import "widgets/caffeine"
import "windows/notificationtoast" import "windows/notificationtoast"
import "windows/workspace-view" import "windows/workspace-view"
import "base" import "base"
import "provider"
import Quickshell // for ShellRoot and PanelWindow import Quickshell // for ShellRoot and PanelWindow
import QtQuick import QtQuick
import QtQuick.Layouts import QtQuick.Layouts
@ -18,9 +19,11 @@ PanelWindow {
anchors { anchors {
top: true top: true
left: true left: Config.alignment === Config.BarAlignment.Left
right: Config.alignment === Config.BarAlignment.Right
bottom: true bottom: true
} }
margins.left: 2 margins.left: 2
margins.top: 2 margins.top: 2
margins.bottom: 2 margins.bottom: 2
@ -68,6 +71,7 @@ PanelWindow {
Layout.fillHeight: true Layout.fillHeight: true
} }
} }
MouseArea { MouseArea {
id: mouse id: mouse
onClicked: lbar.root.enabled = !lbar.root.enabled onClicked: lbar.root.enabled = !lbar.root.enabled

View file

@ -1,4 +1,5 @@
import QtQuick import QtQuick
import "../provider"
Rectangle { Rectangle {
width: parent.width width: parent.width
@ -6,5 +7,5 @@ Rectangle {
border.color: "black" border.color: "black"
border.width: 1 border.width: 1
radius: 5 radius: 5
color: "#BD93F9" color: Config.colours.main
} }

22
src/provider/Config.qml Normal file
View file

@ -0,0 +1,22 @@
pragma Singleton
import QtQuick
import Quickshell
Singleton {
id: config
property Item notifications: Item {
property int toastDuration: 5000
}
enum BarAlignment {
Left,
Right
}
property int alignment: Config.BarAlignment.Left
property Item colours: Item {
property color main: "#BD93F9"
}
}

View file

@ -3,6 +3,7 @@ pragma Singleton
import Quickshell.Services.Notifications import Quickshell.Services.Notifications
import Quickshell import Quickshell
import QtQuick import QtQuick
import "../utils/timer.mjs" as Timer
Singleton { Singleton {
id: notif id: notif
@ -18,10 +19,15 @@ Singleton {
onNotification: n => { onNotification: n => {
n.tracked = true; n.tracked = true;
incoming.push(n);
notif.incomingAdded(n);
Timer.after(1000, notif, () => {
notif.incomingRemoved(n.id);
});
} }
} }
property list<Notification> backlog: notif._.trackedNotifications signal incomingRemoved(id: int)
property list<Notification> incoming: [] signal incomingAdded(id: Notification)
} }

View file

@ -1,4 +1,5 @@
module Provider module Provider
singleton Config 0.1 Config.qml
singleton Player 0.1 Player.qml singleton Player 0.1 Player.qml
singleton Inhibitor 0.1 Inhibitor.qml singleton Inhibitor 0.1 Inhibitor.qml
singleton Notifications 0.1 Notifications.qml singleton Notifications 0.1 Notifications.qml

2
src/utils/qmldir Normal file
View file

@ -0,0 +1,2 @@
module Utils
Timer 0.1 timer.mjs

13
src/utils/timer.mjs Normal file
View file

@ -0,0 +1,13 @@
export function Timer(parent) {
return Qt.createQmlObject("import QtQuick; Timer {}", parent);
}
export function after(delay, parent, callback) {
const timer = new Timer(parent);
timer.interval = delay;
timer.triggered.connect(callback);
timer.start();
}

View file

@ -1,40 +1,30 @@
pragma ComponentBehavior: Bound pragma ComponentBehavior: Bound
import QtQuick import QtQuick
import QtQuick.Controls
import QtQuick.Layouts import QtQuick.Layouts
import QtQuick.Controls
import Quickshell import Quickshell
import Quickshell.Widgets import Quickshell.Widgets
import "../../base" import "../../base"
import "../../provider"
import Quickshell.Services.Notifications import Quickshell.Services.Notifications
MouseArea { MouseArea {
id: toast id: toast
readonly property int lifetime: 5000
property int countdownTime: lifetime
required property string appName required property Notification notif
required property string summary property int actionHeight: 30
required property string body property int expansionSpeed: 200
required property string appIcon property bool showTimeBar: false
required property string image
required property NotificationUrgency urgency
required property bool hasActionIcons
required property var actions
required property int index
Component.onCompleted: () => console.log(toast.actions.values)
function close(): void {
popupcol.model.remove(toast.index, 1);
}
hoverEnabled: true
height: box.height height: box.height
width: popupcol.width hoverEnabled: true
signal close
BRectangle { BRectangle {
id: box id: box
width: parent.width width: parent.width
height: header.height + actions.height + test.height + (5 * 3) height: header.height + toast.actionHeight + bodyBox.height + (5 * 3)
clip: true clip: true
@ -49,32 +39,28 @@ MouseArea {
height: 25 height: 25
IconImage { IconImage {
source: toast.appIcon ? Quickshell.iconPath(toast.appIcon) : "" source: toast.notif.appIcon ? Quickshell.iconPath(toast.notif.appIcon) : ""
height: parent.height height: parent.height
width: height width: height
visible: toast.appIcon visible: toast.notif.appIcon
} }
Text { Text {
text: (toast.appIcon ? " " : toast.appName + ": ") + toast.summary text: `${toast.notif.appIcon ? "" : `${toast.notif.appName}:`} ${toast.notif.summary}`
Layout.fillWidth: true Layout.fillWidth: true
elide: Text.ElideRight elide: Text.ElideRight
font.pointSize: 12.5 font.pointSize: 12.5
} }
Item { Button {
Layout.fillHeight: true onClicked: toast.close()
Layout.rightMargin: 16 height: 16
Button { width: 16
onClicked: toast.close()
height: 16
width: 16
}
} }
} }
Rectangle { Rectangle {
id: test id: bodyBox
width: parent.width width: parent.width
anchors.top: header.bottom anchors.top: header.bottom
height: 60 height: 60
@ -85,17 +71,19 @@ MouseArea {
Text { Text {
id: text id: text
anchors.topMargin: 5 anchors.topMargin: 5
text: toast.body text: toast.notif.body
width: parent.width width: parent.width
height: parent.height height: parent.height
wrapMode: Text.Wrap wrapMode: Text.Wrap
elide: Text.ElideRight elide: Text.ElideRight
font.pointSize: 12.5 font.pointSize: 12.5
Component.onCompleted: () => { Component.onCompleted: () => {
if (text.implicitHeight < test.height) { if (text.implicitHeight < bodyBox.height) {
test.height = text.implicitHeight; bodyBox.height = text.implicitHeight;
} }
test.maxHeight = text.implicitHeight;
bodyBox.maxHeight = Qt.binding(() => text.implicitHeight);
} }
} }
@ -103,62 +91,51 @@ MouseArea {
name: "expand" name: "expand"
when: toast.containsMouse when: toast.containsMouse
PropertyChanges { PropertyChanges {
target: test target: bodyBox
height: test.maxHeight height: bodyBox.maxHeight
} }
} }
transitions: Transition { transitions: Transition {
NumberAnimation { NumberAnimation {
properties: "width,height" properties: "height"
duration: 50 duration: toast.expansionSpeed
easing.type: Easing.InOutQuad
} }
} }
} }
RowLayout { RowLayout {
id: actions id: actions
width: parent.width width: parent.width
anchors.top: test.bottom anchors.top: bodyBox.bottom
anchors.topMargin: 5 anchors.topMargin: 5
anchors.bottomMargin: 5 anchors.bottomMargin: 5
Repeater { Repeater {
model: toast.actions id: rep
model: toast.notif.actions
delegate: ToastAction { delegate: NotificationToastAction {
required property var modelData required property var modelData
notifAction: modelData notifAction: modelData
hasIcons: toast.hasActionIcons hasIcons: toast.notif.hasActionIcons
height: toast.actionHeight
} }
} }
visible: toast?.actions ? true : false visible: toast?.notif.actions ? true : false
}
states: State {
name: "expand"
when: toast.containsMouse
PropertyChanges {
target: box
height: test.height + header.height + actions.height + 15
}
}
transitions: Transition {
NumberAnimation {
properties: "width,height"
duration: 50
easing.type: Easing.InOutQuad
}
} }
} }
NumberAnimation on width {
duration: toast.expansionSpeed
}
Rectangle { Rectangle {
id: timeBar
visible: toast.showTimeBar
anchors.margins: 2 anchors.margins: 2
anchors.bottom: box.bottom anchors.bottom: box.bottom
anchors.right: box.right anchors.right: box.right
width: (box.width - box.border.width - anchors.margins) * (toast.countdownTime / toast.lifetime) width: box.width - box.border.width - anchors.margins
height: 2 height: 2
bottomLeftRadius: box.radius bottomLeftRadius: box.radius
bottomRightRadius: box.radius bottomRightRadius: box.radius
@ -174,21 +151,13 @@ MouseArea {
return "white"; return "white";
} }
} }
}
}
Timer { NumberAnimation on width {
id: timer to: 0
interval: 10 duration: Config.notifications.toastDuration
repeat: !toast.containsMouse paused: toast.containsMouse && timeBar.visible
onTriggered: () => { running: timeBar.visible
toast.countdownTime -= timer.interval;
if (toast.countdownTime <= 0) {
toast.parent.parent.model.remove(toast.index, 1);
timer.repeat = false;
timer.running = false;
} }
} }
running: !toast.containsMouse
} }
} }

View file

@ -3,6 +3,7 @@ import Quickshell
import QtQuick.Controls import QtQuick.Controls
import "root:provider" import "root:provider"
import "root:base" import "root:base"
import "../../widgets/notifications"
PanelWindow { PanelWindow {
id: popups id: popups
@ -36,11 +37,15 @@ PanelWindow {
anchors.margins: lbar.width * 0.2 anchors.margins: lbar.width * 0.2
anchors.fill: parent anchors.fill: parent
focus: true focus: true
spacing: 10
model: ListModel { model: ListModel {
id: data id: data
Component.onCompleted: () => { Component.onCompleted: () => {
Notifications._.notification.connect(e => { Notifications.incomingAdded.connect(n => {
data.insert(0, e); data.insert(0, {
notif: n
});
}); });
} }
} }
@ -80,8 +85,34 @@ PanelWindow {
} }
} }
spacing: 10 delegate: NotificationToast {
delegate: Toast {} id: toast
property int countdownTime: Config.notifications.toastDuration
required property int index
width: ListView.view.width
showTimeBar: true
Timer {
id: timer
interval: 100
onTriggered: () => {
toast.countdownTime -= interval;
if (toast.countdownTime <= 0) {
toast.close();
}
}
repeat: true
running: !toast.containsMouse && toast.countdownTime > 0
}
Component.onCompleted: {
toast.close.connect(() => {
popupcol.model.remove(toast.index, 1);
});
}
}
} }
} }
} }