feat: improve notification popup

This commit is contained in:
Nydragon 2024-11-11 23:37:02 +01:00
parent 9549bd50be
commit 02a72fb5d2
Signed by: nydragon
SSH key fingerprint: SHA256:WcjW5NJPQ8Dx4uQDmoIlVPLWE27Od3fxoe0IUvuoPHE
5 changed files with 137 additions and 52 deletions

View file

@ -2,9 +2,12 @@ pragma Singleton
import Quickshell.Services.Notifications import Quickshell.Services.Notifications
import Quickshell import Quickshell
import QtQuick
Singleton { Singleton {
property var d: NotificationServer { id: notif
property var _: NotificationServer {
actionIconsSupported: true actionIconsSupported: true
actionsSupported: true actionsSupported: true
bodyHyperlinksSupported: true bodyHyperlinksSupported: true
@ -13,4 +16,13 @@ Singleton {
bodySupported: true bodySupported: true
imageSupported: true imageSupported: true
} }
Item {
Component.onCompleted: () => {
notif._.notification.connect(n => {
list.push(n);
});
}
}
property var list: []
} }

View file

@ -2,10 +2,8 @@ import QtQuick.Layouts
import QtQuick import QtQuick
import Quickshell import Quickshell
import QtQuick.Controls import QtQuick.Controls
import Quickshell.Widgets import "../../provider"
import "root:provider" import "../../base"
import "root:base"
import QtQuick.Effects
import Quickshell.Services.Mpris import Quickshell.Services.Mpris
ColumnLayout { ColumnLayout {
@ -74,7 +72,7 @@ ColumnLayout {
, ["media-playlist-repeat", MprisLoopState.Playlist] // , ["media-playlist-repeat", MprisLoopState.Playlist] //
] ]
property int index: map.findIndex(e => e[1] === Player.current?.loopState) property int index: map.findIndex(e => e[1] === Player.current?.loopState)
source: visible ? Quickshell.iconPath(map[index][0]) : "" source: loopButton.visible ? Quickshell.iconPath(map[index][0]) : ""
onClicked: { onClicked: {
const ind = (index + 1) % map.length; const ind = (index + 1) % map.length;
Player.current.loopState = map[ind][1]; Player.current.loopState = map[ind][1];

View file

@ -4,25 +4,27 @@ import QtQuick.Controls
import "root:provider" import "root:provider"
import "root:base" import "root:base"
PopupWindow { PanelWindow {
id: popups id: popups
required property var win required property var win
anchor { visible: true
rect.x: lbar.width * 1.2
window: popups.win anchors {
left: true
top: true
bottom: true
} }
visible: true exclusionMode: ExclusionMode.Normal
mask: Region { mask: Region {
intersection: Intersection.Combine intersection: Intersection.Combine
height: popupcol.count * 26 + popupcol.count * popupcol.spacing + (mouseArea.containsMouse * 114) height: popupcol.contentHeight
width: 300 width: popups.width
} }
color: "transparent" color: "transparent"
height: popupcol.count * 26 + 300 width: 500
width: 300
MouseArea { MouseArea {
id: mouseArea id: mouseArea
@ -33,10 +35,11 @@ PopupWindow {
id: popupcol id: popupcol
anchors.margins: lbar.width * 0.2 anchors.margins: lbar.width * 0.2
anchors.fill: parent anchors.fill: parent
focus: true
model: ListModel { model: ListModel {
id: data id: data
Component.onCompleted: () => { Component.onCompleted: () => {
Notifications.d.notification.connect(e => { Notifications._.notification.connect(e => {
data.insert(0, e); data.insert(0, e);
}); });
} }

View file

@ -1,14 +1,15 @@
pragma ComponentBehavior: Bound
import QtQuick import QtQuick
import QtQuick.Controls import QtQuick.Controls
import QtQuick.Layouts import QtQuick.Layouts
import Quickshell import Quickshell
import Quickshell.Widgets import Quickshell.Widgets
import "root:base" import "../../base"
import Quickshell.Services.Notifications import Quickshell.Services.Notifications
MouseArea { MouseArea {
id: toast id: toast
property int lifetime: 10000 property int lifetime: 5000
property int countdownTime: lifetime property int countdownTime: lifetime
required property string appName required property string appName
@ -17,26 +18,34 @@ MouseArea {
required property string appIcon required property string appIcon
required property string image required property string image
required property NotificationUrgency urgency required property NotificationUrgency urgency
required property bool hasActionIcons
required property var actions
required property int index required property int index
property var close: () => { function close(): void {
toast.parent.parent.model.remove(toast.index, 1); popupcol.model.remove(toast.index, 1);
} }
hoverEnabled: true hoverEnabled: true
height: 26 height: box.height
width: parent.width width: popupcol.width
BRectangle { BRectangle {
id: box id: box
anchors.fill: parent width: parent.width
height: header.height + actions.height + test.height + (5 * 3)
Column { clip: true
anchors.fill: parent
Item {
id: inner
anchors.margins: 5 anchors.margins: 5
anchors.fill: parent
RowLayout { RowLayout {
id: header
width: parent.width width: parent.width
height: 25
IconImage { IconImage {
source: toast.appIcon ? Quickshell.iconPath(toast.appIcon) : "" source: toast.appIcon ? Quickshell.iconPath(toast.appIcon) : ""
@ -48,6 +57,8 @@ MouseArea {
Text { Text {
text: (toast.appIcon ? " " : toast.appName + ": ") + toast.summary text: (toast.appIcon ? " " : toast.appName + ": ") + toast.summary
Layout.fillWidth: true Layout.fillWidth: true
elide: Text.ElideRight
font.pointSize: 12.5
} }
Item { Item {
@ -61,12 +72,29 @@ MouseArea {
} }
} }
Rectangle {
id: test
width: parent.width
anchors.top: header.bottom
height: 60
clip: true
property int maxHeight: 0
color: "transparent"
Text { Text {
id: text
anchors.topMargin: 5
text: toast.body text: toast.body
width: parent.width width: parent.width
visible: box.state === "expand" height: parent.height
wrapMode: Text.Wrap wrapMode: Text.Wrap
Layout.fillWidth: true elide: Text.ElideRight
font.pointSize: 12.5
Component.onCompleted: () => {
if (text.implicitHeight < test.height) {
test.height = text.implicitHeight;
}
test.maxHeight = text.implicitHeight;
} }
} }
@ -74,18 +102,68 @@ MouseArea {
name: "expand" name: "expand"
when: toast.containsMouse when: toast.containsMouse
PropertyChanges { PropertyChanges {
target: toast target: test
height: 140 height: test.maxHeight
} }
} }
transitions: Transition { transitions: Transition {
NumberAnimation { NumberAnimation {
properties: "width,height" properties: "width,height"
duration: 100 duration: 50
easing.type: Easing.InOutQuad easing.type: Easing.InOutQuad
} }
} }
}
RowLayout {
id: actions
width: parent.width
anchors.top: test.bottom
anchors.topMargin: 5
anchors.bottomMargin: 5
Repeater {
model: toast.actions
delegate: Button {
id: actionButton
required property var modelData
IconImage {
visible: toast.hasActionIcons
Component.onCompleted: () => {
if (toast.hasActionIcons) {
source = actionButton.modelData.identifier;
}
}
}
text: modelData.text
onClicked: () => modelData?.invoke()
}
}
visible: toast?.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
}
}
}
Rectangle { Rectangle {
anchors.margins: 2 anchors.margins: 2

View file

@ -59,19 +59,13 @@ PanelWindow {
onClicked: () => I3.dispatch(`workspace ${rec.modelData + 1}`) onClicked: () => I3.dispatch(`workspace ${rec.modelData + 1}`)
} }
} }
Component.onCompleted: () => {
const qs = "/nix/store/78nmm1gzgwfvqgj4pmzi4dgjjzyh8amn-quickshell-0.1.0/bin/quickshell";
const conf = "~/devel/projects/nysh/src/shell.qml";
I3.dispatch(`bindsym Mod4+M exec "${qs} -p ${conf} msg workspace-view toggle"`);
}
} }
color: "transparent" color: "transparent"
width: content?.width ?? 500 width: content?.width ?? 500
height: content?.height ?? 500 height: content?.height ?? 500
visible: true visible: false
Component.onCompleted: () => { Component.onCompleted: () => {
I3.focusedWorkspaceChanged.connect(() => workspaceView.visible = false); I3.focusedWorkspaceChanged.connect(() => workspaceView.visible = false);