notifcenter: group notifications by app with per-group dismiss

This commit is contained in:
Damocles 2026-04-17 09:31:34 +02:00
parent 13372e8055
commit 69e3711aec

View file

@ -68,25 +68,73 @@ M.HoverPanel {
property var _pendingDismissIds: []
function _cascadeDismiss() {
const list = M.NotifService.list;
if (list.length === 0)
return;
_pendingDismissIds = list.map(n => n.id);
const visibles = [];
for (let i = 0; i < list.length; i++) {
const d = notifList.itemAtIndex(i);
if (d && d.modelData?.state !== "dismissing")
visibles.push(d);
// Group notifications by appName, sorted by max urgency desc then most recent time desc
readonly property var _groups: {
const map = {};
for (const n of M.NotifService.list) {
const key = n.appName || "";
if (!map[key])
map[key] = {
appName: key,
appIcon: n.appIcon,
notifs: [],
maxUrgency: 0,
maxTime: 0
};
map[key].notifs.push(n);
if (n.urgency > map[key].maxUrgency)
map[key].maxUrgency = n.urgency;
if (n.time > map[key].maxTime)
map[key].maxTime = n.time;
}
return Object.values(map).sort((a, b) => {
if (b.maxUrgency !== a.maxUrgency)
return b.maxUrgency - a.maxUrgency;
return b.maxTime - a.maxTime;
});
}
// Flat model: group header followed by its notifications
readonly property var _flatModel: {
const arr = [];
for (const g of _groups) {
arr.push({
type: "header",
appName: g.appName,
appIcon: g.appIcon,
count: g.notifs.length
});
for (const n of g.notifs)
arr.push({
type: "notif",
data: n
});
}
return arr;
}
// Collect visible (non-dismissing) notif delegates, optionally filtered by appName
function _getVisibleNotifDelegates(appName) {
const result = [];
for (let i = 0; i < _flatModel.length; i++) {
const item = _flatModel[i];
if (item.type !== "notif")
continue;
if (appName !== undefined && item.data.appName !== appName)
continue;
const d = notifList.itemAtIndex(i);
if (d && d._type === "notif" && d._notif?.state !== "dismissing")
result.push(d);
}
return result;
}
function _startCascade(visibles, ids) {
_pendingDismissIds = ids;
if (visibles.length === 0) {
_finishCascade();
return;
}
for (let i = 0; i < visibles.length; i++) {
_cascadeTimer.createObject(menuWindow, {
_target: visibles[i],
@ -96,6 +144,20 @@ M.HoverPanel {
}
}
function _cascadeDismiss() {
if (M.NotifService.list.length === 0)
return;
const ids = M.NotifService.list.map(n => n.id);
_startCascade(_getVisibleNotifDelegates(), ids);
}
function _cascadeGroupDismiss(appName) {
const ids = M.NotifService.list.filter(n => n.appName === appName).map(n => n.id);
if (ids.length === 0)
return;
_startCascade(_getVisibleNotifDelegates(appName), ids);
}
function _finishCascade() {
const ids = _pendingDismissIds;
_pendingDismissIds = [];
@ -113,10 +175,8 @@ M.HoverPanel {
onTriggered: {
if (_target && _target.dismissVisualOnly)
_target.dismissVisualOnly();
if (_isLast) {
// Wait for the last animation to finish, then bulk dismiss
if (_isLast)
_bulkTimer.createObject(menuWindow, {});
}
destroy();
}
}
@ -148,33 +208,36 @@ M.HoverPanel {
height: Math.min(contentHeight, 60 * (M.Modules.notifications.maxVisible || 10))
clip: true
boundsBehavior: Flickable.StopAtBounds
model: M.NotifService.list
model: menuWindow._flatModel
delegate: Item {
id: notifItem
id: notifDelegate
required property var modelData
required property int index
readonly property string _type: modelData.type
readonly property var _notif: _type === "notif" ? modelData.data : null
width: menuWindow.contentWidth
height: _targetHeight * _heightScale
opacity: 0
clip: true
opacity: 0
readonly property real _targetHeight: notifContent.height + 12
readonly property real _targetHeight: _type === "header" ? 28 : (notifContent.implicitHeight + 12)
property real _heightScale: 1
property bool _skipDismiss: false
function dismiss() {
if (notifItem.modelData.state === "dismissing")
if (_type !== "notif" || _notif.state === "dismissing")
return;
notifItem.modelData.beginDismiss();
_notif.beginDismiss();
_dismissAnim.start();
}
function dismissVisualOnly() {
if (notifItem.modelData.state === "dismissing")
if (_type !== "notif" || _notif.state === "dismissing")
return;
notifItem.modelData.beginDismiss();
_notif.beginDismiss();
_skipDismiss = true;
_dismissAnim.start();
}
@ -183,13 +246,84 @@ M.HoverPanel {
NumberAnimation {
id: fadeIn
target: notifItem
target: notifDelegate
property: "opacity"
to: 1
duration: 150
easing.type: Easing.OutCubic
}
// ---- Group header ----
Item {
visible: notifDelegate._type === "header"
anchors.fill: parent
Image {
id: _headerIcon
anchors.left: parent.left
anchors.leftMargin: 10
anchors.verticalCenter: parent.verticalCenter
width: M.Theme.fontSize + 2
height: M.Theme.fontSize + 2
source: {
if (notifDelegate._type !== "header")
return "";
const ic = notifDelegate.modelData.appIcon;
if (!ic)
return "";
return ic.startsWith("/") ? ic : Quickshell.iconPath(ic, "dialog-information");
}
visible: status === Image.Ready
fillMode: Image.PreserveAspectFit
sourceSize: Qt.size(M.Theme.fontSize + 2, M.Theme.fontSize + 2)
asynchronous: true
}
Text {
anchors.left: _headerIcon.visible ? _headerIcon.right : parent.left
anchors.leftMargin: _headerIcon.visible ? 6 : 10
anchors.right: _groupDismissBtn.left
anchors.rightMargin: 6
anchors.verticalCenter: parent.verticalCenter
text: notifDelegate._type === "header" ? (notifDelegate.modelData.appName || "Unknown") : ""
color: M.Theme.base05
font.pixelSize: M.Theme.fontSize - 1
font.family: M.Theme.fontFamily
font.bold: true
elide: Text.ElideRight
}
Text {
id: _groupDismissBtn
anchors.right: parent.right
anchors.rightMargin: 10
anchors.verticalCenter: parent.verticalCenter
text: "\uF1F8"
color: _groupDismissArea.containsMouse ? M.Theme.base08 : M.Theme.base04
font.pixelSize: M.Theme.fontSize - 1
font.family: M.Theme.iconFontFamily
MouseArea {
id: _groupDismissArea
anchors.fill: parent
anchors.margins: -4
hoverEnabled: true
cursorShape: Qt.PointingHandCursor
onClicked: {
if (notifDelegate._type === "header")
menuWindow._cascadeGroupDismiss(notifDelegate.modelData.appName);
}
}
}
}
// ---- Individual notification ----
Item {
id: _notifRow
visible: notifDelegate._type === "notif"
anchors.fill: parent
// Hover + progress bar background
Rectangle {
anchors.fill: parent
anchors.leftMargin: 4
@ -198,11 +332,11 @@ M.HoverPanel {
radius: M.Theme.radius
Rectangle {
visible: (notifItem.modelData.hints?.value ?? -1) >= 0
visible: (notifDelegate._notif?.hints?.value ?? -1) >= 0
anchors.top: parent.top
anchors.bottom: parent.bottom
anchors.left: parent.left
width: parent.width * Math.min(1, Math.max(0, (notifItem.modelData.hints?.value ?? 0) / 100))
width: parent.width * Math.min(1, Math.max(0, (notifDelegate._notif?.hints?.value ?? 0) / 100))
color: M.Theme.base02
radius: parent.radius
@ -214,7 +348,7 @@ M.HoverPanel {
}
}
// Urgency accent
// Urgency accent bar
Rectangle {
anchors.left: parent.left
anchors.leftMargin: 4
@ -225,11 +359,12 @@ M.HoverPanel {
width: 2
radius: 1
color: {
const u = notifItem.modelData.urgency;
const u = notifDelegate._notif?.urgency ?? NotificationUrgency.Normal;
return u === NotificationUrgency.Critical ? M.Theme.base08 : u === NotificationUrgency.Low ? M.Theme.base04 : M.Theme.base0D;
}
}
// App icon / image
Image {
id: ncIcon
anchors.left: parent.left
@ -239,10 +374,12 @@ M.HoverPanel {
width: 24
height: 24
source: {
const img = notifItem.modelData.image;
if (notifDelegate._type !== "notif")
return "";
const img = notifDelegate._notif?.image;
if (img)
return img;
const ic = notifItem.modelData.appIcon;
const ic = notifDelegate._notif?.appIcon;
if (!ic)
return "";
return ic.startsWith("/") ? ic : Quickshell.iconPath(ic, "dialog-information");
@ -262,29 +399,23 @@ M.HoverPanel {
anchors.topMargin: 6
spacing: 1
// App + time
// Summary + time on the same row
Row {
width: parent.width
Text {
text: notifItem.modelData.appName || "Notification"
color: M.Theme.base04
font.pixelSize: M.Theme.fontSize - 2
text: notifDelegate._notif?.summary ?? ""
color: M.Theme.base05
font.pixelSize: M.Theme.fontSize
font.family: M.Theme.fontFamily
font.bold: true
elide: Text.ElideRight
width: parent.width - ageText.width - 4
width: parent.width - _notifTime.implicitWidth - 4
}
Text {
id: ageText
text: {
const diff = Math.floor((Date.now() - notifItem.modelData.time) / 60000);
if (diff < 1)
return "now";
if (diff < 60)
return diff + "m";
if (diff < 1440)
return Math.floor(diff / 60) + "h";
return Math.floor(diff / 1440) + "d";
}
id: _notifTime
text: notifDelegate._notif?.timeStr ?? ""
color: M.Theme.base03
font.pixelSize: M.Theme.fontSize - 2
font.family: M.Theme.fontFamily
@ -293,17 +424,7 @@ M.HoverPanel {
Text {
width: parent.width
text: notifItem.modelData.summary || ""
color: M.Theme.base05
font.pixelSize: M.Theme.fontSize
font.family: M.Theme.fontFamily
font.bold: true
elide: Text.ElideRight
}
Text {
width: parent.width
text: notifItem.modelData.body || ""
text: notifDelegate._notif?.body ?? ""
color: M.Theme.base04
font.pixelSize: M.Theme.fontSize - 1
font.family: M.Theme.fontFamily
@ -316,10 +437,10 @@ M.HoverPanel {
// Actions
Row {
spacing: 4
visible: notifItem.modelData.actions && notifItem.modelData.actions.length > 0
visible: notifDelegate._notif?.actions && notifDelegate._notif.actions.length > 0
Repeater {
model: notifItem.modelData.actions || []
model: notifDelegate._notif?.actions ?? []
delegate: Rectangle {
required property var modelData
width: actText.implicitWidth + 10
@ -337,6 +458,7 @@ M.HoverPanel {
font.pixelSize: M.Theme.fontSize - 2
font.family: M.Theme.fontFamily
}
MouseArea {
id: actArea
anchors.fill: parent
@ -344,7 +466,7 @@ M.HoverPanel {
cursorShape: Qt.PointingHandCursor
onClicked: {
parent.modelData.invoke();
M.NotifService.dismiss(notifItem.modelData.id);
M.NotifService.dismiss(notifDelegate._notif.id);
}
}
}
@ -352,7 +474,7 @@ M.HoverPanel {
}
}
// Dismiss button
// Per-notification dismiss button
Text {
id: dismissBtn
anchors.right: parent.right
@ -370,7 +492,18 @@ M.HoverPanel {
anchors.margins: -4
hoverEnabled: true
cursorShape: Qt.PointingHandCursor
onClicked: _dismissAnim.start()
onClicked: notifDelegate.dismiss()
}
}
// Right-click to dismiss
MouseArea {
id: notifArea
anchors.fill: parent
z: -1
hoverEnabled: true
acceptedButtons: Qt.RightButton
onClicked: notifDelegate.dismiss()
}
}
@ -378,14 +511,14 @@ M.HoverPanel {
id: _dismissAnim
ParallelAnimation {
NumberAnimation {
target: notifItem
target: notifDelegate
property: "x"
to: menuWindow.contentWidth
duration: 200
easing.type: Easing.InCubic
}
NumberAnimation {
target: notifItem
target: notifDelegate
property: "opacity"
to: 0
duration: 200
@ -393,7 +526,7 @@ M.HoverPanel {
}
}
NumberAnimation {
target: notifItem
target: notifDelegate
property: "_heightScale"
to: 0
duration: 150
@ -401,20 +534,11 @@ M.HoverPanel {
}
ScriptAction {
script: {
if (!notifItem._skipDismiss)
M.NotifService.dismiss(notifItem.modelData.id);
if (notifDelegate._notif && !notifDelegate._skipDismiss)
M.NotifService.dismiss(notifDelegate._notif.id);
}
}
}
MouseArea {
id: notifArea
anchors.fill: parent
z: -1
hoverEnabled: true
acceptedButtons: Qt.RightButton
onClicked: _dismissAnim.start()
}
}
}