// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*- const Clutter = imports.gi.Clutter; const GdkPixbuf = imports.gi.GdkPixbuf; const Gio = imports.gi.Gio; const GLib = imports.gi.GLib; const Gtk = imports.gi.Gtk; const Lang = imports.lang; const Shell = imports.gi.Shell; const Mainloop = imports.mainloop; const St = imports.gi.St; const Config = imports.misc.config; const Main = imports.ui.main; const MessageTray = imports.ui.messageTray; const Params = imports.misc.params; const Util = imports.misc.util; // Should really be defined in Gio.js const BusIface = ' \ \ \ \ \ \ \ '; var BusProxy = Gio.DBusProxy.makeProxyWrapper(BusIface); function Bus() { return new BusProxy(Gio.DBus.session, 'org.freedesktop.DBus', '/org/freedesktop/DBus'); } const FdoNotificationsIface = ' \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ \ '; const NotificationClosedReason = { EXPIRED: 1, DISMISSED: 2, APP_CLOSED: 3, UNDEFINED: 4 }; const Urgency = { LOW: 0, NORMAL: 1, CRITICAL: 2 }; const rewriteRules = { 'XChat': [ { pattern: /^XChat: Private message from: (\S*) \(.*\)$/, replacement: '<$1>' }, { pattern: /^XChat: New public message from: (\S*) \((.*)\)$/, replacement: '$2 <$1>' }, { pattern: /^XChat: Highlighted message from: (\S*) \((.*)\)$/, replacement: '$2 <$1>' } ] }; const FdoNotificationDaemon = new Lang.Class({ Name: 'FdoNotificationDaemon', _init: function() { this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(FdoNotificationsIface, this); this._dbusImpl.export(Gio.DBus.session, '/org/freedesktop/Notifications'); this._sources = []; this._senderToPid = {}; this._notifications = {}; this._busProxy = new Bus(); this._nextNotificationId = 1; this._trayManager = new Shell.TrayManager(); this._trayIconAddedId = this._trayManager.connect('tray-icon-added', Lang.bind(this, this._onTrayIconAdded)); this._trayIconRemovedId = this._trayManager.connect('tray-icon-removed', Lang.bind(this, this._onTrayIconRemoved)); Shell.WindowTracker.get_default().connect('notify::focus-app', Lang.bind(this, this._onFocusAppChanged)); Main.overview.connect('hidden', Lang.bind(this, this._onFocusAppChanged)); this._trayManager.manage_screen(global.screen, Main.messageTray.actor); }, _imageForNotificationData: function(hints) { if (hints['image-data']) { let [width, height, rowStride, hasAlpha, bitsPerSample, nChannels, data] = hints['image-data']; return Shell.util_create_pixbuf_from_data(data, GdkPixbuf.Colorspace.RGB, hasAlpha, bitsPerSample, width, height, rowStride); } else if (hints['image-path']) { return new Gio.FileIcon({ file: Gio.File.new_for_path(hints['image-path']) }); } return null; }, _fallbackIconForNotificationData: function(hints) { let stockIcon; switch (hints.urgency) { case Urgency.LOW: case Urgency.NORMAL: stockIcon = 'gtk-dialog-info'; break; case Urgency.CRITICAL: stockIcon = 'gtk-dialog-error'; break; } return new Gio.ThemedIcon({ name: stockIcon }); }, _iconForNotificationData: function(icon) { if (icon) { if (icon.substr(0, 7) == 'file://') return new Gio.FileIcon({ file: Gio.File.new_for_uri(icon) }); else if (icon[0] == '/') return new Gio.FileIcon({ file: Gio.File.new_for_path(icon) }); else return new Gio.ThemedIcon({ name: icon }); } return null; }, _lookupSource: function(title, pid, trayIcon) { for (let i = 0; i < this._sources.length; i++) { let source = this._sources[i]; if (source.pid == pid && (source.initialTitle == title || source.trayIcon || trayIcon)) return source; } return null; }, // Returns the source associated with ndata.notification if it is set. // If the existing or requested source is associated with a tray icon // and passed in pid matches a pid of an existing source, the title // match is ignored to enable representing a tray icon and notifications // from the same application with a single source. // // If no existing source is found, a new source is created as long as // pid is provided. // // Either a pid or ndata.notification is needed to retrieve or // create a source. _getSource: function(title, pid, ndata, sender, trayIcon) { if (!pid && !(ndata && ndata.notification)) return null; // We use notification's source for the notifications we still have // around that are getting replaced because we don't keep sources // for transient notifications in this._sources, but we still want // the notification associated with them to get replaced correctly. if (ndata && ndata.notification) return ndata.notification.source; let source = this._lookupSource(title, pid, trayIcon); if (source) { source.setTitle(title); return source; } let source = new FdoNotificationDaemonSource(title, pid, sender, trayIcon, ndata ? ndata.hints['desktop-entry'] : null); this._sources.push(source); source.connect('destroy', Lang.bind(this, function() { let index = this._sources.indexOf(source); if (index >= 0) this._sources.splice(index, 1); })); Main.messageTray.add(source); return source; }, NotifyAsync: function(params, invocation) { let [appName, replacesId, icon, summary, body, actions, hints, timeout] = params; let id; for (let hint in hints) { // unpack the variants hints[hint] = hints[hint].deep_unpack(); } hints = Params.parse(hints, { urgency: Urgency.NORMAL }, true); // Filter out chat, presence, calls and invitation notifications from // Empathy, since we handle that information from telepathyClient.js if (appName == 'Empathy' && (hints['category'] == 'im.received' || hints['category'] == 'x-empathy.im.room-invitation' || hints['category'] == 'x-empathy.call.incoming' || hints['category'] == 'x-empathy.transfer.incoming' || hints['category'] == 'x-empathy.im.subscription-request' || hints['category'] == 'presence.online' || hints['category'] == 'presence.offline')) { // Ignore replacesId since we already sent back a // NotificationClosed for that id. id = this._nextNotificationId++; let idle_id = Mainloop.idle_add(Lang.bind(this, function () { this._emitNotificationClosed(id, NotificationClosedReason.DISMISSED); return GLib.SOURCE_REMOVE; })); GLib.Source.set_name_by_id(idle_id, '[gnome-shell] this._emitNotificationClosed'); return invocation.return_value(GLib.Variant.new('(u)', [id])); } let rewrites = rewriteRules[appName]; if (rewrites) { for (let i = 0; i < rewrites.length; i++) { let rule = rewrites[i]; if (summary.search(rule.pattern) != -1) summary = summary.replace(rule.pattern, rule.replacement); } } // Be compatible with the various hints for image data and image path // 'image-data' and 'image-path' are the latest name of these hints, introduced in 1.2 if (!hints['image-path'] && hints['image_path']) hints['image-path'] = hints['image_path']; // version 1.1 of the spec if (!hints['image-data']) { if (hints['image_data']) hints['image-data'] = hints['image_data']; // version 1.1 of the spec else if (hints['icon_data'] && !hints['image-path']) // early versions of the spec; 'icon_data' should only be used if 'image-path' is not available hints['image-data'] = hints['icon_data']; } let ndata = { appName: appName, icon: icon, summary: summary, body: body, actions: actions, hints: hints, timeout: timeout }; if (replacesId != 0 && this._notifications[replacesId]) { ndata.id = id = replacesId; ndata.notification = this._notifications[replacesId].notification; } else { replacesId = 0; ndata.id = id = this._nextNotificationId++; } this._notifications[id] = ndata; let sender = invocation.get_sender(); let pid = this._senderToPid[sender]; let source = this._getSource(appName, pid, ndata, sender, null); if (source) { this._notifyForSource(source, ndata); return invocation.return_value(GLib.Variant.new('(u)', [id])); } if (replacesId) { // There's already a pending call to GetConnectionUnixProcessID, // which will see the new notification data when it finishes, // so we don't have to do anything. return invocation.return_value(GLib.Variant.new('(u)', [id]));; } this._busProxy.GetConnectionUnixProcessIDRemote(sender, Lang.bind(this, function (result, excp) { // The app may have updated or removed the notification ndata = this._notifications[id]; if (!ndata) return; if (excp) { logError(excp, 'Call to GetConnectionUnixProcessID failed'); return; } let [pid] = result; source = this._getSource(appName, pid, ndata, sender, null); this._senderToPid[sender] = pid; source.connect('destroy', Lang.bind(this, function() { delete this._senderToPid[sender]; })); this._notifyForSource(source, ndata); })); return invocation.return_value(GLib.Variant.new('(u)', [id])); }, _makeButton: function(id, label, useActionIcons) { let button = new St.Button({ can_focus: true }); let iconName = id.endsWith('-symbolic') ? id : id + '-symbolic'; if (useActionIcons && Gtk.IconTheme.get_default().has_icon(iconName)) { button.add_style_class_name('notification-icon-button'); button.child = new St.Icon({ icon_name: iconName }); } else { button.add_style_class_name('notification-button'); button.label = label; } return button; }, _notifyForSource: function(source, ndata) { let [id, icon, summary, body, actions, hints, notification] = [ndata.id, ndata.icon, ndata.summary, ndata.body, ndata.actions, ndata.hints, ndata.notification]; if (notification == null) { notification = new MessageTray.Notification(source); ndata.notification = notification; notification.connect('destroy', Lang.bind(this, function(n, reason) { delete this._notifications[ndata.id]; let notificationClosedReason; switch (reason) { case MessageTray.NotificationDestroyedReason.EXPIRED: notificationClosedReason = NotificationClosedReason.EXPIRED; break; case MessageTray.NotificationDestroyedReason.DISMISSED: notificationClosedReason = NotificationClosedReason.DISMISSED; break; case MessageTray.NotificationDestroyedReason.SOURCE_CLOSED: notificationClosedReason = NotificationClosedReason.APP_CLOSED; break; } this._emitNotificationClosed(ndata.id, notificationClosedReason); })); } // Mark music notifications so they can be shown in the screen shield notification.isMusic = (ndata.hints['category'] == 'x-gnome.music'); let gicon = this._iconForNotificationData(icon, hints); let gimage = this._imageForNotificationData(hints); // If an icon is not specified, we use 'image-data' or 'image-path' hint for an icon // and don't show a large image. There are currently many applications that use // notify_notification_set_icon_from_pixbuf() from libnotify, which in turn sets // the 'image-data' hint. These applications don't typically pass in 'app_icon' // argument to Notify() and actually expect the pixbuf to be shown as an icon. // So the logic here does the right thing for this case. If both an icon and either // one of 'image-data' or 'image-path' are specified, we show both an icon and // a large image. if (!gicon && gimage) gicon = gimage; else if (!gicon) gicon = this._fallbackIconForNotificationData(hints); notification.update(summary, body, { gicon: gicon, bannerMarkup: true, clear: true, soundFile: hints['sound-file'], soundName: hints['sound-name'] }); let hasDefaultAction = false; if (actions.length) { let useActionIcons = (hints['action-icons'] == true); for (let i = 0; i < actions.length - 1; i += 2) { let [actionId, label] = [actions[i], actions[i+1]]; if (actionId == 'default') { hasDefaultAction = true; } else { notification.addButton(this._makeButton(actionId, label, useActionIcons), Lang.bind(this, function() { this._emitActionInvoked(ndata.id, actionId); })); } } } if (hasDefaultAction) { notification.connect('clicked', Lang.bind(this, function() { this._emitActionInvoked(ndata.id, 'default'); })); } else { notification.connect('clicked', Lang.bind(this, function() { source.open(); })); } switch (hints.urgency) { case Urgency.LOW: notification.setUrgency(MessageTray.Urgency.LOW); break; case Urgency.NORMAL: notification.setUrgency(MessageTray.Urgency.NORMAL); break; case Urgency.CRITICAL: notification.setUrgency(MessageTray.Urgency.CRITICAL); break; } // 'transient' is a reserved keyword in JS, so we have to retrieve the value // of the 'transient' hint with hints['transient'] rather than hints.transient notification.setTransient(hints['transient'] == true); let sourceGIcon = source.useNotificationIcon ? gicon : null; source.processNotification(notification, sourceGIcon); }, CloseNotification: function(id) { let ndata = this._notifications[id]; if (ndata) { if (ndata.notification) ndata.notification.destroy(MessageTray.NotificationDestroyedReason.SOURCE_CLOSED); delete this._notifications[id]; } }, GetCapabilities: function() { return [ 'actions', 'action-icons', 'body', // 'body-hyperlinks', // 'body-images', 'body-markup', // 'icon-multi', 'icon-static', 'sound', ]; }, GetServerInformation: function() { return [ Config.PACKAGE_NAME, 'GNOME', Config.PACKAGE_VERSION, '1.2' ]; }, _onFocusAppChanged: function() { let tracker = Shell.WindowTracker.get_default(); if (!tracker.focus_app) return; for (let i = 0; i < this._sources.length; i++) { let source = this._sources[i]; if (source.app == tracker.focus_app) { source.destroyNotifications(); return; } } }, _emitNotificationClosed: function(id, reason) { this._dbusImpl.emit_signal('NotificationClosed', GLib.Variant.new('(uu)', [id, reason])); }, _emitActionInvoked: function(id, action) { this._dbusImpl.emit_signal('ActionInvoked', GLib.Variant.new('(us)', [id, action])); }, _onTrayIconAdded: function(o, icon) { let source = this._getSource(icon.title || icon.wm_class || C_("program", "Unknown"), icon.pid, null, null, icon); }, _onTrayIconRemoved: function(o, icon) { let source = this._lookupSource(null, icon.pid, true); if (source) source.destroy(); } }); const FdoNotificationDaemonSource = new Lang.Class({ Name: 'FdoNotificationDaemonSource', Extends: MessageTray.Source, _init: function(title, pid, sender, trayIcon, appId) { // Need to set the app before chaining up, so // methods called from the parent constructor can find it this.trayIcon = trayIcon; this.pid = pid; this.app = this._getApp(appId); this.parent(title); this.initialTitle = title; if (this.app) this.title = this.app.get_name(); else this.useNotificationIcon = true; if (sender) this._nameWatcherId = Gio.DBus.session.watch_name(sender, Gio.BusNameWatcherFlags.NONE, null, Lang.bind(this, this._onNameVanished)); else this._nameWatcherId = 0; if (this.trayIcon) { // Try again finding the app, using the WM_CLASS from the tray icon this._setSummaryIcon(this.trayIcon); this.useNotificationIcon = false; } }, _createPolicy: function() { if (this.app && this.app.get_app_info()) { let id = this.app.get_id().replace(/\.desktop$/,''); return new MessageTray.NotificationApplicationPolicy(id); } else { return new MessageTray.NotificationGenericPolicy(); } }, _onNameVanished: function() { // Destroy the notification source when its sender is removed from DBus. // Only do so if this.app is set to avoid removing "notify-send" sources, senders // of which аre removed from DBus immediately. // Sender being removed from DBus would normally result in a tray icon being removed, // so allow the code path that handles the tray icon being removed to handle that case. if (!this.trayIcon && this.app) this.destroy(); }, processNotification: function(notification, gicon) { if (gicon) this._gicon = gicon; if (!this.trayIcon) this.iconUpdated(); let tracker = Shell.WindowTracker.get_default(); if (this.app && tracker.focus_app == this.app) this.pushNotification(notification); else this.notify(notification); }, handleSummaryClick: function(button) { if (!this.trayIcon) return false; let event = Clutter.get_current_event(); // Left clicks are passed through only where there aren't unacknowledged // notifications, so it possible to open them in summary mode; right // clicks are always forwarded, as the right click menu is not useful for // tray icons if (button == 1 && this.notifications.length > 0) return false; let id = global.stage.connect('deactivate', Lang.bind(this, function () { global.stage.disconnect(id); this.trayIcon.click(event); })); Main.overview.hide(); return true; }, _getApp: function(appId) { let app; app = Shell.WindowTracker.get_default().get_app_from_pid(this.pid); if (app != null) return app; if (this.trayIcon) { app = Shell.AppSystem.get_default().lookup_startup_wmclass(this.trayIcon.wm_class); if (app != null) return app; app = Shell.AppSystem.get_default().lookup_desktop_wmclass(this.trayIcon.wm_class); if (app != null) return app; } if (appId) { app = Shell.AppSystem.get_default().lookup_app(appId + '.desktop'); if (app != null) return app; } return null; }, setTitle: function(title) { // Do nothing if .app is set, we don't want to override the // app name with whatever is provided through libnotify (usually // garbage) if (this.app) return; this.parent(title); }, open: function() { this.openApp(); this.destroyNotifications(); }, _lastNotificationRemoved: function() { if (!this.trayIcon) this.destroy(); }, openApp: function() { if (this.app == null) return; this.app.activate(); Main.overview.hide(); }, destroy: function() { if (this._nameWatcherId) { Gio.DBus.session.unwatch_name(this._nameWatcherId); this._nameWatcherId = 0; } this.parent(); }, createIcon: function(size) { if (this.trayIcon) { return new Clutter.Clone({ width: size, height: size, source: this.trayIcon }); } else if (this.app) { return this.app.create_icon_texture(size); } else if (this._gicon) { return new St.Icon({ gicon: this._gicon, icon_size: size }); } else { return null; } } }); const GtkNotificationDaemonNotification = new Lang.Class({ Name: 'GtkNotificationDaemonNotification', Extends: MessageTray.Notification, _init: function(source, notification) { this.parent(source); this._serialized = GLib.Variant.new('a{sv}', notification); let { "title": title, "body": body, "icon": gicon, "urgent": urgent, "buttons": buttons, "default-action": defaultAction, "default-action-target": defaultActionTarget } = notification; this.setUrgency(urgent.unpack() ? MessageTray.Urgency.CRITICAL : MessageTray.Urgency.NORMAL); if (buttons) { buttons.deep_unpack().forEach(Lang.bind(this, function(button) { this.addAction(button.label.unpack(), Lang.bind(this, this._onButtonClicked, button)); })); } this._defaultAction = defaultAction ? defaultAction.unpack() : null; this._defaultActionTarget = defaultActionTarget; this.update(title.unpack(), body ? body.unpack() : null, { gicon: gicon ? Gio.icon_deserialize(gicon) : null }); }, _activateAction: function(namespacedActionId, target) { if (namespacedActionId) { if (namespacedActionId.startsWith('app.')) { let actionId = namespacedActionId.slice('app.'.length); this.source.activateAction(actionId, target); } } else { this.source.open(); } }, _onButtonClicked: function(button) { let { 'action': action, 'target': actionTarget } = button; this._activateAction(action.unpack(), actionTarget); }, _onClicked: function() { this._activateAction(this._defaultAction, this._defaultActionTarget); this.parent(); }, serialize: function() { return this._serialized; }, }); const FdoApplicationIface = ' \ \ \ \ \ \ \ \ \ \ \ '; const FdoApplicationProxy = Gio.DBusProxy.makeProxyWrapper(FdoApplicationIface); function objectPathFromAppId(appId) { return '/' + appId.replace(/\./g, '/'); } function getPlatformData() { let startupId = GLib.Variant.new('s', '_TIME' + global.get_current_time()); return { "desktop-startup-id": startupId }; } function InvalidAppError() {} const GtkNotificationDaemonAppSource = new Lang.Class({ Name: 'GtkNotificationDaemonAppSource', Extends: MessageTray.Source, _init: function(appId) { this._appId = appId; this._objectPath = objectPathFromAppId(appId); this._app = Shell.AppSystem.get_default().lookup_app(appId + '.desktop'); if (!this._app) throw new InvalidAppError(); this._notifications = {}; this.parent(this._app.get_name()); }, createIcon: function(size) { return this._app.create_icon_texture(size); }, _createPolicy: function() { return new MessageTray.NotificationApplicationPolicy(this._appId); }, _createApp: function() { return new FdoApplicationProxy(Gio.DBus.session, this._appId, this._objectPath); }, activateAction: function(actionId, target) { let app = this._createApp(); app.ActivateActionRemote(actionId, target ? [target] : [], getPlatformData()); }, open: function() { let app = this._createApp(); app.ActivateRemote(getPlatformData()); }, addNotification: function(notificationId, notificationParams, showBanner) { if (this._notifications[notificationId]) this._notifications[notificationId].destroy(); let notification = new GtkNotificationDaemonNotification(this, notificationParams); notification.connect('destroy', Lang.bind(this, function() { delete this._notifications[notificationId]; })); this._notifications[notificationId] = notification; if (showBanner) this.notify(notification); else this.pushNotification(notification); }, removeNotification: function(notificationId) { if (this._notifications[notificationId]) this._notifications[notificationId].destroy(MessageTray.NotificationDestroyedReason.SOURCE_CLOSED); }, serialize: function() { let notifications = []; for (let notificationId in this._notifications) { let notification = this._notifications[notificationId]; notifications.push([notificationId, notification.serialize()]); } return [this._appId, notifications]; }, }); const GtkNotificationsIface = ' \ \ \ \ \ \ \ \ \ \ \ \ '; const GtkNotificationDaemon = new Lang.Class({ Name: 'GtkNotificationDaemon', _init: function() { this._sources = {}; this._loadNotifications(); this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(GtkNotificationsIface, this); this._dbusImpl.export(Gio.DBus.session, '/org/gtk/Notifications'); Gio.DBus.session.own_name('org.gtk.Notifications', Gio.BusNameOwnerFlags.REPLACE, null, null); }, _ensureAppSource: function(appId) { if (this._sources[appId]) return this._sources[appId]; let source = new GtkNotificationDaemonAppSource(appId); source.connect('destroy', Lang.bind(this, function() { delete this._sources[appId]; this._saveNotifications(); })); source.connect('count-updated', Lang.bind(this, this._saveNotifications)); Main.messageTray.add(source); this._sources[appId] = source; return source; }, _loadNotifications: function() { this._isLoading = true; let value = global.get_persistent_state('a(sa(sv))', 'notifications'); if (value) { let sources = value.deep_unpack(); sources.forEach(Lang.bind(this, function([appId, notifications]) { if (notifications.length == 0) return; let source; try { source = this._ensureAppSource(appId); } catch(e if e instanceof InvalidAppError) { return; } notifications.forEach(function([notificationId, notification]) { source.addNotification(notificationId, notification.deep_unpack(), false); }); })); } this._isLoading = false; }, _saveNotifications: function() { if (this._isLoading) return; let sources = []; for (let appId in this._sources) { let source = this._sources[appId]; sources.push(source.serialize()); } global.set_persistent_state('notifications', new GLib.Variant('a(sa(sv))', sources)); }, AddNotificationAsync: function(params, invocation) { let [appId, notificationId, notification] = params; let source; try { source = this._ensureAppSource(appId); } catch(e if e instanceof InvalidAppError) { invocation.return_dbus_error('org.gtk.Notifications.InvalidApp', 'The app by ID "%s" could not be found'.format(appId)); return; } source.addNotification(notificationId, notification, true); invocation.return_value(null); }, RemoveNotificationAsync: function(params, invocation) { let [appId, notificationId] = params; let source = this._sources[appId]; if (source) source.removeNotification(notificationId); invocation.return_value(null); }, }); const NotificationDaemon = new Lang.Class({ Name: 'NotificationDaemon', _init: function() { this._fdoNotificationDaemon = new FdoNotificationDaemon(); this._gtkNotificationDaemon = new GtkNotificationDaemon(); }, });