ags(notifications): reset notification timeout on replaced + better styles

This commit is contained in:
retrozinndev
2025-03-10 21:18:06 -03:00
parent 19c90581ae
commit 8ae855d6e0
4 changed files with 59 additions and 45 deletions
+46 -5
View File
@@ -1,6 +1,11 @@
import { GObject, property, register, signal, timeout } from "astal"; import { AstalIO, GObject, property, register, signal, timeout } from "astal";
import AstalNotifd from "gi://AstalNotifd"; import AstalNotifd from "gi://AstalNotifd";
export const
NOTIFICATION_TIMEOUT_URGENT: number = 0,
NOTIFICATION_TIMEOUT_NORMAL: number = 4000,
NOTIFICATION_TIMEOUT_LOW: number = 2000;
@register({ GTypeName: "Notifications" }) @register({ GTypeName: "Notifications" })
class Notifications extends GObject.Object { class Notifications extends GObject.Object {
private static instance: (Notifications|null) = null; private static instance: (Notifications|null) = null;
@@ -29,6 +34,9 @@ class Notifications extends GObject.Object {
@signal(Number) @signal(Number)
declare historyRemoved: (id: number) => void; declare historyRemoved: (id: number) => void;
@signal(Number)
declare notificationReplaced: (id: number) => void;
constructor() { constructor() {
super(); super();
@@ -36,13 +44,33 @@ class Notifications extends GObject.Object {
this.#connections = [ this.#connections = [
AstalNotifd.get_default().connect("notified", (notifd, id, _replaced) => { AstalNotifd.get_default().connect("notified", (notifd, id, _replaced) => {
const notification = notifd.get_notification(id); const notification = notifd.get_notification(id);
const notifTimeout = 4000; const notifTimeout = notification.urgency === AstalNotifd.Urgency.LOW ?
NOTIFICATION_TIMEOUT_LOW
: (notification.urgency === AstalNotifd.Urgency.CRITICAL ?
NOTIFICATION_TIMEOUT_URGENT
: NOTIFICATION_TIMEOUT_NORMAL);
this.addNotification(notification, () => { this.addNotification(notification, () => {
if(notification.urgency !== AstalNotifd.Urgency.CRITICAL) if(notification.urgency !== AstalNotifd.Urgency.CRITICAL ||
timeout(notifTimeout, () => { (notification.urgency === AstalNotifd.Urgency.CRITICAL &&
NOTIFICATION_TIMEOUT_URGENT > 0)) {
let notifTimer: AstalIO.Time;
let replacedConnectionId: number;
const removeFun = () => { // Funny name haha lmao remove fun :skull:
this.removeNotification(id); this.removeNotification(id);
replacedConnectionId && this.disconnect(replacedConnectionId);
}
replacedConnectionId = this.connect("notification-replaced", (_, id: number) => {
if(notification.id === id) {
notifTimer.cancel();
notifTimer = timeout(notifTimeout, removeFun);
}
}); });
notifTimer = timeout(notifTimeout, removeFun);
}
}); });
}), }),
AstalNotifd.get_default().connect("resolved", (notifd, id, _reason) => { AstalNotifd.get_default().connect("resolved", (notifd, id, _reason) => {
@@ -51,7 +79,8 @@ class Notifications extends GObject.Object {
}) })
]; ];
this.vfunc_dispose = () => { this.run_dispose = () => {
super.run_dispose();
this.#connections.map((id: number) => this.#connections.map((id: number) =>
AstalNotifd.get_default().disconnect(id)); AstalNotifd.get_default().disconnect(id));
}; };
@@ -84,6 +113,10 @@ class Notifications extends GObject.Object {
private addNotification(notif: AstalNotifd.Notification, onAdded?: (notif: AstalNotifd.Notification) => void): void { private addNotification(notif: AstalNotifd.Notification, onAdded?: (notif: AstalNotifd.Notification) => void): void {
const newArray = this.#notifications.reverse().filter((item) => item.id !== notif.id); const newArray = this.#notifications.reverse().filter((item) => item.id !== notif.id);
if(newArray !== this.notifications) {
this.emit("notification-replaced", notif.id);
}
newArray.push(notif); newArray.push(notif);
this.#notifications = newArray.reverse(); this.#notifications = newArray.reverse();
this.notify("notifications"); this.notify("notifications");
@@ -98,6 +131,14 @@ class Notifications extends GObject.Object {
this.notify("notifications"); this.notify("notifications");
this.emit("notification-removed", notifId); this.emit("notification-removed", notifId);
} }
connect(signal: string, callback: (...args: any[]) => void): number {
return super.connect(signal, callback);
}
disconnect(id: number) {
super.disconnect(id);
}
} }
export { Notifications }; export { Notifications };
+3 -14
View File
@@ -7,19 +7,8 @@
top: 6px; top: 6px;
}; };
& revealer { & > .notification {
padding: 6px; margin: 6px;
box-shadow: 0 0 4px .5px colors.$bg-translucent;
& > .notification {
box-shadow: 0 0 4px .5px colors.$bg-translucent;
}
&:first-child {
margin-top: 0;
}
&:last-child {
margin-bottom: 0;
}
} }
} }
+1
View File
@@ -92,6 +92,7 @@ export function NotificationWidget(notification: AstalNotifd.Notification|number
className: "body", className: "body",
useMarkup: true, useMarkup: true,
halign: Gtk.Align.START, halign: Gtk.Align.START,
xalign: 0,
truncate: false, truncate: false,
wrap: true, wrap: true,
wrapMode: Pango.WrapMode.WORD, wrapMode: Pango.WrapMode.WORD,
+9 -26
View File
@@ -3,11 +3,8 @@ import AstalNotifd from "gi://AstalNotifd";
import { bind } from "astal/binding"; import { bind } from "astal/binding";
import { Notifications } from "../scripts/notifications"; import { Notifications } from "../scripts/notifications";
import { NotificationWidget } from "../widget/Notification"; import { NotificationWidget } from "../widget/Notification";
import { timeout } from "astal";
import { VarMap } from "../scripts/varmap";
const connections: Array<number> = []; const connections: Array<number> = [];
const notifWidgets = new VarMap<number, Widget.Revealer>();
export const FloatingNotifications: Widget.Window = new Widget.Window({ export const FloatingNotifications: Widget.Window = new Widget.Window({
namespace: "floating-notifications", namespace: "floating-notifications",
@@ -20,30 +17,11 @@ export const FloatingNotifications: Widget.Window = new Widget.Window({
exclusivity: Astal.Exclusivity.NORMAL, exclusivity: Astal.Exclusivity.NORMAL,
setup: (window) => { setup: (window) => {
connections.push( connections.push(
Notifications.getDefault().connect("notification-added", (_, notif: AstalNotifd.Notification) => { Notifications.getDefault().connect("notification-added", (_, _notif: AstalNotifd.Notification) => {
!window.is_visible() && window.show(); !window.is_visible() && window.show();
notifWidgets.set(notif.id, new Widget.Revealer({
revealChild: false,
transitionDuration: 320,
transitionType: Gtk.RevealerTransitionType.SLIDE_RIGHT,
child: NotificationWidget(notif,
() => Notifications.getDefault().removeNotification(notif.id)),
} as Widget.RevealerProps));
notifWidgets.getValue(notif.id)!.revealChild = true;
}), }),
Notifications.getDefault().connect("notification-removed", (_: Notifications, _id: number) => {
Notifications.getDefault().connect("notification-removed", (_, id: number) => { window.is_visible() && _.notifications.length === 0 && window.hide()
notifWidgets.getValue(id)!.revealChild = false;
timeout(
(notifWidgets.getValue(id)?.get_transition_duration() || 0) + 50,
() => {
notifWidgets.delete(id);
Notifications.getDefault().notifications.length === 0 &&
window.is_visible() && window.hide();
}
);
}) })
); );
}, },
@@ -53,6 +31,11 @@ export const FloatingNotifications: Widget.Window = new Widget.Window({
orientation: Gtk.Orientation.VERTICAL, orientation: Gtk.Orientation.VERTICAL,
homogeneous: false, homogeneous: false,
visible: bind(Notifications.getDefault(), "notifications").as(notifs => notifs.length > 0), visible: bind(Notifications.getDefault(), "notifications").as(notifs => notifs.length > 0),
children: bind(notifWidgets).as((map) => [...map.values()].map((revealer) => revealer)) children: bind(Notifications.getDefault(), "notifications").as((notifs) =>
notifs.map((item) =>
NotificationWidget(item,
() => Notifications.getDefault().removeNotification(item))
)
),
} as Widget.BoxProps) } as Widget.BoxProps)
} as Widget.WindowProps); } as Widget.WindowProps);